import 'dart:async'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:adaptive_dialog/adaptive_dialog.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart'; import 'package:future_loading_dialog/future_loading_dialog.dart'; import 'package:go_router/go_router.dart'; import 'package:keyboard_shortcuts/keyboard_shortcuts.dart'; import 'package:matrix/matrix.dart'; import 'matrix.dart'; class ChatSettingsPopupMenu extends StatefulWidget { final Room room; final bool displayChatDetails; const ChatSettingsPopupMenu(this.room, this.displayChatDetails, {super.key}); @override ChatSettingsPopupMenuState createState() => ChatSettingsPopupMenuState(); } class ChatSettingsPopupMenuState extends State { StreamSubscription? notificationChangeSub; @override void dispose() { notificationChangeSub?.cancel(); super.dispose(); } @override Widget build(BuildContext context) { notificationChangeSub ??= Matrix.of(context) .client .onAccountData .stream .where((u) => u.type == 'm.push_rules') .listen( (u) => setState(() {}), ); final items = >[ widget.room.pushRuleState == PushRuleState.notify ? PopupMenuItem( value: 'mute', child: Row( children: [ const Icon(Icons.notifications_off_outlined), const SizedBox(width: 12), Text(L10n.of(context)!.muteChat), ], ), ) : PopupMenuItem( value: 'unmute', child: Row( children: [ const Icon(Icons.notifications_on_outlined), const SizedBox(width: 12), Text(L10n.of(context)!.unmuteChat), ], ), ), PopupMenuItem( value: 'leave', child: Row( children: [ const Icon(Icons.delete_outlined), const SizedBox(width: 12), Text(L10n.of(context)!.leave), ], ), ), ]; if (widget.displayChatDetails) { items.insert( 0, PopupMenuItem( value: 'details', child: Row( children: [ const Icon(Icons.info_outline_rounded), const SizedBox(width: 12), Text(L10n.of(context)!.chatDetails), ], ), ), ); } return Stack( alignment: Alignment.center, children: [ KeyBoardShortcuts( keysToPress: { LogicalKeyboardKey.controlLeft, LogicalKeyboardKey.keyI, }, helpLabel: L10n.of(context)!.chatDetails, onKeysPressed: _showChatDetails, child: const SizedBox.shrink(), ), PopupMenuButton( onSelected: (String choice) async { switch (choice) { case 'leave': final confirmed = await showOkCancelAlertDialog( useRootNavigator: false, context: context, title: L10n.of(context)!.areYouSure, okLabel: L10n.of(context)!.ok, cancelLabel: L10n.of(context)!.cancel, message: L10n.of(context)!.archiveRoomDescription, ); if (confirmed == OkCancelResult.ok) { final success = await showFutureLoadingDialog( context: context, future: () => widget.room.leave(), ); if (success.error == null) { context.go('/rooms'); } } break; case 'mute': await showFutureLoadingDialog( context: context, future: () => widget.room.setPushRuleState(PushRuleState.mentionsOnly), ); break; case 'unmute': await showFutureLoadingDialog( context: context, future: () => widget.room.setPushRuleState(PushRuleState.notify), ); break; case 'todos': context.go('/rooms/${widget.room.id}/tasks'); break; case 'details': _showChatDetails(); break; } }, itemBuilder: (BuildContext context) => items, ), ], ); } void _showChatDetails() { if (GoRouterState.of(context).uri.path.endsWith('/details')) { context.go('/rooms/${widget.room.id}'); } else { context.go('/rooms/${widget.room.id}/details'); } } }