import 'dart:async'; import 'dart:collection'; import 'package:dynamic_layouts/dynamic_layouts.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_hbb/consts.dart'; import 'package:flutter_hbb/desktop/widgets/scroll_wrapper.dart'; import 'package:flutter_hbb/models/state_model.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'; static const String group = 'load_group_peers'; } /// for peer search text, global obs value final peerSearchText = "".obs; /// for peer sort, global obs value RxString? _peerSort; RxString get peerSort { _peerSort ??= bind.getLocalFlutterOption(k: kOptionPeerSorting).obs; return _peerSort!; } // list for listener RxList get 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, WidgetsBindingObserver { 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 || isWebDesktop) ? 12.0 : 8.0; final _curPeers = {}; var _lastChangeTime = DateTime.now(); var _lastQueryPeers = {}; var _lastQueryTime = DateTime.now(); var _lastWindowRestoreTime = DateTime.now(); var _queryCount = 0; var _exit = false; bool _isActive = true; final _scrollController = ScrollController(); _PeersViewState() { _startCheckOnlines(); } @override void initState() { windowManager.addListener(this); WidgetsBinding.instance.addObserver(this); super.initState(); } @override void dispose() { windowManager.removeListener(this); WidgetsBinding.instance.removeObserver(this); _exit = true; super.dispose(); } @override void onWindowFocus() { _queryCount = 0; _isActive = true; } @override void onWindowBlur() { // We need this comparison because window restore (on Windows) also triggers `onWindowBlur()`. // Maybe it's a bug of the window manager, but the source code seems to be correct. // // Although `onWindowRestore()` is called after `onWindowBlur()` in my test, // we need the following comparison to ensure that `_isActive` is true in the end. if (isWindows && DateTime.now().difference(_lastWindowRestoreTime) < const Duration(milliseconds: 300)) { return; } _queryCount = _maxQueryCount; _isActive = false; } @override void onWindowRestore() { // Window restore (on MacOS and Linux) also triggers `onWindowFocus()`. // But on Windows, it triggers `onWindowBlur()`, mybe it's a bug of the window manager. if (!isWindows) return; _queryCount = 0; _isActive = true; _lastWindowRestoreTime = DateTime.now(); } @override void onWindowMinimize() { // Window minimize also triggers `onWindowBlur()`. } // This function is required for mobile. // `onWindowFocus` works fine for desktop. @override void didChangeAppLifecycleState(AppLifecycleState state) { super.didChangeAppLifecycleState(state); if (isDesktop || isWebDesktop) return; if (state == AppLifecycleState.resumed) { _isActive = true; _queryCount = 0; } else if (state == AppLifecycleState.inactive) { _isActive = false; } } @override Widget build(BuildContext context) { // We should avoid too many rebuilds. MacOS(m1, 14.6.1) on Flutter 3.19.6. // Continious rebuilds of `ChangeNotifierProvider` will cause memory leak. // Simple demo can reproduce this issue. return ChangeNotifierProvider( create: (context) => widget.peers, child: Consumer(builder: (context, peers, child) { if (peers.peers.isEmpty) { gFFI.peerTabModel.setCurrentTabCachedPeers([]); return 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, ), ), ], ), ); } else { return _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) { var peers = snapshot.data!; if (peers.length > 1000) peers = peers.sublist(0, 1000); gFFI.peerTabModel.setCurrentTabCachedPeers(peers); buildOnePeer(Peer peer, bool isPortrait) { final visibilityChild = VisibilityDetector( key: ValueKey(_cardId(peer.id)), onVisibilityChanged: onVisibilityChanged, child: widget.peerCardBuilder(peer), ); // `Provider.of(context)` will causes infinete loop. // Because `gFFI.peerTabModel.setCurrentTabCachedPeers(peers)` will trigger `notifyListeners()`. // // No need to listen the currentTab change event. // Because the currentTab change event will trigger the peers change event, // and the peers change event will trigger _buildPeersView(). return !isPortrait ? Obx(() => peerCardUiType.value == PeerUiType.list ? Container(height: 45, child: visibilityChild) : peerCardUiType.value == PeerUiType.grid ? SizedBox( width: 220, height: 140, child: visibilityChild) : SizedBox( width: 220, height: 42, child: visibilityChild)) : Container(child: visibilityChild); } // We should avoid too many rebuilds. Win10(Some machines) on Flutter 3.19.6. // Continious rebuilds of `ListView.builder` will cause memory leak. // Simple demo can reproduce this issue. final Widget child = Obx(() => stateGlobal.isPortrait.isTrue ? ListView.builder( itemCount: peers.length, itemBuilder: (BuildContext context, int index) { return buildOnePeer(peers[index], true).marginOnly( top: index == 0 ? 0 : space / 2, bottom: space / 2); }, ) : peerCardUiType.value == PeerUiType.list ? DesktopScrollWrapper( scrollController: _scrollController, child: ListView.builder( controller: _scrollController, physics: DraggableNeverScrollableScrollPhysics(), itemCount: peers.length, itemBuilder: (BuildContext context, int index) { return buildOnePeer(peers[index], false) .marginOnly( right: space, top: index == 0 ? 0 : space / 2, bottom: space / 2); }), ) : DesktopScrollWrapper( scrollController: _scrollController, child: DynamicGridView.builder( controller: _scrollController, physics: DraggableNeverScrollableScrollPhysics(), gridDelegate: SliverGridDelegateWithWrapping( mainAxisSpacing: space / 2, crossAxisSpacing: space), itemCount: peers.length, itemBuilder: (BuildContext context, int index) { return buildOnePeer(peers[index], false); }), )); 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; } var _queryInterval = const Duration(seconds: 20); void _startCheckOnlines() { () async { final p = await bind.mainIsUsingPublicServer(); if (!p) { _queryInterval = const Duration(seconds: 6); } while (!_exit) { final now = DateTime.now(); if (!setEquals(_curPeers, _lastQueryPeers)) { if (now.difference(_lastChangeTime) > const Duration(seconds: 1)) { _queryOnlines(false); } } else { if (_isActive && (_queryCount < _maxQueryCount || !p)) { if (now.difference(_lastQueryTime) >= _queryInterval) { if (_curPeers.isNotEmpty) { bind.queryOnlines(ids: _curPeers.toList(growable: false)); _lastQueryTime = DateTime.now(); _queryCount += 1; } } } } await Future.delayed(const Duration(milliseconds: 300)); } }(); } _queryOnlines(bool isLoadEvent) { if (_curPeers.isNotEmpty) { bind.queryOnlines(ids: _curPeers.toList(growable: false)); _queryCount = 0; } _lastQueryPeers = {..._curPeers}; if (isLoadEvent) { _lastChangeTime = DateTime.now(); } else { _lastQueryTime = DateTime.now().subtract(_queryInterval); } } 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.setLocalFlutterOption( k: kOptionPeerSorting, 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 GetInitPeers? getInitPeers; const BasePeersView({ Key? key, required this.name, required this.loadEvent, this.peerFilter, required this.peerCardBuilder, required this.getInitPeers, }) : super(key: key); @override Widget build(BuildContext context) { return _PeersView( peers: Peers(name: name, loadEvent: loadEvent, getInitPeers: getInitPeers), 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, ), getInitPeers: null, ); @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, ), getInitPeers: null, ); @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, ), getInitPeers: null, ); @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 GetInitPeers getInitPeers}) : 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, ), getInitPeers: getInitPeers, ); static bool _hitTag(List selectedTags, List idents) { if (selectedTags.isEmpty) { return true; } if (gFFI.abModel.filterByIntersection.value) { for (final tag in selectedTags) { if (!idents.contains(tag)) { return false; } } return true; } else { for (final tag in selectedTags) { if (idents.contains(tag)) { return true; } } return false; } } } class MyGroupPeerView extends BasePeersView { MyGroupPeerView( {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController, required GetInitPeers getInitPeers}) : super( key: key, name: 'group peer', loadEvent: LoadEvent.group, peerFilter: filter, peerCardBuilder: (Peer peer) => MyGroupPeerCard( peer: peer, menuPadding: menuPadding, ), getInitPeers: getInitPeers, ); static bool filter(Peer peer) { if (gFFI.groupModel.searchUserText.isNotEmpty) { if (!peer.loginName.contains(gFFI.groupModel.searchUserText)) { return false; } } if (gFFI.groupModel.selectedUser.isNotEmpty) { if (gFFI.groupModel.selectedUser.value != peer.loginName) { return false; } } return true; } }