2021-10-26 16:50:34 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2022-03-01 19:14:49 +00:00
|
|
|
import 'package:flutter/services.dart';
|
2021-10-26 16:50:34 +00:00
|
|
|
|
|
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
2023-08-02 10:08:23 +00:00
|
|
|
import 'package:go_router/go_router.dart';
|
2022-03-01 19:14:49 +00:00
|
|
|
import 'package:keyboard_shortcuts/keyboard_shortcuts.dart';
|
2024-07-18 14:45:10 +00:00
|
|
|
import 'package:matrix/matrix.dart';
|
2021-10-26 16:50:34 +00:00
|
|
|
|
2024-07-15 19:14:49 +00:00
|
|
|
import 'package:fluffychat/config/app_config.dart';
|
|
|
|
import 'package:fluffychat/config/themes.dart';
|
2021-11-09 20:32:16 +00:00
|
|
|
import 'package:fluffychat/pages/chat_list/chat_list.dart';
|
2024-07-15 19:14:49 +00:00
|
|
|
import 'package:fluffychat/pages/chat_list/navi_rail_item.dart';
|
|
|
|
import 'package:fluffychat/utils/matrix_sdk_extensions/matrix_locals.dart';
|
2024-07-18 14:45:10 +00:00
|
|
|
import 'package:fluffychat/utils/stream_extension.dart';
|
2024-07-15 19:14:49 +00:00
|
|
|
import 'package:fluffychat/widgets/avatar.dart';
|
2021-10-26 16:50:34 +00:00
|
|
|
import '../../widgets/matrix.dart';
|
2022-05-01 11:03:33 +00:00
|
|
|
import 'chat_list_body.dart';
|
2021-04-14 12:09:46 +00:00
|
|
|
|
2021-05-22 07:13:47 +00:00
|
|
|
class ChatListView extends StatelessWidget {
|
2021-04-14 12:09:46 +00:00
|
|
|
final ChatListController controller;
|
|
|
|
|
2023-10-28 11:03:16 +00:00
|
|
|
const ChatListView(this.controller, {super.key});
|
2021-04-14 12:09:46 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-07-18 14:45:10 +00:00
|
|
|
final client = Matrix.of(context).client;
|
2022-01-29 11:35:03 +00:00
|
|
|
return StreamBuilder<Object?>(
|
2022-04-02 14:18:36 +00:00
|
|
|
stream: Matrix.of(context).onShareContentChanged.stream,
|
|
|
|
builder: (_, __) {
|
|
|
|
final selectMode = controller.selectMode;
|
2023-11-16 11:46:01 +00:00
|
|
|
return PopScope(
|
|
|
|
canPop: controller.selectMode == SelectMode.normal &&
|
2024-07-18 14:47:58 +00:00
|
|
|
!controller.isSearchMode &&
|
|
|
|
controller.activeSpaceId == null,
|
2024-07-15 13:19:50 +00:00
|
|
|
onPopInvoked: (pop) {
|
2023-11-16 11:46:01 +00:00
|
|
|
if (pop) return;
|
2024-07-18 14:47:58 +00:00
|
|
|
if (controller.activeSpaceId != null) {
|
|
|
|
controller.clearActiveSpace();
|
|
|
|
return;
|
|
|
|
}
|
2022-04-02 14:18:36 +00:00
|
|
|
final selMode = controller.selectMode;
|
2023-08-11 07:20:55 +00:00
|
|
|
if (controller.isSearchMode) {
|
|
|
|
controller.cancelSearch();
|
2023-11-16 11:46:01 +00:00
|
|
|
return;
|
2023-08-11 07:20:55 +00:00
|
|
|
}
|
2022-09-11 08:22:05 +00:00
|
|
|
if (selMode != SelectMode.normal) {
|
|
|
|
controller.cancelAction();
|
2023-11-16 11:46:01 +00:00
|
|
|
return;
|
2022-09-11 08:22:05 +00:00
|
|
|
}
|
2022-04-02 14:18:36 +00:00
|
|
|
},
|
2024-07-15 19:14:49 +00:00
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
if (FluffyThemes.isColumnMode(context) &&
|
|
|
|
controller.widget.displayNavigationRail) ...[
|
2024-07-18 14:45:10 +00:00
|
|
|
StreamBuilder(
|
|
|
|
key: ValueKey(
|
|
|
|
client.userID.toString(),
|
|
|
|
),
|
|
|
|
stream: client.onSync.stream
|
|
|
|
.where((s) => s.hasRoomUpdate)
|
|
|
|
.rateLimit(const Duration(seconds: 1)),
|
|
|
|
builder: (context, _) {
|
2024-07-15 19:14:49 +00:00
|
|
|
final allSpaces = Matrix.of(context)
|
|
|
|
.client
|
|
|
|
.rooms
|
|
|
|
.where((room) => room.isSpace);
|
|
|
|
final rootSpaces = allSpaces
|
|
|
|
.where(
|
|
|
|
(space) => !allSpaces.any(
|
|
|
|
(parentSpace) => parentSpace.spaceChildren
|
|
|
|
.any((child) => child.roomId == space.id),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.toList();
|
|
|
|
|
|
|
|
return SizedBox(
|
|
|
|
width: FluffyThemes.navRailWidth,
|
|
|
|
child: ListView.builder(
|
|
|
|
scrollDirection: Axis.vertical,
|
|
|
|
itemCount: rootSpaces.length + 2,
|
|
|
|
itemBuilder: (context, i) {
|
|
|
|
if (i == 0) {
|
|
|
|
return NaviRailItem(
|
|
|
|
isSelected: controller.activeSpaceId == null,
|
|
|
|
onTap: controller.clearActiveSpace,
|
|
|
|
icon: const Icon(Icons.forum_outlined),
|
|
|
|
selectedIcon: const Icon(Icons.forum),
|
|
|
|
toolTip: L10n.of(context)!.chats,
|
|
|
|
unreadBadgeFilter: (room) => true,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
i--;
|
|
|
|
if (i == rootSpaces.length) {
|
|
|
|
return NaviRailItem(
|
|
|
|
isSelected: false,
|
|
|
|
onTap: () => context.go('/rooms/newspace'),
|
|
|
|
icon: const Icon(Icons.add),
|
|
|
|
toolTip: L10n.of(context)!.createNewSpace,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
final space = rootSpaces[i];
|
|
|
|
final displayname =
|
|
|
|
rootSpaces[i].getLocalizedDisplayname(
|
|
|
|
MatrixLocals(L10n.of(context)!),
|
|
|
|
);
|
|
|
|
final spaceChildrenIds =
|
|
|
|
space.spaceChildren.map((c) => c.roomId).toSet();
|
|
|
|
return NaviRailItem(
|
|
|
|
toolTip: displayname,
|
|
|
|
isSelected: controller.activeSpaceId == space.id,
|
|
|
|
onTap: () =>
|
|
|
|
controller.setActiveSpace(rootSpaces[i].id),
|
|
|
|
unreadBadgeFilter: (room) =>
|
|
|
|
spaceChildrenIds.contains(room.id),
|
|
|
|
icon: Avatar(
|
|
|
|
mxContent: rootSpaces[i].avatar,
|
|
|
|
name: displayname,
|
|
|
|
size: 32,
|
|
|
|
borderRadius: BorderRadius.circular(
|
|
|
|
AppConfig.borderRadius / 4,
|
|
|
|
),
|
2023-03-02 09:57:52 +00:00
|
|
|
),
|
2024-07-15 19:14:49 +00:00
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
Container(
|
|
|
|
color: Theme.of(context).dividerColor,
|
|
|
|
width: 1,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
Expanded(
|
|
|
|
child: GestureDetector(
|
|
|
|
onTap: FocusManager.instance.primaryFocus?.unfocus,
|
|
|
|
excludeFromSemantics: true,
|
|
|
|
behavior: HitTestBehavior.translucent,
|
|
|
|
child: Scaffold(
|
|
|
|
body: ChatListViewBody(controller),
|
|
|
|
floatingActionButton: KeyBoardShortcuts(
|
|
|
|
keysToPress: {
|
|
|
|
LogicalKeyboardKey.controlLeft,
|
|
|
|
LogicalKeyboardKey.keyN,
|
|
|
|
},
|
|
|
|
onKeysPressed: () => context.go('/rooms/newprivatechat'),
|
|
|
|
helpLabel: L10n.of(context)!.newChat,
|
|
|
|
child: selectMode == SelectMode.normal &&
|
|
|
|
!controller.isSearchMode
|
|
|
|
? FloatingActionButton.extended(
|
|
|
|
onPressed: controller.addChatAction,
|
|
|
|
icon: const Icon(Icons.add_outlined),
|
|
|
|
label: Text(
|
|
|
|
L10n.of(context)!.chat,
|
|
|
|
overflow: TextOverflow.fade,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
: const SizedBox.shrink(),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
2022-12-26 15:02:45 +00:00
|
|
|
),
|
2024-07-15 19:14:49 +00:00
|
|
|
],
|
2022-12-26 15:02:45 +00:00
|
|
|
),
|
2022-04-02 14:18:36 +00:00
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
2021-04-14 12:09:46 +00:00
|
|
|
}
|
|
|
|
}
|