2023-04-20 18:10:06 +08:00
|
|
|
import 'package:flutter/material.dart';
|
2023-04-20 20:57:47 +08:00
|
|
|
import 'package:flutter/services.dart';
|
|
|
|
import 'package:flutter_hbb/models/model.dart';
|
2023-04-20 18:10:06 +08:00
|
|
|
import 'package:provider/provider.dart';
|
2023-04-21 21:40:34 +08:00
|
|
|
// to-do: do not depend on desktop
|
2023-04-20 20:57:47 +08:00
|
|
|
import 'package:flutter_hbb/desktop/widgets/remote_toolbar.dart';
|
|
|
|
import 'package:flutter_hbb/models/platform_model.dart';
|
2023-04-20 18:10:06 +08:00
|
|
|
|
2023-04-20 22:53:43 +08:00
|
|
|
import './desc.dart';
|
|
|
|
import './model.dart';
|
|
|
|
import './common.dart';
|
2023-04-20 18:10:06 +08:00
|
|
|
|
2023-04-20 22:53:43 +08:00
|
|
|
class LocationItem extends StatelessWidget {
|
2023-04-20 18:10:06 +08:00
|
|
|
final String peerId;
|
2023-04-20 20:57:47 +08:00
|
|
|
final FFI ffi;
|
|
|
|
final String location;
|
2023-04-20 18:10:06 +08:00
|
|
|
final LocationModel locationModel;
|
|
|
|
|
2023-04-20 22:53:43 +08:00
|
|
|
LocationItem({
|
2023-04-20 18:10:06 +08:00
|
|
|
Key? key,
|
|
|
|
required this.peerId,
|
2023-04-20 20:57:47 +08:00
|
|
|
required this.ffi,
|
|
|
|
required this.location,
|
2023-04-20 18:10:06 +08:00
|
|
|
required this.locationModel,
|
|
|
|
}) : super(key: key);
|
|
|
|
|
2023-04-20 20:57:47 +08:00
|
|
|
bool get isEmpty => locationModel.isEmpty;
|
|
|
|
|
2023-04-20 22:53:43 +08:00
|
|
|
static LocationItem createLocationItem(
|
|
|
|
String peerId, FFI ffi, String location) {
|
|
|
|
final model = addLocation(location);
|
|
|
|
return LocationItem(
|
|
|
|
peerId: peerId,
|
|
|
|
ffi: ffi,
|
|
|
|
location: location,
|
|
|
|
locationModel: model,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-20 18:10:06 +08:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return ChangeNotifierProvider.value(
|
|
|
|
value: locationModel,
|
|
|
|
child: Consumer<LocationModel>(builder: (context, model, child) {
|
|
|
|
return Column(
|
2023-04-20 22:53:43 +08:00
|
|
|
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;
|
2023-04-21 21:40:34 +08:00
|
|
|
final KvModel kvModel;
|
2023-04-20 22:53:43 +08:00
|
|
|
|
|
|
|
PluginItem({
|
|
|
|
Key? key,
|
|
|
|
required this.pluginId,
|
|
|
|
required this.peerId,
|
|
|
|
required this.ffi,
|
|
|
|
required this.location,
|
|
|
|
required this.pluginModel,
|
2023-04-21 21:40:34 +08:00
|
|
|
}) : kvModel = addKvModel(location, pluginId, peerId),
|
|
|
|
super(key: key);
|
2023-04-20 22:53:43 +08:00
|
|
|
|
|
|
|
bool get isEmpty => pluginModel.isEmpty;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2023-04-21 21:40:34 +08:00
|
|
|
return MultiProvider(
|
|
|
|
providers: [
|
|
|
|
ChangeNotifierProvider.value(value: pluginModel),
|
|
|
|
ChangeNotifierProvider.value(value: kvModel),
|
|
|
|
],
|
|
|
|
child: Consumer2<PluginModel, KvModel>(
|
|
|
|
builder: (context, pluginModel, kvModel, child) {
|
|
|
|
return Column(
|
|
|
|
children: pluginModel.uiList.map((ui) => _buildItem(ui)).toList(),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
2023-04-20 18:10:06 +08:00
|
|
|
);
|
|
|
|
}
|
2023-04-20 20:57:47 +08:00
|
|
|
|
|
|
|
Widget _buildItem(UiType ui) {
|
2023-04-21 21:40:34 +08:00
|
|
|
late Widget child;
|
2023-04-20 20:57:47 +08:00
|
|
|
switch (ui.runtimeType) {
|
|
|
|
case UiButton:
|
2023-04-21 21:40:34 +08:00
|
|
|
child = _buildMenuButton(ui as UiButton);
|
|
|
|
break;
|
2023-04-20 20:57:47 +08:00
|
|
|
case UiCheckbox:
|
2023-04-21 21:40:34 +08:00
|
|
|
child = _buildCheckboxMenuButton(ui as UiCheckbox);
|
|
|
|
break;
|
2023-04-20 20:57:47 +08:00
|
|
|
default:
|
2023-04-21 21:40:34 +08:00
|
|
|
child = Container();
|
2023-04-20 20:57:47 +08:00
|
|
|
}
|
2023-04-21 21:40:34 +08:00
|
|
|
// to-do: add plugin icon and tooltip
|
|
|
|
return child;
|
2023-04-20 20:57:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildMenuButton(UiButton ui) {
|
|
|
|
return MenuButton(
|
2023-04-21 21:40:34 +08:00
|
|
|
onPressed: () => bind.pluginEvent(
|
|
|
|
id: pluginId,
|
|
|
|
peer: peerId,
|
|
|
|
event: _makeEvent(ui.key),
|
|
|
|
),
|
2023-04-20 22:53:43 +08:00
|
|
|
trailingIcon: Icon(
|
|
|
|
IconData(int.parse(ui.icon, radix: 16), fontFamily: 'MaterialIcons')),
|
|
|
|
// to-do: RustDesk translate or plugin translate ?
|
2023-04-20 20:57:47 +08:00
|
|
|
child: Text(ui.text),
|
|
|
|
ffi: ffi,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildCheckboxMenuButton(UiCheckbox ui) {
|
2023-04-21 21:40:34 +08:00
|
|
|
var v = kvModel.get(ui.key);
|
|
|
|
if (v == null) {
|
|
|
|
if (peerId.isEmpty) {
|
|
|
|
v = bind.pluginGetLocalOption(id: pluginId, key: ui.key);
|
|
|
|
} else {
|
|
|
|
v = bind.pluginGetSessionOption(
|
|
|
|
id: pluginId, peer: peerId, key: ui.key);
|
|
|
|
}
|
|
|
|
}
|
2023-04-20 22:53:43 +08:00
|
|
|
if (v == null) {
|
|
|
|
// session or plugin not found
|
|
|
|
return Container();
|
|
|
|
}
|
2023-04-20 20:57:47 +08:00
|
|
|
return CkbMenuButton(
|
|
|
|
value: ConfigItem.isTrue(v),
|
|
|
|
onChanged: (v) {
|
|
|
|
if (v != null) {
|
2023-04-21 21:40:34 +08:00
|
|
|
bind.pluginEvent(
|
|
|
|
id: pluginId,
|
|
|
|
peer: peerId,
|
|
|
|
event: _makeEvent(ui.key, v: v),
|
|
|
|
);
|
2023-04-20 20:57:47 +08:00
|
|
|
}
|
|
|
|
},
|
2023-04-21 21:40:34 +08:00
|
|
|
// to-do: RustDesk translate or plugin translate ?
|
2023-04-20 20:57:47 +08:00
|
|
|
child: Text(ui.text),
|
|
|
|
ffi: ffi,
|
|
|
|
);
|
|
|
|
}
|
2023-04-20 18:10:06 +08:00
|
|
|
}
|
2023-04-20 22:53:43 +08:00
|
|
|
|
|
|
|
void handleReloading(Map<String, dynamic> evt, String peer) {
|
|
|
|
if (evt['id'] == null || evt['location'] == null) {
|
|
|
|
return;
|
|
|
|
}
|
2023-04-21 21:40:34 +08:00
|
|
|
addLocationUi(evt['location']!, evt['id']!, UiType.fromJson(evt));
|
|
|
|
}
|
|
|
|
|
|
|
|
void handleOption(Map<String, dynamic> evt, String peer) {
|
|
|
|
updateOption(
|
|
|
|
evt['location'], evt['id'], evt['peer'] ?? '', evt['key'], evt['value']);
|
2023-04-20 22:53:43 +08:00
|
|
|
}
|