plugin_framework, ui tmp

Signed-off-by: fufesou <shuanglongchen@yeah.net>
This commit is contained in:
fufesou 2023-04-20 22:53:43 +08:00
parent 9a08e0bed4
commit 1b303b7b27
11 changed files with 147 additions and 93 deletions

View File

@ -2,6 +2,12 @@ import 'dart:convert';
typedef PluginId = String; typedef PluginId = String;
// ui location
const String kLocationHostMainDisplayOthers =
'host|main|settings|display|others';
const String kLocationClientRemoteToolbarDisplay =
'client|remote|toolbar|display';
class MsgFromUi { class MsgFromUi {
String remotePeerId; String remotePeerId;
String localPeerId; String localPeerId;

View File

@ -0,0 +1,44 @@
import 'package:flutter/material.dart';
import './common.dart';
import './desc.dart';
final Map<String, LocationModel> locationModels = {};
class PluginModel with ChangeNotifier {
final List<UiType> uiList = [];
void add(UiType ui) {
uiList.add(ui);
notifyListeners();
}
bool get isEmpty => uiList.isEmpty;
}
class LocationModel with ChangeNotifier {
final Map<PluginId, PluginModel> pluginModels = {};
void add(PluginId id, UiType ui) {
if (pluginModels[id] != null) {
pluginModels[id]!.add(ui);
} else {
var model = PluginModel();
model.add(ui);
pluginModels[id] = model;
notifyListeners();
}
}
bool get isEmpty => pluginModels.isEmpty;
}
void addLocationUi(String location, PluginId id, UiType ui) {
locationModels[location]?.add(id, ui);
}
LocationModel addLocation(String location) {
if (locationModels[location] == null) {
locationModels[location] = LocationModel();
}
return locationModels[location]!;
}

View File

@ -5,20 +5,18 @@ import 'package:provider/provider.dart';
import 'package:flutter_hbb/desktop/widgets/remote_toolbar.dart'; import 'package:flutter_hbb/desktop/widgets/remote_toolbar.dart';
import 'package:flutter_hbb/models/platform_model.dart'; import 'package:flutter_hbb/models/platform_model.dart';
import '../../../desc.dart'; import './desc.dart';
import '../../../model.dart'; import './model.dart';
import '../../../common.dart'; import './common.dart';
class Display extends StatelessWidget { class LocationItem extends StatelessWidget {
final PluginId pluginId;
final String peerId; final String peerId;
final FFI ffi; final FFI ffi;
final String location; final String location;
final LocationModel locationModel; final LocationModel locationModel;
Display({ LocationItem({
Key? key, Key? key,
required this.pluginId,
required this.peerId, required this.peerId,
required this.ffi, required this.ffi,
required this.location, required this.location,
@ -27,18 +25,71 @@ class Display extends StatelessWidget {
bool get isEmpty => locationModel.isEmpty; bool get isEmpty => locationModel.isEmpty;
static LocationItem createLocationItem(
String peerId, FFI ffi, String location) {
final model = addLocation(location);
return LocationItem(
peerId: peerId,
ffi: ffi,
location: location,
locationModel: model,
);
}
@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: locationModel.uiList.map((ui) => _buildItem(ui)).toList(), children: model.pluginModels.entries
.map((entry) => _buildPluginItem(entry.key, entry.value))
.toList(),
); );
}), }),
); );
} }
Widget _buildPluginItem(PluginId id, PluginModel model) => PluginItem(
pluginId: id,
peerId: peerId,
ffi: ffi,
location: location,
pluginModel: model,
);
}
class PluginItem extends StatelessWidget {
final PluginId pluginId;
final String peerId;
final FFI ffi;
final String location;
final PluginModel pluginModel;
PluginItem({
Key? key,
required this.pluginId,
required this.peerId,
required this.ffi,
required this.location,
required this.pluginModel,
}) : super(key: key);
bool get isEmpty => pluginModel.isEmpty;
@override
Widget build(BuildContext context) {
return ChangeNotifierProvider.value(
value: pluginModel,
child: Consumer<PluginModel>(builder: (context, model, child) {
return Column(
children: model.uiList.map((ui) => _buildItem(ui)).toList(),
);
}),
);
}
// to-do: add plugin icon and tooltip
Widget _buildItem(UiType ui) { Widget _buildItem(UiType ui) {
switch (ui.runtimeType) { switch (ui.runtimeType) {
case UiButton: case UiButton:
@ -80,7 +131,9 @@ class Display extends StatelessWidget {
); );
}(); }();
}, },
// to-do: rustdesk translate or plugin translate ? trailingIcon: Icon(
IconData(int.parse(ui.icon, radix: 16), fontFamily: 'MaterialIcons')),
// to-do: RustDesk translate or plugin translate ?
child: Text(ui.text), child: Text(ui.text),
ffi: ffi, ffi: ffi,
); );
@ -89,6 +142,10 @@ class Display extends StatelessWidget {
Widget _buildCheckboxMenuButton(UiCheckbox ui) { Widget _buildCheckboxMenuButton(UiCheckbox ui) {
final v = final v =
bind.pluginGetSessionOption(id: pluginId, peer: peerId, key: ui.key); bind.pluginGetSessionOption(id: pluginId, peer: peerId, key: ui.key);
if (v == null) {
// session or plugin not found
return Container();
}
return CkbMenuButton( return CkbMenuButton(
value: ConfigItem.isTrue(v), value: ConfigItem.isTrue(v),
onChanged: (v) { onChanged: (v) {
@ -108,3 +165,11 @@ class Display extends StatelessWidget {
); );
} }
} }
void handleReloading(Map<String, dynamic> evt, String peer) {
if (evt['id'] == null || evt['location'] == null) {
return;
}
final ui = UiType.fromJson(evt);
addLocationUi(evt['location']!, evt['id']!, ui);
}

View File

@ -8,6 +8,8 @@ import 'package:flutter_hbb/models/chat_model.dart';
import 'package:flutter_hbb/models/state_model.dart'; import 'package:flutter_hbb/models/state_model.dart';
import 'package:flutter_hbb/consts.dart'; import 'package:flutter_hbb/consts.dart';
import 'package:flutter_hbb/utils/multi_window_manager.dart'; import 'package:flutter_hbb/utils/multi_window_manager.dart';
import 'package:flutter_hbb/desktop/plugin/widget.dart';
import 'package:flutter_hbb/desktop/plugin/common.dart';
import 'package:flutter_svg/flutter_svg.dart'; import 'package:flutter_svg/flutter_svg.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@ -655,13 +657,19 @@ class _DisplayMenu extends StatefulWidget {
final FFI ffi; final FFI ffi;
final MenubarState state; final MenubarState state;
final Function(bool) setFullscreen; final Function(bool) setFullscreen;
final LocationItem pluginItem;
_DisplayMenu( _DisplayMenu(
{Key? key, {Key? key,
required this.id, required this.id,
required this.ffi, required this.ffi,
required this.state, required this.state,
required this.setFullscreen}) required this.setFullscreen})
: super(key: key); : pluginItem = LocationItem.createLocationItem(
id,
ffi,
kLocationClientRemoteToolbarDisplay,
),
super(key: key);
@override @override
State<_DisplayMenu> createState() => _DisplayMenuState(); State<_DisplayMenu> createState() => _DisplayMenuState();
@ -699,6 +707,7 @@ class _DisplayMenuState extends State<_DisplayMenu> {
resolutions(), resolutions(),
Divider(), Divider(),
toggles(), toggles(),
widget.pluginItem,
]); ]);
} }

