Merge pull request #1479 from Heap-Hop/master

Update flutter desktop
This commit is contained in:
RustDesk 2022-09-08 22:39:07 +08:00 committed by GitHub
commit f1bbe9ca5e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 273 additions and 203 deletions

View File

@ -14,6 +14,7 @@ import 'package:get/get.dart';
import 'package:shared_preferences/shared_preferences.dart'; import 'package:shared_preferences/shared_preferences.dart';
import 'package:window_manager/window_manager.dart'; import 'package:window_manager/window_manager.dart';
import 'mobile/widgets/overlay.dart';
import 'models/model.dart'; import 'models/model.dart';
import 'models/platform_model.dart'; import 'models/platform_model.dart';
@ -294,9 +295,11 @@ class Dialog<T> {
class OverlayDialogManager { class OverlayDialogManager {
OverlayState? _overlayState; OverlayState? _overlayState;
Map<String, Dialog> _dialogs = Map(); final Map<String, Dialog> _dialogs = {};
int _tagCount = 0; int _tagCount = 0;
OverlayEntry? _mobileActionsOverlayEntry;
/// By default OverlayDialogManager use global overlay /// By default OverlayDialogManager use global overlay
OverlayDialogManager() { OverlayDialogManager() {
_overlayState = globalKey.currentState?.overlay; _overlayState = globalKey.currentState?.overlay;
@ -418,6 +421,60 @@ class OverlayDialogManager {
); );
}); });
} }
void resetMobileActionsOverlay({FFI? ffi}) {
if (_mobileActionsOverlayEntry == null) return;
hideMobileActionsOverlay();
showMobileActionsOverlay(ffi: ffi);
}
void showMobileActionsOverlay({FFI? ffi}) {
if (_mobileActionsOverlayEntry != null) return;
if (_overlayState == null) return;
// compute overlay position
final screenW = MediaQuery.of(globalKey.currentContext!).size.width;
final screenH = MediaQuery.of(globalKey.currentContext!).size.height;
const double overlayW = 200;
const double overlayH = 45;
final left = (screenW - overlayW) / 2;
final top = screenH - overlayH - 80;
final overlay = OverlayEntry(builder: (context) {
final session = ffi ?? gFFI;
return DraggableMobileActions(
position: Offset(left, top),
width: overlayW,
height: overlayH,
onBackPressed: () => session.tap(MouseButtons.right),
onHomePressed: () => session.tap(MouseButtons.wheel),
onRecentPressed: () async {
session.sendMouse('down', MouseButtons.wheel);
await Future.delayed(const Duration(milliseconds: 500));
session.sendMouse('up', MouseButtons.wheel);
},
onHidePressed: () => hideMobileActionsOverlay(),
);
});
_overlayState!.insert(overlay);
_mobileActionsOverlayEntry = overlay;
}
void hideMobileActionsOverlay() {
if (_mobileActionsOverlayEntry != null) {
_mobileActionsOverlayEntry!.remove();
_mobileActionsOverlayEntry = null;
return;
}
}
void toggleMobileActionsOverlay({FFI? ffi}) {
if (_mobileActionsOverlayEntry == null) {
showMobileActionsOverlay(ffi: ffi);
} else {
hideMobileActionsOverlay();
}
}
} }
void showToast(String text, {Duration timeout = const Duration(seconds: 2)}) { void showToast(String text, {Duration timeout = const Duration(seconds: 2)}) {

View File

@ -8,6 +8,8 @@ import 'package:flutter_hbb/desktop/widgets/tabbar_widget.dart';
import 'package:flutter_hbb/utils/multi_window_manager.dart'; import 'package:flutter_hbb/utils/multi_window_manager.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import '../../mobile/widgets/dialog.dart';
/// File Transfer for multi tabs /// File Transfer for multi tabs
class FileManagerTabPage extends StatefulWidget { class FileManagerTabPage extends StatefulWidget {
final Map<String, dynamic> params; final Map<String, dynamic> params;
@ -31,6 +33,7 @@ class _FileManagerTabPageState extends State<FileManagerTabPage> {
label: params['id'], label: params['id'],
selectedIcon: selectedIcon, selectedIcon: selectedIcon,
unselectedIcon: unselectedIcon, unselectedIcon: unselectedIcon,
onTabCloseButton: () => handleTabCloseButton(params['id']),
page: FileManagerPage(key: ValueKey(params['id']), id: params['id']))); page: FileManagerPage(key: ValueKey(params['id']), id: params['id'])));
} }
@ -53,6 +56,7 @@ class _FileManagerTabPageState extends State<FileManagerTabPage> {
label: id, label: id,
selectedIcon: selectedIcon, selectedIcon: selectedIcon,
unselectedIcon: unselectedIcon, unselectedIcon: unselectedIcon,
onTabCloseButton: () => handleTabCloseButton(id),
page: FileManagerPage(key: ValueKey(id), id: id))); page: FileManagerPage(key: ValueKey(id), id: id)));
} else if (call.method == "onDestroy") { } else if (call.method == "onDestroy") {
tabController.clear(); tabController.clear();
@ -71,10 +75,10 @@ class _FileManagerTabPageState extends State<FileManagerTabPage> {
backgroundColor: MyTheme.color(context).bg, backgroundColor: MyTheme.color(context).bg,
body: DesktopTab( body: DesktopTab(
controller: tabController, controller: tabController,
onClose: () { onWindowCloseButton: () {
tabController.clear(); tabController.clear();
}, },
tail: AddButton().paddingOnly(left: 10), tail: const AddButton().paddingOnly(left: 10),
)), )),
), ),
); );
@ -89,4 +93,14 @@ class _FileManagerTabPageState extends State<FileManagerTabPage> {
int windowId() { int windowId() {
return widget.params["windowId"]; return widget.params["windowId"];
} }
void handleTabCloseButton(String peerId) {
final session = ffi('ft_$peerId');
if (session.ffiModel.pi.hostname.isNotEmpty) {
tabController.jumpBy(peerId);
clientClose(session.dialogManager);
} else {
tabController.closeBy(peerId);
}
}
} }

