✨ Join channel hint
🗃️ Realm local db
			
			
This commit is contained in:
		| @@ -866,5 +866,9 @@ | ||||
|     "other": "Maximum ask for {} steps authenticate" | ||||
|   }, | ||||
|   "authAlwaysRisky": "Always Risky", | ||||
|   "authAlwaysRiskyDescription": "Always ask for the highest steps count of authentication when logging in." | ||||
|   "authAlwaysRiskyDescription": "Always ask for the highest steps count of authentication when logging in.", | ||||
|   "chatUnjoined": "Unjoined Channel", | ||||
|   "chatUnjoinedDescription": "You haven't joined this channel, so you can't send messages either view messages in it.", | ||||
|   "chatUnjoinedPublicDescription": "Fortunately, this is a public channel, so you can join it as you want.", | ||||
|   "chatJoin": "Join the Channel" | ||||
| } | ||||
|   | ||||
| @@ -864,5 +864,9 @@ | ||||
|     "other": "登入时最多要求 {} 步验证" | ||||
|   }, | ||||
|   "authAlwaysRisky": "总是风险", | ||||
|   "authAlwaysRiskyDescription": "在登入时始终按最高标准要求验证。" | ||||
|   "authAlwaysRiskyDescription": "在登入时始终按最高标准要求验证。", | ||||
|   "chatUnjoined": "未加入频道", | ||||
|   "chatUnjoinedDescription": "你没有加入这个频道,所以你也无法发送消息或者查看这个频道中的消息。", | ||||
|   "chatUnjoinedPublicDescription": "但幸运的是,这是一个公开频道,所以你可以主动加入。", | ||||
|   "chatJoin": "加入频道" | ||||
| } | ||||
|   | ||||
							
								
								
									
										1
									
								
								drift_schemas/my_database/drift_schema_v4.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								drift_schemas/my_database/drift_schema_v4.json
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -6,10 +6,12 @@ import 'package:surface/database/attachment.dart'; | ||||
| import 'package:surface/database/chat.dart'; | ||||
| import 'package:surface/database/database.steps.dart'; | ||||
| import 'package:surface/database/keypair.dart'; | ||||
| import 'package:surface/database/realm.dart'; | ||||
| import 'package:surface/database/sticker.dart'; | ||||
| import 'package:surface/types/chat.dart'; | ||||
| import 'package:surface/types/attachment.dart'; | ||||
| import 'package:surface/types/account.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
|  | ||||
| part 'database.g.dart'; | ||||
|  | ||||
| @@ -22,12 +24,13 @@ part 'database.g.dart'; | ||||
|   SnLocalAttachment, | ||||
|   SnLocalSticker, | ||||
|   SnLocalStickerPack, | ||||
|   SnLocalRealm, | ||||
| ]) | ||||
| class AppDatabase extends _$AppDatabase { | ||||
|   AppDatabase([QueryExecutor? e]) : super(e ?? _openConnection()); | ||||
|  | ||||
|   @override | ||||
|   int get schemaVersion => 3; | ||||
|   int get schemaVersion => 4; | ||||
|  | ||||
|   static QueryExecutor _openConnection() { | ||||
|     return driftDatabase( | ||||
| @@ -49,6 +52,10 @@ class AppDatabase extends _$AppDatabase { | ||||
|         // Nothing else to do here | ||||
|       }, from2To3: (m, schema) async { | ||||
|         // Nothing else to do here, too | ||||
|       }, from3To4: (m, schema) async { | ||||
|         m.createTable(schema.snLocalRealm); | ||||
|         m.createIndex(schema.idxRealmAccount); | ||||
|         m.createIndex(schema.idxRealmAlias); | ||||
|       }), | ||||
|     ); | ||||
|   } | ||||
|   | ||||
| @@ -2454,6 +2454,351 @@ class SnLocalStickerPackCompanion | ||||
|   } | ||||
| } | ||||
|  | ||||
| class $SnLocalRealmTable extends SnLocalRealm | ||||
|     with TableInfo<$SnLocalRealmTable, SnLocalRealmData> { | ||||
|   @override | ||||
|   final GeneratedDatabase attachedDatabase; | ||||
|   final String? _alias; | ||||
|   $SnLocalRealmTable(this.attachedDatabase, [this._alias]); | ||||
|   static const VerificationMeta _idMeta = const VerificationMeta('id'); | ||||
|   @override | ||||
|   late final GeneratedColumn<int> id = GeneratedColumn<int>( | ||||
|       'id', aliasedName, false, | ||||
|       hasAutoIncrement: true, | ||||
|       type: DriftSqlType.int, | ||||
|       requiredDuringInsert: false, | ||||
|       defaultConstraints: | ||||
|           GeneratedColumn.constraintIsAlways('PRIMARY KEY AUTOINCREMENT')); | ||||
|   static const VerificationMeta _aliasMeta = const VerificationMeta('alias'); | ||||
|   @override | ||||
|   late final GeneratedColumn<String> alias = GeneratedColumn<String>( | ||||
|       'alias', aliasedName, false, | ||||
|       type: DriftSqlType.string, | ||||
|       requiredDuringInsert: true, | ||||
|       defaultConstraints: GeneratedColumn.constraintIsAlways('UNIQUE')); | ||||
|   @override | ||||
|   late final GeneratedColumnWithTypeConverter<SnRealm, String> content = | ||||
|       GeneratedColumn<String>('content', aliasedName, false, | ||||
|               type: DriftSqlType.string, requiredDuringInsert: true) | ||||
|           .withConverter<SnRealm>($SnLocalRealmTable.$convertercontent); | ||||
|   static const VerificationMeta _accountIdMeta = | ||||
|       const VerificationMeta('accountId'); | ||||
|   @override | ||||
|   late final GeneratedColumn<int> accountId = GeneratedColumn<int>( | ||||
|       'account_id', aliasedName, false, | ||||
|       type: DriftSqlType.int, requiredDuringInsert: true); | ||||
|   static const VerificationMeta _createdAtMeta = | ||||
|       const VerificationMeta('createdAt'); | ||||
|   @override | ||||
|   late final GeneratedColumn<DateTime> createdAt = GeneratedColumn<DateTime>( | ||||
|       'created_at', aliasedName, false, | ||||
|       type: DriftSqlType.dateTime, | ||||
|       requiredDuringInsert: false, | ||||
|       defaultValue: currentDateAndTime); | ||||
|   static const VerificationMeta _cacheExpiredAtMeta = | ||||
|       const VerificationMeta('cacheExpiredAt'); | ||||
|   @override | ||||
|   late final GeneratedColumn<DateTime> cacheExpiredAt = | ||||
|       GeneratedColumn<DateTime>('cache_expired_at', aliasedName, false, | ||||
|           type: DriftSqlType.dateTime, requiredDuringInsert: true); | ||||
|   @override | ||||
|   List<GeneratedColumn> get $columns => | ||||
|       [id, alias, content, accountId, createdAt, cacheExpiredAt]; | ||||
|   @override | ||||
|   String get aliasedName => _alias ?? actualTableName; | ||||
|   @override | ||||
|   String get actualTableName => $name; | ||||
|   static const String $name = 'sn_local_realm'; | ||||
|   @override | ||||
|   VerificationContext validateIntegrity(Insertable<SnLocalRealmData> instance, | ||||
|       {bool isInserting = false}) { | ||||
|     final context = VerificationContext(); | ||||
|     final data = instance.toColumns(true); | ||||
|     if (data.containsKey('id')) { | ||||
|       context.handle(_idMeta, id.isAcceptableOrUnknown(data['id']!, _idMeta)); | ||||
|     } | ||||
|     if (data.containsKey('alias')) { | ||||
|       context.handle( | ||||
|           _aliasMeta, alias.isAcceptableOrUnknown(data['alias']!, _aliasMeta)); | ||||
|     } else if (isInserting) { | ||||
|       context.missing(_aliasMeta); | ||||
|     } | ||||
|     if (data.containsKey('account_id')) { | ||||
|       context.handle(_accountIdMeta, | ||||
|           accountId.isAcceptableOrUnknown(data['account_id']!, _accountIdMeta)); | ||||
|     } else if (isInserting) { | ||||
|       context.missing(_accountIdMeta); | ||||
|     } | ||||
|     if (data.containsKey('created_at')) { | ||||
|       context.handle(_createdAtMeta, | ||||
|           createdAt.isAcceptableOrUnknown(data['created_at']!, _createdAtMeta)); | ||||
|     } | ||||
|     if (data.containsKey('cache_expired_at')) { | ||||
|       context.handle( | ||||
|           _cacheExpiredAtMeta, | ||||
|           cacheExpiredAt.isAcceptableOrUnknown( | ||||
|               data['cache_expired_at']!, _cacheExpiredAtMeta)); | ||||
|     } else if (isInserting) { | ||||
|       context.missing(_cacheExpiredAtMeta); | ||||
|     } | ||||
|     return context; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Set<GeneratedColumn> get $primaryKey => {id}; | ||||
|   @override | ||||
|   SnLocalRealmData map(Map<String, dynamic> data, {String? tablePrefix}) { | ||||
|     final effectivePrefix = tablePrefix != null ? '$tablePrefix.' : ''; | ||||
|     return SnLocalRealmData( | ||||
|       id: attachedDatabase.typeMapping | ||||
|           .read(DriftSqlType.int, data['${effectivePrefix}id'])!, | ||||
|       alias: attachedDatabase.typeMapping | ||||
|           .read(DriftSqlType.string, data['${effectivePrefix}alias'])!, | ||||
|       content: $SnLocalRealmTable.$convertercontent.fromSql(attachedDatabase | ||||
|           .typeMapping | ||||
|           .read(DriftSqlType.string, data['${effectivePrefix}content'])!), | ||||
|       accountId: attachedDatabase.typeMapping | ||||
|           .read(DriftSqlType.int, data['${effectivePrefix}account_id'])!, | ||||
|       createdAt: attachedDatabase.typeMapping | ||||
|           .read(DriftSqlType.dateTime, data['${effectivePrefix}created_at'])!, | ||||
|       cacheExpiredAt: attachedDatabase.typeMapping.read( | ||||
|           DriftSqlType.dateTime, data['${effectivePrefix}cache_expired_at'])!, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   $SnLocalRealmTable createAlias(String alias) { | ||||
|     return $SnLocalRealmTable(attachedDatabase, alias); | ||||
|   } | ||||
|  | ||||
|   static JsonTypeConverter2<SnRealm, String, Map<String, Object?>> | ||||
|       $convertercontent = const SnRealmConverter(); | ||||
| } | ||||
|  | ||||
| class SnLocalRealmData extends DataClass | ||||
|     implements Insertable<SnLocalRealmData> { | ||||
|   final int id; | ||||
|   final String alias; | ||||
|   final SnRealm content; | ||||
|   final int accountId; | ||||
|   final DateTime createdAt; | ||||
|   final DateTime cacheExpiredAt; | ||||
|   const SnLocalRealmData( | ||||
|       {required this.id, | ||||
|       required this.alias, | ||||
|       required this.content, | ||||
|       required this.accountId, | ||||
|       required this.createdAt, | ||||
|       required this.cacheExpiredAt}); | ||||
|   @override | ||||
|   Map<String, Expression> toColumns(bool nullToAbsent) { | ||||
|     final map = <String, Expression>{}; | ||||
|     map['id'] = Variable<int>(id); | ||||
|     map['alias'] = Variable<String>(alias); | ||||
|     { | ||||
|       map['content'] = | ||||
|           Variable<String>($SnLocalRealmTable.$convertercontent.toSql(content)); | ||||
|     } | ||||
|     map['account_id'] = Variable<int>(accountId); | ||||
|     map['created_at'] = Variable<DateTime>(createdAt); | ||||
|     map['cache_expired_at'] = Variable<DateTime>(cacheExpiredAt); | ||||
|     return map; | ||||
|   } | ||||
|  | ||||
|   SnLocalRealmCompanion toCompanion(bool nullToAbsent) { | ||||
|     return SnLocalRealmCompanion( | ||||
|       id: Value(id), | ||||
|       alias: Value(alias), | ||||
|       content: Value(content), | ||||
|       accountId: Value(accountId), | ||||
|       createdAt: Value(createdAt), | ||||
|       cacheExpiredAt: Value(cacheExpiredAt), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   factory SnLocalRealmData.fromJson(Map<String, dynamic> json, | ||||
|       {ValueSerializer? serializer}) { | ||||
|     serializer ??= driftRuntimeOptions.defaultSerializer; | ||||
|     return SnLocalRealmData( | ||||
|       id: serializer.fromJson<int>(json['id']), | ||||
|       alias: serializer.fromJson<String>(json['alias']), | ||||
|       content: $SnLocalRealmTable.$convertercontent | ||||
|           .fromJson(serializer.fromJson<Map<String, Object?>>(json['content'])), | ||||
|       accountId: serializer.fromJson<int>(json['accountId']), | ||||
|       createdAt: serializer.fromJson<DateTime>(json['createdAt']), | ||||
|       cacheExpiredAt: serializer.fromJson<DateTime>(json['cacheExpiredAt']), | ||||
|     ); | ||||
|   } | ||||
|   @override | ||||
|   Map<String, dynamic> toJson({ValueSerializer? serializer}) { | ||||
|     serializer ??= driftRuntimeOptions.defaultSerializer; | ||||
|     return <String, dynamic>{ | ||||
|       'id': serializer.toJson<int>(id), | ||||
|       'alias': serializer.toJson<String>(alias), | ||||
|       'content': serializer.toJson<Map<String, Object?>>( | ||||
|           $SnLocalRealmTable.$convertercontent.toJson(content)), | ||||
|       'accountId': serializer.toJson<int>(accountId), | ||||
|       'createdAt': serializer.toJson<DateTime>(createdAt), | ||||
|       'cacheExpiredAt': serializer.toJson<DateTime>(cacheExpiredAt), | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   SnLocalRealmData copyWith( | ||||
|           {int? id, | ||||
|           String? alias, | ||||
|           SnRealm? content, | ||||
|           int? accountId, | ||||
|           DateTime? createdAt, | ||||
|           DateTime? cacheExpiredAt}) => | ||||
|       SnLocalRealmData( | ||||
|         id: id ?? this.id, | ||||
|         alias: alias ?? this.alias, | ||||
|         content: content ?? this.content, | ||||
|         accountId: accountId ?? this.accountId, | ||||
|         createdAt: createdAt ?? this.createdAt, | ||||
|         cacheExpiredAt: cacheExpiredAt ?? this.cacheExpiredAt, | ||||
|       ); | ||||
|   SnLocalRealmData copyWithCompanion(SnLocalRealmCompanion data) { | ||||
|     return SnLocalRealmData( | ||||
|       id: data.id.present ? data.id.value : this.id, | ||||
|       alias: data.alias.present ? data.alias.value : this.alias, | ||||
|       content: data.content.present ? data.content.value : this.content, | ||||
|       accountId: data.accountId.present ? data.accountId.value : this.accountId, | ||||
|       createdAt: data.createdAt.present ? data.createdAt.value : this.createdAt, | ||||
|       cacheExpiredAt: data.cacheExpiredAt.present | ||||
|           ? data.cacheExpiredAt.value | ||||
|           : this.cacheExpiredAt, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return (StringBuffer('SnLocalRealmData(') | ||||
|           ..write('id: $id, ') | ||||
|           ..write('alias: $alias, ') | ||||
|           ..write('content: $content, ') | ||||
|           ..write('accountId: $accountId, ') | ||||
|           ..write('createdAt: $createdAt, ') | ||||
|           ..write('cacheExpiredAt: $cacheExpiredAt') | ||||
|           ..write(')')) | ||||
|         .toString(); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   int get hashCode => | ||||
|       Object.hash(id, alias, content, accountId, createdAt, cacheExpiredAt); | ||||
|   @override | ||||
|   bool operator ==(Object other) => | ||||
|       identical(this, other) || | ||||
|       (other is SnLocalRealmData && | ||||
|           other.id == this.id && | ||||
|           other.alias == this.alias && | ||||
|           other.content == this.content && | ||||
|           other.accountId == this.accountId && | ||||
|           other.createdAt == this.createdAt && | ||||
|           other.cacheExpiredAt == this.cacheExpiredAt); | ||||
| } | ||||
|  | ||||
| class SnLocalRealmCompanion extends UpdateCompanion<SnLocalRealmData> { | ||||
|   final Value<int> id; | ||||
|   final Value<String> alias; | ||||
|   final Value<SnRealm> content; | ||||
|   final Value<int> accountId; | ||||
|   final Value<DateTime> createdAt; | ||||
|   final Value<DateTime> cacheExpiredAt; | ||||
|   const SnLocalRealmCompanion({ | ||||
|     this.id = const Value.absent(), | ||||
|     this.alias = const Value.absent(), | ||||
|     this.content = const Value.absent(), | ||||
|     this.accountId = const Value.absent(), | ||||
|     this.createdAt = const Value.absent(), | ||||
|     this.cacheExpiredAt = const Value.absent(), | ||||
|   }); | ||||
|   SnLocalRealmCompanion.insert({ | ||||
|     this.id = const Value.absent(), | ||||
|     required String alias, | ||||
|     required SnRealm content, | ||||
|     required int accountId, | ||||
|     this.createdAt = const Value.absent(), | ||||
|     required DateTime cacheExpiredAt, | ||||
|   })  : alias = Value(alias), | ||||
|         content = Value(content), | ||||
|         accountId = Value(accountId), | ||||
|         cacheExpiredAt = Value(cacheExpiredAt); | ||||
|   static Insertable<SnLocalRealmData> custom({ | ||||
|     Expression<int>? id, | ||||
|     Expression<String>? alias, | ||||
|     Expression<String>? content, | ||||
|     Expression<int>? accountId, | ||||
|     Expression<DateTime>? createdAt, | ||||
|     Expression<DateTime>? cacheExpiredAt, | ||||
|   }) { | ||||
|     return RawValuesInsertable({ | ||||
|       if (id != null) 'id': id, | ||||
|       if (alias != null) 'alias': alias, | ||||
|       if (content != null) 'content': content, | ||||
|       if (accountId != null) 'account_id': accountId, | ||||
|       if (createdAt != null) 'created_at': createdAt, | ||||
|       if (cacheExpiredAt != null) 'cache_expired_at': cacheExpiredAt, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   SnLocalRealmCompanion copyWith( | ||||
|       {Value<int>? id, | ||||
|       Value<String>? alias, | ||||
|       Value<SnRealm>? content, | ||||
|       Value<int>? accountId, | ||||
|       Value<DateTime>? createdAt, | ||||
|       Value<DateTime>? cacheExpiredAt}) { | ||||
|     return SnLocalRealmCompanion( | ||||
|       id: id ?? this.id, | ||||
|       alias: alias ?? this.alias, | ||||
|       content: content ?? this.content, | ||||
|       accountId: accountId ?? this.accountId, | ||||
|       createdAt: createdAt ?? this.createdAt, | ||||
|       cacheExpiredAt: cacheExpiredAt ?? this.cacheExpiredAt, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Map<String, Expression> toColumns(bool nullToAbsent) { | ||||
|     final map = <String, Expression>{}; | ||||
|     if (id.present) { | ||||
|       map['id'] = Variable<int>(id.value); | ||||
|     } | ||||
|     if (alias.present) { | ||||
|       map['alias'] = Variable<String>(alias.value); | ||||
|     } | ||||
|     if (content.present) { | ||||
|       map['content'] = Variable<String>( | ||||
|           $SnLocalRealmTable.$convertercontent.toSql(content.value)); | ||||
|     } | ||||
|     if (accountId.present) { | ||||
|       map['account_id'] = Variable<int>(accountId.value); | ||||
|     } | ||||
|     if (createdAt.present) { | ||||
|       map['created_at'] = Variable<DateTime>(createdAt.value); | ||||
|     } | ||||
|     if (cacheExpiredAt.present) { | ||||
|       map['cache_expired_at'] = Variable<DateTime>(cacheExpiredAt.value); | ||||
|     } | ||||
|     return map; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String toString() { | ||||
|     return (StringBuffer('SnLocalRealmCompanion(') | ||||
|           ..write('id: $id, ') | ||||
|           ..write('alias: $alias, ') | ||||
|           ..write('content: $content, ') | ||||
|           ..write('accountId: $accountId, ') | ||||
|           ..write('createdAt: $createdAt, ') | ||||
|           ..write('cacheExpiredAt: $cacheExpiredAt') | ||||
|           ..write(')')) | ||||
|         .toString(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| abstract class _$AppDatabase extends GeneratedDatabase { | ||||
|   _$AppDatabase(QueryExecutor e) : super(e); | ||||
|   $AppDatabaseManager get managers => $AppDatabaseManager(this); | ||||
| @@ -2470,6 +2815,7 @@ abstract class _$AppDatabase extends GeneratedDatabase { | ||||
|   late final $SnLocalStickerTable snLocalSticker = $SnLocalStickerTable(this); | ||||
|   late final $SnLocalStickerPackTable snLocalStickerPack = | ||||
|       $SnLocalStickerPackTable(this); | ||||
|   late final $SnLocalRealmTable snLocalRealm = $SnLocalRealmTable(this); | ||||
|   late final Index idxChannelAlias = Index('idx_channel_alias', | ||||
|       'CREATE INDEX idx_channel_alias ON sn_local_chat_channel (alias)'); | ||||
|   late final Index idxChatChannel = Index('idx_chat_channel', | ||||
| @@ -2480,6 +2826,10 @@ abstract class _$AppDatabase extends GeneratedDatabase { | ||||
|       'CREATE INDEX idx_attachment_rid ON sn_local_attachment (rid)'); | ||||
|   late final Index idxAttachmentAccount = Index('idx_attachment_account', | ||||
|       'CREATE INDEX idx_attachment_account ON sn_local_attachment (account_id)'); | ||||
|   late final Index idxRealmAlias = Index('idx_realm_alias', | ||||
|       'CREATE INDEX idx_realm_alias ON sn_local_realm (alias)'); | ||||
|   late final Index idxRealmAccount = Index('idx_realm_account', | ||||
|       'CREATE INDEX idx_realm_account ON sn_local_realm (account_id)'); | ||||
|   @override | ||||
|   Iterable<TableInfo<Table, Object?>> get allTables => | ||||
|       allSchemaEntities.whereType<TableInfo<Table, Object?>>(); | ||||
| @@ -2493,11 +2843,14 @@ abstract class _$AppDatabase extends GeneratedDatabase { | ||||
|         snLocalAttachment, | ||||
|         snLocalSticker, | ||||
|         snLocalStickerPack, | ||||
|         snLocalRealm, | ||||
|         idxChannelAlias, | ||||
|         idxChatChannel, | ||||
|         idxAccountName, | ||||
|         idxAttachmentRid, | ||||
|         idxAttachmentAccount | ||||
|         idxAttachmentAccount, | ||||
|         idxRealmAlias, | ||||
|         idxRealmAccount | ||||
|       ]; | ||||
| } | ||||
|  | ||||
| @@ -3888,6 +4241,192 @@ typedef $$SnLocalStickerPackTableProcessedTableManager = ProcessedTableManager< | ||||
|     ), | ||||
|     SnLocalStickerPackData, | ||||
|     PrefetchHooks Function()>; | ||||
| typedef $$SnLocalRealmTableCreateCompanionBuilder = SnLocalRealmCompanion | ||||
|     Function({ | ||||
|   Value<int> id, | ||||
|   required String alias, | ||||
|   required SnRealm content, | ||||
|   required int accountId, | ||||
|   Value<DateTime> createdAt, | ||||
|   required DateTime cacheExpiredAt, | ||||
| }); | ||||
| typedef $$SnLocalRealmTableUpdateCompanionBuilder = SnLocalRealmCompanion | ||||
|     Function({ | ||||
|   Value<int> id, | ||||
|   Value<String> alias, | ||||
|   Value<SnRealm> content, | ||||
|   Value<int> accountId, | ||||
|   Value<DateTime> createdAt, | ||||
|   Value<DateTime> cacheExpiredAt, | ||||
| }); | ||||
|  | ||||
| class $$SnLocalRealmTableFilterComposer | ||||
|     extends Composer<_$AppDatabase, $SnLocalRealmTable> { | ||||
|   $$SnLocalRealmTableFilterComposer({ | ||||
|     required super.$db, | ||||
|     required super.$table, | ||||
|     super.joinBuilder, | ||||
|     super.$addJoinBuilderToRootComposer, | ||||
|     super.$removeJoinBuilderFromRootComposer, | ||||
|   }); | ||||
|   ColumnFilters<int> get id => $composableBuilder( | ||||
|       column: $table.id, builder: (column) => ColumnFilters(column)); | ||||
|  | ||||
|   ColumnFilters<String> get alias => $composableBuilder( | ||||
|       column: $table.alias, builder: (column) => ColumnFilters(column)); | ||||
|  | ||||
|   ColumnWithTypeConverterFilters<SnRealm, SnRealm, String> get content => | ||||
|       $composableBuilder( | ||||
|           column: $table.content, | ||||
|           builder: (column) => ColumnWithTypeConverterFilters(column)); | ||||
|  | ||||
|   ColumnFilters<int> get accountId => $composableBuilder( | ||||
|       column: $table.accountId, builder: (column) => ColumnFilters(column)); | ||||
|  | ||||
|   ColumnFilters<DateTime> get createdAt => $composableBuilder( | ||||
|       column: $table.createdAt, builder: (column) => ColumnFilters(column)); | ||||
|  | ||||
|   ColumnFilters<DateTime> get cacheExpiredAt => $composableBuilder( | ||||
|       column: $table.cacheExpiredAt, | ||||
|       builder: (column) => ColumnFilters(column)); | ||||
| } | ||||
|  | ||||
| class $$SnLocalRealmTableOrderingComposer | ||||
|     extends Composer<_$AppDatabase, $SnLocalRealmTable> { | ||||
|   $$SnLocalRealmTableOrderingComposer({ | ||||
|     required super.$db, | ||||
|     required super.$table, | ||||
|     super.joinBuilder, | ||||
|     super.$addJoinBuilderToRootComposer, | ||||
|     super.$removeJoinBuilderFromRootComposer, | ||||
|   }); | ||||
|   ColumnOrderings<int> get id => $composableBuilder( | ||||
|       column: $table.id, builder: (column) => ColumnOrderings(column)); | ||||
|  | ||||
|   ColumnOrderings<String> get alias => $composableBuilder( | ||||
|       column: $table.alias, builder: (column) => ColumnOrderings(column)); | ||||
|  | ||||
|   ColumnOrderings<String> get content => $composableBuilder( | ||||
|       column: $table.content, builder: (column) => ColumnOrderings(column)); | ||||
|  | ||||
|   ColumnOrderings<int> get accountId => $composableBuilder( | ||||
|       column: $table.accountId, builder: (column) => ColumnOrderings(column)); | ||||
|  | ||||
|   ColumnOrderings<DateTime> get createdAt => $composableBuilder( | ||||
|       column: $table.createdAt, builder: (column) => ColumnOrderings(column)); | ||||
|  | ||||
|   ColumnOrderings<DateTime> get cacheExpiredAt => $composableBuilder( | ||||
|       column: $table.cacheExpiredAt, | ||||
|       builder: (column) => ColumnOrderings(column)); | ||||
| } | ||||
|  | ||||
| class $$SnLocalRealmTableAnnotationComposer | ||||
|     extends Composer<_$AppDatabase, $SnLocalRealmTable> { | ||||
|   $$SnLocalRealmTableAnnotationComposer({ | ||||
|     required super.$db, | ||||
|     required super.$table, | ||||
|     super.joinBuilder, | ||||
|     super.$addJoinBuilderToRootComposer, | ||||
|     super.$removeJoinBuilderFromRootComposer, | ||||
|   }); | ||||
|   GeneratedColumn<int> get id => | ||||
|       $composableBuilder(column: $table.id, builder: (column) => column); | ||||
|  | ||||
|   GeneratedColumn<String> get alias => | ||||
|       $composableBuilder(column: $table.alias, builder: (column) => column); | ||||
|  | ||||
|   GeneratedColumnWithTypeConverter<SnRealm, String> get content => | ||||
|       $composableBuilder(column: $table.content, builder: (column) => column); | ||||
|  | ||||
|   GeneratedColumn<int> get accountId => | ||||
|       $composableBuilder(column: $table.accountId, builder: (column) => column); | ||||
|  | ||||
|   GeneratedColumn<DateTime> get createdAt => | ||||
|       $composableBuilder(column: $table.createdAt, builder: (column) => column); | ||||
|  | ||||
|   GeneratedColumn<DateTime> get cacheExpiredAt => $composableBuilder( | ||||
|       column: $table.cacheExpiredAt, builder: (column) => column); | ||||
| } | ||||
|  | ||||
| class $$SnLocalRealmTableTableManager extends RootTableManager< | ||||
|     _$AppDatabase, | ||||
|     $SnLocalRealmTable, | ||||
|     SnLocalRealmData, | ||||
|     $$SnLocalRealmTableFilterComposer, | ||||
|     $$SnLocalRealmTableOrderingComposer, | ||||
|     $$SnLocalRealmTableAnnotationComposer, | ||||
|     $$SnLocalRealmTableCreateCompanionBuilder, | ||||
|     $$SnLocalRealmTableUpdateCompanionBuilder, | ||||
|     ( | ||||
|       SnLocalRealmData, | ||||
|       BaseReferences<_$AppDatabase, $SnLocalRealmTable, SnLocalRealmData> | ||||
|     ), | ||||
|     SnLocalRealmData, | ||||
|     PrefetchHooks Function()> { | ||||
|   $$SnLocalRealmTableTableManager(_$AppDatabase db, $SnLocalRealmTable table) | ||||
|       : super(TableManagerState( | ||||
|           db: db, | ||||
|           table: table, | ||||
|           createFilteringComposer: () => | ||||
|               $$SnLocalRealmTableFilterComposer($db: db, $table: table), | ||||
|           createOrderingComposer: () => | ||||
|               $$SnLocalRealmTableOrderingComposer($db: db, $table: table), | ||||
|           createComputedFieldComposer: () => | ||||
|               $$SnLocalRealmTableAnnotationComposer($db: db, $table: table), | ||||
|           updateCompanionCallback: ({ | ||||
|             Value<int> id = const Value.absent(), | ||||
|             Value<String> alias = const Value.absent(), | ||||
|             Value<SnRealm> content = const Value.absent(), | ||||
|             Value<int> accountId = const Value.absent(), | ||||
|             Value<DateTime> createdAt = const Value.absent(), | ||||
|             Value<DateTime> cacheExpiredAt = const Value.absent(), | ||||
|           }) => | ||||
|               SnLocalRealmCompanion( | ||||
|             id: id, | ||||
|             alias: alias, | ||||
|             content: content, | ||||
|             accountId: accountId, | ||||
|             createdAt: createdAt, | ||||
|             cacheExpiredAt: cacheExpiredAt, | ||||
|           ), | ||||
|           createCompanionCallback: ({ | ||||
|             Value<int> id = const Value.absent(), | ||||
|             required String alias, | ||||
|             required SnRealm content, | ||||
|             required int accountId, | ||||
|             Value<DateTime> createdAt = const Value.absent(), | ||||
|             required DateTime cacheExpiredAt, | ||||
|           }) => | ||||
|               SnLocalRealmCompanion.insert( | ||||
|             id: id, | ||||
|             alias: alias, | ||||
|             content: content, | ||||
|             accountId: accountId, | ||||
|             createdAt: createdAt, | ||||
|             cacheExpiredAt: cacheExpiredAt, | ||||
|           ), | ||||
|           withReferenceMapper: (p0) => p0 | ||||
|               .map((e) => (e.readTable(table), BaseReferences(db, table, e))) | ||||
|               .toList(), | ||||
|           prefetchHooksCallback: null, | ||||
|         )); | ||||
| } | ||||
|  | ||||
| typedef $$SnLocalRealmTableProcessedTableManager = ProcessedTableManager< | ||||
|     _$AppDatabase, | ||||
|     $SnLocalRealmTable, | ||||
|     SnLocalRealmData, | ||||
|     $$SnLocalRealmTableFilterComposer, | ||||
|     $$SnLocalRealmTableOrderingComposer, | ||||
|     $$SnLocalRealmTableAnnotationComposer, | ||||
|     $$SnLocalRealmTableCreateCompanionBuilder, | ||||
|     $$SnLocalRealmTableUpdateCompanionBuilder, | ||||
|     ( | ||||
|       SnLocalRealmData, | ||||
|       BaseReferences<_$AppDatabase, $SnLocalRealmTable, SnLocalRealmData> | ||||
|     ), | ||||
|     SnLocalRealmData, | ||||
|     PrefetchHooks Function()>; | ||||
|  | ||||
| class $AppDatabaseManager { | ||||
|   final _$AppDatabase _db; | ||||
| @@ -3908,4 +4447,6 @@ class $AppDatabaseManager { | ||||
|       $$SnLocalStickerTableTableManager(_db, _db.snLocalSticker); | ||||
|   $$SnLocalStickerPackTableTableManager get snLocalStickerPack => | ||||
|       $$SnLocalStickerPackTableTableManager(_db, _db.snLocalStickerPack); | ||||
|   $$SnLocalRealmTableTableManager get snLocalRealm => | ||||
|       $$SnLocalRealmTableTableManager(_db, _db.snLocalRealm); | ||||
| } | ||||
|   | ||||
| @@ -412,9 +412,214 @@ class Shape8 extends i0.VersionedTable { | ||||
|       columnsByName['created_at']! as i1.GeneratedColumn<DateTime>; | ||||
| } | ||||
|  | ||||
| final class Schema4 extends i0.VersionedSchema { | ||||
|   Schema4({required super.database}) : super(version: 4); | ||||
|   @override | ||||
|   late final List<i1.DatabaseSchemaEntity> entities = [ | ||||
|     snLocalChatChannel, | ||||
|     snLocalChatMessage, | ||||
|     snLocalChannelMember, | ||||
|     snLocalKeyPair, | ||||
|     snLocalAccount, | ||||
|     snLocalAttachment, | ||||
|     snLocalSticker, | ||||
|     snLocalStickerPack, | ||||
|     snLocalRealm, | ||||
|     idxChannelAlias, | ||||
|     idxChatChannel, | ||||
|     idxAccountName, | ||||
|     idxAttachmentRid, | ||||
|     idxAttachmentAccount, | ||||
|     idxRealmAlias, | ||||
|     idxRealmAccount, | ||||
|   ]; | ||||
|   late final Shape0 snLocalChatChannel = Shape0( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_chat_channel', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_1, | ||||
|           _column_2, | ||||
|           _column_3, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape3 snLocalChatMessage = Shape3( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_chat_message', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_4, | ||||
|           _column_10, | ||||
|           _column_2, | ||||
|           _column_3, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape4 snLocalChannelMember = Shape4( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_channel_member', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_4, | ||||
|           _column_6, | ||||
|           _column_2, | ||||
|           _column_3, | ||||
|           _column_11, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape2 snLocalKeyPair = Shape2( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_key_pair', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [ | ||||
|           'PRIMARY KEY(id)', | ||||
|         ], | ||||
|         columns: [ | ||||
|           _column_5, | ||||
|           _column_6, | ||||
|           _column_7, | ||||
|           _column_8, | ||||
|           _column_9, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape5 snLocalAccount = Shape5( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_account', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_12, | ||||
|           _column_2, | ||||
|           _column_3, | ||||
|           _column_11, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape6 snLocalAttachment = Shape6( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_attachment', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_13, | ||||
|           _column_14, | ||||
|           _column_2, | ||||
|           _column_6, | ||||
|           _column_3, | ||||
|           _column_11, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape7 snLocalSticker = Shape7( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_sticker', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_1, | ||||
|           _column_15, | ||||
|           _column_2, | ||||
|           _column_3, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape8 snLocalStickerPack = Shape8( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_sticker_pack', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_2, | ||||
|           _column_3, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   late final Shape9 snLocalRealm = Shape9( | ||||
|       source: i0.VersionedTable( | ||||
|         entityName: 'sn_local_realm', | ||||
|         withoutRowId: false, | ||||
|         isStrict: false, | ||||
|         tableConstraints: [], | ||||
|         columns: [ | ||||
|           _column_0, | ||||
|           _column_16, | ||||
|           _column_2, | ||||
|           _column_6, | ||||
|           _column_3, | ||||
|           _column_11, | ||||
|         ], | ||||
|         attachedDatabase: database, | ||||
|       ), | ||||
|       alias: null); | ||||
|   final i1.Index idxChannelAlias = i1.Index('idx_channel_alias', | ||||
|       'CREATE INDEX idx_channel_alias ON sn_local_chat_channel (alias)'); | ||||
|   final i1.Index idxChatChannel = i1.Index('idx_chat_channel', | ||||
|       'CREATE INDEX idx_chat_channel ON sn_local_chat_message (channel_id)'); | ||||
|   final i1.Index idxAccountName = i1.Index('idx_account_name', | ||||
|       'CREATE INDEX idx_account_name ON sn_local_account (name)'); | ||||
|   final i1.Index idxAttachmentRid = i1.Index('idx_attachment_rid', | ||||
|       'CREATE INDEX idx_attachment_rid ON sn_local_attachment (rid)'); | ||||
|   final i1.Index idxAttachmentAccount = i1.Index('idx_attachment_account', | ||||
|       'CREATE INDEX idx_attachment_account ON sn_local_attachment (account_id)'); | ||||
|   final i1.Index idxRealmAlias = i1.Index('idx_realm_alias', | ||||
|       'CREATE INDEX idx_realm_alias ON sn_local_realm (alias)'); | ||||
|   final i1.Index idxRealmAccount = i1.Index('idx_realm_account', | ||||
|       'CREATE INDEX idx_realm_account ON sn_local_realm (account_id)'); | ||||
| } | ||||
|  | ||||
| class Shape9 extends i0.VersionedTable { | ||||
|   Shape9({required super.source, required super.alias}) : super.aliased(); | ||||
|   i1.GeneratedColumn<int> get id => | ||||
|       columnsByName['id']! as i1.GeneratedColumn<int>; | ||||
|   i1.GeneratedColumn<String> get alias => | ||||
|       columnsByName['alias']! as i1.GeneratedColumn<String>; | ||||
|   i1.GeneratedColumn<String> get content => | ||||
|       columnsByName['content']! as i1.GeneratedColumn<String>; | ||||
|   i1.GeneratedColumn<int> get accountId => | ||||
|       columnsByName['account_id']! as i1.GeneratedColumn<int>; | ||||
|   i1.GeneratedColumn<DateTime> get createdAt => | ||||
|       columnsByName['created_at']! as i1.GeneratedColumn<DateTime>; | ||||
|   i1.GeneratedColumn<DateTime> get cacheExpiredAt => | ||||
|       columnsByName['cache_expired_at']! as i1.GeneratedColumn<DateTime>; | ||||
| } | ||||
|  | ||||
| i1.GeneratedColumn<String> _column_16(String aliasedName) => | ||||
|     i1.GeneratedColumn<String>('alias', aliasedName, false, | ||||
|         type: i1.DriftSqlType.string, | ||||
|         defaultConstraints: i1.GeneratedColumn.constraintIsAlways('UNIQUE')); | ||||
| i0.MigrationStepWithVersion migrationSteps({ | ||||
|   required Future<void> Function(i1.Migrator m, Schema2 schema) from1To2, | ||||
|   required Future<void> Function(i1.Migrator m, Schema3 schema) from2To3, | ||||
|   required Future<void> Function(i1.Migrator m, Schema4 schema) from3To4, | ||||
| }) { | ||||
|   return (currentVersion, database) async { | ||||
|     switch (currentVersion) { | ||||
| @@ -428,6 +633,11 @@ i0.MigrationStepWithVersion migrationSteps({ | ||||
|         final migrator = i1.Migrator(database, schema); | ||||
|         await from2To3(migrator, schema); | ||||
|         return 3; | ||||
|       case 3: | ||||
|         final schema = Schema4(database: database); | ||||
|         final migrator = i1.Migrator(database, schema); | ||||
|         await from3To4(migrator, schema); | ||||
|         return 4; | ||||
|       default: | ||||
|         throw ArgumentError.value('Unknown migration from $currentVersion'); | ||||
|     } | ||||
| @@ -437,9 +647,11 @@ i0.MigrationStepWithVersion migrationSteps({ | ||||
| i1.OnUpgrade stepByStep({ | ||||
|   required Future<void> Function(i1.Migrator m, Schema2 schema) from1To2, | ||||
|   required Future<void> Function(i1.Migrator m, Schema3 schema) from2To3, | ||||
|   required Future<void> Function(i1.Migrator m, Schema4 schema) from3To4, | ||||
| }) => | ||||
|     i0.VersionedSchema.stepByStepHelper( | ||||
|         step: migrationSteps( | ||||
|       from1To2: from1To2, | ||||
|       from2To3: from2To3, | ||||
|       from3To4: from3To4, | ||||
|     )); | ||||
|   | ||||
							
								
								
									
										45
									
								
								lib/database/realm.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								lib/database/realm.dart
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| import 'dart:convert'; | ||||
|  | ||||
| import 'package:drift/drift.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
|  | ||||
| class SnRealmConverter extends TypeConverter<SnRealm, String> | ||||
|     with JsonTypeConverter2<SnRealm, String, Map<String, Object?>> { | ||||
|   const SnRealmConverter(); | ||||
|  | ||||
|   @override | ||||
|   SnRealm fromSql(String fromDb) { | ||||
|     return fromJson(jsonDecode(fromDb) as Map<String, dynamic>); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   String toSql(SnRealm value) { | ||||
|     return jsonEncode(toJson(value)); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   SnRealm fromJson(Map<String, Object?> json) { | ||||
|     return SnRealm.fromJson(json); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Map<String, Object?> toJson(SnRealm value) { | ||||
|     return value.toJson(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| @TableIndex(name: 'idx_realm_alias', columns: {#alias}) | ||||
| @TableIndex(name: 'idx_realm_account', columns: {#accountId}) | ||||
| class SnLocalRealm extends Table { | ||||
|   IntColumn get id => integer().autoIncrement()(); | ||||
|  | ||||
|   TextColumn get alias => text().unique()(); | ||||
|  | ||||
|   TextColumn get content => text().map(const SnRealmConverter())(); | ||||
|  | ||||
|   IntColumn get accountId => integer()(); | ||||
|  | ||||
|   DateTimeColumn get createdAt => dateTime().withDefault(currentDateAndTime)(); | ||||
|  | ||||
|   DateTimeColumn get cacheExpiredAt => dateTime()(); | ||||
| } | ||||
| @@ -321,13 +321,13 @@ class SnAttachmentProvider { | ||||
|           uuid: ele.uuid, | ||||
|           content: ele, | ||||
|           accountId: ele.accountId, | ||||
|           cacheExpiredAt: DateTime.now().add(const Duration(days: 7)), | ||||
|           cacheExpiredAt: DateTime.now().add(const Duration(hours: 1)), | ||||
|         ), | ||||
|         onConflict: DoUpdate( | ||||
|           (_) => SnLocalAttachmentCompanion.custom( | ||||
|             content: Constant(jsonEncode(ele.toJson())), | ||||
|             cacheExpiredAt: | ||||
|                 Constant(DateTime.now().add(const Duration(days: 7))), | ||||
|                 Constant(DateTime.now().add(const Duration(hours: 1))), | ||||
|           ), | ||||
|         ), | ||||
|       ); | ||||
|   | ||||
| @@ -1,13 +1,20 @@ | ||||
| import 'dart:convert'; | ||||
|  | ||||
| import 'package:drift/drift.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
| import 'package:surface/database/database.dart'; | ||||
| import 'package:surface/providers/database.dart'; | ||||
| import 'package:surface/providers/sn_network.dart'; | ||||
| import 'package:surface/types/realm.dart'; | ||||
|  | ||||
| class SnRealmProvider { | ||||
|   late final SnNetworkProvider _sn; | ||||
|   late final DatabaseProvider _dt; | ||||
|  | ||||
|   SnRealmProvider(BuildContext context) { | ||||
|     _sn = context.read<SnNetworkProvider>(); | ||||
|     _dt = context.read<DatabaseProvider>(); | ||||
|   } | ||||
|  | ||||
|   final Map<String, SnRealm> _cache = {}; | ||||
| @@ -28,6 +35,7 @@ class SnRealmProvider { | ||||
|       _cache[realm.alias] = realm; | ||||
|       _cache[realm.id.toString()] = realm; | ||||
|     } | ||||
|     _saveToLocal(out); | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
| @@ -35,10 +43,43 @@ class SnRealmProvider { | ||||
|     if (_cache.containsKey(aliasOrId.toString())) { | ||||
|       return _cache[aliasOrId.toString()]!; | ||||
|     } | ||||
|     final localResp = await (_dt.db.snLocalRealm.select() | ||||
|           ..where((e) => | ||||
|               e.id.equals(aliasOrId is int ? aliasOrId : 0) | | ||||
|               e.alias.equals(aliasOrId.toString())) | ||||
|           ..where((e) => e.cacheExpiredAt.isBiggerThanValue(DateTime.now()))) | ||||
|         .getSingleOrNull(); | ||||
|     if (localResp != null) { | ||||
|       _cache[localResp.content.id.toString()] = localResp.content; | ||||
|       _cache[localResp.content.alias] = localResp.content; | ||||
|       return localResp.content; | ||||
|     } | ||||
|     final resp = await _sn.client.get('/cgi/id/realms/$aliasOrId'); | ||||
|     final out = SnRealm.fromJson(resp.data); | ||||
|     _cache[out.alias] = out; | ||||
|     _cache[out.id.toString()] = out; | ||||
|     _saveToLocal([out]); | ||||
|     return out; | ||||
|   } | ||||
|  | ||||
|   Future<void> _saveToLocal(Iterable<SnRealm> out) async { | ||||
|     for (final ele in out) { | ||||
|       await _dt.db.snLocalRealm.insertOne( | ||||
|         SnLocalRealmCompanion.insert( | ||||
|           id: Value(ele.id), | ||||
|           alias: ele.alias, | ||||
|           content: ele, | ||||
|           accountId: ele.accountId, | ||||
|           cacheExpiredAt: DateTime.now().add(const Duration(hours: 1)), | ||||
|         ), | ||||
|         onConflict: DoUpdate( | ||||
|           (_) => SnLocalRealmCompanion.custom( | ||||
|             content: Constant(jsonEncode(ele.toJson())), | ||||
|             cacheExpiredAt: | ||||
|                 Constant(DateTime.now().add(const Duration(hours: 1))), | ||||
|           ), | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -52,8 +52,10 @@ class ChatRoomScreen extends StatefulWidget { | ||||
| class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|   bool _isBusy = false; | ||||
|   bool _isCalling = false; | ||||
|   bool _isJoining = false; | ||||
|  | ||||
|   SnChannel? _channel; | ||||
|   SnChannelMember? _currentMember; | ||||
|   SnChannelMember? _otherMember; | ||||
|   SnChatCall? _ongoingCall; | ||||
|  | ||||
| @@ -67,7 +69,24 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|  | ||||
|   StreamSubscription? _wsSubscription; | ||||
|  | ||||
|   // TODO fetch user identity and ask them to join the channel or not | ||||
|   Future<void> _joinChannel() async { | ||||
|     try { | ||||
|       setState(() => _isJoining = true); | ||||
|       final sn = context.read<SnNetworkProvider>(); | ||||
|       final ua = context.read<UserProvider>(); | ||||
|       await sn.client | ||||
|           .post('/cgi/im/channels/${_channel!.keyPath}/members', data: { | ||||
|         'related': ua.user?.name, | ||||
|       }); | ||||
|       _initializeChat(); | ||||
|     } catch (err) { | ||||
|       if (!mounted) return; | ||||
|       context.showErrorDialog(err); | ||||
|     } finally { | ||||
|       setState(() => _isJoining = true); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   Future<void> _fetchChannel() async { | ||||
|     setState(() => _isBusy = true); | ||||
|  | ||||
| @@ -76,6 +95,12 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|       _channel = await chan.getChannel('${widget.scope}:${widget.alias}'); | ||||
|  | ||||
|       if (!mounted || _channel == null) return; | ||||
|       final ct = context.read<ChatChannelProvider>(); | ||||
|       try { | ||||
|         _currentMember = await ct.getChannelProfile(_channel!); | ||||
|       } catch (_) {} | ||||
|  | ||||
|       if (!mounted || _currentMember == null) return; | ||||
|       final ud = context.read<UserDirectoryProvider>(); | ||||
|       final ua = context.read<UserProvider>(); | ||||
|       if (_channel!.type == 1) { | ||||
| @@ -204,11 +229,9 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|     return a.createdAt.difference(b.createdAt).inMinutes <= 3; | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _messageController = ChatMessageController(context); | ||||
|   Future<void> _initializeChat() async { | ||||
|     _fetchChannel().then((_) async { | ||||
|       if (_currentMember == null) return; | ||||
|       await _messageController.initialize(_channel!); | ||||
|  | ||||
|       if (widget.extra != null) { | ||||
| @@ -230,6 +253,13 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|         _fetchOngoingCall(), | ||||
|       ]); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _messageController = ChatMessageController(context); | ||||
|     _initializeChat(); | ||||
|  | ||||
|     _wsSubscription = _ws.pk.stream.listen((event) { | ||||
|       switch (event.method) { | ||||
| @@ -281,25 +311,27 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|               : _channel?.name ?? 'loading'.tr(), | ||||
|         ), | ||||
|         actions: [ | ||||
|           IconButton( | ||||
|             onPressed: () { | ||||
|               setState(() => _isEncrypted = !_isEncrypted); | ||||
|               _inputGlobalKey.currentState?.setEncrypted(_isEncrypted); | ||||
|             }, | ||||
|             icon: _isEncrypted | ||||
|                 ? const Icon(Symbols.lock) | ||||
|                 : const Icon(Symbols.no_encryption), | ||||
|           ), | ||||
|           IconButton( | ||||
|             icon: _ongoingCall == null | ||||
|                 ? const Icon(Symbols.call) | ||||
|                 : const Icon(Symbols.call_end), | ||||
|             onPressed: _isCalling | ||||
|                 ? null | ||||
|                 : _ongoingCall == null | ||||
|                     ? _makeCall | ||||
|                     : _endCall, | ||||
|           ), | ||||
|           if (_currentMember != null) | ||||
|             IconButton( | ||||
|               onPressed: () { | ||||
|                 setState(() => _isEncrypted = !_isEncrypted); | ||||
|                 _inputGlobalKey.currentState?.setEncrypted(_isEncrypted); | ||||
|               }, | ||||
|               icon: _isEncrypted | ||||
|                   ? const Icon(Symbols.lock) | ||||
|                   : const Icon(Symbols.no_encryption), | ||||
|             ), | ||||
|           if (_currentMember != null) | ||||
|             IconButton( | ||||
|               icon: _ongoingCall == null | ||||
|                   ? const Icon(Symbols.call) | ||||
|                   : const Icon(Symbols.call_end), | ||||
|               onPressed: _isCalling | ||||
|                   ? null | ||||
|                   : _ongoingCall == null | ||||
|                       ? _makeCall | ||||
|                       : _endCall, | ||||
|             ), | ||||
|           IconButton( | ||||
|             icon: const Icon(Symbols.more_vert), | ||||
|             onPressed: () { | ||||
| @@ -348,7 +380,41 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|               ).height(_ongoingCall != null ? 54 : 0, animate: true).animate( | ||||
|                   const Duration(milliseconds: 300), | ||||
|                   Curves.fastLinearToSlowEaseIn), | ||||
|               if (_messageController.isPending) | ||||
|               if (_currentMember == null && !_isBusy) | ||||
|                 Expanded( | ||||
|                   child: Center( | ||||
|                     child: Container( | ||||
|                       constraints: const BoxConstraints(maxWidth: 280), | ||||
|                       child: Column( | ||||
|                         mainAxisSize: MainAxisSize.min, | ||||
|                         children: [ | ||||
|                           const Icon(Symbols.person_remove, size: 40, fill: 1), | ||||
|                           const Gap(8), | ||||
|                           Text('chatUnjoined'.tr(), textAlign: TextAlign.center) | ||||
|                               .fontSize(16) | ||||
|                               .bold(), | ||||
|                           Text('chatUnjoinedDescription'.tr(), | ||||
|                                   textAlign: TextAlign.center) | ||||
|                               .fontSize(13), | ||||
|                           if (_channel!.isPublic) | ||||
|                             Text('chatUnjoinedPublicDescription'.tr(), | ||||
|                                     textAlign: TextAlign.center) | ||||
|                                 .fontSize(13) | ||||
|                                 .padding(top: 8), | ||||
|                           if (_channel!.isPublic) | ||||
|                             TextButton( | ||||
|                               style: ButtonStyle( | ||||
|                                 visualDensity: VisualDensity.compact, | ||||
|                               ), | ||||
|                               onPressed: _isJoining ? null : _joinChannel, | ||||
|                               child: Text('chatJoin').tr(), | ||||
|                             ), | ||||
|                         ], | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               else if (_messageController.isPending) | ||||
|                 Expanded( | ||||
|                   child: const CircularProgressIndicator().center(), | ||||
|                 ) | ||||
| @@ -403,7 +469,7 @@ class _ChatRoomScreenState extends State<ChatRoomScreen> { | ||||
|                     }, | ||||
|                   ), | ||||
|                 ), | ||||
|               if (!_messageController.isPending) | ||||
|               if (!_messageController.isPending && _currentMember != null) | ||||
|                 Material( | ||||
|                   elevation: 2, | ||||
|                   child: Column( | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import 'package:provider/provider.dart'; | ||||
| import 'package:styled_widget/styled_widget.dart'; | ||||
| import 'package:surface/providers/config.dart'; | ||||
| import 'package:surface/providers/navigation.dart'; | ||||
| import 'package:surface/providers/sn_realm.dart'; | ||||
| import 'package:surface/providers/userinfo.dart'; | ||||
| import 'package:surface/widgets/account/account_image.dart'; | ||||
| import 'package:surface/widgets/version_label.dart'; | ||||
| @@ -40,6 +41,7 @@ class _AppNavigationDrawerState extends State<AppNavigationDrawer> { | ||||
|     final ua = context.read<UserProvider>(); | ||||
|     final nav = context.watch<NavigationProvider>(); | ||||
|     final cfg = context.watch<ConfigProvider>(); | ||||
|     final rel = context.read<SnRealmProvider>(); | ||||
|  | ||||
|     final backgroundColor = cfg.drawerIsExpanded ? Colors.transparent : null; | ||||
|  | ||||
| @@ -82,7 +84,31 @@ class _AppNavigationDrawerState extends State<AppNavigationDrawer> { | ||||
|                 vertical: 12, | ||||
|               ), | ||||
|               Expanded( | ||||
|                 child: ListView(), | ||||
|                 child: ListView( | ||||
|                   padding: EdgeInsets.zero, | ||||
|                   children: [ | ||||
|                     ...rel.availableRealms.map((ele) { | ||||
|                       return ListTile( | ||||
|                         minTileHeight: 48, | ||||
|                         contentPadding: EdgeInsets.symmetric(horizontal: 24), | ||||
|                         leading: AccountImage( | ||||
|                           content: ele.avatar, | ||||
|                           radius: 16, | ||||
|                         ), | ||||
|                         title: Text(ele.name), | ||||
|                         onTap: () { | ||||
|                           GoRouter.of(context).goNamed( | ||||
|                             'realmDetail', | ||||
|                             pathParameters: { | ||||
|                               'alias': ele.alias, | ||||
|                             }, | ||||
|                           ); | ||||
|                           Scaffold.of(context).closeDrawer(); | ||||
|                         }, | ||||
|                       ); | ||||
|                     }), | ||||
|                   ], | ||||
|                 ), | ||||
|               ), | ||||
|               Row( | ||||
|                 spacing: 8, | ||||
| @@ -106,7 +132,7 @@ class _AppNavigationDrawerState extends State<AppNavigationDrawer> { | ||||
|                 child: ListTile( | ||||
|                   contentPadding: EdgeInsets.symmetric(horizontal: 24), | ||||
|                   leading: AccountImage(content: ua.user?.avatar), | ||||
|                   title: Text(ua.user?.nick ?? 'unknown').tr().fontSize(15), | ||||
|                   title: Text(ua.user?.nick ?? 'unknown'.tr()).fontSize(15), | ||||
|                   subtitle: | ||||
|                       Text('@${ua.user?.name ?? 'unknown'.tr()}').fontSize(13), | ||||
|                   trailing: Row( | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import 'package:drift/internal/migrations.dart'; | ||||
| import 'schema_v1.dart' as v1; | ||||
| import 'schema_v2.dart' as v2; | ||||
| import 'schema_v3.dart' as v3; | ||||
| import 'schema_v4.dart' as v4; | ||||
|  | ||||
| class GeneratedHelper implements SchemaInstantiationHelper { | ||||
|   @override | ||||
| @@ -17,10 +18,12 @@ class GeneratedHelper implements SchemaInstantiationHelper { | ||||
|         return v2.DatabaseAtV2(db); | ||||
|       case 3: | ||||
|         return v3.DatabaseAtV3(db); | ||||
|       case 4: | ||||
|         return v4.DatabaseAtV4(db); | ||||
|       default: | ||||
|         throw MissingSchemaException(version, versions); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static const versions = const [1, 2, 3]; | ||||
|   static const versions = const [1, 2, 3, 4]; | ||||
| } | ||||
|   | ||||
							
								
								
									
										2391
									
								
								test/drift/my_database/generated/schema_v4.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2391
									
								
								test/drift/my_database/generated/schema_v4.dart
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user