Better Member List #36

Manually merged
Henry-Hiles merged 7 commits from better-member-list into main 2026-06-05 18:39:14 -04:00
4 changed files with 114 additions and 43 deletions
Showing only changes of commit 69312e9b98 - Show all commits

improve UI of member list, sort by power level

Henry Hiles 2026-06-05 14:36:22 -04:00
Signed by: Henry-Hiles
SSH key fingerprint: SHA256:VKQUdS31Q90KvX7EkKMHMBpUspcmItAh86a+v7PGiIs

View file

@ -0,0 +1,49 @@
import "package:fast_immutable_collections/fast_immutable_collections.dart";
import "package:flutter_riverpod/flutter_riverpod.dart";
import "package:nexus/controllers/members_by_status_controller.dart";
import "package:nexus/controllers/rooms_controller.dart";
import "package:nexus/models/configs/members_by_status_config.dart";
import "package:nexus/models/content/content.dart";
import "package:nexus/models/content/power_levels.dart";
import "package:nexus/models/event.dart";
class MembersGroupedController extends AsyncNotifier<IMap<int, ISet<Event>>> {
final MembersByStatusConfig config;
MembersGroupedController(this.config);
@override
Future<IMap<int, ISet<Event>>> build() async {
final event = ref.watch(
RoomsController.provider.select((value) {
final room = value[config.roomId];
final eventRowId = room?.state[EventType.powerLevels.type]?[""];
return eventRowId == null ? null : room?.events[eventRowId];
}),
);
final content = event?.content is PowerLevelsContent
? event!.content as PowerLevelsContent
: PowerLevelsContent();
final members = await ref.watch(
MembersByStatusController.provider(config).future,
);
return members.fold<IMap<int, ISet<Event>>>(.new(), (result, event) {
final groupKey = content.users[event.stateKey!] ?? content.usersDefault;
return result.update(
groupKey,
(value) => value.add(event),
ifAbsent: () => .new({event}),
);
});
}
static final provider =
AsyncNotifierProvider.family<
MembersGroupedController,
IMap<int, ISet<Event>>,
MembersByStatusConfig
>(MembersGroupedController.new);
}

View file

@ -1,13 +1,14 @@
import "package:flutter/material.dart"; import "package:flutter/material.dart";
import "package:flutter_hooks/flutter_hooks.dart"; import "package:flutter_hooks/flutter_hooks.dart";
import "package:hooks_riverpod/hooks_riverpod.dart"; import "package:hooks_riverpod/hooks_riverpod.dart";
import "package:nexus/controllers/members_by_status_controller.dart"; import "package:material_segmented_list/material_segmented_list.dart";
import "package:nexus/controllers/members_grouped_controller.dart";
import "package:nexus/helpers/extensions/get_localpart.dart"; import "package:nexus/helpers/extensions/get_localpart.dart";
import "package:nexus/helpers/extensions/show_user_popover.dart";
import "package:nexus/helpers/extensions/string_to_color.dart"; import "package:nexus/helpers/extensions/string_to_color.dart";
import "package:nexus/models/content/membership.dart"; import "package:nexus/models/content/membership.dart";
import "package:nexus/models/membership_status.dart"; import "package:nexus/models/membership_status.dart";
import "package:nexus/widgets/avatar_or_hash.dart"; import "package:nexus/widgets/avatar_or_hash.dart";
import "package:nexus/widgets/divider_text.dart";
import "package:nexus/widgets/error_dialog.dart"; import "package:nexus/widgets/error_dialog.dart";
import "package:nexus/widgets/loading.dart"; import "package:nexus/widgets/loading.dart";
@ -18,11 +19,6 @@ class MemberList extends HookConsumerWidget {
@override @override
Widget build(BuildContext context, WidgetRef ref) { Widget build(BuildContext context, WidgetRef ref) {
final status = useState(MembershipStatus.join); final status = useState(MembershipStatus.join);
final membersProvider = ref.watch(
MembersByStatusController.provider(
.new(roomId: roomId, status: status.value),
),
);
return Drawer( return Drawer(
shape: Border(), shape: Border(),
@ -64,32 +60,41 @@ class MemberList extends HookConsumerWidget {
), ),
], ],
), ),
switch (membersProvider) { switch (ref.watch(
MembersGroupedController.provider(
.new(roomId: roomId, status: status.value),
),
)) {
AsyncError(:final error, :final stackTrace) => ErrorDialog( AsyncError(:final error, :final stackTrace) => ErrorDialog(
error, error,
stackTrace, stackTrace,
), ),
AsyncData(:final value) || AsyncLoading(:final value?) => Expanded( AsyncData(:final value) || AsyncLoading(:final value?) => Expanded(
child: ListView( child: ListView(
children: value padding: .all(12),
children: [
for (final MapEntry(key: powerLevel, value: members)
in value.toEntryIList(
compare: (a, b) => (b?.key ?? double.negativeInfinity)
.compareTo(a?.key ?? double.negativeInfinity),
)) ...[
DividerText("Power Level $powerLevel"),
SegmentedListSection(
children: members
.map( .map(
(member) => switch (member.content) { (member) => switch (member.content) {
MembershipContent( MembershipContent(
:final avatarUrl, :final avatarUrl,
:final displayName, :final displayName,
) => ) =>
InkWell( SegmentedListTile(
onTapUp: (details) => context.showUserPopover( onTap: () {},
member.content as MembershipContent, // context.showUserPopover(
member.stateKey!, // member.content as MembershipContent,
roomId: roomId, // member.stateKey!,
globalPosition: details.globalPosition, // roomId: roomId,
), // globalPosition: details.globalPosition,
child: ListTile( // ),
leading: AvatarOrHash(
avatarUrl,
displayName ?? member.sender.localpart,
),
title: Text( title: Text(
displayName ?? member.stateKey!.localpart, displayName ?? member.stateKey!.localpart,
overflow: .ellipsis, overflow: .ellipsis,
@ -102,13 +107,21 @@ class MemberList extends HookConsumerWidget {
member.stateKey!, member.stateKey!,
overflow: .ellipsis, overflow: .ellipsis,
), ),
leading: AvatarOrHash(
avatarUrl,
displayName ?? member.sender.localpart,
), ),
), ),
_ => SizedBox.shrink(), _ => throw Exception(
"Member content was not MembershipContent",
),
}, },
) )
.toList(), .toList(),
), ),
],
],
),
), ),
AsyncLoading _ => Loading(), AsyncLoading _ => Loading(),
}, },

View file

@ -760,6 +760,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "0.13.0" version: "0.13.0"
material_segmented_list:
dependency: "direct main"
description:
name: material_segmented_list
sha256: "384bfd41a78e745397ceff1dd39700961e6a5419ad911d1797bcc13ea3824241"
url: "https://pub.dev"
source: hosted
version: "1.0.5"
measure_size: measure_size:
dependency: "direct main" dependency: "direct main"
description: description:

View file

@ -64,6 +64,7 @@ dependencies:
media_kit_video: 2.0.1 media_kit_video: 2.0.1
media_kit_libs_video: 1.0.7 media_kit_libs_video: 1.0.7
measure_size: ^5.0.2 measure_size: ^5.0.2
material_segmented_list: ^1.0.5
dev_dependencies: dev_dependencies:
build_runner: 2.15.0 build_runner: 2.15.0