2023-11-10 14:53:12 +03:30
|
|
|
|
import 'package:flutter/foundation.dart';
|
2026-01-17 13:09:20 +03:00
|
|
|
|
import 'package:umbrix/core/app_info/app_info_provider.dart';
|
|
|
|
|
|
import 'package:umbrix/core/model/environment.dart';
|
|
|
|
|
|
import 'package:umbrix/core/preferences/actions_at_closing.dart';
|
|
|
|
|
|
// import 'package:umbrix/core/model/region.dart';
|
|
|
|
|
|
import 'package:umbrix/core/preferences/preferences_provider.dart';
|
|
|
|
|
|
import 'package:umbrix/core/utils/preferences_utils.dart';
|
|
|
|
|
|
import 'package:umbrix/features/connection/model/connection_status.dart';
|
|
|
|
|
|
import 'package:umbrix/features/connection/notifier/connection_notifier.dart';
|
|
|
|
|
|
import 'package:umbrix/features/per_app_proxy/model/per_app_proxy_mode.dart';
|
|
|
|
|
|
import 'package:umbrix/features/profile/notifier/active_profile_notifier.dart';
|
|
|
|
|
|
import 'package:umbrix/utils/platform_utils.dart';
|
2023-09-07 13:43:46 +03:30
|
|
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
2023-07-15 18:00:44 +03:30
|
|
|
|
|
2023-12-01 12:56:24 +03:30
|
|
|
|
part 'general_preferences.g.dart';
|
|
|
|
|
|
|
2023-11-10 14:53:12 +03:30
|
|
|
|
bool _debugIntroPage = false;
|
|
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
abstract class Preferences {
|
|
|
|
|
|
static final introCompleted = PreferencesNotifier.create(
|
2023-09-17 00:23:31 +03:30
|
|
|
|
"intro_completed",
|
|
|
|
|
|
false,
|
2024-03-02 22:53:14 +03:30
|
|
|
|
overrideValue: _debugIntroPage && kDebugMode ? false : null,
|
2023-09-17 00:23:31 +03:30
|
|
|
|
);
|
|
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
static final silentStart = PreferencesNotifier.create<bool, bool>(
|
2023-12-01 12:56:24 +03:30
|
|
|
|
"silent_start",
|
|
|
|
|
|
false,
|
|
|
|
|
|
);
|
2023-09-07 13:43:46 +03:30
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
static final disableMemoryLimit = PreferencesNotifier.create<bool, bool>(
|
2023-10-26 15:16:25 +03:30
|
|
|
|
"disable_memory_limit",
|
2023-12-14 19:58:15 +03:30
|
|
|
|
// disable memory limit on desktop by default
|
|
|
|
|
|
PlatformUtils.isDesktop,
|
2023-09-16 15:16:20 +03:30
|
|
|
|
);
|
|
|
|
|
|
|
2024-07-04 21:04:44 +02:00
|
|
|
|
static final perAppProxyMode = PreferencesNotifier.create<PerAppProxyMode, String>(
|
2023-09-13 23:19:16 +03:30
|
|
|
|
"per_app_proxy_mode",
|
2026-01-15 12:28:40 +03:00
|
|
|
|
PerAppProxyMode.exclude,
|
2023-09-13 23:19:16 +03:30
|
|
|
|
mapFrom: PerAppProxyMode.values.byName,
|
|
|
|
|
|
mapTo: (value) => value.name,
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
static final markNewProfileActive = PreferencesNotifier.create<bool, bool>(
|
|
|
|
|
|
"mark_new_profile_active",
|
|
|
|
|
|
true,
|
2023-09-13 23:19:16 +03:30
|
|
|
|
);
|
|
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
static final dynamicNotification = PreferencesNotifier.create<bool, bool>(
|
|
|
|
|
|
"dynamic_notification",
|
|
|
|
|
|
true,
|
2023-09-13 23:19:16 +03:30
|
|
|
|
);
|
|
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
static final autoCheckIp = PreferencesNotifier.create<bool, bool>(
|
|
|
|
|
|
"auto_check_ip",
|
2023-09-10 14:16:44 +03:30
|
|
|
|
true,
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
static final startedByUser = PreferencesNotifier.create<bool, bool>(
|
|
|
|
|
|
"started_by_user",
|
|
|
|
|
|
false,
|
|
|
|
|
|
);
|
2024-03-17 14:45:15 +01:00
|
|
|
|
|
|
|
|
|
|
static final storeReviewedByUser = PreferencesNotifier.create<bool, bool>(
|
|
|
|
|
|
"store_reviewed_by_user",
|
|
|
|
|
|
false,
|
|
|
|
|
|
);
|
2024-10-03 19:32:23 +10:00
|
|
|
|
|
|
|
|
|
|
static final actionAtClose = PreferencesNotifier.create<ActionsAtClosing, String>(
|
|
|
|
|
|
"action_at_close",
|
|
|
|
|
|
ActionsAtClosing.ask,
|
|
|
|
|
|
mapFrom: ActionsAtClosing.values.byName,
|
|
|
|
|
|
mapTo: (value) => value.name,
|
|
|
|
|
|
);
|
2023-09-10 14:16:44 +03:30
|
|
|
|
}
|
2023-12-14 14:50:10 +03:30
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
@Riverpod(keepAlive: true)
|
|
|
|
|
|
class DebugModeNotifier extends _$DebugModeNotifier {
|
|
|
|
|
|
late final _pref = PreferencesEntry(
|
|
|
|
|
|
preferences: ref.watch(sharedPreferencesProvider).requireValue,
|
|
|
|
|
|
key: "debug_mode",
|
|
|
|
|
|
defaultValue: ref.read(environmentProvider) == Environment.dev,
|
2023-12-14 14:50:10 +03:30
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
2024-03-02 22:53:14 +03:30
|
|
|
|
bool build() => _pref.read();
|
2023-12-14 14:50:10 +03:30
|
|
|
|
|
|
|
|
|
|
Future<void> update(bool value) {
|
|
|
|
|
|
state = value;
|
2024-03-02 22:53:14 +03:30
|
|
|
|
return _pref.write(value);
|
2023-12-14 14:50:10 +03:30
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-02-17 13:11:50 +03:30
|
|
|
|
|
2024-03-02 22:53:14 +03:30
|
|
|
|
@Riverpod(keepAlive: true)
|
|
|
|
|
|
class PerAppProxyList extends _$PerAppProxyList {
|
|
|
|
|
|
late final _include = PreferencesEntry(
|
|
|
|
|
|
preferences: ref.watch(sharedPreferencesProvider).requireValue,
|
|
|
|
|
|
key: "per_app_proxy_include_list",
|
|
|
|
|
|
defaultValue: <String>[],
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
late final _exclude = PreferencesEntry(
|
|
|
|
|
|
preferences: ref.watch(sharedPreferencesProvider).requireValue,
|
|
|
|
|
|
key: "per_app_proxy_exclude_list",
|
|
|
|
|
|
defaultValue: <String>[],
|
2024-02-17 13:11:50 +03:30
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
2026-01-15 12:28:40 +03:00
|
|
|
|
List<String> build() {
|
|
|
|
|
|
// Слушаем изменения режима и перестраиваем список
|
|
|
|
|
|
final mode = ref.watch(Preferences.perAppProxyMode);
|
|
|
|
|
|
return mode == PerAppProxyMode.include ? _include.read() : _exclude.read();
|
|
|
|
|
|
}
|
2024-02-17 13:11:50 +03:30
|
|
|
|
|
2026-01-15 12:28:40 +03:00
|
|
|
|
Future<void> update(List<String> value) async {
|
|
|
|
|
|
print('[PerAppProxyList] update() вызван с ${value.length} приложениями');
|
|
|
|
|
|
final mode = ref.read(Preferences.perAppProxyMode);
|
|
|
|
|
|
print('[PerAppProxyList] Текущий режим: $mode');
|
|
|
|
|
|
|
|
|
|
|
|
// Сначала сохраняем в SharedPreferences
|
|
|
|
|
|
if (mode == PerAppProxyMode.include) {
|
|
|
|
|
|
await _include.write(value);
|
|
|
|
|
|
print('[PerAppProxyList] Записан include список: $value');
|
|
|
|
|
|
} else {
|
|
|
|
|
|
await _exclude.write(value);
|
|
|
|
|
|
print('[PerAppProxyList] Записан exclude список: $value');
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Затем обновляем локальное состояние
|
2024-02-17 13:11:50 +03:30
|
|
|
|
state = value;
|
2026-01-15 12:28:40 +03:00
|
|
|
|
print('[PerAppProxyList] State обновлён');
|
|
|
|
|
|
|
|
|
|
|
|
// Автоматически перезапускаем VPN если он активен
|
|
|
|
|
|
print('[PerAppProxyList] Вызываю _reconnectVpnIfActive()');
|
|
|
|
|
|
await _reconnectVpnIfActive();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> _reconnectVpnIfActive() async {
|
|
|
|
|
|
try {
|
|
|
|
|
|
final connectionNotifier = await ref.read(connectionNotifierProvider.future);
|
|
|
|
|
|
if (connectionNotifier is Connected) {
|
|
|
|
|
|
final profile = await ref.read(activeProfileProvider.future);
|
|
|
|
|
|
await ref.read(connectionNotifierProvider.notifier).reconnect(profile);
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (_) {
|
|
|
|
|
|
// Игнорируем ошибки если connection provider не инициализирован
|
2024-03-02 22:53:14 +03:30
|
|
|
|
}
|
2024-02-17 13:11:50 +03:30
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-12-26 02:39:35 +03:00
|
|
|
|
|
|
|
|
|
|
@Riverpod(keepAlive: true)
|
|
|
|
|
|
class ExcludedDomainsList extends _$ExcludedDomainsList {
|
|
|
|
|
|
late final _pref = PreferencesEntry(
|
|
|
|
|
|
preferences: ref.watch(sharedPreferencesProvider).requireValue,
|
|
|
|
|
|
key: "excluded_domains_list",
|
|
|
|
|
|
defaultValue: <String>[],
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
List<String> build() => _pref.read();
|
|
|
|
|
|
|
2026-01-15 12:28:40 +03:00
|
|
|
|
Future<void> update(List<String> value) async {
|
2025-12-26 02:39:35 +03:00
|
|
|
|
state = value;
|
2026-01-15 12:28:40 +03:00
|
|
|
|
await _pref.write(value);
|
|
|
|
|
|
// Автоматически перезапускаем VPN если он активен
|
|
|
|
|
|
await _reconnectVpnIfActive();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Future<void> _reconnectVpnIfActive() async {
|
|
|
|
|
|
try {
|
|
|
|
|
|
final connectionNotifier = await ref.read(connectionNotifierProvider.future);
|
|
|
|
|
|
if (connectionNotifier is Connected) {
|
|
|
|
|
|
final profile = await ref.read(activeProfileProvider.future);
|
|
|
|
|
|
await ref.read(connectionNotifierProvider.notifier).reconnect(profile);
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (_) {
|
|
|
|
|
|
// Игнорируем ошибки если connection provider не инициализирован
|
|
|
|
|
|
}
|
2025-12-26 02:39:35 +03:00
|
|
|
|
}
|
|
|
|
|
|
}
|