import 'package:flutter/material.dart'; import 'package:provider/provider.dart'; import 'package:flutter/services.dart'; import 'dart:ui' as ui; import 'package:flutter_easyloading/flutter_easyloading.dart'; import 'dart:async'; import 'package:tuple/tuple.dart'; import 'package:wakelock/wakelock.dart'; import 'common.dart'; import 'model.dart'; final initText = '\1' * 1024; class RemotePage extends StatefulWidget { RemotePage({Key key, this.id}) : super(key: key); final String id; @override _RemotePageState createState() => _RemotePageState(); } class _RemotePageState extends State { Timer _interval; Timer _timer; bool _showBar = true; double _bottom = 0; String _value = ''; double _xOffset = 0; double _yOffset = 0; double _xOffset0 = 0; double _yOffset0 = 0; double _scale = 1; bool _mouseTools = false; var _drag = false; var _right = false; var _scroll = false; var _more = true; var _fn = false; final FocusNode _focusNode = FocusNode(); var _showEdit = true; var _reconnects = 1; @override void initState() { super.initState(); FFI.connect(widget.id); WidgetsBinding.instance.addPostFrameCallback((_) { SystemChrome.setEnabledSystemUIOverlays([]); showLoading('Connecting...', context); _interval = Timer.periodic(Duration(milliseconds: 30), (timer) => interval()); }); Wakelock.enable(); } @override void dispose() { _focusNode.dispose(); super.dispose(); FFI.close(); _interval.cancel(); _timer?.cancel(); dismissLoading(); SystemChrome.setEnabledSystemUIOverlays(SystemUiOverlay.values); Wakelock.disable(); } void resetTool() { _scroll = _drag = _right = false; FFI.resetModifiers(); } void interval() { var v = MediaQuery.of(context).viewInsets.bottom; if (v != _bottom) { resetTool(); setState(() { _bottom = v; if (v < 100) { SystemChrome.setEnabledSystemUIOverlays([]); } }); } FFI.ffiModel.update(widget.id, context, handleMsgbox); } void handleMsgbox(Map evt, String id) { var type = evt['type']; var title = evt['title']; var text = evt['text']; if (type == 're-input-password') { wrongPasswordDialog(id, context); } else if (type == 'input-password') { enterPasswordDialog(id, context); } else { showMsgBox(type, title, text); } } void showMsgBox(String type, String title, String text) { msgbox(type, title, text, context); final hasRetry = type == "error" && title == "Connection Error" && text.toLowerCase().indexOf("offline") < 0 && text.toLowerCase().indexOf("exist") < 0 && text.toLowerCase().indexOf("handshake") < 0 && text.toLowerCase().indexOf("failed") < 0 && text.toLowerCase().indexOf("resolve") < 0 && text.toLowerCase().indexOf("manually") < 0; if (hasRetry) { _timer?.cancel(); _timer = Timer(Duration(seconds: _reconnects), () { FFI.reconnect(); showLoading('Connecting...', context); }); _reconnects *= 2; } else { _reconnects = 1; } } void handleInput(String newValue) { if (_value[0] == '\1' && newValue[0] != '\1') { // clipboard _value = ''; } if (newValue.length <= _value.length) { final char = 'VK_BACK'; FFI.inputKey(char); } else { final content = newValue.substring(_value.length); if (content.length > 1) { if (_value != '' && content.length == 2 && (content == '""' || content == '()' || content == '[]' || content == '<>' || content == "{}" || content == '”“' || content == '《》' || content == '()' || content == '【】')) { // can not only input content[0], because when input ], [ are also auo insert, which cause ] never be input FFI.setByName('input_string', content); openKeyboard(); return; } FFI.setByName('input_string', content); } else { var char = content; if (char == '\n') { char = 'VK_RETURN'; } FFI.inputKey(char); } } _value = newValue; } void openKeyboard() { // destroy first, so that our _value trick can work _value = initText; setState(() => _showEdit = false); _timer?.cancel(); _timer = Timer(Duration(milliseconds: 30), () { // show now, and sleep a while to requestFocus to // make sure edit ready, so that keyboard wont show/hide/show/hide happen setState(() => _showEdit = true); _timer?.cancel(); _timer = Timer(Duration(milliseconds: 30), () { SystemChrome.setEnabledSystemUIOverlays(SystemUiOverlay.values); SystemChannels.textInput.invokeMethod('TextInput.show'); _focusNode.requestFocus(); }); }); } @override Widget build(BuildContext context) { final pi = Provider.of(context).pi; EasyLoading.instance.loadingStyle = EasyLoadingStyle.light; return WillPopScope( onWillPop: () async { close(); return false; }, child: Scaffold( floatingActionButton: _showBar ? null : FloatingActionButton( mini: true, child: Icon(Icons.expand_less), backgroundColor: MyTheme.accent50, onPressed: () { setState(() => _showBar = !_showBar); }), bottomNavigationBar: _showBar && pi.displays != null ? BottomAppBar( elevation: 10, color: MyTheme.accent, child: Row( mainAxisSize: MainAxisSize.max, mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Row(children: [ IconButton( color: Colors.white, icon: Icon(Icons.clear), onPressed: () { close(); }, ), IconButton( color: Colors.white, icon: Icon(Icons.keyboard), onPressed: openKeyboard), IconButton( color: Colors.white, icon: Icon(Icons.tv), onPressed: () { setState(() => _showEdit = false); showOptions(context); }, ), Container( color: _mouseTools ? Colors.blue[500] : null, child: IconButton( color: Colors.white, icon: Icon(Icons.mouse), onPressed: () { setState(() { _mouseTools = !_mouseTools; resetTool(); }); }, )), IconButton( color: Colors.white, icon: Icon(Icons.more_vert), onPressed: () { setState(() => _showEdit = false); showActions(context); }, ), ]), IconButton( color: Colors.white, icon: Icon(Icons.expand_more), onPressed: () { setState(() => _showBar = !_showBar); }), ], ), ) : null, body: FlutterEasyLoading( child: GestureDetector( onTap: () { if (_drag || _scroll) return; FFI.tap(_right); }, onLongPressStart: (_) { if (_drag) { // case: to show password on windows FFI.sendMouse('down', 'left'); } }, onLongPressEnd: (_) { if (_drag) { FFI.sendMouse('up', 'left'); } }, onScaleStart: (details) { _scale = 1; _xOffset = _xOffset0 = details.focalPoint.dx; _yOffset = _yOffset0 = details.focalPoint.dy; if (_drag) { FFI.sendMouse('down', 'left'); } }, onScaleUpdate: (details) { var scale = details.scale; if (scale == 1) { if (!_scroll) { var x = details.focalPoint.dx; var y = details.focalPoint.dy; var dx = x - _xOffset; var dy = y - _yOffset; FFI.cursorModel.updatePan(dx, dy); _xOffset = x; _yOffset = y; } else { _xOffset = details.focalPoint.dx; _yOffset = details.focalPoint.dy; } } else if (!_drag && !_scroll) { FFI.canvasModel.updateScale(scale / _scale); _scale = scale; } }, onScaleEnd: (details) { if (_drag) { FFI.sendMouse('up', 'left'); } else if (_scroll) { var dy = (_yOffset - _yOffset0) / 10; if (dy.abs() > 0.1) { if (dy > 0 && dy < 1) dy = 1; if (dy < 0 && dy > -1) dy = -1; FFI.scroll(dy); } } }, child: Container( color: MyTheme.canvasColor, child: Stack(children: [ ImagePaint(), CursorPaint(), getHelpTools(), SizedBox( width: 0, height: 0, child: !_showEdit ? Container() : TextFormField( textInputAction: TextInputAction.newline, autocorrect: false, enableSuggestions: false, focusNode: _focusNode, maxLines: null, initialValue: _value, // trick way to make backspace work always keyboardType: TextInputType.multiline, onChanged: handleInput, ), ), ]))), )), ); } void close() { msgbox('', 'Close', 'Are you sure to close the connection?', context); } Widget getHelpTools() { final keyboard = _bottom >= 100; if (!_mouseTools && !keyboard) { return SizedBox(); } var wrap = (String text, void Function() onPressed, [bool active, IconData icon]) { return ButtonTheme( padding: EdgeInsets.symmetric( vertical: icon != null ? 3 : 6, horizontal: 6), //adds padding inside the button materialTapTargetSize: MaterialTapTargetSize .shrinkWrap, //limits the touch area to the button area minWidth: 0, //wraps child's width height: 0, child: FlatButton( splashColor: MyTheme.accent, shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(5.0), ), color: active == true ? MyTheme.accent80 : null, child: icon != null ? Icon(icon, size: 17, color: Colors.white) : Text(text, style: TextStyle(color: Colors.white, fontSize: 11)), onPressed: onPressed)); }; final mouse = [ wrap('Drag', () { setState(() { _drag = !_drag; if (_drag) { _scroll = false; _right = false; } }); }, _drag), wrap('Scroll', () { setState(() { _scroll = !_scroll; if (_scroll) { _drag = false; _right = false; } }); }, _scroll), wrap('Right', () { setState(() { _right = !_right; if (_right) { _scroll = false; _drag = false; } }); }, _right) ]; final modifiers = [ wrap('Ctrl', () { setState(() => FFI.ctrl = !FFI.ctrl); }, FFI.ctrl), wrap('Alt', () { setState(() => FFI.alt = !FFI.alt); }, FFI.alt), wrap('Shift', () { setState(() => FFI.shift = !FFI.shift); }, FFI.shift), wrap('Cmd', () { setState(() => FFI.command = !FFI.command); }, FFI.command), ]; final keys = [ wrap( 'Fn', () => setState( () { _fn = !_fn; if (_fn) { _more = false; } }, ), _fn), wrap( '...', () => setState( () { _more = !_more; if (_more) { _fn = false; } }, ), _more), ]; final fn = [ SizedBox(width: 9999), ]; for (var i = 1; i <= 12; ++i) { final name = 'F' + i.toString(); fn.add(wrap(name, () { FFI.inputKey('VK_' + name); })); } final more = [ SizedBox(width: 9999), wrap('Esc', () { FFI.inputKey('VK_ESCAPE'); }), wrap('Tab', () { FFI.inputKey('VK_TAB'); }), wrap('Home', () { FFI.inputKey('VK_HOME'); }), wrap('End', () { FFI.inputKey('VK_END'); }), wrap('Del', () { FFI.inputKey('VK_DELETE'); }), wrap('PgUp', () { FFI.inputKey('VK_PRIOR'); }), wrap('PgDown', () { FFI.inputKey('VK_NEXT'); }), SizedBox(width: 9999), wrap('', () { FFI.inputKey('VK_LEFT'); }, false, Icons.keyboard_arrow_left), wrap('', () { FFI.inputKey('VK_UP'); }, false, Icons.keyboard_arrow_up), wrap('', () { FFI.inputKey('VK_DOWN'); }, false, Icons.keyboard_arrow_down), wrap('', () { FFI.inputKey('VK_RIGHT'); }, false, Icons.keyboard_arrow_right), wrap('Ctrl+C', () { var old = FFI.ctrl; FFI.ctrl = true; FFI.inputKey( 'VK_C', ); FFI.ctrl = old; }), wrap('Ctrl+S', () { var old = FFI.ctrl; FFI.ctrl = true; FFI.inputKey( 'VK_S', ); FFI.ctrl = old; }), ]; return Container( color: Color(0xAA000000), padding: EdgeInsets.only( top: keyboard ? 24 : 4, left: 8, right: 8, bottom: 8), child: Wrap( spacing: 4, runSpacing: 4, children: [SizedBox(width: 9999)] + (keyboard ? modifiers + keys + (_fn ? fn : []) + (_more ? more : []) : mouse + modifiers), )); } } class ImagePaint extends StatelessWidget { @override Widget build(BuildContext context) { final m = Provider.of(context); final c = Provider.of(context); final adjust = FFI.cursorModel.adjustForKeyboard(); var s = c.scale; return CustomPaint( painter: new ImagePainter( image: m.image, x: c.x / s, y: (c.y - adjust) / s, scale: s), ); } } class CursorPaint extends StatelessWidget { @override Widget build(BuildContext context) { final m = Provider.of(context); final c = Provider.of(context); final adjust = FFI.cursorModel.adjustForKeyboard(); var s = c.scale; return CustomPaint( painter: new ImagePainter( image: m.image, x: m.x * s - m.hotx + c.x, y: m.y * s - m.hoty + c.y - adjust, scale: 1), ); } } class ImagePainter extends CustomPainter { ImagePainter({ this.image, this.x, this.y, this.scale, }); ui.Image image; double x; double y; double scale; @override void paint(Canvas canvas, Size size) { if (image == null) return; canvas.scale(scale, scale); canvas.drawImage(image, new Offset(x, y), new Paint()); } @override bool shouldRepaint(CustomPainter oldDelegate) { return oldDelegate != this; } } void enterPasswordDialog(String id, BuildContext context) { final controller = TextEditingController(); var remember = FFI.getByName('remember', id) == 'true'; showAlertDialog( context, (setState) => Tuple3( Text('Password required'), Column(mainAxisSize: MainAxisSize.min, children: [ PasswordWidget(controller: controller), CheckboxListTile( contentPadding: const EdgeInsets.all(0), dense: true, controlAffinity: ListTileControlAffinity.leading, title: Text( 'Remember password', ), value: remember, onChanged: (v) { setState(() => remember = v); }, ), ]), [ FlatButton( textColor: MyTheme.accent, onPressed: () { Navigator.pop(context); Navigator.pop(context); }, child: Text('Cancel'), ), FlatButton( textColor: MyTheme.accent, onPressed: () { var text = controller.text.trim(); if (text == '') return; FFI.login(text, remember); showLoading('Logging in...', null); Navigator.pop(context); }, child: Text('OK'), ), ], )); } void wrongPasswordDialog(String id, BuildContext context) { showAlertDialog( context, (_) => Tuple3(Text('Wrong Password'), Text('Do you want to enter again?'), [ FlatButton( textColor: MyTheme.accent, onPressed: () { Navigator.pop(context); Navigator.pop(context); }, child: Text('Cancel'), ), FlatButton( textColor: MyTheme.accent, onPressed: () { enterPasswordDialog(id, context); }, child: Text('Retry'), ), ])); } void showOptions(BuildContext context) { String quality = FFI.getByName('image_quality'); if (quality == '') quality = 'balanced'; var displays = []; final pi = FFI.ffiModel.pi; final image = FFI.ffiModel.getConnectionImage(); if (image != null) displays.add(Padding(padding: const EdgeInsets.only(top: 8), child: image)); if (pi.displays.length > 1) { final cur = pi.currentDisplay; final children = []; for (var i = 0; i < pi.displays.length; ++i) children.add(InkWell( onTap: () { if (i == cur) return; FFI.setByName('switch_display', i.toString()); Navigator.pop(context); }, child: Ink( width: 40, height: 40, decoration: BoxDecoration( border: Border.all(color: Colors.black87), color: i == cur ? Colors.black87 : Colors.white), child: Center( child: Text((i + 1).toString(), style: TextStyle( color: i == cur ? Colors.white : Colors.black87)))))); displays.add(Padding( padding: const EdgeInsets.only(top: 8), child: Wrap( alignment: WrapAlignment.center, spacing: 8, children: children, ))); } if (displays.isNotEmpty) { displays.add(Divider(color: MyTheme.border)); } showAlertDialog(context, (setState) { final more = []; if (FFI.ffiModel.permissions['audio'] != false) { more.add(CheckboxListTile( value: FFI.getByName('toggle_option', 'disable-audio') == 'true', onChanged: (v) { setState(() { FFI.setByName('toggle_option', 'disable-audio'); }); }, title: Text('Mute'))); } if (FFI.ffiModel.permissions['keyboard'] != false) { more.add(CheckboxListTile( value: FFI.getByName('toggle_option', 'lock-after-session-end') == 'true', onChanged: (v) { setState(() { FFI.setByName('toggle_option', 'lock-after-session-end'); }); }, title: Text('Lock after session end'))); } return Tuple3( null, Column( mainAxisSize: MainAxisSize.min, children: displays + [ RadioListTile( controlAffinity: ListTileControlAffinity.trailing, title: const Text('Good image quality'), value: 'best', groupValue: quality, onChanged: (String value) { setState(() { quality = value; FFI.setByName('image_quality', value); }); }, ), RadioListTile( controlAffinity: ListTileControlAffinity.trailing, title: const Text('Balanced'), value: 'balanced', groupValue: quality, onChanged: (String value) { setState(() { quality = value; FFI.setByName('image_quality', value); }); }, ), RadioListTile( controlAffinity: ListTileControlAffinity.trailing, title: const Text('Optimize reaction time'), value: 'low', groupValue: quality, onChanged: (String value) { setState(() { quality = value; FFI.setByName('image_quality', value); }); }, ), Divider(color: MyTheme.border), CheckboxListTile( value: FFI.getByName( 'toggle_option', 'show-remote-cursor') == 'true', onChanged: (v) { setState(() { FFI.setByName('toggle_option', 'show-remote-cursor'); }); }, title: Text('Show remote cursor')), ] + more), null); }, () async => true, true, 0); } void showActions(BuildContext context) { final size = MediaQuery.of(context).size; final x = 120.0; final y = size.height; final more = >[]; if (FFI.ffiModel.pi.version.isNotEmpty) { more.add(PopupMenuItem(child: Text('Refresh'), value: 'refresh')); } if (FFI.ffiModel.permissions['keyboard'] != false && FFI.ffiModel.permissions['clipboard'] != false) { more.add(PopupMenuItem(child: Text('Paste'), value: 'paste')); } more.add(PopupMenuItem( child: Row( children: ([ Text('OS Password'), FlatButton( textColor: MyTheme.accent, onPressed: () { showSetOSPassword(context); Navigator.pop(context); }, child: Icon(Icons.edit), ) ])), value: 'enter_os_password')); () async { var value = await showMenu( context: context, position: RelativeRect.fromLTRB(x, y, x, y), items: [ PopupMenuItem( child: Text('Insert Ctrl + Alt + Del'), value: 'cad'), PopupMenuItem(child: Text('Insert Lock'), value: 'lock'), ] + more, elevation: 8, ); if (value == 'cad') { FFI.setByName('ctrl_alt_del'); } else if (value == 'lock') { FFI.setByName('lock_screen'); } else if (value == 'refresh') { FFI.setByName('refresh'); } else if (value == 'paste') { () async { ClipboardData data = await Clipboard.getData(Clipboard.kTextPlain); if (data.text != null) { FFI.setByName('input_string', '${data.text}'); } }(); } else if (value == 'enter_os_password') { () async { var password = await getPassword(FFI.id); if (password != "") { var x = FFI.cursorModel.x; var y = FFI.cursorModel.y; FFI.moveMouse(x + 3, y + 3); await Future.delayed(Duration(milliseconds: 50)); FFI.moveMouse(x, y); await Future.delayed(Duration(milliseconds: 50)); FFI.tap(true); await Future.delayed(Duration(milliseconds: 300)); FFI.setByName('input_string', password); FFI.inputKey('VK_RETURN'); } }(); } }(); } void showSetOSPassword(BuildContext context) async { final controller = TextEditingController(); var password = await getPassword(FFI.id); controller.text = password; showAlertDialog( context, (setState) => Tuple3( Text('Password required'), Column(mainAxisSize: MainAxisSize.min, children: [ PasswordWidget(controller: controller), ]), [ FlatButton( textColor: MyTheme.accent, onPressed: () { Navigator.pop(context); }, child: Text('Cancel'), ), FlatButton( textColor: MyTheme.accent, onPressed: () { var text = controller.text.trim(); savePassword(FFI.id, text); Navigator.pop(context); }, child: Text('OK'), ), ], )); }