UNPKG

@clr/angular

Version:

Angular components for Clarity

121 lines 18.8 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 { Keys } from '../../../utils/enums/keys.enum'; import { DefaultKeyNavigationStrategy } from './key-navigation-strategies/default'; import { ExpandedColumnsRowKeyNavigationStrategy } from './key-navigation-strategies/expanded-columns-row'; import { ExpandedRowKeyNavigationStrategy } from './key-navigation-strategies/expanded-row'; export class KeyNavigationUtils { constructor(host, config) { this.host = host; this.config = config; } get grid() { return this.host?.querySelector(this.config.keyGrid); } get rows() { return this.host?.querySelectorAll(this.config.keyGridRows); } get cells() { return this.host?.querySelectorAll(this.config.keyGridCells); } get currentCellCoordinates() { const currentCell = this.cells ? Array.from(this.cells).find(i => i.getAttribute('tabindex') === '0') : null; const currentRow = currentCell ? currentCell.closest(this.config.keyGridRows) : null; const coordinates = { x: currentRow && currentCell ? Array.from(currentRow.querySelectorAll(this.config.keyGridCells)).indexOf(currentCell) : 0, y: currentRow && currentCell && this.rows ? Array.from(this.rows).indexOf(currentRow) : 0, }; return coordinates; } get averageRowHeight() { const heightSum = Array.from(this.rows.values()).reduce((sum, row) => { return sum + row.clientHeight; }, 0); return Math.round(heightSum / this.rows.length); } get itemsPerPage() { return Math.floor(this.host?.querySelector('.datagrid').clientHeight / this.averageRowHeight) - 1 || 0; } setAriaRowIndexTo(cellCoords) { let ariaRowIndex = this.rows[cellCoords.y].getAttribute('aria-rowindex'); if (!ariaRowIndex) { ariaRowIndex = this.rows[cellCoords.y - 1].getAttribute('aria-rowindex'); } cellCoords.ariaRowIndex = ariaRowIndex; } getNextItemCoordinate(e) { const currentCellCoords = this.currentCellCoordinates; const strategy = this.getNavStrategy(currentCellCoords); const inlineStart = this.host.dir === 'rtl' ? Keys.ArrowRight : Keys.ArrowLeft; const inlineEnd = this.host.dir === 'rtl' ? Keys.ArrowLeft : Keys.ArrowRight; switch (e.key) { case Keys.ArrowUp: return strategy.keyUp(currentCellCoords); case Keys.ArrowDown: return strategy.keyDown(currentCellCoords); case inlineStart: return strategy.keyLeft(currentCellCoords); case inlineEnd: return strategy.keyRight(currentCellCoords); case Keys.Home: return strategy.keyHome(currentCellCoords, e.ctrlKey); case Keys.End: return strategy.keyEnd(currentCellCoords, e.ctrlKey); case Keys.PageUp: return strategy.keyPageUp(currentCellCoords); case Keys.PageDown: return strategy.keyPageDown(currentCellCoords); default: return currentCellCoords; } } getCellsForRow(index) { return this.rows[index].querySelectorAll(this.config.keyGridCells); } isExpandedRow(index) { const selectedElement = this.rows[index].querySelector('.datagrid-row-detail'); return selectedElement ? selectedElement.style.display !== 'none' : false; } isDetailsRow(index) { return this.rows[index].classList.contains('datagrid-row-detail'); } isRowReplaced(index) { return !!this.rows[index].closest('clr-dg-row.datagrid-row-replaced'); } isSingleCellExpandedRow(index) { return this.rows[index]?.querySelectorAll(this.config.keyGridCells).length === 1; } actionCellCount(index) { return this.actionCellsAsArray(index).length; } actionCellsAsArray(index) { return Array.from(this.rows[index].querySelectorAll('.datagrid-row-sticky .datagrid-cell, .datagrid-row-sticky .datagrid-column')); } isActionCell(cellCoords) { return !!this.actionCellsAsArray(cellCoords.y)[cellCoords.x]; } createNextCellCoords(cellCoords) { return { x: cellCoords.x, y: cellCoords.y, }; } getNavStrategy(currentCellCoords) { switch (true) { case this.isSingleCellExpandedRow(currentCellCoords.y): return new ExpandedRowKeyNavigationStrategy(this); case this.isDetailsRow(currentCellCoords.y): case this.isExpandedRow(currentCellCoords.y): return new ExpandedColumnsRowKeyNavigationStrategy(this); default: return new DefaultKeyNavigationStrategy(this); } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-navigation-utils.js","sourceRoot":"","sources":["../../../../../../projects/angular/src/data/datagrid/utils/key-navigation-utils.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EAAE,IAAI,EAAE,MAAM,gCAAgC,CAAC;AAGtD,OAAO,EAAE,4BAA4B,EAAE,MAAM,qCAAqC,CAAC;AACnF,OAAO,EAAE,uCAAuC,EAAE,MAAM,kDAAkD,CAAC;AAC3G,OAAO,EAAE,gCAAgC,EAAE,MAAM,0CAA0C,CAAC;AAE5F,MAAM,OAAO,kBAAkB;IAC7B,YAAmB,IAAiB,EAAS,MAA+B;QAAzD,SAAI,GAAJ,IAAI,CAAa;QAAS,WAAM,GAAN,MAAM,CAAyB;IAAG,CAAC;IAEhF,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,IAAI,EAAE,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACvD,CAAC;IAED,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,IAAI,EAAE,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAA4B,CAAC;IACzF,CAAC;IAED,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,IAAI,EAAE,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAA4B,CAAC;IAC1F,CAAC;IAED,IAAI,sBAAsB;QACxB,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC,UAAU,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAC7G,MAAM,UAAU,GAAgB,WAAW,CAAC,CAAC,CAAC,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAElG,MAAM,WAAW,GAAoB;YACnC,CAAC,EACC,UAAU,IAAI,WAAW;gBACvB,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC;gBACxF,CAAC,CAAC,CAAC;YACP,CAAC,EAAE,UAAU,IAAI,WAAW,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;SAC1F,CAAC;QAEF,OAAO,WAAW,CAAC;IACrB,CAAC;IAED,IAAI,gBAAgB;QAClB,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;YACnE,OAAO,GAAG,GAAG,GAAG,CAAC,YAAY,CAAC;QAChC,CAAC,EAAE,CAAC,CAAC,CAAC;QAEN,OAAO,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAClD,CAAC;IAED,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE,aAAa,CAAC,WAAW,CAAC,CAAC,YAAY,GAAG,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACzG,CAAC;IAED,iBAAiB,CAAC,UAA2B;QAC3C,IAAI,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,eAAe,CAAC,CAAC;QAEzE,IAAI,CAAC,YAAY,EAAE;YACjB,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,CAAC,eAAe,CAAC,CAAC;SAC1E;QAED,UAAU,CAAC,YAAY,GAAG,YAAY,CAAC;IACzC,CAAC;IAED,qBAAqB,CAAC,CAAgB;QACpC,MAAM,iBAAiB,GAAG,IAAI,CAAC,sBAAsB,CAAC;QACtD,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,CAAC;QAExD,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,KAAK,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC;QAC/E,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,KAAK,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC;QAE7E,QAAQ,CAAC,CAAC,GAAG,EAAE;YACb,KAAK,IAAI,CAAC,OAAO;gBACf,OAAO,QAAQ,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;YAC3C,KAAK,IAAI,CAAC,SAAS;gBACjB,OAAO,QAAQ,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;YAC7C,KAAK,WAAW;gBACd,OAAO,QAAQ,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;YAC7C,KAAK,SAAS;gBACZ,OAAO,QAAQ,CAAC,QAAQ,CAAC,iBAAiB,CAAC,CAAC;YAC9C,KAAK,IAAI,CAAC,IAAI;gBACZ,OAAO,QAAQ,CAAC,OAAO,CAAC,iBAAiB,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC;YACxD,KAAK,IAAI,CAAC,GAAG;gBACX,OAAO,QAAQ,CAAC,MAAM,CAAC,iBAAiB,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC;YACvD,KAAK,IAAI,CAAC,MAAM;gBACd,OAAO,QAAQ,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;YAC/C,KAAK,IAAI,CAAC,QAAQ;gBAChB,OAAO,QAAQ,CAAC,WAAW,CAAC,iBAAiB,CAAC,CAAC;YACjD;gBACE,OAAO,iBAAiB,CAAC;SAC5B;IACH,CAAC;IAED,cAAc,CAAC,KAAa;QAC1B,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;IACrE,CAAC;IAED,aAAa,CAAC,KAAa;QACzB,MAAM,eAAe,GAAgB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,aAAa,CAAC,sBAAsB,CAAC,CAAC;QAE5F,OAAO,eAAe,CAAC,CAAC,CAAC,eAAe,CAAC,KAAK,CAAC,OAAO,KAAK,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC;IAC5E,CAAC;IAED,YAAY,CAAC,KAAa;QACxB,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,qBAAqB,CAAC,CAAC;IACpE,CAAC;IAED,aAAa,CAAC,KAAa;QACzB,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,kCAAkC,CAAC,CAAC;IACxE,CAAC;IAED,uBAAuB,CAAC,KAAa;QACnC,OAAO,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC;IACnF,CAAC;IAED,eAAe,CAAC,KAAa;QAC3B,OAAO,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC;IAC/C,CAAC;IAED,kBAAkB,CAAC,KAAa;QAC9B,OAAO,KAAK,CAAC,IAAI,CACf,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,gBAAgB,CAAC,4EAA4E,CAAC,CAChH,CAAC;IACJ,CAAC;IAED,YAAY,CAAC,UAA2B;QACtC,OAAO,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;IAC/D,CAAC;IAED,oBAAoB,CAAC,UAA2B;QAC9C,OAAO;YACL,CAAC,EAAE,UAAU,CAAC,CAAC;YACf,CAAC,EAAE,UAAU,CAAC,CAAC;SAChB,CAAC;IACJ,CAAC;IAEO,cAAc,CAAC,iBAAkC;QACvD,QAAQ,IAAI,EAAE;YACZ,KAAK,IAAI,CAAC,uBAAuB,CAAC,iBAAiB,CAAC,CAAC,CAAC;gBACpD,OAAO,IAAI,gCAAgC,CAAC,IAAI,CAAC,CAAC;YACpD,KAAK,IAAI,CAAC,YAAY,CAAC,iBAAiB,CAAC,CAAC,CAAC,CAAC;YAC5C,KAAK,IAAI,CAAC,aAAa,CAAC,iBAAiB,CAAC,CAAC,CAAC;gBAC1C,OAAO,IAAI,uCAAuC,CAAC,IAAI,CAAC,CAAC;YAC3D;gBACE,OAAO,IAAI,4BAA4B,CAAC,IAAI,CAAC,CAAC;SACjD;IACH,CAAC;CACF","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 { Keys } from '../../../utils/enums/keys.enum';\nimport { KeyNavigationGridStrategyInterface } from '../interfaces/key-nav-grid-strategy.interface';\nimport { CellCoordinates, KeyNavigationGridConfig } from './key-navigation-grid.controller';\nimport { DefaultKeyNavigationStrategy } from './key-navigation-strategies/default';\nimport { ExpandedColumnsRowKeyNavigationStrategy } from './key-navigation-strategies/expanded-columns-row';\nimport { ExpandedRowKeyNavigationStrategy } from './key-navigation-strategies/expanded-row';\n\nexport class KeyNavigationUtils {\n  constructor(public host: HTMLElement, public config: KeyNavigationGridConfig) {}\n\n  get grid() {\n    return this.host?.querySelector(this.config.keyGrid);\n  }\n\n  get rows() {\n    return this.host?.querySelectorAll(this.config.keyGridRows) as NodeListOf<HTMLElement>;\n  }\n\n  get cells() {\n    return this.host?.querySelectorAll(this.config.keyGridCells) as NodeListOf<HTMLElement>;\n  }\n\n  get currentCellCoordinates() {\n    const currentCell = this.cells ? Array.from(this.cells).find(i => i.getAttribute('tabindex') === '0') : null;\n    const currentRow: HTMLElement = currentCell ? currentCell.closest(this.config.keyGridRows) : null;\n\n    const coordinates: CellCoordinates = {\n      x:\n        currentRow && currentCell\n          ? Array.from(currentRow.querySelectorAll(this.config.keyGridCells)).indexOf(currentCell)\n          : 0,\n      y: currentRow && currentCell && this.rows ? Array.from(this.rows).indexOf(currentRow) : 0,\n    };\n\n    return coordinates;\n  }\n\n  get averageRowHeight() {\n    const heightSum = Array.from(this.rows.values()).reduce((sum, row) => {\n      return sum + row.clientHeight;\n    }, 0);\n\n    return Math.round(heightSum / this.rows.length);\n  }\n\n  get itemsPerPage() {\n    return Math.floor(this.host?.querySelector('.datagrid').clientHeight / this.averageRowHeight) - 1 || 0;\n  }\n\n  setAriaRowIndexTo(cellCoords: CellCoordinates) {\n    let ariaRowIndex = this.rows[cellCoords.y].getAttribute('aria-rowindex');\n\n    if (!ariaRowIndex) {\n      ariaRowIndex = this.rows[cellCoords.y - 1].getAttribute('aria-rowindex');\n    }\n\n    cellCoords.ariaRowIndex = ariaRowIndex;\n  }\n\n  getNextItemCoordinate(e: KeyboardEvent) {\n    const currentCellCoords = this.currentCellCoordinates;\n    const strategy = this.getNavStrategy(currentCellCoords);\n\n    const inlineStart = this.host.dir === 'rtl' ? Keys.ArrowRight : Keys.ArrowLeft;\n    const inlineEnd = this.host.dir === 'rtl' ? Keys.ArrowLeft : Keys.ArrowRight;\n\n    switch (e.key) {\n      case Keys.ArrowUp:\n        return strategy.keyUp(currentCellCoords);\n      case Keys.ArrowDown:\n        return strategy.keyDown(currentCellCoords);\n      case inlineStart:\n        return strategy.keyLeft(currentCellCoords);\n      case inlineEnd:\n        return strategy.keyRight(currentCellCoords);\n      case Keys.Home:\n        return strategy.keyHome(currentCellCoords, e.ctrlKey);\n      case Keys.End:\n        return strategy.keyEnd(currentCellCoords, e.ctrlKey);\n      case Keys.PageUp:\n        return strategy.keyPageUp(currentCellCoords);\n      case Keys.PageDown:\n        return strategy.keyPageDown(currentCellCoords);\n      default:\n        return currentCellCoords;\n    }\n  }\n\n  getCellsForRow(index: number) {\n    return this.rows[index].querySelectorAll(this.config.keyGridCells);\n  }\n\n  isExpandedRow(index: number) {\n    const selectedElement: HTMLElement = this.rows[index].querySelector('.datagrid-row-detail');\n\n    return selectedElement ? selectedElement.style.display !== 'none' : false;\n  }\n\n  isDetailsRow(index: number) {\n    return this.rows[index].classList.contains('datagrid-row-detail');\n  }\n\n  isRowReplaced(index: number) {\n    return !!this.rows[index].closest('clr-dg-row.datagrid-row-replaced');\n  }\n\n  isSingleCellExpandedRow(index: number) {\n    return this.rows[index]?.querySelectorAll(this.config.keyGridCells).length === 1;\n  }\n\n  actionCellCount(index: number) {\n    return this.actionCellsAsArray(index).length;\n  }\n\n  actionCellsAsArray(index: number) {\n    return Array.from(\n      this.rows[index].querySelectorAll('.datagrid-row-sticky .datagrid-cell, .datagrid-row-sticky .datagrid-column')\n    );\n  }\n\n  isActionCell(cellCoords: CellCoordinates) {\n    return !!this.actionCellsAsArray(cellCoords.y)[cellCoords.x];\n  }\n\n  createNextCellCoords(cellCoords: CellCoordinates): CellCoordinates {\n    return {\n      x: cellCoords.x,\n      y: cellCoords.y,\n    };\n  }\n\n  private getNavStrategy(currentCellCoords: CellCoordinates): KeyNavigationGridStrategyInterface {\n    switch (true) {\n      case this.isSingleCellExpandedRow(currentCellCoords.y):\n        return new ExpandedRowKeyNavigationStrategy(this);\n      case this.isDetailsRow(currentCellCoords.y):\n      case this.isExpandedRow(currentCellCoords.y):\n        return new ExpandedColumnsRowKeyNavigationStrategy(this);\n      default:\n        return new DefaultKeyNavigationStrategy(this);\n    }\n  }\n}\n"]}