Solian/lib/widgets/navigation/app_navigation_drawer.dart

250 lines
8.0 KiB
Dart
Raw Normal View History

2024-08-10 16:36:27 +00:00
import 'dart:math' as math;
2024-07-12 03:39:44 +00:00
import 'package:flutter/material.dart';
2024-07-12 05:15:46 +00: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-08-02 15:15:28 +00:00
import 'package:solian/providers/relation.dart';
2024-07-12 03:39:44 +00:00
import 'package:solian/router.dart';
import 'package:solian/shells/root_shell.dart';
import 'package:solian/theme.dart';
2024-07-12 05:15:46 +00:00
import 'package:solian/widgets/account/account_avatar.dart';
import 'package:solian/widgets/account/account_status_action.dart';
2024-07-12 03:39:44 +00:00
import 'package:solian/widgets/navigation/app_navigation.dart';
2024-07-12 05:15:46 +00:00
import 'package:badges/badges.dart' as badges;
2024-08-07 10:24:16 +00:00
import 'package:solian/widgets/navigation/app_navigation_regions.dart';
2024-07-12 03:39:44 +00:00
class AppNavigationDrawer extends StatefulWidget {
2024-07-12 05:15:46 +00:00
final String? routeName;
const AppNavigationDrawer({super.key, this.routeName});
2024-07-12 03:39:44 +00:00
@override
State<AppNavigationDrawer> createState() => _AppNavigationDrawerState();
}
class _AppNavigationDrawerState extends State<AppNavigationDrawer> {
2024-08-21 09:00:59 +00:00
bool _isCollapsed = true;
2024-07-12 05:15:46 +00:00
AccountStatus? _accountStatus;
2024-07-31 05:29:26 +00:00
Future<void> _getStatus() async {
2024-07-12 05:15:46 +00:00
final StatusProvider provider = Get.find();
final resp = await provider.getCurrentStatus();
final status = AccountStatus.fromJson(resp.body);
if (mounted) {
setState(() {
_accountStatus = status;
});
}
2024-07-12 05:15:46 +00:00
}
2024-08-21 09:00:59 +00:00
Widget _buildUserInfo() {
return Obx(() {
final AuthProvider auth = Get.find();
if (auth.isAuthorized.isFalse || auth.userProfile.value == null) {
if (_isCollapsed) {
return InkWell(
child: const Icon(Icons.account_circle).paddingAll(28),
onTap: () {
AppRouter.instance.goNamed('account');
_closeDrawer();
},
);
}
2024-07-12 05:15:46 +00:00
2024-08-21 09:00:59 +00:00
return ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 28),
leading: const Icon(Icons.account_circle),
title: !_isCollapsed ? Text('guest'.tr) : null,
subtitle: !_isCollapsed ? Text('unsignedIn'.tr) : null,
onTap: () {
AppRouter.instance.goNamed('account');
_closeDrawer();
},
);
}
2024-07-12 05:15:46 +00:00
2024-08-21 09:00:59 +00:00
final leading = Obx(() {
final statusBadgeColor = _accountStatus != null
? StatusProvider.determineStatus(
_accountStatus!,
).$2
: Colors.grey;
2024-07-12 05:15:46 +00:00
2024-08-21 09:00:59 +00:00
final RelationshipProvider relations = Get.find();
final accountNotifications = relations.friendRequestCount.value;
return badges.Badge(
badgeContent: Text(
accountNotifications.toString(),
style: const TextStyle(color: Colors.white),
),
showBadge: accountNotifications > 0,
position: badges.BadgePosition.topEnd(
top: -10,
end: -6,
),
child: badges.Badge(
showBadge: _accountStatus != null,
badgeStyle: badges.BadgeStyle(badgeColor: statusBadgeColor),
position: badges.BadgePosition.bottomEnd(
bottom: 0,
end: -2,
),
child: AccountAvatar(
content: auth.userProfile.value!['avatar'],
),
),
);
});
2024-08-07 10:24:16 +00:00
2024-08-21 09:00:59 +00:00
return InkWell(
child: !_isCollapsed
? ListTile(
2024-08-09 14:59:24 +00:00
contentPadding: const EdgeInsets.only(left: 20, right: 20),
2024-08-07 10:24:16 +00:00
title: Text(
auth.userProfile.value!['nick'],
2024-07-24 17:18:47 +00:00
maxLines: 1,
overflow: TextOverflow.fade,
2024-08-02 15:15:28 +00:00
),
2024-08-07 10:24:16 +00:00
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,
);
},
2024-07-24 17:18:47 +00:00
),
2024-08-21 09:00:59 +00:00
leading: leading,
)
: leading.paddingAll(20),
onTap: () {
AppRouter.instance.goNamed('account');
_closeDrawer();
},
onLongPress: () {
showModalBottomSheet(
useRootNavigator: true,
context: context,
builder: (context) => AccountStatusAction(
currentStatus: _accountStatus!.status,
),
).then((val) {
if (val == true) _getStatus();
});
},
);
});
}
void _closeDrawer() {
rootScaffoldKey.currentState!.closeDrawer();
}
@override
void initState() {
super.initState();
_getStatus();
}
@override
Widget build(BuildContext context) {
return Drawer(
width: _isCollapsed ? 80 : null,
backgroundColor:
SolianTheme.isLargeScreen(context) ? Colors.transparent : null,
child: SafeArea(
bottom: false,
child: Column(
children: [
_buildUserInfo().paddingSymmetric(vertical: 8),
2024-08-07 10:24:16 +00:00
const Divider(thickness: 0.3, height: 1),
Column(
children: AppNavigation.destinations
.map(
2024-08-21 09:00:59 +00:00
(e) => _isCollapsed
? InkWell(
child: Icon(e.icon, size: 22).paddingSymmetric(
horizontal: 28,
vertical: 16,
),
onTap: () {
AppRouter.instance.goNamed(e.page);
_closeDrawer();
},
)
: ListTile(
contentPadding: const EdgeInsets.symmetric(
horizontal: 20,
),
leading: Icon(e.icon, size: 20).paddingAll(2),
title: !_isCollapsed ? Text(e.label) : null,
enabled: true,
onTap: () {
AppRouter.instance.goNamed(e.page);
_closeDrawer();
},
),
2024-08-07 10:24:16 +00:00
)
.toList(),
2024-08-21 09:00:59 +00:00
),
2024-08-07 10:24:16 +00:00
const Divider(thickness: 0.3, height: 1),
Expanded(
child: AppNavigationRegions(
2024-08-21 09:00:59 +00:00
isCollapsed: _isCollapsed,
2024-08-07 10:24:16 +00:00
onSelected: (item) {
_closeDrawer();
},
2024-07-24 17:18:47 +00:00
),
2024-08-07 10:24:16 +00:00
),
2024-08-09 16:51:54 +00:00
const Divider(thickness: 0.3, height: 1),
Column(
children: [
2024-08-21 09:00:59 +00:00
_isCollapsed
? InkWell(
child: const Icon(Icons.settings, size: 22)
.paddingSymmetric(
horizontal: 28,
vertical: 16,
),
onTap: () {
AppRouter.instance.pushNamed('settings');
_closeDrawer();
},
)
: ListTile(
minTileHeight: 0,
contentPadding: const EdgeInsets.symmetric(
horizontal: 20,
),
leading:
const Icon(Icons.settings, size: 20).paddingAll(2),
title: Text('settings'.tr),
onTap: () {
AppRouter.instance.pushNamed('settings');
_closeDrawer();
},
),
2024-08-09 16:51:54 +00:00
],
2024-08-10 16:36:27 +00:00
).paddingOnly(
top: 8,
bottom: math.max(8, MediaQuery.of(context).padding.bottom),
),
2024-08-07 10:24:16 +00:00
],
),
),
2024-07-12 03:39:44 +00:00
);
}
}