@navikt/ds-react 8.10.2 → 8.10.4

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.
Files changed (200) hide show
  1. package/cjs/action-menu/ActionMenu.js +1 -1
  2. package/cjs/action-menu/ActionMenu.js.map +1 -1
  3. package/cjs/data/stories/Data.test-data.d.ts +24 -0
  4. package/cjs/data/stories/Data.test-data.js +1616 -0
  5. package/cjs/data/stories/Data.test-data.js.map +1 -0
  6. package/cjs/data/table/column-header/DataTableColumnHeader.d.ts +4 -1
  7. package/cjs/data/table/column-header/DataTableColumnHeader.js +4 -2
  8. package/cjs/data/table/column-header/DataTableColumnHeader.js.map +1 -1
  9. package/cjs/data/table/column-header/useTableColumnResize.d.ts +39 -14
  10. package/cjs/data/table/column-header/useTableColumnResize.js +37 -39
  11. package/cjs/data/table/column-header/useTableColumnResize.js.map +1 -1
  12. package/cjs/data/table/details-panel-row/DataTableDetailsPanelRow.d.ts +6 -0
  13. package/cjs/data/table/details-panel-row/DataTableDetailsPanelRow.js +32 -0
  14. package/cjs/data/table/details-panel-row/DataTableDetailsPanelRow.js.map +1 -0
  15. package/cjs/data/table/helpers/collectTableRowEntries.d.ts +13 -5
  16. package/cjs/data/table/helpers/collectTableRowEntries.js +26 -19
  17. package/cjs/data/table/helpers/collectTableRowEntries.js.map +1 -1
  18. package/cjs/data/table/helpers/selection/SelectionSubtreeHelper.d.ts +46 -0
  19. package/cjs/data/table/helpers/selection/SelectionSubtreeHelper.js +112 -0
  20. package/cjs/data/table/helpers/selection/SelectionSubtreeHelper.js.map +1 -0
  21. package/cjs/data/table/helpers/selection/getMultipleSelectProps.d.ts +3 -2
  22. package/cjs/data/table/helpers/selection/getMultipleSelectProps.js +43 -19
  23. package/cjs/data/table/helpers/selection/getMultipleSelectProps.js.map +1 -1
  24. package/cjs/data/table/helpers/selection/selection.types.d.ts +1 -0
  25. package/cjs/data/table/helpers/table-keyboard.d.ts +1 -2
  26. package/cjs/data/table/helpers/table-keyboard.js +1 -2
  27. package/cjs/data/table/helpers/table-keyboard.js.map +1 -1
  28. package/cjs/data/table/hooks/useColumnOptions.js +18 -5
  29. package/cjs/data/table/hooks/useColumnOptions.js.map +1 -1
  30. package/cjs/data/table/hooks/useTableDetailsPanel.d.ts +62 -0
  31. package/cjs/data/table/hooks/{useTableExpansion.js → useTableDetailsPanel.js} +26 -19
  32. package/cjs/data/table/hooks/useTableDetailsPanel.js.map +1 -0
  33. package/cjs/data/table/hooks/useTableItems.d.ts +18 -17
  34. package/cjs/data/table/hooks/useTableItems.js +27 -15
  35. package/cjs/data/table/hooks/useTableItems.js.map +1 -1
  36. package/cjs/data/table/hooks/useTableSelection.d.ts +6 -3
  37. package/cjs/data/table/hooks/useTableSelection.js +10 -4
  38. package/cjs/data/table/hooks/useTableSelection.js.map +1 -1
  39. package/cjs/data/table/index.d.ts +1 -2
  40. package/cjs/data/table/index.js +22 -12
  41. package/cjs/data/table/index.js.map +1 -1
  42. package/cjs/data/table/root/DataTable.types.d.ts +12 -10
  43. package/cjs/data/table/root/DataTableRoot.context.d.ts +5 -1
  44. package/cjs/data/table/root/DataTableRoot.context.js.map +1 -1
  45. package/cjs/data/table/root/DataTableRoot.d.ts +79 -115
  46. package/cjs/data/table/root/DataTableRoot.js +167 -39
  47. package/cjs/data/table/root/DataTableRoot.js.map +1 -1
  48. package/cjs/data/table/root/DataTableRoot.legacy.d.ts +177 -0
  49. package/cjs/data/table/root/DataTableRoot.legacy.js +104 -0
  50. package/cjs/data/table/root/DataTableRoot.legacy.js.map +1 -0
  51. package/cjs/data/table/sub-row-toggle/DataTableSubRowToggle.d.ts +6 -0
  52. package/cjs/data/table/sub-row-toggle/DataTableSubRowToggle.js +21 -0
  53. package/cjs/data/table/sub-row-toggle/DataTableSubRowToggle.js.map +1 -0
  54. package/cjs/data/table/tr/DataTableTr.js +11 -11
  55. package/cjs/data/table/tr/DataTableTr.js.map +1 -1
  56. package/cjs/form/checkbox/Checkbox.js +1 -0
  57. package/cjs/form/checkbox/Checkbox.js.map +1 -1
  58. package/cjs/form/radio/Radio.js +7 -1
  59. package/cjs/form/radio/Radio.js.map +1 -1
  60. package/cjs/modal/types.d.ts +8 -4
  61. package/cjs/utils/components/dismissablelayer/DismissableLayer.js +1 -1
  62. package/cjs/utils/components/dismissablelayer/DismissableLayer.js.map +1 -1
  63. package/cjs/utils/components/floating/Floating.d.ts +16 -1
  64. package/cjs/utils/components/floating/Floating.js +50 -13
  65. package/cjs/utils/components/floating/Floating.js.map +1 -1
  66. package/cjs/utils/components/floating-menu/Menu.js +1 -1
  67. package/cjs/utils/components/floating-menu/Menu.js.map +1 -1
  68. package/cjs/utils/helpers/create-strict-context.js +1 -1
  69. package/cjs/utils/helpers/create-strict-context.js.map +1 -1
  70. package/cjs/utils/hooks/useControllableState.d.ts +5 -5
  71. package/cjs/utils/hooks/useControllableState.js.map +1 -1
  72. package/cjs/utils/hooks/useValueAsRef.js +1 -1
  73. package/cjs/utils/hooks/useValueAsRef.js.map +1 -1
  74. package/cjs/utils-external/hooks/useId.js +1 -1
  75. package/cjs/utils-external/hooks/useId.js.map +1 -1
  76. package/esm/action-menu/ActionMenu.js +1 -1
  77. package/esm/action-menu/ActionMenu.js.map +1 -1
  78. package/esm/data/stories/Data.test-data.d.ts +24 -0
  79. package/esm/data/stories/Data.test-data.js +1607 -0
  80. package/esm/data/stories/Data.test-data.js.map +1 -0
  81. package/esm/data/table/column-header/DataTableColumnHeader.d.ts +4 -1
  82. package/esm/data/table/column-header/DataTableColumnHeader.js +4 -2
  83. package/esm/data/table/column-header/DataTableColumnHeader.js.map +1 -1
  84. package/esm/data/table/column-header/useTableColumnResize.d.ts +39 -14
  85. package/esm/data/table/column-header/useTableColumnResize.js +38 -40
  86. package/esm/data/table/column-header/useTableColumnResize.js.map +1 -1
  87. package/esm/data/table/details-panel-row/DataTableDetailsPanelRow.d.ts +6 -0
  88. package/esm/data/table/details-panel-row/DataTableDetailsPanelRow.js +27 -0
  89. package/esm/data/table/details-panel-row/DataTableDetailsPanelRow.js.map +1 -0
  90. package/esm/data/table/helpers/collectTableRowEntries.d.ts +13 -5
  91. package/esm/data/table/helpers/collectTableRowEntries.js +26 -19
  92. package/esm/data/table/helpers/collectTableRowEntries.js.map +1 -1
  93. package/esm/data/table/helpers/selection/SelectionSubtreeHelper.d.ts +46 -0
  94. package/esm/data/table/helpers/selection/SelectionSubtreeHelper.js +109 -0
  95. package/esm/data/table/helpers/selection/SelectionSubtreeHelper.js.map +1 -0
  96. package/esm/data/table/helpers/selection/getMultipleSelectProps.d.ts +3 -2
  97. package/esm/data/table/helpers/selection/getMultipleSelectProps.js +43 -19
  98. package/esm/data/table/helpers/selection/getMultipleSelectProps.js.map +1 -1
  99. package/esm/data/table/helpers/selection/selection.types.d.ts +1 -0
  100. package/esm/data/table/helpers/table-keyboard.d.ts +1 -2
  101. package/esm/data/table/helpers/table-keyboard.js +1 -2
  102. package/esm/data/table/helpers/table-keyboard.js.map +1 -1
  103. package/esm/data/table/hooks/useColumnOptions.js +18 -5
  104. package/esm/data/table/hooks/useColumnOptions.js.map +1 -1
  105. package/esm/data/table/hooks/useTableDetailsPanel.d.ts +62 -0
  106. package/esm/data/table/hooks/useTableDetailsPanel.js +58 -0
  107. package/esm/data/table/hooks/useTableDetailsPanel.js.map +1 -0
  108. package/esm/data/table/hooks/useTableItems.d.ts +18 -17
  109. package/esm/data/table/hooks/useTableItems.js +27 -15
  110. package/esm/data/table/hooks/useTableItems.js.map +1 -1
  111. package/esm/data/table/hooks/useTableSelection.d.ts +6 -3
  112. package/esm/data/table/hooks/useTableSelection.js +10 -4
  113. package/esm/data/table/hooks/useTableSelection.js.map +1 -1
  114. package/esm/data/table/index.d.ts +1 -2
  115. package/esm/data/table/index.js +21 -1
  116. package/esm/data/table/index.js.map +1 -1
  117. package/esm/data/table/root/DataTable.types.d.ts +12 -10
  118. package/esm/data/table/root/DataTableRoot.context.d.ts +5 -1
  119. package/esm/data/table/root/DataTableRoot.context.js.map +1 -1
  120. package/esm/data/table/root/DataTableRoot.d.ts +79 -115
  121. package/esm/data/table/root/DataTableRoot.js +174 -37
  122. package/esm/data/table/root/DataTableRoot.js.map +1 -1
  123. package/esm/data/table/root/DataTableRoot.legacy.d.ts +177 -0
  124. package/esm/data/table/root/DataTableRoot.legacy.js +59 -0
  125. package/esm/data/table/root/DataTableRoot.legacy.js.map +1 -0
  126. package/esm/data/table/sub-row-toggle/DataTableSubRowToggle.d.ts +6 -0
  127. package/esm/data/table/sub-row-toggle/DataTableSubRowToggle.js +16 -0
  128. package/esm/data/table/sub-row-toggle/DataTableSubRowToggle.js.map +1 -0
  129. package/esm/data/table/tr/DataTableTr.js +11 -11
  130. package/esm/data/table/tr/DataTableTr.js.map +1 -1
  131. package/esm/form/checkbox/Checkbox.js +1 -0
  132. package/esm/form/checkbox/Checkbox.js.map +1 -1
  133. package/esm/form/radio/Radio.js +7 -1
  134. package/esm/form/radio/Radio.js.map +1 -1
  135. package/esm/modal/types.d.ts +8 -4
  136. package/esm/utils/components/dismissablelayer/DismissableLayer.js +1 -1
  137. package/esm/utils/components/dismissablelayer/DismissableLayer.js.map +1 -1
  138. package/esm/utils/components/floating/Floating.d.ts +16 -1
  139. package/esm/utils/components/floating/Floating.js +48 -13
  140. package/esm/utils/components/floating/Floating.js.map +1 -1
  141. package/esm/utils/components/floating-menu/Menu.js +2 -2
  142. package/esm/utils/components/floating-menu/Menu.js.map +1 -1
  143. package/esm/utils/helpers/create-strict-context.js +1 -1
  144. package/esm/utils/helpers/create-strict-context.js.map +1 -1
  145. package/esm/utils/hooks/useControllableState.d.ts +5 -5
  146. package/esm/utils/hooks/useControllableState.js.map +1 -1
  147. package/esm/utils/hooks/useValueAsRef.js +1 -1
  148. package/esm/utils/hooks/useValueAsRef.js.map +1 -1
  149. package/esm/utils-external/hooks/useId.js +1 -1
  150. package/esm/utils-external/hooks/useId.js.map +1 -1
  151. package/package.json +3 -3
  152. package/src/action-menu/ActionMenu.tsx +1 -1
  153. package/src/data/stories/Data.test-data.tsx +1703 -0
  154. package/src/data/table/column-header/DataTableColumnHeader.tsx +11 -7
  155. package/src/data/table/column-header/useTableColumnResize.ts +95 -54
  156. package/src/data/table/details-panel-row/DataTableDetailsPanelRow.tsx +53 -0
  157. package/src/data/table/helpers/collectTableRowEntries.ts +55 -31
  158. package/src/data/table/helpers/selection/SelectionSubtreeHelper.test.ts +66 -0
  159. package/src/data/table/helpers/selection/SelectionSubtreeHelper.ts +162 -0
  160. package/src/data/table/helpers/selection/getMultipleSelectProps.ts +57 -20
  161. package/src/data/table/helpers/selection/selection.types.ts +1 -0
  162. package/src/data/table/helpers/table-keyboard.ts +1 -2
  163. package/src/data/table/hooks/__tests__/useTableItems.test.ts +27 -6
  164. package/src/data/table/hooks/__tests__/useTableSelection.test.ts +182 -58
  165. package/src/data/table/hooks/useColumnOptions.ts +19 -5
  166. package/src/data/table/hooks/useTableDetailsPanel.tsx +182 -0
  167. package/src/data/table/hooks/useTableItems.ts +74 -60
  168. package/src/data/table/hooks/useTableSelection.ts +27 -12
  169. package/src/data/table/index.tsx +5 -3
  170. package/src/data/table/root/DataTable.types.ts +25 -10
  171. package/src/data/table/root/DataTableRoot.context.ts +5 -1
  172. package/src/data/table/root/DataTableRoot.legacy.tsx +297 -0
  173. package/src/data/table/root/DataTableRoot.tsx +483 -219
  174. package/src/data/table/sub-row-toggle/DataTableSubRowToggle.tsx +39 -0
  175. package/src/data/table/tr/DataTableTr.tsx +14 -13
  176. package/src/form/checkbox/Checkbox.tsx +1 -0
  177. package/src/form/radio/Radio.tsx +7 -1
  178. package/src/modal/types.ts +8 -4
  179. package/src/utils/components/dismissablelayer/DismissableLayer.tsx +1 -1
  180. package/src/utils/components/floating/Floating.tsx +56 -13
  181. package/src/utils/components/floating-menu/Menu.tsx +4 -1
  182. package/src/utils/helpers/create-strict-context.tsx +1 -1
  183. package/src/utils/hooks/useControllableState.ts +11 -8
  184. package/src/utils/hooks/useValueAsRef.ts +1 -1
  185. package/src/utils-external/hooks/useId.ts +1 -1
  186. package/cjs/data/table/hooks/useTableExpansion.d.ts +0 -29
  187. package/cjs/data/table/hooks/useTableExpansion.js.map +0 -1
  188. package/cjs/data/table/root/DataTableAuto.d.ts +0 -174
  189. package/cjs/data/table/root/DataTableAuto.js +0 -206
  190. package/cjs/data/table/root/DataTableAuto.js.map +0 -1
  191. package/esm/data/table/hooks/useTableExpansion.d.ts +0 -29
  192. package/esm/data/table/hooks/useTableExpansion.js +0 -51
  193. package/esm/data/table/hooks/useTableExpansion.js.map +0 -1
  194. package/esm/data/table/root/DataTableAuto.d.ts +0 -174
  195. package/esm/data/table/root/DataTableAuto.js +0 -170
  196. package/esm/data/table/root/DataTableAuto.js.map +0 -1
  197. package/src/data/table/hooks/__tests__/useTableExpansion.test.tsx +0 -115
  198. package/src/data/table/hooks/useTableExpansion.tsx +0 -141
  199. package/src/data/table/root/DataTableAuto.test.tsx +0 -118
  200. package/src/data/table/root/DataTableAuto.tsx +0 -603