View File

@ -78,7 +78,7 @@ class _PortForwardTabPageState extends State<PortForwardTabPage> {
backgroundColor: MyTheme.color(context).bg, backgroundColor: MyTheme.color(context).bg,
body: DesktopTab( body: DesktopTab(
controller: tabController, controller: tabController,
onClose: () { onWindowCloseButton: () {
tabController.clear(); tabController.clear();
}, },
tail: AddButton().paddingOnly(left: 10), tail: AddButton().paddingOnly(left: 10),
@ -88,7 +88,6 @@ class _PortForwardTabPageState extends State<PortForwardTabPage> {
} }
void onRemoveId(String id) { void onRemoveId(String id) {
ffi("pf_$id").close();
if (tabController.state.value.tabs.isEmpty) { if (tabController.state.value.tabs.isEmpty) {
WindowController.fromWindowId(windowId()).hide(); WindowController.fromWindowId(windowId()).hide();
} }

View File

@ -1,6 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'dart:io'; import 'dart:io';
import 'dart:typed_data';
import 'dart:ui' as ui; import 'dart:ui' as ui;
import 'package:flutter/gestures.dart'; import 'package:flutter/gestures.dart';
@ -16,7 +15,6 @@ import 'package:flutter_custom_cursor/flutter_custom_cursor.dart';
import '../widgets/remote_menubar.dart'; import '../widgets/remote_menubar.dart';
import '../../common.dart'; import '../../common.dart';
import '../../mobile/widgets/dialog.dart'; import '../../mobile/widgets/dialog.dart';
import '../../mobile/widgets/overlay.dart';
import '../../models/model.dart'; import '../../models/model.dart';
import '../../models/platform_model.dart'; import '../../models/platform_model.dart';
import '../../common/shared_state.dart'; import '../../common/shared_state.dart';
@ -107,7 +105,7 @@ class _RemotePageState extends State<RemotePage>
@override @override
void dispose() { void dispose() {
debugPrint("REMOTE PAGE dispose ${widget.id}"); debugPrint("REMOTE PAGE dispose ${widget.id}");
hideMobileActionsOverlay(); _ffi.dialogManager.hideMobileActionsOverlay();
_ffi.listenToMouse(false); _ffi.listenToMouse(false);
_mobileFocusNode.dispose(); _mobileFocusNode.dispose();
_physicalFocusNode.dispose(); _physicalFocusNode.dispose();

View File

@ -10,6 +10,8 @@ import 'package:flutter_hbb/desktop/widgets/tabbar_widget.dart';
import 'package:flutter_hbb/utils/multi_window_manager.dart'; import 'package:flutter_hbb/utils/multi_window_manager.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import '../../mobile/widgets/dialog.dart';
class ConnectionTabPage extends StatefulWidget { class ConnectionTabPage extends StatefulWidget {
final Map<String, dynamic> params; final Map<String, dynamic> params;
@ -37,12 +39,12 @@ class _ConnectionTabPageState extends State<ConnectionTabPage> {
label: peerId, label: peerId,
selectedIcon: selectedIcon, selectedIcon: selectedIcon,
unselectedIcon: unselectedIcon, unselectedIcon: unselectedIcon,
page: Obx(() => RemotePage( onTabCloseButton: () => handleTabCloseButton(peerId),
page: RemotePage(
key: ValueKey(peerId), key: ValueKey(peerId),
id: peerId, id: peerId,
tabBarHeight: tabBarHeight: fullscreen.isTrue ? 0 : kDesktopRemoteTabBarHeight,
fullscreen.isTrue ? 0 : kDesktopRemoteTabBarHeight, )));
))));
} }
} }
@ -69,12 +71,12 @@ class _ConnectionTabPageState extends State<ConnectionTabPage> {
label: id, label: id,
selectedIcon: selectedIcon, selectedIcon: selectedIcon,
unselectedIcon: unselectedIcon, unselectedIcon: unselectedIcon,
page: Obx(() => RemotePage( onTabCloseButton: () => handleTabCloseButton(id),
page: RemotePage(
key: ValueKey(id), key: ValueKey(id),
id: id, id: id,
tabBarHeight: tabBarHeight: fullscreen.isTrue ? 0 : kDesktopRemoteTabBarHeight,
fullscreen.isTrue ? 0 : kDesktopRemoteTabBarHeight, )));
))));
} else if (call.method == "onDestroy") { } else if (call.method == "onDestroy") {
tabController.clear(); tabController.clear();
} }
@ -92,10 +94,10 @@ class _ConnectionTabPageState extends State<ConnectionTabPage> {
border: Border.all(color: MyTheme.color(context).border!)), border: Border.all(color: MyTheme.color(context).border!)),
child: Scaffold( child: Scaffold(
backgroundColor: MyTheme.color(context).bg, backgroundColor: MyTheme.color(context).bg,
body: Obx(() => DesktopTab( body: DesktopTab(
controller: tabController, controller: tabController,
showTabBar: fullscreen.isFalse, showTabBar: fullscreen.isFalse,
onClose: () { onWindowCloseButton: () {
tabController.clear(); tabController.clear();
}, },
tail: AddButton().paddingOnly(left: 10), tail: AddButton().paddingOnly(left: 10),
@ -115,13 +117,11 @@ class _ConnectionTabPageState extends State<ConnectionTabPage> {
], ],
); );
} else { } else {
final msgDirect = translate( final msgDirect = translate(connectionType.direct.value ==
connectionType.direct.value ==
ConnectionType.strDirect ConnectionType.strDirect
? 'Direct Connection' ? 'Direct Connection'
: 'Relay Connection'); : 'Relay Connection');
final msgSecure = translate( final msgSecure = translate(connectionType.secure.value ==
connectionType.secure.value ==
ConnectionType.strSecure ConnectionType.strSecure
? 'Secure Connection' ? 'Secure Connection'
: 'Insecure Connection'); : 'Insecure Connection');
@ -142,7 +142,7 @@ class _ConnectionTabPageState extends State<ConnectionTabPage> {
); );
} }
}), }),
))), )),
), ),
)); ));
} }
@ -157,4 +157,14 @@ class _ConnectionTabPageState extends State<ConnectionTabPage> {
int windowId() { int windowId() {
return widget.params["windowId"]; return widget.params["windowId"];
} }
void handleTabCloseButton(String peerId) {
final session = ffi(peerId);
if (session.ffiModel.pi.hostname.isNotEmpty) {
tabController.jumpBy(peerId);
clientClose(session.dialogManager);
} else {
tabController.closeBy(peerId);
}
}
} }