View File

@ -16,9 +16,9 @@ import 'package:flutter_hbb/models/peer_tab_model.dart';
import 'package:flutter_hbb/models/server_model.dart'; import 'package:flutter_hbb/models/server_model.dart';
import 'package:flutter_hbb/models/user_model.dart'; import 'package:flutter_hbb/models/user_model.dart';
import 'package:flutter_hbb/models/state_model.dart'; import 'package:flutter_hbb/models/state_model.dart';
import 'package:flutter_hbb/plugin/event.dart'; import 'package:flutter_hbb/desktop/plugin/event.dart';
import 'package:flutter_hbb/plugin/desc.dart'; import 'package:flutter_hbb/desktop/plugin/desc.dart';
import 'package:flutter_hbb/plugin/widget.dart'; import 'package:flutter_hbb/desktop/plugin/widget.dart';
import 'package:flutter_hbb/common/shared_state.dart'; import 'package:flutter_hbb/common/shared_state.dart';
import 'package:tuple/tuple.dart'; import 'package:tuple/tuple.dart';
import 'package:image/image.dart' as img2; import 'package:image/image.dart' as img2;

View File

@ -1,30 +0,0 @@
import 'package:flutter/material.dart';
import './common.dart';
import './desc.dart';
// ui location
// host|main|settings|display|others
// client|remote|toolbar|display
final Map<PluginId, Map<String, LocationModel>> locationModels = {};
class LocationModel with ChangeNotifier {
final List<UiType> uiList = [];
void add(UiType ui) {
uiList.add(ui);
notifyListeners();
}
bool get isEmpty => uiList.isEmpty;
}
void addLocation(PluginId id, String location, UiType ui) {
if (!locationModels.containsKey(id)) {
locationModels[id] = {};
}
if (!locationModels[id]!.containsKey(location)) {
locationModels[id]![location] = LocationModel();
}
locationModels[id]![location]!.add(ui);
}

