[Room] Add support for RoomSummary
This commit is contained in:
parent
d2c4f7c6ed
commit
98bb93e657
|
@ -107,7 +107,7 @@ class Client {
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; i < versions.length; i++) {
|
for (int i = 0; i < versions.length; i++) {
|
||||||
if (versions[i] == "r0.4.0")
|
if (versions[i] == "r0.5.0")
|
||||||
break;
|
break;
|
||||||
else if (i == versions.length - 1) {
|
else if (i == versions.length - 1) {
|
||||||
connection.onError.add(ErrorResponse(errcode: "NO_SUPPORT", error: ""));
|
connection.onError.add(ErrorResponse(errcode: "NO_SUPPORT", error: ""));
|
||||||
|
|
|
@ -343,6 +343,12 @@ class Connection {
|
||||||
prev_batch = room["timeline"]["prev_batch"];
|
prev_batch = room["timeline"]["prev_batch"];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
RoomSummary summary;
|
||||||
|
|
||||||
|
if (room["summary"] is Map<String, dynamic>) {
|
||||||
|
summary = RoomSummary.fromJson(room["summary"]);
|
||||||
|
}
|
||||||
|
|
||||||
RoomUpdate update = RoomUpdate(
|
RoomUpdate update = RoomUpdate(
|
||||||
id: id,
|
id: id,
|
||||||
membership: membership,
|
membership: membership,
|
||||||
|
@ -350,6 +356,7 @@ class Connection {
|
||||||
highlight_count: highlight_count,
|
highlight_count: highlight_count,
|
||||||
limitedTimeline: limitedTimeline,
|
limitedTimeline: limitedTimeline,
|
||||||
prev_batch: prev_batch,
|
prev_batch: prev_batch,
|
||||||
|
summary: summary,
|
||||||
);
|
);
|
||||||
client.store?.storeRoomUpdate(update);
|
client.store?.storeRoomUpdate(update);
|
||||||
onRoomUpdate.add(update);
|
onRoomUpdate.add(update);
|
||||||
|
|
|
@ -43,9 +43,6 @@ class Room {
|
||||||
/// The name of the room if set by a participant.
|
/// The name of the room if set by a participant.
|
||||||
String name;
|
String name;
|
||||||
|
|
||||||
/// Whether this room has a name or the name is generated by member names.
|
|
||||||
bool hasName = false;
|
|
||||||
|
|
||||||
/// The topic of the room if set by a participant.
|
/// The topic of the room if set by a participant.
|
||||||
String topic;
|
String topic;
|
||||||
|
|
||||||
|
@ -89,6 +86,10 @@ class Room {
|
||||||
/// The needed power levels for all actions.
|
/// The needed power levels for all actions.
|
||||||
Map<String, int> powerLevels = {};
|
Map<String, int> powerLevels = {};
|
||||||
|
|
||||||
|
List<String> mHeroes;
|
||||||
|
int mJoinedMemberCount;
|
||||||
|
int mInvitedMemberCount;
|
||||||
|
|
||||||
Event lastEvent;
|
Event lastEvent;
|
||||||
|
|
||||||
/// Your current client instance.
|
/// Your current client instance.
|
||||||
|
@ -107,7 +108,6 @@ class Room {
|
||||||
this.id,
|
this.id,
|
||||||
this.membership,
|
this.membership,
|
||||||
this.name,
|
this.name,
|
||||||
this.hasName = false,
|
|
||||||
this.topic,
|
this.topic,
|
||||||
this.avatar,
|
this.avatar,
|
||||||
this.notificationCount,
|
this.notificationCount,
|
||||||
|
@ -125,8 +125,28 @@ class Room {
|
||||||
this.powerLevels,
|
this.powerLevels,
|
||||||
this.lastEvent,
|
this.lastEvent,
|
||||||
this.client,
|
this.client,
|
||||||
|
this.mHeroes,
|
||||||
|
this.mInvitedMemberCount,
|
||||||
|
this.mJoinedMemberCount,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/// Calculates the displayname. First checks if there is a name, then checks for a canonical alias and
|
||||||
|
/// then generates a name from the heroes.
|
||||||
|
String get displayname {
|
||||||
|
if (name != null && !name.isEmpty) return name;
|
||||||
|
if (canonicalAlias != null &&
|
||||||
|
!canonicalAlias.isEmpty &&
|
||||||
|
canonicalAlias.length > 3)
|
||||||
|
return canonicalAlias.substring(1, canonicalAlias.length).split(":")[0];
|
||||||
|
if (mHeroes.length > 0) {
|
||||||
|
String displayname = "";
|
||||||
|
for (int i = 0; i < mHeroes.length; i++)
|
||||||
|
displayname += User(mHeroes[i]).calcDisplayname() + ", ";
|
||||||
|
return displayname.substring(0, displayname.length - 2);
|
||||||
|
}
|
||||||
|
return "Empty chat";
|
||||||
|
}
|
||||||
|
|
||||||
/// The last message sent to this room.
|
/// The last message sent to this room.
|
||||||
String get lastMessage {
|
String get lastMessage {
|
||||||
if (lastEvent != null)
|
if (lastEvent != null)
|
||||||
|
@ -378,23 +398,13 @@ class Room {
|
||||||
/// Returns a Room from a json String which comes normally from the store.
|
/// Returns a Room from a json String which comes normally from the store.
|
||||||
static Future<Room> getRoomFromTableRow(
|
static Future<Room> getRoomFromTableRow(
|
||||||
Map<String, dynamic> row, Client matrix) async {
|
Map<String, dynamic> row, Client matrix) async {
|
||||||
bool newHasName = false;
|
|
||||||
String name = row["topic"];
|
|
||||||
if (name == "" && !row["canonical_alias"].isEmpty)
|
|
||||||
name = row["canonical_alias"];
|
|
||||||
else if (name == "")
|
|
||||||
name = await matrix.store?.getChatNameFromMemberNames(row["id"]) ?? "";
|
|
||||||
else
|
|
||||||
newHasName = true;
|
|
||||||
|
|
||||||
String avatarUrl = row["avatar_url"];
|
String avatarUrl = row["avatar_url"];
|
||||||
if (avatarUrl == "")
|
if (avatarUrl == "")
|
||||||
avatarUrl = await matrix.store?.getAvatarFromSingleChat(row["id"]) ?? "";
|
avatarUrl = await matrix.store?.getAvatarFromSingleChat(row["id"]) ?? "";
|
||||||
|
|
||||||
return Room(
|
return Room(
|
||||||
id: row["id"],
|
id: row["id"],
|
||||||
name: name,
|
name: row["topic"],
|
||||||
hasName: newHasName,
|
|
||||||
membership: Membership.values
|
membership: Membership.values
|
||||||
.firstWhere((e) => e.toString() == 'Membership.' + row["membership"]),
|
.firstWhere((e) => e.toString() == 'Membership.' + row["membership"]),
|
||||||
topic: row["description"],
|
topic: row["description"],
|
||||||
|
@ -411,6 +421,9 @@ class Room {
|
||||||
historyVisibility: row["history_visibility"],
|
historyVisibility: row["history_visibility"],
|
||||||
joinRules: row["join_rules"],
|
joinRules: row["join_rules"],
|
||||||
canonicalAlias: row["canonical_alias"],
|
canonicalAlias: row["canonical_alias"],
|
||||||
|
mInvitedMemberCount: row["invited_member_count"],
|
||||||
|
mJoinedMemberCount: row["joined_member_count"],
|
||||||
|
mHeroes: row["heroes"]?.split(",") ?? [],
|
||||||
powerLevels: {
|
powerLevels: {
|
||||||
"power_events_default": row["power_events_default"],
|
"power_events_default": row["power_events_default"],
|
||||||
"power_state_default": row["power_state_default"],
|
"power_state_default": row["power_state_default"],
|
||||||
|
|
|
@ -92,7 +92,9 @@ class RoomList {
|
||||||
prev_batch: chatUpdate.prev_batch,
|
prev_batch: chatUpdate.prev_batch,
|
||||||
highlightCount: chatUpdate.highlight_count,
|
highlightCount: chatUpdate.highlight_count,
|
||||||
notificationCount: chatUpdate.notification_count,
|
notificationCount: chatUpdate.notification_count,
|
||||||
hasName: false,
|
mHeroes: chatUpdate.summary?.mHeroes,
|
||||||
|
mJoinedMemberCount: chatUpdate.summary?.mJoinedMemberCount,
|
||||||
|
mInvitedMemberCount: chatUpdate.summary?.mInvitedMemberCount,
|
||||||
);
|
);
|
||||||
rooms.insert(position, newRoom);
|
rooms.insert(position, newRoom);
|
||||||
if (onInsert != null) onInsert(position);
|
if (onInsert != null) onInsert(position);
|
||||||
|
@ -110,6 +112,14 @@ class RoomList {
|
||||||
rooms[j].highlightCount != chatUpdate.highlight_count)) {
|
rooms[j].highlightCount != chatUpdate.highlight_count)) {
|
||||||
rooms[j].notificationCount = chatUpdate.notification_count;
|
rooms[j].notificationCount = chatUpdate.notification_count;
|
||||||
rooms[j].highlightCount = chatUpdate.highlight_count;
|
rooms[j].highlightCount = chatUpdate.highlight_count;
|
||||||
|
if (chatUpdate.summary != null) {
|
||||||
|
if (chatUpdate.summary.mHeroes != null)
|
||||||
|
rooms[j].mHeroes = chatUpdate.summary.mHeroes;
|
||||||
|
if (chatUpdate.summary.mJoinedMemberCount != null)
|
||||||
|
rooms[j].mJoinedMemberCount = chatUpdate.summary.mJoinedMemberCount;
|
||||||
|
if (chatUpdate.summary.mInvitedMemberCount != null)
|
||||||
|
rooms[j].mInvitedMemberCount = chatUpdate.summary.mInvitedMemberCount;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
sortAndUpdate();
|
sortAndUpdate();
|
||||||
}
|
}
|
||||||
|
@ -156,8 +166,6 @@ class RoomList {
|
||||||
// Update the room avatar
|
// Update the room avatar
|
||||||
rooms[j].avatar = MxContent(eventUpdate.content["content"]["url"]);
|
rooms[j].avatar = MxContent(eventUpdate.content["content"]["url"]);
|
||||||
}
|
}
|
||||||
if (eventUpdate.eventType == "m.room.member" && !rooms[j].hasName)
|
|
||||||
updateMemberName(j);
|
|
||||||
sortAndUpdate();
|
sortAndUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,11 +174,6 @@ class RoomList {
|
||||||
b.timeCreated.toTimeStamp().compareTo(a.timeCreated.toTimeStamp()));
|
b.timeCreated.toTimeStamp().compareTo(a.timeCreated.toTimeStamp()));
|
||||||
if (onUpdate != null) onUpdate();
|
if (onUpdate != null) onUpdate();
|
||||||
}
|
}
|
||||||
|
|
||||||
void updateMemberName(int position) async {
|
|
||||||
rooms[position].name =
|
|
||||||
await client.store.getChatNameFromMemberNames(rooms[position].id);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef onRoomListUpdateCallback = void Function();
|
typedef onRoomListUpdateCallback = void Function();
|
||||||
|
|
|
@ -55,7 +55,7 @@ class Store {
|
||||||
_init() async {
|
_init() async {
|
||||||
var databasePath = await getDatabasesPath();
|
var databasePath = await getDatabasesPath();
|
||||||
String path = p.join(databasePath, "FluffyMatrix.db");
|
String path = p.join(databasePath, "FluffyMatrix.db");
|
||||||
_db = await openDatabase(path, version: 10,
|
_db = await openDatabase(path, version: 11,
|
||||||
onCreate: (Database db, int version) async {
|
onCreate: (Database db, int version) async {
|
||||||
await createTables(db);
|
await createTables(db);
|
||||||
}, onUpgrade: (Database db, int oldVersion, int newVersion) async {
|
}, onUpgrade: (Database db, int oldVersion, int newVersion) async {
|
||||||
|
@ -164,18 +164,32 @@ class Store {
|
||||||
// Insert the chat into the database if not exists
|
// Insert the chat into the database if not exists
|
||||||
txn.rawInsert(
|
txn.rawInsert(
|
||||||
"INSERT OR IGNORE INTO Rooms " +
|
"INSERT OR IGNORE INTO Rooms " +
|
||||||
"VALUES(?, ?, '', 0, 0, '', '', '', 0, '', '', '', '', '', '', '', '', 0, 50, 50, 0, 50, 50, 0, 50, 100, 50, 50, 50, 100) ",
|
"VALUES(?, ?, '', 0, 0, 0, 0, '', '', '', '', 0, '', '', '', '', '', '', '', '', 0, 50, 50, 0, 50, 50, 0, 50, 100, 50, 50, 50, 100) ",
|
||||||
[roomUpdate.id, roomUpdate.membership.toString().split('.').last]);
|
[roomUpdate.id, roomUpdate.membership.toString().split('.').last]);
|
||||||
|
|
||||||
// Update the notification counts and the limited timeline boolean
|
// Update the notification counts and the limited timeline boolean and the summary
|
||||||
txn.rawUpdate(
|
String updateQuery =
|
||||||
"UPDATE Rooms SET highlight_count=?, notification_count=?, membership=? WHERE id=? ",
|
"UPDATE Rooms SET highlight_count=?, notification_count=?, membership=?";
|
||||||
[
|
List<dynamic> updateArgs = [
|
||||||
roomUpdate.highlight_count,
|
roomUpdate.highlight_count,
|
||||||
roomUpdate.notification_count,
|
roomUpdate.notification_count,
|
||||||
roomUpdate.membership.toString().split('.').last,
|
roomUpdate.membership.toString().split('.').last
|
||||||
roomUpdate.id
|
];
|
||||||
]);
|
if (roomUpdate.summary?.mJoinedMemberCount != null) {
|
||||||
|
updateQuery += ", joined_member_count=?";
|
||||||
|
updateArgs.add(roomUpdate.summary.mJoinedMemberCount);
|
||||||
|
}
|
||||||
|
if (roomUpdate.summary?.mInvitedMemberCount != null) {
|
||||||
|
updateQuery += ", invited_member_count=?";
|
||||||
|
updateArgs.add(roomUpdate.summary.mInvitedMemberCount);
|
||||||
|
}
|
||||||
|
if (roomUpdate.summary?.mHeroes != null) {
|
||||||
|
updateQuery += ", heroes=?";
|
||||||
|
updateArgs.add(roomUpdate.summary.mHeroes.join(","));
|
||||||
|
}
|
||||||
|
updateQuery += " WHERE id=?";
|
||||||
|
updateArgs.add(roomUpdate.id);
|
||||||
|
txn.rawUpdate(updateQuery, updateArgs);
|
||||||
|
|
||||||
// Is the timeline limited? Then all previous messages should be
|
// Is the timeline limited? Then all previous messages should be
|
||||||
// removed from the database!
|
// removed from the database!
|
||||||
|
@ -691,6 +705,9 @@ class Store {
|
||||||
'topic TEXT, ' +
|
'topic TEXT, ' +
|
||||||
'highlight_count INTEGER, ' +
|
'highlight_count INTEGER, ' +
|
||||||
'notification_count INTEGER, ' +
|
'notification_count INTEGER, ' +
|
||||||
|
'joined_member_count INTEGER, ' +
|
||||||
|
'invited_member_count INTEGER, ' +
|
||||||
|
'heroes TEXT, ' +
|
||||||
'prev_batch TEXT, ' +
|
'prev_batch TEXT, ' +
|
||||||
'avatar_url TEXT, ' +
|
'avatar_url TEXT, ' +
|
||||||
'draft TEXT, ' +
|
'draft TEXT, ' +
|
||||||
|
|
|
@ -46,6 +46,8 @@ class RoomUpdate {
|
||||||
/// Represents the current position of the client in the room history.
|
/// Represents the current position of the client in the room history.
|
||||||
final String prev_batch;
|
final String prev_batch;
|
||||||
|
|
||||||
|
final RoomSummary summary;
|
||||||
|
|
||||||
RoomUpdate({
|
RoomUpdate({
|
||||||
this.id,
|
this.id,
|
||||||
this.membership,
|
this.membership,
|
||||||
|
@ -53,5 +55,21 @@ class RoomUpdate {
|
||||||
this.highlight_count,
|
this.highlight_count,
|
||||||
this.limitedTimeline,
|
this.limitedTimeline,
|
||||||
this.prev_batch,
|
this.prev_batch,
|
||||||
|
this.summary,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class RoomSummary {
|
||||||
|
List<String> mHeroes;
|
||||||
|
int mJoinedMemberCount;
|
||||||
|
int mInvitedMemberCount;
|
||||||
|
|
||||||
|
RoomSummary(
|
||||||
|
{this.mHeroes, this.mJoinedMemberCount, this.mInvitedMemberCount});
|
||||||
|
|
||||||
|
RoomSummary.fromJson(Map<String, dynamic> json) {
|
||||||
|
mHeroes = json['m.heroes']?.cast<String>();
|
||||||
|
mJoinedMemberCount = json['m.joined_member_count'];
|
||||||
|
mInvitedMemberCount = json['m.invited_member_count'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -138,7 +138,14 @@ class FakeMatrixApi extends MockClient {
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"/client/versions": (var req) => {
|
"/client/versions": (var req) => {
|
||||||
"versions": ["r0.0.1", "r0.1.0", "r0.2.0", "r0.3.0", "r0.4.0"],
|
"versions": [
|
||||||
|
"r0.0.1",
|
||||||
|
"r0.1.0",
|
||||||
|
"r0.2.0",
|
||||||
|
"r0.3.0",
|
||||||
|
"r0.4.0",
|
||||||
|
"r0.5.0"
|
||||||
|
],
|
||||||
"unstable_features": {"m.lazy_load_members": true},
|
"unstable_features": {"m.lazy_load_members": true},
|
||||||
},
|
},
|
||||||
"/client/r0/login": (var req) => {
|
"/client/r0/login": (var req) => {
|
||||||
|
|
|
@ -68,8 +68,13 @@ void main() {
|
||||||
final String formatted_body = "<b>Hello</b> World";
|
final String formatted_body = "<b>Hello</b> World";
|
||||||
final String contentJson =
|
final String contentJson =
|
||||||
'{"msgtype":"$msgtype","body":"$body","formatted_body":"$formatted_body"}';
|
'{"msgtype":"$msgtype","body":"$body","formatted_body":"$formatted_body"}';
|
||||||
|
final List<String> heroes = [
|
||||||
|
"@alice:matrix.org",
|
||||||
|
"@bob:example.com",
|
||||||
|
"@charley:example.org"
|
||||||
|
];
|
||||||
|
|
||||||
final Map<String, dynamic> jsonObj = {
|
Map<String, dynamic> jsonObj = {
|
||||||
"id": id,
|
"id": id,
|
||||||
"membership": membership.toString().split('.').last,
|
"membership": membership.toString().split('.').last,
|
||||||
"topic": name,
|
"topic": name,
|
||||||
|
@ -101,6 +106,9 @@ void main() {
|
||||||
"power_event_name": 0,
|
"power_event_name": 0,
|
||||||
"power_event_power_levels": 0,
|
"power_event_power_levels": 0,
|
||||||
"content_json": contentJson,
|
"content_json": contentJson,
|
||||||
|
"joined_member_count": notificationCount,
|
||||||
|
"invited_member_count": notificationCount,
|
||||||
|
"heroes": heroes.join(","),
|
||||||
};
|
};
|
||||||
|
|
||||||
room = await Room.getRoomFromTableRow(jsonObj, matrix);
|
room = await Room.getRoomFromTableRow(jsonObj, matrix);
|
||||||
|
@ -108,6 +116,7 @@ void main() {
|
||||||
expect(room.id, id);
|
expect(room.id, id);
|
||||||
expect(room.membership, membership);
|
expect(room.membership, membership);
|
||||||
expect(room.name, name);
|
expect(room.name, name);
|
||||||
|
expect(room.displayname, name);
|
||||||
expect(room.topic, topic);
|
expect(room.topic, topic);
|
||||||
expect(room.avatar.mxc, "");
|
expect(room.avatar.mxc, "");
|
||||||
expect(room.notificationCount, notificationCount);
|
expect(room.notificationCount, notificationCount);
|
||||||
|
@ -127,6 +136,16 @@ void main() {
|
||||||
room.powerLevels.forEach((String key, int value) {
|
room.powerLevels.forEach((String key, int value) {
|
||||||
expect(value, 0);
|
expect(value, 0);
|
||||||
});
|
});
|
||||||
|
expect(room.mJoinedMemberCount, notificationCount);
|
||||||
|
expect(room.mInvitedMemberCount, notificationCount);
|
||||||
|
expect(room.mHeroes, heroes);
|
||||||
|
|
||||||
|
jsonObj["topic"] = "";
|
||||||
|
room = await Room.getRoomFromTableRow(jsonObj, matrix);
|
||||||
|
expect(room.displayname, "testroom");
|
||||||
|
jsonObj["canonical_alias"] = "";
|
||||||
|
room = await Room.getRoomFromTableRow(jsonObj, matrix);
|
||||||
|
expect(room.displayname, "alice, bob, charley");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("sendReadReceipt", () async {
|
test("sendReadReceipt", () async {
|
||||||
|
|
Loading…
Reference in a new issue