import 'package:recon/auxiliary.dart'; import 'package:recon/models/message.dart'; import 'package:recon/models/records/asset_digest.dart'; import 'package:recon/models/records/resonite_db_asset.dart'; import 'package:recon/string_formatter.dart'; import 'package:flutter/material.dart'; import 'package:uuid/uuid.dart'; enum RecordType { unknown, link, object, directory, texture, audio; factory RecordType.fromName(String? name) { return RecordType.values.firstWhere((element) => element.name.toLowerCase() == name?.toLowerCase().trim(), orElse: () => RecordType.unknown); } } class RecordId { final String? id; final String? ownerId; final bool isValid; const RecordId({this.id, this.ownerId, required this.isValid}); factory RecordId.fromMap(Map? map) { return RecordId(id: map?["id"], ownerId: map?["ownerId"], isValid: map?["isValid"] ?? false); } Map toMap() { return { "id": id, "ownerId": ownerId, "isValid": isValid, }; } } class Record { static final _rootRecord = Record( id: "0", combinedRecordId: const RecordId(isValid: false), isSynced: true, fetchedOn: DateTimeX.epoch, path: "Inventory", ownerId: "", assetUri: "", name: "Inventory", description: "", tags: [], recordType: RecordType.directory, thumbnailUri: "", isPublic: false, isListed: false, isForPatreons: false, lastModificationTime: DateTimeX.epoch, resoniteDBManifest: [], lastModifyingUserId: "", lastModifyingMachineId: "", creationTime: DateTimeX.epoch, manifest: [], url: "", isValidOwnerId: true, isValidRecordId: true, globalVersion: 1, localVersion: 1, visits: 0, rating: 0, randomOrder: 0, ); final String id; final RecordId combinedRecordId; final String ownerId; final String assetUri; final int globalVersion; final int localVersion; final String lastModifyingUserId; final String lastModifyingMachineId; final bool isSynced; final DateTime fetchedOn; final String name; final FormatNode formattedName; final String description; final RecordType recordType; final List tags; final String path; final String thumbnailUri; final bool isPublic; final bool isForPatreons; final bool isListed; final DateTime lastModificationTime; final DateTime creationTime; final int visits; final int rating; final int randomOrder; final List manifest; final List resoniteDBManifest; final String url; final bool isValidOwnerId; final bool isValidRecordId; Record({ required this.id, required this.combinedRecordId, required this.isSynced, required this.fetchedOn, required this.path, required this.ownerId, required this.assetUri, required this.name, required this.description, required this.tags, required this.recordType, required this.thumbnailUri, required this.isPublic, required this.isListed, required this.isForPatreons, required this.lastModificationTime, required this.resoniteDBManifest, required this.lastModifyingUserId, required this.lastModifyingMachineId, required this.creationTime, required this.manifest, required this.url, required this.isValidOwnerId, required this.isValidRecordId, required this.globalVersion, required this.localVersion, required this.visits, required this.rating, required this.randomOrder, }) : formattedName = FormatNode.fromText(name); factory Record.fromRequiredData({ required RecordType recordType, required String userId, required String machineId, required String assetUri, required String filename, required String thumbnailUri, required List digests, List? extraTags, }) { final combinedRecordId = RecordId(id: Record.generateId(), ownerId: userId, isValid: true); return Record( id: combinedRecordId.id.toString(), combinedRecordId: combinedRecordId, assetUri: assetUri, name: filename, tags: ([filename, "message_item", "message_id:${Message.generateId()}", "recon"] + (extraTags ?? [])) .unique(), recordType: recordType, thumbnailUri: thumbnailUri, isPublic: false, isForPatreons: false, isListed: false, resoniteDBManifest: digests.map((e) => e.asset).toList(), globalVersion: 0, localVersion: 1, lastModifyingUserId: userId, lastModifyingMachineId: machineId, lastModificationTime: DateTime.now().toUtc(), creationTime: DateTime.now().toUtc(), ownerId: userId, isSynced: false, fetchedOn: DateTimeX.one, path: '', description: '', manifest: digests.map((e) => e.dbUri).toList(), url: "resrec:///$userId/${combinedRecordId.id}", isValidOwnerId: true, isValidRecordId: true, visits: 0, rating: 0, randomOrder: 0, ); } factory Record.fromMap(Map map) { return Record( id: map["id"] ?? "0", combinedRecordId: RecordId.fromMap(map["combinedRecordId"]), ownerId: map["ownerId"] ?? "", assetUri: map["assetUri"] ?? "", globalVersion: map["globalVersion"] ?? 0, localVersion: map["localVersion"] ?? 0, name: map["name"] ?? "", description: map["description"] ?? "", tags: (map["tags"] as List? ?? []).map((e) => e.toString()).toList(), recordType: RecordType.fromName(map["recordType"]), thumbnailUri: map["thumbnailUri"] ?? "", isPublic: map["isPublic"] ?? false, isForPatreons: map["isForPatreons"] ?? false, isListed: map["isListed"] ?? false, lastModificationTime: DateTime.tryParse(map["lastModificationTime"]) ?? DateTimeX.epoch, resoniteDBManifest: (map["resoniteDBManifest"] as List? ?? []).map((e) => ResoniteDBAsset.fromMap(e)).toList(), lastModifyingUserId: map["lastModifyingUserId"] ?? "", lastModifyingMachineId: map["lastModifyingMachineId"] ?? "", creationTime: DateTime.tryParse(map["lastModificationTime"]) ?? DateTimeX.epoch, isSynced: map["isSynced"] ?? false, fetchedOn: DateTime.tryParse(map["fetchedOn"] ?? "") ?? DateTimeX.epoch, path: map["path"] ?? "", manifest: (map["resoniteDBManifest"] as List? ?? []).map((e) => e.toString()).toList(), url: map["url"] ?? "", isValidOwnerId: map["isValidOwnerId"] == "true", isValidRecordId: map["isValidRecordId"] == "true", visits: map["visits"] ?? 0, rating: map["rating"] ?? 0, randomOrder: map["randomOrder"] ?? 0); } factory Record.inventoryRoot() => _rootRecord; bool get isRoot => this == _rootRecord; bool get isLink => assetUri.startsWith("resrec"); bool get isItem => assetUri.startsWith("resdb"); bool get isGroupRecord => linkOwnerId.startsWith("G-"); String get linkRecordId { if (!isLink) { throw "Record is not a link."; } final lastSlashIdx = assetUri.lastIndexOf("/"); if (lastSlashIdx == -1) { throw "Record has invalid assetUri"; } return assetUri.substring(lastSlashIdx+1); } String get linkOwnerId { if (!assetUri.startsWith("resrec")) { throw "Record is not a link."; } String ownerId = assetUri.replaceFirst("resrec:///", ""); final nextSlashIdx = ownerId.indexOf("/"); if (nextSlashIdx == -1) { throw "Record has invalid assetUri"; } return ownerId.substring(0, nextSlashIdx); } Record copyWith({ String? id, String? ownerId, String? recordId, String? assetUri, int? globalVersion, int? localVersion, String? name, TextSpan? formattedName, String? description, List? tags, RecordType? recordType, String? thumbnailUri, bool? isPublic, bool? isForPatreons, bool? isListed, bool? isDeleted, DateTime? lastModificationTime, List? resoniteDBManifest, String? lastModifyingUserId, String? lastModifyingMachineId, DateTime? creationTime, RecordId? combinedRecordId, bool? isSynced, DateTime? fetchedOn, String? path, List? manifest, String? url, bool? isValidOwnerId, bool? isValidRecordId, int? visits, int? rating, int? randomOrder, }) { return Record( id: id ?? this.id, ownerId: ownerId ?? this.ownerId, assetUri: assetUri ?? this.assetUri, globalVersion: globalVersion ?? this.globalVersion, localVersion: localVersion ?? this.localVersion, name: name ?? this.name, description: description ?? this.description, tags: tags ?? this.tags, recordType: recordType ?? this.recordType, thumbnailUri: thumbnailUri ?? this.thumbnailUri, isPublic: isPublic ?? this.isPublic, isForPatreons: isForPatreons ?? this.isForPatreons, isListed: isListed ?? this.isListed, lastModificationTime: lastModificationTime ?? this.lastModificationTime, resoniteDBManifest: resoniteDBManifest ?? this.resoniteDBManifest, lastModifyingUserId: lastModifyingUserId ?? this.lastModifyingUserId, lastModifyingMachineId: lastModifyingMachineId ?? this.lastModifyingMachineId, creationTime: creationTime ?? this.creationTime, combinedRecordId: combinedRecordId ?? this.combinedRecordId, isSynced: isSynced ?? this.isSynced, fetchedOn: fetchedOn ?? this.fetchedOn, path: path ?? this.path, manifest: manifest ?? this.manifest, url: url ?? this.url, isValidOwnerId: isValidOwnerId ?? this.isValidOwnerId, isValidRecordId: isValidRecordId ?? this.isValidRecordId, visits: visits ?? this.visits, rating: rating ?? this.rating, randomOrder: randomOrder ?? this.randomOrder, ); } Map toMap() { return { "id": id, "ownerId": ownerId, "assetUri": assetUri, "globalVersion": globalVersion, "localVersion": localVersion, "name": name, "description": description.asNullable, "tags": tags, "recordType": recordType.name, "thumbnailUri": thumbnailUri.asNullable, "isPublic": isPublic, "isForPatreons": isForPatreons, "isListed": isListed, "lastModificationTime": lastModificationTime.toUtc().toIso8601String(), "resoniteDBManifest": resoniteDBManifest.map((e) => e.toMap()).toList(), "lastModifyingUserId": lastModifyingUserId, "lastModifyingMachineId": lastModifyingMachineId, "creationTime": creationTime.toUtc().toIso8601String(), "combinedRecordId": combinedRecordId.toMap(), "isSynced": isSynced, "fetchedOn": fetchedOn.toUtc().toIso8601String(), "path": path.asNullable, "manifest": manifest, "url": url, "isValidOwnerId": isValidOwnerId, "isValidRecordId": isValidRecordId, "visits": visits, "rating": rating, "randomOrder": randomOrder, }; } static String generateId() { return "R-${const Uuid().v4()}"; } String? extractMessageId() { const key = "message_id:"; for (final tag in tags) { if (tag.startsWith(key)) { return tag.replaceFirst(key, ""); } } return null; } }