View File

@ -1,44 +0,0 @@
import 'package:flutter/material.dart';
import './desc.dart';
import './model.dart';
final Map<String, PluginWidget> pluginWidgets = {};
class PluginWidget {
final String id;
final String name;
final String location;
final Widget widget;
PluginWidget({
required this.id,
required this.name,
required this.location,
required this.widget,
});
// static Widget createButton(UiButton btn) {}
// static Widget createCheckbox(UiCheckbox chk) {}
// // ui location
// // host|main|settings|display|others
// // client|remote|toolbar|display
// static Widget? create(String id, String locatin, UiType ui) {
// if (ui.button != null) {
// return createButton(ui.button!);
// } else if (ui.checkbox != null) {
// return createCheckbox(ui.checkbox!);
// } else {
// return null;
// }
// }
}
void handleReloading(Map<String, dynamic> evt, String peer) {
if (evt['id'] == null || evt['location'] == null) {
return;
}
final ui = UiType.fromJson(evt);
addLocation(evt['id']!, evt['location']!, ui);
}

View File

@ -1406,7 +1406,7 @@ pub fn plugin_event(_id: String, _event: Vec<u8>) {
} }
#[inline] #[inline]
pub fn plugin_get_session_option(_id: String, _peer: String, _key: String) -> SyncReturn<String> { pub fn plugin_get_session_option(_id: String, _peer: String, _key: String) -> SyncReturn<Option<String>> {
#[cfg(feature = "plugin_framework")] #[cfg(feature = "plugin_framework")]
#[cfg(not(any(target_os = "android", target_os = "ios")))] #[cfg(not(any(target_os = "android", target_os = "ios")))]
{ {
@ -1417,7 +1417,9 @@ pub fn plugin_get_session_option(_id: String, _peer: String, _key: String) -> Sy
target_os = "android", target_os = "android",
target_os = "ios" target_os = "ios"
))] ))]
return SyncReturn("".to_owned()); {
return SyncReturn(None);
}
} }
#[inline] #[inline]
@ -1430,18 +1432,20 @@ pub fn plugin_set_session_option(_id: String, _peer: String, _key: String, _valu
} }
#[inline] #[inline]
pub fn plugin_get_local_option(_id: String, _key: String) -> SyncReturn<String> { pub fn plugin_get_local_option(_id: String, _key: String) -> SyncReturn<Option<String>> {
#[cfg(feature = "plugin_framework")] #[cfg(feature = "plugin_framework")]
#[cfg(not(any(target_os = "android", target_os = "ios")))] #[cfg(not(any(target_os = "android", target_os = "ios")))]
{ {
allow_err!(crate::plugin::LocalConfig::get(&_id, &key)); return SyncReturn(crate::plugin::LocalConfig::get(&_id, &_key));
} }
#[cfg(any( #[cfg(any(
not(feature = "plugin_framework"), not(feature = "plugin_framework"),
target_os = "android", target_os = "android",
target_os = "ios" target_os = "ios"
))] ))]
return SyncReturn("".to_owned()); {
return SyncReturn(None);
}
} }
#[inline] #[inline]

View File

@ -8,7 +8,7 @@ use std::ffi::{c_char, CStr};
pub struct UiButton { pub struct UiButton {
key: String, key: String,
text: String, text: String,
icon: String, icon: String, // icon can be int in flutter, but string in other ui framework. And it is flexible to use string.
tooltip: String, tooltip: String,
action: String, // The action to be triggered when the button is clicked. action: String, // The action to be triggered when the button is clicked.
} }