import 'dart:convert'; import 'package:flutter/foundation.dart'; import 'package:get/get.dart'; import 'platform_model.dart'; // ignore: depend_on_referenced_packages import 'package:collection/collection.dart'; class Peer { final String id; String hash; // personal ab hash password String password; // shared ab password String username; // pc username String hostname; String platform; String alias; List tags; bool forceAlwaysRelay = false; String rdpPort; String rdpUsername; bool online = false; String loginName; //login username bool? sameServer; String getId() { if (alias != '') { return alias; } return id; } Peer.fromJson(Map json) : id = json['id'] ?? '', hash = json['hash'] ?? '', password = json['password'] ?? '', username = json['username'] ?? '', hostname = json['hostname'] ?? '', platform = json['platform'] ?? '', alias = json['alias'] ?? '', tags = json['tags'] ?? [], forceAlwaysRelay = json['forceAlwaysRelay'] == 'true', rdpPort = json['rdpPort'] ?? '', rdpUsername = json['rdpUsername'] ?? '', loginName = json['loginName'] ?? '', sameServer = json['same_server']; Map toJson() { return { "id": id, "hash": hash, "password": password, "username": username, "hostname": hostname, "platform": platform, "alias": alias, "tags": tags, "forceAlwaysRelay": forceAlwaysRelay.toString(), "rdpPort": rdpPort, "rdpUsername": rdpUsername, 'loginName': loginName, 'same_server': sameServer, }; } Map toCustomJson({required bool includingHash}) { var res = { "id": id, "username": username, "hostname": hostname, "platform": platform, "alias": alias, "tags": tags, }; if (includingHash) { res['hash'] = hash; } return res; } Map toGroupCacheJson() { return { "id": id, "username": username, "hostname": hostname, "platform": platform, "login_name": loginName, }; } Peer({ required this.id, required this.hash, required this.password, required this.username, required this.hostname, required this.platform, required this.alias, required this.tags, required this.forceAlwaysRelay, required this.rdpPort, required this.rdpUsername, required this.loginName, this.sameServer, }); Peer.loading() : this( id: '...', hash: '', password: '', username: '...', hostname: '...', platform: '...', alias: '', tags: [], forceAlwaysRelay: false, rdpPort: '', rdpUsername: '', loginName: '', ); bool equal(Peer other) { return id == other.id && hash == other.hash && password == other.password && username == other.username && hostname == other.hostname && platform == other.platform && alias == other.alias && tags.equals(other.tags) && forceAlwaysRelay == other.forceAlwaysRelay && rdpPort == other.rdpPort && rdpUsername == other.rdpUsername && loginName == other.loginName; } Peer.copy(Peer other) : this( id: other.id, hash: other.hash, password: other.password, username: other.username, hostname: other.hostname, platform: other.platform, alias: other.alias, tags: other.tags.toList(), forceAlwaysRelay: other.forceAlwaysRelay, rdpPort: other.rdpPort, rdpUsername: other.rdpUsername, loginName: other.loginName, sameServer: other.sameServer); } enum UpdateEvent { online, load } typedef GetInitPeers = RxList Function(); class Peers extends ChangeNotifier { final String name; final String loadEvent; List peers = List.empty(growable: true); final GetInitPeers? getInitPeers; UpdateEvent event = UpdateEvent.load; static const _cbQueryOnlines = 'callback_query_onlines'; Peers( {required this.name, required this.getInitPeers, required this.loadEvent}) { peers = getInitPeers?.call() ?? []; platformFFI.registerEventHandler(_cbQueryOnlines, name, (evt) async { _updateOnlineState(evt); }); platformFFI.registerEventHandler(loadEvent, name, (evt) async { _updatePeers(evt); }); } @override void dispose() { platformFFI.unregisterEventHandler(_cbQueryOnlines, name); platformFFI.unregisterEventHandler(loadEvent, name); super.dispose(); } Peer getByIndex(int index) { if (index < peers.length) { return peers[index]; } else { return Peer.loading(); } } int getPeersCount() { return peers.length; } void _updateOnlineState(Map evt) { int changedCount = 0; evt['onlines'].split(',').forEach((online) { for (var i = 0; i < peers.length; i++) { if (peers[i].id == online) { if (!peers[i].online) { changedCount += 1; peers[i].online = true; } } } }); evt['offlines'].split(',').forEach((offline) { for (var i = 0; i < peers.length; i++) { if (peers[i].id == offline) { if (peers[i].online) { changedCount += 1; peers[i].online = false; } } } }); if (changedCount > 0) { event = UpdateEvent.online; notifyListeners(); } } void _updatePeers(Map evt) { final onlineStates = _getOnlineStates(); if (getInitPeers != null) { peers = getInitPeers?.call() ?? []; } else { peers = _decodePeers(evt['peers']); } for (var peer in peers) { final state = onlineStates[peer.id]; peer.online = state != null && state != false; } event = UpdateEvent.load; notifyListeners(); } Map _getOnlineStates() { var onlineStates = {}; for (var peer in peers) { onlineStates[peer.id] = peer.online; } return onlineStates; } List _decodePeers(String peersStr) { try { if (peersStr == "") return []; List peers = json.decode(peersStr); return peers.map((peer) { return Peer.fromJson(peer as Map); }).toList(); } catch (e) { debugPrint('peers(): $e'); } return []; } }