import 'dart:async'; import 'dart:io'; import 'dart:ui'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_hbb/common/shared_state.dart'; import 'package:flutter_hbb/common/widgets/setting_widgets.dart'; import 'package:flutter_hbb/consts.dart'; import 'package:get/get.dart'; import 'package:qr_flutter/qr_flutter.dart'; import '../../common.dart'; import '../../models/model.dart'; import '../../models/platform_model.dart'; import 'address_book.dart'; void clientClose(SessionID sessionId, OverlayDialogManager dialogManager) { msgBox(sessionId, 'info', 'Close', 'Are you sure to close the connection?', '', dialogManager); } abstract class ValidationRule { String get name; bool validate(String value); } class LengthRangeValidationRule extends ValidationRule { final int _min; final int _max; LengthRangeValidationRule(this._min, this._max); @override String get name => translate('length %min% to %max%') .replaceAll('%min%', _min.toString()) .replaceAll('%max%', _max.toString()); @override bool validate(String value) { return value.length >= _min && value.length <= _max; } } class RegexValidationRule extends ValidationRule { final String _name; final RegExp _regex; RegexValidationRule(this._name, this._regex); @override String get name => translate(_name); @override bool validate(String value) { return value.isNotEmpty ? value.contains(_regex) : false; } } void changeIdDialog() { var newId = ""; var msg = ""; var isInProgress = false; TextEditingController controller = TextEditingController(); final RxString rxId = controller.text.trim().obs; final rules = [ RegexValidationRule('starts with a letter', RegExp(r'^[a-zA-Z]')), LengthRangeValidationRule(6, 16), RegexValidationRule('allowed characters', RegExp(r'^\w*$')) ]; gFFI.dialogManager.show((setState, close, context) { submit() async { debugPrint("onSubmit"); newId = controller.text.trim(); final Iterable violations = rules.where((r) => !r.validate(newId)); if (violations.isNotEmpty) { setState(() { msg = isDesktop ? '${translate('Prompt')}: ${violations.map((r) => r.name).join(', ')}' : violations.map((r) => r.name).join(', '); }); return; } setState(() { msg = ""; isInProgress = true; bind.mainChangeId(newId: newId); }); var status = await bind.mainGetAsyncStatus(); while (status == " ") { await Future.delayed(const Duration(milliseconds: 100)); status = await bind.mainGetAsyncStatus(); } if (status.isEmpty) { // ok close(); return; } setState(() { isInProgress = false; msg = isDesktop ? '${translate('Prompt')}: ${translate(status)}' : translate(status); }); } return CustomAlertDialog( title: Text(translate("Change ID")), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text(translate("id_change_tip")), const SizedBox( height: 12.0, ), TextField( decoration: InputDecoration( labelText: translate('Your new ID'), errorText: msg.isEmpty ? null : translate(msg), suffixText: '${rxId.value.length}/16', suffixStyle: const TextStyle(fontSize: 12, color: Colors.grey)), inputFormatters: [ LengthLimitingTextInputFormatter(16), // FilteringTextInputFormatter(RegExp(r"[a-zA-z][a-zA-z0-9\_]*"), allow: true) ], controller: controller, autofocus: true, onChanged: (value) { setState(() { rxId.value = value.trim(); msg = ''; }); }, ), const SizedBox( height: 8.0, ), isDesktop ? Obx(() => Wrap( runSpacing: 8, spacing: 4, children: rules.map((e) { var checked = e.validate(rxId.value); return Chip( label: Text( e.name, style: TextStyle( color: checked ? const Color(0xFF0A9471) : Color.fromARGB(255, 198, 86, 157)), ), backgroundColor: checked ? const Color(0xFFD0F7ED) : Color.fromARGB(255, 247, 205, 232)); }).toList(), )).marginOnly(bottom: 8) : SizedBox.shrink(), // NOT use Offstage to wrap LinearProgressIndicator if (isInProgress) const LinearProgressIndicator(), ], ), actions: [ dialogButton("Cancel", onPressed: close, isOutline: true), dialogButton("OK", onPressed: submit), ], onSubmit: submit, onCancel: close, ); }); } void changeWhiteList({Function()? callback}) async { var newWhiteList = (await bind.mainGetOption(key: 'whitelist')).split(','); var newWhiteListField = newWhiteList.join('\n'); var controller = TextEditingController(text: newWhiteListField); var msg = ""; var isInProgress = false; gFFI.dialogManager.show((setState, close, context) { return CustomAlertDialog( title: Text(translate("IP Whitelisting")), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text(translate("whitelist_sep")), const SizedBox( height: 8.0, ), Row( children: [ Expanded( child: TextField( maxLines: null, decoration: InputDecoration( errorText: msg.isEmpty ? null : translate(msg), ), controller: controller, autofocus: true), ), ], ), const SizedBox( height: 4.0, ), // NOT use Offstage to wrap LinearProgressIndicator if (isInProgress) const LinearProgressIndicator(), ], ), actions: [ dialogButton("Cancel", onPressed: close, isOutline: true), dialogButton("Clear", onPressed: () async { await bind.mainSetOption(key: 'whitelist', value: ''); callback?.call(); close(); }, isOutline: true), dialogButton( "OK", onPressed: () async { setState(() { msg = ""; isInProgress = true; }); newWhiteListField = controller.text.trim(); var newWhiteList = ""; if (newWhiteListField.isEmpty) { // pass } else { final ips = newWhiteListField.trim().split(RegExp(r"[\s,;\n]+")); // test ip final ipMatch = RegExp( r"^(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]?|0)\.(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]?|0)\.(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]?|0)\.(25[0-5]|2[0-4][0-9]|1[0-9][0-9]|[1-9][0-9]?|0)(\/([1-9]|[1-2][0-9]|3[0-2])){0,1}$"); final ipv6Match = RegExp( r"^(((?:[0-9A-Fa-f]{1,4}))*((?::[0-9A-Fa-f]{1,4}))*::((?:[0-9A-Fa-f]{1,4}))*((?::[0-9A-Fa-f]{1,4}))*|((?:[0-9A-Fa-f]{1,4}))((?::[0-9A-Fa-f]{1,4})){7})(\/([1-9]|[1-9][0-9]|1[0-1][0-9]|12[0-8])){0,1}$"); for (final ip in ips) { if (!ipMatch.hasMatch(ip) && !ipv6Match.hasMatch(ip)) { msg = "${translate("Invalid IP")} $ip"; setState(() { isInProgress = false; }); return; } } newWhiteList = ips.join(','); } await bind.mainSetOption(key: 'whitelist', value: newWhiteList); callback?.call(); close(); }, ), ], onCancel: close, ); }); } Future changeDirectAccessPort( String currentIP, String currentPort) async { final controller = TextEditingController(text: currentPort); await gFFI.dialogManager.show((setState, close, context) { return CustomAlertDialog( title: Text(translate("Change Local Port")), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ const SizedBox(height: 8.0), Row( children: [ Expanded( child: TextField( maxLines: null, keyboardType: TextInputType.number, decoration: InputDecoration( hintText: '21118', isCollapsed: true, prefix: Text('$currentIP : '), suffix: IconButton( padding: EdgeInsets.zero, icon: const Icon(Icons.clear, size: 16), onPressed: () => controller.clear())), inputFormatters: [ FilteringTextInputFormatter.allow(RegExp( r'^([0-9]|[1-9]\d|[1-9]\d{2}|[1-9]\d{3}|[1-5]\d{4}|6[0-4]\d{3}|65[0-4]\d{2}|655[0-2]\d|6553[0-5])$')), ], controller: controller, autofocus: true), ), ], ), ], ), actions: [ dialogButton("Cancel", onPressed: close, isOutline: true), dialogButton("OK", onPressed: () async { await bind.mainSetOption( key: 'direct-access-port', value: controller.text); close(); }), ], onCancel: close, ); }); return controller.text; } Future changeAutoDisconnectTimeout(String old) async { final controller = TextEditingController(text: old); await gFFI.dialogManager.show((setState, close, context) { return CustomAlertDialog( title: Text(translate("Timeout in minutes")), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ const SizedBox(height: 8.0), Row( children: [ Expanded( child: TextField( maxLines: null, keyboardType: TextInputType.number, decoration: InputDecoration( hintText: '10', isCollapsed: true, suffix: IconButton( padding: EdgeInsets.zero, icon: const Icon(Icons.clear, size: 16), onPressed: () => controller.clear())), inputFormatters: [ FilteringTextInputFormatter.allow(RegExp( r'^([0-9]|[1-9]\d|[1-9]\d{2}|[1-9]\d{3}|[1-5]\d{4}|6[0-4]\d{3}|65[0-4]\d{2}|655[0-2]\d|6553[0-5])$')), ], controller: controller, autofocus: true), ), ], ), ], ), actions: [ dialogButton("Cancel", onPressed: close, isOutline: true), dialogButton("OK", onPressed: () async { await bind.mainSetOption( key: 'auto-disconnect-timeout', value: controller.text); close(); }), ], onCancel: close, ); }); return controller.text; } class DialogTextField extends StatelessWidget { final String title; final String? hintText; final bool obscureText; final String? errorText; final String? helperText; final Widget? prefixIcon; final Widget? suffixIcon; final TextEditingController controller; final FocusNode? focusNode; final TextInputType? keyboardType; final List? inputFormatters; static const kUsernameTitle = 'Username'; static const kUsernameIcon = Icon(Icons.account_circle_outlined); static const kPasswordTitle = 'Password'; static const kPasswordIcon = Icon(Icons.lock_outline); DialogTextField( {Key? key, this.focusNode, this.obscureText = false, this.errorText, this.helperText, this.prefixIcon, this.suffixIcon, this.hintText, this.keyboardType, this.inputFormatters, required this.title, required this.controller}) : super(key: key); @override Widget build(BuildContext context) { return Row( children: [ Expanded( child: TextField( decoration: InputDecoration( labelText: title, hintText: hintText, prefixIcon: prefixIcon, suffixIcon: suffixIcon, helperText: helperText, helperMaxLines: 8, errorText: errorText, errorMaxLines: 8, ), controller: controller, focusNode: focusNode, autofocus: true, obscureText: obscureText, keyboardType: keyboardType, inputFormatters: inputFormatters, ), ), ], ).paddingSymmetric(vertical: 4.0); } } abstract class ValidationField extends StatelessWidget { ValidationField({Key? key}) : super(key: key); String? validate(); bool get isReady; } class Dialog2FaField extends ValidationField { Dialog2FaField({ Key? key, required this.controller, this.autoFocus = true, this.reRequestFocus = false, this.title, this.helperText, this.hintText, this.errorText, this.readyCallback, this.onChanged, }) : super(key: key); final TextEditingController controller; final bool autoFocus; final bool reRequestFocus; final String? title; final String? helperText; final String? hintText; final String? errorText; final VoidCallback? readyCallback; final VoidCallback? onChanged; final errMsg = translate('2FA code must be 6 digits.'); @override Widget build(BuildContext context) { return DialogVerificationCodeField( title: title ?? translate('2FA code'), controller: controller, errorText: errorText, autoFocus: autoFocus, reRequestFocus: reRequestFocus, hintText: hintText, readyCallback: readyCallback, helperText: helperText ?? translate('2fa_tip'), onChanged: _onChanged, keyboardType: TextInputType.number, inputFormatters: [ FilteringTextInputFormatter.allow(RegExp(r'[0-9]')), ], ); } String get text => controller.text; bool get isAllDigits => text.codeUnits.every((e) => e >= 48 && e <= 57); @override bool get isReady => text.length == 6 && isAllDigits; @override String? validate() => isReady ? null : errMsg; _onChanged(StateSetter setState, SimpleWrapper errText) { onChanged?.call(); if (text.length > 6) { setState(() => errText.value = errMsg); return; } if (!isAllDigits) { setState(() => errText.value = errMsg); return; } if (isReady) { readyCallback?.call(); return; } if (errText.value != null) { setState(() => errText.value = null); } } } class DialogEmailCodeField extends ValidationField { DialogEmailCodeField({ Key? key, required this.controller, this.autoFocus = true, this.reRequestFocus = false, this.hintText, this.errorText, this.readyCallback, this.onChanged, }) : super(key: key); final TextEditingController controller; final bool autoFocus; final bool reRequestFocus; final String? hintText; final String? errorText; final VoidCallback? readyCallback; final VoidCallback? onChanged; final errMsg = translate('Email verification code must be 6 characters.'); @override Widget build(BuildContext context) { return DialogVerificationCodeField( title: translate('Verification code'), controller: controller, errorText: errorText, autoFocus: autoFocus, reRequestFocus: reRequestFocus, hintText: hintText, readyCallback: readyCallback, helperText: translate('verification_tip'), onChanged: _onChanged, keyboardType: TextInputType.visiblePassword, ); } String get text => controller.text; @override bool get isReady => text.length == 6; @override String? validate() => isReady ? null : errMsg; _onChanged(StateSetter setState, SimpleWrapper errText) { onChanged?.call(); if (text.length > 6) { setState(() => errText.value = errMsg); return; } if (isReady) { readyCallback?.call(); return; } if (errText.value != null) { setState(() => errText.value = null); } } } class DialogVerificationCodeField extends StatefulWidget { DialogVerificationCodeField({ Key? key, required this.controller, required this.title, this.autoFocus = true, this.reRequestFocus = false, this.helperText, this.hintText, this.errorText, this.textLength, this.readyCallback, this.onChanged, this.keyboardType, this.inputFormatters, }) : super(key: key); final TextEditingController controller; final bool autoFocus; final bool reRequestFocus; final String title; final String? helperText; final String? hintText; final String? errorText; final int? textLength; final VoidCallback? readyCallback; final Function(StateSetter setState, SimpleWrapper errText)? onChanged; final TextInputType? keyboardType; final List? inputFormatters; @override State createState() => _DialogVerificationCodeField(); } class _DialogVerificationCodeField extends State { final _focusNode = FocusNode(); Timer? _timer; Timer? _timerReRequestFocus; SimpleWrapper errorText = SimpleWrapper(null); String _preText = ''; @override void initState() { super.initState(); if (widget.autoFocus) { _timer = Timer(Duration(milliseconds: 50), () => _focusNode.requestFocus()); if (widget.onChanged != null) { widget.controller.addListener(() { final text = widget.controller.text.trim(); if (text == _preText) return; widget.onChanged!(setState, errorText); _preText = text; }); } } // software secure keyboard will take the focus since flutter 3.13 // request focus again when android account password obtain focus if (Platform.isAndroid && widget.reRequestFocus) { _focusNode.addListener(() { if (_focusNode.hasFocus) { _timerReRequestFocus?.cancel(); _timerReRequestFocus = Timer( Duration(milliseconds: 100), () => _focusNode.requestFocus()); } }); } } @override void dispose() { _timer?.cancel(); _timerReRequestFocus?.cancel(); _focusNode.unfocus(); _focusNode.dispose(); super.dispose(); } @override Widget build(BuildContext context) { return DialogTextField( title: widget.title, controller: widget.controller, errorText: widget.errorText ?? errorText.value, focusNode: _focusNode, helperText: widget.helperText, keyboardType: widget.keyboardType, inputFormatters: widget.inputFormatters, ); } } class PasswordWidget extends StatefulWidget { PasswordWidget({ Key? key, required this.controller, this.autoFocus = true, this.reRequestFocus = false, this.hintText, this.errorText, }) : super(key: key); final TextEditingController controller; final bool autoFocus; final bool reRequestFocus; final String? hintText; final String? errorText; @override State createState() => _PasswordWidgetState(); } class _PasswordWidgetState extends State { bool _passwordVisible = false; final _focusNode = FocusNode(); Timer? _timer; Timer? _timerReRequestFocus; @override void initState() { super.initState(); if (widget.autoFocus) { _timer = Timer(Duration(milliseconds: 50), () => _focusNode.requestFocus()); } // software secure keyboard will take the focus since flutter 3.13 // request focus again when android account password obtain focus if (Platform.isAndroid && widget.reRequestFocus) { _focusNode.addListener(() { if (_focusNode.hasFocus) { _timerReRequestFocus?.cancel(); _timerReRequestFocus = Timer( Duration(milliseconds: 100), () => _focusNode.requestFocus()); } }); } } @override void dispose() { _timer?.cancel(); _timerReRequestFocus?.cancel(); _focusNode.unfocus(); _focusNode.dispose(); super.dispose(); } @override Widget build(BuildContext context) { return DialogTextField( title: translate(DialogTextField.kPasswordTitle), hintText: translate(widget.hintText ?? 'Enter your password'), controller: widget.controller, prefixIcon: DialogTextField.kPasswordIcon, suffixIcon: IconButton( icon: Icon( // Based on passwordVisible state choose the icon _passwordVisible ? Icons.visibility : Icons.visibility_off, color: MyTheme.lightTheme.primaryColor), onPressed: () { // Update the state i.e. toggle the state of passwordVisible variable setState(() { _passwordVisible = !_passwordVisible; }); }, ), obscureText: !_passwordVisible, errorText: widget.errorText, focusNode: _focusNode, ); } } void wrongPasswordDialog(SessionID sessionId, OverlayDialogManager dialogManager, type, title, text) { dialogManager.dismissAll(); dialogManager.show((setState, close, context) { cancel() { close(); closeConnection(); } submit() { enterPasswordDialog(sessionId, dialogManager); } return CustomAlertDialog( title: null, content: msgboxContent(type, title, text), onSubmit: submit, onCancel: cancel, actions: [ dialogButton( 'Cancel', onPressed: cancel, isOutline: true, ), dialogButton( 'Retry', onPressed: submit, ), ]); }); } void enterPasswordDialog( SessionID sessionId, OverlayDialogManager dialogManager) async { await _connectDialog( sessionId, dialogManager, passwordController: TextEditingController(), ); } void enterUserLoginDialog( SessionID sessionId, OverlayDialogManager dialogManager) async { await _connectDialog( sessionId, dialogManager, osUsernameController: TextEditingController(), osPasswordController: TextEditingController(), ); } void enterUserLoginAndPasswordDialog( SessionID sessionId, OverlayDialogManager dialogManager) async { await _connectDialog( sessionId, dialogManager, osUsernameController: TextEditingController(), osPasswordController: TextEditingController(), passwordController: TextEditingController(), ); } _connectDialog( SessionID sessionId, OverlayDialogManager dialogManager, { TextEditingController? osUsernameController, TextEditingController? osPasswordController, TextEditingController? passwordController, }) async { var rememberPassword = false; if (passwordController != null) { rememberPassword = await bind.sessionGetRemember(sessionId: sessionId) ?? false; } var rememberAccount = false; if (osUsernameController != null) { rememberAccount = await bind.sessionGetRemember(sessionId: sessionId) ?? false; } dialogManager.dismissAll(); dialogManager.show((setState, close, context) { cancel() { close(); closeConnection(); } submit() { final osUsername = osUsernameController?.text.trim() ?? ''; final osPassword = osPasswordController?.text.trim() ?? ''; final password = passwordController?.text.trim() ?? ''; if (passwordController != null && password.isEmpty) return; if (rememberAccount) { bind.sessionPeerOption( sessionId: sessionId, name: 'os-username', value: osUsername); bind.sessionPeerOption( sessionId: sessionId, name: 'os-password', value: osPassword); } gFFI.login( osUsername, osPassword, sessionId, password, rememberPassword, ); close(); dialogManager.showLoading(translate('Logging in...'), onCancel: closeConnection); } descWidget(String text) { return Column( children: [ Align( alignment: Alignment.centerLeft, child: Text( text, maxLines: 3, softWrap: true, overflow: TextOverflow.ellipsis, style: TextStyle(fontSize: 16), ), ), Container( height: 8, ), ], ); } rememberWidget( String desc, bool remember, ValueChanged? onChanged, ) { return CheckboxListTile( contentPadding: const EdgeInsets.all(0), dense: true, controlAffinity: ListTileControlAffinity.leading, title: Text(desc), value: remember, onChanged: onChanged, ); } osAccountWidget() { if (osUsernameController == null || osPasswordController == null) { return Offstage(); } return Column( children: [ descWidget(translate('login_linux_tip')), DialogTextField( title: translate(DialogTextField.kUsernameTitle), controller: osUsernameController, prefixIcon: DialogTextField.kUsernameIcon, errorText: null, ), PasswordWidget( controller: osPasswordController, autoFocus: false, ), rememberWidget( translate('remember_account_tip'), rememberAccount, (v) { if (v != null) { setState(() => rememberAccount = v); } }, ), ], ); } passwdWidget() { if (passwordController == null) { return Offstage(); } return Column( children: [ descWidget(translate('verify_rustdesk_password_tip')), PasswordWidget( controller: passwordController, autoFocus: osUsernameController == null, ), rememberWidget( translate('Remember password'), rememberPassword, (v) { if (v != null) { setState(() => rememberPassword = v); } }, ), ], ); } return CustomAlertDialog( title: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon(Icons.password_rounded, color: MyTheme.accent), Text(translate('Password Required')).paddingOnly(left: 10), ], ), content: Column(mainAxisSize: MainAxisSize.min, children: [ osAccountWidget(), osUsernameController == null || passwordController == null ? Offstage() : Container(height: 12), passwdWidget(), ]), actions: [ dialogButton( 'Cancel', icon: Icon(Icons.close_rounded), onPressed: cancel, isOutline: true, ), dialogButton( 'OK', icon: Icon(Icons.done_rounded), onPressed: submit, ), ], onSubmit: submit, onCancel: cancel, ); }); } void showWaitUacDialog( SessionID sessionId, OverlayDialogManager dialogManager, String type) { dialogManager.dismissAll(); dialogManager.show( tag: '$sessionId-wait-uac', (setState, close, context) => CustomAlertDialog( title: null, content: msgboxContent(type, 'Wait', 'wait_accept_uac_tip'), actions: [ dialogButton( 'OK', icon: Icon(Icons.done_rounded), onPressed: close, ), ], )); } // Another username && password dialog? void showRequestElevationDialog( SessionID sessionId, OverlayDialogManager dialogManager) { RxString groupValue = ''.obs; RxString errUser = ''.obs; RxString errPwd = ''.obs; TextEditingController userController = TextEditingController(); TextEditingController pwdController = TextEditingController(); void onRadioChanged(String? value) { if (value != null) { groupValue.value = value; } } // TODO get from theme final double fontSizeNote = 13.00; Widget OptionRequestPermissions = Obx( () => Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ Radio( visualDensity: VisualDensity(horizontal: -4, vertical: -4), value: '', groupValue: groupValue.value, onChanged: onRadioChanged, ).marginOnly(right: 10), Expanded( child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ InkWell( hoverColor: Colors.transparent, onTap: () => groupValue.value = '', child: Text( translate('Ask the remote user for authentication'), ), ).marginOnly(bottom: 10), Text( translate('Choose this if the remote account is administrator'), style: TextStyle(fontSize: fontSizeNote), ), ], ).marginOnly(top: 3), ), ], ), ); Widget OptionCredentials = Obx( () => Row( crossAxisAlignment: CrossAxisAlignment.start, children: [ Radio( visualDensity: VisualDensity(horizontal: -4, vertical: -4), value: 'logon', groupValue: groupValue.value, onChanged: onRadioChanged, ).marginOnly(right: 10), Expanded( child: InkWell( hoverColor: Colors.transparent, onTap: () => onRadioChanged('logon'), child: Text( translate('Transmit the username and password of administrator'), ), ).marginOnly(top: 4), ), ], ), ); Widget UacNote = Container( padding: EdgeInsets.fromLTRB(10, 8, 8, 8), decoration: BoxDecoration( color: MyTheme.currentThemeMode() == ThemeMode.dark ? Color.fromARGB(135, 87, 87, 90) : Colors.grey[100], borderRadius: BorderRadius.circular(8), border: Border.all(color: Colors.grey), ), child: Row( children: [ Icon(Icons.info_outline_rounded, size: 20).marginOnly(right: 10), Expanded( child: Text( translate('still_click_uac_tip'), style: TextStyle( fontSize: fontSizeNote, fontWeight: FontWeight.normal), ), ) ], ), ); var content = Obx( () => Column( children: [ OptionRequestPermissions.marginOnly(bottom: 15), OptionCredentials, Offstage( offstage: 'logon' != groupValue.value, child: Column( children: [ UacNote.marginOnly(bottom: 10), DialogTextField( controller: userController, title: translate('Username'), hintText: translate('eg: admin'), prefixIcon: DialogTextField.kUsernameIcon, errorText: errUser.isEmpty ? null : errUser.value, ), PasswordWidget( controller: pwdController, autoFocus: false, errorText: errPwd.isEmpty ? null : errPwd.value, ), ], ).marginOnly(left: isDesktop ? 35 : 0), ).marginOnly(top: 10), ], ), ); dialogManager.dismissAll(); dialogManager.show(tag: '$sessionId-request-elevation', (setState, close, context) { void submit() { if (groupValue.value == 'logon') { if (userController.text.isEmpty) { errUser.value = translate('Empty Username'); return; } if (pwdController.text.isEmpty) { errPwd.value = translate('Empty Password'); return; } bind.sessionElevateWithLogon( sessionId: sessionId, username: userController.text, password: pwdController.text); } else { bind.sessionElevateDirect(sessionId: sessionId); } close(); showWaitUacDialog(sessionId, dialogManager, "wait-uac"); } return CustomAlertDialog( title: Text(translate('Request Elevation')), content: content, actions: [ dialogButton( 'Cancel', icon: Icon(Icons.close_rounded), onPressed: close, isOutline: true, ), dialogButton( 'OK', icon: Icon(Icons.done_rounded), onPressed: submit, ) ], onSubmit: submit, onCancel: close, ); }); } void showOnBlockDialog( SessionID sessionId, String type, String title, String text, OverlayDialogManager dialogManager, ) { if (dialogManager.existing('$sessionId-wait-uac') || dialogManager.existing('$sessionId-request-elevation')) { return; } dialogManager.show(tag: '$sessionId-$type', (setState, close, context) { void submit() { close(); showRequestElevationDialog(sessionId, dialogManager); } return CustomAlertDialog( title: null, content: msgboxContent(type, title, "${translate(text)}${type.contains('uac') ? '\n' : '\n\n'}${translate('request_elevation_tip')}"), actions: [ dialogButton('Wait', onPressed: close, isOutline: true), dialogButton('Request Elevation', onPressed: submit), ], onSubmit: submit, onCancel: close, ); }); } void showElevationError(SessionID sessionId, String type, String title, String text, OverlayDialogManager dialogManager) { dialogManager.show(tag: '$sessionId-$type', (setState, close, context) { void submit() { close(); showRequestElevationDialog(sessionId, dialogManager); } return CustomAlertDialog( title: null, content: msgboxContent(type, title, text), actions: [ dialogButton('Cancel', onPressed: () { close(); }, isOutline: true), if (text != 'No permission') dialogButton('Retry', onPressed: submit), ], onSubmit: submit, onCancel: close, ); }); } void showWaitAcceptDialog(SessionID sessionId, String type, String title, String text, OverlayDialogManager dialogManager) { dialogManager.dismissAll(); dialogManager.show((setState, close, context) { onCancel() { closeConnection(); } return CustomAlertDialog( title: null, content: msgboxContent(type, title, text), actions: [ dialogButton('Cancel', onPressed: onCancel, isOutline: true), ], onCancel: onCancel, ); }); } void showRestartRemoteDevice(PeerInfo pi, String id, SessionID sessionId, OverlayDialogManager dialogManager) async { final res = await dialogManager .show((setState, close, context) => CustomAlertDialog( title: Row(children: [ Icon(Icons.warning_rounded, color: Colors.redAccent, size: 28), Flexible( child: Text(translate("Restart remote device")) .paddingOnly(left: 10)), ]), content: Text( "${translate('Are you sure you want to restart')} \n${pi.username}@${pi.hostname}($id) ?"), actions: [ dialogButton( "Cancel", icon: Icon(Icons.close_rounded), onPressed: close, isOutline: true, ), dialogButton( "OK", icon: Icon(Icons.done_rounded), onPressed: () => close(true), ), ], onCancel: close, onSubmit: () => close(true), )); if (res == true) bind.sessionRestartRemoteDevice(sessionId: sessionId); } showSetOSPassword( SessionID sessionId, bool login, OverlayDialogManager dialogManager, String? osPassword, Function()? closeCallback, ) async { final controller = TextEditingController(); osPassword ??= await bind.sessionGetOption(sessionId: sessionId, arg: 'os-password') ?? ''; var autoLogin = await bind.sessionGetOption(sessionId: sessionId, arg: 'auto-login') != ''; controller.text = osPassword; dialogManager.show((setState, close, context) { closeWithCallback([dynamic]) { close(); if (closeCallback != null) closeCallback(); } submit() { var text = controller.text.trim(); bind.sessionPeerOption( sessionId: sessionId, name: 'os-password', value: text); bind.sessionPeerOption( sessionId: sessionId, name: 'auto-login', value: autoLogin ? 'Y' : ''); if (text != '' && login) { bind.sessionInputOsPassword(sessionId: sessionId, value: text); } closeWithCallback(); } return CustomAlertDialog( title: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon(Icons.password_rounded, color: MyTheme.accent), Text(translate('OS Password')).paddingOnly(left: 10), ], ), content: Column( mainAxisSize: MainAxisSize.min, children: [ PasswordWidget(controller: controller), CheckboxListTile( contentPadding: const EdgeInsets.all(0), dense: true, controlAffinity: ListTileControlAffinity.leading, title: Text( translate('Auto Login'), ), value: autoLogin, onChanged: (v) { if (v == null) return; setState(() => autoLogin = v); }, ), ], ), actions: [ dialogButton( "Cancel", icon: Icon(Icons.close_rounded), onPressed: closeWithCallback, isOutline: true, ), dialogButton( "OK", icon: Icon(Icons.done_rounded), onPressed: submit, ), ], onSubmit: submit, onCancel: closeWithCallback, ); }); } showSetOSAccount( SessionID sessionId, OverlayDialogManager dialogManager, ) async { final usernameController = TextEditingController(); final passwdController = TextEditingController(); var username = await bind.sessionGetOption(sessionId: sessionId, arg: 'os-username') ?? ''; var password = await bind.sessionGetOption(sessionId: sessionId, arg: 'os-password') ?? ''; usernameController.text = username; passwdController.text = password; dialogManager.show((setState, close, context) { submit() { final username = usernameController.text.trim(); final password = usernameController.text.trim(); bind.sessionPeerOption( sessionId: sessionId, name: 'os-username', value: username); bind.sessionPeerOption( sessionId: sessionId, name: 'os-password', value: password); close(); } descWidget(String text) { return Column( children: [ Align( alignment: Alignment.centerLeft, child: Text( text, maxLines: 3, softWrap: true, overflow: TextOverflow.ellipsis, style: TextStyle(fontSize: 16), ), ), Container( height: 8, ), ], ); } return CustomAlertDialog( title: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon(Icons.password_rounded, color: MyTheme.accent), Text(translate('OS Account')).paddingOnly(left: 10), ], ), content: Column( mainAxisSize: MainAxisSize.min, children: [ descWidget(translate("os_account_desk_tip")), DialogTextField( title: translate(DialogTextField.kUsernameTitle), controller: usernameController, prefixIcon: DialogTextField.kUsernameIcon, errorText: null, ), PasswordWidget(controller: passwdController), ], ), actions: [ dialogButton( "Cancel", icon: Icon(Icons.close_rounded), onPressed: close, isOutline: true, ), dialogButton( "OK", icon: Icon(Icons.done_rounded), onPressed: submit, ), ], onSubmit: submit, onCancel: close, ); }); } showAuditDialog(FFI ffi) async { final controller = TextEditingController(text: ffi.auditNote); ffi.dialogManager.show((setState, close, context) { submit() { var text = controller.text; bind.sessionSendNote(sessionId: ffi.sessionId, note: text); ffi.auditNote = text; close(); } late final focusNode = FocusNode( onKey: (FocusNode node, RawKeyEvent evt) { if (evt.logicalKey.keyLabel == 'Enter') { if (evt is RawKeyDownEvent) { int pos = controller.selection.base.offset; controller.text = '${controller.text.substring(0, pos)}\n${controller.text.substring(pos)}'; controller.selection = TextSelection.fromPosition(TextPosition(offset: pos + 1)); } return KeyEventResult.handled; } if (evt.logicalKey.keyLabel == 'Esc') { if (evt is RawKeyDownEvent) { close(); } return KeyEventResult.handled; } else { return KeyEventResult.ignored; } }, ); return CustomAlertDialog( title: Text(translate('Note')), content: SizedBox( width: 250, height: 120, child: TextField( autofocus: true, keyboardType: TextInputType.multiline, textInputAction: TextInputAction.newline, decoration: const InputDecoration.collapsed( hintText: 'input note here', ), maxLines: null, maxLength: 256, controller: controller, focusNode: focusNode, )), actions: [ dialogButton('Cancel', onPressed: close, isOutline: true), dialogButton('OK', onPressed: submit) ], onSubmit: submit, onCancel: close, ); }); } void showConfirmSwitchSidesDialog( SessionID sessionId, String id, OverlayDialogManager dialogManager) async { dialogManager.show((setState, close, context) { submit() async { await bind.sessionSwitchSides(sessionId: sessionId); closeConnection(id: id); } return CustomAlertDialog( content: msgboxContent('info', 'Switch Sides', 'Please confirm if you want to share your desktop?'), actions: [ dialogButton('Cancel', onPressed: close, isOutline: true), dialogButton('OK', onPressed: submit), ], onSubmit: submit, onCancel: close, ); }); } customImageQualityDialog(SessionID sessionId, String id, FFI ffi) async { double initQuality = kDefaultQuality; double initFps = kDefaultFps; bool qualitySet = false; bool fpsSet = false; bool? direct; try { direct = ConnectionTypeState.find(id).direct.value == ConnectionType.strDirect; } catch (_) {} bool hideFps = (await bind.mainIsUsingPublicServer() && direct != true) || versionCmp(ffi.ffiModel.pi.version, '1.2.0') < 0; bool hideMoreQuality = (await bind.mainIsUsingPublicServer() && direct != true) || versionCmp(ffi.ffiModel.pi.version, '1.2.2') < 0; setCustomValues({double? quality, double? fps}) async { debugPrint("setCustomValues quality:$quality, fps:$fps"); if (quality != null) { qualitySet = true; await bind.sessionSetCustomImageQuality( sessionId: sessionId, value: quality.toInt()); } if (fps != null) { fpsSet = true; await bind.sessionSetCustomFps(sessionId: sessionId, fps: fps.toInt()); } if (!qualitySet) { qualitySet = true; await bind.sessionSetCustomImageQuality( sessionId: sessionId, value: initQuality.toInt()); } if (!hideFps && !fpsSet) { fpsSet = true; await bind.sessionSetCustomFps( sessionId: sessionId, fps: initFps.toInt()); } } final btnClose = dialogButton('Close', onPressed: () async { await setCustomValues(); ffi.dialogManager.dismissAll(); }); // quality final quality = await bind.sessionGetCustomImageQuality(sessionId: sessionId); initQuality = quality != null && quality.isNotEmpty ? quality[0].toDouble() : kDefaultQuality; if (initQuality < kMinQuality || initQuality > (!hideMoreQuality ? kMaxMoreQuality : kMaxQuality)) { initQuality = kDefaultQuality; } // fps final fpsOption = await bind.sessionGetOption(sessionId: sessionId, arg: 'custom-fps'); initFps = fpsOption == null ? kDefaultFps : double.tryParse(fpsOption) ?? kDefaultFps; if (initFps < kMinFps || initFps > kMaxFps) { initFps = kDefaultFps; } final content = customImageQualityWidget( initQuality: initQuality, initFps: initFps, setQuality: (v) => setCustomValues(quality: v), setFps: (v) => setCustomValues(fps: v), showFps: !hideFps, showMoreQuality: !hideMoreQuality); msgBoxCommon(ffi.dialogManager, 'Custom Image Quality', content, [btnClose]); } void deletePeerConfirmDialog(Function onSubmit, String title) async { gFFI.dialogManager.show( (setState, close, context) { submit() async { await onSubmit(); close(); } return CustomAlertDialog( title: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon( Icons.delete_rounded, color: Colors.red, ), Expanded( child: Text(title, overflow: TextOverflow.ellipsis).paddingOnly( left: 10, ), ), ], ), content: SizedBox.shrink(), actions: [ dialogButton( "Cancel", icon: Icon(Icons.close_rounded), onPressed: close, isOutline: true, ), dialogButton( "OK", icon: Icon(Icons.done_rounded), onPressed: submit, ), ], onSubmit: submit, onCancel: close, ); }, ); } void editAbTagDialog( List currentTags, Function(List) onSubmit) { var isInProgress = false; final tags = List.of(gFFI.abModel.tags); var selectedTag = currentTags.obs; gFFI.dialogManager.show((setState, close, context) { submit() async { setState(() { isInProgress = true; }); await onSubmit(selectedTag); close(); } return CustomAlertDialog( title: Text(translate("Edit Tag")), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( padding: const EdgeInsets.symmetric(vertical: 8.0), child: Wrap( children: tags .map((e) => AddressBookTag( name: e, tags: selectedTag, onTap: () { if (selectedTag.contains(e)) { selectedTag.remove(e); } else { selectedTag.add(e); } }, showActionMenu: false)) .toList(growable: false), ), ), // NOT use Offstage to wrap LinearProgressIndicator if (isInProgress) const LinearProgressIndicator(), ], ), actions: [ dialogButton("Cancel", onPressed: close, isOutline: true), dialogButton("OK", onPressed: submit), ], onSubmit: submit, onCancel: close, ); }); } void renameDialog( {required String oldName, FormFieldValidator? validator, required ValueChanged onSubmit, Function? onCancel}) async { RxBool isInProgress = false.obs; var controller = TextEditingController(text: oldName); final formKey = GlobalKey(); gFFI.dialogManager.show((setState, close, context) { submit() async { String text = controller.text.trim(); if (validator != null && formKey.currentState?.validate() == false) { return; } isInProgress.value = true; onSubmit(text); close(); isInProgress.value = false; } cancel() { onCancel?.call(); close(); } return CustomAlertDialog( title: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Icon(Icons.edit_rounded, color: MyTheme.accent), Text(translate('Rename')).paddingOnly(left: 10), ], ), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Container( child: Form( key: formKey, child: TextFormField( controller: controller, autofocus: true, decoration: InputDecoration(labelText: translate('Name')), validator: validator, ), ), ), // NOT use Offstage to wrap LinearProgressIndicator Obx(() => isInProgress.value ? const LinearProgressIndicator() : Offstage()) ], ), actions: [ dialogButton( "Cancel", icon: Icon(Icons.close_rounded), onPressed: cancel, isOutline: true, ), dialogButton( "OK", icon: Icon(Icons.done_rounded), onPressed: submit, ), ], onSubmit: submit, onCancel: cancel, ); }); } void change2fa({Function()? callback}) async { if (bind.mainHasValid2FaSync()) { await bind.mainSetOption(key: "2fa", value: ""); callback?.call(); return; } var new2fa = (await bind.mainGenerate2Fa()); final secretRegex = RegExp(r'secret=([^&]+)'); final secret = secretRegex.firstMatch(new2fa)?.group(1); String? errorText; final controller = TextEditingController(); gFFI.dialogManager.show((setState, close, context) { onVerify() async { if (await bind.mainVerify2Fa(code: controller.text.trim())) { callback?.call(); close(); } else { errorText = translate('wrong-2fa-code'); } } final codeField = Dialog2FaField( controller: controller, errorText: errorText, onChanged: () => setState(() => errorText = null), title: translate('Verification code'), readyCallback: () { onVerify(); setState(() {}); }, ); getOnSubmit() => codeField.isReady ? onVerify : null; return CustomAlertDialog( title: Text(translate("enable-2fa-title")), content: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ SelectableText(translate("enable-2fa-desc"), style: TextStyle(fontSize: 12)) .marginOnly(bottom: 12), SizedBox( width: 160, height: 160, child: QrImageView( backgroundColor: Colors.white, data: new2fa, version: QrVersions.auto, size: 160, gapless: false, )).marginOnly(bottom: 6), SelectableText(secret ?? '', style: TextStyle(fontSize: 12)) .marginOnly(bottom: 12), Row(children: [Expanded(child: codeField)]), ], ), actions: [ dialogButton("Cancel", onPressed: close, isOutline: true), dialogButton("OK", onPressed: getOnSubmit()), ], onCancel: close, ); }); } void enter2FaDialog( SessionID sessionId, OverlayDialogManager dialogManager) async { final controller = TextEditingController(); final RxBool submitReady = false.obs; dialogManager.dismissAll(); dialogManager.show((setState, close, context) { cancel() { close(); closeConnection(); } submit() { gFFI.send2FA(sessionId, controller.text.trim()); close(); dialogManager.showLoading(translate('Logging in...'), onCancel: closeConnection); } late Dialog2FaField codeField; codeField = Dialog2FaField( controller: controller, title: translate('Verification code'), onChanged: () => submitReady.value = codeField.isReady, ); return CustomAlertDialog( title: Text(translate('enter-2fa-title')), content: codeField, actions: [ dialogButton( 'Cancel', onPressed: cancel, isOutline: true, ), Obx(() => dialogButton( 'OK', onPressed: submitReady.isTrue ? submit : null, )), ]); }); }