2023-02-06 20:31:11 +03:00
import ' dart:async ' ;
2022-05-29 12:19:50 +03:00
import ' dart:convert ' ;
2022-11-05 18:41:22 +03:00
import ' dart:io ' ;
2022-05-29 12:19:50 +03:00
2023-02-03 12:08:40 +03:00
import ' package:bot_toast/bot_toast.dart ' ;
2022-08-18 12:25:47 +03:00
import ' package:desktop_multi_window/desktop_multi_window.dart ' ;
2020-11-06 13:04:04 +03:00
import ' package:flutter/material.dart ' ;
2022-08-11 11:03:04 +03:00
import ' package:flutter_hbb/desktop/pages/desktop_tab_page.dart ' ;
2022-10-09 15:32:28 +03:00
import ' package:flutter_hbb/desktop/pages/install_page.dart ' ;
2023-02-03 12:08:40 +03:00
import ' package:flutter_hbb/desktop/pages/server_page.dart ' ;
2022-06-17 17:57:41 +03:00
import ' package:flutter_hbb/desktop/screen/desktop_file_transfer_screen.dart ' ;
2022-08-26 06:35:28 +03:00
import ' package:flutter_hbb/desktop/screen/desktop_port_forward_screen.dart ' ;
2022-05-29 12:19:50 +03:00
import ' package:flutter_hbb/desktop/screen/desktop_remote_screen.dart ' ;
2022-10-26 09:39:13 +03:00
import ' package:flutter_hbb/desktop/widgets/refresh_wrapper.dart ' ;
2023-02-03 12:08:40 +03:00
import ' package:flutter_hbb/models/state_model.dart ' ;
2022-05-29 12:19:50 +03:00
import ' package:flutter_hbb/utils/multi_window_manager.dart ' ;
2022-09-19 10:46:09 +03:00
import ' package:flutter_localizations/flutter_localizations.dart ' ;
2022-07-29 11:47:24 +03:00
import ' package:get/get.dart ' ;
2020-11-15 15:04:05 +03:00
import ' package:provider/provider.dart ' ;
2022-08-09 08:39:30 +03:00
import ' package:window_manager/window_manager.dart ' ;
2022-07-27 09:29:47 +03:00
2022-06-27 04:48:35 +03:00
// import 'package:window_manager/window_manager.dart';
2022-05-24 19:28:59 +03:00
2022-03-07 17:54:34 +03:00
import ' common.dart ' ;
2022-08-03 17:03:31 +03:00
import ' consts.dart ' ;
2022-05-24 18:33:00 +03:00
import ' mobile/pages/home_page.dart ' ;
import ' mobile/pages/server_page.dart ' ;
2022-08-09 08:50:26 +03:00
import ' models/platform_model.dart ' ;
2022-02-02 12:25:56 +03:00
2023-01-23 17:07:50 +03:00
/// Basic window and launch properties.
int ? kWindowId ;
WindowType ? kWindowType ;
late List < String > kBootArgs ;
2022-05-29 12:19:50 +03:00
2023-02-06 20:31:11 +03:00
/// Uni links.
StreamSubscription ? _uniLinkSubscription ;
2022-09-16 12:14:32 +03:00
Future < void > main ( List < String > args ) async {
2020-11-29 09:00:59 +03:00
WidgetsFlutterBinding . ensureInitialized ( ) ;
2022-09-16 12:14:32 +03:00
debugPrint ( " launch args: $ args " ) ;
2023-01-23 17:07:50 +03:00
kBootArgs = List . from ( args ) ;
2022-05-29 12:19:50 +03:00
if ( ! isDesktop ) {
2022-08-23 09:12:30 +03:00
runMobileApp ( ) ;
2022-05-29 12:19:50 +03:00
return ;
}
2022-05-31 07:09:47 +03:00
// main window
2022-05-29 12:19:50 +03:00
if ( args . isNotEmpty & & args . first = = ' multi_window ' ) {
2023-01-23 17:07:50 +03:00
kWindowId = int . parse ( args [ 1 ] ) ;
stateGlobal . setWindowId ( kWindowId ! ) ;
2022-11-22 18:01:42 +03:00
if ( ! Platform . isMacOS ) {
2023-01-23 17:07:50 +03:00
WindowController . fromWindowId ( kWindowId ! ) . showTitleBar ( false ) ;
2022-11-22 18:01:42 +03:00
}
2022-05-29 12:19:50 +03:00
final argument = args [ 2 ] . isEmpty
2022-09-16 12:14:32 +03:00
? < String , dynamic > { }
2022-05-29 12:19:50 +03:00
: jsonDecode ( args [ 2 ] ) as Map < String , dynamic > ;
int type = argument [ ' type ' ] ? ? - 1 ;
2022-11-01 12:01:43 +03:00
// to-do: No need to parse window id ?
// Because stateGlobal.windowId is a global value.
2023-01-23 17:07:50 +03:00
argument [ ' windowId ' ] = kWindowId ;
kWindowType = type . windowType ;
final windowName = getWindowName ( ) ;
switch ( kWindowType ) {
2022-05-29 12:19:50 +03:00
case WindowType . RemoteDesktop:
2022-09-05 11:01:53 +03:00
desktopType = DesktopType . remote ;
2022-11-02 17:23:23 +03:00
runMultiWindow (
argument ,
kAppTypeDesktopRemote ,
2023-01-23 17:07:50 +03:00
windowName ,
2022-11-02 17:23:23 +03:00
) ;
2022-05-29 12:19:50 +03:00
break ;
2022-06-17 17:57:41 +03:00
case WindowType . FileTransfer:
2022-09-05 11:01:53 +03:00
desktopType = DesktopType . fileTransfer ;
2022-11-02 17:23:23 +03:00
runMultiWindow (
argument ,
kAppTypeDesktopFileTransfer ,
2023-01-23 17:07:50 +03:00
windowName ,
2022-11-02 17:23:23 +03:00
) ;
2022-06-17 17:57:41 +03:00
break ;
2022-08-26 06:35:28 +03:00
case WindowType . PortForward:
2022-09-08 12:22:24 +03:00
desktopType = DesktopType . portForward ;
2022-11-02 17:23:23 +03:00
runMultiWindow (
argument ,
kAppTypeDesktopPortForward ,
2023-01-23 17:07:50 +03:00
windowName ,
2022-11-02 17:23:23 +03:00
) ;
2022-08-26 06:35:28 +03:00
break ;
2022-05-29 12:19:50 +03:00
default :
break ;
}
2022-08-11 13:59:26 +03:00
} else if ( args . isNotEmpty & & args . first = = ' --cm ' ) {
2022-09-16 12:14:32 +03:00
debugPrint ( " --cm started " ) ;
2022-09-05 11:01:53 +03:00
desktopType = DesktopType . cm ;
2022-08-11 13:59:26 +03:00
await windowManager . ensureInitialized ( ) ;
2022-11-23 04:41:05 +03:00
runConnectionManagerScreen ( args . contains ( ' --hide ' ) ) ;
2022-10-09 15:32:28 +03:00
} else if ( args . contains ( ' --install ' ) ) {
runInstallPage ( ) ;
2022-05-29 12:19:50 +03:00
} else {
2022-09-05 11:01:53 +03:00
desktopType = DesktopType . main ;
2022-08-09 08:39:30 +03:00
await windowManager . ensureInitialized ( ) ;
windowManager . setPreventClose ( true ) ;
2022-08-03 17:03:31 +03:00
runMainApp ( true ) ;
}
}
Future < void > initEnv ( String appType ) async {
2022-09-16 07:14:03 +03:00
// global shared preference
2022-08-03 17:03:31 +03:00
await platformFFI . init ( appType ) ;
// 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 03:52:56 +03:00
_registerEventHandler ( ) ;
2023-02-02 08:57:20 +03:00
// Update the system theme.
updateSystemWindowTheme ( ) ;
2023-02-07 13:01:54 +03:00
if ( appType = = kAppTypeConnectionManager ) {
await bind . cmStartListenIpcThread ( ) ;
}
2022-08-03 17:03:31 +03:00
}
void runMainApp ( bool startService ) async {
2022-10-18 05:29:33 +03:00
// register uni links
2022-08-30 15:48:03 +03:00
await initEnv ( kAppTypeMain ) ;
2022-08-30 11:50:25 +03:00
// trigger connection status updater
await bind . mainCheckConnectStatus ( ) ;
2022-08-03 17:03:31 +03:00
if ( startService ) {
2022-06-27 04:48:35 +03:00
// await windowManager.ensureInitialized();
2022-06-13 16:07:26 +03:00
gFFI . serverModel . startService ( ) ;
2022-05-24 19:28:59 +03:00
}
2022-12-11 16:40:35 +03:00
gFFI . userModel . refreshCurrentUser ( ) ;
2022-08-03 17:03:31 +03:00
runApp ( App ( ) ) ;
2023-02-01 09:03:55 +03:00
// Set window option.
2022-09-16 07:14:03 +03:00
WindowOptions windowOptions = getHiddenTitleBarWindowOptions ( ) ;
2022-08-30 15:48:03 +03:00
windowManager . waitUntilReadyToShow ( windowOptions , ( ) async {
2023-02-01 09:03:55 +03: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 13:52:22 +03:00
final handledByUniLinks = await initUniLinks ( ) ;
final handledByCli = checkArguments ( ) ;
debugPrint (
" handled by uni links: $ handledByUniLinks , handled by cli: $ handledByCli " ) ;
if ( handledByUniLinks | | handledByCli ) {
2023-02-01 09:03:55 +03: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 06:40:13 +03:00
windowManager . setOpacity ( 1 ) ;
2023-02-03 12:08:40 +03:00
windowManager . setTitle ( getWindowName ( ) ) ;
2022-08-30 15:48:03 +03:00
} ) ;
2022-08-03 17:03:31 +03:00
}
2022-08-23 09:12:30 +03:00
void runMobileApp ( ) async {
await initEnv ( kAppTypeMain ) ;
if ( isAndroid ) androidChannelInit ( ) ;
runApp ( App ( ) ) ;
}
2022-11-02 17:23:23 +03:00
void runMultiWindow (
Map < String , dynamic > argument ,
String appType ,
String title ,
) async {
await initEnv ( appType ) ;
2022-11-06 12:39:19 +03:00
// set prevent close to true, we handle close event manually
2023-01-23 17:07:50 +03:00
WindowController . fromWindowId ( kWindowId ! ) . setPreventClose ( true ) ;
2022-11-05 18:41:22 +03: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 17:23:23 +03:00
_runApp (
title ,
2022-11-05 18:41:22 +03:00
widget ,
2022-11-02 17:23:23 +03:00
MyTheme . currentThemeMode ( ) ,
2022-08-26 06:35:28 +03:00
) ;
2022-11-29 18:03:16 +03:00
// we do not hide titlebar on win7 because of the frame overflow.
2022-11-30 08:56:02 +03:00
if ( kUseCompatibleUiMode ) {
2023-01-23 17:07:50 +03:00
WindowController . fromWindowId ( kWindowId ! ) . showTitleBar ( true ) ;
2022-11-29 18:03:16 +03:00
}
2022-11-09 10:14:11 +03:00
switch ( appType ) {
case kAppTypeDesktopRemote:
2022-11-22 18:01:42 +03:00
await restoreWindowPosition ( WindowType . RemoteDesktop ,
2023-01-23 17:07:50 +03:00
windowId: kWindowId ! ) ;
2022-11-09 10:14:11 +03:00
break ;
case kAppTypeDesktopFileTransfer:
2023-01-23 17:07:50 +03:00
await restoreWindowPosition ( WindowType . FileTransfer ,
windowId: kWindowId ! ) ;
2022-11-09 10:14:11 +03:00
break ;
case kAppTypeDesktopPortForward:
2023-01-23 17:07:50 +03:00
await restoreWindowPosition ( WindowType . PortForward , windowId: kWindowId ! ) ;
2022-11-09 10:14:11 +03:00
break ;
default :
2023-02-06 20:35:38 +03:00
// no such appType
2022-11-09 10:14:11 +03:00
exit ( 0 ) ;
}
2023-01-07 07:40:29 +03:00
// show window from hidden status
2023-01-23 17:07:50 +03:00
WindowController . fromWindowId ( kWindowId ! ) . show ( ) ;
2022-08-26 06:35:28 +03:00
}
2022-11-23 04:41:05 +03:00
void runConnectionManagerScreen ( bool hide ) async {
2023-02-04 06:23:36 +03:00
await initEnv ( kAppTypeConnectionManager ) ;
2022-11-02 17:23:23 +03:00
_runApp (
' ' ,
const DesktopServerPage ( ) ,
MyTheme . currentThemeMode ( ) ,
) ;
2022-11-23 04:41:05 +03:00
if ( hide ) {
hideCmWindow ( ) ;
} else {
showCmWindow ( ) ;
}
2023-02-06 20:31:11 +03:00
// Start the uni links handler and redirect links to Native, not for Flutter.
_uniLinkSubscription = listenUniLinks ( handleByFlutter: false ) ;
2022-11-23 04:41:05 +03:00
}
void showCmWindow ( ) {
WindowOptions windowOptions =
getHiddenTitleBarWindowOptions ( size: kConnectionManagerWindowSize ) ;
2022-09-16 11:19:15 +03:00
windowManager . waitUntilReadyToShow ( windowOptions , ( ) async {
2023-02-01 12:36:05 +03:00
bind . mainHideDocker ( ) ;
2022-10-17 06:53:15 +03:00
await windowManager . show ( ) ;
2022-11-05 05:35:56 +03:00
await Future . wait ( [ windowManager . focus ( ) , windowManager . setOpacity ( 1 ) ] ) ;
2022-10-17 06:53:15 +03:00
// ensure initial window size to be changed
2022-11-05 05:35:56 +03:00
await windowManager . setSizeAlignment (
kConnectionManagerWindowSize , Alignment . topRight ) ;
2022-09-19 10:46:09 +03:00
} ) ;
2022-08-18 06:07:53 +03:00
}
2022-11-23 04:41:05 +03:00
void hideCmWindow ( ) {
WindowOptions windowOptions =
getHiddenTitleBarWindowOptions ( size: kConnectionManagerWindowSize ) ;
windowManager . setOpacity ( 0 ) ;
windowManager . waitUntilReadyToShow ( windowOptions , ( ) async {
2023-02-01 12:36:05 +03:00
bind . mainHideDocker ( ) ;
2022-11-23 04:41:05 +03:00
await windowManager . hide ( ) ;
} ) ;
}
2022-11-02 17:23:23 +03:00
void _runApp (
String title ,
Widget home ,
ThemeMode themeMode ,
) {
final botToastBuilder = BotToastInit ( ) ;
2022-10-26 09:39:13 +03:00
runApp ( RefreshWrapper (
builder: ( context ) = > GetMaterialApp (
2022-11-02 17:23:23 +03: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 09:39:13 +03:00
) ) ;
2022-11-02 17:23:23 +03:00
}
void runInstallPage ( ) async {
await windowManager . ensureInitialized ( ) ;
await initEnv ( kAppTypeMain ) ;
_runApp ( ' ' , const InstallPage ( ) , ThemeMode . light ) ;
2022-10-09 15:32:28 +03:00
windowManager . waitUntilReadyToShow (
WindowOptions ( size: Size ( 800 , 600 ) , center: true ) , ( ) async {
windowManager . show ( ) ;
windowManager . focus ( ) ;
windowManager . setOpacity ( 1 ) ;
windowManager . setAlignment ( Alignment . center ) ; // ensure
} ) ;
}
2022-09-16 07:14:03 +03:00
WindowOptions getHiddenTitleBarWindowOptions ( { Size ? size } ) {
2022-11-29 18:03:16 +03:00
var defaultTitleBarStyle = TitleBarStyle . hidden ;
// we do not hide titlebar on win7 because of the frame overflow.
2022-11-30 08:56:02 +03:00
if ( kUseCompatibleUiMode ) {
2022-11-29 18:03:16 +03:00
defaultTitleBarStyle = TitleBarStyle . normal ;
}
2022-08-18 06:07:53 +03:00
return WindowOptions (
size: size ,
2022-09-26 12:56:32 +03:00
center: false ,
2022-08-18 06:07:53 +03:00
backgroundColor: Colors . transparent ,
skipTaskbar: false ,
2022-11-29 18:03:16 +03:00
titleBarStyle: defaultTitleBarStyle ,
2022-08-18 06:07:53 +03:00
) ;
2022-08-11 13:59:26 +03:00
}
2022-09-06 17:34:01 +03: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 18:32:59 +03:00
final userPreference = MyTheme . getThemeModePreference ( ) ;
if ( userPreference ! = ThemeMode . system ) return ;
2022-09-06 17:34:01 +03:00
WidgetsBinding . instance . handlePlatformBrightnessChanged ( ) ;
2022-09-21 18:32:59 +03: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 08:57:20 +03:00
// Synchronize the window theme of the system.
updateSystemWindowTheme ( ) ;
2022-09-21 18:32:59 +03:00
if ( desktopType = = DesktopType . main ) {
bind . mainChangeTheme ( dark: to . toShortString ( ) ) ;
2022-09-06 17:34:01 +03:00
}
} ;
}
2020-11-06 13:04:04 +03:00
@ override
Widget build ( BuildContext context ) {
2022-05-23 11:24:56 +03:00
// final analytics = FirebaseAnalytics.instance;
2022-11-02 17:23:23 +03:00
final botToastBuilder = BotToastInit ( ) ;
2022-10-26 09:39:13 +03: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 10:07:45 +03:00
ChangeNotifierProvider . value ( value: gFFI . peerTabModel ) ,
2022-09-19 10:46:09 +03:00
] ,
2022-10-26 09:39:13 +03:00
child: GetMaterialApp (
navigatorKey: globalKey ,
debugShowCheckedModeBanner: false ,
title: ' RustDesk ' ,
theme: MyTheme . lightTheme ,
darkTheme: MyTheme . darkTheme ,
themeMode: MyTheme . currentThemeMode ( ) ,
home: isDesktop
? const DesktopTabPage ( )
: ! isAndroid
? WebHomePage ( )
: HomePage ( ) ,
localizationsDelegates: const [
GlobalMaterialLocalizations . delegate ,
GlobalWidgetsLocalizations . delegate ,
GlobalCupertinoLocalizations . delegate ,
] ,
supportedLocales: supportedLocales ,
2022-11-02 17:23:23 +03:00
navigatorObservers: [
// FirebaseAnalyticsObserver(analytics: analytics),
BotToastNavigatorObserver ( ) ,
] ,
2022-10-26 09:39:13 +03:00
builder: isAndroid
? ( context , child ) = > AccessibilityListener (
child: MediaQuery (
data: MediaQuery . of ( context ) . copyWith (
textScaleFactor: 1.0 ,
) ,
child: child ? ? Container ( ) ,
2022-08-28 17:28:19 +03:00
) ,
2022-10-26 09:39:13 +03:00
)
2022-11-02 17:23:23 +03:00
: ( context , child ) {
child = _keepScaleBuilder ( context , child ) ;
child = botToastBuilder ( context , child ) ;
return child ;
} ,
2022-10-26 09:39:13 +03:00
) ,
) ;
} ) ;
2020-11-06 13:04:04 +03:00
}
}
2022-08-23 14:47:56 +03:00
2022-11-02 17:23:23 +03:00
Widget _keepScaleBuilder ( BuildContext context , Widget ? child ) {
return MediaQuery (
data: MediaQuery . of ( context ) . copyWith (
textScaleFactor: 1.0 ,
) ,
child: child ? ? Container ( ) ,
) ;
2022-08-23 14:47:56 +03:00
}
2022-09-08 03:52:56 +03:00
_registerEventHandler ( ) {
2022-09-15 11:09:07 +03:00
if ( isDesktop & & desktopType ! = DesktopType . main ) {
2022-09-12 05:52:38 +03:00
platformFFI . registerEventHandler ( ' theme ' , ' theme ' , ( evt ) async {
2022-09-08 03:52:56 +03:00
String ? dark = evt [ ' dark ' ] ;
if ( dark ! = null ) {
2022-09-21 18:32:59 +03:00
MyTheme . changeDarkMode ( MyTheme . themeModeFromString ( dark ) ) ;
2022-09-08 03:52:56 +03:00
}
} ) ;
2022-09-12 05:52:38 +03:00
platformFFI . registerEventHandler ( ' language ' , ' language ' , ( _ ) async {
2022-10-26 09:39:13 +03:00
reloadAllWindows ( ) ;
2022-09-08 03:52:56 +03:00
} ) ;
}
}