123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571 |
- /**
- * Copyright (c) Tiny Technologies, Inc. All rights reserved.
- * Licensed under the LGPL or a commercial license.
- * For LGPL see License.txt in the project root for license information.
- * For commercial licenses see https://www.tiny.cloud/
- *
- * Version: 5.2.0 (2020-02-13)
- */
- (function (domGlobals) {
- 'use strict';
- var Cell = function (initial) {
- var value = initial;
- var get = function () {
- return value;
- };
- var set = function (v) {
- value = v;
- };
- var clone = function () {
- return Cell(get());
- };
- return {
- get: get,
- set: set,
- clone: clone
- };
- };
- var global = tinymce.util.Tools.resolve('tinymce.PluginManager');
- var get = function (toggleState) {
- var isEnabled = function () {
- return toggleState.get();
- };
- return { isEnabled: isEnabled };
- };
- var Api = { get: get };
- var fireVisualChars = function (editor, state) {
- return editor.fire('VisualChars', { state: state });
- };
- var Events = { fireVisualChars: fireVisualChars };
- var noop = function () {
- };
- var constant = function (value) {
- return function () {
- return value;
- };
- };
- var never = constant(false);
- var always = constant(true);
- var none = function () {
- return NONE;
- };
- var NONE = function () {
- var eq = function (o) {
- return o.isNone();
- };
- var call = function (thunk) {
- return thunk();
- };
- var id = function (n) {
- return n;
- };
- var me = {
- fold: function (n, s) {
- return n();
- },
- is: never,
- isSome: never,
- isNone: always,
- getOr: id,
- getOrThunk: call,
- getOrDie: function (msg) {
- throw new Error(msg || 'error: getOrDie called on none.');
- },
- getOrNull: constant(null),
- getOrUndefined: constant(undefined),
- or: id,
- orThunk: call,
- map: none,
- each: noop,
- bind: none,
- exists: never,
- forall: always,
- filter: none,
- equals: eq,
- equals_: eq,
- toArray: function () {
- return [];
- },
- toString: constant('none()')
- };
- if (Object.freeze) {
- Object.freeze(me);
- }
- return me;
- }();
- var some = function (a) {
- var constant_a = constant(a);
- var self = function () {
- return me;
- };
- var bind = function (f) {
- return f(a);
- };
- var me = {
- fold: function (n, s) {
- return s(a);
- },
- is: function (v) {
- return a === v;
- },
- isSome: always,
- isNone: never,
- getOr: constant_a,
- getOrThunk: constant_a,
- getOrDie: constant_a,
- getOrNull: constant_a,
- getOrUndefined: constant_a,
- or: self,
- orThunk: self,
- map: function (f) {
- return some(f(a));
- },
- each: function (f) {
- f(a);
- },
- bind: bind,
- exists: bind,
- forall: bind,
- filter: function (f) {
- return f(a) ? me : NONE;
- },
- toArray: function () {
- return [a];
- },
- toString: function () {
- return 'some(' + a + ')';
- },
- equals: function (o) {
- return o.is(a);
- },
- equals_: function (o, elementEq) {
- return o.fold(never, function (b) {
- return elementEq(a, b);
- });
- }
- };
- return me;
- };
- var from = function (value) {
- return value === null || value === undefined ? NONE : some(value);
- };
- var Option = {
- some: some,
- none: none,
- from: from
- };
- var typeOf = function (x) {
- if (x === null) {
- return 'null';
- }
- var t = typeof x;
- if (t === 'object' && (Array.prototype.isPrototypeOf(x) || x.constructor && x.constructor.name === 'Array')) {
- return 'array';
- }
- if (t === 'object' && (String.prototype.isPrototypeOf(x) || x.constructor && x.constructor.name === 'String')) {
- return 'string';
- }
- return t;
- };
- var isType = function (type) {
- return function (value) {
- return typeOf(value) === type;
- };
- };
- var isString = isType('string');
- var isBoolean = isType('boolean');
- var isFunction = isType('function');
- var isNumber = isType('number');
- var nativeSlice = Array.prototype.slice;
- var map = function (xs, f) {
- var len = xs.length;
- var r = new Array(len);
- for (var i = 0; i < len; i++) {
- var x = xs[i];
- r[i] = f(x, i);
- }
- return r;
- };
- var each = function (xs, f) {
- for (var i = 0, len = xs.length; i < len; i++) {
- var x = xs[i];
- f(x, i);
- }
- };
- var filter = function (xs, pred) {
- var r = [];
- for (var i = 0, len = xs.length; i < len; i++) {
- var x = xs[i];
- if (pred(x, i)) {
- r.push(x);
- }
- }
- return r;
- };
- var from$1 = isFunction(Array.from) ? Array.from : function (x) {
- return nativeSlice.call(x);
- };
- var ATTRIBUTE = domGlobals.Node.ATTRIBUTE_NODE;
- var CDATA_SECTION = domGlobals.Node.CDATA_SECTION_NODE;
- var COMMENT = domGlobals.Node.COMMENT_NODE;
- var DOCUMENT = domGlobals.Node.DOCUMENT_NODE;
- var DOCUMENT_TYPE = domGlobals.Node.DOCUMENT_TYPE_NODE;
- var DOCUMENT_FRAGMENT = domGlobals.Node.DOCUMENT_FRAGMENT_NODE;
- var ELEMENT = domGlobals.Node.ELEMENT_NODE;
- var TEXT = domGlobals.Node.TEXT_NODE;
- var PROCESSING_INSTRUCTION = domGlobals.Node.PROCESSING_INSTRUCTION_NODE;
- var ENTITY_REFERENCE = domGlobals.Node.ENTITY_REFERENCE_NODE;
- var ENTITY = domGlobals.Node.ENTITY_NODE;
- var NOTATION = domGlobals.Node.NOTATION_NODE;
- var Global = typeof domGlobals.window !== 'undefined' ? domGlobals.window : Function('return this;')();
- var type = function (element) {
- return element.dom().nodeType;
- };
- var value = function (element) {
- return element.dom().nodeValue;
- };
- var isType$1 = function (t) {
- return function (element) {
- return type(element) === t;
- };
- };
- var isText = isType$1(TEXT);
- var rawSet = function (dom, key, value) {
- if (isString(value) || isBoolean(value) || isNumber(value)) {
- dom.setAttribute(key, value + '');
- } else {
- domGlobals.console.error('Invalid call to Attr.set. Key ', key, ':: Value ', value, ':: Element ', dom);
- throw new Error('Attribute value was not simple');
- }
- };
- var set = function (element, key, value) {
- rawSet(element.dom(), key, value);
- };
- var get$1 = function (element, key) {
- var v = element.dom().getAttribute(key);
- return v === null ? undefined : v;
- };
- var remove = function (element, key) {
- element.dom().removeAttribute(key);
- };
- var read = function (element, attr) {
- var value = get$1(element, attr);
- return value === undefined || value === '' ? [] : value.split(' ');
- };
- var add = function (element, attr, id) {
- var old = read(element, attr);
- var nu = old.concat([id]);
- set(element, attr, nu.join(' '));
- return true;
- };
- var remove$1 = function (element, attr, id) {
- var nu = filter(read(element, attr), function (v) {
- return v !== id;
- });
- if (nu.length > 0) {
- set(element, attr, nu.join(' '));
- } else {
- remove(element, attr);
- }
- return false;
- };
- var supports = function (element) {
- return element.dom().classList !== undefined;
- };
- var get$2 = function (element) {
- return read(element, 'class');
- };
- var add$1 = function (element, clazz) {
- return add(element, 'class', clazz);
- };
- var remove$2 = function (element, clazz) {
- return remove$1(element, 'class', clazz);
- };
- var add$2 = function (element, clazz) {
- if (supports(element)) {
- element.dom().classList.add(clazz);
- } else {
- add$1(element, clazz);
- }
- };
- var cleanClass = function (element) {
- var classList = supports(element) ? element.dom().classList : get$2(element);
- if (classList.length === 0) {
- remove(element, 'class');
- }
- };
- var remove$3 = function (element, clazz) {
- if (supports(element)) {
- var classList = element.dom().classList;
- classList.remove(clazz);
- } else {
- remove$2(element, clazz);
- }
- cleanClass(element);
- };
- var fromHtml = function (html, scope) {
- var doc = scope || domGlobals.document;
- var div = doc.createElement('div');
- div.innerHTML = html;
- if (!div.hasChildNodes() || div.childNodes.length > 1) {
- domGlobals.console.error('HTML does not have a single root node', html);
- throw new Error('HTML must have a single root node');
- }
- return fromDom(div.childNodes[0]);
- };
- var fromTag = function (tag, scope) {
- var doc = scope || domGlobals.document;
- var node = doc.createElement(tag);
- return fromDom(node);
- };
- var fromText = function (text, scope) {
- var doc = scope || domGlobals.document;
- var node = doc.createTextNode(text);
- return fromDom(node);
- };
- var fromDom = function (node) {
- if (node === null || node === undefined) {
- throw new Error('Node cannot be null or undefined');
- }
- return { dom: constant(node) };
- };
- var fromPoint = function (docElm, x, y) {
- var doc = docElm.dom();
- return Option.from(doc.elementFromPoint(x, y)).map(fromDom);
- };
- var Element = {
- fromHtml: fromHtml,
- fromTag: fromTag,
- fromText: fromText,
- fromDom: fromDom,
- fromPoint: fromPoint
- };
- var charMap = {
- '\xA0': 'nbsp',
- '\xAD': 'shy'
- };
- var charMapToRegExp = function (charMap, global) {
- var key, regExp = '';
- for (key in charMap) {
- regExp += key;
- }
- return new RegExp('[' + regExp + ']', global ? 'g' : '');
- };
- var charMapToSelector = function (charMap) {
- var key, selector = '';
- for (key in charMap) {
- if (selector) {
- selector += ',';
- }
- selector += 'span.mce-' + charMap[key];
- }
- return selector;
- };
- var Data = {
- charMap: charMap,
- regExp: charMapToRegExp(charMap),
- regExpGlobal: charMapToRegExp(charMap, true),
- selector: charMapToSelector(charMap),
- nbspClass: 'mce-nbsp',
- charMapToRegExp: charMapToRegExp,
- charMapToSelector: charMapToSelector
- };
- var wrapCharWithSpan = function (value) {
- return '<span data-mce-bogus="1" class="mce-' + Data.charMap[value] + '">' + value + '</span>';
- };
- var Html = { wrapCharWithSpan: wrapCharWithSpan };
- var isMatch = function (n) {
- var value$1 = value(n);
- return isText(n) && value$1 !== undefined && Data.regExp.test(value$1);
- };
- var filterDescendants = function (scope, predicate) {
- var result = [];
- var dom = scope.dom();
- var children = map(dom.childNodes, Element.fromDom);
- each(children, function (x) {
- if (predicate(x)) {
- result = result.concat([x]);
- }
- result = result.concat(filterDescendants(x, predicate));
- });
- return result;
- };
- var findParentElm = function (elm, rootElm) {
- while (elm.parentNode) {
- if (elm.parentNode === rootElm) {
- return elm;
- }
- elm = elm.parentNode;
- }
- };
- var replaceWithSpans = function (text) {
- return text.replace(Data.regExpGlobal, Html.wrapCharWithSpan);
- };
- var Nodes = {
- isMatch: isMatch,
- filterDescendants: filterDescendants,
- findParentElm: findParentElm,
- replaceWithSpans: replaceWithSpans
- };
- var isWrappedNbsp = function (node) {
- return node.nodeName.toLowerCase() === 'span' && node.classList.contains('mce-nbsp-wrap');
- };
- var show = function (editor, rootElm) {
- var nodeList = Nodes.filterDescendants(Element.fromDom(rootElm), Nodes.isMatch);
- each(nodeList, function (n) {
- var parent = n.dom().parentNode;
- if (isWrappedNbsp(parent)) {
- add$2(Element.fromDom(parent), Data.nbspClass);
- } else {
- var withSpans = Nodes.replaceWithSpans(editor.dom.encode(value(n)));
- var div = editor.dom.create('div', null, withSpans);
- var node = void 0;
- while (node = div.lastChild) {
- editor.dom.insertAfter(node, n.dom());
- }
- editor.dom.remove(n.dom());
- }
- });
- };
- var hide = function (editor, rootElm) {
- var nodeList = editor.dom.select(Data.selector, rootElm);
- each(nodeList, function (node) {
- if (isWrappedNbsp(node)) {
- remove$3(Element.fromDom(node), Data.nbspClass);
- } else {
- editor.dom.remove(node, true);
- }
- });
- };
- var toggle = function (editor) {
- var body = editor.getBody();
- var bookmark = editor.selection.getBookmark();
- var parentNode = Nodes.findParentElm(editor.selection.getNode(), body);
- parentNode = parentNode !== undefined ? parentNode : body;
- hide(editor, parentNode);
- show(editor, parentNode);
- editor.selection.moveToBookmark(bookmark);
- };
- var VisualChars = {
- show: show,
- hide: hide,
- toggle: toggle
- };
- var toggleVisualChars = function (editor, toggleState) {
- var body = editor.getBody();
- var selection = editor.selection;
- var bookmark;
- toggleState.set(!toggleState.get());
- Events.fireVisualChars(editor, toggleState.get());
- bookmark = selection.getBookmark();
- if (toggleState.get() === true) {
- VisualChars.show(editor, body);
- } else {
- VisualChars.hide(editor, body);
- }
- selection.moveToBookmark(bookmark);
- };
- var Actions = { toggleVisualChars: toggleVisualChars };
- var register = function (editor, toggleState) {
- editor.addCommand('mceVisualChars', function () {
- Actions.toggleVisualChars(editor, toggleState);
- });
- };
- var Commands = { register: register };
- var global$1 = tinymce.util.Tools.resolve('tinymce.util.Delay');
- var setup = function (editor, toggleState) {
- var debouncedToggle = global$1.debounce(function () {
- VisualChars.toggle(editor);
- }, 300);
- if (editor.settings.forced_root_block !== false) {
- editor.on('keydown', function (e) {
- if (toggleState.get() === true) {
- e.keyCode === 13 ? VisualChars.toggle(editor) : debouncedToggle();
- }
- });
- }
- };
- var Keyboard = { setup: setup };
- var isEnabledByDefault = function (editor) {
- return editor.getParam('visualchars_default_state', false);
- };
- var Settings = { isEnabledByDefault: isEnabledByDefault };
- var setup$1 = function (editor, toggleState) {
- editor.on('init', function () {
- var valueForToggling = !Settings.isEnabledByDefault(editor);
- toggleState.set(valueForToggling);
- Actions.toggleVisualChars(editor, toggleState);
- });
- };
- var Bindings = { setup: setup$1 };
- var toggleActiveState = function (editor, enabledStated) {
- return function (api) {
- api.setActive(enabledStated.get());
- var editorEventCallback = function (e) {
- return api.setActive(e.state);
- };
- editor.on('VisualChars', editorEventCallback);
- return function () {
- return editor.off('VisualChars', editorEventCallback);
- };
- };
- };
- var register$1 = function (editor, toggleState) {
- editor.ui.registry.addToggleButton('visualchars', {
- tooltip: 'Show invisible characters',
- icon: 'visualchars',
- onAction: function () {
- return editor.execCommand('mceVisualChars');
- },
- onSetup: toggleActiveState(editor, toggleState)
- });
- editor.ui.registry.addToggleMenuItem('visualchars', {
- text: 'Show invisible characters',
- onAction: function () {
- return editor.execCommand('mceVisualChars');
- },
- onSetup: toggleActiveState(editor, toggleState)
- });
- };
- function Plugin () {
- global.add('visualchars', function (editor) {
- var toggleState = Cell(false);
- Commands.register(editor, toggleState);
- register$1(editor, toggleState);
- Keyboard.setup(editor, toggleState);
- Bindings.setup(editor, toggleState);
- return Api.get(toggleState);
- });
- }
- Plugin();
- }(window));
|