@clr/angular
Version:
Angular components for Clarity
166 lines (164 loc) • 17.8 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 { isPlatformBrowser } from '@angular/common';
import { Component, EventEmitter, Inject, Input, Output, PLATFORM_ID, ViewChild } from '@angular/core';
import { ClrKeyFocus } from '../../utils/focus/key-focus';
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 * as i0 from "@angular/core";
import * as i1 from "./providers/row-action-service";
import * as i2 from "../../utils/i18n/common-strings.service";
import * as i3 from "../../utils/popover/providers/popover-toggle.service";
import * as i4 from "../../utils/popover/popover-host.directive";
import * as i5 from "../../utils/cdk/cdk-trap-focus.module";
import * as i6 from "../../icon/icon";
import * as i7 from "../../utils/popover/popover-anchor";
import * as i8 from "../../utils/popover/popover-open-close-button";
import * as i9 from "../../utils/popover/popover-content";
import * as i10 from "../../utils/focus/key-focus/key-focus";
let clrDgActionId = 0;
export class ClrDatagridActionOverflow {
constructor(rowActionService, commonStrings, platformId, smartToggleService) {
this.rowActionService = rowActionService;
this.commonStrings = commonStrings;
this.platformId = platformId;
this.smartToggleService = smartToggleService;
this.openChange = new EventEmitter(false);
this.popoverId = uniqueIdFactory();
this.smartPosition = {
axis: ClrAxis.HORIZONTAL,
side: ClrSide.AFTER,
anchor: ClrAlignment.CENTER,
content: ClrAlignment.CENTER,
};
this._open = false;
this.subscriptions = [];
rowActionService.register();
this.subscriptions.push(smartToggleService.openChange.subscribe(openState => {
this.open = openState;
}), smartToggleService.popoverVisible.subscribe(visible => {
if (visible) {
this.initializeFocus();
}
}));
this.popoverId = 'clr-action-menu' + clrDgActionId++;
}
get open() {
return this._open;
}
set open(open) {
const openState = !!open;
if (!!openState !== this.open) {
// prevents chocolate mess
this.smartToggleService.open = openState;
this.openChange.emit(openState);
this._open = openState;
}
}
ngOnDestroy() {
this.rowActionService.unregister();
this.subscriptions.forEach(sub => sub.unsubscribe());
}
closeOverflowContent(event) {
this.smartToggleService.toggleWithEvent(event);
}
initializeFocus() {
if (isPlatformBrowser(this.platformId)) {
const buttons = Array.from(document.querySelectorAll('button.action-item'));
if (buttons.length) {
this.keyFocus.current = 0;
this.keyFocus.focusableItems = buttons;
this.keyFocus.focusCurrent();
}
}
}
}
ClrDatagridActionOverflow.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrDatagridActionOverflow, deps: [{ token: i1.RowActionService }, { token: i2.ClrCommonStringsService }, { token: PLATFORM_ID }, { token: i3.ClrPopoverToggleService }], target: i0.ɵɵFactoryTarget.Component });
ClrDatagridActionOverflow.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrDatagridActionOverflow, selector: "clr-dg-action-overflow", inputs: { buttonLabel: ["clrDgActionOverflowButtonLabel", "buttonLabel"], open: ["clrDgActionOverflowOpen", "open"] }, outputs: { openChange: "clrDgActionOverflowOpenChange" }, viewQueries: [{ propertyName: "keyFocus", first: true, predicate: ClrKeyFocus, descendants: true }], hostDirectives: [{ directive: i4.ClrPopoverHostDirective }], ngImport: i0, template: `
<button
tabindex="-1"
class="datagrid-action-toggle"
type="button"
role="button"
aria-haspopup="true"
#anchor
[attr.aria-controls]="popoverId"
[attr.aria-expanded]="open"
[attr.aria-label]="buttonLabel || commonStrings.keys.rowActions"
clrPopoverAnchor
clrPopoverOpenCloseButton
>
<cds-icon shape="ellipsis-vertical" [attr.title]="buttonLabel || commonStrings.keys.rowActions"></cds-icon>
</button>
<div
class="datagrid-action-overflow"
[id]="popoverId"
[attr.aria-hidden]="!open"
[attr.id]="popoverId"
clrKeyFocus
cdkTrapFocus
(click)="closeOverflowContent($event)"
*clrPopoverContent="open; at: smartPosition; outsideClickToClose: true; scrollToClose: true"
>
<ng-content></ng-content>
</div>
`, isInline: true, dependencies: [{ kind: "directive", type: i5.CdkTrapFocusModule_CdkTrapFocus, selector: "[cdkTrapFocus]" }, { kind: "directive", type: i6.CdsIconCustomTag, selector: "cds-icon" }, { kind: "directive", type: i7.ClrPopoverAnchor, selector: "[clrPopoverAnchor]" }, { kind: "directive", type: i8.ClrPopoverOpenCloseButton, selector: "[clrPopoverOpenCloseButton]", outputs: ["clrPopoverOpenCloseChange"] }, { kind: "directive", type: i9.ClrPopoverContent, selector: "[clrPopoverContent]", inputs: ["clrPopoverContent", "clrPopoverContentAt", "clrPopoverContentOutsideClickToClose", "clrPopoverContentScrollToClose"] }, { kind: "component", type: i10.ClrKeyFocus, selector: "[clrKeyFocus]", inputs: ["clrDirection", "clrFocusOnLoad", "clrKeyFocus"], outputs: ["clrFocusChange"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrDatagridActionOverflow, decorators: [{
type: Component,
args: [{
selector: 'clr-dg-action-overflow',
hostDirectives: [ClrPopoverHostDirective],
template: `
<button
tabindex="-1"
class="datagrid-action-toggle"
type="button"
role="button"
aria-haspopup="true"
#anchor
[attr.aria-controls]="popoverId"
[attr.aria-expanded]="open"
[attr.aria-label]="buttonLabel || commonStrings.keys.rowActions"
clrPopoverAnchor
clrPopoverOpenCloseButton
>
<cds-icon shape="ellipsis-vertical" [attr.title]="buttonLabel || commonStrings.keys.rowActions"></cds-icon>
</button>
<div
class="datagrid-action-overflow"
[id]="popoverId"
[attr.aria-hidden]="!open"
[attr.id]="popoverId"
clrKeyFocus
cdkTrapFocus
(click)="closeOverflowContent($event)"
*clrPopoverContent="open; at: smartPosition; outsideClickToClose: true; scrollToClose: true"
>
<ng-content></ng-content>
</div>
`,
}]
}], ctorParameters: function () { return [{ type: i1.RowActionService }, { type: i2.ClrCommonStringsService }, { type: undefined, decorators: [{
type: Inject,
args: [PLATFORM_ID]
}] }, { type: i3.ClrPopoverToggleService }]; }, propDecorators: { buttonLabel: [{
type: Input,
args: ['clrDgActionOverflowButtonLabel']
}], openChange: [{
type: Output,
args: ['clrDgActionOverflowOpenChange']
}], keyFocus: [{
type: ViewChild,
args: [ClrKeyFocus]
}], open: [{
type: Input,
args: ['clrDgActionOverflowOpen']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datagrid-action-overflow.js","sourceRoot":"","sources":["../../../../../projects/angular/src/data/datagrid/datagrid-action-overflow.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 { isPlatformBrowser } from '@angular/common';\nimport { Component, EventEmitter, Inject, Input, OnDestroy, Output, PLATFORM_ID, ViewChild } from '@angular/core';\nimport { Subscription } from 'rxjs';\n\nimport { ClrKeyFocus } from '../../utils/focus/key-focus';\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 { RowActionService } from './providers/row-action-service';\n\nlet clrDgActionId = 0;\n\n@Component({\n  selector: 'clr-dg-action-overflow',\n  hostDirectives: [ClrPopoverHostDirective],\n  template: `\n    <button\n      tabindex=\"-1\"\n      class=\"datagrid-action-toggle\"\n      type=\"button\"\n      role=\"button\"\n      aria-haspopup=\"true\"\n      #anchor\n      [attr.aria-controls]=\"popoverId\"\n      [attr.aria-expanded]=\"open\"\n      [attr.aria-label]=\"buttonLabel || commonStrings.keys.rowActions\"\n      clrPopoverAnchor\n      clrPopoverOpenCloseButton\n    >\n      <cds-icon shape=\"ellipsis-vertical\" [attr.title]=\"buttonLabel || commonStrings.keys.rowActions\"></cds-icon>\n    </button>\n\n    <div\n      class=\"datagrid-action-overflow\"\n      [id]=\"popoverId\"\n      [attr.aria-hidden]=\"!open\"\n      [attr.id]=\"popoverId\"\n      clrKeyFocus\n      cdkTrapFocus\n      (click)=\"closeOverflowContent($event)\"\n      *clrPopoverContent=\"open; at: smartPosition; outsideClickToClose: true; scrollToClose: true\"\n    >\n      <ng-content></ng-content>\n    </div>\n  `,\n})\nexport class ClrDatagridActionOverflow implements OnDestroy {\n  @Input('clrDgActionOverflowButtonLabel') buttonLabel: string;\n\n  @Output('clrDgActionOverflowOpenChange') openChange = new EventEmitter<boolean>(false);\n\n  popoverId = uniqueIdFactory();\n\n  smartPosition: ClrPopoverPosition = {\n    axis: ClrAxis.HORIZONTAL,\n    side: ClrSide.AFTER,\n    anchor: ClrAlignment.CENTER,\n    content: ClrAlignment.CENTER,\n  };\n\n  @ViewChild(ClrKeyFocus) private readonly keyFocus: ClrKeyFocus;\n\n  private _open = false;\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    private rowActionService: RowActionService,\n    public commonStrings: ClrCommonStringsService,\n    @Inject(PLATFORM_ID) private platformId: any,\n    private smartToggleService: ClrPopoverToggleService\n  ) {\n    rowActionService.register();\n    this.subscriptions.push(\n      smartToggleService.openChange.subscribe(openState => {\n        this.open = openState;\n      }),\n      smartToggleService.popoverVisible.subscribe(visible => {\n        if (visible) {\n          this.initializeFocus();\n        }\n      })\n    );\n    this.popoverId = 'clr-action-menu' + clrDgActionId++;\n  }\n\n  @Input('clrDgActionOverflowOpen')\n  get open() {\n    return this._open;\n  }\n  set open(open: boolean) {\n    const openState = !!open;\n    if (!!openState !== this.open) {\n      // prevents chocolate mess\n      this.smartToggleService.open = openState;\n      this.openChange.emit(openState);\n      this._open = openState;\n    }\n  }\n\n  ngOnDestroy() {\n    this.rowActionService.unregister();\n    this.subscriptions.forEach(sub => sub.unsubscribe());\n  }\n\n  closeOverflowContent(event: Event): void {\n    this.smartToggleService.toggleWithEvent(event);\n  }\n\n  private initializeFocus(): void {\n    if (isPlatformBrowser(this.platformId)) {\n      const buttons = Array.from(document.querySelectorAll<HTMLButtonElement>('button.action-item'));\n\n      if (buttons.length) {\n        this.keyFocus.current = 0;\n        this.keyFocus.focusableItems = buttons;\n\n        this.keyFocus.focusCurrent();\n      }\n    }\n  }\n}\n"]}