chore: Add unread badge to navigation rail and adjust design

This commit is contained in:
Christian Pauly 2022-09-11 11:07:04 +02:00
parent dc0f067b0a
commit d6c7dadb24
8 changed files with 171 additions and 142 deletions

View File

@ -4,7 +4,6 @@ import 'package:flutter/services.dart';
import 'package:vrouter/vrouter.dart'; import 'package:vrouter/vrouter.dart';
import 'package:fluffychat/utils/platform_infos.dart'; import 'package:fluffychat/utils/platform_infos.dart';
import '../widgets/matrix.dart';
import 'app_config.dart'; import 'app_config.dart';
abstract class FluffyThemes { abstract class FluffyThemes {
@ -16,9 +15,7 @@ abstract class FluffyThemes {
isColumnModeByWidth(MediaQuery.of(context).size.width); isColumnModeByWidth(MediaQuery.of(context).size.width);
static bool getDisplayNavigationRail(BuildContext context) => static bool getDisplayNavigationRail(BuildContext context) =>
!VRouter.of(context).path.startsWith('/settings') && !VRouter.of(context).path.startsWith('/settings');
(Matrix.of(context).client.rooms.any((room) => room.isSpace) ||
AppConfig.separateChatTypes);
static const fallbackTextStyle = TextStyle( static const fallbackTextStyle = TextStyle(
fontFamily: 'Roboto', fontFamily: 'Roboto',

View File

@ -1,5 +1,6 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:badges/badges.dart';
import 'package:desktop_drop/desktop_drop.dart'; import 'package:desktop_drop/desktop_drop.dart';
import 'package:flutter_gen/gen_l10n/l10n.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart';
import 'package:future_loading_dialog/future_loading_dialog.dart'; import 'package:future_loading_dialog/future_loading_dialog.dart';
@ -19,7 +20,7 @@ import 'package:fluffychat/pages/chat/tombstone_display.dart';
import 'package:fluffychat/widgets/chat_settings_popup_menu.dart'; import 'package:fluffychat/widgets/chat_settings_popup_menu.dart';
import 'package:fluffychat/widgets/connection_status_header.dart'; import 'package:fluffychat/widgets/connection_status_header.dart';
import 'package:fluffychat/widgets/matrix.dart'; import 'package:fluffychat/widgets/matrix.dart';
import 'package:fluffychat/widgets/unread_badge_back_button.dart'; import 'package:fluffychat/widgets/unread_rooms_badge.dart';
import '../../utils/stream_extension.dart'; import '../../utils/stream_extension.dart';
import '../../widgets/m2_popup_menu_button.dart'; import '../../widgets/m2_popup_menu_button.dart';
import 'chat_emoji_picker.dart'; import 'chat_emoji_picker.dart';
@ -179,7 +180,11 @@ class ChatView extends StatelessWidget {
tooltip: L10n.of(context)!.close, tooltip: L10n.of(context)!.close,
color: Theme.of(context).colorScheme.primary, color: Theme.of(context).colorScheme.primary,
) )
: UnreadBadgeBackButton(roomId: controller.roomId!), : UnreadRoomsBadge(
filter: (r) => r.id != controller.roomId!,
badgePosition: BadgePosition.topEnd(end: 8, top: 4),
child: const Center(child: BackButton()),
),
titleSpacing: 0, titleSpacing: 0,
title: ChatAppBarTitle(controller), title: ChatAppBarTitle(controller),
actions: _appBarActions(context), actions: _appBarActions(context),

View File

@ -101,64 +101,55 @@ class ChatListController extends State<ChatList>
} }
} }
void onDestinationSelected(int? i) { ActiveFilter getActiveFilterByDestination(int? i) {
switch (i) { switch (i) {
case 0:
if (AppConfig.separateChatTypes) {
setState(() {
activeFilter = ActiveFilter.groups;
});
} else {
setState(() {
activeFilter = ActiveFilter.allChats;
});
}
break;
case 1: case 1:
if (AppConfig.separateChatTypes) { if (AppConfig.separateChatTypes) {
setState(() { return ActiveFilter.messages;
activeFilter = ActiveFilter.messages;
});
} else {
setState(() {
activeFilter = ActiveFilter.spaces;
});
} }
break; return ActiveFilter.spaces;
case 2: case 2:
setState(() { return ActiveFilter.spaces;
activeFilter = ActiveFilter.spaces; case 0:
}); default:
break; if (AppConfig.separateChatTypes) {
return ActiveFilter.groups;
}
return ActiveFilter.allChats;
} }
} }
void onDestinationSelected(int? i) {
setState(() {
activeFilter = getActiveFilterByDestination(i);
});
}
ActiveFilter activeFilter = AppConfig.separateChatTypes ActiveFilter activeFilter = AppConfig.separateChatTypes
? ActiveFilter.messages ? ActiveFilter.messages
: ActiveFilter.allChats; : ActiveFilter.allChats;
List<Room> get filteredRooms { bool Function(Room) getRoomFilterByActiveFilter(ActiveFilter activeFilter) {
final rooms = Matrix.of(context).client.rooms;
switch (activeFilter) { switch (activeFilter) {
case ActiveFilter.allChats: case ActiveFilter.allChats:
return rooms return (room) => !room.isSpace && !room.isStoryRoom;
.where((room) => !room.isSpace && !room.isStoryRoom)
.toList();
case ActiveFilter.groups: case ActiveFilter.groups:
return rooms return (room) =>
.where((room) => !room.isSpace && !room.isDirectChat && !room.isStoryRoom;
!room.isSpace && !room.isDirectChat && !room.isStoryRoom)
.toList();
case ActiveFilter.messages: case ActiveFilter.messages:
return rooms return (room) =>
.where((room) => !room.isSpace && room.isDirectChat && !room.isStoryRoom;
!room.isSpace && room.isDirectChat && !room.isStoryRoom)
.toList();
case ActiveFilter.spaces: case ActiveFilter.spaces:
return rooms.where((room) => room.isSpace).toList(); return (r) => r.isSpace;
} }
} }
List<Room> get filteredRooms => Matrix.of(context)
.client
.rooms
.where(getRoomFilterByActiveFilter(activeFilter))
.toList();
bool isSearchMode = false; bool isSearchMode = false;
Future<QueryPublicRoomsResponse>? publicRoomsResponse; Future<QueryPublicRoomsResponse>? publicRoomsResponse;
String? searchServer; String? searchServer;

