import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:flutter_expandable_fab/flutter_expandable_fab.dart'; import 'package:gap/gap.dart'; import 'package:go_router/go_router.dart'; import 'package:material_symbols_icons/symbols.dart'; import 'package:provider/provider.dart'; import 'package:surface/providers/channel.dart'; import 'package:surface/providers/user_directory.dart'; import 'package:surface/types/chat.dart'; import 'package:surface/widgets/account/account_image.dart'; import 'package:surface/widgets/account/account_select.dart'; import 'package:surface/widgets/app_bar_leading.dart'; import 'package:surface/widgets/dialog.dart'; import 'package:surface/widgets/loading_indicator.dart'; import 'package:uuid/uuid.dart'; import '../providers/sn_network.dart'; import '../providers/userinfo.dart'; class ChatScreen extends StatefulWidget { const ChatScreen({super.key}); @override State createState() => _ChatScreenState(); } class _ChatScreenState extends State { final _fabKey = GlobalKey(); bool _isBusy = true; List? _channels; Map? _lastMessages; void _refreshChannels() { final chan = context.read(); chan.fetchChannels().listen((channels) async { final lastMessages = await chan.getLastMessages(channels); _lastMessages = {for (final val in lastMessages) val.channelId: val}; channels.sort((a, b) { if (_lastMessages!.containsKey(a.id) && _lastMessages!.containsKey(b.id)) { return _lastMessages![b.id]!.createdAt.compareTo(_lastMessages![a.id]!.createdAt); } if (_lastMessages!.containsKey(a.id)) return -1; if (_lastMessages!.containsKey(b.id)) return 1; return 0; }); if (!mounted) return; final ud = context.read(); for (final channel in channels) { if (channel.type == 1) { await ud.listAccount( channel.members ?.cast() .map((ele) => ele?.accountId) .where((ele) => ele != null) .toSet() ?? {}, ); } } if (mounted) setState(() => _channels = channels); }) ..onError((err) { if (!mounted) return; context.showErrorDialog(err); setState(() => _isBusy = false); }) ..onDone(() { if (!mounted) return; setState(() => _isBusy = false); }); } void _newDirectMessage() async { final user = await showModalBottomSheet( context: context, builder: (context) => AccountSelect(title: 'channelNewDirectMessage'.tr()), ); if (user == null) return; if (!mounted) return; try { const uuid = Uuid(); final sn = context.read(); final ua = context.read(); final resp = await sn.client.post('/cgi/im/channels/global/dm', data: { 'alias': uuid.v4().replaceAll('-', '').substring(0, 12), 'name': 'DM', 'description': 'A direct message channel between @${ua.user?.name} and @${user.name}', 'related_user': user.id, }); _fabKey.currentState!.toggle(); _refreshChannels(); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } } @override void initState() { super.initState(); _refreshChannels(); } @override Widget build(BuildContext context) { final ud = context.read(); final ua = context.read(); return Scaffold( appBar: AppBar( leading: AutoAppBarLeading(), title: Text('screenChat').tr(), ), floatingActionButtonLocation: ExpandableFab.location, floatingActionButton: ExpandableFab( key: _fabKey, distance: 75, type: ExpandableFabType.up, childrenAnimation: ExpandableFabAnimation.none, overlayStyle: ExpandableFabOverlayStyle( color: Theme.of(context).colorScheme.surface.withAlpha((255 * 0.5).round()), ), openButtonBuilder: RotateFloatingActionButtonBuilder( child: const Icon(Symbols.add, size: 28), fabSize: ExpandableFabSize.regular, foregroundColor: Theme.of(context).floatingActionButtonTheme.foregroundColor, backgroundColor: Theme.of(context).floatingActionButtonTheme.backgroundColor, shape: const CircleBorder(), ), closeButtonBuilder: DefaultFloatingActionButtonBuilder( child: const Icon(Symbols.close, size: 28), fabSize: ExpandableFabSize.regular, foregroundColor: Theme.of(context).floatingActionButtonTheme.foregroundColor, backgroundColor: Theme.of(context).floatingActionButtonTheme.backgroundColor, shape: const CircleBorder(), ), children: [ Row( children: [ Text('channelNewChannel').tr(), const Gap(20), FloatingActionButton( heroTag: null, tooltip: 'channelNewChannel'.tr(), onPressed: () { _fabKey.currentState!.toggle(); GoRouter.of(context).pushNamed('chatManage').then((value) { if (value != null && context.mounted) _refreshChannels(); }); }, child: const Icon(Symbols.chat_add_on), ), ], ), Row( children: [ Text('channelNewDirectMessage').tr(), const Gap(20), FloatingActionButton( heroTag: null, tooltip: 'channelNewDirectMessage'.tr(), onPressed: _newDirectMessage, child: const Icon(Symbols.communication), ), ], ), ], ), body: Column( children: [ LoadingIndicator(isActive: _isBusy), Expanded( child: RefreshIndicator( onRefresh: () => Future.sync(() => _refreshChannels()), child: ListView.builder( itemCount: _channels?.length ?? 0, itemBuilder: (context, idx) { final channel = _channels![idx]; final lastMessage = _lastMessages?[channel.id]; if (channel.type == 1) { final otherMember = channel.members?.cast().firstWhere( (ele) => ele?.accountId != ua.user?.id, orElse: () => null, ); return ListTile( title: Text(ud.getAccountFromCache(otherMember?.accountId)?.nick ?? channel.name), subtitle: lastMessage != null ? Text( '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', maxLines: 1, overflow: TextOverflow.ellipsis, ) : Text( 'channelDirectMessageDescription'.tr(args: [ '@${ud.getAccountFromCache(otherMember?.accountId)?.name}', ]), maxLines: 1, overflow: TextOverflow.ellipsis, ), contentPadding: const EdgeInsets.symmetric(horizontal: 16), leading: AccountImage( content: ud.getAccountFromCache(otherMember?.accountId)?.avatar, ), onTap: () { GoRouter.of(context).pushNamed( 'chatRoom', pathParameters: { 'scope': channel.realm?.alias ?? 'global', 'alias': channel.alias, }, ).then((value) { if (value == true) _refreshChannels(); }); }, ); } return ListTile( title: Text(channel.name), subtitle: lastMessage != null ? Text( '${ud.getAccountFromCache(lastMessage.sender.accountId)?.nick}: ${lastMessage.body['text'] ?? 'Unable preview'}', maxLines: 1, overflow: TextOverflow.ellipsis, ) : Text( channel.description, maxLines: 1, overflow: TextOverflow.ellipsis, ), contentPadding: const EdgeInsets.symmetric(horizontal: 16), leading: AccountImage( content: null, fallbackWidget: const Icon(Symbols.chat, size: 20), ), onTap: () { GoRouter.of(context).pushNamed( 'chatRoom', pathParameters: { 'scope': channel.realm?.alias ?? 'global', 'alias': channel.alias, }, ).then((value) { if (value == true) _refreshChannels(); }); }, ); }, ), ), ), ], ), ); } }