mirror of
https://github.com/FoggedLens/deflock-app.git
synced 2026-03-21 18:33:51 +00:00
181 lines
5.6 KiB
Dart
181 lines
5.6 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
|
|
|
import '../models/node_profile.dart';
|
|
import '../services/profile_service.dart';
|
|
|
|
class ProfileState extends ChangeNotifier {
|
|
static const String _enabledPrefsKey = 'enabled_profiles';
|
|
static const String _profileOrderPrefsKey = 'profile_order';
|
|
|
|
final List<NodeProfile> _profiles = [];
|
|
final Set<NodeProfile> _enabled = {};
|
|
List<String> _customOrder = []; // List of profile IDs in user's preferred order
|
|
|
|
// Test-only getters for accessing private state
|
|
@visibleForTesting
|
|
List<NodeProfile> get internalProfiles => _profiles;
|
|
@visibleForTesting
|
|
Set<NodeProfile> get internalEnabled => _enabled;
|
|
|
|
// Callback for when a profile is deleted (used to clear stale sessions)
|
|
void Function(NodeProfile)? _onProfileDeleted;
|
|
|
|
void setProfileDeletedCallback(void Function(NodeProfile) callback) {
|
|
_onProfileDeleted = callback;
|
|
}
|
|
|
|
// Getters
|
|
List<NodeProfile> get profiles => List.unmodifiable(_getOrderedProfiles());
|
|
bool isEnabled(NodeProfile p) => _enabled.contains(p);
|
|
List<NodeProfile> get enabledProfiles =>
|
|
_getOrderedProfiles().where(isEnabled).toList(growable: false);
|
|
|
|
// Initialize profiles from built-in and custom sources
|
|
Future<void> init({bool addDefaults = false}) async {
|
|
// Load custom profiles from storage
|
|
_profiles.addAll(await ProfileService().load());
|
|
|
|
// Add built-in profiles if this is first launch
|
|
if (addDefaults) {
|
|
_profiles.addAll(NodeProfile.getDefaults());
|
|
await ProfileService().save(_profiles);
|
|
}
|
|
|
|
// Load enabled profile IDs and custom order from prefs
|
|
final prefs = await SharedPreferences.getInstance();
|
|
final enabledIds = prefs.getStringList(_enabledPrefsKey);
|
|
if (enabledIds != null && enabledIds.isNotEmpty) {
|
|
// Restore enabled profiles by id
|
|
_enabled.addAll(_profiles.where((p) => enabledIds.contains(p.id)));
|
|
} else {
|
|
// By default, all are enabled
|
|
_enabled.addAll(_profiles);
|
|
}
|
|
|
|
// Load custom order
|
|
_customOrder = prefs.getStringList(_profileOrderPrefsKey) ?? [];
|
|
}
|
|
|
|
void toggleProfile(NodeProfile p, bool e) {
|
|
if (e) {
|
|
_enabled.add(p);
|
|
} else {
|
|
_enabled.remove(p);
|
|
// Safety: Always have at least one enabled profile
|
|
if (_enabled.isEmpty) {
|
|
final builtIn = _profiles.firstWhere((profile) => profile.builtin, orElse: () => _profiles.first);
|
|
_enabled.add(builtIn);
|
|
}
|
|
}
|
|
_saveEnabledProfiles();
|
|
notifyListeners();
|
|
}
|
|
|
|
void addOrUpdateProfile(NodeProfile p) {
|
|
final idx = _profiles.indexWhere((x) => x.id == p.id);
|
|
if (idx >= 0) {
|
|
_profiles[idx] = p;
|
|
} else {
|
|
_profiles.add(p);
|
|
_enabled.add(p);
|
|
_saveEnabledProfiles();
|
|
}
|
|
_saveProfilesToStorage();
|
|
notifyListeners();
|
|
}
|
|
|
|
void deleteProfile(NodeProfile p) {
|
|
if (!p.editable) return;
|
|
_enabled.remove(p);
|
|
_profiles.removeWhere((x) => x.id == p.id);
|
|
// Safety: Always have at least one enabled profile
|
|
if (_enabled.isEmpty) {
|
|
final builtIn = _profiles.firstWhere((profile) => profile.builtin, orElse: () => _profiles.first);
|
|
_enabled.add(builtIn);
|
|
}
|
|
_saveEnabledProfiles();
|
|
_saveProfilesToStorage();
|
|
|
|
// Notify about profile deletion so other parts can clean up
|
|
_onProfileDeleted?.call(p);
|
|
|
|
notifyListeners();
|
|
}
|
|
|
|
// Reorder profiles (for drag-and-drop in settings)
|
|
void reorderProfiles(int oldIndex, int newIndex) {
|
|
final orderedProfiles = _getOrderedProfiles();
|
|
|
|
// Standard Flutter reordering logic
|
|
if (oldIndex < newIndex) {
|
|
newIndex -= 1;
|
|
}
|
|
final item = orderedProfiles.removeAt(oldIndex);
|
|
orderedProfiles.insert(newIndex, item);
|
|
|
|
// Update custom order with new sequence
|
|
_customOrder = orderedProfiles.map((p) => p.id).toList();
|
|
_saveCustomOrder();
|
|
notifyListeners();
|
|
}
|
|
|
|
// Get profiles in custom order, with unordered profiles at the end
|
|
List<NodeProfile> _getOrderedProfiles() {
|
|
if (_customOrder.isEmpty) {
|
|
return List.from(_profiles);
|
|
}
|
|
|
|
final ordered = <NodeProfile>[];
|
|
final profilesById = {for (final p in _profiles) p.id: p};
|
|
|
|
// Add profiles in custom order
|
|
for (final id in _customOrder) {
|
|
final profile = profilesById[id];
|
|
if (profile != null) {
|
|
ordered.add(profile);
|
|
profilesById.remove(id);
|
|
}
|
|
}
|
|
|
|
// Add any remaining profiles that weren't in the custom order
|
|
ordered.addAll(profilesById.values);
|
|
|
|
return ordered;
|
|
}
|
|
|
|
// Save enabled profile IDs to disk
|
|
Future<void> _saveEnabledProfiles() async {
|
|
try {
|
|
final prefs = await SharedPreferences.getInstance();
|
|
await prefs.setStringList(
|
|
_enabledPrefsKey,
|
|
_enabled.map((p) => p.id).toList(),
|
|
);
|
|
} catch (e) {
|
|
// Fail gracefully in tests or if SharedPreferences isn't available
|
|
debugPrint('[ProfileState] Failed to save enabled profiles: $e');
|
|
}
|
|
}
|
|
|
|
// Save profiles to storage
|
|
Future<void> _saveProfilesToStorage() async {
|
|
try {
|
|
await ProfileService().save(_profiles);
|
|
} catch (e) {
|
|
// Fail gracefully in tests or if storage isn't available
|
|
debugPrint('[ProfileState] Failed to save profiles: $e');
|
|
}
|
|
}
|
|
|
|
// Save custom order to disk
|
|
Future<void> _saveCustomOrder() async {
|
|
try {
|
|
final prefs = await SharedPreferences.getInstance();
|
|
await prefs.setStringList(_profileOrderPrefsKey, _customOrder);
|
|
} catch (e) {
|
|
// Fail gracefully in tests or if SharedPreferences isn't available
|
|
debugPrint('[ProfileState] Failed to save custom order: $e');
|
|
}
|
|
}
|
|
} |