Merge pull request #4179 from fufesou/feat/plugin_framework
plugin_framework, debug config, mid commit
This commit is contained in:
commit
7892008535
@ -12,6 +12,8 @@ import 'package:flutter_hbb/models/platform_model.dart';
|
|||||||
import 'package:flutter_hbb/models/server_model.dart';
|
import 'package:flutter_hbb/models/server_model.dart';
|
||||||
import 'package:flutter_hbb/plugin/desc.dart';
|
import 'package:flutter_hbb/plugin/desc.dart';
|
||||||
import 'package:flutter_hbb/plugin/model.dart';
|
import 'package:flutter_hbb/plugin/model.dart';
|
||||||
|
import 'package:flutter_hbb/plugin/common.dart';
|
||||||
|
import 'package:flutter_hbb/plugin/widget.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
@ -1394,6 +1396,103 @@ class _AccountState extends State<_Account> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _Checkbox extends StatefulWidget {
|
||||||
|
final String label;
|
||||||
|
final bool Function() getValue;
|
||||||
|
final Future<void> Function(bool) setValue;
|
||||||
|
|
||||||
|
const _Checkbox(
|
||||||
|
{Key? key,
|
||||||
|
required this.label,
|
||||||
|
required this.getValue,
|
||||||
|
required this.setValue})
|
||||||
|
: super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<_Checkbox> createState() => _CheckboxState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CheckboxState extends State<_Checkbox> {
|
||||||
|
var value = false;
|
||||||
|
|
||||||
|
@override
|
||||||
|
initState() {
|
||||||
|
super.initState();
|
||||||
|
value = widget.getValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
onChanged(bool b) async {
|
||||||
|
await widget.setValue(b);
|
||||||
|
setState(() {
|
||||||
|
value = widget.getValue();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return GestureDetector(
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Checkbox(
|
||||||
|
value: value,
|
||||||
|
onChanged: (_) => onChanged(!value),
|
||||||
|
).marginOnly(right: 5),
|
||||||
|
Expanded(
|
||||||
|
child: Text(translate(widget.label)),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
).marginOnly(left: _kCheckBoxLeftMargin),
|
||||||
|
onTap: () => onChanged(!value),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class PluginCard extends StatefulWidget {
|
||||||
|
final PluginId pluginId;
|
||||||
|
final Desc desc;
|
||||||
|
const PluginCard({
|
||||||
|
Key? key,
|
||||||
|
required this.pluginId,
|
||||||
|
required this.desc,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<PluginCard> createState() => PluginCardState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class PluginCardState extends State<PluginCard> {
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final children = [
|
||||||
|
_Button(
|
||||||
|
'Reload',
|
||||||
|
() => bind.pluginReload(id: widget.pluginId),
|
||||||
|
),
|
||||||
|
_Checkbox(
|
||||||
|
label: 'Enable',
|
||||||
|
getValue: () => bind.pluginIdIsEnabled(id: widget.pluginId),
|
||||||
|
setValue: (bool v) async {
|
||||||
|
if (!v) {
|
||||||
|
clearPlugin(widget.pluginId);
|
||||||
|
}
|
||||||
|
await bind.pluginIdEnable(id: widget.pluginId, v: v);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
];
|
||||||
|
final model = getPluginModel(kLocationHostMainPlugin, widget.pluginId);
|
||||||
|
if (model != null) {
|
||||||
|
children.add(PluginItem(
|
||||||
|
pluginId: widget.pluginId,
|
||||||
|
peerId: '',
|
||||||
|
location: kLocationHostMainPlugin,
|
||||||
|
pluginModel: model,
|
||||||
|
isMenu: false,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
return _Card(title: widget.desc.name, children: children);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class _Plugin extends StatefulWidget {
|
class _Plugin extends StatefulWidget {
|
||||||
const _Plugin({Key? key}) : super(key: key);
|
const _Plugin({Key? key}) : super(key: key);
|
||||||
|
|
||||||
@ -1403,44 +1502,15 @@ class _Plugin extends StatefulWidget {
|
|||||||
|
|
||||||
class _PluginState extends State<_Plugin> {
|
class _PluginState extends State<_Plugin> {
|
||||||
// temp checkbox widget
|
// temp checkbox widget
|
||||||
Widget _checkbox(
|
|
||||||
String label,
|
|
||||||
bool Function() getValue,
|
|
||||||
Future<void> Function(bool) setValue,
|
|
||||||
) {
|
|
||||||
final value = getValue();
|
|
||||||
onChanged(bool b) async {
|
|
||||||
await setValue(b);
|
|
||||||
setState(() {});
|
|
||||||
}
|
|
||||||
|
|
||||||
return GestureDetector(
|
List<Widget> _buildCards(DescModel model) => [
|
||||||
child: Row(
|
|
||||||
children: [
|
|
||||||
Checkbox(
|
|
||||||
value: bind.pluginIsEnabled(),
|
|
||||||
onChanged: (_) => onChanged(!value),
|
|
||||||
).marginOnly(right: 5),
|
|
||||||
Expanded(
|
|
||||||
child: Text(translate(label)),
|
|
||||||
)
|
|
||||||
],
|
|
||||||
).marginOnly(left: _kCheckBoxLeftMargin),
|
|
||||||
onTap: () => onChanged(!value));
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
final scrollController = ScrollController();
|
|
||||||
buildCards(DescModel model) {
|
|
||||||
final cards = <Widget>[
|
|
||||||
_Card(
|
_Card(
|
||||||
title: 'Plugin',
|
title: 'Plugin',
|
||||||
children: [
|
children: [
|
||||||
_checkbox(
|
_Checkbox(
|
||||||
'Enable',
|
label: 'Enable',
|
||||||
() => bind.pluginIsEnabled() ?? false,
|
getValue: () => bind.pluginIsEnabled() ?? false,
|
||||||
(bool v) async {
|
setValue: (bool v) async {
|
||||||
if (!v) {
|
if (!v) {
|
||||||
clearLocations();
|
clearLocations();
|
||||||
}
|
}
|
||||||
@ -1449,28 +1519,14 @@ class _PluginState extends State<_Plugin> {
|
|||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
...model.all.entries
|
||||||
|
.map((entry) => PluginCard(pluginId: entry.key, desc: entry.value))
|
||||||
|
.toList(),
|
||||||
];
|
];
|
||||||
model.all.forEach((key, value) {
|
|
||||||
cards.add(_Card(title: key, children: [
|
|
||||||
_Button(
|
|
||||||
'Reload',
|
|
||||||
() => bind.pluginReload(id: key),
|
|
||||||
),
|
|
||||||
_checkbox(
|
|
||||||
'Enable',
|
|
||||||
() => bind.pluginIdIsEnabled(id: key),
|
|
||||||
(bool v) async {
|
|
||||||
if (!v) {
|
|
||||||
clearPlugin(key);
|
|
||||||
}
|
|
||||||
await bind.pluginIdEnable(id: key, v: v);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
]));
|
|
||||||
});
|
|
||||||
return cards;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final scrollController = ScrollController();
|
||||||
return DesktopScrollWrapper(
|
return DesktopScrollWrapper(
|
||||||
scrollController: scrollController,
|
scrollController: scrollController,
|
||||||
child: ChangeNotifierProvider.value(
|
child: ChangeNotifierProvider.value(
|
||||||
@ -1479,7 +1535,7 @@ class _PluginState extends State<_Plugin> {
|
|||||||
return ListView(
|
return ListView(
|
||||||
physics: DraggableNeverScrollableScrollPhysics(),
|
physics: DraggableNeverScrollableScrollPhysics(),
|
||||||
controller: scrollController,
|
controller: scrollController,
|
||||||
children: buildCards(model),
|
children: _buildCards(model),
|
||||||
).marginOnly(bottom: _kListViewBottomMargin);
|
).marginOnly(bottom: _kListViewBottomMargin);
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
|
@ -657,7 +657,7 @@ 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;
|
final Widget pluginItem;
|
||||||
_DisplayMenu(
|
_DisplayMenu(
|
||||||
{Key? key,
|
{Key? key,
|
||||||
required this.id,
|
required this.id,
|
||||||
@ -668,6 +668,7 @@ class _DisplayMenu extends StatefulWidget {
|
|||||||
id,
|
id,
|
||||||
ffi,
|
ffi,
|
||||||
kLocationClientRemoteToolbarDisplay,
|
kLocationClientRemoteToolbarDisplay,
|
||||||
|
true,
|
||||||
),
|
),
|
||||||
super(key: key);
|
super(key: key);
|
||||||
|
|
||||||
|
@ -3,8 +3,7 @@ import 'dart:convert';
|
|||||||
typedef PluginId = String;
|
typedef PluginId = String;
|
||||||
|
|
||||||
// ui location
|
// ui location
|
||||||
const String kLocationHostMainDisplayOthers =
|
const String kLocationHostMainPlugin = 'host|main|settings|plugin';
|
||||||
'host|main|settings|display|others';
|
|
||||||
const String kLocationClientRemoteToolbarDisplay =
|
const String kLocationClientRemoteToolbarDisplay =
|
||||||
'client|remote|toolbar|display';
|
'client|remote|toolbar|display';
|
||||||
|
|
||||||
|
@ -2,6 +2,8 @@ import 'dart:convert';
|
|||||||
import 'dart:collection';
|
import 'dart:collection';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
|
|
||||||
|
import './common.dart';
|
||||||
|
|
||||||
const String kValueTrue = '1';
|
const String kValueTrue = '1';
|
||||||
const String kValueFalse = '0';
|
const String kValueFalse = '0';
|
||||||
|
|
||||||
@ -59,7 +61,7 @@ class UiCheckbox extends UiType {
|
|||||||
|
|
||||||
class Location {
|
class Location {
|
||||||
// location key:
|
// location key:
|
||||||
// host|main|settings|display|others
|
// host|main|settings|plugin
|
||||||
// client|remote|toolbar|display
|
// client|remote|toolbar|display
|
||||||
HashMap<String, UiType> ui;
|
HashMap<String, UiType> ui;
|
||||||
|
|
||||||
@ -76,14 +78,12 @@ class Location {
|
|||||||
|
|
||||||
class ConfigItem {
|
class ConfigItem {
|
||||||
String key;
|
String key;
|
||||||
String value;
|
|
||||||
String description;
|
String description;
|
||||||
String defaultValue;
|
String defaultValue;
|
||||||
|
|
||||||
ConfigItem(this.key, this.value, this.defaultValue, this.description);
|
ConfigItem(this.key, this.defaultValue, this.description);
|
||||||
ConfigItem.fromJson(Map<String, dynamic> json)
|
ConfigItem.fromJson(Map<String, dynamic> json)
|
||||||
: key = json['key'] ?? '',
|
: key = json['key'] ?? '',
|
||||||
value = json['value'] ?? '',
|
|
||||||
description = json['description'] ?? '',
|
description = json['description'] ?? '',
|
||||||
defaultValue = json['default'] ?? '';
|
defaultValue = json['default'] ?? '';
|
||||||
|
|
||||||
@ -151,7 +151,7 @@ class Desc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class DescModel with ChangeNotifier {
|
class DescModel with ChangeNotifier {
|
||||||
final data = <String, Desc>{};
|
final data = <PluginId, Desc>{};
|
||||||
|
|
||||||
DescModel._();
|
DescModel._();
|
||||||
|
|
||||||
@ -169,7 +169,7 @@ class DescModel with ChangeNotifier {
|
|||||||
return data[id];
|
return data[id];
|
||||||
}
|
}
|
||||||
|
|
||||||
Map<String, Desc> get all => data;
|
Map<PluginId, Desc> get all => data;
|
||||||
|
|
||||||
static final DescModel _instance = DescModel._();
|
static final DescModel _instance = DescModel._();
|
||||||
static DescModel get instance => _instance;
|
static DescModel get instance => _instance;
|
||||||
|
@ -55,16 +55,17 @@ class LocationModel with ChangeNotifier {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void addLocationUi(String location, PluginId id, UiType ui) {
|
void addLocationUi(String location, PluginId id, UiType ui) {
|
||||||
_locationModels[location]?.add(id, ui);
|
|
||||||
}
|
|
||||||
|
|
||||||
LocationModel addLocation(String location) {
|
|
||||||
if (_locationModels[location] == null) {
|
if (_locationModels[location] == null) {
|
||||||
_locationModels[location] = LocationModel();
|
_locationModels[location] = LocationModel();
|
||||||
}
|
}
|
||||||
return _locationModels[location]!;
|
_locationModels[location]?.add(id, ui);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LocationModel? getLocationModel(String location) => _locationModels[location];
|
||||||
|
|
||||||
|
PluginModel? getPluginModel(String location, PluginId id) =>
|
||||||
|
_locationModels[location]?.pluginModels[id];
|
||||||
|
|
||||||
void clearPlugin(PluginId pluginId) {
|
void clearPlugin(PluginId pluginId) {
|
||||||
for (var element in _locationModels.values) {
|
for (var element in _locationModels.values) {
|
||||||
element.pluginModels.remove(pluginId);
|
element.pluginModels.remove(pluginId);
|
||||||
@ -77,7 +78,7 @@ void clearLocations() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OptionModel addOptionModel(
|
OptionModel getOptionModel(
|
||||||
String location, PluginId pluginId, String peer, String key) {
|
String location, PluginId pluginId, String peer, String key) {
|
||||||
final k = OptionModel.key(location, pluginId, peer, key);
|
final k = OptionModel.key(location, pluginId, peer, key);
|
||||||
if (_optionModels[k] == null) {
|
if (_optionModels[k] == null) {
|
||||||
|
@ -2,8 +2,10 @@ import 'dart:convert';
|
|||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
|
import 'package:flutter_hbb/common.dart';
|
||||||
import 'package:flutter_hbb/models/model.dart';
|
import 'package:flutter_hbb/models/model.dart';
|
||||||
import 'package:provider/provider.dart';
|
import 'package:provider/provider.dart';
|
||||||
|
import 'package:get/get.dart';
|
||||||
// to-do: do not depend on desktop
|
// to-do: do not depend on desktop
|
||||||
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';
|
||||||
@ -12,11 +14,15 @@ import './desc.dart';
|
|||||||
import './model.dart';
|
import './model.dart';
|
||||||
import './common.dart';
|
import './common.dart';
|
||||||
|
|
||||||
|
// dup to flutter\lib\desktop\pages\desktop_setting_page.dart
|
||||||
|
const double _kCheckBoxLeftMargin = 10;
|
||||||
|
|
||||||
class LocationItem extends StatelessWidget {
|
class LocationItem extends StatelessWidget {
|
||||||
final String peerId;
|
final String peerId;
|
||||||
final FFI ffi;
|
final FFI ffi;
|
||||||
final String location;
|
final String location;
|
||||||
final LocationModel locationModel;
|
final LocationModel locationModel;
|
||||||
|
final bool isMenu;
|
||||||
|
|
||||||
LocationItem({
|
LocationItem({
|
||||||
Key? key,
|
Key? key,
|
||||||
@ -24,19 +30,23 @@ class LocationItem extends StatelessWidget {
|
|||||||
required this.ffi,
|
required this.ffi,
|
||||||
required this.location,
|
required this.location,
|
||||||
required this.locationModel,
|
required this.locationModel,
|
||||||
|
required this.isMenu,
|
||||||
}) : super(key: key);
|
}) : super(key: key);
|
||||||
|
|
||||||
bool get isEmpty => locationModel.isEmpty;
|
bool get isEmpty => locationModel.isEmpty;
|
||||||
|
|
||||||
static LocationItem createLocationItem(
|
static Widget createLocationItem(
|
||||||
String peerId, FFI ffi, String location) {
|
String peerId, FFI ffi, String location, bool isMenu) {
|
||||||
final model = addLocation(location);
|
final model = getLocationModel(location);
|
||||||
return LocationItem(
|
return model == null
|
||||||
peerId: peerId,
|
? Container()
|
||||||
ffi: ffi,
|
: LocationItem(
|
||||||
location: location,
|
peerId: peerId,
|
||||||
locationModel: model,
|
ffi: ffi,
|
||||||
);
|
location: location,
|
||||||
|
locationModel: model,
|
||||||
|
isMenu: isMenu,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@ -59,23 +69,26 @@ class LocationItem extends StatelessWidget {
|
|||||||
ffi: ffi,
|
ffi: ffi,
|
||||||
location: location,
|
location: location,
|
||||||
pluginModel: model,
|
pluginModel: model,
|
||||||
|
isMenu: isMenu,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
class PluginItem extends StatelessWidget {
|
class PluginItem extends StatelessWidget {
|
||||||
final PluginId pluginId;
|
final PluginId pluginId;
|
||||||
final String peerId;
|
final String peerId;
|
||||||
final FFI ffi;
|
final FFI? ffi;
|
||||||
final String location;
|
final String location;
|
||||||
final PluginModel pluginModel;
|
final PluginModel pluginModel;
|
||||||
|
final isMenu;
|
||||||
|
|
||||||
PluginItem({
|
PluginItem({
|
||||||
Key? key,
|
Key? key,
|
||||||
required this.pluginId,
|
required this.pluginId,
|
||||||
required this.peerId,
|
required this.peerId,
|
||||||
required this.ffi,
|
this.ffi,
|
||||||
required this.location,
|
required this.location,
|
||||||
required this.pluginModel,
|
required this.pluginModel,
|
||||||
|
required this.isMenu,
|
||||||
}) : super(key: key);
|
}) : super(key: key);
|
||||||
|
|
||||||
bool get isEmpty => pluginModel.isEmpty;
|
bool get isEmpty => pluginModel.isEmpty;
|
||||||
@ -95,74 +108,88 @@ class PluginItem extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildItem(UiType ui) {
|
Widget _buildItem(UiType ui) {
|
||||||
late Widget child;
|
Widget? child;
|
||||||
switch (ui.runtimeType) {
|
switch (ui.runtimeType) {
|
||||||
case UiButton:
|
case UiButton:
|
||||||
child = _buildMenuButton(ui as UiButton);
|
if (isMenu) {
|
||||||
|
if (ffi != null) {
|
||||||
|
child = _buildMenuButton(ui as UiButton, ffi!);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
child = _buildButton(ui as UiButton);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case UiCheckbox:
|
case UiCheckbox:
|
||||||
child = _buildCheckboxMenuButton(ui as UiCheckbox);
|
if (isMenu) {
|
||||||
|
if (ffi != null) {
|
||||||
|
child = _buildCheckboxMenuButton(ui as UiCheckbox, ffi!);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
child = _buildCheckbox(ui as UiCheckbox);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
child = Container();
|
break;
|
||||||
}
|
}
|
||||||
// to-do: add plugin icon and tooltip
|
// to-do: add plugin icon and tooltip
|
||||||
return child;
|
return child ?? Container();
|
||||||
}
|
}
|
||||||
|
|
||||||
Uint8List _makeEvent(
|
Widget _buildButton(UiButton ui) {
|
||||||
String key, {
|
return TextButton(
|
||||||
bool? v,
|
|
||||||
}) {
|
|
||||||
final event = MsgFromUi(
|
|
||||||
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: () => bind.pluginEvent(
|
onPressed: () => bind.pluginEvent(
|
||||||
id: pluginId,
|
id: pluginId,
|
||||||
peer: peerId,
|
peer: peerId,
|
||||||
event: _makeEvent(ui.key),
|
event: _makeEvent(ui.key),
|
||||||
),
|
),
|
||||||
// to-do: support trailing icon, but it will cause tree shake error.
|
|
||||||
// ```
|
|
||||||
// This application cannot tree shake icons fonts. It has non-constant instances of IconData at the following locations:
|
|
||||||
// Target release_macos_bundle_flutter_assets failed: Exception: Avoid non-constant invocations of IconData or try to build again with --no-tree-shake-icons.
|
|
||||||
// ```
|
|
||||||
//
|
|
||||||
// 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,
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
String? getOption(OptionModel model, String key) {
|
Widget _buildCheckbox(UiCheckbox ui) {
|
||||||
var v = model.value;
|
getChild(OptionModel model) {
|
||||||
if (v == null) {
|
final v = _getOption(model, ui.key);
|
||||||
if (peerId.isEmpty) {
|
if (v == null) {
|
||||||
v = bind.pluginGetSharedOption(id: pluginId, key: key);
|
// session or plugin not found
|
||||||
} else {
|
return Container();
|
||||||
v = bind.pluginGetSessionOption(id: pluginId, peer: peerId, key: key);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onChanged(bool value) {
|
||||||
|
bind.pluginEvent(
|
||||||
|
id: pluginId,
|
||||||
|
peer: peerId,
|
||||||
|
event: _makeEvent(ui.key, v: value),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
final value = ConfigItem.isTrue(v);
|
||||||
|
return GestureDetector(
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Checkbox(
|
||||||
|
value: value,
|
||||||
|
onChanged: (_) => onChanged(!value),
|
||||||
|
).marginOnly(right: 5),
|
||||||
|
Expanded(
|
||||||
|
child: Text(translate(ui.text)),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
).marginOnly(left: _kCheckBoxLeftMargin),
|
||||||
|
onTap: () => onChanged(!value),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
return v;
|
|
||||||
|
return ChangeNotifierProvider.value(
|
||||||
|
value: getOptionModel(location, pluginId, peerId, ui.key),
|
||||||
|
child: Consumer<OptionModel>(
|
||||||
|
builder: (context, model, child) => getChild(model),
|
||||||
|
),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildCheckboxMenuButton(UiCheckbox ui) {
|
Widget _buildCheckboxMenuButton(UiCheckbox ui, FFI ffi) {
|
||||||
getChild(OptionModel model) {
|
getChild(OptionModel model) {
|
||||||
final v = getOption(model, ui.key);
|
final v = _getOption(model, ui.key);
|
||||||
if (v == null) {
|
if (v == null) {
|
||||||
// session or plugin not found
|
// session or plugin not found
|
||||||
return Container();
|
return Container();
|
||||||
@ -184,16 +211,63 @@ class PluginItem extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
final optionModel = addOptionModel(location, pluginId, peerId, ui.key);
|
|
||||||
return ChangeNotifierProvider.value(
|
return ChangeNotifierProvider.value(
|
||||||
value: optionModel,
|
value: getOptionModel(location, pluginId, peerId, ui.key),
|
||||||
child: Consumer<OptionModel>(
|
child: Consumer<OptionModel>(
|
||||||
builder: (context, model, child) {
|
builder: (context, model, child) => getChild(model),
|
||||||
return getChild(model);
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildMenuButton(UiButton ui, FFI ffi) {
|
||||||
|
return MenuButton(
|
||||||
|
onPressed: () => bind.pluginEvent(
|
||||||
|
id: pluginId,
|
||||||
|
peer: peerId,
|
||||||
|
event: _makeEvent(ui.key),
|
||||||
|
),
|
||||||
|
// to-do: support trailing icon, but it will cause tree shake error.
|
||||||
|
// ```
|
||||||
|
// This application cannot tree shake icons fonts. It has non-constant instances of IconData at the following locations:
|
||||||
|
// Target release_macos_bundle_flutter_assets failed: Exception: Avoid non-constant invocations of IconData or try to build again with --no-tree-shake-icons.
|
||||||
|
// ```
|
||||||
|
//
|
||||||
|
// trailingIcon: Icon(
|
||||||
|
// IconData(int.parse(ui.icon, radix: 16), fontFamily: 'MaterialIcons')),
|
||||||
|
//
|
||||||
|
// to-do: RustDesk translate or plugin translate ?
|
||||||
|
child: Text(ui.text),
|
||||||
|
ffi: ffi,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Uint8List _makeEvent(
|
||||||
|
String key, {
|
||||||
|
bool? v,
|
||||||
|
}) {
|
||||||
|
final event = MsgFromUi(
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
String? _getOption(OptionModel model, String key) {
|
||||||
|
var v = model.value;
|
||||||
|
if (v == null) {
|
||||||
|
if (peerId.isEmpty) {
|
||||||
|
v = bind.pluginGetSharedOption(id: pluginId, key: key);
|
||||||
|
} else {
|
||||||
|
v = bind.pluginGetSessionOption(id: pluginId, peer: peerId, key: key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return v;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void handleReloading(Map<String, dynamic> evt, String peer) {
|
void handleReloading(Map<String, dynamic> evt, String peer) {
|
||||||
|
@ -59,6 +59,10 @@ lazy_static::lazy_static! {
|
|||||||
pub static ref DEVICE_NAME: Arc<Mutex<String>> = Default::default();
|
pub static ref DEVICE_NAME: Arc<Mutex<String>> = Default::default();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lazy_static::lazy_static! {
|
||||||
|
static ref IS_SERVER: bool = std::env::args().nth(1) == Some("--server".to_owned());
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(not(any(target_os = "android", target_os = "ios")))]
|
#[cfg(not(any(target_os = "android", target_os = "ios")))]
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
static ref ARBOARD_MTX: Arc<Mutex<()>> = Arc::new(Mutex::new(()));
|
static ref ARBOARD_MTX: Arc<Mutex<()>> = Arc::new(Mutex::new(()));
|
||||||
@ -76,6 +80,11 @@ pub fn global_init() -> bool {
|
|||||||
|
|
||||||
pub fn global_clean() {}
|
pub fn global_clean() {}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn is_server() -> bool {
|
||||||
|
*IS_SERVER
|
||||||
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn valid_for_numlock(evt: &KeyEvent) -> bool {
|
pub fn valid_for_numlock(evt: &KeyEvent) -> bool {
|
||||||
if let Some(key_event::Union::ControlKey(ck)) = evt.union {
|
if let Some(key_event::Union::ControlKey(ck)) = evt.union {
|
||||||
|
@ -1466,6 +1466,7 @@ pub fn plugin_reload(_id: 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::ipc::reload_plugin(&_id,));
|
||||||
allow_err!(crate::plugin::reload_plugin(&_id));
|
allow_err!(crate::plugin::reload_plugin(&_id));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1474,7 +1475,8 @@ pub fn plugin_id_uninstall(_id: 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")))]
|
||||||
{
|
{
|
||||||
// to-do: uninstall plugin
|
crate::plugin::unload_plugin(&_id);
|
||||||
|
allow_err!(crate::plugin::ipc::uninstall_plugin(&_id));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1485,11 +1487,11 @@ pub fn plugin_id_enable(_id: String, _v: bool) {
|
|||||||
{
|
{
|
||||||
allow_err!(crate::plugin::ipc::set_manager_plugin_config(
|
allow_err!(crate::plugin::ipc::set_manager_plugin_config(
|
||||||
&_id,
|
&_id,
|
||||||
"enable",
|
"enabled",
|
||||||
_v.to_string()
|
_v.to_string()
|
||||||
));
|
));
|
||||||
if _v {
|
if _v {
|
||||||
allow_err!(crate::plugin::reload_plugin(&_id));
|
allow_err!(crate::plugin::load_plugin(None, Some(&_id)));
|
||||||
} else {
|
} else {
|
||||||
crate::plugin::unload_plugin(&_id);
|
crate::plugin::unload_plugin(&_id);
|
||||||
}
|
}
|
||||||
@ -1522,7 +1524,7 @@ pub fn plugin_enable(_v: bool) {
|
|||||||
#[cfg(not(any(target_os = "android", target_os = "ios")))]
|
#[cfg(not(any(target_os = "android", target_os = "ios")))]
|
||||||
{
|
{
|
||||||
allow_err!(crate::plugin::ipc::set_manager_config(
|
allow_err!(crate::plugin::ipc::set_manager_config(
|
||||||
"enable",
|
"enabled",
|
||||||
_v.to_string()
|
_v.to_string()
|
||||||
));
|
));
|
||||||
if _v {
|
if _v {
|
||||||
|
@ -60,14 +60,14 @@ struct MsgToConfig {
|
|||||||
/// id: The id of this plugin.
|
/// id: The id of this plugin.
|
||||||
/// content: The content.
|
/// content: The content.
|
||||||
/// len: The length of the content.
|
/// len: The length of the content.
|
||||||
pub fn callback_msg(
|
pub fn cb_msg(
|
||||||
peer: *const c_char,
|
peer: *const c_char,
|
||||||
target: *const c_char,
|
target: *const c_char,
|
||||||
id: *const c_char,
|
id: *const c_char,
|
||||||
content: *const c_void,
|
content: *const c_void,
|
||||||
len: usize,
|
len: usize,
|
||||||
) {
|
) {
|
||||||
macro_rules! callback_msg_field {
|
macro_rules! cb_msg_field {
|
||||||
($field: ident) => {
|
($field: ident) => {
|
||||||
let $field = match cstr_to_string($field) {
|
let $field = match cstr_to_string($field) {
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
@ -78,9 +78,9 @@ pub fn callback_msg(
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
callback_msg_field!(peer);
|
cb_msg_field!(peer);
|
||||||
callback_msg_field!(target);
|
cb_msg_field!(target);
|
||||||
callback_msg_field!(id);
|
cb_msg_field!(id);
|
||||||
|
|
||||||
match &target as _ {
|
match &target as _ {
|
||||||
MSG_TO_PEER_TARGET => {
|
MSG_TO_PEER_TARGET => {
|
||||||
|
@ -1,18 +1,20 @@
|
|||||||
use super::desc::ConfigItem;
|
use super::{cstr_to_string, str_to_cstr_ret};
|
||||||
use hbb_common::{allow_err, bail, config::Config as HbbConfig, lazy_static, log, ResultType};
|
use hbb_common::{allow_err, bail, config::Config as HbbConfig, lazy_static, log, ResultType};
|
||||||
use serde_derive::{Deserialize, Serialize};
|
use serde_derive::{Deserialize, Serialize};
|
||||||
use std::{
|
use std::{
|
||||||
|
collections::HashMap,
|
||||||
|
ffi::c_char,
|
||||||
|
fs,
|
||||||
ops::{Deref, DerefMut},
|
ops::{Deref, DerefMut},
|
||||||
|
path::PathBuf,
|
||||||
|
ptr,
|
||||||
str::FromStr,
|
str::FromStr,
|
||||||
sync::{Arc, Mutex},
|
sync::{Arc, Mutex},
|
||||||
{collections::HashMap, path::PathBuf},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
static ref CONFIG_SHARED: Arc<Mutex<HashMap<String, SharedConfig>>> = Default::default();
|
static ref CONFIG_SHARED: Arc<Mutex<HashMap<String, SharedConfig>>> = Default::default();
|
||||||
static ref CONFIG_SHARED_ITEMS: Arc<Mutex<HashMap<String, Vec<ConfigItem>>>> = Default::default();
|
|
||||||
static ref CONFIG_PEERS: Arc<Mutex<HashMap<String, PeersConfig>>> = Default::default();
|
static ref CONFIG_PEERS: Arc<Mutex<HashMap<String, PeersConfig>>> = Default::default();
|
||||||
static ref CONFIG_PEER_ITEMS: Arc<Mutex<HashMap<String, Vec<ConfigItem>>>> = Default::default();
|
|
||||||
static ref CONFIG_MANAGER: Arc<Mutex<ManagerConfig>> = {
|
static ref CONFIG_MANAGER: Arc<Mutex<ManagerConfig>> = {
|
||||||
let conf = hbb_common::config::load_path::<ManagerConfig>(ManagerConfig::path());
|
let conf = hbb_common::config::load_path::<ManagerConfig>(ManagerConfig::path());
|
||||||
Arc::new(Mutex::new(conf))
|
Arc::new(Mutex::new(conf))
|
||||||
@ -68,25 +70,34 @@ impl SharedConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn load(id: &str) {
|
fn load(id: &str) {
|
||||||
|
let mut lock = CONFIG_SHARED.lock().unwrap();
|
||||||
|
if lock.contains_key(id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
let conf = hbb_common::config::load_path::<HashMap<String, String>>(Self::path(id));
|
let conf = hbb_common::config::load_path::<HashMap<String, String>>(Self::path(id));
|
||||||
let mut conf = SharedConfig(conf);
|
let mut conf = SharedConfig(conf);
|
||||||
if let Some(items) = CONFIG_SHARED_ITEMS.lock().unwrap().get(id) {
|
if let Some(desc_conf) = super::plugins::get_desc_conf(id) {
|
||||||
for item in items {
|
for item in desc_conf.shared.iter() {
|
||||||
if !conf.contains_key(&item.key) {
|
if !conf.contains_key(&item.key) {
|
||||||
conf.insert(item.key.to_owned(), item.default.to_owned());
|
conf.insert(item.key.to_owned(), item.default.to_owned());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
CONFIG_SHARED.lock().unwrap().insert(id.to_owned(), conf);
|
lock.insert(id.to_owned(), conf);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn load_if_not_exists(id: &str) {
|
||||||
|
if CONFIG_SHARED.lock().unwrap().contains_key(id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Self::load(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn get(id: &str, key: &str) -> Option<String> {
|
pub fn get(id: &str, key: &str) -> Option<String> {
|
||||||
if let Some(conf) = CONFIG_SHARED.lock().unwrap().get(id) {
|
Self::load_if_not_exists(id);
|
||||||
return conf.get(key).map(|s| s.to_owned());
|
|
||||||
}
|
|
||||||
Self::load(id);
|
|
||||||
CONFIG_SHARED
|
CONFIG_SHARED
|
||||||
.lock()
|
.lock()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
@ -97,12 +108,16 @@ impl SharedConfig {
|
|||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn set(id: &str, key: &str, value: &str) -> ResultType<()> {
|
pub fn set(id: &str, key: &str, value: &str) -> ResultType<()> {
|
||||||
|
Self::load_if_not_exists(id);
|
||||||
match CONFIG_SHARED.lock().unwrap().get_mut(id) {
|
match CONFIG_SHARED.lock().unwrap().get_mut(id) {
|
||||||
Some(config) => {
|
Some(config) => {
|
||||||
config.insert(key.to_owned(), value.to_owned());
|
config.insert(key.to_owned(), value.to_owned());
|
||||||
hbb_common::config::store_path(Self::path(id), config)
|
hbb_common::config::store_path(Self::path(id), config)
|
||||||
}
|
}
|
||||||
None => bail!("No such plugin {}", id),
|
None => {
|
||||||
|
// unreachable
|
||||||
|
bail!("No such plugin {}", id)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -116,35 +131,47 @@ impl PeerConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn load(id: &str, peer: &str) {
|
fn load(id: &str, peer: &str) {
|
||||||
|
let mut lock = CONFIG_PEERS.lock().unwrap();
|
||||||
|
if let Some(peers) = lock.get(id) {
|
||||||
|
if peers.contains_key(peer) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let conf = hbb_common::config::load_path::<HashMap<String, String>>(Self::path(id, peer));
|
let conf = hbb_common::config::load_path::<HashMap<String, String>>(Self::path(id, peer));
|
||||||
let mut conf = PeerConfig(conf);
|
let mut conf = PeerConfig(conf);
|
||||||
if let Some(items) = CONFIG_PEER_ITEMS.lock().unwrap().get(id) {
|
if let Some(desc_conf) = super::plugins::get_desc_conf(id) {
|
||||||
for item in items {
|
for item in desc_conf.peer.iter() {
|
||||||
if !conf.contains_key(&item.key) {
|
if !conf.contains_key(&item.key) {
|
||||||
conf.insert(item.key.to_owned(), item.default.to_owned());
|
conf.insert(item.key.to_owned(), item.default.to_owned());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(peers) = CONFIG_PEERS.lock().unwrap().get_mut(id) {
|
if let Some(peers) = lock.get_mut(id) {
|
||||||
peers.insert(peer.to_owned(), conf);
|
peers.insert(peer.to_owned(), conf);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut peers = HashMap::new();
|
let mut peers = HashMap::new();
|
||||||
peers.insert(peer.to_owned(), conf);
|
peers.insert(peer.to_owned(), conf);
|
||||||
CONFIG_PEERS.lock().unwrap().insert(id.to_owned(), peers);
|
lock.insert(id.to_owned(), peers);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn load_if_not_exists(id: &str, peer: &str) {
|
||||||
|
if let Some(peers) = CONFIG_PEERS.lock().unwrap().get(id) {
|
||||||
|
if peers.contains_key(peer) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Self::load(id, peer);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn get(id: &str, peer: &str, key: &str) -> Option<String> {
|
pub fn get(id: &str, peer: &str, key: &str) -> Option<String> {
|
||||||
if let Some(peers) = CONFIG_PEERS.lock().unwrap().get(id) {
|
Self::load_if_not_exists(id, peer);
|
||||||
if let Some(conf) = peers.get(peer) {
|
|
||||||
return conf.get(key).map(|s| s.to_owned());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Self::load(id, peer);
|
|
||||||
CONFIG_PEERS
|
CONFIG_PEERS
|
||||||
.lock()
|
.lock()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
@ -156,35 +183,26 @@ impl PeerConfig {
|
|||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn set(id: &str, peer: &str, key: &str, value: &str) -> ResultType<()> {
|
pub fn set(id: &str, peer: &str, key: &str, value: &str) -> ResultType<()> {
|
||||||
|
Self::load_if_not_exists(id, peer);
|
||||||
match CONFIG_PEERS.lock().unwrap().get_mut(id) {
|
match CONFIG_PEERS.lock().unwrap().get_mut(id) {
|
||||||
Some(peers) => match peers.get_mut(peer) {
|
Some(peers) => match peers.get_mut(peer) {
|
||||||
Some(config) => {
|
Some(config) => {
|
||||||
config.insert(key.to_owned(), value.to_owned());
|
config.insert(key.to_owned(), value.to_owned());
|
||||||
hbb_common::config::store_path(Self::path(id, peer), config)
|
hbb_common::config::store_path(Self::path(id, peer), config)
|
||||||
}
|
}
|
||||||
None => bail!("No such peer {}", peer),
|
None => {
|
||||||
|
// unreachable
|
||||||
|
bail!("No such peer {}", peer)
|
||||||
|
}
|
||||||
},
|
},
|
||||||
None => bail!("No such plugin {}", id),
|
None => {
|
||||||
|
// unreachable
|
||||||
|
bail!("No such plugin {}", id)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(super) fn set_shared_items(id: &str, items: &Vec<ConfigItem>) {
|
|
||||||
CONFIG_SHARED_ITEMS
|
|
||||||
.lock()
|
|
||||||
.unwrap()
|
|
||||||
.insert(id.to_owned(), items.clone());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(super) fn set_peer_items(id: &str, items: &Vec<ConfigItem>) {
|
|
||||||
CONFIG_PEER_ITEMS
|
|
||||||
.lock()
|
|
||||||
.unwrap()
|
|
||||||
.insert(id.to_owned(), items.clone());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
pub struct PluginStatus {
|
pub struct PluginStatus {
|
||||||
pub enabled: bool,
|
pub enabled: bool,
|
||||||
@ -234,21 +252,31 @@ impl ManagerConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
fn set_option_enabled(enabled: bool) -> ResultType<()> {
|
||||||
pub fn set_option(key: &str, value: &str) -> ResultType<()> {
|
|
||||||
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
||||||
|
lock.enabled = enabled;
|
||||||
|
hbb_common::config::store_path(Self::path(), &*lock)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set_option_not_enabled(key: &str, value: &str) -> ResultType<()> {
|
||||||
|
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
||||||
|
lock.options.insert(key.to_owned(), value.to_owned());
|
||||||
|
hbb_common::config::store_path(Self::path(), &*lock)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn set_option(key: &str, value: &str) {
|
||||||
if key == "enabled" {
|
if key == "enabled" {
|
||||||
let enabled = bool::from_str(value).unwrap_or(false);
|
let enabled = bool::from_str(value).unwrap_or(false);
|
||||||
lock.enabled = enabled;
|
allow_err!(Self::set_option_enabled(enabled));
|
||||||
if enabled {
|
if enabled {
|
||||||
allow_err!(super::load_plugins());
|
allow_err!(super::load_plugins());
|
||||||
} else {
|
} else {
|
||||||
super::unload_plugins();
|
super::unload_plugins();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
lock.options.insert(key.to_owned(), value.to_owned());
|
allow_err!(Self::set_option_not_enabled(key, value));
|
||||||
}
|
}
|
||||||
hbb_common::config::store_path(Self::path(), &*lock)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@ -261,27 +289,31 @@ impl ManagerConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_plugin_option(id: &str, key: &str, value: &str) -> ResultType<()> {
|
fn set_plugin_option_enabled(id: &str, enabled: bool) -> ResultType<()> {
|
||||||
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
||||||
if let Some(status) = lock.plugins.get_mut(id) {
|
if let Some(status) = lock.plugins.get_mut(id) {
|
||||||
match key {
|
status.enabled = enabled;
|
||||||
"enabled" => {
|
|
||||||
let enabled = bool::from_str(value).unwrap_or(false);
|
|
||||||
status.enabled = enabled;
|
|
||||||
if enabled {
|
|
||||||
allow_err!(super::load_plugin(None, Some(id)));
|
|
||||||
} else {
|
|
||||||
super::unload_plugin(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => bail!("No such option {}", key),
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
bail!("No such plugin {}", id)
|
lock.plugins.insert(id.to_owned(), PluginStatus { enabled });
|
||||||
}
|
}
|
||||||
hbb_common::config::store_path(Self::path(), &*lock)
|
hbb_common::config::store_path(Self::path(), &*lock)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn set_plugin_option(id: &str, key: &str, value: &str) {
|
||||||
|
match key {
|
||||||
|
"enabled" => {
|
||||||
|
let enabled = bool::from_str(value).unwrap_or(false);
|
||||||
|
allow_err!(Self::set_plugin_option_enabled(id, enabled));
|
||||||
|
if enabled {
|
||||||
|
allow_err!(super::load_plugin(None, Some(id)));
|
||||||
|
} else {
|
||||||
|
super::unload_plugin(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => log::error!("No such option {}", key),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn add_plugin(id: &str) -> ResultType<()> {
|
pub fn add_plugin(id: &str) -> ResultType<()> {
|
||||||
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
||||||
@ -291,11 +323,58 @@ impl ManagerConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn remove_plugin(id: &str) -> ResultType<()> {
|
pub fn remove_plugin(id: &str, uninstall: bool) -> ResultType<()> {
|
||||||
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
||||||
lock.plugins.remove(id);
|
lock.plugins.remove(id);
|
||||||
hbb_common::config::store_path(Self::path(), &*lock)?;
|
hbb_common::config::store_path(Self::path(), &*lock)?;
|
||||||
// to-do: remove plugin config dir
|
if uninstall {
|
||||||
|
allow_err!(fs::remove_dir_all(path_plugins(id)));
|
||||||
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn remove_plugins(uninstall: bool) {
|
||||||
|
let mut lock = CONFIG_MANAGER.lock().unwrap();
|
||||||
|
lock.plugins.clear();
|
||||||
|
allow_err!(hbb_common::config::store_path(Self::path(), &*lock));
|
||||||
|
if uninstall {
|
||||||
|
allow_err!(fs::remove_dir_all(HbbConfig::path("plugins")));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return shared config if peer is nullptr.
|
||||||
|
pub(super) fn cb_get_conf(
|
||||||
|
peer: *const c_char,
|
||||||
|
id: *const c_char,
|
||||||
|
key: *const c_char,
|
||||||
|
) -> *const c_char {
|
||||||
|
match (cstr_to_string(id), cstr_to_string(key)) {
|
||||||
|
(Ok(id), Ok(key)) => {
|
||||||
|
if peer.is_null() {
|
||||||
|
SharedConfig::load_if_not_exists(&id);
|
||||||
|
if let Some(conf) = CONFIG_SHARED.lock().unwrap().get(&id) {
|
||||||
|
if let Some(value) = conf.get(&key) {
|
||||||
|
return str_to_cstr_ret(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
match cstr_to_string(peer) {
|
||||||
|
Ok(peer) => {
|
||||||
|
PeerConfig::load_if_not_exists(&id, &peer);
|
||||||
|
if let Some(conf) = CONFIG_PEERS.lock().unwrap().get(&id) {
|
||||||
|
if let Some(conf) = conf.get(&peer) {
|
||||||
|
if let Some(value) = conf.get(&key) {
|
||||||
|
return str_to_cstr_ret(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(_) => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
ptr::null()
|
||||||
}
|
}
|
||||||
|
@ -40,7 +40,7 @@ pub struct ConfigItem {
|
|||||||
pub description: String,
|
pub description: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
pub shared: Vec<ConfigItem>,
|
pub shared: Vec<ConfigItem>,
|
||||||
pub peer: Vec<ConfigItem>,
|
pub peer: Vec<ConfigItem>,
|
||||||
|
@ -12,7 +12,7 @@ pub enum Plugin {
|
|||||||
ManagerConfig(String, Option<String>),
|
ManagerConfig(String, Option<String>),
|
||||||
ManagerPluginConfig(String, String, Option<String>),
|
ManagerPluginConfig(String, String, Option<String>),
|
||||||
Reload(String),
|
Reload(String),
|
||||||
Uninstall,
|
Uninstall(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::main(flavor = "current_thread")]
|
#[tokio::main(flavor = "current_thread")]
|
||||||
@ -45,6 +45,16 @@ pub async fn set_manager_plugin_config(id: &str, name: &str, value: String) -> R
|
|||||||
set_manager_plugin_config_async(id, name, value).await
|
set_manager_plugin_config_async(id, name, value).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::main(flavor = "current_thread")]
|
||||||
|
pub async fn reload_plugin(id: &str) -> ResultType<()> {
|
||||||
|
reload_plugin_async(id).await
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::main(flavor = "current_thread")]
|
||||||
|
pub async fn uninstall_plugin(id: &str) -> ResultType<()> {
|
||||||
|
uninstall_plugin_async(id).await
|
||||||
|
}
|
||||||
|
|
||||||
async fn get_config_async(id: &str, name: &str, ms_timeout: u64) -> ResultType<Option<String>> {
|
async fn get_config_async(id: &str, name: &str, ms_timeout: u64) -> ResultType<Option<String>> {
|
||||||
let mut c = connect(ms_timeout, "").await?;
|
let mut c = connect(ms_timeout, "").await?;
|
||||||
c.send(&Data::Plugin(Plugin::Config(
|
c.send(&Data::Plugin(Plugin::Config(
|
||||||
@ -131,6 +141,19 @@ async fn set_manager_plugin_config_async(id: &str, name: &str, value: String) ->
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn reload_plugin_async(id: &str) -> ResultType<()> {
|
||||||
|
let mut c = connect(1000, "").await?;
|
||||||
|
c.send(&Data::Plugin(Plugin::Reload(id.to_owned()))).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn uninstall_plugin_async(id: &str) -> ResultType<()> {
|
||||||
|
let mut c = connect(1000, "").await?;
|
||||||
|
c.send(&Data::Plugin(Plugin::Uninstall(id.to_owned())))
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
pub async fn handle_plugin(plugin: Plugin, stream: &mut Connection) {
|
pub async fn handle_plugin(plugin: Plugin, stream: &mut Connection) {
|
||||||
match plugin {
|
match plugin {
|
||||||
Plugin::Config(id, name, value) => match value {
|
Plugin::Config(id, name, value) => match value {
|
||||||
@ -156,7 +179,7 @@ pub async fn handle_plugin(plugin: Plugin, stream: &mut Connection) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
Some(value) => {
|
Some(value) => {
|
||||||
allow_err!(crate::plugin::ManagerConfig::set_option(&name, &value));
|
crate::plugin::ManagerConfig::set_option(&name, &value);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
Plugin::ManagerPluginConfig(id, name, value) => match value {
|
Plugin::ManagerPluginConfig(id, name, value) => match value {
|
||||||
@ -169,16 +192,16 @@ pub async fn handle_plugin(plugin: Plugin, stream: &mut Connection) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
Some(value) => {
|
Some(value) => {
|
||||||
allow_err!(crate::plugin::ManagerConfig::set_plugin_option(
|
crate::plugin::ManagerConfig::set_plugin_option(&id, &name, &value);
|
||||||
&id, &name, &value
|
|
||||||
));
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
Plugin::Reload(id) => {
|
Plugin::Reload(id) => {
|
||||||
allow_err!(crate::plugin::reload_plugin(&id));
|
allow_err!(crate::plugin::reload_plugin(&id));
|
||||||
}
|
}
|
||||||
Plugin::Uninstall => {
|
Plugin::Uninstall(_id) => {
|
||||||
// to-do: uninstall plugin
|
// to-do: uninstall plugin
|
||||||
|
// 1. unload 2. remove configs 3. remove config files
|
||||||
|
// allow_err!(crate::plugin::unload_plugin(&id));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use hbb_common::ResultType;
|
use hbb_common::{libc, ResultType};
|
||||||
use std::ffi::{c_char, c_void, CStr};
|
use std::ffi::{c_char, c_void, CStr};
|
||||||
|
|
||||||
mod callback_msg;
|
mod callback_msg;
|
||||||
@ -27,6 +27,21 @@ fn cstr_to_string(cstr: *const c_char) -> ResultType<String> {
|
|||||||
})?)
|
})?)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn str_to_cstr_ret(s: &str) -> *const c_char {
|
||||||
|
let mut s = s.as_bytes().to_vec();
|
||||||
|
s.push(0);
|
||||||
|
unsafe {
|
||||||
|
let r = libc::malloc(s.len()) as *mut c_char;
|
||||||
|
libc::memcpy(
|
||||||
|
r as *mut libc::c_void,
|
||||||
|
s.as_ptr() as *const libc::c_void,
|
||||||
|
s.len(),
|
||||||
|
);
|
||||||
|
r
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn get_code_msg_from_ret(ret: *const c_void) -> (i32, String) {
|
fn get_code_msg_from_ret(ret: *const c_void) -> (i32, String) {
|
||||||
assert!(ret.is_null() == false);
|
assert!(ret.is_null() == false);
|
||||||
@ -38,3 +53,12 @@ fn get_code_msg_from_ret(ret: *const c_void) -> (i32, String) {
|
|||||||
.to_string();
|
.to_string();
|
||||||
(code, msg)
|
(code, msg)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn free_c_ptr(ret: *mut c_void) {
|
||||||
|
if !ret.is_null() {
|
||||||
|
unsafe {
|
||||||
|
libc::free(ret);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,3 +1,14 @@
|
|||||||
|
use super::{desc::Desc, errno::*, *};
|
||||||
|
#[cfg(not(debug_assertions))]
|
||||||
|
use crate::common::is_server;
|
||||||
|
use crate::{flutter, ui_interface::get_id};
|
||||||
|
use hbb_common::{
|
||||||
|
allow_err, bail,
|
||||||
|
dlopen::symbor::Library,
|
||||||
|
lazy_static, log,
|
||||||
|
message_proto::{Message, Misc, PluginResponse},
|
||||||
|
ResultType,
|
||||||
|
};
|
||||||
use std::{
|
use std::{
|
||||||
collections::HashMap,
|
collections::HashMap,
|
||||||
ffi::{c_char, c_void},
|
ffi::{c_char, c_void},
|
||||||
@ -5,16 +16,6 @@ use std::{
|
|||||||
sync::{Arc, RwLock},
|
sync::{Arc, RwLock},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::{desc::Desc, errno::*, *};
|
|
||||||
use crate::{flutter, ui_interface::get_id};
|
|
||||||
use hbb_common::{
|
|
||||||
bail,
|
|
||||||
dlopen::symbor::Library,
|
|
||||||
lazy_static, libc, log,
|
|
||||||
message_proto::{Message, Misc, PluginResponse},
|
|
||||||
ResultType,
|
|
||||||
};
|
|
||||||
|
|
||||||
const METHOD_HANDLE_UI: &[u8; 10] = b"handle_ui\0";
|
const METHOD_HANDLE_UI: &[u8; 10] = b"handle_ui\0";
|
||||||
const METHOD_HANDLE_PEER: &[u8; 12] = b"handle_peer\0";
|
const METHOD_HANDLE_PEER: &[u8; 12] = b"handle_peer\0";
|
||||||
|
|
||||||
@ -58,19 +59,27 @@ pub type PluginFuncDesc = fn() -> *const c_char;
|
|||||||
/// id: The id of this plugin.
|
/// id: The id of this plugin.
|
||||||
/// content: The content.
|
/// content: The content.
|
||||||
/// len: The length of the content.
|
/// len: The length of the content.
|
||||||
type PluginFuncCallbackMsg = fn(
|
type CallbackMsg = fn(
|
||||||
peer: *const c_char,
|
peer: *const c_char,
|
||||||
target: *const c_char,
|
target: *const c_char,
|
||||||
id: *const c_char,
|
id: *const c_char,
|
||||||
content: *const c_void,
|
content: *const c_void,
|
||||||
len: usize,
|
len: usize,
|
||||||
);
|
);
|
||||||
pub type PluginFuncSetCallbackMsg = fn(PluginFuncCallbackMsg);
|
|
||||||
/// Callback to get the id of local peer id.
|
/// Callback to get the id of local peer id.
|
||||||
/// The returned string is utf8 string(null terminated).
|
/// The returned string is utf8 string(null terminated).
|
||||||
/// Don't free the returned ptr.
|
/// Don't free the returned ptr.
|
||||||
type GetIdFuncCallback = fn() -> *const c_char;
|
type CallbackGetId = fn() -> *const c_char;
|
||||||
pub type PluginFuncGetIdCallback = fn(GetIdFuncCallback);
|
/// Callback to get the config.
|
||||||
|
/// peer, key are utf8 strings(null terminated).
|
||||||
|
///
|
||||||
|
/// peer: The peer id.
|
||||||
|
/// id: The id of this plugin.
|
||||||
|
/// key: The key of the config.
|
||||||
|
///
|
||||||
|
/// The returned string is utf8 string(null terminated) and must be freed by caller.
|
||||||
|
type CallbackGetConf =
|
||||||
|
fn(peer: *const c_char, id: *const c_char, key: *const c_char) -> *const c_char;
|
||||||
/// The main function of the plugin.
|
/// The main function of the plugin.
|
||||||
/// method: The method. "handle_ui" or "handle_peer"
|
/// method: The method. "handle_ui" or "handle_peer"
|
||||||
/// peer: The peer id.
|
/// peer: The peer id.
|
||||||
@ -86,10 +95,21 @@ pub type PluginFuncCall = fn(
|
|||||||
len: usize,
|
len: usize,
|
||||||
) -> *const c_void;
|
) -> *const c_void;
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
struct Callbacks {
|
||||||
|
msg: CallbackMsg,
|
||||||
|
get_id: CallbackGetId,
|
||||||
|
get_conf: CallbackGetConf,
|
||||||
|
}
|
||||||
|
type PluginFuncSetCallbacks = fn(Callbacks);
|
||||||
|
|
||||||
macro_rules! make_plugin {
|
macro_rules! make_plugin {
|
||||||
($($field:ident : $tp:ty),+) => {
|
($($field:ident : $tp:ty),+) => {
|
||||||
|
#[allow(dead_code)]
|
||||||
pub struct Plugin {
|
pub struct Plugin {
|
||||||
_lib: Library,
|
_lib: Library,
|
||||||
|
id: Option<String>,
|
||||||
|
path: String,
|
||||||
$($field: $tp),+
|
$($field: $tp),+
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -115,9 +135,54 @@ macro_rules! make_plugin {
|
|||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
_lib: lib,
|
_lib: lib,
|
||||||
|
id: None,
|
||||||
|
path: path.to_string(),
|
||||||
$( $field ),+
|
$( $field ),+
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn desc(&self) -> ResultType<Desc> {
|
||||||
|
let desc_ret = (self.desc)();
|
||||||
|
let desc = Desc::from_cstr(desc_ret);
|
||||||
|
free_c_ptr(desc_ret as _);
|
||||||
|
desc
|
||||||
|
}
|
||||||
|
|
||||||
|
fn init(&self, path: &str) -> ResultType<()> {
|
||||||
|
let init_ret = (self.init)();
|
||||||
|
if !init_ret.is_null() {
|
||||||
|
let (code, msg) = get_code_msg_from_ret(init_ret);
|
||||||
|
free_c_ptr(init_ret as _);
|
||||||
|
bail!(
|
||||||
|
"Failed to init plugin {}, code: {}, msg: {}",
|
||||||
|
path,
|
||||||
|
code,
|
||||||
|
msg
|
||||||
|
);
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn clear(&self, id: &str) {
|
||||||
|
let clear_ret = (self.clear)();
|
||||||
|
if !clear_ret.is_null() {
|
||||||
|
let (code, msg) = get_code_msg_from_ret(clear_ret);
|
||||||
|
free_c_ptr(clear_ret as _);
|
||||||
|
log::error!(
|
||||||
|
"Failed to clear plugin {}, code: {}, msg: {}",
|
||||||
|
id,
|
||||||
|
code,
|
||||||
|
msg
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for Plugin {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
let id = self.id.as_ref().unwrap_or(&self.path);
|
||||||
|
self.clear(id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -128,8 +193,7 @@ make_plugin!(
|
|||||||
clear: PluginFuncClear,
|
clear: PluginFuncClear,
|
||||||
desc: PluginFuncDesc,
|
desc: PluginFuncDesc,
|
||||||
call: PluginFuncCall,
|
call: PluginFuncCall,
|
||||||
set_cb_msg: PluginFuncSetCallbackMsg,
|
set_cbs: PluginFuncSetCallbacks
|
||||||
set_cb_get_id: PluginFuncGetIdCallback
|
|
||||||
);
|
);
|
||||||
|
|
||||||
#[cfg(target_os = "windows")]
|
#[cfg(target_os = "windows")]
|
||||||
@ -172,17 +236,18 @@ pub fn load_plugins() -> ResultType<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn unload_plugins() {
|
pub fn unload_plugins() {
|
||||||
let mut plugins = PLUGINS.write().unwrap();
|
log::info!("Plugins unloaded");
|
||||||
for (id, plugin) in plugins.iter() {
|
PLUGINS.write().unwrap().clear();
|
||||||
let _ret = (plugin.clear)();
|
if change_manager() {
|
||||||
log::info!("Plugin {} unloaded", id);
|
super::config::ManagerConfig::remove_plugins(false);
|
||||||
}
|
}
|
||||||
plugins.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn unload_plugin(id: &str) {
|
pub fn unload_plugin(id: &str) {
|
||||||
if let Some(plugin) = PLUGINS.write().unwrap().remove(id) {
|
log::info!("Plugin {} unloaded", id);
|
||||||
let _ret = (plugin.clear)();
|
PLUGINS.write().unwrap().remove(id);
|
||||||
|
if change_manager() {
|
||||||
|
allow_err!(super::config::ManagerConfig::remove_plugin(id, false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -196,7 +261,7 @@ pub fn reload_plugin(id: &str) -> ResultType<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
fn get_local_peer_id() -> *const c_char {
|
fn cb_get_local_peer_id() -> *const c_char {
|
||||||
let mut id = (*LOCAL_PEER_ID.read().unwrap()).clone();
|
let mut id = (*LOCAL_PEER_ID.read().unwrap()).clone();
|
||||||
if id.is_empty() {
|
if id.is_empty() {
|
||||||
let mut lock = LOCAL_PEER_ID.write().unwrap();
|
let mut lock = LOCAL_PEER_ID.write().unwrap();
|
||||||
@ -212,28 +277,38 @@ fn get_local_peer_id() -> *const c_char {
|
|||||||
|
|
||||||
fn load_plugin_path(path: &str) -> ResultType<()> {
|
fn load_plugin_path(path: &str) -> ResultType<()> {
|
||||||
let plugin = Plugin::new(path)?;
|
let plugin = Plugin::new(path)?;
|
||||||
let desc = (plugin.desc)();
|
let desc = plugin.desc()?;
|
||||||
let desc_res = Desc::from_cstr(desc);
|
|
||||||
unsafe {
|
|
||||||
libc::free(desc as _);
|
|
||||||
}
|
|
||||||
let desc = desc_res?;
|
|
||||||
let id = desc.id().to_string();
|
|
||||||
// to-do validate plugin
|
// to-do validate plugin
|
||||||
// to-do check the plugin id (make sure it does not use another plugin's id)
|
// to-do check the plugin id (make sure it does not use another plugin's id)
|
||||||
(plugin.set_cb_msg)(callback_msg::callback_msg);
|
|
||||||
(plugin.set_cb_get_id)(get_local_peer_id as _);
|
plugin.init(path)?;
|
||||||
|
|
||||||
|
if change_manager() {
|
||||||
|
super::config::ManagerConfig::add_plugin(desc.id())?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// set callbacks
|
||||||
|
(plugin.set_cbs)(Callbacks {
|
||||||
|
msg: callback_msg::cb_msg,
|
||||||
|
get_id: cb_get_local_peer_id,
|
||||||
|
get_conf: config::cb_get_conf,
|
||||||
|
});
|
||||||
|
|
||||||
|
// update ui
|
||||||
// Ui may be not ready now, so we need to update again once ui is ready.
|
// Ui may be not ready now, so we need to update again once ui is ready.
|
||||||
update_ui_plugin_desc(&desc, None);
|
update_ui_plugin_desc(&desc, None);
|
||||||
update_config(&desc);
|
|
||||||
// Ui may be not ready now, so we need to reload again once ui is ready.
|
|
||||||
reload_ui(&desc, None);
|
reload_ui(&desc, None);
|
||||||
|
|
||||||
|
// add plugins
|
||||||
|
let id = desc.id().to_string();
|
||||||
let plugin_info = PluginInfo {
|
let plugin_info = PluginInfo {
|
||||||
path: path.to_string(),
|
path: path.to_string(),
|
||||||
desc,
|
desc,
|
||||||
};
|
};
|
||||||
PLUGIN_INFO.write().unwrap().insert(id.clone(), plugin_info);
|
PLUGIN_INFO.write().unwrap().insert(id.clone(), plugin_info);
|
||||||
PLUGINS.write().unwrap().insert(id.clone(), plugin);
|
PLUGINS.write().unwrap().insert(id.clone(), plugin);
|
||||||
|
|
||||||
log::info!("Plugin {} loaded", id);
|
log::info!("Plugin {} loaded", id);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -248,10 +323,13 @@ pub fn sync_ui(sync_to: String) {
|
|||||||
pub fn load_plugin(path: Option<&str>, id: Option<&str>) -> ResultType<()> {
|
pub fn load_plugin(path: Option<&str>, id: Option<&str>) -> ResultType<()> {
|
||||||
match (path, id) {
|
match (path, id) {
|
||||||
(Some(path), _) => load_plugin_path(path),
|
(Some(path), _) => load_plugin_path(path),
|
||||||
(None, Some(id)) => match PLUGIN_INFO.read().unwrap().get(id) {
|
(None, Some(id)) => {
|
||||||
Some(plugin) => load_plugin_path(&plugin.path),
|
let path = match PLUGIN_INFO.read().unwrap().get(id) {
|
||||||
None => bail!("Plugin {} not found", id),
|
Some(plugin) => plugin.path.clone(),
|
||||||
},
|
None => bail!("Plugin {} not found", id),
|
||||||
|
};
|
||||||
|
load_plugin_path(&path)
|
||||||
|
}
|
||||||
(None, None) => {
|
(None, None) => {
|
||||||
bail!("path and id are both None");
|
bail!("path and id are both None");
|
||||||
}
|
}
|
||||||
@ -273,9 +351,7 @@ fn handle_event(method: &[u8], id: &str, peer: &str, event: &[u8]) -> ResultType
|
|||||||
Ok(())
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
let (code, msg) = get_code_msg_from_ret(ret);
|
let (code, msg) = get_code_msg_from_ret(ret);
|
||||||
unsafe {
|
free_c_ptr(ret as _);
|
||||||
libc::free(ret as _);
|
|
||||||
}
|
|
||||||
bail!(
|
bail!(
|
||||||
"Failed to handle plugin event, id: {}, method: {}, code: {}, msg: {}",
|
"Failed to handle plugin event, id: {}, method: {}, code: {}, msg: {}",
|
||||||
id,
|
id,
|
||||||
@ -315,9 +391,7 @@ pub fn handle_client_event(id: &str, peer: &str, event: &[u8]) -> Option<Message
|
|||||||
None
|
None
|
||||||
} else {
|
} else {
|
||||||
let (code, msg) = get_code_msg_from_ret(ret);
|
let (code, msg) = get_code_msg_from_ret(ret);
|
||||||
unsafe {
|
free_c_ptr(ret as _);
|
||||||
libc::free(ret as _);
|
|
||||||
}
|
|
||||||
if code > ERR_RUSTDESK_HANDLE_BASE && code < ERR_PLUGIN_HANDLE_BASE {
|
if code > ERR_RUSTDESK_HANDLE_BASE && code < ERR_PLUGIN_HANDLE_BASE {
|
||||||
let name = match PLUGIN_INFO.read().unwrap().get(id) {
|
let name = match PLUGIN_INFO.read().unwrap().get(id) {
|
||||||
Some(plugin) => plugin.desc.name(),
|
Some(plugin) => plugin.desc.name(),
|
||||||
@ -364,9 +438,13 @@ fn make_plugin_response(id: &str, name: &str, msg: &str) -> Message {
|
|||||||
msg_out
|
msg_out
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update_config(desc: &Desc) {
|
#[inline]
|
||||||
super::config::set_shared_items(desc.id(), &desc.config().shared);
|
fn change_manager() -> bool {
|
||||||
super::config::set_peer_items(desc.id(), &desc.config().peer);
|
#[cfg(debug_assertions)]
|
||||||
|
let change_manager = true;
|
||||||
|
#[cfg(not(debug_assertions))]
|
||||||
|
let change_manager = is_server();
|
||||||
|
change_manager
|
||||||
}
|
}
|
||||||
|
|
||||||
fn reload_ui(desc: &Desc, sync_to: Option<&str>) {
|
fn reload_ui(desc: &Desc, sync_to: Option<&str>) {
|
||||||
@ -417,22 +495,22 @@ fn update_ui_plugin_desc(desc: &Desc, sync_to: Option<&str>) {
|
|||||||
let event = serde_json::to_string(&m).unwrap_or("".to_owned());
|
let event = serde_json::to_string(&m).unwrap_or("".to_owned());
|
||||||
match sync_to {
|
match sync_to {
|
||||||
Some(channel) => {
|
Some(channel) => {
|
||||||
let _res = flutter::push_global_event(channel, serde_json::to_string(&m).unwrap());
|
let _res = flutter::push_global_event(channel, event.clone());
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
let _res = flutter::push_global_event(
|
let _res = flutter::push_global_event(flutter::APP_TYPE_MAIN, event.clone());
|
||||||
flutter::APP_TYPE_MAIN,
|
let _res =
|
||||||
serde_json::to_string(&m).unwrap(),
|
flutter::push_global_event(flutter::APP_TYPE_DESKTOP_REMOTE, event.clone());
|
||||||
);
|
let _res = flutter::push_global_event(flutter::APP_TYPE_CM, event.clone());
|
||||||
let _res = flutter::push_global_event(
|
|
||||||
flutter::APP_TYPE_DESKTOP_REMOTE,
|
|
||||||
serde_json::to_string(&m).unwrap(),
|
|
||||||
);
|
|
||||||
let _res = flutter::push_global_event(
|
|
||||||
flutter::APP_TYPE_CM,
|
|
||||||
serde_json::to_string(&m).unwrap(),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(super) fn get_desc_conf(id: &str) -> Option<super::desc::Config> {
|
||||||
|
PLUGIN_INFO
|
||||||
|
.read()
|
||||||
|
.unwrap()
|
||||||
|
.get(id)
|
||||||
|
.map(|info| info.desc.config().clone())
|
||||||
|
}
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
use super::*;
|
use super::*;
|
||||||
|
#[cfg(target_os = "macos")]
|
||||||
|
use crate::common::is_server;
|
||||||
#[cfg(target_os = "linux")]
|
#[cfg(target_os = "linux")]
|
||||||
use crate::common::IS_X11;
|
use crate::common::IS_X11;
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
@ -377,7 +379,6 @@ pub fn try_stop_record_cursor_pos() {
|
|||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
static ref QUEUE: Queue = Queue::main();
|
static ref QUEUE: Queue = Queue::main();
|
||||||
static ref IS_SERVER: bool = std::env::args().nth(1) == Some("--server".to_owned());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
@ -520,7 +521,7 @@ pub fn handle_mouse(evt: &MouseEvent, conn: i32) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
if !*IS_SERVER {
|
if !is_server() {
|
||||||
// having GUI, run main GUI thread, otherwise crash
|
// having GUI, run main GUI thread, otherwise crash
|
||||||
let evt = evt.clone();
|
let evt = evt.clone();
|
||||||
QUEUE.exec_async(move || handle_mouse_(&evt));
|
QUEUE.exec_async(move || handle_mouse_(&evt));
|
||||||
@ -902,7 +903,7 @@ pub async fn lock_screen() {
|
|||||||
|
|
||||||
pub fn handle_key(evt: &KeyEvent) {
|
pub fn handle_key(evt: &KeyEvent) {
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
if !*IS_SERVER {
|
if !is_server() {
|
||||||
// having GUI, run main GUI thread, otherwise crash
|
// having GUI, run main GUI thread, otherwise crash
|
||||||
let evt = evt.clone();
|
let evt = evt.clone();
|
||||||
QUEUE.exec_async(move || handle_key_(&evt));
|
QUEUE.exec_async(move || handle_key_(&evt));
|
||||||
@ -928,7 +929,7 @@ fn reset_input() {
|
|||||||
|
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
pub fn reset_input_ondisconn() {
|
pub fn reset_input_ondisconn() {
|
||||||
if !*IS_SERVER {
|
if !is_server() {
|
||||||
QUEUE.exec_async(reset_input);
|
QUEUE.exec_async(reset_input);
|
||||||
} else {
|
} else {
|
||||||
reset_input();
|
reset_input();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user