forked from Nexus/nexus
Remove flutter chat (#26)
Had to squash merge manually as Forgejo was erroring
This commit is contained in:
parent
bd1d5ea745
commit
16cf126df4
111 changed files with 3162 additions and 2366 deletions
|
|
@ -1,47 +1,31 @@
|
|||
import "dart:async";
|
||||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_chat_core/flutter_chat_core.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/controllers/user_controller.dart";
|
||||
import "package:nexus/helpers/extensions/get_localpart.dart";
|
||||
import "package:nexus/models/membership.dart";
|
||||
import "package:nexus/models/membership_status.dart";
|
||||
import "package:nexus/models/configs/user_config.dart";
|
||||
import "package:nexus/models/content/membership.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
|
||||
class AuthorController extends AsyncNotifier<Membership> {
|
||||
final Message message;
|
||||
AuthorController(this.message);
|
||||
class AuthorController extends AsyncNotifier<MembershipContent> {
|
||||
final Event event;
|
||||
AuthorController(this.event);
|
||||
|
||||
@override
|
||||
Future<Membership> build() async {
|
||||
Future<MembershipContent> build() async {
|
||||
final member = await ref.watch(
|
||||
UserController.provider(message.authorId).future,
|
||||
UserController.provider(
|
||||
UserConfig(roomId: event.roomId, userId: event.sender),
|
||||
).future,
|
||||
);
|
||||
|
||||
final pmp = message.metadata?["pmp"] == null
|
||||
? null
|
||||
: Membership.fromContent(
|
||||
IMap(message.metadata?["pmp"]),
|
||||
message.authorId,
|
||||
ref.watch(
|
||||
ClientStateController.provider.select(
|
||||
(value) => value?.homeserverUrl,
|
||||
),
|
||||
) ??
|
||||
"",
|
||||
);
|
||||
|
||||
return Membership(
|
||||
status: member?.status ?? MembershipStatus.leave,
|
||||
avatarUrl: pmp?.avatarUrl ?? member?.avatarUrl,
|
||||
displayName:
|
||||
pmp?.displayName ?? member?.displayName ?? message.authorId.localpart,
|
||||
userId: message.authorId,
|
||||
return MembershipContent(
|
||||
status: member.status,
|
||||
avatarUrl: event.pmp?.avatarUrl ?? member.avatarUrl,
|
||||
displayName: event.pmp?.displayName ?? member.displayName,
|
||||
);
|
||||
}
|
||||
|
||||
static final provider =
|
||||
AsyncNotifierProvider.family<AuthorController, Membership, Message>(
|
||||
AsyncNotifierProvider.family<AuthorController, MembershipContent, Event>(
|
||||
AuthorController.new,
|
||||
);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,15 +1,13 @@
|
|||
import "dart:developer";
|
||||
import "dart:ffi";
|
||||
import "dart:io";
|
||||
import "dart:isolate";
|
||||
import "package:collection/collection.dart";
|
||||
import "dart:math";
|
||||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:ffi/ffi.dart";
|
||||
import "package:flutter/foundation.dart";
|
||||
import "package:nexus/controllers/account_data_controller.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/controllers/init_complete_controller.dart";
|
||||
import "package:nexus/controllers/new_events_controller.dart";
|
||||
import "package:nexus/controllers/rooms_controller.dart";
|
||||
import "package:nexus/controllers/space_edges_controller.dart";
|
||||
import "package:nexus/controllers/sync_status_controller.dart";
|
||||
|
|
@ -17,6 +15,7 @@ import "package:nexus/controllers/top_level_spaces_controller.dart";
|
|||
import "package:nexus/helpers/extensions/gomuks_buffer.dart";
|
||||
import "package:nexus/main.dart";
|
||||
import "package:nexus/models/client_state.dart";
|
||||
import "package:nexus/models/content/content.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
import "package:nexus/models/paginate.dart";
|
||||
import "package:nexus/models/requests/get_event_request.dart";
|
||||
|
|
@ -81,12 +80,8 @@ class ClientController extends AsyncNotifier<int> {
|
|||
case "send_complete":
|
||||
final event = Event.fromJson(decodedMuksEvent["event"]);
|
||||
|
||||
if (event.type == "m.room.message") {
|
||||
ref
|
||||
.watch(
|
||||
NewEventsController.provider(event.roomId).notifier,
|
||||
)
|
||||
.add(IList([event]));
|
||||
if (event.type == EventType.message.type) {
|
||||
// ref.watch(provider.notifier).addEvent(event); TODO
|
||||
}
|
||||
break;
|
||||
case "sync_complete":
|
||||
|
|
@ -127,9 +122,12 @@ class ClientController extends AsyncNotifier<int> {
|
|||
}
|
||||
debugPrint("Finished handling $muksEventType...");
|
||||
} catch (error, stackTrace) {
|
||||
debugger();
|
||||
showError(error, stackTrace);
|
||||
debugPrintStack(stackTrace: stackTrace, label: error.toString());
|
||||
if (kDebugMode) {
|
||||
debugPrintStack(stackTrace: stackTrace, label: error.toString());
|
||||
rethrow;
|
||||
} else {
|
||||
showError(error, stackTrace);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
|
|
@ -220,11 +218,6 @@ class ClientController extends AsyncNotifier<int> {
|
|||
}
|
||||
|
||||
Future<Event?> getEvent(GetEventRequest request) async {
|
||||
final event = request.room.events.firstWhereOrNull(
|
||||
(event) => event.eventId == request.eventId,
|
||||
);
|
||||
if (event != null) return event;
|
||||
|
||||
final json = await _sendCommand("get_event", request.toJson());
|
||||
return json == null ? null : Event.fromJson(json);
|
||||
}
|
||||
|
|
@ -232,8 +225,10 @@ class ClientController extends AsyncNotifier<int> {
|
|||
Future<Paginate> paginate(PaginateRequest request) async =>
|
||||
Paginate.fromJson(await _sendCommand("paginate", request.toJson()));
|
||||
|
||||
Future<Profile> getProfile(String userId) async =>
|
||||
Profile.fromJson(await _sendCommand("get_profile", {"user_id": userId}));
|
||||
Future<Profile> getProfile(String userId) async => Profile.fromJsonWithCatch({
|
||||
...(await _sendCommand("get_profile", {"user_id": userId})),
|
||||
"id": userId,
|
||||
});
|
||||
|
||||
Future<void> reportEvent(ReportRequest request) =>
|
||||
_sendCommand("report_event", request.toJson());
|
||||
|
|
@ -242,9 +237,8 @@ class ClientController extends AsyncNotifier<int> {
|
|||
_sendCommand("set_membership", request.toJson());
|
||||
|
||||
Future<void> markRead(Room room) async {
|
||||
final event = room.events.firstWhereOrNull(
|
||||
(event) => event.rowId == room.timeline.last.eventRowId,
|
||||
);
|
||||
final eventRowId = room.timeline[room.timeline.keys.reduce(max)];
|
||||
final event = eventRowId == null ? null : room.events[eventRowId];
|
||||
if (event == null || room.metadata == null) return;
|
||||
|
||||
await _sendCommand("mark_read", {
|
||||
|
|
|
|||
|
|
@ -1,5 +1,7 @@
|
|||
import "package:collection/collection.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_controller.dart";
|
||||
import "package:nexus/controllers/rooms_controller.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
import "package:nexus/models/requests/get_event_request.dart";
|
||||
|
||||
|
|
@ -9,8 +11,18 @@ class EventController extends AsyncNotifier<Event?> {
|
|||
|
||||
@override
|
||||
Future<Event?> build() async {
|
||||
final client = ref.watch(ClientController.provider.notifier);
|
||||
return await client.getEvent(request).onError((_, _) => null);
|
||||
final room = ref.watch(
|
||||
RoomsController.provider.select((value) => value[request.roomId]),
|
||||
);
|
||||
final event = room?.events.values.firstWhereOrNull(
|
||||
(event) => event.eventId == request.eventId,
|
||||
);
|
||||
|
||||
return event ??
|
||||
await ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.getEvent(request)
|
||||
.onError((_, _) => null);
|
||||
}
|
||||
|
||||
static final provider = AsyncNotifierProvider.family
|
||||
|
|
|
|||
|
|
@ -12,14 +12,14 @@ class KeyController extends Notifier<String?> {
|
|||
String? build() =>
|
||||
ref.watch(SharedPrefsController.provider).requireValue.getString(key);
|
||||
|
||||
Future<void> set(String? id) async {
|
||||
Future<void> set(String? value) async {
|
||||
final prefs = ref.watch(SharedPrefsController.provider).requireValue;
|
||||
state = id;
|
||||
state = value;
|
||||
|
||||
if (id == null) {
|
||||
if (value == null) {
|
||||
prefs.remove(key);
|
||||
} else {
|
||||
prefs.setString(key, id);
|
||||
prefs.setString(key, value);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
|||
32
lib/controllers/members_by_status_controller.dart
Normal file
32
lib/controllers/members_by_status_controller.dart
Normal file
|
|
@ -0,0 +1,32 @@
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/members_controller.dart";
|
||||
import "package:nexus/models/configs/members_by_status_config.dart";
|
||||
import "package:nexus/models/content/membership.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
|
||||
class MembersByStatusController extends AsyncNotifier<ISet<Event>> {
|
||||
final MembersByStatusConfig config;
|
||||
MembersByStatusController(this.config);
|
||||
|
||||
@override
|
||||
Future<ISet<Event>> build() => ref.watch(
|
||||
MembersController.provider(config.roomId).selectAsync(
|
||||
(members) => members
|
||||
.where(
|
||||
(membership) => switch (membership.content) {
|
||||
MembershipContent(:final status) => config.status == status,
|
||||
_ => false,
|
||||
},
|
||||
)
|
||||
.toISet(),
|
||||
),
|
||||
);
|
||||
|
||||
static final provider =
|
||||
AsyncNotifierProvider.family<
|
||||
MembersByStatusController,
|
||||
ISet<Event>,
|
||||
MembersByStatusConfig
|
||||
>(MembersByStatusController.new);
|
||||
}
|
||||
|
|
@ -1,25 +0,0 @@
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/members_controller.dart";
|
||||
import "package:nexus/models/membership.dart";
|
||||
import "package:nexus/models/membership_status.dart";
|
||||
|
||||
class MembersByTypeController extends AsyncNotifier<IList<Membership>> {
|
||||
final MembershipStatus status;
|
||||
MembersByTypeController(this.status);
|
||||
|
||||
@override
|
||||
Future<IList<Membership>> build() => ref.watch(
|
||||
MembersController.provider.selectAsync(
|
||||
(members) =>
|
||||
members.where((membership) => membership.status == status).toIList(),
|
||||
),
|
||||
);
|
||||
|
||||
static final provider =
|
||||
AsyncNotifierProvider.family<
|
||||
MembersByTypeController,
|
||||
IList<Membership>,
|
||||
MembershipStatus
|
||||
>(MembersByTypeController.new);
|
||||
}
|
||||
|
|
@ -1,52 +1,46 @@
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_controller.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/controllers/selected_room_controller.dart";
|
||||
import "package:nexus/models/membership.dart";
|
||||
import "package:nexus/controllers/rooms_controller.dart";
|
||||
import "package:nexus/models/content/content.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
import "package:nexus/models/requests/get_room_state_request.dart";
|
||||
|
||||
class MembersController extends AsyncNotifier<IList<Membership>> {
|
||||
class MembersController extends AsyncNotifier<ISet<Event>> {
|
||||
final String roomId;
|
||||
MembersController(this.roomId);
|
||||
|
||||
@override
|
||||
Future<IList<Membership>> build() async {
|
||||
final data = ref.watch(
|
||||
SelectedRoomController.provider.select(
|
||||
(value) => value?.metadata == null
|
||||
? null
|
||||
: (value!.metadata!.id, value.metadata!.hasMemberList),
|
||||
),
|
||||
Future<ISet<Event>> build() async {
|
||||
final room = ref.watch(
|
||||
RoomsController.provider.select((value) => value[roomId]),
|
||||
);
|
||||
if (data == null) return const IList.empty();
|
||||
|
||||
final state = await ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.getRoomState(
|
||||
GetRoomStateRequest(
|
||||
roomId: data.$1,
|
||||
fetchMembers: data.$2 == false,
|
||||
includeMembers: true,
|
||||
),
|
||||
);
|
||||
if (room == null) return const ISet.empty();
|
||||
|
||||
return state.nonNulls
|
||||
.where((state) => state.type == "m.room.member")
|
||||
.map(
|
||||
(membership) => Membership.fromContent(
|
||||
membership.content,
|
||||
membership.stateKey!,
|
||||
ref.watch(
|
||||
ClientStateController.provider.select(
|
||||
(value) => value?.homeserverUrl,
|
||||
),
|
||||
) ??
|
||||
"",
|
||||
),
|
||||
)
|
||||
.toIList();
|
||||
if (!room.hasFetchedMembers) {
|
||||
final fetchedState = await ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.getRoomState(
|
||||
GetRoomStateRequest(
|
||||
roomId: roomId,
|
||||
fetchMembers: room.metadata?.hasMemberList ?? true,
|
||||
includeMembers: true,
|
||||
),
|
||||
);
|
||||
|
||||
await ref
|
||||
.read(RoomsController.provider.notifier)
|
||||
.addState(roomId, fetchedState, isMembers: true);
|
||||
}
|
||||
|
||||
return room.state[EventType.membership.type]?.values
|
||||
.map((rowId) => room.events[rowId])
|
||||
.nonNulls
|
||||
.toISet() ??
|
||||
const ISet.empty();
|
||||
}
|
||||
|
||||
static final provider =
|
||||
AsyncNotifierProvider<MembersController, IList<Membership>>(
|
||||
MembersController.new,
|
||||
);
|
||||
static final provider = AsyncNotifierProvider.autoDispose
|
||||
.family<MembersController, ISet<Event>, String>(MembersController.new);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,214 +0,0 @@
|
|||
import "package:collection/collection.dart";
|
||||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_chat_core/flutter_chat_core.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_controller.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/helpers/extensions/mxc_to_https.dart";
|
||||
import "package:nexus/models/configs/message_config.dart";
|
||||
import "package:nexus/models/requests/get_related_events_request.dart";
|
||||
|
||||
class MessageController extends AsyncNotifier<Message?> {
|
||||
final MessageConfig config;
|
||||
MessageController(this.config);
|
||||
|
||||
@override
|
||||
Future<Message?> build() async {
|
||||
try {
|
||||
final isEdit = config.event.relationType == "m.replace";
|
||||
if ((isEdit && !config.includeEdits) || config.room.metadata == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
final event = config.event.lastEditRowId == null
|
||||
? config.event
|
||||
: config.room.events.firstWhereOrNull(
|
||||
(e) => e.rowId == config.event.lastEditRowId,
|
||||
) ??
|
||||
config.event;
|
||||
|
||||
final decrypted = (event.decrypted ?? event.content);
|
||||
final type = (config.event.decryptedType ?? config.event.type);
|
||||
final content = decrypted["m.new_content"] == null
|
||||
? decrypted
|
||||
: IMap(decrypted["m.new_content"]);
|
||||
|
||||
final homeserver = ref
|
||||
.read(ClientStateController.provider)
|
||||
?.homeserverUrl;
|
||||
final source = homeserver == null || content["url"] == null
|
||||
? "null"
|
||||
: Uri.parse(content["url"]).mxcToHttps(homeserver).toString();
|
||||
|
||||
final metadata = {
|
||||
"body": config.event.redactedBy == null
|
||||
? (content["body"] ?? "")
|
||||
: "Deleted Message",
|
||||
"flashing": false,
|
||||
"timelineId": event.timelineRowId,
|
||||
"big": event.localContent?.bigEmoji == true,
|
||||
"eventType": type,
|
||||
"pmp": content["com.beeper.per_message_profile"],
|
||||
"error": event.sendError,
|
||||
"format": content["format"] ?? content["format"],
|
||||
"editSource": event.localContent?.editSource ?? content["body"],
|
||||
"txnId": config.event.transactionId,
|
||||
};
|
||||
|
||||
final editedAt = event.relationType == "m.replace"
|
||||
? event.timestamp
|
||||
: null;
|
||||
|
||||
if ((event.redactedBy != null && !config.alwaysReturn) ||
|
||||
(!config.includeEdits &&
|
||||
(config.event.relationType == "m.replace"))) {
|
||||
return null;
|
||||
}
|
||||
|
||||
final replyId =
|
||||
config.event.content["m.relates_to"]?["m.in_reply_to"]?["event_id"];
|
||||
|
||||
final reactionEvents = config.event.reactions.isEmpty && !isEdit
|
||||
? null
|
||||
: await ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.getRelatedEvents(
|
||||
GetRelatedEventsRequest(
|
||||
roomId: config.room.metadata!.id,
|
||||
eventId:
|
||||
(isEdit ? config.event.relatesTo : null) ??
|
||||
config.event.eventId,
|
||||
relationType: "m.annotation",
|
||||
),
|
||||
);
|
||||
|
||||
final reactions = reactionEvents
|
||||
?.where((event) => event.redactedBy == null)
|
||||
.fold<IMap<String, IList<String>>>(IMap(), (acc, event) {
|
||||
final key = event.content["m.relates_to"]?["key"];
|
||||
if (key == null) return acc;
|
||||
|
||||
return acc.update(
|
||||
key,
|
||||
(list) => list.add(event.authorId),
|
||||
ifAbsent: () => IList([event.authorId]),
|
||||
);
|
||||
})
|
||||
.map((key, value) => MapEntry(key, value.unlock))
|
||||
.unlock;
|
||||
|
||||
final asText =
|
||||
Message.text(
|
||||
metadata: metadata,
|
||||
id: config.event.eventId,
|
||||
reactions: reactions,
|
||||
authorId: event.authorId,
|
||||
text: content["formatted_body"] ?? content["body"] ?? "",
|
||||
replyToMessageId: replyId,
|
||||
deliveredAt: config.event.timestamp,
|
||||
editedAt: editedAt,
|
||||
)
|
||||
as TextMessage;
|
||||
|
||||
Message toSystemMessage(String content) => Message.system(
|
||||
metadata: {...metadata, "body": content},
|
||||
id: config.event.eventId,
|
||||
reactions: reactions,
|
||||
authorId: event.authorId,
|
||||
deliveredAt: config.event.timestamp,
|
||||
text: content,
|
||||
);
|
||||
|
||||
return switch (type) {
|
||||
"m.room.encrypted" => asText.copyWith(
|
||||
text: "Unable to decrypt message.",
|
||||
metadata: {...metadata, "body": "Unable to decrypt message."},
|
||||
),
|
||||
// "org.matrix.msc3381.poll.start" => Message.custom(
|
||||
// metadata: {
|
||||
// ...metadata,
|
||||
// "poll": event.parsedPollEventContent.pollStartContent,
|
||||
// "responses": event.getPollResponses(timeline),
|
||||
// },
|
||||
// id: eventId,
|
||||
// deliveredAt: originServerTs,
|
||||
// authorId: senderId,
|
||||
// ),
|
||||
("m.sticker" || "m.room.message") => switch (content["msgtype"]) {
|
||||
null || "m.image" => Message.image(
|
||||
id: config.event.eventId,
|
||||
authorId: event.authorId,
|
||||
reactions: reactions,
|
||||
source: source,
|
||||
replyToMessageId: replyId,
|
||||
metadata: metadata,
|
||||
text: asText.text,
|
||||
deliveredAt: config.event.timestamp,
|
||||
blurhash: (content["info"] as Map?)?["xyz.amorgan.blurhash"],
|
||||
),
|
||||
"m.audio" || "m.file" => Message.file(
|
||||
name: content["filename"].toString(),
|
||||
size: content["info"]["size"],
|
||||
metadata: metadata,
|
||||
id: config.event.eventId,
|
||||
reactions: reactions,
|
||||
authorId: event.authorId,
|
||||
source: source,
|
||||
replyToMessageId: replyId,
|
||||
deliveredAt: config.event.timestamp,
|
||||
),
|
||||
_ => asText,
|
||||
},
|
||||
"m.room.member" =>
|
||||
content["membership"] == event.unsigned["prev_content"]?["membership"]
|
||||
? null
|
||||
: toSystemMessage(
|
||||
"${content["displayname"] ?? event.stateKey} ${switch (content["membership"]) {
|
||||
"invite" => "was invited to",
|
||||
"join" => "joined",
|
||||
"leave" => event.authorId == event.stateKey ? "left" : (event.unsigned["prev_content"]?["membership"] == "ban" ? "was unbanned from" : "was kicked from"),
|
||||
"ban" => "was banned from",
|
||||
"knock" => "asked to join",
|
||||
_ => "did something relating to",
|
||||
}} the room. ${content["reason"] ?? ""}",
|
||||
),
|
||||
|
||||
"m.room.server_acl" => toSystemMessage(
|
||||
"${event.authorId} updated the server ban list.",
|
||||
),
|
||||
|
||||
"m.room.redaction" =>
|
||||
config.alwaysReturn
|
||||
? asText.copyWith(
|
||||
metadata: {
|
||||
...(asText.metadata ?? {}),
|
||||
"body": "Deleted Message",
|
||||
},
|
||||
)
|
||||
: null,
|
||||
_ =>
|
||||
config.alwaysReturn
|
||||
? asText
|
||||
: (
|
||||
// Turn this on for debugging purposes
|
||||
false
|
||||
// ignore: dead_code
|
||||
? Message.unsupported(
|
||||
metadata: metadata,
|
||||
reactions: reactions,
|
||||
id: config.event.eventId,
|
||||
authorId: event.authorId,
|
||||
replyToMessageId: replyId,
|
||||
)
|
||||
: null),
|
||||
};
|
||||
} catch (error) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
static final provider = AsyncNotifierProvider.family
|
||||
.autoDispose<MessageController, Message?, MessageConfig>(
|
||||
MessageController.new,
|
||||
);
|
||||
}
|
||||
|
|
@ -1,27 +0,0 @@
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_chat_core/flutter_chat_core.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/message_controller.dart";
|
||||
import "package:nexus/models/configs/message_config.dart";
|
||||
import "package:nexus/models/configs/messages_config.dart";
|
||||
|
||||
class MessagesController extends AsyncNotifier<IList<Message>> {
|
||||
final MessagesConfig config;
|
||||
MessagesController(this.config);
|
||||
|
||||
@override
|
||||
Future<IList<Message>> build() async => (await Future.wait(
|
||||
config.events.map(
|
||||
(event) => ref.watch(
|
||||
MessageController.provider(
|
||||
MessageConfig(event: event, room: config.room),
|
||||
).future,
|
||||
),
|
||||
),
|
||||
)).nonNulls.toIList();
|
||||
|
||||
static final provider = AsyncNotifierProvider.family
|
||||
.autoDispose<MessagesController, IList<Message>, MessagesConfig>(
|
||||
MessagesController.new,
|
||||
);
|
||||
}
|
||||
|
|
@ -1,18 +0,0 @@
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
|
||||
class NewEventsController extends Notifier<IList<Event>> {
|
||||
final String roomId;
|
||||
NewEventsController(this.roomId);
|
||||
|
||||
@override
|
||||
IList<Event> build() => const IList.empty();
|
||||
|
||||
void add(IList<Event> newEvents) => state = newEvents;
|
||||
|
||||
static final provider = NotifierProvider.autoDispose
|
||||
.family<NewEventsController, IList<Event>, String>(
|
||||
NewEventsController.new,
|
||||
);
|
||||
}
|
||||
|
|
@ -1,8 +1,9 @@
|
|||
import "package:collection/collection.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/controllers/selected_room_controller.dart";
|
||||
import "package:nexus/controllers/rooms_controller.dart";
|
||||
import "package:nexus/models/configs/power_level_config.dart";
|
||||
import "package:nexus/models/content/content.dart";
|
||||
import "package:nexus/models/content/power_levels.dart";
|
||||
import "package:nexus/models/requests/membership_action.dart";
|
||||
|
||||
class PowerLevelController extends Notifier<bool> {
|
||||
|
|
@ -11,56 +12,60 @@ class PowerLevelController extends Notifier<bool> {
|
|||
|
||||
@override
|
||||
bool build() {
|
||||
final room = ref.watch(SelectedRoomController.provider);
|
||||
final event = room?.events.firstWhereOrNull(
|
||||
(event) => event.rowId == room.state["m.room.power_levels"]?[""],
|
||||
if (config case EventPowerLevelConfig(:final eventType)) {
|
||||
assert(
|
||||
eventType != EventType.redaction,
|
||||
"Checking power level for a redaction should use [PowerLevelConfig.redaction].",
|
||||
);
|
||||
}
|
||||
|
||||
final room = ref.watch(
|
||||
RoomsController.provider.select((value) => value[config.roomId]),
|
||||
);
|
||||
final user = ref.watch(ClientStateController.provider)?.userId;
|
||||
if (event == null || user == null) return false;
|
||||
|
||||
final users = (event.content["users"] as Map<String, dynamic>? ?? {});
|
||||
final events = (event.content["events"] as Map<String, dynamic>? ?? {});
|
||||
final eventRowId = room?.state[EventType.powerLevels.type]?[""];
|
||||
|
||||
int powerLevelOf(String userId) => users.containsKey(userId)
|
||||
? (users[userId] as int)
|
||||
: (event.content["users_default"] as int? ?? 0);
|
||||
final event = eventRowId == null ? null : room?.events[eventRowId];
|
||||
final content = event?.content is PowerLevelsContent
|
||||
? event!.content
|
||||
: PowerLevelsContent();
|
||||
final user = ref.watch(
|
||||
ClientStateController.provider.select((value) => value?.userId),
|
||||
);
|
||||
if (user == null || content is! PowerLevelsContent) return false;
|
||||
|
||||
int powerLevelOf(String userId) =>
|
||||
content.users[userId] ?? content.usersDefault;
|
||||
|
||||
final userLevel = powerLevelOf(user);
|
||||
final targetLevel = config.targetUser != null
|
||||
? powerLevelOf(config.targetUser!)
|
||||
: null;
|
||||
|
||||
if (config.action != null) {
|
||||
return switch (config.action!) {
|
||||
MembershipAction.invite =>
|
||||
userLevel >= (event.content["invite"] as int? ?? 0),
|
||||
return switch (config) {
|
||||
EventPowerLevelConfig(:final eventType) =>
|
||||
userLevel >= (content.events[eventType.type] ?? content.eventsDefault),
|
||||
|
||||
MembershipAction.kick =>
|
||||
targetLevel != null &&
|
||||
userLevel >= (event.content["kick"] as int? ?? 50) &&
|
||||
userLevel > targetLevel,
|
||||
MembershipActionPowerLevelConfig(:final action, :final targetUser) =>
|
||||
switch (action) {
|
||||
MembershipAction.invite => userLevel >= content.invite,
|
||||
|
||||
MembershipAction.ban =>
|
||||
targetLevel != null &&
|
||||
userLevel >= (event.content["ban"] as int? ?? 50) &&
|
||||
userLevel > targetLevel,
|
||||
MembershipAction.kick =>
|
||||
userLevel >= content.kick && userLevel > powerLevelOf(targetUser),
|
||||
|
||||
MembershipAction.unban =>
|
||||
userLevel >= (event.content["ban"] as int? ?? 50),
|
||||
};
|
||||
}
|
||||
MembershipAction.ban =>
|
||||
userLevel >= content.ban && userLevel > powerLevelOf(targetUser),
|
||||
|
||||
if (config.eventType == "m.room.redaction") {
|
||||
return userLevel >= (event.content["redact"] as int? ?? 50);
|
||||
}
|
||||
MembershipAction.unban => userLevel >= content.ban,
|
||||
},
|
||||
|
||||
final requiredLevel = events.containsKey(config.eventType)
|
||||
? (events[config.eventType] as int)
|
||||
: (config.isStateEvent
|
||||
? (event.content["state_default"] as int? ?? 50)
|
||||
: (event.content["events_default"] as int? ?? 0));
|
||||
StatePowerLevelConfig(:final eventType) =>
|
||||
userLevel >= (content.events[eventType.type] ?? content.stateDefault),
|
||||
|
||||
return userLevel >= requiredLevel;
|
||||
RedactionPowerLevelConfig(:final targetUser) =>
|
||||
userLevel >=
|
||||
(targetUser == user
|
||||
? (content.events[EventType.redaction.type] ??
|
||||
content.eventsDefault)
|
||||
: content.redact),
|
||||
};
|
||||
}
|
||||
|
||||
static final provider = NotifierProvider.autoDispose
|
||||
|
|
|
|||
|
|
@ -12,6 +12,8 @@ class ProfileController extends AsyncNotifier<Profile> {
|
|||
return client.getProfile(userId);
|
||||
}
|
||||
|
||||
static final provider = AsyncNotifierProvider.autoDispose
|
||||
.family<ProfileController, Profile, String>(ProfileController.new);
|
||||
static final provider =
|
||||
AsyncNotifierProvider.family<ProfileController, Profile, String>(
|
||||
ProfileController.new,
|
||||
);
|
||||
}
|
||||
|
|
|
|||
56
lib/controllers/reactions_controller.dart
Normal file
56
lib/controllers/reactions_controller.dart
Normal file
|
|
@ -0,0 +1,56 @@
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_controller.dart";
|
||||
import "package:nexus/controllers/rooms_controller.dart";
|
||||
import "package:nexus/models/configs/reactions_config.dart";
|
||||
import "package:nexus/models/content/reaction.dart";
|
||||
import "package:nexus/models/requests/get_related_events_request.dart";
|
||||
|
||||
class ReactionsController extends AsyncNotifier<IMap<String, IList<String>>> {
|
||||
final ReactionsConfig config;
|
||||
ReactionsController(this.config);
|
||||
|
||||
@override
|
||||
Future<IMap<String, IList<String>>> build() async {
|
||||
final eventInfo = ref.watch(
|
||||
RoomsController.provider.select((value) {
|
||||
final event = value[config.roomId]?.events[config.eventRowId];
|
||||
return event == null ? null : (event.eventId, event.reactions);
|
||||
}),
|
||||
);
|
||||
|
||||
final reactionEvents = eventInfo?.$2.isNotEmpty == true
|
||||
? await ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.getRelatedEvents(
|
||||
GetRelatedEventsRequest(
|
||||
roomId: config.roomId,
|
||||
eventId: eventInfo!.$1,
|
||||
relationType: "m.annotation",
|
||||
),
|
||||
)
|
||||
: null;
|
||||
|
||||
return reactionEvents
|
||||
?.where((event) => event.redactedBy == null)
|
||||
.fold<IMap<String, IList<String>>>(IMap(), (acc, event) {
|
||||
if (event.content case ReactionContent(:final key?)) {
|
||||
return acc.update(
|
||||
key,
|
||||
(list) => list.add(event.sender),
|
||||
ifAbsent: () => IList([event.sender]),
|
||||
);
|
||||
}
|
||||
|
||||
return acc;
|
||||
}) ??
|
||||
const IMap.empty();
|
||||
}
|
||||
|
||||
static final provider =
|
||||
AsyncNotifierProvider.family<
|
||||
ReactionsController,
|
||||
IMap<String, IList<String>>,
|
||||
ReactionsConfig
|
||||
>(ReactionsController.new);
|
||||
}
|
||||
|
|
@ -1,17 +1,13 @@
|
|||
import "dart:async";
|
||||
import "dart:math";
|
||||
import "package:collection/collection.dart";
|
||||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_chat_core/flutter_chat_core.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:fluttertagger/fluttertagger.dart";
|
||||
import "package:nexus/controllers/client_controller.dart";
|
||||
import "package:nexus/controllers/message_controller.dart";
|
||||
import "package:nexus/controllers/messages_controller.dart";
|
||||
import "package:nexus/controllers/new_events_controller.dart";
|
||||
import "package:nexus/controllers/rooms_controller.dart";
|
||||
import "package:nexus/controllers/selected_room_controller.dart";
|
||||
import "package:nexus/models/configs/messages_config.dart";
|
||||
import "package:nexus/models/configs/message_config.dart";
|
||||
import "package:nexus/models/content/reaction.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
import "package:nexus/models/requests/get_related_events_request.dart";
|
||||
import "package:nexus/models/requests/get_room_state_request.dart";
|
||||
import "package:nexus/models/requests/paginate_request.dart";
|
||||
|
|
@ -21,203 +17,75 @@ import "package:nexus/models/requests/send_event_request.dart";
|
|||
import "package:nexus/models/requests/send_message_request.dart";
|
||||
import "package:nexus/models/room.dart";
|
||||
|
||||
class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
||||
class RoomChatController extends AsyncNotifier<IList<Event>> {
|
||||
final String roomId;
|
||||
RoomChatController(this.roomId);
|
||||
|
||||
@override
|
||||
Future<InMemoryChatController> build() async {
|
||||
Future<IList<Event>> build() async {
|
||||
final client = ref.watch(ClientController.provider.notifier);
|
||||
var room = ref.read(RoomsController.provider)[roomId];
|
||||
if (room == null) return InMemoryChatController();
|
||||
final state = await client.getRoomState(
|
||||
GetRoomStateRequest(roomId: roomId),
|
||||
final room = ref.watch(
|
||||
RoomsController.provider.select((rooms) => rooms[roomId]),
|
||||
);
|
||||
if (room == null) return const IList.empty();
|
||||
|
||||
ref
|
||||
.read(RoomsController.provider.notifier)
|
||||
.update(
|
||||
{
|
||||
roomId: Room(
|
||||
events: state,
|
||||
state: state.fold(
|
||||
const IMap.empty(),
|
||||
(previousValue, stateEvent) => previousValue.add(
|
||||
stateEvent.type,
|
||||
(previousValue[stateEvent.type] ?? const IMap.empty()).addAll(
|
||||
IMap({
|
||||
if (stateEvent.stateKey != null)
|
||||
stateEvent.stateKey!: stateEvent.rowId,
|
||||
}),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
}.toIMap(),
|
||||
const ISet.empty(),
|
||||
);
|
||||
|
||||
room = ref.read(RoomsController.provider)[roomId];
|
||||
if (room == null) return InMemoryChatController();
|
||||
|
||||
final messages = await ref.watch(
|
||||
MessagesController.provider(
|
||||
MessagesConfig(
|
||||
room: room,
|
||||
events: room.timeline
|
||||
.map(
|
||||
(timelineRowTuple) => room!.events.firstWhereOrNull(
|
||||
(event) => event.rowId == timelineRowTuple.eventRowId,
|
||||
),
|
||||
)
|
||||
.nonNulls
|
||||
.toIList(),
|
||||
),
|
||||
).future,
|
||||
);
|
||||
final controller = InMemoryChatController(messages: messages.toList());
|
||||
|
||||
ref.onDispose(
|
||||
ref.listen(NewEventsController.provider(roomId), (_, next) async {
|
||||
for (final event in next) {
|
||||
if (event.type == "m.reaction") {
|
||||
final message = controller.messages.firstWhereOrNull(
|
||||
(message) =>
|
||||
message.id == event.content["m.relates_to"]?["event_id"],
|
||||
);
|
||||
final key = event.content["m.relates_to"]?["key"];
|
||||
if (message == null || key == null || !ref.mounted) return;
|
||||
|
||||
return await controller.updateMessage(
|
||||
message,
|
||||
message.copyWith(
|
||||
reactions: IMap(message.reactions)
|
||||
.update(
|
||||
key,
|
||||
(reactors) => [...reactors, event.authorId],
|
||||
ifAbsent: () => [event.authorId],
|
||||
)
|
||||
.unlock,
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (event.type == "m.room.redaction") {
|
||||
final controller = await future;
|
||||
final redactsId = event.content["redacts"];
|
||||
final originalMessage = controller.messages.firstWhereOrNull(
|
||||
(message) => message.id == redactsId,
|
||||
);
|
||||
if (!ref.mounted) return;
|
||||
|
||||
if (originalMessage != null) {
|
||||
return await controller.removeMessage(originalMessage);
|
||||
}
|
||||
|
||||
final redacts = ref
|
||||
.read(SelectedRoomController.provider)
|
||||
?.events
|
||||
.firstWhere((event) => event.eventId == redactsId);
|
||||
|
||||
if (redacts?.type == "m.reaction") {
|
||||
final message = controller.messages.firstWhereOrNull(
|
||||
(message) =>
|
||||
message.id == redacts!.content["m.relates_to"]?["event_id"],
|
||||
);
|
||||
final key = redacts!.content["m.relates_to"]?["key"];
|
||||
if (message == null || key == null || !ref.mounted) return;
|
||||
|
||||
return await controller.updateMessage(
|
||||
message,
|
||||
message.copyWith(
|
||||
reactions: IMap(message.reactions)
|
||||
.update(
|
||||
key,
|
||||
(reactors) =>
|
||||
IList(reactors).remove(redacts.authorId).unlock,
|
||||
)
|
||||
.where((_, value) => value.isNotEmpty)
|
||||
.unlock,
|
||||
),
|
||||
);
|
||||
}
|
||||
} else {
|
||||
final message = await ref.watch(
|
||||
MessageController.provider(
|
||||
MessageConfig(event: event, room: room!, includeEdits: true),
|
||||
).future,
|
||||
);
|
||||
if (event.relationType == "m.replace") {
|
||||
final controller = await future;
|
||||
final oldMessage = controller.messages.firstWhereOrNull(
|
||||
(element) => element.id == event.relatesTo,
|
||||
);
|
||||
if (oldMessage == null || message == null || !ref.mounted) return;
|
||||
|
||||
return await controller.updateMessage(
|
||||
oldMessage,
|
||||
message.copyWith(
|
||||
id: oldMessage.id,
|
||||
replyToMessageId: oldMessage.replyToMessageId,
|
||||
metadata: {
|
||||
...(oldMessage.metadata ?? {}),
|
||||
...(message.metadata ?? {})
|
||||
.toIMap()
|
||||
.where((key, value) => value != null)
|
||||
.unlock,
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
if (message != null && ref.mounted) {
|
||||
await insertMessage(message);
|
||||
}
|
||||
}
|
||||
}
|
||||
}, weak: true).close,
|
||||
);
|
||||
|
||||
ref.onDispose(controller.dispose);
|
||||
|
||||
// While there are under 20 messages, try up to load more messages until theres no more or we have 20 messages.
|
||||
for (var more = true; more == true && controller.messages.length < 20;) {
|
||||
more = await loadOlder(controller);
|
||||
}
|
||||
|
||||
return controller;
|
||||
}
|
||||
|
||||
Future<void> insertMessage(Message message) async {
|
||||
final controller = await future;
|
||||
final oldMessage = message.metadata?["txnId"] == null
|
||||
? null
|
||||
: controller.messages.firstWhereOrNull(
|
||||
(element) =>
|
||||
element.metadata?["txnId"] == message.metadata?["txnId"],
|
||||
);
|
||||
|
||||
return oldMessage == null
|
||||
? controller.insertMessage(message)
|
||||
: controller.updateMessage(oldMessage, message);
|
||||
}
|
||||
|
||||
Future<void> deleteMessage(Message message, {String? reason}) => ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.redactEvent(
|
||||
RedactEventRequest(eventId: message.id, roomId: roomId, reason: reason),
|
||||
if (!room.hasFetchedState) {
|
||||
final state = await client.getRoomState(
|
||||
GetRoomStateRequest(roomId: roomId),
|
||||
);
|
||||
|
||||
Future<bool> loadOlder([InMemoryChatController? chatController]) async {
|
||||
await ref.read(RoomsController.provider.notifier).addState(roomId, state);
|
||||
}
|
||||
|
||||
// While there are under 20 events, try to load more
|
||||
// until there's no more or the conditions are met.
|
||||
if (room.hasMore && room.timeline.length < 20) {
|
||||
loadOlder();
|
||||
}
|
||||
|
||||
return room.timeline
|
||||
.toEntryIList(compare: (a, b) => (b?.key ?? 0).compareTo(a?.key ?? 0))
|
||||
.map((entry) {
|
||||
if (entry.value == null) return null;
|
||||
|
||||
final foundEvent = room.events[entry.value!];
|
||||
|
||||
final editedEvent =
|
||||
foundEvent == null || foundEvent.lastEditRowId == 0
|
||||
? null
|
||||
: room.events[foundEvent.lastEditRowId];
|
||||
|
||||
return editedEvent == null
|
||||
? foundEvent
|
||||
: foundEvent?.copyWith(content: editedEvent.content);
|
||||
})
|
||||
.nonNulls
|
||||
.toIList();
|
||||
}
|
||||
|
||||
Future<void> deleteMessage(Event event, {String? reason}) => ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.redactEvent(
|
||||
RedactEventRequest(
|
||||
eventId: event.eventId,
|
||||
roomId: roomId,
|
||||
reason: reason,
|
||||
),
|
||||
);
|
||||
|
||||
Future<bool> loadOlder() async {
|
||||
final timelineKeys = ref
|
||||
.read(RoomsController.provider.select((value) => value[roomId]))
|
||||
?.timeline
|
||||
.keys;
|
||||
final response = await ref
|
||||
.watch(ClientController.provider.notifier)
|
||||
.paginate(
|
||||
PaginateRequest(
|
||||
roomId: roomId,
|
||||
maxTimelineId: ref
|
||||
.read(RoomsController.provider)[roomId]
|
||||
?.timeline
|
||||
.firstOrNull
|
||||
?.timelineRowId,
|
||||
maxTimelineId: timelineKeys?.isNotEmpty == true
|
||||
? timelineKeys?.reduce(min)
|
||||
: null,
|
||||
),
|
||||
);
|
||||
|
||||
|
|
@ -226,42 +94,22 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
.update(
|
||||
IMap({
|
||||
roomId: Room(
|
||||
events: response.events.addAll(response.relatedEvents),
|
||||
events: IMap.fromIterable(
|
||||
response.events.addAll(response.relatedEvents),
|
||||
keyMapper: (event) => event.rowId,
|
||||
valueMapper: (event) => event,
|
||||
),
|
||||
hasMore: response.hasMore,
|
||||
timeline: response.events
|
||||
.map(
|
||||
(event) => TimelineRowTuple(
|
||||
timelineRowId: event.timelineRowId,
|
||||
eventRowId: event.rowId,
|
||||
),
|
||||
)
|
||||
.toIList(),
|
||||
timeline: IMap.fromIterable(
|
||||
response.events,
|
||||
keyMapper: (event) => event.timelineRowId,
|
||||
valueMapper: (event) => event.rowId,
|
||||
),
|
||||
),
|
||||
}),
|
||||
const ISet.empty(),
|
||||
addToNewEvents: false,
|
||||
);
|
||||
|
||||
final room = ref.read(RoomsController.provider)[roomId];
|
||||
if (room != null) {
|
||||
final messages = await ref.watch(
|
||||
MessagesController.provider(
|
||||
MessagesConfig(room: room, events: response.events.reversed),
|
||||
).future,
|
||||
);
|
||||
|
||||
final controller = chatController ?? await future;
|
||||
await controller.insertAllMessages(
|
||||
messages
|
||||
.where(
|
||||
(newMessage) => !controller.messages.any(
|
||||
(message) => message.id == newMessage.id,
|
||||
),
|
||||
)
|
||||
.toList(),
|
||||
index: 0,
|
||||
);
|
||||
}
|
||||
return response.hasMore;
|
||||
}
|
||||
|
||||
|
|
@ -270,7 +118,7 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
bool shouldMention = true,
|
||||
required IList<Tag> tags,
|
||||
required RelationType relationType,
|
||||
Message? relation,
|
||||
Event? relation,
|
||||
}) async {
|
||||
var taggedMessage = text;
|
||||
|
||||
|
|
@ -285,7 +133,6 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
}
|
||||
|
||||
final client = ref.watch(ClientController.provider.notifier);
|
||||
final room = ref.read(RoomsController.provider)[roomId];
|
||||
final event = await client.sendMessage(
|
||||
SendMessageRequest(
|
||||
roomId: roomId,
|
||||
|
|
@ -294,52 +141,46 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
if (shouldMention == true &&
|
||||
relation != null &&
|
||||
relationType == RelationType.reply)
|
||||
relation.authorId,
|
||||
relation.sender,
|
||||
].toIList(),
|
||||
room: taggedMessage.contains("@room"),
|
||||
),
|
||||
text: taggedMessage,
|
||||
relation: relation == null
|
||||
? null
|
||||
: Relation(eventId: relation.id, relationType: relationType),
|
||||
),
|
||||
);
|
||||
final message = room == null
|
||||
? null
|
||||
: await ref.watch(
|
||||
MessageController.provider(
|
||||
MessageConfig(room: room, event: event),
|
||||
).future,
|
||||
);
|
||||
|
||||
if (message != null) insertMessage(message);
|
||||
}
|
||||
|
||||
Future<void> scrollToMessage(Message message) async {
|
||||
final controller = await future;
|
||||
Future<void> setFlashing(bool flashing) => controller.updateMessage(
|
||||
message,
|
||||
message.copyWith(
|
||||
metadata: {...(message.metadata ?? {}), "flashing": flashing},
|
||||
: Relation(eventId: relation.eventId, relationType: relationType),
|
||||
),
|
||||
);
|
||||
|
||||
await setFlashing(true);
|
||||
Timer(Duration(seconds: 1), () => setFlashing(false));
|
||||
|
||||
return await controller.scrollToMessage(message.id);
|
||||
// TODO: Add new event to timeline whilst its sending
|
||||
// ref
|
||||
// .watch(RoomsController.provider.notifier)
|
||||
// .update(
|
||||
// {
|
||||
// roomId: Room(
|
||||
// events: [event].toIList(),
|
||||
// timeline: [
|
||||
// TimelineRowTuple(
|
||||
// timelineRowId: event.timelineRowId,
|
||||
// eventRowId: event.rowId,
|
||||
// ),
|
||||
// ].toIList(),
|
||||
// ),
|
||||
// }.toIMap(),
|
||||
// const ISet.empty(),
|
||||
// );
|
||||
}
|
||||
|
||||
Future<void> removeReaction(
|
||||
String reaction,
|
||||
Message message,
|
||||
Event event,
|
||||
String userId,
|
||||
) async {
|
||||
final client = ref.watch(ClientController.provider.notifier);
|
||||
final allReactionEvents = await client.getRelatedEvents(
|
||||
GetRelatedEventsRequest(
|
||||
roomId: roomId,
|
||||
eventId: message.id,
|
||||
eventId: event.eventId,
|
||||
relationType: "m.annotation",
|
||||
),
|
||||
);
|
||||
|
|
@ -349,9 +190,11 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
.toIList();
|
||||
|
||||
final reactionEvent = reactionEvents?.firstWhereOrNull(
|
||||
(event) =>
|
||||
event.authorId == userId &&
|
||||
event.content["m.relates_to"]?["key"] == reaction,
|
||||
(event) => switch (event.content) {
|
||||
ReactionContent(:final key) =>
|
||||
key == reaction && event.sender == userId,
|
||||
_ => false,
|
||||
},
|
||||
);
|
||||
|
||||
if (reactionEvent != null) {
|
||||
|
|
@ -363,7 +206,7 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
}
|
||||
}
|
||||
|
||||
Future<void> sendReaction(String reaction, Message message) async {
|
||||
Future<void> sendReaction(String reaction, Event event) async {
|
||||
final client = ref.watch(ClientController.provider.notifier);
|
||||
|
||||
await client.sendEvent(
|
||||
|
|
@ -372,7 +215,7 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
type: "m.reaction",
|
||||
content: {
|
||||
"m.relates_to": {
|
||||
"event_id": message.id,
|
||||
"event_id": event.eventId,
|
||||
"rel_type": "m.annotation",
|
||||
"key": reaction,
|
||||
},
|
||||
|
|
@ -384,7 +227,7 @@ class RoomChatController extends AsyncNotifier<InMemoryChatController> {
|
|||
}
|
||||
|
||||
static final provider = AsyncNotifierProvider.family
|
||||
.autoDispose<RoomChatController, InMemoryChatController, String>(
|
||||
.autoDispose<RoomChatController, IList<Event>, String>(
|
||||
RoomChatController.new,
|
||||
);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,9 +1,8 @@
|
|||
import "package:collection/collection.dart";
|
||||
import "dart:isolate";
|
||||
|
||||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/controllers/new_events_controller.dart";
|
||||
import "package:nexus/helpers/extensions/mxc_to_https.dart";
|
||||
import "package:nexus/models/event.dart";
|
||||
import "package:nexus/models/read_receipt.dart";
|
||||
import "package:nexus/models/room.dart";
|
||||
|
||||
|
|
@ -11,55 +10,50 @@ class RoomsController extends Notifier<IMap<String, Room>> {
|
|||
@override
|
||||
IMap<String, Room> build() => const IMap.empty();
|
||||
|
||||
void update(
|
||||
IMap<String, Room> rooms,
|
||||
ISet<String> leftRooms, {
|
||||
bool addToNewEvents = true,
|
||||
}) {
|
||||
final homeserver =
|
||||
ref.watch(
|
||||
ClientStateController.provider.select(
|
||||
(value) => value?.homeserverUrl,
|
||||
),
|
||||
) ??
|
||||
"";
|
||||
Future<void> addState(
|
||||
String roomId,
|
||||
IList<Event> state, {
|
||||
bool isMembers = false,
|
||||
}) async => update(
|
||||
{
|
||||
roomId: Room(
|
||||
events: IMap.fromEntries(
|
||||
state.map((event) => MapEntry(event.rowId, event)),
|
||||
),
|
||||
hasFetchedState: true,
|
||||
hasFetchedMembers: isMembers,
|
||||
state: await Isolate.run(() {
|
||||
final newState = state.fold(
|
||||
const IMap<String, IMap<String, int>>.empty(),
|
||||
(previousValue, stateEvent) => previousValue.add(
|
||||
stateEvent.type,
|
||||
(previousValue[stateEvent.type] ?? const IMap.empty()).add(
|
||||
stateEvent.stateKey!,
|
||||
stateEvent.rowId,
|
||||
),
|
||||
),
|
||||
);
|
||||
return newState;
|
||||
}),
|
||||
),
|
||||
}.toIMap(),
|
||||
const ISet.empty(),
|
||||
);
|
||||
|
||||
void update(IMap<String, Room> rooms, ISet<String> leftRooms) {
|
||||
final merged = rooms.entries.fold(state, (acc, entry) {
|
||||
final roomId = entry.key;
|
||||
final incoming = entry.value;
|
||||
final existing = acc[roomId];
|
||||
|
||||
final events = existing?.events.updateById(
|
||||
incoming.events,
|
||||
(item) => item.eventId,
|
||||
);
|
||||
|
||||
if (addToNewEvents) {
|
||||
ref
|
||||
.watch(NewEventsController.provider(roomId).notifier)
|
||||
.add(
|
||||
incoming.timeline
|
||||
.map(
|
||||
(timelineTuple) => events?.firstWhereOrNull(
|
||||
(event) => timelineTuple.eventRowId == event.rowId,
|
||||
),
|
||||
)
|
||||
.nonNulls
|
||||
.toIList(),
|
||||
);
|
||||
}
|
||||
|
||||
return acc.add(
|
||||
roomId,
|
||||
existing?.copyWith(
|
||||
hasMore: incoming.hasMore,
|
||||
metadata:
|
||||
incoming.metadata?.copyWith(
|
||||
avatar:
|
||||
incoming.metadata?.avatar?.mxcToHttps(homeserver) ??
|
||||
existing.metadata?.avatar,
|
||||
) ??
|
||||
existing.metadata,
|
||||
events: events!,
|
||||
metadata: incoming.metadata ?? existing.metadata,
|
||||
events: incoming.events.isEmpty
|
||||
? existing.events
|
||||
: existing.events.addAll(incoming.events),
|
||||
state: incoming.state.entries.fold(
|
||||
existing.state,
|
||||
(previousValue, event) => previousValue.add(
|
||||
|
|
@ -69,15 +63,14 @@ class RoomsController extends Notifier<IMap<String, Room>> {
|
|||
),
|
||||
),
|
||||
),
|
||||
timeline:
|
||||
(incoming.reset
|
||||
? incoming.timeline
|
||||
: existing.timeline.updateById(
|
||||
incoming.timeline,
|
||||
(item) => item.timelineRowId,
|
||||
))
|
||||
.sortedBy((element) => element.timelineRowId)
|
||||
.toIList(),
|
||||
reset: false,
|
||||
hasFetchedMembers:
|
||||
incoming.hasFetchedMembers || existing.hasFetchedMembers,
|
||||
hasFetchedState:
|
||||
incoming.hasFetchedState || existing.hasFetchedState,
|
||||
timeline: (incoming.reset
|
||||
? incoming.timeline
|
||||
: existing.timeline.addAll(incoming.timeline)),
|
||||
receipts: incoming.receipts.entries.fold(
|
||||
existing.receipts,
|
||||
(receiptAcc, event) => receiptAcc.add(
|
||||
|
|
@ -88,11 +81,7 @@ class RoomsController extends Notifier<IMap<String, Room>> {
|
|||
),
|
||||
),
|
||||
) ??
|
||||
incoming.copyWith(
|
||||
metadata: incoming.metadata?.copyWith(
|
||||
avatar: incoming.metadata?.avatar?.mxcToHttps(homeserver),
|
||||
),
|
||||
),
|
||||
incoming,
|
||||
);
|
||||
});
|
||||
|
||||
|
|
@ -100,6 +89,7 @@ class RoomsController extends Notifier<IMap<String, Room>> {
|
|||
merged,
|
||||
(acc, roomId) => acc.remove(roomId),
|
||||
);
|
||||
|
||||
state = prunedList;
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -1,24 +0,0 @@
|
|||
import "package:collection/collection.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/key_controller.dart";
|
||||
import "package:nexus/controllers/selected_space_controller.dart";
|
||||
import "package:nexus/models/room.dart";
|
||||
|
||||
class SelectedRoomController extends Notifier<Room?> {
|
||||
@override
|
||||
Room? build() {
|
||||
final space = ref.watch(SelectedSpaceController.provider);
|
||||
final selectedRoomId = ref.watch(
|
||||
KeyController.provider(KeyController.roomKey),
|
||||
);
|
||||
|
||||
return space.children.firstWhereOrNull(
|
||||
(room) => room.metadata?.id == selectedRoomId,
|
||||
) ??
|
||||
space.children.firstOrNull;
|
||||
}
|
||||
|
||||
static final provider = NotifierProvider<SelectedRoomController, Room?>(
|
||||
SelectedRoomController.new,
|
||||
);
|
||||
}
|
||||
|
|
@ -1,22 +0,0 @@
|
|||
import "package:collection/collection.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/key_controller.dart";
|
||||
import "package:nexus/controllers/spaces_controller.dart";
|
||||
import "package:nexus/models/space.dart";
|
||||
|
||||
class SelectedSpaceController extends Notifier<Space> {
|
||||
@override
|
||||
Space build() {
|
||||
final spaces = ref.watch(SpacesController.provider);
|
||||
final selectedSpaceId = ref.watch(
|
||||
KeyController.provider(KeyController.spaceKey),
|
||||
);
|
||||
|
||||
return spaces.firstWhereOrNull((space) => space.id == selectedSpaceId) ??
|
||||
spaces.first;
|
||||
}
|
||||
|
||||
static final provider = NotifierProvider<SelectedSpaceController, Space>(
|
||||
SelectedSpaceController.new,
|
||||
);
|
||||
}
|
||||
|
|
@ -1,18 +1,20 @@
|
|||
import "dart:convert";
|
||||
import "package:flutter_chat_core/flutter_chat_core.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:http/http.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/controllers/header_controller.dart";
|
||||
import "package:nexus/helpers/extensions/mxc_to_https.dart";
|
||||
import "package:nexus/models/open_graph_data.dart";
|
||||
|
||||
class UrlPreviewController extends AsyncNotifier<LinkPreviewData?> {
|
||||
class UrlPreviewController extends AsyncNotifier<OpenGraphData?> {
|
||||
final String link;
|
||||
UrlPreviewController(this.link);
|
||||
|
||||
@override
|
||||
Future<LinkPreviewData?> build() async {
|
||||
final homeserver = ref.watch(ClientStateController.provider)?.homeserverUrl;
|
||||
Future<OpenGraphData?> build() async {
|
||||
final homeserver = ref.watch(
|
||||
ClientStateController.provider.select((value) => value?.homeserverUrl),
|
||||
);
|
||||
|
||||
if (homeserver != null && !link.contains("matrix.to")) {
|
||||
{
|
||||
|
|
@ -25,27 +27,14 @@ class UrlPreviewController extends AsyncNotifier<LinkPreviewData?> {
|
|||
|
||||
if (response.statusCode == 200) {
|
||||
final decodedValue = json.decode(response.body);
|
||||
if (decodedValue is! Map<String, dynamic>) return null;
|
||||
|
||||
final mxc = decodedValue["og:image"];
|
||||
final image = mxc == null
|
||||
? null
|
||||
: Uri.tryParse(mxc)?.mxcToHttps(homeserver);
|
||||
|
||||
return LinkPreviewData(
|
||||
link: link,
|
||||
title: decodedValue["og:title"],
|
||||
description: decodedValue["og:description"],
|
||||
image: image == null
|
||||
? null
|
||||
: ImagePreviewData(
|
||||
url: image.toString(),
|
||||
width:
|
||||
(decodedValue["og:image:width"] as int?)?.toDouble() ??
|
||||
0,
|
||||
height:
|
||||
(decodedValue["og:image:height"] as int?)?.toDouble() ??
|
||||
0,
|
||||
),
|
||||
);
|
||||
return OpenGraphData.fromJson(decodedValue).copyWith(imageUrl: image);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -54,7 +43,7 @@ class UrlPreviewController extends AsyncNotifier<LinkPreviewData?> {
|
|||
}
|
||||
|
||||
static final provider = AsyncNotifierProvider.autoDispose
|
||||
.family<UrlPreviewController, LinkPreviewData?, String>(
|
||||
.family<UrlPreviewController, OpenGraphData?, String>(
|
||||
UrlPreviewController.new,
|
||||
);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -4,37 +4,44 @@ import "package:flutter_riverpod/flutter_riverpod.dart";
|
|||
import "package:nexus/controllers/members_controller.dart";
|
||||
import "package:nexus/controllers/profile_controller.dart";
|
||||
import "package:nexus/helpers/extensions/get_localpart.dart";
|
||||
import "package:nexus/models/membership.dart";
|
||||
import "package:nexus/models/configs/user_config.dart";
|
||||
import "package:nexus/models/content/membership.dart";
|
||||
import "package:nexus/models/membership_status.dart";
|
||||
|
||||
class UserController extends AsyncNotifier<Membership?> {
|
||||
final String userId;
|
||||
UserController(this.userId);
|
||||
class UserController extends AsyncNotifier<MembershipContent> {
|
||||
final UserConfig config;
|
||||
UserController(this.config);
|
||||
|
||||
@override
|
||||
Future<Membership?> build() async {
|
||||
final member = await ref.watch(
|
||||
MembersController.provider.selectAsync(
|
||||
(value) =>
|
||||
value.firstWhereOrNull((membership) => membership.userId == userId),
|
||||
),
|
||||
Future<MembershipContent> build() async {
|
||||
final member = config.roomId == null
|
||||
? null
|
||||
: await ref.watch(
|
||||
MembersController.provider(config.roomId!).selectAsync(
|
||||
(value) => value.firstWhereOrNull(
|
||||
(membership) => membership.stateKey == config.userId,
|
||||
),
|
||||
),
|
||||
);
|
||||
|
||||
if (member?.content case final MembershipContent content) {
|
||||
return content;
|
||||
}
|
||||
|
||||
final profile = await ref.watch(
|
||||
ProfileController.provider(config.userId).future,
|
||||
);
|
||||
|
||||
if (member != null) return member;
|
||||
|
||||
final profile = await ref.watch(ProfileController.provider(userId).future);
|
||||
return Membership(
|
||||
return MembershipContent(
|
||||
status: MembershipStatus.leave,
|
||||
avatarUrl: profile.avatarUrl == null
|
||||
? null
|
||||
: Uri.tryParse(profile.avatarUrl!),
|
||||
displayName: profile.displayName ?? userId.localpart,
|
||||
userId: userId,
|
||||
avatarUrl: profile.avatarUrl,
|
||||
displayName: profile.displayName ?? config.userId.localpart,
|
||||
);
|
||||
}
|
||||
|
||||
static final provider =
|
||||
AsyncNotifierProvider.family<UserController, Membership?, String>(
|
||||
UserController.new,
|
||||
);
|
||||
AsyncNotifierProvider.family<
|
||||
UserController,
|
||||
MembershipContent,
|
||||
UserConfig
|
||||
>(UserController.new);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -2,6 +2,10 @@ import "package:collection/collection.dart";
|
|||
import "package:fast_immutable_collections/fast_immutable_collections.dart";
|
||||
import "package:flutter_riverpod/flutter_riverpod.dart";
|
||||
import "package:nexus/controllers/client_state_controller.dart";
|
||||
import "package:nexus/models/content/content.dart";
|
||||
import "package:nexus/models/content/membership.dart";
|
||||
import "package:nexus/models/content/power_levels.dart";
|
||||
import "package:nexus/models/membership_status.dart";
|
||||
import "package:nexus/models/room.dart";
|
||||
|
||||
class ViaController extends Notifier<String> {
|
||||
|
|
@ -21,23 +25,29 @@ class ViaController extends Notifier<String> {
|
|||
|
||||
addUserId(ref.watch(ClientStateController.provider)?.userId);
|
||||
|
||||
final powerLevels = room.events.firstWhereOrNull(
|
||||
(event) => event.rowId == room.state["m.room.power_levels"]?[""],
|
||||
);
|
||||
final powerLevelsEventId = room.state[EventType.powerLevels.type]?[""];
|
||||
final powerLevels = powerLevelsEventId == null
|
||||
? null
|
||||
: room.events[powerLevelsEventId];
|
||||
|
||||
for (final userId in IMap(powerLevels?.content["users"]).keys) {
|
||||
addUserId(userId);
|
||||
if (servers.length >= 5) break;
|
||||
if (powerLevels?.content case PowerLevelsContent(:final users)) {
|
||||
for (final userId in users.keys) {
|
||||
addUserId(userId);
|
||||
if (servers.length >= 5) break;
|
||||
}
|
||||
}
|
||||
|
||||
final members = room.state["m.room.member"]?.values.toIList();
|
||||
final members = room.state[EventType.membership.type]?.values.toIList();
|
||||
for (var i = 0; servers.length < 5; i++) {
|
||||
final member = room.events.firstWhereOrNull(
|
||||
(event) => event.rowId == members?.getOrNull(i),
|
||||
);
|
||||
final membershipEventId = members?.getOrNull(i);
|
||||
final member = membershipEventId == null
|
||||
? null
|
||||
: room.events[membershipEventId];
|
||||
|
||||
if (member?.content["membership"] == "join") {
|
||||
addUserId(member?.stateKey);
|
||||
if (member?.content case MembershipContent(:final status)) {
|
||||
if (status == MembershipStatus.join) {
|
||||
addUserId(member?.stateKey);
|
||||
}
|
||||
}
|
||||
|
||||
if (members?.getOrNull(i) == null) break;
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue