WuKongIMFlutterSDK/lib/db/conversation.dart

335 lines
13 KiB
Dart

import 'dart:collection';
import 'dart:convert';
import 'package:sqflite/sqflite.dart';
import 'package:wukongimfluttersdk/common/logs.dart';
import 'package:wukongimfluttersdk/db/channel.dart';
import 'package:wukongimfluttersdk/db/const.dart';
import 'package:wukongimfluttersdk/entity/channel.dart';
import '../entity/conversation.dart';
import 'wk_db_helper.dart';
class ConversationDB {
ConversationDB._privateConstructor();
static final ConversationDB _instance = ConversationDB._privateConstructor();
static ConversationDB get shared => _instance;
final String extraCols =
"IFNULL(${WKDBConst.tableConversationExtra}.browse_to,0) AS browse_to,IFNULL(${WKDBConst.tableConversationExtra}.keep_message_seq,0) AS keep_message_seq,IFNULL(${WKDBConst.tableConversationExtra}.keep_offset_y,0) AS keep_offset_y,IFNULL(${WKDBConst.tableConversationExtra}.draft,'') AS draft,IFNULL(${WKDBConst.tableConversationExtra}.draft_updated_at,0) AS draft_updated_at,IFNULL(${WKDBConst.tableConversationExtra}.version,0) AS extra_version";
final String channelCols =
"${WKDBConst.tableChannel}.channel_remark,${WKDBConst.tableChannel}.channel_name,${WKDBConst.tableChannel}.top,${WKDBConst.tableChannel}.mute,${WKDBConst.tableChannel}.save,${WKDBConst.tableChannel}.status as channel_status,${WKDBConst.tableChannel}.forbidden,${WKDBConst.tableChannel}.invite,${WKDBConst.tableChannel}.follow,${WKDBConst.tableChannel}.is_deleted as channel_is_deleted,${WKDBConst.tableChannel}.show_nick,${WKDBConst.tableChannel}.avatar,${WKDBConst.tableChannel}.avatar_cache_key,${WKDBConst.tableChannel}.online,${WKDBConst.tableChannel}.last_offline,${WKDBConst.tableChannel}.category,${WKDBConst.tableChannel}.receipt,${WKDBConst.tableChannel}.robot,${WKDBConst.tableChannel}.parent_channel_id AS c_parent_channel_id,${WKDBConst.tableChannel}.parent_channel_type AS c_parent_channel_type,${WKDBConst.tableChannel}.version AS channel_version,${WKDBConst.tableChannel}.remote_extra AS channel_remote_extra,${WKDBConst.tableChannel}.extra AS channel_extra";
Future<List<WKUIConversationMsg>> queryAll() async {
String sql =
"SELECT ${WKDBConst.tableConversation}.*,$channelCols,$extraCols FROM ${WKDBConst.tableConversation} LEFT JOIN ${WKDBConst.tableChannel} ON ${WKDBConst.tableConversation}.channel_id = ${WKDBConst.tableChannel}.channel_id AND ${WKDBConst.tableConversation}.channel_type = ${WKDBConst.tableChannel}.channel_type LEFT JOIN ${WKDBConst.tableConversationExtra} ON ${WKDBConst.tableConversation}.channel_id=${WKDBConst.tableConversationExtra}.channel_id AND ${WKDBConst.tableConversation}.channel_type=${WKDBConst.tableConversationExtra}.channel_type where ${WKDBConst.tableConversation}.is_deleted=0 order by last_msg_timestamp desc";
List<WKUIConversationMsg> list = [];
if (WKDBHelper.shared.getDB() == null) {
return list;
}
List<Map<String, Object?>> results =
await WKDBHelper.shared.getDB()!.rawQuery(sql);
if (results.isNotEmpty) {
for (Map<String, Object?> data in results) {
WKConversationMsg msg = WKDBConst.serializeCoversation(data);
WKChannel wkChannel = WKDBConst.serializeChannel(data);
wkChannel.remoteExtraMap =
WKDBConst.readDynamic(data, 'channel_remote_extra');
wkChannel.localExtra = WKDBConst.readDynamic(data, 'channel_extra');
WKUIConversationMsg uiMsg = getUIMsg(msg);
uiMsg.setWkChannel(wkChannel);
list.add(uiMsg);
}
}
return list;
}
Future<bool> delete(String channelID, int channelType) async {
if (WKDBHelper.shared.getDB() == null) {
return false;
}
Map<String, dynamic> data = HashMap<String, Object>();
data['is_deleted'] = 1;
int row = await WKDBHelper.shared.getDB()!.update(
WKDBConst.tableConversation, data,
where: "channel_id=? and channel_type=?",
whereArgs: [channelID, channelType]);
return row > 0;
}
Future<WKUIConversationMsg?> insertOrUpdateWithConvMsg(
WKConversationMsg conversationMsg) async {
if (WKDBHelper.shared.getDB() == null) {
return null;
}
int row;
WKConversationMsg? lastMsg = await queryMsgByMsgChannelId(
conversationMsg.channelID, conversationMsg.channelType);
if (lastMsg == null || lastMsg.channelID.isEmpty) {
row = await WKDBHelper.shared.getDB()!.insert(
WKDBConst.tableConversation, getMap(conversationMsg, false),
conflictAlgorithm: ConflictAlgorithm.replace);
} else {
conversationMsg.localExtraMap ??= lastMsg.localExtraMap;
conversationMsg.unreadCount =
lastMsg.unreadCount + conversationMsg.unreadCount;
row = await WKDBHelper.shared.getDB()!.update(
WKDBConst.tableConversation, getMap(conversationMsg, false),
where: "channel_id=? and channel_type=?",
whereArgs: [conversationMsg.channelID, conversationMsg.channelType]);
}
if (row > 0) {
return getUIMsg(conversationMsg);
}
return null;
}
Future<WKConversationMsg?> queryMsgByMsgChannelId(
String channelId, int channelType) async {
WKConversationMsg? msg;
if (WKDBHelper.shared.getDB() == null) {
return msg;
}
List<Map<String, Object?>> list = await WKDBHelper.shared.getDB()!.query(
WKDBConst.tableConversation,
where: "channel_id=? and channel_type=?",
whereArgs: [channelId, channelType]);
if (list.isNotEmpty) {
msg = WKDBConst.serializeCoversation(list[0]);
}
return msg;
}
Future<int> queryAllUnreadCount() async {
int count = 0;
var channels = await ChannelDB.shared.queryWithMuted();
var channelIds = [];
var sql = "";
List<Map<String, Object?>>? list;
if (channels.isNotEmpty) {
for (var channel in channels) {
channelIds.add(channel.channelID);
}
sql =
"select SUM(unread_count) count from ${WKDBConst.tableConversation} where channel_id not in (${WKDBConst.getPlaceholders(channelIds.length)})";
list = await WKDBHelper.shared.getDB()!.rawQuery(sql, channelIds);
} else {
sql =
"select SUM(unread_count) count from ${WKDBConst.tableConversation}";
list = await WKDBHelper.shared.getDB()?.rawQuery(sql);
}
if (list == null || list.isEmpty) {
return count;
}
dynamic data = list[0];
count = WKDBConst.readInt(data, 'count');
Logs.error('总数量$count');
return count;
}
Future<int> getMaxVersion() async {
int maxVersion = 0;
if (WKDBHelper.shared.getDB() == null) {
return maxVersion;
}
String sql =
"select max(version) version from ${WKDBConst.tableConversation} limit 0, 1";
List<Map<String, Object?>> list =
await WKDBHelper.shared.getDB()!.rawQuery(sql);
if (list.isNotEmpty) {
dynamic data = list[0];
maxVersion = WKDBConst.readInt(data, 'version');
}
return maxVersion;
}
Future<String> getLastMsgSeqs() async {
String lastMsgSeqs = "";
if (WKDBHelper.shared.getDB() == null) {
return lastMsgSeqs;
}
String sql =
"select GROUP_CONCAT(channel_id||':'||channel_type||':'|| last_seq,'|') synckey from (select *,(select max(message_seq) from ${WKDBConst.tableMessage} where ${WKDBConst.tableMessage}.channel_id=${WKDBConst.tableConversation}.channel_id and ${WKDBConst.tableMessage}.channel_type=${WKDBConst.tableConversation}.channel_type limit 1) last_seq from ${WKDBConst.tableConversation}) cn where channel_id<>'' AND is_deleted=0";
List<Map<String, Object?>> list =
await WKDBHelper.shared.getDB()!.rawQuery(sql);
if (list.isNotEmpty) {
dynamic data = list[0];
lastMsgSeqs = WKDBConst.readString(data, 'synckey');
}
return lastMsgSeqs;
}
Future<List<WKConversationMsg>> queryWithChannelIds(
List<String> channelIds) async {
List<WKConversationMsg> list = [];
if (WKDBHelper.shared.getDB() == null) {
return list;
}
List<Map<String, Object?>> results = await WKDBHelper.shared.getDB()!.query(
WKDBConst.tableConversation,
where:
"channel_id in (${WKDBConst.getPlaceholders(channelIds.length)})",
whereArgs: channelIds);
if (results.isNotEmpty) {
for (Map<String, Object?> data in results) {
list.add(WKDBConst.serializeCoversation(data));
}
}
return list;
}
insetMsgs(List<WKConversationMsg> list) async {
if (WKDBHelper.shared.getDB() == null) {
return;
}
List<Map<String, dynamic>> insertList = [];
for (WKConversationMsg msg in list) {
insertList.add(getMap(msg, true));
}
WKDBHelper.shared.getDB()!.transaction((txn) async {
if (insertList.isNotEmpty) {
for (int i = 0; i < insertList.length; i++) {
txn.insert(WKDBConst.tableConversation, insertList[i],
conflictAlgorithm: ConflictAlgorithm.replace);
}
}
});
}
insertMsgList(List<WKConversationMsg> list) async {
if (WKDBHelper.shared.getDB() == null) {
return;
}
List<String> channelIds = [];
for (var i = 0; i < list.length; i++) {
if (list[i].channelID != '') {
channelIds.add(list[i].channelID);
}
}
List<WKConversationMsg> existList = await queryWithChannelIds(channelIds);
List<Map<String, dynamic>> insertList = [];
List<Map<String, dynamic>> updateList = [];
for (WKConversationMsg msg in list) {
bool isAdd = true;
if (existList.isNotEmpty) {
for (var i = 0; i < existList.length; i++) {
if (existList[i].channelID == msg.channelID &&
existList[i].channelType == msg.channelType) {
updateList.add(getMap(msg, true));
isAdd = false;
break;
}
}
}
if (isAdd) {
insertList.add(getMap(msg, true));
}
}
if (insertList.isNotEmpty || updateList.isNotEmpty) {
WKDBHelper.shared.getDB()!.transaction((txn) async {
if (insertList.isNotEmpty) {
for (int i = 0; i < insertList.length; i++) {
txn.insert(WKDBConst.tableConversation, insertList[i],
conflictAlgorithm: ConflictAlgorithm.replace);
}
}
if (updateList.isNotEmpty) {
for (Map<String, dynamic> value in updateList) {
txn.update(WKDBConst.tableConversation, value,
where: "channel_id=? and channel_type=?",
whereArgs: [value['channel_id'], value['channel_type']]);
}
}
});
}
}
clearAll() {
if (WKDBHelper.shared.getDB() == null) {
return;
}
WKDBHelper.shared.getDB()!.delete(WKDBConst.tableConversation);
}
Future<int> queryExtraMaxVersion() async {
int maxVersion = 0;
if (WKDBHelper.shared.getDB() == null) {
return maxVersion;
}
String sql =
"select max(version) version from ${WKDBConst.tableConversationExtra}";
List<Map<String, Object?>> list =
await WKDBHelper.shared.getDB()!.rawQuery(sql);
if (list.isNotEmpty) {
dynamic data = list[0];
maxVersion = WKDBConst.readInt(data, 'version');
}
return maxVersion;
}
Future<int> clearAllRedDot() async {
if (WKDBHelper.shared.getDB() == null) {
return 0;
}
var map = <String, Object>{};
map['unread_count'] = 0;
return await WKDBHelper.shared
.getDB()!
.update(WKDBConst.tableConversation, map, where: "unread_count>0");
}
Future<int> updateWithField(
dynamic map, String channelID, int channelType) async {
if (WKDBHelper.shared.getDB() == null) {
return 0;
}
return await WKDBHelper.shared.getDB()!.update(
WKDBConst.tableConversation, map,
where: "channel_id=? and channel_type=?",
whereArgs: [channelID, channelType]);
}
WKUIConversationMsg getUIMsg(WKConversationMsg conversationMsg) {
WKUIConversationMsg msg = WKUIConversationMsg();
msg.lastMsgSeq = conversationMsg.lastMsgSeq;
msg.clientMsgNo = conversationMsg.lastClientMsgNO;
msg.unreadCount = conversationMsg.unreadCount;
msg.lastMsgTimestamp = conversationMsg.lastMsgTimestamp;
msg.channelID = conversationMsg.channelID;
msg.channelType = conversationMsg.channelType;
msg.isDeleted = conversationMsg.isDeleted;
msg.localExtraMap = conversationMsg.localExtraMap;
msg.parentChannelID = conversationMsg.parentChannelID;
msg.parentChannelType = conversationMsg.parentChannelType;
msg.setRemoteMsgExtra(conversationMsg.msgExtra);
return msg;
}
Map<String, dynamic> getMap(WKConversationMsg msg, bool isSync) {
Map<String, dynamic> data = HashMap<String, Object>();
data['channel_id'] = msg.channelID;
data['channel_type'] = msg.channelType;
data['last_client_msg_no'] = msg.lastClientMsgNO;
data['last_msg_timestamp'] = msg.lastMsgTimestamp;
data['last_msg_seq'] = msg.lastMsgSeq;
data['unread_count'] = msg.unreadCount;
data['parent_channel_id'] = msg.parentChannelID;
data['parent_channel_type'] = msg.parentChannelType;
data['is_deleted'] = msg.isDeleted;
if (msg.localExtraMap == null) {
data['extra'] = '';
} else {
data['extra'] = jsonEncode(msg.localExtraMap);
}
if (isSync) {
data['version'] = msg.version;
}
return data;
}
}