import 'package:dropdown_button2/dropdown_button2.dart'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.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:styled_widget/styled_widget.dart'; import 'package:surface/providers/channel.dart'; import 'package:surface/providers/sn_network.dart'; import 'package:surface/providers/user_directory.dart'; import 'package:surface/providers/userinfo.dart'; import 'package:surface/types/chat.dart'; import 'package:surface/widgets/account/account_image.dart'; import 'package:surface/widgets/dialog.dart'; import 'package:surface/widgets/loading_indicator.dart'; import 'package:very_good_infinite_list/very_good_infinite_list.dart'; class ChannelDetailScreen extends StatefulWidget { final String scope; final String alias; const ChannelDetailScreen({ super.key, required this.scope, required this.alias, }); @override State createState() => _ChannelDetailScreenState(); } class _ChannelDetailScreenState extends State { bool _isBusy = false; SnChannel? _channel; SnChannelMember? _profile; Future _fetchChannel() async { setState(() => _isBusy = true); try { final chan = context.read(); _channel = await chan.getChannel('${widget.scope}:${widget.alias}'); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } Future _fetchChannelProfile() async { setState(() => _isBusy = true); try { final sn = context.read(); final resp = await sn.client .get('/cgi/im/channels/${_channel!.keyPath}/members/me'); _profile = SnChannelMember.fromJson(resp.data); _notifyLevel = _profile!.notify; if (!mounted) return; final ud = context.read(); await ud.getAccount(_profile!.accountId); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } Future _deleteChannel() async { final confirm = await context.showConfirmDialog( 'channelDelete'.tr(args: [_channel!.name]), 'channelDeleteDescription'.tr(), ); if (!confirm) return; if (!mounted) return; try { final sn = context.read(); await sn.client.delete( '/cgi/im/channels/${_channel!.realm?.alias ?? 'global'}/${_channel!.id}', ); if (!mounted) return; Navigator.pop(context, false); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } } Future _leaveChannel() async { final confirm = await context.showConfirmDialog( 'channelLeave'.tr(args: [_channel!.name]), 'channelLeaveDescription'.tr(), ); if (!confirm) return; if (!mounted) return; try { final sn = context.read(); await sn.client.delete( '/cgi/im/channels/${_channel!.realm?.alias ?? 'global'}/${_channel!.id}/members/me', ); if (!mounted) return; Navigator.pop(context, false); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } } int _notifyLevel = 0; bool _isUpdatingNotifyLevel = false; final kNotifyLevels = { 0: 'channelNotifyLevelAll'.tr(), 1: 'channelNotifyLevelMentioned'.tr(), 2: 'channelNotifyLevelNone'.tr(), }; Future _updateNotifyLevel(int value) async { if (_isUpdatingNotifyLevel) return; setState(() => _isUpdatingNotifyLevel = true); try { final sn = context.read(); await sn.client.put( '/cgi/im/channels/${_channel!.keyPath}/members/me/notify', data: {'notify_level': value}, ); _notifyLevel = value; if (!mounted) return; context.showSnackbar('channelNotifyLevelApplied'.tr()); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isUpdatingNotifyLevel = false); } } void _showChannelProfileDetail() { showDialog( context: context, builder: (context) => _ChannelProfileDetailDialog( channel: _channel!, current: _profile!, ), ).then((value) { if (value != null && mounted) { Navigator.pop(context, true); } }); } void _showMemberList() { showModalBottomSheet( context: context, builder: (context) => _ChannelMemberListWidget( channel: _channel!, ), ); } void _showMemberAdd() { showModalBottomSheet( context: context, builder: (context) => _NewChannelMemberWidget( channel: _channel!, ), ); } @override void initState() { super.initState(); _fetchChannel().then((_) { _fetchChannelProfile(); }); } @override Widget build(BuildContext context) { final ud = context.read(); final ua = context.read(); final isOwned = ua.isAuthorized && _channel?.accountId == ua.user?.id; return Scaffold( appBar: AppBar( title: _channel != null ? Text(_channel!.name) : Text('loading').tr(), ), body: SingleChildScrollView( child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ LoadingIndicator(isActive: _isBusy), const Gap(24), if (_channel != null) Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( _channel!.name, style: Theme.of(context).textTheme.titleMedium, ), Text( _channel!.description, style: Theme.of(context).textTheme.bodyMedium, ), ], ).padding(horizontal: 24), const Gap(16), const Divider(), const Gap(12), if (_profile != null) Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text('channelDetailPersonalRegion') .bold() .fontSize(17) .tr() .padding(horizontal: 20, bottom: 4), ListTile( leading: const Icon(Symbols.notifications), trailing: DropdownButtonHideUnderline( child: DropdownButton2( isExpanded: true, items: kNotifyLevels.entries .map((item) => DropdownMenuItem( enabled: !_isUpdatingNotifyLevel, value: item.key, child: Text( item.value, style: const TextStyle( fontSize: 14, ), ), )) .toList(), value: _notifyLevel, onChanged: (int? value) { if (value == null) return; _updateNotifyLevel(value); }, buttonStyleData: const ButtonStyleData( padding: EdgeInsets.only(left: 16, right: 1), height: 40, width: 140, ), menuItemStyleData: const MenuItemStyleData( height: 40, ), ), ), title: Text('channelNotifyLevel').tr(), subtitle: Text('channelNotifyLevelDescription').tr(), contentPadding: const EdgeInsets.only(left: 24, right: 20), ), ListTile( leading: AccountImage( content: ud.getAccountFromCache(_profile!.accountId)?.avatar, radius: 18, ), trailing: const Icon(Symbols.chevron_right), title: Text('channelEditProfile').tr(), subtitle: Text( (_profile?.nick?.isEmpty ?? true) ? ud.getAccountFromCache(_profile!.accountId)!.nick : _profile!.nick!, ), contentPadding: const EdgeInsets.only(left: 20, right: 20), onTap: _showChannelProfileDetail, ), if (!isOwned) ListTile( leading: const Icon(Symbols.exit_to_app), trailing: const Icon(Symbols.chevron_right), title: Text('channelActionLeave').tr(), subtitle: Text('channelActionLeaveDescription').tr(), contentPadding: const EdgeInsets.symmetric(horizontal: 24), onTap: _leaveChannel, ), ], ).padding(bottom: 16), Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text('channelDetailMemberRegion') .bold() .fontSize(17) .tr() .padding(horizontal: 20, bottom: 4), ListTile( leading: const Icon(Symbols.group), trailing: const Icon(Symbols.chevron_right), title: Text('channelMemberManage').tr(), subtitle: Text('channelMemberManageDescription').tr(), contentPadding: const EdgeInsets.symmetric(horizontal: 24), onTap: _showMemberList, ), ListTile( leading: const Icon(Symbols.group_add), trailing: const Icon(Symbols.chevron_right), title: Text('channelMemberAdd').tr(), subtitle: Text('channelMemberAddDescription').tr(), contentPadding: const EdgeInsets.symmetric(horizontal: 24), onTap: _showMemberAdd, ), ], ).padding(bottom: 16), Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text('channelDetailAdminRegion') .bold() .fontSize(17) .tr() .padding(horizontal: 20, bottom: 4), ListTile( leading: const Icon(Symbols.edit), trailing: const Icon(Symbols.chevron_right), title: Text('channelEdit').tr(), subtitle: Text('channelEditDescription').tr(), contentPadding: const EdgeInsets.symmetric(horizontal: 24), onTap: () { GoRouter.of(context).pushNamed( 'chatManage', queryParameters: {'editing': _channel!.keyPath}, ).then((value) { if (value != null && context.mounted) { Navigator.pop(context, value); } }); }, ), if (isOwned) ListTile( leading: const Icon(Symbols.delete), trailing: const Icon(Symbols.chevron_right), title: Text('channelActionDelete').tr(), subtitle: Text('channelActionDeleteDescription').tr(), contentPadding: const EdgeInsets.symmetric(horizontal: 24), onTap: _deleteChannel, ), ], ), ], ), ), ); } } class _ChannelProfileDetailDialog extends StatefulWidget { final SnChannel channel; final SnChannelMember current; const _ChannelProfileDetailDialog({ required this.channel, required this.current, }); @override State<_ChannelProfileDetailDialog> createState() => _ChannelProfileDetailDialogState(); } class _ChannelProfileDetailDialogState extends State<_ChannelProfileDetailDialog> { bool _isBusy = false; final TextEditingController _nickController = TextEditingController(); Future _updateProfile() async { setState(() => _isBusy = true); try { final sn = context.read(); await sn.client.put( '/cgi/im/channels/${widget.channel.keyPath}/members/me', data: {'nick': _nickController.text}, ); if (!mounted) return; Navigator.pop(context, true); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } @override void initState() { super.initState(); _nickController.text = widget.current.nick ?? ''; } @override void dispose() { _nickController.dispose(); super.dispose(); } @override Widget build(BuildContext context) { return AlertDialog( title: Text('channelProfileEdit').tr(), content: Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ TextField( controller: _nickController, decoration: InputDecoration( labelText: 'fieldChannelProfileNick'.tr(), helperText: 'fieldChannelProfileNickHint'.tr(), helperMaxLines: 2, ), onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), ), ], ), actions: [ TextButton( onPressed: _isBusy ? null : () => Navigator.pop(context), child: Text('dialogCancel').tr(), ), TextButton( onPressed: _isBusy ? null : _updateProfile, child: Text('apply').tr(), ), ], ); } } class _ChannelMemberListWidget extends StatefulWidget { final SnChannel channel; const _ChannelMemberListWidget({required this.channel}); @override State<_ChannelMemberListWidget> createState() => _ChannelMemberListWidgetState(); } class _ChannelMemberListWidgetState extends State<_ChannelMemberListWidget> { bool _isBusy = false; int? _totalCount; final List _members = List.empty(growable: true); Future _fetchMembers() async { setState(() => _isBusy = true); try { final ud = context.read(); final sn = context.read(); final resp = await sn.client.get( '/cgi/im/channels/${widget.channel.keyPath}/members', queryParameters: { 'take': 10, 'offset': 0, }); final out = List.from( resp.data['data']?.map((e) => SnChannelMember.fromJson(e)) ?? [], ); _totalCount = resp.data['count']; _members.addAll(out); await ud.listAccount(out.map((ele) => ele.accountId).toSet()); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } bool _isUpdating = false; Future _deleteMember(SnChannelMember member) async { if (_isUpdating) return; setState(() => _isUpdating = true); try { final sn = context.read(); await sn.client.delete( '/cgi/im/channels/${widget.channel.keyPath}/members/${member.id}', ); if (!mounted) return; _members.clear(); _fetchMembers(); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isUpdating = false); } } @override void initState() { super.initState(); _fetchMembers(); } @override Widget build(BuildContext context) { final ud = context.read(); return Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Row( crossAxisAlignment: CrossAxisAlignment.center, children: [ const Icon(Symbols.group, size: 24), const Gap(16), Text('channelMemberManage') .tr() .textStyle(Theme.of(context).textTheme.titleLarge!), ], ).padding(horizontal: 20, top: 16, bottom: 12), Expanded( child: RefreshIndicator( onRefresh: () { _members.clear(); return _fetchMembers(); }, child: InfiniteList( itemCount: _members.length, hasReachedMax: _totalCount != null && _members.length >= _totalCount!, isLoading: _isBusy, onFetchData: _fetchMembers, itemBuilder: (context, index) { final member = _members[index]; return ListTile( contentPadding: const EdgeInsets.only(right: 24, left: 16), leading: AccountImage( content: ud.getAccountFromCache(member.accountId)?.avatar, ), title: Text( ud.getAccountFromCache(member.accountId)?.name ?? 'unknown'.tr(), ), subtitle: Text(member.nick ?? 'unknown'.tr()), trailing: SizedBox( height: 48, width: 120, child: Row( mainAxisAlignment: MainAxisAlignment.end, children: [ IconButton( onPressed: _isUpdating ? null : () => _deleteMember(member), icon: const Icon(Symbols.person_remove), ), ], ), ), ); }, ), ), ), ], ); } } class _NewChannelMemberWidget extends StatefulWidget { final SnChannel channel; const _NewChannelMemberWidget({required this.channel}); @override State<_NewChannelMemberWidget> createState() => _NewChannelMemberWidgetState(); } class _NewChannelMemberWidgetState extends State<_NewChannelMemberWidget> { bool _isBusy = false; final TextEditingController _relatedController = TextEditingController(); Future _performAction() async { if (_relatedController.text.isEmpty) return; setState(() => _isBusy = true); try { final sn = context.read(); await sn.client.post( '/cgi/im/channels/${widget.channel.keyPath}/members', data: { 'related': _relatedController.text, }, ); if (!mounted) return; Navigator.pop(context, true); context.showSnackbar('channelMemberAdded'.tr()); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } @override void dispose() { super.dispose(); _relatedController.dispose(); } @override Widget build(BuildContext context) { return StyledWidget(Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( 'channelMemberAdd', style: Theme.of(context).textTheme.titleLarge, ).tr(), const Gap(12), TextField( controller: _relatedController, readOnly: _isBusy, autocorrect: false, autofocus: true, textCapitalization: TextCapitalization.none, decoration: InputDecoration( labelText: 'fieldMemberRelatedName'.tr(), suffix: SizedBox( height: 24, child: IconButton( onPressed: _isBusy ? null : () => _performAction(), icon: Icon(Symbols.send), visualDensity: const VisualDensity(horizontal: -4, vertical: -4), padding: EdgeInsets.zero, ), ), ), onTapOutside: (_) => FocusManager.instance.primaryFocus?.unfocus(), ) ], )).padding(all: 24); } }