mirror of
https://github.com/hiddify/hiddify-next.git
synced 2026-06-05 21:05:07 +08:00
Temporary hotfix to resolve fatal 'setState() called during build' errors caused by lazy evaluation of dirty provider dependency chains during high-frequency status transitions. - Establishes eager evaluation by listening to `activeProxyNotifierProvider` at the root container level during bootstrap to resolve dirty state flushes in microtasks. - Refactors `serviceRunningProvider` and all downstream dependent stream/future notifiers to execute synchronously to prevent build-phase timing hops. - Defers secondary state updates using `Future.microtask()`. Note: This is a temporary architectural patch to bypass Riverpod's lazy widget evaluation mechanics under high-frequency stream updates. A full core communication redesign is recommended for future maintenance to address this natively.
174 lines
6.9 KiB
Dart
174 lines
6.9 KiB
Dart
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:dartx/dartx_io.dart';
|
|
import 'package:file_picker/file_picker.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:hiddify/core/localization/translations.dart';
|
|
import 'package:hiddify/core/notification/in_app_notification_controller.dart';
|
|
import 'package:hiddify/features/connection/data/connection_data_providers.dart';
|
|
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
|
import 'package:hiddify/features/profile/notifier/active_profile_notifier.dart';
|
|
import 'package:hiddify/features/settings/data/config_option_repository.dart';
|
|
import 'package:hiddify/utils/custom_loggers.dart';
|
|
import 'package:hiddify/utils/platform_utils.dart';
|
|
import 'package:json_path/json_path.dart';
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
|
|
part 'config_option_notifier.g.dart';
|
|
|
|
@Riverpod(keepAlive: true)
|
|
class ConfigOptionNotifier extends _$ConfigOptionNotifier with AppLogger {
|
|
@override
|
|
Future<bool> build() async {
|
|
final serviceRunning = ref.watch(serviceRunningProvider);
|
|
final serviceSingboxOptions = ref.read(connectionRepositoryProvider).configOptionsSnapshot;
|
|
|
|
ref.listen(ConfigOptions.singboxConfigOptions, (previous, next) async {
|
|
if (!serviceRunning || previous == null) return;
|
|
if (next != previous && next != serviceSingboxOptions) {
|
|
if (_lastUpdate == null || DateTime.now().difference(_lastUpdate!) > const Duration(milliseconds: 100)) {
|
|
_lastUpdate = DateTime.now();
|
|
if (serviceSingboxOptions?.enableTun != next.enableTun) {
|
|
loggy.debug("tun option changed, reconnecting");
|
|
await ref.read(connectionNotifierProvider.notifier).toggleConnection();
|
|
await ref.read(connectionNotifierProvider.notifier).toggleConnection();
|
|
} else {
|
|
final activeProfile = await ref.read(activeProfileProvider.future);
|
|
return await ref.read(connectionNotifierProvider.notifier).reconnect(activeProfile);
|
|
}
|
|
state = const AsyncData(false);
|
|
}
|
|
}
|
|
}, fireImmediately: true);
|
|
return false;
|
|
}
|
|
|
|
DateTime? _lastUpdate;
|
|
|
|
Future<String?> _exportJson(bool excludePrivate) async {
|
|
try {
|
|
final options = ref.read(ConfigOptions.singboxConfigOptions);
|
|
Map map = options.toJson();
|
|
if (excludePrivate) {
|
|
for (final key in ConfigOptions.privatePreferencesKeys) {
|
|
final query = key.split('.').map((e) => '["$e"]').join();
|
|
final res = JsonPath('\$$query').read(map).firstOrNull;
|
|
if (res != null) {
|
|
map = res.pointer.remove(map)! as Map;
|
|
}
|
|
}
|
|
}
|
|
const encoder = JsonEncoder.withIndent(' ');
|
|
return encoder.convert(map);
|
|
} catch (e, st) {
|
|
loggy.warning("error creating config options json", e, st);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
Future<bool> exportJsonClipboard({bool excludePrivate = true}) async {
|
|
final t = ref.read(translationsProvider).requireValue;
|
|
try {
|
|
final json = await _exportJson(excludePrivate);
|
|
if (json == null) return false;
|
|
await Clipboard.setData(ClipboardData(text: json));
|
|
ref.read(inAppNotificationControllerProvider).showSuccessToast(t.common.msg.export.clipboard.success);
|
|
return true;
|
|
} on PlatformException {
|
|
ref
|
|
.read(inAppNotificationControllerProvider)
|
|
.showInfoToast(t.common.msg.export.clipboard.contentTooLarge, duration: const Duration(seconds: 5));
|
|
return false;
|
|
} catch (e, st) {
|
|
loggy.warning("error exporting config options to clipboard", e, st);
|
|
ref.read(inAppNotificationControllerProvider).showErrorToast(t.common.msg.export.clipboard.failure);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<bool> exportJsonFile({bool excludePrivate = true}) async {
|
|
final t = ref.read(translationsProvider).requireValue;
|
|
try {
|
|
final json = await _exportJson(excludePrivate);
|
|
if (json == null) return false;
|
|
final bytes = utf8.encode(json);
|
|
final outputFile = await FilePicker.platform.saveFile(
|
|
fileName: 'options.json',
|
|
type: FileType.custom,
|
|
allowedExtensions: ['json'],
|
|
bytes: bytes,
|
|
);
|
|
if (outputFile == null) return false;
|
|
if (PlatformUtils.isDesktop) {
|
|
final file = File(outputFile);
|
|
if (file.extension != '.json') return false;
|
|
if (!await file.exists()) await file.parent.create(recursive: true);
|
|
await file.writeAsBytes(bytes);
|
|
}
|
|
ref.read(inAppNotificationControllerProvider).showSuccessToast(t.common.msg.export.file.success);
|
|
return true;
|
|
} catch (e, st) {
|
|
loggy.warning("error exporting config options to json file", e, st);
|
|
ref.read(inAppNotificationControllerProvider).showErrorToast(t.common.msg.export.file.failure);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<void> _importJson(String input) async {
|
|
if (jsonDecode(input) case final Map<String, dynamic> map) {
|
|
for (final option in ConfigOptions.preferences.entries) {
|
|
final query = option.key.split('.').map((e) => '["$e"]').join();
|
|
final res = JsonPath('\$$query').read(map).firstOrNull;
|
|
if (res?.value case final value?) {
|
|
try {
|
|
await ref.read(option.value.notifier).updateRaw(value);
|
|
} catch (e) {
|
|
loggy.debug("error updating [${option.key}]: $e", e);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Future<bool> importFromClipboard() async {
|
|
final t = ref.read(translationsProvider).requireValue;
|
|
try {
|
|
final input = await Clipboard.getData(Clipboard.kTextPlain).then((value) => value?.text);
|
|
if (input == null) return false;
|
|
await _importJson(input);
|
|
ref.read(inAppNotificationControllerProvider).showSuccessToast(t.common.msg.import.success);
|
|
return true;
|
|
} catch (e, st) {
|
|
loggy.warning("error importing config options from clipboard", e, st);
|
|
ref.read(inAppNotificationControllerProvider).showErrorToast(t.common.msg.import.failure);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<bool> importFromJsonFile() async {
|
|
final t = ref.read(translationsProvider).requireValue;
|
|
try {
|
|
final result = await FilePicker.platform.pickFiles(type: FileType.custom, allowedExtensions: ['json']);
|
|
if (result == null) return false;
|
|
final file = File(result.files.single.path!);
|
|
if (!await file.exists()) return false;
|
|
final bytes = await file.readAsBytes();
|
|
await _importJson(utf8.decode(bytes));
|
|
ref.read(inAppNotificationControllerProvider).showSuccessToast(t.common.msg.import.success);
|
|
return true;
|
|
} catch (e, st) {
|
|
loggy.warning("error importing config options from json file", e, st);
|
|
ref.read(inAppNotificationControllerProvider).showErrorToast(t.common.msg.import.failure);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
Future<void> resetOption() async {
|
|
for (final option in ConfigOptions.preferences.values) {
|
|
await ref.read(option.notifier).reset();
|
|
}
|
|
ref.invalidateSelf();
|
|
}
|
|
}
|