@alauda-fe/common
Version:
Alauda frontend team common codes.
116 lines • 14.1 kB
JavaScript
import { NotificationService, coerceAttrBoolean } from '@alauda/ui';
import { ChangeDetectionStrategy, Component, Input, forwardRef, HostListener, ViewChild, ElementRef, } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import { isErrorMessage } from '../../core/public-api';
import { EffectDirectiveModule } from '../../effect-directive/public-api';
import { TranslateService } from '../../translate/public-api';
import * as i0 from "@angular/core";
import * as i1 from "@alauda/ui";
import * as i2 from "../../translate/public-api";
const _c0 = ["fileInput"];
const _c1 = ["*"];
export const FileResultType = {
ARRAY_BUFFER: 'ArrayBuffer',
BINARY_STRING: 'BinaryString',
DATA_URL: 'DataURL',
TEXT: 'Text',
};
export class UploadFileComponent {
constructor(notification, translate) {
this.notification = notification;
this.translate = translate;
this.resultType = FileResultType.TEXT;
}
onClick() {
if (coerceAttrBoolean(this.disabled)) {
return;
}
this.fileInput.nativeElement.click();
}
async onChange(e) {
const el = e.target;
try {
const results = await Promise.all(Array.from(el.files).map(file => new Promise((resolve, reject) => {
const fileReader = new FileReader();
fileReader.addEventListener('load', () => {
const { result } = fileReader;
if (this.resultType === FileResultType.TEXT &&
result.includes('\uFFFD')) {
reject(new Error(this.translate.get('text_file_type_error')));
return;
}
resolve(result);
});
fileReader[`readAs${this.resultType}`](file);
})));
this.valueChange(this.multiple ? results : results[0]);
}
catch (e) {
if (isErrorMessage(e)) {
this.notification.error(e.message);
}
}
el.value = null;
}
writeValue() {
//
}
registerOnChange(fn) {
this.valueChange = fn;
}
registerOnTouched() {
//
}
static { this.ɵfac = function UploadFileComponent_Factory(t) { return new (t || UploadFileComponent)(i0.ɵɵdirectiveInject(i1.NotificationService), i0.ɵɵdirectiveInject(i2.TranslateService)); }; }
static { this.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: UploadFileComponent, selectors: [["acl-upload-file"]], viewQuery: function UploadFileComponent_Query(rf, ctx) { if (rf & 1) {
i0.ɵɵviewQuery(_c0, 5);
} if (rf & 2) {
let _t;
i0.ɵɵqueryRefresh(_t = i0.ɵɵloadQuery()) && (ctx.fileInput = _t.first);
} }, hostBindings: function UploadFileComponent_HostBindings(rf, ctx) { if (rf & 1) {
i0.ɵɵlistener("click", function UploadFileComponent_click_HostBindingHandler() { return ctx.onClick(); });
} }, inputs: { disabled: "disabled", accept: "accept", multiple: "multiple", resultType: "resultType" }, standalone: true, features: [i0.ɵɵProvidersFeature([
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => UploadFileComponent),
multi: true,
},
]), i0.ɵɵStandaloneFeature], ngContentSelectors: _c1, decls: 3, vars: 2, consts: [["fileInput", ""], ["type", "file", 1, "file-input", 3, "change", "multiple"]], template: function UploadFileComponent_Template(rf, ctx) { if (rf & 1) {
const _r1 = i0.ɵɵgetCurrentView();
i0.ɵɵprojectionDef();
i0.ɵɵprojection(0);
i0.ɵɵelementStart(1, "input", 1, 0);
i0.ɵɵlistener("change", function UploadFileComponent_Template_input_change_1_listener($event) { i0.ɵɵrestoreView(_r1); return i0.ɵɵresetView(ctx.onChange($event)); });
i0.ɵɵelementEnd();
} if (rf & 2) {
i0.ɵɵadvance();
i0.ɵɵproperty("multiple", ctx.multiple);
i0.ɵɵattribute("accept", ctx.accept);
} }, dependencies: [EffectDirectiveModule], styles: ["[_nghost-%COMP%]{position:relative;display:inline-flex;overflow:hidden}.file-input[_ngcontent-%COMP%]{position:absolute;top:0;right:0;bottom:0;left:0;opacity:0;z-index:-1}"], changeDetection: 0 }); }
}
(() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(UploadFileComponent, [{
type: Component,
args: [{ selector: 'acl-upload-file', preserveWhitespaces: false, changeDetection: ChangeDetectionStrategy.OnPush, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => UploadFileComponent),
multi: true,
},
], standalone: true, imports: [EffectDirectiveModule], template: "<ng-content></ng-content>\n<input\n #fileInput\n class=\"file-input\"\n type=\"file\"\n [attr.accept]=\"accept\"\n [multiple]=\"multiple\"\n (change)=\"onChange($event)\"\n/>\n", styles: [":host{position:relative;display:inline-flex;overflow:hidden}.file-input{position:absolute;top:0;right:0;bottom:0;left:0;opacity:0;z-index:-1}\n"] }]
}], () => [{ type: i1.NotificationService }, { type: i2.TranslateService }], { disabled: [{
type: Input
}], accept: [{
type: Input
}], multiple: [{
type: Input
}], resultType: [{
type: Input
}], fileInput: [{
type: ViewChild,
args: ['fileInput']
}], onClick: [{
type: HostListener,
args: ['click']
}] }); })();
(() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(UploadFileComponent, { className: "UploadFileComponent" }); })();
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component.js","sourceRoot":"","sources":["../../../../../../libs/common/src/form/upload-file/component.ts","../../../../../../libs/common/src/form/upload-file/template.html"],"names":[],"mappings":"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","sourcesContent":["import { NotificationService, coerceAttrBoolean } from '@alauda/ui';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  forwardRef,\n  HostListener,\n  ViewChild,\n  ElementRef,\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\n\nimport { ValueOf, isErrorMessage } from '../../core/public-api';\nimport { EffectDirectiveModule } from '../../effect-directive/public-api';\nimport { TranslateService } from '../../translate/public-api';\n\nexport const FileResultType = {\n  ARRAY_BUFFER: 'ArrayBuffer',\n  BINARY_STRING: 'BinaryString',\n  DATA_URL: 'DataURL',\n  TEXT: 'Text',\n} as const;\n\nexport type FileResultType = ValueOf<typeof FileResultType>;\n\nexport type FileResult = string | ArrayBuffer;\n\n@Component({\n  selector: 'acl-upload-file',\n  templateUrl: 'template.html',\n  styleUrls: ['styles.scss'],\n  preserveWhitespaces: false,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => UploadFileComponent),\n      multi: true,\n    },\n  ],\n  standalone: true,\n  imports: [EffectDirectiveModule],\n})\nexport class UploadFileComponent implements ControlValueAccessor {\n  @Input()\n  disabled: boolean;\n\n  @Input()\n  accept: string;\n\n  @Input()\n  multiple: boolean;\n\n  @Input()\n  resultType: FileResultType = FileResultType.TEXT;\n\n  @ViewChild('fileInput')\n  fileInput: ElementRef<HTMLInputElement>;\n\n  valueChange: (ev: FileResult | FileResult[]) => void;\n\n  constructor(\n    private readonly notification: NotificationService,\n    private readonly translate: TranslateService,\n  ) {}\n\n  @HostListener('click')\n  onClick() {\n    if (coerceAttrBoolean(this.disabled)) {\n      return;\n    }\n    this.fileInput.nativeElement.click();\n  }\n\n  async onChange(e: Event) {\n    const el = e.target as HTMLInputElement;\n\n    try {\n      const results = await Promise.all(\n        Array.from(el.files).map(\n          file =>\n            new Promise<FileResult>((resolve, reject) => {\n              const fileReader = new FileReader();\n              fileReader.addEventListener('load', () => {\n                const { result } = fileReader;\n\n                if (\n                  this.resultType === FileResultType.TEXT &&\n                  (result as string).includes('\\uFFFD')\n                ) {\n                  reject(new Error(this.translate.get('text_file_type_error')));\n                  return;\n                }\n\n                resolve(result);\n              });\n              fileReader[`readAs${this.resultType}`](file);\n            }),\n        ),\n      );\n      this.valueChange(this.multiple ? results : results[0]);\n    } catch (e) {\n      if (isErrorMessage(e)) {\n        this.notification.error(e.message);\n      }\n    }\n\n    el.value = null;\n  }\n\n  writeValue() {\n    //\n  }\n\n  registerOnChange(fn: any) {\n    this.valueChange = fn;\n  }\n\n  registerOnTouched() {\n    //\n  }\n}\n","<ng-content></ng-content>\n<input\n  #fileInput\n  class=\"file-input\"\n  type=\"file\"\n  [attr.accept]=\"accept\"\n  [multiple]=\"multiple\"\n  (change)=\"onChange($event)\"\n/>\n"]}