You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

227 lines
7.8 KiB

import 'package:flutter/cupertino.dart';
4 months ago
import 'package:huixiang/constant.dart';
import 'package:huixiang/im/database/message.dart';
4 months ago
import 'package:huixiang/im/database/migration.dart';
import 'package:shared_preferences/shared_preferences.dart';
4 months ago
import 'package:sqflite/sqflite.dart';
import '../../data/im_user.dart';
4 months ago
class HxDatabase {
Database? db;
4 months ago
void open({String? key}) async {
4 months ago
// _migrations.add(Migration(3, 4, (Database database) async {
// await database.execute('ALTER TABLE ImUser ADD COLUMN IF NOT EXISTS `isTop` INTEGER DEFAULT 0');
4 months ago
// }));
4 months ago
String databaseName = 'hx.db';
if (key?.isNotEmpty ?? false) {
databaseName = 'hx_$key.db';
}
4 months ago
await openDatabase(databaseName, version: 1, onCreate: (Database db, int version) async {
db.execute('CREATE TABLE IF NOT EXISTS `Message` (`id` INTEGER, `conversationId` VARCHAR(40), `fromId` VARCHAR(20), `toId` VARCHAR(20), `replyId` VARCHAR(20), `content` TEXT, `attach` TEXT, `msgType` INTEGER, `time` VARCHAR(20), `state` INTEGER, `isDelete` INTEGER, PRIMARY KEY (`id`))');
db.execute('CREATE TABLE IF NOT EXISTS `ImUser` (`id` INTEGER, `mid` VARCHAR(20), `nickname` VARCHAR(20), `avatar` VARCHAR(200), `phone` VARCHAR(200), `isDelete` INTEGER, `isTop` INTEGER, PRIMARY KEY (`id`))');
}, onConfigure: (database) async {
await database.execute('PRAGMA foreign_keys = ON');
4 months ago
debugPrint("database-version: ${await database.getVersion()}");
}, onUpgrade: (database, startVersion, endVersion) async {
await runMigrations(database, startVersion, endVersion, _migrations);
}, onOpen: (Database db) {
this.db = db;
});
4 months ago
}
void close() {
db?.close();
4 months ago
}
_dbIsOpen() async {
if (db == null || !db!.isOpen) {
var sp = await SharedPreferences.getInstance();
open(key: sp.getString("userId"));
4 months ago
}
}
Future<Message?> lastMessage(String conversationId) async {
await _dbIsOpen();
4 months ago
String sql = 'SELECT * FROM Message WHERE conversationId = ? ORDER BY time DESC LIMIT 1';
List<Message>? messages = await db?.rawQuery(sql, [conversationId]).then((value) {
4 months ago
return value.map((e) {
4 months ago
debugPrint("Message: $e");
4 months ago
return Message.fromJson(e);
}).toList();
}, onError: (error) {
4 months ago
debugPrint("Message_error: $error");
4 months ago
});
return (messages?.isNotEmpty ?? false) ? (messages?.first ?? null) : null;
4 months ago
}
Future<List<Message>?> queryList() async{
await _dbIsOpen();
String sql = '''SELECT *
FROM `Message`
WHERE ROWID IN (
SELECT ROWID
FROM (
SELECT ROWID, conversationId, MAX(`time`) AS max_time
FROM `Message`
GROUP BY conversationId
) AS grouped_messages
WHERE max_time = (
SELECT MAX(`time`)
FROM `Message`
WHERE conversationId = grouped_messages.conversationId
)
)
ORDER BY `time` DESC;''';
return db?.rawQuery(sql).then((value) {
4 months ago
return value.map((e) {
debugPrint("Message: $e");
4 months ago
return Message.fromJson(e);
}).toList();
}, onError: (error) {
4 months ago
debugPrint("Message-error: $error");
4 months ago
});
}
Future<List<Message>?> queryUList(conversationId, {int page = 1, int pageSize = 10}) async{
await _dbIsOpen();
4 months ago
int start = (page - 1) * pageSize;
String sql = 'SELECT * FROM Message WHERE conversationId = ? ORDER BY time DESC LIMIT ?, ?';
return db?.rawQuery(sql, [conversationId, start, pageSize]).then((value) {
4 months ago
return value.map((e) => Message.fromJson(e)).toList();
}, onError: (error) {
4 months ago
debugPrint("Message-error: $error");
4 months ago
});
}
Future<List<Message>?> queryTList(conversationId) async{
await _dbIsOpen();
String sql = 'SELECT *, time / 300000 * 300000 AS time_interval FROM Message WHERE conversationId = ? GROUP BY time_interval ORDER BY time DESC';
return db?.rawQuery(sql, [conversationId]).then((value) {
return value.map((e) => Message.fromJson(e)).toList();
}, onError: (error) {
4 months ago
debugPrint("Message-error: $error");
});
}
Future<Map<String, int>> messageUnreadCount(List<String> conversationIds) async {
await _dbIsOpen();
4 months ago
String userStr = conversationIds.join("','");
debugPrint("userStr: $userStr");
List<Message>? messages = await db?.query("Message",
4 months ago
where: "conversationId IN ('$userStr') AND state = 0 AND isDelete = 0",
whereArgs: [],
4 months ago
).then((value) {
4 months ago
return value.map((e) => Message.fromJson(e)).toList();
}, onError: (error) {
debugPrint("Message-error: $error");
});
4 months ago
return (messages??[]).lGroupBy((p) => p.conversationId).mGroupCount;
4 months ago
}
Future<List<Map>?> queryListAll() async{
await _dbIsOpen();
4 months ago
String sql = 'SELECT * FROM Message ORDER BY time DESC';
return db?.rawQuery(sql);
4 months ago
}
Future<int?> deleteByUser(String conversationId) async {
return db?.delete("Message",where: "conversationId = ?", whereArgs: [conversationId]);
}
Future<int?> deleteByMsgId(String id) async {
return db?.delete("Message",where: "id = ?", whereArgs: [id]);
}
Future<int?> deleteAll() async {
return db?.delete("Message");
4 months ago
}
update(Map<String, Object?> message) async{
await _dbIsOpen();
4 months ago
debugPrint("Message_insert: $message");
return db?.update("Message", message,
where: 'id = ?', whereArgs: [message['id']]);
4 months ago
}
Future<int?> insert(Map<String, Object?> message) async {
await _dbIsOpen();
4 months ago
debugPrint("Message_insert: $message");
return db?.insert("Message", message);
4 months ago
}
4 months ago
4 months ago
/// update message read state
readMessage(String conversationId) async{
await _dbIsOpen();
db?.update("Message", {"state": 1},
where: "conversationId = ? AND state = 0 AND isDelete = 0",
whereArgs: [conversationId]);
4 months ago
}
4 months ago
Future<int?> insertOrUpdateImUser(Map<String, dynamic> imUserMap) async {
await _dbIsOpen();
debugPrint("imUser_insert: $imUserMap");
if ((await queryImUserById(imUserMap['mid'])) == null)
return db?.insert("ImUser", imUserMap);
else
return db?.update("ImUser", imUserMap,
where: 'mid = ?', whereArgs: [imUserMap['mid']]);
4 months ago
}
Future<List<ImUser>?> queryImUser(List<String> userIds) async {
await _dbIsOpen();
String query =
'SELECT * FROM ImUser WHERE mid IN (${userIds.map((mid) => "'$mid'").join(',')})';
return db?.rawQuery(query).then((value) {
4 months ago
return value.map((e) => ImUser.fromJson(e)).toList();
4 months ago
}, onError: (error) {
debugPrint("ImUser_error: $error");
4 months ago
});
}
Future<ImUser?> queryImUserById(String? userId) async {
await _dbIsOpen();
List<ImUser>? imUser = await db?.query("ImUser",
distinct: true, where: "mid = ?", whereArgs: [userId]).then((value) {
4 months ago
return value.map((e) => ImUser.fromJson(e)).toList();
4 months ago
}, onError: (error) {
debugPrint("ImUser_error: $error");
4 months ago
});
return (imUser?.isNotEmpty ?? false) ? (imUser?.first ?? null) : null;
4 months ago
}
4 months ago
final List<Migration> _migrations = [];
addMigrations(List<Migration> migrations) {
_migrations.addAll(migrations);
return this;
}
4 months ago
Future<void> runMigrations(
final Database migrationDatabase,
final int startVersion,
final int endVersion,
final List<Migration> migrations,
) async {
4 months ago
final relevantMigrations = migrations
.where((migration) => migration.startVersion >= startVersion)
.toList()
..sort(
(first, second) => first.startVersion.compareTo(second.startVersion));
4 months ago
if (relevantMigrations.isEmpty ||
relevantMigrations.last.endVersion != endVersion) {
throw StateError(
'There is no migration supplied to update the database to the current version.'
' Aborting the migration.',
4 months ago
);
}
4 months ago
for (final migration in relevantMigrations) {
await migration.migrate(migrationDatabase);
}
}
}