UNPKG

@tinymce/tinymce-angular

Version:
75 lines 13.7 kB
/** * Copyright (c) 2017-present, Ephox, Inc. * * This source code is licensed under the Apache 2 license found in the * LICENSE file in the root directory of this source tree. * */ import { fromEvent, takeUntil } from 'rxjs'; import { validEvents } from '../editor/Events'; // Caretaker note: `fromEvent` supports passing JQuery-style event targets, the editor has `on` and `off` methods which // will be invoked upon subscription and teardown. const listenTinyMCEEvent = (editor, eventName, destroy$) => fromEvent(editor, eventName).pipe(takeUntil(destroy$)); const bindHandlers = (ctx, editor, destroy$) => { const allowedEvents = getValidEvents(ctx); allowedEvents.forEach((eventName) => { const eventEmitter = ctx[eventName]; listenTinyMCEEvent(editor, eventName.substring(2), destroy$).subscribe((event) => { // Caretaker note: `ngZone.run()` runs change detection since it notifies the forked Angular zone that it's // being re-entered. We don't want to run `ApplicationRef.tick()` if anyone listens to the specific event // within the template. E.g. if the `onSelectionChange` is not listened within the template like: // `<editor (onSelectionChange)="..."></editor>` // then it won't be "observed", and we won't run "dead" change detection. if (isObserved(eventEmitter)) { ctx.ngZone.run(() => eventEmitter.emit({ event, editor })); } }); }); }; const getValidEvents = (ctx) => { const ignoredEvents = parseStringProperty(ctx.ignoreEvents, []); const allowedEvents = parseStringProperty(ctx.allowedEvents, validEvents).filter((event) => validEvents.includes(event) && !ignoredEvents.includes(event)); return allowedEvents; }; const parseStringProperty = (property, defaultValue) => { if (typeof property === 'string') { return property.split(',').map((value) => value.trim()); } if (Array.isArray(property)) { return property; } return defaultValue; }; let unique = 0; const uuid = (prefix) => { const date = new Date(); const time = date.getTime(); const random = Math.floor(Math.random() * 1000000000); unique++; return prefix + '_' + random + unique + String(time); }; const isTextarea = (element) => typeof element !== 'undefined' && element.tagName.toLowerCase() === 'textarea'; const normalizePluginArray = (plugins) => { if (typeof plugins === 'undefined' || plugins === '') { return []; } return Array.isArray(plugins) ? plugins : plugins.split(' '); }; const mergePlugins = (initPlugins, inputPlugins) => normalizePluginArray(initPlugins).concat(normalizePluginArray(inputPlugins)); // eslint-disable-next-line @typescript-eslint/no-empty-function const noop = () => { }; const isNullOrUndefined = (value) => value === null || value === undefined; const isObserved = (o) => // RXJS is making the `observers` property internal in v8. So this is intended as a backwards compatible way of // checking if a subject has observers. o.observed || o.observers?.length > 0; const setMode = (editor, mode) => { if (typeof editor.mode?.set === 'function') { editor.mode.set(mode); } else if ('setMode' in editor && typeof editor.setMode === 'function') { editor.setMode(mode); } }; export { listenTinyMCEEvent, bindHandlers, uuid, isTextarea, normalizePluginArray, mergePlugins, noop, isNullOrUndefined, setMode }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Utils.js","sourceRoot":"","sources":["../../../../tinymce-angular-component/src/main/ts/utils/Utils.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Copyright (c) 2017-present, Ephox, Inc.\n *\n * This source code is licensed under the Apache 2 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { EventEmitter } from '@angular/core';\nimport { fromEvent, Subject, takeUntil } from 'rxjs';\nimport { HasEventTargetAddRemove } from 'rxjs/internal/observable/fromEvent';\n\nimport { EditorComponent } from '../editor/editor.component';\nimport { validEvents, Events } from '../editor/Events';\nimport { Editor } from 'tinymce';\n\n// Caretaker note: `fromEvent` supports passing JQuery-style event targets, the editor has `on` and `off` methods which\n// will be invoked upon subscription and teardown.\nconst listenTinyMCEEvent = (\n  editor: any,\n  eventName: string,\n  destroy$: Subject<void>\n) => fromEvent(editor as HasEventTargetAddRemove<unknown> | ArrayLike<HasEventTargetAddRemove<unknown>>, eventName).pipe(takeUntil(destroy$));\n\nconst bindHandlers = (ctx: EditorComponent, editor: any, destroy$: Subject<void>): void => {\n  const allowedEvents = getValidEvents(ctx);\n  allowedEvents.forEach((eventName) => {\n    const eventEmitter: EventEmitter<any> = ctx[eventName];\n\n    listenTinyMCEEvent(editor, eventName.substring(2), destroy$).subscribe((event) => {\n      // Caretaker note: `ngZone.run()` runs change detection since it notifies the forked Angular zone that it's\n      // being re-entered. We don't want to run `ApplicationRef.tick()` if anyone listens to the specific event\n      // within the template. E.g. if the `onSelectionChange` is not listened within the template like:\n      // `<editor (onSelectionChange)=\"...\"></editor>`\n      // then it won't be \"observed\", and we won't run \"dead\" change detection.\n      if (isObserved(eventEmitter)) {\n        ctx.ngZone.run(() => eventEmitter.emit({ event, editor }));\n      }\n    });\n  });\n};\n\nconst getValidEvents = (ctx: EditorComponent): (keyof Events)[] => {\n  const ignoredEvents = parseStringProperty(ctx.ignoreEvents, []);\n  const allowedEvents = parseStringProperty(ctx.allowedEvents, validEvents).filter(\n    (event) => validEvents.includes(event as (keyof Events)) && !ignoredEvents.includes(event)) as (keyof Events)[];\n  return allowedEvents;\n};\n\nconst parseStringProperty = (property: string | string[] | undefined, defaultValue: (keyof Events)[]): string[] => {\n  if (typeof property === 'string') {\n    return property.split(',').map((value) => value.trim());\n  }\n  if (Array.isArray(property)) {\n    return property;\n  }\n  return defaultValue;\n};\n\nlet unique = 0;\n\nconst uuid = (prefix: string): string => {\n  const date = new Date();\n  const time = date.getTime();\n  const random = Math.floor(Math.random() * 1000000000);\n\n  unique++;\n\n  return prefix + '_' + random + unique + String(time);\n};\n\nconst isTextarea = (element?: Element): element is HTMLTextAreaElement => typeof element !== 'undefined' && element.tagName.toLowerCase() === 'textarea';\n\nconst normalizePluginArray = (plugins?: string | string[]): string[] => {\n  if (typeof plugins === 'undefined' || plugins === '') {\n    return [];\n  }\n\n  return Array.isArray(plugins) ? plugins : plugins.split(' ');\n};\n\nconst mergePlugins = (initPlugins: string | string[], inputPlugins?: string | string[]) =>\n  normalizePluginArray(initPlugins).concat(normalizePluginArray(inputPlugins));\n\n// eslint-disable-next-line @typescript-eslint/no-empty-function\nconst noop: (...args: any[]) => void = () => { };\n\nconst isNullOrUndefined = (value: any): value is null | undefined => value === null || value === undefined;\n\nconst isObserved = (o: Subject<unknown>): boolean =>\n  // RXJS is making the `observers` property internal in v8. So this is intended as a backwards compatible way of\n  // checking if a subject has observers.\n  o.observed || o.observers?.length > 0;\n\nconst setMode = (editor: Editor, mode: 'readonly' | 'design') => {\n  if (typeof editor.mode?.set === 'function') {\n    editor.mode.set(mode);\n  } else if ('setMode' in editor && typeof editor.setMode === 'function') {\n    editor.setMode(mode);\n  }\n};\n\nexport {\n  listenTinyMCEEvent,\n  bindHandlers,\n  uuid,\n  isTextarea,\n  normalizePluginArray,\n  mergePlugins,\n  noop,\n  isNullOrUndefined,\n  setMode\n};\n"]}