From 28a648e5ce929614502b29876ec8d4c52eeec9ca Mon Sep 17 00:00:00 2001 From: Logan Volkers Date: Fri, 20 Jul 2018 14:59:53 -0700 Subject: [PATCH 1/2] Allow plugins to be defined as functions and imported instead of globally namespaced --- src/index.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/index.js b/src/index.js index 1d4a22d028..20b003f2a8 100644 --- a/src/index.js +++ b/src/index.js @@ -58,6 +58,8 @@ module.exports = (() => { if (plugin) { plugin(editor, config.pluginsOpts[pluginId] || {}); + } else if (typeof pluginId === 'function') { + pluginId(editor, config.pluginsOpts[pluginId] || {}); } else { console.warn(`Plugin ${pluginId} not found`); } From 76776a31ebdc941f25b67ed9f257d602a0f19a8a Mon Sep 17 00:00:00 2001 From: Logan Volkers Date: Wed, 1 Aug 2018 16:42:17 -0700 Subject: [PATCH 2/2] Documentation and test cases for unnamed plugins --- docs/modules/Plugins.md | 73 ++++++++++++++++++++++++++++++++---- test/specs/grapesjs/index.js | 22 +++++++++++ 2 files changed, 88 insertions(+), 7 deletions(-) diff --git a/docs/modules/Plugins.md b/docs/modules/Plugins.md index ee5619e0d9..358e82d600 100644 --- a/docs/modules/Plugins.md +++ b/docs/modules/Plugins.md @@ -10,14 +10,46 @@ Creating plugins in GrapesJS is pretty straightforward and here you'll get how t ## Basic plugin -Generally, you would make plugins in separated files to keep thing cleaner, so you'll probably get a similar structure: +The most simple plugins are just functions that are run when the editor is being built. + +```js + function myPlugin(editor){ + editor.BlockManager.add('my-first-block', { + label: 'Simple block', + content: '
This is a simple block
', + }); + } + + var editor = grapesjs.init({ + container : '#gjs', + plugins: [myPlugin] + }); +``` + +This means that plugins can be moved to separate folders to keep thing cleaner or imported from NPM. + +```js + import myPlugin from './plugins/myPlugin' + import npmPackage from '@npm/package' + + var editor = grapesjs.init({ + container : '#gjs', + plugins: [myPlugin, npmPackage] + }); +``` + + + +## Named plugin + +If you're distributing your plugin globally, you may want to make a named plugin. To keep thing cleaner, so you'll probably get a similar structure: ``` /your/path/to/grapesjs.min.js /your/path/to/grapesjs-plugin.js ``` -The order is important as before loading your plugin, GrapesJS have to be loaded first. +**Important:** The order that you load files matters. GrapesJS has to be loaded before the plugin. This sets up the `grapejs` global variable. So, in your `grapesjs-plugin.js` file: @@ -58,10 +90,9 @@ Here is a complete generic example: ## Plugins with options -It's also possible to pass custom parameters to plugins in the way to make them more flexible. +It's also possible to pass custom parameters to plugins in to make them more flexible. -```html - ``` Inside you plugin you'll get those options via `options` argument @@ -83,10 +113,39 @@ export default grapesjs.plugins.add('my-plugin-name', (editor, options) => { }) ``` +This also works with plugins that aren't named. + +```js + import myPlugin from '../plugin' + + var editor = grapesjs.init({ + container : '#gjs', + plugins: [myPlugin], + pluginsOpts: { + [myPlugin]: { + customField: 'customValue' + } + } + }); +``` + + +## Named Plugins vs Non-Named Plugins +When you use a named plugin, then that name must be unique across all other plugins. +```js +grapesjs.plugins.add('my-plugin-name', fn); +``` +In this example, the plugin name is `my-plugin-name` and can't be used by other plugins. To avoid namespace restrictions use basic plugins that are purely functional. ## Boilerplate -If you want to start with a production-ready boilerplate, you might want to try [grapesjs-plugin-boilerplate](https://github.com/artf/grapesjs-plugin-boilerplate) which you can clone and start developing a plugin immediately. For more informations check the repository +If you want to start with a production-ready boilerplate for a named plugin, you might want to try [grapesjs-plugin-boilerplate](https://github.com/artf/grapesjs-plugin-boilerplate) which you can clone and start developing a named plugin immediately. For more informations check the repository + + +## Popular Plugins + + - https://github.com/artf/grapesjs-preset-webpage + - https://github.com/artf/grapesjs-preset-newsletter \ No newline at end of file diff --git a/test/specs/grapesjs/index.js b/test/specs/grapesjs/index.js index fe84599410..52a8492962 100644 --- a/test/specs/grapesjs/index.js +++ b/test/specs/grapesjs/index.js @@ -288,6 +288,28 @@ describe('GrapesJS', () => { expect(editor.customValue).toEqual('TEST'); }); + test('Execute inline plugins with custom options', () => { + const inlinePlugin = (edt, opts) => { + var opts = opts || {}; + edt.customValue = opts.cVal || ''; + }; + config.plugins = [inlinePlugin]; + config.pluginsOpts = {}; + config.pluginsOpts[inlinePlugin] = { cVal: 'TEST' }; + var editor = obj.init(config); + expect(editor.customValue).toEqual('TEST'); + }); + + test('Execute inline plugins without any options', () => { + const inlinePlugin = edt => { + edt.customValue = 'TEST'; + }; + config.plugins = [inlinePlugin]; + config.pluginsOpts = {}; + var editor = obj.init(config); + expect(editor.customValue).toEqual('TEST'); + }); + test('Execute custom command', () => { var editor = obj.init(config); editor.testVal = '';