UNPKG

@clr/angular

Version:

Angular components for Clarity

160 lines (156 loc) 18.6 kB
/* * Copyright (c) 2016-2025 Broadcom. All Rights Reserved. * The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries. * This software is released under MIT license. * The full license information can be found in LICENSE in the root directory of this project. */ import { Component, ContentChild, inject, Injector } from '@angular/core'; import { ClrCommonStringsService } from '../../utils/i18n/common-strings.service'; import { NgControlService } from '../common/providers/ng-control.service'; import { ClrFileInputContainer } from './file-input-container'; import { selectFiles } from './file-input.helpers'; import { CLR_FILE_MESSAGES_TEMPLATE_CONTEXT } from './file-messages'; import { ClrFileMessagesTemplate, } from './file-messages-template'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; import * as i2 from "../../icon/icon"; export class ClrFileList { constructor() { this.injector = inject(Injector); this.commonStrings = inject(ClrCommonStringsService); this.ngControlService = inject(NgControlService, { optional: true }); this.fileInputContainer = inject(ClrFileInputContainer, { optional: true }); if (!this.ngControlService || !this.fileInputContainer) { throw new Error('The clr-file-list component can only be used within a clr-file-input-container.'); } } get files() { if (!this.fileInputContainer.fileInput) { return []; } const fileInputElement = this.fileInputContainer.fileInput.elementRef.nativeElement; return Array.from(fileInputElement.files).sort((a, b) => a.name.localeCompare(b.name)); } getClearFileLabel(filename) { return this.commonStrings.parse(this.commonStrings.keys.clearFile, { FILE: filename, }); } clearFile(fileToRemove) { if (!this.fileInputContainer.fileInput) { return; } const fileInputElement = this.fileInputContainer.fileInput.elementRef.nativeElement; const files = Array.from(fileInputElement.files); const newFiles = files.filter(file => file !== fileToRemove); selectFiles(fileInputElement, newFiles); this.fileInputContainer.focusBrowseButton(); } createFileMessagesTemplateContext(file) { const fileInputErrors = this.ngControlService.control.errors || {}; const errors = { accept: fileInputErrors.accept?.find(error => error.name === file.name), minFileSize: fileInputErrors.minFileSize?.find(error => error.name === file.name), maxFileSize: fileInputErrors.maxFileSize?.find(error => error.name === file.name), }; const success = Object.values(errors).every(error => !error); return { $implicit: file, success, errors }; } createFileMessagesTemplateInjector(fileMessagesTemplateContext) { return Injector.create({ parent: this.injector, providers: [{ provide: CLR_FILE_MESSAGES_TEMPLATE_CONTEXT, useValue: fileMessagesTemplateContext }], }); } } ClrFileList.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrFileList, deps: [], target: i0.ɵɵFactoryTarget.Component }); ClrFileList.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrFileList, selector: "clr-file-list", host: { properties: { "attr.role": "\"list\"", "class.clr-file-list": "true" } }, queries: [{ propertyName: "fileMessagesTemplate", first: true, predicate: ClrFileMessagesTemplate, descendants: true }], ngImport: i0, template: ` <ng-container *ngFor="let file of files"> <div *ngIf="createFileMessagesTemplateContext(file); let fileMessagesTemplateContext" role="listitem" class="clr-file-list-item" [ngClass]="{ 'clr-error': !fileMessagesTemplateContext.success, 'clr-success': fileMessagesTemplateContext.success }" > <div class="clr-file-label-and-status-icon"> <span class="label clr-file-label"> {{ file.name }} <button class="btn btn-sm btn-link-neutral btn-icon clr-file-clear-button" [attr.aria-label]="getClearFileLabel(file.name)" (click)="clearFile(file)" > <cds-icon shape="times"></cds-icon> </button> </span> <cds-icon class="clr-validate-icon" [attr.shape]="fileMessagesTemplateContext.success ? 'check-circle' : 'exclamation-circle'" [attr.status]="fileMessagesTemplateContext.success ? 'success' : 'danger'" aria-hidden="true" ></cds-icon> </div> <ng-container *ngIf="fileMessagesTemplate" [ngTemplateOutlet]="fileMessagesTemplate.templateRef" [ngTemplateOutletContext]="fileMessagesTemplateContext" [ngTemplateOutletInjector]="createFileMessagesTemplateInjector(fileMessagesTemplateContext)" ></ng-container> </div> </ng-container> `, isInline: true, dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i2.CdsIconCustomTag, selector: "cds-icon" }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrFileList, decorators: [{ type: Component, args: [{ selector: 'clr-file-list', template: ` <ng-container *ngFor="let file of files"> <div *ngIf="createFileMessagesTemplateContext(file); let fileMessagesTemplateContext" role="listitem" class="clr-file-list-item" [ngClass]="{ 'clr-error': !fileMessagesTemplateContext.success, 'clr-success': fileMessagesTemplateContext.success }" > <div class="clr-file-label-and-status-icon"> <span class="label clr-file-label"> {{ file.name }} <button class="btn btn-sm btn-link-neutral btn-icon clr-file-clear-button" [attr.aria-label]="getClearFileLabel(file.name)" (click)="clearFile(file)" > <cds-icon shape="times"></cds-icon> </button> </span> <cds-icon class="clr-validate-icon" [attr.shape]="fileMessagesTemplateContext.success ? 'check-circle' : 'exclamation-circle'" [attr.status]="fileMessagesTemplateContext.success ? 'success' : 'danger'" aria-hidden="true" ></cds-icon> </div> <ng-container *ngIf="fileMessagesTemplate" [ngTemplateOutlet]="fileMessagesTemplate.templateRef" [ngTemplateOutletContext]="fileMessagesTemplateContext" [ngTemplateOutletInjector]="createFileMessagesTemplateInjector(fileMessagesTemplateContext)" ></ng-container> </div> </ng-container> `, host: { '[attr.role]': '"list"', '[class.clr-file-list]': 'true', }, }] }], ctorParameters: function () { return []; }, propDecorators: { fileMessagesTemplate: [{ type: ContentChild, args: [ClrFileMessagesTemplate] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-list.js","sourceRoot":"","sources":["../../../../../projects/angular/src/forms/file-input/file-list.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2016-2025 Broadcom. All Rights Reserved.\n * The term \"Broadcom\" refers to Broadcom Inc. and/or its subsidiaries.\n * This software is released under MIT license.\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport { Component, ContentChild, inject, Injector } from '@angular/core';\n\nimport { ClrCommonStringsService } from '../../utils/i18n/common-strings.service';\nimport { NgControlService } from '../common/providers/ng-control.service';\nimport { ClrFileInputContainer } from './file-input-container';\nimport { ClrFileListValidationErrors } from './file-input-validator-errors';\nimport { selectFiles } from './file-input.helpers';\nimport { CLR_FILE_MESSAGES_TEMPLATE_CONTEXT } from './file-messages';\nimport {\n  ClrFileMessagesTemplate,\n  ClrFileMessagesTemplateContext,\n  ClrSingleFileValidationErrors,\n} from './file-messages-template';\n\n@Component({\n  selector: 'clr-file-list',\n  template: `\n    <ng-container *ngFor=\"let file of files\">\n      <div\n        *ngIf=\"createFileMessagesTemplateContext(file); let fileMessagesTemplateContext\"\n        role=\"listitem\"\n        class=\"clr-file-list-item\"\n        [ngClass]=\"{\n          'clr-error': !fileMessagesTemplateContext.success,\n          'clr-success': fileMessagesTemplateContext.success\n        }\"\n      >\n        <div class=\"clr-file-label-and-status-icon\">\n          <span class=\"label clr-file-label\">\n            {{ file.name }}\n            <button\n              class=\"btn btn-sm btn-link-neutral btn-icon clr-file-clear-button\"\n              [attr.aria-label]=\"getClearFileLabel(file.name)\"\n              (click)=\"clearFile(file)\"\n            >\n              <cds-icon shape=\"times\"></cds-icon>\n            </button>\n          </span>\n\n          <cds-icon\n            class=\"clr-validate-icon\"\n            [attr.shape]=\"fileMessagesTemplateContext.success ? 'check-circle' : 'exclamation-circle'\"\n            [attr.status]=\"fileMessagesTemplateContext.success ? 'success' : 'danger'\"\n            aria-hidden=\"true\"\n          ></cds-icon>\n        </div>\n\n        <ng-container\n          *ngIf=\"fileMessagesTemplate\"\n          [ngTemplateOutlet]=\"fileMessagesTemplate.templateRef\"\n          [ngTemplateOutletContext]=\"fileMessagesTemplateContext\"\n          [ngTemplateOutletInjector]=\"createFileMessagesTemplateInjector(fileMessagesTemplateContext)\"\n        ></ng-container>\n      </div>\n    </ng-container>\n  `,\n  host: {\n    '[attr.role]': '\"list\"',\n    '[class.clr-file-list]': 'true',\n  },\n})\nexport class ClrFileList {\n  @ContentChild(ClrFileMessagesTemplate) protected readonly fileMessagesTemplate: ClrFileMessagesTemplate;\n\n  private readonly injector = inject(Injector);\n  private readonly commonStrings = inject(ClrCommonStringsService);\n  private readonly ngControlService = inject(NgControlService, { optional: true });\n  private readonly fileInputContainer = inject(ClrFileInputContainer, { optional: true });\n\n  constructor() {\n    if (!this.ngControlService || !this.fileInputContainer) {\n      throw new Error('The clr-file-list component can only be used within a clr-file-input-container.');\n    }\n  }\n\n  protected get files() {\n    if (!this.fileInputContainer.fileInput) {\n      return [];\n    }\n\n    const fileInputElement = this.fileInputContainer.fileInput.elementRef.nativeElement;\n\n    return Array.from(fileInputElement.files).sort((a, b) => a.name.localeCompare(b.name));\n  }\n\n  protected getClearFileLabel(filename: string) {\n    return this.commonStrings.parse(this.commonStrings.keys.clearFile, {\n      FILE: filename,\n    });\n  }\n\n  protected clearFile(fileToRemove: File) {\n    if (!this.fileInputContainer.fileInput) {\n      return;\n    }\n\n    const fileInputElement = this.fileInputContainer.fileInput.elementRef.nativeElement;\n    const files = Array.from(fileInputElement.files);\n    const newFiles = files.filter(file => file !== fileToRemove);\n\n    selectFiles(fileInputElement, newFiles);\n    this.fileInputContainer.focusBrowseButton();\n  }\n\n  protected createFileMessagesTemplateContext(file: File): ClrFileMessagesTemplateContext {\n    const fileInputErrors: ClrFileListValidationErrors = this.ngControlService.control.errors || {};\n\n    const errors: ClrSingleFileValidationErrors = {\n      accept: fileInputErrors.accept?.find(error => error.name === file.name),\n      minFileSize: fileInputErrors.minFileSize?.find(error => error.name === file.name),\n      maxFileSize: fileInputErrors.maxFileSize?.find(error => error.name === file.name),\n    };\n\n    const success = Object.values(errors).every(error => !error);\n\n    return { $implicit: file, success, errors };\n  }\n\n  protected createFileMessagesTemplateInjector(fileMessagesTemplateContext: ClrFileMessagesTemplateContext) {\n    return Injector.create({\n      parent: this.injector,\n      providers: [{ provide: CLR_FILE_MESSAGES_TEMPLATE_CONTEXT, useValue: fileMessagesTemplateContext }],\n    });\n  }\n}\n"]}