pagecut
v1.0.0-beta.74
Published
Extensible web content editor
Downloads
17
Maintainers
Readme
pagecut -- Extensible web content editor
Warning: this is a work in progress. Documentation is out of date.
It is mostly meant to be used by pageboard, with a lot of elements already defined here.
An easy to setup and easy to extend html content editor built upon ProseMirror.
Objects
Viewer
renders blocks to DOM using elements and modifiers.Editor
parse DOM into blocks using resolvers, render blocks into DOM (edit version), parse that DOM into an internal editor object model, which is in turn used by prosemirror to render that internal model into a DOM view.Resolvers
Return synchronously or asynchronously a block from a url or from a DOM node.Modifiers
Change a block or its rendered attributes on the fly.Elements
Blocks are instances of elements.
An element comes in two parts:- its definition (name, group, contents, and json schema properties)
- its edit and view methods, with signatures (document, block).
Modules
A simple extension system for defining resolvers and elements. A module can be a simple object in which case it is added as an element having the name of the module, or it can be a class constructor function, instantiated by the viewer and accessible underpagecut.modules[name]
.Blocks
The core data structure for holding elements instances, and persisting content. A block has an type, content, data, and an optional id.
A block can be rendered to DOM using pagecut.render(block, forEdition)
.
An editor instance can be rendered to DOM using pagecut.get()
.
Basic setup
Bunbled files are available as
- dist/editor.js (already containing the viewer)
- dist/viewer.js (for view-only purpose)
- dist/editor.css (for editor)
A basic editor setup without anything interesting:
var pagecut = new Pagecut.Editor({
place: '#editable',
content: document.querySelector('.some.selector')
});
Yet such a setup isn't really useful without modules.
Modules
A module constructor receives a viewer instance as argument, so it can set up:
- elements
- resolvers
- modifiers
Example:
module.exports = function(main) {
this.main = main;
main.elements.video = VideoElement;
main.resolvers.youtube = YoutubeResolver;
main.modifiers.oddity = MyOwnModifier;
};
and is typically exported using browserify:
browserify --standalone Pagecut.modules.video modules/video.js
The viewer instance initializes the globally installed modules automatically.
The order of modules, and in particular, the order of resolvers or modifiers, should not change the result - be sure to define independent functions that can be run in any order.
If a module is an object, it is directly added as an element, with
element.name
set to be the key under which the object was set on modules.
Resolvers
Elements convert blocks to DOM, and resolvers are there to do the reverse.
A resolver is a function(main, obj, cb)
,
- that can return a block immediately.
- that can return a block asynchronously using
cb(err, block)
; - the synchronous block is automatically replaced by the asynchronous block
obj can have obj.url or obj.node, depending on wether a url was pasted or a node is being processed.
Elements
An element is a simple object with properties and methods, and must be added
to the elements
array, mapping types to elements.
Mandatory property:
- name
the element type name
Properties for the editor:
- group (optional)
the group as defined by the prosemirror editor - contents
an object matching contents names to an object having a 'spec' property being a prosemirror content expression.
Or,contents
can be a string, meaning content is not labelled. - inplace
A boolean indicating the block is not stored, implying it is entirely defined by its DOM representation. - inline
A boolean indicating the block has only one content (what's inside its tag), and that this content is stored in place, not in the block. The attributes are still kept in the block.
Properties for content management:
- properties
a json schema object defining the format of block's data object, but can actually only hold strings (empty string being the default value). - required or other json-schema keywords
anything being optional here
Mandatory viewing method:
- view(document, block)
renders a block to DOM
Methods for editing:
- edit(document, block)
optional, defaults to view().
renders a block to editable DOM.
The edit
method must return a DOM with block-content
attributes placed on
DOM Nodes that have editable content, and does not need to actually merge the
block content into it (it's done automatically).
The view
method, on the other hand, can merge the content the way it wants,
or add block-content
attributes and let the merge be done automatically.
It can also place a block-handle
boolean attribute on a DOM Node to facilitate
selection and drag and drop of the block DOM representation.
Blocks
A simple object with:
- type
- data object
- content object (mapping names with content)
and optionally, some extensions:
- id (used by id module)
- focused (used by focus modifier)
- foreign: the block is not editable and the element defining the block exports
a
foreign(dom, block)
function that can change the dom content.
Application data should be stored in the data object, but the block object itself can be used to store runtime variables like these.
The content object holds html content, which itself can be in two states:
- serialized, as html text, with unresolved sub-blocks in it
- parsed as DOM with resolved sub-blocks in it
When serializing to blocks, it is essential to also get all the blocks that where (un)resolved by the resolvers, or else it's like getting a tree without its apples. See the "id module" below.
Modifiers
After an element renders a block, modifiers can act upon the returned DOM node. A modifier is a function(main, block, dom) {} that returns nothing.
Typical modifiers:
- add a block-id attribute if block.id is set
- add a block-focused attribute if block.focused is set
Pagecut.Editor
Pagecut main editor instance conveniently exposes underlying prosemirror editor modules: Menu, Commands, State, Transform, Model, Pos (from dompos), keymap.
Pagecut.Editor options:
- update(main, transaction): called upon each transaction
if it returns true, the transaction is not applied to the editor view.
This gives a way to override underlying editor dispatchTransaction event. - change(main, block): called when a block has changed
the ancestor block, if any, in which the current action is applied.
Pagecut.Editor.defaults holds some default options values:
- marks, nodes: the schema specifications
- plugins: array of plugins for ProseMirror
- menu: function(coed, items) { return items.fullMenu; }
Pagecut.Editor methods:
- set(dom) - set the dom content of the editor
- get(edition) - get the dom content of the editor
Pagecut.Viewer
An instance of viewer is created by default by an instance of Pagecut.Editor.
Its purpose it to keep track of elements and resolvers, and mainly to render
blocks to DOM using render
method.
A separate instance can be created using new Pagecut.Viewer(opts)
where opts:
- document: a DOM Document (a new one is created if none is given)
- render(block, edition) returns a DOM node
Calls the element edit or view function, and modifiers. Merges content. Not recursive.
A default fragment
type is available to be able to render a fragment of html:
var domWrapper = pagecut.render({
type: 'fragment',
content: {fragment: 'some html <p>string</p>'}
});
// domWrapper.innerHTML == 'some html <p>string</p>'
Prosemirror modules
The Prosemirror modules that are used by Editor are accessible through Pagecut variable.
The id module
The id module provides:
IdResolver for edition that maps block-id attributes values to blocks stored in a shared cache,
IdModifier for edition that adds block-id attributes when block.id is defined,
id.to(store?) returns a block fragment of the editor root content, and optionally populates store with the blocks (by id) that have been referenced. id.to also calls elements's to() function if present, and do not add blocks to the root.children list when they have a true
orphan
property.id.from(block or html, store, resolver?) renders the block, searches all descendents with a
block-id
attribute in the store, and if it doesn't find a matching block, optionally calls the resolver(id) function which can return a promise, then replaces each block with its rendered DOM. The async resolver allows one to fetch remote data during initial rendering of the view. It's similar (but different in the details and applications) to the editor's resolvers functions.
It's up to the custom resolver to store fetched blocks in the id module cache.
Returns a promise that resolves to a DOM node.