UNPKG

@clr/angular

Version:

Angular components for Clarity

204 lines 23 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, ElementRef, ViewChild } from '@angular/core'; import { uniqueIdFactory } from '../../utils/id-generator/id-generator.service'; import { ClrAlignment } from '../../utils/popover/enums/alignment.enum'; import { ClrAxis } from '../../utils/popover/enums/axis.enum'; import { ClrSide } from '../../utils/popover/enums/side.enum'; import { ClrPopoverHostDirective } from '../../utils/popover/popover-host.directive'; import { columnToggleTrackByFn } from './datagrid-column-toggle-trackby'; import { DatagridColumnChanges } from './enums/column-changes.enum'; import * as i0 from "@angular/core"; import * as i1 from "../../utils/i18n/common-strings.service"; import * as i2 from "./providers/columns.service"; import * as i3 from "../../utils/popover/providers/popover-toggle.service"; import * as i4 from "../../utils/popover/popover-host.directive"; import * as i5 from "@angular/common"; import * as i6 from "../../utils/cdk/cdk-trap-focus.module"; import * as i7 from "../../icon/icon"; import * as i8 from "../../forms/common/label"; import * as i9 from "../../forms/checkbox/checkbox"; import * as i10 from "../../forms/checkbox/checkbox-wrapper"; import * as i11 from "@angular/forms"; import * as i12 from "../../utils/popover/popover-anchor"; import * as i13 from "../../utils/popover/popover-close-button"; import * as i14 from "../../utils/popover/popover-open-close-button"; import * as i15 from "../../utils/popover/popover-content"; import * as i16 from "./datagrid-column-toggle-button"; export class ClrDatagridColumnToggle { constructor(commonStrings, columnsService, popoverToggleService) { this.commonStrings = commonStrings; this.columnsService = columnsService; this.popoverId = uniqueIdFactory(); this.openState = false; // Smart Popover this.smartPosition = { axis: ClrAxis.VERTICAL, side: ClrSide.BEFORE, anchor: ClrAlignment.START, content: ClrAlignment.START, }; // Without tracking the checkboxes get rerendered on model update, which leads // to loss of focus after checkbox toggle. this.trackByFn = columnToggleTrackByFn; this.subscription = popoverToggleService.openChange.subscribe(change => (this.openState = change)); } get allColumnsVisible() { return this._allColumnsVisible; } set allColumnsVisible(value) { this._allColumnsVisible = value; } get hideableColumnStates() { const hideables = this.columnsService.columns.filter(column => column.value.hideable); return hideables.map(column => column.value); } get hasOnlyOneVisibleColumn() { const nbNonHideableColumns = this.columnsService.columns.length - this.hideableColumnStates.length; // this should only return true when there is no non-hideable columns. return (nbNonHideableColumns === 0 && this.hideableColumnStates.filter(columnState => !columnState.hidden).length === 1); } ngOnDestroy() { this.subscription.unsubscribe(); } toggleColumnState(columnState, event) { const columnToToggle = this.columnsService.columns.filter(column => column.value === columnState)[0]; this.columnsService.emitStateChange(columnToToggle, { hidden: event, changes: [DatagridColumnChanges.HIDDEN], }); } toggleSwitchPanel() { this.openState = !this.openState; } allColumnsSelected() { this.allSelectedElement.nativeElement.focus(); } } ClrDatagridColumnToggle.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrDatagridColumnToggle, deps: [{ token: i1.ClrCommonStringsService }, { token: i2.ColumnsService }, { token: i3.ClrPopoverToggleService }], target: i0.ɵɵFactoryTarget.Component }); ClrDatagridColumnToggle.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrDatagridColumnToggle, selector: "clr-dg-column-toggle", host: { properties: { "class.column-switch-wrapper": "true", "class.active": "openState" } }, viewQueries: [{ propertyName: "allSelectedElement", first: true, predicate: ["allSelected"], descendants: true, read: ElementRef }], hostDirectives: [{ directive: i4.ClrPopoverHostDirective }], ngImport: i0, template: ` <button role="button" type="button" class="btn btn-sm column-toggle--action" clrPopoverAnchor clrPopoverOpenCloseButton [attr.aria-controls]="popoverId" [attr.aria-expanded]="openState" [attr.aria-haspopup]="'menu'" > {{ commonStrings.keys.pickColumns }} </button> <div class="column-switch" role="dialog" [attr.aria-label]="commonStrings.keys.showColumnsMenuDescription" [id]="popoverId" cdkTrapFocus *clrPopoverContent="openState; at: smartPosition; outsideClickToClose: true; scrollToClose: true" > <div class="switch-header"> <div class="clr-sr-only" tabindex="-1" #allSelected>{{ commonStrings.keys.allColumnsSelected }}</div> <h2>{{ commonStrings.keys.showColumns }}</h2> <button class="btn btn-sm btn-link toggle-switch-close-button" clrPopoverCloseButton type="button" [attr.aria-label]="commonStrings.keys.close" > <cds-icon shape="window-close" aria-hidden="true" [attr.title]="commonStrings.keys.close"></cds-icon> <span class="clr-sr-only">{{ commonStrings.keys.close }}</span> </button> </div> <ul class="switch-content list-unstyled"> <li *ngFor="let columnState of hideableColumnStates; trackBy: trackByFn"> <clr-checkbox-wrapper> <input clrCheckbox type="checkbox" [disabled]="hasOnlyOneVisibleColumn && !columnState.hidden" [ngModel]="!columnState.hidden" (ngModelChange)="toggleColumnState(columnState, !$event)" /> <label> <ng-template [ngTemplateOutlet]="columnState.titleTemplateRef"></ng-template> </label> </clr-checkbox-wrapper> </li> </ul> <div class="switch-footer"> <clr-dg-column-toggle-button (clrAllSelected)="allColumnsSelected()"></clr-dg-column-toggle-button> </div> </div> `, isInline: true, dependencies: [{ kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i6.CdkTrapFocusModule_CdkTrapFocus, selector: "[cdkTrapFocus]" }, { kind: "directive", type: i7.CdsIconCustomTag, selector: "cds-icon" }, { kind: "directive", type: i8.ClrLabel, selector: "label", inputs: ["id", "for"] }, { kind: "directive", type: i9.ClrCheckbox, selector: "[clrCheckbox],[clrToggle]" }, { kind: "component", type: i10.ClrCheckboxWrapper, selector: "clr-checkbox-wrapper,clr-toggle-wrapper" }, { kind: "directive", type: i11.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i11.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i11.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i12.ClrPopoverAnchor, selector: "[clrPopoverAnchor]" }, { kind: "directive", type: i13.ClrPopoverCloseButton, selector: "[clrPopoverCloseButton]", outputs: ["clrPopoverOnCloseChange"] }, { kind: "directive", type: i14.ClrPopoverOpenCloseButton, selector: "[clrPopoverOpenCloseButton]", outputs: ["clrPopoverOpenCloseChange"] }, { kind: "directive", type: i15.ClrPopoverContent, selector: "[clrPopoverContent]", inputs: ["clrPopoverContent", "clrPopoverContentAt", "clrPopoverContentOutsideClickToClose", "clrPopoverContentScrollToClose"] }, { kind: "component", type: i16.ClrDatagridColumnToggleButton, selector: "clr-dg-column-toggle-button", outputs: ["clrAllSelected"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrDatagridColumnToggle, decorators: [{ type: Component, args: [{ selector: 'clr-dg-column-toggle', template: ` <button role="button" type="button" class="btn btn-sm column-toggle--action" clrPopoverAnchor clrPopoverOpenCloseButton [attr.aria-controls]="popoverId" [attr.aria-expanded]="openState" [attr.aria-haspopup]="'menu'" > {{ commonStrings.keys.pickColumns }} </button> <div class="column-switch" role="dialog" [attr.aria-label]="commonStrings.keys.showColumnsMenuDescription" [id]="popoverId" cdkTrapFocus *clrPopoverContent="openState; at: smartPosition; outsideClickToClose: true; scrollToClose: true" > <div class="switch-header"> <div class="clr-sr-only" tabindex="-1" #allSelected>{{ commonStrings.keys.allColumnsSelected }}</div> <h2>{{ commonStrings.keys.showColumns }}</h2> <button class="btn btn-sm btn-link toggle-switch-close-button" clrPopoverCloseButton type="button" [attr.aria-label]="commonStrings.keys.close" > <cds-icon shape="window-close" aria-hidden="true" [attr.title]="commonStrings.keys.close"></cds-icon> <span class="clr-sr-only">{{ commonStrings.keys.close }}</span> </button> </div> <ul class="switch-content list-unstyled"> <li *ngFor="let columnState of hideableColumnStates; trackBy: trackByFn"> <clr-checkbox-wrapper> <input clrCheckbox type="checkbox" [disabled]="hasOnlyOneVisibleColumn && !columnState.hidden" [ngModel]="!columnState.hidden" (ngModelChange)="toggleColumnState(columnState, !$event)" /> <label> <ng-template [ngTemplateOutlet]="columnState.titleTemplateRef"></ng-template> </label> </clr-checkbox-wrapper> </li> </ul> <div class="switch-footer"> <clr-dg-column-toggle-button (clrAllSelected)="allColumnsSelected()"></clr-dg-column-toggle-button> </div> </div> `, host: { '[class.column-switch-wrapper]': 'true', '[class.active]': 'openState' }, hostDirectives: [ClrPopoverHostDirective], }] }], ctorParameters: function () { return [{ type: i1.ClrCommonStringsService }, { type: i2.ColumnsService }, { type: i3.ClrPopoverToggleService }]; }, propDecorators: { allSelectedElement: [{ type: ViewChild, args: ['allSelected', { read: ElementRef }] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datagrid-column-toggle.js","sourceRoot":"","sources":["../../../../../projects/angular/src/data/datagrid/datagrid-column-toggle.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, ElementRef, OnDestroy, ViewChild } from '@angular/core';\nimport { Subscription } from 'rxjs';\n\nimport { ClrCommonStringsService } from '../../utils/i18n/common-strings.service';\nimport { uniqueIdFactory } from '../../utils/id-generator/id-generator.service';\nimport { ClrAlignment } from '../../utils/popover/enums/alignment.enum';\nimport { ClrAxis } from '../../utils/popover/enums/axis.enum';\nimport { ClrSide } from '../../utils/popover/enums/side.enum';\nimport { ClrPopoverPosition } from '../../utils/popover/interfaces/popover-position.interface';\nimport { ClrPopoverHostDirective } from '../../utils/popover/popover-host.directive';\nimport { ClrPopoverToggleService } from '../../utils/popover/providers/popover-toggle.service';\nimport { columnToggleTrackByFn } from './datagrid-column-toggle-trackby';\nimport { DatagridColumnChanges } from './enums/column-changes.enum';\nimport { ColumnState } from './interfaces/column-state.interface';\nimport { ColumnsService } from './providers/columns.service';\n\n@Component({\n  selector: 'clr-dg-column-toggle',\n  template: `\n    <button\n      role=\"button\"\n      type=\"button\"\n      class=\"btn btn-sm column-toggle--action\"\n      clrPopoverAnchor\n      clrPopoverOpenCloseButton\n      [attr.aria-controls]=\"popoverId\"\n      [attr.aria-expanded]=\"openState\"\n      [attr.aria-haspopup]=\"'menu'\"\n    >\n      {{ commonStrings.keys.pickColumns }}\n    </button>\n    <div\n      class=\"column-switch\"\n      role=\"dialog\"\n      [attr.aria-label]=\"commonStrings.keys.showColumnsMenuDescription\"\n      [id]=\"popoverId\"\n      cdkTrapFocus\n      *clrPopoverContent=\"openState; at: smartPosition; outsideClickToClose: true; scrollToClose: true\"\n    >\n      <div class=\"switch-header\">\n        <div class=\"clr-sr-only\" tabindex=\"-1\" #allSelected>{{ commonStrings.keys.allColumnsSelected }}</div>\n        <h2>{{ commonStrings.keys.showColumns }}</h2>\n        <button\n          class=\"btn btn-sm btn-link toggle-switch-close-button\"\n          clrPopoverCloseButton\n          type=\"button\"\n          [attr.aria-label]=\"commonStrings.keys.close\"\n        >\n          <cds-icon shape=\"window-close\" aria-hidden=\"true\" [attr.title]=\"commonStrings.keys.close\"></cds-icon>\n          <span class=\"clr-sr-only\">{{ commonStrings.keys.close }}</span>\n        </button>\n      </div>\n      <ul class=\"switch-content list-unstyled\">\n        <li *ngFor=\"let columnState of hideableColumnStates; trackBy: trackByFn\">\n          <clr-checkbox-wrapper>\n            <input\n              clrCheckbox\n              type=\"checkbox\"\n              [disabled]=\"hasOnlyOneVisibleColumn && !columnState.hidden\"\n              [ngModel]=\"!columnState.hidden\"\n              (ngModelChange)=\"toggleColumnState(columnState, !$event)\"\n            />\n            <label>\n              <ng-template [ngTemplateOutlet]=\"columnState.titleTemplateRef\"></ng-template>\n            </label>\n          </clr-checkbox-wrapper>\n        </li>\n      </ul>\n      <div class=\"switch-footer\">\n        <clr-dg-column-toggle-button (clrAllSelected)=\"allColumnsSelected()\"></clr-dg-column-toggle-button>\n      </div>\n    </div>\n  `,\n  host: { '[class.column-switch-wrapper]': 'true', '[class.active]': 'openState' },\n  hostDirectives: [ClrPopoverHostDirective],\n})\nexport class ClrDatagridColumnToggle implements OnDestroy {\n  popoverId = uniqueIdFactory();\n  openState = false;\n\n  // Smart Popover\n  smartPosition: ClrPopoverPosition = {\n    axis: ClrAxis.VERTICAL,\n    side: ClrSide.BEFORE,\n    anchor: ClrAlignment.START,\n    content: ClrAlignment.START,\n  };\n\n  // Without tracking the checkboxes get rerendered on model update, which leads\n  // to loss of focus after checkbox toggle.\n  readonly trackByFn = columnToggleTrackByFn;\n\n  private _allColumnsVisible: boolean;\n  private subscription: Subscription;\n\n  @ViewChild('allSelected', { read: ElementRef }) private allSelectedElement: ElementRef<HTMLElement>;\n\n  constructor(\n    public commonStrings: ClrCommonStringsService,\n    private columnsService: ColumnsService,\n    popoverToggleService: ClrPopoverToggleService\n  ) {\n    this.subscription = popoverToggleService.openChange.subscribe(change => (this.openState = change));\n  }\n\n  get allColumnsVisible(): boolean {\n    return this._allColumnsVisible;\n  }\n  set allColumnsVisible(value: boolean) {\n    this._allColumnsVisible = value;\n  }\n\n  get hideableColumnStates(): ColumnState[] {\n    const hideables = this.columnsService.columns.filter(column => column.value.hideable);\n    return hideables.map(column => column.value);\n  }\n\n  get hasOnlyOneVisibleColumn(): boolean {\n    const nbNonHideableColumns = this.columnsService.columns.length - this.hideableColumnStates.length;\n    // this should only return true when there is no non-hideable columns.\n    return (\n      nbNonHideableColumns === 0 && this.hideableColumnStates.filter(columnState => !columnState.hidden).length === 1\n    );\n  }\n\n  ngOnDestroy() {\n    this.subscription.unsubscribe();\n  }\n\n  toggleColumnState(columnState: ColumnState, event: boolean) {\n    const columnToToggle = this.columnsService.columns.filter(column => column.value === columnState)[0];\n    this.columnsService.emitStateChange(columnToToggle, {\n      hidden: event,\n      changes: [DatagridColumnChanges.HIDDEN],\n    });\n  }\n\n  toggleSwitchPanel() {\n    this.openState = !this.openState;\n  }\n\n  allColumnsSelected() {\n    this.allSelectedElement.nativeElement.focus();\n  }\n}\n"]}