@clr/angular
Version:
Angular components for Clarity
155 lines • 22.2 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 { Injectable } from '@angular/core';
import { fromEvent, Subject } from 'rxjs';
import { debounceTime, takeUntil } from 'rxjs/operators';
import { Keys } from '../../../utils/enums/keys.enum';
import { KeyNavigationUtils } from './key-navigation-utils';
import * as i0 from "@angular/core";
const actionableItemSelectors = [
'a[href]',
'area[href]',
'input:not([disabled])',
'button:not([disabled])',
'select:not([disabled])',
'textarea:not([disabled])',
'iframe',
'object',
'embed',
'[contenteditable=true]',
'[role=button]:not([disabled])',
];
export function getTabbableItems(el) {
const tabbableItemSelectors = [...actionableItemSelectors, '[tabindex="0"]:not([disabled])'];
const tabbableSelector = tabbableItemSelectors.join(',');
return Array.from(el.querySelectorAll(tabbableSelector));
}
function isActionableItem(el) {
const actionableSelector = actionableItemSelectors.join(',');
return el.matches(actionableSelector);
}
export class KeyNavigationGridController {
constructor(zone) {
this.zone = zone;
this.skipItemFocus = false;
this.listenersAdded = false;
this.destroy$ = new Subject();
this._activeCell = null;
this.config = {
keyGridRows: '[role=row]:not(.datagrid-placeholder):not([style*="display: none"])',
keyGridCells: '[role=gridcell]:not(.datagrid-hidden-column):not(.datagrid-placeholder-content), [role=columnheader]:not(.datagrid-hidden-column):not(.datagrid-placeholder-content), .datagrid-detail-caret',
keyGrid: '[role=grid]',
};
}
ngOnDestroy() {
this.destroy$.next();
this.destroy$.complete();
}
addListeners() {
if (this.listenersAdded) {
return;
}
this.zone.runOutsideAngular(() => {
fromEvent(this.keyNavUtils.grid, 'mousedown')
.pipe(takeUntil(this.destroy$))
.subscribe((e) => {
// preserve right click for context menus & keyboard mouse control https://apple.stackexchange.com/questions/32715/how-do-i-open-the-context-menu-from-a-mac-keyboard
if (e.buttons === 1 && !e.ctrlKey) {
const activeCell = this.keyNavUtils.cells
? Array.from(this.keyNavUtils.cells).find(c => c === e.target || c === e.target.closest(this.config.keyGridCells))
: null;
if (activeCell) {
this.setActiveCell(activeCell, { keepFocus: isActionableItem(e.target) });
}
}
});
fromEvent(this.keyNavUtils.grid, 'wheel')
.pipe(takeUntil(this.destroy$))
.subscribe(() => {
this.removeActiveCell();
});
fromEvent(this.keyNavUtils.grid, 'focusout')
.pipe(debounceTime(0), takeUntil(this.destroy$))
.subscribe(() => {
if (this.keyNavUtils.grid.contains(document.activeElement)) {
return;
}
this.removeActiveCell();
});
fromEvent(this.keyNavUtils.grid, 'keydown')
.pipe(takeUntil(this.destroy$))
.subscribe((e) => {
// Skip column resize events
if (e.target.classList.contains('drag-handle') &&
(e.key === Keys.ArrowLeft || e.key === Keys.ArrowRight)) {
return;
}
if (e.key === Keys.ArrowUp ||
e.key === Keys.ArrowDown ||
e.key === Keys.ArrowLeft ||
e.key === Keys.ArrowRight ||
e.key === Keys.End ||
e.key === Keys.Home ||
e.key === Keys.PageUp ||
e.key === Keys.PageDown) {
const nextCellCoords = this.keyNavUtils.getNextItemCoordinate(e);
const activeItem = this.keyNavUtils.rows
? Array.from(this.keyNavUtils.getCellsForRow(nextCellCoords.y))[nextCellCoords.x]
: null;
if (activeItem) {
this.setActiveCell(activeItem);
}
e.preventDefault();
}
});
});
this.listenersAdded = true;
}
initializeKeyGrid(host) {
this.keyNavUtils = new KeyNavigationUtils(host, this.config);
this.addListeners();
this.resetKeyGrid();
}
resetKeyGrid() {
this.keyNavUtils.cells?.forEach((i) => i.setAttribute('tabindex', '-1'));
const firstCell = this.keyNavUtils.cells ? this.keyNavUtils.cells[0] : null;
firstCell?.setAttribute('tabindex', '0');
}
removeActiveCell() {
this._activeCell = null;
}
getActiveCell() {
return this._activeCell;
}
setActiveCell(activeCell, { keepFocus } = { keepFocus: false }) {
const prior = this.keyNavUtils.cells
? Array.from(this.keyNavUtils.cells).find(c => c.getAttribute('tabindex') === '0')
: null;
if (prior) {
prior.setAttribute('tabindex', '-1');
}
activeCell.setAttribute('tabindex', '0');
this._activeCell = activeCell;
if (!this.skipItemFocus && !keepFocus) {
let elementToFocus;
if (activeCell.getAttribute('role') === 'columnheader') {
elementToFocus = activeCell;
}
else {
const tabbableElements = getTabbableItems(activeCell);
elementToFocus = tabbableElements.length ? tabbableElements[0] : activeCell;
}
elementToFocus.focus();
}
}
}
KeyNavigationGridController.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: KeyNavigationGridController, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
KeyNavigationGridController.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: KeyNavigationGridController });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: KeyNavigationGridController, decorators: [{
type: Injectable
}], ctorParameters: function () { return [{ type: i0.NgZone }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-navigation-grid.controller.js","sourceRoot":"","sources":["../../../../../../projects/angular/src/data/datagrid/utils/key-navigation-grid.controller.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 { Injectable, NgZone, OnDestroy } from '@angular/core';\nimport { fromEvent, Subject } from 'rxjs';\nimport { debounceTime, takeUntil } from 'rxjs/operators';\n\nimport { Keys } from '../../../utils/enums/keys.enum';\nimport { KeyNavigationUtils } from './key-navigation-utils';\n\nconst actionableItemSelectors = [\n  'a[href]',\n  'area[href]',\n  'input:not([disabled])',\n  'button:not([disabled])',\n  'select:not([disabled])',\n  'textarea:not([disabled])',\n  'iframe',\n  'object',\n  'embed',\n  '[contenteditable=true]',\n  '[role=button]:not([disabled])',\n];\n\nexport function getTabbableItems(el: HTMLElement) {\n  const tabbableItemSelectors = [...actionableItemSelectors, '[tabindex=\"0\"]:not([disabled])'];\n  const tabbableSelector = tabbableItemSelectors.join(',');\n  return Array.from(el.querySelectorAll(tabbableSelector)) as HTMLElement[];\n}\n\nfunction isActionableItem(el: HTMLElement) {\n  const actionableSelector = actionableItemSelectors.join(',');\n  return el.matches(actionableSelector);\n}\n\nexport interface KeyNavigationGridConfig {\n  keyGrid: string;\n  keyGridRows: string;\n  keyGridCells: string;\n}\n\nexport interface CellCoordinates {\n  x: number;\n  y: number;\n}\n\n@Injectable()\nexport class KeyNavigationGridController implements OnDestroy {\n  skipItemFocus = false;\n\n  private keyNavUtils: KeyNavigationUtils;\n  private config: KeyNavigationGridConfig;\n  private listenersAdded = false;\n  private destroy$ = new Subject<void>();\n  private _activeCell: HTMLElement = null;\n\n  constructor(private zone: NgZone) {\n    this.config = {\n      keyGridRows: '[role=row]:not(.datagrid-placeholder):not([style*=\"display: none\"])',\n      keyGridCells:\n        '[role=gridcell]:not(.datagrid-hidden-column):not(.datagrid-placeholder-content), [role=columnheader]:not(.datagrid-hidden-column):not(.datagrid-placeholder-content), .datagrid-detail-caret',\n      keyGrid: '[role=grid]',\n    };\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  addListeners() {\n    if (this.listenersAdded) {\n      return;\n    }\n\n    this.zone.runOutsideAngular(() => {\n      fromEvent(this.keyNavUtils.grid, 'mousedown')\n        .pipe(takeUntil(this.destroy$))\n        .subscribe((e: MouseEvent) => {\n          // preserve right click for context menus & keyboard mouse control https://apple.stackexchange.com/questions/32715/how-do-i-open-the-context-menu-from-a-mac-keyboard\n          if (e.buttons === 1 && !e.ctrlKey) {\n            const activeCell = this.keyNavUtils.cells\n              ? Array.from(this.keyNavUtils.cells).find(\n                  c => c === e.target || c === (e.target as HTMLElement).closest(this.config.keyGridCells)\n                )\n              : null;\n            if (activeCell) {\n              this.setActiveCell(activeCell, { keepFocus: isActionableItem(e.target as HTMLElement) });\n            }\n          }\n        });\n\n      fromEvent(this.keyNavUtils.grid, 'wheel')\n        .pipe(takeUntil(this.destroy$))\n        .subscribe(() => {\n          this.removeActiveCell();\n        });\n\n      fromEvent(this.keyNavUtils.grid, 'focusout')\n        .pipe(debounceTime(0), takeUntil(this.destroy$))\n        .subscribe(() => {\n          if (this.keyNavUtils.grid.contains(document.activeElement)) {\n            return;\n          }\n\n          this.removeActiveCell();\n        });\n\n      fromEvent(this.keyNavUtils.grid, 'keydown')\n        .pipe(takeUntil(this.destroy$))\n        .subscribe((e: KeyboardEvent) => {\n          // Skip column resize events\n          if (\n            (e.target as HTMLElement).classList.contains('drag-handle') &&\n            (e.key === Keys.ArrowLeft || e.key === Keys.ArrowRight)\n          ) {\n            return;\n          }\n          if (\n            e.key === Keys.ArrowUp ||\n            e.key === Keys.ArrowDown ||\n            e.key === Keys.ArrowLeft ||\n            e.key === Keys.ArrowRight ||\n            e.key === Keys.End ||\n            e.key === Keys.Home ||\n            e.key === Keys.PageUp ||\n            e.key === Keys.PageDown\n          ) {\n            const nextCellCoords = this.keyNavUtils.getNextItemCoordinate(e);\n\n            const activeItem = this.keyNavUtils.rows\n              ? (Array.from(this.keyNavUtils.getCellsForRow(nextCellCoords.y))[nextCellCoords.x] as HTMLElement)\n              : null;\n\n            if (activeItem) {\n              this.setActiveCell(activeItem);\n            }\n\n            e.preventDefault();\n          }\n        });\n    });\n    this.listenersAdded = true;\n  }\n\n  initializeKeyGrid(host: HTMLElement) {\n    this.keyNavUtils = new KeyNavigationUtils(host, this.config);\n    this.addListeners();\n    this.resetKeyGrid();\n  }\n\n  resetKeyGrid() {\n    this.keyNavUtils.cells?.forEach((i: HTMLElement) => i.setAttribute('tabindex', '-1'));\n    const firstCell = this.keyNavUtils.cells ? this.keyNavUtils.cells[0] : null;\n    firstCell?.setAttribute('tabindex', '0');\n  }\n\n  removeActiveCell() {\n    this._activeCell = null;\n  }\n\n  getActiveCell() {\n    return this._activeCell;\n  }\n\n  setActiveCell(activeCell: HTMLElement, { keepFocus } = { keepFocus: false }) {\n    const prior = this.keyNavUtils.cells\n      ? Array.from(this.keyNavUtils.cells).find(c => c.getAttribute('tabindex') === '0')\n      : null;\n\n    if (prior) {\n      prior.setAttribute('tabindex', '-1');\n    }\n\n    activeCell.setAttribute('tabindex', '0');\n    this._activeCell = activeCell;\n\n    if (!this.skipItemFocus && !keepFocus) {\n      let elementToFocus: HTMLElement;\n\n      if (activeCell.getAttribute('role') === 'columnheader') {\n        elementToFocus = activeCell;\n      } else {\n        const tabbableElements = getTabbableItems(activeCell);\n        elementToFocus = tabbableElements.length ? tabbableElements[0] : activeCell;\n      }\n\n      elementToFocus.focus();\n    }\n  }\n}\n"]}