- 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
192 lines
6.4 KiB
Dart
192 lines
6.4 KiB
Dart
import 'dart:io';
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:umbrix/core/localization/translations.dart';
|
|
import 'package:umbrix/core/model/constants.dart';
|
|
import 'package:umbrix/core/router/router.dart';
|
|
import 'package:umbrix/features/config_option/data/config_option_repository.dart';
|
|
import 'package:umbrix/features/connection/model/connection_status.dart';
|
|
import 'package:umbrix/features/connection/notifier/connection_notifier.dart';
|
|
import 'package:umbrix/features/proxy/active/active_proxy_notifier.dart';
|
|
import 'package:umbrix/features/window/notifier/window_notifier.dart';
|
|
import 'package:umbrix/gen/assets.gen.dart';
|
|
import 'package:umbrix/singbox/model/singbox_config_enum.dart';
|
|
import 'package:umbrix/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 'system_tray_notifier.g.dart';
|
|
|
|
@Riverpod(keepAlive: true)
|
|
class SystemTrayNotifier extends _$SystemTrayNotifier with AppLogger {
|
|
@override
|
|
Future<void> build() async {
|
|
if (!PlatformUtils.isDesktop) return;
|
|
|
|
final activeProxy = ref.watch(activeProxyNotifierProvider);
|
|
final delay = activeProxy.value?.urlTestDelay ?? 0;
|
|
final newConnectionStatus = delay > 0 && delay < 65000;
|
|
ConnectionStatus connection;
|
|
try {
|
|
connection = await ref.watch(connectionNotifierProvider.future);
|
|
} catch (e) {
|
|
loggy.warning("error getting connection status", e);
|
|
connection = const ConnectionStatus.disconnected();
|
|
}
|
|
|
|
final t = ref.watch(translationsProvider);
|
|
|
|
var tooltip = Constants.appName;
|
|
final serviceMode = ref.watch(ConfigOptions.serviceMode);
|
|
if (connection == const Disconnected()) {
|
|
setIcon(connection);
|
|
} else if (newConnectionStatus) {
|
|
setIcon(const Connected());
|
|
tooltip = "$tooltip - ${connection.present(t)}";
|
|
if (newConnectionStatus) {
|
|
tooltip = "$tooltip : ${delay}ms";
|
|
} else {
|
|
tooltip = "$tooltip : -";
|
|
}
|
|
// else if (delay>1000)
|
|
// SystemTrayNotifier.setIcon(timeout ? Disconnecting() : Connecting());
|
|
} else {
|
|
setIcon(const Disconnecting());
|
|
tooltip = "$tooltip - ${connection.present(t)}";
|
|
}
|
|
if (Platform.isMacOS) {
|
|
windowManager.setBadgeLabel("${delay}ms");
|
|
}
|
|
if (!Platform.isLinux) await trayManager.setToolTip(tooltip);
|
|
|
|
final destinations = <(String label, String location)>[
|
|
(t.home.pageTitle, const HomeRoute().location),
|
|
(t.proxies.pageTitle, const ProxiesRoute().location),
|
|
(t.logs.pageTitle, const LogsOverviewRoute().location),
|
|
(t.settings.pageTitle, const SettingsRoute().location),
|
|
(t.about.pageTitle, const AboutRoute().location),
|
|
];
|
|
|
|
// loggy.debug('updating system tray');
|
|
|
|
final menu = Menu(
|
|
items: [
|
|
MenuItem(
|
|
label: t.tray.dashboard,
|
|
onClick: (_) async {
|
|
await ref.read(windowNotifierProvider.notifier).open();
|
|
},
|
|
),
|
|
MenuItem.separator(),
|
|
MenuItem.checkbox(
|
|
label: switch (connection) {
|
|
Disconnected() => t.tray.status.connect,
|
|
Connecting() => t.tray.status.connecting,
|
|
Connected() => t.tray.status.disconnect,
|
|
Disconnecting() => t.tray.status.disconnecting,
|
|
},
|
|
// checked: connection.isConnected,
|
|
checked: false,
|
|
disabled: connection.isSwitching,
|
|
onClick: (_) async {
|
|
await ref.read(connectionNotifierProvider.notifier).toggleConnection();
|
|
},
|
|
),
|
|
MenuItem.separator(),
|
|
MenuItem(
|
|
label: t.config.serviceMode,
|
|
icon: Assets.images.trayIconIco,
|
|
disabled: true,
|
|
),
|
|
|
|
...ServiceMode.values.map(
|
|
(e) => MenuItem.checkbox(
|
|
checked: e == serviceMode,
|
|
key: e.name,
|
|
label: e.present(t),
|
|
onClick: (menuItem) async {
|
|
final newMode = ServiceMode.values.byName(menuItem.key!);
|
|
loggy.debug("switching service mode: [$newMode]");
|
|
await ref.read(ConfigOptions.serviceMode.notifier).update(newMode);
|
|
},
|
|
),
|
|
),
|
|
|
|
// MenuItem.submenu(
|
|
// label: t.tray.open,
|
|
// submenu: Menu(
|
|
// items: [
|
|
// ...destinations.map(
|
|
// (e) => MenuItem(
|
|
// label: e.$1,
|
|
// onClick: (_) async {
|
|
// await ref.read(windowNotifierProvider.notifier).open();
|
|
// ref.read(routerProvider).go(e.$2);
|
|
// },
|
|
// ),
|
|
// ),
|
|
// ],
|
|
// ),
|
|
// ),
|
|
MenuItem.separator(),
|
|
MenuItem(
|
|
label: t.tray.quit,
|
|
onClick: (_) async {
|
|
return ref.read(windowNotifierProvider.notifier).quit();
|
|
},
|
|
),
|
|
],
|
|
);
|
|
|
|
await trayManager.setContextMenu(menu);
|
|
}
|
|
|
|
static void setIcon(ConnectionStatus status) {
|
|
if (!PlatformUtils.isDesktop) return;
|
|
trayManager
|
|
.setIcon(
|
|
_trayIconPath(status),
|
|
isTemplate: Platform.isMacOS,
|
|
)
|
|
.asStream();
|
|
}
|
|
|
|
static String _trayIconPath(ConnectionStatus status) {
|
|
if (Platform.isWindows) {
|
|
final Brightness brightness = WidgetsBinding.instance.platformDispatcher.platformBrightness;
|
|
final isDarkMode = brightness == Brightness.dark;
|
|
switch (status) {
|
|
case Connected():
|
|
return Assets.images.trayIconConnectedIco;
|
|
case Connecting():
|
|
return Assets.images.trayIconDisconnectedIco;
|
|
case Disconnecting():
|
|
return Assets.images.trayIconDisconnectedIco;
|
|
case Disconnected():
|
|
if (isDarkMode) {
|
|
return Assets.images.trayIconIco;
|
|
} else {
|
|
return Assets.images.trayIconDarkIco;
|
|
}
|
|
}
|
|
}
|
|
const isDarkMode = false;
|
|
switch (status) {
|
|
case Connected():
|
|
return Assets.images.trayIconConnectedPng.path;
|
|
case Connecting():
|
|
return Assets.images.trayIconDisconnectedPng.path;
|
|
case Disconnecting():
|
|
return Assets.images.trayIconDisconnectedPng.path;
|
|
case Disconnected():
|
|
if (isDarkMode) {
|
|
return Assets.images.trayIconDarkPng.path;
|
|
} else {
|
|
return Assets.images.trayIconPng.path;
|
|
}
|
|
}
|
|
// return Assets.images.trayIconPng.path;
|
|
}
|
|
}
|