@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
185 lines • 32 kB
JavaScript
import { CommonModule } from '@angular/common';
import { ChangeDetectionStrategy, Component, Input, Optional, Self, ViewEncapsulation, } from '@angular/core';
import { FormsModule } from '@angular/forms';
import { BehaviorSubject, ReplaySubject, combineLatest, distinctUntilChanged, takeUntil, map, delay, of, switchMap, } from 'rxjs';
import { DOKU_FORM_FIELD_ACCESSOR, } from '../form-field';
import { DokuSpinner } from '../spinner';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
import * as i2 from "@angular/common";
export class DokuInputFileUpload {
/**
* Whether input file upload should be on disabled state. The default is `false`.
*/
get disabled() {
return this._disabled;
}
set disabled(val) {
this._disabled = val;
setTimeout(() => {
this.onDisable?.(this._disabled);
}, 0);
}
/**
* Whether input file upload should be on loading state. The default is `false`.
*/
get loading() {
return this._loading;
}
set loading(val) {
this._loading = val != null && `${val}` !== 'false';
}
constructor(cdr, ngControl) {
this.cdr = cdr;
this.ngControl = ngControl;
this.value = undefined;
this.fieldOptions = { withoutInputStyle: true };
this.loadingChanges$ = new BehaviorSubject(false);
this.destroy$ = new ReplaySubject();
/**
* Text that is shown as an input title.
*/
this.title = 'Upload File';
/**
* Text that is shown as an input subtitle.
*/
this.subtitle = 'Tap here to upload file';
/**
* Text that is shown in loading state.
*/
this.loadingText = 'loading...';
/**
* Comma-separated list of one or more file types, or unique file type specifiers,
* describing which file types to allow. The default is "*".
*/
this.accept = '*';
this._disabled = false;
this._loading = false;
if (this.ngControl) {
this.ngControl.valueAccessor = this;
}
}
ngAfterViewInit() {
this._listenNgControlStatus();
}
onFileSelected(e) {
const files = e.target.files;
if (!files)
return;
const file = files[0];
if (this._checkFileIsImage(file)) {
this._showImagePreview(file);
}
this.title = file.name;
this.subtitle = 'Tap here to change file';
this.value = file;
this.onChange?.(file);
}
ngOnChanges(changes) {
if (changes['loading']?.currentValue !== changes['loading']?.previousValue) {
this.loadingChanges$.next(changes['loading']?.currentValue);
}
if (changes['disabled']?.previousValue !== changes['disabled']?.currentValue) {
this.onDisable?.(!!this.disabled);
}
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
this.cdr.detectChanges();
}
/**
* Listen and assign NgControl status to `doku-form-field`
* to complete behavior and functionality of the field
* (hint, error, success message, and styling).
*/
_listenNgControlStatus() {
combineLatest([
this.loadingChanges$,
of(true).pipe(switchMap(() => this.ngControl?.statusChanges || of(null)), map((status) => ({
status: status,
state: {
pristine: this.ngControl?.control?.pristine,
untouched: this.ngControl?.control?.untouched,
},
})), delay(0), distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr))),
])
?.pipe(distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr)), takeUntil(this.destroy$))
.subscribe(([loading, statusObj]) => {
if (!this.value)
return;
const status = statusObj?.status;
const state = statusObj?.state;
if (status === 'VALID' && !loading) {
this.onValidate?.('valid', state);
}
else if (status === 'INVALID' && !loading) {
this.onValidate?.('invalid', state);
}
else {
this.onValidate?.(undefined, state);
}
});
}
_showImagePreview(file) {
if (file instanceof File) {
this.imagePreviewSrc = URL.createObjectURL(file);
return;
}
this.imagePreviewSrc = file;
}
_checkFileIsImage(file) {
if (file instanceof File) {
return file.type.startsWith('image/');
}
return false;
}
_checkStringIsImage(str) {
return /\.(jpg|jpeg|png|webp|avif|gif|svg)(\??.*)$/.test(str);
}
ngOnDestroy() {
this.destroy$.next(true);
this.destroy$.complete();
}
writeValue(value) {
if ((typeof value === 'string' && this._checkStringIsImage(value)) ||
(value instanceof File && this._checkFileIsImage(value))) {
this._showImagePreview(value);
}
this.value = value;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
registerOnDisable(fn) {
this.onDisable = fn;
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnValidate(fn) {
this.onValidate = fn;
}
}
DokuInputFileUpload.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuInputFileUpload, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component });
DokuInputFileUpload.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuInputFileUpload, isStandalone: true, selector: "doku-input-file-upload", inputs: { title: "title", subtitle: "subtitle", loadingText: "loadingText", accept: "accept", disabled: "disabled", loading: "loading" }, providers: [{ provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuInputFileUpload }], exportAs: ["dokuInputFileUpload"], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"d-input-file-container\"\n [class.d-input-file-loading]=\"loading\"\n (click)=\"!loading && inputFileDefault.click()\"\n>\n <div class=\"d-input-file-preview\">\n <img *ngIf=\"imagePreviewSrc\" [src]=\"imagePreviewSrc\" />\n <ng-container *ngIf=\"!imagePreviewSrc && value\" [ngTemplateOutlet]=\"iconDocs\"></ng-container>\n <ng-container *ngIf=\"!imagePreviewSrc && !value\" [ngTemplateOutlet]=\"iconPlus\"></ng-container>\n </div>\n <div class=\"d-input-file-wrapper\">\n <div class=\"d-input-file-desc\">\n <span class=\"d-input-file-name\">{{ title }}</span>\n <span class=\"d-input-file-status\">{{ loading ? loadingText : subtitle }}</span>\n </div>\n <doku-spinner *ngIf=\"loading\" [diameter]=\"16\" [strokeWidth]=\"1\"></doku-spinner>\n </div>\n</div>\n<input\n #inputFileDefault\n type=\"file\"\n class=\"d-input-file-default\"\n (change)=\"onFileSelected($event)\"\n [disabled]=\"disabled\"\n [accept]=\"accept\"\n/>\n\n<ng-template #iconPlus>\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M8.92849 6.57147V7.07147H9.42849H13.7142C13.964 7.07147 14.1666 7.2741 14.1666 7.52385V8.47623C14.1666 8.72598 13.964 8.92861 13.7142 8.92861H9.42849H8.92849V9.42861V13.7143C8.92849 13.9641 8.72586 14.1667 8.47611 14.1667H7.52373C7.27398 14.1667 7.07135 13.9641 7.07135 13.7143V9.42861V8.92861H6.57135H2.28563C2.03588 8.92861 1.83325 8.72598 1.83325 8.47623V7.52385C1.83325 7.2741 2.03588 7.07147 2.28563 7.07147H6.57135H7.07135V6.57147V2.28576C7.07135 2.036 7.27398 1.83337 7.52373 1.83337H8.47611C8.72586 1.83337 8.92849 2.036 8.92849 2.28576V6.57147Z\"\n fill=\"currentColor\"\n stroke=\"currentColor\"\n />\n </svg>\n</ng-template>\n\n<ng-template #iconDocs>\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M8.83333 4.87498V1.33331H3.625C3.27865 1.33331 3 1.61196 3 1.95831V14.0416C3 14.388 3.27865 14.6666 3.625 14.6666H12.375C12.7214 14.6666 13 14.388 13 14.0416V5.49998H9.45833C9.11458 5.49998 8.83333 5.21873 8.83333 4.87498ZM13 4.50779V4.66665H9.66667V1.33331H9.82552C9.99219 1.33331 10.151 1.39842 10.2682 1.5156L12.8177 4.06769C12.9349 4.18488 13 4.34373 13 4.50779Z\"\n fill=\"currentColor\"\n />\n </svg>\n</ng-template>\n", dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: FormsModule }, { kind: "component", type: DokuSpinner, selector: "doku-spinner", inputs: ["size", "diameter", "strokeWidth"], exportAs: ["dokuSpinner"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuInputFileUpload, decorators: [{
type: Component,
args: [{ selector: 'doku-input-file-upload', exportAs: 'dokuInputFileUpload', standalone: true, imports: [CommonModule, FormsModule, DokuSpinner], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [{ provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuInputFileUpload }], template: "<div\n class=\"d-input-file-container\"\n [class.d-input-file-loading]=\"loading\"\n (click)=\"!loading && inputFileDefault.click()\"\n>\n <div class=\"d-input-file-preview\">\n <img *ngIf=\"imagePreviewSrc\" [src]=\"imagePreviewSrc\" />\n <ng-container *ngIf=\"!imagePreviewSrc && value\" [ngTemplateOutlet]=\"iconDocs\"></ng-container>\n <ng-container *ngIf=\"!imagePreviewSrc && !value\" [ngTemplateOutlet]=\"iconPlus\"></ng-container>\n </div>\n <div class=\"d-input-file-wrapper\">\n <div class=\"d-input-file-desc\">\n <span class=\"d-input-file-name\">{{ title }}</span>\n <span class=\"d-input-file-status\">{{ loading ? loadingText : subtitle }}</span>\n </div>\n <doku-spinner *ngIf=\"loading\" [diameter]=\"16\" [strokeWidth]=\"1\"></doku-spinner>\n </div>\n</div>\n<input\n #inputFileDefault\n type=\"file\"\n class=\"d-input-file-default\"\n (change)=\"onFileSelected($event)\"\n [disabled]=\"disabled\"\n [accept]=\"accept\"\n/>\n\n<ng-template #iconPlus>\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M8.92849 6.57147V7.07147H9.42849H13.7142C13.964 7.07147 14.1666 7.2741 14.1666 7.52385V8.47623C14.1666 8.72598 13.964 8.92861 13.7142 8.92861H9.42849H8.92849V9.42861V13.7143C8.92849 13.9641 8.72586 14.1667 8.47611 14.1667H7.52373C7.27398 14.1667 7.07135 13.9641 7.07135 13.7143V9.42861V8.92861H6.57135H2.28563C2.03588 8.92861 1.83325 8.72598 1.83325 8.47623V7.52385C1.83325 7.2741 2.03588 7.07147 2.28563 7.07147H6.57135H7.07135V6.57147V2.28576C7.07135 2.036 7.27398 1.83337 7.52373 1.83337H8.47611C8.72586 1.83337 8.92849 2.036 8.92849 2.28576V6.57147Z\"\n fill=\"currentColor\"\n stroke=\"currentColor\"\n />\n </svg>\n</ng-template>\n\n<ng-template #iconDocs>\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M8.83333 4.87498V1.33331H3.625C3.27865 1.33331 3 1.61196 3 1.95831V14.0416C3 14.388 3.27865 14.6666 3.625 14.6666H12.375C12.7214 14.6666 13 14.388 13 14.0416V5.49998H9.45833C9.11458 5.49998 8.83333 5.21873 8.83333 4.87498ZM13 4.50779V4.66665H9.66667V1.33331H9.82552C9.99219 1.33331 10.151 1.39842 10.2682 1.5156L12.8177 4.06769C12.9349 4.18488 13 4.34373 13 4.50779Z\"\n fill=\"currentColor\"\n />\n </svg>\n</ng-template>\n" }]
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.NgControl, decorators: [{
type: Optional
}, {
type: Self
}] }]; }, propDecorators: { title: [{
type: Input
}], subtitle: [{
type: Input
}], loadingText: [{
type: Input
}], accept: [{
type: Input
}], disabled: [{
type: Input
}], loading: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/input-file-upload/input-file-upload.component.ts","../../../../../../projects/doku-fragment/src/lib/input-file-upload/input-file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Optional,\n  Self,\n  SimpleChanges,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ControlValueAccessor, FormsModule, NgControl } from '@angular/forms';\nimport {\n  BehaviorSubject,\n  ReplaySubject,\n  combineLatest,\n  distinctUntilChanged,\n  takeUntil,\n  map,\n  delay,\n  of,\n  switchMap,\n} from 'rxjs';\nimport {\n  DOKU_FORM_FIELD_ACCESSOR,\n  DokuFormFieldAccessor,\n  DokuFormFieldAccessorValidateState,\n  DokuFormFieldAccessorValidateValue,\n  DokuFormFieldOptions,\n} from '../form-field';\nimport { DokuSpinner } from '../spinner';\n\n@Component({\n  selector: 'doku-input-file-upload',\n  exportAs: 'dokuInputFileUpload',\n  standalone: true,\n  imports: [CommonModule, FormsModule, DokuSpinner],\n  templateUrl: './input-file-upload.component.html',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [{ provide: DOKU_FORM_FIELD_ACCESSOR, useExisting: DokuInputFileUpload }],\n})\nexport class DokuInputFileUpload\n  implements DokuFormFieldAccessor, ControlValueAccessor, AfterViewInit, OnChanges, OnDestroy\n{\n  onChange?: (value: File) => void;\n  onTouched?: () => void;\n  onDisable?: (value: boolean) => void;\n  onValidate?: (\n    value?: DokuFormFieldAccessorValidateValue,\n    state?: DokuFormFieldAccessorValidateState\n  ) => void;\n\n  value?: File | string = undefined;\n  fieldOptions?: DokuFormFieldOptions = { withoutInputStyle: true };\n  private loadingChanges$ = new BehaviorSubject(false);\n  private destroy$ = new ReplaySubject();\n  protected imagePreviewSrc?: string;\n\n  /**\n   * Text that is shown as an input title.\n   */\n  @Input() title = 'Upload File';\n\n  /**\n   * Text that is shown as an input subtitle.\n   */\n  @Input() subtitle = 'Tap here to upload file';\n\n  /**\n   * Text that is shown in loading state.\n   */\n  @Input() loadingText = 'loading...';\n\n  /**\n   * Comma-separated list of one or more file types, or unique file type specifiers,\n   * describing which file types to allow. The default is \"*\".\n   */\n  @Input() accept = '*';\n\n  /**\n   * Whether input file upload should be on disabled state. The default is `false`.\n   */\n  @Input()\n  get disabled() {\n    return this._disabled;\n  }\n  set disabled(val: boolean) {\n    this._disabled = val;\n\n    setTimeout(() => {\n      this.onDisable?.(this._disabled);\n    }, 0);\n  }\n  private _disabled = false;\n\n  /**\n   * Whether input file upload should be on loading state. The default is `false`.\n   */\n  @Input()\n  get loading() {\n    return this._loading;\n  }\n  set loading(val: unknown) {\n    this._loading = val != null && `${val}` !== 'false';\n  }\n  private _loading = false;\n\n  constructor(private cdr: ChangeDetectorRef, @Optional() @Self() private ngControl?: NgControl) {\n    if (this.ngControl) {\n      this.ngControl.valueAccessor = this;\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this._listenNgControlStatus();\n  }\n\n  onFileSelected(e: Event) {\n    const files = (e.target as HTMLInputElement).files;\n\n    if (!files) return;\n\n    const file = files[0];\n\n    if (this._checkFileIsImage(file)) {\n      this._showImagePreview(file);\n    }\n\n    this.title = file.name;\n    this.subtitle = 'Tap here to change file';\n    this.value = file;\n    this.onChange?.(file);\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['loading']?.currentValue !== changes['loading']?.previousValue) {\n      this.loadingChanges$.next(changes['loading']?.currentValue);\n    }\n\n    if (changes['disabled']?.previousValue !== changes['disabled']?.currentValue) {\n      this.onDisable?.(!!this.disabled);\n    }\n  }\n\n  setDisabledState?(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n\n    this.cdr.detectChanges();\n  }\n\n  /**\n   * Listen and assign NgControl status to `doku-form-field`\n   * to complete behavior and functionality of the field\n   * (hint, error, success message, and styling).\n   */\n  private _listenNgControlStatus() {\n    combineLatest([\n      this.loadingChanges$,\n      of(true).pipe(\n        switchMap(() => this.ngControl?.statusChanges || of(null)),\n        map((status) => ({\n          status: status,\n          state: {\n            pristine: this.ngControl?.control?.pristine,\n            untouched: this.ngControl?.control?.untouched,\n          } as DokuFormFieldAccessorValidateState,\n        })),\n        delay(0),\n        distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr))\n      ),\n    ])\n      ?.pipe(\n        distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr)),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(([loading, statusObj]) => {\n        if (!this.value) return;\n\n        const status = statusObj?.status;\n        const state = statusObj?.state;\n        if (status === 'VALID' && !loading) {\n          this.onValidate?.('valid', state);\n        } else if (status === 'INVALID' && !loading) {\n          this.onValidate?.('invalid', state);\n        } else {\n          this.onValidate?.(undefined, state);\n        }\n      });\n  }\n\n  private _showImagePreview(file: File | string) {\n    if (file instanceof File) {\n      this.imagePreviewSrc = URL.createObjectURL(file);\n\n      return;\n    }\n\n    this.imagePreviewSrc = file;\n  }\n\n  private _checkFileIsImage(file: File): boolean {\n    if (file instanceof File) {\n      return file.type.startsWith('image/');\n    }\n\n    return false;\n  }\n\n  private _checkStringIsImage(str: string): boolean {\n    return /\\.(jpg|jpeg|png|webp|avif|gif|svg)(\\??.*)$/.test(str);\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  writeValue(value: File | string): void {\n    if (\n      (typeof value === 'string' && this._checkStringIsImage(value)) ||\n      (value instanceof File && this._checkFileIsImage(value))\n    ) {\n      this._showImagePreview(value);\n    }\n\n    this.value = value;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  registerOnDisable(fn: (value: boolean) => void): void {\n    this.onDisable = fn;\n  }\n\n  registerOnChange(fn: (value: File) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnValidate(fn: (value?: DokuFormFieldAccessorValidateValue) => void): void {\n    this.onValidate = fn;\n  }\n}\n","<div\n  class=\"d-input-file-container\"\n  [class.d-input-file-loading]=\"loading\"\n  (click)=\"!loading && inputFileDefault.click()\"\n>\n  <div class=\"d-input-file-preview\">\n    <img *ngIf=\"imagePreviewSrc\" [src]=\"imagePreviewSrc\" />\n    <ng-container *ngIf=\"!imagePreviewSrc && value\" [ngTemplateOutlet]=\"iconDocs\"></ng-container>\n    <ng-container *ngIf=\"!imagePreviewSrc && !value\" [ngTemplateOutlet]=\"iconPlus\"></ng-container>\n  </div>\n  <div class=\"d-input-file-wrapper\">\n    <div class=\"d-input-file-desc\">\n      <span class=\"d-input-file-name\">{{ title }}</span>\n      <span class=\"d-input-file-status\">{{ loading ? loadingText : subtitle }}</span>\n    </div>\n    <doku-spinner *ngIf=\"loading\" [diameter]=\"16\" [strokeWidth]=\"1\"></doku-spinner>\n  </div>\n</div>\n<input\n  #inputFileDefault\n  type=\"file\"\n  class=\"d-input-file-default\"\n  (change)=\"onFileSelected($event)\"\n  [disabled]=\"disabled\"\n  [accept]=\"accept\"\n/>\n\n<ng-template #iconPlus>\n  <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n    <path\n      d=\"M8.92849 6.57147V7.07147H9.42849H13.7142C13.964 7.07147 14.1666 7.2741 14.1666 7.52385V8.47623C14.1666 8.72598 13.964 8.92861 13.7142 8.92861H9.42849H8.92849V9.42861V13.7143C8.92849 13.9641 8.72586 14.1667 8.47611 14.1667H7.52373C7.27398 14.1667 7.07135 13.9641 7.07135 13.7143V9.42861V8.92861H6.57135H2.28563C2.03588 8.92861 1.83325 8.72598 1.83325 8.47623V7.52385C1.83325 7.2741 2.03588 7.07147 2.28563 7.07147H6.57135H7.07135V6.57147V2.28576C7.07135 2.036 7.27398 1.83337 7.52373 1.83337H8.47611C8.72586 1.83337 8.92849 2.036 8.92849 2.28576V6.57147Z\"\n      fill=\"currentColor\"\n      stroke=\"currentColor\"\n    />\n  </svg>\n</ng-template>\n\n<ng-template #iconDocs>\n  <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n    <path\n      d=\"M8.83333 4.87498V1.33331H3.625C3.27865 1.33331 3 1.61196 3 1.95831V14.0416C3 14.388 3.27865 14.6666 3.625 14.6666H12.375C12.7214 14.6666 13 14.388 13 14.0416V5.49998H9.45833C9.11458 5.49998 8.83333 5.21873 8.83333 4.87498ZM13 4.50779V4.66665H9.66667V1.33331H9.82552C9.99219 1.33331 10.151 1.39842 10.2682 1.5156L12.8177 4.06769C12.9349 4.18488 13 4.34373 13 4.50779Z\"\n      fill=\"currentColor\"\n    />\n  </svg>\n</ng-template>\n"]}