285 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			285 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:convert';
 | |
| import 'dart:io';
 | |
| 
 | |
| import 'package:desktop_multi_window/desktop_multi_window.dart';
 | |
| import 'package:flutter/foundation.dart';
 | |
| import 'package:flutter/material.dart';
 | |
| import 'package:flutter/services.dart';
 | |
| import 'package:flutter_hbb/common.dart';
 | |
| 
 | |
| /// must keep the order
 | |
| enum WindowType { Main, RemoteDesktop, FileTransfer, PortForward, Unknown }
 | |
| 
 | |
| extension Index on int {
 | |
|   WindowType get windowType {
 | |
|     switch (this) {
 | |
|       case 0:
 | |
|         return WindowType.Main;
 | |
|       case 1:
 | |
|         return WindowType.RemoteDesktop;
 | |
|       case 2:
 | |
|         return WindowType.FileTransfer;
 | |
|       case 3:
 | |
|         return WindowType.PortForward;
 | |
|       default:
 | |
|         return WindowType.Unknown;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /// Window Manager
 | |
| /// mainly use it in `Main Window`
 | |
| /// use it in sub window is not recommended
 | |
| class RustDeskMultiWindowManager {
 | |
|   RustDeskMultiWindowManager._();
 | |
| 
 | |
|   static final instance = RustDeskMultiWindowManager._();
 | |
| 
 | |
|   final List<int> _activeWindows = List.empty(growable: true);
 | |
|   final List<AsyncCallback> _windowActiveCallbacks = List.empty(growable: true);
 | |
|   int? _remoteDesktopWindowId;
 | |
|   int? _fileTransferWindowId;
 | |
|   int? _portForwardWindowId;
 | |
| 
 | |
|   Future<dynamic> newRemoteDesktop(
 | |
|     String remoteId, {
 | |
|     String? switch_uuid,
 | |
|     bool? forceRelay,
 | |
|   }) async {
 | |
|     var params = {
 | |
|       "type": WindowType.RemoteDesktop.index,
 | |
|       "id": remoteId,
 | |
|       "forceRelay": forceRelay
 | |
|     };
 | |
|     if (switch_uuid != null) {
 | |
|       params['switch_uuid'] = switch_uuid;
 | |
|     }
 | |
|     final msg = jsonEncode(params);
 | |
| 
 | |
|     try {
 | |
|       final ids = await DesktopMultiWindow.getAllSubWindowIds();
 | |
|       if (!ids.contains(_remoteDesktopWindowId)) {
 | |
|         _remoteDesktopWindowId = null;
 | |
|       }
 | |
|     } on Error {
 | |
|       _remoteDesktopWindowId = null;
 | |
|     }
 | |
|     if (_remoteDesktopWindowId == null) {
 | |
|       final remoteDesktopController =
 | |
|           await DesktopMultiWindow.createWindow(msg);
 | |
|       remoteDesktopController
 | |
|         ..setFrame(const Offset(0, 0) & const Size(1280, 720))
 | |
|         ..center()
 | |
|         ..setTitle(getWindowNameWithId(remoteId,
 | |
|             overrideType: WindowType.RemoteDesktop));
 | |
|       if (Platform.isMacOS) {
 | |
|         Future.microtask(() => remoteDesktopController.show());
 | |
|       }
 | |
|       registerActiveWindow(remoteDesktopController.windowId);
 | |
|       _remoteDesktopWindowId = remoteDesktopController.windowId;
 | |
|     } else {
 | |
|       return call(WindowType.RemoteDesktop, "new_remote_desktop", msg);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<dynamic> newFileTransfer(String remoteId, {bool? forceRelay}) async {
 | |
|     var msg = jsonEncode({
 | |
|       "type": WindowType.FileTransfer.index,
 | |
|       "id": remoteId,
 | |
|       "forceRelay": forceRelay,
 | |
|     });
 | |
| 
 | |
|     try {
 | |
|       final ids = await DesktopMultiWindow.getAllSubWindowIds();
 | |
|       if (!ids.contains(_fileTransferWindowId)) {
 | |
|         _fileTransferWindowId = null;
 | |
|       }
 | |
|     } on Error {
 | |
|       _fileTransferWindowId = null;
 | |
|     }
 | |
|     if (_fileTransferWindowId == null) {
 | |
|       final fileTransferController = await DesktopMultiWindow.createWindow(msg);
 | |
|       fileTransferController
 | |
|         ..setFrame(const Offset(0, 0) & const Size(1280, 720))
 | |
|         ..center()
 | |
|         ..setTitle(getWindowNameWithId(remoteId,
 | |
|             overrideType: WindowType.FileTransfer));
 | |
|       if (Platform.isMacOS) {
 | |
|         Future.microtask(() => fileTransferController.show());
 | |
|       }
 | |
|       registerActiveWindow(fileTransferController.windowId);
 | |
|       _fileTransferWindowId = fileTransferController.windowId;
 | |
|     } else {
 | |
|       return call(WindowType.FileTransfer, "new_file_transfer", msg);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<dynamic> newPortForward(String remoteId, bool isRDP,
 | |
|       {bool? forceRelay}) async {
 | |
|     final msg = jsonEncode({
 | |
|       "type": WindowType.PortForward.index,
 | |
|       "id": remoteId,
 | |
|       "isRDP": isRDP,
 | |
|       "forceRelay": forceRelay,
 | |
|     });
 | |
| 
 | |
|     try {
 | |
|       final ids = await DesktopMultiWindow.getAllSubWindowIds();
 | |
|       if (!ids.contains(_portForwardWindowId)) {
 | |
|         _portForwardWindowId = null;
 | |
|       }
 | |
|     } on Error {
 | |
|       _portForwardWindowId = null;
 | |
|     }
 | |
|     if (_portForwardWindowId == null) {
 | |
|       final portForwardController = await DesktopMultiWindow.createWindow(msg);
 | |
|       portForwardController
 | |
|         ..setFrame(const Offset(0, 0) & const Size(1280, 720))
 | |
|         ..center()
 | |
|         ..setTitle(getWindowNameWithId(remoteId,
 | |
|             overrideType: WindowType.PortForward));
 | |
|       if (Platform.isMacOS) {
 | |
|         Future.microtask(() => portForwardController.show());
 | |
|       }
 | |
|       registerActiveWindow(portForwardController.windowId);
 | |
|       _portForwardWindowId = portForwardController.windowId;
 | |
|     } else {
 | |
|       return call(WindowType.PortForward, "new_port_forward", msg);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<dynamic> call(WindowType type, String methodName, dynamic args) async {
 | |
|     int? windowId = findWindowByType(type);
 | |
|     if (windowId == null) {
 | |
|       return;
 | |
|     }
 | |
|     return await DesktopMultiWindow.invokeMethod(windowId, methodName, args);
 | |
|   }
 | |
| 
 | |
|   int? findWindowByType(WindowType type) {
 | |
|     switch (type) {
 | |
|       case WindowType.Main:
 | |
|         return 0;
 | |
|       case WindowType.RemoteDesktop:
 | |
|         return _remoteDesktopWindowId;
 | |
|       case WindowType.FileTransfer:
 | |
|         return _fileTransferWindowId;
 | |
|       case WindowType.PortForward:
 | |
|         return _portForwardWindowId;
 | |
|       case WindowType.Unknown:
 | |
|         break;
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   void clearWindowType(WindowType type) {
 | |
|     switch (type) {
 | |
|       case WindowType.Main:
 | |
|         return;
 | |
|       case WindowType.RemoteDesktop:
 | |
|         _remoteDesktopWindowId = null;
 | |
|         break;
 | |
|       case WindowType.FileTransfer:
 | |
|         _fileTransferWindowId = null;
 | |
|         break;
 | |
|       case WindowType.PortForward:
 | |
|         _portForwardWindowId = null;
 | |
|         break;
 | |
|       case WindowType.Unknown:
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void setMethodHandler(
 | |
|       Future<dynamic> Function(MethodCall call, int fromWindowId)? handler) {
 | |
|     DesktopMultiWindow.setMethodHandler(handler);
 | |
|   }
 | |
| 
 | |
|   Future<void> closeAllSubWindows() async {
 | |
|     await Future.wait(WindowType.values.map((e) => closeWindows(e)));
 | |
|   }
 | |
| 
 | |
|   Future<void> closeWindows(WindowType type) async {
 | |
|     if (type == WindowType.Main) {
 | |
|       // skip main window, use window manager instead
 | |
|       return;
 | |
|     }
 | |
|     int? wId = findWindowByType(type);
 | |
|     if (wId != null) {
 | |
|       debugPrint("closing multi window: ${type.toString()}");
 | |
|       await saveWindowPosition(type, windowId: wId);
 | |
|       try {
 | |
|         final ids = await DesktopMultiWindow.getAllSubWindowIds();
 | |
|         if (!ids.contains(wId)) {
 | |
|           // no such window already
 | |
|           return;
 | |
|         }
 | |
|         await WindowController.fromWindowId(wId).setPreventClose(false);
 | |
|         await WindowController.fromWindowId(wId).close();
 | |
|       } catch (e) {
 | |
|         debugPrint("$e");
 | |
|         return;
 | |
|       } finally {
 | |
|         clearWindowType(type);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<List<int>> getAllSubWindowIds() async {
 | |
|     try {
 | |
|       final windows = await DesktopMultiWindow.getAllSubWindowIds();
 | |
|       return windows;
 | |
|     } catch (err) {
 | |
|       if (err is AssertionError) {
 | |
|         return [];
 | |
|       } else {
 | |
|         rethrow;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   List<int> getActiveWindows() {
 | |
|     return _activeWindows;
 | |
|   }
 | |
| 
 | |
|   Future<void> _notifyActiveWindow() async {
 | |
|     for (final callback in _windowActiveCallbacks) {
 | |
|       await callback.call();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<void> registerActiveWindow(int windowId) async {
 | |
|     if (_activeWindows.contains(windowId)) {
 | |
|       // ignore
 | |
|     } else {
 | |
|       _activeWindows.add(windowId);
 | |
|     }
 | |
|     await _notifyActiveWindow();
 | |
|   }
 | |
| 
 | |
|   /// Remove active window which has [`windowId`]
 | |
|   ///
 | |
|   /// [Availability]
 | |
|   /// This function should only be called from main window.
 | |
|   /// For other windows, please post a unregister(hide) event to main window handler:
 | |
|   /// `rustDeskWinManager.call(WindowType.Main, kWindowEventHide, {"id": windowId!});`
 | |
|   Future<void> unregisterActiveWindow(int windowId) async {
 | |
|     if (!_activeWindows.contains(windowId)) {
 | |
|       // ignore
 | |
|     } else {
 | |
|       _activeWindows.remove(windowId);
 | |
|     }
 | |
|     await _notifyActiveWindow();
 | |
|   }
 | |
| 
 | |
|   void registerActiveWindowListener(AsyncCallback callback) {
 | |
|     _windowActiveCallbacks.add(callback);
 | |
|   }
 | |
| 
 | |
|   void unregisterActiveWindowListener(AsyncCallback callback) {
 | |
|     _windowActiveCallbacks.remove(callback);
 | |
|   }
 | |
| }
 | |
| 
 | |
| final rustDeskWinManager = RustDeskMultiWindowManager.instance;
 |