plugin.js 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969
  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 Cell = function (initial) {
  12. var value = initial;
  13. var get = function () {
  14. return value;
  15. };
  16. var set = function (v) {
  17. value = v;
  18. };
  19. return {
  20. get: get,
  21. set: set
  22. };
  23. };
  24. var global = tinymce.util.Tools.resolve('tinymce.PluginManager');
  25. var get = function (fullscreenState) {
  26. return {
  27. isFullscreen: function () {
  28. return fullscreenState.get() !== null;
  29. }
  30. };
  31. };
  32. var noop = function () {
  33. };
  34. var compose = function (fa, fb) {
  35. return function () {
  36. var args = [];
  37. for (var _i = 0; _i < arguments.length; _i++) {
  38. args[_i] = arguments[_i];
  39. }
  40. return fa(fb.apply(null, args));
  41. };
  42. };
  43. var compose1 = function (fbc, fab) {
  44. return function (a) {
  45. return fbc(fab(a));
  46. };
  47. };
  48. var constant = function (value) {
  49. return function () {
  50. return value;
  51. };
  52. };
  53. function curry(fn) {
  54. var initialArgs = [];
  55. for (var _i = 1; _i < arguments.length; _i++) {
  56. initialArgs[_i - 1] = arguments[_i];
  57. }
  58. return function () {
  59. var restArgs = [];
  60. for (var _i = 0; _i < arguments.length; _i++) {
  61. restArgs[_i] = arguments[_i];
  62. }
  63. var all = initialArgs.concat(restArgs);
  64. return fn.apply(null, all);
  65. };
  66. }
  67. var never = constant(false);
  68. var always = constant(true);
  69. var none = function () {
  70. return NONE;
  71. };
  72. var NONE = function () {
  73. var eq = function (o) {
  74. return o.isNone();
  75. };
  76. var call = function (thunk) {
  77. return thunk();
  78. };
  79. var id = function (n) {
  80. return n;
  81. };
  82. var me = {
  83. fold: function (n, _s) {
  84. return n();
  85. },
  86. is: never,
  87. isSome: never,
  88. isNone: always,
  89. getOr: id,
  90. getOrThunk: call,
  91. getOrDie: function (msg) {
  92. throw new Error(msg || 'error: getOrDie called on none.');
  93. },
  94. getOrNull: constant(null),
  95. getOrUndefined: constant(undefined),
  96. or: id,
  97. orThunk: call,
  98. map: none,
  99. each: noop,
  100. bind: none,
  101. exists: never,
  102. forall: always,
  103. filter: none,
  104. equals: eq,
  105. equals_: eq,
  106. toArray: function () {
  107. return [];
  108. },
  109. toString: constant('none()')
  110. };
  111. return me;
  112. }();
  113. var some = function (a) {
  114. var constant_a = constant(a);
  115. var self = function () {
  116. return me;
  117. };
  118. var bind = function (f) {
  119. return f(a);
  120. };
  121. var me = {
  122. fold: function (n, s) {
  123. return s(a);
  124. },
  125. is: function (v) {
  126. return a === v;
  127. },
  128. isSome: always,
  129. isNone: never,
  130. getOr: constant_a,
  131. getOrThunk: constant_a,
  132. getOrDie: constant_a,
  133. getOrNull: constant_a,
  134. getOrUndefined: constant_a,
  135. or: self,
  136. orThunk: self,
  137. map: function (f) {
  138. return some(f(a));
  139. },
  140. each: function (f) {
  141. f(a);
  142. },
  143. bind: bind,
  144. exists: bind,
  145. forall: bind,
  146. filter: function (f) {
  147. return f(a) ? me : NONE;
  148. },
  149. toArray: function () {
  150. return [a];
  151. },
  152. toString: function () {
  153. return 'some(' + a + ')';
  154. },
  155. equals: function (o) {
  156. return o.is(a);
  157. },
  158. equals_: function (o, elementEq) {
  159. return o.fold(never, function (b) {
  160. return elementEq(a, b);
  161. });
  162. }
  163. };
  164. return me;
  165. };
  166. var from = function (value) {
  167. return value === null || value === undefined ? NONE : some(value);
  168. };
  169. var Optional = {
  170. some: some,
  171. none: none,
  172. from: from
  173. };
  174. var revocable = function (doRevoke) {
  175. var subject = Cell(Optional.none());
  176. var revoke = function () {
  177. return subject.get().each(doRevoke);
  178. };
  179. var clear = function () {
  180. revoke();
  181. subject.set(Optional.none());
  182. };
  183. var isSet = function () {
  184. return subject.get().isSome();
  185. };
  186. var set = function (s) {
  187. revoke();
  188. subject.set(Optional.some(s));
  189. };
  190. return {
  191. clear: clear,
  192. isSet: isSet,
  193. set: set
  194. };
  195. };
  196. var unbindable = function () {
  197. return revocable(function (s) {
  198. return s.unbind();
  199. });
  200. };
  201. var value = function () {
  202. var subject = Cell(Optional.none());
  203. var clear = function () {
  204. return subject.set(Optional.none());
  205. };
  206. var set = function (s) {
  207. return subject.set(Optional.some(s));
  208. };
  209. var isSet = function () {
  210. return subject.get().isSome();
  211. };
  212. var on = function (f) {
  213. return subject.get().each(f);
  214. };
  215. return {
  216. clear: clear,
  217. set: set,
  218. isSet: isSet,
  219. on: on
  220. };
  221. };
  222. var typeOf = function (x) {
  223. var t = typeof x;
  224. if (x === null) {
  225. return 'null';
  226. } else if (t === 'object' && (Array.prototype.isPrototypeOf(x) || x.constructor && x.constructor.name === 'Array')) {
  227. return 'array';
  228. } else if (t === 'object' && (String.prototype.isPrototypeOf(x) || x.constructor && x.constructor.name === 'String')) {
  229. return 'string';
  230. } else {
  231. return t;
  232. }
  233. };
  234. var isType = function (type) {
  235. return function (value) {
  236. return typeOf(value) === type;
  237. };
  238. };
  239. var isSimpleType = function (type) {
  240. return function (value) {
  241. return typeof value === type;
  242. };
  243. };
  244. var isString = isType('string');
  245. var isArray = isType('array');
  246. var isBoolean = isSimpleType('boolean');
  247. var isNullable = function (a) {
  248. return a === null || a === undefined;
  249. };
  250. var isNonNullable = function (a) {
  251. return !isNullable(a);
  252. };
  253. var isFunction = isSimpleType('function');
  254. var isNumber = isSimpleType('number');
  255. var nativePush = Array.prototype.push;
  256. var map = function (xs, f) {
  257. var len = xs.length;
  258. var r = new Array(len);
  259. for (var i = 0; i < len; i++) {
  260. var x = xs[i];
  261. r[i] = f(x, i);
  262. }
  263. return r;
  264. };
  265. var each = function (xs, f) {
  266. for (var i = 0, len = xs.length; i < len; i++) {
  267. var x = xs[i];
  268. f(x, i);
  269. }
  270. };
  271. var filter = function (xs, pred) {
  272. var r = [];
  273. for (var i = 0, len = xs.length; i < len; i++) {
  274. var x = xs[i];
  275. if (pred(x, i)) {
  276. r.push(x);
  277. }
  278. }
  279. return r;
  280. };
  281. var flatten = function (xs) {
  282. var r = [];
  283. for (var i = 0, len = xs.length; i < len; ++i) {
  284. if (!isArray(xs[i])) {
  285. throw new Error('Arr.flatten item ' + i + ' was not an array, input: ' + xs);
  286. }
  287. nativePush.apply(r, xs[i]);
  288. }
  289. return r;
  290. };
  291. var bind = function (xs, f) {
  292. return flatten(map(xs, f));
  293. };
  294. var head = function (xs) {
  295. return xs.length === 0 ? Optional.none() : Optional.some(xs[0]);
  296. };
  297. var keys = Object.keys;
  298. var each$1 = function (obj, f) {
  299. var props = keys(obj);
  300. for (var k = 0, len = props.length; k < len; k++) {
  301. var i = props[k];
  302. var x = obj[i];
  303. f(x, i);
  304. }
  305. };
  306. var isSupported = function (dom) {
  307. return dom.style !== undefined && isFunction(dom.style.getPropertyValue);
  308. };
  309. var fromHtml = function (html, scope) {
  310. var doc = scope || document;
  311. var div = doc.createElement('div');
  312. div.innerHTML = html;
  313. if (!div.hasChildNodes() || div.childNodes.length > 1) {
  314. console.error('HTML does not have a single root node', html);
  315. throw new Error('HTML must have a single root node');
  316. }
  317. return fromDom(div.childNodes[0]);
  318. };
  319. var fromTag = function (tag, scope) {
  320. var doc = scope || document;
  321. var node = doc.createElement(tag);
  322. return fromDom(node);
  323. };
  324. var fromText = function (text, scope) {
  325. var doc = scope || document;
  326. var node = doc.createTextNode(text);
  327. return fromDom(node);
  328. };
  329. var fromDom = function (node) {
  330. if (node === null || node === undefined) {
  331. throw new Error('Node cannot be null or undefined');
  332. }
  333. return { dom: node };
  334. };
  335. var fromPoint = function (docElm, x, y) {
  336. return Optional.from(docElm.dom.elementFromPoint(x, y)).map(fromDom);
  337. };
  338. var SugarElement = {
  339. fromHtml: fromHtml,
  340. fromTag: fromTag,
  341. fromText: fromText,
  342. fromDom: fromDom,
  343. fromPoint: fromPoint
  344. };
  345. var Global = typeof window !== 'undefined' ? window : Function('return this;')();
  346. var DOCUMENT = 9;
  347. var DOCUMENT_FRAGMENT = 11;
  348. var ELEMENT = 1;
  349. var TEXT = 3;
  350. var type = function (element) {
  351. return element.dom.nodeType;
  352. };
  353. var isType$1 = function (t) {
  354. return function (element) {
  355. return type(element) === t;
  356. };
  357. };
  358. var isElement = isType$1(ELEMENT);
  359. var isText = isType$1(TEXT);
  360. var isDocument = isType$1(DOCUMENT);
  361. var isDocumentFragment = isType$1(DOCUMENT_FRAGMENT);
  362. var is = function (element, selector) {
  363. var dom = element.dom;
  364. if (dom.nodeType !== ELEMENT) {
  365. return false;
  366. } else {
  367. var elem = dom;
  368. if (elem.matches !== undefined) {
  369. return elem.matches(selector);
  370. } else if (elem.msMatchesSelector !== undefined) {
  371. return elem.msMatchesSelector(selector);
  372. } else if (elem.webkitMatchesSelector !== undefined) {
  373. return elem.webkitMatchesSelector(selector);
  374. } else if (elem.mozMatchesSelector !== undefined) {
  375. return elem.mozMatchesSelector(selector);
  376. } else {
  377. throw new Error('Browser lacks native selectors');
  378. }
  379. }
  380. };
  381. var bypassSelector = function (dom) {
  382. return dom.nodeType !== ELEMENT && dom.nodeType !== DOCUMENT && dom.nodeType !== DOCUMENT_FRAGMENT || dom.childElementCount === 0;
  383. };
  384. var all = function (selector, scope) {
  385. var base = scope === undefined ? document : scope.dom;
  386. return bypassSelector(base) ? [] : map(base.querySelectorAll(selector), SugarElement.fromDom);
  387. };
  388. var eq = function (e1, e2) {
  389. return e1.dom === e2.dom;
  390. };
  391. var owner = function (element) {
  392. return SugarElement.fromDom(element.dom.ownerDocument);
  393. };
  394. var documentOrOwner = function (dos) {
  395. return isDocument(dos) ? dos : owner(dos);
  396. };
  397. var parent = function (element) {
  398. return Optional.from(element.dom.parentNode).map(SugarElement.fromDom);
  399. };
  400. var parents = function (element, isRoot) {
  401. var stop = isFunction(isRoot) ? isRoot : never;
  402. var dom = element.dom;
  403. var ret = [];
  404. while (dom.parentNode !== null && dom.parentNode !== undefined) {
  405. var rawParent = dom.parentNode;
  406. var p = SugarElement.fromDom(rawParent);
  407. ret.push(p);
  408. if (stop(p) === true) {
  409. break;
  410. } else {
  411. dom = rawParent;
  412. }
  413. }
  414. return ret;
  415. };
  416. var siblings = function (element) {
  417. var filterSelf = function (elements) {
  418. return filter(elements, function (x) {
  419. return !eq(element, x);
  420. });
  421. };
  422. return parent(element).map(children).map(filterSelf).getOr([]);
  423. };
  424. var children = function (element) {
  425. return map(element.dom.childNodes, SugarElement.fromDom);
  426. };
  427. var isShadowRoot = function (dos) {
  428. return isDocumentFragment(dos);
  429. };
  430. var supported = isFunction(Element.prototype.attachShadow) && isFunction(Node.prototype.getRootNode);
  431. var isSupported$1 = constant(supported);
  432. var getRootNode = supported ? function (e) {
  433. return SugarElement.fromDom(e.dom.getRootNode());
  434. } : documentOrOwner;
  435. var getShadowRoot = function (e) {
  436. var r = getRootNode(e);
  437. return isShadowRoot(r) ? Optional.some(r) : Optional.none();
  438. };
  439. var getShadowHost = function (e) {
  440. return SugarElement.fromDom(e.dom.host);
  441. };
  442. var getOriginalEventTarget = function (event) {
  443. if (isSupported$1() && isNonNullable(event.target)) {
  444. var el = SugarElement.fromDom(event.target);
  445. if (isElement(el) && isOpenShadowHost(el)) {
  446. if (event.composed && event.composedPath) {
  447. var composedPath = event.composedPath();
  448. if (composedPath) {
  449. return head(composedPath);
  450. }
  451. }
  452. }
  453. }
  454. return Optional.from(event.target);
  455. };
  456. var isOpenShadowHost = function (element) {
  457. return isNonNullable(element.dom.shadowRoot);
  458. };
  459. var inBody = function (element) {
  460. var dom = isText(element) ? element.dom.parentNode : element.dom;
  461. if (dom === undefined || dom === null || dom.ownerDocument === null) {
  462. return false;
  463. }
  464. var doc = dom.ownerDocument;
  465. return getShadowRoot(SugarElement.fromDom(dom)).fold(function () {
  466. return doc.body.contains(dom);
  467. }, compose1(inBody, getShadowHost));
  468. };
  469. var getBody = function (doc) {
  470. var b = doc.dom.body;
  471. if (b === null || b === undefined) {
  472. throw new Error('Body is not available yet');
  473. }
  474. return SugarElement.fromDom(b);
  475. };
  476. var rawSet = function (dom, key, value) {
  477. if (isString(value) || isBoolean(value) || isNumber(value)) {
  478. dom.setAttribute(key, value + '');
  479. } else {
  480. console.error('Invalid call to Attribute.set. Key ', key, ':: Value ', value, ':: Element ', dom);
  481. throw new Error('Attribute value was not simple');
  482. }
  483. };
  484. var set = function (element, key, value) {
  485. rawSet(element.dom, key, value);
  486. };
  487. var get$1 = function (element, key) {
  488. var v = element.dom.getAttribute(key);
  489. return v === null ? undefined : v;
  490. };
  491. var remove = function (element, key) {
  492. element.dom.removeAttribute(key);
  493. };
  494. var internalSet = function (dom, property, value) {
  495. if (!isString(value)) {
  496. console.error('Invalid call to CSS.set. Property ', property, ':: Value ', value, ':: Element ', dom);
  497. throw new Error('CSS value must be a string: ' + value);
  498. }
  499. if (isSupported(dom)) {
  500. dom.style.setProperty(property, value);
  501. }
  502. };
  503. var setAll = function (element, css) {
  504. var dom = element.dom;
  505. each$1(css, function (v, k) {
  506. internalSet(dom, k, v);
  507. });
  508. };
  509. var get$2 = function (element, property) {
  510. var dom = element.dom;
  511. var styles = window.getComputedStyle(dom);
  512. var r = styles.getPropertyValue(property);
  513. return r === '' && !inBody(element) ? getUnsafeProperty(dom, property) : r;
  514. };
  515. var getUnsafeProperty = function (dom, property) {
  516. return isSupported(dom) ? dom.style.getPropertyValue(property) : '';
  517. };
  518. var mkEvent = function (target, x, y, stop, prevent, kill, raw) {
  519. return {
  520. target: target,
  521. x: x,
  522. y: y,
  523. stop: stop,
  524. prevent: prevent,
  525. kill: kill,
  526. raw: raw
  527. };
  528. };
  529. var fromRawEvent = function (rawEvent) {
  530. var target = SugarElement.fromDom(getOriginalEventTarget(rawEvent).getOr(rawEvent.target));
  531. var stop = function () {
  532. return rawEvent.stopPropagation();
  533. };
  534. var prevent = function () {
  535. return rawEvent.preventDefault();
  536. };
  537. var kill = compose(prevent, stop);
  538. return mkEvent(target, rawEvent.clientX, rawEvent.clientY, stop, prevent, kill, rawEvent);
  539. };
  540. var handle = function (filter, handler) {
  541. return function (rawEvent) {
  542. if (filter(rawEvent)) {
  543. handler(fromRawEvent(rawEvent));
  544. }
  545. };
  546. };
  547. var binder = function (element, event, filter, handler, useCapture) {
  548. var wrapped = handle(filter, handler);
  549. element.dom.addEventListener(event, wrapped, useCapture);
  550. return { unbind: curry(unbind, element, event, wrapped, useCapture) };
  551. };
  552. var bind$1 = function (element, event, filter, handler) {
  553. return binder(element, event, filter, handler, false);
  554. };
  555. var unbind = function (element, event, handler, useCapture) {
  556. element.dom.removeEventListener(event, handler, useCapture);
  557. };
  558. var filter$1 = always;
  559. var bind$2 = function (element, event, handler) {
  560. return bind$1(element, event, filter$1, handler);
  561. };
  562. var r = function (left, top) {
  563. var translate = function (x, y) {
  564. return r(left + x, top + y);
  565. };
  566. return {
  567. left: left,
  568. top: top,
  569. translate: translate
  570. };
  571. };
  572. var SugarPosition = r;
  573. var get$3 = function (_DOC) {
  574. var doc = _DOC !== undefined ? _DOC.dom : document;
  575. var x = doc.body.scrollLeft || doc.documentElement.scrollLeft;
  576. var y = doc.body.scrollTop || doc.documentElement.scrollTop;
  577. return SugarPosition(x, y);
  578. };
  579. var get$4 = function (_win) {
  580. var win = _win === undefined ? window : _win;
  581. return Optional.from(win['visualViewport']);
  582. };
  583. var bounds = function (x, y, width, height) {
  584. return {
  585. x: x,
  586. y: y,
  587. width: width,
  588. height: height,
  589. right: x + width,
  590. bottom: y + height
  591. };
  592. };
  593. var getBounds = function (_win) {
  594. var win = _win === undefined ? window : _win;
  595. var doc = win.document;
  596. var scroll = get$3(SugarElement.fromDom(doc));
  597. return get$4(win).fold(function () {
  598. var html = win.document.documentElement;
  599. var width = html.clientWidth;
  600. var height = html.clientHeight;
  601. return bounds(scroll.left, scroll.top, width, height);
  602. }, function (visualViewport) {
  603. return bounds(Math.max(visualViewport.pageLeft, scroll.left), Math.max(visualViewport.pageTop, scroll.top), visualViewport.width, visualViewport.height);
  604. });
  605. };
  606. var bind$3 = function (name, callback, _win) {
  607. return get$4(_win).map(function (visualViewport) {
  608. var handler = function (e) {
  609. return callback(fromRawEvent(e));
  610. };
  611. visualViewport.addEventListener(name, handler);
  612. return {
  613. unbind: function () {
  614. return visualViewport.removeEventListener(name, handler);
  615. }
  616. };
  617. }).getOrThunk(function () {
  618. return { unbind: noop };
  619. });
  620. };
  621. var global$1 = tinymce.util.Tools.resolve('tinymce.dom.DOMUtils');
  622. var global$2 = tinymce.util.Tools.resolve('tinymce.Env');
  623. var global$3 = tinymce.util.Tools.resolve('tinymce.util.Delay');
  624. var fireFullscreenStateChanged = function (editor, state) {
  625. editor.fire('FullscreenStateChanged', { state: state });
  626. };
  627. var getFullscreenNative = function (editor) {
  628. return editor.getParam('fullscreen_native', false, 'boolean');
  629. };
  630. var getFullscreenRoot = function (editor) {
  631. var elem = SugarElement.fromDom(editor.getElement());
  632. return getShadowRoot(elem).map(getShadowHost).getOrThunk(function () {
  633. return getBody(owner(elem));
  634. });
  635. };
  636. var getFullscreenElement = function (root) {
  637. if (root.fullscreenElement !== undefined) {
  638. return root.fullscreenElement;
  639. } else if (root.msFullscreenElement !== undefined) {
  640. return root.msFullscreenElement;
  641. } else if (root.webkitFullscreenElement !== undefined) {
  642. return root.webkitFullscreenElement;
  643. } else {
  644. return null;
  645. }
  646. };
  647. var getFullscreenchangeEventName = function () {
  648. if (document.fullscreenElement !== undefined) {
  649. return 'fullscreenchange';
  650. } else if (document.msFullscreenElement !== undefined) {
  651. return 'MSFullscreenChange';
  652. } else if (document.webkitFullscreenElement !== undefined) {
  653. return 'webkitfullscreenchange';
  654. } else {
  655. return 'fullscreenchange';
  656. }
  657. };
  658. var requestFullscreen = function (sugarElem) {
  659. var elem = sugarElem.dom;
  660. if (elem.requestFullscreen) {
  661. elem.requestFullscreen();
  662. } else if (elem.msRequestFullscreen) {
  663. elem.msRequestFullscreen();
  664. } else if (elem.webkitRequestFullScreen) {
  665. elem.webkitRequestFullScreen();
  666. }
  667. };
  668. var exitFullscreen = function (sugarDoc) {
  669. var doc = sugarDoc.dom;
  670. if (doc.exitFullscreen) {
  671. doc.exitFullscreen();
  672. } else if (doc.msExitFullscreen) {
  673. doc.msExitFullscreen();
  674. } else if (doc.webkitCancelFullScreen) {
  675. doc.webkitCancelFullScreen();
  676. }
  677. };
  678. var isFullscreenElement = function (elem) {
  679. return elem.dom === getFullscreenElement(owner(elem).dom);
  680. };
  681. var ancestors = function (scope, predicate, isRoot) {
  682. return filter(parents(scope, isRoot), predicate);
  683. };
  684. var siblings$1 = function (scope, predicate) {
  685. return filter(siblings(scope), predicate);
  686. };
  687. var all$1 = function (selector) {
  688. return all(selector);
  689. };
  690. var ancestors$1 = function (scope, selector, isRoot) {
  691. return ancestors(scope, function (e) {
  692. return is(e, selector);
  693. }, isRoot);
  694. };
  695. var siblings$2 = function (scope, selector) {
  696. return siblings$1(scope, function (e) {
  697. return is(e, selector);
  698. });
  699. };
  700. var attr = 'data-ephox-mobile-fullscreen-style';
  701. var siblingStyles = 'display:none!important;';
  702. var ancestorPosition = 'position:absolute!important;';
  703. var ancestorStyles = 'top:0!important;left:0!important;margin:0!important;padding:0!important;width:100%!important;height:100%!important;overflow:visible!important;';
  704. var bgFallback = 'background-color:rgb(255,255,255)!important;';
  705. var isAndroid = global$2.os.isAndroid();
  706. var matchColor = function (editorBody) {
  707. var color = get$2(editorBody, 'background-color');
  708. return color !== undefined && color !== '' ? 'background-color:' + color + '!important' : bgFallback;
  709. };
  710. var clobberStyles = function (dom, container, editorBody) {
  711. var gatherSiblings = function (element) {
  712. return siblings$2(element, '*:not(.tox-silver-sink)');
  713. };
  714. var clobber = function (clobberStyle) {
  715. return function (element) {
  716. var styles = get$1(element, 'style');
  717. var backup = styles === undefined ? 'no-styles' : styles.trim();
  718. if (backup === clobberStyle) {
  719. return;
  720. } else {
  721. set(element, attr, backup);
  722. setAll(element, dom.parseStyle(clobberStyle));
  723. }
  724. };
  725. };
  726. var ancestors = ancestors$1(container, '*');
  727. var siblings = bind(ancestors, gatherSiblings);
  728. var bgColor = matchColor(editorBody);
  729. each(siblings, clobber(siblingStyles));
  730. each(ancestors, clobber(ancestorPosition + ancestorStyles + bgColor));
  731. var containerStyles = isAndroid === true ? '' : ancestorPosition;
  732. clobber(containerStyles + ancestorStyles + bgColor)(container);
  733. };
  734. var restoreStyles = function (dom) {
  735. var clobberedEls = all$1('[' + attr + ']');
  736. each(clobberedEls, function (element) {
  737. var restore = get$1(element, attr);
  738. if (restore !== 'no-styles') {
  739. setAll(element, dom.parseStyle(restore));
  740. } else {
  741. remove(element, 'style');
  742. }
  743. remove(element, attr);
  744. });
  745. };
  746. var DOM = global$1.DOM;
  747. var getScrollPos = function () {
  748. var vp = getBounds(window);
  749. return {
  750. x: vp.x,
  751. y: vp.y
  752. };
  753. };
  754. var setScrollPos = function (pos) {
  755. window.scrollTo(pos.x, pos.y);
  756. };
  757. var viewportUpdate = get$4().fold(function () {
  758. return {
  759. bind: noop,
  760. unbind: noop
  761. };
  762. }, function (visualViewport) {
  763. var editorContainer = value();
  764. var resizeBinder = unbindable();
  765. var scrollBinder = unbindable();
  766. var refreshScroll = function () {
  767. document.body.scrollTop = 0;
  768. document.documentElement.scrollTop = 0;
  769. };
  770. var refreshVisualViewport = function () {
  771. window.requestAnimationFrame(function () {
  772. editorContainer.on(function (container) {
  773. return setAll(container, {
  774. top: visualViewport.offsetTop + 'px',
  775. left: visualViewport.offsetLeft + 'px',
  776. height: visualViewport.height + 'px',
  777. width: visualViewport.width + 'px'
  778. });
  779. });
  780. });
  781. };
  782. var update = global$3.throttle(function () {
  783. refreshScroll();
  784. refreshVisualViewport();
  785. }, 50);
  786. var bind = function (element) {
  787. editorContainer.set(element);
  788. update();
  789. resizeBinder.set(bind$3('resize', update));
  790. scrollBinder.set(bind$3('scroll', update));
  791. };
  792. var unbind = function () {
  793. editorContainer.on(function () {
  794. resizeBinder.clear();
  795. scrollBinder.clear();
  796. });
  797. editorContainer.clear();
  798. };
  799. return {
  800. bind: bind,
  801. unbind: unbind
  802. };
  803. });
  804. var toggleFullscreen = function (editor, fullscreenState) {
  805. var body = document.body;
  806. var documentElement = document.documentElement;
  807. var editorContainer = editor.getContainer();
  808. var editorContainerS = SugarElement.fromDom(editorContainer);
  809. var fullscreenRoot = getFullscreenRoot(editor);
  810. var fullscreenInfo = fullscreenState.get();
  811. var editorBody = SugarElement.fromDom(editor.getBody());
  812. var isTouch = global$2.deviceType.isTouch();
  813. var editorContainerStyle = editorContainer.style;
  814. var iframe = editor.iframeElement;
  815. var iframeStyle = iframe.style;
  816. var cleanup = function () {
  817. if (isTouch) {
  818. restoreStyles(editor.dom);
  819. }
  820. DOM.removeClass(body, 'tox-fullscreen');
  821. DOM.removeClass(documentElement, 'tox-fullscreen');
  822. DOM.removeClass(editorContainer, 'tox-fullscreen');
  823. viewportUpdate.unbind();
  824. Optional.from(fullscreenState.get()).each(function (info) {
  825. return info.fullscreenChangeHandler.unbind();
  826. });
  827. };
  828. if (!fullscreenInfo) {
  829. var fullscreenChangeHandler = bind$2(owner(fullscreenRoot), getFullscreenchangeEventName(), function (_evt) {
  830. if (getFullscreenNative(editor)) {
  831. if (!isFullscreenElement(fullscreenRoot) && fullscreenState.get() !== null) {
  832. toggleFullscreen(editor, fullscreenState);
  833. }
  834. }
  835. });
  836. var newFullScreenInfo = {
  837. scrollPos: getScrollPos(),
  838. containerWidth: editorContainerStyle.width,
  839. containerHeight: editorContainerStyle.height,
  840. containerTop: editorContainerStyle.top,
  841. containerLeft: editorContainerStyle.left,
  842. iframeWidth: iframeStyle.width,
  843. iframeHeight: iframeStyle.height,
  844. fullscreenChangeHandler: fullscreenChangeHandler
  845. };
  846. if (isTouch) {
  847. clobberStyles(editor.dom, editorContainerS, editorBody);
  848. }
  849. iframeStyle.width = iframeStyle.height = '100%';
  850. editorContainerStyle.width = editorContainerStyle.height = '';
  851. DOM.addClass(body, 'tox-fullscreen');
  852. DOM.addClass(documentElement, 'tox-fullscreen');
  853. DOM.addClass(editorContainer, 'tox-fullscreen');
  854. viewportUpdate.bind(editorContainerS);
  855. editor.on('remove', cleanup);
  856. fullscreenState.set(newFullScreenInfo);
  857. if (getFullscreenNative(editor)) {
  858. requestFullscreen(fullscreenRoot);
  859. }
  860. fireFullscreenStateChanged(editor, true);
  861. } else {
  862. fullscreenInfo.fullscreenChangeHandler.unbind();
  863. if (getFullscreenNative(editor) && isFullscreenElement(fullscreenRoot)) {
  864. exitFullscreen(owner(fullscreenRoot));
  865. }
  866. iframeStyle.width = fullscreenInfo.iframeWidth;
  867. iframeStyle.height = fullscreenInfo.iframeHeight;
  868. editorContainerStyle.width = fullscreenInfo.containerWidth;
  869. editorContainerStyle.height = fullscreenInfo.containerHeight;
  870. editorContainerStyle.top = fullscreenInfo.containerTop;
  871. editorContainerStyle.left = fullscreenInfo.containerLeft;
  872. setScrollPos(fullscreenInfo.scrollPos);
  873. fullscreenState.set(null);
  874. fireFullscreenStateChanged(editor, false);
  875. cleanup();
  876. editor.off('remove', cleanup);
  877. }
  878. };
  879. var register = function (editor, fullscreenState) {
  880. editor.addCommand('mceFullScreen', function () {
  881. toggleFullscreen(editor, fullscreenState);
  882. });
  883. };
  884. var makeSetupHandler = function (editor, fullscreenState) {
  885. return function (api) {
  886. api.setActive(fullscreenState.get() !== null);
  887. var editorEventCallback = function (e) {
  888. return api.setActive(e.state);
  889. };
  890. editor.on('FullscreenStateChanged', editorEventCallback);
  891. return function () {
  892. return editor.off('FullscreenStateChanged', editorEventCallback);
  893. };
  894. };
  895. };
  896. var register$1 = function (editor, fullscreenState) {
  897. editor.ui.registry.addToggleMenuItem('fullscreen', {
  898. text: 'Fullscreen',
  899. icon: 'fullscreen',
  900. shortcut: 'Meta+Shift+F',
  901. onAction: function () {
  902. return editor.execCommand('mceFullScreen');
  903. },
  904. onSetup: makeSetupHandler(editor, fullscreenState)
  905. });
  906. editor.ui.registry.addToggleButton('fullscreen', {
  907. tooltip: 'Fullscreen',
  908. icon: 'fullscreen',
  909. onAction: function () {
  910. return editor.execCommand('mceFullScreen');
  911. },
  912. onSetup: makeSetupHandler(editor, fullscreenState)
  913. });
  914. };
  915. function Plugin () {
  916. global.add('fullscreen', function (editor) {
  917. var fullscreenState = Cell(null);
  918. if (editor.inline) {
  919. return get(fullscreenState);
  920. }
  921. register(editor, fullscreenState);
  922. register$1(editor, fullscreenState);
  923. editor.addShortcut('Meta+Shift+F', '', 'mceFullScreen');
  924. return get(fullscreenState);
  925. });
  926. }
  927. Plugin();
  928. }());