2023-02-07 01:31:11 +08:00
|
|
|
import 'dart:async';
|
2022-05-29 17:19:50 +08:00
|
|
|
import 'dart:convert';
|
2022-11-05 23:41:22 +08:00
|
|
|
import 'dart:io';
|
2022-05-29 17:19:50 +08:00
|
|
|
|
2023-02-03 17:08:40 +08:00
|
|
|
import 'package:bot_toast/bot_toast.dart';
|
2022-08-18 17:25:47 +08:00
|
|
|
import 'package:desktop_multi_window/desktop_multi_window.dart';
|
2020-11-06 18:04:04 +08:00
|
|
|
import 'package:flutter/material.dart';
|
2023-08-04 13:11:24 +08:00
|
|
|
import 'package:flutter/services.dart';
|
2022-08-11 16:03:04 +08:00
|
|
|
import 'package:flutter_hbb/desktop/pages/desktop_tab_page.dart';
|
2022-10-09 20:32:28 +08:00
|
|
|
import 'package:flutter_hbb/desktop/pages/install_page.dart';
|
2023-02-03 17:08:40 +08:00
|
|
|
import 'package:flutter_hbb/desktop/pages/server_page.dart';
|
2022-06-17 22:57:41 +08:00
|
|
|
import 'package:flutter_hbb/desktop/screen/desktop_file_transfer_screen.dart';
|
2022-08-26 11:35:28 +08:00
|
|
|
import 'package:flutter_hbb/desktop/screen/desktop_port_forward_screen.dart';
|
2022-05-29 17:19:50 +08:00
|
|
|
import 'package:flutter_hbb/desktop/screen/desktop_remote_screen.dart';
|
2022-10-26 14:39:13 +08:00
|
|
|
import 'package:flutter_hbb/desktop/widgets/refresh_wrapper.dart';
|
2023-02-03 17:08:40 +08:00
|
|
|
import 'package:flutter_hbb/models/state_model.dart';
|
2022-05-29 17:19:50 +08:00
|
|
|
import 'package:flutter_hbb/utils/multi_window_manager.dart';
|
2022-09-19 15:46:09 +08:00
|
|
|
import 'package:flutter_localizations/flutter_localizations.dart';
|
2022-07-29 16:47:24 +08:00
|
|
|
import 'package:get/get.dart';
|
2020-11-15 20:04:05 +08:00
|
|
|
import 'package:provider/provider.dart';
|
2022-08-09 13:39:30 +08:00
|
|
|
import 'package:window_manager/window_manager.dart';
|
2022-07-27 14:29:47 +08:00
|
|
|
|
2022-03-07 22:54:34 +08:00
|
|
|
import 'common.dart';
|
2022-08-03 22:03:31 +08:00
|
|
|
import 'consts.dart';
|
2022-05-24 23:33:00 +08:00
|
|
|
import 'mobile/pages/home_page.dart';
|
|
|
|
import 'mobile/pages/server_page.dart';
|
2022-08-09 13:50:26 +08:00
|
|
|
import 'models/platform_model.dart';
|
2022-02-02 17:25:56 +08:00
|
|
|
|
2024-03-22 13:16:37 +08:00
|
|
|
import 'package:flutter_hbb/plugin/handlers.dart'
|
|
|
|
if (dart.library.html) 'package:flutter_hbb/web/plugin/handlers.dart';
|
|
|
|
|
2023-01-23 22:07:50 +08:00
|
|
|
/// Basic window and launch properties.
|
|
|
|
int? kWindowId;
|
|
|
|
WindowType? kWindowType;
|
|
|
|
late List<String> kBootArgs;
|
2022-05-29 17:19:50 +08:00
|
|
|
|
2022-09-16 17:14:32 +08:00
|
|
|
Future<void> main(List<String> args) async {
|
2020-11-29 14:00:59 +08:00
|
|
|
WidgetsFlutterBinding.ensureInitialized();
|
2024-03-22 13:16:37 +08:00
|
|
|
|
2022-09-16 17:14:32 +08:00
|
|
|
debugPrint("launch args: $args");
|
2023-01-23 22:07:50 +08:00
|
|
|
kBootArgs = List.from(args);
|
2022-05-29 17:19:50 +08:00
|
|
|
|
|
|
|
if (!isDesktop) {
|
2022-08-23 14:12:30 +08:00
|
|
|
runMobileApp();
|
2022-05-29 17:19:50 +08:00
|
|
|
return;
|
|
|
|
}
|
2022-05-31 12:09:47 +08:00
|
|
|
// main window
|
2022-05-29 17:19:50 +08:00
|
|
|
if (args.isNotEmpty && args.first == 'multi_window') {
|
2023-01-23 22:07:50 +08:00
|
|
|
kWindowId = int.parse(args[1]);
|
|
|
|
stateGlobal.setWindowId(kWindowId!);
|
2024-03-24 11:23:06 +08:00
|
|
|
if (!isMacOS) {
|
2023-01-23 22:07:50 +08:00
|
|
|
WindowController.fromWindowId(kWindowId!).showTitleBar(false);
|
2022-11-22 23:01:42 +08:00
|
|
|
}
|
2022-05-29 17:19:50 +08:00
|
|
|
final argument = args[2].isEmpty
|
2022-09-16 17:14:32 +08:00
|
|
|
? <String, dynamic>{}
|
2022-05-29 17:19:50 +08:00
|
|
|
: jsonDecode(args[2]) as Map<String, dynamic>;
|
|
|
|
int type = argument['type'] ?? -1;
|
2022-11-01 17:01:43 +08:00
|
|
|
// to-do: No need to parse window id ?
|
|
|
|
// Because stateGlobal.windowId is a global value.
|
2023-01-23 22:07:50 +08:00
|
|
|
argument['windowId'] = kWindowId;
|
|
|
|
kWindowType = type.windowType;
|
|
|
|
switch (kWindowType) {
|
2022-05-29 17:19:50 +08:00
|
|
|
case WindowType.RemoteDesktop:
|
2022-09-05 16:01:53 +08:00
|
|
|
desktopType = DesktopType.remote;
|
2022-11-02 22:23:23 +08:00
|
|
|
runMultiWindow(
|
|
|
|
argument,
|
|
|
|
kAppTypeDesktopRemote,
|
|
|
|
);
|
2022-05-29 17:19:50 +08:00
|
|
|
break;
|
2022-06-17 22:57:41 +08:00
|
|
|
case WindowType.FileTransfer:
|
2022-09-05 16:01:53 +08:00
|
|
|
desktopType = DesktopType.fileTransfer;
|
2022-11-02 22:23:23 +08:00
|
|
|
runMultiWindow(
|
|
|
|
argument,
|
|
|
|
kAppTypeDesktopFileTransfer,
|
|
|
|
);
|
2022-06-17 22:57:41 +08:00
|
|
|
break;
|
2022-08-26 11:35:28 +08:00
|
|
|
case WindowType.PortForward:
|
2022-09-08 17:22:24 +08:00
|
|
|
desktopType = DesktopType.portForward;
|
2022-11-02 22:23:23 +08:00
|
|
|
runMultiWindow(
|
|
|
|
argument,
|
|
|
|
kAppTypeDesktopPortForward,
|
|
|
|
);
|
2022-08-26 11:35:28 +08:00
|
|
|
break;
|
2022-05-29 17:19:50 +08:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2022-08-11 18:59:26 +08:00
|
|
|
} else if (args.isNotEmpty && args.first == '--cm') {
|
2022-09-16 17:14:32 +08:00
|
|
|
debugPrint("--cm started");
|
2022-09-05 16:01:53 +08:00
|
|
|
desktopType = DesktopType.cm;
|
2022-08-11 18:59:26 +08:00
|
|
|
await windowManager.ensureInitialized();
|
2023-11-08 11:28:59 +08:00
|
|
|
runConnectionManagerScreen();
|
2022-10-09 20:32:28 +08:00
|
|
|
} else if (args.contains('--install')) {
|
|
|
|
runInstallPage();
|
2022-05-29 17:19:50 +08:00
|
|
|
} else {
|
2022-09-05 16:01:53 +08:00
|
|
|
desktopType = DesktopType.main;
|
2022-08-09 13:39:30 +08:00
|
|
|
await windowManager.ensureInitialized();
|
|
|
|
windowManager.setPreventClose(true);
|
2022-08-03 22:03:31 +08:00
|
|
|
runMainApp(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-27 17:29:12 +08:00
|
|
|
Future<void> initEnv(String appType) async {
|
2022-09-16 12:14:03 +08:00
|
|
|
// global shared preference
|
2024-02-27 17:29:12 +08:00
|
|
|
await platformFFI.init(appType);
|
2022-08-03 22:03:31 +08:00
|
|
|
// global FFI, use this **ONLY** for global configuration
|
|
|
|
// for convenience, use global FFI on mobile platform
|
|
|
|
// focus on multi-ffi on desktop first
|
|
|
|
await initGlobalFFI();
|
|
|
|
// await Firebase.initializeApp();
|
2022-09-08 08:52:56 +08:00
|
|
|
_registerEventHandler();
|
2023-02-02 13:57:20 +08:00
|
|
|
// Update the system theme.
|
|
|
|
updateSystemWindowTheme();
|
2022-08-03 22:03:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void runMainApp(bool startService) async {
|
2022-10-18 10:29:33 +08:00
|
|
|
// register uni links
|
2024-02-27 17:29:12 +08:00
|
|
|
await initEnv(kAppTypeMain);
|
2022-08-30 16:50:25 +08:00
|
|
|
// trigger connection status updater
|
|
|
|
await bind.mainCheckConnectStatus();
|
2022-08-03 22:03:31 +08:00
|
|
|
if (startService) {
|
2022-06-13 21:07:26 +08:00
|
|
|
gFFI.serverModel.startService();
|
2023-04-23 20:53:51 +08:00
|
|
|
bind.pluginSyncUi(syncTo: kAppTypeMain);
|
2023-05-09 22:59:38 +08:00
|
|
|
bind.pluginListReload();
|
2022-05-25 00:28:59 +08:00
|
|
|
}
|
2023-09-24 17:56:35 +08:00
|
|
|
await Future.wait([gFFI.abModel.loadCache(), gFFI.groupModel.loadCache()]);
|
2022-12-11 21:40:35 +08:00
|
|
|
gFFI.userModel.refreshCurrentUser();
|
2022-08-03 22:03:31 +08:00
|
|
|
runApp(App());
|
2024-03-28 11:38:11 +08:00
|
|
|
|
2023-02-01 14:03:55 +08:00
|
|
|
// Set window option.
|
2022-09-16 12:14:03 +08:00
|
|
|
WindowOptions windowOptions = getHiddenTitleBarWindowOptions();
|
2022-08-30 20:48:03 +08:00
|
|
|
windowManager.waitUntilReadyToShow(windowOptions, () async {
|
2023-02-01 14:03:55 +08:00
|
|
|
// Restore the location of the main window before window hide or show.
|
|
|
|
await restoreWindowPosition(WindowType.Main);
|
|
|
|
// Check the startup argument, if we successfully handle the argument, we keep the main window hidden.
|
2023-02-03 18:52:22 +08:00
|
|
|
final handledByUniLinks = await initUniLinks();
|
2023-04-11 09:56:35 +08:00
|
|
|
debugPrint("handled by uni links: $handledByUniLinks");
|
2023-07-07 12:22:39 +08:00
|
|
|
if (handledByUniLinks || handleUriLink(cmdArgs: kBootArgs)) {
|
2023-02-01 14:03:55 +08:00
|
|
|
windowManager.hide();
|
|
|
|
} else {
|
|
|
|
windowManager.show();
|
|
|
|
windowManager.focus();
|
|
|
|
// Move registration of active main window here to prevent from async visible check.
|
|
|
|
rustDeskWinManager.registerActiveWindow(kWindowMainId);
|
|
|
|
}
|
2022-11-26 11:40:13 +08:00
|
|
|
windowManager.setOpacity(1);
|
2023-02-03 17:08:40 +08:00
|
|
|
windowManager.setTitle(getWindowName());
|
2024-05-10 16:40:29 +08:00
|
|
|
// Do not use `windowManager.setResizable()` here.
|
|
|
|
setResizable(!bind.isIncomingOnly());
|
2022-08-30 20:48:03 +08:00
|
|
|
});
|
2022-08-03 22:03:31 +08:00
|
|
|
}
|
|
|
|
|
2022-08-23 14:12:30 +08:00
|
|
|
void runMobileApp() async {
|
2024-02-27 17:29:12 +08:00
|
|
|
await initEnv(kAppTypeMain);
|
2022-08-23 14:12:30 +08:00
|
|
|
if (isAndroid) androidChannelInit();
|
2024-03-23 10:08:55 +08:00
|
|
|
if (isAndroid) platformFFI.syncAndroidServiceAppDirConfigPath();
|
2023-09-24 17:56:35 +08:00
|
|
|
await Future.wait([gFFI.abModel.loadCache(), gFFI.groupModel.loadCache()]);
|
2023-06-23 18:29:15 +08:00
|
|
|
gFFI.userModel.refreshCurrentUser();
|
2022-08-23 14:12:30 +08:00
|
|
|
runApp(App());
|
2024-03-23 10:08:55 +08:00
|
|
|
if (!isWeb) await initUniLinks();
|
2022-08-23 14:12:30 +08:00
|
|
|
}
|
|
|
|
|
2022-11-02 22:23:23 +08:00
|
|
|
void runMultiWindow(
|
|
|
|
Map<String, dynamic> argument,
|
|
|
|
String appType,
|
|
|
|
) async {
|
2024-02-27 17:29:12 +08:00
|
|
|
await initEnv(appType);
|
2024-02-26 00:02:42 +08:00
|
|
|
final title = getWindowName();
|
2022-11-06 17:39:19 +08:00
|
|
|
// set prevent close to true, we handle close event manually
|
2023-01-23 22:07:50 +08:00
|
|
|
WindowController.fromWindowId(kWindowId!).setPreventClose(true);
|
2022-11-05 23:41:22 +08:00
|
|
|
late Widget widget;
|
|
|
|
switch (appType) {
|
|
|
|
case kAppTypeDesktopRemote:
|
|
|
|
widget = DesktopRemoteScreen(
|
|
|
|
params: argument,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case kAppTypeDesktopFileTransfer:
|
|
|
|
widget = DesktopFileTransferScreen(
|
|
|
|
params: argument,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case kAppTypeDesktopPortForward:
|
|
|
|
widget = DesktopPortForwardScreen(
|
|
|
|
params: argument,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// no such appType
|
|
|
|
exit(0);
|
|
|
|
}
|
2022-11-02 22:23:23 +08:00
|
|
|
_runApp(
|
|
|
|
title,
|
2022-11-05 23:41:22 +08:00
|
|
|
widget,
|
2022-11-02 22:23:23 +08:00
|
|
|
MyTheme.currentThemeMode(),
|
2022-08-26 11:35:28 +08:00
|
|
|
);
|
2022-11-29 23:03:16 +08:00
|
|
|
// we do not hide titlebar on win7 because of the frame overflow.
|
2022-11-30 13:56:02 +08:00
|
|
|
if (kUseCompatibleUiMode) {
|
2023-01-23 22:07:50 +08:00
|
|
|
WindowController.fromWindowId(kWindowId!).showTitleBar(true);
|
2022-11-29 23:03:16 +08:00
|
|
|
}
|
2022-11-09 15:14:11 +08:00
|
|
|
switch (appType) {
|
|
|
|
case kAppTypeDesktopRemote:
|
2023-10-17 00:30:34 +08:00
|
|
|
// If screen rect is set, the window will be moved to the target screen and then set fullscreen.
|
|
|
|
if (argument['screen_rect'] == null) {
|
2023-10-19 07:55:55 +08:00
|
|
|
// display can be used to control the offset of the window.
|
2023-10-19 07:50:59 +08:00
|
|
|
await restoreWindowPosition(
|
|
|
|
WindowType.RemoteDesktop,
|
|
|
|
windowId: kWindowId!,
|
|
|
|
peerId: argument['id'] as String?,
|
|
|
|
display: argument['display'] as int?,
|
|
|
|
);
|
2023-10-17 00:30:34 +08:00
|
|
|
}
|
2022-11-09 15:14:11 +08:00
|
|
|
break;
|
|
|
|
case kAppTypeDesktopFileTransfer:
|
2023-01-23 22:07:50 +08:00
|
|
|
await restoreWindowPosition(WindowType.FileTransfer,
|
|
|
|
windowId: kWindowId!);
|
2022-11-09 15:14:11 +08:00
|
|
|
break;
|
|
|
|
case kAppTypeDesktopPortForward:
|
2023-01-23 22:07:50 +08:00
|
|
|
await restoreWindowPosition(WindowType.PortForward, windowId: kWindowId!);
|
2022-11-09 15:14:11 +08:00
|
|
|
break;
|
|
|
|
default:
|
2023-02-07 01:35:38 +08:00
|
|
|
// no such appType
|
2022-11-09 15:14:11 +08:00
|
|
|
exit(0);
|
|
|
|
}
|
2023-01-07 12:40:29 +08:00
|
|
|
// show window from hidden status
|
2023-01-23 22:07:50 +08:00
|
|
|
WindowController.fromWindowId(kWindowId!).show();
|
2022-08-26 11:35:28 +08:00
|
|
|
}
|
|
|
|
|
2023-11-08 11:28:59 +08:00
|
|
|
void runConnectionManagerScreen() async {
|
2024-02-27 17:29:12 +08:00
|
|
|
await initEnv(kAppTypeConnectionManager);
|
2022-11-02 22:23:23 +08:00
|
|
|
_runApp(
|
|
|
|
'',
|
|
|
|
const DesktopServerPage(),
|
|
|
|
MyTheme.currentThemeMode(),
|
|
|
|
);
|
2023-11-08 11:28:59 +08:00
|
|
|
final hide = await bind.cmGetConfig(name: "hide_cm") == 'true';
|
2023-08-30 11:48:42 +08:00
|
|
|
gFFI.serverModel.hideCm = hide;
|
2022-11-23 09:41:05 +08:00
|
|
|
if (hide) {
|
2023-06-20 12:43:38 +08:00
|
|
|
await hideCmWindow(isStartup: true);
|
2022-11-23 09:41:05 +08:00
|
|
|
} else {
|
2023-06-20 12:43:38 +08:00
|
|
|
await showCmWindow(isStartup: true);
|
2022-11-23 09:41:05 +08:00
|
|
|
}
|
2024-05-10 16:40:29 +08:00
|
|
|
setResizable(false);
|
2023-02-07 01:31:11 +08:00
|
|
|
// Start the uni links handler and redirect links to Native, not for Flutter.
|
2023-06-22 23:19:26 +08:00
|
|
|
listenUniLinks(handleByFlutter: false);
|
2022-11-23 09:41:05 +08:00
|
|
|
}
|
|
|
|
|
2023-09-27 23:51:27 +08:00
|
|
|
bool _isCmReadyToShow = false;
|
|
|
|
|
2023-06-20 12:43:38 +08:00
|
|
|
showCmWindow({bool isStartup = false}) async {
|
|
|
|
if (isStartup) {
|
|
|
|
WindowOptions windowOptions = getHiddenTitleBarWindowOptions(
|
2024-05-08 12:08:37 +08:00
|
|
|
size: kConnectionManagerWindowSizeClosedChat, alwaysOnTop: true);
|
2023-09-27 23:51:27 +08:00
|
|
|
await windowManager.waitUntilReadyToShow(windowOptions, null);
|
|
|
|
bind.mainHideDocker();
|
|
|
|
await Future.wait([
|
|
|
|
windowManager.show(),
|
|
|
|
windowManager.focus(),
|
|
|
|
windowManager.setOpacity(1)
|
|
|
|
]);
|
|
|
|
// ensure initial window size to be changed
|
|
|
|
await windowManager.setSizeAlignment(
|
|
|
|
kConnectionManagerWindowSizeClosedChat, Alignment.topRight);
|
|
|
|
_isCmReadyToShow = true;
|
|
|
|
} else if (_isCmReadyToShow) {
|
2023-06-20 12:43:38 +08:00
|
|
|
if (await windowManager.getOpacity() != 1) {
|
|
|
|
await windowManager.setOpacity(1);
|
|
|
|
await windowManager.focus();
|
|
|
|
await windowManager.minimize(); //needed
|
|
|
|
await windowManager.setSizeAlignment(
|
|
|
|
kConnectionManagerWindowSizeClosedChat, Alignment.topRight);
|
2023-08-02 20:38:09 +08:00
|
|
|
windowOnTop(null);
|
2023-06-20 12:43:38 +08:00
|
|
|
}
|
|
|
|
}
|
2022-08-18 11:07:53 +08:00
|
|
|
}
|
|
|
|
|
2023-06-20 12:43:38 +08:00
|
|
|
hideCmWindow({bool isStartup = false}) async {
|
|
|
|
if (isStartup) {
|
|
|
|
WindowOptions windowOptions = getHiddenTitleBarWindowOptions(
|
|
|
|
size: kConnectionManagerWindowSizeClosedChat);
|
|
|
|
windowManager.setOpacity(0);
|
2023-09-27 23:51:27 +08:00
|
|
|
await windowManager.waitUntilReadyToShow(windowOptions, null);
|
|
|
|
bind.mainHideDocker();
|
|
|
|
await windowManager.minimize();
|
|
|
|
await windowManager.hide();
|
|
|
|
_isCmReadyToShow = true;
|
|
|
|
} else if (_isCmReadyToShow) {
|
2023-08-26 18:49:23 +08:00
|
|
|
if (await windowManager.getOpacity() != 0) {
|
|
|
|
await windowManager.setOpacity(0);
|
|
|
|
bind.mainHideDocker();
|
|
|
|
await windowManager.minimize();
|
|
|
|
await windowManager.hide();
|
|
|
|
}
|
2023-06-20 12:43:38 +08:00
|
|
|
}
|
2022-11-23 09:41:05 +08:00
|
|
|
}
|
|
|
|
|
2022-11-02 22:23:23 +08:00
|
|
|
void _runApp(
|
|
|
|
String title,
|
|
|
|
Widget home,
|
|
|
|
ThemeMode themeMode,
|
|
|
|
) {
|
|
|
|
final botToastBuilder = BotToastInit();
|
2022-10-26 14:39:13 +08:00
|
|
|
runApp(RefreshWrapper(
|
|
|
|
builder: (context) => GetMaterialApp(
|
2022-11-02 22:23:23 +08:00
|
|
|
navigatorKey: globalKey,
|
|
|
|
debugShowCheckedModeBanner: false,
|
|
|
|
title: title,
|
|
|
|
theme: MyTheme.lightTheme,
|
|
|
|
darkTheme: MyTheme.darkTheme,
|
|
|
|
themeMode: themeMode,
|
|
|
|
home: home,
|
|
|
|
localizationsDelegates: const [
|
|
|
|
GlobalMaterialLocalizations.delegate,
|
|
|
|
GlobalWidgetsLocalizations.delegate,
|
|
|
|
GlobalCupertinoLocalizations.delegate,
|
|
|
|
],
|
|
|
|
supportedLocales: supportedLocales,
|
|
|
|
navigatorObservers: [
|
|
|
|
// FirebaseAnalyticsObserver(analytics: analytics),
|
|
|
|
BotToastNavigatorObserver(),
|
|
|
|
],
|
|
|
|
builder: (context, child) {
|
|
|
|
child = _keepScaleBuilder(context, child);
|
|
|
|
child = botToastBuilder(context, child);
|
|
|
|
return child;
|
|
|
|
},
|
|
|
|
),
|
2022-10-26 14:39:13 +08:00
|
|
|
));
|
2022-11-02 22:23:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void runInstallPage() async {
|
|
|
|
await windowManager.ensureInitialized();
|
2024-02-27 17:29:12 +08:00
|
|
|
await initEnv(kAppTypeMain);
|
2023-02-27 17:54:18 +08:00
|
|
|
_runApp('', const InstallPage(), MyTheme.currentThemeMode());
|
2023-03-01 14:18:46 +08:00
|
|
|
WindowOptions windowOptions =
|
|
|
|
getHiddenTitleBarWindowOptions(size: Size(800, 600), center: true);
|
|
|
|
windowManager.waitUntilReadyToShow(windowOptions, () async {
|
2022-10-09 20:32:28 +08:00
|
|
|
windowManager.show();
|
|
|
|
windowManager.focus();
|
|
|
|
windowManager.setOpacity(1);
|
|
|
|
windowManager.setAlignment(Alignment.center); // ensure
|
2023-03-01 14:18:46 +08:00
|
|
|
windowManager.setTitle(getWindowName());
|
2022-10-09 20:32:28 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-03-01 14:18:46 +08:00
|
|
|
WindowOptions getHiddenTitleBarWindowOptions(
|
2024-05-08 12:08:37 +08:00
|
|
|
{Size? size, bool center = false, bool? alwaysOnTop}) {
|
2022-11-29 23:03:16 +08:00
|
|
|
var defaultTitleBarStyle = TitleBarStyle.hidden;
|
|
|
|
// we do not hide titlebar on win7 because of the frame overflow.
|
2022-11-30 13:56:02 +08:00
|
|
|
if (kUseCompatibleUiMode) {
|
2022-11-29 23:03:16 +08:00
|
|
|
defaultTitleBarStyle = TitleBarStyle.normal;
|
|
|
|
}
|
2022-08-18 11:07:53 +08:00
|
|
|
return WindowOptions(
|
|
|
|
size: size,
|
2023-03-01 14:18:46 +08:00
|
|
|
center: center,
|
2022-08-18 11:07:53 +08:00
|
|
|
backgroundColor: Colors.transparent,
|
|
|
|
skipTaskbar: false,
|
2022-11-29 23:03:16 +08:00
|
|
|
titleBarStyle: defaultTitleBarStyle,
|
2024-05-08 12:08:37 +08:00
|
|
|
alwaysOnTop: alwaysOnTop,
|
2022-08-18 11:07:53 +08:00
|
|
|
);
|
2022-08-11 18:59:26 +08:00
|
|
|
}
|
|
|
|
|
2022-09-06 22:34:01 +08:00
|
|
|
class App extends StatefulWidget {
|
|
|
|
@override
|
|
|
|
State<App> createState() => _AppState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _AppState extends State<App> {
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
WidgetsBinding.instance.window.onPlatformBrightnessChanged = () {
|
2022-09-21 23:32:59 +08:00
|
|
|
final userPreference = MyTheme.getThemeModePreference();
|
|
|
|
if (userPreference != ThemeMode.system) return;
|
2022-09-06 22:34:01 +08:00
|
|
|
WidgetsBinding.instance.handlePlatformBrightnessChanged();
|
2022-09-21 23:32:59 +08:00
|
|
|
final systemIsDark =
|
|
|
|
WidgetsBinding.instance.platformDispatcher.platformBrightness ==
|
|
|
|
Brightness.dark;
|
|
|
|
final ThemeMode to;
|
|
|
|
if (systemIsDark) {
|
|
|
|
to = ThemeMode.dark;
|
|
|
|
} else {
|
|
|
|
to = ThemeMode.light;
|
|
|
|
}
|
|
|
|
Get.changeThemeMode(to);
|
2023-02-02 13:57:20 +08:00
|
|
|
// Synchronize the window theme of the system.
|
|
|
|
updateSystemWindowTheme();
|
2022-09-21 23:32:59 +08:00
|
|
|
if (desktopType == DesktopType.main) {
|
|
|
|
bind.mainChangeTheme(dark: to.toShortString());
|
2022-09-06 22:34:01 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-11-06 18:04:04 +08:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2022-05-23 16:24:56 +08:00
|
|
|
// final analytics = FirebaseAnalytics.instance;
|
2022-11-02 22:23:23 +08:00
|
|
|
final botToastBuilder = BotToastInit();
|
2022-10-26 14:39:13 +08:00
|
|
|
return RefreshWrapper(builder: (context) {
|
|
|
|
return MultiProvider(
|
|
|
|
providers: [
|
|
|
|
// global configuration
|
|
|
|
// use session related FFI when in remote control or file transfer page
|
|
|
|
ChangeNotifierProvider.value(value: gFFI.ffiModel),
|
|
|
|
ChangeNotifierProvider.value(value: gFFI.imageModel),
|
|
|
|
ChangeNotifierProvider.value(value: gFFI.cursorModel),
|
|
|
|
ChangeNotifierProvider.value(value: gFFI.canvasModel),
|
2023-02-03 15:07:45 +08:00
|
|
|
ChangeNotifierProvider.value(value: gFFI.peerTabModel),
|
2022-09-19 15:46:09 +08:00
|
|
|
],
|
2022-10-26 14:39:13 +08:00
|
|
|
child: GetMaterialApp(
|
|
|
|
navigatorKey: globalKey,
|
|
|
|
debugShowCheckedModeBanner: false,
|
|
|
|
title: 'RustDesk',
|
|
|
|
theme: MyTheme.lightTheme,
|
|
|
|
darkTheme: MyTheme.darkTheme,
|
|
|
|
themeMode: MyTheme.currentThemeMode(),
|
|
|
|
home: isDesktop
|
|
|
|
? const DesktopTabPage()
|
2023-08-30 21:59:25 +08:00
|
|
|
: isWeb
|
2022-10-26 14:39:13 +08:00
|
|
|
? WebHomePage()
|
|
|
|
: HomePage(),
|
|
|
|
localizationsDelegates: const [
|
|
|
|
GlobalMaterialLocalizations.delegate,
|
|
|
|
GlobalWidgetsLocalizations.delegate,
|
|
|
|
GlobalCupertinoLocalizations.delegate,
|
|
|
|
],
|
|
|
|
supportedLocales: supportedLocales,
|
2022-11-02 22:23:23 +08:00
|
|
|
navigatorObservers: [
|
|
|
|
// FirebaseAnalyticsObserver(analytics: analytics),
|
|
|
|
BotToastNavigatorObserver(),
|
|
|
|
],
|
2022-10-26 14:39:13 +08:00
|
|
|
builder: isAndroid
|
|
|
|
? (context, child) => AccessibilityListener(
|
|
|
|
child: MediaQuery(
|
|
|
|
data: MediaQuery.of(context).copyWith(
|
2024-02-12 21:39:19 +08:00
|
|
|
textScaler: TextScaler.linear(1.0),
|
2022-10-26 14:39:13 +08:00
|
|
|
),
|
|
|
|
child: child ?? Container(),
|
2022-08-28 22:28:19 +08:00
|
|
|
),
|
2022-10-26 14:39:13 +08:00
|
|
|
)
|
2022-11-02 22:23:23 +08:00
|
|
|
: (context, child) {
|
|
|
|
child = _keepScaleBuilder(context, child);
|
|
|
|
child = botToastBuilder(context, child);
|
2023-08-09 22:00:15 +08:00
|
|
|
if (isDesktop && desktopType == DesktopType.main) {
|
2023-08-04 13:11:24 +08:00
|
|
|
child = keyListenerBuilder(context, child);
|
|
|
|
}
|
2024-05-08 09:59:05 +08:00
|
|
|
if (isLinux) {
|
|
|
|
child = buildVirtualWindowFrame(context, child);
|
|
|
|
}
|
2022-11-02 22:23:23 +08:00
|
|
|
return child;
|
|
|
|
},
|
2022-10-26 14:39:13 +08:00
|
|
|
),
|
|
|
|
);
|
|
|
|
});
|
2020-11-06 18:04:04 +08:00
|
|
|
}
|
|
|
|
}
|
2022-08-23 19:47:56 +08:00
|
|
|
|
2022-11-02 22:23:23 +08:00
|
|
|
Widget _keepScaleBuilder(BuildContext context, Widget? child) {
|
|
|
|
return MediaQuery(
|
|
|
|
data: MediaQuery.of(context).copyWith(
|
2024-02-12 21:39:19 +08:00
|
|
|
textScaler: TextScaler.linear(1.0),
|
2022-11-02 22:23:23 +08:00
|
|
|
),
|
|
|
|
child: child ?? Container(),
|
|
|
|
);
|
2022-08-23 19:47:56 +08:00
|
|
|
}
|
2022-09-08 08:52:56 +08:00
|
|
|
|
|
|
|
_registerEventHandler() {
|
2022-09-15 16:09:07 +08:00
|
|
|
if (isDesktop && desktopType != DesktopType.main) {
|
2022-09-12 10:52:38 +08:00
|
|
|
platformFFI.registerEventHandler('theme', 'theme', (evt) async {
|
2022-09-08 08:52:56 +08:00
|
|
|
String? dark = evt['dark'];
|
|
|
|
if (dark != null) {
|
2022-09-21 23:32:59 +08:00
|
|
|
MyTheme.changeDarkMode(MyTheme.themeModeFromString(dark));
|
2022-09-08 08:52:56 +08:00
|
|
|
}
|
|
|
|
});
|
2022-09-12 10:52:38 +08:00
|
|
|
platformFFI.registerEventHandler('language', 'language', (_) async {
|
2022-10-26 14:39:13 +08:00
|
|
|
reloadAllWindows();
|
2022-09-08 08:52:56 +08:00
|
|
|
});
|
|
|
|
}
|
2023-05-06 02:29:01 +08:00
|
|
|
// Register native handlers.
|
|
|
|
if (isDesktop) {
|
|
|
|
platformFFI.registerEventHandler('native_ui', 'native_ui', (evt) async {
|
|
|
|
NativeUiHandler.instance.onEvent(evt);
|
|
|
|
});
|
|
|
|
}
|
2022-09-08 08:52:56 +08:00
|
|
|
}
|
2023-08-04 13:11:24 +08:00
|
|
|
|
|
|
|
Widget keyListenerBuilder(BuildContext context, Widget? child) {
|
|
|
|
return RawKeyboardListener(
|
|
|
|
focusNode: FocusNode(),
|
|
|
|
child: child ?? Container(),
|
|
|
|
onKey: (RawKeyEvent event) {
|
|
|
|
if (event.logicalKey == LogicalKeyboardKey.shiftLeft) {
|
|
|
|
if (event is RawKeyDownEvent) {
|
2023-08-09 22:00:15 +08:00
|
|
|
gFFI.peerTabModel.setShiftDown(true);
|
2023-08-04 13:11:24 +08:00
|
|
|
} else if (event is RawKeyUpEvent) {
|
2023-08-09 22:00:15 +08:00
|
|
|
gFFI.peerTabModel.setShiftDown(false);
|
2023-08-04 13:11:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|