Solian/lib/screens/account.dart

288 lines
8.3 KiB
Dart
Raw Normal View History

import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
import 'package:solian/providers/auth.dart';
2024-04-21 12:55:17 +00:00
import 'package:solian/router.dart';
2024-05-02 04:51:16 +00:00
import 'package:solian/screens/account/friend.dart';
2024-05-02 10:56:40 +00:00
import 'package:solian/screens/account/personalize.dart';
2024-04-26 12:49:21 +00:00
import 'package:solian/widgets/account/avatar.dart';
2024-04-25 15:03:16 +00:00
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
2024-05-02 04:51:16 +00:00
import 'package:solian/widgets/empty.dart';
import 'package:solian/widgets/indent_wrapper.dart';
class AccountScreen extends StatefulWidget {
const AccountScreen({super.key});
@override
State<AccountScreen> createState() => _AccountScreenState();
}
class _AccountScreenState extends State<AccountScreen> {
2024-05-02 04:51:16 +00:00
String? _title;
String? _selectedTab;
@override
Widget build(BuildContext context) {
final screenWidth = MediaQuery.of(context).size.width;
final isLargeScreen = screenWidth >= 600;
Widget renderContent() {
switch (_selectedTab) {
case 'account.friend':
return const FriendScreenWidget();
2024-05-02 10:56:40 +00:00
case 'account.personalize':
return const PersonalizeScreenWidget();
2024-05-02 04:51:16 +00:00
default:
return const SelectionEmptyWidget();
}
}
return IndentWrapper(
title: _title ?? AppLocalizations.of(context)!.account,
noSafeArea: true,
fixedAppBarColor: true,
child: isLargeScreen
? Row(
children: [
Flexible(
flex: 2,
child: AccountScreenWidget(
onSelect: (item, title) {
setState(() {
_selectedTab = item;
_title = title;
});
},
),
),
const VerticalDivider(thickness: 0.3, width: 0.3),
Flexible(flex: 4, child: renderContent()),
],
)
: AccountScreenWidget(
onSelect: (item, _) {
router.pushNamed(item);
},
),
);
}
}
class AccountScreenWidget extends StatefulWidget {
final Function(String item, String title) onSelect;
const AccountScreenWidget({super.key, required this.onSelect});
@override
State<AccountScreenWidget> createState() => _AccountScreenWidgetState();
}
class _AccountScreenWidgetState extends State<AccountScreenWidget> {
bool _isAuthorized = false;
@override
void initState() {
Future.delayed(Duration.zero, () async {
var authorized = await context.read<AuthProvider>().isAuthorized();
2024-05-02 04:51:16 +00:00
setState(() => _isAuthorized = authorized);
});
super.initState();
}
@override
Widget build(BuildContext context) {
final auth = context.watch<AuthProvider>();
2024-05-02 04:51:16 +00:00
if (_isAuthorized) {
return Column(
children: [
const Padding(
2024-05-02 10:56:40 +00:00
padding: EdgeInsets.only(top: 16, bottom: 8, left: 24, right: 24),
2024-05-02 04:51:16 +00:00
child: NameCard(),
),
2024-05-02 10:56:40 +00:00
ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 34),
leading: const Icon(Icons.color_lens),
title: Text(AppLocalizations.of(context)!.personalize),
onTap: () {
widget.onSelect('account.personalize', AppLocalizations.of(context)!.personalize);
},
),
2024-05-02 04:51:16 +00:00
ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 34),
leading: const Icon(Icons.diversity_1),
title: Text(AppLocalizations.of(context)!.friend),
onTap: () {
widget.onSelect('account.friend', AppLocalizations.of(context)!.friend);
},
),
ListTile(
contentPadding: const EdgeInsets.symmetric(horizontal: 34),
leading: const Icon(Icons.logout),
title: Text(AppLocalizations.of(context)!.signOut),
onTap: () {
auth.signoff();
setState(() {
_isAuthorized = false;
});
},
),
],
);
} else {
return Center(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
ActionCard(
icon: const Icon(Icons.login, color: Colors.white),
title: AppLocalizations.of(context)!.signIn,
caption: AppLocalizations.of(context)!.signInCaption,
onTap: () {
router.pushNamed('auth.sign-in').then((did) {
auth.isAuthorized().then((value) {
setState(() => _isAuthorized = value);
});
});
},
),
2024-05-02 04:51:16 +00:00
ActionCard(
icon: const Icon(Icons.add, color: Colors.white),
title: AppLocalizations.of(context)!.signUp,
caption: AppLocalizations.of(context)!.signUpCaption,
onTap: () {
router.pushNamed('auth.sign-up');
},
),
],
),
);
}
}
}
class NameCard extends StatelessWidget {
const NameCard({super.key});
2024-04-14 07:58:27 +00:00
Future<Widget> renderAvatar(BuildContext context) async {
final auth = context.read<AuthProvider>();
final profiles = await auth.getProfiles();
2024-05-01 16:49:38 +00:00
return AccountAvatar(source: profiles['picture'], direct: true);
}
2024-04-14 07:58:27 +00:00
Future<Column> renderLabel(BuildContext context) async {
final auth = context.read<AuthProvider>();
final profiles = await auth.getProfiles();
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
2024-05-01 16:49:38 +00:00
profiles['nick'],
2024-05-02 10:56:40 +00:00
maxLines: 1,
style: const TextStyle(
fontSize: 20,
fontWeight: FontWeight.bold,
2024-05-02 10:56:40 +00:00
overflow: TextOverflow.ellipsis,
),
),
2024-05-02 10:56:40 +00:00
Text(
profiles['email'],
maxLines: 1,
style: const TextStyle(overflow: TextOverflow.ellipsis),
)
],
);
}
@override
Widget build(BuildContext context) {
return Card(
child: InkWell(
splashColor: Colors.indigo.withAlpha(30),
child: Padding(
padding: const EdgeInsets.all(20),
child: Row(
children: [
FutureBuilder(
2024-04-14 07:58:27 +00:00
future: renderAvatar(context),
2024-05-02 04:51:16 +00:00
builder: (BuildContext context, AsyncSnapshot<Widget> snapshot) {
if (snapshot.hasData) {
return snapshot.data!;
} else {
return const CircularProgressIndicator();
}
},
),
const SizedBox(width: 20),
FutureBuilder(
2024-04-14 07:58:27 +00:00
future: renderLabel(context),
2024-05-02 04:51:16 +00:00
builder: (BuildContext context, AsyncSnapshot<Column> snapshot) {
if (snapshot.hasData) {
2024-05-02 10:56:40 +00:00
return Expanded(child: snapshot.data!);
} else {
return const Column();
}
},
)
],
),
),
),
);
}
}
class ActionCard extends StatelessWidget {
final Widget icon;
final String title;
final String caption;
final Function onTap;
2024-05-02 10:56:40 +00:00
const ActionCard({
super.key,
required this.onTap,
required this.title,
required this.caption,
required this.icon,
});
@override
Widget build(BuildContext context) {
return Card(
child: InkWell(
borderRadius: BorderRadius.circular(10),
onTap: () => onTap(),
child: Container(
width: 320,
padding: const EdgeInsets.all(20),
child: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Padding(
padding: const EdgeInsets.only(bottom: 12),
child: CircleAvatar(
backgroundColor: Colors.indigo,
child: icon,
),
),
Text(
title,
style: const TextStyle(
fontSize: 18,
fontWeight: FontWeight.w900,
2024-05-02 10:56:40 +00:00
overflow: TextOverflow.clip,
),
),
2024-05-02 10:56:40 +00:00
Text(
caption,
style: const TextStyle(overflow: TextOverflow.clip),
),
],
),
),
),
);
}
}