writer
writer copied to clipboard
A small, modern WYSIWYG editor for inline formats – by the team of Kirby CMS
Kirby Writer
A small, modern WYSIWYG editor for inline formats – by the team of Kirby CMS
About this editor
This library is still experimental and looking for help to get finished. DO NOT USE IN PRODUCTION!
Yes, you read that correctly. We are trying to build our own contenteditable wrapper in Javascript. It seems like a stupid idea. Maybe it is. But we are drastically limiting the scope of this project:
Inline formats only
Modern content editing tools have moved away from the idea to solve everything within a single contenteditable element. Notion's editor, something like Gutenberg or our own Kirby Editor turn to a new block editor model, in which each block element (heading, images, videos, etc.) are handled seperately and new block types can be added to extend the editor feature set. This has a lot of benefits. It massively reduces the complexity of what each block component has to solve and also leads to a more controllable content structure that can be exported as something like JSON instead of HTML.
For such block editors, a fully fledged WYSIWYG editor is way too much. What they need is a WYSIWYG editor that only handles inline elements (strong, em, code, sub, sup, etc.) in a reliable and clean way. This is exactly what this editor implementation does. By completely ignoring any kind of block elements, we can make this library a lot smaller and simpler and focus on perfect structured inline content, selection and event APIs that help to build great block types.
Modern browsers only
This library also does not support any legacy browsers. We don't have to care about IE and some outdated selection APIs. It's fully written in ES6 and is aimed at browsers that support ES6 modules by default. https://caniuse.com/#feat=es6-module
It's pretty small
14 kb compressed / 39 kb uncompressed – we are really proud of its small footprint.
Demo
https://writer.getkirby.com
Usage
<div class="writer" contenteditable>Hello <b>world</b></div>
<script type="module">
import Writer from "https://cdn.jsdelivr.net/gh/getkirby/writer@latest/dist/Writer.min.js";
const writer = Writer(".writer", {
onChange() {
console.log(writer.toJson());
}
});
</script>
Install via npm
npm i @getkirby/writer
Import it into your project …
import Writer from "@getkirby/writer";
const writer = Writer(".writer");
Documentation
The API is not stable yet. Methods and properties are very likely to change.
How to create a Writer instance
To create a Writer instance, you need to pass a HTML node or query selector for the element that should be editable.
<div class="writer" contenteditable></div>
<script type="module">
import Writer from "https://cdn.jsdelivr.net/gh/getkirby/writer@latest/dist/Writer.min.js";
const writer = Writer(".writer");
</script>
Options
You can pass additional options to the Writer as second argument
<div class="writer" contenteditable></div>
<script type="module">
import Writer from "https://cdn.jsdelivr.net/gh/getkirby/writer@latest/dist/Writer.min.js";
const writer = Writer(".writer", {
breaks: false,
onChange() {
console.log(writer.toJson());
}
});
</script>
autofocus: true
Enables/disables autofocus for the Writer element as soon as it is initialized
breaks: true
Enables/disables line breaks within the text. Line breaks are enabled by default.
formats: {}
You can overwrite or extend the available inline formats with the formats object. Check out src/Formats.js
for all default formats.
history: 100
The number of steps that can be undone in the history.
onBlur: () => {}
Add an event when the Writer looses focus
onChange: () => {}
React on any content changes in the Writer. This is the method to be used if you want to preview or save the Writer content. You probably want to use writer.toHtml()
, writer.toJson()
or writer.toText()
in this method.
onFocus: () => {}
Add an event when the Writer gains focus
onKeydown: () => {}
This event is triggered when a native keydown event happens in the Writer. This event is triggered before the any Writer shortcut.
onKeyup: () => {}
This event is triggered when a native keyup event happens in the Writer.
onMousedown: () => {}
This event is triggered when a native mousedown event happens in the Writer.
onMouseup: () => {}
This event is triggered when a native mouseup event happens in the Writer.
onRedo: () => {}
This event is triggered when history changes are reverted
onSelection: () => {}
This event is triggered when a selection is made and changed.
onSelectionEnd: () => {}
This event is only fired when the selection no longer changes (on mouseup)
onSelectionStart: () => {}
This event is fired when the selection starts
onUndo: () => {}
This event is triggered when content changes are undone.
placeholder: ""
Add a placeholder text to the Writer when there's no content.
shortcuts: {}
You can pass your own keyboard shortcuts or overwrite existing shortcuts with this object. Keyboard shortcuts are defined like this:
const writer = Writer(".writer", {
shortcuts: {
"Meta+b": () => {
// make something bold
}
}
});
The following special keywords for key combinations are automatically injected when pressed (in the following order): Meta
, Alt
, Ctrl
, Shift
spellcheck: true
Enable/disable native spellchecking
triggers: {}
Add custom key combinations, which will trigger a command, when being entered at the beginning of the text. This can be used to trigger a command when "/" is entered or to use typical Markdown syntax to trigger format changes (i.e. ## would convert the block to h2).
const writer = Writer(".writer", {
triggers: {
"/": () => {
// open the command dropdown
}
}
});
Methods
writer.activeFormats()
Returns an array of all active formats at the current selection. A format must be present at all characters in the selection to be included.
writer.activeLink()
Returns an object with attributes of the active link if the selected text has a link. The object contains href
, rel
, title
, and target
:
// example return value of writer.activeLink() if a link has been found
{
href: "https://getkirby.com",
rel: "me",
target: "_blank",
title: "Kirby"
}
writer.command(commandName, ...args)
Executes the given command with the optional arguments. Available commands:
writer.command('bold')
Wraps the selected text in a <strong>
tag.
writer.command('code')
Wraps the selected text in a <code>
tag.
writer.command('delete')
Deletes the selected text before the cursor
writer.command('deleteForward')
Deletes the selected text after the cursor
writer.command('enter')
Adds a line break if breaks are enabled.
writer.command('insert', text, at)
Inserts text at the given position.
writer.command('italic')
Wraps the selected text in an <em>
tag
writer.command('link', href)
Wraps the selected text in an <a>
tag with the given value for the href
attribute.
writer.command('paste', html)
Pastes any unsanitized html at the given selection/cursor. The html will be handled by the parser and all unwanted formats will be stripped. Block elements are of ignored and converted to line breaks if it makes sense.
writer.command('strikeThrough')
Wraps the selected text in an <del>
tag
writer.command('subscript')
Wraps the selected text in an <sub>
tag
writer.command('superscript')
Wraps the selected text in an <sup>
tag
writer.command('unlink')
Removes a link from the selected text, if it exists.
writer.cursor()
Returns the cursor object with additional methods to inspect and manipulate the cursor:
writer.cursor().isInFirstLine()
Checks if the cursor is in the first line of text
writer.cursor().isInLastLine()
Checks if the cursor is in the last line of text
writer.cursor().rect()
Returns the DOMRect object for the absolute cursor position
writer.cursor().set(position)
Moves the cursor to the given position
writer.focus(position)
Focusses the element and sets the cursor at the optional position.
You can use index numbers for the position as well as the
start
and end
keywords.
writer.redo()
Reverts the last undo event
writer.select(start, length)
Select the text from the start position for the given length. If you only specify a start position, the cursor will be set to that point and there will be no spanning selection.
writer.selection()
Returns the current selection object with additional methods to inspect and manipulate the selection
writer.selection().ancestor()
Returns the common ancestor element of the current selection
writer.selection().container()
Returns the writer element
writer.selection().containerRect()
Returns the largest possible selection DOMRect for the element. This is useful if you want to compare the current selection to the rest of the content
writer.selection().end()
Returns the position of the selection end
writer.selection().isWithin()
Checks if the selection is within the writer container
writer.selection().length()
Returns the length of the selected text
writer.selection().object()
Returns the native selection object
writer.selection().range(clone = false)
Returns the currently active range object (or null)
writer.selection().rangeAfterCursor()
Returns the range object before the cursor
writer.selection().rangeBeforeCursor()
Returns the range object after the cursor
writer.selection().rect()
Returns the DOMRect of the current selection
writer.selection().select(start, end)
Creates a new selection at the start and end point
writer.selection().start()
Returns the start position of the selection
writer.selection().text()
Returns the selected text
writer.toHtml(start, length)
Returns the current Writer content as sanitized HTML
writer.toJson(start, length)
Returns the current Writer content as JSON object
writer.toText(start, length)
Returns the current Writer content as plain text.
writer.undo()
Reverts the last step in history
writer.update()
Triggers a writer update manually. This will update the HTML in the writer according to the current document state. It will also trigger the onChange
event.
CSS
The Writer does not need a lot of CSS to work. You can find the suggested CSS in writer.css
:
.writer {
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
white-space: pre-wrap;
line-height: 1.5em;
}
.writer:empty::after {
content: attr(data-placeholder);
color: rgba(0,0,0, .5);
}
.writer code {
font-family: "SFMono-Regular", Consolas, Liberation Mono, Menlo, Courier, monospace;
background: #efefef;
font-size: .925rem;
padding: 0 .125em;
display: inline-block;
border-radius: 3px;
}
.writer strong {
font-weight: bold;
}
.writer em {
font-style: italic;
}
.writer a {
text-decoration: underline;
color: currentColor;
}
Contributing
There are still many things to get right before we can launch this project. It would be amazing to have you on board. Just get in contact if you don't know where to start: [email protected]
Developing
- Clone the repository
-
npm run i
-
npm run start
- Start writing code
Testing
We are using Cypress to run e2e and unit tests. Make sure to install the dependencies first with npm i
.
via Cypress app
Open the Cypress app with npm run cy:open
via command line
Start all tests on the command line with npm run test
License
This editor is licensed under the MIT license and will stay open. It will not fall under our proprietary Kirby license. Promised!
Authors
Bastian Allgeier [email protected]
... join me!