Solian/lib/widgets/channel/channel_list.dart

278 lines
8.5 KiB
Dart
Raw Normal View History

import 'dart:math';
2024-05-29 15:22:24 +00:00
import 'package:flutter/material.dart';
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
import 'package:gap/gap.dart';
2024-05-29 15:22:24 +00:00
import 'package:get/get.dart';
2024-10-05 15:12:23 +00:00
import 'package:intl/intl.dart';
2024-08-01 16:54:19 +00:00
import 'package:solian/controllers/chat_events_controller.dart';
2024-05-29 15:22:24 +00:00
import 'package:solian/models/channel.dart';
2024-08-01 16:54:19 +00:00
import 'package:solian/platform.dart';
2024-10-05 07:11:48 +00:00
import 'package:solian/providers/database/database.dart';
2024-05-29 15:22:24 +00:00
import 'package:solian/router.dart';
import 'package:solian/widgets/account/account_avatar.dart';
2024-10-05 15:12:23 +00:00
import 'package:badges/badges.dart' as badges;
2024-05-29 15:22:24 +00:00
2024-05-30 14:02:54 +00:00
class ChannelListWidget extends StatefulWidget {
2024-05-29 15:22:24 +00:00
final List<Channel> channels;
final int selfId;
2024-07-12 13:59:16 +00:00
final bool useReplace;
final Function(Channel)? onSelected;
2024-05-29 15:22:24 +00:00
2024-05-30 14:02:54 +00:00
const ChannelListWidget({
super.key,
required this.channels,
required this.selfId,
2024-07-12 13:59:16 +00:00
this.useReplace = false,
this.onSelected,
2024-05-30 14:02:54 +00:00
});
2024-05-29 15:22:24 +00:00
@override
2024-05-30 14:02:54 +00:00
State<ChannelListWidget> createState() => _ChannelListWidgetState();
}
2024-05-29 15:22:24 +00:00
2024-05-30 14:02:54 +00:00
class _ChannelListWidgetState extends State<ChannelListWidget> {
2024-10-05 07:11:48 +00:00
Map<int, LocalMessageEventTableData>? _lastMessages;
2024-08-01 16:54:19 +00:00
final ChatEventController _eventController = ChatEventController();
2024-10-05 07:11:48 +00:00
Future<void> _loadLastMessages() async {
final messages = await _eventController.src.getLastInAllChannels();
setState(() {
_lastMessages = messages
.map((k, v) => MapEntry(k, v.firstOrNull))
.cast<int, LocalMessageEventTableData>();
});
}
2024-06-22 18:25:45 +00:00
@override
void initState() {
super.initState();
2024-10-05 07:11:48 +00:00
_eventController.initialize().then((_) {
_loadLastMessages();
});
2024-06-22 18:25:45 +00:00
}
2024-08-01 16:54:19 +00:00
void _gotoChannel(Channel item) {
2024-07-12 13:59:16 +00:00
if (widget.useReplace) {
AppRouter.instance.pushReplacementNamed(
'channelChat',
pathParameters: {'alias': item.alias},
queryParameters: {
if (item.realmId != null) 'realm': item.realm!.alias,
},
);
} else {
AppRouter.instance.pushNamed(
'channelChat',
pathParameters: {'alias': item.alias},
queryParameters: {
if (item.realmId != null) 'realm': item.realm!.alias,
},
);
}
if (widget.onSelected != null) {
widget.onSelected!(item);
}
2024-07-12 13:59:16 +00:00
}
2024-10-05 15:12:23 +00:00
Widget _buildTitle(Channel item, ChannelMember? otherside) {
if (otherside != null) {
return Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Expanded(child: Text(otherside.account.nick)),
if (_lastMessages != null && _lastMessages![item.id] != null)
Text(
DateFormat('MM/dd').format(
_lastMessages![item.id]!.createdAt.toLocal(),
),
style: TextStyle(
fontSize: 12,
color:
Theme.of(context).colorScheme.onSurface.withOpacity(0.75),
),
),
],
);
}
return Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Expanded(child: Text(item.name)),
if (_lastMessages != null && _lastMessages![item.id] != null)
Text(
DateFormat('MM/dd').format(
_lastMessages![item.id]!.createdAt.toLocal(),
),
style: TextStyle(
fontSize: 12,
color: Theme.of(context).colorScheme.onSurface.withOpacity(0.75),
),
),
],
);
}
Widget _buildSubtitle(Channel item, ChannelMember? otherside) {
2024-08-01 16:54:19 +00:00
if (PlatformInfo.isWeb) {
2024-10-05 07:11:48 +00:00
return otherside != null
? Text(
'channelDirectDescription'.trParams(
{'username': '@${otherside.account.name}'},
),
maxLines: 1,
overflow: TextOverflow.ellipsis,
)
: Text(
item.description,
maxLines: 1,
overflow: TextOverflow.ellipsis,
);
2024-08-01 16:54:19 +00:00
}
2024-10-05 07:11:48 +00:00
return AnimatedSwitcher(
switchInCurve: Curves.easeIn,
switchOutCurve: Curves.easeOut,
transitionBuilder: (child, animation) {
return FadeTransition(opacity: animation, child: child);
},
duration: const Duration(milliseconds: 300),
child: (_lastMessages == null || _lastMessages![item.id] == null)
2024-10-05 15:12:23 +00:00
? Builder(
builder: (context) {
return otherside != null
? Text(
'channelDirectDescription'.trParams(
{'username': '@${otherside.account.name}'},
),
maxLines: 1,
overflow: TextOverflow.ellipsis,
)
: Text(
item.description,
maxLines: 1,
overflow: TextOverflow.ellipsis,
);
},
)
2024-10-05 07:11:48 +00:00
: Builder(
builder: (context) {
final data = _lastMessages![item.id]!.data!;
2024-10-05 15:12:23 +00:00
return Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
if (item.type == 0)
Badge(
label: Text(data.sender.account.nick),
backgroundColor:
Theme.of(context).colorScheme.secondaryContainer,
textColor:
Theme.of(context).colorScheme.onSecondaryContainer,
),
if (item.type == 0) const Gap(6),
if (data.body['text'] != null)
Expanded(
child: Text(
data.body['text'],
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
)
else
Badge(label: Text('unablePreview'.tr)),
],
2024-10-05 07:11:48 +00:00
);
},
),
layoutBuilder: (currentChild, previousChildren) {
return Stack(
alignment: Alignment.centerLeft,
children: <Widget>[
...previousChildren,
if (currentChild != null) currentChild,
],
2024-08-01 16:54:19 +00:00
);
},
);
}
Widget _buildEntry(Channel item) {
2024-10-05 15:12:23 +00:00
const padding = EdgeInsets.symmetric(horizontal: 20);
2024-07-12 13:59:16 +00:00
final otherside =
item.members!.where((e) => e.account.id != widget.selfId).firstOrNull;
2024-05-30 14:02:54 +00:00
if (item.type == 1 && otherside != null) {
2024-10-11 16:41:03 +00:00
final avatar = AttachedCircleAvatar(
content: otherside.account.avatar,
2024-10-05 15:12:23 +00:00
radius: 20,
bgColor: Theme.of(context).colorScheme.primary,
feColor: Theme.of(context).colorScheme.onPrimary,
);
2024-05-30 14:02:54 +00:00
return ListTile(
leading: avatar,
2024-07-12 13:59:16 +00:00
contentPadding: padding,
2024-10-05 15:12:23 +00:00
title: _buildTitle(item, otherside),
subtitle: _buildSubtitle(item, otherside),
2024-08-01 16:54:19 +00:00
onTap: () => _gotoChannel(item),
2024-07-12 13:59:16 +00:00
);
} else {
final avatar = CircleAvatar(
2024-10-05 15:12:23 +00:00
backgroundColor: Theme.of(context).colorScheme.primary,
radius: 20,
child: FaIcon(
FontAwesomeIcons.hashtag,
2024-10-05 15:12:23 +00:00
color: Theme.of(context).colorScheme.onPrimary,
size: 16,
),
);
2024-07-12 13:59:16 +00:00
return ListTile(
2024-10-05 15:12:23 +00:00
minTileHeight: null,
leading: item.realmId == null
? avatar
: badges.Badge(
position: badges.BadgePosition.bottomEnd(bottom: -4, end: -6),
badgeStyle: badges.BadgeStyle(
badgeColor: Theme.of(context).colorScheme.secondaryContainer,
padding: const EdgeInsets.all(2),
elevation: 8,
),
2024-10-11 16:41:03 +00:00
badgeContent: AttachedCircleAvatar(
2024-10-05 15:12:23 +00:00
content: item.realm?.avatar,
radius: 10,
fallbackWidget: const Icon(
Icons.workspaces,
size: 16,
),
),
child: avatar,
),
2024-07-12 13:59:16 +00:00
contentPadding: padding,
2024-10-05 15:12:23 +00:00
title: _buildTitle(item, null),
subtitle: _buildSubtitle(item, null),
2024-08-01 16:54:19 +00:00
onTap: () => _gotoChannel(item),
2024-05-30 14:02:54 +00:00
);
}
2024-05-29 15:22:24 +00:00
}
2024-05-30 14:02:54 +00:00
@override
Widget build(BuildContext context) {
2024-07-06 10:17:54 +00:00
return CustomScrollView(
slivers: [
SliverList.builder(
2024-10-05 15:12:23 +00:00
itemCount: widget.channels.length,
2024-07-06 10:17:54 +00:00
itemBuilder: (context, index) {
2024-10-05 15:12:23 +00:00
final element = widget.channels[index];
2024-08-01 16:54:19 +00:00
return _buildEntry(element);
2024-07-06 10:17:54 +00:00
},
),
2024-10-05 15:12:23 +00:00
SliverGap(max(16, MediaQuery.of(context).padding.bottom)),
2024-05-30 14:02:54 +00:00
],
);
}
2024-05-29 15:22:24 +00:00
}