View File

@ -1,6 +1,7 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:badges/badges.dart';
import 'package:flutter_gen/gen_l10n/l10n.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart';
import 'package:keyboard_shortcuts/keyboard_shortcuts.dart'; import 'package:keyboard_shortcuts/keyboard_shortcuts.dart';
import 'package:vrouter/vrouter.dart'; import 'package:vrouter/vrouter.dart';
@ -9,6 +10,7 @@ import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/config/themes.dart'; import 'package:fluffychat/config/themes.dart';
import 'package:fluffychat/pages/chat_list/chat_list.dart'; import 'package:fluffychat/pages/chat_list/chat_list.dart';
import 'package:fluffychat/widgets/avatar.dart'; import 'package:fluffychat/widgets/avatar.dart';
import 'package:fluffychat/widgets/unread_rooms_badge.dart';
import '../../widgets/matrix.dart'; import '../../widgets/matrix.dart';
import 'chat_list_body.dart'; import 'chat_list_body.dart';
import 'chat_list_header.dart'; import 'chat_list_header.dart';
@ -19,32 +21,62 @@ class ChatListView extends StatelessWidget {
const ChatListView(this.controller, {Key? key}) : super(key: key); const ChatListView(this.controller, {Key? key}) : super(key: key);
List<NavigationDestination> getNavigationDestinations(BuildContext context) => List<NavigationDestination> getNavigationDestinations(BuildContext context) {
[ final badgePosition = BadgePosition.topEnd(top: -12, end: -8);
if (AppConfig.separateChatTypes) ...[ return [
NavigationDestination( if (AppConfig.separateChatTypes) ...[
icon: const Icon(Icons.groups_outlined), NavigationDestination(
selectedIcon: const Icon(Icons.groups), icon: UnreadRoomsBadge(
label: L10n.of(context)!.groups, badgePosition: badgePosition,
filter: controller.getRoomFilterByActiveFilter(ActiveFilter.groups),
child: const Icon(Icons.groups_outlined),
), ),
NavigationDestination( selectedIcon: UnreadRoomsBadge(
icon: const Icon(Icons.chat_outlined), badgePosition: badgePosition,
selectedIcon: const Icon(Icons.chat), filter: controller.getRoomFilterByActiveFilter(ActiveFilter.groups),
label: L10n.of(context)!.messages, child: const Icon(Icons.groups),
), ),
] else label: L10n.of(context)!.groups,
NavigationDestination( ),
icon: const Icon(Icons.chat_outlined), NavigationDestination(
selectedIcon: const Icon(Icons.chat), icon: UnreadRoomsBadge(
label: L10n.of(context)!.chats, badgePosition: badgePosition,
filter:
controller.getRoomFilterByActiveFilter(ActiveFilter.messages),
child: const Icon(Icons.chat_outlined),
), ),
if (controller.spaces.isNotEmpty) selectedIcon: UnreadRoomsBadge(
const NavigationDestination( badgePosition: badgePosition,
icon: Icon(Icons.workspaces_outlined), filter:
selectedIcon: Icon(Icons.workspaces), controller.getRoomFilterByActiveFilter(ActiveFilter.messages),
label: 'Spaces', child: const Icon(Icons.chat),
), ),
]; label: L10n.of(context)!.messages,
),
] else
NavigationDestination(
icon: UnreadRoomsBadge(
badgePosition: badgePosition,
filter:
controller.getRoomFilterByActiveFilter(ActiveFilter.allChats),
child: const Icon(Icons.chat_outlined),
),
selectedIcon: UnreadRoomsBadge(
badgePosition: badgePosition,
filter:
controller.getRoomFilterByActiveFilter(ActiveFilter.allChats),
child: const Icon(Icons.chat),
),
label: L10n.of(context)!.chats,
),
if (controller.spaces.isNotEmpty)
const NavigationDestination(
icon: Icon(Icons.workspaces_outlined),
selectedIcon: Icon(Icons.workspaces),
label: 'Spaces',
),
];
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -102,11 +134,6 @@ class ChatListView extends StatelessWidget {
height: 64, height: 64,
width: 64, width: 64,
decoration: BoxDecoration( decoration: BoxDecoration(
color: isSelected
? Theme.of(context)
.colorScheme
.secondaryContainer
: Theme.of(context).colorScheme.background,
border: Border( border: Border(
bottom: i == (destinations.length - 1) bottom: i == (destinations.length - 1)
? BorderSide( ? BorderSide(
@ -129,15 +156,21 @@ class ChatListView extends StatelessWidget {
alignment: Alignment.center, alignment: Alignment.center,
child: IconButton( child: IconButton(
color: isSelected color: isSelected
? Theme.of(context).colorScheme.primary ? Theme.of(context).colorScheme.secondary
: null, : null,
icon: CircleAvatar( icon: CircleAvatar(
backgroundColor: Theme.of(context) backgroundColor: isSelected
.colorScheme ? Theme.of(context).colorScheme.secondary
.secondaryContainer, : Theme.of(context)
foregroundColor: Theme.of(context) .colorScheme
.colorScheme .background,
.onSecondaryContainer, foregroundColor: isSelected
? Theme.of(context)
.colorScheme
.onSecondary
: Theme.of(context)
.colorScheme
.onBackground,
child: i == controller.selectedIndex child: i == controller.selectedIndex
? destinations[i].selectedIcon ?? ? destinations[i].selectedIcon ??
destinations[i].icon destinations[i].icon
@ -156,15 +189,10 @@ class ChatListView extends StatelessWidget {
height: 64, height: 64,
width: 64, width: 64,
decoration: BoxDecoration( decoration: BoxDecoration(
color: isSelected
? Theme.of(context)
.colorScheme
.secondaryContainer
: Theme.of(context).colorScheme.background,
border: Border( border: Border(
left: BorderSide( left: BorderSide(
color: isSelected color: isSelected
? Theme.of(context).colorScheme.primary ? Theme.of(context).colorScheme.secondary
: Colors.transparent, : Colors.transparent,
width: 4, width: 4,
), ),

View File

@ -1,58 +0,0 @@
import 'package:flutter/material.dart';
import 'package:matrix/matrix.dart';
import '../config/app_config.dart';
import 'matrix.dart';
class UnreadBadgeBackButton extends StatelessWidget {
final String roomId;
const UnreadBadgeBackButton({
Key? key,
required this.roomId,
}) : super(key: key);
@override
Widget build(BuildContext context) {
return Stack(
children: [
StreamBuilder(
stream: Matrix.of(context).client.onSync.stream,
builder: (context, _) {
final unreadCount = Matrix.of(context)
.client
.rooms
.where((r) =>
r.id != roomId &&
(r.isUnread || r.membership == Membership.invite))
.length;
return unreadCount > 0
? Align(
alignment: Alignment.bottomRight,
child: Container(
padding: const EdgeInsets.all(4),
margin: const EdgeInsets.only(bottom: 4, right: 8),
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.primaryContainer,
borderRadius:
BorderRadius.circular(AppConfig.borderRadius),
),
child: Text(
'$unreadCount',
style: TextStyle(
fontSize: 12,
color: Theme.of(context)
.colorScheme
.onPrimaryContainer,
),
),
),
)
: Container();
}),
const Center(child: BackButton()),
],
);
}
}

View File

@ -0,0 +1,58 @@
import 'package:flutter/material.dart';
import 'package:badges/badges.dart';
import 'package:matrix/matrix.dart';
import 'matrix.dart';
class UnreadRoomsBadge extends StatelessWidget {
final bool Function(Room) filter;
final BadgePosition? badgePosition;
final Widget? child;
const UnreadRoomsBadge({
Key? key,
required this.filter,
this.badgePosition,
this.child,
}) : super(key: key);
@override
Widget build(BuildContext context) {
return StreamBuilder(
stream: Matrix.of(context)
.client
.onSync
.stream
.where((syncUpdate) => syncUpdate.hasRoomUpdate),
builder: (context, _) {
final unreadCount = Matrix.of(context)
.client
.rooms
.where(filter)
.where((r) => (r.isUnread || r.membership == Membership.invite))
.length;
return Badge(
alignment: Alignment.bottomRight,
badgeContent: Text(
unreadCount.toString(),
style: TextStyle(
color: Theme.of(context).colorScheme.onPrimary,
fontSize: 12,
),
),
showBadge: unreadCount != 0,
animationType: BadgeAnimationType.scale,
badgeColor: Theme.of(context).colorScheme.primary,
position: badgePosition,
elevation: 4,
borderSide: BorderSide(
color: Theme.of(context).colorScheme.background,
width: 2,
strokeAlign: StrokeAlign.outside,
),
child: child,
);
});
}
}

View File

@ -85,6 +85,13 @@ packages:
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
source: hosted source: hosted
version: "0.1.6+1" version: "0.1.6+1"
badges:
dependency: "direct main"
description:
name: badges
url: "https://pub.dartlang.org"
source: hosted
version: "2.0.3"
base58check: base58check:
dependency: transitive dependency: transitive
description: description:

View File

@ -11,6 +11,7 @@ dependencies:
adaptive_theme: ^3.0.0 adaptive_theme: ^3.0.0
animations: ^2.0.2 animations: ^2.0.2
async: ^2.8.2 async: ^2.8.2
badges: ^2.0.3
blurhash_dart: ^1.1.0 blurhash_dart: ^1.1.0
callkeep: ^0.3.2 callkeep: ^0.3.2
chewie: ^1.2.2 chewie: ^1.2.2