@clr/angular
Version:
Angular components for Clarity
245 lines (239 loc) • 25.5 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, forwardRef, inject, Input, ViewChild } from '@angular/core';
import { ClrCommonStringsService } from '../../utils/i18n/common-strings.service';
import { ClrAbstractContainer } from '../common/abstract-container';
import { IfControlStateService } from '../common/if-control-state/if-control-state.service';
import { ControlClassService } from '../common/providers/control-class.service';
import { ControlIdService } from '../common/providers/control-id.service';
import { NgControlService } from '../common/providers/ng-control.service';
import { ClrFileInput } from './file-input';
import { selectFiles } from './file-input.helpers';
import { ClrFileList } from './file-list';
import { ClrFileError, ClrFileSuccess } from './file-messages';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "../../icon/icon";
import * as i3 from "../common/label";
export class ClrFileInputContainer extends ClrAbstractContainer {
constructor() {
super(...arguments);
this.commonStrings = inject(ClrCommonStringsService);
}
get accept() {
return this.fileInput.elementRef.nativeElement.accept;
}
get multiple() {
return this.fileInput.elementRef.nativeElement.multiple;
}
get disabled() {
return this.fileInput.elementRef.nativeElement.disabled;
}
get browseButtonText() {
const selectionButtonLabel = this.fileList ? undefined : this.fileInput?.selection?.buttonLabel;
return selectionButtonLabel || this.customButtonLabel || this.commonStrings.keys.browse;
}
get browseButtonDescribedBy() {
return `${this.label?.forAttr} ${this.fileInput.elementRef.nativeElement.getAttribute('aria-describedby')}`;
}
get successMessagePresent() {
return super.successMessagePresent || !!this.fileSuccessComponent;
}
get errorMessagePresent() {
return super.errorMessagePresent || !!this.fileErrorComponent;
}
focusBrowseButton() {
this.browseButtonElementRef.nativeElement.focus();
}
browse() {
const fileInputElementRef = this.fileList && this.multiple ? this.fileListFileInputElementRef : this.fileInput.elementRef;
fileInputElementRef.nativeElement.click();
}
clearSelectedFiles() {
this.fileInput.elementRef.nativeElement.value = '';
this.fileInput.elementRef.nativeElement.dispatchEvent(new Event('change', { bubbles: true, cancelable: true }));
this.focusBrowseButton();
}
addFilesToSelection(newFiles) {
if (!newFiles.length) {
return;
}
// start with new files
const mergedFiles = [...newFiles];
// add existing files if a new file doesn't have the same name
for (const existingFile of this.fileInput.elementRef.nativeElement.files) {
if (!mergedFiles.some(file => file.name === existingFile.name)) {
mergedFiles.push(existingFile);
}
}
// update file selection
selectFiles(this.fileInput.elementRef.nativeElement, mergedFiles);
}
}
ClrFileInputContainer.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrFileInputContainer, deps: null, target: i0.ɵɵFactoryTarget.Component });
ClrFileInputContainer.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrFileInputContainer, selector: "clr-file-input-container", inputs: { customButtonLabel: ["clrButtonLabel", "customButtonLabel"] }, host: { properties: { "class.clr-form-control": "true", "class.clr-form-control-disabled": "disabled", "class.clr-row": "addGrid()" } }, providers: [IfControlStateService, NgControlService, ControlIdService, ControlClassService], queries: [{ propertyName: "fileInput", first: true, predicate: i0.forwardRef(function () { return ClrFileInput; }), descendants: true }, { propertyName: "fileList", first: true, predicate: i0.forwardRef(function () { return ClrFileList; }), descendants: true }, { propertyName: "fileSuccessComponent", first: true, predicate: ClrFileSuccess, descendants: true }, { propertyName: "fileErrorComponent", first: true, predicate: ClrFileError, descendants: true }], viewQueries: [{ propertyName: "browseButtonElementRef", first: true, predicate: ["browseButton"], descendants: true }, { propertyName: "fileListFileInputElementRef", first: true, predicate: ["fileListFileInput"], descendants: true }], usesInheritance: true, ngImport: i0, template: `
<ng-content select="label"></ng-content>
<label *ngIf="!label && addGrid()"></label>
<div class="clr-control-container" [ngClass]="controlClass()">
<div class="clr-file-input-wrapper">
<ng-content select="[clrFileInput]"></ng-content>
<!-- file input to handle adding new files to selection when file list is present (prevent replacing selected files on the main file input) -->
<input
*ngIf="fileList"
#fileListFileInput
type="file"
class="clr-file-input"
tabindex="-1"
aria-hidden="true"
[accept]="accept"
[multiple]="multiple"
[disabled]="disabled"
(change)="addFilesToSelection(fileListFileInput.files)"
/>
<button
#browseButton
type="button"
class="btn btn-sm clr-file-input-browse-button"
[attr.aria-describedby]="browseButtonDescribedBy"
[disabled]="disabled"
(click)="browse()"
>
<cds-icon shape="folder-open"></cds-icon>
<span class="clr-file-input-browse-button-text">{{ browseButtonText }}</span>
</button>
<button
*ngIf="!fileList && fileInput?.selection?.fileCount"
type="button"
class="btn btn-sm clr-file-input-clear-button"
[attr.aria-label]="fileInput?.selection?.clearFilesButtonLabel"
(click)="clearSelectedFiles()"
>
<cds-icon shape="times" status="neutral" size="md"></cds-icon>
</button>
<cds-icon
*ngIf="showInvalid"
class="clr-validate-icon"
shape="exclamation-circle"
status="danger"
aria-hidden="true"
></cds-icon>
<cds-icon
*ngIf="showValid"
class="clr-validate-icon"
shape="check-circle"
status="success"
aria-hidden="true"
></cds-icon>
</div>
<ng-content select="clr-control-helper" *ngIf="showHelper"></ng-content>
<ng-content select="clr-control-error" *ngIf="showInvalid"></ng-content>
<ng-content select="clr-control-success" *ngIf="showValid"></ng-content>
<!-- If this is present, this file input becomes an "advanced" file input. -->
<ng-container>
<div class="clr-file-list-break"></div>
<ng-content select="clr-file-list"></ng-content>
</ng-container>
</div>
`, isInline: true, dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.CdsIconCustomTag, selector: "cds-icon" }, { kind: "directive", type: i3.ClrLabel, selector: "label", inputs: ["id", "for"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrFileInputContainer, decorators: [{
type: Component,
args: [{
selector: 'clr-file-input-container',
template: `
<ng-content select="label"></ng-content>
<label *ngIf="!label && addGrid()"></label>
<div class="clr-control-container" [ngClass]="controlClass()">
<div class="clr-file-input-wrapper">
<ng-content select="[clrFileInput]"></ng-content>
<!-- file input to handle adding new files to selection when file list is present (prevent replacing selected files on the main file input) -->
<input
*ngIf="fileList"
#fileListFileInput
type="file"
class="clr-file-input"
tabindex="-1"
aria-hidden="true"
[accept]="accept"
[multiple]="multiple"
[disabled]="disabled"
(change)="addFilesToSelection(fileListFileInput.files)"
/>
<button
#browseButton
type="button"
class="btn btn-sm clr-file-input-browse-button"
[attr.aria-describedby]="browseButtonDescribedBy"
[disabled]="disabled"
(click)="browse()"
>
<cds-icon shape="folder-open"></cds-icon>
<span class="clr-file-input-browse-button-text">{{ browseButtonText }}</span>
</button>
<button
*ngIf="!fileList && fileInput?.selection?.fileCount"
type="button"
class="btn btn-sm clr-file-input-clear-button"
[attr.aria-label]="fileInput?.selection?.clearFilesButtonLabel"
(click)="clearSelectedFiles()"
>
<cds-icon shape="times" status="neutral" size="md"></cds-icon>
</button>
<cds-icon
*ngIf="showInvalid"
class="clr-validate-icon"
shape="exclamation-circle"
status="danger"
aria-hidden="true"
></cds-icon>
<cds-icon
*ngIf="showValid"
class="clr-validate-icon"
shape="check-circle"
status="success"
aria-hidden="true"
></cds-icon>
</div>
<ng-content select="clr-control-helper" *ngIf="showHelper"></ng-content>
<ng-content select="clr-control-error" *ngIf="showInvalid"></ng-content>
<ng-content select="clr-control-success" *ngIf="showValid"></ng-content>
<!-- If this is present, this file input becomes an "advanced" file input. -->
<ng-container>
<div class="clr-file-list-break"></div>
<ng-content select="clr-file-list"></ng-content>
</ng-container>
</div>
`,
host: {
'[class.clr-form-control]': 'true',
'[class.clr-form-control-disabled]': 'disabled',
'[class.clr-row]': 'addGrid()',
},
providers: [IfControlStateService, NgControlService, ControlIdService, ControlClassService],
}]
}], propDecorators: { customButtonLabel: [{
type: Input,
args: ['clrButtonLabel']
}], fileInput: [{
type: ContentChild,
args: [forwardRef(() => ClrFileInput)]
}], fileList: [{
type: ContentChild,
args: [forwardRef(() => ClrFileList)]
}], browseButtonElementRef: [{
type: ViewChild,
args: ['browseButton']
}], fileListFileInputElementRef: [{
type: ViewChild,
args: ['fileListFileInput']
}], fileSuccessComponent: [{
type: ContentChild,
args: [ClrFileSuccess]
}], fileErrorComponent: [{
type: ContentChild,
args: [ClrFileError]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-input-container.js","sourceRoot":"","sources":["../../../../../projects/angular/src/forms/file-input/file-input-container.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, ElementRef, forwardRef, inject, Input, ViewChild } from '@angular/core';\n\nimport { ClrCommonStringsService } from '../../utils/i18n/common-strings.service';\nimport { ClrAbstractContainer } from '../common/abstract-container';\nimport { IfControlStateService } from '../common/if-control-state/if-control-state.service';\nimport { ControlClassService } from '../common/providers/control-class.service';\nimport { ControlIdService } from '../common/providers/control-id.service';\nimport { NgControlService } from '../common/providers/ng-control.service';\nimport { ClrFileInput } from './file-input';\nimport { selectFiles } from './file-input.helpers';\nimport { ClrFileList } from './file-list';\nimport { ClrFileError, ClrFileSuccess } from './file-messages';\n\n@Component({\n  selector: 'clr-file-input-container',\n  template: `\n    <ng-content select=\"label\"></ng-content>\n    <label *ngIf=\"!label && addGrid()\"></label>\n    <div class=\"clr-control-container\" [ngClass]=\"controlClass()\">\n      <div class=\"clr-file-input-wrapper\">\n        <ng-content select=\"[clrFileInput]\"></ng-content>\n\n        <!-- file input to handle adding new files to selection when file list is present (prevent replacing selected files on the main file input) -->\n        <input\n          *ngIf=\"fileList\"\n          #fileListFileInput\n          type=\"file\"\n          class=\"clr-file-input\"\n          tabindex=\"-1\"\n          aria-hidden=\"true\"\n          [accept]=\"accept\"\n          [multiple]=\"multiple\"\n          [disabled]=\"disabled\"\n          (change)=\"addFilesToSelection(fileListFileInput.files)\"\n        />\n\n        <button\n          #browseButton\n          type=\"button\"\n          class=\"btn btn-sm clr-file-input-browse-button\"\n          [attr.aria-describedby]=\"browseButtonDescribedBy\"\n          [disabled]=\"disabled\"\n          (click)=\"browse()\"\n        >\n          <cds-icon shape=\"folder-open\"></cds-icon>\n          <span class=\"clr-file-input-browse-button-text\">{{ browseButtonText }}</span>\n        </button>\n        <button\n          *ngIf=\"!fileList && fileInput?.selection?.fileCount\"\n          type=\"button\"\n          class=\"btn btn-sm clr-file-input-clear-button\"\n          [attr.aria-label]=\"fileInput?.selection?.clearFilesButtonLabel\"\n          (click)=\"clearSelectedFiles()\"\n        >\n          <cds-icon shape=\"times\" status=\"neutral\" size=\"md\"></cds-icon>\n        </button>\n        <cds-icon\n          *ngIf=\"showInvalid\"\n          class=\"clr-validate-icon\"\n          shape=\"exclamation-circle\"\n          status=\"danger\"\n          aria-hidden=\"true\"\n        ></cds-icon>\n        <cds-icon\n          *ngIf=\"showValid\"\n          class=\"clr-validate-icon\"\n          shape=\"check-circle\"\n          status=\"success\"\n          aria-hidden=\"true\"\n        ></cds-icon>\n      </div>\n      <ng-content select=\"clr-control-helper\" *ngIf=\"showHelper\"></ng-content>\n      <ng-content select=\"clr-control-error\" *ngIf=\"showInvalid\"></ng-content>\n      <ng-content select=\"clr-control-success\" *ngIf=\"showValid\"></ng-content>\n\n      <!-- If this is present, this file input becomes an \"advanced\" file input. -->\n      <ng-container>\n        <div class=\"clr-file-list-break\"></div>\n        <ng-content select=\"clr-file-list\"></ng-content>\n      </ng-container>\n    </div>\n  `,\n  host: {\n    '[class.clr-form-control]': 'true',\n    '[class.clr-form-control-disabled]': 'disabled',\n    '[class.clr-row]': 'addGrid()',\n  },\n  providers: [IfControlStateService, NgControlService, ControlIdService, ControlClassService],\n})\nexport class ClrFileInputContainer extends ClrAbstractContainer {\n  @Input('clrButtonLabel') customButtonLabel: string;\n\n  @ContentChild(forwardRef(() => ClrFileInput)) readonly fileInput: ClrFileInput;\n  @ContentChild(forwardRef(() => ClrFileList)) protected readonly fileList: ClrFileList;\n\n  @ViewChild('browseButton') private browseButtonElementRef: ElementRef<HTMLButtonElement>;\n  @ViewChild('fileListFileInput') private fileListFileInputElementRef: ElementRef<HTMLInputElement>;\n\n  // These are for the \"message present\" override properties\n  @ContentChild(ClrFileSuccess) private readonly fileSuccessComponent: ClrFileSuccess;\n  @ContentChild(ClrFileError) private readonly fileErrorComponent: ClrFileError;\n\n  private readonly commonStrings = inject(ClrCommonStringsService);\n\n  protected get accept() {\n    return this.fileInput.elementRef.nativeElement.accept;\n  }\n\n  protected get multiple() {\n    return this.fileInput.elementRef.nativeElement.multiple;\n  }\n\n  protected get disabled() {\n    return this.fileInput.elementRef.nativeElement.disabled;\n  }\n\n  protected get browseButtonText() {\n    const selectionButtonLabel = this.fileList ? undefined : this.fileInput?.selection?.buttonLabel;\n\n    return selectionButtonLabel || this.customButtonLabel || this.commonStrings.keys.browse;\n  }\n\n  protected get browseButtonDescribedBy() {\n    return `${this.label?.forAttr} ${this.fileInput.elementRef.nativeElement.getAttribute('aria-describedby')}`;\n  }\n\n  protected override get successMessagePresent() {\n    return super.successMessagePresent || !!this.fileSuccessComponent;\n  }\n\n  protected override get errorMessagePresent() {\n    return super.errorMessagePresent || !!this.fileErrorComponent;\n  }\n\n  focusBrowseButton() {\n    this.browseButtonElementRef.nativeElement.focus();\n  }\n\n  protected browse() {\n    const fileInputElementRef =\n      this.fileList && this.multiple ? this.fileListFileInputElementRef : this.fileInput.elementRef;\n\n    fileInputElementRef.nativeElement.click();\n  }\n\n  protected clearSelectedFiles() {\n    this.fileInput.elementRef.nativeElement.value = '';\n    this.fileInput.elementRef.nativeElement.dispatchEvent(new Event('change', { bubbles: true, cancelable: true }));\n\n    this.focusBrowseButton();\n  }\n\n  protected addFilesToSelection(newFiles: FileList) {\n    if (!newFiles.length) {\n      return;\n    }\n\n    // start with new files\n    const mergedFiles = [...newFiles];\n\n    // add existing files if a new file doesn't have the same name\n    for (const existingFile of this.fileInput.elementRef.nativeElement.files) {\n      if (!mergedFiles.some(file => file.name === existingFile.name)) {\n        mergedFiles.push(existingFile);\n      }\n    }\n\n    // update file selection\n    selectFiles(this.fileInput.elementRef.nativeElement, mergedFiles);\n  }\n}\n"]}