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 ' ;
2024-06-23 11:06:47 +08:00
import ' package:flutter_hbb/common/widgets/overlay.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 ) ;
2024-06-05 14:52:56 +08:00
if ( isMacOS ) {
disableWindowMovable ( kWindowId ) ;
}
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 ( ) ;
2024-06-23 11:06:47 +08:00
draggablePositions . load ( ) ;
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 ) ;
2024-06-05 14:52:56 +08:00
if ( isMacOS ) {
disableWindowMovable ( kWindowId ) ;
}
2022-11-05 23:41:22 +08:00
late Widget widget ;
switch ( appType ) {
case kAppTypeDesktopRemote:
2024-06-23 11:06:47 +08:00
draggablePositions . load ( ) ;
2022-11-05 23:41:22 +08:00
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-06 20: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 ) ;
2024-08-16 12:20:40 +08:00
bind . mainHideDock ( ) ;
2023-09-27 23:51:27 +08:00
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 ) ;
2024-08-16 12:20:40 +08:00
bind . mainHideDock ( ) ;
2023-09-27 23:51:27 +08:00
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 ) ;
2024-08-16 12:20:40 +08:00
bind . mainHideDock ( ) ;
2023-08-26 18:49:23 +08:00
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
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 ( ) ;
}
2024-09-03 19:06:11 +08:00
class _AppState extends State < App > with WidgetsBindingObserver {
2022-09-06 22:34:01 +08:00
@ 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
}
} ;
2024-09-03 19:06:11 +08:00
WidgetsBinding . instance . addObserver ( this ) ;
WidgetsBinding . instance . addPostFrameCallback ( ( _ ) = > _updateOrientation ( ) ) ;
}
@ override
void dispose ( ) {
WidgetsBinding . instance . removeObserver ( this ) ;
super . dispose ( ) ;
}
@ override
void didChangeMetrics ( ) {
_updateOrientation ( ) ;
}
void _updateOrientation ( ) {
if ( isDesktop ) return ;
// Don't use `MediaQuery.of(context).orientation` in `didChangeMetrics()`,
// my test (Flutter 3.19.6, Android 14) is always the reverse value.
// https://github.com/flutter/flutter/issues/60899
// stateGlobal.isPortrait.value =
// MediaQuery.of(context).orientation == Orientation.portrait;
final orientation = View . of ( context ) . physicalSize . aspectRatio > 1
? Orientation . landscape
: Orientation . portrait ;
stateGlobal . isPortrait . value = orientation = = Orientation . portrait ;
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-11 19:52:38 -07: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-11 19:52:38 -07: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 01:29:01 +07: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
}
}
} ,
) ;
}