plugin.js 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281
  1. /**
  2. * Copyright (c) Tiny Technologies, Inc. All rights reserved.
  3. * Licensed under the LGPL or a commercial license.
  4. * For LGPL see License.txt in the project root for license information.
  5. * For commercial licenses see https://www.tiny.cloud/
  6. *
  7. * Version: 5.5.1 (2020-10-01)
  8. */
  9. (function () {
  10. 'use strict';
  11. var global = tinymce.util.Tools.resolve('tinymce.PluginManager');
  12. var global$1 = tinymce.util.Tools.resolve('tinymce.util.VK');
  13. var typeOf = function (x) {
  14. var t = typeof x;
  15. if (x === null) {
  16. return 'null';
  17. } else if (t === 'object' && (Array.prototype.isPrototypeOf(x) || x.constructor && x.constructor.name === 'Array')) {
  18. return 'array';
  19. } else if (t === 'object' && (String.prototype.isPrototypeOf(x) || x.constructor && x.constructor.name === 'String')) {
  20. return 'string';
  21. } else {
  22. return t;
  23. }
  24. };
  25. var isType = function (type) {
  26. return function (value) {
  27. return typeOf(value) === type;
  28. };
  29. };
  30. var isSimpleType = function (type) {
  31. return function (value) {
  32. return typeof value === type;
  33. };
  34. };
  35. var eq = function (t) {
  36. return function (a) {
  37. return t === a;
  38. };
  39. };
  40. var isString = isType('string');
  41. var isArray = isType('array');
  42. var isNull = eq(null);
  43. var isBoolean = isSimpleType('boolean');
  44. var isFunction = isSimpleType('function');
  45. var assumeExternalTargets = function (editor) {
  46. var externalTargets = editor.getParam('link_assume_external_targets', false);
  47. if (isBoolean(externalTargets) && externalTargets) {
  48. return 1;
  49. } else if (isString(externalTargets) && (externalTargets === 'http' || externalTargets === 'https')) {
  50. return externalTargets;
  51. }
  52. return 0;
  53. };
  54. var hasContextToolbar = function (editor) {
  55. return editor.getParam('link_context_toolbar', false, 'boolean');
  56. };
  57. var getLinkList = function (editor) {
  58. return editor.getParam('link_list');
  59. };
  60. var getDefaultLinkTarget = function (editor) {
  61. return editor.getParam('default_link_target');
  62. };
  63. var getTargetList = function (editor) {
  64. return editor.getParam('target_list', true);
  65. };
  66. var getRelList = function (editor) {
  67. return editor.getParam('rel_list', [], 'array');
  68. };
  69. var getLinkClassList = function (editor) {
  70. return editor.getParam('link_class_list', [], 'array');
  71. };
  72. var shouldShowLinkTitle = function (editor) {
  73. return editor.getParam('link_title', true, 'boolean');
  74. };
  75. var allowUnsafeLinkTarget = function (editor) {
  76. return editor.getParam('allow_unsafe_link_target', false, 'boolean');
  77. };
  78. var useQuickLink = function (editor) {
  79. return editor.getParam('link_quicklink', false, 'boolean');
  80. };
  81. var getDefaultLinkProtocol = function (editor) {
  82. return editor.getParam('link_default_protocol', 'http', 'string');
  83. };
  84. var noop = function () {
  85. };
  86. var constant = function (value) {
  87. return function () {
  88. return value;
  89. };
  90. };
  91. var never = constant(false);
  92. var always = constant(true);
  93. var none = function () {
  94. return NONE;
  95. };
  96. var NONE = function () {
  97. var eq = function (o) {
  98. return o.isNone();
  99. };
  100. var call = function (thunk) {
  101. return thunk();
  102. };
  103. var id = function (n) {
  104. return n;
  105. };
  106. var me = {
  107. fold: function (n, _s) {
  108. return n();
  109. },
  110. is: never,
  111. isSome: never,
  112. isNone: always,
  113. getOr: id,
  114. getOrThunk: call,
  115. getOrDie: function (msg) {
  116. throw new Error(msg || 'error: getOrDie called on none.');
  117. },
  118. getOrNull: constant(null),
  119. getOrUndefined: constant(undefined),
  120. or: id,
  121. orThunk: call,
  122. map: none,
  123. each: noop,
  124. bind: none,
  125. exists: never,
  126. forall: always,
  127. filter: none,
  128. equals: eq,
  129. equals_: eq,
  130. toArray: function () {
  131. return [];
  132. },
  133. toString: constant('none()')
  134. };
  135. return me;
  136. }();
  137. var some = function (a) {
  138. var constant_a = constant(a);
  139. var self = function () {
  140. return me;
  141. };
  142. var bind = function (f) {
  143. return f(a);
  144. };
  145. var me = {
  146. fold: function (n, s) {
  147. return s(a);
  148. },
  149. is: function (v) {
  150. return a === v;
  151. },
  152. isSome: always,
  153. isNone: never,
  154. getOr: constant_a,
  155. getOrThunk: constant_a,
  156. getOrDie: constant_a,
  157. getOrNull: constant_a,
  158. getOrUndefined: constant_a,
  159. or: self,
  160. orThunk: self,
  161. map: function (f) {
  162. return some(f(a));
  163. },
  164. each: function (f) {
  165. f(a);
  166. },
  167. bind: bind,
  168. exists: bind,
  169. forall: bind,
  170. filter: function (f) {
  171. return f(a) ? me : NONE;
  172. },
  173. toArray: function () {
  174. return [a];
  175. },
  176. toString: function () {
  177. return 'some(' + a + ')';
  178. },
  179. equals: function (o) {
  180. return o.is(a);
  181. },
  182. equals_: function (o, elementEq) {
  183. return o.fold(never, function (b) {
  184. return elementEq(a, b);
  185. });
  186. }
  187. };
  188. return me;
  189. };
  190. var from = function (value) {
  191. return value === null || value === undefined ? NONE : some(value);
  192. };
  193. var Optional = {
  194. some: some,
  195. none: none,
  196. from: from
  197. };
  198. var nativeIndexOf = Array.prototype.indexOf;
  199. var nativePush = Array.prototype.push;
  200. var rawIndexOf = function (ts, t) {
  201. return nativeIndexOf.call(ts, t);
  202. };
  203. var contains = function (xs, x) {
  204. return rawIndexOf(xs, x) > -1;
  205. };
  206. var map = function (xs, f) {
  207. var len = xs.length;
  208. var r = new Array(len);
  209. for (var i = 0; i < len; i++) {
  210. var x = xs[i];
  211. r[i] = f(x, i);
  212. }
  213. return r;
  214. };
  215. var each = function (xs, f) {
  216. for (var i = 0, len = xs.length; i < len; i++) {
  217. var x = xs[i];
  218. f(x, i);
  219. }
  220. };
  221. var foldl = function (xs, f, acc) {
  222. each(xs, function (x) {
  223. acc = f(acc, x);
  224. });
  225. return acc;
  226. };
  227. var flatten = function (xs) {
  228. var r = [];
  229. for (var i = 0, len = xs.length; i < len; ++i) {
  230. if (!isArray(xs[i])) {
  231. throw new Error('Arr.flatten item ' + i + ' was not an array, input: ' + xs);
  232. }
  233. nativePush.apply(r, xs[i]);
  234. }
  235. return r;
  236. };
  237. var bind = function (xs, f) {
  238. return flatten(map(xs, f));
  239. };
  240. var findMap = function (arr, f) {
  241. for (var i = 0; i < arr.length; i++) {
  242. var r = f(arr[i], i);
  243. if (r.isSome()) {
  244. return r;
  245. }
  246. }
  247. return Optional.none();
  248. };
  249. var cat = function (arr) {
  250. var r = [];
  251. var push = function (x) {
  252. r.push(x);
  253. };
  254. for (var i = 0; i < arr.length; i++) {
  255. arr[i].each(push);
  256. }
  257. return r;
  258. };
  259. var someIf = function (b, a) {
  260. return b ? Optional.some(a) : Optional.none();
  261. };
  262. var global$2 = tinymce.util.Tools.resolve('tinymce.util.Tools');
  263. var getValue = function (item) {
  264. return isString(item.value) ? item.value : '';
  265. };
  266. var sanitizeList = function (list, extractValue) {
  267. var out = [];
  268. global$2.each(list, function (item) {
  269. var text = isString(item.text) ? item.text : isString(item.title) ? item.title : '';
  270. if (item.menu !== undefined) {
  271. var items = sanitizeList(item.menu, extractValue);
  272. out.push({
  273. text: text,
  274. items: items
  275. });
  276. } else {
  277. var value = extractValue(item);
  278. out.push({
  279. text: text,
  280. value: value
  281. });
  282. }
  283. });
  284. return out;
  285. };
  286. var sanitizeWith = function (extracter) {
  287. if (extracter === void 0) {
  288. extracter = getValue;
  289. }
  290. return function (list) {
  291. return Optional.from(list).map(function (list) {
  292. return sanitizeList(list, extracter);
  293. });
  294. };
  295. };
  296. var sanitize = function (list) {
  297. return sanitizeWith(getValue)(list);
  298. };
  299. var createUi = function (name, label) {
  300. return function (items) {
  301. return {
  302. name: name,
  303. type: 'listbox',
  304. label: label,
  305. items: items
  306. };
  307. };
  308. };
  309. var ListOptions = {
  310. sanitize: sanitize,
  311. sanitizeWith: sanitizeWith,
  312. createUi: createUi,
  313. getValue: getValue
  314. };
  315. var __assign = function () {
  316. __assign = Object.assign || function __assign(t) {
  317. for (var s, i = 1, n = arguments.length; i < n; i++) {
  318. s = arguments[i];
  319. for (var p in s)
  320. if (Object.prototype.hasOwnProperty.call(s, p))
  321. t[p] = s[p];
  322. }
  323. return t;
  324. };
  325. return __assign.apply(this, arguments);
  326. };
  327. var keys = Object.keys;
  328. var hasOwnProperty = Object.hasOwnProperty;
  329. var each$1 = function (obj, f) {
  330. var props = keys(obj);
  331. for (var k = 0, len = props.length; k < len; k++) {
  332. var i = props[k];
  333. var x = obj[i];
  334. f(x, i);
  335. }
  336. };
  337. var objAcc = function (r) {
  338. return function (x, i) {
  339. r[i] = x;
  340. };
  341. };
  342. var internalFilter = function (obj, pred, onTrue, onFalse) {
  343. var r = {};
  344. each$1(obj, function (x, i) {
  345. (pred(x, i) ? onTrue : onFalse)(x, i);
  346. });
  347. return r;
  348. };
  349. var filter = function (obj, pred) {
  350. var t = {};
  351. internalFilter(obj, pred, objAcc(t), noop);
  352. return t;
  353. };
  354. var has = function (obj, key) {
  355. return hasOwnProperty.call(obj, key);
  356. };
  357. var hasNonNullableKey = function (obj, key) {
  358. return has(obj, key) && obj[key] !== undefined && obj[key] !== null;
  359. };
  360. var global$3 = tinymce.util.Tools.resolve('tinymce.dom.TreeWalker');
  361. var isAnchor = function (elm) {
  362. return elm && elm.nodeName.toLowerCase() === 'a';
  363. };
  364. var isLink = function (elm) {
  365. return isAnchor(elm) && !!getHref(elm);
  366. };
  367. var collectNodesInRange = function (rng, predicate) {
  368. if (rng.collapsed) {
  369. return [];
  370. } else {
  371. var contents = rng.cloneContents();
  372. var walker = new global$3(contents.firstChild, contents);
  373. var elements = [];
  374. var current = contents.firstChild;
  375. do {
  376. if (predicate(current)) {
  377. elements.push(current);
  378. }
  379. } while (current = walker.next());
  380. return elements;
  381. }
  382. };
  383. var hasProtocol = function (url) {
  384. return /^\w+:/i.test(url);
  385. };
  386. var getHref = function (elm) {
  387. var href = elm.getAttribute('data-mce-href');
  388. return href ? href : elm.getAttribute('href');
  389. };
  390. var applyRelTargetRules = function (rel, isUnsafe) {
  391. var rules = ['noopener'];
  392. var rels = rel ? rel.split(/\s+/) : [];
  393. var toString = function (rels) {
  394. return global$2.trim(rels.sort().join(' '));
  395. };
  396. var addTargetRules = function (rels) {
  397. rels = removeTargetRules(rels);
  398. return rels.length > 0 ? rels.concat(rules) : rules;
  399. };
  400. var removeTargetRules = function (rels) {
  401. return rels.filter(function (val) {
  402. return global$2.inArray(rules, val) === -1;
  403. });
  404. };
  405. var newRels = isUnsafe ? addTargetRules(rels) : removeTargetRules(rels);
  406. return newRels.length > 0 ? toString(newRels) : '';
  407. };
  408. var trimCaretContainers = function (text) {
  409. return text.replace(/\uFEFF/g, '');
  410. };
  411. var getAnchorElement = function (editor, selectedElm) {
  412. selectedElm = selectedElm || editor.selection.getNode();
  413. if (isImageFigure(selectedElm)) {
  414. return editor.dom.select('a[href]', selectedElm)[0];
  415. } else {
  416. return editor.dom.getParent(selectedElm, 'a[href]');
  417. }
  418. };
  419. var getAnchorText = function (selection, anchorElm) {
  420. var text = anchorElm ? anchorElm.innerText || anchorElm.textContent : selection.getContent({ format: 'text' });
  421. return trimCaretContainers(text);
  422. };
  423. var hasLinks = function (elements) {
  424. return global$2.grep(elements, isLink).length > 0;
  425. };
  426. var hasLinksInSelection = function (rng) {
  427. return collectNodesInRange(rng, isLink).length > 0;
  428. };
  429. var isOnlyTextSelected = function (editor) {
  430. var inlineTextElements = editor.schema.getTextInlineElements();
  431. var isElement = function (elm) {
  432. return elm.nodeType === 1 && !isAnchor(elm) && !has(inlineTextElements, elm.nodeName.toLowerCase());
  433. };
  434. var elements = collectNodesInRange(editor.selection.getRng(), isElement);
  435. return elements.length === 0;
  436. };
  437. var isImageFigure = function (elm) {
  438. return elm && elm.nodeName === 'FIGURE' && /\bimage\b/i.test(elm.className);
  439. };
  440. var getLinkAttrs = function (data) {
  441. return foldl([
  442. 'title',
  443. 'rel',
  444. 'class',
  445. 'target'
  446. ], function (acc, key) {
  447. data[key].each(function (value) {
  448. acc[key] = value.length > 0 ? value : null;
  449. });
  450. return acc;
  451. }, { href: data.href });
  452. };
  453. var handleExternalTargets = function (href, assumeExternalTargets) {
  454. if ((assumeExternalTargets === 'http' || assumeExternalTargets === 'https') && !hasProtocol(href)) {
  455. return assumeExternalTargets + '://' + href;
  456. }
  457. return href;
  458. };
  459. var applyLinkOverrides = function (editor, linkAttrs) {
  460. var newLinkAttrs = __assign({}, linkAttrs);
  461. if (!(getRelList(editor).length > 0) && allowUnsafeLinkTarget(editor) === false) {
  462. var newRel = applyRelTargetRules(newLinkAttrs.rel, newLinkAttrs.target === '_blank');
  463. newLinkAttrs.rel = newRel ? newRel : null;
  464. }
  465. if (Optional.from(newLinkAttrs.target).isNone() && getTargetList(editor) === false) {
  466. newLinkAttrs.target = getDefaultLinkTarget(editor);
  467. }
  468. newLinkAttrs.href = handleExternalTargets(newLinkAttrs.href, assumeExternalTargets(editor));
  469. return newLinkAttrs;
  470. };
  471. var updateLink = function (editor, anchorElm, text, linkAttrs) {
  472. text.each(function (text) {
  473. if (anchorElm.hasOwnProperty('innerText')) {
  474. anchorElm.innerText = text;
  475. } else {
  476. anchorElm.textContent = text;
  477. }
  478. });
  479. editor.dom.setAttribs(anchorElm, linkAttrs);
  480. editor.selection.select(anchorElm);
  481. };
  482. var createLink = function (editor, selectedElm, text, linkAttrs) {
  483. if (isImageFigure(selectedElm)) {
  484. linkImageFigure(editor, selectedElm, linkAttrs);
  485. } else {
  486. text.fold(function () {
  487. editor.execCommand('mceInsertLink', false, linkAttrs);
  488. }, function (text) {
  489. editor.insertContent(editor.dom.createHTML('a', linkAttrs, editor.dom.encode(text)));
  490. });
  491. }
  492. };
  493. var linkDomMutation = function (editor, attachState, data) {
  494. var selectedElm = editor.selection.getNode();
  495. var anchorElm = getAnchorElement(editor, selectedElm);
  496. var linkAttrs = applyLinkOverrides(editor, getLinkAttrs(data));
  497. editor.undoManager.transact(function () {
  498. if (data.href === attachState.href) {
  499. attachState.attach();
  500. }
  501. if (anchorElm) {
  502. editor.focus();
  503. updateLink(editor, anchorElm, data.text, linkAttrs);
  504. } else {
  505. createLink(editor, selectedElm, data.text, linkAttrs);
  506. }
  507. });
  508. };
  509. var unlinkSelection = function (editor) {
  510. var dom = editor.dom, selection = editor.selection;
  511. var bookmark = selection.getBookmark();
  512. var rng = selection.getRng().cloneRange();
  513. var startAnchorElm = dom.getParent(rng.startContainer, 'a[href]', editor.getBody());
  514. var endAnchorElm = dom.getParent(rng.endContainer, 'a[href]', editor.getBody());
  515. if (startAnchorElm) {
  516. rng.setStartBefore(startAnchorElm);
  517. }
  518. if (endAnchorElm) {
  519. rng.setEndAfter(endAnchorElm);
  520. }
  521. selection.setRng(rng);
  522. editor.execCommand('unlink');
  523. selection.moveToBookmark(bookmark);
  524. };
  525. var unlinkDomMutation = function (editor) {
  526. editor.undoManager.transact(function () {
  527. var node = editor.selection.getNode();
  528. if (isImageFigure(node)) {
  529. unlinkImageFigure(editor, node);
  530. } else {
  531. unlinkSelection(editor);
  532. }
  533. editor.focus();
  534. });
  535. };
  536. var unwrapOptions = function (data) {
  537. var cls = data.class, href = data.href, rel = data.rel, target = data.target, text = data.text, title = data.title;
  538. return filter({
  539. class: cls.getOrNull(),
  540. href: href,
  541. rel: rel.getOrNull(),
  542. target: target.getOrNull(),
  543. text: text.getOrNull(),
  544. title: title.getOrNull()
  545. }, function (v, _k) {
  546. return isNull(v) === false;
  547. });
  548. };
  549. var link = function (editor, attachState, data) {
  550. editor.hasPlugin('rtc', true) ? editor.execCommand('createlink', false, unwrapOptions(data)) : linkDomMutation(editor, attachState, data);
  551. };
  552. var unlink = function (editor) {
  553. editor.hasPlugin('rtc', true) ? editor.execCommand('unlink') : unlinkDomMutation(editor);
  554. };
  555. var unlinkImageFigure = function (editor, fig) {
  556. var img = editor.dom.select('img', fig)[0];
  557. if (img) {
  558. var a = editor.dom.getParents(img, 'a[href]', fig)[0];
  559. if (a) {
  560. a.parentNode.insertBefore(img, a);
  561. editor.dom.remove(a);
  562. }
  563. }
  564. };
  565. var linkImageFigure = function (editor, fig, attrs) {
  566. var img = editor.dom.select('img', fig)[0];
  567. if (img) {
  568. var a = editor.dom.create('a', attrs);
  569. img.parentNode.insertBefore(a, img);
  570. a.appendChild(img);
  571. }
  572. };
  573. var isListGroup = function (item) {
  574. return hasNonNullableKey(item, 'items');
  575. };
  576. var findTextByValue = function (value, catalog) {
  577. return findMap(catalog, function (item) {
  578. if (isListGroup(item)) {
  579. return findTextByValue(value, item.items);
  580. } else {
  581. return someIf(item.value === value, item);
  582. }
  583. });
  584. };
  585. var getDelta = function (persistentText, fieldName, catalog, data) {
  586. var value = data[fieldName];
  587. var hasPersistentText = persistentText.length > 0;
  588. return value !== undefined ? findTextByValue(value, catalog).map(function (i) {
  589. return {
  590. url: {
  591. value: i.value,
  592. meta: {
  593. text: hasPersistentText ? persistentText : i.text,
  594. attach: noop
  595. }
  596. },
  597. text: hasPersistentText ? persistentText : i.text
  598. };
  599. }) : Optional.none();
  600. };
  601. var findCatalog = function (catalogs, fieldName) {
  602. if (fieldName === 'link') {
  603. return catalogs.link;
  604. } else if (fieldName === 'anchor') {
  605. return catalogs.anchor;
  606. } else {
  607. return Optional.none();
  608. }
  609. };
  610. var init = function (initialData, linkCatalog) {
  611. var persistentData = {
  612. text: initialData.text,
  613. title: initialData.title
  614. };
  615. var getTitleFromUrlChange = function (url) {
  616. return someIf(persistentData.title.length <= 0, Optional.from(url.meta.title).getOr(''));
  617. };
  618. var getTextFromUrlChange = function (url) {
  619. return someIf(persistentData.text.length <= 0, Optional.from(url.meta.text).getOr(url.value));
  620. };
  621. var onUrlChange = function (data) {
  622. var text = getTextFromUrlChange(data.url);
  623. var title = getTitleFromUrlChange(data.url);
  624. if (text.isSome() || title.isSome()) {
  625. return Optional.some(__assign(__assign({}, text.map(function (text) {
  626. return { text: text };
  627. }).getOr({})), title.map(function (title) {
  628. return { title: title };
  629. }).getOr({})));
  630. } else {
  631. return Optional.none();
  632. }
  633. };
  634. var onCatalogChange = function (data, change) {
  635. var catalog = findCatalog(linkCatalog, change.name).getOr([]);
  636. return getDelta(persistentData.text, change.name, catalog, data);
  637. };
  638. var onChange = function (getData, change) {
  639. var name = change.name;
  640. if (name === 'url') {
  641. return onUrlChange(getData());
  642. } else if (contains([
  643. 'anchor',
  644. 'link'
  645. ], name)) {
  646. return onCatalogChange(getData(), change);
  647. } else if (name === 'text' || name === 'title') {
  648. persistentData[name] = getData()[name];
  649. return Optional.none();
  650. } else {
  651. return Optional.none();
  652. }
  653. };
  654. return { onChange: onChange };
  655. };
  656. var DialogChanges = {
  657. init: init,
  658. getDelta: getDelta
  659. };
  660. var global$4 = tinymce.util.Tools.resolve('tinymce.util.Delay');
  661. var global$5 = tinymce.util.Tools.resolve('tinymce.util.Promise');
  662. var delayedConfirm = function (editor, message, callback) {
  663. var rng = editor.selection.getRng();
  664. global$4.setEditorTimeout(editor, function () {
  665. editor.windowManager.confirm(message, function (state) {
  666. editor.selection.setRng(rng);
  667. callback(state);
  668. });
  669. });
  670. };
  671. var tryEmailTransform = function (data) {
  672. var url = data.href;
  673. var suggestMailTo = url.indexOf('@') > 0 && url.indexOf('/') === -1 && url.indexOf('mailto:') === -1;
  674. return suggestMailTo ? Optional.some({
  675. message: 'The URL you entered seems to be an email address. Do you want to add the required mailto: prefix?',
  676. preprocess: function (oldData) {
  677. return __assign(__assign({}, oldData), { href: 'mailto:' + url });
  678. }
  679. }) : Optional.none();
  680. };
  681. var tryProtocolTransform = function (assumeExternalTargets, defaultLinkProtocol) {
  682. return function (data) {
  683. var url = data.href;
  684. var suggestProtocol = assumeExternalTargets === 1 && !hasProtocol(url) || assumeExternalTargets === 0 && /^\s*www(\.|\d\.)/i.test(url);
  685. return suggestProtocol ? Optional.some({
  686. message: 'The URL you entered seems to be an external link. Do you want to add the required ' + defaultLinkProtocol + ':// prefix?',
  687. preprocess: function (oldData) {
  688. return __assign(__assign({}, oldData), { href: defaultLinkProtocol + '://' + url });
  689. }
  690. }) : Optional.none();
  691. };
  692. };
  693. var preprocess = function (editor, data) {
  694. return findMap([
  695. tryEmailTransform,
  696. tryProtocolTransform(assumeExternalTargets(editor), getDefaultLinkProtocol(editor))
  697. ], function (f) {
  698. return f(data);
  699. }).fold(function () {
  700. return global$5.resolve(data);
  701. }, function (transform) {
  702. return new global$5(function (callback) {
  703. delayedConfirm(editor, transform.message, function (state) {
  704. callback(state ? transform.preprocess(data) : data);
  705. });
  706. });
  707. });
  708. };
  709. var DialogConfirms = { preprocess: preprocess };
  710. var getAnchors = function (editor) {
  711. var anchorNodes = editor.dom.select('a:not([href])');
  712. var anchors = bind(anchorNodes, function (anchor) {
  713. var id = anchor.name || anchor.id;
  714. return id ? [{
  715. text: id,
  716. value: '#' + id
  717. }] : [];
  718. });
  719. return anchors.length > 0 ? Optional.some([{
  720. text: 'None',
  721. value: ''
  722. }].concat(anchors)) : Optional.none();
  723. };
  724. var AnchorListOptions = { getAnchors: getAnchors };
  725. var getClasses = function (editor) {
  726. var list = getLinkClassList(editor);
  727. if (list.length > 0) {
  728. return ListOptions.sanitize(list);
  729. }
  730. return Optional.none();
  731. };
  732. var ClassListOptions = { getClasses: getClasses };
  733. var global$6 = tinymce.util.Tools.resolve('tinymce.util.XHR');
  734. var parseJson = function (text) {
  735. try {
  736. return Optional.some(JSON.parse(text));
  737. } catch (err) {
  738. return Optional.none();
  739. }
  740. };
  741. var getLinks = function (editor) {
  742. var extractor = function (item) {
  743. return editor.convertURL(item.value || item.url, 'href');
  744. };
  745. var linkList = getLinkList(editor);
  746. return new global$5(function (callback) {
  747. if (isString(linkList)) {
  748. global$6.send({
  749. url: linkList,
  750. success: function (text) {
  751. return callback(parseJson(text));
  752. },
  753. error: function (_) {
  754. return callback(Optional.none());
  755. }
  756. });
  757. } else if (isFunction(linkList)) {
  758. linkList(function (output) {
  759. return callback(Optional.some(output));
  760. });
  761. } else {
  762. callback(Optional.from(linkList));
  763. }
  764. }).then(function (optItems) {
  765. return optItems.bind(ListOptions.sanitizeWith(extractor)).map(function (items) {
  766. if (items.length > 0) {
  767. var noneItem = [{
  768. text: 'None',
  769. value: ''
  770. }];
  771. return noneItem.concat(items);
  772. } else {
  773. return items;
  774. }
  775. });
  776. });
  777. };
  778. var LinkListOptions = { getLinks: getLinks };
  779. var getRels = function (editor, initialTarget) {
  780. var list = getRelList(editor);
  781. if (list.length > 0) {
  782. var isTargetBlank_1 = initialTarget.is('_blank');
  783. var enforceSafe = allowUnsafeLinkTarget(editor) === false;
  784. var safeRelExtractor = function (item) {
  785. return applyRelTargetRules(ListOptions.getValue(item), isTargetBlank_1);
  786. };
  787. var sanitizer = enforceSafe ? ListOptions.sanitizeWith(safeRelExtractor) : ListOptions.sanitize;
  788. return sanitizer(list);
  789. }
  790. return Optional.none();
  791. };
  792. var RelOptions = { getRels: getRels };
  793. var fallbacks = [
  794. {
  795. text: 'Current window',
  796. value: ''
  797. },
  798. {
  799. text: 'New window',
  800. value: '_blank'
  801. }
  802. ];
  803. var getTargets = function (editor) {
  804. var list = getTargetList(editor);
  805. if (isArray(list)) {
  806. return ListOptions.sanitize(list).orThunk(function () {
  807. return Optional.some(fallbacks);
  808. });
  809. } else if (list === false) {
  810. return Optional.none();
  811. }
  812. return Optional.some(fallbacks);
  813. };
  814. var TargetOptions = { getTargets: getTargets };
  815. var nonEmptyAttr = function (dom, elem, name) {
  816. var val = dom.getAttrib(elem, name);
  817. return val !== null && val.length > 0 ? Optional.some(val) : Optional.none();
  818. };
  819. var extractFromAnchor = function (editor, anchor) {
  820. var dom = editor.dom;
  821. var onlyText = isOnlyTextSelected(editor);
  822. var text = onlyText ? Optional.some(getAnchorText(editor.selection, anchor)) : Optional.none();
  823. var url = anchor ? Optional.some(dom.getAttrib(anchor, 'href')) : Optional.none();
  824. var target = anchor ? Optional.from(dom.getAttrib(anchor, 'target')) : Optional.none();
  825. var rel = nonEmptyAttr(dom, anchor, 'rel');
  826. var linkClass = nonEmptyAttr(dom, anchor, 'class');
  827. var title = nonEmptyAttr(dom, anchor, 'title');
  828. return {
  829. url: url,
  830. text: text,
  831. title: title,
  832. target: target,
  833. rel: rel,
  834. linkClass: linkClass
  835. };
  836. };
  837. var collect = function (editor, linkNode) {
  838. return LinkListOptions.getLinks(editor).then(function (links) {
  839. var anchor = extractFromAnchor(editor, linkNode);
  840. return {
  841. anchor: anchor,
  842. catalogs: {
  843. targets: TargetOptions.getTargets(editor),
  844. rels: RelOptions.getRels(editor, anchor.target),
  845. classes: ClassListOptions.getClasses(editor),
  846. anchor: AnchorListOptions.getAnchors(editor),
  847. link: links
  848. },
  849. optNode: Optional.from(linkNode),
  850. flags: { titleEnabled: shouldShowLinkTitle(editor) }
  851. };
  852. });
  853. };
  854. var DialogInfo = { collect: collect };
  855. var handleSubmit = function (editor, info) {
  856. return function (api) {
  857. var data = api.getData();
  858. if (!data.url.value) {
  859. unlink(editor);
  860. api.close();
  861. return;
  862. }
  863. var getChangedValue = function (key) {
  864. return Optional.from(data[key]).filter(function (value) {
  865. return !info.anchor[key].is(value);
  866. });
  867. };
  868. var changedData = {
  869. href: data.url.value,
  870. text: getChangedValue('text'),
  871. target: getChangedValue('target'),
  872. rel: getChangedValue('rel'),
  873. class: getChangedValue('linkClass'),
  874. title: getChangedValue('title')
  875. };
  876. var attachState = {
  877. href: data.url.value,
  878. attach: data.url.meta !== undefined && data.url.meta.attach ? data.url.meta.attach : function () {
  879. }
  880. };
  881. DialogConfirms.preprocess(editor, changedData).then(function (pData) {
  882. link(editor, attachState, pData);
  883. });
  884. api.close();
  885. };
  886. };
  887. var collectData = function (editor) {
  888. var anchorNode = getAnchorElement(editor);
  889. return DialogInfo.collect(editor, anchorNode);
  890. };
  891. var getInitialData = function (info, defaultTarget) {
  892. var anchor = info.anchor;
  893. var url = anchor.url.getOr('');
  894. return {
  895. url: {
  896. value: url,
  897. meta: { original: { value: url } }
  898. },
  899. text: anchor.text.getOr(''),
  900. title: anchor.title.getOr(''),
  901. anchor: url,
  902. link: url,
  903. rel: anchor.rel.getOr(''),
  904. target: anchor.target.or(defaultTarget).getOr(''),
  905. linkClass: anchor.linkClass.getOr('')
  906. };
  907. };
  908. var makeDialog = function (settings, onSubmit, editor) {
  909. var urlInput = [{
  910. name: 'url',
  911. type: 'urlinput',
  912. filetype: 'file',
  913. label: 'URL'
  914. }];
  915. var displayText = settings.anchor.text.map(function () {
  916. return {
  917. name: 'text',
  918. type: 'input',
  919. label: 'Text to display'
  920. };
  921. }).toArray();
  922. var titleText = settings.flags.titleEnabled ? [{
  923. name: 'title',
  924. type: 'input',
  925. label: 'Title'
  926. }] : [];
  927. var defaultTarget = Optional.from(getDefaultLinkTarget(editor));
  928. var initialData = getInitialData(settings, defaultTarget);
  929. var catalogs = settings.catalogs;
  930. var dialogDelta = DialogChanges.init(initialData, catalogs);
  931. var body = {
  932. type: 'panel',
  933. items: flatten([
  934. urlInput,
  935. displayText,
  936. titleText,
  937. cat([
  938. catalogs.anchor.map(ListOptions.createUi('anchor', 'Anchors')),
  939. catalogs.rels.map(ListOptions.createUi('rel', 'Rel')),
  940. catalogs.targets.map(ListOptions.createUi('target', 'Open link in...')),
  941. catalogs.link.map(ListOptions.createUi('link', 'Link list')),
  942. catalogs.classes.map(ListOptions.createUi('linkClass', 'Class'))
  943. ])
  944. ])
  945. };
  946. return {
  947. title: 'Insert/Edit Link',
  948. size: 'normal',
  949. body: body,
  950. buttons: [
  951. {
  952. type: 'cancel',
  953. name: 'cancel',
  954. text: 'Cancel'
  955. },
  956. {
  957. type: 'submit',
  958. name: 'save',
  959. text: 'Save',
  960. primary: true
  961. }
  962. ],
  963. initialData: initialData,
  964. onChange: function (api, _a) {
  965. var name = _a.name;
  966. dialogDelta.onChange(api.getData, { name: name }).each(function (newData) {
  967. api.setData(newData);
  968. });
  969. },
  970. onSubmit: onSubmit
  971. };
  972. };
  973. var open = function (editor) {
  974. var data = collectData(editor);
  975. data.then(function (info) {
  976. var onSubmit = handleSubmit(editor, info);
  977. return makeDialog(info, onSubmit, editor);
  978. }).then(function (spec) {
  979. editor.windowManager.open(spec);
  980. });
  981. };
  982. var appendClickRemove = function (link, evt) {
  983. document.body.appendChild(link);
  984. link.dispatchEvent(evt);
  985. document.body.removeChild(link);
  986. };
  987. var open$1 = function (url) {
  988. var link = document.createElement('a');
  989. link.target = '_blank';
  990. link.href = url;
  991. link.rel = 'noreferrer noopener';
  992. var evt = document.createEvent('MouseEvents');
  993. evt.initMouseEvent('click', true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
  994. appendClickRemove(link, evt);
  995. };
  996. var getLink = function (editor, elm) {
  997. return editor.dom.getParent(elm, 'a[href]');
  998. };
  999. var getSelectedLink = function (editor) {
  1000. return getLink(editor, editor.selection.getStart());
  1001. };
  1002. var hasOnlyAltModifier = function (e) {
  1003. return e.altKey === true && e.shiftKey === false && e.ctrlKey === false && e.metaKey === false;
  1004. };
  1005. var gotoLink = function (editor, a) {
  1006. if (a) {
  1007. var href = getHref(a);
  1008. if (/^#/.test(href)) {
  1009. var targetEl = editor.$(href);
  1010. if (targetEl.length) {
  1011. editor.selection.scrollIntoView(targetEl[0], true);
  1012. }
  1013. } else {
  1014. open$1(a.href);
  1015. }
  1016. }
  1017. };
  1018. var openDialog = function (editor) {
  1019. return function () {
  1020. open(editor);
  1021. };
  1022. };
  1023. var gotoSelectedLink = function (editor) {
  1024. return function () {
  1025. gotoLink(editor, getSelectedLink(editor));
  1026. };
  1027. };
  1028. var setupGotoLinks = function (editor) {
  1029. editor.on('click', function (e) {
  1030. var link = getLink(editor, e.target);
  1031. if (link && global$1.metaKeyPressed(e)) {
  1032. e.preventDefault();
  1033. gotoLink(editor, link);
  1034. }
  1035. });
  1036. editor.on('keydown', function (e) {
  1037. var link = getSelectedLink(editor);
  1038. if (link && e.keyCode === 13 && hasOnlyAltModifier(e)) {
  1039. e.preventDefault();
  1040. gotoLink(editor, link);
  1041. }
  1042. });
  1043. };
  1044. var toggleState = function (editor, toggler) {
  1045. editor.on('NodeChange', toggler);
  1046. return function () {
  1047. return editor.off('NodeChange', toggler);
  1048. };
  1049. };
  1050. var toggleActiveState = function (editor) {
  1051. return function (api) {
  1052. return toggleState(editor, function () {
  1053. api.setActive(!editor.mode.isReadOnly() && getAnchorElement(editor, editor.selection.getNode()) !== null);
  1054. });
  1055. };
  1056. };
  1057. var toggleEnabledState = function (editor) {
  1058. return function (api) {
  1059. var updateState = function () {
  1060. return api.setDisabled(getAnchorElement(editor, editor.selection.getNode()) === null);
  1061. };
  1062. updateState();
  1063. return toggleState(editor, updateState);
  1064. };
  1065. };
  1066. var toggleUnlinkState = function (editor) {
  1067. return function (api) {
  1068. var hasLinks$1 = function (parents) {
  1069. return hasLinks(parents) || hasLinksInSelection(editor.selection.getRng());
  1070. };
  1071. var parents = editor.dom.getParents(editor.selection.getStart());
  1072. api.setDisabled(!hasLinks$1(parents));
  1073. return toggleState(editor, function (e) {
  1074. return api.setDisabled(!hasLinks$1(e.parents));
  1075. });
  1076. };
  1077. };
  1078. var register = function (editor) {
  1079. editor.addCommand('mceLink', function () {
  1080. if (useQuickLink(editor)) {
  1081. editor.fire('contexttoolbar-show', { toolbarKey: 'quicklink' });
  1082. } else {
  1083. openDialog(editor)();
  1084. }
  1085. });
  1086. };
  1087. var setup = function (editor) {
  1088. editor.addShortcut('Meta+K', '', function () {
  1089. editor.execCommand('mceLink');
  1090. });
  1091. };
  1092. var setupButtons = function (editor) {
  1093. editor.ui.registry.addToggleButton('link', {
  1094. icon: 'link',
  1095. tooltip: 'Insert/edit link',
  1096. onAction: openDialog(editor),
  1097. onSetup: toggleActiveState(editor)
  1098. });
  1099. editor.ui.registry.addButton('openlink', {
  1100. icon: 'new-tab',
  1101. tooltip: 'Open link',
  1102. onAction: gotoSelectedLink(editor),
  1103. onSetup: toggleEnabledState(editor)
  1104. });
  1105. editor.ui.registry.addButton('unlink', {
  1106. icon: 'unlink',
  1107. tooltip: 'Remove link',
  1108. onAction: function () {
  1109. return unlink(editor);
  1110. },
  1111. onSetup: toggleUnlinkState(editor)
  1112. });
  1113. };
  1114. var setupMenuItems = function (editor) {
  1115. editor.ui.registry.addMenuItem('openlink', {
  1116. text: 'Open link',
  1117. icon: 'new-tab',
  1118. onAction: gotoSelectedLink(editor),
  1119. onSetup: toggleEnabledState(editor)
  1120. });
  1121. editor.ui.registry.addMenuItem('link', {
  1122. icon: 'link',
  1123. text: 'Link...',
  1124. shortcut: 'Meta+K',
  1125. onAction: openDialog(editor)
  1126. });
  1127. editor.ui.registry.addMenuItem('unlink', {
  1128. icon: 'unlink',
  1129. text: 'Remove link',
  1130. onAction: function () {
  1131. return unlink(editor);
  1132. },
  1133. onSetup: toggleUnlinkState(editor)
  1134. });
  1135. };
  1136. var setupContextMenu = function (editor) {
  1137. var inLink = 'link unlink openlink';
  1138. var noLink = 'link';
  1139. editor.ui.registry.addContextMenu('link', {
  1140. update: function (element) {
  1141. return hasLinks(editor.dom.getParents(element, 'a')) ? inLink : noLink;
  1142. }
  1143. });
  1144. };
  1145. var setupContextToolbars = function (editor) {
  1146. var collapseSelectionToEnd = function (editor) {
  1147. editor.selection.collapse(false);
  1148. };
  1149. var onSetupLink = function (buttonApi) {
  1150. var node = editor.selection.getNode();
  1151. buttonApi.setDisabled(!getAnchorElement(editor, node));
  1152. return function () {
  1153. };
  1154. };
  1155. editor.ui.registry.addContextForm('quicklink', {
  1156. launch: {
  1157. type: 'contextformtogglebutton',
  1158. icon: 'link',
  1159. tooltip: 'Link',
  1160. onSetup: toggleActiveState(editor)
  1161. },
  1162. label: 'Link',
  1163. predicate: function (node) {
  1164. return !!getAnchorElement(editor, node) && hasContextToolbar(editor);
  1165. },
  1166. initValue: function () {
  1167. var elm = getAnchorElement(editor);
  1168. return !!elm ? getHref(elm) : '';
  1169. },
  1170. commands: [
  1171. {
  1172. type: 'contextformtogglebutton',
  1173. icon: 'link',
  1174. tooltip: 'Link',
  1175. primary: true,
  1176. onSetup: function (buttonApi) {
  1177. var node = editor.selection.getNode();
  1178. buttonApi.setActive(!!getAnchorElement(editor, node));
  1179. return toggleActiveState(editor)(buttonApi);
  1180. },
  1181. onAction: function (formApi) {
  1182. var anchor = getAnchorElement(editor);
  1183. var value = formApi.getValue();
  1184. if (!anchor) {
  1185. var attachState = {
  1186. href: value,
  1187. attach: function () {
  1188. }
  1189. };
  1190. var onlyText = isOnlyTextSelected(editor);
  1191. var text = onlyText ? Optional.some(getAnchorText(editor.selection, anchor)).filter(function (t) {
  1192. return t.length > 0;
  1193. }).or(Optional.from(value)) : Optional.none();
  1194. link(editor, attachState, {
  1195. href: value,
  1196. text: text,
  1197. title: Optional.none(),
  1198. rel: Optional.none(),
  1199. target: Optional.none(),
  1200. class: Optional.none()
  1201. });
  1202. formApi.hide();
  1203. } else {
  1204. editor.undoManager.transact(function () {
  1205. editor.dom.setAttrib(anchor, 'href', value);
  1206. collapseSelectionToEnd(editor);
  1207. formApi.hide();
  1208. });
  1209. }
  1210. }
  1211. },
  1212. {
  1213. type: 'contextformbutton',
  1214. icon: 'unlink',
  1215. tooltip: 'Remove link',
  1216. onSetup: onSetupLink,
  1217. onAction: function (formApi) {
  1218. unlink(editor);
  1219. formApi.hide();
  1220. }
  1221. },
  1222. {
  1223. type: 'contextformbutton',
  1224. icon: 'new-tab',
  1225. tooltip: 'Open link',
  1226. onSetup: onSetupLink,
  1227. onAction: function (formApi) {
  1228. gotoSelectedLink(editor)();
  1229. formApi.hide();
  1230. }
  1231. }
  1232. ]
  1233. });
  1234. };
  1235. function Plugin () {
  1236. global.add('link', function (editor) {
  1237. setupButtons(editor);
  1238. setupMenuItems(editor);
  1239. setupContextMenu(editor);
  1240. setupContextToolbars(editor);
  1241. setupGotoLinks(editor);
  1242. register(editor);
  1243. setup(editor);
  1244. });
  1245. }
  1246. Plugin();
  1247. }());