commit
f1bbe9ca5e
@ -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)}) {
|
||||||
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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();
|
||||||
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
@ -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';
|
||||||
|
|
||||||
|
@ -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));
|
||||||
|
@ -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
|
||||||
|
@ -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();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
: [
|
: [
|
||||||
|
@ -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,
|
||||||
|
@ -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();
|
||||||
|
@ -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 =
|
||||||
|
@ -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:
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user