(function (global, factory) { typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('@tiptap/core'), require('prosemirror-state')) : typeof define === 'function' && define.amd ? define(['exports', '@tiptap/core', 'prosemirror-state'], factory) : (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global["@tiptap/extension-character-count"] = {}, global.core, global.prosemirrorState)); })(this, (function (exports, core, prosemirrorState) { 'use strict'; const CharacterCount = core.Extension.create({ name: 'characterCount', addOptions() { return { limit: null, mode: 'textSize', }; }, addStorage() { return { characters: () => 0, words: () => 0, }; }, onBeforeCreate() { this.storage.characters = options => { const node = (options === null || options === void 0 ? void 0 : options.node) || this.editor.state.doc; const mode = (options === null || options === void 0 ? void 0 : options.mode) || this.options.mode; if (mode === 'textSize') { const text = node.textBetween(0, node.content.size, undefined, ' '); return text.length; } return node.nodeSize; }; this.storage.words = options => { const node = (options === null || options === void 0 ? void 0 : options.node) || this.editor.state.doc; const text = node.textBetween(0, node.content.size, ' ', ' '); const words = text .split(' ') .filter(word => word !== ''); return words.length; }; }, addProseMirrorPlugins() { return [ new prosemirrorState.Plugin({ key: new prosemirrorState.PluginKey('characterCount'), filterTransaction: (transaction, state) => { const limit = this.options.limit; // Nothing has changed or no limit is defined. Ignore it. if (!transaction.docChanged || limit === 0 || limit === null || limit === undefined) { return true; } const oldSize = this.storage.characters({ node: state.doc }); const newSize = this.storage.characters({ node: transaction.doc }); // Everything is in the limit. Good. if (newSize <= limit) { return true; } // The limit has already been exceeded but will be reduced. if (oldSize > limit && newSize > limit && newSize <= oldSize) { return true; } // The limit has already been exceeded and will be increased further. if (oldSize > limit && newSize > limit && newSize > oldSize) { return false; } const isPaste = transaction.getMeta('paste'); // Block all exceeding transactions that were not pasted. if (!isPaste) { return false; } // For pasted content, we try to remove the exceeding content. const pos = transaction.selection.$head.pos; const over = newSize - limit; const from = pos - over; const to = pos; // It’s probably a bad idea to mutate transactions within `filterTransaction` // but for now this is working fine. transaction.deleteRange(from, to); // In some situations, the limit will continue to be exceeded after trimming. // This happens e.g. when truncating within a complex node (e.g. table) // and ProseMirror has to close this node again. // If this is the case, we prevent the transaction completely. const updatedSize = this.storage.characters({ node: transaction.doc }); if (updatedSize > limit) { return false; } return true; }, }), ]; }, }); exports.CharacterCount = CharacterCount; exports["default"] = CharacterCount; Object.defineProperty(exports, '__esModule', { value: true }); })); //# sourceMappingURL=tiptap-extension-character-count.umd.js.map