- Changed window size to mobile phone format (400x800) - Removed width condition for ActiveProxyFooter - now always visible - Added run-umbrix.sh launch script with icon copying - Stats cards now display on all screen sizes
160 lines
5.2 KiB
Dart
160 lines
5.2 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:dartx/dartx.dart';
|
|
|
|
import 'package:umbrix/core/haptic/haptic_service.dart';
|
|
import 'package:umbrix/core/localization/translations.dart';
|
|
import 'package:umbrix/core/preferences/preferences_provider.dart';
|
|
import 'package:umbrix/core/utils/preferences_utils.dart';
|
|
import 'package:umbrix/features/connection/notifier/connection_notifier.dart';
|
|
import 'package:umbrix/features/proxy/data/proxy_data_providers.dart';
|
|
import 'package:umbrix/features/proxy/model/proxy_entity.dart';
|
|
import 'package:umbrix/features/proxy/model/proxy_failure.dart';
|
|
import 'package:umbrix/utils/riverpod_utils.dart';
|
|
import 'package:umbrix/utils/utils.dart';
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
import 'package:rxdart/rxdart.dart';
|
|
|
|
part 'proxies_overview_notifier.g.dart';
|
|
|
|
enum ProxiesSort {
|
|
unsorted,
|
|
name,
|
|
delay;
|
|
|
|
String present(TranslationsEn t) => switch (this) {
|
|
ProxiesSort.unsorted => t.proxies.sortOptions.unsorted,
|
|
ProxiesSort.name => t.proxies.sortOptions.name,
|
|
ProxiesSort.delay => t.proxies.sortOptions.delay,
|
|
};
|
|
}
|
|
|
|
@Riverpod(keepAlive: true)
|
|
class ProxiesSortNotifier extends _$ProxiesSortNotifier with AppLogger {
|
|
late final _pref = PreferencesEntry(
|
|
preferences: ref.watch(sharedPreferencesProvider).requireValue,
|
|
key: "proxies_sort_mode",
|
|
defaultValue: ProxiesSort.delay,
|
|
mapFrom: ProxiesSort.values.byName,
|
|
mapTo: (value) => value.name,
|
|
);
|
|
|
|
@override
|
|
ProxiesSort build() {
|
|
final sortBy = _pref.read();
|
|
loggy.info("sort proxies by: [${sortBy.name}]");
|
|
return sortBy;
|
|
}
|
|
|
|
Future<void> update(ProxiesSort value) {
|
|
state = value;
|
|
return _pref.write(value);
|
|
}
|
|
}
|
|
|
|
@riverpod
|
|
class ProxiesOverviewNotifier extends _$ProxiesOverviewNotifier with AppLogger {
|
|
@override
|
|
Stream<List<ProxyGroupEntity>> build() async* {
|
|
ref.disposeDelay(const Duration(seconds: 15));
|
|
final serviceRunning = await ref.watch(serviceRunningProvider.future);
|
|
if (!serviceRunning) {
|
|
throw const ServiceNotRunning();
|
|
}
|
|
final sortBy = ref.watch(proxiesSortNotifierProvider);
|
|
yield* ref
|
|
.watch(proxyRepositoryProvider)
|
|
.watchProxies()
|
|
.throttleTime(
|
|
const Duration(milliseconds: 100),
|
|
leading: false,
|
|
trailing: true,
|
|
)
|
|
.map(
|
|
(event) => event.getOrElse(
|
|
(err) {
|
|
loggy.warning("error receiving proxies", err);
|
|
throw err;
|
|
},
|
|
),
|
|
)
|
|
.asyncMap((proxies) async => _sortOutbounds(proxies, sortBy));
|
|
}
|
|
|
|
Future<List<ProxyGroupEntity>> _sortOutbounds(
|
|
List<ProxyGroupEntity> proxies,
|
|
ProxiesSort sortBy,
|
|
) async {
|
|
final groupWithSelected = {
|
|
for (final o in proxies) o.tag: o.selected,
|
|
};
|
|
final sortedProxies = <ProxyGroupEntity>[];
|
|
for (final group in proxies) {
|
|
final sortedItems = switch (sortBy) {
|
|
ProxiesSort.name => group.items.sortedWith((a, b) {
|
|
if (a.type.isGroup && !b.type.isGroup) return -1;
|
|
if (!a.type.isGroup && b.type.isGroup) return 1;
|
|
return a.tag.compareTo(b.tag);
|
|
}),
|
|
ProxiesSort.delay => group.items.sortedWith((a, b) {
|
|
if (a.type.isGroup && !b.type.isGroup) return -1;
|
|
if (!a.type.isGroup && b.type.isGroup) return 1;
|
|
|
|
final ai = a.urlTestDelay;
|
|
final bi = b.urlTestDelay;
|
|
if (ai == 0 && bi == 0) return -1;
|
|
if (ai == 0 && bi > 0) return 1;
|
|
if (ai > 0 && bi == 0) return -1;
|
|
return ai.compareTo(bi);
|
|
}),
|
|
ProxiesSort.unsorted => group.items,
|
|
};
|
|
final items = <ProxyItemEntity>[];
|
|
for (final item in sortedItems) {
|
|
if (groupWithSelected.keys.contains(item.tag)) {
|
|
items.add(item.copyWith(selectedTag: groupWithSelected[item.tag]));
|
|
} else {
|
|
items.add(item);
|
|
}
|
|
}
|
|
sortedProxies.add(group.copyWith(items: items));
|
|
}
|
|
return sortedProxies;
|
|
}
|
|
|
|
Future<void> changeProxy(String groupTag, String outboundTag) async {
|
|
loggy.debug(
|
|
"changing proxy, group: [$groupTag] - outbound: [$outboundTag]",
|
|
);
|
|
if (state case AsyncData(value: final outbounds)) {
|
|
await ref.read(hapticServiceProvider.notifier).lightImpact();
|
|
await ref.read(proxyRepositoryProvider).selectProxy(groupTag, outboundTag).getOrElse((err) {
|
|
loggy.warning("error selecting outbound", err);
|
|
throw err;
|
|
}).run();
|
|
state = AsyncData(
|
|
[
|
|
...outbounds.map(
|
|
(e) => e.tag == groupTag ? e.copyWith(selected: outboundTag) : e,
|
|
),
|
|
],
|
|
).copyWithPrevious(state);
|
|
|
|
// Автоматически запускаем URL test после переключения
|
|
// чтобы обновить задержки для всех прокси
|
|
await urlTest(groupTag);
|
|
}
|
|
}
|
|
|
|
Future<void> urlTest(String groupTag) async {
|
|
loggy.debug("testing group: [$groupTag]");
|
|
if (state case AsyncData()) {
|
|
await ref.read(hapticServiceProvider.notifier).lightImpact();
|
|
await ref.read(proxyRepositoryProvider).urlTest(groupTag).getOrElse((err) {
|
|
loggy.error("error testing group", err);
|
|
throw err;
|
|
}).run();
|
|
}
|
|
}
|
|
}
|