@@ -0,0 +1,162 @@
1
+ type SelectionKey = string | number;
2
+
3
+ type SelectionStats = {
4
+ selectableCount: number;
5
+ selectedCount: number;
6
+ };
7
+
8
+ type SelectionSubtreeHelperArgs = {
9
+ childRowIdsById?: Map<SelectionKey, SelectionKey[]>;
10
+ disabledKeysSet: Set<SelectionKey>;
11
+ selectedKeysSet: Set<SelectionKey>;
12
+ };
13
+
14
+ /**
15
+ * Helper class for managing selection state in a tree structure.
16
+ * - It provides methods to get selectable keys in a subtree
17
+ * - Compute selection statistics for a subtree
18
+ * - Determine if a subtree is fully selected.
19
+ *
20
+ * Results of selection statistics are cached to optimize performance for repeated calls on the same subtree.
21
+ */
22
+ class SelectionSubtreeHelper {
23
+ private childRowIdsById: Map<SelectionKey, SelectionKey[]>;
24
+ private disabledKeysSet: Set<SelectionKey>;
25
+ private selectedKeysSet: Set<SelectionKey>;
26
+ private selectionStatsCache = new Map<SelectionKey, SelectionStats>();
27
+
28
+ constructor({
29
+ childRowIdsById,
30
+ disabledKeysSet,
31
+ selectedKeysSet,
32
+ }: SelectionSubtreeHelperArgs) {
33
+ this.childRowIdsById = childRowIdsById ?? new Map();
34
+ this.disabledKeysSet = disabledKeysSet;
35
+ this.selectedKeysSet = selectedKeysSet;
36
+ }
37
+
38
+ getSelectableKeys(rootIds: SelectionKey[]): SelectionKey[] {
39
+ const visitedKeys = new Set<SelectionKey>();
40
+ const selectableKeys: SelectionKey[] = [];
41
+ const stack = [...rootIds].reverse();
42
+
43
+ while (stack.length > 0) {
44
+ const key = stack.pop();
45
+
46
+ if (key == null || visitedKeys.has(key)) {
47
+ continue;
48
+ }
49
+
50
+ visitedKeys.add(key);
51
+
52
+ if (!this.disabledKeysSet.has(key)) {
53
+ selectableKeys.push(key);
54
+ }
55
+
56
+ const childRowIds = this.childRowIdsById.get(key) ?? [];
57
+
58
+ for (
59
+ let childIndex = childRowIds.length - 1;
60
+ childIndex >= 0;
61
+ childIndex--
62
+ ) {
63
+ stack.push(childRowIds[childIndex]);
64
+ }
65
+ }
66
+
67
+ return selectableKeys;
68
+ }
69
+
70
+ /**
71
+ * Returns the number of selectable and selected rows in the subtree of the given root ID.
72
+ * Results are cached after the first computation to optimize repeated calls for the same root ID.
73
+ *
74
+ * The selectable count excludes disabled rows, and the selected count excludes disabled rows that are selected.
75
+ * The method is implemented iteratively to handle deep trees without hitting call stack limits.
76
+ *
77
+ * How it works:
78
+ * - Manually add root ID to stack to get processing going. Note that the ready-flag is `false`.
79
+ * - Pop stack until empty. For each entry:
80
+ * - - If entry is already cached, skip it.
81
+ * - - If entry is not ready, push it back as ready and push all its children as not ready.
82
+ * - - If entry is ready, compute its stats based on its own state and the stats of its children, then cache the result.
83
+ * - Since we add all the children to the stack after pushing element with ready: true, while "popping" the stack we will always encounter the children before their parent is ready, ensuring that the stats for all children are computed and cached before computing the stats for their parent.
84
+ * - Finally, return the cached stats for the root ID.
85
+ */
86
+ getSelectionStats(rootId: SelectionKey): SelectionStats {
87
+ const cachedStats = this.selectionStatsCache.get(rootId);
88
+
89
+ if (cachedStats) {
90
+ return cachedStats;
91
+ }
92
+
93
+ /* Compute subtree totals iteratively so deep trees do not depend on call stack depth. */
94
+ const stack: { key: SelectionKey; ready: boolean }[] = [
95
+ { key: rootId, ready: false },
96
+ ];
97
+
98
+ while (stack.length > 0) {
99
+ const entry = stack.pop();
100
+
101
+ if (!entry) {
102
+ continue;
103
+ }
104
+
105
+ if (this.selectionStatsCache.has(entry.key)) {
106
+ continue;
107
+ }
108
+
109
+ if (entry.ready) {
110
+ let selectableCount = this.disabledKeysSet.has(entry.key) ? 0 : 1;
111
+ let selectedCount =
112
+ !this.disabledKeysSet.has(entry.key) &&
113
+ this.selectedKeysSet.has(entry.key)
114
+ ? 1
115
+ : 0;
116
+
117
+ for (const childKey of this.childRowIdsById.get(entry.key) ?? []) {
118
+ const childStats = this.selectionStatsCache.get(childKey);
119
+
120
+ if (!childStats) {
121
+ continue;
122
+ }
123
+
124
+ selectableCount += childStats.selectableCount;
125
+ selectedCount += childStats.selectedCount;
126
+ }
127
+
128
+ this.selectionStatsCache.set(entry.key, {
129
+ selectableCount,
130
+ selectedCount,
131
+ });
132
+ continue;
133
+ }
134
+
135
+ stack.push({ key: entry.key, ready: true });
136
+
137
+ for (const childKey of this.childRowIdsById.get(entry.key) ?? []) {
138
+ if (!this.selectionStatsCache.has(childKey)) {
139
+ stack.push({ key: childKey, ready: false });
140
+ }
141
+ }
142
+ }
143
+
144
+ return (
145
+ this.selectionStatsCache.get(rootId) ?? {
146
+ selectableCount: 0,
147
+ selectedCount: 0,
148
+ }
149
+ );
150
+ }
151
+
152
+ isFullySelected(rootId: SelectionKey): boolean {
153
+ const stats = this.getSelectionStats(rootId);
154
+
155
+ return (
156
+ stats.selectableCount > 0 && stats.selectedCount === stats.selectableCount
157
+ );
158
+ }
159
+ }
160
+
161
+ export { SelectionSubtreeHelper };
162
+ export type { SelectionKey, SelectionStats, SelectionSubtreeHelperArgs };
@@ -1,11 +1,13 @@
1
1
  import type { CheckboxInputProps } from "../../../../form/checkbox/checkbox-input/CheckboxInput";
