@freshworks/crayons
Version:
Crayons Web Components library
454 lines (449 loc) • 41.6 kB
JavaScript
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 };