refactor: room details page redesign
parent
bbd3d29f55
commit
bc77056b96
@ -1,40 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
|
|
||||||
import 'package:fluffychat/pages/chat/chat.dart';
|
|
||||||
import 'package:fluffychat/pangea/extensions/pangea_room_extension.dart';
|
|
||||||
|
|
||||||
class ActivityPlanPageLaunchIconButton extends StatelessWidget {
|
|
||||||
const ActivityPlanPageLaunchIconButton({
|
|
||||||
super.key,
|
|
||||||
required this.controller,
|
|
||||||
});
|
|
||||||
|
|
||||||
final ChatController controller;
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
if (!controller.room.canSendDefaultStates) {
|
|
||||||
return const SizedBox();
|
|
||||||
}
|
|
||||||
|
|
||||||
return FutureBuilder<bool>(
|
|
||||||
future: controller.room.isBotDM,
|
|
||||||
builder: (BuildContext context, snapshot) {
|
|
||||||
final isBotDM = snapshot.data;
|
|
||||||
if (isBotDM == true || isBotDM == null) {
|
|
||||||
return const SizedBox();
|
|
||||||
}
|
|
||||||
return IconButton(
|
|
||||||
icon: const Icon(Icons.event_note_outlined),
|
|
||||||
tooltip: L10n.of(context).activityPlannerTitle,
|
|
||||||
onPressed: () {
|
|
||||||
context.go('/rooms/${controller.room.id}/planner');
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,71 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import 'package:matrix/matrix.dart';
|
|
||||||
|
|
||||||
import 'package:fluffychat/pangea/spaces/pages/pangea_space_page_view.dart';
|
|
||||||
import 'package:fluffychat/pangea/spaces/utils/load_participants_util.dart';
|
|
||||||
|
|
||||||
class PangeaSpacePage extends StatefulWidget {
|
|
||||||
final Room space;
|
|
||||||
|
|
||||||
const PangeaSpacePage({
|
|
||||||
required this.space,
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<PangeaSpacePage> createState() => PangeaSpacePageState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class PangeaSpacePageState extends State<PangeaSpacePage> {
|
|
||||||
bool expanded = true;
|
|
||||||
|
|
||||||
final TextEditingController searchController = TextEditingController();
|
|
||||||
final FocusNode searchFocusNode = FocusNode();
|
|
||||||
|
|
||||||
@override
|
|
||||||
void initState() {
|
|
||||||
super.initState();
|
|
||||||
searchController.addListener(() {
|
|
||||||
if (mounted) {
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
void dispose() {
|
|
||||||
searchController.dispose();
|
|
||||||
searchFocusNode.dispose();
|
|
||||||
super.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
void startSearch() {
|
|
||||||
setState(() {});
|
|
||||||
searchFocusNode.requestFocus();
|
|
||||||
}
|
|
||||||
|
|
||||||
void cancelSearch({bool unfocus = true}) {
|
|
||||||
setState(() {
|
|
||||||
searchController.clear();
|
|
||||||
});
|
|
||||||
if (unfocus) searchFocusNode.unfocus();
|
|
||||||
}
|
|
||||||
|
|
||||||
void toggleExpanded() {
|
|
||||||
setState(() {
|
|
||||||
expanded = !expanded;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return LoadParticipantsUtil(
|
|
||||||
space: widget.space,
|
|
||||||
builder: (util) => PangeaSpacePageView(
|
|
||||||
this,
|
|
||||||
participantsLoader: util,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,622 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import 'package:collection/collection.dart';
|
|
||||||
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
||||||
import 'package:flutter_linkify/flutter_linkify.dart';
|
|
||||||
import 'package:go_router/go_router.dart';
|
|
||||||
import 'package:material_symbols_icons/symbols.dart';
|
|
||||||
import 'package:matrix/matrix.dart';
|
|
||||||
|
|
||||||
import 'package:fluffychat/config/app_config.dart';
|
|
||||||
import 'package:fluffychat/config/themes.dart';
|
|
||||||
import 'package:fluffychat/pages/chat_details/participant_list_item.dart';
|
|
||||||
import 'package:fluffychat/pangea/bot/utils/bot_name.dart';
|
|
||||||
import 'package:fluffychat/pangea/spaces/pages/pangea_space_page.dart';
|
|
||||||
import 'package:fluffychat/pangea/spaces/utils/load_participants_util.dart';
|
|
||||||
import 'package:fluffychat/utils/fluffy_share.dart';
|
|
||||||
import 'package:fluffychat/utils/matrix_sdk_extensions/matrix_locals.dart';
|
|
||||||
import 'package:fluffychat/utils/url_launcher.dart';
|
|
||||||
import 'package:fluffychat/widgets/adaptive_dialogs/user_dialog.dart';
|
|
||||||
import 'package:fluffychat/widgets/avatar.dart';
|
|
||||||
import 'package:fluffychat/widgets/layouts/max_width_body.dart';
|
|
||||||
|
|
||||||
class PangeaSpacePageView extends StatelessWidget {
|
|
||||||
final PangeaSpacePageState controller;
|
|
||||||
final LoadParticipantsUtilState participantsLoader;
|
|
||||||
const PangeaSpacePageView(
|
|
||||||
this.controller, {
|
|
||||||
required this.participantsLoader,
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final theme = Theme.of(context);
|
|
||||||
final room = controller.widget.space;
|
|
||||||
|
|
||||||
final displayname = room.getLocalizedDisplayname(
|
|
||||||
MatrixLocals(L10n.of(context)),
|
|
||||||
);
|
|
||||||
|
|
||||||
final filteredParticipants = participantsLoader
|
|
||||||
.filteredParticipants("")
|
|
||||||
.where((u) => u.id != BotName.byEnvironment)
|
|
||||||
.toList();
|
|
||||||
|
|
||||||
final bool showMedals = !participantsLoader.loading &&
|
|
||||||
controller.searchController.text.isEmpty &&
|
|
||||||
filteredParticipants.isNotEmpty;
|
|
||||||
|
|
||||||
final Widget leaderboardHeader = ListTile(
|
|
||||||
tileColor: Color.lerp(AppConfig.gold, Colors.black, 0.3),
|
|
||||||
contentPadding: const EdgeInsets.symmetric(horizontal: 16.0),
|
|
||||||
visualDensity: const VisualDensity(vertical: -4.0),
|
|
||||||
title: Text(
|
|
||||||
L10n.of(context).leaderboard,
|
|
||||||
style: Theme.of(context).textTheme.headlineSmall,
|
|
||||||
),
|
|
||||||
trailing: Icon(
|
|
||||||
controller.expanded
|
|
||||||
? Icons.keyboard_arrow_down_outlined
|
|
||||||
: Icons.keyboard_arrow_right_outlined,
|
|
||||||
),
|
|
||||||
onTap: controller.toggleExpanded,
|
|
||||||
);
|
|
||||||
|
|
||||||
return LayoutBuilder(
|
|
||||||
builder: (context, constraints) {
|
|
||||||
return Scaffold(
|
|
||||||
appBar: AppBar(
|
|
||||||
automaticallyImplyLeading: false,
|
|
||||||
elevation: theme.appBarTheme.elevation,
|
|
||||||
backgroundColor: theme.appBarTheme.backgroundColor,
|
|
||||||
actions: [
|
|
||||||
IconButton(
|
|
||||||
icon: const Icon(Icons.settings_outlined),
|
|
||||||
onPressed: () => context.go(
|
|
||||||
'/rooms/${room.id}/details',
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
shape: Border(
|
|
||||||
bottom: BorderSide(
|
|
||||||
color: theme.dividerColor,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
body: Row(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Expanded(
|
|
||||||
child: MaxWidthBody(
|
|
||||||
child: Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|
||||||
children: <Widget>[
|
|
||||||
Row(
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.all(32.0),
|
|
||||||
child: Avatar(
|
|
||||||
mxContent: room.avatar,
|
|
||||||
name: displayname,
|
|
||||||
size: Avatar.defaultSize * 2.5,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
TextButton.icon(
|
|
||||||
onPressed: () => FluffyShare.share(
|
|
||||||
displayname,
|
|
||||||
context,
|
|
||||||
copyOnly: true,
|
|
||||||
),
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.copy_outlined,
|
|
||||||
size: 16,
|
|
||||||
),
|
|
||||||
style: TextButton.styleFrom(
|
|
||||||
foregroundColor:
|
|
||||||
theme.colorScheme.onSurface,
|
|
||||||
),
|
|
||||||
label: Text(
|
|
||||||
displayname,
|
|
||||||
maxLines: 1,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
style: const TextStyle(fontSize: 18),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Row(
|
|
||||||
spacing: 8.0,
|
|
||||||
children: [
|
|
||||||
TextButton.icon(
|
|
||||||
onPressed: () => context.push(
|
|
||||||
'/rooms/${room.id}/details/members',
|
|
||||||
),
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.group_outlined,
|
|
||||||
size: 14,
|
|
||||||
),
|
|
||||||
style: TextButton.styleFrom(
|
|
||||||
foregroundColor:
|
|
||||||
theme.colorScheme.secondary,
|
|
||||||
),
|
|
||||||
label: Text(
|
|
||||||
L10n.of(context).countParticipants(
|
|
||||||
(room.summary.mInvitedMemberCount ??
|
|
||||||
0) +
|
|
||||||
(room.summary
|
|
||||||
.mJoinedMemberCount ??
|
|
||||||
0),
|
|
||||||
),
|
|
||||||
maxLines: 1,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
TextButton.icon(
|
|
||||||
onPressed: () => context.push(
|
|
||||||
'/rooms/${room.id}/details/invite',
|
|
||||||
),
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.group_add_outlined,
|
|
||||||
size: 14,
|
|
||||||
),
|
|
||||||
style: TextButton.styleFrom(
|
|
||||||
foregroundColor:
|
|
||||||
theme.colorScheme.secondary,
|
|
||||||
),
|
|
||||||
label: Text(
|
|
||||||
L10n.of(context).invite,
|
|
||||||
maxLines: 1,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
Divider(color: theme.dividerColor, height: 1),
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
left: 24.0,
|
|
||||||
right: 24.0,
|
|
||||||
top: 16.0,
|
|
||||||
bottom: 16.0,
|
|
||||||
),
|
|
||||||
child: SelectableLinkify(
|
|
||||||
text: room.topic.isEmpty
|
|
||||||
? room.isSpace
|
|
||||||
? L10n.of(context).noSpaceDescriptionYet
|
|
||||||
: L10n.of(context).noChatDescriptionYet
|
|
||||||
: room.topic,
|
|
||||||
options: const LinkifyOptions(humanize: false),
|
|
||||||
linkStyle: const TextStyle(
|
|
||||||
color: Colors.blueAccent,
|
|
||||||
decorationColor: Colors.blueAccent,
|
|
||||||
),
|
|
||||||
style: TextStyle(
|
|
||||||
fontSize: 14,
|
|
||||||
fontStyle: room.topic.isEmpty
|
|
||||||
? FontStyle.italic
|
|
||||||
: FontStyle.normal,
|
|
||||||
color: theme.textTheme.bodyMedium!.color,
|
|
||||||
decorationColor: theme.textTheme.bodyMedium!.color,
|
|
||||||
),
|
|
||||||
onOpen: (url) =>
|
|
||||||
UrlLauncher(context, url.url).launchUrl(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (constraints.maxWidth <= 800) leaderboardHeader,
|
|
||||||
if (constraints.maxWidth <= 800 && controller.expanded)
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.all(16.0),
|
|
||||||
child: Row(
|
|
||||||
spacing: 16.0,
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 200.0,
|
|
||||||
child: LeaderboardMedals(
|
|
||||||
isVisible: showMedals,
|
|
||||||
participants: filteredParticipants,
|
|
||||||
smallRadius: Avatar.defaultSize * 0.7,
|
|
||||||
largeRadius: Avatar.defaultSize,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (filteredParticipants.isNotEmpty)
|
|
||||||
Expanded(
|
|
||||||
child: Column(
|
|
||||||
children: filteredParticipants
|
|
||||||
.take(3)
|
|
||||||
.mapIndexed((i, user) {
|
|
||||||
return TrophyParticipantListItem(
|
|
||||||
index: i,
|
|
||||||
user: user,
|
|
||||||
);
|
|
||||||
}).toList(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (constraints.maxWidth > 800)
|
|
||||||
Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
border: Border(
|
|
||||||
left: BorderSide(
|
|
||||||
color: theme.dividerColor,
|
|
||||||
width: 1.0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
width: 350.0,
|
|
||||||
child: Column(
|
|
||||||
spacing: 16.0,
|
|
||||||
children: [
|
|
||||||
leaderboardHeader,
|
|
||||||
if (controller.expanded)
|
|
||||||
Expanded(
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
LeaderboardMedals(
|
|
||||||
isVisible: showMedals,
|
|
||||||
participants: filteredParticipants,
|
|
||||||
padding: EdgeInsets.only(
|
|
||||||
top: showMedals ? 16.0 : 0,
|
|
||||||
left: showMedals ? 42.0 : 0,
|
|
||||||
right: showMedals ? 42.0 : 0,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.all(16.0),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Row(
|
|
||||||
mainAxisAlignment:
|
|
||||||
MainAxisAlignment.spaceBetween,
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 8.0,
|
|
||||||
),
|
|
||||||
child: participantsLoader.loading
|
|
||||||
? const CircularProgressIndicator
|
|
||||||
.adaptive()
|
|
||||||
: Text(
|
|
||||||
L10n.of(context)
|
|
||||||
.countParticipants(
|
|
||||||
participantsLoader
|
|
||||||
.participants.length,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
IconButton(
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.group_add_outlined,
|
|
||||||
),
|
|
||||||
iconSize: 20.0,
|
|
||||||
onPressed: () => context.push(
|
|
||||||
'/rooms/${room.id}/details/members',
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
TextField(
|
|
||||||
controller: controller.searchController,
|
|
||||||
focusNode: controller.searchFocusNode,
|
|
||||||
textInputAction: TextInputAction.search,
|
|
||||||
decoration: InputDecoration(
|
|
||||||
filled: true,
|
|
||||||
fillColor: theme
|
|
||||||
.colorScheme.secondaryContainer,
|
|
||||||
border: OutlineInputBorder(
|
|
||||||
borderSide: BorderSide.none,
|
|
||||||
borderRadius:
|
|
||||||
BorderRadius.circular(99),
|
|
||||||
),
|
|
||||||
contentPadding: EdgeInsets.zero,
|
|
||||||
hintText: L10n.of(context).search,
|
|
||||||
hintStyle: TextStyle(
|
|
||||||
color: theme
|
|
||||||
.colorScheme.onPrimaryContainer,
|
|
||||||
fontWeight: FontWeight.normal,
|
|
||||||
),
|
|
||||||
prefixIcon: controller.searchController
|
|
||||||
.text.isNotEmpty
|
|
||||||
? IconButton(
|
|
||||||
tooltip:
|
|
||||||
L10n.of(context).cancel,
|
|
||||||
icon: const Icon(
|
|
||||||
Icons.close_outlined,
|
|
||||||
),
|
|
||||||
onPressed:
|
|
||||||
controller.cancelSearch,
|
|
||||||
color: theme.colorScheme
|
|
||||||
.onPrimaryContainer,
|
|
||||||
)
|
|
||||||
: IconButton(
|
|
||||||
onPressed:
|
|
||||||
controller.startSearch,
|
|
||||||
icon: Icon(
|
|
||||||
Icons.search_outlined,
|
|
||||||
color: theme.colorScheme
|
|
||||||
.onPrimaryContainer,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: Builder(
|
|
||||||
builder: (context) {
|
|
||||||
if (participantsLoader.loading) {
|
|
||||||
return const Column(
|
|
||||||
children: [
|
|
||||||
CircularProgressIndicator.adaptive(),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (participantsLoader.error != null) {
|
|
||||||
return Text(
|
|
||||||
L10n.of(context).oopsSomethingWentWrong,
|
|
||||||
style: TextStyle(
|
|
||||||
color: theme.colorScheme.error,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ListView.builder(
|
|
||||||
itemCount: filteredParticipants.length,
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
return TrophyParticipantListItem(
|
|
||||||
index: index,
|
|
||||||
user: filteredParticipants[index],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class LeaderboardMedal extends StatelessWidget {
|
|
||||||
final User user;
|
|
||||||
final Color color;
|
|
||||||
final double radius;
|
|
||||||
final double iconSize;
|
|
||||||
final double iconRadius;
|
|
||||||
|
|
||||||
final double? top;
|
|
||||||
final double? left;
|
|
||||||
final double? right;
|
|
||||||
final double? bottom;
|
|
||||||
|
|
||||||
const LeaderboardMedal(
|
|
||||||
this.user, {
|
|
||||||
required this.color,
|
|
||||||
required this.radius,
|
|
||||||
required this.iconSize,
|
|
||||||
required this.iconRadius,
|
|
||||||
this.top,
|
|
||||||
this.left,
|
|
||||||
this.right,
|
|
||||||
this.bottom,
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Stack(
|
|
||||||
children: [
|
|
||||||
Positioned(
|
|
||||||
top: top,
|
|
||||||
left: left,
|
|
||||||
right: right,
|
|
||||||
bottom: bottom != null ? bottom! + 10.0 : null,
|
|
||||||
child: CircleAvatar(
|
|
||||||
radius: radius + 3.0,
|
|
||||||
child: Container(
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
gradient: LinearGradient(
|
|
||||||
begin: const Alignment(0.5, -0.5),
|
|
||||||
end: const Alignment(-0.5, 0.5),
|
|
||||||
colors: <Color>[
|
|
||||||
color,
|
|
||||||
Colors.white,
|
|
||||||
color,
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: top != null ? 3.0 : null,
|
|
||||||
left: left != null ? 3.0 : null,
|
|
||||||
right: right != null ? 3.0 : null,
|
|
||||||
bottom: bottom != null ? bottom! + 10.0 + 3.0 : null,
|
|
||||||
child: MouseRegion(
|
|
||||||
cursor: SystemMouseCursors.click,
|
|
||||||
child: GestureDetector(
|
|
||||||
onTap: () => UserDialog.show(
|
|
||||||
context: context,
|
|
||||||
profile: Profile(
|
|
||||||
userId: user.id,
|
|
||||||
displayName: user.displayName,
|
|
||||||
avatarUrl: user.avatarUrl,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: Center(
|
|
||||||
child: Avatar(
|
|
||||||
mxContent: user.avatarUrl,
|
|
||||||
name: user.calcDisplayname(),
|
|
||||||
size: radius * 2,
|
|
||||||
presenceUserId: user.id,
|
|
||||||
showPresence: false,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Positioned(
|
|
||||||
top: top != null ? ((radius + 3.0) * 2) - iconRadius : null,
|
|
||||||
left: left != null ? radius + 3.0 - iconRadius : null,
|
|
||||||
right: right != null ? radius + 3.0 - iconRadius : null,
|
|
||||||
bottom: bottom,
|
|
||||||
child: CircleAvatar(
|
|
||||||
backgroundColor: color,
|
|
||||||
radius: iconRadius,
|
|
||||||
child: Icon(
|
|
||||||
Symbols.trophy,
|
|
||||||
color: Colors.white,
|
|
||||||
size: iconSize,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class LeaderboardMedals extends StatelessWidget {
|
|
||||||
final bool isVisible;
|
|
||||||
final List<User> participants;
|
|
||||||
final EdgeInsets? padding;
|
|
||||||
|
|
||||||
final double? largeRadius;
|
|
||||||
final double? smallRadius;
|
|
||||||
|
|
||||||
const LeaderboardMedals({
|
|
||||||
super.key,
|
|
||||||
required this.isVisible,
|
|
||||||
required this.participants,
|
|
||||||
this.largeRadius,
|
|
||||||
this.smallRadius,
|
|
||||||
this.padding,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return AnimatedContainer(
|
|
||||||
duration: FluffyThemes.animationDuration,
|
|
||||||
height: isVisible ? Avatar.defaultSize * 3.5 : 0.0,
|
|
||||||
// padding: EdgeInsets.only(
|
|
||||||
// top: isVisible ? 16.0 : 0,
|
|
||||||
// left: isVisible ? 42.0 : 0,
|
|
||||||
// right: isVisible ? 42.0 : 0,
|
|
||||||
// ),
|
|
||||||
padding: padding,
|
|
||||||
child: !isVisible
|
|
||||||
? const SizedBox.shrink()
|
|
||||||
: Stack(
|
|
||||||
children: [
|
|
||||||
if (participants.length > 1)
|
|
||||||
LeaderboardMedal(
|
|
||||||
participants[1],
|
|
||||||
color: Colors.grey[400]!,
|
|
||||||
radius: smallRadius ?? Avatar.defaultSize * 0.75,
|
|
||||||
iconSize: 16.0,
|
|
||||||
iconRadius: 10.0,
|
|
||||||
bottom: 0.0,
|
|
||||||
left: 0.0,
|
|
||||||
),
|
|
||||||
if (participants.isNotEmpty)
|
|
||||||
LeaderboardMedal(
|
|
||||||
participants[0],
|
|
||||||
color: AppConfig.gold,
|
|
||||||
radius: largeRadius ?? Avatar.defaultSize * 1.25,
|
|
||||||
iconSize: 20.0,
|
|
||||||
iconRadius: 16.0,
|
|
||||||
top: 0.0,
|
|
||||||
right: 0.0,
|
|
||||||
left: 0.0,
|
|
||||||
),
|
|
||||||
if (participants.length > 2)
|
|
||||||
LeaderboardMedal(
|
|
||||||
participants[2],
|
|
||||||
color: Colors.brown[400]!,
|
|
||||||
radius: smallRadius ?? Avatar.defaultSize * 0.75,
|
|
||||||
bottom: 0.0,
|
|
||||||
right: 0.0,
|
|
||||||
iconSize: 16.0,
|
|
||||||
iconRadius: 10.0,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class TrophyParticipantListItem extends StatelessWidget {
|
|
||||||
final int index;
|
|
||||||
final User user;
|
|
||||||
|
|
||||||
const TrophyParticipantListItem({
|
|
||||||
required this.index,
|
|
||||||
required this.user,
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return InkWell(
|
|
||||||
onTap: () => UserDialog.show(
|
|
||||||
context: context,
|
|
||||||
profile: Profile(
|
|
||||||
userId: user.id,
|
|
||||||
displayName: user.displayName,
|
|
||||||
avatarUrl: user.avatarUrl,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
alignment: Alignment.centerRight,
|
|
||||||
width: 32.0,
|
|
||||||
child: (index < 3)
|
|
||||||
? Icon(
|
|
||||||
Symbols.trophy,
|
|
||||||
color: index == 0
|
|
||||||
? AppConfig.gold
|
|
||||||
: index == 1
|
|
||||||
? Colors.grey[400]
|
|
||||||
: index == 2
|
|
||||||
? Colors.brown[400]
|
|
||||||
: null,
|
|
||||||
)
|
|
||||||
: null,
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: AbsorbPointer(
|
|
||||||
child: ParticipantListItem(user),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,96 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
|
|
||||||
import 'package:collection/collection.dart';
|
|
||||||
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
||||||
import 'package:matrix/matrix.dart';
|
|
||||||
|
|
||||||
import 'package:fluffychat/config/app_config.dart';
|
|
||||||
import 'package:fluffychat/config/themes.dart';
|
|
||||||
import 'package:fluffychat/pangea/bot/utils/bot_name.dart';
|
|
||||||
import 'package:fluffychat/pangea/spaces/pages/pangea_space_page_view.dart';
|
|
||||||
import 'package:fluffychat/pangea/spaces/utils/load_participants_util.dart';
|
|
||||||
import 'package:fluffychat/widgets/avatar.dart';
|
|
||||||
|
|
||||||
class SpaceViewLeaderboard extends StatefulWidget {
|
|
||||||
final Room space;
|
|
||||||
|
|
||||||
const SpaceViewLeaderboard({
|
|
||||||
required this.space,
|
|
||||||
super.key,
|
|
||||||
});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<SpaceViewLeaderboard> createState() => SpaceViewLeaderboardState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class SpaceViewLeaderboardState extends State<SpaceViewLeaderboard> {
|
|
||||||
bool _expanded = true;
|
|
||||||
|
|
||||||
void _toggleExpanded() {
|
|
||||||
setState(() => _expanded = !_expanded);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
if (FluffyThemes.isColumnMode(context)) {
|
|
||||||
return const SizedBox.shrink();
|
|
||||||
}
|
|
||||||
|
|
||||||
return LoadParticipantsUtil(
|
|
||||||
space: widget.space,
|
|
||||||
builder: (participantsLoader) {
|
|
||||||
final filteredParticipants = participantsLoader
|
|
||||||
.filteredParticipants("")
|
|
||||||
.where((u) => u.id != BotName.byEnvironment)
|
|
||||||
.toList();
|
|
||||||
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.center,
|
|
||||||
spacing: 16.0,
|
|
||||||
children: [
|
|
||||||
ListTile(
|
|
||||||
tileColor: Color.lerp(AppConfig.gold, Colors.black, 0.3),
|
|
||||||
contentPadding: const EdgeInsets.symmetric(horizontal: 16.0),
|
|
||||||
visualDensity: const VisualDensity(vertical: -4.0),
|
|
||||||
title: Text(
|
|
||||||
L10n.of(context).leaderboard,
|
|
||||||
style: Theme.of(context).textTheme.headlineSmall,
|
|
||||||
),
|
|
||||||
trailing: Icon(
|
|
||||||
_expanded
|
|
||||||
? Icons.keyboard_arrow_down_outlined
|
|
||||||
: Icons.keyboard_arrow_right_outlined,
|
|
||||||
),
|
|
||||||
onTap: _toggleExpanded,
|
|
||||||
),
|
|
||||||
if (_expanded)
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
SizedBox(
|
|
||||||
width: 225.0,
|
|
||||||
child: LeaderboardMedals(
|
|
||||||
isVisible: !participantsLoader.loading &&
|
|
||||||
filteredParticipants.isNotEmpty,
|
|
||||||
participants: filteredParticipants,
|
|
||||||
smallRadius: Avatar.defaultSize * 0.7,
|
|
||||||
largeRadius: Avatar.defaultSize,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Column(
|
|
||||||
children: filteredParticipants.take(3).mapIndexed(
|
|
||||||
(index, user) {
|
|
||||||
return TrophyParticipantListItem(
|
|
||||||
index: index,
|
|
||||||
user: filteredParticipants[index],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
).toList(),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Loading…
Reference in New Issue