UNPKG

slate

Version:

A completely customizable framework for building rich text editors.

658 lines (520 loc) 42.5 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); var _isPlainObject = require('is-plain-object'); var _isPlainObject2 = _interopRequireDefault(_isPlainObject); var _immutable = require('immutable'); var _character = require('./character'); var _character2 = _interopRequireDefault(_character); var _mark = require('./mark'); var _mark2 = _interopRequireDefault(_mark); var _leaf = require('./leaf'); var _leaf2 = _interopRequireDefault(_leaf); var _modelTypes = require('../constants/model-types'); var _modelTypes2 = _interopRequireDefault(_modelTypes); var _generateKey = require('../utils/generate-key'); var _generateKey2 = _interopRequireDefault(_generateKey); var _memoize = require('../utils/memoize'); var _memoize2 = _interopRequireDefault(_memoize); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } /** * Default properties. * * @type {Object} */ var DEFAULTS = { characters: new _immutable.List(), key: undefined }; /** * Text. * * @type {Text} */ var Text = function (_Record) { _inherits(Text, _Record); function Text() { _classCallCheck(this, Text); return _possibleConstructorReturn(this, (Text.__proto__ || Object.getPrototypeOf(Text)).apply(this, arguments)); } _createClass(Text, [{ key: 'addMark', /** * Add a `mark` at `index` and `length`. * * @param {Number} index * @param {Number} length * @param {Mark} mark * @return {Text} */ value: function addMark(index, length, mark) { var marks = new _immutable.Set([mark]); return this.addMarks(index, length, marks); } /** * Add a `set` of marks at `index` and `length`. * * @param {Number} index * @param {Number} length * @param {Set<Mark>} set * @return {Text} */ }, { key: 'addMarks', value: function addMarks(index, length, set) { var characters = this.characters.map(function (char, i) { if (i < index) return char; if (i >= index + length) return char; var _char = char, marks = _char.marks; marks = marks.union(set); char = char.set('marks', marks); return char; }); return this.set('characters', characters); } /** * Derive a set of decorated characters with `decorations`. * * @param {List<Decoration>} decorations * @return {List<Character>} */ }, { key: 'getDecoratedCharacters', value: function getDecoratedCharacters(decorations) { var node = this; var _node = node, key = _node.key, characters = _node.characters; // PERF: Exit early if there are no characters to be decorated. if (characters.size == 0) return characters; decorations.forEach(function (range) { var startKey = range.startKey, endKey = range.endKey, startOffset = range.startOffset, endOffset = range.endOffset, marks = range.marks; var hasStart = startKey == key; var hasEnd = endKey == key; var index = hasStart ? startOffset : 0; var length = hasEnd ? endOffset - index : characters.size; node = node.addMarks(index, length, marks); }); return node.characters; } /** * Get the decorations for the node from a `schema`. * * @param {Schema} schema * @return {Array} */ }, { key: 'getDecorations', value: function getDecorations(schema) { return schema.__getDecorations(this); } /** * Derive the leaves for a list of `characters`. * * @param {Array|Void} decorations (optional) * @return {List<Leaf>} */ }, { key: 'getLeaves', value: function getLeaves() { var decorations = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; var characters = this.getDecoratedCharacters(decorations); var leaves = []; // PERF: cache previous values for faster lookup. var prevChar = void 0; var prevLeaf = void 0; // If there are no characters, return one empty range. if (characters.size == 0) { leaves.push({}); } // Otherwise, loop the characters and build the leaves... else { characters.forEach(function (char, i) { var marks = char.marks, text = char.text; // The first one can always just be created. if (i == 0) { prevChar = char; prevLeaf = { text: text, marks: marks }; leaves.push(prevLeaf); return; } // Otherwise, compare the current and previous marks. var prevMarks = prevChar.marks; var isSame = (0, _immutable.is)(marks, prevMarks); // If the marks are the same, add the text to the previous range. if (isSame) { prevChar = char; prevLeaf.text += text; return; } // Otherwise, create a new range. prevChar = char; prevLeaf = { text: text, marks: marks }; leaves.push(prevLeaf); }, []); } // PERF: convert the leaves to immutable objects after iterating. leaves = new _immutable.List(leaves.map(function (object) { return new _leaf2.default(object); })); // Return the leaves. return leaves; } /** * Get all of the marks on the text. * * @return {OrderedSet<Mark>} */ }, { key: 'getMarks', value: function getMarks() { var array = this.getMarksAsArray(); return new _immutable.OrderedSet(array); } /** * Get all of the marks on the text as an array * * @return {Array} */ }, { key: 'getMarksAsArray', value: function getMarksAsArray() { return this.characters.reduce(function (array, char) { return array.concat(char.marks.toArray()); }, []); } /** * Get the marks on the text at `index`. * * @param {Number} index * @return {Set<Mark>} */ }, { key: 'getMarksAtIndex', value: function getMarksAtIndex(index) { if (index == 0) return _mark2.default.createSet(); var characters = this.characters; var char = characters.get(index - 1); if (!char) return _mark2.default.createSet(); return char.marks; } /** * Get a node by `key`, to parallel other nodes. * * @param {String} key * @return {Node|Null} */ }, { key: 'getNode', value: function getNode(key) { return this.key == key ? this : null; } /** * Check if the node has a node by `key`, to parallel other nodes. * * @param {String} key * @return {Boolean} */ }, { key: 'hasNode', value: function hasNode(key) { return !!this.getNode(key); } /** * Insert `text` at `index`. * * @param {Numbder} index * @param {String} text * @param {String} marks (optional) * @return {Text} */ }, { key: 'insertText', value: function insertText(index, text, marks) { var characters = this.characters; var chars = _character2.default.createList(text.split('').map(function (char) { return { text: char, marks: marks }; })); characters = characters.slice(0, index).concat(chars).concat(characters.slice(index)); return this.set('characters', characters); } /** * Regenerate the node's key. * * @return {Text} */ }, { key: 'regenerateKey', value: function regenerateKey() { var key = (0, _generateKey2.default)(); return this.set('key', key); } /** * Remove a `mark` at `index` and `length`. * * @param {Number} index * @param {Number} length * @param {Mark} mark * @return {Text} */ }, { key: 'removeMark', value: function removeMark(index, length, mark) { var characters = this.characters.map(function (char, i) { if (i < index) return char; if (i >= index + length) return char; var _char2 = char, marks = _char2.marks; marks = marks.remove(mark); char = char.set('marks', marks); return char; }); return this.set('characters', characters); } /** * Remove text from the text node at `index` for `length`. * * @param {Number} index * @param {Number} length * @return {Text} */ }, { key: 'removeText', value: function removeText(index, length) { var characters = this.characters; var start = index; var end = index + length; characters = characters.filterNot(function (char, i) { return start <= i && i < end; }); return this.set('characters', characters); } /** * Return a JSON representation of the text. * * @param {Object} options * @return {Object} */ }, { key: 'toJSON', value: function toJSON() { var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; var object = { kind: this.kind, leaves: this.getLeaves().toArray().map(function (r) { return r.toJSON(); }) }; if (options.preserveKeys) { object.key = this.key; } return object; } /** * Alias `toJS`. */ }, { key: 'toJS', value: function toJS(options) { return this.toJSON(options); } /** * Update a `mark` at `index` and `length` with `properties`. * * @param {Number} index * @param {Number} length * @param {Mark} mark * @param {Object} properties * @return {Text} */ }, { key: 'updateMark', value: function updateMark(index, length, mark, properties) { var newMark = mark.merge(properties); var characters = this.characters.map(function (char, i) { if (i < index) return char; if (i >= index + length) return char; var _char3 = char, marks = _char3.marks; if (!marks.has(mark)) return char; marks = marks.remove(mark); marks = marks.add(newMark); char = char.set('marks', marks); return char; }); return this.set('characters', characters); } /** * Validate the text node against a `schema`. * * @param {Schema} schema * @return {Object|Void} */ }, { key: 'validate', value: function validate(schema) { return schema.validateNode(this); } }, { key: 'kind', /** * Get the node's kind. * * @return {String} */ get: function get() { return 'text'; } /** * Is the node empty? * * @return {Boolean} */ }, { key: 'isEmpty', get: function get() { return this.text == ''; } /** * Get the concatenated text of the node. * * @return {String} */ }, { key: 'text', get: function get() { return this.characters.reduce(function (string, char) { return string + char.text; }, ''); } }], [{ key: 'create', /** * Create a new `Text` with `attrs`. * * @param {Object|Array|List|String|Text} attrs * @return {Text} */ value: function create() { var attrs = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : ''; if (Text.isText(attrs)) { return attrs; } if (typeof attrs == 'string') { attrs = { leaves: [{ text: attrs }] }; } if ((0, _isPlainObject2.default)(attrs)) { if (attrs.text) { var _attrs = attrs, text = _attrs.text, marks = _attrs.marks, key = _attrs.key; attrs = { key: key, leaves: [{ text: text, marks: marks }] }; } return Text.fromJSON(attrs); } throw new Error('`Text.create` only accepts objects, arrays, strings or texts, but you passed it: ' + attrs); } /** * Create a list of `Texts` from `elements`. * * @param {Array<Text|Object>|List<Text|Object>} elements * @return {List<Text>} */ }, { key: 'createList', value: function createList() { var elements = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; if (_immutable.List.isList(elements) || Array.isArray(elements)) { var list = new _immutable.List(elements.map(Text.create)); return list; } throw new Error('`Text.createList` only accepts arrays or lists, but you passed it: ' + elements); } /** * Create a `Text` from a JSON `object`. * * @param {Object|Text} object * @return {Text} */ }, { key: 'fromJSON', value: function fromJSON(object) { if (Text.isText(object)) { return object; } var _object$leaves = object.leaves, leaves = _object$leaves === undefined ? [] : _object$leaves, _object$key = object.key, key = _object$key === undefined ? (0, _generateKey2.default)() : _object$key; var characters = leaves.map(_leaf2.default.fromJSON).reduce(function (l, r) { return l.concat(r.getCharacters()); }, new _immutable.List()); var node = new Text({ characters: characters, key: key }); return node; } /** * Alias `fromJS`. */ }, { key: 'isText', /** * Check if `any` is a `Text`. * * @param {Any} any * @return {Boolean} */ value: function isText(any) { return !!(any && any[_modelTypes2.default.TEXT]); } /** * Check if `any` is a list of texts. * * @param {Any} any * @return {Boolean} */ }, { key: 'isTextList', value: function isTextList(any) { return _immutable.List.isList(any) && any.every(function (item) { return Text.isText(item); }); } }]); return Text; }((0, _immutable.Record)(DEFAULTS)); /** * Attach a pseudo-symbol for type checking. */ Text.fromJS = Text.fromJSON; Text.prototype[_modelTypes2.default.TEXT] = true; /** * Memoize read methods. */ (0, _memoize2.default)(Text.prototype, ['getMarks', 'getMarksAsArray'], { takesArguments: false }); (0, _memoize2.default)(Text.prototype, ['getDecoratedCharacters', 'getDecorations', 'getLeaves', 'getMarksAtIndex', 'validate'], { takesArguments: true }); /** * Export. * * @type {Text} */ exports.default = Text; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/models/text.js"],"names":["DEFAULTS","characters","key","undefined","Text","index","length","mark","marks","addMarks","set","map","char","i","union","decorations","node","size","forEach","range","startKey","endKey","startOffset","endOffset","hasStart","hasEnd","schema","__getDecorations","getDecoratedCharacters","leaves","prevChar","prevLeaf","push","text","prevMarks","isSame","object","array","getMarksAsArray","reduce","concat","toArray","createSet","get","getNode","chars","createList","split","slice","remove","start","end","filterNot","options","kind","getLeaves","r","toJSON","preserveKeys","properties","newMark","merge","has","add","validateNode","string","attrs","isText","fromJSON","Error","elements","isList","Array","isArray","list","create","l","getCharacters","any","TEXT","every","item","fromJS","prototype","takesArguments"],"mappings":"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","file":"text.js","sourcesContent":["\nimport isPlainObject from 'is-plain-object'\nimport { List, OrderedSet, Record, Set, is } from 'immutable'\n\nimport Character from './character'\nimport Mark from './mark'\nimport Leaf from './leaf'\nimport MODEL_TYPES from '../constants/model-types'\nimport generateKey from '../utils/generate-key'\nimport memoize from '../utils/memoize'\n\n/**\n * Default properties.\n *\n * @type {Object}\n */\n\nconst DEFAULTS = {\n  characters: new List(),\n  key: undefined,\n}\n\n/**\n * Text.\n *\n * @type {Text}\n */\n\nclass Text extends Record(DEFAULTS) {\n\n  /**\n   * Create a new `Text` with `attrs`.\n   *\n   * @param {Object|Array|List|String|Text} attrs\n   * @return {Text}\n   */\n\n  static create(attrs = '') {\n    if (Text.isText(attrs)) {\n      return attrs\n    }\n\n    if (typeof attrs == 'string') {\n      attrs = { leaves: [{ text: attrs }] }\n    }\n\n    if (isPlainObject(attrs)) {\n      if (attrs.text) {\n        const { text, marks, key } = attrs\n        attrs = { key, leaves: [{ text, marks }] }\n      }\n\n      return Text.fromJSON(attrs)\n    }\n\n    throw new Error(`\\`Text.create\\` only accepts objects, arrays, strings or texts, but you passed it: ${attrs}`)\n  }\n\n  /**\n   * Create a list of `Texts` from `elements`.\n   *\n   * @param {Array<Text|Object>|List<Text|Object>} elements\n   * @return {List<Text>}\n   */\n\n  static createList(elements = []) {\n    if (List.isList(elements) || Array.isArray(elements)) {\n      const list = new List(elements.map(Text.create))\n      return list\n    }\n\n    throw new Error(`\\`Text.createList\\` only accepts arrays or lists, but you passed it: ${elements}`)\n  }\n\n  /**\n   * Create a `Text` from a JSON `object`.\n   *\n   * @param {Object|Text} object\n   * @return {Text}\n   */\n\n  static fromJSON(object) {\n    if (Text.isText(object)) {\n      return object\n    }\n\n    const {\n      leaves = [],\n      key = generateKey(),\n    } = object\n\n    const characters = leaves\n      .map(Leaf.fromJSON)\n      .reduce((l, r) => l.concat(r.getCharacters()), new List())\n\n    const node = new Text({\n      characters,\n      key,\n    })\n\n    return node\n  }\n\n  /**\n   * Alias `fromJS`.\n   */\n\n  static fromJS = Text.fromJSON\n\n  /**\n   * Check if `any` is a `Text`.\n   *\n   * @param {Any} any\n   * @return {Boolean}\n   */\n\n  static isText(any) {\n    return !!(any && any[MODEL_TYPES.TEXT])\n  }\n\n  /**\n   * Check if `any` is a list of texts.\n   *\n   * @param {Any} any\n   * @return {Boolean}\n   */\n\n  static isTextList(any) {\n    return List.isList(any) && any.every(item => Text.isText(item))\n  }\n\n  /**\n   * Get the node's kind.\n   *\n   * @return {String}\n   */\n\n  get kind() {\n    return 'text'\n  }\n\n  /**\n   * Is the node empty?\n   *\n   * @return {Boolean}\n   */\n\n  get isEmpty() {\n    return this.text == ''\n  }\n\n  /**\n   * Get the concatenated text of the node.\n   *\n   * @return {String}\n   */\n\n  get text() {\n    return this.characters.reduce((string, char) => string + char.text, '')\n  }\n\n  /**\n   * Add a `mark` at `index` and `length`.\n   *\n   * @param {Number} index\n   * @param {Number} length\n   * @param {Mark} mark\n   * @return {Text}\n   */\n\n  addMark(index, length, mark) {\n    const marks = new Set([mark])\n    return this.addMarks(index, length, marks)\n  }\n\n  /**\n   * Add a `set` of marks at `index` and `length`.\n   *\n   * @param {Number} index\n   * @param {Number} length\n   * @param {Set<Mark>} set\n   * @return {Text}\n   */\n\n  addMarks(index, length, set) {\n    const characters = this.characters.map((char, i) => {\n      if (i < index) return char\n      if (i >= index + length) return char\n      let { marks } = char\n      marks = marks.union(set)\n      char = char.set('marks', marks)\n      return char\n    })\n\n    return this.set('characters', characters)\n  }\n\n  /**\n   * Derive a set of decorated characters with `decorations`.\n   *\n   * @param {List<Decoration>} decorations\n   * @return {List<Character>}\n   */\n\n  getDecoratedCharacters(decorations) {\n    let node = this\n    const { key, characters } = node\n\n    // PERF: Exit early if there are no characters to be decorated.\n    if (characters.size == 0) return characters\n\n    decorations.forEach((range) => {\n      const { startKey, endKey, startOffset, endOffset, marks } = range\n      const hasStart = startKey == key\n      const hasEnd = endKey == key\n      const index = hasStart ? startOffset : 0\n      const length = hasEnd ? endOffset - index : characters.size\n      node = node.addMarks(index, length, marks)\n    })\n\n    return node.characters\n  }\n\n  /**\n   * Get the decorations for the node from a `schema`.\n   *\n   * @param {Schema} schema\n   * @return {Array}\n   */\n\n  getDecorations(schema) {\n    return schema.__getDecorations(this)\n  }\n\n  /**\n   * Derive the leaves for a list of `characters`.\n   *\n   * @param {Array|Void} decorations (optional)\n   * @return {List<Leaf>}\n   */\n\n  getLeaves(decorations = []) {\n    const characters = this.getDecoratedCharacters(decorations)\n    let leaves = []\n\n    // PERF: cache previous values for faster lookup.\n    let prevChar\n    let prevLeaf\n\n    // If there are no characters, return one empty range.\n    if (characters.size == 0) {\n      leaves.push({})\n    }\n\n    // Otherwise, loop the characters and build the leaves...\n    else {\n      characters.forEach((char, i) => {\n        const { marks, text } = char\n\n        // The first one can always just be created.\n        if (i == 0) {\n          prevChar = char\n          prevLeaf = { text, marks }\n          leaves.push(prevLeaf)\n          return\n        }\n\n        // Otherwise, compare the current and previous marks.\n        const prevMarks = prevChar.marks\n        const isSame = is(marks, prevMarks)\n\n        // If the marks are the same, add the text to the previous range.\n        if (isSame) {\n          prevChar = char\n          prevLeaf.text += text\n          return\n        }\n\n        // Otherwise, create a new range.\n        prevChar = char\n        prevLeaf = { text, marks }\n        leaves.push(prevLeaf)\n      }, [])\n    }\n\n    // PERF: convert the leaves to immutable objects after iterating.\n    leaves = new List(leaves.map(object => new Leaf(object)))\n\n    // Return the leaves.\n    return leaves\n  }\n\n  /**\n   * Get all of the marks on the text.\n   *\n   * @return {OrderedSet<Mark>}\n   */\n\n  getMarks() {\n    const array = this.getMarksAsArray()\n    return new OrderedSet(array)\n  }\n\n  /**\n   * Get all of the marks on the text as an array\n   *\n   * @return {Array}\n   */\n\n  getMarksAsArray() {\n    return this.characters.reduce((array, char) => {\n      return array.concat(char.marks.toArray())\n    }, [])\n  }\n\n  /**\n   * Get the marks on the text at `index`.\n   *\n   * @param {Number} index\n   * @return {Set<Mark>}\n   */\n\n  getMarksAtIndex(index) {\n    if (index == 0) return Mark.createSet()\n    const { characters } = this\n    const char = characters.get(index - 1)\n    if (!char) return Mark.createSet()\n    return char.marks\n  }\n\n  /**\n   * Get a node by `key`, to parallel other nodes.\n   *\n   * @param {String} key\n   * @return {Node|Null}\n   */\n\n  getNode(key) {\n    return this.key == key\n      ? this\n      : null\n  }\n\n  /**\n   * Check if the node has a node by `key`, to parallel other nodes.\n   *\n   * @param {String} key\n   * @return {Boolean}\n   */\n\n  hasNode(key) {\n    return !!this.getNode(key)\n  }\n\n  /**\n   * Insert `text` at `index`.\n   *\n   * @param {Numbder} index\n   * @param {String} text\n   * @param {String} marks (optional)\n   * @return {Text}\n   */\n\n  insertText(index, text, marks) {\n    let { characters } = this\n    const chars = Character.createList(text.split('').map(char => ({ text: char, marks })))\n\n    characters = characters.slice(0, index)\n      .concat(chars)\n      .concat(characters.slice(index))\n\n    return this.set('characters', characters)\n  }\n\n  /**\n   * Regenerate the node's key.\n   *\n   * @return {Text}\n   */\n\n  regenerateKey() {\n    const key = generateKey()\n    return this.set('key', key)\n  }\n\n  /**\n   * Remove a `mark` at `index` and `length`.\n   *\n   * @param {Number} index\n   * @param {Number} length\n   * @param {Mark} mark\n   * @return {Text}\n   */\n\n  removeMark(index, length, mark) {\n    const characters = this.characters.map((char, i) => {\n      if (i < index) return char\n      if (i >= index + length) return char\n      let { marks } = char\n      marks = marks.remove(mark)\n      char = char.set('marks', marks)\n      return char\n    })\n\n    return this.set('characters', characters)\n  }\n\n  /**\n   * Remove text from the text node at `index` for `length`.\n   *\n   * @param {Number} index\n   * @param {Number} length\n   * @return {Text}\n   */\n\n  removeText(index, length) {\n    let { characters } = this\n    const start = index\n    const end = index + length\n    characters = characters.filterNot((char, i) => start <= i && i < end)\n    return this.set('characters', characters)\n  }\n\n  /**\n   * Return a JSON representation of the text.\n   *\n   * @param {Object} options\n   * @return {Object}\n   */\n\n  toJSON(options = {}) {\n    const object = {\n      kind: this.kind,\n      leaves: this.getLeaves().toArray().map(r => r.toJSON()),\n    }\n\n    if (options.preserveKeys) {\n      object.key = this.key\n    }\n\n    return object\n  }\n\n  /**\n   * Alias `toJS`.\n   */\n\n  toJS(options) {\n    return this.toJSON(options)\n  }\n\n  /**\n   * Update a `mark` at `index` and `length` with `properties`.\n   *\n   * @param {Number} index\n   * @param {Number} length\n   * @param {Mark} mark\n   * @param {Object} properties\n   * @return {Text}\n   */\n\n  updateMark(index, length, mark, properties) {\n    const newMark = mark.merge(properties)\n\n    const characters = this.characters.map((char, i) => {\n      if (i < index) return char\n      if (i >= index + length) return char\n      let { marks } = char\n      if (!marks.has(mark)) return char\n      marks = marks.remove(mark)\n      marks = marks.add(newMark)\n      char = char.set('marks', marks)\n      return char\n    })\n\n    return this.set('characters', characters)\n  }\n\n  /**\n   * Validate the text node against a `schema`.\n   *\n   * @param {Schema} schema\n   * @return {Object|Void}\n   */\n\n  validate(schema) {\n    return schema.validateNode(this)\n  }\n\n}\n\n/**\n * Attach a pseudo-symbol for type checking.\n */\n\nText.prototype[MODEL_TYPES.TEXT] = true\n\n/**\n * Memoize read methods.\n */\n\nmemoize(Text.prototype, [\n  'getMarks',\n  'getMarksAsArray',\n], {\n  takesArguments: false,\n})\n\nmemoize(Text.prototype, [\n  'getDecoratedCharacters',\n  'getDecorations',\n  'getLeaves',\n  'getMarksAtIndex',\n  'validate'\n], {\n  takesArguments: true,\n})\n\n/**\n * Export.\n *\n * @type {Text}\n */\n\nexport default Text\n"]}