2
+ import { SelectionSubtreeHelper } from "./SelectionSubtreeHelper";
2
3
 
3
4
  type GetMultipleSelectPropsArgs = {
4
5
  selectedKeysSet: Set<string | number>;
5
6
  selectedKeys: (string | number)[];
6
7
  setSelectedKeys: (keys: (string | number)[]) => void;
7
8
  disabledKeysSet: Set<string | number>;
8
- allRowKeys: (string | number)[];
9
+ visibleRowIds: (string | number)[];
10
+ childRowIdsById?: Map<string | number, (string | number)[]>;
9
11
  };
10
12
 
11
13
  function getMultipleSelectProps({
@@ -13,34 +15,56 @@ function getMultipleSelectProps({
13
15
  selectedKeys,
14
16
  setSelectedKeys,
15
17
  disabledKeysSet,
16
- allRowKeys,
18
+ visibleRowIds,
19
+ childRowIdsById,
17
20
  }: GetMultipleSelectPropsArgs) {
18
- const allRowKeysSet = new Set(allRowKeys);
19
- const selectableKeys = allRowKeys.filter((k) => !disabledKeysSet.has(k));
21
+ const subtreeHelper = new SelectionSubtreeHelper({
22
+ childRowIdsById,
23
+ disabledKeysSet,
24
+ selectedKeysSet,
25
+ });
20
26
 
21
- const selectedSelectableCount = selectableKeys.filter((k) =>
27
+ // Header selection traverses the visible roots and skips already visited
28
+ // descendants, so expanded trees stay linear in the number of rows.
29
+ const headerSelectableKeys = subtreeHelper.getSelectableKeys(visibleRowIds);
30
+ const headerSelectableKeysSet = new Set(headerSelectableKeys);
31
+
32
+ const selectedSelectableCount = headerSelectableKeys.filter((k) =>
22
33
  selectedKeysSet.has(k),
23
34
  ).length;
24
35
 
25
36
  const allSelectableSelected =
26
- selectableKeys.length > 0 &&
27
- selectedSelectableCount === selectableKeys.length;
37
+ headerSelectableKeys.length > 0 &&
38
+ selectedSelectableCount === headerSelectableKeys.length;
28
39
 
29
40
  const indeterminate =
30
41
  selectedSelectableCount > 0 &&
31
- selectedSelectableCount < selectableKeys.length;
42
+ selectedSelectableCount < headerSelectableKeys.length;
32
43
 
33
44
  const selectedKeysNotInView = selectedKeys.filter(
34
- (k) => !allRowKeysSet.has(k),
45
+ (k) => !headerSelectableKeysSet.has(k),
35
46
  );
36
47
  const disabledSelected = selectedKeys.filter((k) => disabledKeysSet.has(k));
37
- const preservedKeys = [...selectedKeysNotInView, ...disabledSelected];
48
+ const preservedKeys = [
49
+ ...new Set([...selectedKeysNotInView, ...disabledSelected]),
50
+ ];
51
+
52
+ const isGroupFullySelected = (key: string | number) => {
53
+ const groupStats = subtreeHelper.getSelectionStats(key);
54
+
55
+ return (
56
+ groupStats.selectableCount > 0 &&
57
+ groupStats.selectedCount === groupStats.selectableCount
58
+ );
59
+ };
38
60
 
39
61
  const handleToggleAll = () => {
40
62
  if (allSelectableSelected) {
41
63
  setSelectedKeys(preservedKeys);
42
64
  } else {
43
- setSelectedKeys([...new Set([...preservedKeys, ...selectableKeys])]);
65
+ setSelectedKeys([
66
+ ...new Set([...preservedKeys, ...headerSelectableKeys]),
67
+ ]);
44
68
  }
45
69
  };
46
70
 
@@ -48,10 +72,16 @@ function getMultipleSelectProps({
48
72
  if (disabledKeysSet.has(key)) {
49
73
  return;
50
74
  }
51
- if (selectedKeysSet.has(key)) {
52
- setSelectedKeys(selectedKeys.filter((k) => k !== key));
75
+
76
+ const groupKeys = subtreeHelper.getSelectableKeys([key]);
77
+
78
+ if (isGroupFullySelected(key)) {
79
+ const groupKeysSet = new Set(groupKeys);
80
+ setSelectedKeys(
81
+ selectedKeys.filter((selectedKey) => !groupKeysSet.has(selectedKey)),
82
+ );
53
83
  } else {
54
- setSelectedKeys([...selectedKeys, key]);
84
+ setSelectedKeys([...new Set([...selectedKeys, ...groupKeys])]);
55
85
  }
56
86
  };
57
87
 
@@ -60,13 +90,20 @@ function getMultipleSelectProps({
60
90
  onChange: handleToggleAll,
61
91
  checked: allSelectableSelected,
62
92
  indeterminate,
63
- disabled: selectableKeys.length === 0,
64
- }),
65
- getRowCheckboxProps: (key: string | number): CheckboxInputProps => ({
66
- onChange: () => handleToggleRow(key),
67
- checked: selectedKeysSet.has(key),
68
- disabled: disabledKeysSet.has(key),
93
+ disabled: headerSelectableKeys.length === 0,
69
94
  }),
95
+ getRowCheckboxProps: (key: string | number): CheckboxInputProps => {
96
+ const groupStats = subtreeHelper.getSelectionStats(key);
97
+
98
+ return {
99
+ onChange: () => handleToggleRow(key),
100
+ checked: isGroupFullySelected(key),
101
+ indeterminate:
102
+ groupStats.selectedCount > 0 &&
103
+ groupStats.selectedCount < groupStats.selectableCount,
104
+ disabled: disabledKeysSet.has(key),
105
+ };
106
+ },
70
107
  toggleSelection: handleToggleRow,
71
108
  };
72
109
  }
@@ -10,6 +10,7 @@ type SelectionProps = {
10
10
  * When set to "single", only one row can be selected at a time (renders radio buttons).
11
11
  *
12
12
  * When set to "multiple", multiple rows can be selected (renders checkboxes).
13
+ * Nested rows use cascading selection, so selecting a parent toggles its descendants too.
13
14
  *
14
15
  * @default "none"
15
16
  */
@@ -16,8 +16,7 @@ type NavigationAction =
16
16
 
17
17
  /**
18
18
  * Maps keyboard events to navigation actions.
19
- * Supports arrow keys, Home/End (row navigation), Ctrl/Cmd+Home/End (table navigation),
20
- * and PageUp/PageDown (multi-row navigation).
19
+ * Supports arrow keys, Home/End (row navigation), Ctrl/Cmd+Home/End (table navigation).
21
20
  */
22
21
  function getNavigationAction(event: KeyboardEvent): NavigationAction | null {
23
22
  const key = event.key;
@@ -78,20 +78,39 @@ describe("useTableItems", () => {
78
78
  useTableItems({
79
79
  items: nestedRows,
80
80
  getRowId: (row) => row.id,
81
- getSubRows,
82
- defaultExpandedSubRowIds: ["a"],
81
+ subRows: {
82
+ getRows: getSubRows,
83
+ defaultExpandedRowIds: ["a"],
84
+ },
83
85
  }),
84
86
  );
85
87
 
86
88
  expect(getVisibleIds(result.current.items)).toEqual(["a", "a1", "a2", "b"]);
87
89
  });
88
90
 
91
+ test("collects direct child row ids even when nested rows are collapsed", () => {
92
+ const { result } = renderHook(() =>
93
+ useTableItems({
94
+ items: nestedRows,
95
+ getRowId: (row) => row.id,
96
+ subRows: { getRows: getSubRows },
97
+ }),
98
+ );
99
+
100
+ expect(result.current.childRowIdsById.get("a")).toEqual(["a1", "a2"]);
101
+ expect(result.current.childRowIdsById.get("a2")).toEqual(["a2a"]);
102
+ expect(result.current.childRowIdsById.get("b")).toEqual(["b1"]);
103
+ });
104
+
89
105
  test("uses the same fallback root id to reveal child rows when getRowId is omitted", () => {
90
106
  const { result } = renderHook(() =>
91
107
  useTableItems({
92
108
  items: fallbackRows,
93
- getSubRows: (row) => row.subRows ?? [],
94
- defaultExpandedSubRowIds: [0],
109
+ getRowId: (_, index) => index,
110
+ subRows: {
111
+ getRows: (row: any) => row.subRows ?? [],
112
+ defaultExpandedRowIds: [0],
113
+ },
95
114
  }),
96
115
  );
97
116
 
@@ -107,8 +126,10 @@ describe("useTableItems", () => {
107
126
  useTableItems({
108
127
  items: nestedRows,
109
128
  getRowId: (row) => row.id,
110
- getSubRows,
111
- expandedSubRowIds: expandedIds,
129
+ subRows: {
130
+ getRows: getSubRows,
131
+ expandedRowIds: expandedIds,
132
+ },
112
133
  }),
113
134
  {
114
135
  initialProps: { expandedIds: [] as (string | number)[] },