* fix: web, skip querying onlines, if not in main page Signed-off-by: fufesou <linlong1266@gmail.com> * fix: web/mobile, skip querying onlines Signed-off-by: fufesou <linlong1266@gmail.com> * Set isInMainPage to false after router is changed. Signed-off-by: fufesou <linlong1266@gmail.com> --------- Signed-off-by: fufesou <linlong1266@gmail.com>
		
			
				
	
	
		
			580 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			580 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| 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/peer_tab_model.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<String> 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';
 | |
| }
 | |
| 
 | |
| class PeersModelName {
 | |
|   static const String recent = 'recent peer';
 | |
|   static const String favorite = 'fav peer';
 | |
|   static const String lan = 'discovered peer';
 | |
|   static const String addressBook = 'address book peer';
 | |
|   static const String group = 'group peer';
 | |
| }
 | |
| 
 | |
| /// 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<RxString> 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<String, String> _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 = <String>{};
 | |
|   var _lastChangeTime = DateTime.now();
 | |
|   var _lastQueryPeers = <String>{};
 | |
|   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<Peers>.value(
 | |
|       value: widget.peers,
 | |
|       child: Consumer<Peers>(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<RxList>((filters) {
 | |
|       return FutureBuilder<List<Peer>>(
 | |
|         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<PeerTabModel>(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 {
 | |
|           final skipIfIsWeb =
 | |
|               isWeb && !(stateGlobal.isWebVisible && stateGlobal.isInMainPage);
 | |
|           final skipIfMobile =
 | |
|               (isAndroid || isIOS) && !stateGlobal.isInMainPage;
 | |
|           final skipIfNotActive = skipIfIsWeb || skipIfMobile || !_isActive;
 | |
|           if (!skipIfNotActive && (_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<List<Peer>>? matchPeers(
 | |
|       String searchText, String sortedBy, List<Peer> 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<Peer>.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 PeerTabIndex peerTabIndex;
 | |
|   final PeerFilter? peerFilter;
 | |
|   final PeerCardBuilder peerCardBuilder;
 | |
| 
 | |
|   const BasePeersView({
 | |
|     Key? key,
 | |
|     required this.peerTabIndex,
 | |
|     this.peerFilter,
 | |
|     required this.peerCardBuilder,
 | |
|   }) : super(key: key);
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context) {
 | |
|     Peers peers;
 | |
|     switch (peerTabIndex) {
 | |
|       case PeerTabIndex.recent:
 | |
|         peers = gFFI.recentPeersModel;
 | |
|         break;
 | |
|       case PeerTabIndex.fav:
 | |
|         peers = gFFI.favoritePeersModel;
 | |
|         break;
 | |
|       case PeerTabIndex.lan:
 | |
|         peers = gFFI.lanPeersModel;
 | |
|         break;
 | |
|       case PeerTabIndex.ab:
 | |
|         peers = gFFI.abModel.peersModel;
 | |
|         break;
 | |
|       case PeerTabIndex.group:
 | |
|         peers = gFFI.groupModel.peersModel;
 | |
|         break;
 | |
|     }
 | |
|     return _PeersView(
 | |
|         peers: peers, peerFilter: peerFilter, peerCardBuilder: peerCardBuilder);
 | |
|   }
 | |
| }
 | |
| 
 | |
| class RecentPeersView extends BasePeersView {
 | |
|   RecentPeersView(
 | |
|       {Key? key, EdgeInsets? menuPadding, ScrollController? scrollController})
 | |
|       : super(
 | |
|           key: key,
 | |
|           peerTabIndex: PeerTabIndex.recent,
 | |
|           peerCardBuilder: (Peer peer) => RecentPeerCard(
 | |
|             peer: peer,
 | |
|             menuPadding: menuPadding,
 | |
|           ),
 | |
|         );
 | |
| 
 | |
|   @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,
 | |
|           peerTabIndex: PeerTabIndex.fav,
 | |
|           peerCardBuilder: (Peer peer) => FavoritePeerCard(
 | |
|             peer: peer,
 | |
|             menuPadding: menuPadding,
 | |
|           ),
 | |
|         );
 | |
| 
 | |
|   @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,
 | |
|           peerTabIndex: PeerTabIndex.lan,
 | |
|           peerCardBuilder: (Peer peer) => DiscoveredPeerCard(
 | |
|             peer: peer,
 | |
|             menuPadding: menuPadding,
 | |
|           ),
 | |
|         );
 | |
| 
 | |
|   @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})
 | |
|       : super(
 | |
|           key: key,
 | |
|           peerTabIndex: PeerTabIndex.ab,
 | |
|           peerFilter: (Peer peer) =>
 | |
|               _hitTag(gFFI.abModel.selectedTags, peer.tags),
 | |
|           peerCardBuilder: (Peer peer) => AddressBookPeerCard(
 | |
|             peer: peer,
 | |
|             menuPadding: menuPadding,
 | |
|           ),
 | |
|         );
 | |
| 
 | |
|   static bool _hitTag(List<dynamic> selectedTags, List<dynamic> 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})
 | |
|       : super(
 | |
|           key: key,
 | |
|           peerTabIndex: PeerTabIndex.group,
 | |
|           peerFilter: filter,
 | |
|           peerCardBuilder: (Peer peer) => MyGroupPeerCard(
 | |
|             peer: peer,
 | |
|             menuPadding: menuPadding,
 | |
|           ),
 | |
|         );
 | |
| 
 | |
|   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;
 | |
|   }
 | |
| }
 |