import 'dart:convert'; import 'dart:math'; import 'package:flutter/material.dart'; import 'package:flutter_hbb/consts.dart'; import 'package:flutter_hbb/models/peer_model.dart'; import 'package:flutter_hbb/models/platform_model.dart'; import 'package:get/get.dart'; import '../common.dart'; import 'model.dart'; enum PeerTabIndex { recent, fav, lan, ab, group, } class PeerTabModel with ChangeNotifier { WeakReference parent; int get currentTab => _currentTab; int _currentTab = 0; // index in tabNames static const int maxTabCount = 5; static const List tabNames = [ 'Recent sessions', 'Favorites', 'Discovered', 'Address book', 'Group', ]; static const List icons = [ Icons.access_time_filled, Icons.star, Icons.explore, IconFont.addressBook, Icons.group, ]; List isEnabled = List.from([ true, true, !isWeb, !(bind.isDisableAb() || bind.isDisableAccount()), !(bind.isDisableGroupPanel() || bind.isDisableAccount()), ]); final List _isVisible = List.filled(maxTabCount, true, growable: false); List get isVisibleEnabled => () { final list = _isVisible.toList(); for (int i = 0; i < maxTabCount; i++) { list[i] = list[i] && isEnabled[i]; } return list; }(); final List orders = List.generate(maxTabCount, (index) => index, growable: false); List get visibleEnabledOrderedIndexs => orders.where((e) => isVisibleEnabled[e]).toList(); List _selectedPeers = List.empty(growable: true); List get selectedPeers => _selectedPeers; bool _multiSelectionMode = false; bool get multiSelectionMode => _multiSelectionMode; List _currentTabCachedPeers = List.empty(growable: true); List get currentTabCachedPeers => _currentTabCachedPeers; bool _isShiftDown = false; bool get isShiftDown => _isShiftDown; String _lastId = ''; String get lastId => _lastId; PeerTabModel(this.parent) { // visible try { final option = bind.getLocalFlutterOption(k: kOptionPeerTabVisible); if (option.isNotEmpty) { List decodeList = jsonDecode(option); if (decodeList.length == _isVisible.length) { for (int i = 0; i < _isVisible.length; i++) { if (decodeList[i] is bool) { _isVisible[i] = decodeList[i]; } } } } } catch (e) { debugPrint("failed to get peer tab visible list:$e"); } // order try { final option = bind.getLocalFlutterOption(k: kOptionPeerTabOrder); if (option.isNotEmpty) { List decodeList = jsonDecode(option); if (decodeList.length == maxTabCount) { var sortedList = decodeList.toList(); sortedList.sort(); bool valid = true; for (int i = 0; i < maxTabCount; i++) { if (sortedList[i] is! int || sortedList[i] != i) { valid = false; } } if (valid) { for (int i = 0; i < orders.length; i++) { orders[i] = decodeList[i]; } } } } } catch (e) { debugPrint("failed to get peer tab order list: $e"); } // init currentTab _currentTab = int.tryParse(bind.getLocalFlutterOption(k: kOptionPeerTabIndex)) ?? 0; if (_currentTab < 0 || _currentTab >= maxTabCount) { _currentTab = 0; } _trySetCurrentTabToFirstVisibleEnabled(); } setCurrentTab(int index) { if (_currentTab != index) { _currentTab = index; notifyListeners(); } } String tabTooltip(int index) { if (index >= 0 && index < tabNames.length) { return translate(tabNames[index]); } return index.toString(); } IconData tabIcon(int index) { if (index >= 0 && index < icons.length) { return icons[index]; } return Icons.help; } setMultiSelectionMode(bool mode) { _multiSelectionMode = mode; if (!mode) { _selectedPeers.clear(); _lastId = ''; } notifyListeners(); } select(Peer peer) { if (!_multiSelectionMode) { // https://github.com/flutter/flutter/issues/101275#issuecomment-1604541700 // After onTap, the shift key should be pressed for a while when not in multiselection mode, // because onTap is delayed when onDoubleTap is not null if (isDesktop && !_isShiftDown) return; _multiSelectionMode = true; } final cached = _currentTabCachedPeers.map((e) => e.id).toList(); int thisIndex = cached.indexOf(peer.id); int lastIndex = cached.indexOf(_lastId); if (_isShiftDown && thisIndex >= 0 && lastIndex >= 0) { int start = min(thisIndex, lastIndex); int end = max(thisIndex, lastIndex); bool remove = isPeerSelected(peer.id); for (var i = start; i <= end; i++) { if (remove) { if (isPeerSelected(cached[i])) { _selectedPeers.removeWhere((p) => p.id == cached[i]); } } else { if (!isPeerSelected(cached[i])) { _selectedPeers.add(_currentTabCachedPeers[i]); } } } } else { if (isPeerSelected(peer.id)) { _selectedPeers.removeWhere((p) => p.id == peer.id); } else { _selectedPeers.add(peer); } } _lastId = peer.id; notifyListeners(); } setCurrentTabCachedPeers(List peers) { Future.delayed(Duration.zero, () { _currentTabCachedPeers = peers; notifyListeners(); }); } selectAll() { _selectedPeers = _currentTabCachedPeers.toList(); notifyListeners(); } bool isPeerSelected(String id) { return selectedPeers.firstWhereOrNull((p) => p.id == id) != null; } setShiftDown(bool v) { if (_isShiftDown != v) { _isShiftDown = v; if (_multiSelectionMode) { notifyListeners(); } } } setTabVisible(int index, bool visible) { if (index >= 0 && index < maxTabCount) { if (_isVisible[index] != visible) { _isVisible[index] = visible; if (index == _currentTab && !visible) { _trySetCurrentTabToFirstVisibleEnabled(); } else if (visible && visibleEnabledOrderedIndexs.length == 1) { _currentTab = index; } try { bind.setLocalFlutterOption( k: kOptionPeerTabVisible, v: jsonEncode(_isVisible)); } catch (_) {} notifyListeners(); } } } _trySetCurrentTabToFirstVisibleEnabled() { if (!visibleEnabledOrderedIndexs.contains(_currentTab)) { if (visibleEnabledOrderedIndexs.isNotEmpty) { _currentTab = visibleEnabledOrderedIndexs.first; } } } reorder(int oldIndex, int newIndex) { if (oldIndex < newIndex) { newIndex -= 1; } if (oldIndex < 0 || oldIndex >= visibleEnabledOrderedIndexs.length) { return; } if (newIndex < 0 || newIndex >= visibleEnabledOrderedIndexs.length) { return; } final oldTabValue = visibleEnabledOrderedIndexs[oldIndex]; final newTabValue = visibleEnabledOrderedIndexs[newIndex]; int oldValueIndex = orders.indexOf(oldTabValue); int newValueIndex = orders.indexOf(newTabValue); final list = orders.toList(); if (oldIndex != -1 && newIndex != -1) { list.removeAt(oldValueIndex); list.insert(newValueIndex, oldTabValue); for (int i = 0; i < list.length; i++) { orders[i] = list[i]; } bind.setLocalFlutterOption(k: kOptionPeerTabOrder, v: jsonEncode(orders)); notifyListeners(); } } }