Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Editable table cells IxD spec #2477

Open
wants to merge 9 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
{
"type": "none",
"comment": "IxD spec for editable table",
"packageName": "@ni/nimble-components",
"email": "[email protected]",
"dependentChangeType": "none"
}
18 changes: 0 additions & 18 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

129 changes: 129 additions & 0 deletions packages/nimble-components/src/table/specs/editable-table-ixd.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,129 @@
# Nimble Editable Table (IxD)

## Overview

Nimble table support for interactive cell editing.

### Background

- [Figma worksheet](https://www.figma.com/design/r2yGNQNVFdE7cBO9CyHmQx/Nimble---IxD?node-id=1221-36463)
- [Input table columns (select, text field, etc) #1190](https://github.com/ni/nimble/issues/1190)

## Usage

**When to use:**

- When the table is used to display and edit structured engineering data such as measurements, specifications, or configurations.
- When inline editing can improve the user experience by reducing the need for separate forms or dialogs.
- When data accuracy can be ensured through validation and error handling mechanisms.

**When not to use:**
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we state some non-goals to make it clear we're not trying to replicate a spreadsheet experience? For example:

  • pasting values into a range of cells
  • adding/deleting/reordering rows or columns
  • advanced edits like formulas or autoformatting values to the expected type

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added to both "When not to use" and "Out of scope"

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

After reading the DFF tabular data spec I am more nervous that adding/deleting/reordering rows or columns is out of scope and expected to be implemented by clients. It's clear that they expect those operations to be possible and I think they'll seek our guidance about how to do them if we don't provide them. Re-opening in case we want to revisit this now, but I'm also fine to move it to another venue.


- When the table is used for displaying static or read-only data.
- When the user requires guidance entering or modifying data and would be best served by a form or other UX pattern.
- When the application requires spreadsheet features like pasting or filling data into a range of cells

## Requirements

- Inline editing support for table cells
- Validation and error handling for input data
- Initiate cell editing via keyboard or mouse
- Support for strings and numbers (1st priority)
- Support for enumerated values (2nd priority)
- All cells within a particular column share the same edit view and configuration
- Compatibility with assistive technologies (e.g., screen readers)

### Out of scope

- Support for complex data types (e.g., dates, times, custom objects)
- Advanced spreadsheet functionalities (e.g., formulas, cell references)
- Real-time collaboration or multi-user editing
- Custom cell styling or formatting

## Anatomy

![Editable table anatomy](./spec-images/editable-cell-anatomy.png)

| Element | Description |
| --------------------- | ------------------------------------------------------------------------ |
| Static cell value | Non-editable display of cell value |
| Action button | Button to open cell context menu. Appears on row hover or keyboard focus |
| Editable cell control | Text or numeric control that supports entering or editing cell data |

## Behavior
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What's the scope of "editability"?

  1. Do apps mark the entire table as editable? Specific rows? Specific columns? Specific cells?
  2. Does every column type support editing? For pretty much everything except text columns that opens the door to questions about what the editor should look like and what kind of validation is available.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Specific columns and we'll need to start with text and numeric, and add select and checkbox later. We shouldn't need to support every column type (at least right away.)


### States

![Editable cell states for text](./spec-images/text-editable-cell-states.png)
![Editable cell states for numeric](./spec-images/numeric-editable-cell-states.png)
![Editable cell states for select](./spec-images/select-editable-cell-states.png)
![Editable cell states for checkbox](./spec-images/checkbox-editable-cell-states.png)

#### Error State
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do apps need to be able to do their own validation and explicitly put a cell in an error state? Or is the table completely in control of the error state?

Copy link
Contributor

@atmgrifter00 atmgrifter00 Dec 2, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is ultimately a discussion for the HLD for the implementation of this feature. My preference, at the moment, would be for our error APIs to be very general provided via the table, and not by various columns (i.e. the table is not in complete control of the error state, it is managed by the user via APIs provided at the table level).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Requirements-wise: Client apps need to be able to specify their own cell validation rules (e.g., max/min/invalid characters/format) at the column level and they should get that feedback while editing the cell.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems like a call to essentially surface the appropriate HTML input attributes on the relevant column types. I'm concerned if we feel that there are validation APIs beyond those captured by the input attributes. The "format" you mention is one that may fall under that category. For such cases, I would prefer that clients manage their own custom validation logic, and simply rely on the information provided via events that fire during input, and then call general APIs to set the cell in an error state.


![Editable cell error states](./spec-images/editable-cell-error-data.png)

### ARIA Considerations

- Maintain compatibility with existing keyboard navigation behavior
- Consider resolving [existing ARIA gaps](https://github.com/ni/nimble/issues/2285)
- Consider adding `aria-readonly` for non-editable cells

### Mouse Interactions
fredvisser marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we cover mousewheel behavior in this section? Due to the table's virtualization we need to be explicit about the behavior we want.

If a cell has Edit focus (potentially with an edited cell value), then the user scrolls the mousewheel, is that equivalent to committing or reverting the edited value?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd strongly prefer that we maintain the edit state even if the cell scrolls out of view, such that it can scroll back into view in the same state. While it's true the menu button dropdown closes on scroll, setting or reverting an edited cell value is a much more aggressive action in response to something as simple as scrolling.

Do we have any options for persisting this type of state as the cell is scrolled out of view?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If that proves too expensive, we should prevent the table from scrolling while a cell is in edit focus.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Discussed this a bit with Fred today. Maintaining an edited cell as 'still being edited' after a scroll will probably be difficult due to virtualization - as soon as you scroll more than 1 row's height, even if the row is still visible in the table viewport, the row elements have been re-bound to different data at that point. We can remember which row/cell was being edited (and the current edited cell value), but we'd need to reapply that to a new row visual again (since now a different row visual represents the data row that was being edited, after the scroll), and there might be visual artifacts/ visible flashing if we do that.

Assuming we can prevent mousewheel scrolling the table while editing a cell, that's probably the simplest approach (least work to implement).


![Editable cell mouse navigation](./spec-images/editable-cell-mouse-navigation.png)

- Editable table cells show the same row hover state as non-editable cells on mouse hover.
- I.e. When hovering over a row, the action menu button appears. Depending on the table's selection mode, the row may be highlighted.
- The cursor will change from an *arrow cursor* when hovering over editable cells
- To a *text cursor* when hovering over cells with editable text or numeric controls
- To a *hand cursor* when hovering over select or boolean controls
- Single clicking editable cell shows the *edit focus* state.
- Clicking away from a focused editable cell sets the value and removes *edit focus*.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To be crystal clear, when you click on a non-editable cell while an editable cell has focus, and the row is selectable, do we expect the row to become selected? This is what I expect to happen with no extra work.


### Keyboard Interactions

![Editable cell key navigation](./spec-images/editable-cell-key-navigation.png)

- When a cell has keyboard focus (*cell focus* state), pressing `TAB` or `ENTER` transforms the cell into an input control in the *edit focus* state.
- When a cell has *edit focus*, pressing `TAB` moves the focus to the action button, or to the next available focus target.

![Editable cell enter key](./spec-images/editable-cell-enter-key.png)

- When a cell has *edit focus*, pressing `ENTER` sets the value and transforms it into the *cell focus* state.

![Editable cell escape key](./spec-images/editable-cell-escape-key.png)

- When a cell has *edit focus*, pressing `ESCAPE` reverts any value change and transforms it into the *cell focus* state.

![Editable cell data error](./spec-images/editable-cell-error-data.png)

- User edits are validated on entry. If the entered data is invalid,
- the error is shown in the *edit focus + error* state.
- pressing `TAB`, `ENTER` or `ESC` resets the cell to the most recent valid value.
- Other invalid cell data is shown in the *cell error* state.

### Out of scope

The client application is responsible for defining and implementing workflows for adding, deleting, and moving rows, as well as saving data.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We probably need to think through situations like if editing a value causes its sort order or grouping to change. Do we have a recommended experience? Are we ok with the value you just edited disappearing off screen? Should we scroll to it? Wait to apply changes until they finish a batch of edits?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, it would be nice to know what our UX options are here, and some reasoning behind our preferred choice. For instance, here are two different approaches that the Radzen DataGrid provides:

What isn't made clear in these examples is how changes made in the table component make their way to the "bound" data (if they do at all). One thing you can see, however, is that changes made to the cells have no affect on their current sort position. This would be a pretty tricky proposition for us to "guarantee" if we liked that behavior.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The least disruptive option is keep the new edited value in the same relative location after completing the edit, even if the column sort would move the row.

If that's not possible, and the row needs to immediately sort to the correct position, some affordance to indicate that the row is moving would be helpful. E.g., an animation of the row moving to it's new location. We shouldn't scroll to the new position.


Example row editing UX:

![Table row editing](./spec-images/table-row-editing.png)

Example save workflow:

![Table save workflow](./spec-images/table-save-workflow.png)

#### Touch-Screen Devices

- The edit workflow should support touch screen devices.

## Open Issues

See content marked "**QUESTION**" or "**NOTE**".

## References

- [AG Grid](https://www.ag-grid.com/example/)
- [Patternfly](https://www.patternfly.org/components/table/react-deprecated/#editable-rows)
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
22 changes: 8 additions & 14 deletions packages/storybook/.storybook/main.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
import { dirname, join } from 'path';
import remarkGfm from 'remark-gfm';
import CircularDependencyPlugin from 'circular-dependency-plugin';
import TerserPlugin from 'terser-webpack-plugin';
Expand All @@ -12,13 +11,13 @@ export const stories = [
];
export const addons = [
{
name: getAbsolutePath('@storybook/addon-essentials'),
name: '@storybook/addon-essentials',
options: {
outline: false,
docs: false
}
}, {
name: getAbsolutePath('@storybook/addon-docs'),
name: '@storybook/addon-docs',
options: {
mdxPluginOptions: {
mdxCompileOptions: {
Expand All @@ -27,12 +26,11 @@ export const addons = [
}
}
},
getAbsolutePath('@storybook/addon-a11y'),
getAbsolutePath('@storybook/addon-interactions'),
getAbsolutePath('@chromatic-com/storybook'),
getAbsolutePath('@storybook/addon-webpack5-compiler-swc'),
getAbsolutePath('storybook-addon-pseudo-states'),
getAbsolutePath('@storybook/addon-mdx-gfm')
'@storybook/addon-a11y',
'@storybook/addon-interactions',
'@chromatic-com/storybook',
'@storybook/addon-webpack5-compiler-swc',
'storybook-addon-pseudo-states'
];

export function webpackFinal(config) {
Expand Down Expand Up @@ -65,9 +63,5 @@ export function webpackFinal(config) {
}
export const staticDirs = ['public'];
export const framework = {
name: getAbsolutePath('@storybook/html-webpack5')
name: '@storybook/html-webpack5'
};

function getAbsolutePath(value) {
return dirname(require.resolve(join(value, 'package.json')));
}
1 change: 0 additions & 1 deletion packages/storybook/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,6 @@
"@storybook/addon-essentials": "^8.4.3",
"@storybook/addon-interactions": "^8.4.3",
"@storybook/addon-links": "^8.4.3",
"@storybook/addon-mdx-gfm": "^8.4.3",
"@storybook/addon-webpack5-compiler-swc": "^1.0.5",
"@storybook/cli": "^8.4.3",
"@storybook/csf": "^0.1.11",
Expand Down
Loading