- 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
206 lines
7.4 KiB
Dart
206 lines
7.4 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:dio/dio.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:fpdart/fpdart.dart';
|
|
import 'package:umbrix/core/haptic/haptic_service.dart';
|
|
import 'package:umbrix/core/localization/translations.dart';
|
|
import 'package:umbrix/core/model/failures.dart';
|
|
import 'package:umbrix/core/notification/in_app_notification_controller.dart';
|
|
import 'package:umbrix/core/preferences/general_preferences.dart';
|
|
import 'package:umbrix/core/preferences/preferences_provider.dart';
|
|
import 'package:umbrix/features/common/adaptive_root_scaffold.dart';
|
|
import 'package:umbrix/features/config_option/notifier/warp_option_notifier.dart';
|
|
import 'package:umbrix/features/config_option/overview/warp_options_widgets.dart';
|
|
import 'package:umbrix/features/connection/notifier/connection_notifier.dart';
|
|
import 'package:umbrix/features/profile/data/profile_data_providers.dart';
|
|
import 'package:umbrix/features/profile/data/profile_repository.dart';
|
|
import 'package:umbrix/features/profile/model/profile_entity.dart';
|
|
import 'package:umbrix/features/profile/model/profile_failure.dart';
|
|
import 'package:umbrix/features/profile/notifier/active_profile_notifier.dart';
|
|
import 'package:umbrix/utils/riverpod_utils.dart';
|
|
import 'package:umbrix/utils/utils.dart';
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
|
|
part 'profile_notifier.g.dart';
|
|
|
|
@riverpod
|
|
class AddProfile extends _$AddProfile with AppLogger {
|
|
@override
|
|
AsyncValue<Unit?> build() {
|
|
ref.disposeDelay(const Duration(minutes: 1));
|
|
ref.onDispose(() {
|
|
loggy.debug("disposing");
|
|
_cancelToken?.cancel();
|
|
});
|
|
ref.listenSelf(
|
|
(previous, next) {
|
|
final t = ref.read(translationsProvider);
|
|
final notification = ref.read(inAppNotificationControllerProvider);
|
|
switch (next) {
|
|
case AsyncData(value: final _?):
|
|
notification.showSuccessToast(t.profile.save.successMsg);
|
|
case AsyncError(:final error):
|
|
if (error case ProfileInvalidUrlFailure()) {
|
|
notification.showErrorToast(t.failure.profiles.invalidUrl);
|
|
} else {
|
|
notification.showErrorDialog(
|
|
t.presentError(error, action: t.profile.add.failureMsg),
|
|
);
|
|
}
|
|
}
|
|
},
|
|
);
|
|
return const AsyncData(null);
|
|
}
|
|
|
|
ProfileRepository get _profilesRepo => ref.read(profileRepositoryProvider).requireValue;
|
|
CancelToken? _cancelToken;
|
|
|
|
Future<void> add(String rawInput) async {
|
|
if (state.isLoading) return;
|
|
state = const AsyncLoading();
|
|
// await check4Warp(rawInput);
|
|
state = await AsyncValue.guard(
|
|
() async {
|
|
final activeProfile = await ref.read(activeProfileProvider.future);
|
|
final markAsActive = activeProfile == null || ref.read(Preferences.markNewProfileActive);
|
|
final TaskEither<ProfileFailure, Unit> task;
|
|
if (LinkParser.parse(rawInput) case (final link)?) {
|
|
loggy.debug("adding profile, url: [${link.url}]");
|
|
task = _profilesRepo.addByUrl(
|
|
link.url,
|
|
markAsActive: markAsActive,
|
|
cancelToken: _cancelToken = CancelToken(),
|
|
);
|
|
} else if (LinkParser.protocol(rawInput) case (final parsed)?) {
|
|
loggy.debug("adding profile, content");
|
|
var name = parsed.name;
|
|
final oldItem = await _profilesRepo.getByName(name);
|
|
if (name == "Hiddify WARP" && oldItem != null) {
|
|
_profilesRepo.deleteById(oldItem.id).run();
|
|
}
|
|
while (await _profilesRepo.getByName(name) != null) {
|
|
name += '${randomInt(0, 9).run()}';
|
|
}
|
|
task = _profilesRepo.addByContent(
|
|
parsed.content,
|
|
name: name,
|
|
markAsActive: markAsActive,
|
|
);
|
|
} else {
|
|
loggy.debug("invalid content");
|
|
throw const ProfileInvalidUrlFailure();
|
|
}
|
|
return task.match(
|
|
(err) {
|
|
loggy.warning("failed to add profile", err);
|
|
throw err;
|
|
},
|
|
(_) {
|
|
loggy.info(
|
|
"successfully added profile, mark as active? [$markAsActive]",
|
|
);
|
|
return unit;
|
|
},
|
|
).run();
|
|
},
|
|
);
|
|
}
|
|
|
|
Future<void> check4Warp(String rawInput) async {
|
|
for (final line in rawInput.split("\n")) {
|
|
if (line.toLowerCase().startsWith("warp://")) {
|
|
final prefs = ref.read(sharedPreferencesProvider).requireValue;
|
|
final warp = ref.read(warpOptionNotifierProvider.notifier);
|
|
|
|
final consent = false && (prefs.getBool(WarpOptionNotifier.warpConsentGiven) ?? false);
|
|
|
|
final t = ref.read(translationsProvider);
|
|
final notification = ref.read(inAppNotificationControllerProvider);
|
|
|
|
if (!consent) {
|
|
final agreed = await showDialog<bool>(
|
|
context: RootScaffold.stateKey.currentContext!,
|
|
builder: (context) => const WarpLicenseAgreementModal(),
|
|
);
|
|
|
|
if (agreed ?? false) {
|
|
await prefs.setBool(WarpOptionNotifier.warpConsentGiven, true);
|
|
final toast = notification.showInfoToast(t.profile.add.addingWarpMsg, duration: const Duration(milliseconds: 100));
|
|
toast?.pause();
|
|
await warp.generateWarpConfig();
|
|
toast?.start();
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
|
|
final accountId = prefs.getString("warp2-account-id");
|
|
final accessToken = prefs.getString("warp2-access-token");
|
|
final hasWarp2Config = accountId != null && accessToken != null;
|
|
|
|
if (!hasWarp2Config || true) {
|
|
final toast = notification.showInfoToast(t.profile.add.addingWarpMsg, duration: const Duration(milliseconds: 100));
|
|
toast?.pause();
|
|
await warp.generateWarp2Config();
|
|
toast?.start();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@riverpod
|
|
class UpdateProfile extends _$UpdateProfile with AppLogger {
|
|
@override
|
|
AsyncValue<Unit?> build(String id) {
|
|
ref.disposeDelay(const Duration(minutes: 1));
|
|
ref.listenSelf(
|
|
(previous, next) {
|
|
final t = ref.read(translationsProvider);
|
|
final notification = ref.read(inAppNotificationControllerProvider);
|
|
switch (next) {
|
|
case AsyncData(value: final _?):
|
|
notification.showSuccessToast(t.profile.update.successMsg);
|
|
case AsyncError(:final error):
|
|
notification.showErrorDialog(
|
|
t.presentError(error, action: t.profile.update.failureMsg),
|
|
);
|
|
}
|
|
},
|
|
);
|
|
return const AsyncData(null);
|
|
}
|
|
|
|
ProfileRepository get _profilesRepo => ref.read(profileRepositoryProvider).requireValue;
|
|
|
|
Future<void> updateProfile(RemoteProfileEntity profile) async {
|
|
if (state.isLoading) return;
|
|
state = const AsyncLoading();
|
|
await ref.read(hapticServiceProvider.notifier).lightImpact();
|
|
state = await AsyncValue.guard(
|
|
() async {
|
|
return await _profilesRepo.updateSubscription(profile).match(
|
|
(err) {
|
|
loggy.warning("failed to update profile", err);
|
|
throw err;
|
|
},
|
|
(_) async {
|
|
loggy.info(
|
|
'successfully updated profile, was active? [${profile.active}]',
|
|
);
|
|
|
|
await ref.read(activeProfileProvider.future).then((active) async {
|
|
if (active != null && active.id == profile.id) {
|
|
await ref.read(connectionNotifierProvider.notifier).reconnect(profile);
|
|
}
|
|
});
|
|
return unit;
|
|
},
|
|
).run();
|
|
},
|
|
);
|
|
}
|
|
}
|