'use strict';
Object.defineProperty(exports, '__esModule', { value: true });
function detect(enumerable, callback) {
if ('detect' in enumerable) {
return enumerable.detect(callback);
}
else {
for (let i = 0; i < enumerable.length; i++) {
if (callback(enumerable[i])) {
return enumerable[i];
}
}
}
}
function any(enumerable, callback) {
if ('any' in enumerable) {
return enumerable.any(callback);
}
for (let i = 0; i < enumerable.length; i++) {
if (callback(enumerable[i])) {
return true;
}
}
return false;
}
/**
* Useful for array-like things that aren't
* actually arrays, like NodeList
* @private
*/
function forEach(enumerable, callback) {
if ('forEach' in enumerable) {
enumerable.forEach(callback);
}
else {
for (let i = 0; i < enumerable.length; i++) {
callback(enumerable[i], i);
}
}
}
function filter(enumerable, conditionFn) {
const filtered = [];
forEach(enumerable, i => {
if (conditionFn(i)) {
filtered.push(i);
}
});
return filtered;
}
/**
* @return {Integer} the number of items that are the same, starting from the 0th index, in a and b
* @private
*/
function commonItemLength(listA, listB) {
let offset = 0;
while (offset < listA.length && offset < listB.length) {
if (listA[offset] !== listB[offset]) {
break;
}
offset++;
}
return offset;
}
/**
* @return {Array} the items that are the same, starting from the 0th index, in a and b
* @private
*/
function commonItems(listA, listB) {
let offset = 0;
while (offset < listA.length && offset < listB.length) {
if (listA[offset] !== listB[offset]) {
break;
}
offset++;
}
return listA.slice(0, offset);
}
function reduce(enumerable, callback, initialValue) {
let previousValue = initialValue;
forEach(enumerable, (val, index) => {
previousValue = callback(previousValue, val, index);
});
return previousValue;
}
/**
* @param {Array} array of key1,value1,key2,value2,...
* @return {Object} {key1:value1, key2:value2, ...}
* @private
*/
function kvArrayToObject(array) {
const obj = {};
for (let i = 0; i < array.length; i += 2) {
let [key, value] = [array[i], array[i + 1]];
obj[key] = value;
}
return obj;
}
function objectToSortedKVArray(obj) {
const keys = Object.keys(obj).sort();
const result = [];
keys.forEach(k => {
result.push(k);
result.push(obj[k]);
});
return result;
}
// check shallow equality of two non-nested arrays
function isArrayEqual(arr1, arr2) {
let l1 = arr1.length;
let l2 = arr2.length;
if (l1 !== l2) {
return false;
}
for (let i = 0; i < l1; i++) {
if (arr1[i] !== arr2[i]) {
return false;
}
}
return true;
}
// return an object with only the valid keys
function filterObject(object, validKeys = []) {
let result = {};
forEach(filter(Object.keys(object), key => validKeys.indexOf(key) !== -1), key => (result[key] = object[key]));
return result;
}
function contains(array, item) {
return array.indexOf(item) !== -1;
}
function values(object) {
return Object.keys(object).map(key => object[key]);
}
const NODE_TYPES = {
ELEMENT: 1,
TEXT: 3,
COMMENT: 8,
};
function isTextNode(node) {
return node.nodeType === NODE_TYPES.TEXT;
}
function isCommentNode(node) {
return node.nodeType === NODE_TYPES.COMMENT;
}
function isElementNode(node) {
return node.nodeType === NODE_TYPES.ELEMENT;
}
function clearChildNodes(element) {
while (element.childNodes.length) {
element.removeChild(element.childNodes[0]);
}
}
/**
* @return {Boolean} true when the child node is contained or the same as
* (e.g., inclusive containment) the parent node
* see https://github.com/webmodules/node-contains/blob/master/index.js
* Mimics the behavior of `Node.contains`, which is broken in IE 10
* @private
*/
function containsNode(parentNode, childNode) {
if (parentNode === childNode) {
return true;
}
const position = parentNode.compareDocumentPosition(childNode);
return !!(position & Node.DOCUMENT_POSITION_CONTAINED_BY);
}
/**
* converts the element's NamedNodeMap of attrs into
* an object with key-value pairs
* @param {DOMNode} element
* @return {Object} key-value pairs
* @private
*/
function getAttributes(element) {
const result = {};
if (element.hasAttributes()) {
forEach(element.attributes, ({ name, value }) => {
result[name] = value;
});
}
return result;
}
function addClassName(element, className) {
element.classList.add(className);
}
function removeClassName(element, className) {
element.classList.remove(className);
}
function normalizeTagName(tagName) {
return tagName.toLowerCase();
}
function parseHTML(html) {
const div = document.createElement('div');
div.innerHTML = html;
return div;
}
function serializeHTML(node) {
const div = document.createElement('div');
div.appendChild(node);
return div.innerHTML;
}
class View {
constructor(options = {}) {
this.isShowing = false;
this.isDestroyed = false;
options.tagName = options.tagName || 'div';
options.container = options.container || document.body;
this.element = document.createElement(options.tagName);
this.container = options.container;
let classNames = options.classNames || [];
classNames.forEach(name => addClassName(this.element, name));
this._eventListeners = [];
}
addEventListener(element, type, listener) {
element.addEventListener(type, listener);
this._eventListeners.push([element, type, listener]);
}
removeAllEventListeners() {
this._eventListeners.forEach(([element, type, listener]) => {
element.removeEventListener(type, listener);
});
}
show() {
if (!this.isShowing) {
this.container.appendChild(this.element);
this.isShowing = true;
return true;
}
}
hide() {
if (this.isShowing) {
this.container.removeChild(this.element);
this.isShowing = false;
return true;
}
}
destroy() {
this.removeAllEventListeners();
this.hide();
this.isDestroyed = true;
}
}
/*
* @param {String} string
* @return {String} a dasherized string. 'modelIndex' -> 'model-index', etc
*/
function dasherize(string) {
return string.replace(/[A-Z]/g, (match, offset) => {
const lower = match.toLowerCase();
return offset === 0 ? lower : '-' + lower;
});
}
function startsWith(string, character) {
return string.charAt(0) === character;
}
function endsWith(string, endString) {
let index = string.lastIndexOf(endString);
return index !== -1 && index === string.length - endString.length;
}
function getEventTargetMatchingTag(tagName, target, container) {
tagName = normalizeTagName(tagName);
// Traverses up DOM from an event target to find the node matching specifed tag
while (target && target !== container) {
if (normalizeTagName(target.tagName) === tagName) {
return target;
}
target = target.parentElement;
}
}
function getElementRelativeOffset(element) {
const offset = { left: 0, top: -window.pageYOffset };
const offsetParent = element.offsetParent;
const offsetParentPosition = window.getComputedStyle(offsetParent).position;
let offsetParentRect;
if (offsetParentPosition === 'relative') {
offsetParentRect = offsetParent.getBoundingClientRect();
offset.left = offsetParentRect.left;
offset.top = offsetParentRect.top;
}
return offset;
}
function getElementComputedStyleNumericProp(element, prop) {
return parseFloat(window.getComputedStyle(element)[prop]);
}
function positionElementToRect(element, rect, topOffset, leftOffset) {
const relativeOffset = getElementRelativeOffset(element);
const style = element.style;
const round = Math.round;
let left, top;
topOffset = topOffset || 0;
leftOffset = leftOffset || 0;
left = round(rect.left - relativeOffset.left - leftOffset);
top = round(rect.top + rect.height - relativeOffset.top - topOffset);
style.left = `${left}px`;
style.top = `${top}px`;
return { left: left, top: top };
}
function positionElementHorizontallyCenteredToRect(element, rect, topOffset) {
const horizontalCenter = element.offsetWidth / 2 - rect.width / 2;
return positionElementToRect(element, rect, topOffset, horizontalCenter);
}
function positionElementCenteredBelow(element, belowElement) {
const elementMargin = getElementComputedStyleNumericProp(element, 'marginTop');
return positionElementHorizontallyCenteredToRect(element, belowElement.getBoundingClientRect(), -elementMargin);
}
function setData(element, name, value) {
if (element.dataset) {
element.dataset[name] = value;
}
else {
const dataName = dasherize(name);
return element.setAttribute(dataName, value);
}
}
function whenElementIsNotInDOM(element, callback) {
let isCanceled = false;
const observerFn = () => {
if (isCanceled) {
return;
}
if (!element.parentNode) {
callback();
}
else {
window.requestAnimationFrame(observerFn);
}
};
observerFn();
return { cancel: () => (isCanceled = true) };
}
class Markuperable {
constructor() {
this.markups = [];
this.prev = null;
this.next = null;
this.isAtom = false;
this.isMarker = false;
this.section = null;
this.parent = null;
this.renderNode = null;
}
charAt(offset) {
return this.value.slice(offset, offset + 1);
}
clearMarkups() {
this.markups = [];
}
addMarkup(markup) {
this.markups.push(markup);
}
addMarkupAtIndex(markup, index) {
this.markups.splice(index, 0, markup);
}
removeMarkup(markupOrMarkupCallback) {
let callback;
if (typeof markupOrMarkupCallback === 'function') {
callback = markupOrMarkupCallback;
}
else {
let markup = markupOrMarkupCallback;
callback = _markup => _markup === markup;
}
forEach(filter(this.markups, callback), m => this._removeMarkup(m));
}
_removeMarkup(markup) {
const index = this.markups.indexOf(markup);
if (index !== -1) {
this.markups.splice(index, 1);
}
}
hasMarkup(tagNameOrMarkup) {
return !!this.getMarkup(tagNameOrMarkup);
}
getMarkup(tagNameOrMarkup) {
if (typeof tagNameOrMarkup === 'string') {
let tagName = normalizeTagName(tagNameOrMarkup);
return detect(this.markups, markup => markup.tagName === tagName);
}
else {
let targetMarkup = tagNameOrMarkup;
return detect(this.markups, markup => markup === targetMarkup);
}
}
get openedMarkups() {
let count = 0;
if (this.prev) {
count = commonItemLength(this.markups, this.prev.markups);
}
return this.markups.slice(count);
}
get closedMarkups() {
let count = 0;
if (this.next) {
count = commonItemLength(this.markups, this.next.markups);
}
return this.markups.slice(count);
}
}
class MobiledocError extends Error {
}
function assert(message, conditional) {
if (!conditional) {
throw new MobiledocError(message);
}
}
function assertExistsIn(message, key, object) {
assert(message, key in object);
}
function assertNotNull(message, value) {
if (value === null) {
throw new MobiledocError(message);
}
}
function assertType(message, _value, conditional) {
assert(message, conditional);
}
function expect(value, message) {
if (value === null || value === undefined) {
throw new MobiledocError(message);
}
return value;
}
function unwrap(value) {
return expect(value, 'expected value to not be null or undefined');
}
// Unicode uses a pair of "surrogate" characters" (a high- and low-surrogate)
// to encode characters outside the basic multilingual plane (like emoji and
// some languages).
// These values are the unicode code points for the start and end of the
// high- and low-surrogate characters.
// See "high surrogate" and "low surrogate" on
// https://en.wikipedia.org/wiki/Unicode_block
const HIGH_SURROGATE_RANGE = [0xd800, 0xdbff];
const LOW_SURROGATE_RANGE = [0xdc00, 0xdfff];
class Marker extends Markuperable {
constructor(value = '', markups = []) {
super();
this.type = "marker" /* MARKER */;
this.isMarker = true;
this.markups = [];
this.renderNode = null;
this.value = value;
assert('Marker must have value', value !== undefined && value !== null);
markups.forEach(m => this.addMarkup(m));
}
clone() {
const clonedMarkups = this.markups.slice();
return this.builder.createMarker(this.value, clonedMarkups);
}
get isEmpty() {
return this.isBlank;
}
get isBlank() {
return this.length === 0;
}
/**
* A marker's text is equal to its value.
* Compare with an Atom which distinguishes between text and value
*/
get text() {
return this.value;
}
get length() {
return this.value.length;
}
// delete the character at this offset,
// update the value with the new value
deleteValueAtOffset(offset) {
assert('Cannot delete value at offset outside bounds', offset >= 0 && offset <= this.length);
let width = 1;
let code = this.value.charCodeAt(offset);
if (code >= HIGH_SURROGATE_RANGE[0] && code <= HIGH_SURROGATE_RANGE[1]) {
width = 2;
}
else if (code >= LOW_SURROGATE_RANGE[0] && code <= LOW_SURROGATE_RANGE[1]) {
width = 2;
offset = offset - 1;
}
const [left, right] = [this.value.slice(0, offset), this.value.slice(offset + width)];
this.value = left + right;
return width;
}
canJoin(other) {
return other && other.isMarker && isArrayEqual(this.markups, other.markups);
}
textUntil(offset) {
return this.value.slice(0, offset);
}
split(offset = 0, endOffset = this.length) {
let markers = [
this.builder.createMarker(this.value.substring(0, offset)),
this.builder.createMarker(this.value.substring(offset, endOffset)),
this.builder.createMarker(this.value.substring(endOffset)),
];
this.markups.forEach(mu => markers.forEach(m => m.addMarkup(mu)));
return markers;
}
/**
* @return {Array} 2 markers either or both of which could be blank
*/
splitAtOffset(offset) {
assert('Cannot split a marker at an offset > its length', offset <= this.length);
let { value, builder } = this;
let pre = builder.createMarker(value.substring(0, offset));
let post = builder.createMarker(value.substring(offset));
this.markups.forEach(markup => {
pre.addMarkup(markup);
post.addMarkup(markup);
});
return [pre, post];
}
}
function isMarker(postNode) {
return postNode.type === "marker" /* MARKER */;
}
var Keycodes = {
BACKSPACE: 8,
SPACE: 32,
ENTER: 13,
SHIFT: 16,
ESC: 27,
DELETE: 46,
'0': 48,
'9': 57,
A: 65,
Z: 90,
a: 97,
z: 122,
NUMPAD_0: 186,
NUMPAD_9: 111,
';': 186,
'.': 190,
'`': 192,
'[': 219,
'"': 222,
// Input Method Editor uses multiple keystrokes to display characters.
// Example on mac: press option-i then i. This fires 2 key events in Chrome
// with keyCode 229 and displays ˆ and then î.
// See http://lists.w3.org/Archives/Public/www-dom/2010JulSep/att-0182/keyCode-spec.html#fixed-virtual-key-codes
IME: 229,
TAB: 9,
CLEAR: 12,
PAUSE: 19,
PAGEUP: 33,
PAGEDOWN: 34,
END: 35,
HOME: 36,
LEFT: 37,
UP: 38,
RIGHT: 39,
DOWN: 40,
INS: 45,
META: 91,
ALT: 18,
CTRL: 17,
};
var Keys = {
BACKSPACE: 'Backspace',
SPACE: ' ',
ENTER: 'Enter',
SHIFT: 'Shift',
ESC: 'Escape',
DELETE: 'Delete',
INS: 'Insert',
HOME: 'Home',
END: 'End',
PAGEUP: 'PageUp',
PAGEDOWN: 'PageDown',
CLEAR: 'Clear',
PAUSE: 'Pause',
TAB: 'Tab',
ALT: 'Alt',
CTRL: 'Control',
LEFT: 'ArrowLeft',
RIGHT: 'ArrowRight',
UP: 'ArrowUp',
DOWN: 'ArrowDown',
};
const TAB = '\t';
const ENTER = '\n';
var Direction;
(function (Direction) {
Direction[Direction["FORWARD"] = 1] = "FORWARD";
Direction[Direction["BACKWARD"] = -1] = "BACKWARD";
})(Direction || (Direction = {}));
const MODIFIERS = {
META: 1,
CTRL: 2,
SHIFT: 4,
ALT: 8,
};
function modifierMask(event) {
let { metaKey, shiftKey, ctrlKey, altKey } = event;
let modVal = (val, modifier) => {
return (val && modifier) || 0;
};
return (modVal(metaKey, MODIFIERS.META) +
modVal(shiftKey, MODIFIERS.SHIFT) +
modVal(ctrlKey, MODIFIERS.CTRL) +
modVal(altKey, MODIFIERS.ALT));
}
const SPECIAL_KEYS = {
BACKSPACE: Keycodes.BACKSPACE,
TAB: Keycodes.TAB,
ENTER: Keycodes.ENTER,
ESC: Keycodes.ESC,
SPACE: Keycodes.SPACE,
PAGEUP: Keycodes.PAGEUP,
PAGEDOWN: Keycodes.PAGEDOWN,
END: Keycodes.END,
HOME: Keycodes.HOME,
LEFT: Keycodes.LEFT,
UP: Keycodes.UP,
RIGHT: Keycodes.RIGHT,
DOWN: Keycodes.DOWN,
INS: Keycodes.INS,
DEL: Keycodes.DELETE,
};
function specialCharacterToCode(specialCharacter) {
return SPECIAL_KEYS[specialCharacter];
}
// heuristic for determining if `event` is a key event
function isKeyEvent(event) {
return /^key/.test(event.type);
}
/**
* An abstraction around a KeyEvent
* that key listeners in the editor can use
* to determine what sort of key was pressed
*/
class Key {
constructor(event) {
this.key = event.key;
this.keyCode = event.keyCode;
this.charCode = event.charCode;
this.event = event;
this.modifierMask = modifierMask(event);
}
static fromEvent(event) {
assert('Must pass a Key event to Key.fromEvent', event && isKeyEvent(event));
return new Key(event);
}
toString() {
if (this.isTab()) {
return TAB;
}
return String.fromCharCode(this.charCode);
}
// See https://caniuse.com/#feat=keyboardevent-key for browser support.
isKeySupported() {
return this.key;
}
isKey(identifier) {
if (this.isKeySupported()) {
assert(`Must define Keys.${identifier}.`, !!Keys[identifier]);
return this.key === Keys[identifier];
}
else {
assert(`Must define Keycodes.${identifier}.`, !!Keycodes[identifier]);
return this.keyCode === Keycodes[identifier];
}
}
isEscape() {
return this.isKey('ESC');
}
isDelete() {
return this.isKey('BACKSPACE') || this.isForwardDelete();
}
isForwardDelete() {
return this.isKey('DELETE');
}
isArrow() {
return this.isHorizontalArrow() || this.isVerticalArrow();
}
isHorizontalArrow() {
return this.isLeftArrow() || this.isRightArrow();
}
isHorizontalArrowWithoutModifiersOtherThanShift() {
return this.isHorizontalArrow() && !(this.ctrlKey || this.metaKey || this.altKey);
}
isVerticalArrow() {
return this.isKey('UP') || this.isKey('DOWN');
}
isLeftArrow() {
return this.isKey('LEFT');
}
isRightArrow() {
return this.isKey('RIGHT');
}
isHome() {
return this.isKey('HOME');
}
isEnd() {
return this.isKey('END');
}
isPageUp() {
return this.isKey('PAGEUP');
}
isPageDown() {
return this.isKey('PAGEDOWN');
}
isInsert() {
return this.isKey('INS');
}
isClear() {
return this.isKey('CLEAR');
}
isPause() {
return this.isKey('PAUSE');
}
isSpace() {
return this.isKey('SPACE');
}
// In Firefox, pressing ctrl-TAB will switch to another open browser tab, but
// it will also fire a keydown event for the tab+modifier (ctrl). This causes
// Mobiledoc to erroneously insert a tab character before FF switches to the
// new browser tab. Chrome doesn't fire this event so the issue doesn't
// arise there. Fix this by returning false when the TAB key event includes a
// modifier.
// See: https://github.com/bustle/mobiledoc-kit/issues/565
isTab() {
return !this.hasAnyModifier() && this.isKey('TAB');
}
isEnter() {
return this.isKey('ENTER');
}
/*
* If the key is the actual shift key. This is false when the shift key
* is held down and the source `event` is not the shift key.
* @see {isShift}
* @return {bool}
*/
isShiftKey() {
return this.isKey('SHIFT');
}
/*
* If the key is the actual alt key (aka "option" on mac). This is false when the alt key
* is held down and the source `event` is not the alt key.
* @return {bool}
*/
isAltKey() {
return this.isKey('ALT');
}
/*
* If the key is the actual ctrl key. This is false when the ctrl key
* is held down and the source `event` is not the ctrl key.
* @return {bool}
*/
isCtrlKey() {
return this.isKey('CTRL');
}
isIME() {
// FIXME the IME action seems to get lost when we issue an
// `editor.deleteSelection` before it (in Chrome)
return this.keyCode === Keycodes.IME;
}
get direction() {
switch (true) {
case this.isDelete():
return this.isForwardDelete() ? Direction.FORWARD : Direction.BACKWARD;
case this.isHorizontalArrow():
return this.isRightArrow() ? Direction.FORWARD : Direction.BACKWARD;
default:
return Direction.FORWARD;
}
}
/**
* If the shift key is depressed.
* For example, while holding down meta+shift, pressing the "v"
* key would result in an event whose `Key` had `isShift()` with a truthy value,
* because the shift key is down when pressing the "v".
* @see {isShiftKey} which checks if the key is actually the shift key itself.
* @return {bool}
*/
isShift() {
return this.shiftKey;
}
hasModifier(modifier) {
return modifier & this.modifierMask;
}
hasAnyModifier() {
return !!this.modifierMask;
}
get ctrlKey() {
return MODIFIERS.CTRL & this.modifierMask;
}
get metaKey() {
return MODIFIERS.META & this.modifierMask;
}
get shiftKey() {
return MODIFIERS.SHIFT & this.modifierMask;
}
get altKey() {
return MODIFIERS.ALT & this.modifierMask;
}
isPrintableKey() {
return !(this.isArrow() ||
this.isHome() ||
this.isEnd() ||
this.isPageUp() ||
this.isPageDown() ||
this.isInsert() ||
this.isClear() ||
this.isPause() ||
this.isEscape());
}
isNumberKey() {
if (this.isKeySupported()) {
return this.key >= '0' && this.key <= '9';
}
else {
const code = this.keyCode;
return ((code >= Keycodes['0'] && code <= Keycodes['9']) || (code >= Keycodes.NUMPAD_0 && code <= Keycodes.NUMPAD_9)); // numpad keys
}
}
isLetterKey() {
if (this.isKeySupported()) {
const key = this.key;
return (key >= 'a' && key <= 'z') || (key >= 'A' && key <= 'Z');
}
else {
const code = this.keyCode;
return (code >= Keycodes.A && code <= Keycodes.Z) || (code >= Keycodes.a && code <= Keycodes.z);
}
}
isPunctuation() {
if (this.isKeySupported()) {
const key = this.key;
return (key >= ';' && key <= '`') || (key >= '[' && key <= '"');
}
else {
const code = this.keyCode;
return (code >= Keycodes[';'] && code <= Keycodes['`']) || (code >= Keycodes['['] && code <= Keycodes['"']);
}
}
/**
* See https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode#Printable_keys_in_standard_position
* and http://stackoverflow.com/a/12467610/137784
*/
isPrintable() {
if (this.ctrlKey || this.metaKey) {
return false;
}
// Firefox calls keypress events for some keys that should not be printable
if (!this.isPrintableKey()) {
return false;
}
return (this.keyCode !== 0 ||
this.toString().length > 0 ||
this.isNumberKey() ||
this.isSpace() ||
this.isTab() ||
this.isEnter() ||
this.isLetterKey() ||
this.isPunctuation() ||
this.isIME());
}
}
function clearSelection(root = window) {
const selection = root.getSelection();
selection && selection.removeAllRanges();
}
function textNodeRects(node) {
let range = document.createRange();
range.setEnd(node, node.nodeValue.length);
range.setStart(node, 0);
return range.getClientRects();
}
function findOffsetInTextNode(node, coords) {
let len = node.nodeValue.length;
let range = document.createRange();
for (let i = 0; i < len; i++) {
range.setEnd(node, i + 1);
range.setStart(node, i);
let rect = range.getBoundingClientRect();
if (rect.top === rect.bottom) {
continue;
}
if (rect.left <= coords.left && rect.right >= coords.left && rect.top <= coords.top && rect.bottom >= coords.top) {
return { node, offset: i + (coords.left >= (rect.left + rect.right) / 2 ? 1 : 0) };
}
}
return { node, offset: 0 };
}
/*
* @param {Object} coords with `top` and `left`
* @see https://github.com/ProseMirror/prosemirror/blob/4c22e3fe97d87a355a0534e25d65aaf0c0d83e57/src/edit/dompos.js
* @return {Object} {node, offset}
*/
/* eslint-disable complexity */
function findOffsetInNode(node, coords) {
let closest, dyClosest = 1e8, coordsClosest, offset = 0;
for (let child = node.firstChild; child; child = child.nextSibling) {
let rects;
if (isElementNode(child)) {
rects = child.getClientRects();
}
else if (isTextNode(child)) {
rects = textNodeRects(child);
}
else {
continue;
}
for (let i = 0; i < rects.length; i++) {
let rect = rects[i];
if (rect.left <= coords.left && rect.right >= coords.left) {
let dy = rect.top > coords.top ? rect.top - coords.top : rect.bottom < coords.top ? coords.top - rect.bottom : 0;
if (dy < dyClosest) {
closest = child;
dyClosest = dy;
coordsClosest = dy ? { left: coords.left, top: rect.top } : coords;
if (isElementNode(child) && !child.firstChild) {
offset = i + (coords.left >= (rect.left + rect.right) / 2 ? 1 : 0);
}
continue;
}
}
if (!closest && (coords.top >= rect.bottom || (coords.top >= rect.top && coords.left >= rect.right))) {
offset = i + 1;
}
}
}
if (!closest) {
return { node, offset };
}
if (isTextNode(closest)) {
return findOffsetInTextNode(closest, coordsClosest);
}
if (closest.firstChild) {
return findOffsetInNode(closest, coordsClosest);
}
return { node, offset };
}
/* eslint-enable complexity */
function constrainNodeTo(node, parentNode, existingOffset) {
let compare = parentNode.compareDocumentPosition(node);
if (compare & Node.DOCUMENT_POSITION_CONTAINED_BY) {
// the node is inside parentNode, do nothing
return { node, offset: existingOffset };
}
else if (compare & Node.DOCUMENT_POSITION_CONTAINS) {
// the node contains parentNode. This shouldn't happen.
return { node, offset: existingOffset };
}
else if (compare & Node.DOCUMENT_POSITION_PRECEDING) {
// node is before parentNode. return start of deepest first child
let child = parentNode.firstChild;
while (child && child.firstChild) {
child = child.firstChild;
}
return { node: child, offset: 0 };
}
else if (compare & Node.DOCUMENT_POSITION_FOLLOWING) {
// node is after parentNode. return end of deepest last child
let child = parentNode.lastChild;
while (child.lastChild) {
child = child.lastChild;
}
let offset = isTextNode(child) ? child.textContent.length : 1;
return { node: child, offset };
}
else {
return { node, offset: existingOffset };
}
}
/*
* Returns a new selection that is constrained within parentNode.
* If the anchorNode or focusNode are outside the parentNode, they are replaced with the beginning
* or end of the parentNode's children
*/
function constrainSelectionTo(selection, parentNode) {
assertNotNull('selection anchorNode should not be null', selection.anchorNode);
assertNotNull('selection focusNode should not be null', selection.focusNode);
let { node: anchorNode, offset: anchorOffset } = constrainNodeTo(selection.anchorNode, parentNode, selection.anchorOffset);
let { node: focusNode, offset: focusOffset } = constrainNodeTo(selection.focusNode, parentNode, selection.focusOffset);
return { anchorNode, anchorOffset, focusNode, focusOffset };
}
function isFullSelection(selection) {
return selection instanceof Selection;
}
function comparePosition(selection) {
assertNotNull('selection anchorNode should not be null', selection.anchorNode);
assertNotNull('selection focusNode should not be null', selection.focusNode);
let { anchorNode, focusNode, anchorOffset, focusOffset } = selection;
let headNode, tailNode, headOffset, tailOffset, direction;
const position = anchorNode.compareDocumentPosition(focusNode);
// IE may select return focus and anchor nodes far up the DOM tree instead of
// picking the deepest, most specific possible node. For example in
//
//
abcdef
//
// with a cursor between c and d, IE might say the focusNode is with
// an offset of 1. However the anchorNode for a selection might still be
//
2 if there was a selection.
//
// This code walks down the DOM tree until a good comparison of position can be
// made.
//
if (position & Node.DOCUMENT_POSITION_CONTAINS) {
if (focusOffset < focusNode.childNodes.length) {
focusNode = focusNode.childNodes[focusOffset];
focusOffset = 0;
}
else {
// This situation happens on IE when triple-clicking to select.
// Set the focus to the very last character inside the node.
while (focusNode.lastChild) {
focusNode = focusNode.lastChild;
}
focusOffset = focusNode.textContent.length;
}
return comparePosition({
focusNode,
focusOffset,
anchorNode,
anchorOffset,
});
}
else if (position & Node.DOCUMENT_POSITION_CONTAINED_BY) {
let offset = anchorOffset - 1;
if (offset < 0) {
offset = 0;
}
return comparePosition({
anchorNode: anchorNode.childNodes[offset],
anchorOffset: 0,
focusNode,
focusOffset,
});
// The meat of translating anchor and focus nodes to head and tail nodes
}
else if (position & Node.DOCUMENT_POSITION_FOLLOWING) {
headNode = anchorNode;
tailNode = focusNode;
headOffset = anchorOffset;
tailOffset = focusOffset;
direction = Direction.FORWARD;
}
else if (position & Node.DOCUMENT_POSITION_PRECEDING) {
headNode = focusNode;
tailNode = anchorNode;
headOffset = focusOffset;
tailOffset = anchorOffset;
direction = Direction.BACKWARD;
}
else {
// same node
headNode = tailNode = anchorNode;
headOffset = anchorOffset;
tailOffset = focusOffset;
if (tailOffset < headOffset) {
// Swap the offset order
headOffset = focusOffset;
tailOffset = anchorOffset;
direction = Direction.BACKWARD;
}
else if (headOffset < tailOffset) {
direction = Direction.FORWARD;
}
else {
direction = null;
}
}
return { headNode, headOffset, tailNode, tailOffset, direction };
}
/**
* A logical range of a {@link Post}.
* Usually an instance of Range will be read from the {@link Editor#range} property,
* but it may be useful to instantiate a range directly when programmatically modifying a Post.
*/
class Range {
/**
* @param {Position} head
* @param {Position} [tail=head]
* @param {Direction} [direction=null]
* @private
*/
constructor(head, tail = head, direction = null) {
/** @property {Position} head */
this.head = head;
/** @property {Position} tail */
this.tail = tail;
/** @property {Direction} direction */
this.direction = direction;
}
/**
* Shorthand to create a new range from a section(s) and offset(s).
* When given only a head section and offset, creates a collapsed range.
* @param {Section} headSection
* @param {number} headOffset
* @param {Section} [tailSection=headSection]
* @param {number} [tailOffset=headOffset]
* @param {Direction} [direction=null]
* @return {Range}
*/
static create(headSection, headOffset, tailSection = headSection, tailOffset = headOffset, direction = null) {
return new Range(new Position(headSection, headOffset), new Position(tailSection, tailOffset), direction);
}
static blankRange() {
return new Range(Position.blankPosition(), Position.blankPosition());
}
/**
* @param {Markerable} section
* @return {Range} A range that is constrained to only the part that
* includes the section.
* FIXME -- if the section isn't the head or tail, it's assumed to be
* wholly contained. It's possible to call `trimTo` with a selection that is
* outside of the range, though, which would invalidate that assumption.
* There's no efficient way to determine if a section is within a range, yet.
* @private
*/
trimTo(section) {
const length = section.length;
let headOffset = section === this.head.section ? Math.min(this.head.offset, length) : 0;
let tailOffset = section === this.tail.section ? Math.min(this.tail.offset, length) : length;
return Range.create(section, headOffset, section, tailOffset);
}
/**
* Expands the range 1 unit in the given direction
* If the range is expandable in the given direction, always returns a
* non-collapsed range.
* @param {Number} units If units is > 0, the range is extended to the right,
* otherwise range is extended to the left.
* @return {Range}
* @public
*/
extend(units) {
assert(`Must pass integer to Range#extend`, typeof units === 'number');
if (units === 0) {
return this;
}
let { head, tail, direction: currentDirection } = this;
switch (currentDirection) {
case Direction.FORWARD:
return new Range(head, tail.move(units), currentDirection);
case Direction.BACKWARD:
return new Range(head.move(units), tail, currentDirection);
default: {
let newDirection = units > 0 ? Direction.FORWARD : Direction.BACKWARD;
return new Range(head, tail, newDirection).extend(units);
}
}
}
/**
* Moves this range 1 unit in the given direction.
* If the range is collapsed, returns a collapsed range shifted by 1 unit,
* otherwise collapses this range to the position at the `direction` end of the range.
* Always returns a collapsed range.
* @param {Direction} direction
* @return {Range}
* @public
*/
move(direction) {
assert(`Must pass DIRECTION.FORWARD (${Direction.FORWARD}) or DIRECTION.BACKWARD (${Direction.BACKWARD}) to Range#move`, direction === Direction.FORWARD || direction === Direction.BACKWARD);
let { focusedPosition, isCollapsed } = this;
if (isCollapsed) {
return new Range(focusedPosition.move(direction));
}
else {
return this._collapse(direction);
}
}
/**
* expand a range to all markers matching a given check
*
* @param {Function} detectMarker
* @return {Range} The expanded range
*
* @public
*/
expandByMarker(detectMarker) {
let { head, tail, direction } = this;
let { section: headSection } = head;
assertNotNull('expected range section to not be null', headSection);
assertMarkerable(headSection);
if (headSection !== tail.section) {
throw new Error('#expandByMarker does not work across sections. Perhaps you should confirm the range is collapsed');
}
let firstNotMatchingDetect = (i) => {
return !detectMarker(i);
};
let headMarker = headSection.markers.detect(firstNotMatchingDetect, head.marker, true);
if (!headMarker && detectMarker(headSection.markers.head)) {
headMarker = headSection.markers.head;
}
else {
headMarker = unwrap(headMarker).next || head.marker;
}
let headPosition = new Position(headSection, headSection.offsetOfMarker(unwrap(headMarker)));
assertMarkerable(tail.section);
let tailMarker = tail.section.markers.detect(firstNotMatchingDetect, tail.marker);
if (!tailMarker && detectMarker(unwrap(headSection.markers.tail))) {
tailMarker = unwrap(headSection.markers.tail);
}
else {
tailMarker = unwrap(tailMarker).prev || unwrap(tail.marker);
}
let tailPosition = new Position(tail.section, tail.section.offsetOfMarker(tailMarker) + tailMarker.length);
return headPosition.toRange(tailPosition, direction);
}
_collapse(direction) {
return new Range(direction === Direction.BACKWARD ? this.head : this.tail);
}
get focusedPosition() {
return this.direction === Direction.BACKWARD ? this.head : this.tail;
}
isEqual(other) {
return other && this.head.isEqual(other.head) && this.tail.isEqual(other.tail);
}
get isBlank() {
return this.head.isBlank && this.tail.isBlank;
}
// "legacy" APIs
get headSection() {
return this.head.section;
}
get tailSection() {
return this.tail.section;
}
get headSectionOffset() {
return this.head.offset;
}
get tailSectionOffset() {
return this.tail.offset;
}
get isCollapsed() {
return this.head.isEqual(this.tail);
}
get headMarker() {
return this.head.marker;
}
get tailMarker() {
return this.tail.marker;
}
get headMarkerOffset() {
return this.head.offsetInMarker;
}
get tailMarkerOffset() {
return this.tail.offsetInMarker;
}
}
function assertMarkerable(section) {
if (!('markers' in section)) {
throw new MobiledocError('Expected position section to be markerable');
}
}
class LinkedItem {
constructor() {
this.next = null;
this.prev = null;
}
}
function isListSection(item) {
return 'items' in item && item.items;
}
class Section extends LinkedItem {
constructor(type) {
super();
this.isSection = true;
this.isMarkerable = false;
this.isNested = false;
this.isListItem = false;
this.isListSection = false;
this.isLeafSection = true;
this.isCardSection = false;
this._parent = null;
assert('Cannot create section without type', !!type);
this.type = type;
}
get parent() {
return expect(this._parent, 'expected section parent to be assigned');
}
get isBlank() {
return false;
}
get length() {
return 0;
}
/**
* @return {Position} The position at the start of this section
* @public
*/
headPosition() {
return this.toPosition(0);
}
/**
* @return {Position} The position at the end of this section
* @public
*/
tailPosition() {
return this.toPosition(this.length);
}
/**
* @param {Number} offset
* @return {Position} The position in this section at the given offset
* @public
*/
toPosition(offset) {
assert('Must pass number to `toPosition`', typeof offset === 'number');
assert('Cannot call `toPosition` with offset > length', offset <= this.length);
return new Position(this, offset);
}
/**
* @return {Range} A range from this section's head to tail positions
* @public
*/
toRange() {
return this.headPosition().toRange(this.tailPosition());
}
/**
* Markerable sections should override this method
*/
// eslint-disable-next-line @typescript-eslint/no-unused-vars
splitMarkerAtOffset(_offset) {
let blankEdit = { added: [], removed: [] };
return blankEdit;
}
nextLeafSection() {
const next = this.next;
if (next) {
if (isListSection(next)) {
return next.items.head;
}
else {
return next;
}
}
else {
if (isNested(this)) {
return this.parent.nextLeafSection();
}
}
return null;
}
immediatelyNextMarkerableSection() {
let next = this.nextLeafSection();
while (next && !next.isMarkerable) {
next = next.nextLeafSection();
}
return next;
}
previousLeafSection() {
const prev = this.prev;
if (prev) {
if (isListSection(prev)) {
return prev.items.tail;
}
else {
return prev;
}
}
else {
if (isNested(this)) {
return this.parent.previousLeafSection();
}
}
return null;
}
}
function isNested(section) {
return section.isNested;
}
function shallowCopyObject(object) {
return { ...object };
}
var CardMode;
(function (CardMode) {
CardMode["DISPLAY"] = "display";
CardMode["EDIT"] = "edit";
})(CardMode || (CardMode = {}));
const CARD_LENGTH = 1;
function isCardSection(section) {
return section.isCardSection;
}
class Card extends Section {
constructor(name, payload) {
super("card-section" /* CARD */);
this._initialMode = CardMode.DISPLAY;
this.isCardSection = true;
this.name = name;
this.payload = payload;
this.isCardSection = true;
}
textUntil() {
return '';
}
canJoin() {
return false;
}
get length() {
return CARD_LENGTH;
}
clone() {
let payload = shallowCopyObject(this.payload);
let card = this.builder.createCardSection(this.name, payload);
// If this card is currently rendered, clone the mode it is
// currently in as the default mode of the new card.
let mode = this._initialMode;
if (this.renderNode && this.renderNode.cardNode) {
mode = this.renderNode.cardNode.mode;
}
card.setInitialMode(mode);
return card;
}
/**
* set the mode that this will be rendered into initially
* @private
*/
setInitialMode(initialMode) {
// TODO validate initialMode
this._initialMode = initialMode;
}
}
const ATOM_LENGTH = 1;
class Atom extends Markuperable {
constructor(name, value, payload, markups = []) {
super();
this.type = "atom" /* ATOM */;
this.isAtom = true;
this.name = name;
this.value = value;
this.text = ''; // An atom never has text, but it does have a value
assert('Atom must have value', value !== undefined && value !== null);
this.payload = payload;
this.type = "atom" /* ATOM */;
this.isMarker = false;
this.isAtom = true;
this.markups = [];
markups.forEach(m => this.addMarkup(m));
}
clone() {
let clonedMarkups = this.markups.slice();
return this.builder.createAtom(this.name, this.value, this.payload, clonedMarkups);
}
get isBlank() {
return false;
}
get length() {
return ATOM_LENGTH;
}
canJoin( /* other */) {
return false;
}
textUntil( /* offset */) {
return '';
}
split(offset = 0, endOffset = offset) {
let markers = [];
if (endOffset === 0) {
markers.push(this.builder.createMarker('', this.markups.slice()));
}
markers.push(this.clone());
if (offset === ATOM_LENGTH) {
markers.push(this.builder.createMarker('', this.markups.slice()));
}
return markers;
}
splitAtOffset(offset) {
assert('Cannot split a marker at an offset > its length', offset <= this.length);
let { builder } = this;
let clone = this.clone();
let blankMarker = builder.createMarker('');
let pre, post;
if (offset === 0) {
[pre, post] = [blankMarker, clone];
}
else if (offset === ATOM_LENGTH) {
[pre, post] = [clone, blankMarker];
}
else {
assert(`Invalid offset given to Atom#splitAtOffset: "${offset}"`, false);
}
this.markups.forEach(markup => {
pre.addMarkup(markup);
post.addMarkup(markup);
});
return [pre, post];
}
}
function isAtom(postNode) {
return postNode.type === "atom" /* ATOM */;
}
const { FORWARD, BACKWARD } = Direction;
// generated via http://xregexp.com/ to cover chars that \w misses
// (new XRegExp('\\p{Alphabetic}|[0-9]|_|:')).toString()
// eslint-disable-next-line no-misleading-character-class
const WORD_CHAR_REGEX = /[A-Za-zªµºÀ-ÖØ-öø-ˁˆ-ˑˠ-ˤˬˮͅͰ-ʹͶͷͺ-ͽͿΆΈ-ΊΌΎ-ΡΣ-ϵϷ-ҁҊ-ԯԱ-Ֆՙա-ևְ-ׇֽֿׁׂׅׄא-תװ-ײؐ-ؚؠ-ٗٙ-ٟٮ-ۓە-ۜۡ-ۭۨ-ۯۺ-ۼۿܐ-ܿݍ-ޱߊ-ߪߴߵߺࠀ-ࠗࠚ-ࠬࡀ-ࡘࢠ-ࢴࣣ-ࣰࣩ-ऻऽ-ौॎ-ॐॕ-ॣॱ-ঃঅ-ঌএঐও-নপ-রলশ-হঽ-ৄেৈোৌৎৗড়ঢ়য়-ৣৰৱਁ-ਃਅ-ਊਏਐਓ-ਨਪ-ਰਲਲ਼ਵਸ਼ਸਹਾ-ੂੇੈੋੌੑਖ਼-ੜਫ਼ੰ-ੵઁ-ઃઅ-ઍએ-ઑઓ-નપ-રલળવ-હઽ-ૅે-ૉોૌૐૠ-ૣૹଁ-ଃଅ-ଌଏଐଓ-ନପ-ରଲଳଵ-ହଽ-ୄେୈୋୌୖୗଡ଼ଢ଼ୟ-ୣୱஂஃஅ-ஊஎ-ஐஒ-கஙசஜஞடணதந-பம-ஹா-ூெ-ைொ-ௌௐௗఀ-ఃఅ-ఌఎ-ఐఒ-నప-హఽ-ౄె-ైొ-ౌౕౖౘ-ౚౠ-ౣಁ-ಃಅ-ಌಎ-ಐಒ-ನಪ-ಳವ-ಹಽ-ೄೆ-ೈೊ-ೌೕೖೞೠ-ೣೱೲഁ-ഃഅ-ഌഎ-ഐഒ-ഺഽ-ൄെ-ൈൊ-ൌൎൗൟ-ൣൺ-ൿංඃඅ-ඖක-නඳ-රලව-ෆා-ුූෘ-ෟෲෳก-ฺเ-ๆํກຂຄງຈຊຍດ-ທນ-ຟມ-ຣລວສຫອ-ູົ-ຽເ-ໄໆໍໜ-ໟༀཀ-ཇཉ-ཬཱ-ཱྀྈ-ྗྙ-ྼက-ံးျ-ဿၐ-ၢၥ-ၨၮ-ႆႎႜႝႠ-ჅჇჍა-ჺჼ-ቈቊ-ቍቐ-ቖቘቚ-ቝበ-ኈኊ-ኍነ-ኰኲ-ኵኸ-ኾዀዂ-ዅወ-ዖዘ-ጐጒ-ጕጘ-ፚ፟ᎀ-ᎏᎠ-Ᏽᏸ-ᏽᐁ-ᙬᙯ-ᙿᚁ-ᚚᚠ-ᛪᛮ-ᛸᜀ-ᜌᜎ-ᜓᜠ-ᜳᝀ-ᝓᝠ-ᝬᝮ-ᝰᝲᝳក-ឳា-ៈៗៜᠠ-ᡷᢀ-ᢪᢰ-ᣵᤀ-ᤞᤠ-ᤫᤰ-ᤸᥐ-ᥭᥰ-ᥴᦀ-ᦫᦰ-ᧉᨀ-ᨛᨠ-ᩞᩡ-ᩴᪧᬀ-ᬳᬵ-ᭃᭅ-ᭋᮀ-ᮩᮬ-ᮯᮺ-ᯥᯧ-ᯱᰀ-ᰵᱍ-ᱏᱚ-ᱽᳩ-ᳬᳮ-ᳳᳵᳶᴀ-ᶿᷧ-ᷴḀ-ἕἘ-Ἕἠ-ὅὈ-Ὅὐ-ὗὙὛὝὟ-ώᾀ-ᾴᾶ-ᾼιῂ-ῄῆ-ῌῐ-ΐῖ-Ίῠ-Ῥῲ-ῴῶ-ῼⁱⁿₐ-ₜℂℇℊ-ℓℕℙ-ℝℤΩℨK-ℭℯ-ℹℼ-ℿⅅ-ⅉⅎⅠ-ↈⒶ-ⓩⰀ-Ⱞⰰ-ⱞⱠ-ⳤⳫ-ⳮⳲⳳⴀ-ⴥⴧⴭⴰ-ⵧⵯⶀ-ⶖⶠ-ⶦⶨ-ⶮⶰ-ⶶⶸ-ⶾⷀ-ⷆⷈ-ⷎⷐ-ⷖⷘ-ⷞⷠ-ⷿⸯ々-〇〡-〩〱-〵〸-〼ぁ-ゖゝ-ゟァ-ヺー-ヿㄅ-ㄭㄱ-ㆎㆠ-ㆺㇰ-ㇿ㐀-䶵一-鿕ꀀ-ꒌꓐ-ꓽꔀ-ꘌꘐ-ꘟꘪꘫꙀ-ꙮꙴ-ꙻꙿ-ꛯꜗ-ꜟꜢ-ꞈꞋ-ꞭꞰ-ꞷꟷ-ꠁꠃ-ꠅꠇ-ꠊꠌ-ꠧꡀ-ꡳꢀ-ꣃꣲ-ꣷꣻꣽꤊ-ꤪꤰ-ꥒꥠ-ꥼꦀ-ꦲꦴ-ꦿꧏꧠ-ꧤꧦ-ꧯꧺ-ꧾꨀ-ꨶꩀ-ꩍꩠ-ꩶꩺꩾ-ꪾꫀꫂꫛ-ꫝꫠ-ꫯꫲ-ꫵꬁ-ꬆꬉ-ꬎꬑ-ꬖꬠ-ꬦꬨ-ꬮꬰ-ꭚꭜ-ꭥꭰ-ꯪ가-힣ힰ-ퟆퟋ-ퟻ豈-舘並-龎ff-stﬓ-ﬗיִ-ﬨשׁ-זּטּ-לּמּנּסּףּפּצּ-ﮱﯓ-ﴽﵐ-ﶏﶒ-ﷇﷰ-ﷻﹰ-ﹴﹶ-ﻼA-Za-zヲ-하-ᅦᅧ-ᅬᅭ-ᅲᅳ-ᅵ]|[0-9]|_|:/;
function findParentSectionFromNode(renderTree, node) {
let renderNode = renderTree.findRenderNodeFromElement(node, renderNode => renderNode.postNode.isSection);
return renderNode && renderNode.postNode;
}
function findOffsetInMarkerable(markerable, node, offset = 0) {
let offsetInSection = 0;
let marker = markerable.markers.head;
while (marker) {
assertHasRenderNode(marker.renderNode);
let markerNode = marker.renderNode.element;
if (markerNode === node) {
return offsetInSection + offset;
}
else if (marker.isAtom) {
if (marker.renderNode.headTextNode === node) {
return offsetInSection;
}
else if (marker.renderNode.tailTextNode === node) {
return offsetInSection + 1;
}
}
offsetInSection += marker.length;
marker = marker.next;
}
return offsetInSection;
}
function assertHasRenderNode(renderNode) {
if (!renderNode) {
throw new Error('expected marker to have render node');
}
}
function findOffsetInSection(section, node, offset) {
if (isMarkerable(section)) {
return findOffsetInMarkerable(section, node, offset);
}
else {
assertIsCard(section);
assertHasRenderNode(section.renderNode);
let wrapperNode = section.renderNode.element;
let endTextNode = wrapperNode.lastChild;
if (node === endTextNode) {
return 1;
}
return 0;
}
}
function assertIsCard(section) {
assert('findOffsetInSection must be called with markerable or card section', section && section.isCardSection);
}
function isMarkerable(section) {
return section.isMarkerable;
}
class Position {
/**
* A position is a logical location (zero-width, or "collapsed") in a post,
* typically between two characters in a section.
* Two positions (a head and a tail) make up a {@link Range}.
* @constructor
*/
constructor(section, offset = 0, isBlank = false) {
if (!isBlank) {
assert('Position must have a section that is addressable by the cursor', section && section.isLeafSection);
assert('Position must have numeric offset', typeof offset === 'number');
}
this.section = section;
this.offset = offset;
this.isBlank = isBlank;
}
/**
* @param {integer} x x-position in current viewport
* @param {integer} y y-position in current viewport
* @param {Editor} editor
* @return {Position|null}
*/
static atPoint(x, y, editor) {
let { _renderTree, element: rootElement } = editor;
let elementFromPoint = document.elementFromPoint(x, y);
if (!elementFromPoint || !containsNode(rootElement, elementFromPoint)) {
return null;
}
let { node, offset } = findOffsetInNode(elementFromPoint, { left: x, top: y });
return Position.fromNode(_renderTree, node, offset);
}
static blankPosition() {
return new BlankPosition();
}
/**
* Returns a range from this position to the given tail. If no explicit
* tail is given this returns a collapsed range focused on this position.
* @param {Position} [tail=this] The ending position
* @return {Range}
* @public
*/
toRange(tail = this, direction = null) {
return new Range(this, tail, direction);
}
get leafSectionIndex() {
let post = this.section.post;
let leafSectionIndex;
post.walkAllLeafSections((section, index) => {
if (section === this.section) {
leafSectionIndex = index;
}
});
return leafSectionIndex;
}
get isMarkerable() {
return this.section && this.section.isMarkerable;
}
/**
* Returns the marker at this position, in the backward direction
* (i.e., the marker to the left of the cursor if the cursor is on a marker boundary and text is left-to-right)
* @return {Marker|undefined}
*/
get marker() {
return (this.isMarkerable && this.markerPosition.marker) || null;
}
/**
* Returns the marker in `direction` from this position.
* If the position is in the middle of a marker, the direction is irrelevant.
* Otherwise, if the position is at a boundary between two markers, returns the
* marker to the left if `direction` === BACKWARD and the marker to the right
* if `direction` === FORWARD (assuming left-to-right text direction).
* @param {Direction}
* @return {Marker|undefined}
*/
markerIn(direction) {
if (!this.isMarkerable) {
return;
}
let { marker, offsetInMarker } = this;
if (!marker) {
return;
}
if (offsetInMarker > 0 && offsetInMarker < marker.length) {
return marker;
}
else if (offsetInMarker === 0) {
return direction === BACKWARD ? marker : marker.prev;
}
else if (offsetInMarker === marker.length) {
return direction === FORWARD ? marker.next : marker;
}
}
get offsetInMarker() {
return this.markerPosition.offset;
}
isEqual(position) {
return this.section === position.section && this.offset === position.offset;
}
/**
* @return {Boolean} If this position is at the head of the post
*/
isHeadOfPost() {
return this.move(BACKWARD).isEqual(this);
}
/**
* @return {Boolean} If this position is at the tail of the post
*/
isTailOfPost() {
return this.move(FORWARD).isEqual(this);
}
/**
* @return {Boolean} If this position is at the head of its section
*/
isHead() {
return this.isEqual(this.section.headPosition());
}
/**
* @return {Boolean} If this position is at the tail of its section
*/
isTail() {
return this.isEqual(this.section.tailPosition());
}
/**
* Move the position 1 unit in `direction`.
*
* @param {Number} units to move. > 0 moves right, < 0 moves left
* @return {Position} Return a new position one unit in the given
* direction. If the position is moving left and at the beginning of the post,
* the same position will be returned. Same if the position is moving right and
* at the end of the post.
*/
move(units) {
assert('Must pass integer to Position#move', typeof units === 'number');
if (units < 0) {
return this.moveLeft().move(++units);
}
else if (units > 0) {
return this.moveRight().move(--units);
}
else {
return this;
}
}
/**
* @param {Number} direction (FORWARD or BACKWARD)
* @return {Position} The result of moving 1 "word" unit in `direction`
*/
moveWord(direction) {
let isPostBoundary = direction === BACKWARD ? this.isHeadOfPost() : this.isTailOfPost();
if (isPostBoundary) {
return this;
}
if (!this.isMarkerable) {
return this.move(direction);
}
let pos = this;
// Helper fn to check if the pos is at the `dir` boundary of its section
let isBoundary = (pos, dir) => {
return dir === BACKWARD ? pos.isHead() : pos.isTail();
};
// Get the char at this position (looking forward/right)
let getChar = (pos) => {
let { marker, offsetInMarker } = pos;
return marker.charAt(offsetInMarker);
};
// Get the char in `dir` at this position
let peekChar = (pos, dir) => {
return dir === BACKWARD ? getChar(pos.move(BACKWARD)) : getChar(pos);
};
// Whether there is an atom in `dir` from this position
let isAtom = (pos, dir) => {
// Special case when position is at end, the marker associated with it is
// the marker to its left. Normally `pos#marker` is the marker to the right of the pos's offset.
if (dir === BACKWARD && pos.isTail() && pos.marker.isAtom) {
return true;
}
return dir === BACKWARD ? pos.move(BACKWARD).marker.isAtom : pos.marker.isAtom;
};
if (isBoundary(pos, direction)) {
// extend movement into prev/next section
return pos.move(direction).moveWord(direction);
}
let seekWord = (pos) => {
return !isBoundary(pos, direction) && !isAtom(pos, direction) && !WORD_CHAR_REGEX.test(peekChar(pos, direction));
};
// move(dir) while we are seeking the first word char
while (seekWord(pos)) {
pos = pos.move(direction);
}
if (isAtom(pos, direction)) {
return pos.move(direction);
}
let seekBoundary = (pos) => {
return !isBoundary(pos, direction) && !isAtom(pos, direction) && WORD_CHAR_REGEX.test(peekChar(pos, direction));
};
// move(dir) while we are seeking the first boundary position
while (seekBoundary(pos)) {
pos = pos.move(direction);
}
return pos;
}
/**
* The position to the left of this position.
* If this position is the post's headPosition it returns itself.
* @return {Position}
* @private
*/
moveLeft() {
if (this.isHead()) {
let prev = this.section.previousLeafSection();
return prev ? prev.tailPosition() : this;
}
else {
let offset = this.offset - 1;
if (this.isMarkerable && this.marker) {
let code = this.marker.value.charCodeAt(offset);
if (code >= LOW_SURROGATE_RANGE[0] && code <= LOW_SURROGATE_RANGE[1]) {
offset = offset - 1;
}
}
return new Position(this.section, offset);
}
}
/**
* The position to the right of this position.
* If this position is the post's tailPosition it returns itself.
* @return {Position}
* @private
*/
moveRight() {
if (this.isTail()) {
let next = this.section.nextLeafSection();
return next ? next.headPosition() : this;
}
else {
let offset = this.offset + 1;
if (this.isMarkerable && this.marker) {
let code = this.marker.value.charCodeAt(offset - 1);
if (code >= HIGH_SURROGATE_RANGE[0] && code <= HIGH_SURROGATE_RANGE[1]) {
offset = offset + 1;
}
}
return new Position(this.section, offset);
}
}
static fromNode(renderTree, node, offset) {
if (isTextNode(node)) {
return Position.fromTextNode(renderTree, node, offset);
}
else if (isElementNode(node)) {
return Position.fromElementNode(renderTree, node, offset);
}
assert('Positions can only be created from text nodes or elements', false);
}
static fromTextNode(renderTree, textNode, offsetInNode) {
const renderNode = renderTree.getElementRenderNode(textNode);
let section, offsetInSection;
if (renderNode) {
const marker = renderNode.postNode;
section = marker.section;
assert(`Could not find parent section for mapped text node "${textNode.textContent}"`, !!section);
offsetInSection = marker.section.offsetOfMarker(marker, offsetInNode);
}
else {
// all text nodes should be rendered by markers except:
// * text nodes inside cards
// * text nodes created by the browser during text input
// both of these should have rendered parent sections, though
section = findParentSectionFromNode(renderTree, textNode);
assert(`Could not find parent section for un-mapped text node "${textNode.textContent}"`, !!section);
offsetInSection = findOffsetInSection(section, textNode, offsetInNode);
}
return new Position(section, offsetInSection);
}
static fromElementNode(renderTree, elementNode, offset = 0) {
let position;
// The browser may change the reported selection to equal the editor's root
// element if the user clicks an element that is immediately removed,
// which can happen when clicking to remove a card.
if (elementNode === renderTree.rootElement) {
let post = renderTree.rootNode.postNode;
position = offset === 0 ? post.headPosition() : post.tailPosition();
}
else {
let section = findParentSectionFromNode(renderTree, elementNode);
assert('Could not find parent section from element node', !!section);
if (isCardSection(section)) {
// Selections in cards are usually made on a text node
// containing a on one side or the other of the card but
// some scenarios (Firefox) will result in selecting the
// card's wrapper div. If the offset is 2 we've selected
// the final zwnj and should consider the cursor at the
// end of the card (offset 1). Otherwise, the cursor is at
// the start of the card
position = offset < 2 ? section.headPosition() : section.tailPosition();
}
else {
// In Firefox it is possible for the cursor to be on an atom's wrapper
// element. (In Chrome/Safari, the browser corrects this to be on
// one of the text nodes surrounding the wrapper).
// This code corrects for when the browser reports the cursor position
// to be on the wrapper element itself
let renderNode = renderTree.getElementRenderNode(elementNode);
let postNode = renderNode && renderNode.postNode;
if (postNode && isAtom(postNode)) {
let sectionOffset = section.offsetOfMarker(postNode);
if (offset > 1) {
// we are on the tail side of the atom
sectionOffset += postNode.length;
}
position = new Position(section, sectionOffset);
}
else if (offset >= elementNode.childNodes.length) {
// This is to deal with how Firefox handles triple-click selections.
// See https://stackoverflow.com/a/21234837/1269194 for an
// explanation.
position = section.tailPosition();
}
else {
// The offset is 0 if the cursor is on a non-atom-wrapper element node
// (e.g., a
tag in a blank markup section)
position = section.headPosition();
}
}
}
return position;
}
/**
* @private
*/
get markerPosition() {
assert('Cannot get markerPosition without a section', !!this.section);
assertType('cannot get markerPosition of a non-markerable', this.section, !!this.section.isMarkerable);
return this.section.markerPositionAtOffset(this.offset);
}
}
class BlankPosition extends Position {
constructor() {
super(null, 0, true);
}
isEqual(other) {
return other && other.isBlank;
}
toRange() {
return Range.blankRange();
}
get leafSectionIndex() {
throw new Error('must implement get leafSectionIndex');
}
get isMarkerable() {
return false;
}
get marker() {
return null;
}
isHeadOfPost() {
return false;
}
isTailOfPost() {
return false;
}
isHead() {
return false;
}
isTail() {
return false;
}
move() {
return this;
}
moveWord() {
return this;
}
get markerPosition() {
return {};
}
}
/**
* @module UI
*/
const defaultShowPrompt = (message, defaultValue, callback) => callback(window.prompt(message, defaultValue));
/**
* @callback promptCallback
* @param {String} url The URL to pass back to the editor for linking
* to the selected text.
*/
/**
* @callback showPrompt
* @param {String} message The text of the prompt.
* @param {String} defaultValue The initial URL to display in the prompt.
* @param {module:UI~promptCallback} callback Once your handler has accepted a URL,
* it should pass it to `callback` so that the editor may link the
* selected text.
*/
/**
* Exposes the core behavior for linking and unlinking text, and allows for
* customization of the URL input handler.
* @param {Editor} editor An editor instance to operate on. If a range is selected,
* either prompt for a URL and add a link or un-link the
* currently linked text.
* @param {module:UI~showPrompt} [showPrompt] An optional custom input handler. Defaults
* to using `window.prompt`.
* @example
* let myPrompt = (message, defaultURL, promptCallback) => {
* let url = window.prompt("Overriding the defaults", "http://placekitten.com");
* promptCallback(url);
* };
*
* editor.registerKeyCommand({
* str: "META+K",
* run(editor) {
* toggleLink(editor, myPrompt);
* }
* });
* @public
*/
function toggleLink(editor, showPrompt = defaultShowPrompt) {
if (editor.range.isCollapsed) {
return;
}
let selectedText = editor.cursor.selectedText();
let defaultUrl = '';
if (selectedText.indexOf('http') !== -1) {
defaultUrl = selectedText;
}
let { range } = editor;
let hasLink = editor.detectMarkupInRange(range, 'a');
if (hasLink) {
editor.toggleMarkup('a');
}
else {
showPrompt('Enter a URL', defaultUrl, url => {
if (!url) {
return;
}
editor.toggleMarkup('a', { href: url });
});
}
}
/**
* Exposes the core behavior for editing an existing link, and allows for
* customization of the URL input handler.
* @param {HTMLAnchorElement} target The anchor () DOM element whose URL should be edited.
* @param {Editor} editor An editor instance to operate on. If a range is selected,
* either prompt for a URL and add a link or un-link the
* currently linked text.
* @param {module:UI~showPrompt} [showPrompt] An optional custom input handler. Defaults
* to using `window.prompt`.
*
* @public
*/
function editLink(target, editor, showPrompt = defaultShowPrompt) {
showPrompt('Enter a URL', target.href, url => {
if (!url) {
return;
}
const position = Position.fromNode(editor._renderTree, target.firstChild);
const range = new Range(position, new Position(position.section, position.offset + target.textContent.length));
editor.run(post => {
let markup = editor.builder.createMarkup('a', { href: url });
// This is the only way to "update" a markup with new attributes in the
// current API.
post.toggleMarkup(markup, range);
post.toggleMarkup(markup, range);
});
});
}
var ui = {
toggleLink,
editLink,
};
const SHOW_DELAY = 200;
const HIDE_DELAY = 600;
class Tooltip extends View {
constructor(options) {
super({ ...options, classNames: ['__mobiledoc-tooltip'] });
this.elementObserver = null;
this.rootElement = options.rootElement;
this.editor = options.editor;
this.addListeners(options);
}
showLink(linkEl) {
const { editor, element: tooltipEl } = this;
const { tooltipPlugin } = editor;
tooltipPlugin.renderLink(tooltipEl, linkEl, {
editLink: () => {
editLink(linkEl, editor);
this.hide();
},
});
this.show();
positionElementCenteredBelow(this.element, linkEl);
this.elementObserver = whenElementIsNotInDOM(linkEl, () => this.hide());
}
addListeners(options) {
const { rootElement, element: tooltipElement } = this;
let showTimeout, hideTimeout;
const scheduleHide = () => {
clearTimeout(hideTimeout);
hideTimeout = setTimeout(() => {
this.hide();
}, HIDE_DELAY);
};
this.addEventListener(tooltipElement, 'mouseenter', () => {
clearTimeout(hideTimeout);
});
this.addEventListener(tooltipElement, 'mouseleave', () => {
scheduleHide();
});
this.addEventListener(rootElement, 'mouseover', event => {
let target = getEventTargetMatchingTag(options.showForTag, event.target, rootElement);
if (target && target.isContentEditable) {
clearTimeout(hideTimeout);
showTimeout = setTimeout(() => {
target && this.showLink(target);
}, SHOW_DELAY);
}
});
this.addEventListener(rootElement, 'mouseout', () => {
clearTimeout(showTimeout);
if (this.elementObserver) {
this.elementObserver.cancel();
}
scheduleHide();
});
}
}
const DEFAULT_TOOLTIP_PLUGIN = {
renderLink(tooltipEl, linkEl, { editLink }) {
const { href } = linkEl;
tooltipEl.innerHTML = `${href}`;
const button = document.createElement('button');
button.classList.add('__mobiledoc-tooltip__edit-link');
button.innerText = 'Edit Link';
button.addEventListener('click', editLink);
tooltipEl.append(button);
},
};
class LifecycleCallbacks {
constructor(queueNames = []) {
this.callbackQueues = {};
this.removalQueues = {};
queueNames.forEach(name => {
this.callbackQueues[name] = [];
this.removalQueues[name] = [];
});
}
runCallbacks(queueName, args = []) {
let queue = this._getQueue(queueName);
queue.forEach(cb => cb(...args));
let toRemove = this.removalQueues[queueName];
toRemove.forEach(cb => {
let index = queue.indexOf(cb);
if (index !== -1) {
queue.splice(index, 1);
}
});
this.removalQueues[queueName] = [];
}
addCallback(queueName, callback) {
this._getQueue(queueName).push(callback);
}
_scheduleCallbackForRemoval(queueName, callback) {
this.removalQueues[queueName].push(callback);
}
addCallbackOnce(queueName, callback) {
let queue = this._getQueue(queueName);
if (queue.indexOf(callback) === -1) {
queue.push(callback);
this._scheduleCallbackForRemoval(queueName, callback);
}
}
_getQueue(queueName) {
let queue = this.callbackQueues[queueName];
assert(`No queue found for "${queueName}"`, !!queue);
return queue;
}
}
function hasChildSections(section) {
return 'sections' in section;
}
const MARKERABLE = 'markerable';
const NESTED_MARKERABLE = 'nested_markerable';
const NON_MARKERABLE = 'non_markerable';
class Visitor {
constructor({ postEditor, post }, cursorPosition) {
this.postEditor = postEditor;
this._post = post;
this.cursorPosition = cursorPosition;
this.builder = this.postEditor.builder;
this._hasInsertedFirstLeafSection = false;
}
get cursorPosition() {
return this._cursorPosition;
}
set cursorPosition(position) {
this._cursorPosition = position;
this.postEditor.setRange(position);
}
visit(node) {
let method = node.type;
assertType(`Cannot visit node of type ${node.type}`, method, method in this);
this[method](node);
}
_canMergeSection(section) {
if (this._hasInsertedFirstLeafSection) {
return false;
}
else {
return this._isMarkerable && section.isMarkerable;
}
}
get _isMarkerable() {
return this.cursorSection.isMarkerable;
}
get cursorSection() {
return this.cursorPosition.section;
}
get cursorOffset() {
return this.cursorPosition.offset;
}
get _isNested() {
return this.cursorSection.isNested;
}
["post" /* POST */](node) {
let { cursorSection } = this;
if (cursorSection.isBlank && !cursorSection.isNested) {
// replace blank section with entire post
let newSections = node.sections.map(s => s.clone());
this._replaceSection(cursorSection, newSections);
}
else {
node.sections.forEach(section => this.visit(section));
}
}
["markup-section" /* MARKUP_SECTION */](node) {
this[MARKERABLE](node);
}
["list-section" /* LIST_SECTION */](node) {
let hasNext = !!node.next;
node.items.forEach(item => this.visit(item));
if (this._isNested && hasNext) {
this._breakNestedAtCursor();
}
}
["list-item" /* LIST_ITEM */](node) {
this[NESTED_MARKERABLE](node);
}
["card-section" /* CARD */](node) {
this[NON_MARKERABLE](node);
}
["image-section" /* IMAGE_SECTION */](node) {
this[NON_MARKERABLE](node);
}
[NON_MARKERABLE](section) {
if (this._isNested) {
this._breakNestedAtCursor();
}
else if (!this.cursorSection.isBlank) {
this._breakAtCursor();
}
this._insertLeafSection(section);
}
[MARKERABLE](section) {
if (this._canMergeSection(section)) {
this._mergeSection(section);
}
else if (this._isNested && this._isMarkerable) {
// If we are attaching a markerable section to a list item,
// insert a linebreak then merge the section onto the resulting blank list item
this._breakAtCursor();
// Advance the cursor to the head of the blank list item
let nextPosition = this.cursorSection.next.headPosition();
this.cursorPosition = nextPosition;
// Merge this section onto the list item
this._mergeSection(section);
}
else {
this._breakAtCursor();
this._insertLeafSection(section);
}
}
[NESTED_MARKERABLE](section) {
if (this._canMergeSection(section)) {
this._mergeSection(section);
return;
}
let insertedSection = this._isNested ? section : this._wrapNestedSection(section);
this._breakAtCursor();
this._insertLeafSection(insertedSection);
}
// break out of a nested cursor position
_breakNestedAtCursor() {
assert('Cannot call _breakNestedAtCursor if not nested', this._isNested);
let parent = this.cursorSection.parent;
let cursorAtEndOfList = this.cursorPosition.isEqual(parent.tailPosition());
if (cursorAtEndOfList) {
let blank = this.builder.createMarkupSection();
this._insertSectionAfter(blank, parent);
}
else {
let [, blank] = this._breakListAtCursor();
this.cursorPosition = blank.tailPosition();
}
}
_breakListAtCursor() {
assert('Cannot _splitParentSection if cursor position is not nested', this._isNested);
const list = this.cursorSection.parent;
const position = this.cursorPosition;
const blank = this.builder.createMarkupSection();
let [pre, post] = this.postEditor._splitListAtPosition(list, position);
let collection = this._post.sections, reference = post;
this.postEditor.insertSectionBefore(collection, blank, reference);
return [pre, blank, post];
}
_wrapNestedSection(section) {
let tagName = section.parent.tagName;
let parent = this.builder.createListSection(tagName);
parent.items.append(section.clone());
return parent;
}
_mergeSection(section) {
assert('Can only merge markerable sections', this._isMarkerable && section.isMarkerable);
this._hasInsertedFirstLeafSection = true;
let markers = section.markers.map(m => m.clone());
let position = this.postEditor.insertMarkers(this.cursorPosition, markers);
this.cursorPosition = position;
}
// Can be called to add a line break when in a nested section or a parent
// section.
_breakAtCursor() {
if (this.cursorSection.isBlank) {
return;
}
else if (this._isMarkerable) {
this._breakMarkerableAtCursor();
}
else {
this._breakNonMarkerableAtCursor();
}
}
// Inserts a blank section before/after the cursor,
// depending on cursor position.
_breakNonMarkerableAtCursor() {
const collection = this._post.sections;
const blank = this.builder.createMarkupSection();
const reference = this.cursorPosition.isHead() ? this.cursorSection : this.cursorSection.next;
this.postEditor.insertSectionBefore(collection, blank, reference);
this.cursorPosition = blank.tailPosition();
}
_breakMarkerableAtCursor() {
let [pre] = this.postEditor.splitSection(this.cursorPosition);
this.cursorPosition = pre.tailPosition();
}
_replaceSection(section, newSections) {
assert('Cannot replace section that does not have parent.sections', hasChildSections(section.parent));
assert('Must pass enumerable to _replaceSection', !!newSections.forEach);
let collection = section.parent.sections;
let reference = section.next;
this.postEditor.removeSection(section);
newSections.forEach(section => {
this.postEditor.insertSectionBefore(collection, section, reference);
});
let lastSection = newSections[newSections.length - 1];
this.cursorPosition = lastSection.tailPosition();
}
_insertSectionBefore(section, reference) {
assert('Cannot insert into section that does not have parent.sections', hasChildSections(this.cursorSection.parent));
let collection = this.cursorSection.parent.sections;
this.postEditor.insertSectionBefore(collection, section, reference);
this.cursorPosition = section.tailPosition();
}
// Insert a section after the parent section.
// E.g., add a markup section after a list section
_insertSectionAfter(section, parent) {
assert('Cannot _insertSectionAfter nested section', !parent.isNested);
let reference = parent.next;
let collection = this._post.sections;
this.postEditor.insertSectionBefore(collection, section, reference);
this.cursorPosition = section.tailPosition();
}
_insertLeafSection(section) {
assert('Can only _insertLeafSection when cursor is at end of section', this.cursorPosition.isTail());
this._hasInsertedFirstLeafSection = true;
section = section.clone();
if (this.cursorSection.isBlank) {
assert('Cannot insert leaf non-markerable section when cursor is nested', !(section.isMarkerable && this._isNested));
this._replaceSection(this.cursorSection, [section]);
}
else if (this.cursorSection.next && this.cursorSection.next.isBlank) {
this._replaceSection(this.cursorSection.next, [section]);
}
else {
let reference = this.cursorSection.next;
this._insertSectionBefore(section, reference);
}
}
}
class Inserter {
constructor(postEditor, post) {
this.postEditor = postEditor;
this.post = post;
}
insert(cursorPosition, newPost) {
let visitor = new Visitor(this, cursorPosition);
if (!newPost.isBlank) {
visitor.visit(newPost);
}
return visitor.cursorPosition;
}
}
/**
* Usage:
* Without a conditional, always prints deprecate message:
* `deprecate('This is deprecated')`
*
* Conditional deprecation, works similarly to `assert`, prints deprecation if
* conditional is false:
* `deprecate('Deprecated only if foo !== bar', foo === bar)`
*/
function deprecate(message, conditional = false) {
if (!conditional) {
// eslint-disable-next-line no-console
console.log(`[mobiledoc-kit] [DEPRECATED]: ${message}`);
}
}
function toRange(rangeLike) {
assert(`Must pass non-blank object to "toRange"`, !!rangeLike);
if (rangeLike instanceof Range) {
return rangeLike;
}
else if (rangeLike instanceof Position) {
return rangeLike.toRange();
}
assert(`Incorrect structure for rangeLike: ${rangeLike}`, false);
}
var Browser = {
isMac() {
return typeof window !== 'undefined' && window.navigator && /Mac/.test(window.navigator.platform);
},
isWin() {
return typeof window !== 'undefined' && window.navigator && /Win/.test(window.navigator.platform);
},
isChrome() {
return typeof window !== 'undefined' && 'chrome' in window;
},
};
class Cursor {
constructor(editor) {
this.editor = editor;
this.renderTree = editor._renderTree;
this.post = editor.post;
}
clearSelection() {
clearSelection(this.editor.root);
}
/**
* @return {Boolean} true when there is either a collapsed cursor in the
* editor's element or a selection that is contained in the editor's element
*/
hasCursor() {
return this.editor.hasRendered && (this._hasCollapsedSelection() || this._hasSelection());
}
hasSelection() {
return this.editor.hasRendered && this._hasSelection();
}
/**
* @return {Boolean} Can the cursor be on this element?
*/
isAddressable(element) {
let { renderTree } = this;
let renderNode = renderTree.findRenderNodeFromElement(element);
if (renderNode && renderNode.postNode.isCardSection) {
let renderedElement = renderNode.element;
// card sections have addressable text nodes containing
// as their first and last child
if (element !== renderedElement &&
element !== renderedElement.firstChild &&
element !== renderedElement.lastChild) {
return false;
}
}
return !!renderNode;
}
/*
* @return {Range} Cursor#Range object
*/
get offsets() {
if (!this.hasCursor()) {
return Range.blankRange();
}
let { renderTree } = this;
let parentNode = unwrap(this.editor.element);
let selection = constrainSelectionTo(this.selection, parentNode);
const { headNode, headOffset, tailNode, tailOffset, direction } = comparePosition(selection);
const headPosition = Position.fromNode(renderTree, headNode, headOffset);
const tailPosition = Position.fromNode(renderTree, tailNode, tailOffset);
return new Range(headPosition, tailPosition, direction);
}
_findNodeForPosition(position) {
let section = unwrap(position.section);
let node, offset;
assertNotNull('expected section to have render node', section.renderNode);
if (isCardSection(section)) {
offset = 0;
if (position.offset === 0) {
node = section.renderNode.element.firstChild;
}
else {
node = section.renderNode.element.lastChild;
}
}
else if (section.isBlank || section.type === "image-section" /* IMAGE_SECTION */) {
node = section.renderNode.cursorElement;
offset = 0;
}
else {
let { marker, offsetInMarker } = position;
assertNotNull('expected position to have marker', marker);
assertNotNull('expected marker to have render node', marker.renderNode);
if (marker.isAtom) {
if (offsetInMarker > 0) {
// FIXME -- if there is a next marker, focus on it?
offset = 0;
node = marker.renderNode.tailTextNode;
}
else {
offset = 0;
node = marker.renderNode.headTextNode;
}
}
else {
node = marker.renderNode.element;
offset = offsetInMarker;
}
}
return { node, offset };
}
selectRange(range) {
if (range.isBlank) {
this.clearSelection();
return;
}
const { head, tail, direction } = range;
const { node: headNode, offset: headOffset } = this._findNodeForPosition(head), { node: tailNode, offset: tailOffset } = this._findNodeForPosition(tail);
this._moveToNode(headNode, headOffset, tailNode, tailOffset, direction);
// Firefox sometimes doesn't keep focus in the editor after adding a card
this.editor._ensureFocus();
}
get selection() {
return expect(this.editor.root.getSelection(), 'expected editor.root selection to not be null');
}
selectedText() {
// FIXME remove this
return this.selection.toString();
}
/**
* @param {textNode} node
* @param {integer} offset
* @param {textNode} endNode
* @param {integer} endOffset
* @param {integer} direction forward or backward, default forward
* @private
*/
_moveToNode(node, offset, endNode, endOffset, direction = Direction.FORWARD) {
this.clearSelection();
if (direction === Direction.BACKWARD) {
[node, offset, endNode, endOffset] = [endNode, endOffset, node, offset];
}
const range = document.createRange();
range.setStart(node, offset);
if (direction === Direction.BACKWARD && isFullSelection(this.selection)) {
this.selection.addRange(range);
this.selection.extend(endNode, endOffset);
}
else {
range.setEnd(endNode, endOffset);
this.selection.addRange(range);
}
}
_hasSelection() {
const element = unwrap(this.editor.element);
const { _selectionRange } = this;
// COMPAT: There's a bug in chrome that always returns `true` for
// `isCollapsed` for a Selection that comes from a ShadowRoot.
// (2020/08/08)
// https://bugs.chromium.org/p/chromium/issues/detail?id=447523
let isCollapsed;
const hasShadowRoot = () => {
return !!(window.document.activeElement && window.document.activeElement.shadowRoot);
};
if (_selectionRange) {
if (Browser.isChrome() && hasShadowRoot()) {
isCollapsed =
_selectionRange.anchorNode === _selectionRange.focusNode &&
_selectionRange.anchorOffset === _selectionRange.focusOffset;
}
else {
isCollapsed = _selectionRange.isCollapsed;
}
}
if (!_selectionRange || isCollapsed) {
return false;
}
return (containsNode(element, unwrap(this.selection.anchorNode)) &&
containsNode(element, unwrap(this.selection.focusNode)));
}
_hasCollapsedSelection() {
const { _selectionRange } = this;
if (!_selectionRange) {
return false;
}
const element = this.editor.element;
return containsNode(unwrap(element), unwrap(this.selection.anchorNode));
}
get _selectionRange() {
const { selection } = this;
if (selection.rangeCount === 0) {
return null;
}
return selection.getRangeAt(0);
}
}
class Set {
constructor(items = []) {
this.items = [];
items.forEach(i => this.add(i));
}
add(item) {
if (!this.has(item)) {
this.items.push(item);
}
}
get length() {
return this.items.length;
}
has(item) {
return this.items.indexOf(item) !== -1;
}
toArray() {
return this.items;
}
}
const PARENT_PROP = '__parent';
class LinkedList {
constructor(options) {
this.head = null;
this.tail = null;
this.length = 0;
if (options) {
const { adoptItem, freeItem } = options;
this._adoptItem = adoptItem;
this._freeItem = freeItem;
}
}
adoptItem(item) {
item[PARENT_PROP] = this;
this.length++;
if (this._adoptItem) {
this._adoptItem(item);
}
}
freeItem(item) {
item[PARENT_PROP] = null;
this.length--;
if (this._freeItem) {
this._freeItem(item);
}
}
get isEmpty() {
return this.length === 0;
}
prepend(item) {
this.insertBefore(item, this.head);
}
append(item) {
this.insertBefore(item, null);
}
insertAfter(item, prevItem) {
let nextItem = prevItem ? prevItem.next : this.head;
this.insertBefore(item, nextItem);
}
_ensureItemIsNotAlreadyInList(item) {
assert('Cannot insert an item into a list if it is already in a list', !item.next && !item.prev && this.head !== item);
}
insertBefore(item, nextItem) {
this._ensureItemIsNotInList(item);
this.adoptItem(item);
let insertPos;
if (nextItem && nextItem.prev) {
insertPos = 'middle';
}
else if (nextItem) {
insertPos = 'start';
}
else {
insertPos = 'end';
}
switch (insertPos) {
case 'start':
if (this.head) {
item.next = this.head;
this.head.prev = item;
}
this.head = item;
break;
case 'middle': {
let prevItem = nextItem.prev;
item.next = nextItem;
item.prev = prevItem;
nextItem.prev = item;
prevItem.next = item;
break;
}
case 'end': {
let tail = this.tail;
item.prev = tail;
if (tail) {
tail.next = item;
}
else {
this.head = item;
}
this.tail = item;
break;
}
}
}
remove(item) {
if (!getParent(item)) {
return;
}
this._ensureItemIsInThisList(item);
this.freeItem(item);
let [prev, next] = [item.prev, item.next];
item.prev = null;
item.next = null;
if (prev) {
prev.next = next;
}
else {
this.head = next;
}
if (next) {
next.prev = prev;
}
else {
this.tail = prev;
}
}
forEach(callback) {
let item = this.head;
let index = 0;
while (item) {
callback(item, index++);
item = item.next;
}
}
map(callback) {
let result = [];
this.forEach(i => result.push(callback(i)));
return result;
}
walk(startItem, endItem, callback) {
let item = startItem || this.head;
while (item) {
callback(item);
if (item === endItem) {
break;
}
item = item.next;
}
}
readRange(startItem, endItem) {
let items = [];
this.walk(startItem, endItem, item => {
items.push(item);
});
return items;
}
toArray() {
return this.readRange();
}
detect(callback, item = this.head, reverse = false) {
while (item) {
if (callback(item)) {
return item;
}
item = reverse ? item.prev : item.next;
}
}
any(callback) {
return !!this.detect(callback);
}
every(callback) {
let item = this.head;
while (item) {
if (!callback(item)) {
return false;
}
item = item.next;
}
return true;
}
objectAt(targetIndex) {
let index = -1;
return this.detect(() => {
index++;
return targetIndex === index;
});
}
splice(targetItem, removalCount, newItems) {
let item = targetItem;
let nextItem = item.next;
let count = 0;
while (item && count < removalCount) {
count++;
nextItem = item.next;
this.remove(item);
item = nextItem;
}
newItems.forEach(newItem => {
this.insertBefore(newItem, nextItem);
});
}
removeBy(conditionFn) {
let item = this.head;
while (item) {
let nextItem = item.next;
if (conditionFn(item)) {
this.remove(item);
}
item = nextItem;
}
}
_ensureItemIsNotInList(item) {
assert('Cannot insert an item into a list if it is already in a list', !item[PARENT_PROP]);
}
_ensureItemIsInThisList(item) {
assert('Cannot remove item that is in another list', getParent(item) === this);
}
}
function getParent(item) {
return item[PARENT_PROP] || null;
}
function tagNameable(Base) {
class TagNameable extends Base {
constructor() {
super(...arguments);
this._tagName = null;
}
set tagName(val) {
let normalizedTagName = normalizeTagName(val);
assert(`Cannot set section tagName to ${val}`, this.isValidTagName(normalizedTagName));
this._tagName = normalizedTagName;
}
get tagName() {
return this._tagName;
}
}
return TagNameable;
}
class Markerable extends tagNameable(Section) {
constructor(type, tagName, markers = []) {
super(type);
this.type = type;
this.isMarkerable = true;
this.tagName = tagName;
this.markers = new LinkedList({
adoptItem: m => {
assert(`Can only insert markers and atoms into markerable (was: ${m.type})`, m.isMarker || m.isAtom);
m.section = m.parent = this;
},
freeItem: m => (m.section = m.parent = null),
});
markers.forEach(m => this.markers.append(m));
}
canJoin(other) {
return other.isMarkerable && other.type === this.type && other.tagName === this.tagName;
}
clone() {
const newMarkers = this.markers.map(m => m.clone());
return this.builder.createMarkerableSection(this.type, this.tagName, newMarkers);
}
get isBlank() {
if (!this.markers.length) {
return true;
}
return this.markers.every(m => m.isBlank);
}
textUntil(position) {
assert(`Cannot get textUntil for a position not in this section`, position.section === this);
let { marker, offsetInMarker } = position;
let text = '';
let currentMarker = this.markers.head;
while (currentMarker) {
if (currentMarker === marker) {
text += currentMarker.textUntil(offsetInMarker);
break;
}
else {
text += currentMarker.text;
currentMarker = currentMarker.next;
}
}
return text;
}
/**
* @param {Marker}
* @param {Number} markerOffset The offset relative to the start of the marker
*
* @return {Number} The offset relative to the start of this section
*/
offsetOfMarker(marker, markerOffset = 0) {
assert(`Cannot get offsetOfMarker for marker that is not child of this`, marker.section === this);
// FIXME it is possible, when we get a cursor position before having finished reparsing,
// for markerOffset to be > marker.length. We shouldn't rely on this functionality.
let offset = 0;
let currentMarker = this.markers.head;
while (currentMarker && currentMarker !== marker.next) {
let length = currentMarker === marker ? markerOffset : currentMarker.length;
offset += length;
currentMarker = currentMarker.next;
}
return offset;
}
// puts clones of this.markers into beforeSection and afterSection,
// all markers before the marker/offset split go in beforeSection, and all
// after the marker/offset split go in afterSection
// @return {Array} [beforeSection, afterSection], two new sections
_redistributeMarkers(beforeSection, afterSection, marker, offset = 0) {
let currentSection = beforeSection;
forEach(this.markers, m => {
if (m === marker) {
const [beforeMarker, ...afterMarkers] = marker.split(offset);
beforeSection.markers.append(beforeMarker);
forEach(afterMarkers, _m => afterSection.markers.append(_m));
currentSection = afterSection;
}
else {
currentSection.markers.append(m.clone());
}
});
return [beforeSection, afterSection];
}
/**
* Split this section's marker (if any) at the given offset, so that
* there is now a marker boundary at that offset (useful for later applying
* a markup to a range)
* @param {Number} sectionOffset The offset relative to start of this section
* @return {EditObject} An edit object with 'removed' and 'added' keys with arrays of Markers. The added markers may be blank.
* After calling `splitMarkerAtOffset(offset)`, there will always be a valid
* result returned from `markerBeforeOffset(offset)`.
*/
splitMarkerAtOffset(sectionOffset) {
assert('Cannot splitMarkerAtOffset when offset is > length', sectionOffset <= this.length);
let markerOffset;
let len = 0;
let currentMarker = this.markers.head;
let edit = { added: [], removed: [] };
if (!currentMarker) {
let blankMarker = this.builder.createMarker();
this.markers.prepend(blankMarker);
edit.added.push(blankMarker);
}
else {
while (currentMarker) {
len += currentMarker.length;
if (len === sectionOffset) {
// nothing to do, there is a gap at the requested offset
break;
}
else if (len > sectionOffset) {
markerOffset = currentMarker.length - (len - sectionOffset);
let newMarkers = currentMarker.splitAtOffset(markerOffset);
edit.added.push(...newMarkers);
edit.removed.push(currentMarker);
this.markers.splice(currentMarker, 1, newMarkers);
break;
}
else {
currentMarker = currentMarker.next;
}
}
}
return edit;
}
splitAtPosition(position) {
const { marker, offsetInMarker } = position;
return this.splitAtMarker(marker, offsetInMarker);
}
// returns the marker just before this offset.
// It is an error to call this method with an offset that is in the middle
// of a marker.
markerBeforeOffset(sectionOffset) {
let len = 0;
let currentMarker = this.markers.head;
while (currentMarker) {
len += currentMarker.length;
if (len === sectionOffset) {
return currentMarker;
}
else {
assert('markerBeforeOffset called with sectionOffset not between markers', len < sectionOffset);
currentMarker = currentMarker.next;
}
}
}
markerPositionAtOffset(offset) {
let currentOffset = 0;
let currentMarker = null;
let remaining = offset;
this.markers.detect(marker => {
currentOffset = Math.min(remaining, marker.length);
remaining -= currentOffset;
if (remaining === 0) {
currentMarker = marker;
return true; // break out of detect
}
return false;
});
return { marker: currentMarker, offset: currentOffset };
}
get text() {
return reduce(this.markers, (prev, m) => prev + m.value, '');
}
get length() {
return reduce(this.markers, (prev, m) => prev + m.length, 0);
}
/**
* @return {Array} New markers that match the boundaries of the
* range. Does not change the existing markers in this section.
*/
markersFor(headOffset, tailOffset) {
const range = Range.create(this, headOffset, this, tailOffset);
let markers = [];
this._markersInRange(range, (marker, { markerHead, markerTail, isContained }) => {
const cloned = marker.clone();
if (!isContained) {
// cannot do marker.value.slice if the marker is an atom -- this breaks the atom's "atomic" value
// If a marker is an atom `isContained` should always be true so
// we shouldn't hit this code path. FIXME add tests
cloned.value = marker.value.slice(markerHead, markerTail);
}
markers.push(cloned);
});
return markers;
}
markupsInRange(range) {
const markups = new Set();
this._markersInRange(range, marker => {
marker.markups.forEach(m => markups.add(m));
});
return markups.toArray();
}
// calls the callback with (marker, {markerHead, markerTail, isContained})
// for each marker that is wholly or partially contained in the range.
_markersInRange(range, callback) {
const { head, tail } = range;
assert('Cannot call #_markersInRange if range expands beyond this section', head.section === this && tail.section === this);
const { offset: headOffset } = head, { offset: tailOffset } = tail;
let currentHead = 0, currentTail = 0, currentMarker = this.markers.head;
while (currentMarker) {
currentTail += currentMarker.length;
if (currentTail > headOffset && currentHead < tailOffset) {
let markerHead = Math.max(headOffset - currentHead, 0);
let markerTail = currentMarker.length - Math.max(currentTail - tailOffset, 0);
let isContained = markerHead === 0 && markerTail === currentMarker.length;
callback(currentMarker, { markerHead, markerTail, isContained });
}
currentHead += currentMarker.length;
currentMarker = currentMarker.next;
if (currentHead > tailOffset) {
break;
}
}
}
// mutates this by appending the other section's (cloned) markers to it
join(otherSection) {
let beforeMarker = this.markers.tail;
let afterMarker = null;
otherSection.markers.forEach(m => {
if (!m.isBlank) {
m = m.clone();
this.markers.append(m);
if (!afterMarker) {
afterMarker = m;
}
}
});
return { beforeMarker, afterMarker };
}
}
function isMarkerable$1(section) {
return section.isMarkerable;
}
const MARKUP_SECTION_TYPE = 'markup-section';
const LIST_SECTION_TYPE = 'list-section';
const MARKUP_TYPE = 'markup';
const LIST_ITEM_TYPE = 'list-item';
function entries(obj) {
const ownProps = Object.keys(obj);
let i = ownProps.length;
const resArray = new Array(i);
while (i--) {
resArray[i] = [ownProps[i], obj[ownProps[i]]];
}
return resArray;
}
const VALID_ATTRIBUTES = ['data-md-text-align'];
/*
* A "mixin" to add section attribute support
* to markup and list sections.
*/
function attributable(Base) {
return class extends Base {
constructor() {
super(...arguments);
this.attributes = {};
}
hasAttribute(key) {
return key in this.attributes;
}
setAttribute(key, value) {
if (!contains(VALID_ATTRIBUTES, key)) {
throw new Error(`Invalid attribute "${key}" was passed. Constrain attributes to the spec-compliant whitelist.`);
}
this.attributes[key] = value;
}
removeAttribute(key) {
delete this.attributes[key];
}
getAttribute(key) {
return this.attributes[key];
}
eachAttribute(cb) {
entries(this.attributes).forEach(([k, v]) => cb(k, v));
}
};
}
function getSectionAttributes(section) {
if (isNested(section)) {
return section.parent.attributes || {};
}
return section.attributes || {};
}
const VALID_LIST_SECTION_TAGNAMES = ['ul', 'ol'].map(normalizeTagName);
const DEFAULT_TAG_NAME = VALID_LIST_SECTION_TAGNAMES[0];
class ListSection extends attributable(tagNameable(Section)) {
constructor(tagName = DEFAULT_TAG_NAME, items = [], attributes = {}) {
super(LIST_SECTION_TYPE);
this.isListSection = true;
this.isLeafSection = false;
this.tagName = tagName;
entries(attributes).forEach(([k, v]) => this.setAttribute(k, v));
this.items = new LinkedList({
adoptItem: i => {
assert(`Cannot insert non-list-item to list (is: ${i.type})`, i.isListItem);
i.section = i._parent = this;
},
freeItem: i => (i.section = i._parent = null),
});
this.sections = this.items;
items.forEach(i => this.items.append(i));
}
canJoin() {
return false;
}
isValidTagName(normalizedTagName) {
return contains(VALID_LIST_SECTION_TAGNAMES, normalizedTagName);
}
headPosition() {
return this.items.head.headPosition();
}
tailPosition() {
return this.items.tail.tailPosition();
}
get isBlank() {
return this.items.isEmpty;
}
clone() {
let newSection = this.builder.createListSection(this.tagName);
forEach(this.items, i => newSection.items.append(i.clone()));
return newSection;
}
/**
* Mutates this list
* @param {ListSection|Markerable}
* @return null
*/
join(other) {
if (isListSection$1(other)) {
other.items.forEach(i => this.join(i));
}
else if (other.isMarkerable) {
let item = this.builder.createListItem();
item.join(other);
this.items.append(item);
}
}
}
function isListSection$1(section) {
return section.isListSection;
}
const VALID_LIST_ITEM_TAGNAMES = ['li'].map(normalizeTagName);
class ListItem extends Markerable {
constructor(tagName, markers = []) {
super("list-item" /* LIST_ITEM */, tagName, markers);
this.isListItem = true;
this.isNested = true;
this.section = null;
}
isValidTagName(normalizedTagName) {
return contains(VALID_LIST_ITEM_TAGNAMES, normalizedTagName);
}
splitAtMarker(marker, offset = 0) {
// FIXME need to check if we are going to split into two list items
// or a list item and a new markup section:
const isLastItem = !this.next;
const createNewSection = !marker && offset === 0 && isLastItem;
let [beforeSection, afterSection] = [
this.builder.createListItem(),
createNewSection ? this.builder.createMarkupSection() : this.builder.createListItem(),
];
return this._redistributeMarkers(beforeSection, afterSection, marker, offset);
}
get post() {
return expect(this.section, 'expected list item to have section').post;
}
}
function isListItem(section) {
return section.isListItem;
}
// valid values of `tagName` for a MarkupSection
const VALID_MARKUP_SECTION_TAGNAMES = ['aside', 'blockquote', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p'].map(normalizeTagName);
// valid element names for a MarkupSection. A MarkupSection with a tagName
// not in this will be rendered as a div with a className matching the
// tagName
const MARKUP_SECTION_ELEMENT_NAMES = ['aside', 'blockquote', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p'].map(normalizeTagName);
const DEFAULT_TAG_NAME$1 = VALID_MARKUP_SECTION_TAGNAMES[8];
class MarkupSection extends attributable(Markerable) {
constructor(tagName = DEFAULT_TAG_NAME$1, markers = [], attributes = {}) {
super(MARKUP_SECTION_TYPE, tagName, markers);
this.isMarkupSection = true;
this.isGenerated = false;
this._inferredTagName = false;
entries(attributes).forEach(([k, v]) => this.setAttribute(k, v));
}
isValidTagName(normalizedTagName) {
return contains(VALID_MARKUP_SECTION_TAGNAMES, normalizedTagName);
}
splitAtMarker(marker, offset = 0) {
let [beforeSection, afterSection] = [
this.builder.createMarkupSection(this.tagName, [], false, this.attributes),
this.builder.createMarkupSection(),
];
return this._redistributeMarkers(beforeSection, afterSection, marker, offset);
}
}
function isMarkupSection(section) {
return section.isMarkupSection;
}
function hasInferredTagName(section) {
return isMarkupSection(section) && section._inferredTagName;
}
const { FORWARD: FORWARD$1, BACKWARD: BACKWARD$1 } = Direction;
function isListSectionTagName(tagName) {
return tagName === 'ul' || tagName === 'ol';
}
function shrinkRange(range) {
const { head, tail } = range;
if (tail.offset === 0 && head.section !== tail.section) {
range.tail = new Position(tail.section.prev, tail.section.prev.length);
}
return range;
}
const CALLBACK_QUEUES = {
BEFORE_COMPLETE: 'beforeComplete',
COMPLETE: 'complete',
AFTER_COMPLETE: 'afterComplete',
};
/**
* The PostEditor is used to modify a post. It should not be instantiated directly.
* Instead, a new instance of a PostEditor is created by the editor and passed
* as the argument to the callback in {@link Editor#run}.
*
* Usage:
* ```
* editor.run((postEditor) => {
* // postEditor is an instance of PostEditor that can operate on the
* // editor's post
* });
* ```
*/
class PostEditor {
constructor(editor) {
this.editor = editor;
this.builder = this.editor.builder;
this._callbacks = new LifecycleCallbacks(values(CALLBACK_QUEUES));
this._didComplete = false;
this.editActionTaken = null;
this._renderRange = () => this.editor.selectRange(this._range);
this._postDidChange = () => this.editor._postDidChange();
this._rerender = () => this.editor.rerender();
}
addCallback(queueName, callback) {
this._callbacks.addCallback(queueName, callback);
}
addCallbackOnce(queueName, callback) {
this._callbacks.addCallbackOnce(queueName, callback);
}
runCallbacks(queueName) {
this._callbacks.runCallbacks(queueName);
}
begin() {
// cache the editor's range
this._range = this.editor.range;
}
/**
* Schedules to select the given range on the editor after the postEditor
* has completed its work. This also updates the postEditor's active range
* (so that multiple calls to range-changing methods on the postEditor will
* update the correct range).
*
* Usage:
* let range = editor.range;
* editor.run(postEditor => {
* let nextPosition = postEditor.deleteRange(range);
*
* // Will position the editor's cursor at `nextPosition` after
* // the postEditor finishes work and the editor rerenders.
* postEditor.setRange(nextPosition);
* });
* @param {Range|Position} range
* @public
*/
setRange(range) {
range = toRange(range);
// TODO validate that the range is valid
// (does not contain marked-for-removal head or tail sections?)
this._range = range;
this.scheduleAfterRender(this._renderRange, true);
}
/**
* Delete a range from the post
*
* Usage:
* ```
* let { range } = editor;
* editor.run((postEditor) => {
* let nextPosition = postEditor.deleteRange(range);
* postEditor.setRange(nextPosition);
* });
* ```
* @param {Range} range Cursor Range object with head and tail Positions
* @return {Position} The position where the cursor would go after deletion
* @public
*/
deleteRange(range) {
assert('Must pass MobiledocKit Range to `deleteRange`', range instanceof Range);
this.editActionTaken = 2 /* DELETE */;
const { head, tail } = range;
let headSection = head.section;
let tailSection = tail.section;
const { editor } = this;
const { post } = editor;
if (headSection === tailSection) {
return this.cutSection(headSection, head, tail);
}
let nextSection = headSection.nextLeafSection();
let nextPos = this.cutSection(headSection, head, headSection.tailPosition());
// cutSection can replace the section, so re-read headSection here
headSection = nextPos.section;
// Remove sections in the middle of the range
while (nextSection !== tailSection) {
let tmp = nextSection;
nextSection = nextSection.nextLeafSection();
this.removeSection(tmp);
}
let tailPos = this.cutSection(tailSection, tailSection.headPosition(), tail);
// cutSection can replace the section, so re-read tailSection here
tailSection = tailPos.section;
if (tailSection.isBlank) {
this.removeSection(tailSection);
}
else {
// If head and tail sections are markerable, join them
// Note: They may not be the same section type. E.g. this may join
// a tail section that was a list item onto a markup section, or vice versa.
// (This is the desired behavior.)
if (isMarkerable$1(headSection) && isMarkerable$1(tailSection)) {
headSection.join(tailSection);
this._markDirty(headSection);
this.removeSection(tailSection);
}
else if (headSection.isBlank) {
this.removeSection(headSection);
nextPos = tailPos;
}
}
if (post.isBlank) {
post.sections.append(this.builder.createMarkupSection('p'));
nextPos = post.headPosition();
}
return nextPos;
}
/**
* Note: This method may replace `section` with a different section.
*
* "Cut" out the part of the section inside `headOffset` and `tailOffset`.
* If section is markerable this splits markers that straddle the head or tail (if necessary),
* and removes markers that are wholly inside the offsets.
* If section is a card, this may replace it with a blank markup section if the
* positions contain the entire card.
*
* @param {Section} section
* @param {Position} head
* @param {Position} tail
* @return {Position}
* @private
*/
cutSection(section, head, tail) {
assert('Must pass head position and tail position to `cutSection`', head instanceof Position && tail instanceof Position);
assert('Must pass positions within same section to `cutSection`', head.section === tail.section);
if (section.isBlank || head.isEqual(tail)) {
return head;
}
if (section.isCardSection) {
if (head.isHead() && tail.isTail()) {
let newSection = this.builder.createMarkupSection();
this.replaceSection(section, newSection);
return newSection.headPosition();
}
else {
return tail;
}
}
let range = head.toRange(tail);
this.splitMarkers(range).forEach(m => this.removeMarker(m));
return head;
}
_coalesceMarkers(section) {
if (isMarkerable$1(section)) {
this._removeBlankMarkers(section);
this._joinSimilarMarkers(section);
}
}
_removeBlankMarkers(section) {
forEach(filter(section.markers, m => m.isBlank), m => this.removeMarker(m));
}
// joins markers that have identical markups
_joinSimilarMarkers(section) {
let marker = section.markers.head;
let nextMarker;
while (marker && marker.next) {
nextMarker = marker.next;
if (marker.canJoin(nextMarker)) {
nextMarker.value = marker.value + nextMarker.value;
this._markDirty(nextMarker);
this.removeMarker(marker);
}
marker = nextMarker;
}
}
removeMarker(marker) {
this._scheduleForRemoval(marker);
if (marker.section) {
this._markDirty(marker.section);
marker.section.markers.remove(marker);
}
}
_scheduleForRemoval(postNode) {
if (postNode.renderNode) {
postNode.renderNode.scheduleForRemoval();
this.scheduleRerender();
this.scheduleDidUpdate();
}
let removedAdjacentToList = (postNode.prev && isListSection$1(postNode.prev)) ||
(postNode.next && isListSection$1(postNode.next));
if (removedAdjacentToList) {
this.addCallback(CALLBACK_QUEUES.BEFORE_COMPLETE, () => this._joinContiguousListSections());
}
}
_joinContiguousListSections() {
let { post } = this.editor;
let range = this._range;
let prev;
let groups = [];
let currentGroup;
// FIXME do we need to force a re-render of the range if changed sections
// are contained within the range?
let updatedHead = null;
forEach(post.sections, section => {
if (prev && isListSection$1(prev) && isListSection$1(section) && prev.tagName === section.tagName) {
currentGroup = currentGroup || [prev];
currentGroup.push(section);
}
else {
if (currentGroup) {
groups.push(currentGroup);
}
currentGroup = null;
}
prev = section;
});
if (currentGroup) {
groups.push(currentGroup);
}
forEach(groups, group => {
let list = group[0];
forEach(group, listSection => {
if (listSection === list) {
return;
}
let currentHead = range.head;
let prevPosition;
// FIXME is there a currentHead if there is no range?
// is the current head a list item in the section
if (!range.isBlank && isListItem(currentHead.section) && currentHead.section.parent === listSection) {
prevPosition = list.tailPosition();
}
this._joinListSections(list, listSection);
if (prevPosition) {
updatedHead = prevPosition.move(FORWARD$1);
}
});
});
if (updatedHead) {
this.setRange(updatedHead);
}
}
_joinListSections(baseList, nextList) {
baseList.join(nextList);
this._markDirty(baseList);
this.removeSection(nextList);
}
_markDirty(postNode) {
if (postNode.renderNode) {
postNode.renderNode.markDirty();
this.scheduleRerender();
this.scheduleDidUpdate();
}
if ('section' in postNode && postNode.section) {
this._markDirty(postNode.section);
}
if (isMarkerable$1(postNode)) {
this.addCallback(CALLBACK_QUEUES.BEFORE_COMPLETE, () => this._coalesceMarkers(postNode));
}
}
/**
* @param {Position} position object with {section, offset} the marker and offset to delete from
* @param {Number} direction The direction to delete in (default is BACKWARD)
* @return {Position} for positioning the cursor
* @public
* @deprecated after v0.10.3
*/
deleteFrom(position, direction = Direction.BACKWARD) {
deprecate("`postEditor#deleteFrom is deprecated. Use `deleteAtPosition(position, direction=BACKWARD, {unit}={unit: 'char'})` instead");
return this.deleteAtPosition(position, direction, { unit: TextUnit.CHAR });
}
/**
* Delete 1 `unit` (can be 'char' or 'word') in the given `direction` at the given
* `position`. In almost all cases this will be equivalent to deleting the range formed
* by expanding the position 1 unit in the given direction. The exception is when deleting
* backward from the beginning of a list item, which reverts the list item into a markup section
* instead of joining it with its previous list item (if any).
*
* Usage:
*
* let position = section.tailPosition();
* // Section has text of "Howdy!"
* editor.run((postEditor) => {
* postEditor.deleteAtPosition(position);
* });
* // section has text of "Howdy"
*
* @param {Position} position The position to delete at
* @param {Direction} [direction=DIRECTION.BACKWARD] direction The direction to delete in
* @param {Object} [options]
* @param {String} [options.unit="char"] The unit of deletion ("word" or "char")
* @return {Position}
*/
deleteAtPosition(position, direction = Direction.BACKWARD, { unit } = { unit: TextUnit.CHAR }) {
if (direction === Direction.BACKWARD) {
return this._deleteAtPositionBackward(position, unit);
}
else {
return this._deleteAtPositionForward(position, unit);
}
}
_deleteAtPositionBackward(position, unit) {
if (position.isHead() && isListItem(position.section)) {
this.toggleSection('p', position);
return this._range.head;
}
else {
let prevPosition = unit === 'word' ? position.moveWord(BACKWARD$1) : position.move(BACKWARD$1);
let range = prevPosition.toRange(position);
return this.deleteRange(range);
}
}
_deleteAtPositionForward(position, unit) {
let nextPosition = unit === 'word' ? position.moveWord(FORWARD$1) : position.move(FORWARD$1);
let range = position.toRange(nextPosition);
return this.deleteRange(range);
}
/**
* Split markers at two positions, once at the head, and if necessary once
* at the tail.
*
* Usage:
* ```
* let range = editor.range;
* editor.run((postEditor) => {
* postEditor.splitMarkers(range);
* });
* ```
* The return value will be marker object completely inside the offsets
* provided. Markers outside of the split may also have been modified.
*
* @param {Range} markerRange
* @return {Array} of markers that are inside the split
* @private
*/
splitMarkers(range) {
const { post } = this.editor;
const { head, tail } = range;
this.splitSectionMarkerAtOffset(head.section, head.offset);
this.splitSectionMarkerAtOffset(tail.section, tail.offset);
return post.markersContainedByRange(range);
}
splitSectionMarkerAtOffset(section, offset) {
const edit = section.splitMarkerAtOffset(offset);
edit.removed.forEach(m => this.removeMarker(m));
}
/**
* Split the section at the position.
*
* Usage:
* ```
* let position = editor.cursor.offsets.head;
* editor.run((postEditor) => {
* postEditor.splitSection(position);
* });
* // Will result in the creation of two new sections
* // replacing the old one at the cursor position
* ```
* The return value will be the two new sections. One or both of these
* sections can be blank (contain only a blank marker), for example if the
* headMarkerOffset is 0.
*
* @param {Position} position
* @return {Array} new sections, one for the first half and one for the second (either one can be null)
* @public
*/
splitSection(position) {
const section = position.section;
if (isCardSection(section)) {
return this._splitCardSection(section, position);
}
else if (isListItem(section)) {
let isLastAndBlank = section.isBlank && !section.next;
if (isLastAndBlank) {
// if is last, replace the item with a blank markup section
let parent = section.parent;
let collection = this.editor.post.sections;
let blank = this.builder.createMarkupSection();
this.removeSection(section);
this.insertSectionBefore(collection, blank, parent.next);
return [null, blank];
}
else {
let [pre, post] = this._splitListItem(section, position);
return [pre, post];
}
}
else {
let splitSections = section.splitAtPosition(position);
splitSections.forEach(s => this._coalesceMarkers(s));
this._replaceSection(section, splitSections);
return splitSections;
}
}
/**
* @param {Section} cardSection
* @param {Position} position to split at
* @return {Section[]} 2-item array of pre and post-split sections
* @private
*/
_splitCardSection(cardSection, position) {
let { offset } = position;
assert('Cards section must be split at offset 0 or 1', offset === 0 || offset === 1);
let newSection = this.builder.createMarkupSection();
let nextSection;
let surroundingSections;
if (offset === 0) {
nextSection = cardSection;
surroundingSections = [newSection, cardSection];
}
else {
nextSection = cardSection.next;
surroundingSections = [cardSection, newSection];
}
let collection = this.editor.post.sections;
this.insertSectionBefore(collection, newSection, nextSection);
return surroundingSections;
}
/**
* @param {Section} section
* @param {Section} newSection
* @public
*/
replaceSection(section, newSection) {
if (!section) {
// FIXME should a falsy section be a valid argument?
this.insertSectionBefore(this.editor.post.sections, newSection, null);
}
else {
this._replaceSection(section, [newSection]);
}
}
moveSectionBefore(collection, renderedSection, beforeSection) {
const newSection = renderedSection.clone();
this.removeSection(renderedSection);
this.insertSectionBefore(collection, newSection, beforeSection);
return newSection;
}
/**
* @param {Section} section A section that is already in DOM
* @public
*/
moveSectionUp(renderedSection) {
const isFirst = !renderedSection.prev;
if (isFirst) {
return renderedSection;
}
const collection = renderedSection.parent.sections;
const beforeSection = renderedSection.prev;
return this.moveSectionBefore(collection, renderedSection, beforeSection);
}
/**
* @param {Section} section A section that is already in DOM
* @public
*/
moveSectionDown(renderedSection) {
const isLast = !renderedSection.next;
if (isLast) {
return renderedSection;
}
const beforeSection = renderedSection.next.next;
const collection = renderedSection.parent.sections;
return this.moveSectionBefore(collection, renderedSection, beforeSection);
}
/**
* Insert an array of markers at the given position. If the position is in
* a non-markerable section (like a card section), this method throws an error.
*
* @param {Position} position
* @param {Marker[]} markers
* @return {Position} The position that represents the end of the inserted markers.
* @public
*/
insertMarkers(position, markers) {
const section = position.section;
let offset = position.offset;
assert('Cannot insert markers at non-markerable position', section.isMarkerable);
this.editActionTaken = 1 /* INSERT_TEXT */;
let edit = section.splitMarkerAtOffset(offset);
edit.removed.forEach(marker => this._scheduleForRemoval(marker));
let prevMarker = section.markerBeforeOffset(offset);
markers.forEach(marker => {
section.markers.insertAfter(marker, prevMarker);
offset += marker.length;
prevMarker = marker;
});
this._coalesceMarkers(section);
this._markDirty(section);
let nextPosition = section.toPosition(offset);
this.setRange(nextPosition);
return nextPosition;
}
/**
* Inserts text with the given markups, ignoring the existing markups at
* the position, if any.
*
* @param {Position} position
* @param {String} text
* @param {Markup[]} markups
* @return {Position} position at the end of the inserted text
*/
insertTextWithMarkup(position, text, markups = []) {
let { section } = position;
if (!section.isMarkerable) {
return;
}
let marker = this.builder.createMarker(text, markups);
return this.insertMarkers(position, [marker]);
}
/**
* Insert the text at the given position
* Inherits the markups already at that position, if any.
*
* @param {Position} position
* @param {String} text
* @return {Position} position at the end of the inserted text.
*/
insertText(position, text) {
let { section } = position;
if (!section.isMarkerable) {
return;
}
let markups = position.marker && position.marker.markups;
markups = markups || [];
return this.insertTextWithMarkup(position, text, markups);
}
_replaceSection(section, newSections) {
let nextSection = section.next;
let collection = section.parent.sections;
let nextNewSection = newSections[0];
if (isMarkupSection(nextNewSection) && isListItem(section)) {
// put the new section after the ListSection (section.parent)
// instead of after the ListItem
collection = section.parent.parent.sections;
nextSection = section.parent.next;
}
newSections.forEach(s => this.insertSectionBefore(collection, s, nextSection));
this.removeSection(section);
}
/**
* Given a markerRange (for example `editor.range`) mark all markers
* inside it as a given markup. The markup must be provided as a post
* abstract node.
*
* Usage:
*
* let range = editor.range;
* let strongMarkup = editor.builder.createMarkup('strong');
* editor.run((postEditor) => {
* postEditor.addMarkupToRange(range, strongMarkup);
* });
* // Will result some markers possibly being split, and the markup
* // being applied to all markers between the split.
*
* @param {Range} range
* @param {Markup} markup A markup post abstract node
* @public
*/
addMarkupToRange(range, markup) {
if (range.isCollapsed) {
return;
}
let markers = this.splitMarkers(range);
if (markers.length) {
// We insert the new markup at a consistent index across the range.
// If we just push on the end of the list, it can end up in different positions
// of the markup stack. This results in unnecessary closing and re-opening of
// the markup each time it changes position.
// If we just push it at the beginning of the list, this causes unnecessary closing
// and re-opening of surrounding tags.
// So, we look for any tags open across the whole range, and push into the stack
// at the end of those.
// Prompted by https://github.com/bustle/mobiledoc-kit/issues/360
let markupsOpenAcrossRange = reduce(markers, function (soFar, marker) {
return commonItems(soFar, marker.markups);
}, markers[0].markups);
let indexToInsert = markupsOpenAcrossRange.length;
markers.forEach(marker => {
marker.addMarkupAtIndex(markup, indexToInsert);
this._markDirty(marker);
});
}
}
/**
* Given a markerRange (for example `editor.range`) remove the given
* markup from all contained markers.
*
* Usage:
* ```
* let { range } = editor;
* let markup = markerRange.headMarker.markups[0];
* editor.run(postEditor => {
* postEditor.removeMarkupFromRange(range, markup);
* });
* // Will result in some markers possibly being split, and the markup
* // being removed from all markers between the split.
* ```
* @param {Range} range Object with offsets
* @param {Markup|Function} markupOrCallback A markup post abstract node or
* a function that returns true when passed a markup that should be removed
* @private
*/
removeMarkupFromRange(range, markupOrMarkupCallback) {
if (range.isCollapsed) {
return;
}
this.splitMarkers(range).forEach(marker => {
marker.removeMarkup(markupOrMarkupCallback);
this._markDirty(marker);
});
}
/**
* Toggle the given markup in the given range (or at the position given). If the range/position
* has the markup, the markup will be removed. If nothing in the range/position
* has the markup, the markup will be added to everything in the range/position.
*
* Usage:
* ```
* // Remove any 'strong' markup if it exists in the selection, otherwise
* // make it all 'strong'
* editor.run(postEditor => postEditor.toggleMarkup('strong'));
*
* // add/remove a link to 'bustle.com' to the selection
* editor.run(postEditor => {
* const linkMarkup = postEditor.builder.createMarkup('a', {href: 'http://bustle.com'});
* postEditor.toggleMarkup(linkMarkup);
* });
* ```
* @param {Markup|String} markupOrString Either a markup object created using
* the builder (useful when adding a markup with attributes, like an 'a' markup),
* or, if a string, the tag name of the markup (e.g. 'strong', 'em') to toggle.
* @param {Range|Position} range in which to toggle. Defaults to current editor range.
* @public
*/
toggleMarkup(markupOrMarkupString, range = this._range) {
range = toRange(range);
const markup = typeof markupOrMarkupString === 'string' ? this.builder.createMarkup(markupOrMarkupString) : markupOrMarkupString;
const hasMarkup = this.editor.detectMarkupInRange(range, markup.tagName);
// FIXME: This implies only a single markup in a range. This may not be
// true for links (which are not the same object instance like multiple
// strong tags would be).
if (hasMarkup) {
this.removeMarkupFromRange(range, hasMarkup);
}
else {
this.addMarkupToRange(range, markup);
}
this.setRange(range);
}
/**
* Toggles the tagName of the active section or sections in the given range/position.
* If every section has the tag name, they will all be reset to default sections.
* Otherwise, every section will be changed to the requested type
*
* @param {String} sectionTagName A valid markup section or
* list section tag name (e.g. 'blockquote', 'h2', 'ul')
* @param {Range|Position} range The range over which to toggle.
* Defaults to the current editor range.
* @public
*/
toggleSection(sectionTagName, range = this._range) {
range = shrinkRange(toRange(range));
sectionTagName = normalizeTagName(sectionTagName);
let { post } = this.editor;
let everySectionHasTagName = true;
post.walkMarkerableSections(range, section => {
if (!this._isSameSectionType(section, sectionTagName)) {
everySectionHasTagName = false;
}
});
let tagName = everySectionHasTagName ? 'p' : sectionTagName;
let sectionTransformations = [];
post.walkMarkerableSections(range, section => {
let changedSection = this.changeSectionTagName(section, tagName);
sectionTransformations.push({
from: section,
to: changedSection,
});
});
let nextRange = this._determineNextRangeAfterToggleSection(range, sectionTransformations);
this.setRange(nextRange);
}
_determineNextRangeAfterToggleSection(range, sectionTransformations) {
if (sectionTransformations.length) {
let changedHeadSection = detect(sectionTransformations, ({ from }) => {
return from === range.headSection;
}).to;
let changedTailSection = detect(sectionTransformations, ({ from }) => {
return from === range.tailSection;
}).to;
if (changedHeadSection.isListSection || changedTailSection.isListSection) {
// We don't know to which ListItem's the original sections point at, so
// we don't have enough information to reconstruct the range when
// dealing with lists.
return sectionTransformations[0].to.headPosition().toRange();
}
else {
return Range.create(changedHeadSection, range.headSectionOffset, changedTailSection, range.tailSectionOffset, range.direction);
}
}
else {
return range;
}
}
setAttribute(key, value, range = this._range) {
this._mutateAttribute(key, range, (section, attribute) => {
if (section.getAttribute(attribute) !== value) {
section.setAttribute(attribute, value);
return true;
}
});
}
removeAttribute(key, range = this._range) {
this._mutateAttribute(key, range, (section, attribute) => {
if (section.hasAttribute(attribute)) {
section.removeAttribute(attribute);
return true;
}
});
}
_mutateAttribute(key, range, cb) {
range = toRange(range);
let { post } = this.editor;
let attribute = `data-md-${key}`;
post.walkMarkerableSections(range, section => {
const cbSection = isListItem(section) ? section.parent : section;
if (cb(cbSection, attribute) === true) {
this._markDirty(section);
}
});
this.setRange(range);
}
_isSameSectionType(section, sectionTagName) {
return isListItem(section) ? section.parent.tagName === sectionTagName : section.tagName === sectionTagName;
}
/**
* @param {Markerable} section
* @private
*/
changeSectionTagName(section, newTagName) {
assert('Cannot pass non-markerable section to `changeSectionTagName`', section.isMarkerable);
if (isListSectionTagName(newTagName)) {
return this._changeSectionToListItem(section, newTagName);
}
else if (isListItem(section)) {
return this._changeSectionFromListItem(section, newTagName);
}
else {
section.tagName = newTagName;
this._markDirty(section);
return section;
}
}
/**
* Splits the item at the position given.
* If the position is at the start or end of the item, the pre- or post-item
* will contain a single empty ("") marker.
* @param {ListItem} item
* @param {Position} position
* @return {Array} the pre-item and post-item on either side of the split
* @private
*/
_splitListItem(item, position) {
let { section, offset } = position;
assert('Cannot split list item at position that does not include item', item === section);
item.splitMarkerAtOffset(offset);
let prevMarker = item.markerBeforeOffset(offset);
let preItem = this.builder.createListItem(), postItem = this.builder.createListItem();
let currentItem = preItem;
item.markers.forEach(marker => {
currentItem.markers.append(marker.clone());
if (marker === prevMarker) {
currentItem = postItem;
}
});
this._replaceSection(item, [preItem, postItem]);
return [preItem, postItem];
}
/**
* Splits the list at the position given.
* @return {Array} pre-split list and post-split list, either of which could
* be blank (0-item list) if the position is at the start or end of the list.
*
* Note: Contiguous list sections will be joined in the before_complete queue
* of the postEditor.
*
* @private
*/
_splitListAtPosition(list, position) {
assert('Cannot split list at position not in list', position.section.parent === list);
let positionIsMiddle = !position.isHead() && !position.isTail();
if (positionIsMiddle) {
let item = position.section;
let [pre] = this._splitListItem(item, position);
position = pre.tailPosition();
}
let preList = this.builder.createListSection(list.tagName);
let postList = this.builder.createListSection(list.tagName);
let preItem = position.section;
let currentList = preList;
list.items.forEach(item => {
// If this item matches the start item and the position is at its start,
// it should be appended to the postList instead of the preList
if (item === preItem && position.isEqual(item.headPosition())) {
currentList = postList;
}
currentList.items.append(item.clone());
// If we just appended the preItem, append the remaining items to the postList
if (item === preItem) {
currentList = postList;
}
});
this._replaceSection(list, [preList, postList]);
return [preList, postList];
}
/**
* @return Array of [prev, mid, next] lists. `prev` and `next` can
* be blank, depending on the position of `item`. `mid` will always
* be a 1-item list containing `item`. `prev` and `next` will be
* removed in the before_complete queue if they are blank
* (and still attached).
*
* @private
*/
_splitListAtItem(list, item) {
let next = list;
let prev = this.builder.createListSection(next.tagName, [], next.attributes);
let mid = this.builder.createListSection(next.tagName);
let addToPrev = true;
// must turn the LinkedList into an array so that we can remove items
// as we iterate through it
let items = next.items.toArray();
items.forEach(i => {
let listToAppend;
if (i === item) {
addToPrev = false;
listToAppend = mid;
}
else if (addToPrev) {
listToAppend = prev;
}
else {
return; // break after iterating prev and mid parts of the list
}
listToAppend.join(i);
this.removeSection(i);
});
let found = !addToPrev;
assert('Cannot split list at item that is not present in the list', found);
let collection = this.editor.post.sections;
this.insertSectionBefore(collection, mid, next);
this.insertSectionBefore(collection, prev, mid);
// Remove possibly blank prev/next lists
this.addCallback(CALLBACK_QUEUES.BEFORE_COMPLETE, () => {
[prev, next].forEach(_list => {
let isAttached = !!_list._parent;
if (_list.isBlank && isAttached) {
this.removeSection(_list);
}
});
});
return [prev, mid, next];
}
_changeSectionFromListItem(section, newTagName) {
assertType('Must pass list item to `_changeSectionFromListItem`', section, isListItem(section));
let listSection = section.parent;
let markupSection = this.builder.createMarkupSection(newTagName);
markupSection.join(section);
let [, mid] = this._splitListAtItem(listSection, section);
this.replaceSection(mid, markupSection);
return markupSection;
}
_changeSectionToListItem(section, newTagName) {
let isAlreadyCorrectListItem = section.isListItem && section.parent.tagName === newTagName;
if (isAlreadyCorrectListItem) {
return section;
}
let listSection = this.builder.createListSection(newTagName);
listSection.join(section);
let sectionToReplace;
if (isListItem(section)) {
let [, mid] = this._splitListAtItem(section.parent, section);
sectionToReplace = mid;
}
else {
sectionToReplace = section;
}
this.replaceSection(sectionToReplace, listSection);
return listSection;
}
/**
* Insert a given section before another one, updating the post abstract
* and the rendered UI.
*
* Usage:
* ```
* let markerRange = editor.range;
* let sectionWithCursor = markerRange.headMarker.section;
* let section = editor.builder.createCardSection('my-image');
* let collection = sectionWithCursor.parent.sections;
* editor.run((postEditor) => {
* postEditor.insertSectionBefore(collection, section, sectionWithCursor);
* });
* ```
* @param {LinkedList} collection The list of sections to insert into
* @param {Object} section The new section
* @param {Object} beforeSection Optional The section "before" is relative to,
* if falsy the new section will be appended to the collection
* @public
*/
insertSectionBefore(collection, section, beforeSection) {
collection.insertBefore(section, beforeSection);
this._markDirty(section.parent);
}
/**
* Insert the given section after the current active section, or, if no
* section is active, at the end of the document.
* @param {Section} section
* @public
*/
insertSection(section) {
const activeSection = this.editor.activeSection;
const nextSection = activeSection && activeSection.next;
const collection = this.editor.post.sections;
this.insertSectionBefore(collection, section, nextSection);
}
/**
* Insert the given section at the end of the document.
* @param {Section} section
* @public
*/
insertSectionAtEnd(section) {
this.insertSectionBefore(this.editor.post.sections, section, null);
}
/**
* Insert the `post` at the given position in the editor's post.
* @param {Position} position
* @param {Post} post
* @private
*/
insertPost(position, newPost) {
let post = this.editor.post;
let inserter = new Inserter(this, post);
let nextPosition = inserter.insert(position, newPost);
return nextPosition;
}
/**
* Remove a given section from the post abstract and the rendered UI.
*
* Usage:
* ```
* let { range } = editor;
* let sectionWithCursor = range.head.section;
* editor.run((postEditor) => {
* postEditor.removeSection(sectionWithCursor);
* });
* ```
* @param {Object} section The section to remove
* @public
*/
removeSection(section) {
let parent = section.parent;
assertType('expected section to have child sections', parent, hasChildSections(parent));
this._scheduleForRemoval(section);
parent.sections.remove(section);
if (isListSection$1(parent)) {
this._scheduleListRemovalIfEmpty(parent);
}
}
removeAllSections() {
this.editor.post.sections.toArray().forEach(section => {
this.removeSection(section);
});
}
migrateSectionsFromPost(post) {
post.sections.toArray().forEach(section => {
post.sections.remove(section);
this.insertSectionBefore(this.editor.post.sections, section, null);
});
}
_scheduleListRemovalIfEmpty(listSection) {
this.addCallback(CALLBACK_QUEUES.BEFORE_COMPLETE, () => {
// if the list is attached and blank after we do other rendering stuff,
// remove it
let isAttached = !!listSection._parent;
if (isAttached && listSection.isBlank) {
this.removeSection(listSection);
}
});
}
/**
* A method for adding work the deferred queue
*
* @param {Function} callback to run during completion
* @param {Boolean} [once=false] Whether to only schedule the callback once.
* @public
*/
schedule(callback, once = false) {
assert('Work can only be scheduled before a post edit has completed', !this._didComplete);
if (once) {
this.addCallbackOnce(CALLBACK_QUEUES.COMPLETE, callback);
}
else {
this.addCallback(CALLBACK_QUEUES.COMPLETE, callback);
}
}
/**
* A method for adding work the deferred queue. The callback will only
* be added to the queue once, even if `scheduleOnce` is called multiple times.
* The function cannot be an anonymous function.
*
* @param {Function} callback to run during completion
* @public
*/
scheduleOnce(callback) {
this.schedule(callback, true);
}
/**
* Add a rerender job to the queue
*
* @public
*/
scheduleRerender() {
this.scheduleOnce(this._rerender);
}
/**
* Schedule a notification that the post has been changed.
* The notification will result in the editor firing its `postDidChange`
* hook after the postEditor completes its work (at the end of {@link Editor#run}).
*
* @public
*/
scheduleDidUpdate() {
this.scheduleOnce(this._postDidChange);
}
scheduleAfterRender(callback, once = false) {
if (once) {
this.addCallbackOnce(CALLBACK_QUEUES.AFTER_COMPLETE, callback);
}
else {
this.addCallback(CALLBACK_QUEUES.AFTER_COMPLETE, callback);
}
}
/**
* Flush any work on the queue. {@link Editor#run} calls this method; it
* should not be called directly.
*
* @private
*/
complete() {
assert('Post editing can only be completed once', !this._didComplete);
this.runCallbacks(CALLBACK_QUEUES.BEFORE_COMPLETE);
this._didComplete = true;
this.runCallbacks(CALLBACK_QUEUES.COMPLETE);
this.runCallbacks(CALLBACK_QUEUES.AFTER_COMPLETE);
}
undoLastChange() {
this.editor._editHistory.stepBackward(this);
}
redoLastChange() {
this.editor._editHistory.stepForward(this);
}
cancelSnapshot() {
this._shouldCancelSnapshot = true;
}
}
const placeholderImageSrc = '';
var ImageCard = {
name: 'image',
type: 'dom',
render({ payload }) {
let img = document.createElement('img');
img.src = payload.src || placeholderImageSrc;
return img;
},
};
/*
* Parses from mobiledoc -> post
*/
class MobiledocParser {
constructor(builder) {
this.builder = builder;
}
/**
* @param {Mobiledoc}
* @return {Post}
*/
parse({ sections: sectionData }) {
try {
const markerTypes = sectionData[0];
const sections = sectionData[1];
const post = this.builder.createPost();
this.markups = [];
this.markerTypes = this.parseMarkerTypes(markerTypes);
this.parseSections(sections, post);
return post;
}
catch (e) {
assert(`Unable to parse mobiledoc: ${e.message}`, false);
}
}
parseMarkerTypes(markerTypes) {
return markerTypes.map(markerType => this.parseMarkerType(markerType));
}
parseMarkerType([tagName, attributesArray]) {
const attributesObject = kvArrayToObject(attributesArray || []);
return this.builder.createMarkup(tagName, attributesObject);
}
parseSections(sections, post) {
sections.forEach(section => this.parseSection(section, post));
}
parseSection(section, post) {
switch (section[0]) {
case 1 /* MARKUP */:
this.parseMarkupSection(section, post);
break;
case 2 /* IMAGE */:
this.parseImageSection(section, post);
break;
case 10 /* CARD */:
this.parseCardSection(section, post);
break;
case 3 /* LIST */:
this.parseListSection(section, post);
break;
default:
assert(`Unexpected section type ${section[0]}`, false);
}
}
parseCardSection([, name, payload], post) {
const section = this.builder.createCardSection(name, payload);
post.sections.append(section);
}
parseImageSection([, src], post) {
const section = this.builder.createImageSection(src);
post.sections.append(section);
}
parseMarkupSection([, tagName, markers], post) {
const section = this.builder.createMarkupSection(tagName.toLowerCase() === 'pull-quote' ? 'aside' : tagName);
post.sections.append(section);
this.parseMarkers(markers, section);
// Strip blank markers after they have been created. This ensures any
// markup they include has been correctly populated.
filter(section.markers, m => m.isBlank).forEach(m => {
section.markers.remove(m);
});
}
parseListSection([, tagName, items], post) {
const section = this.builder.createListSection(tagName);
post.sections.append(section);
this.parseListItems(items, section);
}
parseListItems(items, section) {
items.forEach(i => this.parseListItem(i, section));
}
parseListItem(markers, section) {
const item = this.builder.createListItem();
this.parseMarkers(markers, item);
section.items.append(item);
}
parseMarkers(markers, parent) {
markers.forEach(m => this.parseMarker(m, parent));
}
parseMarker([markerTypeIndexes, closeCount, value], parent) {
markerTypeIndexes.forEach(index => {
this.markups.push(this.markerTypes[index]);
});
const marker = this.builder.createMarker(value, this.markups.slice());
parent.markers.append(marker);
this.markups = this.markups.slice(0, this.markups.length - closeCount);
}
}
/*
* Parses from mobiledoc -> post
*/
class MobiledocParser$1 {
constructor(builder) {
this.builder = builder;
}
/**
* @param {Mobiledoc}
* @return {Post}
*/
parse({ sections, markups: markerTypes, cards: cardTypes, atoms: atomTypes }) {
try {
const post = this.builder.createPost();
this.markups = [];
this.markerTypes = this.parseMarkerTypes(markerTypes);
this.cardTypes = this.parseCardTypes(cardTypes);
this.atomTypes = this.parseAtomTypes(atomTypes);
this.parseSections(sections, post);
return post;
}
catch (e) {
assert(`Unable to parse mobiledoc: ${e.message}`, false);
}
}
parseMarkerTypes(markerTypes) {
return markerTypes.map(markerType => this.parseMarkerType(markerType));
}
parseMarkerType([tagName, attributesArray]) {
const attributesObject = kvArrayToObject(attributesArray || []);
return this.builder.createMarkup(tagName, attributesObject);
}
parseCardTypes(cardTypes) {
return cardTypes.map(cardType => this.parseCardType(cardType));
}
parseCardType([cardName, cardPayload]) {
return [cardName, cardPayload];
}
parseAtomTypes(atomTypes) {
return atomTypes.map(atomType => this.parseAtomType(atomType));
}
parseAtomType([atomName, atomValue, atomPayload]) {
return [atomName, atomValue, atomPayload];
}
parseSections(sections, post) {
sections.forEach(section => this.parseSection(section, post));
}
parseSection(section, post) {
switch (section[0]) {
case 1 /* MARKUP */:
this.parseMarkupSection(section, post);
break;
case 2 /* IMAGE */:
this.parseImageSection(section, post);
break;
case 10 /* CARD */:
this.parseCardSection(section, post);
break;
case 3 /* LIST */:
this.parseListSection(section, post);
break;
default:
assert(`Unexpected section type ${section[0]}`, false);
}
}
getAtomTypeFromIndex(index) {
const atomType = this.atomTypes[index];
assert(`No atom definition found at index ${index}`, !!atomType);
return atomType;
}
getCardTypeFromIndex(index) {
const cardType = this.cardTypes[index];
assert(`No card definition found at index ${index}`, !!cardType);
return cardType;
}
parseCardSection([, cardIndex], post) {
const [name, payload] = this.getCardTypeFromIndex(cardIndex);
const section = this.builder.createCardSection(name, payload);
post.sections.append(section);
}
parseImageSection([, src], post) {
const section = this.builder.createImageSection(src);
post.sections.append(section);
}
parseMarkupSection([, tagName, markers], post) {
const section = this.builder.createMarkupSection(tagName.toLowerCase() === 'pull-quote' ? 'aside' : tagName);
post.sections.append(section);
this.parseMarkers(markers, section);
// Strip blank markers after they have been created. This ensures any
// markup they include has been correctly populated.
filter(section.markers, m => m.isBlank).forEach(m => {
section.markers.remove(m);
});
}
parseListSection([, tagName, items], post) {
const section = this.builder.createListSection(tagName);
post.sections.append(section);
this.parseListItems(items, section);
}
parseListItems(items, section) {
items.forEach(i => this.parseListItem(i, section));
}
parseListItem(markers, section) {
const item = this.builder.createListItem();
this.parseMarkers(markers, item);
section.items.append(item);
}
parseMarkers(markers, parent) {
markers.forEach(m => this.parseMarker(m, parent));
}
parseMarker([type, markerTypeIndexes, closeCount, value], parent) {
markerTypeIndexes.forEach(index => {
this.markups.push(this.markerTypes[index]);
});
const marker = this.buildMarkerType(type, value);
parent.markers.append(marker);
this.markups = this.markups.slice(0, this.markups.length - closeCount);
}
buildMarkerType(type, value) {
switch (type) {
case 0 /* MARKUP */:
return this.builder.createMarker(value, this.markups.slice());
case 1 /* ATOM */: {
const [atomName, atomValue, atomPayload] = this.getAtomTypeFromIndex(value);
return this.builder.createAtom(atomName, atomValue, atomPayload, this.markups.slice());
}
default:
assert(`Unexpected marker type ${type}`, false);
}
}
}
/*
* Parses from mobiledoc -> post
*/
class MobiledocParser$2 {
constructor(builder) {
this.builder = builder;
}
/**
* @param {Mobiledoc}
* @return {Post}
*/
parse({ sections, markups: markerTypes, cards: cardTypes, atoms: atomTypes }) {
try {
const post = this.builder.createPost();
this.markups = [];
this.markerTypes = this.parseMarkerTypes(markerTypes);
this.cardTypes = this.parseCardTypes(cardTypes);
this.atomTypes = this.parseAtomTypes(atomTypes);
this.parseSections(sections, post);
return post;
}
catch (e) {
assert(`Unable to parse mobiledoc: ${e.message}`, false);
}
}
parseMarkerTypes(markerTypes) {
return markerTypes.map(markerType => this.parseMarkerType(markerType));
}
parseMarkerType([tagName, attributesArray]) {
const attributesObject = kvArrayToObject(attributesArray || []);
return this.builder.createMarkup(tagName, attributesObject);
}
parseCardTypes(cardTypes) {
return cardTypes.map(cardType => this.parseCardType(cardType));
}
parseCardType([cardName, cardPayload]) {
return [cardName, cardPayload];
}
parseAtomTypes(atomTypes) {
return atomTypes.map(atomType => this.parseAtomType(atomType));
}
parseAtomType([atomName, atomValue, atomPayload]) {
return [atomName, atomValue, atomPayload];
}
parseSections(sections, post) {
sections.forEach(section => this.parseSection(section, post));
}
parseSection(section, post) {
switch (section[0]) {
case 1 /* MARKUP */:
this.parseMarkupSection(section, post);
break;
case 2 /* IMAGE */:
this.parseImageSection(section, post);
break;
case 10 /* CARD */:
this.parseCardSection(section, post);
break;
case 3 /* LIST */:
this.parseListSection(section, post);
break;
default:
assert(`Unexpected section type ${section[0]}`, false);
}
}
getAtomTypeFromIndex(index) {
const atomType = this.atomTypes[index];
assert(`No atom definition found at index ${index}`, !!atomType);
return atomType;
}
getCardTypeFromIndex(index) {
const cardType = this.cardTypes[index];
assert(`No card definition found at index ${index}`, !!cardType);
return cardType;
}
parseCardSection([, cardIndex], post) {
const [name, payload] = this.getCardTypeFromIndex(cardIndex);
const section = this.builder.createCardSection(name, payload);
post.sections.append(section);
}
parseImageSection([, src], post) {
const section = this.builder.createImageSection(src);
post.sections.append(section);
}
parseMarkupSection([, tagName, markers], post) {
const section = this.builder.createMarkupSection(tagName);
post.sections.append(section);
this.parseMarkers(markers, section);
// Strip blank markers after they have been created. This ensures any
// markup they include has been correctly populated.
filter(section.markers, m => m.isBlank).forEach(m => {
section.markers.remove(m);
});
}
parseListSection([, tagName, items], post) {
const section = this.builder.createListSection(tagName);
post.sections.append(section);
this.parseListItems(items, section);
}
parseListItems(items, section) {
items.forEach(i => this.parseListItem(i, section));
}
parseListItem(markers, section) {
const item = this.builder.createListItem();
this.parseMarkers(markers, item);
section.items.append(item);
}
parseMarkers(markers, parent) {
markers.forEach(m => this.parseMarker(m, parent));
}
parseMarker([type, markerTypeIndexes, closeCount, value], parent) {
markerTypeIndexes.forEach(index => {
this.markups.push(this.markerTypes[index]);
});
const marker = this.buildMarkerType(type, value);
parent.markers.append(marker);
this.markups = this.markups.slice(0, this.markups.length - closeCount);
}
buildMarkerType(type, value) {
switch (type) {
case 0 /* MARKUP */:
return this.builder.createMarker(value, this.markups.slice());
case 1 /* ATOM */: {
const [atomName, atomValue, atomPayload] = this.getAtomTypeFromIndex(value);
return this.builder.createAtom(atomName, atomValue, atomPayload, this.markups.slice());
}
default:
assert(`Unexpected marker type ${type}`, false);
}
}
}
/*
* Parses from mobiledoc -> post
*/
class MobiledocParser$3 {
constructor(builder) {
this.builder = builder;
}
/**
* @param {Mobiledoc}
* @return {Post}
*/
parse({ sections, markups: markerTypes, cards: cardTypes, atoms: atomTypes }) {
try {
const post = this.builder.createPost();
this.markups = [];
this.markerTypes = this.parseMarkerTypes(markerTypes);
this.cardTypes = this.parseCardTypes(cardTypes);
this.atomTypes = this.parseAtomTypes(atomTypes);
this.parseSections(sections, post);
return post;
}
catch (e) {
assert(`Unable to parse mobiledoc: ${e.message}`, false);
}
}
parseMarkerTypes(markerTypes) {
return markerTypes.map(markerType => this.parseMarkerType(markerType));
}
parseMarkerType([tagName, attributesArray]) {
const attributesObject = kvArrayToObject(attributesArray || []);
return this.builder.createMarkup(tagName, attributesObject);
}
parseCardTypes(cardTypes) {
return cardTypes.map(cardType => this.parseCardType(cardType));
}
parseCardType([cardName, cardPayload]) {
return [cardName, cardPayload];
}
parseAtomTypes(atomTypes) {
return atomTypes.map(atomType => this.parseAtomType(atomType));
}
parseAtomType([atomName, atomValue, atomPayload]) {
return [atomName, atomValue, atomPayload];
}
parseSections(sections, post) {
sections.forEach(section => this.parseSection(section, post));
}
parseSection(section, post) {
switch (section[0]) {
case 1 /* MARKUP */:
this.parseMarkupSection(section, post);
break;
case 2 /* IMAGE */:
this.parseImageSection(section, post);
break;
case 10 /* CARD */:
this.parseCardSection(section, post);
break;
case 3 /* LIST */:
this.parseListSection(section, post);
break;
default:
assert(`Unexpected section type ${section[0]}`, false);
}
}
getAtomTypeFromIndex(index) {
const atomType = this.atomTypes[index];
assert(`No atom definition found at index ${index}`, !!atomType);
return atomType;
}
getCardTypeFromIndex(index) {
const cardType = this.cardTypes[index];
assert(`No card definition found at index ${index}`, !!cardType);
return cardType;
}
parseCardSection([, cardIndex], post) {
const [name, payload] = this.getCardTypeFromIndex(cardIndex);
const section = this.builder.createCardSection(name, payload);
post.sections.append(section);
}
parseImageSection([, src], post) {
const section = this.builder.createImageSection(src);
post.sections.append(section);
}
parseMarkupSection([, tagName, markers, attributesArray], post) {
const section = this.builder.createMarkupSection(tagName);
post.sections.append(section);
if (attributesArray) {
entries(kvArrayToObject(attributesArray)).forEach(([key, value]) => {
section.setAttribute(key, value);
});
}
this.parseMarkers(markers, section);
// Strip blank markers after they have been created. This ensures any
// markup they include has been correctly populated.
filter(section.markers, m => m.isBlank).forEach(m => {
section.markers.remove(m);
});
}
parseListSection([, tagName, items, attributesArray], post) {
const section = this.builder.createListSection(tagName);
post.sections.append(section);
if (attributesArray) {
entries(kvArrayToObject(attributesArray)).forEach(([key, value]) => {
section.setAttribute(key, value);
});
}
this.parseListItems(items, section);
}
parseListItems(items, section) {
items.forEach(i => this.parseListItem(i, section));
}
parseListItem(markers, section) {
const item = this.builder.createListItem();
this.parseMarkers(markers, item);
section.items.append(item);
}
parseMarkers(markers, parent) {
markers.forEach(m => this.parseMarker(m, parent));
}
parseMarker([type, markerTypeIndexes, closeCount, value], parent) {
markerTypeIndexes.forEach(index => {
this.markups.push(this.markerTypes[index]);
});
const marker = this.buildMarkerType(type, value);
parent.markers.append(marker);
this.markups = this.markups.slice(0, this.markups.length - closeCount);
}
buildMarkerType(type, value) {
switch (type) {
case 0 /* MARKUP */:
return this.builder.createMarker(value, this.markups.slice());
case 1 /* ATOM */: {
const [atomName, atomValue, atomPayload] = this.getAtomTypeFromIndex(value);
return this.builder.createAtom(atomName, atomValue, atomPayload, this.markups.slice());
}
default:
assert(`Unexpected marker type ${type}`, false);
}
}
}
function visit(visitor, node, opcodes) {
const method = node.type;
assertExistsIn(`Cannot visit unknown type ${method}`, method, visitor);
visitor[method](node, opcodes);
}
function compile(compiler, opcodes) {
for (let i = 0, l = opcodes.length; i < l; i++) {
let [method, ...params] = opcodes[i];
compiler[method].apply(compiler, params);
}
}
function visitArray(visitor, nodes, opcodes) {
if (!nodes || nodes.length === 0) {
return;
}
forEach(nodes, node => {
visit(visitor, node, opcodes);
});
}
const MOBILEDOC_VERSION = '0.2.0';
const visitor = {
["post" /* POST */](node, opcodes) {
opcodes.push(['openPost']);
visitArray(visitor, node.sections, opcodes);
},
["markup-section" /* MARKUP_SECTION */](node, opcodes) {
opcodes.push(['openMarkupSection', node.tagName]);
visitArray(visitor, node.markers, opcodes);
},
["list-section" /* LIST_SECTION */](node, opcodes) {
opcodes.push(['openListSection', node.tagName]);
visitArray(visitor, node.items, opcodes);
},
["list-item" /* LIST_ITEM */](node, opcodes) {
opcodes.push(['openListItem']);
visitArray(visitor, node.markers, opcodes);
},
["image-section" /* IMAGE_SECTION */](node, opcodes) {
opcodes.push(['openImageSection', node.src]);
},
["card-section" /* CARD */](node, opcodes) {
opcodes.push(['openCardSection', node.name, node.payload]);
},
["marker" /* MARKER */](node, opcodes) {
opcodes.push(['openMarker', node.closedMarkups.length, node.value]);
visitArray(visitor, node.openedMarkups, opcodes);
},
["markup" /* MARKUP */](node, opcodes) {
opcodes.push(['openMarkup', node.tagName, objectToSortedKVArray(node.attributes)]);
},
};
class PostOpcodeCompiler {
openMarker(closeCount, value) {
this.markupMarkerIds = [];
this.markers.push([this.markupMarkerIds, closeCount, value || '']);
}
openMarkupSection(tagName) {
this.markers = [];
this.sections.push([1 /* MARKUP */, tagName, this.markers]);
}
openListSection(tagName) {
this.items = [];
this.sections.push([3 /* LIST */, tagName, this.items]);
}
openListItem() {
this.markers = [];
this.items.push(this.markers);
}
openImageSection(url) {
this.sections.push([2 /* IMAGE */, url]);
}
openCardSection(name, payload) {
this.sections.push([10 /* CARD */, name, payload]);
}
openPost() {
this.markerTypes = [];
this.sections = [];
this.result = {
version: MOBILEDOC_VERSION,
sections: [this.markerTypes, this.sections],
};
}
openMarkup(tagName, attributes) {
const index = this._findOrAddMarkerTypeIndex(tagName, attributes);
this.markupMarkerIds.push(index);
}
_findOrAddMarkerTypeIndex(tagName, attributesArray) {
if (!this._markerTypeCache) {
this._markerTypeCache = {};
}
const key = `${tagName}-${attributesArray.join('-')}`;
let index = this._markerTypeCache[key];
if (index === undefined) {
let markerType = [tagName];
if (attributesArray.length) {
markerType.push(attributesArray);
}
this.markerTypes.push(markerType);
index = this.markerTypes.length - 1;
this._markerTypeCache[key] = index;
}
return index;
}
}
/**
* Render from post -> mobiledoc
*/
var MobiledocRenderer_0_2 = {
/**
* @param {Post}
* @return {Mobiledoc}
*/
render(post) {
let opcodes = [];
visit(visitor, post, opcodes);
let compiler = new PostOpcodeCompiler();
compile(compiler, opcodes);
return compiler.result;
},
};
const MOBILEDOC_VERSION$1 = '0.3.0';
const visitor$1 = {
["post" /* POST */](node, opcodes) {
opcodes.push(['openPost']);
visitArray(visitor$1, node.sections, opcodes);
},
["markup-section" /* MARKUP_SECTION */](node, opcodes) {
opcodes.push(['openMarkupSection', node.tagName]);
visitArray(visitor$1, node.markers, opcodes);
},
["list-section" /* LIST_SECTION */](node, opcodes) {
opcodes.push(['openListSection', node.tagName]);
visitArray(visitor$1, node.items, opcodes);
},
["list-item" /* LIST_ITEM */](node, opcodes) {
opcodes.push(['openListItem']);
visitArray(visitor$1, node.markers, opcodes);
},
["image-section" /* IMAGE_SECTION */](node, opcodes) {
opcodes.push(['openImageSection', node.src]);
},
["card-section" /* CARD */](node, opcodes) {
opcodes.push(['openCardSection', node.name, node.payload]);
},
["marker" /* MARKER */](node, opcodes) {
opcodes.push(['openMarker', node.closedMarkups.length, node.value]);
visitArray(visitor$1, node.openedMarkups, opcodes);
},
["markup" /* MARKUP */](node, opcodes) {
opcodes.push(['openMarkup', node.tagName, objectToSortedKVArray(node.attributes)]);
},
["atom" /* ATOM */](node, opcodes) {
opcodes.push(['openAtom', node.closedMarkups.length, node.name, node.value, node.payload]);
visitArray(visitor$1, node.openedMarkups, opcodes);
},
};
class PostOpcodeCompiler$1 {
openMarker(closeCount, value) {
this.markupMarkerIds = [];
this.markers.push([0 /* MARKUP */, this.markupMarkerIds, closeCount, value || '']);
}
openAtom(closeCount, name, value, payload) {
const index = this._addAtomTypeIndex(name, value, payload);
this.markupMarkerIds = [];
this.markers.push([1 /* ATOM */, this.markupMarkerIds, closeCount, index]);
}
openMarkupSection(tagName) {
this.markers = [];
this.sections.push([1 /* MARKUP */, tagName, this.markers]);
}
openListSection(tagName) {
this.items = [];
this.sections.push([3 /* LIST */, tagName, this.items]);
}
openListItem() {
this.markers = [];
this.items.push(this.markers);
}
openImageSection(url) {
this.sections.push([2 /* IMAGE */, url]);
}
openCardSection(name, payload) {
const index = this._addCardTypeIndex(name, payload);
this.sections.push([10 /* CARD */, index]);
}
openPost() {
this.atomTypes = [];
this.cardTypes = [];
this.markerTypes = [];
this.sections = [];
this.result = {
version: MOBILEDOC_VERSION$1,
atoms: this.atomTypes,
cards: this.cardTypes,
markups: this.markerTypes,
sections: this.sections,
};
}
openMarkup(tagName, attributes) {
const index = this._findOrAddMarkerTypeIndex(tagName, attributes);
this.markupMarkerIds.push(index);
}
_addCardTypeIndex(cardName, payload) {
let cardType = [cardName, payload];
this.cardTypes.push(cardType);
return this.cardTypes.length - 1;
}
_addAtomTypeIndex(atomName, atomValue, payload) {
let atomType = [atomName, atomValue, payload];
this.atomTypes.push(atomType);
return this.atomTypes.length - 1;
}
_findOrAddMarkerTypeIndex(tagName, attributesArray) {
if (!this._markerTypeCache) {
this._markerTypeCache = {};
}
const key = `${tagName}-${attributesArray.join('-')}`;
let index = this._markerTypeCache[key];
if (index === undefined) {
let markerType = [tagName];
if (attributesArray.length) {
markerType.push(attributesArray);
}
this.markerTypes.push(markerType);
index = this.markerTypes.length - 1;
this._markerTypeCache[key] = index;
}
return index;
}
}
/**
* Render from post -> mobiledoc
*/
var MobiledocRenderer_0_3 = {
/**
* @param {Post}
* @return {Mobiledoc}
*/
render(post) {
let opcodes = [];
visit(visitor$1, post, opcodes);
let compiler = new PostOpcodeCompiler$1();
compile(compiler, opcodes);
return compiler.result;
},
};
const MOBILEDOC_VERSION$2 = '0.3.1';
const visitor$2 = {
["post" /* POST */](node, opcodes) {
opcodes.push(['openPost']);
visitArray(visitor$2, node.sections, opcodes);
},
["markup-section" /* MARKUP_SECTION */](node, opcodes) {
opcodes.push(['openMarkupSection', node.tagName]);
visitArray(visitor$2, node.markers, opcodes);
},
["list-section" /* LIST_SECTION */](node, opcodes) {
opcodes.push(['openListSection', node.tagName]);
visitArray(visitor$2, node.items, opcodes);
},
["list-item" /* LIST_ITEM */](node, opcodes) {
opcodes.push(['openListItem']);
visitArray(visitor$2, node.markers, opcodes);
},
["image-section" /* IMAGE_SECTION */](node, opcodes) {
opcodes.push(['openImageSection', node.src]);
},
["card-section" /* CARD */](node, opcodes) {
opcodes.push(['openCardSection', node.name, node.payload]);
},
["marker" /* MARKER */](node, opcodes) {
opcodes.push(['openMarker', node.closedMarkups.length, node.value]);
visitArray(visitor$2, node.openedMarkups, opcodes);
},
["markup" /* MARKUP */](node, opcodes) {
opcodes.push(['openMarkup', node.tagName, objectToSortedKVArray(node.attributes)]);
},
["atom" /* ATOM */](node, opcodes) {
opcodes.push(['openAtom', node.closedMarkups.length, node.name, node.value, node.payload]);
visitArray(visitor$2, node.openedMarkups, opcodes);
},
};
class PostOpcodeCompiler$2 {
openMarker(closeCount, value) {
this.markupMarkerIds = [];
this.markers.push([0 /* MARKUP */, this.markupMarkerIds, closeCount, value || '']);
}
openAtom(closeCount, name, value, payload) {
const index = this._addAtomTypeIndex(name, value, payload);
this.markupMarkerIds = [];
this.markers.push([1 /* ATOM */, this.markupMarkerIds, closeCount, index]);
}
openMarkupSection(tagName) {
this.markers = [];
this.sections.push([1 /* MARKUP */, tagName, this.markers]);
}
openListSection(tagName) {
this.items = [];
this.sections.push([3 /* LIST */, tagName, this.items]);
}
openListItem() {
this.markers = [];
this.items.push(this.markers);
}
openImageSection(url) {
this.sections.push([2 /* IMAGE */, url]);
}
openCardSection(name, payload) {
const index = this._addCardTypeIndex(name, payload);
this.sections.push([10 /* CARD */, index]);
}
openPost() {
this.atomTypes = [];
this.cardTypes = [];
this.markerTypes = [];
this.sections = [];
this.result = {
version: MOBILEDOC_VERSION$2,
atoms: this.atomTypes,
cards: this.cardTypes,
markups: this.markerTypes,
sections: this.sections,
};
}
openMarkup(tagName, attributes) {
const index = this._findOrAddMarkerTypeIndex(tagName, attributes);
this.markupMarkerIds.push(index);
}
_addCardTypeIndex(cardName, payload) {
let cardType = [cardName, payload];
this.cardTypes.push(cardType);
return this.cardTypes.length - 1;
}
_addAtomTypeIndex(atomName, atomValue, payload) {
let atomType = [atomName, atomValue, payload];
this.atomTypes.push(atomType);
return this.atomTypes.length - 1;
}
_findOrAddMarkerTypeIndex(tagName, attributesArray) {
if (!this._markerTypeCache) {
this._markerTypeCache = {};
}
const key = `${tagName}-${attributesArray.join('-')}`;
let index = this._markerTypeCache[key];
if (index === undefined) {
let markerType = [tagName];
if (attributesArray.length) {
markerType.push(attributesArray);
}
this.markerTypes.push(markerType);
index = this.markerTypes.length - 1;
this._markerTypeCache[key] = index;
}
return index;
}
}
/**
* Render from post -> mobiledoc
*/
var MobiledocRenderer_0_3_1 = {
/**
* @param {Post}
* @return {Mobiledoc}
*/
render(post) {
let opcodes = [];
visit(visitor$2, post, opcodes);
let compiler = new PostOpcodeCompiler$2();
compile(compiler, opcodes);
return compiler.result;
},
};
const MOBILEDOC_VERSION$3 = '0.3.2';
const visitor$3 = {
["post" /* POST */](node, opcodes) {
opcodes.push(['openPost']);
visitArray(visitor$3, node.sections, opcodes);
},
["markup-section" /* MARKUP_SECTION */](node, opcodes) {
opcodes.push(['openMarkupSection', node.tagName, objectToSortedKVArray(node.attributes)]);
visitArray(visitor$3, node.markers, opcodes);
},
["list-section" /* LIST_SECTION */](node, opcodes) {
opcodes.push(['openListSection', node.tagName, objectToSortedKVArray(node.attributes)]);
visitArray(visitor$3, node.items, opcodes);
},
["list-item" /* LIST_ITEM */](node, opcodes) {
opcodes.push(['openListItem']);
visitArray(visitor$3, node.markers, opcodes);
},
["image-section" /* IMAGE_SECTION */](node, opcodes) {
opcodes.push(['openImageSection', node.src]);
},
["card-section" /* CARD */](node, opcodes) {
opcodes.push(['openCardSection', node.name, node.payload]);
},
["marker" /* MARKER */](node, opcodes) {
opcodes.push(['openMarker', node.closedMarkups.length, node.value]);
visitArray(visitor$3, node.openedMarkups, opcodes);
},
["markup" /* MARKUP */](node, opcodes) {
opcodes.push(['openMarkup', node.tagName, objectToSortedKVArray(node.attributes)]);
},
["atom" /* ATOM */](node, opcodes) {
opcodes.push(['openAtom', node.closedMarkups.length, node.name, node.value, node.payload]);
visitArray(visitor$3, node.openedMarkups, opcodes);
},
};
class PostOpcodeCompiler$3 {
openMarker(closeCount, value) {
this.markupMarkerIds = [];
this.markers.push([0 /* MARKUP */, this.markupMarkerIds, closeCount, value || '']);
}
openAtom(closeCount, name, value, payload) {
const index = this._addAtomTypeIndex(name, value, payload);
this.markupMarkerIds = [];
this.markers.push([1 /* ATOM */, this.markupMarkerIds, closeCount, index]);
}
openMarkupSection(tagName, attributes) {
this.markers = [];
if (attributes && attributes.length !== 0) {
this.sections.push([1 /* MARKUP */, tagName, this.markers, attributes]);
}
else {
this.sections.push([1 /* MARKUP */, tagName, this.markers]);
}
}
openListSection(tagName, attributes) {
this.items = [];
if (attributes && attributes.length !== 0) {
this.sections.push([3 /* LIST */, tagName, this.items, attributes]);
}
else {
this.sections.push([3 /* LIST */, tagName, this.items]);
}
}
openListItem() {
this.markers = [];
this.items.push(this.markers);
}
openImageSection(url) {
this.sections.push([2 /* IMAGE */, url]);
}
openCardSection(name, payload) {
const index = this._addCardTypeIndex(name, payload);
this.sections.push([10 /* CARD */, index]);
}
openPost() {
this.atomTypes = [];
this.cardTypes = [];
this.markerTypes = [];
this.sections = [];
this.result = {
version: MOBILEDOC_VERSION$3,
atoms: this.atomTypes,
cards: this.cardTypes,
markups: this.markerTypes,
sections: this.sections,
};
}
openMarkup(tagName, attributes) {
const index = this._findOrAddMarkerTypeIndex(tagName, attributes);
this.markupMarkerIds.push(index);
}
_addCardTypeIndex(cardName, payload) {
let cardType = [cardName, payload];
this.cardTypes.push(cardType);
return this.cardTypes.length - 1;
}
_addAtomTypeIndex(atomName, atomValue, payload) {
let atomType = [atomName, atomValue, payload];
this.atomTypes.push(atomType);
return this.atomTypes.length - 1;
}
_findOrAddMarkerTypeIndex(tagName, attributesArray) {
if (!this._markerTypeCache) {
this._markerTypeCache = {};
}
const key = `${tagName}-${attributesArray.join('-')}`;
let index = this._markerTypeCache[key];
if (index === undefined) {
let markerType = [tagName];
if (attributesArray.length) {
markerType.push(attributesArray);
}
this.markerTypes.push(markerType);
index = this.markerTypes.length - 1;
this._markerTypeCache[key] = index;
}
return index;
}
}
/**
* Render from post -> mobiledoc
*/
var MobiledocRenderer_0_3_2 = {
/**
* @param {Post}
* @return {Mobiledoc}
*/
render(post) {
let opcodes = [];
visit(visitor$3, post, opcodes);
let compiler = new PostOpcodeCompiler$3();
compile(compiler, opcodes);
return compiler.result;
},
};
var mobiledocParsers = {
parse(builder, mobiledoc) {
switch (mobiledoc.version) {
case MOBILEDOC_VERSION:
return new MobiledocParser(builder).parse(mobiledoc);
case MOBILEDOC_VERSION$1:
return new MobiledocParser$1(builder).parse(mobiledoc);
case MOBILEDOC_VERSION$2:
return new MobiledocParser$2(builder).parse(mobiledoc);
case MOBILEDOC_VERSION$3:
return new MobiledocParser$3(builder).parse(mobiledoc);
default:
assert(`Unknown version of mobiledoc parser requested: ${mobiledoc.version}`, false);
}
},
};
class CardNode {
constructor(editor, card, section, element, options) {
this._rendered = null;
this._teardownCallback = null;
this._didRenderCallback = null;
this.editor = editor;
this.card = card;
this.section = section;
this.element = element;
this.options = options;
}
render(mode) {
if (this.mode === mode) {
return;
}
this.teardown();
this.mode = mode;
let methodName = mode === 'display' ? 'render' : 'edit';
let method = this.card[methodName];
assert(`Card is missing "${methodName}" (tried to render mode: "${mode}")`, !!method);
let rendered = method({
env: this.env,
options: this.options,
payload: this.section.payload,
}) || null;
this._validateAndAppendRenderResult(rendered);
}
teardown() {
if (this._teardownCallback) {
this._teardownCallback();
this._teardownCallback = null;
}
if (this._rendered) {
this.element.removeChild(this._rendered);
this._rendered = null;
}
}
didRender() {
if (this._didRenderCallback) {
this._didRenderCallback();
}
}
get env() {
return {
name: this.card.name,
isInEditor: true,
onTeardown: (callback) => (this._teardownCallback = callback),
didRender: (callback) => (this._didRenderCallback = callback),
edit: () => this.edit(),
save: (payload, transition = true) => {
this.section.payload = payload;
this.editor._postDidChange();
if (transition) {
this.display();
}
},
cancel: () => this.display(),
remove: () => this.remove(),
postModel: this.section,
};
}
display() {
this.render(CardMode.DISPLAY);
}
edit() {
this.render(CardMode.EDIT);
}
remove() {
this.editor.run((postEditor) => postEditor.removeSection(this.section));
}
_validateAndAppendRenderResult(rendered) {
if (!rendered) {
return;
}
let { card: { name }, } = this;
assert(`Card "${name}" must render dom (render value was: "${rendered}")`, !!rendered.nodeType);
this.element.appendChild(rendered);
this._rendered = rendered;
this.didRender();
}
}
class AtomNode {
constructor(editor, atom, model, element, atomOptions) {
this._teardownCallback = null;
this.editor = editor;
this.atom = atom;
this.model = model;
this.atomOptions = atomOptions;
this.element = element;
}
render() {
if (!this._rendered) {
let { atomOptions: options, env, model: { value, payload }, } = this;
// cache initial render
this._rendered = this.atom.render({ options, env, value, payload }) || null;
}
this._validateAndAppendRenderResult(this._rendered);
}
get env() {
return {
name: this.atom.name,
onTeardown: (callback) => (this._teardownCallback = callback),
save: (value, payload = {}) => {
this.model.value = value;
this.model.payload = payload;
this.editor._postDidChange();
this.teardown();
this.render();
},
};
}
teardown() {
if (this._teardownCallback) {
this._teardownCallback();
this._teardownCallback = null;
}
if (this._rendered) {
this.element.removeChild(this._rendered);
this._rendered = null;
}
}
_validateAndAppendRenderResult(rendered) {
if (!rendered) {
return;
}
let { atom: { name }, } = this;
assert(`Atom "${name}" must return a DOM node (returned value was: "${rendered}")`, !!rendered.nodeType);
this.element.appendChild(rendered);
}
}
const CARD_ELEMENT_CLASS_NAME = '__mobiledoc-card';
const NO_BREAK_SPACE = '\u00A0';
const TAB_CHARACTER = '\u2003';
const SPACE = ' ';
const ZWNJ = '\u200c';
const ATOM_CLASS_NAME = '-mobiledoc-kit__atom';
const EDITOR_HAS_NO_CONTENT_CLASS_NAME = '__has-no-content';
const EDITOR_ELEMENT_CLASS_NAME = '__mobiledoc-editor';
function createElementFromMarkup(doc, markup) {
let element = doc.createElement(markup.tagName);
Object.keys(markup.attributes).forEach(k => {
element.setAttribute(k, markup.attributes[k]);
});
return element;
}
const TWO_SPACES = `${SPACE}${SPACE}`;
const SPACE_AND_NO_BREAK = `${SPACE}${NO_BREAK_SPACE}`;
const SPACES_REGEX = new RegExp(TWO_SPACES, 'g');
const TAB_REGEX = new RegExp(TAB, 'g');
const endsWithSpace = function (text) {
return endsWith(text, SPACE);
};
const startsWithSpace = function (text) {
return startsWith(text, SPACE);
};
// FIXME: This can be done more efficiently with a single pass
// building a correct string based on the original.
function renderHTMLText(marker) {
let text = marker.value;
text = text.replace(SPACES_REGEX, SPACE_AND_NO_BREAK).replace(TAB_REGEX, TAB_CHARACTER);
// If the first marker has a leading space or the last marker has a
// trailing space, the browser will collapse the space when we position
// the cursor.
// See https://github.com/bustle/mobiledoc-kit/issues/68
// and https://github.com/bustle/mobiledoc-kit/issues/75
if (marker.isMarker && endsWithSpace(text) && !marker.next) {
text = text.substr(0, text.length - 1) + NO_BREAK_SPACE;
}
if (marker.isMarker &&
startsWithSpace(text) &&
(!marker.prev || (marker.prev.isMarker && endsWithSpace(marker.prev.value)))) {
text = NO_BREAK_SPACE + text.substr(1);
}
return text;
}
// ascends from element upward, returning the last parent node that is not
// parentElement
function penultimateParentOf(element, parentElement) {
while (parentElement &&
element.parentNode !== parentElement &&
element.parentNode !== document.body // ensure the while loop stops
) {
element = element.parentNode;
}
return element;
}
function setSectionAttributesOnElement(section, element) {
section.eachAttribute((key, value) => {
element.setAttribute(key, value);
});
}
function renderMarkupSection(section) {
let element;
if (MARKUP_SECTION_ELEMENT_NAMES.indexOf(section.tagName) !== -1) {
element = document.createElement(section.tagName);
}
else {
element = document.createElement('div');
addClassName(element, section.tagName);
}
setSectionAttributesOnElement(section, element);
return element;
}
function renderListSection(section) {
let element = document.createElement(section.tagName);
setSectionAttributesOnElement(section, element);
return element;
}
function renderListItem() {
return document.createElement('li');
}
function renderCursorPlaceholder() {
return document.createElement('br');
}
function renderInlineCursorPlaceholder() {
return document.createTextNode(ZWNJ);
}
function renderCard() {
let wrapper = document.createElement('div');
let cardElement = document.createElement('div');
cardElement.contentEditable = 'false';
addClassName(cardElement, CARD_ELEMENT_CLASS_NAME);
wrapper.appendChild(renderInlineCursorPlaceholder());
wrapper.appendChild(cardElement);
wrapper.appendChild(renderInlineCursorPlaceholder());
return { wrapper, cardElement };
}
/**
* Wrap the element in all of the opened markups
* @return {DOMElement} the wrapped element
* @private
*/
function wrapElement(element, openedMarkups) {
let wrappedElement = element;
for (let i = openedMarkups.length - 1; i >= 0; i--) {
let markup = openedMarkups[i];
let openedElement = createElementFromMarkup(document, markup);
openedElement.appendChild(wrappedElement);
wrappedElement = openedElement;
}
return wrappedElement;
}
// Attach the element to its parent element at the correct position based on the
// previousRenderNode
function attachElementToParent(element, parentElement, previousRenderNode = null) {
if (previousRenderNode) {
let previousSibling = previousRenderNode.element;
let previousSiblingPenultimate = penultimateParentOf(previousSibling, parentElement);
parentElement.insertBefore(element, previousSiblingPenultimate.nextSibling);
}
else {
parentElement.insertBefore(element, parentElement.firstChild);
}
}
function renderAtom(atom, element, previousRenderNode) {
let atomElement = document.createElement('span');
atomElement.contentEditable = 'false';
let wrapper = document.createElement('span');
addClassName(wrapper, ATOM_CLASS_NAME);
let headTextNode = renderInlineCursorPlaceholder();
let tailTextNode = renderInlineCursorPlaceholder();
wrapper.appendChild(headTextNode);
wrapper.appendChild(atomElement);
wrapper.appendChild(tailTextNode);
let wrappedElement = wrapElement(wrapper, atom.openedMarkups);
attachElementToParent(wrappedElement, element, previousRenderNode);
return {
markupElement: wrappedElement,
wrapper,
atomElement,
headTextNode,
tailTextNode,
};
}
function getNextMarkerElement(renderNode) {
let element = renderNode.element.parentNode;
let marker = renderNode.postNode;
let closedCount = marker.closedMarkups.length;
while (closedCount--) {
element = element.parentNode;
}
return element;
}
/**
* Render the marker
* @param {Marker} marker the marker to render
* @param {DOMNode} element the element to attach the rendered marker to
* @param {RenderNode} [previousRenderNode] The render node before this one, which
* affects the determination of where to insert this rendered marker.
* @return {Object} With properties `element` and `markupElement`.
* The node (textNode) that has the text for
* this marker, and the outermost rendered element. If the marker has no
* markups, element and markupElement will be the same textNode
* @private
*/
function renderMarker(marker, parentElement, previousRenderNode) {
let text = renderHTMLText(marker);
let element = document.createTextNode(text);
let markupElement = wrapElement(element, marker.openedMarkups);
attachElementToParent(markupElement, parentElement, previousRenderNode);
return { element, markupElement };
}
// Attach the render node's element to the DOM,
// replacing the originalElement if it exists
function attachRenderNodeElementToDOM(renderNode, originalElement = null) {
const element = unwrap(renderNode.element);
assertNotNull('expected RenderNode to have a parent', renderNode.parent);
if (originalElement) {
// RenderNode has already rendered
let parentElement = renderNode.parent.element;
parentElement.replaceChild(element, originalElement);
}
else {
// RenderNode has not yet been rendered
let parentElement;
let nextSiblingElement;
if (renderNode.prev) {
let previousElement = unwrap(renderNode.prev.element);
parentElement = unwrap(previousElement.parentNode);
nextSiblingElement = previousElement.nextSibling;
}
else {
parentElement = renderNode.parent.element;
nextSiblingElement = parentElement.firstChild;
}
parentElement.insertBefore(element, nextSiblingElement);
}
}
function removeRenderNodeSectionFromParent(renderNode, section) {
assertNotNull('expected RenderNode to have a parent', renderNode.parent);
assertNotNull('expected parent RenderNode to have a PostNode', renderNode.parent.postNode);
const parent = renderNode.parent.postNode;
assert('expected PostNode to have sections', hasChildSections(parent));
parent.sections.remove(section);
}
function removeRenderNodeElementFromParent(renderNode) {
if (renderNode.element && renderNode.element.parentNode) {
renderNode.element.parentNode.removeChild(renderNode.element);
}
}
function validateCards(cards = []) {
forEach(cards, card => {
assert(`Card "${card.name}" must define type "dom", has: "${card.type}"`, card.type === 'dom');
assert(`Card "${card.name}" must define \`render\` method`, !!card.render);
});
return cards;
}
function validateAtoms(atoms = []) {
forEach(atoms, atom => {
assert(`Atom "${atom.name}" must define type "dom", has: "${atom.type}"`, atom.type === 'dom');
assert(`Atom "${atom.name}" must define \`render\` method`, !!atom.render);
});
return atoms;
}
class Visitor$1 {
constructor(editor, cards, atoms, unknownCardHandler, unknownAtomHandler, options) {
this.editor = editor;
this.cards = validateCards(cards);
this.atoms = validateAtoms(atoms);
this.unknownCardHandler = unknownCardHandler;
this.unknownAtomHandler = unknownAtomHandler;
this.options = options;
}
_findCard(cardName) {
let card = detect(this.cards, card => card.name === cardName);
return card || this._createUnknownCard(cardName);
}
_createUnknownCard(cardName) {
assert(`Unknown card "${cardName}" found, but no unknownCardHandler is defined`, !!this.unknownCardHandler);
return {
name: cardName,
type: 'dom',
render: this.unknownCardHandler,
edit: this.unknownCardHandler,
};
}
_findAtom(atomName) {
let atom = detect(this.atoms, atom => atom.name === atomName);
return atom || this._createUnknownAtom(atomName);
}
_createUnknownAtom(atomName) {
assert(`Unknown atom "${atomName}" found, but no unknownAtomHandler is defined`, !!this.unknownAtomHandler);
return {
name: atomName,
type: 'dom',
render: this.unknownAtomHandler,
};
}
["post" /* POST */](renderNode, post, visit) {
if (!renderNode.element) {
renderNode.element = document.createElement('div');
}
let element = renderNode.element;
addClassName(element, EDITOR_ELEMENT_CLASS_NAME);
if (post.hasContent) {
removeClassName(element, EDITOR_HAS_NO_CONTENT_CLASS_NAME);
}
else {
addClassName(element, EDITOR_HAS_NO_CONTENT_CLASS_NAME);
}
visit(renderNode, post.sections);
}
["markup-section" /* MARKUP_SECTION */](renderNode, section, visit) {
const originalElement = renderNode.element;
// Always rerender the section -- its tag name or attributes may have changed.
// TODO make this smarter, only rerendering and replacing the element when necessary
renderNode.element = renderMarkupSection(section);
renderNode.cursorElement = null;
attachRenderNodeElementToDOM(renderNode, originalElement);
if (section.isBlank) {
let cursorPlaceholder = renderCursorPlaceholder();
renderNode.element.appendChild(cursorPlaceholder);
renderNode.cursorElement = cursorPlaceholder;
}
else {
const visitAll = true;
visit(renderNode, section.markers, visitAll);
}
}
["list-section" /* LIST_SECTION */](renderNode, section, visit) {
const originalElement = renderNode.element;
renderNode.element = renderListSection(section);
attachRenderNodeElementToDOM(renderNode, originalElement);
const visitAll = true;
visit(renderNode, section.items, visitAll);
}
["list-item" /* LIST_ITEM */](renderNode, item, visit) {
// FIXME do we need to do anything special for rerenders?
renderNode.element = renderListItem();
renderNode.cursorElement = null;
attachRenderNodeElementToDOM(renderNode, null);
if (item.isBlank) {
let cursorPlaceholder = renderCursorPlaceholder();
renderNode.element.appendChild(cursorPlaceholder);
renderNode.cursorElement = cursorPlaceholder;
}
else {
const visitAll = true;
visit(renderNode, item.markers, visitAll);
}
}
["marker" /* MARKER */](renderNode, marker) {
let parentElement;
if (renderNode.prev) {
parentElement = getNextMarkerElement(renderNode.prev);
}
else {
parentElement = renderNode.parent.element;
}
let { element, markupElement } = renderMarker(marker, parentElement, renderNode.prev);
renderNode.element = element;
renderNode.markupElement = markupElement;
}
["image-section" /* IMAGE_SECTION */](renderNode, section) {
if (renderNode.element) {
if (renderNode.element.src !== section.src) {
renderNode.element.src = section.src || '';
}
}
else {
let element = document.createElement('img');
element.src = section.src || '';
if (renderNode.prev) {
let previousElement = renderNode.prev.element;
let nextElement = previousElement.nextSibling;
if (nextElement) {
nextElement.parentNode.insertBefore(element, nextElement);
}
}
if (!element.parentNode) {
renderNode.parent.element.appendChild(element);
}
renderNode.element = element;
}
}
["card-section" /* CARD */](renderNode, section) {
const originalElement = renderNode.element;
const { editor, options } = this;
const card = this._findCard(section.name);
let { wrapper, cardElement } = renderCard();
renderNode.element = wrapper;
attachRenderNodeElementToDOM(renderNode, originalElement);
const cardNode = new CardNode(editor, card, section, cardElement, options);
renderNode.cardNode = cardNode;
const initialMode = section._initialMode;
cardNode[initialMode]();
}
["atom" /* ATOM */](renderNode, atomModel) {
let parentElement;
if (renderNode.prev) {
parentElement = getNextMarkerElement(renderNode.prev);
}
else {
parentElement = renderNode.parent.element;
}
const { editor, options } = this;
const { wrapper, markupElement, atomElement, headTextNode, tailTextNode } = renderAtom(atomModel, parentElement, renderNode.prev);
const atom = this._findAtom(atomModel.name);
let atomNode = renderNode.atomNode;
if (!atomNode) {
// create new AtomNode
atomNode = new AtomNode(editor, atom, atomModel, atomElement, options);
}
else {
// retarget atomNode to new atom element
atomNode.element = atomElement;
}
atomNode.render();
renderNode.atomNode = atomNode;
renderNode.element = wrapper;
renderNode.headTextNode = headTextNode;
renderNode.tailTextNode = tailTextNode;
renderNode.markupElement = markupElement;
}
}
let destroyHooks = {
["post" /* POST */]( /*renderNode, post*/) {
assert('post destruction is not supported by the renderer', false);
},
["markup-section" /* MARKUP_SECTION */](renderNode, section) {
removeRenderNodeSectionFromParent(renderNode, section);
removeRenderNodeElementFromParent(renderNode);
},
["list-section" /* LIST_SECTION */](renderNode, section) {
removeRenderNodeSectionFromParent(renderNode, section);
removeRenderNodeElementFromParent(renderNode);
},
["list-item" /* LIST_ITEM */](renderNode, li) {
removeRenderNodeSectionFromParent(renderNode, li);
removeRenderNodeElementFromParent(renderNode);
},
["marker" /* MARKER */](renderNode, marker) {
// FIXME before we render marker, should delete previous renderNode's element
// and up until the next marker element
// If an atom throws during render we may end up later destroying a renderNode
// that has not rendered yet, so exit early here if so.
if (!renderNode.isRendered) {
return;
}
let { markupElement } = renderNode;
if (marker.section) {
marker.section.markers.remove(marker);
}
if (markupElement.parentNode) {
// if no parentNode, the browser already removed this element
markupElement.parentNode.removeChild(markupElement);
}
},
["image-section" /* IMAGE_SECTION */](renderNode, section) {
removeRenderNodeSectionFromParent(renderNode, section);
removeRenderNodeElementFromParent(renderNode);
},
["card-section" /* CARD */](renderNode, section) {
if (renderNode.cardNode) {
renderNode.cardNode.teardown();
}
removeRenderNodeSectionFromParent(renderNode, section);
removeRenderNodeElementFromParent(renderNode);
},
["atom" /* ATOM */](renderNode, atom) {
if (renderNode.atomNode) {
renderNode.atomNode.teardown();
}
// an atom is a kind of marker so just call its destroy hook vs copying here
destroyHooks["marker" /* MARKER */](renderNode, atom);
},
};
// removes children from parentNode (a RenderNode) that are scheduled for removal
function removeDestroyedChildren(parentNode, forceRemoval = false) {
let child = parentNode.childNodes.head;
let nextChild, method;
while (child) {
nextChild = child.next;
if (child.isRemoved || forceRemoval) {
removeDestroyedChildren(child, true);
method = child.postNode.type;
assertExistsIn(`editor-dom cannot destroy "${method}"`, method, destroyHooks);
destroyHooks[method](child, child.postNode);
parentNode.childNodes.remove(child);
}
child = nextChild;
}
}
// Find an existing render node for the given postNode, or
// create one, insert it into the tree, and return it
function lookupNode(renderTree, parentNode, postNode, previousNode) {
if (postNode.renderNode) {
return postNode.renderNode;
}
else {
const renderNode = renderTree.buildRenderNode(postNode);
parentNode.childNodes.insertAfter(renderNode, previousNode);
return renderNode;
}
}
class Renderer {
constructor(editor, cards, atoms, unknownCardHandler, unknownAtomHandler, options) {
this.renderTree = null;
this.editor = editor;
this.visitor = new Visitor$1(editor, cards, atoms, unknownCardHandler, unknownAtomHandler, options);
this.nodes = [];
this.hasRendered = false;
}
destroy() {
if (!this.hasRendered) {
return;
}
let renderNode = unwrap(this.renderTree).rootNode;
let force = true;
removeDestroyedChildren(renderNode, force);
}
visit(renderTree, parentNode, postNodes, visitAll = false) {
let previousNode;
postNodes.forEach(postNode => {
let node = lookupNode(renderTree, parentNode, postNode, previousNode);
if (node.isDirty || visitAll) {
this.nodes.push(node);
}
previousNode = node;
});
}
render(renderTree) {
this.hasRendered = true;
this.renderTree = renderTree;
let renderNode = renderTree.rootNode;
let method;
let postNode;
while (renderNode) {
removeDestroyedChildren(renderNode);
postNode = renderNode.postNode;
method = postNode.type;
assertExistsIn(`EditorDom visitor cannot handle type ${method}`, method, this.visitor);
this.visitor[method](renderNode, postNode, (...args) => this.visit(renderTree, ...args));
renderNode.markClean();
renderNode = this.nodes.shift();
}
}
}
const VALID_MARKUP_TAGNAMES = [
'a',
'b',
'code',
'em',
'i',
's',
'del',
'strong',
'sub',
'sup',
'u',
].map(normalizeTagName);
const VALID_ATTRIBUTES$1 = ['href', 'rel'];
/**
* A Markup is similar with an inline HTML tag that might be added to
* text to modify its meaning and/or display. Examples of types of markup
* that could be added are bold ('b'), italic ('i'), strikethrough ('s'), and `a` tags (links).
* @property {String} tagName
*/
class Markup {
/*
* @param {Object} attributes key-values
*/
constructor(tagName, attributes = {}) {
this.type = MARKUP_TYPE;
assert(`Cannot create markup of tagName ${tagName}`, VALID_MARKUP_TAGNAMES.indexOf(tagName) !== -1);
this.tagName = normalizeTagName(tagName);
assert('Must use attributes object param (not array) for Markup', !Array.isArray(attributes));
this.attributes = filterObject(attributes, VALID_ATTRIBUTES$1);
}
/**
* Whether text in the forward direction of the cursor (i.e. to the right in ltr text)
* should be considered to have this markup applied to it.
* @private
*/
isForwardInclusive() {
return this.tagName === normalizeTagName('a') ? false : true;
}
isBackwardInclusive() {
return false;
}
hasTag(tagName) {
return this.tagName === normalizeTagName(tagName);
}
/**
* Returns the attribute value
* @param {String} name, e.g. "href"
*/
getAttribute(name) {
return this.attributes[name];
}
static isValidElement(element) {
const tagName = normalizeTagName(element.tagName);
return VALID_MARKUP_TAGNAMES.indexOf(tagName) !== -1;
}
}
const SKIPPABLE_ELEMENT_TAG_NAMES = ['style', 'head', 'title', 'meta'].map(normalizeTagName);
const NEWLINES = /\s*\n\s*/g;
function sanitize(text) {
return text.replace(NEWLINES, ' ');
}
class SectionParser {
constructor(builder, options = {}) {
this.builder = builder;
this.plugins = options.plugins || [];
}
parse(element) {
if (this._isSkippable(element)) {
return [];
}
this.sections = [];
this.state = {};
this._updateStateFromElement(element);
let finished = false;
// top-level text nodes will be run through parseNode later so avoid running
// the node through parserPlugins twice
if (!isTextNode(element)) {
finished = this.runPlugins(element);
}
if (!finished) {
let childNodes = isTextNode(element) ? [element] : element.childNodes;
forEach(childNodes, el => {
this.parseNode(el);
});
}
this._closeCurrentSection();
return this.sections;
}
runPlugins(node) {
let isNodeFinished = false;
let env = {
addSection: (section) => {
// avoid creating empty paragraphs due to wrapper elements around
// parser-plugin-handled elements
if (this.state.section && isMarkerable$1(this.state.section) && !this.state.section.text && !this.state.text) {
this.state.section = null;
}
else {
this._closeCurrentSection();
}
this.sections.push(section);
},
addMarkerable: (marker) => {
let { state } = this;
let { section } = state;
// if the first element doesn't create it's own state and it's plugin
// handler uses `addMarkerable` we won't have a section yet
if (!section) {
state.text = '';
state.section = this.builder.createMarkupSection(normalizeTagName('p'));
section = state.section;
}
assertType('Markerables can only be appended to markup sections and list item sections', section, section && section.isMarkerable);
if (state.text) {
this._createMarker();
}
section.markers.append(marker);
},
nodeFinished() {
isNodeFinished = true;
},
};
for (let i = 0; i < this.plugins.length; i++) {
let plugin = this.plugins[i];
plugin(node, this.builder, env);
if (isNodeFinished) {
return true;
}
}
return false;
}
/* eslint-disable complexity */
parseNode(node) {
if (!this.state.section) {
this._updateStateFromElement(node);
}
let nodeFinished = this.runPlugins(node);
if (nodeFinished) {
return;
}
// handle closing the current section and starting a new one if we hit a
// new-section-creating element.
if (this.state.section && isElementNode(node) && node.tagName) {
let tagName = normalizeTagName(node.tagName);
let isListSection = contains(VALID_LIST_SECTION_TAGNAMES, tagName);
let isListItem = contains(VALID_LIST_ITEM_TAGNAMES, tagName);
let isMarkupSection$1 = contains(VALID_MARKUP_SECTION_TAGNAMES, tagName);
let isNestedListSection = isListSection && this.state.section.isListItem;
let lastSection = this.sections[this.sections.length - 1];
// lists can continue after breaking out for a markup section,
// in that situation, start a new list using the same list type
if (isListItem && isMarkupSection(this.state.section)) {
this._closeCurrentSection();
this._updateStateFromElement(node.parentElement);
}
// we can hit a list item after parsing a nested list, when that happens
// and the lists are of different types we need to make sure we switch
// the list type back
if (isListItem && lastSection && isListSection$1(lastSection)) {
let parentElement = expect(node.parentElement, 'expected node to have parent element');
let parentElementTagName = normalizeTagName(parentElement.tagName);
if (parentElementTagName !== lastSection.tagName) {
this._closeCurrentSection();
this._updateStateFromElement(parentElement);
}
}
// if we've broken out of a list due to nested section-level elements we
// can hit the next list item without having a list section in the current
// state. In this instance we find the parent list node and use it to
// re-initialize the state with a new list section
if (isListItem &&
!(this.state.section.isListItem || this.state.section.isListSection) &&
!lastSection.isListSection) {
this._closeCurrentSection();
this._updateStateFromElement(node.parentElement);
}
// if we have consecutive list sections of different types (ul, ol) then
// ensure we close the current section and start a new one
let isNewListSection = lastSection &&
isListSection$1(lastSection) &&
this.state.section.isListItem &&
isListSection &&
tagName !== lastSection.tagName;
if (isNewListSection || (isListSection && !isNestedListSection) || isMarkupSection$1 || isListItem) {
// don't break out of the list for list items that contain a single .
// deals with typical case of
Text
Text
if (this.state.section.isListItem &&
tagName === 'p' &&
!node.nextSibling &&
contains(VALID_LIST_ITEM_TAGNAMES, normalizeTagName(expect(node.parentElement, 'expected node to have parent element').tagName))) {
this.parseElementNode(node);
return;
}
// avoid creating empty paragraphs due to wrapper elements around
// section-creating elements
if (isMarkerable$1(this.state.section) && !this.state.text && this.state.section.markers.length === 0) {
this.state.section = null;
}
else {
this._closeCurrentSection();
}
this._updateStateFromElement(node);
}
if (this.state.section && this.state.section.isListSection) {
// ensure the list section is closed and added to the sections list.
// _closeCurrentSection handles pushing list items onto the list section
this._closeCurrentSection();
forEach(node.childNodes, node => {
this.parseNode(node);
});
return;
}
}
switch (node.nodeType) {
case NODE_TYPES.TEXT:
this.parseTextNode(node);
break;
case NODE_TYPES.ELEMENT:
this.parseElementNode(node);
break;
}
}
parseElementNode(element) {
let { state } = this;
assert('expected markups to be non-null', state.markups);
const markups = this._markupsFromElement(element);
if (markups.length && state.text.length && isMarkerable$1(state.section)) {
this._createMarker();
}
state.markups.push(...markups);
forEach(element.childNodes, node => {
this.parseNode(node);
});
if (markups.length && state.text.length && state.section.isMarkerable) {
// create the marker started for this node
this._createMarker();
}
// pop the current markups from the stack
state.markups.splice(-markups.length, markups.length);
}
parseTextNode(textNode) {
let { state } = this;
state.text += sanitize(textNode.textContent);
}
_updateStateFromElement(element) {
if (isCommentNode(element)) {
return;
}
let { state } = this;
state.section = this._createSectionFromElement(element);
state.markups = this._markupsFromElement(element);
state.text = '';
}
_closeCurrentSection() {
let { sections, state } = this;
let lastSection = sections[sections.length - 1];
if (!state.section) {
return;
}
// close a trailing text node if it exists
if (state.text.length && state.section.isMarkerable) {
this._createMarker();
}
// push listItems onto the listSection or add a new section
if (isListItem(state.section) && lastSection && isListSection$1(lastSection)) {
trimSectionText(state.section);
lastSection.items.append(state.section);
}
else {
// avoid creating empty markup sections, especially useful for indented source
if (isMarkerable$1(state.section) &&
!state.section.text.trim() &&
!any(state.section.markers, marker => marker.isAtom)) {
state.section = null;
state.text = '';
return;
}
// remove empty list sections before creating a new section
if (lastSection && isListSection$1(lastSection) && lastSection.items.length === 0) {
sections.pop();
}
sections.push(state.section);
}
state.section = null;
state.text = '';
}
_markupsFromElement(element) {
let { builder } = this;
let markups = [];
if (isTextNode(element)) {
return markups;
}
const tagName = normalizeTagName(element.tagName);
if (this._isValidMarkupForElement(tagName, element)) {
markups.push(builder.createMarkup(tagName, getAttributes(element)));
}
this._markupsFromElementStyle(element).forEach(markup => markups.push(markup));
return markups;
}
_isValidMarkupForElement(tagName, element) {
if (VALID_MARKUP_TAGNAMES.indexOf(tagName) === -1) {
return false;
}
else if (tagName === 'b') {
// google docs add a that should not
// create a "b" markup
return element.style.fontWeight !== 'normal';
}
return true;
}
_markupsFromElementStyle(element) {
let { builder } = this;
let markups = [];
let { fontStyle, fontWeight } = element.style;
if (fontStyle === 'italic') {
markups.push(builder.createMarkup('em'));
}
if (fontWeight === 'bold' || fontWeight === '700') {
markups.push(builder.createMarkup('strong'));
}
return markups;
}
_createMarker() {
let { state } = this;
let text = transformHTMLText(state.text);
let marker = this.builder.createMarker(text, state.markups);
assertType('expected section to be markerable', state.section, isMarkerable$1(state.section));
state.section.markers.append(marker);
state.text = '';
}
_getSectionDetails(element) {
let sectionType, tagName, inferredTagName = false;
if (isTextNode(element)) {
tagName = DEFAULT_TAG_NAME$1;
sectionType = MARKUP_SECTION_TYPE;
inferredTagName = true;
}
else {
tagName = normalizeTagName(element.tagName);
// blockquote>p is valid html and should be treated as a blockquote section
// rather than a plain markup section
if (tagName === 'p' &&
element.parentElement &&
normalizeTagName(element.parentElement.tagName) === 'blockquote') {
tagName = 'blockquote';
}
if (contains(VALID_LIST_SECTION_TAGNAMES, tagName)) {
sectionType = LIST_SECTION_TYPE;
}
else if (contains(VALID_LIST_ITEM_TAGNAMES, tagName)) {
sectionType = LIST_ITEM_TYPE;
}
else if (contains(VALID_MARKUP_SECTION_TAGNAMES, tagName)) {
sectionType = MARKUP_SECTION_TYPE;
}
else {
sectionType = MARKUP_SECTION_TYPE;
tagName = DEFAULT_TAG_NAME$1;
inferredTagName = true;
}
}
return { sectionType, tagName, inferredTagName };
}
_createSectionFromElement(element) {
if (isCommentNode(element)) {
return;
}
let { builder } = this;
let section;
let { tagName, sectionType, inferredTagName } = this._getSectionDetails(element);
switch (sectionType) {
case LIST_SECTION_TYPE:
section = builder.createListSection(tagName);
break;
case LIST_ITEM_TYPE:
section = builder.createListItem();
break;
case MARKUP_SECTION_TYPE:
section = builder.createMarkupSection(tagName);
section._inferredTagName = inferredTagName;
break;
default:
assert('Cannot parse section from element', false);
}
return section;
}
_isSkippable(element) {
return isElementNode(element) && contains(SKIPPABLE_ELEMENT_TAG_NAMES, normalizeTagName(element.tagName));
}
}
const GOOGLE_DOCS_CONTAINER_ID_REGEX = /^docs-internal-guid/;
const NO_BREAK_SPACE_REGEX = new RegExp(NO_BREAK_SPACE, 'g');
const TAB_CHARACTER_REGEX = new RegExp(TAB_CHARACTER, 'g');
function transformHTMLText(textContent) {
let text = textContent;
text = text.replace(NO_BREAK_SPACE_REGEX, ' ');
text = text.replace(TAB_CHARACTER_REGEX, TAB);
return text;
}
function trimSectionText(section) {
if (isMarkerable$1(section) && section.markers.length) {
let { head, tail } = section.markers;
head.value = head.value.replace(/^\s+/, '');
tail.value = tail.value.replace(/\s+$/, '');
}
}
function isGoogleDocsContainer(element) {
return (isElementNode(element) &&
normalizeTagName(element.tagName) === normalizeTagName('b') &&
GOOGLE_DOCS_CONTAINER_ID_REGEX.test(element.id));
}
function detectRootElement(element) {
let childNodes = element.childNodes || [];
let googleDocsContainer = detect(childNodes, isGoogleDocsContainer);
if (googleDocsContainer) {
return googleDocsContainer;
}
else {
return element;
}
}
const TAG_REMAPPING = {
b: 'strong',
i: 'em',
};
function remapTagName(tagName) {
let normalized = normalizeTagName(tagName);
let remapped = TAG_REMAPPING[normalized];
return remapped || normalized;
}
function trim(str) {
return str.replace(/^\s+/, '').replace(/\s+$/, '');
}
function walkMarkerableNodes(parent, callback) {
let currentNode = parent;
if (isTextNode(currentNode) || (isElementNode(currentNode) && currentNode.classList.contains(ATOM_CLASS_NAME))) {
callback(currentNode);
}
else {
currentNode = currentNode.firstChild;
while (currentNode) {
walkMarkerableNodes(currentNode, callback);
currentNode = currentNode.nextSibling;
}
}
}
/**
* Parses DOM element -> Post
* @private
*/
class DOMParser {
constructor(builder, options = {}) {
this.builder = builder;
this.sectionParser = new SectionParser(this.builder, options);
}
parse(element) {
const post = this.builder.createPost();
let rootElement = detectRootElement(element);
this._eachChildNode(rootElement, child => {
let sections = this.parseSections(child);
this.appendSections(post, sections);
});
// trim leading/trailing whitespace of markerable sections to avoid
// unnessary whitespace from indented HTML input
forEach(post.sections, section => trimSectionText(section));
return post;
}
appendSections(post, sections) {
forEach(sections, section => this.appendSection(post, section));
}
appendSection(post, section) {
if (section.isBlank ||
(isMarkerable$1(section) && trim(section.text) === '' && !any(section.markers, marker => marker.isAtom))) {
return;
}
let lastSection = post.sections.tail;
if (lastSection &&
hasInferredTagName(lastSection) &&
hasInferredTagName(section) &&
lastSection.tagName === section.tagName) {
lastSection.join(section);
}
else {
post.sections.append(section);
}
}
_eachChildNode(element, callback) {
let nodes = isTextNode(element) ? [element] : element.childNodes;
forEach(nodes, node => callback(node));
}
parseSections(element) {
return this.sectionParser.parse(element);
}
// walk up from the textNode until the rootNode, converting each
// parentNode into a markup
collectMarkups(textNode, rootNode) {
let markups = [];
let currentNode = textNode.parentNode;
while (currentNode && currentNode !== rootNode) {
let markup = this.markupFromNode(currentNode);
if (markup) {
markups.push(markup);
}
currentNode = currentNode.parentNode;
}
return markups;
}
// Turn an element node into a markup
markupFromNode(node) {
if (isElementNode(node) && Markup.isValidElement(node)) {
let tagName = remapTagName(node.tagName);
let attributes = getAttributes(node);
return this.builder.createMarkup(tagName, attributes);
}
}
// FIXME should move to the section parser?
// FIXME the `collectMarkups` logic could simplify the section parser?
reparseSection(section, renderTree) {
switch (section.type) {
case LIST_SECTION_TYPE:
return this.reparseListSection(section, renderTree);
case LIST_ITEM_TYPE:
return this.reparseListItem(section, renderTree);
case MARKUP_SECTION_TYPE:
return this.reparseMarkupSection(section, renderTree);
default:
return; // can only parse the above types
}
}
reparseMarkupSection(section, renderTree) {
return this._reparseSectionContainingMarkers(section, renderTree);
}
reparseListItem(listItem, renderTree) {
return this._reparseSectionContainingMarkers(listItem, renderTree);
}
reparseListSection(listSection, renderTree) {
listSection.items.forEach(li => this.reparseListItem(li, renderTree));
}
_reparseSectionContainingMarkers(section, renderTree) {
let element = section.renderNode.element;
let seenRenderNodes = [];
let previousMarker;
walkMarkerableNodes(element, node => {
let marker;
let renderNode = renderTree.getElementRenderNode(node);
if (renderNode) {
if (isMarker(renderNode.postNode)) {
let text = transformHTMLText(node.textContent || '');
let markups = this.collectMarkups(node, element);
if (text.length) {
marker = renderNode.postNode;
marker.value = text;
marker.markups = markups;
}
else {
renderNode.scheduleForRemoval();
}
}
else if (isAtom(renderNode.postNode)) {
let { headTextNode, tailTextNode } = renderNode;
if (headTextNode.textContent !== ZWNJ) {
let value = headTextNode.textContent.replace(new RegExp(ZWNJ, 'g'), '');
headTextNode.textContent = ZWNJ;
if (previousMarker && previousMarker.isMarker) {
previousMarker.value += value;
if (previousMarker.renderNode) {
previousMarker.renderNode.markDirty();
}
}
else {
let postNode = renderNode.postNode;
let newMarkups = postNode.markups.slice();
let newPreviousMarker = this.builder.createMarker(value, newMarkups);
section.markers.insertBefore(newPreviousMarker, postNode);
let newPreviousRenderNode = renderTree.buildRenderNode(newPreviousMarker);
newPreviousRenderNode.markDirty();
section.renderNode.markDirty();
seenRenderNodes.push(newPreviousRenderNode);
section.renderNode.childNodes.insertBefore(newPreviousRenderNode, renderNode);
}
}
if (tailTextNode.textContent !== ZWNJ) {
let value = tailTextNode.textContent.replace(new RegExp(ZWNJ, 'g'), '');
tailTextNode.textContent = ZWNJ;
if (renderNode.postNode.next && renderNode.postNode.next.isMarker) {
let nextMarker = renderNode.postNode.next;
if (nextMarker.renderNode) {
let nextValue = nextMarker.renderNode.element.textContent;
nextMarker.renderNode.element.textContent = value + nextValue;
}
else {
let nextValue = value + nextMarker.value;
nextMarker.value = nextValue;
}
}
else {
let postNode = renderNode.postNode;
let newMarkups = postNode.markups.slice();
let newMarker = this.builder.createMarker(value, newMarkups);
section.markers.insertAfter(newMarker, postNode);
let newRenderNode = renderTree.buildRenderNode(newMarker);
seenRenderNodes.push(newRenderNode);
newRenderNode.markDirty();
section.renderNode.markDirty();
section.renderNode.childNodes.insertAfter(newRenderNode, renderNode);
}
}
if (renderNode) {
marker = renderNode.postNode;
}
}
}
else if (isTextNode(node)) {
let text = transformHTMLText(node.textContent);
let markups = this.collectMarkups(node, element);
marker = this.builder.createMarker(text, markups);
renderNode = renderTree.buildRenderNode(marker);
renderNode.element = node;
renderNode.markClean();
section.renderNode.markDirty();
let previousRenderNode = previousMarker && previousMarker.renderNode;
section.markers.insertAfter(marker, previousMarker);
section.renderNode.childNodes.insertAfter(renderNode, previousRenderNode);
}
if (renderNode) {
seenRenderNodes.push(renderNode);
}
previousMarker = marker;
});
let renderNode = section.renderNode.childNodes.head;
while (renderNode) {
if (seenRenderNodes.indexOf(renderNode) === -1) {
renderNode.scheduleForRemoval();
}
renderNode = renderNode.next;
}
}
}
class HTMLParser {
constructor(builder, options = {}) {
assert('Must pass builder to HTMLParser', builder);
this.builder = builder;
this.options = options;
}
/**
* @param {String} html to parse
* @return {Post} A post abstract
*/
parse(html) {
let dom = parseHTML(html);
let parser = new DOMParser(this.builder, this.options);
return parser.parse(dom);
}
}
class RenderNode extends LinkedItem {
constructor(postNode, renderTree) {
super();
this.parent = null;
this.isDirty = true;
this.isRemoved = false;
// RenderNodes for Markers keep track of their markupElement
this.markupElement = null;
// RenderNodes for Atoms use these properties
this.headTextNode = null;
this.tailTextNode = null;
this.atomNode = null;
// RenderNodes for cards use this property
this.cardNode = null;
this._childNodes = null;
this._element = null;
this._cursorElement = null; // blank render nodes need a cursor element
this.postNode = postNode;
this.renderTree = renderTree;
}
isAttached() {
assert('Cannot check if a renderNode is attached without an element.', !!this.element);
return containsNode(unwrap(unwrap(this.renderTree).rootElement), this.element);
}
get childNodes() {
if (!this._childNodes) {
this._childNodes = new LinkedList({
adoptItem: item => (item.parent = this),
freeItem: item => item.destroy(),
});
}
return this._childNodes;
}
scheduleForRemoval() {
this.isRemoved = true;
if (this.parent) {
this.parent.markDirty();
}
}
markDirty() {
this.isDirty = true;
if (this.parent) {
this.parent.markDirty();
}
}
get isRendered() {
return !!this.element;
}
markClean() {
this.isDirty = false;
}
get element() {
return this._element;
}
set element(element) {
const currentElement = this._element;
this._element = element;
if (currentElement) {
this.renderTree.removeElementRenderNode(currentElement);
}
if (element) {
this.renderTree.setElementRenderNode(element, this);
}
}
set cursorElement(cursorElement) {
this._cursorElement = cursorElement;
}
get cursorElement() {
return this._cursorElement || this.element;
}
destroy() {
this.element = null;
this.parent = null;
this.postNode = null;
this.renderTree = null;
}
reparsesMutationOfChildNode(node) {
if (this.postNode.isCardSection) {
return !containsNode(this.cardNode.element, node);
}
else if (this.postNode.isAtom) {
return !containsNode(this.atomNode.element, node);
}
return true;
}
}
// start at one to make the falsy semantics easier
let uuidGenerator = 1;
class ElementMap {
constructor() {
this._map = {};
}
set(key, value) {
let uuid = key._uuid;
if (!uuid) {
key._uuid = uuid = '' + uuidGenerator++;
}
this._map[uuid] = value;
}
get(key) {
if (key._uuid) {
return this._map[key._uuid];
}
return null;
}
remove(key) {
assertHasUuid(key);
delete this._map[key._uuid];
}
}
function assertHasUuid(key) {
assert('tried to fetch a value for an element not seen before', !!key._uuid);
}
class RenderTree {
constructor(rootPostNode) {
this._rootNode = this.buildRenderNode(rootPostNode);
this._elements = new ElementMap();
}
/*
* @return {RenderNode} The root render node in this tree
*/
get rootNode() {
return this._rootNode;
}
/**
* @return {Boolean}
*/
get isDirty() {
return this.rootNode && this.rootNode.isDirty;
}
/*
* @return {DOMNode} The root DOM element in this tree
*/
get rootElement() {
return this.rootNode.element;
}
/*
* @param {DOMNode} element
* @return {RenderNode} The renderNode for this element, if any
*/
getElementRenderNode(element) {
return this._elements.get(element);
}
setElementRenderNode(element, renderNode) {
this._elements.set(element, renderNode);
}
removeElementRenderNode(element) {
this._elements.remove(element);
}
/**
* @param {DOMNode} element
* Walk up from the dom element until we find a renderNode element
*/
findRenderNodeFromElement(element, conditionFn = () => true) {
let renderNode;
let _element = element;
while (_element) {
renderNode = this.getElementRenderNode(_element);
if (renderNode && conditionFn(renderNode)) {
return renderNode;
}
// continue loop
_element = _element.parentElement;
// stop if we are at the root element
if (_element === this.rootElement) {
if (conditionFn(this.rootNode)) {
return this.rootNode;
}
else {
return;
}
}
}
}
buildRenderNode(postNode) {
const renderNode = new RenderNode(postNode, this);
postNode.renderNode = renderNode;
return renderNode;
}
}
const MOBILEDOC_VERSION$4 = MOBILEDOC_VERSION$3;
var mobiledocRenderers = {
render(post, version = MOBILEDOC_VERSION$3) {
switch (version) {
case MOBILEDOC_VERSION:
return MobiledocRenderer_0_2.render(post);
case MOBILEDOC_VERSION$1:
return MobiledocRenderer_0_3.render(post);
case MOBILEDOC_VERSION$2:
return MobiledocRenderer_0_3_1.render(post);
case undefined:
case null:
case MOBILEDOC_VERSION$3:
return MobiledocRenderer_0_3_2.render(post);
default:
assert(`Unknown version of mobiledoc renderer requested: ${version}`, false);
}
},
};
function mergeWithOptions(original, updates, options) {
return Object.assign(original, updates, options);
}
var Environment = {
hasDOM() {
return typeof document !== 'undefined';
},
};
function expectCloneable(section) {
if (!('clone' in section)) {
throw new Error('Expected section to be cloneable');
}
return section;
}
/**
* The Post is an in-memory representation of an editor's document.
* An editor always has a single post. The post is organized into a list of
* sections. Each section may be markerable (contains "markers", aka editable
* text) or non-markerable (e.g., a card).
* When persisting a post, it must first be serialized (loss-lessly) into
* mobiledoc using {@link Editor#serialize}.
*/
class Post {
constructor() {
this.type = "post" /* POST */;
this.sections = new LinkedList({
adoptItem: s => (s.post = s._parent = this),
freeItem: s => (s.post = s._parent = null),
});
}
/**
* @return {Position} The position at the start of the post (will be a {@link BlankPosition}
* if the post is blank)
* @public
*/
headPosition() {
if (this.isBlank) {
return Position.blankPosition();
}
else {
return this.sections.head.headPosition();
}
}
/**
* @return {Position} The position at the end of the post (will be a {@link BlankPosition}
* if the post is blank)
* @public
*/
tailPosition() {
if (this.isBlank) {
return Position.blankPosition();
}
else {
return this.sections.tail.tailPosition();
}
}
/**
* @return {Range} A range encompassing the entire post
* @public
*/
toRange() {
return this.headPosition().toRange(this.tailPosition());
}
get isBlank() {
return this.sections.isEmpty;
}
/**
* If the post has no sections, or only has one, blank section, then it does
* not have content and this method returns false. Otherwise it is true.
* @return {Boolean}
* @public
*/
get hasContent() {
if (this.sections.length > 1 || (this.sections.length === 1 && !this.sections.head.isBlank)) {
return true;
}
else {
return false;
}
}
/**
* @param {Range} range
* @return {Array} markers that are completely contained by the range
*/
markersContainedByRange(range) {
const markers = [];
this.walkMarkerableSections(range, (section) => {
section._markersInRange(range.trimTo(section), (m, { isContained }) => {
if (isContained) {
markers.push(m);
}
});
});
return markers;
}
markupsInRange(range) {
const markups = new Set();
if (range.isCollapsed) {
let pos = range.head;
if (pos.isMarkerable) {
let [back, forward] = [pos.markerIn(-1), pos.markerIn(1)];
if (back && forward && back === forward) {
back.markups.forEach(m => markups.add(m));
}
else {
((back && back.markups) || []).forEach(m => {
if (m.isForwardInclusive()) {
markups.add(m);
}
});
((forward && forward.markups) || []).forEach(m => {
if (m.isBackwardInclusive()) {
markups.add(m);
}
});
}
}
}
else {
this.walkMarkerableSections(range, section => {
forEach(section.markupsInRange(range.trimTo(section)), m => markups.add(m));
});
}
return markups.toArray();
}
walkAllLeafSections(callback) {
let range = this.headPosition().toRange(this.tailPosition());
return this.walkLeafSections(range, callback);
}
walkLeafSections(range, callback) {
const { head, tail } = range;
let index = 0;
let nextSection;
let shouldStop;
let currentSection = head.section;
while (currentSection) {
nextSection = this._nextLeafSection(currentSection);
shouldStop = currentSection === tail.section;
callback(currentSection, index);
index++;
if (shouldStop) {
break;
}
else {
currentSection = nextSection;
}
}
}
walkMarkerableSections(range, callback) {
this.walkLeafSections(range, section => {
if (isMarkerable$1(section)) {
callback(section);
}
});
}
// return the next section that has markers after this one,
// possibly skipping non-markerable sections
_nextLeafSection(section) {
if (!section) {
return null;
}
const next = section.next;
if (next) {
if (next.isLeafSection) {
return next;
}
else if (isListSection$1(next)) {
return next.items.head;
}
else {
assert('Cannot determine next section from non-leaf-section', false);
}
}
else if (isNested(section)) {
// if there is no section after this, but this section is a child
// (e.g. a ListItem inside a ListSection), check for a markerable
// section after its parent
return this._nextLeafSection(section.parent);
}
else {
return null;
}
}
/**
* @param {Range} range
* @return {Post} A new post, constrained to {range}
*/
trimTo(range) {
const { builder } = this;
const post = builder.createPost();
const { head, tail } = range;
const tailNotSelected = tail.offset === 0 && head.section !== tail.section;
let sectionParent = post, listParent = null;
this.walkLeafSections(range, section => {
let newSection;
if (isMarkerable$1(section)) {
if (isListItem(section)) {
if (listParent) {
sectionParent = null;
}
else {
listParent = builder.createListSection(section.parent.tagName);
post.sections.append(listParent);
sectionParent = null;
}
newSection = builder.createListItem();
listParent.items.append(newSection);
}
else {
listParent = null;
sectionParent = post;
const tagName = tailNotSelected && tail.section === section ? 'p' : section.tagName;
newSection = builder.createMarkupSection(tagName);
}
let currentRange = range.trimTo(section);
forEach(section.markersFor(currentRange.headSectionOffset, currentRange.tailSectionOffset), m => newSection.markers.append(m));
}
else {
newSection =
tailNotSelected && tail.section === section
? builder.createMarkupSection('p')
: expectCloneable(section).clone();
sectionParent = post;
}
if (sectionParent) {
sectionParent.sections.append(newSection);
}
});
return post;
}
}
class Image extends Section {
constructor() {
super("image-section" /* IMAGE_SECTION */);
this.src = null;
}
clone() {
return this.builder.createImageSection(this.src);
}
canJoin() {
return false;
}
get length() {
return 1;
}
}
function cacheKey(tagName, attributes) {
return `${normalizeTagName(tagName)}-${objectToSortedKVArray(attributes).join('-')}`;
}
function addMarkupToCache(cache, markup) {
cache[cacheKey(markup.tagName, markup.attributes)] = markup;
}
function findMarkupInCache(cache, tagName, attributes) {
const key = cacheKey(tagName, attributes);
return cache[key];
}
/**
* The PostNodeBuilder is used to create new {@link Post} primitives, such
* as a MarkupSection, a CardSection, a Markup, etc. Every instance of an
* {@link Editor} has its own builder instance. The builder can be used
* inside an {@link Editor#run} callback to programmatically create new
* Post primitives to insert into the document.
* A PostNodeBuilder should be read from the Editor, *not* instantiated on its own.
*/
class PostNodeBuilder {
constructor() {
this.markupCache = {};
}
/**
* @return {Post} A new, blank post
*/
createPost(sections = []) {
const post = new Post();
post.builder = this;
sections.forEach(s => post.sections.append(s));
return post;
}
createMarkerableSection(type, tagName, markers = []) {
switch (type) {
case LIST_ITEM_TYPE:
return this.createListItem(markers);
case MARKUP_SECTION_TYPE:
return this.createMarkupSection(tagName, markers);
default:
assert(`Cannot create markerable section of type ${type}`, false);
}
}
/**
* @param {tagName} [tagName='P']
* @param {Marker[]} [markers=[]]
* @return {MarkupSection}
*/
createMarkupSection(tagName = DEFAULT_TAG_NAME$1, markers = [], isGenerated = false, attributes = {}) {
tagName = normalizeTagName(tagName);
const section = new MarkupSection(tagName, markers, attributes);
if (isGenerated) {
section.isGenerated = true;
}
section.builder = this;
return section;
}
createListSection(tagName = DEFAULT_TAG_NAME, items = [], attributes = {}) {
tagName = normalizeTagName(tagName);
const section = new ListSection(tagName, items, attributes);
section.builder = this;
return section;
}
createListItem(markers = []) {
const tagName = normalizeTagName('li');
const item = new ListItem(tagName, markers);
item.builder = this;
return item;
}
createImageSection(url) {
let section = new Image();
if (url) {
section.src = url;
}
section.builder = this;
return section;
}
/**
* @param {String} name
* @param {Object} [payload={}]
* @return {CardSection}
*/
createCardSection(name, payload = {}) {
const card = new Card(name, payload);
card.builder = this;
return card;
}
/**
* @param {String} value
* @param {Markup[]} [markups=[]]
* @return {Marker}
*/
createMarker(value, markups = []) {
const marker = new Marker(value, markups);
marker.builder = this;
return marker;
}
/**
* @param {String} name
* @param {String} [value='']
* @param {Object} [payload={}]
* @param {Markup[]} [markups=[]]
* @return {Atom}
*/
createAtom(name, value = '', payload = {}, markups = []) {
const atom = new Atom(name, value, payload, markups);
atom.builder = this;
return atom;
}
/**
* @param {String} tagName
* @param {Object} attributes Key-value pairs of attributes for the markup
* @return {Markup}
*/
createMarkup(tagName, attributes = {}) {
tagName = normalizeTagName(tagName);
let markup = findMarkupInCache(this.markupCache, tagName, attributes);
if (!markup) {
markup = new Markup(tagName, attributes);
markup.builder = this;
addMarkupToCache(this.markupCache, markup);
}
return markup;
}
}
/**
* Convert section at the editor's cursor position into a list.
* Does nothing if the cursor position is not at the start of the section,
* or if the section is already a list item.
*
* @param {Editor} editor
* @param {String} listTagName ("ul" or "ol")
* @public
*/
function replaceWithListSection(editor, listTagName) {
const { range } = editor;
const { head } = range;
const { section } = head;
// Skip if cursor is not at end of section
if (!head.isTail()) {
return;
}
if (section.isListItem) {
return;
}
editor.run(postEditor => {
let { builder } = postEditor;
let item = builder.createListItem();
let listSection = builder.createListSection(listTagName, [item]);
postEditor.replaceSection(section, listSection);
postEditor.setRange(listSection.headPosition());
});
}
/**
* Convert section at the editor's cursor position into a header section.
* Does nothing if the cursor position is not at the start of the section.
*
* @param {Editor} editor
* @param {String} headingTagName ('h1', 'h2', 'h3', 'h4', 'h5', 'h6')
* @public
*/
function replaceWithHeaderSection(editor, headingTagName) {
let { range: { head, head: { section }, }, } = editor;
// Skip if cursor is not at end of section
if (!head.isTail()) {
return;
}
editor.run(postEditor => {
let { builder } = postEditor;
let newSection = builder.createMarkupSection(headingTagName);
postEditor.replaceSection(section, newSection);
postEditor.setRange(newSection.headPosition());
});
}
const DEFAULT_TEXT_INPUT_HANDLERS = [
{
name: 'ul',
// "* " -> ul
match: /^\* $/,
run(editor) {
replaceWithListSection(editor, 'ul');
},
},
{
name: 'ol',
// "1" -> ol, "1." -> ol
match: /^1\.? $/,
run(editor) {
replaceWithListSection(editor, 'ol');
},
},
{
name: 'heading',
/*
* "# " -> h1
* "## " -> h2
* "### " -> h3
* "#### " -> h4
* "##### " -> h5
* "###### " -> h6
*/
match: /^(#{1,6}) $/,
run(editor, matches) {
let capture = matches[1];
let headingTag = 'h' + capture.length;
replaceWithHeaderSection(editor, headingTag);
},
},
];
function selectAll(editor) {
let { post } = editor;
editor.selectRange(post.toRange());
}
function gotoStartOfLine(editor) {
let { range } = editor;
let { tail: { section }, } = range;
editor.run(postEditor => {
postEditor.setRange(section.headPosition());
});
}
function gotoEndOfLine(editor) {
let { range } = editor;
let { tail: { section }, } = range;
editor.run(postEditor => {
postEditor.setRange(section.tailPosition());
});
}
function deleteToEndOfSection(editor) {
let { range } = editor;
if (range.isCollapsed) {
let { head, head: { section }, } = range;
range = head.toRange(section.tailPosition());
}
editor.run(postEditor => {
let nextPosition = postEditor.deleteRange(range);
postEditor.setRange(nextPosition);
});
}
const DEFAULT_KEY_COMMANDS = [
{
str: 'META+B',
run(editor) {
editor.toggleMarkup('strong');
},
},
{
str: 'CTRL+B',
run(editor) {
editor.toggleMarkup('strong');
},
},
{
str: 'META+I',
run(editor) {
editor.toggleMarkup('em');
},
},
{
str: 'CTRL+I',
run(editor) {
editor.toggleMarkup('em');
},
},
{
str: 'META+U',
run(editor) {
editor.toggleMarkup('u');
},
},
{
str: 'CTRL+U',
run(editor) {
editor.toggleMarkup('u');
},
},
{
str: 'CTRL+K',
run(editor) {
if (Browser.isMac()) {
return deleteToEndOfSection(editor);
}
else if (Browser.isWin()) {
return toggleLink(editor);
}
},
},
{
str: 'CTRL+A',
run(editor) {
if (Browser.isMac()) {
gotoStartOfLine(editor);
}
else {
selectAll(editor);
}
},
},
{
str: 'META+A',
run(editor) {
if (Browser.isMac()) {
selectAll(editor);
}
},
},
{
str: 'CTRL+E',
run(editor) {
if (Browser.isMac()) {
gotoEndOfLine(editor);
}
},
},
{
str: 'META+K',
run(editor) {
return toggleLink(editor);
},
},
{
str: 'META+Z',
run(editor) {
editor.run(postEditor => {
postEditor.undoLastChange();
});
},
},
{
str: 'META+SHIFT+Z',
run(editor) {
editor.run(postEditor => {
postEditor.redoLastChange();
});
},
},
{
str: 'CTRL+Z',
run(editor) {
if (Browser.isMac()) {
return false;
}
editor.run(postEditor => postEditor.undoLastChange());
},
},
{
str: 'CTRL+SHIFT+Z',
run(editor) {
if (Browser.isMac()) {
return false;
}
editor.run(postEditor => postEditor.redoLastChange());
},
},
];
function modifierNamesToMask(modiferNames) {
let defaultVal = 0;
return reduce(modiferNames, (sum, name) => {
let modifier = MODIFIERS[name.toUpperCase()];
assert(`No modifier named "${name}" found`, !!modifier);
return sum + modifier;
}, defaultVal);
}
function characterToCode(character) {
const upperCharacter = character.toUpperCase();
const special = specialCharacterToCode(upperCharacter);
if (special) {
return special;
}
else {
assert(`Only 1 character can be used in a key command str (got "${character}")`, character.length === 1);
return upperCharacter.charCodeAt(0);
}
}
function buildKeyCommand(keyCommand) {
if (isCompiledKeyCommand(keyCommand)) {
return keyCommand;
}
assert('[deprecation] Key commands no longer use the `modifier` property', !keyCommand.modifier);
let { str, run, name } = keyCommand;
let [character, ...modifierNames] = str.split('+').reverse();
return {
name,
run,
modifierMask: modifierNamesToMask(modifierNames),
code: characterToCode(character),
};
}
function isCompiledKeyCommand(keyCommand) {
return keyCommand.str === undefined;
}
function validateKeyCommand(keyCommand) {
return !!keyCommand.code && !!keyCommand.run;
}
function findKeyCommands(keyCommands, keyEvent) {
const key = Key.fromEvent(keyEvent);
return filter(keyCommands, ({ modifierMask, code }) => {
return key.keyCode === code && key.modifierMask === modifierMask;
});
}
class MutationHandler {
constructor(editor) {
this.editor = editor;
this.logger = editor.loggerFor('mutation-handler');
this.renderTree = null;
this._isObserving = false;
this._observer = new MutationObserver(mutations => {
this._handleMutations(mutations);
});
}
init() {
this.startObserving();
}
destroy() {
this.stopObserving();
this._observer = null;
}
suspendObservation(callback) {
this.stopObserving();
callback();
this.startObserving();
}
stopObserving() {
if (this._isObserving) {
this._isObserving = false;
this._observer.disconnect();
}
}
startObserving() {
if (!this._isObserving) {
let { editor } = this;
assert('Cannot observe un-rendered editor', editor.hasRendered);
this._isObserving = true;
this.renderTree = editor._renderTree;
this._observer.observe(editor.element, {
characterData: true,
childList: true,
subtree: true,
});
}
}
reparsePost() {
this.editor._reparsePost();
}
reparseSections(sections) {
this.editor._reparseSections(sections);
}
/**
* for each mutation:
* * find the target nodes:
* * if nodes changed, target nodes are:
* * added nodes
* * the target from which removed nodes were removed
* * if character data changed
* * target node is the mutation event's target (text node)
* * filter out nodes that are no longer attached (parentNode is null)
* * for each remaining node:
* * find its section, add to sections-to-reparse
* * if no section, reparse all (and break)
*/
_handleMutations(mutations) {
let reparsePost = false;
let sections = new Set();
for (let i = 0; i < mutations.length; i++) {
if (reparsePost) {
break;
}
let nodes = this._findTargetNodes(mutations[i]);
for (let j = 0; j < nodes.length; j++) {
let node = nodes[j];
let renderNode = this._findRenderNodeFromNode(node);
if (renderNode) {
if (renderNode.reparsesMutationOfChildNode(node)) {
let section = this._findSectionFromRenderNode(renderNode);
if (section) {
sections.add(section);
}
else {
reparsePost = true;
}
}
}
else {
reparsePost = true;
break;
}
}
}
if (reparsePost) {
this.logger.log(`reparsePost (${mutations.length} mutations)`);
this.reparsePost();
}
else if (sections.length) {
this.logger.log(`reparse ${sections.length} sections (${mutations.length} mutations)`);
this.reparseSections(sections.toArray());
}
}
_findTargetNodes(mutation) {
let nodes = [];
switch (mutation.type) {
case "characterData" /* CHARACTER_DATA */:
nodes.push(mutation.target);
break;
case "childList" /* NODES_CHANGED */:
forEach(mutation.addedNodes, n => nodes.push(n));
if (mutation.removedNodes.length) {
nodes.push(mutation.target);
}
break;
}
let element = this.editor.element;
let attachedNodes = filter(nodes, node => containsNode(element, node));
return attachedNodes;
}
_findSectionRenderNodeFromNode(node) {
return this.renderTree.findRenderNodeFromElement(node, rn => {
return rn.postNode.isSection;
});
}
_findRenderNodeFromNode(node) {
return this.renderTree.findRenderNodeFromElement(node);
}
_findSectionFromRenderNode(renderNode) {
let sectionRenderNode = this._findSectionRenderNodeFromNode(renderNode.element);
return sectionRenderNode && sectionRenderNode.postNode;
}
}
class FixedQueue {
constructor(length = 0) {
this._maxLength = length;
this._items = [];
}
get length() {
return this._items.length;
}
pop() {
return this._items.pop();
}
push(item) {
this._items.push(item);
if (this.length > this._maxLength) {
this._items.shift();
}
}
clear() {
this._items = [];
}
toArray() {
return this._items;
}
}
function findLeafSectionAtIndex(post, index) {
let section;
post.walkAllLeafSections((_section, _index) => {
if (index === _index) {
section = _section;
}
});
return section;
}
class Snapshot {
constructor(takenAt, editor, editAction = null) {
this.mobiledoc = editor.serialize();
this.editor = editor;
this.editAction = editAction;
this.takenAt = takenAt;
this.snapshotRange();
}
snapshotRange() {
let { range, cursor } = this.editor;
if (cursor.hasCursor() && !range.isBlank) {
let { head, tail } = range;
this.range = {
head: [head.leafSectionIndex, head.offset],
tail: [tail.leafSectionIndex, tail.offset],
};
}
}
getRange(post) {
if (this.range) {
let { head, tail } = this.range;
let [headLeafSectionIndex, headOffset] = head;
let [tailLeafSectionIndex, tailOffset] = tail;
let headSection = findLeafSectionAtIndex(post, headLeafSectionIndex);
let tailSection = findLeafSectionAtIndex(post, tailLeafSectionIndex);
let headPosition = headSection.toPosition(headOffset);
let tailPosition = tailSection.toPosition(tailOffset);
return headPosition.toRange(tailPosition);
}
}
groupsWith(groupingTimeout, editAction, takenAt) {
return editAction !== null && this.editAction === editAction && this.takenAt + groupingTimeout > takenAt;
}
}
class EditHistory {
constructor(editor, queueLength, groupingTimeout) {
this.editor = editor;
this._undoStack = new FixedQueue(queueLength);
this._redoStack = new FixedQueue(queueLength);
this._pendingSnapshot = null;
this._groupingTimeout = groupingTimeout;
}
snapshot() {
// update the current snapshot with the range read from DOM
if (this._pendingSnapshot) {
this._pendingSnapshot.snapshotRange();
}
}
storeSnapshot(editAction = null) {
let now = Date.now();
// store pending snapshot
let pendingSnapshot = this._pendingSnapshot;
if (pendingSnapshot) {
if (!pendingSnapshot.groupsWith(this._groupingTimeout, editAction, now)) {
this._undoStack.push(pendingSnapshot);
}
this._redoStack.clear();
}
// take new pending snapshot to store next time `storeSnapshot` is called
this._pendingSnapshot = new Snapshot(now, this.editor, editAction);
}
stepBackward(postEditor) {
// Throw away the pending snapshot
this._pendingSnapshot = null;
let snapshot = this._undoStack.pop();
if (snapshot) {
this._redoStack.push(new Snapshot(Date.now(), this.editor));
this._restoreFromSnapshot(snapshot, postEditor);
}
}
stepForward(postEditor) {
let snapshot = this._redoStack.pop();
if (snapshot) {
this._undoStack.push(new Snapshot(Date.now(), this.editor));
this._restoreFromSnapshot(snapshot, postEditor);
}
postEditor.cancelSnapshot();
}
_restoreFromSnapshot(snapshot, postEditor) {
let { mobiledoc } = snapshot;
let { editor } = this;
let { builder, post } = editor;
let restoredPost = mobiledocParsers.parse(builder, mobiledoc);
postEditor.removeAllSections();
postEditor.migrateSectionsFromPost(restoredPost);
// resurrect snapshotted range if it exists
let newRange = snapshot.getRange(post);
if (newRange) {
postEditor.setRange(newRange);
}
}
}
const UL_LI_REGEX = /^\* (.*)$/;
const OL_LI_REGEX = /^\d\.? (.*)$/;
const CR = '\r';
const LF = '\n';
const CR_REGEX = new RegExp(CR, 'g');
const CR_LF_REGEX = new RegExp(CR + LF, 'g');
const SECTION_BREAK = LF;
function normalizeLineEndings(text) {
return text.replace(CR_LF_REGEX, LF).replace(CR_REGEX, LF);
}
class TextParser {
constructor(builder, options) {
this.builder = builder;
this.options = options;
this.post = this.builder.createPost();
this.prevSection = null;
}
/**
* @param {String} text to parse
* @return {Post} a post abstract
*/
parse(text) {
text = normalizeLineEndings(text);
text.split(SECTION_BREAK).forEach(text => {
let section = this._parseSection(text);
this._appendSection(section);
});
return this.post;
}
_parseSection(text) {
let tagName = DEFAULT_TAG_NAME$1, type = MARKUP_SECTION_TYPE, section;
if (UL_LI_REGEX.test(text)) {
tagName = 'ul';
type = LIST_SECTION_TYPE;
text = text.match(UL_LI_REGEX)[1];
}
else if (OL_LI_REGEX.test(text)) {
tagName = 'ol';
type = LIST_SECTION_TYPE;
text = text.match(OL_LI_REGEX)[1];
}
let markers = [this.builder.createMarker(text)];
switch (type) {
case LIST_SECTION_TYPE: {
let item = this.builder.createListItem(markers);
let list = this.builder.createListSection(tagName, [item]);
section = list;
break;
}
case MARKUP_SECTION_TYPE:
section = this.builder.createMarkupSection(tagName, markers);
break;
default:
assert(`Unknown type encountered ${type}`, false);
}
return section;
}
_appendSection(section) {
let isSameListSection = isListSection$1(section) &&
this.prevSection &&
isListSection$1(this.prevSection) &&
this.prevSection.tagName === section.tagName;
if (isSameListSection) {
section.items.forEach(item => {
this.prevSection.items.append(item.clone());
});
}
else {
this.post.sections.insertAfter(section, this.prevSection);
this.prevSection = section;
}
}
}
const MIME_TEXT_PLAIN = 'text/plain';
const MIME_TEXT_HTML = 'text/html';
const NONSTANDARD_IE_TEXT_TYPE = 'Text';
const MOBILEDOC_REGEX = new RegExp(/data-mobiledoc='(.*?)'>/);
/**
* @return {Post}
* @private
*/
function parsePostFromHTML(html, builder, plugins) {
let post;
if (MOBILEDOC_REGEX.test(html)) {
let mobiledocString = html.match(MOBILEDOC_REGEX)[1];
let mobiledoc = JSON.parse(mobiledocString);
post = mobiledocParsers.parse(builder, mobiledoc);
}
else {
post = new HTMLParser(builder, { plugins }).parse(html);
}
return post;
}
/**
* @return {Post}
* @private
*/
function parsePostFromText(text, builder, plugins) {
let parser = new TextParser(builder, { plugins });
let post = parser.parse(text);
return post;
}
/**
* @return {{html: String, text: String}}
* @private
*/
function getContentFromPasteEvent(event, window) {
let html = '', text = '';
let { clipboardData } = event;
if (clipboardData && clipboardData.getData) {
html = clipboardData.getData(MIME_TEXT_HTML);
text = clipboardData.getData(MIME_TEXT_PLAIN);
}
else if (window.clipboardData && window.clipboardData.getData) {
// IE
// The Internet Explorers (including Edge) have a non-standard way of interacting with the
// Clipboard API (see http://caniuse.com/#feat=clipboard). In short, they expose a global window.clipboardData
// object instead of the per-event event.clipboardData object on the other browsers.
html = window.clipboardData.getData(NONSTANDARD_IE_TEXT_TYPE);
}
return { html, text };
}
/**
* @return {{html: String, text: String}}
* @private
*/
function getContentFromDropEvent(event, logger) {
let html = '', text = '';
try {
html = event.dataTransfer.getData(MIME_TEXT_HTML);
text = event.dataTransfer.getData(MIME_TEXT_PLAIN);
}
catch (e) {
// FIXME IE11 does not include any data in the 'text/html' or 'text/plain'
// mimetypes. It throws an error 'Invalid argument' when attempting to read
// these properties.
if (logger) {
logger.log('Error getting drop data: ', e);
}
}
return { html, text };
}
/**
* @param {CopyEvent|CutEvent}
* @param {Editor}
* @param {Window}
* @private
*/
function setClipboardData(event, { mobiledoc, html, text }, window) {
if (mobiledoc && html) {
html = `${html}
`;
}
let { clipboardData } = event;
let { clipboardData: nonstandardClipboardData } = window;
if (clipboardData && clipboardData.setData) {
clipboardData.setData(MIME_TEXT_HTML, html);
clipboardData.setData(MIME_TEXT_PLAIN, text);
}
else if (nonstandardClipboardData && nonstandardClipboardData.setData) {
// The Internet Explorers (including Edge) have a non-standard way of interacting with the
// Clipboard API (see http://caniuse.com/#feat=clipboard). In short, they expose a global window.clipboardData
// object instead of the per-event event.clipboardData object on the other browsers.
nonstandardClipboardData.setData(NONSTANDARD_IE_TEXT_TYPE, html);
}
}
/**
* @param {PasteEvent}
* @param {{builder: Builder, _parserPlugins: Array}} options
* @return {Post}
* @private
*/
function parsePostFromPaste(pasteEvent, { builder, _parserPlugins: plugins }, { targetFormat } = { targetFormat: 'html' }) {
let { html, text } = getContentFromPasteEvent(pasteEvent, window);
if (targetFormat === 'html' && html && html.length) {
return parsePostFromHTML(html, builder, plugins);
}
else if (text && text.length) {
return parsePostFromText(text, builder, plugins);
}
}
/**
* @param {DropEvent}
* @param {Editor} editor
* @param {Object} [options={}] Can pass a logger
* @return {Post}
* @private
*/
function parsePostFromDrop(dropEvent, editor, { logger } = {}) {
let { builder, _parserPlugins: plugins } = editor;
let { html, text } = getContentFromDropEvent(dropEvent, logger);
if (html && html.length) {
return parsePostFromHTML(html, builder, plugins);
}
else if (text && text.length) {
return parsePostFromText(text, builder, plugins);
}
}
class TextInputHandler {
constructor(editor) {
this.editor = editor;
this._handlers = [];
}
register(handler) {
assert(`Input Handler is not valid`, this._validateHandler(handler));
this._handlers.push(handler);
}
unregister(name) {
let handlers = this._handlers;
for (let i = 0; i < handlers.length; i++) {
if (handlers[i].name === name) {
handlers.splice(i, 1);
}
}
}
handle(string) {
let { editor } = this;
editor.insertText(string);
let matchedHandler = this._findHandler();
if (matchedHandler) {
let [handler, matches] = matchedHandler;
handler.run(editor, matches);
}
}
handleNewLine() {
let { editor } = this;
let matchedHandler = this._findHandler(ENTER);
if (matchedHandler) {
let [handler, matches] = matchedHandler;
handler.run(editor, matches);
}
}
_findHandler(string = '') {
const { editor } = this;
const { range } = editor;
const { head } = range;
const { section } = head;
let preText = section.textUntil(head) + string;
for (let i = 0; i < this._handlers.length; i++) {
let handler = this._handlers[i];
if ('text' in handler && endsWith(preText, handler.text)) {
return [handler, [handler.text]];
}
else if ('match' in handler && handler.match.test(preText)) {
return [handler, handler.match.exec(preText)];
}
}
}
_validateHandler(handler) {
deprecate('Registered input handlers require a "name" property so that they can be unregistered', !!handler.name);
return (!!handler.run && // has `run`
(!!handler.text || !!handler.match) && // and `text` or `match`
!(!!handler.text && !!handler.match)); // not both `text` and `match`
}
destroy() {
this._handlers = [];
}
}
let instance;
class SelectionChangeObserver {
constructor({ editor }) {
this.started = false;
this.listeners = [];
this.selection = {};
this.editor = editor;
}
static getInstance({ editor } = {}) {
if (!instance) {
instance = new SelectionChangeObserver({ editor });
}
return instance;
}
static addListener(listener) {
SelectionChangeObserver.getInstance({ editor: listener.editor }).addListener(listener);
}
addListener(listener) {
if (this.listeners.indexOf(listener) === -1) {
this.listeners.push(listener);
this.start();
}
}
static removeListener(listener) {
SelectionChangeObserver.getInstance().removeListener(listener);
}
removeListener(listener) {
let index = this.listeners.indexOf(listener);
if (index !== -1) {
this.listeners.splice(index, 1);
if (this.listeners.length === 0) {
this.stop();
}
}
}
start() {
if (this.started) {
return;
}
this.started = true;
this.poll();
}
stop() {
this.started = false;
this.selection = {};
}
notifyListeners(newSelection, prevSelection) {
this.listeners.forEach(listener => {
listener.selectionDidChange(newSelection, prevSelection);
});
}
destroy() {
this.stop();
this.listeners = [];
}
getSelection(root = window) {
let selection = root.getSelection();
let { anchorNode, focusNode, anchorOffset, focusOffset } = selection;
return { anchorNode, focusNode, anchorOffset, focusOffset };
}
poll() {
if (this.started) {
this.update();
this.runNext(() => this.poll());
}
}
runNext(fn) {
window.requestAnimationFrame(fn);
}
update() {
let prevSelection = this.selection;
let curSelection = this.getSelection(this.editor.root);
if (!this.selectionIsEqual(prevSelection, curSelection)) {
this.selection = curSelection;
this.notifyListeners(curSelection, prevSelection);
}
}
selectionIsEqual(s1, s2) {
return (s1.anchorNode === s2.anchorNode &&
s1.anchorOffset === s2.anchorOffset &&
s1.focusNode === s2.focusNode &&
s1.focusOffset === s2.focusOffset);
}
}
class SelectionManager {
constructor(editor, callback) {
this.editor = editor;
this.callback = callback;
this.started = false;
}
start() {
if (this.started) {
return;
}
SelectionChangeObserver.addListener(this);
this.started = true;
}
stop() {
this.started = false;
SelectionChangeObserver.removeListener(this);
}
destroy() {
this.stop();
}
selectionDidChange(curSelection, prevSelection) {
if (this.started) {
this.callback(curSelection, prevSelection);
}
}
}
const ELEMENT_EVENT_TYPES = [
'keydown',
'keyup',
'cut',
'copy',
'paste',
'keypress',
'drop',
'compositionstart',
'compositionend',
];
class EventManager {
constructor(editor) {
this.editor = editor;
this.logger = editor.loggerFor('event-manager');
this._textInputHandler = new TextInputHandler(editor);
this._listeners = [];
this.modifierKeys = {
shift: false,
};
this._selectionManager = new SelectionManager(this.editor, this.selectionDidChange.bind(this));
this.started = true;
this._isComposingOnBlankLine = false;
}
init() {
let { editor: { element }, } = this;
assert(`Cannot init EventManager without element`, !!element);
ELEMENT_EVENT_TYPES.forEach(type => {
this._addListener(element, type);
});
this._selectionManager.start();
}
start() {
this.started = true;
}
stop() {
this.started = false;
}
registerInputHandler(inputHandler) {
this._textInputHandler.register(inputHandler);
}
unregisterInputHandler(name) {
this._textInputHandler.unregister(name);
}
unregisterAllTextInputHandlers() {
this._textInputHandler.destroy();
this._textInputHandler = new TextInputHandler(this.editor);
}
_addListener(context, type) {
assert(`Missing listener for ${type}`, !!this[type]);
let listener = event => this._handleEvent(type, event);
context.addEventListener(type, listener);
this._listeners.push([context, type, listener]);
}
_removeListeners() {
this._listeners.forEach(([context, type, listener]) => {
context.removeEventListener(type, listener);
});
this._listeners = [];
}
// This is primarily useful for programmatically simulating events on the
// editor from the tests.
_trigger(context, type, event) {
forEach(filter(this._listeners, ([_context, _type]) => {
return _context === context && _type === type;
}), ([context, , listener]) => {
listener.call(context, event);
});
}
destroy() {
this._textInputHandler.destroy();
this._selectionManager.destroy();
this._removeListeners();
}
_handleEvent(type, event) {
let { target: element } = event;
if (!this.started) {
// abort handling this event
return true;
}
if (!this.isElementAddressable(element)) {
// abort handling this event
return true;
}
this[type](event);
}
isElementAddressable(element) {
return this.editor.cursor.isAddressable(element);
}
selectionDidChange(selection /*, prevSelection */) {
let shouldNotify = true;
let { anchorNode } = selection;
if (!this.isElementAddressable(anchorNode)) {
if (!this.editor.range.isBlank) {
// Selection changed from something addressable to something
// not-addressable -- e.g., blur event, user clicked outside editor,
// etc
shouldNotify = true;
}
else {
// selection changes wholly outside the editor should not trigger
// change notifications
shouldNotify = false;
}
}
if (shouldNotify) {
this.editor._readRangeFromDOM();
}
}
keypress(event) {
let { editor, _textInputHandler } = this;
if (!editor.hasCursor()) {
return;
}
let key = Key.fromEvent(event);
if (!key.isPrintable()) {
return;
}
else {
event.preventDefault();
}
// Handle carriage returns
if (!key.isEnter() && key.keyCode === 13) {
_textInputHandler.handleNewLine();
editor.handleNewline(event);
return;
}
_textInputHandler.handle(key.toString());
}
keydown(event) {
let { editor } = this;
if (!editor.hasCursor()) {
return;
}
if (!editor.isEditable) {
return;
}
let key = Key.fromEvent(event);
this._updateModifiersFromKey(key, { isDown: true });
if (editor.handleKeyCommand(event)) {
return;
}
if (editor.post.isBlank) {
editor._insertEmptyMarkupSectionAtCursor();
}
let range = editor.range;
switch (true) {
// Ignore keydown events when using an IME
case key.isIME(): {
break;
}
// FIXME This should be restricted to only card/atom boundaries
case key.isHorizontalArrowWithoutModifiersOtherThanShift(): {
let newRange;
if (key.isShift()) {
newRange = range.extend(key.direction * 1);
}
else {
newRange = range.move(key.direction);
}
editor.selectRange(newRange);
event.preventDefault();
break;
}
case key.isDelete(): {
let { direction } = key;
let unit = TextUnit.CHAR;
if (key.altKey && Browser.isMac()) {
unit = TextUnit.WORD;
}
else if (key.ctrlKey && !Browser.isMac()) {
unit = TextUnit.WORD;
}
editor.performDelete({ direction, unit });
event.preventDefault();
break;
}
case key.isEnter():
this._textInputHandler.handleNewLine();
editor.handleNewline(event);
break;
case key.isTab():
// Handle tab here because it does not fire a `keypress` event
event.preventDefault();
this._textInputHandler.handle(key.toString());
break;
}
}
keyup(event) {
let { editor } = this;
if (!editor.hasCursor()) {
return;
}
let key = Key.fromEvent(event);
this._updateModifiersFromKey(key, { isDown: false });
}
// The mutation handler interferes with IMEs when composing
// on a blank line. These two event handlers are for suppressing
// mutation handling in this scenario.
compositionstart(_event) {
let { editor } = this;
// Ignore compositionstart if not on a blank line
if (editor.range.headMarker) {
return;
}
this._isComposingOnBlankLine = true;
if (editor.post.isBlank) {
editor._insertEmptyMarkupSectionAtCursor();
}
// Stop listening for mutations on Chrome browsers and suppress
// mutations by prepending a character for other browsers.
// The reason why we treat these separately is because
// of the way each browser processes IME inputs.
if (Browser.isChrome()) {
editor.setPlaceholder('');
editor._mutationHandler.stopObserving();
}
else {
this._textInputHandler.handle(' ');
}
}
compositionend(event) {
const { editor } = this;
// Ignore compositionend if not composing on blank line
if (!this._isComposingOnBlankLine) {
return;
}
this._isComposingOnBlankLine = false;
// Start listening for mutations on Chrome browsers and
// delete the prepended character introduced by compositionstart
// for other browsers.
if (Browser.isChrome()) {
editor.insertText(event.data);
editor.setPlaceholder(editor.placeholder);
editor._mutationHandler.startObserving();
}
else {
let startOfCompositionLine = editor.range.headSection.toPosition(0);
let endOfCompositionLine = editor.range.headSection.toPosition(event.data.length);
editor.run(postEditor => {
postEditor.deleteAtPosition(startOfCompositionLine, 1, { unit: TextUnit.CHAR });
postEditor.setRange(endOfCompositionLine);
});
}
}
cut(event) {
event.preventDefault();
this.copy(event);
this.editor.performDelete();
}
copy(event) {
event.preventDefault();
let { editor, editor: { range, post }, } = this;
post = post.trimTo(range);
let data = {
html: editor.serializePost(post, Format.HTML),
text: editor.serializePost(post, Format.TEXT),
mobiledoc: editor.serializePost(post, Format.MOBILEDOC),
};
editor.runCallbacks('willCopy', [data]);
setClipboardData(event, data, window);
}
paste(event) {
event.preventDefault();
let { editor } = this;
let range = editor.range;
if (!range.isCollapsed) {
editor.performDelete();
}
if (editor.post.isBlank) {
editor._insertEmptyMarkupSectionAtCursor();
}
let position = editor.range.head;
let targetFormat = this.modifierKeys.shift ? 'text' : 'html';
let pastedPost = parsePostFromPaste(event, editor, { targetFormat });
editor.runCallbacks('willPaste', [pastedPost]);
editor.run(postEditor => {
let nextPosition = postEditor.insertPost(position, pastedPost);
postEditor.setRange(nextPosition);
});
}
drop(event) {
event.preventDefault();
let { clientX: x, clientY: y } = event;
let { editor } = this;
let position = editor.positionAtPoint(x, y);
if (!position) {
this.logger.log('Could not find drop position');
return;
}
let post = parsePostFromDrop(event, editor, { logger: this.logger });
if (!post) {
this.logger.log('Could not determine post from drop event');
return;
}
editor.run(postEditor => {
let nextPosition = postEditor.insertPost(position, post);
postEditor.setRange(nextPosition);
});
}
_updateModifiersFromKey(key, { isDown }) {
if (key.isShiftKey()) {
this.modifierKeys.shift = isDown;
}
}
}
/**
* Used by {@link Editor} to manage its current state (cursor, active markups
* and active sections).
* @private
*/
class EditState {
constructor(editor) {
this.editor = editor;
let defaultState = {
range: Range.blankRange(),
activeMarkups: [],
activeSections: [],
activeSectionTagNames: [],
activeSectionAttributes: {},
};
this.prevState = this.state = defaultState;
}
updateRange(newRange) {
this.prevState = this.state;
this.state = this._readState(newRange);
}
destroy() {
this.editor = null;
this.prevState = this.state = null;
}
/**
* @return {Boolean}
*/
rangeDidChange() {
const { state, prevState } = this;
const { range } = state;
const { range: prevRange } = prevState;
return !prevRange.isEqual(range);
}
/**
* @return {Boolean} Whether the input mode (active markups or active section tag names)
* has changed.
*/
inputModeDidChange() {
const state = this.state;
const prevState = this.prevState;
return (!isArrayEqual(state.activeMarkups, prevState.activeMarkups) ||
!isArrayEqual(state.activeSectionTagNames, prevState.activeSectionTagNames) ||
!isArrayEqual(objectToSortedKVArray(state.activeSectionAttributes), objectToSortedKVArray(prevState.activeSectionAttributes)));
}
/**
* @return {Range}
*/
get range() {
return this.state.range;
}
/**
* @return {Section[]}
*/
get activeSections() {
return this.state.activeSections;
}
/**
* @return {Object}
*/
get activeSectionAttributes() {
return this.state.activeSectionAttributes;
}
/**
* @return {Markup[]}
*/
get activeMarkups() {
return this.state.activeMarkups;
}
/**
* Update the editor's markup state. This is used when, e.g.,
* a user types meta+B when the editor has a cursor but no selected text;
* in this case the editor needs to track that it has an active "b" markup
* and apply it to the next text the user types.
*/
toggleMarkupState(markup) {
if (contains(this.activeMarkups, markup)) {
this._removeActiveMarkup(markup);
}
else {
this._addActiveMarkup(markup);
}
}
_readState(range) {
let state = {
range,
activeMarkups: this._readActiveMarkups(range),
activeSections: this._readActiveSections(range),
};
// Section objects are 'live', so to check that they changed, we
// need to map their tagNames now (and compare to mapped tagNames later).
// In addition, to catch changes from ul -> ol, we keep track of the
// un-nested tag names (otherwise we'd only see li -> li change)
state.activeSectionTagNames = state.activeSections.map(s => {
return s.isNested ? s.parent.tagName : s.tagName;
});
state.activeSectionAttributes = this._readSectionAttributes(state.activeSections);
return state;
}
_readActiveSections(range) {
const { head, tail } = range;
const { editor } = this;
const { post } = editor;
if (range.isBlank) {
return [];
}
else {
return post.sections.readRange(head.section, tail.section);
}
}
_readActiveMarkups(range) {
const { editor } = this;
const { post } = editor;
return post.markupsInRange(range);
}
_readSectionAttributes(sections) {
return sections.reduce((sectionAttributes, s) => {
let attributes = getSectionAttributes(s);
Object.keys(attributes).forEach(attrName => {
let camelizedAttrName = attrName.replace(/^data-md-/, '');
let attrValue = attributes[attrName];
sectionAttributes[camelizedAttrName] = sectionAttributes[camelizedAttrName] || [];
if (!contains(sectionAttributes[camelizedAttrName], attrValue)) {
sectionAttributes[camelizedAttrName].push(attrValue);
}
});
return sectionAttributes;
}, {});
}
_removeActiveMarkup(markup) {
let index = this.state.activeMarkups.indexOf(markup);
this.state.activeMarkups.splice(index, 1);
}
_addActiveMarkup(markup) {
this.state.activeMarkups.push(markup);
}
}
function addHTMLSpaces(text) {
let nbsp = '\u00A0';
return text.replace(/ /g, ' ' + nbsp);
}
function createTextNode(dom, text) {
return dom.createTextNode(addHTMLSpaces(text));
}
function normalizeTagName$1(tagName) {
return tagName.toLowerCase();
}
var RENDER_TYPE = 'dom';
var ImageCard$1 = {
name: 'image',
type: RENDER_TYPE,
render({payload, env: {dom}}) {
let img = dom.createElement('img');
img.src = payload.src;
return img;
}
};
const MARKUP_SECTION_TYPE$1 = 1;
const IMAGE_SECTION_TYPE = 2;
const LIST_SECTION_TYPE$1 = 3;
const CARD_SECTION_TYPE = 10;
const MARKUP_SECTION_TAG_NAMES = [
'p', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'blockquote', 'pull-quote', 'aside'
].map(normalizeTagName$1);
const MARKUP_SECTION_ELEMENT_NAMES$1 = [
'p', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'blockquote', 'aside'
].map(normalizeTagName$1);
const LIST_SECTION_TAG_NAMES = [
'ul', 'ol'
].map(normalizeTagName$1);
const MARKUP_TYPES = [
'b', 'i', 'strong', 'em', 'a', 'u', 'sub', 'sup', 's', 'code'
].map(normalizeTagName$1);
function contains$1(array, item) {
return array.indexOf(item) !== -1;
}
function isValidSectionTagName(tagName, sectionType) {
tagName = normalizeTagName$1(tagName);
switch (sectionType) {
case MARKUP_SECTION_TYPE$1:
return contains$1(MARKUP_SECTION_TAG_NAMES, tagName);
case LIST_SECTION_TYPE$1:
return contains$1(LIST_SECTION_TAG_NAMES, tagName);
default:
throw new Error(`Cannot validate tagName for unknown section type "${sectionType}"`);
}
}
function isMarkupSectionElementName(tagName) {
tagName = normalizeTagName$1(tagName);
return contains$1(MARKUP_SECTION_ELEMENT_NAMES$1, tagName);
}
function isValidMarkerType(type) {
type = normalizeTagName$1(type);
return contains$1(MARKUP_TYPES, type);
}
function includes(array, detectValue) {
for (let i=0;i < array.length;i++) {
let value = array[i];
if (value === detectValue) {
return true;
}
}
return false;
}
const PROTOCOL_REGEXP = /^([a-z0-9.+-]+:)/i;
const badProtocols = [
'javascript:', // jshint ignore:line
'vbscript:' // jshint ignore:line
];
function getProtocol(url) {
let matches = url && url.match(PROTOCOL_REGEXP);
let protocol = (matches && matches[0]) || ':';
return protocol;
}
function sanitizeHref(url) {
let protocol = getProtocol(url).toLowerCase();
if (includes(badProtocols, protocol)) {
return `unsafe:${url}`;
}
return url;
}
/**
* @param attributes array
* @return obj with normalized attribute names (lowercased)
*/
function reduceAttributes(attributes) {
let obj = {};
for (let i = 0; i < attributes.length; i += 2) {
let key = attributes[i];
let val = attributes[i+1];
obj[key.toLowerCase()] = val;
}
return obj;
}
const VALID_ATTRIBUTES$2 = [
'data-md-text-align'
];
function _isValidAttribute(attr) {
return VALID_ATTRIBUTES$2.indexOf(attr) !== -1;
}
function handleMarkupSectionAttribute(element, attributeKey, attributeValue) {
if (!_isValidAttribute(attributeKey)) {
throw new Error(`Cannot use attribute: ${attributeKey}`);
}
element.setAttribute(attributeKey, attributeValue);
}
function defaultSectionElementRenderer(tagName, dom, attrsObj = {}) {
let element;
if (isMarkupSectionElementName(tagName)) {
element = dom.createElement(tagName);
Object.keys(attrsObj).forEach(k => {
handleMarkupSectionAttribute(element, k, attrsObj[k]);
});
} else {
element = dom.createElement('div');
element.setAttribute('class', tagName);
}
return element;
}
function sanitizeAttribute(tagName, attrName, attrValue) {
if (tagName === 'a' && attrName === 'href') {
return sanitizeHref(attrValue);
} else {
return attrValue;
}
}
function defaultMarkupElementRenderer(tagName, dom, attrsObj) {
let element = dom.createElement(tagName);
Object.keys(attrsObj).forEach(attrName => {
let attrValue = attrsObj[attrName];
attrValue = sanitizeAttribute(tagName, attrName, attrValue);
element.setAttribute(attrName, attrValue);
});
return element;
}
const MOBILEDOC_VERSION$5 = '0.2.0';
const IMAGE_SECTION_TAG_NAME = 'img';
function validateVersion(version) {
if (version !== MOBILEDOC_VERSION$5) {
throw new Error(`Unexpected Mobiledoc version "${version}"`);
}
}
class Renderer$1 {
constructor(mobiledoc, options) {
let {
cards,
cardOptions,
unknownCardHandler,
markupElementRenderer,
sectionElementRenderer,
dom
} = options;
let {
version,
sections: sectionData
} = mobiledoc;
validateVersion(version);
const [markerTypes, sections] = sectionData;
this.dom = dom;
this.root = dom.createDocumentFragment();
this.markerTypes = markerTypes;
this.sections = sections;
this.cards = cards;
this.cardOptions = cardOptions;
this.unknownCardHandler = unknownCardHandler || this._defaultUnknownCardHandler;
this.sectionElementRenderer = {
'__default__': defaultSectionElementRenderer
};
Object.keys(sectionElementRenderer).forEach(key => {
this.sectionElementRenderer[key.toLowerCase()] = sectionElementRenderer[key];
});
this.markupElementRenderer = {
'__default__': defaultMarkupElementRenderer
};
Object.keys(markupElementRenderer).forEach(key => {
this.markupElementRenderer[key.toLowerCase()] = markupElementRenderer[key];
});
this._renderCallbacks = [];
this._teardownCallbacks = [];
this._renderedChildNodes = [];
}
get _defaultUnknownCardHandler() {
return ({env: {name}}) => {
throw new Error(`Card "${name}" not found but no unknownCardHandler was registered`);
};
}
render() {
this.sections.forEach(section => {
let rendered = this.renderSection(section);
if (rendered) {
this.root.appendChild(rendered);
}
});
for (let i = 0; i < this._renderCallbacks.length; i++) {
this._renderCallbacks[i]();
}
// maintain a reference to child nodes so they can be cleaned up later by teardown
this._renderedChildNodes = [];
let node = this.root.firstChild;
while (node) {
this._renderedChildNodes.push(node);
node = node.nextSibling;
}
return { result: this.root, teardown: () => this.teardown() };
}
teardown() {
for (let i=0; i < this._teardownCallbacks.length; i++) {
this._teardownCallbacks[i]();
}
for (let i=0; i < this._renderedChildNodes.length; i++) {
let node = this._renderedChildNodes[i];
if (node.parentNode) {
node.parentNode.removeChild(node);
}
}
}
renderSection(section) {
const [type] = section;
switch (type) {
case MARKUP_SECTION_TYPE$1:
return this.renderMarkupSection(section);
case IMAGE_SECTION_TYPE:
return this.renderImageSection(section);
case LIST_SECTION_TYPE$1:
return this.renderListSection(section);
case CARD_SECTION_TYPE:
return this.renderCardSection(section);
default:
throw new Error(`Cannot render mobiledoc section of type "${type}"`);
}
}
renderMarkersOnElement(element, markers) {
let elements = [element];
let currentElement = element;
let pushElement = (openedElement) => {
currentElement.appendChild(openedElement);
elements.push(openedElement);
currentElement = openedElement;
};
for (let i=0, l=markers.length; i {
element.appendChild(this.renderListItem(li));
});
return element;
}
renderImageSection([type, src]) {
let element = this.dom.createElement(IMAGE_SECTION_TAG_NAME);
element.src = src;
return element;
}
findCard(name) {
for (let i=0; i < this.cards.length; i++) {
if (this.cards[i].name === name) {
return this.cards[i];
}
}
if (name === ImageCard$1.name) {
return ImageCard$1;
}
return this._createUnknownCard(name);
}
_createUnknownCard(name) {
return {
name,
type: RENDER_TYPE,
render: this.unknownCardHandler
};
}
_createCardArgument(card, payload={}) {
let env = {
name: card.name,
isInEditor: false,
dom: this.dom,
didRender: (callback) => this._registerRenderCallback(callback),
onTeardown: (callback) => this._registerTeardownCallback(callback)
};
let options = this.cardOptions;
return { env, options, payload };
}
_registerRenderCallback(callback) {
this._renderCallbacks.push(callback);
}
_registerTeardownCallback(callback) {
this._teardownCallbacks.push(callback);
}
renderCardSection([type, name, payload]) {
let card = this.findCard(name);
let cardArg = this._createCardArgument(card, payload);
let rendered = card.render(cardArg);
this._validateCardRender(rendered, card.name);
return rendered;
}
_validateCardRender(rendered, cardName) {
if (!rendered) {
return;
}
if (typeof rendered !== 'object') {
throw new Error(`Card "${cardName}" must render ${RENDER_TYPE}, but result was "${rendered}"`);
}
}
renderMarkupSection([type, tagName, markers]) {
tagName = tagName.toLowerCase();
if (!isValidSectionTagName(tagName, MARKUP_SECTION_TYPE$1)) {
return;
}
let renderer = this.sectionElementRendererFor(tagName);
let element = renderer(tagName, this.dom);
this.renderMarkersOnElement(element, markers);
return element;
}
sectionElementRendererFor(tagName) {
return this.sectionElementRenderer[tagName] ||
this.sectionElementRenderer.__default__;
}
}
const MARKUP_MARKER_TYPE = 0;
const ATOM_MARKER_TYPE = 1;
const MOBILEDOC_VERSION_0_3_0 = '0.3.0';
const MOBILEDOC_VERSION_0_3_1 = '0.3.1';
const MOBILEDOC_VERSION_0_3_2 = '0.3.2';
const IMAGE_SECTION_TAG_NAME$1 = 'img';
function validateVersion$1(version) {
switch (version) {
case MOBILEDOC_VERSION_0_3_0:
case MOBILEDOC_VERSION_0_3_1:
case MOBILEDOC_VERSION_0_3_2:
return;
default:
throw new Error(`Unexpected Mobiledoc version "${version}"`);
}
}
class Renderer$2 {
constructor(mobiledoc, state) {
let {
cards,
cardOptions,
atoms,
unknownCardHandler,
unknownAtomHandler,
markupElementRenderer,
sectionElementRenderer,
dom
} = state;
let {
version,
sections,
atoms: atomTypes,
cards: cardTypes,
markups: markerTypes
} = mobiledoc;
validateVersion$1(version);
this.dom = dom;
this.root = this.dom.createDocumentFragment();
this.sections = sections;
this.atomTypes = atomTypes;
this.cardTypes = cardTypes;
this.markerTypes = markerTypes;
this.cards = cards;
this.atoms = atoms;
this.cardOptions = cardOptions;
this.unknownCardHandler = unknownCardHandler || this._defaultUnknownCardHandler;
this.unknownAtomHandler = unknownAtomHandler || this._defaultUnknownAtomHandler;
this.sectionElementRenderer = {
'__default__': defaultSectionElementRenderer
};
Object.keys(sectionElementRenderer).forEach(key => {
this.sectionElementRenderer[key.toLowerCase()] = sectionElementRenderer[key];
});
this.markupElementRenderer = {
'__default__': defaultMarkupElementRenderer
};
Object.keys(markupElementRenderer).forEach(key => {
this.markupElementRenderer[key.toLowerCase()] = markupElementRenderer[key];
});
this._renderCallbacks = [];
this._teardownCallbacks = [];
}
get _defaultUnknownCardHandler() {
return ({env: {name}}) => {
throw new Error(`Card "${name}" not found but no unknownCardHandler was registered`);
};
}
get _defaultUnknownAtomHandler() {
return ({env: {name}}) => {
throw new Error(`Atom "${name}" not found but no unknownAtomHandler was registered`);
};
}
render() {
this.sections.forEach(section => {
let rendered = this.renderSection(section);
if (rendered) {
this.root.appendChild(rendered);
}
});
for (let i=0; i < this._renderCallbacks.length; i++) {
this._renderCallbacks[i]();
}
// maintain a reference to child nodes so they can be cleaned up later by teardown
this._renderedChildNodes = Array.prototype.slice.call(this.root.childNodes);
return { result: this.root, teardown: () => this.teardown() };
}
teardown() {
for (let i=0; i < this._teardownCallbacks.length; i++) {
this._teardownCallbacks[i]();
}
for (let i=0; i < this._renderedChildNodes.length; i++) {
let node = this._renderedChildNodes[i];
if (node.parentNode) {
node.parentNode.removeChild(node);
}
}
}
renderSection(section) {
const [type] = section;
switch (type) {
case MARKUP_SECTION_TYPE$1:
return this.renderMarkupSection(section);
case IMAGE_SECTION_TYPE:
return this.renderImageSection(section);
case LIST_SECTION_TYPE$1:
return this.renderListSection(section);
case CARD_SECTION_TYPE:
return this.renderCardSection(section);
default:
throw new Error(`Cannot render mobiledoc section of type "${type}"`);
}
}
renderMarkersOnElement(element, markers) {
let elements = [element];
let currentElement = element;
let pushElement = (openedElement) => {
currentElement.appendChild(openedElement);
elements.push(openedElement);
currentElement = openedElement;
};
for (let i=0, l=markers.length; i {
element.appendChild(this.renderListItem(li));
});
return element;
}
renderImageSection([type, src]) {
let element = this.dom.createElement(IMAGE_SECTION_TAG_NAME$1);
element.src = src;
return element;
}
findCard(name) {
for (let i=0; i < this.cards.length; i++) {
if (this.cards[i].name === name) {
return this.cards[i];
}
}
if (name === ImageCard$1.name) {
return ImageCard$1;
}
return this._createUnknownCard(name);
}
_findCardByIndex(index) {
let cardType = this.cardTypes[index];
if (!cardType) {
throw new Error(`No card definition found at index ${index}`);
}
let [ name, payload ] = cardType;
let card = this.findCard(name);
return {
card,
payload
};
}
_createUnknownCard(name) {
return {
name,
type: RENDER_TYPE,
render: this.unknownCardHandler
};
}
_createCardArgument(card, payload={}) {
let env = {
name: card.name,
isInEditor: false,
dom: this.dom,
didRender: (callback) => this._registerRenderCallback(callback),
onTeardown: (callback) => this._registerTeardownCallback(callback)
};
let options = this.cardOptions;
return { env, options, payload };
}
_registerTeardownCallback(callback) {
this._teardownCallbacks.push(callback);
}
_registerRenderCallback(callback) {
this._renderCallbacks.push(callback);
}
renderCardSection([type, index]) {
let { card, payload } = this._findCardByIndex(index);
let cardArg = this._createCardArgument(card, payload);
let rendered = card.render(cardArg);
this._validateCardRender(rendered, card.name);
return rendered;
}
_validateCardRender(rendered, cardName) {
if (!rendered) {
return;
}
if (typeof rendered !== 'object') {
throw new Error(`Card "${cardName}" must render ${RENDER_TYPE}, but result was "${rendered}"`);
}
}
findAtom(name) {
for (let i=0; i < this.atoms.length; i++) {
if (this.atoms[i].name === name) {
return this.atoms[i];
}
}
return this._createUnknownAtom(name);
}
_createUnknownAtom(name) {
return {
name,
type: RENDER_TYPE,
render: this.unknownAtomHandler
};
}
_createAtomArgument(atom, value, payload) {
let env = {
name: atom.name,
isInEditor: false,
dom: this.dom,
onTeardown: (callback) => this._registerTeardownCallback(callback)
};
let options = this.cardOptions;
return { env, options, value, payload };
}
_validateAtomRender(rendered, atomName) {
if (!rendered) {
return;
}
if (typeof rendered !== 'object') {
throw new Error(`Atom "${atomName}" must render ${RENDER_TYPE}, but result was "${rendered}"`);
}
}
_findAtomByIndex(index) {
let atomType = this.atomTypes[index];
if (!atomType) {
throw new Error(`No atom definition found at index ${index}`);
}
let [ name, value, payload ] = atomType;
let atom = this.findAtom(name);
return {
atom,
value,
payload
};
}
_renderAtom(index) {
let { atom, value, payload } = this._findAtomByIndex(index);
let atomArg = this._createAtomArgument(atom, value, payload);
let rendered = atom.render(atomArg);
this._validateAtomRender(rendered, atom.name);
return rendered || createTextNode(this.dom, '');
}
renderMarkupSection([type, tagName, markers, attributes = []]) {
tagName = tagName.toLowerCase();
if (!isValidSectionTagName(tagName, MARKUP_SECTION_TYPE$1)) {
return;
}
let attrsObj = reduceAttributes(attributes);
let renderer = this.sectionElementRendererFor(tagName);
let element = renderer(tagName, this.dom, attrsObj);
this.renderMarkersOnElement(element, markers);
return element;
}
sectionElementRendererFor(tagName) {
return this.sectionElementRenderer[tagName] ||
this.sectionElementRenderer.__default__;
}
}
/**
* runtime DOM renderer
* renders a mobiledoc to DOM
*
* input: mobiledoc
* output: DOM
*/
function validateCards$1(cards) {
if (!Array.isArray(cards)) {
throw new Error('`cards` must be passed as an array');
}
for (let i=0; i < cards.length; i++) {
let card = cards[i];
if (card.type !== RENDER_TYPE) {
throw new Error(`Card "${card.name}" must be of type "${RENDER_TYPE}", was "${card.type}"`);
}
if (!card.render) {
throw new Error(`Card "${card.name}" must define \`render\``);
}
}
}
function validateAtoms$1(atoms) {
if (!Array.isArray(atoms)) {
throw new Error('`atoms` must be passed as an array');
}
for (let i=0; i < atoms.length; i++) {
let atom = atoms[i];
if (atom.type !== RENDER_TYPE) {
throw new Error(`Atom "${atom.name}" must be type "${RENDER_TYPE}", was "${atom.type}"`);
}
if (!atom.render) {
throw new Error(`Atom "${atom.name}" must define \`render\``);
}
}
}
class RendererFactory {
constructor({
cards=[],
atoms=[],
cardOptions={},
unknownCardHandler,
unknownAtomHandler,
markupElementRenderer={},
sectionElementRenderer={},
dom,
markupSanitizer=null
}={}) {
validateCards$1(cards);
validateAtoms$1(atoms);
if (!dom) {
if (typeof window === 'undefined') {
throw new Error('A `dom` option must be provided to the renderer when running without window.document');
}
dom = window.document;
}
this.options = {
cards,
atoms,
cardOptions,
unknownCardHandler,
unknownAtomHandler,
markupElementRenderer,
sectionElementRenderer,
dom,
markupSanitizer
};
}
render(mobiledoc) {
let { version } = mobiledoc;
switch (version) {
case MOBILEDOC_VERSION$5:
case undefined:
case null:
return new Renderer$1(mobiledoc, this.options).render();
case MOBILEDOC_VERSION_0_3_0:
case MOBILEDOC_VERSION_0_3_1:
case MOBILEDOC_VERSION_0_3_2:
return new Renderer$2(mobiledoc, this.options).render();
default:
throw new Error(`Unexpected Mobiledoc version "${version}"`);
}
}
}
var ImageCard$2 = {
name: 'image-card',
type: 'text',
render() {}
};
var RENDER_TYPE$1 = 'text';
const MARKUP_SECTION_TYPE$2 = 1;
const IMAGE_SECTION_TYPE$1 = 2;
const LIST_SECTION_TYPE$2 = 3;
const CARD_SECTION_TYPE$1 = 10;
/**
* runtime Text renderer
* renders a mobiledoc to Text
*
* input: mobiledoc
* output: Text (string)
*/
const LINE_BREAK = '\n';
const MOBILEDOC_VERSION$6 = '0.2.0';
function validateVersion$2(version) {
if (version !== MOBILEDOC_VERSION$6) {
throw new Error(`Unexpected Mobiledoc version "${version}"`);
}
}
class Renderer$3 {
constructor(mobiledoc, state) {
let { cards, cardOptions, atoms, unknownCardHandler } = state;
let { version, sections: sectionData } = mobiledoc;
validateVersion$2(version);
let [, sections] = sectionData;
this.root = [];
this.sections = sections;
this.cards = cards;
this.atoms = atoms;
this.cardOptions = cardOptions;
this.unknownCardHandler = unknownCardHandler || this._defaultUnknownCardHandler;
this._teardownCallbacks = [];
}
render() {
this.sections.forEach(section => {
this.root.push(this.renderSection(section));
});
let result = this.root.join(LINE_BREAK);
return { result, teardown: () => this.teardown() };
}
teardown() {
for (let i=0; i < this._teardownCallbacks.length; i++) {
this._teardownCallbacks[i]();
}
}
get _defaultUnknownCardHandler() {
return () => {
// for the text renderer, a missing card is a no-op
};
}
renderSection(section) {
const [type] = section;
switch (type) {
case MARKUP_SECTION_TYPE$2:
return this.renderMarkupSection(section);
case IMAGE_SECTION_TYPE$1:
return this.renderImageSection(section);
case LIST_SECTION_TYPE$2:
return this.renderListSection(section);
case CARD_SECTION_TYPE$1:
return this.renderCardSection(section);
default:
throw new Error('Unimplemented renderer for type ' + type);
}
}
renderImageSection() {
return '';
}
renderListSection([type, tagName, items]) {
return items.map(
li => this.renderListItem(li)
).join(LINE_BREAK);
}
renderListItem(markers) {
return this.renderMarkers(markers);
}
findCard(name) {
for (let i=0; i < this.cards.length; i++) {
if (this.cards[i].name === name) {
return this.cards[i];
}
}
if (name === ImageCard$2.name) {
return ImageCard$2;
}
return this._createUnknownCard(name);
}
_createUnknownCard(name) {
return {
name,
type: RENDER_TYPE$1,
render: this.unknownCardHandler
};
}
renderCardSection([type, name, payload]) {
let card = this.findCard(name);
let cardArg = this._createCardArgument(card, payload);
let rendered = card.render(cardArg);
this._validateCardRender(rendered, card.name);
return rendered || '';
}
_validateCardRender(rendered, cardName) {
if (!rendered) {
return;
}
if (typeof rendered !== 'string') {
throw new Error(`Card "${cardName}" must render ${RENDER_TYPE$1}, but result was ${typeof rendered}"`);
}
}
_registerTeardownCallback(callback) {
this._teardownCallbacks.push(callback);
}
_createCardArgument(card, payload={}) {
let env = {
name: card.name,
isInEditor: false,
onTeardown: (callback) => this._registerTeardownCallback(callback)
};
let options = this.cardOptions;
return { env, options, payload };
}
renderMarkupSection([type, tagName, markers]) {
return this.renderMarkers(markers);
}
renderMarkers(markers) {
let str = '';
markers.forEach(m => {
let [, , text] = m;
str += text;
});
return str;
}
}
const MARKUP_MARKER_TYPE$1 = 0;
const ATOM_MARKER_TYPE$1 = 1;
/**
* runtime Text renderer
* renders a mobiledoc to Text
*
* input: mobiledoc
* output: Text (string)
*/
const LINE_BREAK$1 = '\n';
const MOBILEDOC_VERSION_0_3 = '0.3.0';
const MOBILEDOC_VERSION_0_3_1$1 = '0.3.1';
const MOBILEDOC_VERSION_0_3_2$1 = '0.3.2';
function validateVersion$3(version) {
if (
version !== MOBILEDOC_VERSION_0_3 &&
version !== MOBILEDOC_VERSION_0_3_1$1 &&
version !== MOBILEDOC_VERSION_0_3_2$1
) {
throw new Error(`Unexpected Mobiledoc version "${version}"`);
}
}
class Renderer$4 {
constructor(mobiledoc, state) {
let { cards, cardOptions, atoms, unknownCardHandler, unknownAtomHandler } = state;
let { version, sections, atoms: atomTypes, cards: cardTypes } = mobiledoc;
validateVersion$3(version);
this.root = [];
this.sections = sections;
this.atomTypes = atomTypes;
this.cardTypes = cardTypes;
this.cards = cards;
this.atoms = atoms;
this.cardOptions = cardOptions;
this.unknownCardHandler = unknownCardHandler || this._defaultUnknownCardHandler;
this.unknownAtomHandler = unknownAtomHandler || this._defaultUnknownAtomHandler;
this._teardownCallbacks = [];
}
render() {
this.sections.forEach(section => {
this.root.push(this.renderSection(section));
});
let result = this.root.join(LINE_BREAK$1);
return { result, teardown: () => this.teardown() };
}
teardown() {
for (let i=0; i < this._teardownCallbacks.length; i++) {
this._teardownCallbacks[i]();
}
}
get _defaultUnknownCardHandler() {
return () => {
// for the text renderer, a missing card is a no-op
};
}
get _defaultUnknownAtomHandler() {
return ({ value }) => {
return value || '';
};
}
renderSection(section) {
const [type] = section;
switch (type) {
case MARKUP_SECTION_TYPE$2:
return this.renderMarkupSection(section);
case IMAGE_SECTION_TYPE$1:
return this.renderImageSection(section);
case LIST_SECTION_TYPE$2:
return this.renderListSection(section);
case CARD_SECTION_TYPE$1:
return this.renderCardSection(section);
default:
throw new Error('Unimplemented renderer for type ' + type);
}
}
renderImageSection() {
return '';
}
renderListSection([type, tagName, items]) {
return items.map(
li => this.renderListItem(li)
).join(LINE_BREAK$1);
}
renderListItem(markers) {
return this.renderMarkers(markers);
}
findCard(name) {
for (let i=0; i < this.cards.length; i++) {
if (this.cards[i].name === name) {
return this.cards[i];
}
}
if (name === ImageCard$2.name) {
return ImageCard$2;
}
return this._createUnknownCard(name);
}
_findCardByIndex(index) {
let cardType = this.cardTypes[index];
if (!cardType) {
throw new Error(`No card definition found at index ${index}`);
}
let [ name, payload ] = cardType;
let card = this.findCard(name);
return {
card,
payload
};
}
_createUnknownCard(name) {
return {
name,
type: RENDER_TYPE$1,
render: this.unknownCardHandler
};
}
renderCardSection([type, index]) {
let { card, payload } = this._findCardByIndex(index);
let cardArg = this._createCardArgument(card, payload);
let rendered = card.render(cardArg);
this._validateCardRender(rendered, card.name);
return rendered || '';
}
_validateCardRender(rendered, cardName) {
if (!rendered) {
return;
}
if (typeof rendered !== 'string') {
throw new Error(`Card "${cardName}" must render ${RENDER_TYPE$1}, but result was ${typeof rendered}"`);
}
}
_registerTeardownCallback(callback) {
this._teardownCallbacks.push(callback);
}
_createCardArgument(card, payload={}) {
let env = {
name: card.name,
isInEditor: false,
onTeardown: (callback) => this._registerTeardownCallback(callback)
};
let options = this.cardOptions;
return { env, options, payload };
}
renderMarkupSection([type, tagName, markers]) {
return this.renderMarkers(markers);
}
findAtom(name) {
for (let i=0; i < this.atoms.length; i++) {
if (this.atoms[i].name === name) {
return this.atoms[i];
}
}
return this._createUnknownAtom(name);
}
_createUnknownAtom(name) {
return {
name,
type: RENDER_TYPE$1,
render: this.unknownAtomHandler
};
}
_createAtomArgument(atom, value, payload) {
let env = {
name: atom.name,
onTeardown: (callback) => this._registerTeardownCallback(callback)
};
let options = this.cardOptions;
return { env, options, value, payload };
}
_validateAtomRender(rendered, atomName) {
if (!rendered) {
return;
}
if (typeof rendered !== 'string') {
throw new Error(`Atom "${atomName}" must render ${RENDER_TYPE$1}, but result was ${typeof rendered}"`);
}
}
_findAtomByIndex(index) {
let atomType = this.atomTypes[index];
if (!atomType) {
throw new Error(`No atom definition found at index ${index}`);
}
let [ name, value, payload ] = atomType;
let atom = this.findAtom(name);
return {
atom,
value,
payload
};
}
_renderAtom(index) {
let { atom, value, payload } = this._findAtomByIndex(index);
let atomArg = this._createAtomArgument(atom, value, payload);
let rendered = atom.render(atomArg);
this._validateAtomRender(rendered, atom.name);
return rendered || '';
}
renderMarkers(markers) {
let str = '';
markers.forEach(m => {
let [type, , , value] = m;
switch (type) {
case MARKUP_MARKER_TYPE$1:
str += value;
break;
case ATOM_MARKER_TYPE$1:
str += this._renderAtom(value);
break;
default:
throw new Error(`Unknown markup type (${type})`);
}
});
return str;
}
}
/**
* runtime Text renderer
* renders a mobiledoc to Text
*
* input: mobiledoc
* output: Text (string)
*/
function validateCards$2(cards) {
if (!Array.isArray(cards)) {
throw new Error('`cards` must be passed as an array');
}
for (let i=0; i < cards.length; i++) {
let card = cards[i];
if (card.type !== RENDER_TYPE$1) {
throw new Error(`Card "${card.name}" must be type "${RENDER_TYPE$1}", was "${card.type}"`);
}
if (!card.render) {
throw new Error(`Card "${card.name}" must define \`render\``);
}
}
}
function validateAtoms$2(atoms) {
if (!Array.isArray(atoms)) {
throw new Error('`atoms` must be passed as an array');
}
for (let i=0; i < atoms.length; i++) {
let atom = atoms[i];
if (atom.type !== RENDER_TYPE$1) {
throw new Error(`Atom "${atom.name}" must be type "${RENDER_TYPE$1}", was "${atom.type}"`);
}
if (!atom.render) {
throw new Error(`Atom "${atom.name}" must define \`render\``);
}
}
}
class RendererFactory$1 {
constructor({cards, atoms, cardOptions, unknownCardHandler, unknownAtomHandler}={}) {
cards = cards || [];
validateCards$2(cards);
atoms = atoms || [];
validateAtoms$2(atoms);
cardOptions = cardOptions || {};
this.state = {cards, atoms, cardOptions, unknownCardHandler, unknownAtomHandler};
}
render(mobiledoc) {
let { version } = mobiledoc;
switch (version) {
case MOBILEDOC_VERSION$6:
return new Renderer$3(mobiledoc, this.state).render();
case undefined:
case null:
case MOBILEDOC_VERSION_0_3:
case MOBILEDOC_VERSION_0_3_1$1:
case MOBILEDOC_VERSION_0_3_2$1:
return new Renderer$4(mobiledoc, this.state).render();
default:
throw new Error(`Unexpected Mobiledoc version "${version}"`);
}
}
}
class Logger {
constructor(type, manager) {
this.type = type;
this.manager = manager;
}
isEnabled() {
return this.manager.isEnabled(this.type);
}
log(...args) {
args.unshift(`[${this.type}]`);
if (this.isEnabled()) {
window.console.log(...args);
}
}
}
class LogManager {
constructor() {
this.enabledTypes = [];
this.allEnabled = false;
}
for(type) {
return new Logger(type, this);
}
enableAll() {
this.allEnabled = true;
}
enableTypes(types) {
this.enabledTypes = this.enabledTypes.concat(types);
}
disable() {
this.enabledTypes = [];
this.allEnabled = false;
}
isEnabled(type) {
return this.allEnabled || this.enabledTypes.indexOf(type) !== -1;
}
}
const defaults = {
placeholder: 'Write here...',
spellcheck: true,
autofocus: true,
showLinkTooltips: true,
undoDepth: 5,
undoBlockTimeout: 5000,
cards: [],
atoms: [],
cardOptions: {},
unknownCardHandler: ({ env }) => {
throw new MobiledocError(`Unknown card encountered: ${env.name}`);
},
unknownAtomHandler: ({ env }) => {
throw new MobiledocError(`Unknown atom encountered: ${env.name}`);
},
mobiledoc: null,
html: null,
tooltipPlugin: DEFAULT_TOOLTIP_PLUGIN,
};
const CALLBACK_QUEUES$1 = {
DID_UPDATE: 'didUpdate',
WILL_RENDER: 'willRender',
DID_RENDER: 'didRender',
WILL_DELETE: 'willDelete',
DID_DELETE: 'didDelete',
WILL_HANDLE_NEWLINE: 'willHandleNewline',
CURSOR_DID_CHANGE: 'cursorDidChange',
DID_REPARSE: 'didReparse',
POST_DID_CHANGE: 'postDidChange',
INPUT_MODE_DID_CHANGE: 'inputModeDidChange',
WILL_COPY: 'willCopy',
WILL_PASTE: 'willPaste',
};
var Format;
(function (Format) {
Format["MOBILEDOC"] = "mobiledoc";
Format["HTML"] = "html";
Format["TEXT"] = "text";
})(Format || (Format = {}));
var TextUnit;
(function (TextUnit) {
TextUnit["CHAR"] = "char";
TextUnit["WORD"] = "word";
})(TextUnit || (TextUnit = {}));
/**
* The Editor is a core component of mobiledoc-kit. After instantiating
* an editor, use {@link Editor#render} to display the editor on the web page.
*
* An editor uses a {@link Post} internally to represent the displayed document.
* The post can be serialized as mobiledoc using {@link Editor#serialize}. Mobiledoc
* is the transportable "over-the-wire" format (JSON) that is suited for persisting
* and sharing between editors and renderers (for display, e.g.), whereas the Post
* model is better suited for programmatic editing.
*
* The editor will call registered callbacks for certain state changes. These are:
* * {@link Editor#cursorDidChange} -- The cursor position or selection changed.
* * {@link Editor#postDidChange} -- The contents of the post changed due to user input or
* programmatic editing. This hook can be used with {@link Editor#serialize}
* to auto-save a post as it is being edited.
* * {@link Editor#inputModeDidChange} -- The active section(s) or markup(s) at the current cursor
* position or selection have changed. This hook can be used with
* {@link Editor#activeMarkups} and {@link Editor#activeSections} to implement
* a custom toolbar.
* * {@link Editor#onTextInput} -- Register callbacks when the user enters text
* that matches a given string or regex.
* * {@link Editor#beforeToggleMarkup} -- Register callbacks that will be run before
* applying changes from {@link Editor#toggleMarkup}
*/
class Editor {
/**
* @param {Object} [options]
* @param {Object} [options.mobiledoc] The mobiledoc to load into the editor.
* Supersedes `options.html`.
* @param {String|DOM} [options.html] The html (as a string or DOM fragment)
* to parse and load into the editor.
* Will be ignored if `options.mobiledoc` is also passed.
* @param {Array} [options.parserPlugins=[]]
* @param {Array} [options.cards=[]] The cards that the editor may render.
* @param {Array} [options.atoms=[]] The atoms that the editor may render.
* @param {Function} [options.unknownCardHandler] Invoked by the editor's renderer
* whenever it encounters an unknown card.
* @param {Function} [options.unknownAtomHandler] Invoked by the editor's renderer
* whenever it encounters an unknown atom.
* @param {String} [options.placeholder] Default text to show before user starts typing.
* @param {Boolean} [options.spellcheck=true] Whether to enable spellcheck
* @param {Boolean} [options.autofocus=true] Whether to focus the editor when it is first rendered.
* @param {Boolean} [options.showLinkTooltips=true] Whether to show the url tooltip for links
* @param {number} [options.undoDepth=5] How many undo levels will be available.
* Set to 0 to disable undo/redo functionality.
* @public
*/
constructor(options = {}) {
assert('editor create accepts an options object. For legacy usage passing an element for the first argument, consider the `html` option for loading DOM or HTML posts. For other cases call `editor.render(domNode)` after editor creation', options && !options.nodeType);
this._views = [];
this.isEditable = true;
this._parserPlugins = options.parserPlugins || [];
// FIXME: This should merge onto this.options
mergeWithOptions(this, defaults, options);
this.cards.push(ImageCard);
DEFAULT_KEY_COMMANDS.forEach(kc => this.registerKeyCommand(kc));
this._logManager = new LogManager();
this._parser = new DOMParser(this.builder);
let { cards, atoms, unknownCardHandler, unknownAtomHandler, cardOptions } = this;
this._renderer = new Renderer(this, cards, atoms, unknownCardHandler, unknownAtomHandler, cardOptions);
this.post = this.loadPost();
this._renderTree = new RenderTree(this.post);
this._editHistory = new EditHistory(this, this.undoDepth, this.undoBlockTimeout);
this._eventManager = new EventManager(this);
this._mutationHandler = new MutationHandler(this);
this._editState = new EditState(this);
this._callbacks = new LifecycleCallbacks(values(CALLBACK_QUEUES$1));
this._beforeHooks = { toggleMarkup: [] };
this._isComposingOnBlankLine = false;
DEFAULT_TEXT_INPUT_HANDLERS.forEach(handler => this.onTextInput(handler));
this.hasRendered = false;
this.isDestroyed = false;
}
/**
* Turns on verbose logging for the editor.
* @param {Array} [logTypes=[]] If present, only the given log types will be logged.
* @public
*/
enableLogging(logTypes = []) {
if (logTypes.length === 0) {
this._logManager.enableAll();
}
else {
this._logManager.enableTypes(logTypes);
}
}
/**
* Disable all logging
* @public
*/
disableLogging() {
this._logManager.disable();
}
/**
* @private
*/
loggerFor(type) {
return this._logManager.for(type);
}
/**
* The editor's instance of a post node builder.
* @type {PostNodeBuilder}
*/
get builder() {
if (!this._builder) {
this._builder = new PostNodeBuilder();
}
return this._builder;
}
loadPost() {
let { mobiledoc, html } = this;
if (mobiledoc) {
return mobiledocParsers.parse(this.builder, mobiledoc);
}
else if (html) {
if (typeof html === 'string') {
let options = { plugins: this._parserPlugins };
return new HTMLParser(this.builder, options).parse(html);
}
else {
let dom = html;
return this._parser.parse(dom);
}
}
else {
return this.builder.createPost([this.builder.createMarkupSection()]);
}
}
rerender() {
let postRenderNode = this.post.renderNode;
// if we haven't rendered this post's renderNode before, mark it dirty
if (!postRenderNode.element) {
assert('Must call `render` before `rerender` can be called', this.hasRendered);
postRenderNode.element = this.element;
postRenderNode.markDirty();
}
this.runCallbacks(CALLBACK_QUEUES$1.WILL_RENDER);
this._mutationHandler.suspendObservation(() => {
this._renderer.render(this._renderTree);
});
this.runCallbacks(CALLBACK_QUEUES$1.DID_RENDER);
}
/**
* @param {Element} element The DOM element to render into.
* Its contents will be replaced by the editor's rendered post.
* @public
*/
render(element) {
assert('Cannot render an editor twice. Use `rerender` to update the ' + 'rendering of an existing editor instance.', !this.hasRendered);
element.spellcheck = this.spellcheck;
clearChildNodes(element);
this.element = element;
if (this.showLinkTooltips) {
this._addTooltip();
}
// A call to `run` will trigger the didUpdatePostCallbacks hooks with a
// postEditor.
this.run(() => { });
// Only set `hasRendered` to true after calling `run` to ensure that
// no cursorDidChange or other callbacks get fired before the editor is
// done rendering
this.hasRendered = true;
this.rerender();
this._mutationHandler.init();
this._eventManager.init();
if (this.isEditable === false) {
this.disableEditing();
}
else {
this.enableEditing();
}
if (this.autofocus) {
this.selectRange(this.post.headPosition());
}
}
_addTooltip() {
this.addView(new Tooltip({
rootElement: this.element,
showForTag: 'a',
editor: this,
}));
}
get keyCommands() {
if (!this._keyCommands) {
this._keyCommands = [];
}
return this._keyCommands;
}
/**
* @param {Object} keyCommand The key command to register. It must specify a
* modifier key (meta, ctrl, etc), a string representing the ascii key, and
* a `run` method that will be passed the editor instance when the key command
* is invoked
* @public
*/
registerKeyCommand(rawKeyCommand) {
const keyCommand = buildKeyCommand(rawKeyCommand);
assert('Key Command is not valid', validateKeyCommand(keyCommand));
this.keyCommands.unshift(keyCommand);
}
/**
* @param {String} name If the keyCommand event has a name attribute it can be removed.
* @public
*/
unregisterKeyCommands(name) {
for (let i = this.keyCommands.length - 1; i > -1; i--) {
let keyCommand = this.keyCommands[i];
if (keyCommand.name === name) {
this.keyCommands.splice(i, 1);
}
}
}
/**
* Convenience for {@link PostEditor#deleteAtPosition}. Deletes and puts the
* cursor in the new position.
* @public
*/
deleteAtPosition(position, direction, { unit }) {
this.run(postEditor => {
let nextPosition = postEditor.deleteAtPosition(position, direction, { unit });
postEditor.setRange(nextPosition);
});
}
/**
* Convenience for {@link PostEditor#deleteRange}. Deletes and puts the
* cursor in the new position.
* @param {Range} range
* @public
*/
deleteRange(range) {
this.run(postEditor => {
let nextPosition = postEditor.deleteRange(range);
postEditor.setRange(nextPosition);
});
}
/**
* @private
*/
performDelete({ direction, unit } = { direction: Direction.BACKWARD, unit: TextUnit.CHAR }) {
const { range } = this;
this.runCallbacks(CALLBACK_QUEUES$1.WILL_DELETE, [range, direction, unit]);
if (range.isCollapsed) {
this.deleteAtPosition(range.head, direction, { unit });
}
else {
this.deleteRange(range);
}
this.runCallbacks(CALLBACK_QUEUES$1.DID_DELETE, [range, direction, unit]);
}
handleNewline(event) {
if (!this.hasCursor()) {
return;
}
event.preventDefault();
let { range } = this;
this.run(postEditor => {
let cursorSection;
if (!range.isCollapsed) {
let nextPosition = postEditor.deleteRange(range);
cursorSection = nextPosition.section;
if (cursorSection && cursorSection.isBlank) {
postEditor.setRange(cursorSection.headPosition());
return;
}
}
// Above logic might delete redundant range, so callback must run after it.
let defaultPrevented = false;
const event = {
preventDefault() {
defaultPrevented = true;
},
};
this.runCallbacks(CALLBACK_QUEUES$1.WILL_HANDLE_NEWLINE, [event]);
if (defaultPrevented) {
return;
}
cursorSection = postEditor.splitSection(range.head)[1];
postEditor.setRange(cursorSection.headPosition());
});
}
/**
* Notify the editor that the post did change, and run associated
* callbacks.
* @private
*/
_postDidChange() {
this.runCallbacks(CALLBACK_QUEUES$1.POST_DID_CHANGE);
}
/**
* Selects the given range or position. If given a collapsed range or a position, this positions the cursor
* at the range's position. Otherwise a selection is created in the editor
* surface encompassing the range.
* @param {Range|Position} range
*/
selectRange(range) {
range = toRange(range);
this.cursor.selectRange(range);
this.range = range;
}
get cursor() {
return new Cursor(this);
}
/**
* Return the current range for the editor (may be cached).
* @return {Range}
*/
get range() {
return this._editState.range;
}
set range(newRange) {
this._editState.updateRange(newRange);
if (this._editState.rangeDidChange()) {
this._rangeDidChange();
}
if (this._editState.inputModeDidChange()) {
this._inputModeDidChange();
}
}
_readRangeFromDOM() {
this.range = this.cursor.offsets;
}
setPlaceholder(placeholder) {
setData(this.element, 'placeholder', placeholder);
}
_reparsePost() {
let post = this._parser.parse(this.element);
this.run(postEditor => {
postEditor.removeAllSections();
postEditor.migrateSectionsFromPost(post);
postEditor.setRange(Range.blankRange());
});
this.runCallbacks(CALLBACK_QUEUES$1.DID_REPARSE);
this._postDidChange();
}
_reparseSections(sections = []) {
let currentRange;
sections.forEach(section => {
this._parser.reparseSection(section, this._renderTree);
});
this._removeDetachedSections();
if (this._renderTree.isDirty) {
currentRange = this.range;
}
// force the current snapshot's range to remain the same rather than
// rereading it from DOM after the new character is applied and the browser
// updates the cursor position
const editHistory = this._editHistory;
const pendingSnapshot = editHistory._pendingSnapshot;
const range = pendingSnapshot.range;
this.run(() => {
pendingSnapshot.range = range;
});
this.rerender();
if (currentRange) {
this.selectRange(currentRange);
}
this.runCallbacks(CALLBACK_QUEUES$1.DID_REPARSE);
this._postDidChange();
}
// FIXME this should be able to be removed now -- if any sections are detached,
// it's due to a bug in the code.
_removeDetachedSections() {
forEach(filter(this.post.sections, s => !s.renderNode.isAttached()), s => s.renderNode.scheduleForRemoval());
}
/**
* The sections from the cursor's selection start to the selection end
* @type {Section[]}
*/
get activeSections() {
return this._editState.activeSections;
}
get activeSection() {
const { activeSections } = this;
return activeSections[activeSections.length - 1];
}
get activeSectionAttributes() {
return this._editState.activeSectionAttributes;
}
detectMarkupInRange(range, markupTagName) {
let markups = this.post.markupsInRange(range);
return detect(markups, markup => {
return markup.hasTag(markupTagName);
});
}
/**
* @type {Markup[]}
* @public
*/
get activeMarkups() {
return this._editState.activeMarkups;
}
/**
* @param {Markup|String} markup A markup instance, or a string (e.g. "b")
* @return {boolean}
*/
hasActiveMarkup(markup) {
let matchesFn;
if (typeof markup === 'string') {
let tagName = normalizeTagName(markup);
matchesFn = m => m.tagName === tagName;
}
else {
matchesFn = m => m === markup;
}
return !!detect(this.activeMarkups, matchesFn);
}
/**
* @param {String} version The mobiledoc version to serialize to.
* @return {Mobiledoc} Serialized mobiledoc
* @public
*/
serialize(version = MOBILEDOC_VERSION$4) {
return this.serializePost(this.post, Format.MOBILEDOC, { version });
}
serializeTo(format) {
let post = this.post;
return this.serializePost(post, format);
}
serializePost(post, format, options = {}) {
assert(`Unrecognized serialization format ${format}`, contains(Object.values(Format), format));
if (format === Format.MOBILEDOC) {
let version = options.version || MOBILEDOC_VERSION$4;
return mobiledocRenderers.render(post, version);
}
else {
let mobiledoc = this.serializePost(post, Format.MOBILEDOC);
let unknownCardHandler = () => { };
let unknownAtomHandler = () => { };
let rendererOptions = { unknownCardHandler, unknownAtomHandler };
switch (format) {
case Format.HTML: {
if (Environment.hasDOM()) {
const rendered = new RendererFactory(rendererOptions).render(mobiledoc);
return `${serializeHTML(rendered.result)}
`;
}
else {
// Fallback to text serialization
return this.serializePost(post, Format.TEXT, options);
}
}
case Format.TEXT: {
let rendered = new RendererFactory$1(rendererOptions).render(mobiledoc);
return rendered.result;
}
}
}
}
addView(view) {
this._views.push(view);
}
removeAllViews() {
this._views.forEach(v => v.destroy());
this._views = [];
}
/**
* Whether the editor has a cursor (or a selected range).
* It is possible for the editor to be focused but not have a selection.
* In this case, key events will fire but the editor will not be able to
* determine a cursor position, so they will be ignored.
* @return {boolean}
* @public
*/
hasCursor() {
return this.cursor.hasCursor();
}
/**
* Tears down the editor's attached event listeners and views.
* @public
*/
destroy() {
this.isDestroyed = true;
if (this._hasSelection()) {
this.cursor.clearSelection();
}
if (this._hasFocus()) {
this.element.blur(); // FIXME This doesn't blur the element on IE11
}
this._mutationHandler.destroy();
this._eventManager.destroy();
this.removeAllViews();
this._renderer.destroy();
this._editState.destroy();
}
/**
* Keep the user from directly editing the post using the keyboard and mouse.
* Modification via the programmatic API is still permitted.
* @see Editor#enableEditing
* @public
*/
disableEditing() {
this.isEditable = false;
if (this.hasRendered) {
this._eventManager.stop();
this.element.setAttribute('contentEditable', 'false');
this.setPlaceholder('');
this.selectRange(Range.blankRange());
}
}
/**
* Allow the user to directly interact with editing a post via keyboard and mouse input.
* Editor instances are editable by default. Use this method to re-enable
* editing after disabling it.
* @see Editor#disableEditing
* @public
*/
enableEditing() {
this.isEditable = true;
if (this.hasRendered) {
this._eventManager.start();
this.element.setAttribute('contentEditable', 'true');
this.setPlaceholder(this.placeholder);
}
}
/**
* Change a cardSection into edit mode
* If called before the card has been rendered, it will be marked so that
* it is rendered in edit mode when it gets rendered.
* @param {CardSection} cardSection
* @public
*/
editCard(cardSection) {
this._setCardMode(cardSection, CardMode.EDIT);
}
/**
* Change a cardSection into display mode
* If called before the card has been rendered, it will be marked so that
* it is rendered in display mode when it gets rendered.
* @param {CardSection} cardSection
* @return undefined
* @public
*/
displayCard(cardSection) {
this._setCardMode(cardSection, CardMode.DISPLAY);
}
/**
* Run a new post editing session. Yields a block with a new {@link PostEditor}
* instance. This instance can be used to interact with the post abstract.
* Rendering will be deferred until after the callback is completed.
*
* Usage:
* ```
* let markerRange = this.range;
* editor.run((postEditor) => {
* postEditor.deleteRange(markerRange);
* // editing surface not updated yet
* postEditor.schedule(() => {
* console.log('logs during rerender flush');
* });
* // logging not yet flushed
* });
* // editing surface now updated.
* // logging now flushed
* ```
*
* @param {Function} callback Called with an instance of
* {@link PostEditor} as its argument.
* @return {Mixed} The return value of `callback`.
* @public
*/
run(callback) {
const postEditor = new PostEditor(this);
postEditor.begin();
this._editHistory.snapshot();
const result = callback(postEditor);
this.runCallbacks(CALLBACK_QUEUES$1.DID_UPDATE, [postEditor]);
postEditor.complete();
this._readRangeFromDOM();
if (postEditor._shouldCancelSnapshot) {
this._editHistory._pendingSnapshot = null;
}
this._editHistory.storeSnapshot(postEditor.editActionTaken);
return result;
}
/**
* @param {Function} callback Called with `postEditor` as its argument.
* @public
*/
didUpdatePost(callback) {
this.addCallback(CALLBACK_QUEUES$1.DID_UPDATE, callback);
}
/**
* @param {Function} callback Called when the post has changed, either via
* user input or programmatically. Use with {@link Editor#serialize} to
* retrieve the post in portable mobiledoc format.
*/
postDidChange(callback) {
this.addCallback(CALLBACK_QUEUES$1.POST_DID_CHANGE, callback);
}
/**
* Register a handler that will be invoked by the editor after the user enters
* matching text.
* @param {Object} inputHandler
* @param {String} inputHandler.name Required. Used by identifying handlers.
* @param {String} [inputHandler.text] Required if `match` is not provided
* @param {RegExp} [inputHandler.match] Required if `text` is not provided
* @param {Function} inputHandler.run This callback is invoked with the {@link Editor}
* instance and an array of matches. If `text` was provided,
* the matches array will equal [`text`], and if a `match`
* regex was provided the matches array will be the result of
* `match.exec` on the matching text. The callback is called
* after the matching text has been inserted.
* @public
*/
onTextInput(inputHandler) {
this._eventManager.registerInputHandler(inputHandler);
}
/**
* Unregister all text input handlers
*
* @public
*/
unregisterAllTextInputHandlers() {
this._eventManager.unregisterAllTextInputHandlers();
}
/**
* Unregister text input handler by name
* @param {String} name The name of handler to be removed
*
* @public
*/
unregisterTextInputHandler(name) {
this._eventManager.unregisterInputHandler(name);
}
/**
* @param {Function} callback Called when the editor's state (active markups or
* active sections) has changed, either via user input or programmatically
*/
inputModeDidChange(callback) {
this.addCallback(CALLBACK_QUEUES$1.INPUT_MODE_DID_CHANGE, callback);
}
/**
* @param {Function} callback This callback will be called before the editor
* is rendered.
* @public
*/
willRender(callback) {
this.addCallback(CALLBACK_QUEUES$1.WILL_RENDER, callback);
}
/**
* @param {Function} callback This callback will be called after the editor
* is rendered.
* @public
*/
didRender(callback) {
this.addCallback(CALLBACK_QUEUES$1.DID_RENDER, callback);
}
willCopy(callback) {
this.addCallback(CALLBACK_QUEUES$1.WILL_COPY, callback);
}
/**
* @param {Function} callback This callback will be called before pasting.
* @public
*/
willPaste(callback) {
this.addCallback(CALLBACK_QUEUES$1.WILL_PASTE, callback);
}
/**
* @param {Function} callback This callback will be called before deleting.
* @public
*/
willDelete(callback) {
this.addCallback(CALLBACK_QUEUES$1.WILL_DELETE, callback);
}
/**
* @param {Function} callback This callback will be called after deleting.
* @public
*/
didDelete(callback) {
this.addCallback(CALLBACK_QUEUES$1.DID_DELETE, callback);
}
/**
* @param {Function} callback This callback will be called before handling new line.
* @public
*/
willHandleNewline(callback) {
this.addCallback(CALLBACK_QUEUES$1.WILL_HANDLE_NEWLINE, callback);
}
/**
* @param {Function} callback This callback will be called every time the cursor
* position (or selection) changes.
* @public
*/
cursorDidChange(callback) {
this.addCallback(CALLBACK_QUEUES$1.CURSOR_DID_CHANGE, callback);
}
_rangeDidChange() {
if (this.hasRendered) {
this.runCallbacks(CALLBACK_QUEUES$1.CURSOR_DID_CHANGE);
}
}
_inputModeDidChange() {
this.runCallbacks(CALLBACK_QUEUES$1.INPUT_MODE_DID_CHANGE);
}
_insertEmptyMarkupSectionAtCursor() {
this.run(postEditor => {
const section = postEditor.builder.createMarkupSection('p');
postEditor.insertSectionBefore(this.post.sections, section);
postEditor.setRange(section.toRange());
});
}
/**
* @callback editorBeforeCallback
* @param { Object } details
* @param { Markup } details.markup
* @param { Range } details.range
* @param { boolean } details.willAdd Whether the markup will be applied
*/
/**
* Register a callback that will be run before {@link Editor#toggleMarkup} is applied.
* If any callback returns literal `false`, the toggling of markup will be canceled.
* Note this only applies to calling `editor#toggleMarkup`. Using `editor.run` and
* modifying markup with the `postEditor` will skip any `beforeToggleMarkup` callbacks.
* @param {editorBeforeCallback}
*/
beforeToggleMarkup(callback) {
this._beforeHooks.toggleMarkup.push(callback);
}
/**
* Toggles the given markup at the editor's current {@link Range}.
* If the range is collapsed this changes the editor's state so that the
* next characters typed will be affected. If there is text selected
* (aka a non-collapsed range), the selections' markup will be toggled.
* If the editor is not focused and has no active range, nothing happens.
* Hooks added using #beforeToggleMarkup will be run before toggling,
* and if any of them returns literal false, toggling the markup will be canceled
* and no change will be applied.
* @param {String} markup E.g. "b", "em", "a"
* @param {Object} [attributes={}] E.g. {href: "http://bustle.com"}
* @public
* @see PostEditor#toggleMarkup
*/
toggleMarkup(markupTag, attributes = {}) {
const markup = this.builder.createMarkup(markupTag, attributes);
const { range } = this;
const willAdd = !this.detectMarkupInRange(range, markup.tagName);
const shouldCancel = this._runBeforeHooks('toggleMarkup', { markup, range, willAdd });
if (shouldCancel) {
return;
}
if (range.isCollapsed) {
this._editState.toggleMarkupState(markup);
this._inputModeDidChange();
// when clicking a button to toggle markup, the button can end up being focused,
// so ensure the editor is focused
this._ensureFocus();
}
else {
this.run(postEditor => postEditor.toggleMarkup(markup, range));
}
}
// If the editor has a selection but is not focused, focus it
_ensureFocus() {
if (this._hasSelection() && !this._hasFocus()) {
this.focus();
}
}
focus() {
this.element.focus();
}
/**
* Whether there is a selection inside the editor's element.
* It's possible to have a selection but not have focus.
* @see #_hasFocus
* @return {Boolean}
*/
_hasSelection() {
const { cursor } = this;
return this.hasRendered && (cursor._hasCollapsedSelection() || cursor._hasSelection());
}
/**
* Whether the editor's element is focused
* It's possible to be focused but have no selection
* @see #_hasSelection
* @return {Boolean}
*/
_hasFocus() {
return this.root.activeElement === this.element;
}
/**
* Toggles the tagName for the current active section(s). This will skip
* non-markerable sections. E.g. if the editor's range includes a "P" MarkupSection
* and a CardSection, only the MarkupSection will be toggled.
* @param {String} tagName The new tagname to change to.
* @public
* @see PostEditor#toggleSection
*/
toggleSection(tagName) {
this.run(postEditor => postEditor.toggleSection(tagName, this.range));
}
/**
* Sets an attribute for the current active section(s).
*
* @param {String} key The attribute. The only valid attribute is 'text-align'.
* @param {String} value The value of the attribute.
* @public
* @see PostEditor#setAttribute
*/
setAttribute(key, value) {
this.run(postEditor => postEditor.setAttribute(key, value, this.range));
}
/**
* Removes an attribute from the current active section(s).
*
* @param {String} key The attribute. The only valid attribute is 'text-align'.
* @public
* @see PostEditor#removeAttribute
*/
removeAttribute(key) {
this.run(postEditor => postEditor.removeAttribute(key, this.range));
}
/**
* Finds and runs the first matching key command for the event
*
* If multiple commands are bound to a key combination, the
* first matching one is run.
*
* If a command returns `false` then the next matching command
* is run instead.
*
* @param {Event} event The keyboard event triggered by the user
* @return {Boolean} true when a command was successfully run
* @private
*/
handleKeyCommand(event) {
const keyCommands = findKeyCommands(this.keyCommands, event);
for (let i = 0; i < keyCommands.length; i++) {
let keyCommand = keyCommands[i];
if (keyCommand.run(this) !== false) {
event.preventDefault();
return true;
}
}
return false;
}
/**
* Inserts the text at the current cursor position. If the editor has
* no current cursor position, nothing will be inserted. If the editor's
* range is not collapsed, it will be deleted before insertion.
*
* @param {String} text
* @public
*/
insertText(text) {
if (!this.hasCursor()) {
return;
}
if (this.post.isBlank) {
this._insertEmptyMarkupSectionAtCursor();
}
let { activeMarkups, range, range: { head: position }, } = this;
this.run(postEditor => {
if (!range.isCollapsed) {
position = postEditor.deleteRange(range);
}
postEditor.insertTextWithMarkup(position, text, activeMarkups);
});
}
/**
* Inserts an atom at the current cursor position. If the editor has
* no current cursor position, nothing will be inserted. If the editor's
* range is not collapsed, it will be deleted before insertion.
* @param {String} atomName
* @param {String} [atomText='']
* @param {Object} [atomPayload={}]
* @return {Atom} The inserted atom.
* @public
*/
insertAtom(atomName, atomText = '', atomPayload = {}) {
if (!this.hasCursor()) {
return;
}
if (this.post.isBlank) {
this._insertEmptyMarkupSectionAtCursor();
}
let atom;
let { range } = this;
this.run(postEditor => {
let position = range.head;
atom = postEditor.builder.createAtom(atomName, atomText, atomPayload);
if (!range.isCollapsed) {
position = postEditor.deleteRange(range);
}
postEditor.insertMarkers(position, [atom]);
});
return atom;
}
/**
* Inserts a card at the section after the current cursor position. If the editor has
* no current cursor position, nothing will be inserted. If the editor's
* range is not collapsed, it will be deleted before insertion. If the cursor is in
* a blank section, it will be replaced with a card section.
* The editor's cursor will be placed at the end of the inserted card.
* @param {String} cardName
* @param {Object} [cardPayload={}]
* @param {Boolean} [inEditMode=false] Whether the card should be inserted in edit mode.
* @return {Card} The inserted Card section.
* @public
*/
insertCard(cardName, cardPayload = {}, inEditMode = false) {
if (!this.hasCursor()) {
return;
}
if (this.post.isBlank) {
this._insertEmptyMarkupSectionAtCursor();
}
let card;
let { range } = this;
this.run(postEditor => {
let position = range.tail;
card = postEditor.builder.createCardSection(cardName, cardPayload);
if (inEditMode) {
this.editCard(card);
}
if (!range.isCollapsed) {
position = postEditor.deleteRange(range);
}
let section = position.section;
if (isNested(section)) {
section = section.parent;
}
if (section.isBlank) {
postEditor.replaceSection(section, card);
}
else {
let collection = this.post.sections;
postEditor.insertSectionBefore(collection, card, section.next);
}
// It is important to explicitly set the range to the end of the card.
// Otherwise it is possible to create an inconsistent state in the
// browser. For instance, if the user clicked a button that
// called `editor.insertCard`, the editor surface may retain
// the selection but lose focus, and the next keystroke by the user
// will cause an unexpected DOM mutation (which can wipe out the
// card).
// See: https://github.com/bustle/mobiledoc-kit/issues/286
postEditor.setRange(card.tailPosition());
});
return card;
}
/**
* @param {integer} x x-position in viewport
* @param {integer} y y-position in viewport
* @return {Position|null}
*/
positionAtPoint(x, y) {
return Position.atPoint(x, y, this);
}
/**
* @private
*/
_setCardMode(cardSection, mode) {
const renderNode = cardSection.renderNode;
if (renderNode && renderNode.isRendered) {
const cardNode = renderNode.cardNode;
cardNode[mode]();
}
else {
cardSection.setInitialMode(mode);
}
}
triggerEvent(context, eventName, event) {
this._eventManager._trigger(context, eventName, event);
}
addCallback(queueName, callback) {
this._callbacks.addCallback(queueName, callback);
}
addCallbackOnce(queueName, callback) {
this._callbacks.addCallbackOnce(queueName, callback);
}
runCallbacks(queueName, args) {
if (this.isDestroyed) {
// TODO warn that callback attempted after editor was destroyed
return;
}
this._callbacks.runCallbacks(queueName, args);
}
/**
* Runs each callback for the given hookName.
* Only the hookName 'toggleMarkup' is currently supported
* @return {Boolean} shouldCancel Whether the action in `hookName` should be canceled
* @private
*/
_runBeforeHooks(hookName, ...args) {
let hooks = this._beforeHooks[hookName] || [];
for (let i = 0; i < hooks.length; i++) {
if (hooks[i](...args) === false) {
return true;
}
}
}
get root() {
const root = this.element.getRootNode();
// COMPAT: Only Chrome implements the DocumentOrShadowRoot mixin for
// ShadowRoot; other browsers still implement it on the Document
// interface. (2020/08/08)
// https://developer.mozilla.org/en-US/docs/Web/API/ShadowRoot#Properties
if (root.getSelection === undefined && this.element.ownerDocument !== null) {
return this.element.ownerDocument;
}
return root;
}
}
var version = '##VERSION##';
exports.DOMParser = DOMParser;
exports.Editor = Editor;
exports.Error = MobiledocError;
exports.ImageCard = ImageCard;
exports.MOBILEDOC_VERSION = MOBILEDOC_VERSION$4;
exports.Position = Position;
exports.PostNodeBuilder = PostNodeBuilder;
exports.Range = Range;
exports.Renderer = mobiledocRenderers;
exports.UI = ui;
exports.VERSION = version;
//# sourceMappingURL=mobiledoc.cjs.map