import 'dart:async'; import 'dart:collection'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:get/get.dart'; import 'package:provider/provider.dart'; import 'package:visibility_detector/visibility_detector.dart'; import 'package:window_manager/window_manager.dart'; import '../../common.dart'; import '../../models/peer_model.dart'; import '../../models/platform_model.dart'; import 'peer_card.dart'; typedef PeerFilter = bool Function(Peer peer); typedef PeerCardBuilder = Widget Function(Peer peer); class PeerSortType { static const String remoteId = 'Remote ID'; static const String remoteHost = 'Remote Host'; static const String username = 'Username'; // static const String status = 'Status'; static List values = [ PeerSortType.remoteId, PeerSortType.remoteHost, PeerSortType.username, // PeerSortType.status ]; } class LoadEvent { static const String recent = 'load_recent_peers'; static const String favorite = 'load_fav_peers'; static const String lan = 'load_lan_peers'; static const String addressBook = 'load_address_book_peers'; } /// for peer search text, global obs value final peerSearchText = "".obs; /// for peer sort, global obs value final peerSort = bind.getLocalFlutterConfig(k: 'peer-sorting').obs; // list for listener final obslist = [peerSearchText, peerSort].obs; final peerSearchTextController = TextEditingController(text: peerSearchText.value); class _PeersView extends StatefulWidget { final Peers peers; final PeerFilter? peerFilter; final PeerCardBuilder peerCardBuilder; const _PeersView( {required this.peers, required this.peerCardBuilder, this.peerFilter, Key? key}) : super(key: key); @override _PeersViewState createState() => _PeersViewState(); } /// State for the peer widget. class _PeersViewState extends State<_PeersView> with WindowListener { static const int _maxQueryCount = 3; final HashMap _emptyMessages = HashMap.from({ LoadEvent.recent: 'empty_recent_tip', LoadEvent.favorite: 'empty_favorite_tip', LoadEvent.lan: 'empty_lan_tip', LoadEvent.addressBook: 'empty_address_book_tip', }); final space = isDesktop ? 12.0 : 8.0; final _curPeers = {}; var _lastChangeTime = DateTime.now(); var _lastQueryPeers = {}; var _lastQueryTime = DateTime.now().subtract(const Duration(hours: 1)); var _queryCount = 0; var _exit = false; late final mobileWidth = () { const minWidth = 320.0; final windowWidth = MediaQuery.of(context).size.width; var width = windowWidth - 2 * space; if (windowWidth > minWidth + 2 * space) { final n = (windowWidth / (minWidth + 2 * space)).floor(); width = windowWidth / n - 2 * space; } return width; }(); _PeersViewState() { _startCheckOnlines(); } @override void initState() { windowManager.addListener(this); super.initState(); } @override void dispose() { windowManager.removeListener(this); _exit = true; super.dispose(); } @override void onWindowFocus() { _queryCount = 0; } @override void onWindowMinimize() { _queryCount = _maxQueryCount; } @override Widget build(BuildContext context) { return ChangeNotifierProvider( create: (context) => widget.peers, child: Consumer( builder: (context, peers, child) => peers.peers.isEmpty ? Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon( Icons.sentiment_very_dissatisfied_rounded, color: Theme.of(context).tabBarTheme.labelColor, size: 40, ).paddingOnly(bottom: 10), Text( translate( _emptyMessages[widget.peers.loadEvent] ?? 'Empty', ), textAlign: TextAlign.center, style: TextStyle( color: Theme.of(context).tabBarTheme.labelColor, ), ), ], ), ) : _buildPeersView(peers), ), ); } onVisibilityChanged(VisibilityInfo info) { final peerId = _peerId((info.key as ValueKey).value); if (info.visibleFraction > 0.00001) { _curPeers.add(peerId); } else { _curPeers.remove(peerId); } _lastChangeTime = DateTime.now(); } String _cardId(String id) => widget.peers.name + id; String _peerId(String cardId) => cardId.replaceAll(widget.peers.name, ''); Widget _buildPeersView(Peers peers) { final updateEvent = peers.event; final body = ObxValue((filters) { return FutureBuilder>( builder: (context, snapshot) { if (snapshot.hasData) { final peers = snapshot.data!; gFFI.peerTabModel.setCurrentTabCachedPeers(peers); final cards = []; for (final peer in peers) { final visibilityChild = VisibilityDetector( key: ValueKey(_cardId(peer.id)), onVisibilityChanged: onVisibilityChanged, child: widget.peerCardBuilder(peer), ); cards.add(isDesktop ? Obx( () => SizedBox( width: 220, height: peerCardUiType.value == PeerUiType.grid ? 140 : 42, child: visibilityChild, ), ) : SizedBox(width: mobileWidth, child: visibilityChild)); } final child = Wrap(spacing: space, runSpacing: space, children: cards); if (updateEvent == UpdateEvent.load) { _curPeers.clear(); _curPeers.addAll(peers.map((e) => e.id)); _queryOnlines(true); } return child; } else { return const Center( child: CircularProgressIndicator(), ); } }, future: matchPeers(filters[0].value, filters[1].value, peers.peers), ); }, obslist); return body; } final _queryInterval = const Duration(seconds: 20); void _startCheckOnlines() { () async { while (!_exit) { final now = DateTime.now(); if (!setEquals(_curPeers, _lastQueryPeers)) { if (now.difference(_lastChangeTime) > const Duration(seconds: 1)) { _queryOnlines(false); } } else { if (_queryCount < _maxQueryCount) { if (now.difference(_lastQueryTime) >= _queryInterval) { if (_curPeers.isNotEmpty) { platformFFI.ffiBind .queryOnlines(ids: _curPeers.toList(growable: false)); _lastQueryTime = DateTime.now(); _queryCount += 1; } } } } await Future.delayed(const Duration(milliseconds: 300)); } }(); } _queryOnlines(bool isLoadEvent) { if (_curPeers.isNotEmpty) { platformFFI.ffiBind.queryOnlines(ids: _curPeers.toList(growable: false)); _lastQueryPeers = {..._curPeers}; if (isLoadEvent) { _lastChangeTime = DateTime.now(); } else { _lastQueryTime = DateTime.now().subtract(_queryInterval); } _queryCount = 0; } } Future>? matchPeers( String searchText, String sortedBy, List peers) async { if (widget.peerFilter != null) { peers = peers.where((peer) => widget.peerFilter!(peer)).toList(); } // fallback to id sorting if (!PeerSortType.values.contains(sortedBy)) { sortedBy = PeerSortType.remoteId; bind.setLocalFlutterConfig( k: "peer-sorting", v: sortedBy, ); } if (widget.peers.loadEvent != LoadEvent.recent) { switch (sortedBy) { case PeerSortType.remoteId: peers.sort((p1, p2) => p1.getId().compareTo(p2.getId())); break; case PeerSortType.remoteHost: peers.sort((p1, p2) => p1.hostname.toLowerCase().compareTo(p2.hostname.toLowerCase())); break; case PeerSortType.username: peers.sort((p1, p2) => p1.username.toLowerCase().compareTo(p2.username.toLowerCase())); break; // case PeerSortType.status: // peers.sort((p1, p2) => p1.online ? -1 : 1); // break; } } searchText = searchText.trim(); if (searchText.isEmpty) { return peers; } searchText = searchText.toLowerCase(); final matches = await Future.wait(peers.map((peer) => matchPeer(searchText, peer))); final filteredList = List.empty(growable: true); for (var i = 0; i < peers.length; i++) { if (matches[i]) { filteredList.add(peers[i]); } } return filteredList; } } abstract class BasePeersView extends StatelessWidget { final String name; final String loadEvent; final PeerFilter? peerFilter; final PeerCardBuilder peerCardBuilder; final List initPeers; const BasePeersView({ Key? key, required this.name, required this.loadEvent, this.peerFilter, required this.peerCardBuilder, required this.initPeers, }) : super(key: key); @override Widget build(BuildContext context) { return _PeersView( peers: Peers(name: name, loadEvent: loadEvent, peers: initPeers), peerFilter: peerFilter, peerCardBuilder: peerCardBuilder); } } class RecentPeersView extends BasePeersView { RecentPeersView( {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController}) : super( key: key, name: 'recent peer', loadEvent: LoadEvent.recent, peerCardBuilder: (Peer peer) => RecentPeerCard( peer: peer, menuPadding: menuPadding, ), initPeers: [], ); @override Widget build(BuildContext context) { final widget = super.build(context); bind.mainLoadRecentPeers(); return widget; } } class FavoritePeersView extends BasePeersView { FavoritePeersView( {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController}) : super( key: key, name: 'favorite peer', loadEvent: LoadEvent.favorite, peerCardBuilder: (Peer peer) => FavoritePeerCard( peer: peer, menuPadding: menuPadding, ), initPeers: [], ); @override Widget build(BuildContext context) { final widget = super.build(context); bind.mainLoadFavPeers(); return widget; } } class DiscoveredPeersView extends BasePeersView { DiscoveredPeersView( {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController}) : super( key: key, name: 'discovered peer', loadEvent: LoadEvent.lan, peerCardBuilder: (Peer peer) => DiscoveredPeerCard( peer: peer, menuPadding: menuPadding, ), initPeers: [], ); @override Widget build(BuildContext context) { final widget = super.build(context); bind.mainLoadLanPeers(); return widget; } } class AddressBookPeersView extends BasePeersView { AddressBookPeersView( {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController, required List initPeers}) : super( key: key, name: 'address book peer', loadEvent: LoadEvent.addressBook, peerFilter: (Peer peer) => _hitTag(gFFI.abModel.selectedTags, peer.tags), peerCardBuilder: (Peer peer) => AddressBookPeerCard( peer: peer, menuPadding: menuPadding, ), initPeers: initPeers, ); static bool _hitTag(List selectedTags, List idents) { if (selectedTags.isEmpty) { return true; } if (idents.isEmpty) { return false; } for (final tag in selectedTags) { if (!idents.contains(tag)) { return false; } } return true; } } class MyGroupPeerView extends BasePeersView { MyGroupPeerView( {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController, required List initPeers}) : super( key: key, name: 'my group peer', loadEvent: 'load_my_group_peers', peerFilter: filter, peerCardBuilder: (Peer peer) => MyGroupPeerCard( peer: peer, menuPadding: menuPadding, ), initPeers: initPeers, ); static bool filter(Peer peer) { if (gFFI.groupModel.searchUserText.isNotEmpty) { if (!peer.username.contains(gFFI.groupModel.searchUserText)) { return false; } } if (gFFI.groupModel.selectedUser.isNotEmpty) { if (gFFI.groupModel.selectedUser.value != peer.username) { return false; } } return true; } }