173 lines
6.3 KiB
Dart
173 lines
6.3 KiB
Dart
import 'package:easy_localization/easy_localization.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:gap/gap.dart';
|
|
import 'package:go_router/go_router.dart';
|
|
import 'package:material_symbols_icons/symbols.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'package:relative_time/relative_time.dart';
|
|
import 'package:styled_widget/styled_widget.dart';
|
|
import 'package:surface/providers/experience.dart';
|
|
import 'package:surface/providers/sn_network.dart';
|
|
import 'package:surface/types/account.dart';
|
|
import 'package:surface/widgets/account/account_image.dart';
|
|
import 'package:surface/widgets/account/badge.dart';
|
|
import 'package:surface/widgets/universal_image.dart';
|
|
|
|
class AccountPopoverCard extends StatelessWidget {
|
|
final SnAccount data;
|
|
|
|
const AccountPopoverCard({super.key, required this.data});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final sn = context.read<SnNetworkProvider>();
|
|
|
|
return SingleChildScrollView(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
if (data.banner.isNotEmpty)
|
|
ClipRRect(
|
|
borderRadius: BorderRadius.circular(16),
|
|
child: Container(
|
|
color: Theme.of(context).colorScheme.surfaceContainer,
|
|
child: AspectRatio(
|
|
aspectRatio: 16 / 7,
|
|
child: AutoResizeUniversalImage(
|
|
sn.getAttachmentUrl(data.banner),
|
|
fit: BoxFit.cover,
|
|
),
|
|
),
|
|
),
|
|
).padding(all: 16)
|
|
else
|
|
const Gap(16),
|
|
// Top padding
|
|
Row(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
children: [
|
|
AccountImage(
|
|
content: data.avatar,
|
|
radius: 20,
|
|
),
|
|
Gap(16),
|
|
Expanded(
|
|
child: Column(
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
Text(data.nick).bold(),
|
|
Text('@${data.name}').fontSize(13).opacity(0.75),
|
|
],
|
|
),
|
|
),
|
|
IconButton(
|
|
onPressed: () {
|
|
Navigator.pop(context);
|
|
GoRouter.of(context).pushReplacementNamed(
|
|
'accountProfilePage',
|
|
pathParameters: {'name': data.name},
|
|
);
|
|
},
|
|
icon: const Icon(Symbols.chevron_right),
|
|
padding: EdgeInsets.zero,
|
|
visualDensity:
|
|
const VisualDensity(horizontal: -4, vertical: -4),
|
|
),
|
|
const Gap(8)
|
|
],
|
|
).padding(horizontal: 16),
|
|
if (data.badges.isNotEmpty)
|
|
Wrap(
|
|
spacing: 4,
|
|
children: data.badges
|
|
.map(
|
|
(ele) => AccountBadge(badge: ele),
|
|
)
|
|
.toList(),
|
|
).padding(horizontal: 24, bottom: 12, top: 12),
|
|
if (data.profile?.description.isNotEmpty ?? false)
|
|
Text(
|
|
data.profile?.description ?? '',
|
|
maxLines: 2,
|
|
overflow: TextOverflow.ellipsis,
|
|
).padding(horizontal: 26, bottom: 8)
|
|
else
|
|
const Gap(12),
|
|
Row(
|
|
crossAxisAlignment: CrossAxisAlignment.center,
|
|
children: [
|
|
const Icon(Symbols.star),
|
|
const Gap(8),
|
|
Text('Lv${getLevelFromExp(data.profile?.experience ?? 0)}'),
|
|
const Gap(8),
|
|
Text(calcLevelUpProgressLevel(data.profile?.experience ?? 0))
|
|
.fontSize(11)
|
|
.opacity(0.5),
|
|
const Gap(8),
|
|
Container(
|
|
width: double.infinity,
|
|
constraints: const BoxConstraints(maxWidth: 160),
|
|
child: LinearProgressIndicator(
|
|
value: calcLevelUpProgress(data.profile?.experience ?? 0),
|
|
borderRadius: BorderRadius.circular(8),
|
|
backgroundColor:
|
|
Theme.of(context).colorScheme.surfaceContainer,
|
|
).alignment(Alignment.centerLeft),
|
|
),
|
|
],
|
|
).padding(horizontal: 24),
|
|
FutureBuilder(
|
|
future: sn.client.get('/cgi/id/users/${data.name}/status'),
|
|
builder: (context, snapshot) {
|
|
final SnAccountStatusInfo? status = snapshot.hasData
|
|
? SnAccountStatusInfo.fromJson(snapshot.data!.data)
|
|
: null;
|
|
return Row(
|
|
children: [
|
|
Icon(
|
|
(status?.isDisturbable ?? true)
|
|
? Symbols.circle
|
|
: Symbols.do_not_disturb_on,
|
|
fill: (status?.isOnline ?? false) ? 1 : 0,
|
|
size: 16,
|
|
color: (status?.isOnline ?? false)
|
|
? (status?.isDisturbable ?? true)
|
|
? Colors.green
|
|
: Colors.red
|
|
: Colors.grey,
|
|
).padding(all: 4),
|
|
const Gap(8),
|
|
Text(
|
|
status != null
|
|
? (status.status?.label.isNotEmpty ?? false)
|
|
? status.status!.label
|
|
: status.isOnline
|
|
? 'accountStatusOnline'.tr()
|
|
: 'accountStatusOffline'.tr()
|
|
: 'loading'.tr(),
|
|
),
|
|
if (status != null &&
|
|
!status.isOnline &&
|
|
status.lastSeenAt != null)
|
|
Text(
|
|
'accountStatusLastSeen'.tr(args: [
|
|
status.lastSeenAt != null
|
|
? RelativeTime(context).format(
|
|
status.lastSeenAt!.toLocal(),
|
|
)
|
|
: 'unknown',
|
|
]),
|
|
).padding(left: 6).opacity(0.75),
|
|
],
|
|
).padding(horizontal: 24);
|
|
},
|
|
),
|
|
// Bottom padding
|
|
const Gap(64),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|