/* MIT License * * Copyright (C) 2019, 2020, 2021 Famedly GmbH * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ import 'package:matrix/matrix_api_lite/model/basic_event.dart'; import 'package:matrix/matrix_api_lite/utils/filter_map_extension.dart'; import 'package:matrix/matrix_api_lite/utils/try_get_map_extension.dart'; extension ImagePackContentBasicEventExtension on BasicEvent { ImagePackContent get parsedImagePackContent => ImagePackContent.fromJson(content); } enum ImagePackUsage { sticker, emoticon, } List? imagePackUsageFromJson(List? json) => json ?.map( (v) => { 'sticker': ImagePackUsage.sticker, 'emoticon': ImagePackUsage.emoticon, }[v], ) .whereType() .toList(); List imagePackUsageToJson( List? usage, List? prevUsage, ) { final knownUsages = {'sticker', 'emoticon'}; final usagesStr = usage ?.map( (v) => { ImagePackUsage.sticker: 'sticker', ImagePackUsage.emoticon: 'emoticon', }[v], ) .whereType() .toList() ?? []; // first we add all the unknown usages and the previous known usages which are new again final newUsages = prevUsage ?.where((v) => !knownUsages.contains(v) || usagesStr.contains(v)) .toList() ?? []; // now we need to add the new usages that we didn't add yet newUsages.addAll(usagesStr.where((v) => !newUsages.contains(v))); return newUsages; } class ImagePackContent { // we want to preserve potential custom keys in this object final Map _json; Map images; ImagePackPackContent pack; ImagePackContent({required this.images, required this.pack}) : _json = {}; ImagePackContent.fromJson(Map json) : _json = Map.fromEntries( json.entries.where( (e) => !['images', 'pack', 'emoticons', 'short'].contains(e.key), ), ), pack = ImagePackPackContent.fromJson( json.tryGetMap('pack') ?? {}, ), images = json.tryGetMap('images')?.catchMap( (k, v) => MapEntry( k, ImagePackImageContent.fromJson( v as Map, ), ), ) ?? // the "emoticons" key needs a small migration on the key, ":string:" --> "string" json.tryGetMap('emoticons')?.catchMap( (k, v) => MapEntry( k.startsWith(':') && k.endsWith(':') ? k.substring(1, k.length - 1) : k, ImagePackImageContent.fromJson( v as Map, ), ), ) ?? // the "short" key was still just a map from shortcode to mxc uri json.tryGetMap('short')?.catchMap( (k, v) => MapEntry( k.startsWith(':') && k.endsWith(':') ? k.substring(1, k.length - 1) : k, ImagePackImageContent(url: Uri.parse(v)), ), ) ?? {}; Map toJson() => { ..._json, 'images': images.map((k, v) => MapEntry(k, v.toJson())), 'pack': pack.toJson(), }; } class ImagePackImageContent { // we want to preserve potential custom keys in this object final Map _json; Uri url; String? body; Map? info; List? usage; ImagePackImageContent({required this.url, this.body, this.info, this.usage}) : _json = {}; ImagePackImageContent.fromJson(Map json) : _json = Map.fromEntries( json.entries.where((e) => !['url', 'body', 'info'].contains(e.key)), ), url = Uri.parse(json['url'] as String), body = json.tryGet('body'), info = json.tryGetMap('info'), usage = imagePackUsageFromJson(json.tryGetList('usage')); Map toJson() { return { ...Map.from(_json)..remove('usage'), 'url': url.toString(), if (body != null) 'body': body, if (info != null) 'info': info, if (usage != null) 'usage': imagePackUsageToJson(usage, _json.tryGetList('usage')), }; } } class ImagePackPackContent { // we want to preserve potential custom keys in this object final Map _json; String? displayName; Uri? avatarUrl; List? usage; String? attribution; ImagePackPackContent({ this.displayName, this.avatarUrl, this.usage, this.attribution, }) : _json = {}; ImagePackPackContent.fromJson(Map json) : _json = Map.fromEntries( json.entries.where( (e) => !['display_name', 'avatar_url', 'attribution'].contains(e.key), ), ), displayName = json.tryGet('display_name'), // we default to an invalid uri avatarUrl = Uri.tryParse(json.tryGet('avatar_url') ?? '.::'), usage = imagePackUsageFromJson(json.tryGetList('usage')), attribution = json.tryGet('attribution'); Map toJson() { return { ...Map.from(_json)..remove('usage'), if (displayName != null) 'display_name': displayName, if (avatarUrl != null) 'avatar_url': avatarUrl.toString(), if (usage != null) 'usage': imagePackUsageToJson(usage, _json.tryGetList('usage')), if (attribution != null) 'attribution': attribution, }; } }