import 'package:awesome_extensions/awesome_extensions.dart'; import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_translate/flutter_translate.dart'; import 'package:flutter_zoom_drawer/flutter_zoom_drawer.dart'; import '../../account_manager/account_manager.dart'; import '../../chat/chat.dart'; import '../../chat_list/chat_list.dart'; import '../../contacts/contacts.dart'; import '../../proto/proto.dart' as proto; import '../../theme/theme.dart'; class HomeAccountReady extends StatefulWidget { const HomeAccountReady({super.key}); @override State createState() => _HomeAccountReadyState(); } class _HomeAccountReadyState extends State { @override void initState() { super.initState(); } Widget buildMenuButton() => Builder(builder: (context) { final theme = Theme.of(context); final scale = theme.extension()!; final scaleConfig = theme.extension()!; return IconButton( icon: const Icon(Icons.menu), color: scaleConfig.preferBorders ? scale.primaryScale.border : scale.primaryScale.borderText, constraints: const BoxConstraints.expand(height: 48, width: 48), style: ButtonStyle( backgroundColor: WidgetStateProperty.all( scaleConfig.preferBorders ? scale.primaryScale.hoverElementBackground : scale.primaryScale.hoverBorder), shape: WidgetStateProperty.all( RoundedRectangleBorder( side: !scaleConfig.useVisualIndicators ? BorderSide.none : BorderSide( strokeAlign: BorderSide.strokeAlignCenter, color: scaleConfig.preferBorders ? scale.primaryScale.border : scale.primaryScale.borderText, width: 2), borderRadius: BorderRadius.all( Radius.circular(12 * scaleConfig.borderRadiusScale))), )), tooltip: translate('menu.accounts_menu_tooltip'), onPressed: () async { final ctrl = context.read(); await ctrl.toggle?.call(); }); }); Widget buildContactsButton() => Builder(builder: (context) { final theme = Theme.of(context); final scale = theme.extension()!; final scaleConfig = theme.extension()!; return IconButton( icon: const Icon(Icons.contacts), color: scaleConfig.preferBorders ? scale.primaryScale.border : scale.primaryScale.borderText, constraints: const BoxConstraints.expand(height: 48, width: 48), style: ButtonStyle( backgroundColor: WidgetStateProperty.all( scaleConfig.preferBorders ? scale.primaryScale.hoverElementBackground : scale.primaryScale.hoverBorder), shape: WidgetStateProperty.all( RoundedRectangleBorder( side: !scaleConfig.useVisualIndicators ? BorderSide.none : BorderSide( strokeAlign: BorderSide.strokeAlignCenter, color: scaleConfig.preferBorders ? scale.primaryScale.border : scale.primaryScale.borderText, width: 2), borderRadius: BorderRadius.all( Radius.circular(12 * scaleConfig.borderRadiusScale))), )), tooltip: translate('menu.contacts_tooltip'), onPressed: () async { await ContactsDialog.show(context); }); }); Widget buildUserPanel() => Builder(builder: (context) { final profile = context.select( (c) => c.state.asData!.value.profile); final theme = Theme.of(context); final scale = theme.extension()!; final scaleConfig = theme.extension()!; return ColoredBox( color: scaleConfig.preferBorders ? scale.primaryScale.subtleBackground : scale.primaryScale.subtleBorder, child: Column(children: [ Row(children: [ buildMenuButton().paddingLTRB(0, 0, 8, 0), ProfileWidget( profile: profile, showPronouns: false, ).expanded(), buildContactsButton().paddingLTRB(8, 0, 0, 0), ]).paddingAll(8), const ChatListWidget().expanded() ])); }); Widget buildLeftPane(BuildContext context) => Builder( builder: (context) => Material(color: Colors.transparent, child: buildUserPanel())); Widget buildRightPane(BuildContext context) { final activeChatLocalConversationKey = context.watch().state; if (activeChatLocalConversationKey == null) { return const NoConversationWidget(); } return ChatComponentWidget( localConversationRecordKey: activeChatLocalConversationKey, key: ValueKey(activeChatLocalConversationKey)); } @override Widget build(BuildContext context) { final isLarge = responsiveVisibility( context: context, phone: false, ); final theme = Theme.of(context); final scale = theme.extension()!; final scaleConfig = theme.extension()!; final activeChat = context.watch().state; final hasActiveChat = activeChat != null; return LayoutBuilder(builder: (context, constraints) { const leftColumnSize = 300.0; late final bool visibleLeft; late final bool visibleRight; late final double leftWidth; late final double rightWidth; if (isLarge) { visibleLeft = true; visibleRight = true; leftWidth = leftColumnSize; rightWidth = constraints.maxWidth - leftColumnSize - 2; } else { if (hasActiveChat) { visibleLeft = false; visibleRight = true; leftWidth = leftColumnSize; rightWidth = constraints.maxWidth; } else { visibleLeft = true; visibleRight = false; leftWidth = constraints.maxWidth; rightWidth = 400; // whatever } } return Row(crossAxisAlignment: CrossAxisAlignment.stretch, children: [ Offstage( offstage: !visibleLeft, child: ConstrainedBox( constraints: BoxConstraints(maxWidth: leftWidth), child: buildLeftPane(context))), Offstage( offstage: !(visibleLeft && visibleRight), child: SizedBox( width: 2, height: double.infinity, child: ColoredBox( color: scaleConfig.preferBorders ? scale.primaryScale.subtleBorder : scale.primaryScale.subtleBackground))), Offstage( offstage: !visibleRight, child: ConstrainedBox( constraints: BoxConstraints( maxHeight: constraints.maxHeight, maxWidth: rightWidth), child: buildRightPane(context), )), ]); }); } }