library flowy_plugin; import 'package:appflowy/startup/plugin/plugin.dart'; import 'package:appflowy/startup/startup.dart'; import 'package:appflowy/workspace/presentation/home/home_stack.dart'; import 'package:appflowy_backend/protobuf/flowy-folder2/view.pb.dart'; import 'package:flutter/widgets.dart'; export "./src/sandbox.dart"; enum PluginType { editor, blank, trash, grid, board, calendar, } typedef PluginId = String; abstract class Plugin { PluginId get id; PluginWidgetBuilder get widgetBuilder; PluginNotifier? get notifier => null; PluginType get pluginType; void dispose() { notifier?.dispose(); } } abstract class PluginNotifier { /// Notify if the plugin get deleted ValueNotifier get isDeleted; /// Notify if the [PluginWidgetBuilder]'s content was changed ValueNotifier get isDisplayChanged; void dispose() {} } abstract class PluginBuilder { Plugin build(dynamic data); String get menuName; String get menuIcon; /// The type of this [Plugin]. Each [Plugin] should have a unique [PluginType] PluginType get pluginType; /// The layoutType is used in the backend to determine the layout of the view. /// Currrently, AppFlowy supports 4 layout types: Document, Grid, Board, Calendar. ViewLayoutPB? get layoutType => ViewLayoutPB.Document; } abstract class PluginConfig { // Return false will disable the user to create it. For example, a trash plugin shouldn't be created by the user, bool get creatable => true; } abstract class PluginWidgetBuilder with NavigationItem { List get navigationItems; EdgeInsets get contentPadding => const EdgeInsets.symmetric(horizontal: 40, vertical: 28); Widget buildWidget(PluginContext context); } class PluginContext { // calls when widget of the plugin get deleted final Function(ViewPB, int?) onDeleted; PluginContext({required this.onDeleted}); } void registerPlugin({required PluginBuilder builder, PluginConfig? config}) { getIt() .registerPlugin(builder.pluginType, builder, config: config); } /// Make the correct plugin from the [pluginType] and [data]. If the plugin /// is not registered, it will return a blank plugin. Plugin makePlugin({required PluginType pluginType, dynamic data}) { final plugin = getIt().buildPlugin(pluginType, data); return plugin; } List pluginBuilders() { final pluginBuilders = getIt().builders; final pluginConfigs = getIt().pluginConfigs; return pluginBuilders.where( (builder) { final config = pluginConfigs[builder.pluginType]?.creatable; return config ?? true; }, ).toList(); } enum FlowyPluginException { invalidData, }