:memo::cocktail: A configurable rich text editor based on Draft.js, built for Wagtail.
data:image/s3,"s3://crabby-images/38be4/38be4d7b956a0e8b2b78ccbf32e4543a3369b2b8" alt="Screenshot of Draftail"
Itโs developed alongside our Python Draft.js exporter, for integration into Wagtail. Check out WagtailDraftail, and the editorโs online demo!
Features
This project adheres to Semantic Versioning, and measures performance and code coverage.
Draftail aims for a mouse-free, keyboard-centric experience. Most formatting can be done by using common keyboard shortcuts, inspired by Google Docs and Markdown.
Here are important features worth highlighting:
- Support for keyboard shortcuts. Lots of them!
- Paste from Word. Or any other editor.
- Autolists โ start a line with
-
, *
, 1.
to create a list item. - Shortcuts for heading levels
##
, code blocks ```
, and more. - Undo / redo โ until the end of times.
- Common text types: headings, paragraphs, quotes, lists.
- Common text styles: Bold, italic, and many more.
- API to build custom controls for links, images, and more.
Using Draftail
Draftail is meant to be used in scenarios where not all formatting should be available, and where custom formatting can be necessary. Available formats, built-in and custom, can be specificed declaratively for each editor instance.
Built-in formats
- Block types: H1, H2, H3, H4, H5, H6, Blockquote, Code, UL, OL, P
- Inline styles: Bold, Italic, Underline, Code, Strikethrough, Mark, Keyboard, Superscript, Subscript
- Entities (things with data): Images, Embeds, Links, Documents
- And HR, BR
Custom formats
Your mileage may vary! There is good support for custom block-level and inline formatting. Custom entities or decorators require knowledge of the Draft.js API, which is very low-level.
Getting started
First, grab the package from npm:
npm install --save draft-js@^0.10.4 react react-dom prop-types
npm install --save draftail
Import the styles for Draft.js, and the editor:
@import 'draft-js/dist/Draft.css';
@import 'draftail/dist/draftail.css';
Then, import the editor and use it in your code. Here is a simple example:
import React from 'react';
import ReactDOM from 'react-dom';
import DraftailEditor, { BLOCK_TYPE, INLINE_STYLE } from '../lib';
const initial = JSON.parse(sessionStorage.getItem('draftail:content'));
const onSave = content => {
console.log('saving', content);
sessionStorage.setItem('draftail:content', JSON.stringify(content));
};
const editor = (
<DraftailEditor
rawContentState={initial || null}
onSave={onSave}
blockTypes={[
{ type: BLOCK_TYPE.HEADER_THREE, label: 'H3' },
{ type: BLOCK_TYPE.UNORDERED_LIST_ITEM, label: 'UL' },
]}
inlineStyles={[
{ type: INLINE_STYLE.BOLD, label: 'B' },
{ type: INLINE_STYLE.ITALIC, label: 'I' },
]}
/>
);
ReactDOM.render(editor, document.querySelector('[data-mount]'));
Finally, be sure to check out the required polyfills.
Options and configuration
To change the behavior of the editor, pass props to DraftailEditor
. Here are the available props, and their default values:
rawContentState: null,
placeholder: null,
onSave: () => {},
enableHorizontalRule: false,
enableLineBreak: false,
showUndoRedoControls: false,
stripPastedStyles: true,
spellCheck: false,
blockTypes: [],
inlineStyles: [],
entityTypes: [],
decorators: [],
maxListNesting: 1,
stateSaveInterval: 250,
Formatting options
Draftail, like Draft.js, distinguishes between 3 content formats:
- Blocks, that provide structure to the content. Blocks do not overlap โ no content can be both a paragraph and a title.
- Inline styles, providing inline formatting for text. Styles can overlap: a piece of text can be both bold and italic.
- Entities, annotating content with metadata to represent rich content beyond text. Entities can be inline (eg. a link applied on a portion of text), or block-based (eg. an embedded video).
Configuring available formats
By default, the editor provides the least amount of rich text features. Formats have to be explicitly enabled by the developer, so they have as much control over what rich content is available as possible.
To use a given format, add it to the corresponding list, following the options detailed in the next sections.
blockTypes: [],
inlineStyles: [],
entityTypes: [],
Blocks
type: PropTypes.string.isRequired,
label: PropTypes.string,
description: PropTypes.string,
icon: iconPropType,
element: PropTypes.string,
className: PropTypes.string,
Inline styles
type: PropTypes.string.isRequired,
label: PropTypes.string,
description: PropTypes.string,
icon: iconPropType,
style: PropTypes.Object,
Entities
type: PropTypes.string.isRequired,
label: PropTypes.string,
description: PropTypes.string,
icon: iconPropType,
source: PropTypes.func.isRequired,
decorator: PropTypes.func,
block: PropTypes.func,
Decorators
strategy: PropTypes.func,
component: PropTypes.func,
Custom formats
Draftail is meant to provide a consistent editing experience regardless of what formats (blocks, inline styles, entities) are available. It should be simple for developers to enable/disable a certain format, or to create new ones.
Here are quick questions to help you determine which formatting to use, depending on the use case:
In order to... | Use |
---|
Indicate the structure of the content | Blocks |
Enter additional data/metadata | Entities |
Format a portion of a line | Inline styles |
Custom blocks
Simple blocks are very easy to create. Add a new block type to blockTypes
, specifying which element
and className
to display the block as.
Here is an example, creating a "Tiny text" block:
blockTypes={[
{
type: 'tiny-text',
label: 'Tiny',
element: 'div',
className: 'u-tinytext',
},
]}
More advanced blocks, requiring custom React components, aren't supported at the moment. If you need this, feel free to create an issue.
Custom inline styles
Custom inline styles only require a style
prop, defining which CSS properties to apply when the format is active.
Here is a basic example:
inlineStyles={[
{
label: 'Redacted',
type: 'REDACTED',
style: {
backgroundColor: 'currentcolor',
},
},
]}
It is also possible to override the styling of predefined inline styles:
inlineStyles={[
{
label: 'Bold',
type: INLINE_STYLE.BOLD,
style: {
fontWeight: 'bold',
textShadow: '1px 1px 1px black',
},
},
]}
Custom entity types
Creating custom entity types is a bit more involved because entities aren't simply on/off: they often need additional data (thus a UI to enter this data), and can be edited.
Apart from the usual label/type/icon options, entities need:
- A
source
, a React component that will be used to create/edit the entity from a specific data source (could be an API, or a form inside of a modal). - A
decorator
, a React component to display the entity within the editor area.
Sources
For now, please refer to the examples available with the project.
Decorators
Decorators receive the current textual content as children
, as well as the entity key and content state. They can then render the entity based on its data:
const Link = ({ entityKey, contentState, children }) => {
const { url } = contentState.getEntity(entityKey).getData();
return (
<span title={url} className="Link">
{children}
</span>
);
};
Custom text decorators
Custom decorators follow the Draft.js CompositeDecorator API.
UI and styling
Without custom controls, the editor has a very simple UI and its styles are relatively straightforward. To make sure everything works, use a CSS reset, like Normalize.css.
To tweak the editor UI, Draftail uses old-fashioned stable, namespaced class names that you are free to add more styles to. For example, the toolbar uses .Draftail-Toolbar
.
Draftail also makes its Sass stylesheets available for extension:
$editor-z-index: 100;
@import 'draftail/lib/index';
@import 'draftail/lib/api/constants';
List nesting levels
Draft.js only provides default styles for list nesting up to five levels (depth 0 to 4). If you want to allow more nesting, you will need to add the list styles.
Draftail provides a helper Sass mixin which adds OL counters and indentation, and can be used like:
@import 'draftail/lib/api/nested-list-item';
@for $depth from 5 through 6 {
@include nested-list-item($depth);
}
Browser support and polyfills
Supported browser / device versions:
Browser | Device/OS | Version |
---|
Mobile Safari | iOS Phone | latest |
Mobile Safari | iOS Tablet | latest |
Chrome | Android | latest |
IE | Desktop | 11 |
Chrome | Desktop | latest |
MS Edge | Windows | latest |
Firefox | Desktop | latest |
Safari | OSX | latest |
Polyfills
Draft.js and Draftail build upon ES6 language features. If targeting browsers that do not support them, have a look at:
The Draftail demo site lists minimum polyfills for IE11 support: examples/utils/polyfills.js
.
Contributing
See anything you like in here? Anything missing? We welcome all support, whether on bug reports, feature requests, code, design, reviews, tests, documentation, and more. Please have a look at our contribution guidelines.
If you just want to set up the project on your own computer, the contribution guidelines also contain all of the setup commands.
Credits
Draftail is made possible by the work of Springload, a New Zealand digital agency, and core contributors to the Wagtail CMS. The beautiful demo site is the work of @thibaudcolas.
View the full list of contributors. MIT licensed. Website content available as CC0.