2023-01-25 01:53:55 +00:00
|
|
|
import 'package:flutter/material.dart';
|
2024-10-02 17:17:48 +00:00
|
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
2023-01-25 01:53:55 +00:00
|
|
|
import 'package:go_router/go_router.dart';
|
|
|
|
|
|
|
|
import '../controls/image_control.dart';
|
2023-04-13 14:30:09 +00:00
|
|
|
import '../controls/responsive_max_width.dart';
|
2023-01-25 01:53:55 +00:00
|
|
|
import '../controls/standard_appbar.dart';
|
2023-02-08 15:41:29 +00:00
|
|
|
import '../controls/status_and_refresh_button.dart';
|
2024-10-02 17:17:48 +00:00
|
|
|
import '../models/auth/profile.dart';
|
2024-12-10 11:54:45 +00:00
|
|
|
import '../riverpod_controllers/account_services.dart';
|
2024-12-20 01:23:23 +00:00
|
|
|
import '../riverpod_controllers/connection_manager_services.dart';
|
2024-10-02 17:17:48 +00:00
|
|
|
import '../riverpod_controllers/direct_message_services.dart';
|
2024-12-17 00:54:31 +00:00
|
|
|
import '../riverpod_controllers/networking/network_status_services.dart';
|
2023-01-25 01:53:55 +00:00
|
|
|
import '../routes.dart';
|
|
|
|
import '../utils/dateutils.dart';
|
|
|
|
|
2024-10-02 17:17:48 +00:00
|
|
|
class MessagesScreen extends ConsumerWidget {
|
2023-10-31 01:44:16 +00:00
|
|
|
const MessagesScreen({super.key});
|
|
|
|
|
2023-01-25 01:53:55 +00:00
|
|
|
@override
|
2024-10-02 17:17:48 +00:00
|
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
2024-12-10 11:54:45 +00:00
|
|
|
final profile = ref.watch(activeProfileProvider);
|
2024-12-20 01:23:23 +00:00
|
|
|
ref.watch(directMessageThreadIdsProvider(profile));
|
|
|
|
final loading = ref.watch(directMessageLoadingProvider(profile));
|
2024-12-12 14:23:13 +00:00
|
|
|
|
2023-01-25 01:53:55 +00:00
|
|
|
return Scaffold(
|
2023-01-25 18:51:53 +00:00
|
|
|
appBar: StandardAppBar.build(context, 'Direct Message Threads', actions: [
|
2024-12-20 01:30:14 +00:00
|
|
|
StatusAndRefreshButton(
|
2024-12-20 01:23:23 +00:00
|
|
|
executing: loading,
|
|
|
|
refreshFunction: () async => await ref
|
|
|
|
.read(directMessageThreadIdsProvider(profile).notifier)
|
|
|
|
.update(),
|
2024-05-19 15:07:15 +00:00
|
|
|
busyColor: Theme.of(context).colorScheme.surface,
|
2023-02-08 15:41:29 +00:00
|
|
|
),
|
2023-01-25 18:51:53 +00:00
|
|
|
IconButton(
|
|
|
|
onPressed: () {
|
|
|
|
context.push('/messages/new_thread');
|
|
|
|
},
|
2023-10-31 01:44:16 +00:00
|
|
|
icon: const Icon(Icons.add),
|
2023-01-25 18:51:53 +00:00
|
|
|
)
|
|
|
|
]),
|
2023-01-25 01:53:55 +00:00
|
|
|
body: RefreshIndicator(
|
|
|
|
onRefresh: () async {
|
2024-12-20 01:23:23 +00:00
|
|
|
await ref
|
|
|
|
.read(directMessageThreadIdsProvider(profile).notifier)
|
|
|
|
.update();
|
2023-01-25 01:53:55 +00:00
|
|
|
},
|
2024-12-20 01:23:23 +00:00
|
|
|
child: Center(
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
if (loading) const LinearProgressIndicator(),
|
|
|
|
Expanded(child: buildBody(profile, ref))
|
|
|
|
],
|
|
|
|
)),
|
2023-01-25 01:53:55 +00:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-10-02 17:17:48 +00:00
|
|
|
Widget buildBody(Profile profile, WidgetRef ref) {
|
|
|
|
final threadIds = ref.watch(directMessageThreadIdsProvider(profile));
|
2024-12-20 01:23:23 +00:00
|
|
|
|
2024-10-02 17:17:48 +00:00
|
|
|
return threadIds.isEmpty
|
2023-01-25 01:53:55 +00:00
|
|
|
? const Text('No Direct Message Threads')
|
2023-04-13 14:30:09 +00:00
|
|
|
: ResponsiveMaxWidth(
|
|
|
|
child: ListView.separated(
|
2024-10-02 17:17:48 +00:00
|
|
|
itemCount: threadIds.length,
|
2024-12-20 01:23:23 +00:00
|
|
|
itemBuilder: (context, i) {
|
|
|
|
final index = i;
|
2024-10-02 17:17:48 +00:00
|
|
|
final threadId = threadIds[index];
|
|
|
|
final thread = ref.watch(
|
|
|
|
directMessageThreadServiceProvider(profile, threadId));
|
2023-04-13 14:30:09 +00:00
|
|
|
final style = thread.allSeen
|
|
|
|
? null
|
|
|
|
: const TextStyle(fontWeight: FontWeight.bold);
|
2024-12-20 01:23:23 +00:00
|
|
|
final participantsAvatars = <String>[];
|
|
|
|
final participantNames = <String>[];
|
|
|
|
for (final pid in thread.participantIds) {
|
|
|
|
if (pid == profile.userId) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
ref.watch(connectionByIdProvider(profile, pid)).match(
|
|
|
|
onSuccess: (c) {
|
|
|
|
participantsAvatars.add(c.avatarUrl);
|
|
|
|
participantNames.add('${c.name}(${c.handle})');
|
|
|
|
},
|
|
|
|
onError: (_) => participantNames.add('Person 1'));
|
|
|
|
}
|
2023-04-13 14:30:09 +00:00
|
|
|
return ListTile(
|
|
|
|
onTap: () => context.pushNamed(
|
|
|
|
ScreenPaths.thread,
|
2023-10-31 01:44:16 +00:00
|
|
|
queryParameters: {'uri': thread.parentUri},
|
2023-04-13 14:30:09 +00:00
|
|
|
),
|
2024-12-20 01:23:23 +00:00
|
|
|
leading: thread.participantIds.isEmpty
|
|
|
|
? null
|
|
|
|
: ImageControl(
|
|
|
|
imageUrl: participantsAvatars.first,
|
|
|
|
iconOverride: const Icon(Icons.person),
|
|
|
|
width: 32.0,
|
|
|
|
onTap: null,
|
|
|
|
),
|
2023-04-13 14:30:09 +00:00
|
|
|
title: Text(
|
|
|
|
[
|
|
|
|
'You',
|
2024-12-20 01:23:23 +00:00
|
|
|
...participantNames,
|
|
|
|
].join(participantNames.length == 1 ? ' & ' : ', '),
|
2023-04-13 14:30:09 +00:00
|
|
|
softWrap: true,
|
|
|
|
style: style,
|
|
|
|
),
|
|
|
|
subtitle: Text(
|
|
|
|
thread.title,
|
|
|
|
style: style,
|
|
|
|
),
|
|
|
|
trailing: Text(
|
2024-12-20 01:23:23 +00:00
|
|
|
thread.messages.isEmpty
|
|
|
|
? ''
|
|
|
|
: ElapsedDateUtils.elapsedTimeStringFromEpochSeconds(
|
|
|
|
thread.messages.last.createdAt),
|
2023-04-13 14:30:09 +00:00
|
|
|
style: style,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
separatorBuilder: (_, __) => const Divider(),
|
|
|
|
),
|
2023-01-25 01:53:55 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|