Files
umbrix/lib/features/system_tray/notifier/system_tray_notifier.dart
Umbrix Developer 76a374950f feat: mobile-like window size and always-visible stats
- 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
2026-01-17 13:09:20 +03:00

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;
}
}