@acrodata/gui
Version: 
JSON powered GUI for configurable panels.
139 lines • 25.3 kB
JavaScript
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewChild, ViewEncapsulation, forwardRef, } from '@angular/core';
import { FormsModule, NG_VALUE_ACCESSOR } from '@angular/forms';
import { MatIconButton } from '@angular/material/button';
import { MatFormField, MatHint, MatPrefix, MatSuffix } from '@angular/material/form-field';
import { MatIcon } from '@angular/material/icon';
import { MatInput } from '@angular/material/input';
import { finalize } from 'rxjs/operators';
import { GuiFieldLabel } from '../field-label/field-label';
import { GuiIconButtonWrapper } from '../icon-button-wrapper/icon-button-wrapper';
import * as i0 from "@angular/core";
import * as i1 from "./file-uploader-config";
import * as i2 from "../gui-icons";
import * as i3 from "@angular/forms";
export class GuiFileUploader {
    constructor(fileUploaderCfg, cdr, iconsRegistry) {
        this.fileUploaderCfg = fileUploaderCfg;
        this.cdr = cdr;
        this.config = {};
        this.disabled = false;
        this.type = '*';
        this.name = '';
        this.accept = '';
        this.fileChange = new EventEmitter();
        // file url that returned from the server
        this.url = '';
        this.onChange = () => { };
        this.onTouched = () => { };
        iconsRegistry.add('link', 'clear', 'file', 'upload');
    }
    ngOnChanges(changes) {
        if (changes['config'] || changes['accept'] || changes['type']) {
            this.accept = this.config.accept || this.accept || this.type + '/*';
        }
    }
    writeValue(value) {
        this.url = value;
        this.cdr.markForCheck();
    }
    registerOnChange(fn) {
        this.onChange = fn;
    }
    registerOnTouched(fn) {
        this.onTouched = fn;
    }
    setDisabledState(isDisabled) {
        this.disabled = isDisabled;
        this.cdr.markForCheck();
    }
    upload(fileUpload) {
        const formData = new FormData();
        formData.append('file', fileUpload.data);
        fileUpload.inProgress = true;
        this.fileUploaderCfg
            .upload(formData, this.config)
            .pipe(finalize(() => {
            fileUpload.inProgress = false;
        }))
            .subscribe(result => {
            if (result) {
                this.url = result;
                this.cdr.detectChanges();
                this.onChange(this.url);
                this.onTouched();
                this.fileChange.emit(this.url);
            }
        });
    }
    onUrlChange(e) {
        this.url = e.target.value;
        this.onChange(this.url);
        this.fileChange.emit(this.url);
    }
    onFileChange(e) {
        this.fileUpload = {
            data: e.target.files[0],
            inProgress: false,
            progress: 0,
        };
        this.upload(this.fileUpload);
        // reset input value
        e.target.value = '';
    }
    onBlur() {
        this.onTouched();
    }
    onClear() {
        this.url = '';
        this.onChange(this.url);
        this.onTouched();
        this.fileChange.emit(this.url);
    }
    static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: GuiFileUploader, deps: [{ token: i1.GuiFileUploaderConfig }, { token: i0.ChangeDetectorRef }, { token: i2.GuiIconsRegistry }], target: i0.ɵɵFactoryTarget.Component }); }
    static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: GuiFileUploader, isStandalone: true, selector: "gui-file-uploader", inputs: { config: "config", disabled: "disabled", type: "type", name: "name", accept: "accept" }, outputs: { fileChange: "fileChange" }, host: { classAttribute: "gui-field gui-file-uploader" }, providers: [
            {
                provide: NG_VALUE_ACCESSOR,
                useExisting: forwardRef(() => GuiFileUploader),
                multi: true,
            },
        ], viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<mat-form-field>\n  <gui-icon-button-wrapper matPrefix>\n    <mat-icon svgIcon=\"link\" />\n  </gui-icon-button-wrapper>\n\n  <input\n    matInput\n    type=\"text\"\n    [ngModel]=\"url\"\n    [disabled]=\"disabled\"\n    [placeholder]=\"config.placeholder || ''\"\n    (change)=\"onUrlChange($event)\"\n    (blur)=\"onBlur()\"\n  />\n\n  @if (url) {\n    <gui-icon-button-wrapper matSuffix>\n      <button mat-icon-button type=\"button\" color=\"warn\" [disabled]=\"disabled\" (click)=\"onClear()\">\n        <mat-icon svgIcon=\"clear\" />\n      </button>\n    </gui-icon-button-wrapper>\n  }\n</mat-form-field>\n\n<figure class=\"gui-file-content\">\n  @if (url) {\n    @switch (type) {\n      @case ('image') {\n        <img [src]=\"url\" alt=\"\" />\n      }\n      @case ('video') {\n        <video [src]=\"url\"></video>\n      }\n      @case ('audio') {\n        <audio [src]=\"url\" controls></audio>\n      }\n      @default {\n        <mat-icon svgIcon=\"file\" />\n      }\n    }\n  } @else {\n    <div class=\"gui-file-placeholder\">\n      <mat-icon svgIcon=\"upload\" />\n    </div>\n  }\n\n  <input\n    #fileInput\n    type=\"file\"\n    [accept]=\"accept\"\n    [name]=\"name\"\n    [disabled]=\"disabled\"\n    (change)=\"onFileChange($event)\"\n  />\n</figure>\n\n@if (config.parentType === 'inline') {\n  <mat-hint>\n    <gui-field-label [config]=\"config\" />\n  </mat-hint>\n}\n", styles: [".gui-file-uploader{display:block;overflow:auto;line-height:1}.gui-file-uploader .mat-mdc-form-field{width:100%}.gui-file-uploader .gui-file-content{position:relative;display:flex;align-items:center;justify-content:center;height:7.5rem;margin:.25rem 0 0;padding:.25rem;background-color:var(--mdc-filled-text-field-container-color);border:1px solid transparent;border-radius:.25rem}.gui-file-uploader .gui-file-content:hover,.gui-file-uploader .gui-file-content:focus-within{border-color:currentColor}.gui-file-uploader .gui-file-content img,.gui-file-uploader .gui-file-content video{width:100%;height:100%;object-fit:contain}.gui-file-uploader .gui-file-content input[type=file]{position:absolute;top:0;left:0;width:100%;height:100%;opacity:0;cursor:pointer}.gui-file-uploader .gui-file-content .mat-icon svg{width:1.5rem;height:1.5rem}\n"], dependencies: [{ kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "component", type: MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: MatPrefix, selector: "[matPrefix], [matIconPrefix], [matTextPrefix]", inputs: ["matTextPrefix"] }, { kind: "directive", type: MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl],      input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "directive", type: MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "directive", type: MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "component", type: GuiFieldLabel, selector: "gui-field-label", inputs: ["config", "index"] }, { kind: "component", type: GuiIconButtonWrapper, selector: "gui-icon-button-wrapper" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: GuiFileUploader, decorators: [{
            type: Component,
            args: [{ selector: 'gui-file-uploader', host: {
                        class: 'gui-field gui-file-uploader',
                    }, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [
                        {
                            provide: NG_VALUE_ACCESSOR,
                            useExisting: forwardRef(() => GuiFileUploader),
                            multi: true,
                        },
                    ], standalone: true, imports: [
                        FormsModule,
                        MatFormField,
                        MatIcon,
                        MatPrefix,
                        MatInput,
                        MatIconButton,
                        MatSuffix,
                        MatHint,
                        GuiFieldLabel,
                        GuiIconButtonWrapper,
                    ], template: "<mat-form-field>\n  <gui-icon-button-wrapper matPrefix>\n    <mat-icon svgIcon=\"link\" />\n  </gui-icon-button-wrapper>\n\n  <input\n    matInput\n    type=\"text\"\n    [ngModel]=\"url\"\n    [disabled]=\"disabled\"\n    [placeholder]=\"config.placeholder || ''\"\n    (change)=\"onUrlChange($event)\"\n    (blur)=\"onBlur()\"\n  />\n\n  @if (url) {\n    <gui-icon-button-wrapper matSuffix>\n      <button mat-icon-button type=\"button\" color=\"warn\" [disabled]=\"disabled\" (click)=\"onClear()\">\n        <mat-icon svgIcon=\"clear\" />\n      </button>\n    </gui-icon-button-wrapper>\n  }\n</mat-form-field>\n\n<figure class=\"gui-file-content\">\n  @if (url) {\n    @switch (type) {\n      @case ('image') {\n        <img [src]=\"url\" alt=\"\" />\n      }\n      @case ('video') {\n        <video [src]=\"url\"></video>\n      }\n      @case ('audio') {\n        <audio [src]=\"url\" controls></audio>\n      }\n      @default {\n        <mat-icon svgIcon=\"file\" />\n      }\n    }\n  } @else {\n    <div class=\"gui-file-placeholder\">\n      <mat-icon svgIcon=\"upload\" />\n    </div>\n  }\n\n  <input\n    #fileInput\n    type=\"file\"\n    [accept]=\"accept\"\n    [name]=\"name\"\n    [disabled]=\"disabled\"\n    (change)=\"onFileChange($event)\"\n  />\n</figure>\n\n@if (config.parentType === 'inline') {\n  <mat-hint>\n    <gui-field-label [config]=\"config\" />\n  </mat-hint>\n}\n", styles: [".gui-file-uploader{display:block;overflow:auto;line-height:1}.gui-file-uploader .mat-mdc-form-field{width:100%}.gui-file-uploader .gui-file-content{position:relative;display:flex;align-items:center;justify-content:center;height:7.5rem;margin:.25rem 0 0;padding:.25rem;background-color:var(--mdc-filled-text-field-container-color);border:1px solid transparent;border-radius:.25rem}.gui-file-uploader .gui-file-content:hover,.gui-file-uploader .gui-file-content:focus-within{border-color:currentColor}.gui-file-uploader .gui-file-content img,.gui-file-uploader .gui-file-content video{width:100%;height:100%;object-fit:contain}.gui-file-uploader .gui-file-content input[type=file]{position:absolute;top:0;left:0;width:100%;height:100%;opacity:0;cursor:pointer}.gui-file-uploader .gui-file-content .mat-icon svg{width:1.5rem;height:1.5rem}\n"] }]
        }], ctorParameters: () => [{ type: i1.GuiFileUploaderConfig }, { type: i0.ChangeDetectorRef }, { type: i2.GuiIconsRegistry }], propDecorators: { fileInput: [{
                type: ViewChild,
                args: ['fileInput']
            }], config: [{
                type: Input
            }], disabled: [{
                type: Input
            }], type: [{
                type: Input
            }], name: [{
                type: Input
            }], accept: [{
                type: Input
            }], fileChange: [{
                type: Output
            }] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-uploader.js","sourceRoot":"","sources":["../../../../projects/gui/file-uploader/file-uploader.ts","../../../../projects/gui/file-uploader/file-uploader.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n  forwardRef,\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { MatIconButton } from '@angular/material/button';\nimport { MatFormField, MatHint, MatPrefix, MatSuffix } from '@angular/material/form-field';\nimport { MatIcon } from '@angular/material/icon';\nimport { MatInput } from '@angular/material/input';\nimport { finalize } from 'rxjs/operators';\n\nimport { GuiFieldLabel } from '../field-label/field-label';\nimport { GuiIconsRegistry } from '../gui-icons';\nimport { GuiIconButtonWrapper } from '../icon-button-wrapper/icon-button-wrapper';\nimport { GuiControl } from '../interface';\nimport { GuiFileUploaderConfig } from './file-uploader-config';\n\nexport type FileUploadType = 'image' | 'video' | 'audio' | '*';\n\nexport interface FileUploadContent {\n  data: File;\n  progress: number;\n  inProgress: boolean;\n}\n\n@Component({\n  selector: 'gui-file-uploader',\n  templateUrl: './file-uploader.html',\n  styleUrl: './file-uploader.scss',\n  host: {\n    class: 'gui-field gui-file-uploader',\n  },\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => GuiFileUploader),\n      multi: true,\n    },\n  ],\n  standalone: true,\n  imports: [\n    FormsModule,\n    MatFormField,\n    MatIcon,\n    MatPrefix,\n    MatInput,\n    MatIconButton,\n    MatSuffix,\n    MatHint,\n    GuiFieldLabel,\n    GuiIconButtonWrapper,\n  ],\n})\nexport class GuiFileUploader implements ControlValueAccessor, OnChanges {\n  @ViewChild('fileInput') fileInput!: ElementRef<HTMLInputElement>;\n\n  @Input() config: Partial<GuiControl> = {};\n  @Input() disabled = false;\n  @Input() type: FileUploadType = '*';\n  @Input() name = '';\n  @Input() accept = '';\n\n  @Output() fileChange = new EventEmitter<string>();\n\n  // file url that returned from the server\n  url = '';\n\n  // file to upload\n  fileUpload!: FileUploadContent;\n\n  private onChange: (value: string) => void = () => {};\n  private onTouched: () => void = () => {};\n\n  constructor(\n    private fileUploaderCfg: GuiFileUploaderConfig,\n    private cdr: ChangeDetectorRef,\n    iconsRegistry: GuiIconsRegistry\n  ) {\n    iconsRegistry.add('link', 'clear', 'file', 'upload');\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['config'] || changes['accept'] || changes['type']) {\n      this.accept = this.config.accept || this.accept || this.type + '/*';\n    }\n  }\n\n  writeValue(value: string) {\n    this.url = value;\n    this.cdr.markForCheck();\n  }\n\n  registerOnChange(fn: (value: string) => void) {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void) {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n    this.cdr.markForCheck();\n  }\n\n  upload(fileUpload: FileUploadContent) {\n    const formData = new FormData();\n    formData.append('file', fileUpload.data);\n\n    fileUpload.inProgress = true;\n\n    this.fileUploaderCfg\n      .upload(formData, this.config)\n      .pipe(\n        finalize(() => {\n          fileUpload.inProgress = false;\n        })\n      )\n      .subscribe(result => {\n        if (result) {\n          this.url = result;\n          this.cdr.detectChanges();\n\n          this.onChange(this.url);\n          this.onTouched();\n\n          this.fileChange.emit(this.url);\n        }\n      });\n  }\n\n  onUrlChange(e: Event) {\n    this.url = (e.target as HTMLInputElement).value;\n\n    this.onChange(this.url);\n\n    this.fileChange.emit(this.url);\n  }\n\n  onFileChange(e: Event) {\n    this.fileUpload = {\n      data: (e.target as HTMLInputElement).files![0],\n      inProgress: false,\n      progress: 0,\n    };\n\n    this.upload(this.fileUpload);\n\n    // reset input value\n    (e.target as HTMLInputElement).value = '';\n  }\n\n  onBlur() {\n    this.onTouched();\n  }\n\n  onClear() {\n    this.url = '';\n\n    this.onChange(this.url);\n    this.onTouched();\n\n    this.fileChange.emit(this.url);\n  }\n}\n","<mat-form-field>\n  <gui-icon-button-wrapper matPrefix>\n    <mat-icon svgIcon=\"link\" />\n  </gui-icon-button-wrapper>\n\n  <input\n    matInput\n    type=\"text\"\n    [ngModel]=\"url\"\n    [disabled]=\"disabled\"\n    [placeholder]=\"config.placeholder || ''\"\n    (change)=\"onUrlChange($event)\"\n    (blur)=\"onBlur()\"\n  />\n\n  @if (url) {\n    <gui-icon-button-wrapper matSuffix>\n      <button mat-icon-button type=\"button\" color=\"warn\" [disabled]=\"disabled\" (click)=\"onClear()\">\n        <mat-icon svgIcon=\"clear\" />\n      </button>\n    </gui-icon-button-wrapper>\n  }\n</mat-form-field>\n\n<figure class=\"gui-file-content\">\n  @if (url) {\n    @switch (type) {\n      @case ('image') {\n        <img [src]=\"url\" alt=\"\" />\n      }\n      @case ('video') {\n        <video [src]=\"url\"></video>\n      }\n      @case ('audio') {\n        <audio [src]=\"url\" controls></audio>\n      }\n      @default {\n        <mat-icon svgIcon=\"file\" />\n      }\n    }\n  } @else {\n    <div class=\"gui-file-placeholder\">\n      <mat-icon svgIcon=\"upload\" />\n    </div>\n  }\n\n  <input\n    #fileInput\n    type=\"file\"\n    [accept]=\"accept\"\n    [name]=\"name\"\n    [disabled]=\"disabled\"\n    (change)=\"onFileChange($event)\"\n  />\n</figure>\n\n@if (config.parentType === 'inline') {\n  <mat-hint>\n    <gui-field-label [config]=\"config\" />\n  </mat-hint>\n}\n"]}