@clr/angular
Version:
Angular components for Clarity
160 lines (156 loc) • 18.6 kB
JavaScript
/*
* 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":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAE1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,yCAAyC,CAAC;AAClF,OAAO,EAAE,gBAAgB,EAAE,MAAM,wCAAwC,CAAC;AAC1E,OAAO,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAE/D,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,kCAAkC,EAAE,MAAM,iBAAiB,CAAC;AACrE,OAAO,EACL,uBAAuB,GAGxB,MAAM,0BAA0B,CAAC;;;;AAiDlC,MAAM,OAAO,WAAW;IAQtB;QALiB,aAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC5B,kBAAa,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAChD,qBAAgB,GAAG,MAAM,CAAC,gBAAgB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QAChE,uBAAkB,GAAG,MAAM,CAAC,qBAAqB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QAGtF,IAAI,CAAC,IAAI,CAAC,gBAAgB,IAAI,CAAC,IAAI,CAAC,kBAAkB,EAAE;YACtD,MAAM,IAAI,KAAK,CAAC,iFAAiF,CAAC,CAAC;SACpG;IACH,CAAC;IAED,IAAc,KAAK;QACjB,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE;YACtC,OAAO,EAAE,CAAC;SACX;QAED,MAAM,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,UAAU,CAAC,aAAa,CAAC;QAEpF,OAAO,KAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;IACzF,CAAC;IAES,iBAAiB,CAAC,QAAgB;QAC1C,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,EAAE;YACjE,IAAI,EAAE,QAAQ;SACf,CAAC,CAAC;IACL,CAAC;IAES,SAAS,CAAC,YAAkB;QACpC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE;YACtC,OAAO;SACR;QAED,MAAM,gBAAgB,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,UAAU,CAAC,aAAa,CAAC;QACpF,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QACjD,MAAM,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,KAAK,YAAY,CAAC,CAAC;QAE7D,WAAW,CAAC,gBAAgB,EAAE,QAAQ,CAAC,CAAC;QACxC,IAAI,CAAC,kBAAkB,CAAC,iBAAiB,EAAE,CAAC;IAC9C,CAAC;IAES,iCAAiC,CAAC,IAAU;QACpD,MAAM,eAAe,GAAgC,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC;QAEhG,MAAM,MAAM,GAAkC;YAC5C,MAAM,EAAE,eAAe,CAAC,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC;YACvE,WAAW,EAAE,eAAe,CAAC,WAAW,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC;YACjF,WAAW,EAAE,eAAe,CAAC,WAAW,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC;SAClF,CAAC;QAEF,MAAM,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC;QAE7D,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC;IAC9C,CAAC;IAES,kCAAkC,CAAC,2BAA2D;QACtG,OAAO,QAAQ,CAAC,MAAM,CAAC;YACrB,MAAM,EAAE,IAAI,CAAC,QAAQ;YACrB,SAAS,EAAE,CAAC,EAAE,OAAO,EAAE,kCAAkC,EAAE,QAAQ,EAAE,2BAA2B,EAAE,CAAC;SACpG,CAAC,CAAC;IACL,CAAC;;wGA9DU,WAAW;4FAAX,WAAW,yLACR,uBAAuB,gDA9C3B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuCT;2FAMU,WAAW;kBA/CvB,SAAS;mBAAC;oBACT,QAAQ,EAAE,eAAe;oBACzB,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAuCT;oBACD,IAAI,EAAE;wBACJ,aAAa,EAAE,QAAQ;wBACvB,uBAAuB,EAAE,MAAM;qBAChC;iBACF;0EAE2D,oBAAoB;sBAA7E,YAAY;uBAAC,uBAAuB","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"]}