add ui event

Signed-off-by: fufesou <shuanglongchen@yeah.net>
This commit is contained in:
fufesou 2023-04-20 20:57:47 +08:00
parent d9755abbc2
commit 9a08e0bed4
9 changed files with 262 additions and 63 deletions

View File

@ -1,5 +1,4 @@
import 'dart:convert'; import 'dart:convert';
import 'dart:io';
import 'dart:ui' as ui; import 'dart:ui' as ui;
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@ -641,7 +640,7 @@ class _ControlMenu extends StatelessWidget {
if (e.divider) { if (e.divider) {
return Divider(); return Divider();
} else { } else {
return _MenuItemButton( return MenuButton(
child: e.child, child: e.child,
onPressed: e.onPressed, onPressed: e.onPressed,
ffi: ffi, ffi: ffi,
@ -711,7 +710,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
if (!visible) return Offstage(); if (!visible) return Offstage();
return Column( return Column(
children: [ children: [
_MenuItemButton( MenuButton(
child: Text(translate('Adjust Window')), child: Text(translate('Adjust Window')),
onPressed: _doAdjustWindow, onPressed: _doAdjustWindow,
ffi: widget.ffi), ffi: widget.ffi),
@ -828,7 +827,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
final v = data as List<TRadioMenu<String>>; final v = data as List<TRadioMenu<String>>;
return Column(children: [ return Column(children: [
...v ...v
.map((e) => _RadioMenuButton<String>( .map((e) => RdoMenuButton<String>(
value: e.value, value: e.value,
groupValue: e.groupValue, groupValue: e.groupValue,
onChanged: e.onChanged, onChanged: e.onChanged,
@ -858,14 +857,14 @@ class _DisplayMenuState extends State<_DisplayMenu> {
final enabled = widget.ffi.canvasModel.imageOverflow.value; final enabled = widget.ffi.canvasModel.imageOverflow.value;
return Column(children: [ return Column(children: [
_RadioMenuButton<String>( RdoMenuButton<String>(
child: Text(translate('ScrollAuto')), child: Text(translate('ScrollAuto')),
value: kRemoteScrollStyleAuto, value: kRemoteScrollStyleAuto,
groupValue: groupValue, groupValue: groupValue,
onChanged: enabled ? (value) => onChange(value) : null, onChanged: enabled ? (value) => onChange(value) : null,
ffi: widget.ffi, ffi: widget.ffi,
), ),
_RadioMenuButton<String>( RdoMenuButton<String>(
child: Text(translate('Scrollbar')), child: Text(translate('Scrollbar')),
value: kRemoteScrollStyleBar, value: kRemoteScrollStyleBar,
groupValue: groupValue, groupValue: groupValue,
@ -886,7 +885,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
ffi: widget.ffi, ffi: widget.ffi,
child: Text(translate('Image Quality')), child: Text(translate('Image Quality')),
menuChildren: v menuChildren: v
.map((e) => _RadioMenuButton<String>( .map((e) => RdoMenuButton<String>(
value: e.value, value: e.value,
groupValue: e.groupValue, groupValue: e.groupValue,
onChanged: e.onChanged, onChanged: e.onChanged,
@ -908,7 +907,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
ffi: widget.ffi, ffi: widget.ffi,
child: Text(translate('Codec')), child: Text(translate('Codec')),
menuChildren: v menuChildren: v
.map((e) => _RadioMenuButton( .map((e) => RdoMenuButton(
value: e.value, value: e.value,
groupValue: e.groupValue, groupValue: e.groupValue,
onChanged: e.onChanged, onChanged: e.onChanged,
@ -948,7 +947,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
return _SubmenuButton( return _SubmenuButton(
ffi: widget.ffi, ffi: widget.ffi,
menuChildren: resolutions menuChildren: resolutions
.map((e) => _RadioMenuButton( .map((e) => RdoMenuButton(
value: '${e.width}x${e.height}', value: '${e.width}x${e.height}',
groupValue: groupValue, groupValue: groupValue,
onChanged: onChanged, onChanged: onChanged,
@ -966,7 +965,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
if (v.isEmpty) return Offstage(); if (v.isEmpty) return Offstage();
return Column( return Column(
children: v children: v
.map((e) => _CheckboxMenuButton( .map((e) => CkbMenuButton(
value: e.value, value: e.value,
onChanged: e.onChanged, onChanged: e.onChanged,
child: e.child, child: e.child,
@ -1026,7 +1025,7 @@ class _KeyboardMenu extends StatelessWidget {
KeyboardModeMenu(key: _kKeyMapMode, menu: 'Map mode'), KeyboardModeMenu(key: _kKeyMapMode, menu: 'Map mode'),
KeyboardModeMenu(key: _kKeyTranslateMode, menu: 'Translate mode'), KeyboardModeMenu(key: _kKeyTranslateMode, menu: 'Translate mode'),
]; ];
List<_RadioMenuButton> list = []; List<RdoMenuButton> list = [];
final enabled = !ffi.ffiModel.viewOnly; final enabled = !ffi.ffiModel.viewOnly;
onChanged(String? value) async { onChanged(String? value) async {
if (value == null) return; if (value == null) return;
@ -1049,7 +1048,7 @@ class _KeyboardMenu extends StatelessWidget {
if (mode.key == _kKeyTranslateMode) { if (mode.key == _kKeyTranslateMode) {
text = '$text beta'; text = '$text beta';
} }
list.add(_RadioMenuButton<String>( list.add(RdoMenuButton<String>(
child: Text(text), child: Text(text),
value: mode.key, value: mode.key,
groupValue: groupValue, groupValue: groupValue,
@ -1069,7 +1068,7 @@ class _KeyboardMenu extends StatelessWidget {
return Column( return Column(
children: [ children: [
Divider(), Divider(),
_MenuItemButton( MenuButton(
child: Text( child: Text(
'${translate('Local keyboard type')}: ${KBLayoutType.value}'), '${translate('Local keyboard type')}: ${KBLayoutType.value}'),
trailingIcon: const Icon(Icons.settings), trailingIcon: const Icon(Icons.settings),
@ -1085,7 +1084,7 @@ class _KeyboardMenu extends StatelessWidget {
view_mode() { view_mode() {
final ffiModel = ffi.ffiModel; final ffiModel = ffi.ffiModel;
final enabled = version_cmp(pi.version, '1.2.0') >= 0 && ffiModel.keyboard; final enabled = version_cmp(pi.version, '1.2.0') >= 0 && ffiModel.keyboard;
return _CheckboxMenuButton( return CkbMenuButton(
value: ffiModel.viewOnly, value: ffiModel.viewOnly,
onChanged: enabled onChanged: enabled
? (value) async { ? (value) async {
@ -1129,7 +1128,7 @@ class _ChatMenuState extends State<_ChatMenu> {
} }
textChat() { textChat() {
return _MenuItemButton( return MenuButton(
child: Text(translate('Text chat')), child: Text(translate('Text chat')),
ffi: widget.ffi, ffi: widget.ffi,
onPressed: () { onPressed: () {
@ -1148,7 +1147,7 @@ class _ChatMenuState extends State<_ChatMenu> {
} }
voiceCall() { voiceCall() {
return _MenuItemButton( return MenuButton(
child: Text(translate('Voice call')), child: Text(translate('Voice call')),
ffi: widget.ffi, ffi: widget.ffi,
onPressed: () => bind.sessionRequestVoiceCall(id: widget.id), onPressed: () => bind.sessionRequestVoiceCall(id: widget.id),
@ -1403,12 +1402,12 @@ class _SubmenuButton extends StatelessWidget {
} }
} }
class _MenuItemButton extends StatelessWidget { class MenuButton extends StatelessWidget {
final VoidCallback? onPressed; final VoidCallback? onPressed;
final Widget? trailingIcon; final Widget? trailingIcon;
final Widget? child; final Widget? child;
final FFI ffi; final FFI ffi;
_MenuItemButton( MenuButton(
{Key? key, {Key? key,
this.onPressed, this.onPressed,
this.trailingIcon, this.trailingIcon,
@ -1431,12 +1430,12 @@ class _MenuItemButton extends StatelessWidget {
} }
} }
class _CheckboxMenuButton extends StatelessWidget { class CkbMenuButton extends StatelessWidget {
final bool? value; final bool? value;
final ValueChanged<bool?>? onChanged; final ValueChanged<bool?>? onChanged;
final Widget? child; final Widget? child;
final FFI ffi; final FFI ffi;
const _CheckboxMenuButton( const CkbMenuButton(
{Key? key, {Key? key,
required this.value, required this.value,
required this.onChanged, required this.onChanged,
@ -1460,13 +1459,13 @@ class _CheckboxMenuButton extends StatelessWidget {
} }
} }
class _RadioMenuButton<T> extends StatelessWidget { class RdoMenuButton<T> extends StatelessWidget {
final T value; final T value;
final T? groupValue; final T? groupValue;
final ValueChanged<T?>? onChanged; final ValueChanged<T?>? onChanged;
final Widget? child; final Widget? child;
final FFI ffi; final FFI ffi;
const _RadioMenuButton( const RdoMenuButton(
{Key? key, {Key? key,
required this.value, required this.value,
required this.groupValue, required this.groupValue,

View File

@ -1 +1,43 @@
import 'dart:convert';
typedef PluginId = String; typedef PluginId = String;
class MsgFromUi {
String remotePeerId;
String localPeerId;
String id;
String name;
String location;
String key;
String value;
String action;
MsgFromUi({
required this.remotePeerId,
required this.localPeerId,
required this.id,
required this.name,
required this.location,
required this.key,
required this.value,
required this.action,
});
Map<String, dynamic> toJson() {
return <String, dynamic>{
'remote_peer_id': remotePeerId,
'local_peer_id': localPeerId,
'id': id,
'name': name,
'location': location,
'key': key,
'value': value,
'action': action,
};
}
@override
String toString() {
return jsonEncode(toJson());
}
}

View File

@ -1,45 +1,58 @@
import 'dart:collection'; import 'dart:collection';
class UiButton { const String kValueTrue = '1';
String key; const String kValueFalse = '0';
String text;
String icon;
String tooltip;
String action;
UiButton(this.key, this.text, this.icon, this.tooltip, this.action);
UiButton.fromJson(Map<String, dynamic> json)
: key = json['key'] ?? '',
text = json['text'] ?? '',
icon = json['icon'] ?? '',
tooltip = json['tooltip'] ?? '',
action = json['action'] ?? '';
}
class UiCheckbox {
String key;
String text;
String tooltip;
String action;
UiCheckbox(this.key, this.text, this.tooltip, this.action);
UiCheckbox.fromJson(Map<String, dynamic> json)
: key = json['key'] ?? '',
text = json['text'] ?? '',
tooltip = json['tooltip'] ?? '',
action = json['action'] ?? '';
}
class UiType { class UiType {
UiButton? button; String key;
UiCheckbox? checkbox; String text;
String tooltip;
String action;
UiType(this.key, this.text, this.tooltip, this.action);
UiType.fromJson(Map<String, dynamic> json) UiType.fromJson(Map<String, dynamic> json)
: button = json['t'] == 'Button' ? UiButton.fromJson(json['c']) : null, : key = json['key'] ?? '',
checkbox = text = json['text'] ?? '',
json['t'] != 'Checkbox' ? UiCheckbox.fromJson(json['c']) : null; tooltip = json['tooltip'] ?? '',
action = json['action'] ?? '';
bool get isValid => button != null || checkbox != null; static UiType? create(Map<String, dynamic> json) {
if (json['t'] == 'Button') {
return UiButton.fromJson(json['c']);
} else if (json['t'] == 'Checkbox') {
return UiCheckbox.fromJson(json['c']);
} else {
return null;
}
}
}
class UiButton extends UiType {
String icon;
UiButton(
{required String key,
required String text,
required this.icon,
required String tooltip,
required String action})
: super(key, text, tooltip, action);
UiButton.fromJson(Map<String, dynamic> json)
: icon = json['icon'] ?? '',
super.fromJson(json);
}
class UiCheckbox extends UiType {
UiCheckbox(
{required String key,
required String text,
required String tooltip,
required String action})
: super(key, text, tooltip, action);
UiCheckbox.fromJson(Map<String, dynamic> json) : super.fromJson(json);
} }
class Location { class Location {
@ -49,6 +62,14 @@ class Location {
HashMap<String, UiType> ui; HashMap<String, UiType> ui;
Location(this.ui); Location(this.ui);
Location.fromJson(Map<String, dynamic> json) : ui = HashMap() {
json.forEach((key, value) {
var ui = UiType.create(value);
if (ui != null) {
this.ui[ui.key] = ui;
}
});
}
} }
class ConfigItem { class ConfigItem {
@ -63,6 +84,11 @@ class ConfigItem {
value = json['value'] ?? '', value = json['value'] ?? '',
description = json['description'] ?? '', description = json['description'] ?? '',
defaultValue = json['default'] ?? ''; defaultValue = json['default'] ?? '';
static String get trueValue => kValueTrue;
static String get falseValue => kValueFalse;
static bool isTrue(String value) => value == kValueTrue;
static bool isFalse(String value) => value == kValueFalse;
} }
class Config { class Config {

View File

@ -15,6 +15,8 @@ class LocationModel with ChangeNotifier {
uiList.add(ui); uiList.add(ui);
notifyListeners(); notifyListeners();
} }
bool get isEmpty => uiList.isEmpty;
} }
void addLocation(PluginId id, String location, UiType ui) { void addLocation(PluginId id, String location, UiType ui) {

View File

@ -40,8 +40,5 @@ void handleReloading(Map<String, dynamic> evt, String peer) {
return; return;
} }
final ui = UiType.fromJson(evt); final ui = UiType.fromJson(evt);
if (!ui.isValid) {
return;
}
addLocation(evt['id']!, evt['location']!, ui); addLocation(evt['id']!, evt['location']!, ui);
} }

View File

@ -1,27 +1,110 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_hbb/models/model.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'package:flutter_hbb/desktop/widgets/remote_toolbar.dart';
import 'package:flutter_hbb/models/platform_model.dart';
import '../../../desc.dart';
import '../../../model.dart'; import '../../../model.dart';
import '../../../common.dart';
class Display extends StatelessWidget { class Display extends StatelessWidget {
final PluginId pluginId;
final String peerId; final String peerId;
final FFI ffi;
final String location;
final LocationModel locationModel; final LocationModel locationModel;
Display({ Display({
Key? key, Key? key,
required this.pluginId,
required this.peerId, required this.peerId,
required this.ffi,
required this.location,
required this.locationModel, required this.locationModel,
}) : super(key: key); }) : super(key: key);
bool get isEmpty => locationModel.isEmpty;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return ChangeNotifierProvider.value( return ChangeNotifierProvider.value(
value: locationModel, value: locationModel,
child: Consumer<LocationModel>(builder: (context, model, child) { child: Consumer<LocationModel>(builder: (context, model, child) {
return Column( return Column(
children: [], children: locationModel.uiList.map((ui) => _buildItem(ui)).toList(),
); );
}), }),
); );
} }
Widget _buildItem(UiType ui) {
switch (ui.runtimeType) {
case UiButton:
return _buildMenuButton(ui as UiButton);
case UiCheckbox:
return _buildCheckboxMenuButton(ui as UiCheckbox);
default:
return Container();
}
}
Uint8List _makeEvent(
String localPeerId,
String key, {
bool? v,
}) {
final event = MsgFromUi(
remotePeerId: peerId,
localPeerId: localPeerId,
id: pluginId,
name: getDesc(pluginId)?.name ?? '',
location: location,
key: key,
value:
v != null ? (v ? ConfigItem.trueValue : ConfigItem.falseValue) : '',
action: '',
);
return Uint8List.fromList(event.toString().codeUnits);
}
Widget _buildMenuButton(UiButton ui) {
return MenuButton(
onPressed: () {
() async {
final localPeerId = await bind.mainGetMyId();
bind.pluginEvent(
id: pluginId,
event: _makeEvent(localPeerId, ui.key),
);
}();
},
// to-do: rustdesk translate or plugin translate ?
child: Text(ui.text),
ffi: ffi,
);
}
Widget _buildCheckboxMenuButton(UiCheckbox ui) {
final v =
bind.pluginGetSessionOption(id: pluginId, peer: peerId, key: ui.key);
return CkbMenuButton(
value: ConfigItem.isTrue(v),
onChanged: (v) {
if (v != null) {
() async {
final localPeerId = await bind.mainGetMyId();
bind.pluginEvent(
id: pluginId,
event: _makeEvent(localPeerId, ui.key, v: v),
);
}();
}
},
// to-do: rustdesk translate or plugin translate ?
child: Text(ui.text),
ffi: ffi,
);
}
} }

View File

@ -1405,6 +1405,54 @@ pub fn plugin_event(_id: String, _event: Vec<u8>) {
} }
} }
#[inline]
pub fn plugin_get_session_option(_id: String, _peer: String, _key: String) -> SyncReturn<String> {
#[cfg(feature = "plugin_framework")]
#[cfg(not(any(target_os = "android", target_os = "ios")))]
{
return SyncReturn(crate::plugin::PeerConfig::get(&_id, &_peer, &_key));
}
#[cfg(any(
not(feature = "plugin_framework"),
target_os = "android",
target_os = "ios"
))]
return SyncReturn("".to_owned());
}
#[inline]
pub fn plugin_set_session_option(_id: String, _peer: String, _key: String, _value: String) {
#[cfg(feature = "plugin_framework")]
#[cfg(not(any(target_os = "android", target_os = "ios")))]
{
crate::plugin::PeerConfig::set(&_id, &_peer, &_key, &_value);
}
}
#[inline]
pub fn plugin_get_local_option(_id: String, _key: String) -> SyncReturn<String> {
#[cfg(feature = "plugin_framework")]
#[cfg(not(any(target_os = "android", target_os = "ios")))]
{
allow_err!(crate::plugin::LocalConfig::get(&_id, &key));
}
#[cfg(any(
not(feature = "plugin_framework"),
target_os = "android",
target_os = "ios"
))]
return SyncReturn("".to_owned());
}
#[inline]
pub fn plugin_set_local_option(_id: String, _key: String, _value: String) {
#[cfg(feature = "plugin_framework")]
#[cfg(not(any(target_os = "android", target_os = "ios")))]
{
crate::plugin::LocalConfig::set(&_id, &_key, &_value);
}
}
#[cfg(target_os = "android")] #[cfg(target_os = "android")]
pub mod server_side { pub mod server_side {
use hbb_common::{config, log}; use hbb_common::{config, log};

View File

@ -15,9 +15,9 @@ lazy_static::lazy_static! {
} }
#[derive(Debug, Default, Serialize, Deserialize)] #[derive(Debug, Default, Serialize, Deserialize)]
struct LocalConfig(HashMap<String, String>); pub struct LocalConfig(HashMap<String, String>);
#[derive(Debug, Default, Serialize, Deserialize)] #[derive(Debug, Default, Serialize, Deserialize)]
struct PeerConfig(HashMap<String, String>); pub struct PeerConfig(HashMap<String, String>);
type PeersConfig = HashMap<String, PeerConfig>; type PeersConfig = HashMap<String, PeerConfig>;
#[inline] #[inline]

View File

@ -12,6 +12,8 @@ pub use plugins::{
reload_plugin, unload_plugin, reload_plugin, unload_plugin,
}; };
pub use config::{LocalConfig, PeerConfig};
#[inline] #[inline]
fn cstr_to_string(cstr: *const c_char) -> ResultType<String> { fn cstr_to_string(cstr: *const c_char) -> ResultType<String> {
Ok(String::from_utf8(unsafe { Ok(String::from_utf8(unsafe {