UNPKG

@freshworks/crayons

Version:
454 lines (449 loc) 41.6 kB
import { attachShadow, createEvent, h, proxyCustomElement } from '@stencil/core/internal/client'; import { T as TranslationController, i as i18n } from './Translation.js'; import { d as defineCustomElement$6 } from './file-uploader-file.js'; import { d as defineCustomElement$5 } from './file-uploader-progress.js'; import { d as defineCustomElement$2, a as defineCustomElement$4 } from './icon.js'; import { d as defineCustomElement$3 } from './spinner.js'; const fileUploaderCss = ":host{font-family:var(--fw-font-family, -apple-system, blinkmacsystemfont, \"Segoe UI\", roboto, oxygen, ubuntu, cantarell, \"Open Sans\", \"Helvetica Neue\", sans-serif);-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;-webkit-box-sizing:border-box;box-sizing:border-box}:host{display:block}div.file-uploader-container{display:-ms-flexbox;display:flex;width:100%;min-height:153px;border:1px dashed var(--fw-file-uploader-border, #bbdcfe);background:#fff;-ms-flex-pack:center;justify-content:center}div.file-uploader-container .dropzone,div.file-uploader-container .progress,div.file-uploader-container .files{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;width:100%;-ms-flex-pack:center;justify-content:center}div.file-uploader-container .dropzone{-ms-flex-align:center;align-items:center;cursor:pointer}div.file-uploader-container .dropzone .dropzone-center{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;width:500px;height:100%;-ms-flex-pack:justify;justify-content:space-between}div.file-uploader-container .dropzone .dropzone-center .drop-clickable{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;margin:auto 0;-webkit-transform:translateY(-5px);transform:translateY(-5px)}div.file-uploader-container .dropzone .dropzone-center .drop-clickable .drop-clickable-text{line-height:20px;font-size:14px;font-weight:500;color:#2c5cc5;margin-top:9px;margin-bottom:3px}div.file-uploader-container .dropzone .dropzone-center .drop-clickable .drop-clickable-hint{line-height:20px;font-size:14px;color:#92a2b1}div.file-uploader-container .dropzone .dropzone-center .dropzone-hint{line-height:20px;font-size:10px;color:#345c7c;text-align:center}div.file-uploader-container .dropzone .dropzone-center .dropzone-error{line-height:12px;font-size:10px;text-align:center;padding:5px 0px;-webkit-box-sizing:border-box;box-sizing:border-box}div.file-uploader-container .dropzone .dropzone-center .dropzone-error span{display:block;color:#d72d30}div.file-uploader-container .progress,div.file-uploader-container .files{padding:28px;-webkit-box-sizing:border-box;box-sizing:border-box;-ms-flex-pack:center;justify-content:center}div.file-uploader-container .progress-center,div.file-uploader-container .files-center{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}div.file-uploader-container .progress-title,div.file-uploader-container .files-title{line-height:20px;font-size:12px;color:#475867;font-weight:600;letter-spacing:0.2px}"; var __decorate = (undefined && undefined.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; let fileCount = 1; let FileUploader = class extends HTMLElement { constructor() { super(); this.__registerHost(); attachShadow(this); this.fwFilesUploaded = createEvent(this, "fwFilesUploaded", 7); this.fwFileReuploaded = createEvent(this, "fwFileReuploaded", 7); this.fwStageChanged = createEvent(this, "fwStageChanged", 7); /** * stage - different stages in file uploader. */ this.stage = 'dropzone'; /** * hint - file uploader hint text. */ this.hint = ''; /** * accept - comma separated string. tells us what file formats file uploader should accept. */ this.accept = ''; /** * maxFileSize - maximum file size the file uploader must accept. */ this.maxFileSize = 0; /** * actionURL - URL to make server call. */ this.actionURL = ''; /** * actionParams - additional information to send to server other than the file. */ this.actionParams = {}; /** * multiple - upload multiple files. */ this.multiple = false; /** * Max files allowed to upload. */ this.filesLimit = 10; /** * modify request * @param xhr * @returns xhr */ this.modifyRequest = (xhr) => xhr; /** * files - files collection. */ this.files = []; /** * errors - errors collection. */ this.errors = []; /** * private * fileInputElement */ this.fileInputElement = null; /** * private * isFileUploadInProgress */ this.isFileUploadInProgress = false; /** * private * fileUploadPromises */ this.fileUploadPromises = []; /** * private * formDataCollection */ this.formDataCollection = {}; } stageChange(newStage) { switch (newStage) { case 'dropzone': this.formDataCollection = {}; this.fileUploadPromises = []; this.errors = []; this.files = []; break; } this.fwStageChanged.emit({ stage: newStage }); } /** * private * uploadFileLocally - upload the files locally and add it to form for sending to server * @param file */ uploadFileLocally(file) { const formData = new FormData(); formData.append('file', file); this.formDataCollection[fileCount] = formData; this.files.push({ id: fileCount, name: file.name, progress: 0, error: '', }); fileCount = fileCount + 1; } /** * uploadFile * @param fileId * @returns fileUploadPromise */ uploadFile(fileId) { const formData = this.formDataCollection[fileId]; // adding extra information to formData before uploading for (const key in this.actionParams) { if (Object.prototype.hasOwnProperty.call(this.actionParams, key)) { formData.append(key, this.actionParams[key]); } } // creating and sending xhr requests const xhr = new XMLHttpRequest(); xhr.upload.addEventListener('progress', this.progressHandler.bind(this, fileId), false); const fileUploadPromise = new Promise((resolve, reject) => { xhr.onreadystatechange = () => { if (xhr.readyState === 4) { if (xhr.status === 200) { resolve({ uploadStatus: xhr.status, response: xhr.response }); } else { this.setFile(fileId, { error: this.fileUploadError }); reject({ uploadStatus: xhr.status, response: xhr.response }); } } }; }); xhr.open('POST', this.actionURL); const modifiedRequest = this.modifyRequest(xhr); modifiedRequest.send(formData); return fileUploadPromise; } /** * private * retryFileUpload retry a file upload * @param fileId */ retryFileUpload(fileId) { this.setFile(fileId, { error: '' }); const uploadPromise = this.uploadFile(fileId); this.fileUploadPromises = [uploadPromise]; Promise.allSettled(this.fileUploadPromises).then((responses) => { this.fwFileReuploaded.emit(responses[0].value); }); } /** * uploadFiles - uploads the files to the server. emits an after file is uploaded. */ async uploadFiles() { if (this.files.length && !this.isFileUploadInProgress) { this.stage = 'progress'; this.isFileUploadInProgress = true; for (const fileId in this.formDataCollection) { if (Object.prototype.hasOwnProperty.call(this.formDataCollection, fileId)) { const uploadPromise = this.uploadFile(parseInt(fileId)); this.fileUploadPromises.push(uploadPromise); } } Promise.allSettled(this.fileUploadPromises).then((responses) => { const responseValues = responses.map((response) => response.value); const responseValue = this.multiple ? responseValues : responseValues[0]; this.fwFilesUploaded.emit(responseValue); this.isFileUploadInProgress = false; }); } } /** * private * removeFile - remove a file from the form and files collection. * @param fileId */ removeFile(fileId) { const fileIndex = this.files.findIndex((file) => file.id === fileId); if (fileIndex >= 0) { const beforeFiles = this.files.slice(0, fileIndex); const afterFiles = this.files.slice(fileIndex + 1, this.files.length + 1); this.files = [...beforeFiles, ...afterFiles]; delete this.formDataCollection[fileId]; if (!this.files.length) { this.stage = 'dropzone'; } } } /** * private * fileValidation validate a file for upload * @param file * @returns */ fileValidation(file) { let isPassed = true; const fileExtension = file.name; const fileSize = file.size; const errors = []; if (this.accept) { isPassed = this.accept .split(',') .filter((fileType) => fileType !== '') .some((fileType) => fileExtension.includes(fileType.trim())); if (!isPassed) { errors.push(this.acceptError); } } if (this.maxFileSize !== 0) { if (fileSize > this.maxFileSize * 1024 * 1024) { isPassed = false; errors.push(this.maxFileSizeError); } } this.errors = [...this.errors, ...errors]; return isPassed; } /** * private * setFile - update the file object in files collection. */ setFile(fileId, errorObject) { let change; const fileIndex = this.files.findIndex((file) => file.id === fileId); if (fileIndex >= 0) { this.files = [ ...this.files.slice(0, fileIndex), Object.assign(this.files[fileIndex], errorObject), ...this.files.slice(fileIndex + 1, this.files.length), ]; change = true; } else { change = false; } return change; } /** * private * drag and drop handler * @param event */ dropHandler(event) { event.preventDefault(); this.fileHandler(event); } /** * private * fileHandler - handler for both drop and input change * @param event */ fileHandler(event) { let passed = true; const tempFiles = event.target.files || event.dataTransfer.files; const files = this.multiple ? tempFiles : [tempFiles[0]]; this.errors = []; if (files.length <= this.filesLimit) { for (let index = 0; index < files.length; index++) { const file = files[index]; passed = this.fileValidation(file); if (!passed) { break; } } } else { this.errors = [this.maxFilesLimitError]; passed = false; } if (passed) { for (let index = 0; index < files.length; index++) { const file = files[index]; this.uploadFileLocally(file); this.stage = 'files'; } } } /** * private * progressHandler - update the progress on files * @param fileId * @param event */ progressHandler(fileId, event) { const fileIndex = this.files.findIndex((file) => fileId === file.id); if (fileIndex >= 0) { const progressPercentage = (event.loaded / event.total) * 100; const file = Object.assign(Object.assign({}, this.files[fileIndex]), { progress: progressPercentage }); const beforeFiles = this.files.slice(0, fileIndex); const afterFiles = this.files.slice(fileIndex + 1, this.files.length + 1); this.files = [...beforeFiles, file, ...afterFiles]; } } /** * renderFileUploader * @returns {JSX.Element} */ renderFileUploader() { let template = null; switch (this.stage) { case 'dropzone': template = this.renderDropzone(); break; case 'progress': template = this.renderProgress(); break; case 'files': template = this.renderFiles(); break; } return template; } /** * renderDropzone * @returns {JSX.Element} */ renderDropzone() { const multipleFiles = this.multiple ? { multiple: true } : {}; return (h("div", { class: 'dropzone', key: 'dropzone', tabIndex: 0, onDrop: (event) => this.dropHandler(event), onDragOver: (event) => event.preventDefault(), onClick: () => this.fileInputElement.click(), onKeyUp: (event) => { if (event.key === 'Enter' || event.key === 'Space') { this.fileInputElement.click(); } }, role: 'button' }, h("div", { class: 'dropzone-center' }, h("div", { class: 'drop-clickable' }, h("div", { class: 'drop-clickable-icon' }, h("svg", { width: '32', height: '32', viewBox: '0 0 32 32', fill: 'none', xmlns: 'http://www.w3.org/2000/svg' }, h("rect", { width: '32', height: '32', fill: 'url(#pattern0)' }), h("defs", null, h("pattern", { id: 'pattern0', patternContentUnits: 'objectBoundingBox', width: '1', height: '1' }, h("use", { xlinkHref: '#image0_1441_50512', transform: 'scale(0.00195312)' })), h("image", { id: 'image0_1441_50512', width: '512', height: '512', xlinkHref: '' })))), h("div", { class: 'drop-clickable-text' }, h("input", Object.assign({ type: 'file', hidden: true }, multipleFiles, { style: { display: 'none' }, onChange: (ev) => this.fileHandler(ev), ref: (el) => (this.fileInputElement = el) })), this.text), h("div", { class: 'drop-clickable-hint' }, h("span", null, this.description))), this.errors.length ? (h("div", { class: 'dropzone-error' }, this.errors.map((message) => (h("span", null, message))))) : (h("div", { class: 'dropzone-hint' }, this.hint))))); } /** * renderProgress * @returns {JSX.Element} */ renderProgress() { return (h("div", { class: 'progress', key: 'progress' }, h("div", { class: 'progress-center' }, h("div", { class: 'progress-title' }, TranslationController.t('fileUploader.uploading')), this.files.map((file) => (h("fw-file-uploader-progress", { fileId: file.id, fileName: file.name, progress: file.progress, error: file.error, onFwRetryUpload: (event) => this.retryFileUpload(event.detail.fileId) })))))); } /** * renderFiles * @returns {JSX.Element} */ renderFiles() { return (h("div", { class: 'files', key: 'files' }, h("div", { class: 'files-center' }, h("div", { class: 'files-title' }, TranslationController.t('fileUploader.selectedFiles')), this.files.map((file) => (h("fw-file-uploader-file", { fileId: file.id, name: file.name, onFwRemoveFile: (event) => { event.stopPropagation(); this.removeFile(event.detail.fileId); } })))))); } /** * render * @returns {JSX.Element} */ render() { return (h("div", { class: 'file-uploader-container' }, this.renderFileUploader())); } static get watchers() { return { "stage": ["stageChange"] }; } static get style() { return fileUploaderCss; } }; __decorate([ i18n({ keyName: 'fileUploader.text' }) ], FileUploader.prototype, "text", void 0); __decorate([ i18n({ keyName: 'fileUploader.description' }) ], FileUploader.prototype, "description", void 0); __decorate([ i18n({ keyName: 'fileUploader.acceptError' }) ], FileUploader.prototype, "acceptError", void 0); __decorate([ i18n({ keyName: 'fileUploader.maxFileSizeError' }) ], FileUploader.prototype, "maxFileSizeError", void 0); __decorate([ i18n({ keyName: 'fileUploader.maxFilesLimitError' }) ], FileUploader.prototype, "maxFilesLimitError", void 0); __decorate([ i18n({ keyName: 'fileUploader.fileUploadError' }) ], FileUploader.prototype, "fileUploadError", void 0); FileUploader = /*@__PURE__*/ proxyCustomElement(FileUploader, [1, "fw-file-uploader", { "text": [1032], "description": [1032], "hint": [1], "accept": [1], "maxFileSize": [2, "max-file-size"], "acceptError": [1032, "accept-error"], "maxFileSizeError": [1032, "max-file-size-error"], "maxFilesLimitError": [1032, "max-files-limit-error"], "fileUploadError": [1032, "file-upload-error"], "actionURL": [1, "action-u-r-l"], "actionParams": [8, "action-params"], "multiple": [4], "filesLimit": [2, "files-limit"], "modifyRequest": [16], "stage": [32], "files": [32], "errors": [32], "uploadFiles": [64] }]); function defineCustomElement$1() { const components = ["fw-file-uploader", "fw-file-uploader-file", "fw-file-uploader-progress", "fw-icon", "fw-spinner", "fw-toast-message"]; components.forEach(tagName => { switch (tagName) { case "fw-file-uploader": if (!customElements.get(tagName)) { customElements.define(tagName, FileUploader); } break; case "fw-file-uploader-file": if (!customElements.get(tagName)) { defineCustomElement$6(); } break; case "fw-file-uploader-progress": if (!customElements.get(tagName)) { defineCustomElement$5(); } break; case "fw-icon": if (!customElements.get(tagName)) { defineCustomElement$4(); } break; case "fw-spinner": if (!customElements.get(tagName)) { defineCustomElement$3(); } break; case "fw-toast-message": if (!customElements.get(tagName)) { defineCustomElement$2(); } break; } }); } const FwFileUploader = FileUploader; const defineCustomElement = defineCustomElement$1; export { FwFileUploader, defineCustomElement };