rustdesk/lib/model.dart

807 lines
21 KiB
Dart
Raw Normal View History

2020-11-28 13:22:19 +08:00
import 'package:flutter/services.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'dart:math';
2020-11-19 00:32:46 +08:00
import 'dart:convert';
import 'dart:typed_data';
import 'dart:ui' as ui;
import 'package:flutter/material.dart';
2020-11-23 23:18:42 +08:00
import 'package:tuple/tuple.dart';
2020-11-19 00:32:46 +08:00
import 'dart:async';
import 'common.dart';
2022-01-26 12:48:16 +08:00
import 'native_model.dart' if (dart.library.html) 'web_model.dart';
2020-11-19 00:32:46 +08:00
class FfiModel with ChangeNotifier {
2020-11-22 21:08:19 +08:00
PeerInfo _pi;
Display _display;
2020-11-28 17:42:29 +08:00
var _decoding = false;
2020-11-22 21:08:19 +08:00
bool _waitForImage;
2020-11-24 22:03:04 +08:00
bool _initialized = false;
2022-02-02 00:46:21 +08:00
var _inputBlocked = false;
2020-11-22 21:08:19 +08:00
final _permissions = Map<String, bool>();
bool _secure;
bool _direct;
2020-11-22 21:08:19 +08:00
get permissions => _permissions;
2020-11-24 22:03:04 +08:00
get initialized => _initialized;
get display => _display;
get secure => _secure;
get direct => _direct;
2020-11-25 23:52:58 +08:00
get pi => _pi;
2022-02-02 00:46:21 +08:00
get inputBlocked => _inputBlocked;
set inputBlocked(v) {
_inputBlocked = v;
}
2020-11-19 00:32:46 +08:00
FfiModel() {
2021-08-02 20:54:56 +08:00
Translator.call = translate;
2020-11-22 21:08:19 +08:00
clear();
() async {
2022-01-26 12:48:16 +08:00
await PlatformFFI.init();
2020-11-24 22:03:04 +08:00
_initialized = true;
2021-08-05 01:38:20 +08:00
print("FFI initialized");
2020-11-22 21:08:19 +08:00
notifyListeners();
}();
2020-11-19 00:32:46 +08:00
}
2020-11-22 21:08:19 +08:00
void updatePermission(Map<String, dynamic> evt) {
evt.forEach((k, v) {
if (k == 'name') return;
_permissions[k] = v == 'true';
});
2020-11-23 17:45:38 +08:00
print('$_permissions');
2020-11-19 00:32:46 +08:00
}
2020-11-24 23:36:46 +08:00
bool keyboard() => _permissions['keyboard'] != false;
2020-11-23 23:18:42 +08:00
2020-11-19 00:32:46 +08:00
void clear() {
2020-11-22 21:08:19 +08:00
_pi = PeerInfo();
_display = Display();
_waitForImage = false;
_secure = null;
_direct = null;
2022-02-02 00:46:21 +08:00
_inputBlocked = false;
2020-11-28 13:22:19 +08:00
clearPermissions();
}
void setConnectionType(bool secure, bool direct) {
_secure = secure;
_direct = direct;
}
Image getConnectionImage() {
String icon;
if (secure == true && direct == true) {
icon = 'secure';
} else if (secure == false && direct == true) {
icon = 'insecure';
} else if (secure == false && direct == false) {
icon = 'insecure_relay';
} else if (secure == true && direct == false) {
icon = 'secure_relay';
}
return icon == null
? null
: Image.asset('assets/$icon.png', width: 48, height: 48);
}
2020-11-28 13:22:19 +08:00
void clearPermissions() {
2022-02-02 00:46:21 +08:00
_inputBlocked = false;
2020-11-22 21:08:19 +08:00
_permissions.clear();
2020-11-19 00:32:46 +08:00
}
2020-11-27 10:52:09 +08:00
void update(
2020-11-19 21:59:49 +08:00
String id,
BuildContext context,
void Function(
Map<String, dynamic> evt,
String id,
)
2020-11-27 10:52:09 +08:00
handleMsgbox) {
2020-11-23 23:52:30 +08:00
var pos;
2020-11-19 18:22:06 +08:00
for (;;) {
var evt = FFI.popEvent();
if (evt == null) break;
2020-11-19 00:32:46 +08:00
var name = evt['name'];
if (name == 'msgbox') {
handleMsgbox(evt, id);
2020-11-19 00:32:46 +08:00
} else if (name == 'peer_info') {
2020-11-26 21:41:25 +08:00
handlePeerInfo(evt, context);
} else if (name == 'connection_ready') {
FFI.ffiModel.setConnectionType(
evt['secure'] == 'true', evt['direct'] == 'true');
2020-11-19 00:32:46 +08:00
} else if (name == 'switch_display') {
handleSwitchDisplay(evt);
} else if (name == 'cursor_data') {
FFI.cursorModel.updateCursorData(evt);
} else if (name == 'cursor_id') {
FFI.cursorModel.updateCursorId(evt);
} else if (name == 'cursor_position') {
2020-11-23 23:52:30 +08:00
pos = evt;
2020-11-28 13:22:19 +08:00
} else if (name == 'clipboard') {
Clipboard.setData(ClipboardData(text: evt['content']));
2020-11-22 21:08:19 +08:00
} else if (name == 'permission') {
FFI.ffiModel.updatePermission(evt);
2020-11-19 00:32:46 +08:00
}
}
2020-11-23 23:52:30 +08:00
if (pos != null) FFI.cursorModel.updateCursorPosition(pos);
2020-11-19 00:32:46 +08:00
if (!_decoding) {
2022-01-26 12:48:16 +08:00
var rgba = PlatformFFI.getRgba();
2020-11-19 00:32:46 +08:00
if (rgba != null) {
2020-11-19 00:53:10 +08:00
if (_waitForImage) {
_waitForImage = false;
dismissLoading();
}
2020-11-19 00:32:46 +08:00
_decoding = true;
2020-11-28 17:42:29 +08:00
final pid = FFI.id;
2020-11-19 00:32:46 +08:00
ui.decodeImageFromPixels(
rgba, _display.width, _display.height, ui.PixelFormat.bgra8888,
(image) {
2022-01-26 12:48:16 +08:00
PlatformFFI.clearRgbaFrame();
2020-11-19 00:32:46 +08:00
_decoding = false;
2020-11-28 17:42:29 +08:00
if (FFI.id != pid) return;
2020-11-19 00:32:46 +08:00
try {
// my throw exception, because the listener maybe already dispose
FFI.imageModel.update(image);
} catch (e) {
print('update image: $e');
}
2020-11-19 00:32:46 +08:00
});
}
2020-11-27 10:52:09 +08:00
}
2020-11-19 00:32:46 +08:00
}
void handleSwitchDisplay(Map<String, dynamic> evt) {
var old = _pi.currentDisplay;
2020-11-19 00:32:46 +08:00
_pi.currentDisplay = int.parse(evt['display']);
_display.x = double.parse(evt['x']);
_display.y = double.parse(evt['y']);
_display.width = int.parse(evt['width']);
_display.height = int.parse(evt['height']);
if (old != _pi.currentDisplay)
FFI.cursorModel.updateDisplayOrigin(_display.x, _display.y);
2020-11-25 23:52:58 +08:00
notifyListeners();
2020-11-19 00:32:46 +08:00
}
2020-11-26 21:41:25 +08:00
void handlePeerInfo(Map<String, dynamic> evt, BuildContext context) {
2020-11-19 00:32:46 +08:00
dismissLoading();
2020-11-27 17:34:09 +08:00
_pi.version = evt['version'];
2020-11-19 00:32:46 +08:00
_pi.username = evt['username'];
_pi.hostname = evt['hostname'];
_pi.platform = evt['platform'];
_pi.sasEnabled = evt['sas_enabled'] == "true";
_pi.currentDisplay = int.parse(evt['current_display']);
List<dynamic> displays = json.decode(evt['displays']);
_pi.displays = [];
2020-11-19 00:32:46 +08:00
for (int i = 0; i < displays.length; ++i) {
Map<String, dynamic> d0 = displays[i];
var d = Display();
d.x = d0['x'].toDouble();
d.y = d0['y'].toDouble();
d.width = d0['width'];
d.height = d0['height'];
_pi.displays.add(d);
}
if (_pi.currentDisplay < _pi.displays.length) {
_display = _pi.displays[_pi.currentDisplay];
}
2020-11-23 01:16:17 +08:00
if (displays.length > 0) {
2021-08-02 20:54:56 +08:00
showLoading(translate('Connected, waiting for image...'), context);
2020-11-19 00:53:10 +08:00
_waitForImage = true;
}
2020-11-29 14:28:07 +08:00
notifyListeners();
2020-11-19 00:32:46 +08:00
}
}
class ImageModel with ChangeNotifier {
ui.Image _image;
ui.Image get image => _image;
void update(ui.Image image) {
2020-11-24 22:03:04 +08:00
if (_image == null && image != null) {
2022-02-03 00:53:59 +08:00
if (isDesktop) {
FFI.canvasModel.updateViewStyle();
} else {
final size = MediaQueryData.fromWindow(ui.window).size;
final xscale = size.width / image.width;
final yscale = size.height / image.height;
FFI.canvasModel.scale = max(xscale, yscale);
}
2022-01-31 16:22:05 +08:00
initializeCursorAndCanvas();
2020-11-24 22:03:04 +08:00
}
2020-11-19 00:32:46 +08:00
_image = image;
2020-11-19 18:22:06 +08:00
if (image != null) notifyListeners();
2020-11-19 00:32:46 +08:00
}
double get maxScale {
if (_image == null) return 1.0;
final size = MediaQueryData.fromWindow(ui.window).size;
final xscale = size.width / _image.width;
final yscale = size.height / _image.height;
return max(1.0, max(xscale, yscale));
}
double get minScale {
if (_image == null) return 1.0;
final size = MediaQueryData.fromWindow(ui.window).size;
final xscale = size.width / _image.width;
final yscale = size.height / _image.height;
return min(xscale, yscale);
}
2020-11-19 00:32:46 +08:00
}
2020-11-23 23:18:42 +08:00
class CanvasModel with ChangeNotifier {
double _x;
double _y;
double _scale;
CanvasModel() {
clear();
}
double get x => _x;
double get y => _y;
double get scale => _scale;
2022-02-03 00:53:59 +08:00
void updateViewStyle() {
final s = FFI.getByName('peer_option', 'view-style');
final size = MediaQueryData.fromWindow(ui.window).size;
final s1 = size.width / FFI.ffiModel.display.width;
final s2 = size.height / FFI.ffiModel.display.height;
if (s == 'shrink') {
final s = s1 < s2 ? s1 : s2;
if (s < 1) {
_scale = s;
}
} else if (s == 'stretch') {
final s = s1 > s2 ? s1 : s2;
if (s > 1) {
_scale = s;
}
} else {
_scale = 1;
}
_x = (size.width - FFI.ffiModel.display.width * _scale) / 2;
_y = (size.height - FFI.ffiModel.display.height * _scale) / 2;
notifyListeners();
}
void update(double x, double y, double scale) {
_x = x;
_y = y;
_scale = scale;
notifyListeners();
}
set scale(v) {
_scale = v;
notifyListeners();
}
2020-11-24 22:03:04 +08:00
void panX(double dx) {
_x += dx;
notifyListeners();
2020-11-23 23:18:42 +08:00
}
2020-11-27 17:59:42 +08:00
void resetOffset() {
2022-02-03 00:53:59 +08:00
if (isDesktop) {
updateViewStyle();
} else {
_x = 0;
_y = 0;
}
2020-11-27 17:59:42 +08:00
notifyListeners();
}
2020-11-24 22:03:04 +08:00
void panY(double dy) {
2020-11-23 23:18:42 +08:00
_y += dy;
notifyListeners();
}
void updateScale(double v) {
2020-11-27 10:52:09 +08:00
if (FFI.imageModel.image == null) return;
2020-11-25 14:41:57 +08:00
final offset = FFI.cursorModel.offset;
var r = FFI.cursorModel.getVisibleRect();
final px0 = (offset.dx - r.left) * _scale;
final py0 = (offset.dy - r.top) * _scale;
2020-11-23 23:18:42 +08:00
_scale *= v;
final maxs = FFI.imageModel.maxScale;
final mins = FFI.imageModel.minScale;
if (_scale > maxs) _scale = maxs;
if (_scale < mins) _scale = mins;
2020-11-25 14:41:57 +08:00
r = FFI.cursorModel.getVisibleRect();
final px1 = (offset.dx - r.left) * _scale;
final py1 = (offset.dy - r.top) * _scale;
_x -= px1 - px0;
_y -= py1 - py0;
2020-11-23 23:18:42 +08:00
notifyListeners();
}
2022-01-25 18:13:11 +08:00
void clear([bool notify = false]) {
2020-11-23 23:18:42 +08:00
_x = 0;
_y = 0;
_scale = 1.0;
2021-08-22 07:50:12 +08:00
if (notify) notifyListeners();
2020-11-23 23:18:42 +08:00
}
}
2020-11-19 00:32:46 +08:00
class CursorModel with ChangeNotifier {
ui.Image _image;
2020-11-23 23:18:42 +08:00
final _images = Map<int, Tuple3<ui.Image, double, double>>();
2020-11-22 18:29:04 +08:00
double _x = -10000;
double _y = -10000;
2020-11-19 00:32:46 +08:00
double _hotx = 0;
double _hoty = 0;
double _displayOriginX = 0;
double _displayOriginY = 0;
ui.Image get image => _image;
2020-11-23 23:18:42 +08:00
double get x => _x - _displayOriginX;
double get y => _y - _displayOriginY;
2020-11-25 14:41:57 +08:00
Offset get offset => Offset(_x, _y);
2020-11-23 23:18:42 +08:00
double get hotx => _hotx;
double get hoty => _hoty;
2020-11-19 00:32:46 +08:00
2020-11-25 14:41:57 +08:00
// remote physical display coordinate
2020-11-24 22:03:04 +08:00
Rect getVisibleRect() {
final size = MediaQueryData.fromWindow(ui.window).size;
final xoffset = FFI.canvasModel.x;
final yoffset = FFI.canvasModel.y;
final scale = FFI.canvasModel.scale;
final x0 = _displayOriginX - xoffset / scale;
final y0 = _displayOriginY - yoffset / scale;
return Rect.fromLTWH(x0, y0, size.width / scale, size.height / scale);
}
2020-11-25 11:20:40 +08:00
double adjustForKeyboard() {
2020-12-21 17:26:23 +08:00
final m = MediaQueryData.fromWindow(ui.window);
var keyboardHeight = m.viewInsets.bottom;
final size = m.size;
2020-11-25 11:20:40 +08:00
if (keyboardHeight < 100) return 0;
2020-11-25 14:41:57 +08:00
final s = FFI.canvasModel.scale;
2020-12-21 17:26:23 +08:00
final thresh = (size.height - keyboardHeight) / 2;
2020-11-25 14:41:57 +08:00
var h = (_y - getVisibleRect().top) * s; // local physical display height
2020-11-27 12:05:23 +08:00
return h - thresh;
2020-11-25 11:20:40 +08:00
}
2021-08-21 17:18:14 +08:00
void touch(double x, double y, bool right) {
final scale = FFI.canvasModel.scale;
final xoffset = FFI.canvasModel.x;
final yoffset = FFI.canvasModel.y;
_x = (x - xoffset) / scale + _displayOriginX;
_y = (y - yoffset) / scale + _displayOriginY;
FFI.moveMouse(_x, _y);
FFI.tap(right);
notifyListeners();
}
void reset() {
_x = _displayOriginX;
_y = _displayOriginY;
FFI.moveMouse(_x, _y);
2021-08-22 07:50:12 +08:00
FFI.canvasModel.clear(true);
2021-08-21 17:18:14 +08:00
notifyListeners();
}
void updatePan(double dx, double dy, bool touchMode, bool drag) {
2020-11-27 10:52:09 +08:00
if (FFI.imageModel.image == null) return;
2021-08-21 17:18:14 +08:00
if (touchMode) {
if (drag) {
final scale = FFI.canvasModel.scale;
_x += dx / scale;
_y += dy / scale;
FFI.moveMouse(_x, _y);
notifyListeners();
} else {
FFI.canvasModel.panX(dx);
FFI.canvasModel.panY(dy);
}
return;
}
2020-11-25 00:13:23 +08:00
final scale = FFI.canvasModel.scale;
dx /= scale;
dy /= scale;
2020-11-24 22:03:04 +08:00
final r = getVisibleRect();
var cx = r.center.dx;
var cy = r.center.dy;
var tryMoveCanvasX = false;
if (dx > 0) {
final maxCanvasCanMove =
_displayOriginX + FFI.imageModel.image.width - r.right;
tryMoveCanvasX = _x + dx > cx && maxCanvasCanMove > 0;
if (tryMoveCanvasX) {
dx = min(dx, maxCanvasCanMove);
} else {
final maxCursorCanMove = r.right - _x;
dx = min(dx, maxCursorCanMove);
}
} else if (dx < 0) {
final maxCanvasCanMove = _displayOriginX - r.left;
tryMoveCanvasX = _x + dx < cx && maxCanvasCanMove < 0;
if (tryMoveCanvasX) {
dx = max(dx, maxCanvasCanMove);
} else {
final maxCursorCanMove = r.left - _x;
dx = max(dx, maxCursorCanMove);
}
}
var tryMoveCanvasY = false;
if (dy > 0) {
final mayCanvasCanMove =
2020-11-24 23:36:46 +08:00
_displayOriginY + FFI.imageModel.image.height - r.bottom;
2020-11-24 22:03:04 +08:00
tryMoveCanvasY = _y + dy > cy && mayCanvasCanMove > 0;
if (tryMoveCanvasY) {
dy = min(dy, mayCanvasCanMove);
} else {
2020-11-24 23:36:46 +08:00
final mayCursorCanMove = r.bottom - _y;
2020-11-24 22:03:04 +08:00
dy = min(dy, mayCursorCanMove);
}
} else if (dy < 0) {
2020-11-24 23:36:46 +08:00
final mayCanvasCanMove = _displayOriginY - r.top;
2020-11-24 22:03:04 +08:00
tryMoveCanvasY = _y + dy < cy && mayCanvasCanMove < 0;
if (tryMoveCanvasY) {
dy = max(dy, mayCanvasCanMove);
} else {
2020-11-24 23:36:46 +08:00
final mayCursorCanMove = r.top - _y;
2020-11-24 22:03:04 +08:00
dy = max(dy, mayCursorCanMove);
}
}
if (dx == 0 && dy == 0) return;
_x += dx;
_y += dy;
if (tryMoveCanvasX && dx != 0) {
2020-11-24 23:36:46 +08:00
FFI.canvasModel.panX(-dx);
2020-11-24 22:03:04 +08:00
}
if (tryMoveCanvasY && dy != 0) {
2020-11-24 23:36:46 +08:00
FFI.canvasModel.panY(-dy);
2020-11-24 22:03:04 +08:00
}
2020-11-25 00:13:23 +08:00
FFI.moveMouse(_x, _y);
2020-11-24 22:03:04 +08:00
notifyListeners();
}
2020-11-19 00:32:46 +08:00
void updateCursorData(Map<String, dynamic> evt) {
var id = int.parse(evt['id']);
_hotx = double.parse(evt['hotx']);
_hoty = double.parse(evt['hoty']);
var width = int.parse(evt['width']);
var height = int.parse(evt['height']);
List<dynamic> colors = json.decode(evt['colors']);
final rgba = Uint8List.fromList(colors.map((s) => s as int).toList());
2020-11-28 17:42:29 +08:00
var pid = FFI.id;
2020-11-19 00:32:46 +08:00
ui.decodeImageFromPixels(rgba, width, height, ui.PixelFormat.rgba8888,
(image) {
2020-11-28 17:42:29 +08:00
if (FFI.id != pid) return;
2020-11-19 00:32:46 +08:00
_image = image;
2020-11-23 23:18:42 +08:00
_images[id] = Tuple3(image, _hotx, _hoty);
2020-11-19 00:32:46 +08:00
try {
// my throw exception, because the listener maybe already dispose
notifyListeners();
} catch (e) {
print('notify cursor: $e');
}
2020-11-19 00:32:46 +08:00
});
}
void updateCursorId(Map<String, dynamic> evt) {
final tmp = _images[int.parse(evt['id'])];
if (tmp != null) {
2020-11-23 23:18:42 +08:00
_image = tmp.item1;
_hotx = tmp.item2;
_hoty = tmp.item3;
2020-11-19 00:32:46 +08:00
notifyListeners();
}
}
void updateCursorPosition(Map<String, dynamic> evt) {
_x = double.parse(evt['x']);
_y = double.parse(evt['y']);
notifyListeners();
}
void updateDisplayOrigin(double x, double y) {
_displayOriginX = x;
_displayOriginY = y;
2020-12-21 21:52:20 +08:00
_x = x + 1;
_y = y + 1;
2020-11-24 23:36:46 +08:00
FFI.moveMouse(x, y);
2020-11-27 22:50:24 +08:00
FFI.canvasModel.resetOffset();
2020-11-19 00:32:46 +08:00
notifyListeners();
}
void updateDisplayOriginWithCursor(
double x, double y, double xCursor, double yCursor) {
_displayOriginX = x;
_displayOriginY = y;
_x = xCursor;
_y = yCursor;
FFI.moveMouse(x, y);
notifyListeners();
}
2020-11-19 00:32:46 +08:00
void clear() {
2020-11-22 18:29:04 +08:00
_x = -10000;
_x = -10000;
2020-11-19 00:32:46 +08:00
_image = null;
_images.clear();
}
}
class FFI {
2020-11-28 17:42:29 +08:00
static String id = "";
2020-11-25 16:28:46 +08:00
static var shift = false;
static var ctrl = false;
static var alt = false;
static var command = false;
2020-11-19 00:32:46 +08:00
static final imageModel = ImageModel();
static final ffiModel = FfiModel();
static final cursorModel = CursorModel();
2020-11-23 23:18:42 +08:00
static final canvasModel = CanvasModel();
2020-11-19 00:32:46 +08:00
static String getId() {
return getByName('remote_id');
}
2020-11-25 17:02:27 +08:00
static void tap(bool right) {
sendMouse('down', right ? 'right' : 'left');
sendMouse('up', right ? 'right' : 'left');
}
static void scroll(double y) {
var y2 = y.round();
if (y2 == 0) return;
setByName('send_mouse',
json.encode(modify({'type': 'wheel', 'y': y2.toString()})));
2020-11-25 16:28:46 +08:00
}
2020-11-28 13:22:19 +08:00
static void reconnect() {
setByName('reconnect');
FFI.ffiModel.clearPermissions();
}
2020-11-25 16:28:46 +08:00
static void resetModifiers() {
shift = ctrl = alt = command = false;
}
static Map<String, String> modify(Map<String, String> evt) {
if (ctrl) evt['ctrl'] = 'true';
if (shift) evt['shift'] = 'true';
if (alt) evt['alt'] = 'true';
if (command) evt['command'] = 'true';
return evt;
}
static void sendMouse(String type, String buttons) {
2020-11-25 16:28:46 +08:00
if (!ffiModel.keyboard()) return;
setByName(
'send_mouse', json.encode(modify({'type': type, 'buttons': buttons})));
}
static void inputKey(String name) {
if (!ffiModel.keyboard()) return;
2022-02-04 11:52:54 +08:00
setByName(
'input_key', json.encode(modify({'name': name, 'press': 'true'})));
}
2020-11-24 23:36:46 +08:00
static void moveMouse(double x, double y) {
2020-11-25 16:28:46 +08:00
if (!ffiModel.keyboard()) return;
2020-11-24 23:36:46 +08:00
var x2 = x.toInt();
var y2 = y.toInt();
2020-11-25 16:28:46 +08:00
setByName('send_mouse', json.encode(modify({'x': '$x2', 'y': '$y2'})));
2020-11-24 23:36:46 +08:00
}
2020-11-19 00:32:46 +08:00
static List<Peer> peers() {
try {
List<dynamic> peers = json.decode(getByName('peers'));
return peers
.map((s) => s as List<dynamic>)
.map((s) =>
Peer.fromJson(s[0] as String, s[1] as Map<String, dynamic>))
.toList();
} catch (e) {
print('peers(): $e');
2020-11-19 00:32:46 +08:00
}
return [];
}
static void connect(String id) {
setByName('connect', id);
2020-11-28 17:42:29 +08:00
FFI.id = id;
2020-11-19 00:32:46 +08:00
}
static Map<String, dynamic> popEvent() {
var s = getByName('event');
if (s == '') return null;
try {
Map<String, dynamic> event = json.decode(s);
return event;
} catch (e) {
print('popEvent(): $e');
2020-11-19 00:32:46 +08:00
}
return null;
}
static void login(String password, bool remember) {
setByName(
'login',
json.encode({
'password': password,
'remember': remember ? 'true' : 'false',
}));
}
static void close() {
2022-02-03 17:19:25 +08:00
if (FFI.imageModel.image != null && !isDesktop) {
savePreference(id, cursorModel.x, cursorModel.y, canvasModel.x,
canvasModel.y, canvasModel.scale, ffiModel.pi.currentDisplay);
}
2020-11-28 17:42:29 +08:00
id = "";
2020-11-19 00:32:46 +08:00
setByName('close', '');
2020-11-25 16:28:46 +08:00
imageModel.update(null);
cursorModel.clear();
ffiModel.clear();
canvasModel.clear();
resetModifiers();
2020-11-19 00:32:46 +08:00
}
2022-01-26 12:48:16 +08:00
static String getByName(String name, [String arg = '']) {
return PlatformFFI.getByName(name, arg);
2020-11-19 00:32:46 +08:00
}
2022-01-26 12:48:16 +08:00
static void setByName(String name, [String value = '']) {
PlatformFFI.setByName(name, value);
2020-11-19 00:32:46 +08:00
}
2022-01-26 19:00:23 +08:00
static Future<String> getVersion() async {
return await PlatformFFI.getVersion();
}
2022-02-03 00:53:59 +08:00
static handleMouse(Map<String, dynamic> evt) {
2022-02-03 17:19:25 +08:00
var type = '';
switch (evt['type']) {
case 'mousedown':
type = 'down';
break;
case 'mouseup':
type = 'up';
break;
case 'mousemove':
break;
default:
return;
}
evt['type'] = type;
var x = evt['x'];
var y = evt['y'];
final d = FFI.ffiModel.display;
x -= FFI.canvasModel.x;
y -= FFI.canvasModel.y;
if (x < 0 || x > d.width || y < 0 || y > d.height) {
return;
}
x += d.x;
y += d.y;
if (type != '') {
x = 0;
y = 0;
}
evt['x'] = '$x';
evt['y'] = '$y';
var buttons = '';
switch (evt['buttons']) {
case 1:
buttons = 'left';
break;
case 2:
buttons = 'right';
break;
case 4:
buttons = 'wheel';
break;
}
evt['buttons'] = buttons;
if (evt['ctrl'] != true) evt.remove('ctrl');
if (evt['shift'] != true) evt.remove('shift');
if (evt['alt'] != true) evt.remove('alt');
if (evt['command'] != true) evt.remove('command');
setByName('send_mouse', json.encode(evt));
2022-02-03 00:53:59 +08:00
}
static listenToMouse(bool yesOrNo) {
if (yesOrNo) {
PlatformFFI.startDesktopWebListener(handleMouse);
} else {
PlatformFFI.stopDesktopWebListener();
}
}
2020-11-19 00:32:46 +08:00
}
class Peer {
final String id;
final String username;
final String hostname;
final String platform;
Peer.fromJson(String id, Map<String, dynamic> json)
: id = id,
username = json['username'],
hostname = json['hostname'],
platform = json['platform'];
}
class Display {
double x = 0;
double y = 0;
int width = 0;
int height = 0;
}
class PeerInfo {
2020-11-27 17:34:09 +08:00
String version;
2020-11-19 00:32:46 +08:00
String username;
String hostname;
String platform;
bool sasEnabled;
int currentDisplay;
List<Display> displays;
}
void savePreference(String id, double xCursor, double yCursor, double xCanvas,
double yCanvas, double scale, int currentDisplay) async {
SharedPreferences prefs = await SharedPreferences.getInstance();
final p = Map<String, dynamic>();
p['xCursor'] = xCursor;
p['yCursor'] = yCursor;
p['xCanvas'] = xCanvas;
p['yCanvas'] = yCanvas;
p['scale'] = scale;
p['currentDisplay'] = currentDisplay;
prefs.setString('peer' + id, json.encode(p));
}
Future<Map<String, dynamic>> getPreference(String id) async {
2022-02-03 17:19:25 +08:00
if (!isDesktop) return null;
SharedPreferences prefs = await SharedPreferences.getInstance();
var p = prefs.getString('peer' + id);
if (p == null) return null;
Map<String, dynamic> m = json.decode(p);
return m;
}
2021-08-06 21:18:06 +08:00
void removePreference(String id) async {
SharedPreferences prefs = await SharedPreferences.getInstance();
prefs.remove('peer' + id);
2020-12-21 19:05:31 +08:00
}
void initializeCursorAndCanvas() async {
var p = await getPreference(FFI.id);
int currentDisplay = 0;
if (p != null) {
currentDisplay = p['currentDisplay'];
}
if (p == null || currentDisplay != FFI.ffiModel.pi.currentDisplay) {
FFI.cursorModel
.updateDisplayOrigin(FFI.ffiModel.display.x, FFI.ffiModel.display.y);
return;
}
double xCursor = p['xCursor'];
double yCursor = p['yCursor'];
double xCanvas = p['xCanvas'];
double yCanvas = p['yCanvas'];
double scale = p['scale'];
FFI.cursorModel.updateDisplayOriginWithCursor(
FFI.ffiModel.display.x, FFI.ffiModel.display.y, xCursor, yCursor);
FFI.canvasModel.update(xCanvas, yCanvas, scale);
}
2021-08-02 20:54:56 +08:00
String translate(String name) {
2022-01-27 01:28:32 +08:00
if (name.startsWith('Failed to') && name.contains(': ')) {
return name.split(': ').map((x) => translate(x)).join(': ');
}
2022-01-26 19:00:23 +08:00
var a = 'translate';
var b = '{"locale": "$localeName", "text": "$name"}';
return FFI.getByName(a, b);
2021-08-02 20:54:56 +08:00
}