Solian/lib/widgets/navigation/app_navigation_drawer.dart

228 lines
7.0 KiB
Dart
Raw Normal View History

2024-07-12 11:39:44 +08:00
import 'package:flutter/material.dart';
2024-07-12 13:15:46 +08:00
import 'package:get/get.dart';
import 'package:solian/models/account_status.dart';
import 'package:solian/providers/account_status.dart';
import 'package:solian/providers/auth.dart';
2024-07-12 21:59:16 +08:00
import 'package:solian/providers/content/channel.dart';
2024-07-12 11:39:44 +08:00
import 'package:solian/router.dart';
import 'package:solian/shells/root_shell.dart';
import 'package:solian/theme.dart';
2024-07-12 13:15:46 +08:00
import 'package:solian/widgets/account/account_avatar.dart';
import 'package:solian/widgets/account/account_status_action.dart';
2024-07-12 21:59:16 +08:00
import 'package:solian/widgets/channel/channel_list.dart';
2024-07-12 11:39:44 +08:00
import 'package:solian/widgets/navigation/app_navigation.dart';
2024-07-12 13:15:46 +08:00
import 'package:badges/badges.dart' as badges;
2024-07-12 11:39:44 +08:00
class AppNavigationDrawer extends StatefulWidget {
2024-07-12 13:15:46 +08:00
final String? routeName;
const AppNavigationDrawer({super.key, this.routeName});
2024-07-12 11:39:44 +08:00
@override
State<AppNavigationDrawer> createState() => _AppNavigationDrawerState();
}
class _AppNavigationDrawerState extends State<AppNavigationDrawer> {
2024-07-12 16:19:54 +08:00
int? _selectedIndex = 0;
2024-07-12 21:59:16 +08:00
2024-07-12 13:15:46 +08:00
AccountStatus? _accountStatus;
2024-07-12 21:59:16 +08:00
late final ChannelProvider _channels;
2024-07-31 13:29:26 +08:00
Future<void> _getStatus() async {
2024-07-12 13:15:46 +08:00
final StatusProvider provider = Get.find();
final resp = await provider.getCurrentStatus();
final status = AccountStatus.fromJson(resp.body);
setState(() {
_accountStatus = status;
});
}
2024-07-31 13:29:26 +08:00
void _detectSelectedIndex() {
2024-07-12 13:15:46 +08:00
if (widget.routeName == null) return;
final nameList = AppNavigation.destinations.map((x) => x.page).toList();
final idx = nameList.indexOf(widget.routeName!);
2024-07-12 16:19:54 +08:00
_selectedIndex = idx != -1 ? idx : null;
}
2024-07-31 13:29:26 +08:00
void _closeDrawer() {
2024-07-12 16:19:54 +08:00
rootScaffoldKey.currentState!.closeDrawer();
2024-07-12 13:15:46 +08:00
}
2024-07-31 13:29:26 +08:00
Widget _buildSettingButton() {
return IconButton(
icon: const Icon(Icons.settings),
onPressed: () {
AppRouter.instance.pushNamed('settings');
setState(() => _selectedIndex = null);
_closeDrawer();
}
);
}
2024-07-12 13:15:46 +08:00
@override
void initState() {
super.initState();
2024-07-12 21:59:16 +08:00
_channels = Get.find();
2024-07-31 13:29:26 +08:00
_detectSelectedIndex();
_getStatus();
2024-07-12 13:15:46 +08:00
}
@override
void didChangeDependencies() {
super.didChangeDependencies();
2024-07-31 13:29:26 +08:00
_detectSelectedIndex();
2024-07-12 13:15:46 +08:00
}
2024-07-12 11:39:44 +08:00
@override
Widget build(BuildContext context) {
2024-07-12 13:15:46 +08:00
final AuthProvider auth = Get.find();
2024-07-12 11:39:44 +08:00
return NavigationDrawer(
backgroundColor:
SolianTheme.isLargeScreen(context) ? Colors.transparent : null,
2024-07-12 11:39:44 +08:00
selectedIndex: _selectedIndex,
onDestinationSelected: (idx) {
setState(() => _selectedIndex = idx);
AppRouter.instance.goNamed(AppNavigation.destinations[idx].page);
2024-07-31 13:29:26 +08:00
_closeDrawer();
2024-07-12 11:39:44 +08:00
},
children: [
2024-07-25 01:18:47 +08:00
Obx(() {
2024-07-26 14:21:00 +08:00
if (auth.isAuthorized.isFalse || auth.userProfile.value == null) {
2024-07-12 16:19:54 +08:00
return ListTile(
2024-07-25 01:18:47 +08:00
contentPadding: const EdgeInsets.symmetric(horizontal: 28),
leading: const Icon(Icons.account_circle),
title: Text('guest'.tr),
subtitle: Text('unsignedIn'.tr),
2024-07-31 13:29:26 +08:00
trailing: _buildSettingButton(),
2024-07-12 16:19:54 +08:00
onTap: () {
AppRouter.instance.goNamed('account');
setState(() => _selectedIndex = null);
2024-07-31 13:29:26 +08:00
_closeDrawer();
2024-07-12 16:19:54 +08:00
},
);
2024-07-25 01:18:47 +08:00
}
return ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 24),
title: Text(
auth.userProfile.value!['nick'],
maxLines: 1,
overflow: TextOverflow.fade,
),
subtitle: Builder(
builder: (context) {
if (_accountStatus == null) {
return Text('loading'.tr);
}
final info = StatusProvider.determineStatus(
_accountStatus!,
);
return Text(
info.$3,
maxLines: 1,
overflow: TextOverflow.fade,
);
},
),
leading: Builder(builder: (context) {
final badgeColor = _accountStatus != null
? StatusProvider.determineStatus(
_accountStatus!,
).$2
: Colors.grey;
return badges.Badge(
showBadge: _accountStatus != null,
badgeStyle: badges.BadgeStyle(badgeColor: badgeColor),
position: badges.BadgePosition.bottomEnd(
bottom: 0,
end: -2,
),
child: AccountAvatar(
content: auth.userProfile.value!['avatar'],
),
);
}),
2024-07-31 13:29:26 +08:00
trailing: _buildSettingButton(),
2024-07-25 01:18:47 +08:00
onTap: () {
AppRouter.instance.goNamed('account');
setState(() => _selectedIndex = null);
2024-07-31 13:29:26 +08:00
_closeDrawer();
2024-07-25 01:18:47 +08:00
},
2024-07-31 02:44:49 +08:00
onLongPress: () {
showModalBottomSheet(
useRootNavigator: true,
context: context,
builder: (context) => AccountStatusAction(
currentStatus: _accountStatus!.status,
),
).then((val) {
2024-07-31 13:29:26 +08:00
if (val == true) _getStatus();
2024-07-31 02:44:49 +08:00
});
},
2024-07-25 01:18:47 +08:00
);
}).paddingOnly(top: 8),
2024-07-12 16:19:54 +08:00
const Divider(thickness: 0.3, height: 1).paddingOnly(
bottom: 12,
top: 8,
),
2024-07-12 11:39:44 +08:00
...AppNavigation.destinations.map(
(e) => NavigationDrawerDestination(
icon: e.icon,
label: Text(e.label),
),
2024-07-12 16:19:54 +08:00
),
const Divider(thickness: 0.3, height: 1).paddingOnly(
top: 12,
2024-07-12 21:59:16 +08:00
),
2024-07-25 01:18:47 +08:00
Obx(() {
2024-07-26 14:21:00 +08:00
if (auth.isAuthorized.isFalse || auth.userProfile.value == null) {
2024-07-25 01:18:47 +08:00
return const SizedBox();
}
final selfId = auth.userProfile.value!['id'];
return Column(
children: [
Theme(
data: Theme.of(context)
.copyWith(dividerColor: Colors.transparent),
child: ExpansionTile(
title: Text('channels'.tr),
tilePadding: const EdgeInsets.symmetric(horizontal: 24),
children: [
Obx(
() => SizedBox(
height: 360,
child: RefreshIndicator(
onRefresh: () => _channels.refreshAvailableChannel(),
child: ChannelListWidget(
channels: _channels.groupChannels,
selfId: selfId,
isDense: true,
useReplace: true,
onSelected: (_) {
setState(() => _selectedIndex = null);
2024-07-31 13:29:26 +08:00
_closeDrawer();
2024-07-25 01:18:47 +08:00
},
),
2024-07-12 21:59:16 +08:00
),
),
2024-07-25 01:18:47 +08:00
),
],
2024-07-12 21:59:16 +08:00
),
2024-07-25 01:18:47 +08:00
),
],
);
}),
2024-07-12 11:39:44 +08:00
],
);
}
}