Refactor desktop window management and tray
This commit is contained in:
@@ -14,20 +14,19 @@ import 'package:hiddify/core/preferences/general_preferences.dart';
|
|||||||
import 'package:hiddify/core/preferences/preferences_migration.dart';
|
import 'package:hiddify/core/preferences/preferences_migration.dart';
|
||||||
import 'package:hiddify/core/preferences/preferences_provider.dart';
|
import 'package:hiddify/core/preferences/preferences_provider.dart';
|
||||||
import 'package:hiddify/features/app/widget/app.dart';
|
import 'package:hiddify/features/app/widget/app.dart';
|
||||||
import 'package:hiddify/features/common/window/window_controller.dart';
|
import 'package:hiddify/features/auto_start/notifier/auto_start_notifier.dart';
|
||||||
import 'package:hiddify/features/geo_asset/data/geo_asset_data_providers.dart';
|
import 'package:hiddify/features/geo_asset/data/geo_asset_data_providers.dart';
|
||||||
import 'package:hiddify/features/log/data/log_data_providers.dart';
|
import 'package:hiddify/features/log/data/log_data_providers.dart';
|
||||||
import 'package:hiddify/features/profile/data/profile_data_providers.dart';
|
import 'package:hiddify/features/profile/data/profile_data_providers.dart';
|
||||||
import 'package:hiddify/features/profile/notifier/active_profile_notifier.dart';
|
import 'package:hiddify/features/profile/notifier/active_profile_notifier.dart';
|
||||||
import 'package:hiddify/features/system_tray/system_tray_controller.dart';
|
import 'package:hiddify/features/system_tray/notifier/system_tray_notifier.dart';
|
||||||
import 'package:hiddify/services/auto_start_service.dart';
|
import 'package:hiddify/features/window/notifier/window_notifier.dart';
|
||||||
import 'package:hiddify/services/deep_link_service.dart';
|
import 'package:hiddify/services/deep_link_service.dart';
|
||||||
import 'package:hiddify/singbox/service/singbox_service_provider.dart';
|
import 'package:hiddify/singbox/service/singbox_service_provider.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:loggy/loggy.dart';
|
import 'package:loggy/loggy.dart';
|
||||||
import 'package:sentry_flutter/sentry_flutter.dart';
|
import 'package:sentry_flutter/sentry_flutter.dart';
|
||||||
import 'package:window_manager/window_manager.dart';
|
|
||||||
|
|
||||||
Future<void> lazyBootstrap(
|
Future<void> lazyBootstrap(
|
||||||
WidgetsBinding widgetsBinding,
|
WidgetsBinding widgetsBinding,
|
||||||
@@ -42,7 +41,6 @@ Future<void> lazyBootstrap(
|
|||||||
Logger.logPlatformDispatcherError;
|
Logger.logPlatformDispatcherError;
|
||||||
|
|
||||||
final stopWatch = Stopwatch()..start();
|
final stopWatch = Stopwatch()..start();
|
||||||
if (PlatformUtils.isDesktop) await windowManager.ensureInitialized();
|
|
||||||
|
|
||||||
final container = ProviderContainer(
|
final container = ProviderContainer(
|
||||||
overrides: [
|
overrides: [
|
||||||
@@ -95,6 +93,27 @@ Future<void> lazyBootstrap(
|
|||||||
|
|
||||||
final debug = container.read(debugModeNotifierProvider) || kDebugMode;
|
final debug = container.read(debugModeNotifierProvider) || kDebugMode;
|
||||||
|
|
||||||
|
if (PlatformUtils.isDesktop) {
|
||||||
|
await _init(
|
||||||
|
"window controller",
|
||||||
|
() => container.read(windowNotifierProvider.future),
|
||||||
|
);
|
||||||
|
|
||||||
|
final silentStart = container.read(silentStartNotifierProvider);
|
||||||
|
Logger.bootstrap
|
||||||
|
.debug("silent start [${silentStart ? "Enabled" : "Disabled"}]");
|
||||||
|
if (!silentStart) {
|
||||||
|
await container.read(windowNotifierProvider.notifier).open(focus: false);
|
||||||
|
} else {
|
||||||
|
Logger.bootstrap.debug("silent start, remain hidden accessible via tray");
|
||||||
|
}
|
||||||
|
|
||||||
|
await _init(
|
||||||
|
"auto start service",
|
||||||
|
() => container.read(autoStartNotifierProvider.future),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
await _init(
|
await _init(
|
||||||
"logs repository",
|
"logs repository",
|
||||||
() => container.read(logRepositoryProvider.future),
|
() => container.read(logRepositoryProvider.future),
|
||||||
@@ -111,24 +130,6 @@ Future<void> lazyBootstrap(
|
|||||||
() => container.read(profileRepositoryProvider.future),
|
() => container.read(profileRepositoryProvider.future),
|
||||||
);
|
);
|
||||||
|
|
||||||
final silentStart = container.read(silentStartNotifierProvider);
|
|
||||||
Logger.bootstrap
|
|
||||||
.debug("silent start [${silentStart ? "Enabled" : "Disabled"}]");
|
|
||||||
if (silentStart) {
|
|
||||||
FlutterNativeSplash.remove();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (PlatformUtils.isDesktop) {
|
|
||||||
await _init(
|
|
||||||
"auto start service",
|
|
||||||
() => container.read(autoStartServiceProvider.future),
|
|
||||||
);
|
|
||||||
await _init(
|
|
||||||
"window controller",
|
|
||||||
() => container.read(windowControllerProvider.future),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
await _init(
|
await _init(
|
||||||
"sing-box",
|
"sing-box",
|
||||||
() => container.read(singboxServiceProvider).init(),
|
() => container.read(singboxServiceProvider).init(),
|
||||||
@@ -146,7 +147,7 @@ Future<void> lazyBootstrap(
|
|||||||
if (PlatformUtils.isDesktop) {
|
if (PlatformUtils.isDesktop) {
|
||||||
await _safeInit(
|
await _safeInit(
|
||||||
"system tray",
|
"system tray",
|
||||||
() => container.read(systemTrayControllerProvider.future),
|
() => container.read(systemTrayNotifierProvider.future),
|
||||||
timeout: 1000,
|
timeout: 1000,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -163,7 +164,7 @@ Future<void> lazyBootstrap(
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!silentStart) FlutterNativeSplash.remove();
|
FlutterNativeSplash.remove();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<T> _init<T>(
|
Future<T> _init<T>(
|
||||||
|
|||||||
@@ -6,11 +6,15 @@ import 'package:hiddify/core/localization/locale_extensions.dart';
|
|||||||
import 'package:hiddify/core/localization/locale_preferences.dart';
|
import 'package:hiddify/core/localization/locale_preferences.dart';
|
||||||
import 'package:hiddify/core/localization/translations.dart';
|
import 'package:hiddify/core/localization/translations.dart';
|
||||||
import 'package:hiddify/core/model/constants.dart';
|
import 'package:hiddify/core/model/constants.dart';
|
||||||
|
import 'package:hiddify/core/preferences/general_preferences.dart';
|
||||||
import 'package:hiddify/core/router/router.dart';
|
import 'package:hiddify/core/router/router.dart';
|
||||||
import 'package:hiddify/core/theme/app_theme.dart';
|
import 'package:hiddify/core/theme/app_theme.dart';
|
||||||
import 'package:hiddify/core/theme/theme_preferences.dart';
|
import 'package:hiddify/core/theme/theme_preferences.dart';
|
||||||
import 'package:hiddify/features/app_update/notifier/app_update_notifier.dart';
|
import 'package:hiddify/features/app_update/notifier/app_update_notifier.dart';
|
||||||
import 'package:hiddify/features/common/common_controllers.dart';
|
import 'package:hiddify/features/connection/widget/connection_wrapper.dart';
|
||||||
|
import 'package:hiddify/features/profile/notifier/profiles_update_notifier.dart';
|
||||||
|
import 'package:hiddify/features/system_tray/widget/system_tray_wrapper.dart';
|
||||||
|
import 'package:hiddify/features/window/widget/window_wrapper.dart';
|
||||||
import 'package:hiddify/features/wrapper/shortcut/shortcut_wrapper.dart';
|
import 'package:hiddify/features/wrapper/shortcut/shortcut_wrapper.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
@@ -28,35 +32,48 @@ class App extends HookConsumerWidget with PresLogger {
|
|||||||
final themeMode = ref.watch(themePreferencesProvider);
|
final themeMode = ref.watch(themePreferencesProvider);
|
||||||
final theme = AppTheme(themeMode, locale.preferredFontFamily);
|
final theme = AppTheme(themeMode, locale.preferredFontFamily);
|
||||||
|
|
||||||
ref.watch(commonControllersProvider);
|
|
||||||
|
|
||||||
final upgrader = ref.watch(upgraderProvider);
|
final upgrader = ref.watch(upgraderProvider);
|
||||||
|
|
||||||
return ShortcutWrapper(
|
ref.listen(
|
||||||
MaterialApp.router(
|
introCompletedProvider,
|
||||||
routerConfig: router,
|
(_, completed) async {
|
||||||
locale: locale.flutterLocale,
|
if (completed) {
|
||||||
supportedLocales: AppLocaleUtils.supportedLocales,
|
await ref.read(foregroundProfilesUpdateNotifierProvider.future);
|
||||||
localizationsDelegates: GlobalMaterialLocalizations.delegates,
|
}
|
||||||
debugShowCheckedModeBanner: false,
|
},
|
||||||
themeMode: themeMode.flutterThemeMode,
|
);
|
||||||
theme: theme.light(),
|
|
||||||
darkTheme: theme.dark(),
|
return WindowWrapper(
|
||||||
title: Constants.appName,
|
TrayWrapper(
|
||||||
builder: (context, child) {
|
ShortcutWrapper(
|
||||||
child = UpgradeAlert(
|
ConnectionWrapper(
|
||||||
upgrader: upgrader,
|
MaterialApp.router(
|
||||||
navigatorKey: router.routerDelegate.navigatorKey,
|
routerConfig: router,
|
||||||
child: child ?? const SizedBox(),
|
locale: locale.flutterLocale,
|
||||||
);
|
supportedLocales: AppLocaleUtils.supportedLocales,
|
||||||
if (kDebugMode && _debugAccessibility) {
|
localizationsDelegates: GlobalMaterialLocalizations.delegates,
|
||||||
return AccessibilityTools(
|
debugShowCheckedModeBanner: false,
|
||||||
checkFontOverflows: true,
|
themeMode: themeMode.flutterThemeMode,
|
||||||
child: child,
|
theme: theme.light(),
|
||||||
);
|
darkTheme: theme.dark(),
|
||||||
}
|
title: Constants.appName,
|
||||||
return child;
|
builder: (context, child) {
|
||||||
},
|
child = UpgradeAlert(
|
||||||
|
upgrader: upgrader,
|
||||||
|
navigatorKey: router.routerDelegate.navigatorKey,
|
||||||
|
child: child ?? const SizedBox(),
|
||||||
|
);
|
||||||
|
if (kDebugMode && _debugAccessibility) {
|
||||||
|
return AccessibilityTools(
|
||||||
|
checkFontOverflows: true,
|
||||||
|
child: child,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return child;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,14 +5,14 @@ import 'package:hiddify/utils/utils.dart';
|
|||||||
import 'package:launch_at_startup/launch_at_startup.dart';
|
import 'package:launch_at_startup/launch_at_startup.dart';
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||||
|
|
||||||
part 'auto_start_service.g.dart';
|
part 'auto_start_notifier.g.dart';
|
||||||
|
|
||||||
@Riverpod(keepAlive: true)
|
@Riverpod(keepAlive: true)
|
||||||
class AutoStartService extends _$AutoStartService with InfraLogger {
|
class AutoStartNotifier extends _$AutoStartNotifier with InfraLogger {
|
||||||
@override
|
@override
|
||||||
Future<bool> build() async {
|
Future<bool> build() async {
|
||||||
loggy.debug("initializing");
|
|
||||||
if (!PlatformUtils.isDesktop) return false;
|
if (!PlatformUtils.isDesktop) return false;
|
||||||
|
|
||||||
final appInfo = ref.watch(appInfoProvider).requireValue;
|
final appInfo = ref.watch(appInfoProvider).requireValue;
|
||||||
launchAtStartup.setup(
|
launchAtStartup.setup(
|
||||||
appName: appInfo.name,
|
appName: appInfo.name,
|
||||||
@@ -1,41 +0,0 @@
|
|||||||
import 'package:hiddify/core/preferences/general_preferences.dart';
|
|
||||||
import 'package:hiddify/features/common/window/window_controller.dart';
|
|
||||||
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
|
||||||
import 'package:hiddify/features/profile/notifier/profiles_update_notifier.dart';
|
|
||||||
import 'package:hiddify/features/system_tray/system_tray_controller.dart';
|
|
||||||
import 'package:hiddify/utils/platform_utils.dart';
|
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
||||||
|
|
||||||
part 'common_controllers.g.dart';
|
|
||||||
|
|
||||||
// this is a temporary solution to keep providers running even when there are no active listeners
|
|
||||||
// https://github.com/rrousselGit/riverpod/discussions/2730
|
|
||||||
@Riverpod(keepAlive: true)
|
|
||||||
void commonControllers(CommonControllersRef ref) {
|
|
||||||
ref.listen(
|
|
||||||
introCompletedProvider,
|
|
||||||
(_, completed) async {
|
|
||||||
if (completed) {
|
|
||||||
await ref.read(foregroundProfilesUpdateNotifierProvider.future);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
fireImmediately: true,
|
|
||||||
);
|
|
||||||
ref.listen(
|
|
||||||
connectionNotifierProvider,
|
|
||||||
(previous, next) {},
|
|
||||||
fireImmediately: true,
|
|
||||||
);
|
|
||||||
if (PlatformUtils.isDesktop) {
|
|
||||||
ref.listen(
|
|
||||||
windowControllerProvider,
|
|
||||||
(previous, next) {},
|
|
||||||
fireImmediately: true,
|
|
||||||
);
|
|
||||||
ref.listen(
|
|
||||||
systemTrayControllerProvider,
|
|
||||||
(previous, next) {},
|
|
||||||
fireImmediately: true,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
import 'package:flutter/material.dart';
|
|
||||||
import 'package:hiddify/core/app_info/app_info_provider.dart';
|
|
||||||
import 'package:hiddify/core/model/constants.dart';
|
|
||||||
import 'package:hiddify/core/preferences/general_preferences.dart';
|
|
||||||
import 'package:hiddify/core/preferences/service_preferences.dart';
|
|
||||||
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
|
||||||
import 'package:hiddify/utils/utils.dart';
|
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
||||||
import 'package:window_manager/window_manager.dart';
|
|
||||||
|
|
||||||
part 'window_controller.g.dart';
|
|
||||||
|
|
||||||
// TODO improve
|
|
||||||
@Riverpod(keepAlive: true)
|
|
||||||
class WindowController extends _$WindowController
|
|
||||||
with WindowListener, AppLogger {
|
|
||||||
@override
|
|
||||||
Future<bool> build() async {
|
|
||||||
await windowManager.ensureInitialized();
|
|
||||||
const size = Size(868, 668);
|
|
||||||
const minimumSize = Size(368, 568);
|
|
||||||
const windowOptions = WindowOptions(
|
|
||||||
size: size,
|
|
||||||
minimumSize: minimumSize,
|
|
||||||
center: true,
|
|
||||||
);
|
|
||||||
await windowManager.setPreventClose(true);
|
|
||||||
await windowManager.waitUntilReadyToShow(
|
|
||||||
windowOptions,
|
|
||||||
() async {
|
|
||||||
final version = await ref.watch(appInfoProvider.future);
|
|
||||||
await windowManager
|
|
||||||
.setTitle("${Constants.appName} ${version.presentVersion}");
|
|
||||||
|
|
||||||
if (ref.read(silentStartNotifierProvider)) {
|
|
||||||
loggy.debug("silent start is enabled, hiding window");
|
|
||||||
await windowManager.hide();
|
|
||||||
}
|
|
||||||
await Future.delayed(
|
|
||||||
const Duration(seconds: 3),
|
|
||||||
() async {
|
|
||||||
if (ref.read(startedByUserProvider)) {
|
|
||||||
loggy.debug("previously started by user, trying to connect");
|
|
||||||
return ref.read(connectionNotifierProvider.notifier).mayConnect();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
windowManager.addListener(this);
|
|
||||||
|
|
||||||
ref.onDispose(() {
|
|
||||||
loggy.debug("disposing");
|
|
||||||
windowManager.removeListener(this);
|
|
||||||
});
|
|
||||||
return windowManager.isVisible();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> show() async {
|
|
||||||
await windowManager.show();
|
|
||||||
await windowManager.focus();
|
|
||||||
state = const AsyncData(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> hide() async {
|
|
||||||
await windowManager.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> quit() async {
|
|
||||||
loggy.debug("quitting");
|
|
||||||
await windowManager.close();
|
|
||||||
await windowManager.destroy();
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<void> onWindowClose() async {
|
|
||||||
await windowManager.hide();
|
|
||||||
state = const AsyncData(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
39
lib/features/connection/widget/connection_wrapper.dart
Normal file
39
lib/features/connection/widget/connection_wrapper.dart
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:hiddify/core/preferences/service_preferences.dart';
|
||||||
|
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
||||||
|
import 'package:hiddify/utils/custom_loggers.dart';
|
||||||
|
import 'package:hiddify/utils/platform_utils.dart';
|
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
|
||||||
|
class ConnectionWrapper extends StatefulHookConsumerWidget {
|
||||||
|
const ConnectionWrapper(this.child, {super.key});
|
||||||
|
|
||||||
|
final Widget child;
|
||||||
|
|
||||||
|
@override
|
||||||
|
ConsumerState<ConsumerStatefulWidget> createState() =>
|
||||||
|
_ConnectionWrapperState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ConnectionWrapperState extends ConsumerState<ConnectionWrapper>
|
||||||
|
with AppLogger {
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
ref.listen(connectionNotifierProvider, (_, __) {});
|
||||||
|
|
||||||
|
return widget.child;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
Future.delayed(const Duration(seconds: 2)).then(
|
||||||
|
(_) async {
|
||||||
|
if (ref.read(startedByUserProvider) && PlatformUtils.isDesktop) {
|
||||||
|
loggy.debug("previously started by user, trying to connect");
|
||||||
|
return ref.read(connectionNotifierProvider.notifier).mayConnect();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,8 +5,8 @@ import 'package:hiddify/core/localization/translations.dart';
|
|||||||
import 'package:hiddify/core/preferences/general_preferences.dart';
|
import 'package:hiddify/core/preferences/general_preferences.dart';
|
||||||
import 'package:hiddify/core/theme/app_theme_mode.dart';
|
import 'package:hiddify/core/theme/app_theme_mode.dart';
|
||||||
import 'package:hiddify/core/theme/theme_preferences.dart';
|
import 'package:hiddify/core/theme/theme_preferences.dart';
|
||||||
|
import 'package:hiddify/features/auto_start/notifier/auto_start_notifier.dart';
|
||||||
import 'package:hiddify/features/common/general_pref_tiles.dart';
|
import 'package:hiddify/features/common/general_pref_tiles.dart';
|
||||||
import 'package:hiddify/services/auto_start_service.dart';
|
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
|
||||||
@@ -67,12 +67,12 @@ class GeneralSettingTiles extends HookConsumerWidget {
|
|||||||
if (PlatformUtils.isDesktop) ...[
|
if (PlatformUtils.isDesktop) ...[
|
||||||
SwitchListTile(
|
SwitchListTile(
|
||||||
title: Text(t.settings.general.autoStart),
|
title: Text(t.settings.general.autoStart),
|
||||||
value: ref.watch(autoStartServiceProvider).asData!.value,
|
value: ref.watch(autoStartNotifierProvider).asData!.value,
|
||||||
onChanged: (value) async {
|
onChanged: (value) async {
|
||||||
if (value) {
|
if (value) {
|
||||||
await ref.read(autoStartServiceProvider.notifier).enable();
|
await ref.read(autoStartNotifierProvider.notifier).enable();
|
||||||
} else {
|
} else {
|
||||||
await ref.read(autoStartServiceProvider.notifier).disable();
|
await ref.read(autoStartNotifierProvider.notifier).disable();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -3,34 +3,30 @@ import 'dart:io';
|
|||||||
import 'package:hiddify/core/localization/translations.dart';
|
import 'package:hiddify/core/localization/translations.dart';
|
||||||
import 'package:hiddify/core/model/constants.dart';
|
import 'package:hiddify/core/model/constants.dart';
|
||||||
import 'package:hiddify/core/router/router.dart';
|
import 'package:hiddify/core/router/router.dart';
|
||||||
import 'package:hiddify/features/common/window/window_controller.dart';
|
|
||||||
import 'package:hiddify/features/config_option/model/config_option_patch.dart';
|
import 'package:hiddify/features/config_option/model/config_option_patch.dart';
|
||||||
import 'package:hiddify/features/config_option/notifier/config_option_notifier.dart';
|
import 'package:hiddify/features/config_option/notifier/config_option_notifier.dart';
|
||||||
import 'package:hiddify/features/connection/model/connection_status.dart';
|
import 'package:hiddify/features/connection/model/connection_status.dart';
|
||||||
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
||||||
|
import 'package:hiddify/features/window/notifier/window_notifier.dart';
|
||||||
import 'package:hiddify/gen/assets.gen.dart';
|
import 'package:hiddify/gen/assets.gen.dart';
|
||||||
import 'package:hiddify/singbox/model/singbox_config_enum.dart';
|
import 'package:hiddify/singbox/model/singbox_config_enum.dart';
|
||||||
import 'package:hiddify/utils/utils.dart';
|
import 'package:hiddify/utils/utils.dart';
|
||||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||||
import 'package:tray_manager/tray_manager.dart';
|
import 'package:tray_manager/tray_manager.dart';
|
||||||
|
|
||||||
part 'system_tray_controller.g.dart';
|
part 'system_tray_notifier.g.dart';
|
||||||
|
|
||||||
@Riverpod(keepAlive: true)
|
@Riverpod(keepAlive: true)
|
||||||
class SystemTrayController extends _$SystemTrayController
|
class SystemTrayNotifier extends _$SystemTrayNotifier with AppLogger {
|
||||||
with TrayListener, AppLogger {
|
|
||||||
@override
|
@override
|
||||||
Future<void> build() async {
|
Future<void> build() async {
|
||||||
if (!_initialized) {
|
if (!PlatformUtils.isDesktop) return;
|
||||||
loggy.debug('initializing');
|
|
||||||
await trayManager.setIcon(
|
await trayManager.setIcon(
|
||||||
_trayIconPath,
|
_trayIconPath,
|
||||||
isTemplate: Platform.isMacOS,
|
isTemplate: Platform.isMacOS,
|
||||||
);
|
);
|
||||||
if (!Platform.isLinux) await trayManager.setToolTip(Constants.appName);
|
if (!Platform.isLinux) await trayManager.setToolTip(Constants.appName);
|
||||||
trayManager.addListener(this);
|
|
||||||
_initialized = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
ConnectionStatus connection;
|
ConnectionStatus connection;
|
||||||
try {
|
try {
|
||||||
@@ -55,11 +51,13 @@ class SystemTrayController extends _$SystemTrayController
|
|||||||
|
|
||||||
loggy.debug('updating system tray');
|
loggy.debug('updating system tray');
|
||||||
|
|
||||||
final trayMenu = Menu(
|
final menu = Menu(
|
||||||
items: [
|
items: [
|
||||||
MenuItem(
|
MenuItem(
|
||||||
label: t.tray.dashboard,
|
label: t.tray.dashboard,
|
||||||
onClick: handleClickShowApp,
|
onClick: (_) async {
|
||||||
|
await ref.read(windowNotifierProvider.notifier).open();
|
||||||
|
},
|
||||||
),
|
),
|
||||||
MenuItem.separator(),
|
MenuItem.separator(),
|
||||||
MenuItem.checkbox(
|
MenuItem.checkbox(
|
||||||
@@ -71,7 +69,11 @@ class SystemTrayController extends _$SystemTrayController
|
|||||||
},
|
},
|
||||||
checked: connection.isConnected,
|
checked: connection.isConnected,
|
||||||
disabled: connection.isSwitching,
|
disabled: connection.isSwitching,
|
||||||
onClick: handleClickSetAsSystemProxy,
|
onClick: (_) async {
|
||||||
|
await ref
|
||||||
|
.read(connectionNotifierProvider.notifier)
|
||||||
|
.toggleConnection();
|
||||||
|
},
|
||||||
),
|
),
|
||||||
MenuItem.submenu(
|
MenuItem.submenu(
|
||||||
label: t.settings.config.serviceMode,
|
label: t.settings.config.serviceMode,
|
||||||
@@ -102,7 +104,7 @@ class SystemTrayController extends _$SystemTrayController
|
|||||||
(e) => MenuItem(
|
(e) => MenuItem(
|
||||||
label: e.$1,
|
label: e.$1,
|
||||||
onClick: (_) async {
|
onClick: (_) async {
|
||||||
await ref.read(windowControllerProvider.notifier).show();
|
await ref.read(windowNotifierProvider.notifier).open();
|
||||||
ref.read(routerProvider).go(e.$2);
|
ref.read(routerProvider).go(e.$2);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
@@ -113,46 +115,18 @@ class SystemTrayController extends _$SystemTrayController
|
|||||||
MenuItem.separator(),
|
MenuItem.separator(),
|
||||||
MenuItem(
|
MenuItem(
|
||||||
label: t.tray.quit,
|
label: t.tray.quit,
|
||||||
onClick: handleClickExitApp,
|
onClick: (_) async {
|
||||||
|
return ref.read(windowNotifierProvider.notifier).quit();
|
||||||
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
await trayManager.setContextMenu(trayMenu);
|
|
||||||
|
await trayManager.setContextMenu(menu);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool _initialized = false;
|
static String get _trayIconPath {
|
||||||
|
|
||||||
String get _trayIconPath {
|
|
||||||
if (Platform.isWindows) return Assets.images.trayIconIco;
|
if (Platform.isWindows) return Assets.images.trayIconIco;
|
||||||
return Assets.images.trayIconPng.path;
|
return Assets.images.trayIconPng.path;
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
|
||||||
Future<void> onTrayIconMouseDown() async {
|
|
||||||
if (Platform.isMacOS) {
|
|
||||||
await trayManager.popUpContextMenu();
|
|
||||||
} else {
|
|
||||||
await ref.read(windowControllerProvider.notifier).show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Future<void> onTrayIconRightMouseDown() async {
|
|
||||||
super.onTrayIconRightMouseDown();
|
|
||||||
await trayManager.popUpContextMenu();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> handleClickShowApp(MenuItem menuItem) async {
|
|
||||||
await ref.read(windowControllerProvider.notifier).show();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> handleClickSetAsSystemProxy(MenuItem menuItem) async {
|
|
||||||
return ref.read(connectionNotifierProvider.notifier).toggleConnection();
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> handleClickExitApp(MenuItem menuItem) async {
|
|
||||||
await ref.read(connectionNotifierProvider.notifier).abortConnection();
|
|
||||||
await trayManager.destroy();
|
|
||||||
return ref.read(windowControllerProvider.notifier).quit();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
53
lib/features/system_tray/widget/system_tray_wrapper.dart
Normal file
53
lib/features/system_tray/widget/system_tray_wrapper.dart
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:hiddify/features/system_tray/notifier/system_tray_notifier.dart';
|
||||||
|
import 'package:hiddify/features/window/notifier/window_notifier.dart';
|
||||||
|
import 'package:hiddify/utils/custom_loggers.dart';
|
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
import 'package:tray_manager/tray_manager.dart';
|
||||||
|
|
||||||
|
class TrayWrapper extends StatefulHookConsumerWidget {
|
||||||
|
const TrayWrapper(this.child, {super.key});
|
||||||
|
|
||||||
|
final Widget child;
|
||||||
|
|
||||||
|
@override
|
||||||
|
ConsumerState<ConsumerStatefulWidget> createState() => _TrayWrapperState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _TrayWrapperState extends ConsumerState<TrayWrapper>
|
||||||
|
with TrayListener, AppLogger {
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
ref.listen(systemTrayNotifierProvider, (_, __) {});
|
||||||
|
|
||||||
|
return widget.child;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
trayManager.addListener(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
trayManager.removeListener(this);
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> onTrayIconMouseDown() async {
|
||||||
|
if (Platform.isMacOS) {
|
||||||
|
await trayManager.popUpContextMenu();
|
||||||
|
} else {
|
||||||
|
await ref.read(windowNotifierProvider.notifier).open();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> onTrayIconRightMouseDown() async {
|
||||||
|
await trayManager.popUpContextMenu();
|
||||||
|
}
|
||||||
|
}
|
||||||
54
lib/features/window/notifier/window_notifier.dart
Normal file
54
lib/features/window/notifier/window_notifier.dart
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import 'dart:ui';
|
||||||
|
|
||||||
|
import 'package:hiddify/core/app_info/app_info_provider.dart';
|
||||||
|
import 'package:hiddify/core/model/constants.dart';
|
||||||
|
import 'package:hiddify/features/connection/notifier/connection_notifier.dart';
|
||||||
|
import 'package:hiddify/utils/utils.dart';
|
||||||
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||||
|
import 'package:tray_manager/tray_manager.dart';
|
||||||
|
import 'package:window_manager/window_manager.dart';
|
||||||
|
|
||||||
|
part 'window_notifier.g.dart';
|
||||||
|
|
||||||
|
const minimumWindowSize = Size(368, 568);
|
||||||
|
const defaultWindowSize = Size(868, 668);
|
||||||
|
|
||||||
|
@Riverpod(keepAlive: true)
|
||||||
|
class WindowNotifier extends _$WindowNotifier with AppLogger {
|
||||||
|
@override
|
||||||
|
Future<void> build() async {
|
||||||
|
if (!PlatformUtils.isDesktop) return;
|
||||||
|
|
||||||
|
await windowManager.ensureInitialized();
|
||||||
|
await windowManager.setMinimumSize(minimumWindowSize);
|
||||||
|
await windowManager.setSize(defaultWindowSize);
|
||||||
|
|
||||||
|
final appInfo = await ref.watch(appInfoProvider.future);
|
||||||
|
await windowManager
|
||||||
|
.setTitle("${Constants.appName} v${appInfo.presentVersion}");
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> open({bool focus = true}) async {
|
||||||
|
await windowManager.show();
|
||||||
|
if (focus) await windowManager.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO add option to quit or minimize to tray
|
||||||
|
Future<void> close() async {
|
||||||
|
await windowManager.hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> quit() async {
|
||||||
|
await ref
|
||||||
|
.read(connectionNotifierProvider.notifier)
|
||||||
|
.abortConnection()
|
||||||
|
.timeout(const Duration(seconds: 2))
|
||||||
|
.catchError(
|
||||||
|
(e) {
|
||||||
|
loggy.warning("error aborting connection on quit", e);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
await trayManager.destroy();
|
||||||
|
await windowManager.destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
54
lib/features/window/widget/window_wrapper.dart
Normal file
54
lib/features/window/widget/window_wrapper.dart
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import 'dart:async';
|
||||||
|
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:hiddify/features/window/notifier/window_notifier.dart';
|
||||||
|
import 'package:hiddify/utils/custom_loggers.dart';
|
||||||
|
import 'package:hiddify/utils/platform_utils.dart';
|
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
import 'package:window_manager/window_manager.dart';
|
||||||
|
|
||||||
|
class WindowWrapper extends StatefulHookConsumerWidget {
|
||||||
|
const WindowWrapper(this.child, {super.key});
|
||||||
|
|
||||||
|
final Widget child;
|
||||||
|
|
||||||
|
@override
|
||||||
|
ConsumerState<ConsumerStatefulWidget> createState() => _WindowWrapperState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _WindowWrapperState extends ConsumerState<WindowWrapper>
|
||||||
|
with WindowListener, AppLogger {
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
ref.watch(windowNotifierProvider);
|
||||||
|
|
||||||
|
return widget.child;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
windowManager.addListener(this);
|
||||||
|
if (PlatformUtils.isDesktop) {
|
||||||
|
WidgetsBinding.instance.addPostFrameCallback((_) async {
|
||||||
|
await windowManager.setPreventClose(true);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
windowManager.removeListener(this);
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<void> onWindowClose() async {
|
||||||
|
await ref.read(windowNotifierProvider.notifier).close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void onWindowFocus() {
|
||||||
|
setState(() {});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,7 +3,7 @@ import 'dart:io';
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:hiddify/core/router/router.dart';
|
import 'package:hiddify/core/router/router.dart';
|
||||||
import 'package:hiddify/features/common/window/window_controller.dart';
|
import 'package:hiddify/features/window/notifier/window_notifier.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
|
||||||
class ShortcutWrapper extends HookConsumerWidget {
|
class ShortcutWrapper extends HookConsumerWidget {
|
||||||
@@ -44,13 +44,13 @@ class ShortcutWrapper extends HookConsumerWidget {
|
|||||||
actions: {
|
actions: {
|
||||||
CloseWindowIntent: CallbackAction(
|
CloseWindowIntent: CallbackAction(
|
||||||
onInvoke: (_) async {
|
onInvoke: (_) async {
|
||||||
await ref.read(windowControllerProvider.notifier).hide();
|
await ref.read(windowNotifierProvider.notifier).close();
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
QuitAppIntent: CallbackAction(
|
QuitAppIntent: CallbackAction(
|
||||||
onInvoke: (_) async {
|
onInvoke: (_) async {
|
||||||
await ref.read(windowControllerProvider.notifier).quit();
|
await ref.read(windowNotifierProvider.notifier).quit();
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ static void my_application_activate(GApplication* application) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
gtk_window_set_default_size(window, 1280, 720);
|
gtk_window_set_default_size(window, 1280, 720);
|
||||||
gtk_widget_show(GTK_WIDGET(window));
|
gtk_widget_realize(GTK_WIDGET(window));
|
||||||
|
|
||||||
g_autoptr(FlDartProject) project = fl_dart_project_new();
|
g_autoptr(FlDartProject) project = fl_dart_project_new();
|
||||||
fl_dart_project_set_dart_entrypoint_arguments(project, self->dart_entrypoint_arguments);
|
fl_dart_project_set_dart_entrypoint_arguments(project, self->dart_entrypoint_arguments);
|
||||||
|
|||||||
@@ -8,17 +8,17 @@ class AppDelegate: FlutterAppDelegate {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// window manager restore from dock: https://leanflutter.dev/blog/click-dock-icon-to-restore-after-closing-the-window
|
// // window manager restore from dock: https://leanflutter.dev/blog/click-dock-icon-to-restore-after-closing-the-window
|
||||||
override func applicationShouldHandleReopen(_ sender: NSApplication, hasVisibleWindows flag: Bool) -> Bool {
|
// override func applicationShouldHandleReopen(_ sender: NSApplication, hasVisibleWindows flag: Bool) -> Bool {
|
||||||
if !flag {
|
// if !flag {
|
||||||
for window in NSApp.windows {
|
// for window in NSApp.windows {
|
||||||
if !window.isVisible {
|
// if !window.isVisible {
|
||||||
window.setIsVisible(true)
|
// window.setIsVisible(true)
|
||||||
}
|
// }
|
||||||
window.makeKeyAndOrderFront(self)
|
// window.makeKeyAndOrderFront(self)
|
||||||
NSApp.activate(ignoringOtherApps: true)
|
// NSApp.activate(ignoringOtherApps: true)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
return true
|
// return true
|
||||||
}
|
// }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import Cocoa
|
import Cocoa
|
||||||
import FlutterMacOS
|
import FlutterMacOS
|
||||||
|
import window_manager
|
||||||
|
|
||||||
class MainFlutterWindow: NSWindow {
|
class MainFlutterWindow: NSWindow {
|
||||||
override func awakeFromNib() {
|
override func awakeFromNib() {
|
||||||
@@ -12,4 +13,10 @@ class MainFlutterWindow: NSWindow {
|
|||||||
|
|
||||||
super.awakeFromNib()
|
super.awakeFromNib()
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
// window manager hidden at launch
|
||||||
|
override public func order(_ place: NSWindow.OrderingMode, relativeTo otherWin: Int) {
|
||||||
|
super.order(place, relativeTo: otherWin)
|
||||||
|
hiddenWindowAtLaunch()
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -28,7 +28,8 @@ bool FlutterWindow::OnCreate() {
|
|||||||
SetChildContent(flutter_controller_->view()->GetNativeWindow());
|
SetChildContent(flutter_controller_->view()->GetNativeWindow());
|
||||||
|
|
||||||
flutter_controller_->engine()->SetNextFrameCallback([&]() {
|
flutter_controller_->engine()->SetNextFrameCallback([&]() {
|
||||||
this->Show();
|
// this->Show(); window_manager hidden at launch
|
||||||
|
""
|
||||||
});
|
});
|
||||||
|
|
||||||
// Flutter can complete the first frame before the "show window" callback is
|
// Flutter can complete the first frame before the "show window" callback is
|
||||||
|
|||||||
@@ -135,7 +135,9 @@ bool Win32Window::Create(const std::wstring& title,
|
|||||||
double scale_factor = dpi / 96.0;
|
double scale_factor = dpi / 96.0;
|
||||||
|
|
||||||
HWND window = CreateWindow(
|
HWND window = CreateWindow(
|
||||||
window_class, title.c_str(), WS_OVERLAPPEDWINDOW,
|
// window_class, title.c_str(), WS_OVERLAPPEDWINDOW, // window_manager hidden at launch
|
||||||
|
window_class, title.c_str(),
|
||||||
|
WS_OVERLAPPEDWINDOW, // do not add WS_VISIBLE since the window will be shown later
|
||||||
Scale(origin.x, scale_factor), Scale(origin.y, scale_factor),
|
Scale(origin.x, scale_factor), Scale(origin.y, scale_factor),
|
||||||
Scale(size.width, scale_factor), Scale(size.height, scale_factor),
|
Scale(size.width, scale_factor), Scale(size.height, scale_factor),
|
||||||
nullptr, nullptr, GetModuleHandle(nullptr), this);
|
nullptr, nullptr, GetModuleHandle(nullptr), this);
|
||||||
|
|||||||
Reference in New Issue
Block a user