2022-02-28 14:38:53 +00:00
|
|
|
library flowy_plugin;
|
|
|
|
|
2024-06-03 06:27:28 +00:00
|
|
|
import 'package:appflowy_backend/protobuf/flowy-user/user_profile.pb.dart';
|
2024-02-25 15:46:13 +00:00
|
|
|
import 'package:flutter/widgets.dart';
|
|
|
|
|
2023-08-14 20:34:01 +00:00
|
|
|
import 'package:appflowy/generated/flowy_svgs.g.dart';
|
2023-02-26 08:27:17 +00:00
|
|
|
import 'package:appflowy/startup/plugin/plugin.dart';
|
|
|
|
import 'package:appflowy/startup/startup.dart';
|
|
|
|
import 'package:appflowy/workspace/presentation/home/home_stack.dart';
|
2023-12-30 23:29:40 +00:00
|
|
|
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
|
2022-02-28 14:38:53 +00:00
|
|
|
|
|
|
|
export "./src/sandbox.dart";
|
|
|
|
|
2022-08-18 11:32:08 +00:00
|
|
|
enum PluginType {
|
2024-04-29 05:44:42 +00:00
|
|
|
document,
|
2022-03-02 03:38:22 +00:00
|
|
|
blank,
|
|
|
|
trash,
|
2022-03-04 00:22:49 +00:00
|
|
|
grid,
|
2022-06-11 07:51:53 +00:00
|
|
|
board,
|
2023-01-08 12:51:19 +00:00
|
|
|
calendar,
|
2024-04-29 05:44:42 +00:00
|
|
|
databaseDocument,
|
2024-06-03 06:27:28 +00:00
|
|
|
chat,
|
2022-03-02 03:38:22 +00:00
|
|
|
}
|
2022-02-28 14:38:53 +00:00
|
|
|
|
2022-03-01 08:05:45 +00:00
|
|
|
typedef PluginId = String;
|
|
|
|
|
2024-04-29 05:44:42 +00:00
|
|
|
abstract class Plugin {
|
2022-03-02 03:38:22 +00:00
|
|
|
PluginId get id;
|
2022-03-01 08:05:45 +00:00
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
PluginWidgetBuilder get widgetBuilder;
|
2022-02-28 14:38:53 +00:00
|
|
|
|
2022-09-22 05:08:48 +00:00
|
|
|
PluginNotifier? get notifier => null;
|
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
PluginType get pluginType;
|
2022-02-28 14:38:53 +00:00
|
|
|
|
2024-02-25 15:46:13 +00:00
|
|
|
void init() {}
|
|
|
|
|
2022-09-22 05:08:48 +00:00
|
|
|
void dispose() {
|
|
|
|
notifier?.dispose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-26 08:59:58 +00:00
|
|
|
abstract class PluginNotifier<T> {
|
2022-09-22 05:08:48 +00:00
|
|
|
/// Notify if the plugin get deleted
|
2022-09-26 08:59:58 +00:00
|
|
|
ValueNotifier<T> get isDeleted;
|
2022-09-22 05:08:48 +00:00
|
|
|
|
2022-03-02 03:38:22 +00:00
|
|
|
void dispose() {}
|
2022-02-28 14:38:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
abstract class PluginBuilder {
|
|
|
|
Plugin build(dynamic data);
|
|
|
|
|
2022-03-01 08:05:45 +00:00
|
|
|
String get menuName;
|
2022-02-28 14:38:53 +00:00
|
|
|
|
2023-08-14 20:34:01 +00:00
|
|
|
FlowySvgData get icon;
|
2022-11-15 03:45:23 +00:00
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
/// The type of this [Plugin]. Each [Plugin] should have a unique [PluginType]
|
2022-02-28 14:38:53 +00:00
|
|
|
PluginType get pluginType;
|
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
/// The layoutType is used in the backend to determine the layout of the view.
|
2024-04-18 12:52:58 +00:00
|
|
|
/// Currently, AppFlowy supports 4 layout types: Document, Grid, Board, Calendar.
|
2024-06-03 06:27:28 +00:00
|
|
|
ViewLayoutPB? get layoutType;
|
2022-02-28 14:38:53 +00:00
|
|
|
}
|
|
|
|
|
2022-03-01 03:22:39 +00:00
|
|
|
abstract class PluginConfig {
|
2022-03-02 03:38:22 +00:00
|
|
|
// Return false will disable the user to create it. For example, a trash plugin shouldn't be created by the user,
|
2022-03-01 03:22:39 +00:00
|
|
|
bool get creatable => true;
|
|
|
|
}
|
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
abstract class PluginWidgetBuilder with NavigationItem {
|
2022-02-28 14:38:53 +00:00
|
|
|
List<NavigationItem> get navigationItems;
|
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
EdgeInsets get contentPadding =>
|
|
|
|
const EdgeInsets.symmetric(horizontal: 40, vertical: 28);
|
|
|
|
|
2024-06-03 06:27:28 +00:00
|
|
|
Widget buildWidget({
|
|
|
|
required PluginContext context,
|
|
|
|
required bool shrinkWrap,
|
|
|
|
});
|
2022-09-22 05:08:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class PluginContext {
|
2024-06-03 06:27:28 +00:00
|
|
|
PluginContext({
|
|
|
|
this.userProfile,
|
|
|
|
this.onDeleted,
|
|
|
|
});
|
2024-01-25 15:37:36 +00:00
|
|
|
|
2022-09-22 05:08:48 +00:00
|
|
|
// calls when widget of the plugin get deleted
|
2024-06-03 06:27:28 +00:00
|
|
|
final Function(ViewPB, int?)? onDeleted;
|
|
|
|
final UserProfilePB? userProfile;
|
2022-02-28 14:38:53 +00:00
|
|
|
}
|
|
|
|
|
2022-03-01 03:22:39 +00:00
|
|
|
void registerPlugin({required PluginBuilder builder, PluginConfig? config}) {
|
2022-08-09 02:35:27 +00:00
|
|
|
getIt<PluginSandbox>()
|
|
|
|
.registerPlugin(builder.pluginType, builder, config: config);
|
2022-02-28 14:38:53 +00:00
|
|
|
}
|
|
|
|
|
2023-06-01 12:23:27 +00:00
|
|
|
/// Make the correct plugin from the [pluginType] and [data]. If the plugin
|
|
|
|
/// is not registered, it will return a blank plugin.
|
2022-03-01 02:25:21 +00:00
|
|
|
Plugin makePlugin({required PluginType pluginType, dynamic data}) {
|
2022-02-28 14:38:53 +00:00
|
|
|
final plugin = getIt<PluginSandbox>().buildPlugin(pluginType, data);
|
|
|
|
return plugin;
|
|
|
|
}
|
|
|
|
|
2022-03-01 03:22:39 +00:00
|
|
|
List<PluginBuilder> pluginBuilders() {
|
|
|
|
final pluginBuilders = getIt<PluginSandbox>().builders;
|
|
|
|
final pluginConfigs = getIt<PluginSandbox>().pluginConfigs;
|
|
|
|
return pluginBuilders.where(
|
|
|
|
(builder) {
|
|
|
|
final config = pluginConfigs[builder.pluginType]?.creatable;
|
|
|
|
return config ?? true;
|
|
|
|
},
|
|
|
|
).toList();
|
|
|
|
}
|
|
|
|
|
2022-02-28 14:38:53 +00:00
|
|
|
enum FlowyPluginException {
|
|
|
|
invalidData,
|
|
|
|
}
|