View File

@ -1,6 +1,6 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_hbb/common.dart'; import 'package:flutter_hbb/common.dart';
import 'package:flutter_hbb/desktop/pages/connection_tab_page.dart'; import 'package:flutter_hbb/desktop/pages/remote_tab_page.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';

View File

@ -6,7 +6,6 @@ import 'package:rxdart/rxdart.dart' as rxdart;
import '../../common.dart'; import '../../common.dart';
import '../../mobile/widgets/dialog.dart'; import '../../mobile/widgets/dialog.dart';
import '../../mobile/widgets/overlay.dart';
import '../../models/model.dart'; import '../../models/model.dart';
import '../../models/platform_model.dart'; import '../../models/platform_model.dart';
import '../../common/shared_state.dart'; import '../../common/shared_state.dart';
@ -75,20 +74,17 @@ class _RemoteMenubarState extends State<RemoteMenubar> {
final List<Widget> menubarItems = []; final List<Widget> menubarItems = [];
if (!isWebDesktop) { if (!isWebDesktop) {
menubarItems.add(_buildFullscreen(context)); menubarItems.add(_buildFullscreen(context));
//if (widget.ffi.ffiModel.isPeerAndroid) { if (widget.ffi.ffiModel.isPeerAndroid) {
menubarItems.add(IconButton( menubarItems.add(IconButton(
tooltip: translate('Mobile Actions'), tooltip: translate('Mobile Actions'),
color: _MenubarTheme.commonColor, color: _MenubarTheme.commonColor,
icon: const Icon(Icons.build), icon: const Icon(Icons.build),
onPressed: () { onPressed: () {
if (mobileActionsOverlayEntry == null) { widget.ffi.dialogManager
showMobileActionsOverlay(); .toggleMobileActionsOverlay(ffi: widget.ffi);
} else {
hideMobileActionsOverlay();
}
}, },
)); ));
//} }
} }
menubarItems.add(_buildMonitor(context)); menubarItems.add(_buildMonitor(context));
menubarItems.add(_buildControl(context)); menubarItems.add(_buildControl(context));

View File

@ -25,6 +25,7 @@ class TabInfo {
final IconData? selectedIcon; final IconData? selectedIcon;
final IconData? unselectedIcon; final IconData? unselectedIcon;
final bool closable; final bool closable;
final VoidCallback? onTabCloseButton;
final Widget page; final Widget page;
TabInfo( TabInfo(
@ -33,6 +34,7 @@ class TabInfo {
this.selectedIcon, this.selectedIcon,
this.unselectedIcon, this.unselectedIcon,
this.closable = true, this.closable = true,
this.onTabCloseButton,
required this.page}); required this.page});
} }
@ -137,6 +139,12 @@ class DesktopTabController {
} }
} }
void jumpBy(String key) {
if (!isDesktop) return;
final index = state.value.tabs.indexWhere((tab) => tab.key == key);
jumpTo(index);
}
void closeBy(String? key) { void closeBy(String? key) {
if (!isDesktop) return; if (!isDesktop) return;
assert(onRemove != null); assert(onRemove != null);
@ -145,8 +153,8 @@ class DesktopTabController {
remove(state.value.selected); remove(state.value.selected);
} }
} else { } else {
state.value.tabs.indexWhere((tab) => tab.key == key); final index = state.value.tabs.indexWhere((tab) => tab.key == key);
remove(state.value.selected); remove(index);
} }
} }
@ -175,7 +183,7 @@ class DesktopTab extends StatelessWidget {
final bool showClose; final bool showClose;
final Widget Function(Widget pageView)? pageViewBuilder; final Widget Function(Widget pageView)? pageViewBuilder;
final Widget? tail; final Widget? tail;
final VoidCallback? onClose; final VoidCallback? onWindowCloseButton;
final TabBuilder? tabBuilder; final TabBuilder? tabBuilder;
final LabelGetter? labelGetter; final LabelGetter? labelGetter;
@ -196,7 +204,7 @@ class DesktopTab extends StatelessWidget {
this.showClose = true, this.showClose = true,
this.pageViewBuilder, this.pageViewBuilder,
this.tail, this.tail,
this.onClose, this.onWindowCloseButton,
this.tabBuilder, this.tabBuilder,
this.labelGetter, this.labelGetter,
}) : super(key: key) { }) : super(key: key) {
@ -333,7 +341,7 @@ class DesktopTab extends StatelessWidget {
showMinimize: showMinimize, showMinimize: showMinimize,
showMaximize: showMaximize, showMaximize: showMaximize,
showClose: showClose, showClose: showClose,
onClose: onClose, onClose: onWindowCloseButton,
) )
], ],
); );
@ -511,7 +519,13 @@ class _ListView extends StatelessWidget {
unselectedIcon: tab.unselectedIcon, unselectedIcon: tab.unselectedIcon,
closable: tab.closable, closable: tab.closable,
selected: state.value.selected, selected: state.value.selected,
onClose: () => controller.remove(index), onClose: () {
if (tab.onTabCloseButton != null) {
tab.onTabCloseButton!();
} else {
controller.remove(index);
}
},
onSelected: () => controller.jumpTo(index), onSelected: () => controller.jumpTo(index),
tabBuilder: tabBuilder == null tabBuilder: tabBuilder == null
? null ? null

View File

@ -14,7 +14,6 @@ import '../../models/model.dart';
import '../../models/platform_model.dart'; import '../../models/platform_model.dart';
import '../widgets/dialog.dart'; import '../widgets/dialog.dart';
import '../widgets/gestures.dart'; import '../widgets/gestures.dart';
import '../widgets/overlay.dart';
final initText = '\1' * 1024; final initText = '\1' * 1024;
@ -64,7 +63,7 @@ class _RemotePageState extends State<RemotePage> {
@override @override
void dispose() { void dispose() {
hideMobileActionsOverlay(); gFFI.dialogManager.hideMobileActionsOverlay();
gFFI.listenToMouse(false); gFFI.listenToMouse(false);
gFFI.invokeMethod("enable_soft_keyboard", true); gFFI.invokeMethod("enable_soft_keyboard", true);
_mobileFocusNode.dispose(); _mobileFocusNode.dispose();
@ -266,8 +265,9 @@ class _RemotePageState extends State<RemotePage> {
: SafeArea(child: : SafeArea(child:
OrientationBuilder(builder: (ctx, orientation) { OrientationBuilder(builder: (ctx, orientation) {
if (_currentOrientation != orientation) { if (_currentOrientation != orientation) {
Timer(Duration(milliseconds: 200), () { Timer(const Duration(milliseconds: 200), () {
resetMobileActionsOverlay(); gFFI.dialogManager
.resetMobileActionsOverlay(ffi: gFFI);
_currentOrientation = orientation; _currentOrientation = orientation;
gFFI.canvasModel.updateViewStyle(); gFFI.canvasModel.updateViewStyle();
}); });
@ -422,14 +422,9 @@ class _RemotePageState extends State<RemotePage> {
? [ ? [
IconButton( IconButton(
color: Colors.white, color: Colors.white,
icon: Icon(Icons.build), icon: const Icon(Icons.build),
onPressed: () { onPressed: () => gFFI.dialogManager
if (mobileActionsOverlayEntry == null) { .toggleMobileActionsOverlay(ffi: gFFI),
showMobileActionsOverlay();
} else {
hideMobileActionsOverlay();
}
},
) )
] ]
: [ : [

View File

@ -2,11 +2,8 @@ import 'package:flutter/material.dart';
import 'package:flutter_hbb/common.dart'; import 'package:flutter_hbb/common.dart';
import '../../models/chat_model.dart'; import '../../models/chat_model.dart';
import '../../models/model.dart';
import '../pages/chat_page.dart'; import '../pages/chat_page.dart';
OverlayEntry? mobileActionsOverlayEntry;
class DraggableChatWindow extends StatelessWidget { class DraggableChatWindow extends StatelessWidget {
DraggableChatWindow( DraggableChatWindow(
{this.position = Offset.zero, {this.position = Offset.zero,
@ -99,6 +96,7 @@ class DraggableMobileActions extends StatelessWidget {
this.onBackPressed, this.onBackPressed,
this.onRecentPressed, this.onRecentPressed,
this.onHomePressed, this.onHomePressed,
this.onHidePressed,
required this.width, required this.width,
required this.height}); required this.height});
@ -108,6 +106,7 @@ class DraggableMobileActions extends StatelessWidget {
final VoidCallback? onBackPressed; final VoidCallback? onBackPressed;
final VoidCallback? onHomePressed; final VoidCallback? onHomePressed;
final VoidCallback? onRecentPressed; final VoidCallback? onRecentPressed;
final VoidCallback? onHidePressed;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -118,6 +117,9 @@ class DraggableMobileActions extends StatelessWidget {
builder: (_, onPanUpdate) { builder: (_, onPanUpdate) {
return GestureDetector( return GestureDetector(
onPanUpdate: onPanUpdate, onPanUpdate: onPanUpdate,
child: Card(
color: Colors.transparent,
shadowColor: Colors.transparent,
child: Container( child: Container(
decoration: BoxDecoration( decoration: BoxDecoration(
color: MyTheme.accent.withOpacity(0.4), color: MyTheme.accent.withOpacity(0.4),
@ -128,16 +130,19 @@ class DraggableMobileActions extends StatelessWidget {
IconButton( IconButton(
color: MyTheme.white, color: MyTheme.white,
onPressed: onBackPressed, onPressed: onBackPressed,
icon: Icon(Icons.arrow_back)), splashRadius: 20,
icon: const Icon(Icons.arrow_back)),
IconButton( IconButton(
color: MyTheme.white, color: MyTheme.white,
onPressed: onHomePressed, onPressed: onHomePressed,
icon: Icon(Icons.home)), splashRadius: 20,
icon: const Icon(Icons.home)),
IconButton( IconButton(
color: MyTheme.white, color: MyTheme.white,
onPressed: onRecentPressed, onPressed: onRecentPressed,
icon: Icon(Icons.more_horiz)), splashRadius: 20,
VerticalDivider( icon: const Icon(Icons.more_horiz)),
const VerticalDivider(
width: 0, width: 0,
thickness: 2, thickness: 2,
indent: 10, indent: 10,
@ -145,62 +150,16 @@ class DraggableMobileActions extends StatelessWidget {
), ),
IconButton( IconButton(
color: MyTheme.white, color: MyTheme.white,
onPressed: hideMobileActionsOverlay, onPressed: onHidePressed,
icon: Icon(Icons.keyboard_arrow_down)), splashRadius: 20,
icon: const Icon(Icons.keyboard_arrow_down)),
], ],
), ),
)); )));
}); });
} }
} }
resetMobileActionsOverlay() {
if (mobileActionsOverlayEntry == null) return;
hideMobileActionsOverlay();
showMobileActionsOverlay();
}
showMobileActionsOverlay() {
if (mobileActionsOverlayEntry != null) return;
if (globalKey.currentContext == null ||
globalKey.currentState == null ||
globalKey.currentState!.overlay == null) return;
final globalOverlayState = globalKey.currentState!.overlay!;
// compute overlay position
final screenW = MediaQuery.of(globalKey.currentContext!).size.width;
final screenH = MediaQuery.of(globalKey.currentContext!).size.height;
final double overlayW = 200;
final double overlayH = 45;
final left = (screenW - overlayW) / 2;
final top = screenH - overlayH - 80;
final overlay = OverlayEntry(builder: (context) {
return DraggableMobileActions(
position: Offset(left, top),
width: overlayW,
height: overlayH,
onBackPressed: () => gFFI.tap(MouseButtons.right),
onHomePressed: () => gFFI.tap(MouseButtons.wheel),
onRecentPressed: () async {
gFFI.sendMouse('down', MouseButtons.wheel);
await Future.delayed(Duration(milliseconds: 500));
gFFI.sendMouse('up', MouseButtons.wheel);
},
);
});
globalOverlayState.insert(overlay);
mobileActionsOverlayEntry = overlay;
}
hideMobileActionsOverlay() {
if (mobileActionsOverlayEntry != null) {
mobileActionsOverlayEntry!.remove();
mobileActionsOverlayEntry = null;
return;
}
}
class Draggable extends StatefulWidget { class Draggable extends StatefulWidget {
Draggable( Draggable(
{this.checkKeyboard = false, {this.checkKeyboard = false,

View File

@ -143,9 +143,12 @@ class ChatModel with ChangeNotifier {
} }
toggleChatOverlay() { toggleChatOverlay() {
if (chatIconOverlayEntry == null || chatWindowOverlayEntry == null) { if ((!isDesktop && chatIconOverlayEntry == null) ||
chatWindowOverlayEntry == null) {
gFFI.invokeMethod("enable_soft_keyboard", true); gFFI.invokeMethod("enable_soft_keyboard", true);
if (!isDesktop) {
showChatIconOverlay(); showChatIconOverlay();
}
showChatWindowOverlay(); showChatWindowOverlay();
} else { } else {
hideChatIconOverlay(); hideChatIconOverlay();

View File

@ -22,7 +22,6 @@ import 'package:flutter_custom_cursor/flutter_custom_cursor.dart';
import '../common.dart'; import '../common.dart';
import '../common/shared_state.dart'; import '../common/shared_state.dart';
import '../mobile/widgets/dialog.dart'; import '../mobile/widgets/dialog.dart';
import '../mobile/widgets/overlay.dart';
import 'peer_model.dart'; import 'peer_model.dart';
import 'platform_model.dart'; import 'platform_model.dart';
@ -267,8 +266,10 @@ class FfiModel with ChangeNotifier {
if (isPeerAndroid) { if (isPeerAndroid) {
_touchMode = true; _touchMode = true;
if (parent.target?.ffiModel.permissions['keyboard'] != false) { if (parent.target != null &&
Timer(const Duration(milliseconds: 100), showMobileActionsOverlay); parent.target!.ffiModel.permissions['keyboard'] != false) {
Timer(const Duration(milliseconds: 100),
parent.target!.dialogManager.showMobileActionsOverlay);
} }
} else { } else {
_touchMode = _touchMode =

View File

@ -414,6 +414,15 @@ packages:
url: "https://pub.flutter-io.cn" url: "https://pub.flutter-io.cn"
source: hosted source: hosted
version: "3.3.0" version: "3.3.0"
flutter_custom_cursor:
dependency: "direct main"
description:
path: "."
ref: "9021e21de36c84edf01d5034f38eda580463163b"
resolved-ref: "9021e21de36c84edf01d5034f38eda580463163b"
url: "https://github.com/Kingtous/rustdesk_flutter_custom_cursor"
source: git
version: "0.0.1"
flutter_launcher_icons: flutter_launcher_icons:
dependency: "direct dev" dependency: "direct dev"
description: description:

View File

@ -1,6 +1,6 @@
use hbb_common::log; use hbb_common::log;
use crate::{start_os_service, flutter::connection_manager}; use crate::{start_os_service, flutter::connection_manager, start_server};
/// Main entry of the RustDesk Core. /// Main entry of the RustDesk Core.
/// Return true if the app should continue running with UI(possibly Flutter), false if the app should exit. /// Return true if the app should continue running with UI(possibly Flutter), false if the app should exit.
@ -20,7 +20,15 @@ pub fn core_main() -> bool {
return false; return false;
} }
if args[1] == "--server" { if args[1] == "--server" {
// TODO: server log::info!("start --server");
#[cfg(not(target_os = "macos"))]
{
start_server(true);
}
#[cfg(target_os = "macos")]
{
std::thread::spawn(move || start_server(true));
}
return false; return false;
} }
} }

View File

@ -525,20 +525,27 @@ pub fn is_root() -> bool {
crate::username() == "root" crate::username() == "root"
} }
fn is_opensuse() -> bool {
if let Ok(res) = run_cmds("cat /etc/os-release | grep opensuse".to_owned()) {
if !res.is_empty() {
return true;
}
}
false
}
pub fn run_as_user(arg: &str) -> ResultType<Option<std::process::Child>> { pub fn run_as_user(arg: &str) -> ResultType<Option<std::process::Child>> {
let uid = get_active_userid(); let uid = get_active_userid();
let cmd = std::env::current_exe()?; let cmd = std::env::current_exe()?;
let xdg = &format!("XDG_RUNTIME_DIR=/run/user/{}", uid) as &str;
let username = &get_active_username();
let mut args = vec![xdg, "-u", username, cmd.to_str().unwrap_or(""), arg];
// -E required for opensuse // -E required for opensuse
let task = std::process::Command::new("sudo") if is_opensuse() {
.args(vec![ args.insert(0, "-E");
"-E", }
&format!("XDG_RUNTIME_DIR=/run/user/{}", uid) as &str,
"-u", let task = std::process::Command::new("sudo").args(args).spawn()?;
&get_active_username(),
cmd.to_str().unwrap_or(""),
arg,
])
.spawn()?;
Ok(Some(task)) Ok(Some(task))
} }