@toolbox-web/grid 1.21.2 → 1.22.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -8
- package/all.js +1 -1
- package/all.js.map +1 -1
- package/index.js +1 -1
- package/index.js.map +1 -1
- package/lib/core/grid.d.ts +53 -2
- package/lib/core/grid.d.ts.map +1 -1
- package/lib/core/internal/rows.d.ts.map +1 -1
- package/lib/core/plugin/base-plugin.d.ts +8 -2
- package/lib/core/plugin/base-plugin.d.ts.map +1 -1
- package/lib/core/types.d.ts +84 -2
- package/lib/core/types.d.ts.map +1 -1
- package/lib/plugins/clipboard/index.js.map +1 -1
- package/lib/plugins/clipboard/types.d.ts +5 -0
- package/lib/plugins/clipboard/types.d.ts.map +1 -1
- package/lib/plugins/column-virtualization/index.js.map +1 -1
- package/lib/plugins/column-virtualization/types.d.ts +5 -0
- package/lib/plugins/column-virtualization/types.d.ts.map +1 -1
- package/lib/plugins/context-menu/index.js.map +1 -1
- package/lib/plugins/context-menu/types.d.ts +5 -0
- package/lib/plugins/context-menu/types.d.ts.map +1 -1
- package/lib/plugins/editing/EditingPlugin.d.ts.map +1 -1
- package/lib/plugins/editing/index.js +1 -1
- package/lib/plugins/editing/index.js.map +1 -1
- package/lib/plugins/editing/types.d.ts +26 -0
- package/lib/plugins/editing/types.d.ts.map +1 -1
- package/lib/plugins/export/index.js.map +1 -1
- package/lib/plugins/export/types.d.ts +5 -0
- package/lib/plugins/export/types.d.ts.map +1 -1
- package/lib/plugins/filtering/index.js.map +1 -1
- package/lib/plugins/filtering/types.d.ts +3 -0
- package/lib/plugins/filtering/types.d.ts.map +1 -1
- package/lib/plugins/grouping-columns/index.js.map +1 -1
- package/lib/plugins/grouping-columns/types.d.ts +3 -0
- package/lib/plugins/grouping-columns/types.d.ts.map +1 -1
- package/lib/plugins/grouping-rows/index.js.map +1 -1
- package/lib/plugins/grouping-rows/types.d.ts +5 -0
- package/lib/plugins/grouping-rows/types.d.ts.map +1 -1
- package/lib/plugins/master-detail/index.js.map +1 -1
- package/lib/plugins/master-detail/types.d.ts +5 -0
- package/lib/plugins/master-detail/types.d.ts.map +1 -1
- package/lib/plugins/multi-sort/index.js.map +1 -1
- package/lib/plugins/multi-sort/types.d.ts +5 -0
- package/lib/plugins/multi-sort/types.d.ts.map +1 -1
- package/lib/plugins/pinned-columns/index.js.map +1 -1
- package/lib/plugins/pinned-columns/types.d.ts +3 -0
- package/lib/plugins/pinned-columns/types.d.ts.map +1 -1
- package/lib/plugins/pinned-rows/index.js.map +1 -1
- package/lib/plugins/pinned-rows/types.d.ts +5 -0
- package/lib/plugins/pinned-rows/types.d.ts.map +1 -1
- package/lib/plugins/pivot/index.js.map +1 -1
- package/lib/plugins/pivot/types.d.ts +5 -0
- package/lib/plugins/pivot/types.d.ts.map +1 -1
- package/lib/plugins/print/index.js.map +1 -1
- package/lib/plugins/print/types.d.ts +3 -0
- package/lib/plugins/print/types.d.ts.map +1 -1
- package/lib/plugins/reorder/index.js.map +1 -1
- package/lib/plugins/reorder/types.d.ts +5 -0
- package/lib/plugins/reorder/types.d.ts.map +1 -1
- package/lib/plugins/responsive/index.js.map +1 -1
- package/lib/plugins/responsive/types.d.ts +5 -0
- package/lib/plugins/responsive/types.d.ts.map +1 -1
- package/lib/plugins/row-reorder/index.js.map +1 -1
- package/lib/plugins/row-reorder/types.d.ts +5 -0
- package/lib/plugins/row-reorder/types.d.ts.map +1 -1
- package/lib/plugins/selection/index.js.map +1 -1
- package/lib/plugins/selection/types.d.ts +3 -0
- package/lib/plugins/selection/types.d.ts.map +1 -1
- package/lib/plugins/server-side/index.js.map +1 -1
- package/lib/plugins/server-side/types.d.ts +5 -0
- package/lib/plugins/server-side/types.d.ts.map +1 -1
- package/lib/plugins/tree/index.js.map +1 -1
- package/lib/plugins/tree/types.d.ts +5 -0
- package/lib/plugins/tree/types.d.ts.map +1 -1
- package/lib/plugins/undo-redo/index.js.map +1 -1
- package/lib/plugins/undo-redo/types.d.ts +5 -0
- package/lib/plugins/undo-redo/types.d.ts.map +1 -1
- package/lib/plugins/visibility/index.js.map +1 -1
- package/lib/plugins/visibility/types.d.ts +5 -0
- package/lib/plugins/visibility/types.d.ts.map +1 -1
- package/package.json +1 -1
- package/umd/grid.all.umd.js +1 -1
- package/umd/grid.all.umd.js.map +1 -1
- package/umd/grid.umd.js +1 -1
- package/umd/grid.umd.js.map +1 -1
- package/umd/plugins/clipboard.umd.js.map +1 -1
- package/umd/plugins/editing.umd.js +1 -1
- package/umd/plugins/editing.umd.js.map +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"editing.umd.js","sources":["../../../../../libs/grid/src/lib/plugins/editing/editors.ts","../../../../../libs/grid/src/lib/plugins/editing/EditingPlugin.ts"],"sourcesContent":["/**\n * Default Editors Module\n *\n * Provides built-in editor factories for different column types.\n * Each editor type has its own factory function for consistency and readability.\n *\n * IMPORTANT: Editor factories should NOT call focus() on elements - they are called\n * before the element is appended to the DOM. The calling code (beginBulkEdit,\n * inlineEnterEdit) is responsible for focusing the correct editor after insertion.\n */\n\nimport type { ColumnConfig, ColumnEditorContext } from '../../core/types';\nimport type { DateEditorParams, NumberEditorParams, SelectEditorParams, TextEditorParams } from './types';\n\n// ============================================================================\n// Type Aliases\n// ============================================================================\n\n/** Option shape used by select editor (matches column.options) */\ntype ColumnOption = { label: string; value: unknown };\n\n/** Column with any row type (used for editor factories) */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype AnyColumn = ColumnConfig<any>;\n\n// ============================================================================\n// Utilities\n// ============================================================================\n\n/** Resolve column.options (handles both array and function forms) */\nfunction resolveOptions(column: AnyColumn): ColumnOption[] {\n const raw = column.options;\n if (!raw) return [];\n return typeof raw === 'function' ? raw() : raw;\n}\n\n// ============================================================================\n// Editor Factories\n// ============================================================================\n\n/** Creates a number input editor */\nfunction createNumberEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as NumberEditorParams | undefined;\n const input = document.createElement('input');\n input.type = 'number';\n input.value = ctx.value != null ? String(ctx.value) : '';\n\n if (params?.min !== undefined) input.min = String(params.min);\n if (params?.max !== undefined) input.max = String(params.max);\n if (params?.step !== undefined) input.step = String(params.step);\n if (params?.placeholder) input.placeholder = params.placeholder;\n\n const commit = () => ctx.commit(input.value === '' ? null : Number(input.value));\n input.addEventListener('blur', commit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Enter') commit();\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return input;\n };\n}\n\n/** Creates a checkbox editor for boolean values */\nfunction createBooleanEditor(): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const input = document.createElement('input');\n input.type = 'checkbox';\n input.checked = !!ctx.value;\n input.addEventListener('change', () => ctx.commit(input.checked));\n return input;\n };\n}\n\n/** Creates a date input editor */\nfunction createDateEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as DateEditorParams | undefined;\n const input = document.createElement('input');\n input.type = 'date';\n\n // Set initial value - handle both Date objects and string dates\n if (ctx.value instanceof Date) {\n input.valueAsDate = ctx.value;\n } else if (typeof ctx.value === 'string' && ctx.value) {\n // String date like \"2019-10-09\" - set directly as value\n input.value = ctx.value.split('T')[0]; // Handle ISO strings too\n }\n if (params?.min) input.min = params.min;\n if (params?.max) input.max = params.max;\n if (params?.placeholder) input.placeholder = params.placeholder;\n\n // Commit function preserves original type (string vs Date)\n const commit = () => {\n if (typeof ctx.value === 'string') {\n // Original was string, return string in YYYY-MM-DD format\n ctx.commit(input.value);\n } else {\n ctx.commit(input.valueAsDate);\n }\n };\n\n input.addEventListener('change', commit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return input;\n };\n}\n\n/** Creates a select dropdown editor */\nfunction createSelectEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as SelectEditorParams | undefined;\n const select = document.createElement('select');\n if (column.multi) select.multiple = true;\n\n // Add empty option if requested\n if (params?.includeEmpty) {\n const emptyOpt = document.createElement('option');\n emptyOpt.value = '';\n emptyOpt.textContent = params.emptyLabel ?? '';\n select.appendChild(emptyOpt);\n }\n\n // Populate options from column.options\n const options = resolveOptions(column);\n options.forEach((opt) => {\n const o = document.createElement('option');\n o.value = String(opt.value);\n o.textContent = opt.label;\n if (column.multi && Array.isArray(ctx.value) && ctx.value.includes(opt.value)) {\n o.selected = true;\n } else if (!column.multi && ctx.value === opt.value) {\n o.selected = true;\n }\n select.appendChild(o);\n });\n\n const commit = () => {\n if (column.multi) {\n const values = Array.from(select.selectedOptions).map((o) => o.value);\n ctx.commit(values);\n } else {\n ctx.commit(select.value);\n }\n };\n\n select.addEventListener('change', commit);\n select.addEventListener('blur', commit);\n select.addEventListener('keydown', (e) => {\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return select;\n };\n}\n\n/** Creates a text input editor (default) */\nfunction createTextEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as TextEditorParams | undefined;\n const input = document.createElement('input');\n input.type = 'text';\n input.value = ctx.value != null ? String(ctx.value) : '';\n\n if (params?.maxLength !== undefined) input.maxLength = params.maxLength;\n if (params?.pattern) input.pattern = params.pattern;\n if (params?.placeholder) input.placeholder = params.placeholder;\n\n // Commit function preserves original type when possible\n const commit = () => {\n const inputVal = input.value;\n // Preserve null/undefined: empty input on a null/undefined field means no change\n if ((ctx.value === null || ctx.value === undefined) && inputVal === '') {\n return;\n }\n // Preserve values with characters that <input> can't represent (newlines, etc.).\n // If stripping those characters produces the same string, the user didn't change anything.\n if (typeof ctx.value === 'string' && inputVal === ctx.value.replace(/[\\n\\r]/g, '')) {\n return;\n }\n // Preserve numeric type for custom column types (e.g., currency)\n if (typeof ctx.value === 'number' && inputVal !== '') {\n ctx.commit(Number(inputVal));\n } else {\n ctx.commit(inputVal);\n }\n };\n\n input.addEventListener('blur', commit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Enter') commit();\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return input;\n };\n}\n\n// ============================================================================\n// Main Entry Point\n// ============================================================================\n\n/**\n * Returns a default editor factory function for the given column type.\n * Each editor handles commit on blur/Enter, and cancel on Escape.\n *\n * Note: Focus is NOT called here - the calling code handles focusing after DOM insertion.\n */\nexport function defaultEditorFor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement | string {\n switch (column.type) {\n case 'number':\n return createNumberEditor(column);\n case 'boolean':\n return createBooleanEditor();\n case 'date':\n return createDateEditor(column);\n case 'select':\n return createSelectEditor(column);\n default:\n return createTextEditor(column);\n }\n}\n\n// ============================================================================\n// Utility Export (used by EditingPlugin)\n// ============================================================================\n\n/**\n * Gets the current value from an input element, with type coercion based on column type.\n */\nexport function getInputValue(\n input: HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement,\n col: AnyColumn,\n): unknown {\n if (input instanceof HTMLSelectElement) {\n if (col.multi) {\n return Array.from(input.selectedOptions).map((o) => o.value);\n }\n return input.value;\n }\n if (input instanceof HTMLInputElement) {\n if (input.type === 'checkbox') return input.checked;\n if (input.type === 'number') return input.value === '' ? null : Number(input.value);\n if (input.type === 'date') return input.valueAsDate;\n }\n return input.value;\n}\n","/**\n * Editing Plugin\n *\n * Provides complete editing functionality for tbw-grid.\n * This plugin is FULLY SELF-CONTAINED - the grid has ZERO editing knowledge.\n *\n * The plugin:\n * - Owns all editing state (active cell, snapshots, changed rows)\n * - Uses event distribution (onCellClick, onKeyDown) to handle edit lifecycle\n * - Uses afterRender() hook to inject editors into cells\n * - Uses processColumns() to augment columns with editing metadata\n * - Emits its own events (cell-commit, row-commit, changed-rows-reset)\n *\n * Without this plugin, the grid cannot edit. With this plugin, editing\n * is fully functional without any core changes.\n */\n\nimport { ensureCellVisible } from '../../core/internal/keyboard';\nimport { FOCUSABLE_EDITOR_SELECTOR } from '../../core/internal/rows';\nimport type { AfterCellRenderContext, PluginManifest, PluginQuery } from '../../core/plugin/base-plugin';\nimport { BaseGridPlugin, type CellClickEvent, type GridElement } from '../../core/plugin/base-plugin';\nimport type {\n ColumnConfig,\n ColumnEditorSpec,\n ColumnInternal,\n InternalGrid,\n RowElementInternal,\n} from '../../core/types';\nimport styles from './editing.css?inline';\nimport { defaultEditorFor } from './editors';\nimport type {\n BeforeEditCloseDetail,\n CellCommitDetail,\n ChangedRowsResetDetail,\n EditCloseDetail,\n EditingConfig,\n EditOpenDetail,\n EditorContext,\n RowCommitDetail,\n} from './types';\n\n// ============================================================================\n// Helper Functions\n// ============================================================================\n\n/**\n * Resolves the editor for a column using the priority chain:\n * 1. Column-level (`column.editor`)\n * 2. Light DOM template (`__editorTemplate` → returns 'template')\n * 3. Grid-level (`gridConfig.typeDefaults[column.type]`)\n * 4. App-level (framework adapter's `getTypeDefault`)\n * 5. Returns undefined (caller uses built-in defaultEditorFor)\n */\nfunction resolveEditor<TRow>(\n grid: InternalGrid<TRow>,\n col: ColumnInternal<TRow>,\n): ColumnEditorSpec<TRow, unknown> | 'template' | undefined {\n // 1. Column-level editor (highest priority)\n if (col.editor) return col.editor;\n\n // 2. Light DOM template\n const tplHolder = col.__editorTemplate;\n if (tplHolder) return 'template';\n\n // No type specified - no type defaults to check\n if (!col.type) return undefined;\n\n // 3. Grid-level typeDefaults (access via effectiveConfig)\n const gridTypeDefaults = (grid as any).effectiveConfig?.typeDefaults;\n if (gridTypeDefaults?.[col.type]?.editor) {\n return gridTypeDefaults[col.type].editor as ColumnEditorSpec<TRow, unknown>;\n }\n\n // 4. App-level registry (via framework adapter)\n const adapter = grid.__frameworkAdapter;\n if (adapter?.getTypeDefault) {\n const appDefault = adapter.getTypeDefault<TRow>(col.type);\n if (appDefault?.editor) {\n return appDefault.editor as ColumnEditorSpec<TRow, unknown>;\n }\n }\n\n // 5. No custom editor - caller uses built-in defaultEditorFor\n return undefined;\n}\n\n/**\n * Returns true if the given property key is safe to use on a plain object.\n */\nfunction isSafePropertyKey(key: unknown): key is string {\n if (typeof key !== 'string') return false;\n if (key === '__proto__' || key === 'constructor' || key === 'prototype') return false;\n return true;\n}\n\n/**\n * Check if a row element has any cells in editing mode.\n */\nexport function hasEditingCells(rowEl: RowElementInternal): boolean {\n return (rowEl.__editingCellCount ?? 0) > 0;\n}\n\n/**\n * Increment the editing cell count on a row element.\n */\nfunction incrementEditingCount(rowEl: RowElementInternal): void {\n const count = (rowEl.__editingCellCount ?? 0) + 1;\n rowEl.__editingCellCount = count;\n rowEl.setAttribute('data-has-editing', '');\n}\n\n/**\n * Clear all editing state from a row element.\n */\nexport function clearEditingState(rowEl: RowElementInternal): void {\n rowEl.__editingCellCount = 0;\n rowEl.removeAttribute('data-has-editing');\n}\n\n/**\n * Get the typed value from an input element based on its type, column config, and original value.\n * Preserves the type of the original value (e.g., numeric currency values stay as numbers,\n * string dates stay as strings).\n */\nfunction getInputValue(\n input: HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement,\n column?: ColumnConfig<any>,\n originalValue?: unknown,\n): unknown {\n if (input instanceof HTMLInputElement) {\n if (input.type === 'checkbox') return input.checked;\n if (input.type === 'number') return input.value === '' ? null : Number(input.value);\n if (input.type === 'date') {\n // Preserve original type: if original was a string, return string (YYYY-MM-DD format)\n if (typeof originalValue === 'string') {\n return input.value; // input.value is already in YYYY-MM-DD format\n }\n return input.valueAsDate;\n }\n // For text inputs, check if original value was a number to preserve type\n if (typeof originalValue === 'number') {\n return input.value === '' ? null : Number(input.value);\n }\n // Preserve null/undefined: if original was null/undefined and input is empty, return original\n if ((originalValue === null || originalValue === undefined) && input.value === '') {\n return originalValue;\n }\n // Preserve values with characters <input> can't represent (newlines, etc.)\n if (typeof originalValue === 'string' && input.value === originalValue.replace(/[\\n\\r]/g, '')) {\n return originalValue;\n }\n return input.value;\n }\n // For textarea/select, check column type OR original value type\n if (column?.type === 'number' && input.value !== '') {\n return Number(input.value);\n }\n // Preserve numeric type for custom column types (e.g., currency)\n if (typeof originalValue === 'number' && input.value !== '') {\n return Number(input.value);\n }\n // Preserve null/undefined: if original was null/undefined and input is empty, return original\n if ((originalValue === null || originalValue === undefined) && input.value === '') {\n return originalValue;\n }\n return input.value;\n}\n\n/**\n * No-op updateRow function for rows without IDs.\n * Extracted to a named function to satisfy eslint no-empty-function.\n */\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nfunction noopUpdateRow(_changes: unknown): void {\n // Row has no ID - cannot update\n}\n\n/**\n * Auto-wire commit/cancel lifecycle for input elements in string-returned editors.\n */\nfunction wireEditorInputs(\n editorHost: HTMLElement,\n column: ColumnConfig<unknown>,\n commit: (value: unknown) => void,\n originalValue?: unknown,\n): void {\n const input = editorHost.querySelector('input,textarea,select') as\n | HTMLInputElement\n | HTMLTextAreaElement\n | HTMLSelectElement\n | null;\n if (!input) return;\n\n input.addEventListener('blur', () => {\n commit(getInputValue(input, column, originalValue));\n });\n\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.addEventListener('change', () => commit(input.checked));\n } else if (input instanceof HTMLSelectElement) {\n input.addEventListener('change', () => commit(getInputValue(input, column, originalValue)));\n }\n}\n\n// ============================================================================\n// EditingPlugin\n// ============================================================================\n\n/**\n * Editing Plugin for tbw-grid\n *\n * Enables inline cell editing in the grid. Provides built-in editors for common data types\n * and supports custom editor functions for specialized input scenarios.\n *\n * ## Why Opt-In?\n *\n * Editing is delivered as a plugin rather than built into the core grid:\n *\n * - **Smaller bundle** — Apps that only display data don't pay for editing code\n * - **Clear intent** — Explicit plugin registration makes editing capability obvious\n * - **Runtime validation** — Using `editable: true` without the plugin throws a helpful error\n *\n * ## Installation\n *\n * ```ts\n * import { EditingPlugin } from '@toolbox-web/grid/plugins/editing';\n * ```\n *\n * ## Edit Triggers\n *\n * Configure how editing is triggered with the `editOn` option:\n *\n * | Value | Behavior |\n * |-------|----------|\n * | `'click'` | Single click enters edit mode (default) |\n * | `'dblclick'` | Double-click enters edit mode |\n *\n * ## Keyboard Shortcuts\n *\n * | Key | Action |\n * |-----|--------|\n * | `Enter` | Commit edit and move down |\n * | `Tab` | Commit edit and move right |\n * | `Escape` | Cancel edit, restore original value |\n * | `Arrow Keys` | Navigate between cells (when not editing) |\n *\n * @example Basic editing with double-click trigger\n * ```ts\n * grid.gridConfig = {\n * columns: [\n * { field: 'name', editable: true },\n * { field: 'price', type: 'number', editable: true },\n * { field: 'active', type: 'boolean', editable: true },\n * ],\n * plugins: [new EditingPlugin({ editOn: 'dblclick' })],\n * };\n *\n * grid.addEventListener('cell-commit', (e) => {\n * const { field, oldValue, newValue } = e.detail;\n * console.log(`${field}: ${oldValue} → ${newValue}`);\n * });\n * ```\n *\n * @example Custom editor function\n * ```ts\n * columns: [\n * {\n * field: 'status',\n * editable: true,\n * editor: (ctx) => {\n * const select = document.createElement('select');\n * ['pending', 'active', 'completed'].forEach(opt => {\n * const option = document.createElement('option');\n * option.value = opt;\n * option.textContent = opt;\n * option.selected = ctx.value === opt;\n * select.appendChild(option);\n * });\n * select.addEventListener('change', () => ctx.commit(select.value));\n * return select;\n * },\n * },\n * ]\n * ```\n *\n * @see {@link EditingConfig} for configuration options\n * @see {@link EditorContext} for custom editor context\n * @see [Live Demos](?path=/docs/grid-plugins-editing--docs) for interactive examples\n */\nexport class EditingPlugin<T = unknown> extends BaseGridPlugin<EditingConfig> {\n /**\n * Plugin manifest - declares owned properties for configuration validation.\n * @internal\n */\n static override readonly manifest: PluginManifest = {\n ownedProperties: [\n {\n property: 'editable',\n level: 'column',\n description: 'the \"editable\" column property',\n isUsed: (v) => v === true,\n },\n {\n property: 'editor',\n level: 'column',\n description: 'the \"editor\" column property',\n },\n {\n property: 'editorParams',\n level: 'column',\n description: 'the \"editorParams\" column property',\n },\n ],\n events: [\n {\n type: 'cell-edit-committed',\n description: 'Emitted when a cell edit is committed (for plugin-to-plugin coordination)',\n },\n ],\n queries: [\n {\n type: 'isEditing',\n description: 'Returns whether any cell is currently being edited',\n },\n ],\n };\n\n /** @internal */\n readonly name = 'editing';\n /** @internal */\n override readonly styles = styles;\n\n /** @internal */\n protected override get defaultConfig(): Partial<EditingConfig> {\n return {\n mode: 'row',\n editOn: 'click',\n };\n }\n\n /**\n * Whether the grid is in 'grid' mode (all cells always editable).\n */\n get #isGridMode(): boolean {\n return this.config.mode === 'grid';\n }\n\n // #region Editing State (fully owned by plugin)\n\n /** Currently active edit row index, or -1 if not editing */\n #activeEditRow = -1;\n\n /** Row ID of the currently active edit row (stable across _rows replacement) */\n #activeEditRowId: string | undefined;\n\n /** Reference to the row object at edit-open time. Used as fallback in\n * #exitRowEdit when no row ID is available (prevents stale-index access). */\n #activeEditRowRef: T | undefined;\n\n /** Currently active edit column index, or -1 if not editing */\n #activeEditCol = -1;\n\n /** Snapshots of row data before editing started */\n #rowEditSnapshots = new Map<number, T>();\n\n /** Set of row IDs that have been modified (ID-based for stability) */\n #changedRowIds = new Set<string>();\n\n /** Set of cells currently in edit mode: \"rowIndex:colIndex\" */\n #editingCells = new Set<string>();\n\n /**\n * Value-change callbacks for active editors.\n * Keyed by \"rowIndex:field\" → callback that pushes updated values to the editor.\n * Populated during #injectEditor, cleaned up when editors are removed.\n */\n #editorValueCallbacks = new Map<string, (newValue: unknown) => void>();\n\n /** Flag to restore focus after next render (used when exiting edit mode) */\n #pendingFocusRestore = false;\n\n /** Row index pending animation after render, or -1 if none */\n #pendingRowAnimation = -1;\n\n /**\n * Invalid cell tracking: Map<rowId, Map<field, message>>\n * Used for validation feedback without canceling edits.\n */\n #invalidCells = new Map<string, Map<string, string>>();\n\n /**\n * In grid mode, tracks whether an input field is currently focused.\n * When true: arrow keys work within input (edit mode).\n * When false: arrow keys navigate between cells (navigation mode).\n * Escape switches to navigation mode, Enter switches to edit mode.\n */\n #gridModeInputFocused = false;\n\n /**\n * In grid mode, when true, prevents inputs from auto-focusing.\n * This is set when Escape is pressed (navigation mode) and cleared\n * when Enter is pressed or user explicitly clicks an input.\n */\n #gridModeEditLocked = false;\n\n /**\n * When true, only a single cell is being edited (triggered by F2 or `beginCellEdit`).\n * Tab and Arrow keys commit and close the editor instead of navigating to adjacent cells.\n */\n #singleCellEdit = false;\n\n // #endregion\n\n // #region Lifecycle\n\n /** @internal */\n override attach(grid: GridElement): void {\n super.attach(grid);\n\n const signal = this.disconnectSignal;\n const internalGrid = grid as unknown as InternalGrid<T>;\n\n // Inject editing state and methods onto grid for backward compatibility\n internalGrid._activeEditRows = -1;\n internalGrid._rowEditSnapshots = new Map();\n\n // Inject changedRows getter\n Object.defineProperty(grid, 'changedRows', {\n get: () => this.changedRows,\n configurable: true,\n });\n\n // Inject changedRowIds getter (new ID-based API)\n Object.defineProperty(grid, 'changedRowIds', {\n get: () => this.changedRowIds,\n configurable: true,\n });\n\n // Inject resetChangedRows method\n (grid as any).resetChangedRows = (silent?: boolean) => this.resetChangedRows(silent);\n\n // Inject beginBulkEdit method (for backward compatibility)\n (grid as any).beginBulkEdit = (rowIndex: number, field?: string) => {\n if (field) {\n this.beginCellEdit(rowIndex, field);\n }\n // If no field specified, we can't start editing without a specific cell\n };\n\n // Document-level Escape to cancel editing (only in 'row' mode)\n document.addEventListener(\n 'keydown',\n (e: KeyboardEvent) => {\n // In grid mode, Escape doesn't exit edit mode\n if (this.#isGridMode) return;\n if (e.key === 'Escape' && this.#activeEditRow !== -1) {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return;\n }\n }\n this.#exitRowEdit(this.#activeEditRow, true);\n }\n },\n { capture: true, signal },\n );\n\n // Click outside to commit editing (only in 'row' mode)\n // Use queueMicrotask to allow pending change events to fire first.\n // This is important for Angular/React editors where the (change) event\n // fires after mousedown but before mouseup/click.\n document.addEventListener(\n 'mousedown',\n (e: MouseEvent) => {\n // In grid mode, clicking outside doesn't exit edit mode\n if (this.#isGridMode) return;\n if (this.#activeEditRow === -1) return;\n const rowEl = internalGrid.findRenderedRowElement?.(this.#activeEditRow);\n if (!rowEl) return;\n const path = (e.composedPath && e.composedPath()) || [];\n if (path.includes(rowEl)) return;\n\n // Allow users to prevent edit close via callback (e.g., when click is inside an overlay)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return;\n }\n }\n\n // Delay exit to allow pending change/commit events to fire\n queueMicrotask(() => {\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, false);\n }\n });\n },\n { signal },\n );\n\n // Listen for external row mutations to push updated values to active editors.\n // When field A commits and sets field B via updateRow(), field B's editor\n // (if open) must reflect the new value.\n this.gridElement.addEventListener(\n 'cell-change',\n (e: Event) => {\n const detail = (e as CustomEvent).detail as {\n rowIndex: number;\n field: string;\n newValue: unknown;\n source: string;\n };\n // Only push updates from cascade/api sources — not from the editor's own commit\n if (detail.source === 'user') return;\n const key = `${detail.rowIndex}:${detail.field}`;\n const cb = this.#editorValueCallbacks.get(key);\n if (cb) cb(detail.newValue);\n },\n { signal },\n );\n\n // In grid mode, request a full render to trigger afterCellRender hooks\n if (this.#isGridMode) {\n internalGrid._isGridEditMode = true;\n this.gridElement.classList.add('tbw-grid-mode');\n this.requestRender();\n\n // Track focus/blur on inputs to maintain navigation vs edit mode state\n this.gridElement.addEventListener(\n 'focusin',\n (e: FocusEvent) => {\n const target = e.target as HTMLElement;\n if (target.matches(FOCUSABLE_EDITOR_SELECTOR)) {\n // If edit is locked (navigation mode), blur the input immediately\n if (this.#gridModeEditLocked) {\n target.blur();\n this.gridElement.focus();\n return;\n }\n this.#gridModeInputFocused = true;\n }\n },\n { signal },\n );\n\n this.gridElement.addEventListener(\n 'focusout',\n (e: FocusEvent) => {\n const related = e.relatedTarget as HTMLElement | null;\n // Only clear if focus went outside grid or to a non-input element\n if (!related || !this.gridElement.contains(related) || !related.matches(FOCUSABLE_EDITOR_SELECTOR)) {\n this.#gridModeInputFocused = false;\n }\n },\n { signal },\n );\n\n // Handle Escape key directly on the grid element (capture phase)\n // This ensures we intercept Escape even when focus is inside an input\n this.gridElement.addEventListener(\n 'keydown',\n (e: KeyboardEvent) => {\n if (e.key === 'Escape' && this.#gridModeInputFocused) {\n // Allow users to prevent Escape handling via callback (e.g., when overlay is open).\n // In grid mode, Escape transitions from editing to navigation mode, so we check\n // onBeforeEditClose to let overlays (dropdowns, autocompletes) close first.\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the overlay handle Escape\n }\n }\n const activeEl = document.activeElement as HTMLElement;\n if (activeEl && this.gridElement.contains(activeEl)) {\n activeEl.blur();\n // Move focus to the grid container so arrow keys work\n this.gridElement.focus();\n }\n this.#gridModeInputFocused = false;\n this.#gridModeEditLocked = true; // Lock edit mode until Enter/click\n e.preventDefault();\n e.stopPropagation();\n }\n },\n { capture: true, signal },\n );\n\n // Handle click on inputs - unlock edit mode when user explicitly clicks\n this.gridElement.addEventListener(\n 'mousedown',\n (e: MouseEvent) => {\n const target = e.target as HTMLElement;\n if (target.matches(FOCUSABLE_EDITOR_SELECTOR)) {\n this.#gridModeEditLocked = false; // User clicked input - allow edit\n }\n },\n { signal },\n );\n }\n }\n\n /** @internal */\n override detach(): void {\n const internalGrid = this.gridElement as unknown as InternalGrid<T>;\n internalGrid._isGridEditMode = false;\n this.gridElement.classList.remove('tbw-grid-mode');\n this.#activeEditRow = -1;\n this.#activeEditRowId = undefined;\n this.#activeEditRowRef = undefined;\n this.#activeEditCol = -1;\n this.#rowEditSnapshots.clear();\n this.#changedRowIds.clear();\n this.#editingCells.clear();\n this.#editorValueCallbacks.clear();\n this.#gridModeInputFocused = false;\n this.#gridModeEditLocked = false;\n this.#singleCellEdit = false;\n super.detach();\n }\n\n /**\n * Handle plugin queries.\n * @internal\n */\n override handleQuery(query: PluginQuery): unknown {\n if (query.type === 'isEditing') {\n // In grid mode, we're always editing\n return this.#isGridMode || this.#activeEditRow !== -1;\n }\n return undefined;\n }\n\n // #endregion\n\n // #region Event Handlers (event distribution)\n\n /**\n * Handle cell clicks - start editing if configured for click mode.\n * Both click and dblclick events come through this handler.\n * Starts row-based editing (all editable cells in the row get editors).\n * @internal\n */\n override onCellClick(event: CellClickEvent): boolean | void {\n // In grid mode, all cells are already editable - no need to trigger row edit\n if (this.#isGridMode) return false;\n\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n\n // Check if editing is disabled\n if (editOn === false || editOn === 'manual') return false;\n\n // Check if this is click or dblclick mode\n if (editOn !== 'click' && editOn !== 'dblclick') return false;\n\n // Check if the event type matches the edit mode\n const isDoubleClick = event.originalEvent.type === 'dblclick';\n if (editOn === 'click' && isDoubleClick) return false; // In click mode, only handle single clicks\n if (editOn === 'dblclick' && !isDoubleClick) return false; // In dblclick mode, only handle double clicks\n\n const { rowIndex } = event;\n\n // Check if any column in the row is editable\n const hasEditableColumn = internalGrid._columns?.some((col) => col.editable);\n if (!hasEditableColumn) return false;\n\n // Start row-based editing (all editable cells get editors)\n event.originalEvent.stopPropagation();\n this.beginBulkEdit(rowIndex);\n return true; // Handled\n }\n\n /**\n * Handle keyboard events for edit lifecycle.\n * @internal\n */\n override onKeyDown(event: KeyboardEvent): boolean | void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n\n // Escape: cancel current edit (row mode) or exit edit mode (grid mode)\n if (event.key === 'Escape') {\n // In grid mode: blur input to enable arrow key navigation\n if (this.#isGridMode && this.#gridModeInputFocused) {\n // Allow users to prevent Escape handling via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let overlay handle Escape\n }\n }\n const activeEl = document.activeElement as HTMLElement;\n if (activeEl && this.gridElement.contains(activeEl)) {\n activeEl.blur();\n }\n this.#gridModeInputFocused = false;\n // Update focus styling\n this.requestAfterRender();\n return true;\n }\n\n // In row mode: cancel edit\n if (this.#activeEditRow !== -1 && !this.#isGridMode) {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let event reach overlay\n }\n }\n this.#exitRowEdit(this.#activeEditRow, true);\n return true;\n }\n }\n\n // Arrow keys in grid mode when not editing input: navigate cells\n if (\n this.#isGridMode &&\n !this.#gridModeInputFocused &&\n (event.key === 'ArrowUp' || event.key === 'ArrowDown' || event.key === 'ArrowLeft' || event.key === 'ArrowRight')\n ) {\n // Let the grid's default keyboard navigation handle this\n return false;\n }\n\n // Arrow Up/Down in grid mode when input is focused: let the editor handle it\n // (e.g., ArrowDown opens autocomplete/datepicker overlays, ArrowUp/Down navigates options)\n if (this.#isGridMode && this.#gridModeInputFocused && (event.key === 'ArrowUp' || event.key === 'ArrowDown')) {\n return true; // Handled: block grid navigation, let event reach editor\n }\n\n // Arrow Up/Down while editing: commit and exit edit mode, move to adjacent row (only in 'row' mode)\n if ((event.key === 'ArrowUp' || event.key === 'ArrowDown') && this.#activeEditRow !== -1 && !this.#isGridMode) {\n // Allow users to prevent row navigation via callback (e.g., when dropdown is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let event reach dropdown\n }\n }\n\n const maxRow = internalGrid._rows.length - 1;\n const currentRow = this.#activeEditRow;\n\n // Commit the current edit\n this.#exitRowEdit(currentRow, false);\n\n // Move focus to adjacent row (same column)\n if (event.key === 'ArrowDown') {\n internalGrid._focusRow = Math.min(maxRow, internalGrid._focusRow + 1);\n } else {\n internalGrid._focusRow = Math.max(0, internalGrid._focusRow - 1);\n }\n\n event.preventDefault();\n // Ensure the focused cell is scrolled into view\n ensureCellVisible(internalGrid);\n // Request render to update focus styling\n this.requestAfterRender();\n return true;\n }\n\n // Tab/Shift+Tab while editing: move to next/prev editable cell\n if (event.key === 'Tab' && (this.#activeEditRow !== -1 || this.#isGridMode)) {\n event.preventDefault();\n\n // In single-cell edit mode (F2), commit and close instead of navigating\n if (this.#singleCellEdit) {\n this.#exitRowEdit(this.#activeEditRow, false);\n return true;\n }\n\n const forward = !event.shiftKey;\n this.#handleTabNavigation(forward);\n return true;\n }\n\n // Space: toggle boolean cells (only when not in edit mode - let editors handle their own space)\n if (event.key === ' ' || event.key === 'Spacebar') {\n // If we're in row edit mode, let the event pass through to the editor (e.g., checkbox)\n if (this.#activeEditRow !== -1) {\n return false;\n }\n\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n if (focusRow >= 0 && focusCol >= 0) {\n const column = internalGrid._visibleColumns[focusCol];\n const rowData = internalGrid._rows[focusRow];\n if (column?.editable && column.type === 'boolean' && rowData) {\n const field = column.field;\n if (isSafePropertyKey(field)) {\n const currentValue = (rowData as Record<string, unknown>)[field];\n const newValue = !currentValue;\n this.#commitCellValue(focusRow, column, newValue, rowData);\n event.preventDefault();\n // Re-render to update the UI\n this.requestRender();\n return true;\n }\n }\n }\n // Space on non-boolean cell - don't block keyboard navigation\n return false;\n }\n\n // Enter (unmodified): start row edit, commit, or enter edit mode in grid mode\n if (event.key === 'Enter' && !event.shiftKey && !event.ctrlKey && !event.altKey && !event.metaKey) {\n // In grid mode when not editing: focus the current cell's input\n if (this.#isGridMode && !this.#gridModeInputFocused) {\n this.#focusCurrentCellEditor();\n return true;\n }\n\n if (this.#activeEditRow !== -1) {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n // This lets Enter select an item in a dropdown instead of committing the row\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let event reach overlay\n }\n }\n // Already editing - let cell handlers deal with it\n return false;\n }\n\n // Start row-based editing (not just the focused cell)\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n if (editOn === false || editOn === 'manual') return false;\n\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n if (focusRow >= 0) {\n // Check if ANY column in the row is editable\n const hasEditableColumn = internalGrid._columns?.some((col) => col.editable);\n if (hasEditableColumn) {\n // Emit cell-activate event BEFORE starting edit\n // This ensures consumers always get the activation event\n const column = internalGrid._visibleColumns[focusCol];\n const row = internalGrid._rows[focusRow];\n const field = column?.field ?? '';\n const value = field && row ? (row as Record<string, unknown>)[field] : undefined;\n const cellEl = this.gridElement.querySelector(`[data-row=\"${focusRow}\"][data-col=\"${focusCol}\"]`) as\n | HTMLElement\n | undefined;\n\n const activateEvent = new CustomEvent('cell-activate', {\n cancelable: true,\n bubbles: true,\n detail: {\n rowIndex: focusRow,\n colIndex: focusCol,\n field,\n value,\n row,\n cellEl,\n trigger: 'keyboard' as const,\n originalEvent: event,\n },\n });\n this.gridElement.dispatchEvent(activateEvent);\n\n // Also emit deprecated activate-cell for backwards compatibility\n const legacyEvent = new CustomEvent('activate-cell', {\n cancelable: true,\n bubbles: true,\n detail: { row: focusRow, col: focusCol },\n });\n this.gridElement.dispatchEvent(legacyEvent);\n\n // If consumer canceled the activation, don't start editing\n if (activateEvent.defaultPrevented || legacyEvent.defaultPrevented) {\n event.preventDefault();\n return true;\n }\n\n this.beginBulkEdit(focusRow);\n return true;\n }\n }\n // No editable columns - don't block keyboard navigation\n return false;\n }\n\n // F2: begin single-cell edit on the focused cell\n if (event.key === 'F2') {\n if (this.#activeEditRow !== -1 || this.#isGridMode) return false;\n\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n if (editOn === false) return false;\n\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n if (focusRow >= 0 && focusCol >= 0) {\n const column = internalGrid._visibleColumns[focusCol];\n if (column?.editable && column.field) {\n event.preventDefault();\n this.beginCellEdit(focusRow, column.field);\n return true;\n }\n }\n return false;\n }\n\n // Don't block other keyboard events\n return false;\n }\n\n // #endregion\n\n // #region Render Hooks\n\n /**\n * Process columns to merge type-level editorParams with column-level.\n * Column-level params take precedence.\n * @internal\n */\n override processColumns(columns: ColumnConfig<T>[]): ColumnConfig<T>[] {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const typeDefaults = (internalGrid as any).effectiveConfig?.typeDefaults;\n const adapter = internalGrid.__frameworkAdapter;\n\n // If no type defaults configured anywhere, skip processing\n if (!typeDefaults && !adapter?.getTypeDefault) return columns;\n\n return columns.map((col) => {\n if (!col.type) return col;\n\n // Get type-level editorParams\n let typeEditorParams: Record<string, unknown> | undefined;\n\n // Check grid-level typeDefaults first\n if (typeDefaults?.[col.type]?.editorParams) {\n typeEditorParams = typeDefaults[col.type].editorParams;\n }\n\n // Then check app-level (adapter) typeDefaults\n if (!typeEditorParams && adapter?.getTypeDefault) {\n const appDefault = adapter.getTypeDefault<T>(col.type);\n if (appDefault?.editorParams) {\n typeEditorParams = appDefault.editorParams;\n }\n }\n\n // No type-level params to merge\n if (!typeEditorParams) return col;\n\n // Merge: type-level as base, column-level wins on conflicts\n return {\n ...col,\n editorParams: { ...typeEditorParams, ...col.editorParams },\n };\n });\n }\n\n /**\n * After render, reapply editors to cells in edit mode.\n * This handles virtualization - when a row scrolls back into view,\n * we need to re-inject the editor.\n * @internal\n */\n override afterRender(): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n\n // Restore focus after exiting edit mode\n if (this.#pendingFocusRestore) {\n this.#pendingFocusRestore = false;\n this.#restoreCellFocus(internalGrid);\n }\n\n // Animate the row after render completes (so the row element exists)\n if (this.#pendingRowAnimation !== -1) {\n const rowIndex = this.#pendingRowAnimation;\n this.#pendingRowAnimation = -1;\n internalGrid.animateRow?.(rowIndex, 'change');\n }\n\n // In 'grid' mode, editors are injected via afterCellRender hook during render\n if (this.#isGridMode) return;\n\n if (this.#editingCells.size === 0) return;\n\n // Re-inject editors for any editing cells that are visible\n for (const cellKey of this.#editingCells) {\n const [rowStr, colStr] = cellKey.split(':');\n const rowIndex = parseInt(rowStr, 10);\n const colIndex = parseInt(colStr, 10);\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n if (!rowEl) continue;\n\n const cellEl = rowEl.querySelector(`.cell[data-col=\"${colIndex}\"]`) as HTMLElement | null;\n if (!cellEl || cellEl.classList.contains('editing')) continue;\n\n // Cell is visible but not in editing mode - reinject editor\n const rowData = internalGrid._rows[rowIndex];\n const column = internalGrid._visibleColumns[colIndex];\n if (rowData && column) {\n this.#injectEditor(rowData, rowIndex, column, colIndex, cellEl, true);\n }\n }\n }\n\n /**\n * Hook called after each cell is rendered.\n * In grid mode, injects editors into editable cells during render (no DOM queries needed).\n * @internal\n */\n override afterCellRender(context: AfterCellRenderContext): void {\n // Only inject editors in grid mode\n if (!this.#isGridMode) return;\n\n const { row, rowIndex, column, colIndex, cellElement } = context;\n\n // Skip non-editable columns\n if (!column.editable) return;\n\n // Skip if already has editor\n if (cellElement.classList.contains('editing')) return;\n\n // Inject editor (don't track in editingCells - we're always editing in grid mode)\n this.#injectEditor(row as T, rowIndex, column as ColumnConfig<T>, colIndex, cellElement, true);\n }\n\n /**\n * On scroll render, reapply editors to recycled cells.\n * @internal\n */\n override onScrollRender(): void {\n this.afterRender();\n }\n\n // #endregion\n\n // #region Public API\n\n /**\n * Get all rows that have been modified.\n * Uses ID-based lookup for stability when rows are reordered.\n */\n get changedRows(): T[] {\n const rows: T[] = [];\n for (const id of this.#changedRowIds) {\n const row = this.grid.getRow(id) as T | undefined;\n if (row) rows.push(row);\n }\n return rows;\n }\n\n /**\n * Get IDs of all modified rows.\n */\n get changedRowIds(): string[] {\n return Array.from(this.#changedRowIds);\n }\n\n /**\n * Get the currently active edit row index, or -1 if not editing.\n */\n get activeEditRow(): number {\n return this.#activeEditRow;\n }\n\n /**\n * Get the currently active edit column index, or -1 if not editing.\n */\n get activeEditCol(): number {\n return this.#activeEditCol;\n }\n\n /**\n * Check if a specific row is currently being edited.\n */\n isRowEditing(rowIndex: number): boolean {\n return this.#activeEditRow === rowIndex;\n }\n\n /**\n * Check if a specific cell is currently being edited.\n */\n isCellEditing(rowIndex: number, colIndex: number): boolean {\n return this.#editingCells.has(`${rowIndex}:${colIndex}`);\n }\n\n /**\n * Check if a specific row has been modified.\n * @param rowIndex - Row index to check (will be converted to ID internally)\n */\n isRowChanged(rowIndex: number): boolean {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const row = internalGrid._rows[rowIndex];\n if (!row) return false;\n try {\n const rowId = internalGrid.getRowId?.(row);\n return rowId ? this.#changedRowIds.has(rowId) : false;\n } catch {\n return false;\n }\n }\n\n /**\n * Check if a row with the given ID has been modified.\n * @param rowId - Row ID to check\n */\n isRowChangedById(rowId: string): boolean {\n return this.#changedRowIds.has(rowId);\n }\n\n // #region Cell Validation\n\n /**\n * Mark a cell as invalid with an optional validation message.\n * Invalid cells are marked with a `data-invalid` attribute for styling.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n * @param message - Optional validation message (for tooltips or display)\n *\n * @example\n * ```typescript\n * // In cell-commit handler:\n * grid.addEventListener('cell-commit', (e) => {\n * if (e.detail.field === 'email' && !isValidEmail(e.detail.value)) {\n * e.detail.setInvalid('Invalid email format');\n * }\n * });\n *\n * // Or programmatically:\n * editingPlugin.setInvalid('row-123', 'email', 'Invalid email format');\n * ```\n */\n setInvalid(rowId: string, field: string, message = ''): void {\n let rowInvalids = this.#invalidCells.get(rowId);\n if (!rowInvalids) {\n rowInvalids = new Map();\n this.#invalidCells.set(rowId, rowInvalids);\n }\n rowInvalids.set(field, message);\n this.#syncInvalidCellAttribute(rowId, field, true);\n }\n\n /**\n * Clear the invalid state for a specific cell.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n */\n clearInvalid(rowId: string, field: string): void {\n const rowInvalids = this.#invalidCells.get(rowId);\n if (rowInvalids) {\n rowInvalids.delete(field);\n if (rowInvalids.size === 0) {\n this.#invalidCells.delete(rowId);\n }\n }\n this.#syncInvalidCellAttribute(rowId, field, false);\n }\n\n /**\n * Clear all invalid cells for a specific row.\n *\n * @param rowId - The row ID (from getRowId)\n */\n clearRowInvalid(rowId: string): void {\n const rowInvalids = this.#invalidCells.get(rowId);\n if (rowInvalids) {\n const fields = Array.from(rowInvalids.keys());\n this.#invalidCells.delete(rowId);\n fields.forEach((field) => this.#syncInvalidCellAttribute(rowId, field, false));\n }\n }\n\n /**\n * Clear all invalid cell states across all rows.\n */\n clearAllInvalid(): void {\n const entries = Array.from(this.#invalidCells.entries());\n this.#invalidCells.clear();\n entries.forEach(([rowId, fields]) => {\n fields.forEach((_, field) => this.#syncInvalidCellAttribute(rowId, field, false));\n });\n }\n\n /**\n * Check if a specific cell is marked as invalid.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n * @returns True if the cell is marked as invalid\n */\n isCellInvalid(rowId: string, field: string): boolean {\n return this.#invalidCells.get(rowId)?.has(field) ?? false;\n }\n\n /**\n * Get the validation message for an invalid cell.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n * @returns The validation message, or undefined if cell is valid\n */\n getInvalidMessage(rowId: string, field: string): string | undefined {\n return this.#invalidCells.get(rowId)?.get(field);\n }\n\n /**\n * Check if a row has any invalid cells.\n *\n * @param rowId - The row ID (from getRowId)\n * @returns True if the row has at least one invalid cell\n */\n hasInvalidCells(rowId: string): boolean {\n const rowInvalids = this.#invalidCells.get(rowId);\n return rowInvalids ? rowInvalids.size > 0 : false;\n }\n\n /**\n * Get all invalid fields for a row.\n *\n * @param rowId - The row ID (from getRowId)\n * @returns Map of field names to validation messages\n */\n getInvalidFields(rowId: string): Map<string, string> {\n return new Map(this.#invalidCells.get(rowId) ?? []);\n }\n\n /**\n * Sync the data-invalid attribute on a cell element.\n */\n #syncInvalidCellAttribute(rowId: string, field: string, invalid: boolean): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const colIndex = internalGrid._visibleColumns?.findIndex((c) => c.field === field);\n if (colIndex === -1 || colIndex === undefined) return;\n\n // Find the row element by rowId\n const rows = internalGrid._rows;\n const rowIndex = rows?.findIndex((r) => {\n try {\n return internalGrid.getRowId?.(r) === rowId;\n } catch {\n return false;\n }\n });\n if (rowIndex === -1 || rowIndex === undefined) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${colIndex}\"]`) as HTMLElement | null;\n if (!cellEl) return;\n\n if (invalid) {\n cellEl.setAttribute('data-invalid', 'true');\n const message = this.#invalidCells.get(rowId)?.get(field);\n if (message) {\n cellEl.setAttribute('title', message);\n }\n } else {\n cellEl.removeAttribute('data-invalid');\n cellEl.removeAttribute('title');\n }\n }\n\n // #endregion\n\n /**\n * Reset all change tracking.\n * @param silent - If true, suppresses the `changed-rows-reset` event\n * @fires changed-rows-reset - Emitted when tracking is reset (unless silent)\n */\n resetChangedRows(silent?: boolean): void {\n const rows = this.changedRows;\n const ids = this.changedRowIds;\n this.#changedRowIds.clear();\n this.#syncGridEditState();\n\n if (!silent) {\n this.emit<ChangedRowsResetDetail<T>>('changed-rows-reset', { rows, ids });\n }\n\n // Clear visual indicators\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n internalGrid._rowPool?.forEach((r) => r.classList.remove('changed'));\n }\n\n /**\n * Programmatically begin editing a cell.\n * @param rowIndex - Index of the row to edit\n * @param field - Field name of the column to edit\n * @fires cell-commit - Emitted when the cell value is committed (on blur or Enter)\n */\n beginCellEdit(rowIndex: number, field: string): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const colIndex = internalGrid._visibleColumns.findIndex((c) => c.field === field);\n if (colIndex === -1) return;\n\n const column = internalGrid._visibleColumns[colIndex];\n if (!column?.editable) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${colIndex}\"]`) as HTMLElement | null;\n if (!cellEl) return;\n\n this.#singleCellEdit = true;\n this.#beginCellEdit(rowIndex, colIndex, cellEl);\n }\n\n /**\n * Programmatically begin editing all editable cells in a row.\n * @param rowIndex - Index of the row to edit\n * @fires cell-commit - Emitted for each cell value that is committed\n * @fires row-commit - Emitted when focus leaves the row\n */\n beginBulkEdit(rowIndex: number): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n if (editOn === false) return;\n\n const hasEditableColumn = internalGrid._columns?.some((col) => col.editable);\n if (!hasEditableColumn) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n if (!rowEl) return;\n\n // Bulk edit clears single-cell mode\n this.#singleCellEdit = false;\n\n // Start row edit\n const rowData = internalGrid._rows[rowIndex];\n this.#startRowEdit(rowIndex, rowData);\n\n // Enter edit mode on all editable cells\n Array.from(rowEl.children).forEach((cell, i) => {\n const col = internalGrid._visibleColumns[i];\n if (col?.editable) {\n const cellEl = cell as HTMLElement;\n if (!cellEl.classList.contains('editing')) {\n this.#injectEditor(rowData, rowIndex, col, i, cellEl, true);\n }\n }\n });\n\n // Focus the first editable cell\n setTimeout(() => {\n let targetCell = rowEl.querySelector(`.cell[data-col=\"${internalGrid._focusCol}\"]`);\n if (!targetCell?.classList.contains('editing')) {\n targetCell = rowEl.querySelector('.cell.editing');\n }\n if (targetCell?.classList.contains('editing')) {\n const editor = (targetCell as HTMLElement).querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n try {\n editor?.focus({ preventScroll: true });\n } catch {\n /* empty */\n }\n }\n }, 0);\n }\n\n /**\n * Commit the currently active row edit.\n * @fires row-commit - Emitted after the row edit is committed\n */\n commitActiveRowEdit(): void {\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, false);\n }\n }\n\n /**\n * Cancel the currently active row edit.\n */\n cancelActiveRowEdit(): void {\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, true);\n }\n }\n\n // #endregion\n\n // #region Internal Methods\n\n /**\n * Begin editing a single cell.\n */\n #beginCellEdit(rowIndex: number, colIndex: number, cellEl: HTMLElement): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const rowData = internalGrid._rows[rowIndex];\n const column = internalGrid._visibleColumns[colIndex];\n\n if (!rowData || !column?.editable) return;\n if (cellEl.classList.contains('editing')) return;\n\n // Start row edit if not already\n if (this.#activeEditRow !== rowIndex) {\n this.#startRowEdit(rowIndex, rowData);\n }\n\n this.#activeEditCol = colIndex;\n this.#injectEditor(rowData, rowIndex, column, colIndex, cellEl, false);\n }\n\n /**\n * Focus the editor input in the currently focused cell (grid mode only).\n * Used when pressing Enter to enter edit mode from navigation mode.\n */\n #focusCurrentCellEditor(): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n\n if (focusRow < 0 || focusCol < 0) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(focusRow);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${focusCol}\"]`) as HTMLElement | null;\n\n if (cellEl?.classList.contains('editing')) {\n const editor = cellEl.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n if (editor) {\n this.#gridModeEditLocked = false; // Unlock edit mode - user pressed Enter\n editor.focus();\n this.#gridModeInputFocused = true;\n // Select all text in text inputs for quick replacement\n if (editor instanceof HTMLInputElement && (editor.type === 'text' || editor.type === 'number')) {\n editor.select();\n }\n }\n }\n }\n\n /**\n * Handle Tab/Shift+Tab navigation while editing.\n * Moves to next/previous editable cell, staying in edit mode.\n * Wraps to next/previous row when reaching row boundaries.\n */\n #handleTabNavigation(forward: boolean): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const rows = internalGrid._rows;\n // In grid mode, use focusRow since there's no active edit row\n const currentRow = this.#isGridMode ? internalGrid._focusRow : this.#activeEditRow;\n\n // Get editable column indices\n const editableCols = internalGrid._visibleColumns.map((c, i) => (c.editable ? i : -1)).filter((i) => i >= 0);\n if (editableCols.length === 0) return;\n\n const currentIdx = editableCols.indexOf(internalGrid._focusCol);\n const nextIdx = currentIdx + (forward ? 1 : -1);\n\n // Can move within same row?\n if (nextIdx >= 0 && nextIdx < editableCols.length) {\n internalGrid._focusCol = editableCols[nextIdx];\n const rowEl = internalGrid.findRenderedRowElement?.(currentRow);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${editableCols[nextIdx]}\"]`) as HTMLElement | null;\n if (cellEl?.classList.contains('editing')) {\n const editor = cellEl.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n editor?.focus({ preventScroll: true });\n }\n ensureCellVisible(internalGrid, { forceHorizontalScroll: true });\n return;\n }\n\n // Can move to adjacent row?\n const nextRow = currentRow + (forward ? 1 : -1);\n if (nextRow >= 0 && nextRow < rows.length) {\n // In grid mode, just move focus (all rows are always editable)\n if (this.#isGridMode) {\n internalGrid._focusRow = nextRow;\n internalGrid._focusCol = forward ? editableCols[0] : editableCols[editableCols.length - 1];\n ensureCellVisible(internalGrid, { forceHorizontalScroll: true });\n // Focus the editor in the new cell after render\n this.requestAfterRender();\n setTimeout(() => {\n const rowEl = internalGrid.findRenderedRowElement?.(nextRow);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${internalGrid._focusCol}\"]`) as HTMLElement | null;\n if (cellEl?.classList.contains('editing')) {\n const editor = cellEl.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n editor?.focus({ preventScroll: true });\n }\n }, 0);\n } else {\n // In row mode, commit current row and enter next row\n this.#exitRowEdit(currentRow, false);\n internalGrid._focusRow = nextRow;\n internalGrid._focusCol = forward ? editableCols[0] : editableCols[editableCols.length - 1];\n this.beginBulkEdit(nextRow);\n ensureCellVisible(internalGrid, { forceHorizontalScroll: true });\n }\n }\n // else: at boundary - stay put\n }\n\n /**\n * Sync the internal grid state with the plugin's editing state.\n */\n #syncGridEditState(): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n internalGrid._activeEditRows = this.#activeEditRow;\n internalGrid._rowEditSnapshots = this.#rowEditSnapshots;\n }\n\n /**\n * Snapshot original row data and mark as editing.\n */\n #startRowEdit(rowIndex: number, rowData: T): void {\n if (this.#activeEditRow !== rowIndex) {\n this.#rowEditSnapshots.set(rowIndex, { ...rowData });\n this.#activeEditRow = rowIndex;\n this.#activeEditRowRef = rowData;\n\n // Store stable row ID for resilience against _rows replacement during editing\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n try {\n this.#activeEditRowId = internalGrid.getRowId?.(rowData) ?? undefined;\n } catch {\n this.#activeEditRowId = undefined;\n }\n\n this.#syncGridEditState();\n\n // Emit edit-open event (row mode only)\n if (!this.#isGridMode) {\n this.emit<EditOpenDetail<T>>('edit-open', {\n rowIndex,\n rowId: this.#activeEditRowId ?? '',\n row: rowData,\n });\n }\n }\n }\n\n /**\n * Exit editing for a row.\n */\n #exitRowEdit(rowIndex: number, revert: boolean): void {\n if (this.#activeEditRow !== rowIndex) return;\n\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const snapshot = this.#rowEditSnapshots.get(rowIndex);\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n\n // Resolve the row being edited using the stored row ID.\n // The _rows array may have been replaced (e.g. Angular pushing new rows\n // via directive effect) since editing started, so _rows[rowIndex] could\n // point to a completely different row. The ID map is always up-to-date.\n // Without an ID we fall back to the stored row reference from edit-open\n // (#activeEditRowRef) — safer than _rows[rowIndex] which may be stale.\n let rowId = this.#activeEditRowId;\n const entry = rowId ? internalGrid._getRowEntry(rowId) : undefined;\n const current = entry?.row ?? this.#activeEditRowRef ?? internalGrid._rows[rowIndex];\n\n if (!rowId && current) {\n try {\n rowId = internalGrid.getRowId?.(current);\n } catch {\n // Row has no ID - skip ID-based tracking\n }\n }\n\n // Collect and commit values from active editors before re-rendering\n if (!revert && rowEl && current) {\n const editingCells = rowEl.querySelectorAll('.cell.editing');\n editingCells.forEach((cell) => {\n const colIndex = Number((cell as HTMLElement).getAttribute('data-col'));\n if (isNaN(colIndex)) return;\n const col = internalGrid._visibleColumns[colIndex];\n if (!col) return;\n\n // Skip cells with externally-managed editors (framework adapters like Angular/React/Vue).\n // These editors handle their own commits via the commit() callback - we should NOT\n // try to read values from their DOM inputs (which may contain formatted display values).\n if ((cell as HTMLElement).hasAttribute('data-editor-managed')) {\n return;\n }\n\n const input = cell.querySelector('input,textarea,select') as\n | HTMLInputElement\n | HTMLTextAreaElement\n | HTMLSelectElement\n | null;\n if (input) {\n const field = col.field as keyof T;\n const originalValue = current[field];\n const val = getInputValue(input, col, originalValue);\n if (originalValue !== val) {\n this.#commitCellValue(rowIndex, col, val, current);\n }\n }\n });\n }\n\n // Flush managed editors (framework adapters) before clearing state.\n // At this point the commit() callback is still active, so editors can\n // synchronously commit their pending values in response to this event.\n if (!revert && !this.#isGridMode && current) {\n this.emit<BeforeEditCloseDetail<T>>('before-edit-close', {\n rowIndex,\n rowId: rowId ?? '',\n row: current,\n });\n }\n\n // Revert if requested\n if (revert && snapshot && current) {\n Object.keys(snapshot as object).forEach((k) => {\n (current as Record<string, unknown>)[k] = (snapshot as Record<string, unknown>)[k];\n });\n if (rowId) {\n this.#changedRowIds.delete(rowId);\n this.clearRowInvalid(rowId);\n }\n } else if (!revert && current) {\n // Compare snapshot vs current to detect if changes were made during THIS edit session\n const changedThisSession = this.#hasRowChanged(snapshot, current);\n\n // Check if this row has any cumulative changes (via ID tracking)\n // Fall back to session-based detection when no row ID is available\n const changed = rowId ? this.#changedRowIds.has(rowId) : changedThisSession;\n\n // Emit cancelable row-commit event\n const cancelled = this.emitCancelable<RowCommitDetail<T>>('row-commit', {\n rowIndex,\n rowId: rowId ?? '',\n row: current,\n oldValue: snapshot,\n newValue: current,\n changed,\n changedRows: this.changedRows,\n changedRowIds: this.changedRowIds,\n });\n\n // If consumer called preventDefault(), revert the row\n if (cancelled && snapshot) {\n Object.keys(snapshot as object).forEach((k) => {\n (current as Record<string, unknown>)[k] = (snapshot as Record<string, unknown>)[k];\n });\n if (rowId) {\n this.#changedRowIds.delete(rowId);\n this.clearRowInvalid(rowId);\n }\n } else if (!cancelled && changedThisSession && this.isAnimationEnabled) {\n // Animate the row only if changes were made during this edit session\n // (deferred to afterRender so the row element exists after re-render)\n this.#pendingRowAnimation = rowIndex;\n }\n }\n\n // Clear editing state\n this.#rowEditSnapshots.delete(rowIndex);\n this.#activeEditRow = -1;\n this.#activeEditRowId = undefined;\n this.#activeEditRowRef = undefined;\n this.#activeEditCol = -1;\n this.#singleCellEdit = false;\n this.#syncGridEditState();\n\n // Remove all editing cells for this row.\n // Note: these keys use the rowIndex captured at edit-open time. Even if _rows\n // was replaced and the row moved to a different index, the keys still match\n // what was inserted during this edit session (same captured rowIndex).\n for (const cellKey of this.#editingCells) {\n if (cellKey.startsWith(`${rowIndex}:`)) {\n this.#editingCells.delete(cellKey);\n }\n }\n // Remove value-change callbacks for this row (same captured-index rationale)\n for (const callbackKey of this.#editorValueCallbacks.keys()) {\n if (callbackKey.startsWith(`${rowIndex}:`)) {\n this.#editorValueCallbacks.delete(callbackKey);\n }\n }\n\n // Mark that focus should be restored after the upcoming render completes.\n // This must be set BEFORE refreshVirtualWindow because it calls afterRender()\n // synchronously, which reads this flag.\n this.#pendingFocusRestore = true;\n\n // Re-render the row to remove editors\n if (rowEl) {\n // Remove editing class and re-render cells\n rowEl.querySelectorAll('.cell.editing').forEach((cell) => {\n cell.classList.remove('editing');\n clearEditingState(cell.parentElement as RowElementInternal);\n });\n\n // Refresh the virtual window to restore cell content WITHOUT rebuilding\n // the row model. requestRender() would trigger processRows (ROWS phase)\n // which re-sorts — causing the edited row to jump to a new position and\n // disappear from view. refreshVirtualWindow re-renders visible cells from\n // the current _rows order, keeping the row in place until the user\n // explicitly sorts again or new data arrives.\n internalGrid.refreshVirtualWindow(true);\n } else {\n // Row not visible - restore focus immediately (no render will happen)\n this.#restoreCellFocus(internalGrid);\n this.#pendingFocusRestore = false;\n }\n\n // Emit edit-close event (row mode only, fires for both commit and revert)\n if (!this.#isGridMode && current) {\n this.emit<EditCloseDetail<T>>('edit-close', {\n rowIndex,\n rowId: rowId ?? '',\n row: current,\n reverted: revert,\n });\n }\n }\n\n /**\n * Commit a single cell value change.\n * Uses ID-based change tracking for stability when rows are reordered.\n */\n #commitCellValue(rowIndex: number, column: ColumnConfig<T>, newValue: unknown, rowData: T): void {\n const field = column.field;\n if (!isSafePropertyKey(field)) return;\n const oldValue = (rowData as Record<string, unknown>)[field];\n if (oldValue === newValue) return;\n\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n\n // Get row ID for change tracking (may not exist if getRowId not configured)\n let rowId: string | undefined;\n try {\n rowId = this.grid.getRowId(rowData);\n } catch {\n // Row has no ID - will still work but won't be tracked in changedRowIds\n }\n\n const firstTime = rowId ? !this.#changedRowIds.has(rowId) : true;\n\n // Create updateRow helper for cascade updates (noop if row has no ID)\n const updateRow: (changes: Partial<T>) => void = rowId\n ? (changes) => this.grid.updateRow(rowId!, changes as Record<string, unknown>, 'cascade')\n : noopUpdateRow;\n\n // Track whether setInvalid was called during event handling\n let invalidWasSet = false;\n\n // Create setInvalid callback for validation (noop if row has no ID)\n const setInvalid = rowId\n ? (message?: string) => {\n invalidWasSet = true;\n this.setInvalid(rowId!, field, message ?? '');\n }\n : () => {}; // eslint-disable-line @typescript-eslint/no-empty-function\n\n // Emit cancelable event BEFORE applying the value\n const cancelled = this.emitCancelable<CellCommitDetail<T>>('cell-commit', {\n row: rowData,\n rowId: rowId ?? '',\n field,\n oldValue,\n value: newValue,\n rowIndex,\n changedRows: this.changedRows,\n changedRowIds: this.changedRowIds,\n firstTimeForRow: firstTime,\n updateRow,\n setInvalid,\n });\n\n // If consumer called preventDefault(), abort the commit\n if (cancelled) return;\n\n // Clear any previous invalid state for this cell ONLY if setInvalid wasn't called\n // (if setInvalid was called, the handler wants it to remain invalid)\n if (rowId && !invalidWasSet && this.isCellInvalid(rowId, field)) {\n this.clearInvalid(rowId, field);\n }\n\n // Apply the value and mark row as changed\n (rowData as Record<string, unknown>)[field] = newValue;\n if (rowId) {\n this.#changedRowIds.add(rowId);\n }\n this.#syncGridEditState();\n\n // Notify other plugins (e.g., UndoRedoPlugin) about the committed edit\n this.emitPluginEvent('cell-edit-committed', {\n rowIndex,\n field,\n oldValue,\n newValue,\n });\n\n // Mark the row visually as changed (animation happens when row edit closes)\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n if (rowEl) {\n rowEl.classList.add('changed');\n }\n }\n\n /**\n * Inject an editor into a cell.\n */\n #injectEditor(\n rowData: T,\n rowIndex: number,\n column: ColumnConfig<T>,\n colIndex: number,\n cell: HTMLElement,\n skipFocus: boolean,\n ): void {\n if (!column.editable) return;\n if (cell.classList.contains('editing')) return;\n\n // Get row ID for updateRow helper (may not exist)\n let rowId: string | undefined;\n try {\n rowId = this.grid.getRowId(rowData);\n } catch {\n // Row has no ID\n }\n\n // Create updateRow helper for cascade updates (noop if row has no ID)\n const updateRow: (changes: Partial<T>) => void = rowId\n ? (changes) => this.grid.updateRow(rowId!, changes as Record<string, unknown>, 'cascade')\n : noopUpdateRow;\n\n const originalValue = isSafePropertyKey(column.field)\n ? (rowData as Record<string, unknown>)[column.field]\n : undefined;\n\n cell.classList.add('editing');\n this.#editingCells.add(`${rowIndex}:${colIndex}`);\n\n const rowEl = cell.parentElement as RowElementInternal | null;\n if (rowEl) incrementEditingCount(rowEl);\n\n let editFinalized = false;\n const commit = (newValue: unknown) => {\n // In grid mode, always allow commits (we're always editing)\n // In row mode, only allow commits if we're in an active edit session\n if (editFinalized || (!this.#isGridMode && this.#activeEditRow === -1)) return;\n // Resolve row and index fresh at commit time.\n // With a row ID we use _getRowEntry for O(1) lookup — this is resilient\n // against _rows being replaced (e.g. Angular directive effect).\n // Without a row ID we fall back to the captured rowData reference.\n // Using _rows[rowIndex] without an ID is unsafe: the index may be stale\n // after _rows replacement, which would commit to the WRONG row.\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const entry = rowId ? internalGrid._getRowEntry(rowId) : undefined;\n const currentRowData = (entry?.row ?? rowData) as T;\n const currentIndex = entry?.index ?? rowIndex;\n this.#commitCellValue(currentIndex, column, newValue, currentRowData);\n };\n const cancel = () => {\n editFinalized = true;\n if (isSafePropertyKey(column.field)) {\n // Same ID-first / captured-rowData fallback as commit — see comment above.\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const entry = rowId ? internalGrid._getRowEntry(rowId) : undefined;\n const currentRowData = (entry?.row ?? rowData) as T;\n (currentRowData as Record<string, unknown>)[column.field] = originalValue;\n }\n };\n\n const editorHost = document.createElement('div');\n editorHost.className = 'tbw-editor-host';\n cell.innerHTML = '';\n cell.appendChild(editorHost);\n\n // Keydown handler for Enter/Escape\n editorHost.addEventListener('keydown', (e: KeyboardEvent) => {\n if (e.key === 'Enter') {\n // In grid mode, Enter just commits without exiting\n if (this.#isGridMode) {\n e.stopPropagation();\n e.preventDefault();\n // Get current value and commit\n const input = editorHost.querySelector('input,textarea,select') as\n | HTMLInputElement\n | HTMLTextAreaElement\n | HTMLSelectElement\n | null;\n if (input) {\n commit(getInputValue(input, column as ColumnConfig<unknown>, originalValue));\n }\n return;\n }\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n editFinalized = true;\n this.#exitRowEdit(rowIndex, false);\n }\n if (e.key === 'Escape') {\n // In grid mode, Escape doesn't exit edit mode\n if (this.#isGridMode) {\n e.stopPropagation();\n e.preventDefault();\n return;\n }\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n cancel();\n this.#exitRowEdit(rowIndex, true);\n }\n });\n\n const colInternal = column as ColumnInternal<T>;\n const tplHolder = colInternal.__editorTemplate;\n // Resolve editor using priority chain: column → template → typeDefaults → adapter → built-in\n const editorSpec = resolveEditor(this.grid as unknown as InternalGrid<T>, colInternal) ?? defaultEditorFor(column);\n const value = originalValue;\n\n // Value-change callback registration.\n // Editors call onValueChange(cb) to receive pushes when the underlying row\n // is mutated externally (e.g., via updateRow from another cell's commit).\n // Multiple callbacks can be registered (user + auto-wire).\n const callbackKey = `${rowIndex}:${column.field}`;\n const callbacks: Array<(newValue: unknown) => void> = [];\n this.#editorValueCallbacks.set(callbackKey, (newVal) => {\n for (const cb of callbacks) cb(newVal);\n });\n const onValueChange = (cb: (newValue: unknown) => void) => {\n callbacks.push(cb);\n };\n\n if (editorSpec === 'template' && tplHolder) {\n this.#renderTemplateEditor(editorHost, colInternal, rowData, originalValue, commit, cancel, skipFocus, rowIndex);\n // Auto-update built-in template editors when value changes externally\n onValueChange((newVal) => {\n const input = editorHost.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(\n 'input,textarea,select',\n );\n if (input) {\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.checked = !!newVal;\n } else {\n input.value = String(newVal ?? '');\n }\n }\n });\n } else if (typeof editorSpec === 'string') {\n const el = document.createElement(editorSpec) as HTMLElement & { value?: unknown };\n el.value = value;\n el.addEventListener('change', () => commit(el.value));\n // Auto-update custom element editors when value changes externally\n onValueChange((newVal) => {\n el.value = newVal;\n });\n editorHost.appendChild(el);\n if (!skipFocus) {\n queueMicrotask(() => {\n const focusable = editorHost.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n focusable?.focus({ preventScroll: true });\n });\n }\n } else if (typeof editorSpec === 'function') {\n const ctx: EditorContext<T> = {\n row: rowData,\n rowId: rowId ?? '',\n value,\n field: column.field,\n column,\n commit,\n cancel,\n updateRow,\n onValueChange,\n };\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const produced = (editorSpec as any)(ctx);\n if (typeof produced === 'string') {\n editorHost.innerHTML = produced;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n wireEditorInputs(editorHost, column as any, commit, originalValue);\n // Auto-update wired inputs when value changes externally\n onValueChange((newVal) => {\n const input = editorHost.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(\n 'input,textarea,select',\n );\n if (input) {\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.checked = !!newVal;\n } else {\n input.value = String(newVal ?? '');\n }\n }\n });\n } else if (produced instanceof Node) {\n editorHost.appendChild(produced);\n const isSimpleInput =\n produced instanceof HTMLInputElement ||\n produced instanceof HTMLSelectElement ||\n produced instanceof HTMLTextAreaElement;\n if (!isSimpleInput) {\n cell.setAttribute('data-editor-managed', '');\n } else {\n // Auto-update simple inputs returned by factory functions\n onValueChange((newVal) => {\n if (produced instanceof HTMLInputElement && produced.type === 'checkbox') {\n produced.checked = !!newVal;\n } else {\n (produced as HTMLInputElement).value = String(newVal ?? '');\n }\n });\n }\n }\n if (!skipFocus) {\n queueMicrotask(() => {\n const focusable = editorHost.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n focusable?.focus({ preventScroll: true });\n });\n }\n } else if (editorSpec && typeof editorSpec === 'object') {\n const placeholder = document.createElement('div');\n placeholder.setAttribute('data-external-editor', '');\n placeholder.setAttribute('data-field', column.field);\n editorHost.appendChild(placeholder);\n cell.setAttribute('data-editor-managed', '');\n const context: EditorContext<T> = {\n row: rowData,\n rowId: rowId ?? '',\n value,\n field: column.field,\n column,\n commit,\n cancel,\n updateRow,\n onValueChange,\n };\n if (editorSpec.mount) {\n try {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n editorSpec.mount({ placeholder, context: context as any, spec: editorSpec });\n } catch (e) {\n console.warn(`[tbw-grid] External editor mount error for column '${column.field}':`, e);\n }\n } else {\n (this.grid as unknown as HTMLElement).dispatchEvent(\n new CustomEvent('mount-external-editor', { detail: { placeholder, spec: editorSpec, context } }),\n );\n }\n }\n }\n\n /**\n * Render a template-based editor.\n */\n #renderTemplateEditor(\n editorHost: HTMLElement,\n column: ColumnInternal<T>,\n rowData: T,\n originalValue: unknown,\n commit: (value: unknown) => void,\n cancel: () => void,\n skipFocus: boolean,\n rowIndex: number,\n ): void {\n const tplHolder = column.__editorTemplate;\n if (!tplHolder) return;\n\n const clone = tplHolder.cloneNode(true) as HTMLElement;\n const compiledEditor = column.__compiledEditor;\n\n if (compiledEditor) {\n clone.innerHTML = compiledEditor({\n row: rowData,\n value: originalValue,\n field: column.field,\n column,\n commit,\n cancel,\n });\n } else {\n clone.querySelectorAll<HTMLElement>('*').forEach((node) => {\n if (node.childNodes.length === 1 && node.firstChild?.nodeType === Node.TEXT_NODE) {\n node.textContent =\n node.textContent\n ?.replace(/{{\\s*value\\s*}}/g, originalValue == null ? '' : String(originalValue))\n .replace(/{{\\s*row\\.([a-zA-Z0-9_]+)\\s*}}/g, (_m, g: string) => {\n if (!isSafePropertyKey(g)) return '';\n const v = (rowData as Record<string, unknown>)[g];\n return v == null ? '' : String(v);\n }) || '';\n }\n });\n }\n\n const input = clone.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(\n 'input,textarea,select',\n );\n if (input) {\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.checked = !!originalValue;\n } else {\n input.value = String(originalValue ?? '');\n }\n\n let editFinalized = false;\n input.addEventListener('blur', () => {\n if (editFinalized) return;\n commit(getInputValue(input, column, originalValue));\n });\n input.addEventListener('keydown', (evt) => {\n const e = evt as KeyboardEvent;\n if (e.key === 'Enter') {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n editFinalized = true;\n commit(getInputValue(input, column, originalValue));\n this.#exitRowEdit(rowIndex, false);\n }\n if (e.key === 'Escape') {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n cancel();\n this.#exitRowEdit(rowIndex, true);\n }\n });\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.addEventListener('change', () => commit(input.checked));\n }\n if (!skipFocus) {\n setTimeout(() => input.focus({ preventScroll: true }), 0);\n }\n }\n editorHost.appendChild(clone);\n }\n\n /**\n * Compare snapshot vs current row to detect if any values changed during this edit session.\n * Uses shallow comparison of all properties.\n */\n #hasRowChanged(snapshot: T | undefined, current: T): boolean {\n if (!snapshot) return false;\n\n const snapshotObj = snapshot as Record<string, unknown>;\n const currentObj = current as Record<string, unknown>;\n\n // Check all keys in both objects\n const allKeys = new Set([...Object.keys(snapshotObj), ...Object.keys(currentObj)]);\n for (const key of allKeys) {\n if (snapshotObj[key] !== currentObj[key]) {\n return true;\n }\n }\n return false;\n }\n\n /**\n * Restore focus to cell after exiting edit mode.\n */\n #restoreCellFocus(internalGrid: InternalGrid<T>): void {\n queueMicrotask(() => {\n try {\n const rowIdx = internalGrid._focusRow;\n const colIdx = internalGrid._focusCol;\n const rowEl = internalGrid.findRenderedRowElement?.(rowIdx);\n if (rowEl) {\n Array.from(internalGrid._bodyEl.querySelectorAll('.cell-focus')).forEach((el) =>\n el.classList.remove('cell-focus'),\n );\n const cell = rowEl.querySelector(`.cell[data-row=\"${rowIdx}\"][data-col=\"${colIdx}\"]`) as HTMLElement | null;\n if (cell) {\n cell.classList.add('cell-focus');\n cell.setAttribute('aria-selected', 'true');\n if (!cell.hasAttribute('tabindex')) cell.setAttribute('tabindex', '-1');\n cell.focus({ preventScroll: true });\n }\n }\n } catch {\n /* empty */\n }\n });\n }\n\n // #endregion\n}\n"],"names":["defaultEditorFor","column","type","ctx","params","editorParams","input","document","createElement","value","String","min","max","step","placeholder","commit","Number","addEventListener","e","key","cancel","createNumberEditor","checked","Date","valueAsDate","split","createDateEditor","select","multi","multiple","includeEmpty","emptyOpt","textContent","emptyLabel","appendChild","options","raw","resolveOptions","forEach","opt","o","label","Array","isArray","includes","selected","values","from","selectedOptions","map","createSelectEditor","maxLength","pattern","inputVal","replace","createTextEditor","isSafePropertyKey","getInputValue","originalValue","HTMLInputElement","noopUpdateRow","_changes","EditingPlugin","BaseGridPlugin","static","ownedProperties","property","level","description","isUsed","v","events","queries","name","styles","defaultConfig","mode","editOn","isGridMode","this","config","activeEditRow","activeEditRowId","activeEditRowRef","activeEditCol","rowEditSnapshots","Map","changedRowIds","Set","editingCells","editorValueCallbacks","pendingFocusRestore","pendingRowAnimation","invalidCells","gridModeInputFocused","gridModeEditLocked","singleCellEdit","attach","grid","super","signal","disconnectSignal","internalGrid","_activeEditRows","_rowEditSnapshots","Object","defineProperty","get","changedRows","configurable","resetChangedRows","silent","beginBulkEdit","rowIndex","field","beginCellEdit","onBeforeEditClose","exitRowEdit","capture","rowEl","findRenderedRowElement","composedPath","queueMicrotask","gridElement","detail","source","cb","newValue","_isGridEditMode","classList","add","requestRender","target","matches","FOCUSABLE_EDITOR_SELECTOR","blur","focus","related","relatedTarget","contains","activeEl","activeElement","preventDefault","stopPropagation","detach","remove","clear","handleQuery","query","onCellClick","event","effectiveConfig","isDoubleClick","originalEvent","hasEditableColumn","_columns","some","col","editable","onKeyDown","requestAfterRender","maxRow","_rows","length","currentRow","_focusRow","Math","ensureCellVisible","forward","shiftKey","handleTabNavigation","focusRow","focusCol","_focusCol","_visibleColumns","rowData","commitCellValue","ctrlKey","altKey","metaKey","focusCurrentCellEditor","row","cellEl","querySelector","activateEvent","CustomEvent","cancelable","bubbles","colIndex","trigger","dispatchEvent","legacyEvent","defaultPrevented","processColumns","columns","typeDefaults","adapter","__frameworkAdapter","getTypeDefault","typeEditorParams","appDefault","afterRender","restoreCellFocus","animateRow","size","cellKey","rowStr","colStr","parseInt","injectEditor","afterCellRender","context","cellElement","onScrollRender","rows","id","getRow","push","isRowEditing","isCellEditing","has","isRowChanged","rowId","getRowId","isRowChangedById","setInvalid","message","rowInvalids","set","syncInvalidCellAttribute","clearInvalid","delete","clearRowInvalid","fields","keys","clearAllInvalid","entries","_","isCellInvalid","getInvalidMessage","hasInvalidCells","getInvalidFields","invalid","findIndex","c","r","setAttribute","removeAttribute","ids","syncGridEditState","emit","_rowPool","startRowEdit","children","cell","i","setTimeout","targetCell","editor","preventScroll","commitActiveRowEdit","cancelActiveRowEdit","editableCols","filter","nextIdx","indexOf","forceHorizontalScroll","nextRow","revert","snapshot","entry","_getRowEntry","current","querySelectorAll","getAttribute","isNaN","hasAttribute","val","k","changedThisSession","hasRowChanged","changed","cancelled","emitCancelable","oldValue","isAnimationEnabled","startsWith","callbackKey","__editingCellCount","clearEditingState","parentElement","refreshVirtualWindow","reverted","firstTime","updateRow","changes","invalidWasSet","firstTimeForRow","emitPluginEvent","skipFocus","count","editFinalized","currentRowData","currentIndex","index","editorHost","className","innerHTML","colInternal","tplHolder","__editorTemplate","editorSpec","gridTypeDefaults","resolveEditor","callbacks","newVal","onValueChange","renderTemplateEditor","el","focusable","produced","HTMLSelectElement","wireEditorInputs","Node","HTMLTextAreaElement","mount","spec","console","warn","clone","cloneNode","compiledEditor","__compiledEditor","node","childNodes","firstChild","nodeType","TEXT_NODE","_m","g","evt","snapshotObj","currentObj","allKeys","rowIdx","colIdx","_bodyEl"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"editing.umd.js","sources":["../../../../../libs/grid/src/lib/plugins/editing/editors.ts","../../../../../libs/grid/src/lib/plugins/editing/EditingPlugin.ts"],"sourcesContent":["/**\n * Default Editors Module\n *\n * Provides built-in editor factories for different column types.\n * Each editor type has its own factory function for consistency and readability.\n *\n * IMPORTANT: Editor factories should NOT call focus() on elements - they are called\n * before the element is appended to the DOM. The calling code (beginBulkEdit,\n * inlineEnterEdit) is responsible for focusing the correct editor after insertion.\n */\n\nimport type { ColumnConfig, ColumnEditorContext } from '../../core/types';\nimport type { DateEditorParams, NumberEditorParams, SelectEditorParams, TextEditorParams } from './types';\n\n// ============================================================================\n// Type Aliases\n// ============================================================================\n\n/** Option shape used by select editor (matches column.options) */\ntype ColumnOption = { label: string; value: unknown };\n\n/** Column with any row type (used for editor factories) */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype AnyColumn = ColumnConfig<any>;\n\n// ============================================================================\n// Utilities\n// ============================================================================\n\n/** Resolve column.options (handles both array and function forms) */\nfunction resolveOptions(column: AnyColumn): ColumnOption[] {\n const raw = column.options;\n if (!raw) return [];\n return typeof raw === 'function' ? raw() : raw;\n}\n\n// ============================================================================\n// Editor Factories\n// ============================================================================\n\n/** Creates a number input editor */\nfunction createNumberEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as NumberEditorParams | undefined;\n const input = document.createElement('input');\n input.type = 'number';\n input.value = ctx.value != null ? String(ctx.value) : '';\n\n if (params?.min !== undefined) input.min = String(params.min);\n if (params?.max !== undefined) input.max = String(params.max);\n if (params?.step !== undefined) input.step = String(params.step);\n if (params?.placeholder) input.placeholder = params.placeholder;\n\n const commit = () => ctx.commit(input.value === '' ? null : Number(input.value));\n input.addEventListener('blur', commit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Enter') commit();\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return input;\n };\n}\n\n/** Creates a checkbox editor for boolean values */\nfunction createBooleanEditor(): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const input = document.createElement('input');\n input.type = 'checkbox';\n input.checked = !!ctx.value;\n input.addEventListener('change', () => ctx.commit(input.checked));\n return input;\n };\n}\n\n/** Creates a date input editor */\nfunction createDateEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as DateEditorParams | undefined;\n const input = document.createElement('input');\n input.type = 'date';\n\n // Set initial value - handle both Date objects and string dates\n if (ctx.value instanceof Date) {\n input.valueAsDate = ctx.value;\n } else if (typeof ctx.value === 'string' && ctx.value) {\n // String date like \"2019-10-09\" - set directly as value\n input.value = ctx.value.split('T')[0]; // Handle ISO strings too\n }\n if (params?.min) input.min = params.min;\n if (params?.max) input.max = params.max;\n if (params?.placeholder) input.placeholder = params.placeholder;\n\n // Commit function preserves original type (string vs Date)\n const commit = () => {\n if (typeof ctx.value === 'string') {\n // Original was string, return string in YYYY-MM-DD format\n ctx.commit(input.value);\n } else {\n ctx.commit(input.valueAsDate);\n }\n };\n\n input.addEventListener('change', commit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return input;\n };\n}\n\n/** Creates a select dropdown editor */\nfunction createSelectEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as SelectEditorParams | undefined;\n const select = document.createElement('select');\n if (column.multi) select.multiple = true;\n\n // Add empty option if requested\n if (params?.includeEmpty) {\n const emptyOpt = document.createElement('option');\n emptyOpt.value = '';\n emptyOpt.textContent = params.emptyLabel ?? '';\n select.appendChild(emptyOpt);\n }\n\n // Populate options from column.options\n const options = resolveOptions(column);\n options.forEach((opt) => {\n const o = document.createElement('option');\n o.value = String(opt.value);\n o.textContent = opt.label;\n if (column.multi && Array.isArray(ctx.value) && ctx.value.includes(opt.value)) {\n o.selected = true;\n } else if (!column.multi && ctx.value === opt.value) {\n o.selected = true;\n }\n select.appendChild(o);\n });\n\n const commit = () => {\n if (column.multi) {\n const values = Array.from(select.selectedOptions).map((o) => o.value);\n ctx.commit(values);\n } else {\n ctx.commit(select.value);\n }\n };\n\n select.addEventListener('change', commit);\n select.addEventListener('blur', commit);\n select.addEventListener('keydown', (e) => {\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return select;\n };\n}\n\n/** Creates a text input editor (default) */\nfunction createTextEditor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement {\n return (ctx) => {\n const params = column.editorParams as TextEditorParams | undefined;\n const input = document.createElement('input');\n input.type = 'text';\n input.value = ctx.value != null ? String(ctx.value) : '';\n\n if (params?.maxLength !== undefined) input.maxLength = params.maxLength;\n if (params?.pattern) input.pattern = params.pattern;\n if (params?.placeholder) input.placeholder = params.placeholder;\n\n // Commit function preserves original type when possible\n const commit = () => {\n const inputVal = input.value;\n // Preserve null/undefined: empty input on a null/undefined field means no change\n if ((ctx.value === null || ctx.value === undefined) && inputVal === '') {\n return;\n }\n // Preserve values with characters that <input> can't represent (newlines, etc.).\n // If stripping those characters produces the same string, the user didn't change anything.\n if (typeof ctx.value === 'string' && inputVal === ctx.value.replace(/[\\n\\r]/g, '')) {\n return;\n }\n // Preserve numeric type for custom column types (e.g., currency)\n if (typeof ctx.value === 'number' && inputVal !== '') {\n ctx.commit(Number(inputVal));\n } else {\n ctx.commit(inputVal);\n }\n };\n\n input.addEventListener('blur', commit);\n input.addEventListener('keydown', (e) => {\n if (e.key === 'Enter') commit();\n if (e.key === 'Escape') ctx.cancel();\n });\n\n return input;\n };\n}\n\n// ============================================================================\n// Main Entry Point\n// ============================================================================\n\n/**\n * Returns a default editor factory function for the given column type.\n * Each editor handles commit on blur/Enter, and cancel on Escape.\n *\n * Note: Focus is NOT called here - the calling code handles focusing after DOM insertion.\n */\nexport function defaultEditorFor(column: AnyColumn): (ctx: ColumnEditorContext) => HTMLElement | string {\n switch (column.type) {\n case 'number':\n return createNumberEditor(column);\n case 'boolean':\n return createBooleanEditor();\n case 'date':\n return createDateEditor(column);\n case 'select':\n return createSelectEditor(column);\n default:\n return createTextEditor(column);\n }\n}\n\n// ============================================================================\n// Utility Export (used by EditingPlugin)\n// ============================================================================\n\n/**\n * Gets the current value from an input element, with type coercion based on column type.\n */\nexport function getInputValue(\n input: HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement,\n col: AnyColumn,\n): unknown {\n if (input instanceof HTMLSelectElement) {\n if (col.multi) {\n return Array.from(input.selectedOptions).map((o) => o.value);\n }\n return input.value;\n }\n if (input instanceof HTMLInputElement) {\n if (input.type === 'checkbox') return input.checked;\n if (input.type === 'number') return input.value === '' ? null : Number(input.value);\n if (input.type === 'date') return input.valueAsDate;\n }\n return input.value;\n}\n","/**\n * Editing Plugin\n *\n * Provides complete editing functionality for tbw-grid.\n * This plugin is FULLY SELF-CONTAINED - the grid has ZERO editing knowledge.\n *\n * The plugin:\n * - Owns all editing state (active cell, snapshots, changed rows)\n * - Uses event distribution (onCellClick, onKeyDown) to handle edit lifecycle\n * - Uses afterRender() hook to inject editors into cells\n * - Uses processColumns() to augment columns with editing metadata\n * - Emits its own events (cell-commit, row-commit, changed-rows-reset)\n *\n * Without this plugin, the grid cannot edit. With this plugin, editing\n * is fully functional without any core changes.\n */\n\nimport { ensureCellVisible } from '../../core/internal/keyboard';\nimport { FOCUSABLE_EDITOR_SELECTOR } from '../../core/internal/rows';\nimport type { AfterCellRenderContext, PluginManifest, PluginQuery } from '../../core/plugin/base-plugin';\nimport { BaseGridPlugin, type CellClickEvent, type GridElement } from '../../core/plugin/base-plugin';\nimport type {\n ColumnConfig,\n ColumnEditorSpec,\n ColumnInternal,\n InternalGrid,\n RowElementInternal,\n} from '../../core/types';\nimport styles from './editing.css?inline';\nimport { defaultEditorFor } from './editors';\nimport type {\n BeforeEditCloseDetail,\n CellCommitDetail,\n ChangedRowsResetDetail,\n EditCloseDetail,\n EditingConfig,\n EditOpenDetail,\n EditorContext,\n RowCommitDetail,\n} from './types';\n\n// ============================================================================\n// Helper Functions\n// ============================================================================\n\n/**\n * Resolves the editor for a column using the priority chain:\n * 1. Column-level (`column.editor`)\n * 2. Light DOM template (`__editorTemplate` → returns 'template')\n * 3. Grid-level (`gridConfig.typeDefaults[column.type]`)\n * 4. App-level (framework adapter's `getTypeDefault`)\n * 5. Returns undefined (caller uses built-in defaultEditorFor)\n */\nfunction resolveEditor<TRow>(\n grid: InternalGrid<TRow>,\n col: ColumnInternal<TRow>,\n): ColumnEditorSpec<TRow, unknown> | 'template' | undefined {\n // 1. Column-level editor (highest priority)\n if (col.editor) return col.editor;\n\n // 2. Light DOM template\n const tplHolder = col.__editorTemplate;\n if (tplHolder) return 'template';\n\n // No type specified - no type defaults to check\n if (!col.type) return undefined;\n\n // 3. Grid-level typeDefaults (access via effectiveConfig)\n const gridTypeDefaults = (grid as any).effectiveConfig?.typeDefaults;\n if (gridTypeDefaults?.[col.type]?.editor) {\n return gridTypeDefaults[col.type].editor as ColumnEditorSpec<TRow, unknown>;\n }\n\n // 4. App-level registry (via framework adapter)\n const adapter = grid.__frameworkAdapter;\n if (adapter?.getTypeDefault) {\n const appDefault = adapter.getTypeDefault<TRow>(col.type);\n if (appDefault?.editor) {\n return appDefault.editor as ColumnEditorSpec<TRow, unknown>;\n }\n }\n\n // 5. No custom editor - caller uses built-in defaultEditorFor\n return undefined;\n}\n\n/**\n * Returns true if the given property key is safe to use on a plain object.\n */\nfunction isSafePropertyKey(key: unknown): key is string {\n if (typeof key !== 'string') return false;\n if (key === '__proto__' || key === 'constructor' || key === 'prototype') return false;\n return true;\n}\n\n/**\n * Check if a row element has any cells in editing mode.\n */\nexport function hasEditingCells(rowEl: RowElementInternal): boolean {\n return (rowEl.__editingCellCount ?? 0) > 0;\n}\n\n/**\n * Increment the editing cell count on a row element.\n */\nfunction incrementEditingCount(rowEl: RowElementInternal): void {\n const count = (rowEl.__editingCellCount ?? 0) + 1;\n rowEl.__editingCellCount = count;\n rowEl.setAttribute('data-has-editing', '');\n}\n\n/**\n * Clear all editing state from a row element.\n */\nexport function clearEditingState(rowEl: RowElementInternal): void {\n rowEl.__editingCellCount = 0;\n rowEl.removeAttribute('data-has-editing');\n}\n\n/**\n * Get the typed value from an input element based on its type, column config, and original value.\n * Preserves the type of the original value (e.g., numeric currency values stay as numbers,\n * string dates stay as strings).\n */\nfunction getInputValue(\n input: HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement,\n column?: ColumnConfig<any>,\n originalValue?: unknown,\n): unknown {\n if (input instanceof HTMLInputElement) {\n if (input.type === 'checkbox') return input.checked;\n if (input.type === 'number') return input.value === '' ? null : Number(input.value);\n if (input.type === 'date') {\n // Preserve original type: if original was a string, return string (YYYY-MM-DD format)\n if (typeof originalValue === 'string') {\n return input.value; // input.value is already in YYYY-MM-DD format\n }\n return input.valueAsDate;\n }\n // For text inputs, check if original value was a number to preserve type\n if (typeof originalValue === 'number') {\n return input.value === '' ? null : Number(input.value);\n }\n // Preserve null/undefined: if original was null/undefined and input is empty, return original\n if ((originalValue === null || originalValue === undefined) && input.value === '') {\n return originalValue;\n }\n // Preserve values with characters <input> can't represent (newlines, etc.)\n if (typeof originalValue === 'string' && input.value === originalValue.replace(/[\\n\\r]/g, '')) {\n return originalValue;\n }\n return input.value;\n }\n // For textarea/select, check column type OR original value type\n if (column?.type === 'number' && input.value !== '') {\n return Number(input.value);\n }\n // Preserve numeric type for custom column types (e.g., currency)\n if (typeof originalValue === 'number' && input.value !== '') {\n return Number(input.value);\n }\n // Preserve null/undefined: if original was null/undefined and input is empty, return original\n if ((originalValue === null || originalValue === undefined) && input.value === '') {\n return originalValue;\n }\n return input.value;\n}\n\n/**\n * No-op updateRow function for rows without IDs.\n * Extracted to a named function to satisfy eslint no-empty-function.\n */\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nfunction noopUpdateRow(_changes: unknown): void {\n // Row has no ID - cannot update\n}\n\n/**\n * Auto-wire commit/cancel lifecycle for input elements in string-returned editors.\n */\nfunction wireEditorInputs(\n editorHost: HTMLElement,\n column: ColumnConfig<unknown>,\n commit: (value: unknown) => void,\n originalValue?: unknown,\n): void {\n const input = editorHost.querySelector('input,textarea,select') as\n | HTMLInputElement\n | HTMLTextAreaElement\n | HTMLSelectElement\n | null;\n if (!input) return;\n\n input.addEventListener('blur', () => {\n commit(getInputValue(input, column, originalValue));\n });\n\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.addEventListener('change', () => commit(input.checked));\n } else if (input instanceof HTMLSelectElement) {\n input.addEventListener('change', () => commit(getInputValue(input, column, originalValue)));\n }\n}\n\n// ============================================================================\n// EditingPlugin\n// ============================================================================\n\n/**\n * Editing Plugin for tbw-grid\n *\n * Enables inline cell editing in the grid. Provides built-in editors for common data types\n * and supports custom editor functions for specialized input scenarios.\n *\n * ## Why Opt-In?\n *\n * Editing is delivered as a plugin rather than built into the core grid:\n *\n * - **Smaller bundle** — Apps that only display data don't pay for editing code\n * - **Clear intent** — Explicit plugin registration makes editing capability obvious\n * - **Runtime validation** — Using `editable: true` without the plugin throws a helpful error\n *\n * ## Installation\n *\n * ```ts\n * import { EditingPlugin } from '@toolbox-web/grid/plugins/editing';\n * ```\n *\n * ## Edit Triggers\n *\n * Configure how editing is triggered with the `editOn` option:\n *\n * | Value | Behavior |\n * |-------|----------|\n * | `'click'` | Single click enters edit mode (default) |\n * | `'dblclick'` | Double-click enters edit mode |\n *\n * ## Keyboard Shortcuts\n *\n * | Key | Action |\n * |-----|--------|\n * | `Enter` | Commit edit and move down |\n * | `Tab` | Commit edit and move right |\n * | `Escape` | Cancel edit, restore original value |\n * | `Arrow Keys` | Navigate between cells (when not editing) |\n *\n * @example Basic editing with double-click trigger\n * ```ts\n * grid.gridConfig = {\n * columns: [\n * { field: 'name', editable: true },\n * { field: 'price', type: 'number', editable: true },\n * { field: 'active', type: 'boolean', editable: true },\n * ],\n * plugins: [new EditingPlugin({ editOn: 'dblclick' })],\n * };\n *\n * grid.addEventListener('cell-commit', (e) => {\n * const { field, oldValue, newValue } = e.detail;\n * console.log(`${field}: ${oldValue} → ${newValue}`);\n * });\n * ```\n *\n * @example Custom editor function\n * ```ts\n * columns: [\n * {\n * field: 'status',\n * editable: true,\n * editor: (ctx) => {\n * const select = document.createElement('select');\n * ['pending', 'active', 'completed'].forEach(opt => {\n * const option = document.createElement('option');\n * option.value = opt;\n * option.textContent = opt;\n * option.selected = ctx.value === opt;\n * select.appendChild(option);\n * });\n * select.addEventListener('change', () => ctx.commit(select.value));\n * return select;\n * },\n * },\n * ]\n * ```\n *\n * @see {@link EditingConfig} for configuration options\n * @see {@link EditorContext} for custom editor context\n * @see [Live Demos](?path=/docs/grid-plugins-editing--docs) for interactive examples\n */\nexport class EditingPlugin<T = unknown> extends BaseGridPlugin<EditingConfig> {\n /**\n * Plugin manifest - declares owned properties for configuration validation.\n * @internal\n */\n static override readonly manifest: PluginManifest = {\n ownedProperties: [\n {\n property: 'editable',\n level: 'column',\n description: 'the \"editable\" column property',\n isUsed: (v) => v === true,\n },\n {\n property: 'editor',\n level: 'column',\n description: 'the \"editor\" column property',\n },\n {\n property: 'editorParams',\n level: 'column',\n description: 'the \"editorParams\" column property',\n },\n ],\n events: [\n {\n type: 'cell-edit-committed',\n description: 'Emitted when a cell edit is committed (for plugin-to-plugin coordination)',\n },\n ],\n queries: [\n {\n type: 'isEditing',\n description: 'Returns whether any cell is currently being edited',\n },\n ],\n };\n\n /** @internal */\n readonly name = 'editing';\n /** @internal */\n override readonly styles = styles;\n\n /** @internal */\n protected override get defaultConfig(): Partial<EditingConfig> {\n return {\n mode: 'row',\n editOn: 'click',\n };\n }\n\n /**\n * Whether the grid is in 'grid' mode (all cells always editable).\n */\n get #isGridMode(): boolean {\n return this.config.mode === 'grid';\n }\n\n // #region Editing State (fully owned by plugin)\n\n /** Currently active edit row index, or -1 if not editing */\n #activeEditRow = -1;\n\n /** Row ID of the currently active edit row (stable across _rows replacement) */\n #activeEditRowId: string | undefined;\n\n /** Reference to the row object at edit-open time. Used as fallback in\n * #exitRowEdit when no row ID is available (prevents stale-index access). */\n #activeEditRowRef: T | undefined;\n\n /** Currently active edit column index, or -1 if not editing */\n #activeEditCol = -1;\n\n /** Snapshots of row data before editing started */\n #rowEditSnapshots = new Map<number, T>();\n\n /** Set of row IDs that have been modified (ID-based for stability) */\n #changedRowIds = new Set<string>();\n\n /** Set of cells currently in edit mode: \"rowIndex:colIndex\" */\n #editingCells = new Set<string>();\n\n /**\n * Value-change callbacks for active editors.\n * Keyed by \"rowIndex:field\" → callback that pushes updated values to the editor.\n * Populated during #injectEditor, cleaned up when editors are removed.\n */\n #editorValueCallbacks = new Map<string, (newValue: unknown) => void>();\n\n /** Flag to restore focus after next render (used when exiting edit mode) */\n #pendingFocusRestore = false;\n\n /** Row index pending animation after render, or -1 if none */\n #pendingRowAnimation = -1;\n\n /**\n * Invalid cell tracking: Map<rowId, Map<field, message>>\n * Used for validation feedback without canceling edits.\n */\n #invalidCells = new Map<string, Map<string, string>>();\n\n /**\n * In grid mode, tracks whether an input field is currently focused.\n * When true: arrow keys work within input (edit mode).\n * When false: arrow keys navigate between cells (navigation mode).\n * Escape switches to navigation mode, Enter switches to edit mode.\n */\n #gridModeInputFocused = false;\n\n /**\n * In grid mode, when true, prevents inputs from auto-focusing.\n * This is set when Escape is pressed (navigation mode) and cleared\n * when Enter is pressed or user explicitly clicks an input.\n */\n #gridModeEditLocked = false;\n\n /**\n * When true, only a single cell is being edited (triggered by F2 or `beginCellEdit`).\n * Tab and Arrow keys commit and close the editor instead of navigating to adjacent cells.\n */\n #singleCellEdit = false;\n\n // #endregion\n\n // #region Lifecycle\n\n /** @internal */\n override attach(grid: GridElement): void {\n super.attach(grid);\n\n const signal = this.disconnectSignal;\n const internalGrid = grid as unknown as InternalGrid<T>;\n\n // Inject editing state and methods onto grid for backward compatibility\n internalGrid._activeEditRows = -1;\n internalGrid._rowEditSnapshots = new Map();\n\n // Inject changedRows getter\n Object.defineProperty(grid, 'changedRows', {\n get: () => this.changedRows,\n configurable: true,\n });\n\n // Inject changedRowIds getter (new ID-based API)\n Object.defineProperty(grid, 'changedRowIds', {\n get: () => this.changedRowIds,\n configurable: true,\n });\n\n // Inject resetChangedRows method\n (grid as any).resetChangedRows = (silent?: boolean) => this.resetChangedRows(silent);\n\n // Inject beginBulkEdit method (for backward compatibility)\n (grid as any).beginBulkEdit = (rowIndex: number, field?: string) => {\n if (field) {\n this.beginCellEdit(rowIndex, field);\n }\n // If no field specified, we can't start editing without a specific cell\n };\n\n // Document-level Escape to cancel editing (only in 'row' mode)\n document.addEventListener(\n 'keydown',\n (e: KeyboardEvent) => {\n // In grid mode, Escape doesn't exit edit mode\n if (this.#isGridMode) return;\n if (e.key === 'Escape' && this.#activeEditRow !== -1) {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return;\n }\n }\n this.#exitRowEdit(this.#activeEditRow, true);\n }\n },\n { capture: true, signal },\n );\n\n // Click outside to commit editing (only in 'row' mode)\n // Use queueMicrotask to allow pending change events to fire first.\n // This is important for Angular/React editors where the (change) event\n // fires after mousedown but before mouseup/click.\n document.addEventListener(\n 'mousedown',\n (e: MouseEvent) => {\n // In grid mode, clicking outside doesn't exit edit mode\n if (this.#isGridMode) return;\n if (this.#activeEditRow === -1) return;\n const rowEl = internalGrid.findRenderedRowElement?.(this.#activeEditRow);\n if (!rowEl) return;\n const path = (e.composedPath && e.composedPath()) || [];\n if (path.includes(rowEl)) return;\n\n // Check if click is inside a registered external focus container\n // (e.g., overlays, datepickers, dropdowns at <body> level).\n // Only check targets OUTSIDE the grid — clicks on other rows inside\n // the grid should still commit the active edit row.\n const target = e.target as Node | null;\n if (target && !this.gridElement.contains(target) && this.grid.containsFocus?.(target)) {\n return;\n }\n\n // Allow users to prevent edit close via callback (e.g., when click is inside an overlay)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return;\n }\n }\n\n // Delay exit to allow pending change/commit events to fire\n queueMicrotask(() => {\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, false);\n }\n });\n },\n { signal },\n );\n\n // Focus trap: when enabled, prevent focus from leaving the grid\n // while a row is being edited. If focus moves outside the grid\n // (and its registered external containers), reclaim it.\n if (this.config.focusTrap) {\n this.gridElement.addEventListener(\n 'focusout',\n (e: FocusEvent) => {\n // Only trap in row mode when actively editing\n if (this.#isGridMode) return;\n if (this.#activeEditRow === -1) return;\n\n const related = e.relatedTarget as Node | null;\n // If focus is going to an external container, that's fine\n if (related && this.grid.containsFocus?.(related)) return;\n // If focus is going to another element inside the grid, allow it\n if (related && this.gridElement.contains(related)) return;\n\n // Focus left the grid entirely — reclaim it\n queueMicrotask(() => {\n // Re-check in case editing was committed in the meantime\n if (this.#activeEditRow === -1) return;\n this.#focusCurrentCellEditor();\n });\n },\n { signal },\n );\n }\n\n // Listen for external row mutations to push updated values to active editors.\n // When field A commits and sets field B via updateRow(), field B's editor\n // (if open) must reflect the new value.\n this.gridElement.addEventListener(\n 'cell-change',\n (e: Event) => {\n const detail = (e as CustomEvent).detail as {\n rowIndex: number;\n field: string;\n newValue: unknown;\n source: string;\n };\n // Only push updates from cascade/api sources — not from the editor's own commit\n if (detail.source === 'user') return;\n const key = `${detail.rowIndex}:${detail.field}`;\n const cb = this.#editorValueCallbacks.get(key);\n if (cb) cb(detail.newValue);\n },\n { signal },\n );\n\n // In grid mode, request a full render to trigger afterCellRender hooks\n if (this.#isGridMode) {\n internalGrid._isGridEditMode = true;\n this.gridElement.classList.add('tbw-grid-mode');\n this.requestRender();\n\n // Track focus/blur on inputs to maintain navigation vs edit mode state\n this.gridElement.addEventListener(\n 'focusin',\n (e: FocusEvent) => {\n const target = e.target as HTMLElement;\n if (target.matches(FOCUSABLE_EDITOR_SELECTOR)) {\n // If edit is locked (navigation mode), blur the input immediately\n if (this.#gridModeEditLocked) {\n target.blur();\n this.gridElement.focus();\n return;\n }\n this.#gridModeInputFocused = true;\n }\n },\n { signal },\n );\n\n this.gridElement.addEventListener(\n 'focusout',\n (e: FocusEvent) => {\n const related = e.relatedTarget as HTMLElement | null;\n // Only clear if focus went outside grid (and external containers) or to a non-input element\n if (\n !related ||\n (!this.gridElement.contains(related) && !this.grid.containsFocus?.(related)) ||\n !related.matches(FOCUSABLE_EDITOR_SELECTOR)\n ) {\n this.#gridModeInputFocused = false;\n }\n },\n { signal },\n );\n\n // Handle Escape key directly on the grid element (capture phase)\n // This ensures we intercept Escape even when focus is inside an input\n this.gridElement.addEventListener(\n 'keydown',\n (e: KeyboardEvent) => {\n if (e.key === 'Escape' && this.#gridModeInputFocused) {\n // Allow users to prevent Escape handling via callback (e.g., when overlay is open).\n // In grid mode, Escape transitions from editing to navigation mode, so we check\n // onBeforeEditClose to let overlays (dropdowns, autocompletes) close first.\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the overlay handle Escape\n }\n }\n const activeEl = document.activeElement as HTMLElement;\n if (activeEl && this.gridElement.contains(activeEl)) {\n activeEl.blur();\n // Move focus to the grid container so arrow keys work\n this.gridElement.focus();\n }\n this.#gridModeInputFocused = false;\n this.#gridModeEditLocked = true; // Lock edit mode until Enter/click\n e.preventDefault();\n e.stopPropagation();\n }\n },\n { capture: true, signal },\n );\n\n // Handle click on inputs - unlock edit mode when user explicitly clicks\n this.gridElement.addEventListener(\n 'mousedown',\n (e: MouseEvent) => {\n const target = e.target as HTMLElement;\n if (target.matches(FOCUSABLE_EDITOR_SELECTOR)) {\n this.#gridModeEditLocked = false; // User clicked input - allow edit\n }\n },\n { signal },\n );\n }\n }\n\n /** @internal */\n override detach(): void {\n const internalGrid = this.gridElement as unknown as InternalGrid<T>;\n internalGrid._isGridEditMode = false;\n this.gridElement.classList.remove('tbw-grid-mode');\n this.#activeEditRow = -1;\n this.#activeEditRowId = undefined;\n this.#activeEditRowRef = undefined;\n this.#activeEditCol = -1;\n this.#rowEditSnapshots.clear();\n this.#changedRowIds.clear();\n this.#editingCells.clear();\n this.#editorValueCallbacks.clear();\n this.#gridModeInputFocused = false;\n this.#gridModeEditLocked = false;\n this.#singleCellEdit = false;\n super.detach();\n }\n\n /**\n * Handle plugin queries.\n * @internal\n */\n override handleQuery(query: PluginQuery): unknown {\n if (query.type === 'isEditing') {\n // In grid mode, we're always editing\n return this.#isGridMode || this.#activeEditRow !== -1;\n }\n return undefined;\n }\n\n // #endregion\n\n // #region Event Handlers (event distribution)\n\n /**\n * Handle cell clicks - start editing if configured for click mode.\n * Both click and dblclick events come through this handler.\n * Starts row-based editing (all editable cells in the row get editors).\n * @internal\n */\n override onCellClick(event: CellClickEvent): boolean | void {\n // In grid mode, all cells are already editable - no need to trigger row edit\n if (this.#isGridMode) return false;\n\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n\n // Check if editing is disabled\n if (editOn === false || editOn === 'manual') return false;\n\n // Check if this is click or dblclick mode\n if (editOn !== 'click' && editOn !== 'dblclick') return false;\n\n // Check if the event type matches the edit mode\n const isDoubleClick = event.originalEvent.type === 'dblclick';\n if (editOn === 'click' && isDoubleClick) return false; // In click mode, only handle single clicks\n if (editOn === 'dblclick' && !isDoubleClick) return false; // In dblclick mode, only handle double clicks\n\n const { rowIndex } = event;\n\n // Check if any column in the row is editable\n const hasEditableColumn = internalGrid._columns?.some((col) => col.editable);\n if (!hasEditableColumn) return false;\n\n // Start row-based editing (all editable cells get editors)\n event.originalEvent.stopPropagation();\n this.beginBulkEdit(rowIndex);\n return true; // Handled\n }\n\n /**\n * Handle keyboard events for edit lifecycle.\n * @internal\n */\n override onKeyDown(event: KeyboardEvent): boolean | void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n\n // Escape: cancel current edit (row mode) or exit edit mode (grid mode)\n if (event.key === 'Escape') {\n // In grid mode: blur input to enable arrow key navigation\n if (this.#isGridMode && this.#gridModeInputFocused) {\n // Allow users to prevent Escape handling via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let overlay handle Escape\n }\n }\n const activeEl = document.activeElement as HTMLElement;\n if (activeEl && this.gridElement.contains(activeEl)) {\n activeEl.blur();\n }\n this.#gridModeInputFocused = false;\n // Update focus styling\n this.requestAfterRender();\n return true;\n }\n\n // In row mode: cancel edit\n if (this.#activeEditRow !== -1 && !this.#isGridMode) {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let event reach overlay\n }\n }\n this.#exitRowEdit(this.#activeEditRow, true);\n return true;\n }\n }\n\n // Arrow keys in grid mode when not editing input: navigate cells\n if (\n this.#isGridMode &&\n !this.#gridModeInputFocused &&\n (event.key === 'ArrowUp' || event.key === 'ArrowDown' || event.key === 'ArrowLeft' || event.key === 'ArrowRight')\n ) {\n // Let the grid's default keyboard navigation handle this\n return false;\n }\n\n // Arrow Up/Down in grid mode when input is focused: let the editor handle it\n // (e.g., ArrowDown opens autocomplete/datepicker overlays, ArrowUp/Down navigates options)\n if (this.#isGridMode && this.#gridModeInputFocused && (event.key === 'ArrowUp' || event.key === 'ArrowDown')) {\n return true; // Handled: block grid navigation, let event reach editor\n }\n\n // Arrow Up/Down while editing: commit and exit edit mode, move to adjacent row (only in 'row' mode)\n if ((event.key === 'ArrowUp' || event.key === 'ArrowDown') && this.#activeEditRow !== -1 && !this.#isGridMode) {\n // Allow users to prevent row navigation via callback (e.g., when dropdown is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let event reach dropdown\n }\n }\n\n const maxRow = internalGrid._rows.length - 1;\n const currentRow = this.#activeEditRow;\n\n // Commit the current edit\n this.#exitRowEdit(currentRow, false);\n\n // Move focus to adjacent row (same column)\n if (event.key === 'ArrowDown') {\n internalGrid._focusRow = Math.min(maxRow, internalGrid._focusRow + 1);\n } else {\n internalGrid._focusRow = Math.max(0, internalGrid._focusRow - 1);\n }\n\n event.preventDefault();\n // Ensure the focused cell is scrolled into view\n ensureCellVisible(internalGrid);\n // Request render to update focus styling\n this.requestAfterRender();\n return true;\n }\n\n // Tab/Shift+Tab while editing: move to next/prev editable cell\n if (event.key === 'Tab' && (this.#activeEditRow !== -1 || this.#isGridMode)) {\n event.preventDefault();\n\n // In single-cell edit mode (F2), commit and close instead of navigating\n if (this.#singleCellEdit) {\n this.#exitRowEdit(this.#activeEditRow, false);\n return true;\n }\n\n const forward = !event.shiftKey;\n this.#handleTabNavigation(forward);\n return true;\n }\n\n // Space: toggle boolean cells (only when not in edit mode - let editors handle their own space)\n if (event.key === ' ' || event.key === 'Spacebar') {\n // If we're in row edit mode, let the event pass through to the editor (e.g., checkbox)\n if (this.#activeEditRow !== -1) {\n return false;\n }\n\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n if (focusRow >= 0 && focusCol >= 0) {\n const column = internalGrid._visibleColumns[focusCol];\n const rowData = internalGrid._rows[focusRow];\n if (column?.editable && column.type === 'boolean' && rowData) {\n const field = column.field;\n if (isSafePropertyKey(field)) {\n const currentValue = (rowData as Record<string, unknown>)[field];\n const newValue = !currentValue;\n this.#commitCellValue(focusRow, column, newValue, rowData);\n event.preventDefault();\n // Re-render to update the UI\n this.requestRender();\n return true;\n }\n }\n }\n // Space on non-boolean cell - don't block keyboard navigation\n return false;\n }\n\n // Enter (unmodified): start row edit, commit, or enter edit mode in grid mode\n if (event.key === 'Enter' && !event.shiftKey && !event.ctrlKey && !event.altKey && !event.metaKey) {\n // In grid mode when not editing: focus the current cell's input\n if (this.#isGridMode && !this.#gridModeInputFocused) {\n this.#focusCurrentCellEditor();\n return true;\n }\n\n if (this.#activeEditRow !== -1) {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n // This lets Enter select an item in a dropdown instead of committing the row\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(event);\n if (shouldClose === false) {\n return true; // Handled: block grid navigation, let event reach overlay\n }\n }\n // Already editing - let cell handlers deal with it\n return false;\n }\n\n // Start row-based editing (not just the focused cell)\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n if (editOn === false || editOn === 'manual') return false;\n\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n if (focusRow >= 0) {\n // Check if ANY column in the row is editable\n const hasEditableColumn = internalGrid._columns?.some((col) => col.editable);\n if (hasEditableColumn) {\n // Emit cell-activate event BEFORE starting edit\n // This ensures consumers always get the activation event\n const column = internalGrid._visibleColumns[focusCol];\n const row = internalGrid._rows[focusRow];\n const field = column?.field ?? '';\n const value = field && row ? (row as Record<string, unknown>)[field] : undefined;\n const cellEl = this.gridElement.querySelector(`[data-row=\"${focusRow}\"][data-col=\"${focusCol}\"]`) as\n | HTMLElement\n | undefined;\n\n const activateEvent = new CustomEvent('cell-activate', {\n cancelable: true,\n bubbles: true,\n detail: {\n rowIndex: focusRow,\n colIndex: focusCol,\n field,\n value,\n row,\n cellEl,\n trigger: 'keyboard' as const,\n originalEvent: event,\n },\n });\n this.gridElement.dispatchEvent(activateEvent);\n\n // Also emit deprecated activate-cell for backwards compatibility\n const legacyEvent = new CustomEvent('activate-cell', {\n cancelable: true,\n bubbles: true,\n detail: { row: focusRow, col: focusCol },\n });\n this.gridElement.dispatchEvent(legacyEvent);\n\n // If consumer canceled the activation, don't start editing\n if (activateEvent.defaultPrevented || legacyEvent.defaultPrevented) {\n event.preventDefault();\n return true;\n }\n\n this.beginBulkEdit(focusRow);\n return true;\n }\n }\n // No editable columns - don't block keyboard navigation\n return false;\n }\n\n // F2: begin single-cell edit on the focused cell\n if (event.key === 'F2') {\n if (this.#activeEditRow !== -1 || this.#isGridMode) return false;\n\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n if (editOn === false) return false;\n\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n if (focusRow >= 0 && focusCol >= 0) {\n const column = internalGrid._visibleColumns[focusCol];\n if (column?.editable && column.field) {\n event.preventDefault();\n this.beginCellEdit(focusRow, column.field);\n return true;\n }\n }\n return false;\n }\n\n // Don't block other keyboard events\n return false;\n }\n\n // #endregion\n\n // #region Render Hooks\n\n /**\n * Process columns to merge type-level editorParams with column-level.\n * Column-level params take precedence.\n * @internal\n */\n override processColumns(columns: ColumnConfig<T>[]): ColumnConfig<T>[] {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const typeDefaults = (internalGrid as any).effectiveConfig?.typeDefaults;\n const adapter = internalGrid.__frameworkAdapter;\n\n // If no type defaults configured anywhere, skip processing\n if (!typeDefaults && !adapter?.getTypeDefault) return columns;\n\n return columns.map((col) => {\n if (!col.type) return col;\n\n // Get type-level editorParams\n let typeEditorParams: Record<string, unknown> | undefined;\n\n // Check grid-level typeDefaults first\n if (typeDefaults?.[col.type]?.editorParams) {\n typeEditorParams = typeDefaults[col.type].editorParams;\n }\n\n // Then check app-level (adapter) typeDefaults\n if (!typeEditorParams && adapter?.getTypeDefault) {\n const appDefault = adapter.getTypeDefault<T>(col.type);\n if (appDefault?.editorParams) {\n typeEditorParams = appDefault.editorParams;\n }\n }\n\n // No type-level params to merge\n if (!typeEditorParams) return col;\n\n // Merge: type-level as base, column-level wins on conflicts\n return {\n ...col,\n editorParams: { ...typeEditorParams, ...col.editorParams },\n };\n });\n }\n\n /**\n * After render, reapply editors to cells in edit mode.\n * This handles virtualization - when a row scrolls back into view,\n * we need to re-inject the editor.\n * @internal\n */\n override afterRender(): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n\n // Restore focus after exiting edit mode\n if (this.#pendingFocusRestore) {\n this.#pendingFocusRestore = false;\n this.#restoreCellFocus(internalGrid);\n }\n\n // Animate the row after render completes (so the row element exists)\n if (this.#pendingRowAnimation !== -1) {\n const rowIndex = this.#pendingRowAnimation;\n this.#pendingRowAnimation = -1;\n internalGrid.animateRow?.(rowIndex, 'change');\n }\n\n // In 'grid' mode, editors are injected via afterCellRender hook during render\n if (this.#isGridMode) return;\n\n if (this.#editingCells.size === 0) return;\n\n // Re-inject editors for any editing cells that are visible\n for (const cellKey of this.#editingCells) {\n const [rowStr, colStr] = cellKey.split(':');\n const rowIndex = parseInt(rowStr, 10);\n const colIndex = parseInt(colStr, 10);\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n if (!rowEl) continue;\n\n const cellEl = rowEl.querySelector(`.cell[data-col=\"${colIndex}\"]`) as HTMLElement | null;\n if (!cellEl || cellEl.classList.contains('editing')) continue;\n\n // Cell is visible but not in editing mode - reinject editor\n const rowData = internalGrid._rows[rowIndex];\n const column = internalGrid._visibleColumns[colIndex];\n if (rowData && column) {\n this.#injectEditor(rowData, rowIndex, column, colIndex, cellEl, true);\n }\n }\n }\n\n /**\n * Hook called after each cell is rendered.\n * In grid mode, injects editors into editable cells during render (no DOM queries needed).\n * @internal\n */\n override afterCellRender(context: AfterCellRenderContext): void {\n // Only inject editors in grid mode\n if (!this.#isGridMode) return;\n\n const { row, rowIndex, column, colIndex, cellElement } = context;\n\n // Skip non-editable columns\n if (!column.editable) return;\n\n // Skip if already has editor\n if (cellElement.classList.contains('editing')) return;\n\n // Inject editor (don't track in editingCells - we're always editing in grid mode)\n this.#injectEditor(row as T, rowIndex, column as ColumnConfig<T>, colIndex, cellElement, true);\n }\n\n /**\n * On scroll render, reapply editors to recycled cells.\n * @internal\n */\n override onScrollRender(): void {\n this.afterRender();\n }\n\n // #endregion\n\n // #region Public API\n\n /**\n * Get all rows that have been modified.\n * Uses ID-based lookup for stability when rows are reordered.\n */\n get changedRows(): T[] {\n const rows: T[] = [];\n for (const id of this.#changedRowIds) {\n const row = this.grid.getRow(id) as T | undefined;\n if (row) rows.push(row);\n }\n return rows;\n }\n\n /**\n * Get IDs of all modified rows.\n */\n get changedRowIds(): string[] {\n return Array.from(this.#changedRowIds);\n }\n\n /**\n * Get the currently active edit row index, or -1 if not editing.\n */\n get activeEditRow(): number {\n return this.#activeEditRow;\n }\n\n /**\n * Get the currently active edit column index, or -1 if not editing.\n */\n get activeEditCol(): number {\n return this.#activeEditCol;\n }\n\n /**\n * Check if a specific row is currently being edited.\n */\n isRowEditing(rowIndex: number): boolean {\n return this.#activeEditRow === rowIndex;\n }\n\n /**\n * Check if a specific cell is currently being edited.\n */\n isCellEditing(rowIndex: number, colIndex: number): boolean {\n return this.#editingCells.has(`${rowIndex}:${colIndex}`);\n }\n\n /**\n * Check if a specific row has been modified.\n * @param rowIndex - Row index to check (will be converted to ID internally)\n */\n isRowChanged(rowIndex: number): boolean {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const row = internalGrid._rows[rowIndex];\n if (!row) return false;\n try {\n const rowId = internalGrid.getRowId?.(row);\n return rowId ? this.#changedRowIds.has(rowId) : false;\n } catch {\n return false;\n }\n }\n\n /**\n * Check if a row with the given ID has been modified.\n * @param rowId - Row ID to check\n */\n isRowChangedById(rowId: string): boolean {\n return this.#changedRowIds.has(rowId);\n }\n\n // #region Cell Validation\n\n /**\n * Mark a cell as invalid with an optional validation message.\n * Invalid cells are marked with a `data-invalid` attribute for styling.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n * @param message - Optional validation message (for tooltips or display)\n *\n * @example\n * ```typescript\n * // In cell-commit handler:\n * grid.addEventListener('cell-commit', (e) => {\n * if (e.detail.field === 'email' && !isValidEmail(e.detail.value)) {\n * e.detail.setInvalid('Invalid email format');\n * }\n * });\n *\n * // Or programmatically:\n * editingPlugin.setInvalid('row-123', 'email', 'Invalid email format');\n * ```\n */\n setInvalid(rowId: string, field: string, message = ''): void {\n let rowInvalids = this.#invalidCells.get(rowId);\n if (!rowInvalids) {\n rowInvalids = new Map();\n this.#invalidCells.set(rowId, rowInvalids);\n }\n rowInvalids.set(field, message);\n this.#syncInvalidCellAttribute(rowId, field, true);\n }\n\n /**\n * Clear the invalid state for a specific cell.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n */\n clearInvalid(rowId: string, field: string): void {\n const rowInvalids = this.#invalidCells.get(rowId);\n if (rowInvalids) {\n rowInvalids.delete(field);\n if (rowInvalids.size === 0) {\n this.#invalidCells.delete(rowId);\n }\n }\n this.#syncInvalidCellAttribute(rowId, field, false);\n }\n\n /**\n * Clear all invalid cells for a specific row.\n *\n * @param rowId - The row ID (from getRowId)\n */\n clearRowInvalid(rowId: string): void {\n const rowInvalids = this.#invalidCells.get(rowId);\n if (rowInvalids) {\n const fields = Array.from(rowInvalids.keys());\n this.#invalidCells.delete(rowId);\n fields.forEach((field) => this.#syncInvalidCellAttribute(rowId, field, false));\n }\n }\n\n /**\n * Clear all invalid cell states across all rows.\n */\n clearAllInvalid(): void {\n const entries = Array.from(this.#invalidCells.entries());\n this.#invalidCells.clear();\n entries.forEach(([rowId, fields]) => {\n fields.forEach((_, field) => this.#syncInvalidCellAttribute(rowId, field, false));\n });\n }\n\n /**\n * Check if a specific cell is marked as invalid.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n * @returns True if the cell is marked as invalid\n */\n isCellInvalid(rowId: string, field: string): boolean {\n return this.#invalidCells.get(rowId)?.has(field) ?? false;\n }\n\n /**\n * Get the validation message for an invalid cell.\n *\n * @param rowId - The row ID (from getRowId)\n * @param field - The field name\n * @returns The validation message, or undefined if cell is valid\n */\n getInvalidMessage(rowId: string, field: string): string | undefined {\n return this.#invalidCells.get(rowId)?.get(field);\n }\n\n /**\n * Check if a row has any invalid cells.\n *\n * @param rowId - The row ID (from getRowId)\n * @returns True if the row has at least one invalid cell\n */\n hasInvalidCells(rowId: string): boolean {\n const rowInvalids = this.#invalidCells.get(rowId);\n return rowInvalids ? rowInvalids.size > 0 : false;\n }\n\n /**\n * Get all invalid fields for a row.\n *\n * @param rowId - The row ID (from getRowId)\n * @returns Map of field names to validation messages\n */\n getInvalidFields(rowId: string): Map<string, string> {\n return new Map(this.#invalidCells.get(rowId) ?? []);\n }\n\n /**\n * Sync the data-invalid attribute on a cell element.\n */\n #syncInvalidCellAttribute(rowId: string, field: string, invalid: boolean): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const colIndex = internalGrid._visibleColumns?.findIndex((c) => c.field === field);\n if (colIndex === -1 || colIndex === undefined) return;\n\n // Find the row element by rowId\n const rows = internalGrid._rows;\n const rowIndex = rows?.findIndex((r) => {\n try {\n return internalGrid.getRowId?.(r) === rowId;\n } catch {\n return false;\n }\n });\n if (rowIndex === -1 || rowIndex === undefined) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${colIndex}\"]`) as HTMLElement | null;\n if (!cellEl) return;\n\n if (invalid) {\n cellEl.setAttribute('data-invalid', 'true');\n const message = this.#invalidCells.get(rowId)?.get(field);\n if (message) {\n cellEl.setAttribute('title', message);\n }\n } else {\n cellEl.removeAttribute('data-invalid');\n cellEl.removeAttribute('title');\n }\n }\n\n // #endregion\n\n /**\n * Reset all change tracking.\n * @param silent - If true, suppresses the `changed-rows-reset` event\n * @fires changed-rows-reset - Emitted when tracking is reset (unless silent)\n */\n resetChangedRows(silent?: boolean): void {\n const rows = this.changedRows;\n const ids = this.changedRowIds;\n this.#changedRowIds.clear();\n this.#syncGridEditState();\n\n if (!silent) {\n this.emit<ChangedRowsResetDetail<T>>('changed-rows-reset', { rows, ids });\n }\n\n // Clear visual indicators\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n internalGrid._rowPool?.forEach((r) => r.classList.remove('changed'));\n }\n\n /**\n * Programmatically begin editing a cell.\n * @param rowIndex - Index of the row to edit\n * @param field - Field name of the column to edit\n * @fires cell-commit - Emitted when the cell value is committed (on blur or Enter)\n */\n beginCellEdit(rowIndex: number, field: string): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const colIndex = internalGrid._visibleColumns.findIndex((c) => c.field === field);\n if (colIndex === -1) return;\n\n const column = internalGrid._visibleColumns[colIndex];\n if (!column?.editable) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${colIndex}\"]`) as HTMLElement | null;\n if (!cellEl) return;\n\n this.#singleCellEdit = true;\n this.#beginCellEdit(rowIndex, colIndex, cellEl);\n }\n\n /**\n * Programmatically begin editing all editable cells in a row.\n * @param rowIndex - Index of the row to edit\n * @fires cell-commit - Emitted for each cell value that is committed\n * @fires row-commit - Emitted when focus leaves the row\n */\n beginBulkEdit(rowIndex: number): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const editOn = this.config.editOn ?? internalGrid.effectiveConfig?.editOn;\n if (editOn === false) return;\n\n const hasEditableColumn = internalGrid._columns?.some((col) => col.editable);\n if (!hasEditableColumn) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n if (!rowEl) return;\n\n // Bulk edit clears single-cell mode\n this.#singleCellEdit = false;\n\n // Start row edit\n const rowData = internalGrid._rows[rowIndex];\n this.#startRowEdit(rowIndex, rowData);\n\n // Enter edit mode on all editable cells\n Array.from(rowEl.children).forEach((cell, i) => {\n const col = internalGrid._visibleColumns[i];\n if (col?.editable) {\n const cellEl = cell as HTMLElement;\n if (!cellEl.classList.contains('editing')) {\n this.#injectEditor(rowData, rowIndex, col, i, cellEl, true);\n }\n }\n });\n\n // Focus the first editable cell\n setTimeout(() => {\n let targetCell = rowEl.querySelector(`.cell[data-col=\"${internalGrid._focusCol}\"]`);\n if (!targetCell?.classList.contains('editing')) {\n targetCell = rowEl.querySelector('.cell.editing');\n }\n if (targetCell?.classList.contains('editing')) {\n const editor = (targetCell as HTMLElement).querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n try {\n editor?.focus({ preventScroll: true });\n } catch {\n /* empty */\n }\n }\n }, 0);\n }\n\n /**\n * Commit the currently active row edit.\n * @fires row-commit - Emitted after the row edit is committed\n */\n commitActiveRowEdit(): void {\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, false);\n }\n }\n\n /**\n * Cancel the currently active row edit.\n */\n cancelActiveRowEdit(): void {\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, true);\n }\n }\n\n // #endregion\n\n // #region Internal Methods\n\n /**\n * Begin editing a single cell.\n */\n #beginCellEdit(rowIndex: number, colIndex: number, cellEl: HTMLElement): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const rowData = internalGrid._rows[rowIndex];\n const column = internalGrid._visibleColumns[colIndex];\n\n if (!rowData || !column?.editable) return;\n if (cellEl.classList.contains('editing')) return;\n\n // Start row edit if not already\n if (this.#activeEditRow !== rowIndex) {\n this.#startRowEdit(rowIndex, rowData);\n }\n\n this.#activeEditCol = colIndex;\n this.#injectEditor(rowData, rowIndex, column, colIndex, cellEl, false);\n }\n\n /**\n * Focus the editor input in the currently focused cell (grid mode only).\n * Used when pressing Enter to enter edit mode from navigation mode.\n */\n #focusCurrentCellEditor(): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const focusRow = internalGrid._focusRow;\n const focusCol = internalGrid._focusCol;\n\n if (focusRow < 0 || focusCol < 0) return;\n\n const rowEl = internalGrid.findRenderedRowElement?.(focusRow);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${focusCol}\"]`) as HTMLElement | null;\n\n if (cellEl?.classList.contains('editing')) {\n const editor = cellEl.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n if (editor) {\n this.#gridModeEditLocked = false; // Unlock edit mode - user pressed Enter\n editor.focus();\n this.#gridModeInputFocused = true;\n // Select all text in text inputs for quick replacement\n if (editor instanceof HTMLInputElement && (editor.type === 'text' || editor.type === 'number')) {\n editor.select();\n }\n }\n }\n }\n\n /**\n * Handle Tab/Shift+Tab navigation while editing.\n * Moves to next/previous editable cell, staying in edit mode.\n * Wraps to next/previous row when reaching row boundaries.\n */\n #handleTabNavigation(forward: boolean): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const rows = internalGrid._rows;\n // In grid mode, use focusRow since there's no active edit row\n const currentRow = this.#isGridMode ? internalGrid._focusRow : this.#activeEditRow;\n\n // Get editable column indices\n const editableCols = internalGrid._visibleColumns.map((c, i) => (c.editable ? i : -1)).filter((i) => i >= 0);\n if (editableCols.length === 0) return;\n\n const currentIdx = editableCols.indexOf(internalGrid._focusCol);\n const nextIdx = currentIdx + (forward ? 1 : -1);\n\n // Can move within same row?\n if (nextIdx >= 0 && nextIdx < editableCols.length) {\n internalGrid._focusCol = editableCols[nextIdx];\n const rowEl = internalGrid.findRenderedRowElement?.(currentRow);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${editableCols[nextIdx]}\"]`) as HTMLElement | null;\n if (cellEl?.classList.contains('editing')) {\n const editor = cellEl.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n editor?.focus({ preventScroll: true });\n }\n ensureCellVisible(internalGrid, { forceHorizontalScroll: true });\n return;\n }\n\n // Can move to adjacent row?\n const nextRow = currentRow + (forward ? 1 : -1);\n if (nextRow >= 0 && nextRow < rows.length) {\n // In grid mode, just move focus (all rows are always editable)\n if (this.#isGridMode) {\n internalGrid._focusRow = nextRow;\n internalGrid._focusCol = forward ? editableCols[0] : editableCols[editableCols.length - 1];\n ensureCellVisible(internalGrid, { forceHorizontalScroll: true });\n // Focus the editor in the new cell after render\n this.requestAfterRender();\n setTimeout(() => {\n const rowEl = internalGrid.findRenderedRowElement?.(nextRow);\n const cellEl = rowEl?.querySelector(`.cell[data-col=\"${internalGrid._focusCol}\"]`) as HTMLElement | null;\n if (cellEl?.classList.contains('editing')) {\n const editor = cellEl.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n editor?.focus({ preventScroll: true });\n }\n }, 0);\n } else {\n // In row mode, commit current row and enter next row\n this.#exitRowEdit(currentRow, false);\n internalGrid._focusRow = nextRow;\n internalGrid._focusCol = forward ? editableCols[0] : editableCols[editableCols.length - 1];\n this.beginBulkEdit(nextRow);\n ensureCellVisible(internalGrid, { forceHorizontalScroll: true });\n }\n }\n // else: at boundary - stay put\n }\n\n /**\n * Sync the internal grid state with the plugin's editing state.\n */\n #syncGridEditState(): void {\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n internalGrid._activeEditRows = this.#activeEditRow;\n internalGrid._rowEditSnapshots = this.#rowEditSnapshots;\n }\n\n /**\n * Snapshot original row data and mark as editing.\n */\n #startRowEdit(rowIndex: number, rowData: T): void {\n if (this.#activeEditRow !== rowIndex) {\n // Commit the previous row before starting a new one\n if (this.#activeEditRow !== -1) {\n this.#exitRowEdit(this.#activeEditRow, false);\n }\n this.#rowEditSnapshots.set(rowIndex, { ...rowData });\n this.#activeEditRow = rowIndex;\n this.#activeEditRowRef = rowData;\n\n // Store stable row ID for resilience against _rows replacement during editing\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n try {\n this.#activeEditRowId = internalGrid.getRowId?.(rowData) ?? undefined;\n } catch {\n this.#activeEditRowId = undefined;\n }\n\n this.#syncGridEditState();\n\n // Emit edit-open event (row mode only)\n if (!this.#isGridMode) {\n this.emit<EditOpenDetail<T>>('edit-open', {\n rowIndex,\n rowId: this.#activeEditRowId ?? '',\n row: rowData,\n });\n }\n }\n }\n\n /**\n * Exit editing for a row.\n */\n #exitRowEdit(rowIndex: number, revert: boolean): void {\n if (this.#activeEditRow !== rowIndex) return;\n\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const snapshot = this.#rowEditSnapshots.get(rowIndex);\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n\n // Resolve the row being edited using the stored row ID.\n // The _rows array may have been replaced (e.g. Angular pushing new rows\n // via directive effect) since editing started, so _rows[rowIndex] could\n // point to a completely different row. The ID map is always up-to-date.\n // Without an ID we fall back to the stored row reference from edit-open\n // (#activeEditRowRef) — safer than _rows[rowIndex] which may be stale.\n let rowId = this.#activeEditRowId;\n const entry = rowId ? internalGrid._getRowEntry(rowId) : undefined;\n const current = entry?.row ?? this.#activeEditRowRef ?? internalGrid._rows[rowIndex];\n\n if (!rowId && current) {\n try {\n rowId = internalGrid.getRowId?.(current);\n } catch {\n // Row has no ID - skip ID-based tracking\n }\n }\n\n // Collect and commit values from active editors before re-rendering\n if (!revert && rowEl && current) {\n const editingCells = rowEl.querySelectorAll('.cell.editing');\n editingCells.forEach((cell) => {\n const colIndex = Number((cell as HTMLElement).getAttribute('data-col'));\n if (isNaN(colIndex)) return;\n const col = internalGrid._visibleColumns[colIndex];\n if (!col) return;\n\n // Skip cells with externally-managed editors (framework adapters like Angular/React/Vue).\n // These editors handle their own commits via the commit() callback - we should NOT\n // try to read values from their DOM inputs (which may contain formatted display values).\n if ((cell as HTMLElement).hasAttribute('data-editor-managed')) {\n return;\n }\n\n const input = cell.querySelector('input,textarea,select') as\n | HTMLInputElement\n | HTMLTextAreaElement\n | HTMLSelectElement\n | null;\n if (input) {\n const field = col.field as keyof T;\n const originalValue = current[field];\n const val = getInputValue(input, col, originalValue);\n if (originalValue !== val) {\n this.#commitCellValue(rowIndex, col, val, current);\n }\n }\n });\n }\n\n // Flush managed editors (framework adapters) before clearing state.\n // At this point the commit() callback is still active, so editors can\n // synchronously commit their pending values in response to this event.\n if (!revert && !this.#isGridMode && current) {\n this.emit<BeforeEditCloseDetail<T>>('before-edit-close', {\n rowIndex,\n rowId: rowId ?? '',\n row: current,\n });\n }\n\n // Revert if requested\n if (revert && snapshot && current) {\n Object.keys(snapshot as object).forEach((k) => {\n (current as Record<string, unknown>)[k] = (snapshot as Record<string, unknown>)[k];\n });\n if (rowId) {\n this.#changedRowIds.delete(rowId);\n this.clearRowInvalid(rowId);\n }\n } else if (!revert && current) {\n // Compare snapshot vs current to detect if changes were made during THIS edit session\n const changedThisSession = this.#hasRowChanged(snapshot, current);\n\n // Check if this row has any cumulative changes (via ID tracking)\n // Fall back to session-based detection when no row ID is available\n const changed = rowId ? this.#changedRowIds.has(rowId) : changedThisSession;\n\n // Emit cancelable row-commit event\n const cancelled = this.emitCancelable<RowCommitDetail<T>>('row-commit', {\n rowIndex,\n rowId: rowId ?? '',\n row: current,\n oldValue: snapshot,\n newValue: current,\n changed,\n changedRows: this.changedRows,\n changedRowIds: this.changedRowIds,\n });\n\n // If consumer called preventDefault(), revert the row\n if (cancelled && snapshot) {\n Object.keys(snapshot as object).forEach((k) => {\n (current as Record<string, unknown>)[k] = (snapshot as Record<string, unknown>)[k];\n });\n if (rowId) {\n this.#changedRowIds.delete(rowId);\n this.clearRowInvalid(rowId);\n }\n } else if (!cancelled && changedThisSession && this.isAnimationEnabled) {\n // Animate the row only if changes were made during this edit session\n // (deferred to afterRender so the row element exists after re-render)\n this.#pendingRowAnimation = rowIndex;\n }\n }\n\n // Clear editing state\n this.#rowEditSnapshots.delete(rowIndex);\n this.#activeEditRow = -1;\n this.#activeEditRowId = undefined;\n this.#activeEditRowRef = undefined;\n this.#activeEditCol = -1;\n this.#singleCellEdit = false;\n this.#syncGridEditState();\n\n // Remove all editing cells for this row.\n // Note: these keys use the rowIndex captured at edit-open time. Even if _rows\n // was replaced and the row moved to a different index, the keys still match\n // what was inserted during this edit session (same captured rowIndex).\n for (const cellKey of this.#editingCells) {\n if (cellKey.startsWith(`${rowIndex}:`)) {\n this.#editingCells.delete(cellKey);\n }\n }\n // Remove value-change callbacks for this row (same captured-index rationale)\n for (const callbackKey of this.#editorValueCallbacks.keys()) {\n if (callbackKey.startsWith(`${rowIndex}:`)) {\n this.#editorValueCallbacks.delete(callbackKey);\n }\n }\n\n // Mark that focus should be restored after the upcoming render completes.\n // This must be set BEFORE refreshVirtualWindow because it calls afterRender()\n // synchronously, which reads this flag.\n this.#pendingFocusRestore = true;\n\n // Re-render the row to remove editors\n if (rowEl) {\n // Remove editing class and re-render cells\n rowEl.querySelectorAll('.cell.editing').forEach((cell) => {\n cell.classList.remove('editing');\n clearEditingState(cell.parentElement as RowElementInternal);\n });\n\n // Refresh the virtual window to restore cell content WITHOUT rebuilding\n // the row model. requestRender() would trigger processRows (ROWS phase)\n // which re-sorts — causing the edited row to jump to a new position and\n // disappear from view. refreshVirtualWindow re-renders visible cells from\n // the current _rows order, keeping the row in place until the user\n // explicitly sorts again or new data arrives.\n internalGrid.refreshVirtualWindow(true);\n } else {\n // Row not visible - restore focus immediately (no render will happen)\n this.#restoreCellFocus(internalGrid);\n this.#pendingFocusRestore = false;\n }\n\n // Emit edit-close event (row mode only, fires for both commit and revert)\n if (!this.#isGridMode && current) {\n this.emit<EditCloseDetail<T>>('edit-close', {\n rowIndex,\n rowId: rowId ?? '',\n row: current,\n reverted: revert,\n });\n }\n }\n\n /**\n * Commit a single cell value change.\n * Uses ID-based change tracking for stability when rows are reordered.\n */\n #commitCellValue(rowIndex: number, column: ColumnConfig<T>, newValue: unknown, rowData: T): void {\n const field = column.field;\n if (!isSafePropertyKey(field)) return;\n const oldValue = (rowData as Record<string, unknown>)[field];\n if (oldValue === newValue) return;\n\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n\n // Get row ID for change tracking (may not exist if getRowId not configured)\n let rowId: string | undefined;\n try {\n rowId = this.grid.getRowId(rowData);\n } catch {\n // Row has no ID - will still work but won't be tracked in changedRowIds\n }\n\n const firstTime = rowId ? !this.#changedRowIds.has(rowId) : true;\n\n // Create updateRow helper for cascade updates (noop if row has no ID)\n const updateRow: (changes: Partial<T>) => void = rowId\n ? (changes) => this.grid.updateRow(rowId!, changes as Record<string, unknown>, 'cascade')\n : noopUpdateRow;\n\n // Track whether setInvalid was called during event handling\n let invalidWasSet = false;\n\n // Create setInvalid callback for validation (noop if row has no ID)\n const setInvalid = rowId\n ? (message?: string) => {\n invalidWasSet = true;\n this.setInvalid(rowId!, field, message ?? '');\n }\n : () => {}; // eslint-disable-line @typescript-eslint/no-empty-function\n\n // Emit cancelable event BEFORE applying the value\n const cancelled = this.emitCancelable<CellCommitDetail<T>>('cell-commit', {\n row: rowData,\n rowId: rowId ?? '',\n field,\n oldValue,\n value: newValue,\n rowIndex,\n changedRows: this.changedRows,\n changedRowIds: this.changedRowIds,\n firstTimeForRow: firstTime,\n updateRow,\n setInvalid,\n });\n\n // If consumer called preventDefault(), abort the commit\n if (cancelled) return;\n\n // Clear any previous invalid state for this cell ONLY if setInvalid wasn't called\n // (if setInvalid was called, the handler wants it to remain invalid)\n if (rowId && !invalidWasSet && this.isCellInvalid(rowId, field)) {\n this.clearInvalid(rowId, field);\n }\n\n // Apply the value and mark row as changed\n (rowData as Record<string, unknown>)[field] = newValue;\n if (rowId) {\n this.#changedRowIds.add(rowId);\n }\n this.#syncGridEditState();\n\n // Notify other plugins (e.g., UndoRedoPlugin) about the committed edit\n this.emitPluginEvent('cell-edit-committed', {\n rowIndex,\n field,\n oldValue,\n newValue,\n });\n\n // Mark the row visually as changed (animation happens when row edit closes)\n const rowEl = internalGrid.findRenderedRowElement?.(rowIndex);\n if (rowEl) {\n rowEl.classList.add('changed');\n }\n }\n\n /**\n * Inject an editor into a cell.\n */\n #injectEditor(\n rowData: T,\n rowIndex: number,\n column: ColumnConfig<T>,\n colIndex: number,\n cell: HTMLElement,\n skipFocus: boolean,\n ): void {\n if (!column.editable) return;\n if (cell.classList.contains('editing')) return;\n\n // Get row ID for updateRow helper (may not exist)\n let rowId: string | undefined;\n try {\n rowId = this.grid.getRowId(rowData);\n } catch {\n // Row has no ID\n }\n\n // Create updateRow helper for cascade updates (noop if row has no ID)\n const updateRow: (changes: Partial<T>) => void = rowId\n ? (changes) => this.grid.updateRow(rowId!, changes as Record<string, unknown>, 'cascade')\n : noopUpdateRow;\n\n const originalValue = isSafePropertyKey(column.field)\n ? (rowData as Record<string, unknown>)[column.field]\n : undefined;\n\n cell.classList.add('editing');\n this.#editingCells.add(`${rowIndex}:${colIndex}`);\n\n const rowEl = cell.parentElement as RowElementInternal | null;\n if (rowEl) incrementEditingCount(rowEl);\n\n let editFinalized = false;\n const commit = (newValue: unknown) => {\n // In grid mode, always allow commits (we're always editing)\n // In row mode, only allow commits if we're in an active edit session\n if (editFinalized || (!this.#isGridMode && this.#activeEditRow === -1)) return;\n // Resolve row and index fresh at commit time.\n // With a row ID we use _getRowEntry for O(1) lookup — this is resilient\n // against _rows being replaced (e.g. Angular directive effect).\n // Without a row ID we fall back to the captured rowData reference.\n // Using _rows[rowIndex] without an ID is unsafe: the index may be stale\n // after _rows replacement, which would commit to the WRONG row.\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const entry = rowId ? internalGrid._getRowEntry(rowId) : undefined;\n const currentRowData = (entry?.row ?? rowData) as T;\n const currentIndex = entry?.index ?? rowIndex;\n this.#commitCellValue(currentIndex, column, newValue, currentRowData);\n };\n const cancel = () => {\n editFinalized = true;\n if (isSafePropertyKey(column.field)) {\n // Same ID-first / captured-rowData fallback as commit — see comment above.\n const internalGrid = this.grid as unknown as InternalGrid<T>;\n const entry = rowId ? internalGrid._getRowEntry(rowId) : undefined;\n const currentRowData = (entry?.row ?? rowData) as T;\n (currentRowData as Record<string, unknown>)[column.field] = originalValue;\n }\n };\n\n const editorHost = document.createElement('div');\n editorHost.className = 'tbw-editor-host';\n cell.innerHTML = '';\n cell.appendChild(editorHost);\n\n // Keydown handler for Enter/Escape\n editorHost.addEventListener('keydown', (e: KeyboardEvent) => {\n if (e.key === 'Enter') {\n // In grid mode, Enter just commits without exiting\n if (this.#isGridMode) {\n e.stopPropagation();\n e.preventDefault();\n // Get current value and commit\n const input = editorHost.querySelector('input,textarea,select') as\n | HTMLInputElement\n | HTMLTextAreaElement\n | HTMLSelectElement\n | null;\n if (input) {\n commit(getInputValue(input, column as ColumnConfig<unknown>, originalValue));\n }\n return;\n }\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n editFinalized = true;\n this.#exitRowEdit(rowIndex, false);\n }\n if (e.key === 'Escape') {\n // In grid mode, Escape doesn't exit edit mode\n if (this.#isGridMode) {\n e.stopPropagation();\n e.preventDefault();\n return;\n }\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n cancel();\n this.#exitRowEdit(rowIndex, true);\n }\n });\n\n const colInternal = column as ColumnInternal<T>;\n const tplHolder = colInternal.__editorTemplate;\n // Resolve editor using priority chain: column → template → typeDefaults → adapter → built-in\n const editorSpec = resolveEditor(this.grid as unknown as InternalGrid<T>, colInternal) ?? defaultEditorFor(column);\n const value = originalValue;\n\n // Value-change callback registration.\n // Editors call onValueChange(cb) to receive pushes when the underlying row\n // is mutated externally (e.g., via updateRow from another cell's commit).\n // Multiple callbacks can be registered (user + auto-wire).\n const callbackKey = `${rowIndex}:${column.field}`;\n const callbacks: Array<(newValue: unknown) => void> = [];\n this.#editorValueCallbacks.set(callbackKey, (newVal) => {\n for (const cb of callbacks) cb(newVal);\n });\n const onValueChange = (cb: (newValue: unknown) => void) => {\n callbacks.push(cb);\n };\n\n if (editorSpec === 'template' && tplHolder) {\n this.#renderTemplateEditor(editorHost, colInternal, rowData, originalValue, commit, cancel, skipFocus, rowIndex);\n // Auto-update built-in template editors when value changes externally\n onValueChange((newVal) => {\n const input = editorHost.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(\n 'input,textarea,select',\n );\n if (input) {\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.checked = !!newVal;\n } else {\n input.value = String(newVal ?? '');\n }\n }\n });\n } else if (typeof editorSpec === 'string') {\n const el = document.createElement(editorSpec) as HTMLElement & { value?: unknown };\n el.value = value;\n el.addEventListener('change', () => commit(el.value));\n // Auto-update custom element editors when value changes externally\n onValueChange((newVal) => {\n el.value = newVal;\n });\n editorHost.appendChild(el);\n if (!skipFocus) {\n queueMicrotask(() => {\n const focusable = editorHost.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n focusable?.focus({ preventScroll: true });\n });\n }\n } else if (typeof editorSpec === 'function') {\n const ctx: EditorContext<T> = {\n row: rowData,\n rowId: rowId ?? '',\n value,\n field: column.field,\n column,\n commit,\n cancel,\n updateRow,\n onValueChange,\n };\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const produced = (editorSpec as any)(ctx);\n if (typeof produced === 'string') {\n editorHost.innerHTML = produced;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n wireEditorInputs(editorHost, column as any, commit, originalValue);\n // Auto-update wired inputs when value changes externally\n onValueChange((newVal) => {\n const input = editorHost.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(\n 'input,textarea,select',\n );\n if (input) {\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.checked = !!newVal;\n } else {\n input.value = String(newVal ?? '');\n }\n }\n });\n } else if (produced instanceof Node) {\n editorHost.appendChild(produced);\n const isSimpleInput =\n produced instanceof HTMLInputElement ||\n produced instanceof HTMLSelectElement ||\n produced instanceof HTMLTextAreaElement;\n if (!isSimpleInput) {\n cell.setAttribute('data-editor-managed', '');\n } else {\n // Auto-update simple inputs returned by factory functions\n onValueChange((newVal) => {\n if (produced instanceof HTMLInputElement && produced.type === 'checkbox') {\n produced.checked = !!newVal;\n } else {\n (produced as HTMLInputElement).value = String(newVal ?? '');\n }\n });\n }\n }\n if (!skipFocus) {\n queueMicrotask(() => {\n const focusable = editorHost.querySelector(FOCUSABLE_EDITOR_SELECTOR) as HTMLElement | null;\n focusable?.focus({ preventScroll: true });\n });\n }\n } else if (editorSpec && typeof editorSpec === 'object') {\n const placeholder = document.createElement('div');\n placeholder.setAttribute('data-external-editor', '');\n placeholder.setAttribute('data-field', column.field);\n editorHost.appendChild(placeholder);\n cell.setAttribute('data-editor-managed', '');\n const context: EditorContext<T> = {\n row: rowData,\n rowId: rowId ?? '',\n value,\n field: column.field,\n column,\n commit,\n cancel,\n updateRow,\n onValueChange,\n };\n if (editorSpec.mount) {\n try {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n editorSpec.mount({ placeholder, context: context as any, spec: editorSpec });\n } catch (e) {\n console.warn(`[tbw-grid] External editor mount error for column '${column.field}':`, e);\n }\n } else {\n (this.grid as unknown as HTMLElement).dispatchEvent(\n new CustomEvent('mount-external-editor', { detail: { placeholder, spec: editorSpec, context } }),\n );\n }\n }\n }\n\n /**\n * Render a template-based editor.\n */\n #renderTemplateEditor(\n editorHost: HTMLElement,\n column: ColumnInternal<T>,\n rowData: T,\n originalValue: unknown,\n commit: (value: unknown) => void,\n cancel: () => void,\n skipFocus: boolean,\n rowIndex: number,\n ): void {\n const tplHolder = column.__editorTemplate;\n if (!tplHolder) return;\n\n const clone = tplHolder.cloneNode(true) as HTMLElement;\n const compiledEditor = column.__compiledEditor;\n\n if (compiledEditor) {\n clone.innerHTML = compiledEditor({\n row: rowData,\n value: originalValue,\n field: column.field,\n column,\n commit,\n cancel,\n });\n } else {\n clone.querySelectorAll<HTMLElement>('*').forEach((node) => {\n if (node.childNodes.length === 1 && node.firstChild?.nodeType === Node.TEXT_NODE) {\n node.textContent =\n node.textContent\n ?.replace(/{{\\s*value\\s*}}/g, originalValue == null ? '' : String(originalValue))\n .replace(/{{\\s*row\\.([a-zA-Z0-9_]+)\\s*}}/g, (_m, g: string) => {\n if (!isSafePropertyKey(g)) return '';\n const v = (rowData as Record<string, unknown>)[g];\n return v == null ? '' : String(v);\n }) || '';\n }\n });\n }\n\n const input = clone.querySelector<HTMLInputElement | HTMLTextAreaElement | HTMLSelectElement>(\n 'input,textarea,select',\n );\n if (input) {\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.checked = !!originalValue;\n } else {\n input.value = String(originalValue ?? '');\n }\n\n let editFinalized = false;\n input.addEventListener('blur', () => {\n if (editFinalized) return;\n commit(getInputValue(input, column, originalValue));\n });\n input.addEventListener('keydown', (evt) => {\n const e = evt as KeyboardEvent;\n if (e.key === 'Enter') {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n editFinalized = true;\n commit(getInputValue(input, column, originalValue));\n this.#exitRowEdit(rowIndex, false);\n }\n if (e.key === 'Escape') {\n // Allow users to prevent edit close via callback (e.g., when overlay is open)\n if (this.config.onBeforeEditClose) {\n const shouldClose = this.config.onBeforeEditClose(e);\n if (shouldClose === false) {\n return; // Let the event propagate to overlay\n }\n }\n e.stopPropagation();\n e.preventDefault();\n cancel();\n this.#exitRowEdit(rowIndex, true);\n }\n });\n if (input instanceof HTMLInputElement && input.type === 'checkbox') {\n input.addEventListener('change', () => commit(input.checked));\n }\n if (!skipFocus) {\n setTimeout(() => input.focus({ preventScroll: true }), 0);\n }\n }\n editorHost.appendChild(clone);\n }\n\n /**\n * Compare snapshot vs current row to detect if any values changed during this edit session.\n * Uses shallow comparison of all properties.\n */\n #hasRowChanged(snapshot: T | undefined, current: T): boolean {\n if (!snapshot) return false;\n\n const snapshotObj = snapshot as Record<string, unknown>;\n const currentObj = current as Record<string, unknown>;\n\n // Check all keys in both objects\n const allKeys = new Set([...Object.keys(snapshotObj), ...Object.keys(currentObj)]);\n for (const key of allKeys) {\n if (snapshotObj[key] !== currentObj[key]) {\n return true;\n }\n }\n return false;\n }\n\n /**\n * Restore focus to cell after exiting edit mode.\n */\n #restoreCellFocus(internalGrid: InternalGrid<T>): void {\n queueMicrotask(() => {\n try {\n const rowIdx = internalGrid._focusRow;\n const colIdx = internalGrid._focusCol;\n const rowEl = internalGrid.findRenderedRowElement?.(rowIdx);\n if (rowEl) {\n Array.from(internalGrid._bodyEl.querySelectorAll('.cell-focus')).forEach((el) =>\n el.classList.remove('cell-focus'),\n );\n const cell = rowEl.querySelector(`.cell[data-row=\"${rowIdx}\"][data-col=\"${colIdx}\"]`) as HTMLElement | null;\n if (cell) {\n cell.classList.add('cell-focus');\n cell.setAttribute('aria-selected', 'true');\n if (!cell.hasAttribute('tabindex')) cell.setAttribute('tabindex', '-1');\n cell.focus({ preventScroll: true });\n }\n }\n } catch {\n /* empty */\n }\n });\n }\n\n // #endregion\n}\n"],"names":["defaultEditorFor","column","type","ctx","params","editorParams","input","document","createElement","value","String","min","max","step","placeholder","commit","Number","addEventListener","e","key","cancel","createNumberEditor","checked","Date","valueAsDate","split","createDateEditor","select","multi","multiple","includeEmpty","emptyOpt","textContent","emptyLabel","appendChild","options","raw","resolveOptions","forEach","opt","o","label","Array","isArray","includes","selected","values","from","selectedOptions","map","createSelectEditor","maxLength","pattern","inputVal","replace","createTextEditor","isSafePropertyKey","getInputValue","originalValue","HTMLInputElement","noopUpdateRow","_changes","EditingPlugin","BaseGridPlugin","static","ownedProperties","property","level","description","isUsed","v","events","queries","name","styles","defaultConfig","mode","editOn","isGridMode","this","config","activeEditRow","activeEditRowId","activeEditRowRef","activeEditCol","rowEditSnapshots","Map","changedRowIds","Set","editingCells","editorValueCallbacks","pendingFocusRestore","pendingRowAnimation","invalidCells","gridModeInputFocused","gridModeEditLocked","singleCellEdit","attach","grid","super","signal","disconnectSignal","internalGrid","_activeEditRows","_rowEditSnapshots","Object","defineProperty","get","changedRows","configurable","resetChangedRows","silent","beginBulkEdit","rowIndex","field","beginCellEdit","onBeforeEditClose","exitRowEdit","capture","rowEl","findRenderedRowElement","composedPath","target","gridElement","contains","containsFocus","queueMicrotask","focusTrap","related","relatedTarget","focusCurrentCellEditor","detail","source","cb","newValue","_isGridEditMode","classList","add","requestRender","matches","FOCUSABLE_EDITOR_SELECTOR","blur","focus","activeEl","activeElement","preventDefault","stopPropagation","detach","remove","clear","handleQuery","query","onCellClick","event","effectiveConfig","isDoubleClick","originalEvent","hasEditableColumn","_columns","some","col","editable","onKeyDown","requestAfterRender","maxRow","_rows","length","currentRow","_focusRow","Math","ensureCellVisible","forward","shiftKey","handleTabNavigation","focusRow","focusCol","_focusCol","_visibleColumns","rowData","commitCellValue","ctrlKey","altKey","metaKey","row","cellEl","querySelector","activateEvent","CustomEvent","cancelable","bubbles","colIndex","trigger","dispatchEvent","legacyEvent","defaultPrevented","processColumns","columns","typeDefaults","adapter","__frameworkAdapter","getTypeDefault","typeEditorParams","appDefault","afterRender","restoreCellFocus","animateRow","size","cellKey","rowStr","colStr","parseInt","injectEditor","afterCellRender","context","cellElement","onScrollRender","rows","id","getRow","push","isRowEditing","isCellEditing","has","isRowChanged","rowId","getRowId","isRowChangedById","setInvalid","message","rowInvalids","set","syncInvalidCellAttribute","clearInvalid","delete","clearRowInvalid","fields","keys","clearAllInvalid","entries","_","isCellInvalid","getInvalidMessage","hasInvalidCells","getInvalidFields","invalid","findIndex","c","r","setAttribute","removeAttribute","ids","syncGridEditState","emit","_rowPool","startRowEdit","children","cell","i","setTimeout","targetCell","editor","preventScroll","commitActiveRowEdit","cancelActiveRowEdit","editableCols","filter","nextIdx","indexOf","forceHorizontalScroll","nextRow","revert","snapshot","entry","_getRowEntry","current","querySelectorAll","getAttribute","isNaN","hasAttribute","val","k","changedThisSession","hasRowChanged","changed","cancelled","emitCancelable","oldValue","isAnimationEnabled","startsWith","callbackKey","__editingCellCount","clearEditingState","parentElement","refreshVirtualWindow","reverted","firstTime","updateRow","changes","invalidWasSet","firstTimeForRow","emitPluginEvent","skipFocus","count","editFinalized","currentRowData","currentIndex","index","editorHost","className","innerHTML","colInternal","tplHolder","__editorTemplate","editorSpec","gridTypeDefaults","resolveEditor","callbacks","newVal","onValueChange","renderTemplateEditor","el","focusable","produced","HTMLSelectElement","wireEditorInputs","Node","HTMLTextAreaElement","mount","spec","console","warn","clone","cloneNode","compiledEditor","__compiledEditor","node","childNodes","firstChild","nodeType","TEXT_NODE","_m","g","evt","snapshotObj","currentObj","allKeys","rowIdx","colIdx","_bodyEl"],"mappings":"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"}
|