UNPKG

ngx-quill-upload

Version:

A module for images and videos to be uploaded to a server instead of being base64 encoded, in ngx-quill from toolbar editor.

122 lines 20.3 kB
import Styled from './Styled'; import { Constants, Helpers } from '../utils'; class BaseHandler { constructor(quill, options) { this.helpers = new Helpers(); this.quill = quill; this.options = options; this.range = null; new Styled().apply(); if (this.isNotExistLoading()) { const node = document.createElement('div'); node.innerHTML = this.helpers.loadingHTML(); this.quill.container.appendChild(node); } if (typeof this.options.upload !== 'function') { console.warn('[Missing config] upload function that returns a promise is required'); } setTimeout(() => { if (!this.options.accepts) { if (this.handler === Constants.blots.image) { this.options.accepts = ['jpg', 'jpeg', 'png']; } if (this.handler === Constants.blots.video) { this.options.accepts = ['mp4', 'webm']; } } if (this.handler === Constants.blots.image) { this.possibleExtension = new Set(['apng', 'bmp', 'gif', 'ico', 'cur', 'jpg', 'jpeg', 'jfif', 'pjpeg', 'pjp', 'png', 'svg', 'tif', 'tiff', 'webp']); } if (this.handler === Constants.blots.video) { this.possibleExtension = new Set(['mp4', 'webm', '3gp', 'mp4', 'mpeg', 'quickTime', 'ogg']); } this.allowedFormatRegex = new RegExp('^(' + this.options.accepts.filter((el) => this.possibleExtension.has(el.toLowerCase())) .reduce((acc, el, i) => acc.concat(i !== 0 ? `|${el}` : `${el}`), '') + ')$', 'i'); }, 1); } applyForToolbar() { const toolbar = this.quill.getModule('toolbar'); this.loading = document.getElementById(`${Constants.ID_SPLIT_FLAG}.QUILL-LOADING`); toolbar.addHandler(this.handler, this.selectLocalFile.bind(this)); } selectLocalFile() { this.range = this.quill.getSelection(); this.fileHolder = document.createElement('input'); this.fileHolder.setAttribute('type', 'file'); this.fileHolder.setAttribute('accept', `${this.handler}/*`); this.fileHolder.onchange = this.fileChanged.bind(this); this.fileHolder.click(); } loadFile(context) { this.loading.removeAttribute('class'); this.loading.setAttribute('class', Constants.LOADING_CLASS_NAME); const file = context.fileHolder.files[0]; this.handlerId = this.helpers.generateID(); const fileReader = new FileReader(); fileReader.addEventListener('load', () => { this.insertBase64Data(fileReader.result, this.handlerId); }, false); if (!file) { console.warn('[File not found] Something was wrong, please try again!!'); return null; } fileReader.readAsDataURL(file); return { file, handlerId: this.handlerId }; } fileChanged() { const { file, handlerId } = this.loadFile(this); if (!file) { return; } const extension = file.name.split('.').pop(); if (!this.isValidExtension(extension)) { console.warn('[Wrong Format] Format was wrong, please try with correct format!!'); } if (!this.hasValidMimeType(file.type)) { console.warn(`[Incorrect Mime Type] The MIME Type of uploaded file is not ${this.handler}!!`); } this.embedFile(file, handlerId); } embedFile(file, handlerId) { this.options.upload(file).then((url) => { this.insertFileToEditor(url, handlerId); this.loading.removeAttribute('class'); this.loading.setAttribute('class', Constants.NONE_DISPLAY_CLASS_NAME); }, (error) => { this.loading.removeAttribute('class'); this.loading.setAttribute('class', Constants.NONE_DISPLAY_CLASS_NAME); setTimeout(() => { const el = document.getElementById(handlerId); el.remove(); }, 1000); }); } insertBase64Data(url, handlerId) { const range = this.range; this.quill.insertEmbed(range.index, this.handler, `${handlerId}${Constants.ID_SPLIT_FLAG}${url}`); const el = document.getElementById(handlerId); if (el) { el.setAttribute('class', Constants.QUILL_UPLOAD_HOLDER_CLASS_NAME); } } insertFileToEditor(url, handlerId) { const el = document.getElementById(handlerId); if (el) { el.setAttribute('src', url); el.removeAttribute('id'); el.removeAttribute('class'); } } isValidExtension(extension) { return extension && this.allowedFormatRegex.test(extension); } hasValidMimeType(type) { return type && type.startsWith(this.handler); } isNotExistLoading() { const loading = document.getElementById(`${Constants.ID_SPLIT_FLAG}.QUILL-LOADING`); return loading == null; } } export default BaseHandler; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BaseHandler.js","sourceRoot":"","sources":["../../../../src/handlers/BaseHandler.ts"],"names":[],"mappings":"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","sourcesContent":["import Styled from './Styled';\r\nimport { Constants, Helpers } from '../utils';\r\n\r\ninterface Range {\r\n  index: number;\r\n  length: number;\r\n}\r\n\r\nexport interface Options {\r\n  accepts: string[];\r\n  upload(file: File): Promise<string>;\r\n}\r\n\r\nclass BaseHandler {\r\n  quill: any;\r\n  options: Options;\r\n  range: Range | null;\r\n  handler: string;\r\n  loading: HTMLElement;\r\n  fileHolder: HTMLInputElement;\r\n  handlerId: string;\r\n  helpers = new Helpers();\r\n  allowedFormatRegex: RegExp;\r\n  possibleExtension: Set<string>;\r\n\r\n  constructor(quill, options: Options) {\r\n    this.quill = quill;\r\n    this.options = options;\r\n    this.range = null;\r\n\r\n    new Styled().apply();\r\n\r\n    if (this.isNotExistLoading()) {\r\n      const node = document.createElement('div');\r\n      node.innerHTML = this.helpers.loadingHTML();\r\n\r\n      this.quill.container.appendChild(node);\r\n    }\r\n\r\n    if (typeof this.options.upload !== 'function') {\r\n      console.warn('[Missing config] upload function that returns a promise is required');\r\n    }\r\n\r\n    setTimeout(() => {\r\n      if (!this.options.accepts) {\r\n        if (this.handler === Constants.blots.image) {\r\n          this.options.accepts = ['jpg', 'jpeg', 'png'];\r\n        }\r\n        if (this.handler === Constants.blots.video) {\r\n          this.options.accepts = ['mp4', 'webm'];\r\n        }\r\n      }\r\n\r\n      if (this.handler === Constants.blots.image) {\r\n        this.possibleExtension = new Set(['apng', 'bmp', 'gif', 'ico', 'cur', 'jpg', 'jpeg', 'jfif', 'pjpeg', 'pjp', 'png', 'svg', 'tif', 'tiff', 'webp']);\r\n      }\r\n      if (this.handler === Constants.blots.video) {\r\n        this.possibleExtension = new Set(['mp4', 'webm', '3gp', 'mp4', 'mpeg', 'quickTime', 'ogg']);\r\n      }\r\n\r\n      this.allowedFormatRegex = new RegExp('^(' + this.options.accepts.filter((el) => this.possibleExtension.has(el.toLowerCase()))\r\n      .reduce((acc, el, i) => acc.concat(i !== 0 ? `|${el}` : `${el}`), '') + ')$', 'i');\r\n    }, 1);\r\n  }\r\n\r\n  applyForToolbar() {\r\n    const toolbar = this.quill.getModule('toolbar');\r\n    this.loading = document.getElementById(\r\n      `${Constants.ID_SPLIT_FLAG}.QUILL-LOADING`\r\n    );\r\n    toolbar.addHandler(this.handler, this.selectLocalFile.bind(this));\r\n  }\r\n\r\n  selectLocalFile() {\r\n    this.range = this.quill.getSelection();\r\n    this.fileHolder = document.createElement('input');\r\n    this.fileHolder.setAttribute('type', 'file');\r\n    this.fileHolder.setAttribute('accept', `${this.handler}/*`);\r\n    this.fileHolder.onchange = this.fileChanged.bind(this);\r\n    this.fileHolder.click();\r\n  }\r\n\r\n  loadFile(context) {\r\n    this.loading.removeAttribute('class');\r\n    this.loading.setAttribute('class', Constants.LOADING_CLASS_NAME);\r\n\r\n    const file = context.fileHolder.files[0];\r\n    this.handlerId = this.helpers.generateID();\r\n\r\n    const fileReader = new FileReader();\r\n    fileReader.addEventListener('load', () => {\r\n      this.insertBase64Data(fileReader.result, this.handlerId);\r\n    }, false);\r\n\r\n    if (!file) {\r\n      console.warn('[File not found] Something was wrong, please try again!!');\r\n      return null;\r\n    }\r\n\r\n    fileReader.readAsDataURL(file);\r\n\r\n    return { file, handlerId: this.handlerId };\r\n  }\r\n\r\n  fileChanged() {\r\n    const { file, handlerId } = this.loadFile(this);\r\n\r\n    if (!file) {\r\n      return;\r\n    }\r\n\r\n    const extension = file.name.split('.').pop();\r\n\r\n    if (!this.isValidExtension(extension)) {\r\n      console.warn(\r\n        '[Wrong Format] Format was wrong, please try with correct format!!'\r\n      );\r\n    }\r\n\r\n    if (!this.hasValidMimeType(file.type)) {\r\n      console.warn(\r\n        `[Incorrect Mime Type] The MIME Type of uploaded file is not ${this.handler}!!`\r\n      );\r\n    }\r\n\r\n    this.embedFile(file, handlerId);\r\n  }\r\n\r\n  embedFile(file: File, handlerId: string) {\r\n    this.options.upload(file).then(\r\n      (url) => {\r\n        this.insertFileToEditor(url, handlerId);\r\n        this.loading.removeAttribute('class');\r\n        this.loading.setAttribute('class', Constants.NONE_DISPLAY_CLASS_NAME);\r\n      },\r\n      (error) => {\r\n        this.loading.removeAttribute('class');\r\n        this.loading.setAttribute('class', Constants.NONE_DISPLAY_CLASS_NAME);\r\n        setTimeout(() => {\r\n          const el = document.getElementById(handlerId);\r\n          el.remove();\r\n        }, 1000);\r\n      }\r\n    );\r\n  }\r\n\r\n  insertBase64Data(url: string | ArrayBuffer, handlerId: string) {\r\n    const range = this.range;\r\n    this.quill.insertEmbed(\r\n      range.index,\r\n      this.handler,\r\n      `${handlerId}${Constants.ID_SPLIT_FLAG}${url}`\r\n    );\r\n\r\n    const el = document.getElementById(handlerId);\r\n\r\n    if (el) {\r\n      el.setAttribute('class', Constants.QUILL_UPLOAD_HOLDER_CLASS_NAME);\r\n    }\r\n  }\r\n\r\n  insertFileToEditor(url: string, handlerId: string) {\r\n    const el = document.getElementById(handlerId);\r\n    if (el) {\r\n      el.setAttribute('src', url);\r\n      el.removeAttribute('id');\r\n      el.removeAttribute('class');\r\n    }\r\n  }\r\n\r\n  isValidExtension(extension: string) {\r\n    return extension && this.allowedFormatRegex.test(extension);\r\n  }\r\n\r\n  hasValidMimeType(type: string) {\r\n    return type && type.startsWith(this.handler);\r\n  }\r\n\r\n  isNotExistLoading() {\r\n    const loading = document.getElementById(\r\n      `${Constants.ID_SPLIT_FLAG}.QUILL-LOADING`\r\n    );\r\n\r\n    return loading == null;\r\n  }\r\n}\r\n\r\nexport default BaseHandler;\r\n"]}