0
Fork 0
mirror of https://github.com/fastmail/Squire.git synced 2025-01-05 06:10:07 -05:00
Squire/README.md

483 lines
22 KiB
Markdown
Raw Normal View History

# Squire
2011-10-28 22:15:21 -05:00
Squire is an HTML5 rich text editor, which provides powerful cross-browser normalisation in a flexible lightweight package (only 16KB of JS after minification and gzip, with no dependencies!).
2011-10-28 22:15:21 -05:00
It was designed to handle email composition for the [Fastmail](https://www.fastmail.com) web app. The most important consequence of this (and where Squire differs from most other modern rich text editors) is that it must handle arbitrary HTML, because it may be used to forward or quote emails from third-parties and must be able to preserve their HTML without breaking the formatting. This means that it can't use a more structured (but limited) internal data model (as most other modern HTML editors do) and the HTML remains the source-of-truth. The other consequence is excellent handling of multiple levels of blockquotes.
Squire is designed to be integrated with your own UI framework, and so does not provide its own UI toolbar, widgets or overlays. Instead, you get a component you can insert in place of a `<textarea>` and manipulate programmatically, allowing you to integrate seamlessly with the rest of your application and lose the bloat of having two UI toolkits loaded.
2011-10-28 22:15:21 -05:00
Squire supports all reasonably recent browsers. It no longer supports any version of IE.
2011-10-28 22:15:21 -05:00
In addition to its use at [Fastmail](https://www.fastmail.com), it is also currently used in production at [ProtonMail](https://protonmail.com/), [SnappyMail](https://github.com/the-djmaze/snappymail), [StartMail](https://startmail.com/), [Tutanota](https://tutanota.com), [Zoho Mail](https://www.zoho.com/mail/), [Superhuman](https://superhuman.com/) and [Teamwork Desk](https://www.teamwork.com/desk/), as well as other non-mail apps including [Google Earth](https://www.google.com/earth/) (drop me a line if you're using Squire elsewhere, I'm always interested to hear about it!).
2011-10-28 22:15:21 -05:00
For a demo of the latest version with a production-level UI integration, [sign up for a free Fastmail trial](https://www.fastmail.com/signup/) :). There's also a very bare-bones integration in the repo; just clone it and open `Demo.html`. If you are reporting a bug, please report the steps to reproduce using `Demo.html`, to make sure it's not a bug in your integration.
2011-10-28 22:15:21 -05:00
## Installation and usage
2011-10-28 22:15:21 -05:00
1. Add Squire to your project: `npm install squire-rte`
2. In your code, `import Squire from 'squire-rte';`
3. Create your editor by calling `editor = new Squire(node);`.
2011-10-28 22:15:21 -05:00
### Invoke with script tag
Squire can also be used in a script tag:
1. Add a `<script>` tag to load in `dist/squire.js` (or `squire-raw.js` for the debuggable unminified version):
```
<script type="text/javascript" src="dist/squire.js"></script>
```
2. Get a reference to the DOM node in the document that you want to make into the rich textarea, e.g. `node = document.getElementById('editor-div')`.
3. Call `editor = new Squire(node)`. This will instantiate a new Squire instance. Please note, this will remove any current children of the node; you must use the `setHTML` command after initialising to set any content.
2016-12-06 22:31:44 -05:00
## Editor lifecycle
2016-12-06 22:31:44 -05:00
You can have multiple Squire instances in a single page without issue. If you are using the editor as part of a long lived single-page app, be sure to call `editor.destroy()` once you have finished using an instance to ensure it doesn't leak resources.
### Security
2016-12-06 22:31:44 -05:00
Malicious HTML can be a source of XSS and other security issues. You MUST provide a method to safely convert raw HTML into DOM nodes to use Squire. Squire will automatically integrate with [DOMPurify](https://github.com/cure53/DOMPurify) to do this if present in the page. Otherwise you must set a custom `sanitizeToDOMFragment` function in your config.
- **sanitizeToDOMFragment**: `(html: string, editor: Squire) => DocumentFragment`
A custom sanitization function. This will be called instead of the default call to DOMPurify to sanitize the potentially dangerous HTML. It is passed two arguments: the first is the string of HTML, the second is the Squire instance. It must return a DOM Fragment node belonging to the same document as the editor's root node, with the contents being clean DOM nodes to set/insert.
2018-10-05 02:01:12 -05:00
## Advanced usage
Squire provides an engine that handles the heavy work for you, making it easy to add extra features. With the `changeFormat` method you can easily add or remove any inline formatting you wish. And the `modifyBlocks` method can be used to make complicated block-level changes in a relatively easy manner.
2018-10-05 02:01:12 -05:00
If you need more commands than in the simple API, I suggest you check out the source code (it's not very long), and see how a lot of the other API methods are implemented in terms of these two methods.
The general philosophy of Squire is to allow the browser to do as much as it can (which unfortunately is not very much), but take control anywhere it deviates from what is required, or there are significant cross-browser differences. As such, the [`document.execCommand`](https://developer.mozilla.org/en-US/docs/Web/API/Document/execCommand) method is not used at all; instead all formatting is done via custom functions, and certain keys, such as 'enter' and 'backspace' are handled by the editor.
### Setting the default block style
By default, the editor will use a `<div>` for blank lines, as most users have been conditioned by Microsoft Word to expect <kbd>Enter</kbd> to act like pressing <kbd>return</kbd> on a typewriter. If you would like to use `<p>` tags (or anything else) for the default block type instead, you can pass a config object as the second parameter to the Squire constructor. You can also
pass a set of attributes to apply to each default block:
var editor = new Squire(document, {
blockTag: 'P',
blockAttributes: { style: 'font-size: 16px;' }
});
### Determining button state
If you are adding a UI to Squire, you'll probably want to show a button in different states depending on whether a particular style is active in the current selection or not. For example, a "Bold" button would be in a depressed state if the text under the cursor is already bold.
The efficient way to determine the state for most buttons is to monitor the "pathChange" event in the editor, and determine the state from the new path. If the selection goes across nodes, you will need to call the `hasFormat` method for each of your buttons to determine whether the styles are active. See the `getPath` and `hasFormat` documentation for more information.
## License
2011-10-28 22:15:21 -05:00
Squire is released under the MIT license. See LICENSE for full license.
2011-10-28 22:15:21 -05:00
## API
2011-10-28 22:15:21 -05:00
### addEventListener
2011-10-28 22:15:21 -05:00
Attach an event listener to the editor. The handler can be either a function or an object with a `handleEvent` method. This function or method will be called whenever the event fires, with an event object as the sole argument. The following events may be observed:
- **focus**: The editor gained focus.
- **blur**: The editor lost focus
- **keydown**: Standard [DOM keydown event](https://developer.mozilla.org/en-US/docs/Web/Events/keydown).
- **keypress**: Standard [DOM keypress event](https://developer.mozilla.org/en-US/docs/Web/Events/keypress).
- **keyup**: Standard [DOM keyup event](https://developer.mozilla.org/en-US/docs/Web/Events/keyup).
- **input**: The user inserted, deleted or changed the style of some text; in other words, the result for `editor.getHTML()` will have changed.
- **pathChange**: The path (see getPath documentation) to the cursor has changed. The new path is available as the `path` property on the event's `detail` property object.
- **select**: The user selected some text.
- **cursor**: The user cleared their selection or moved the cursor to a different position.
- **undoStateChange**: The availability of undo and/or redo has changed. The event object has a `detail` property, which is an object with two boolean properties, `canUndo` and `canRedo` to let you know the new state.
- **willPaste**: The user is pasting content into the document. The content that will be inserted is available as either the `fragment` property, or the `text` property for plain text, on the `detail` property of the event. You can modify this text/fragment in your event handler to change what will be pasted. You can also call the `preventDefault` on the event object to cancel the paste operation.
- **pasteImage**: The user is pasting image content into the document.
2011-10-28 22:15:21 -05:00
The method takes two arguments:
2011-10-28 22:15:21 -05:00
- **type**: The event to listen for. e.g. 'focus'.
- **handler**: The callback function to invoke
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### removeEventListener
2011-10-28 22:15:21 -05:00
Remove an event listener attached via the addEventListener method.
The method takes two arguments:
2011-10-28 22:15:21 -05:00
- **type**: The event type the handler was registered for.
- **handler**: The handler to remove.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### setKeyHandler
Adds or removes a keyboard shortcut. You can use this to override the default keyboard shortcuts (e.g. Ctrl-B for bold  see the bottom of KeyHandlers.js for the list).
This method takes two arguments:
- **key**: The key to handle, including any modifiers in alphabetical order. e.g. `"Alt-Ctrl-Meta-Shift-Enter"`
- **fn**: The function to be called when this key is pressed, or `null` if removing a key handler. The function will be passed three arguments when called:
- **self**: A reference to the Squire instance.
- **event**: The key event object.
- **range**: A Range object representing the current selection.
Returns self (the Squire instance).
### focus
2011-10-28 22:15:21 -05:00
Focuses the editor.
The method takes no arguments.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### blur
2011-10-28 22:15:21 -05:00
Removes focus from the editor.
2011-10-28 22:15:21 -05:00
The method takes no arguments.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### getHTML
2011-10-28 22:15:21 -05:00
Returns the HTML value of the editor in its current state. This value is equivalent to the contents of the `<body>` tag and does not include any surrounding boilerplate.
### setHTML
2011-10-28 22:15:21 -05:00
Sets the HTML value for the editor. The value supplied should not contain `<body>` tags or anything outside of that.
The method takes one argument:
2011-10-28 22:15:21 -05:00
- **html**: The html to set.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
### getSelectedText
Returns the text currently selected in the editor.
2012-03-29 00:26:01 -05:00
### insertImage
2011-10-28 22:15:21 -05:00
Inserts an image at the current cursor location.
The method takes two arguments:
2011-10-28 22:15:21 -05:00
- **src**: The source path for the image.
- **attributes**: (optional) An object containing other attributes to set on the `<img>` node. e.g. `{ class: 'class-name' }`. Any `src` attribute will be overwritten by the url given as the first argument.
2011-10-28 22:15:21 -05:00
Returns a reference to the newly inserted image element.
### insertHTML
Inserts an HTML fragment at the current cursor location, or replaces the selection if selected. The value supplied should not contain `<body>` tags or anything outside of that.
The method takes one argument:
- **html**: The html to insert.
Returns self (the Squire instance).
### getPath
2011-10-28 22:15:21 -05:00
2016-07-07 08:38:06 -05:00
Returns the path through the DOM tree from the `<body>` element to the current current cursor position. This is a string consisting of the tag, id, class, font, and color names in CSS format. For example `BODY>BLOCKQUOTE>DIV#id>STRONG>SPAN.font[fontFamily=Arial,sans-serif]>EM`. If a selection has been made, so different parts of the selection may have different paths, the value will be `(selection)`. The path is useful for efficiently determining the current formatting for bold, italic, underline etc, and thus determining button state. If a selection has been made, you can has the `hasFormat` method instead to get the current state for the properties you care about.
2011-10-28 22:15:21 -05:00
### getFontInfo
Returns an object containing the active font family, size, color and background color for the the current cursor position, if any are set. The property names are respectively `fontFamily`, `fontSize`, `color` and `backgroundColor` (matching the CSS property names). It looks at style attributes to detect this, so will not detect `<FONT>` tags or non-inline styles. If a selection across multiple elements has been made, it will return an empty object.
2018-05-09 02:43:11 -05:00
### createRange
Creates a range in the document belonging to the editor. Takes 4 arguments, matching the [W3C Range properties](https://developer.mozilla.org/en-US/docs/Web/API/Range) they set:
- **startContainer**
- **startOffset**
- **endContainer** (optional; if not collapsed)
- **endOffset** (optional; if not collapsed)
2018-05-09 02:43:11 -05:00
### getCursorPosition
Returns a bounding client rect (top/left/right/bottom properties relative to
the viewport) for the current selection/cursor.
### getSelection
2011-10-28 22:15:21 -05:00
Returns a [W3C Range object](https://developer.mozilla.org/en-US/docs/Web/API/Range) representing the current selection/cursor position.
2011-10-28 22:15:21 -05:00
### setSelection
2011-10-28 22:15:21 -05:00
Changes the current selection/cursor position.
The method takes one argument:
2011-10-28 22:15:21 -05:00
- **range**: The [W3C Range object](https://developer.mozilla.org/en-US/docs/Web/API/Range) representing the desired selection.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### moveCursorToStart
Removes any current selection and moves the cursor to the very beginning of the
document.
Returns self (the Squire instance).
### moveCursorToEnd
Removes any current selection and moves the cursor to the very end of the
document.
Returns self (the Squire instance).
### saveUndoState
Saves an undo checkpoint with the current editor state. Methods that modify the
state (e.g. bold/setHighlightColor/modifyBlocks) will automatically save undo
checkpoints; you only need this method if you want to modify the DOM outside of
one of these methods, and you want to save an undo checkpoint first.
Returns self (the Squire instance).
### undo
2011-10-28 22:15:21 -05:00
Undoes the most recent change.
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### redo
2011-10-28 22:15:21 -05:00
If the user has just undone a change, this will reapply that change.
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### hasFormat
2011-10-28 22:15:21 -05:00
Queries the editor for whether a particular format is applied anywhere in the current selection.
The method takes two arguments:
2011-10-28 22:15:21 -05:00
- **tag**: The tag of the format
- **attributes**: (optional) Any attributes the format.
2011-10-28 22:15:21 -05:00
2018-04-03 23:41:18 -05:00
Returns `true` if the entire selection is contained within an element with the specified tag and attributes, otherwise returns `false`.
2011-10-28 22:15:21 -05:00
### bold
2011-10-28 22:15:21 -05:00
Makes any non-bold currently selected text bold (by wrapping it in a `<b>` tag).
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### italic
2011-10-28 22:15:21 -05:00
Makes any non-italic currently selected text italic (by wrapping it in an `<i>` tag).
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### underline
2011-10-28 22:15:21 -05:00
Makes any non-underlined currently selected text underlined (by wrapping it in a `<u>` tag).
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### removeBold
2011-10-28 22:15:21 -05:00
Removes any bold formatting from the selected text.
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### removeItalic
2011-10-28 22:15:21 -05:00
Removes any italic formatting from the selected text.
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### removeUnderline
2011-10-28 22:15:21 -05:00
Removes any underline formatting from the selected text.
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### makeLink
2011-10-28 22:15:21 -05:00
Makes the currently selected text a link. If no text is selected, the URL or email will be inserted as text at the current cursor point and made into a link.
2011-10-28 22:15:21 -05:00
2014-12-20 01:07:11 -05:00
This method takes two arguments:
2011-10-28 22:15:21 -05:00
- **url**: The url or email to link to.
- **attributes**: (optional) An object containing other attributes to set on the `<a>` node. e.g. `{ target: '_blank' }`. Any `href` attribute will be overwritten by the url given as the first argument.
Returns self (the Squire instance).
### removeLink
Removes any link that is currently at least partially selected.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### setFontFace
2011-10-28 22:15:21 -05:00
Sets the font face for the selected text.
This method takes one argument:
2011-10-28 22:15:21 -05:00
- **font**: A comma-separated list of fonts (in order of preference) to set.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### setFontSize
2011-10-28 22:15:21 -05:00
Sets the font size for the selected text.
This method takes one argument:
2011-10-28 22:15:21 -05:00
- **size**: A size to set. Any CSS [length value](https://developer.mozilla.org/en-US/docs/Web/CSS/length) or [absolute-size value](https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_values_syntax#syntax-absolute-size) is accepted, e.g. '13px', or 'small'.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### setTextColor
Sets the color of the selected text.
This method takes one argument:
- **color**: The color to set. Any [CSS color value](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value) is accepted, e.g. '#f00', or 'hsl(0,0,0)'.
Returns self (the Squire instance).
### setHighlightColor
Sets the color of the background of the selected text.
This method takes one argument:
- **color**: The color to set. Any [CSS color value](https://developer.mozilla.org/en-US/docs/Web/CSS/color_value) is accepted, e.g. '#f00', or 'hsl(0,0,0)'.
Returns self (the Squire instance).
### setTextAlignment
2011-10-28 22:15:21 -05:00
Sets the text alignment in all blocks at least partially contained by the selection.
This method takes one argument:
2011-10-28 22:15:21 -05:00
- **alignment**: The direction to align to. Can be 'left', 'right', 'center' or 'justify'.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### setTextDirection
Sets the text direction in all blocks at least partially contained by the selection.
This method takes one argument:
- **direction**: The text direction. Can be 'ltr' or 'rtl'.
Returns self (the Squire instance).
### forEachBlock
Executes a function on each block in the current selection, or until the function returns a truthy value.
This method takes two arguments:
- **fn** The function to execute on each block node at least partially contained in the current selection. The function will be called with the block node as the only argument.
- **mutates** A boolean indicating whether your function may modify anything in the document in any way.
Returns self (the Squire instance).
### modifyBlocks
2011-10-28 22:15:21 -05:00
Extracts a portion of the DOM tree (up to the block boundaries of the current selection), modifies it and then reinserts it and merges the edges. See the code for examples if you're interested in using this function.
This method takes one argument:
2011-10-28 22:15:21 -05:00
- **modify** The function to apply to the extracted DOM tree; gets a document fragment as a sole argument. `this` is bound to the Squire instance. Should return the node or fragment to be reinserted in the DOM.
2011-10-28 22:15:21 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### increaseQuoteLevel
2011-10-28 22:15:21 -05:00
Increases by 1 the quote level (number of `<blockquote>` tags wrapping) all blocks at least partially selected.
2012-03-29 00:26:01 -05:00
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### decreaseQuoteLevel
2011-10-28 22:15:21 -05:00
Decreases by 1 the quote level (number of `<blockquote>` tags wrapping) all blocks at least partially selected.
Returns self (the Squire instance).
2012-03-29 00:26:01 -05:00
### makeUnorderedList
2011-10-28 22:15:21 -05:00
Changes all at-least-partially selected blocks to be part of an unordered list.
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### makeOrderedList
2011-10-28 22:15:21 -05:00
Changes all at-least-partially selected blocks to be part of an ordered list.
Returns self (the Squire instance).
2011-10-28 22:15:21 -05:00
### removeList
2011-10-28 22:15:21 -05:00
Changes any at-least-partially selected blocks which are part of a list to no longer be part of a list.
Returns self (the Squire instance).
### increaseListLevel
Increases by 1 the nesting level of any at-least-partially selected blocks which are part of a list.
Returns self (the Squire instance).
### decreaseListLevel
Decreases by 1 the nesting level of any at-least-partially selected blocks which are part of a list.
2014-12-20 01:07:11 -05:00
Returns self (the Squire instance).
### code
If no selection, or selection across blocks, converts the block to a `<pre>` to format the text as fixed-width. If a selection within a single block is present, wraps that in `<code>` tags for inline formatting instead.
Returns self (the Squire instance).
### removeCode
If inside a `<pre>`, converts that to the default block type instead. Otherwise, removes any `<code>` tags.
Returns self (the Squire instance).
### toggleCode
If inside a `<pre>` or `<code>`, calls `removeCode()`, otherwise callse `code()`.
Returns self (the Squire instance).
### removeAllFormatting
Removes all formatting from the selection. Block elements (list items, table cells, etc.) are kept as separate blocks.
Returns self (the Squire instance).
### changeFormat
Change the **inline** formatting of the current selection. This is a high-level method which is used to implement the bold, italic etc. helper methods. THIS METHOD IS ONLY FOR USE WITH INLINE TAGS, NOT BLOCK TAGS. It takes 4 arguments:
1. An object describing the formatting to add, or `null` if you only wish to remove formatting. If supplied, this object should have a `tag` property with the string name of the tag to wrap around the selected text (e.g. `"STRONG"`) and optionally an `attributes` property, consisting of an object of attributes to apply to the tag (e.g. `{"class": "bold"}`).
2. An object describing the formatting to remove, in the same format as the object given to add formatting, or `null` if you only wish to add formatting.
3. A Range object with the range to apply the formatting changes to (or `null`/omit to apply to current selection).
4. A boolean (defaults to `false` if omitted). If `true`, any formatting nodes that cover at least part of the selected range will be removed entirely (so will potentially be removed from text outside the selected range as well). If `false`, the formatting nodes will continue to apply to any text outside the selection. This is useful, for example, when removing links. If any of the text in the selection is part of a link, the whole link is removed, rather than the link continuing to apply to bits of text outside the selection.
### modifyDocument
Takes in a function that can modify the document without the modifications being treated as input.
This is useful when the document needs to be changed programmatically, but those changes should not raise input events or modify the undo state.
### linkRegExp
This is the regular expression used to automatically mark up links when inserting HTML or after pressing space. You can change it if you want to use a custom regular expression for detecting links, or set to `/[]/` to turn off link detection. To append to the existing regex, set it to `linkRegExp.source + '|' + newLinkRegExp`. For compatibility with linkRegExpHandlers use named capture groups (`?<name>`).
### linkRegExpHandlers
This is a map of handlers for different types of matches in linkRegExp. For example, linkRegExp has a named group 'url' that matches urls, and a named group 'email' that matches emails. linkRegExpHandler['url'] and linkRegExpHandler['email'] are functions that take in the matching string and returns what the link should have in its 'href'.