playbook_ui 14.19.0 → 14.20.0.pre.alpha.PLAY2178advancedtablerowpinning7978

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 (162) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/pb_advanced_table/Components/RegularTableView.tsx +92 -5
  3. data/app/pb_kits/playbook/pb_advanced_table/Components/TableActionBar.tsx +175 -16
  4. data/app/pb_kits/playbook/pb_advanced_table/Components/TableHeaderCell.tsx +56 -25
  5. data/app/pb_kits/playbook/pb_advanced_table/Context/AdvancedTableContext.tsx +58 -2
  6. data/app/pb_kits/playbook/pb_advanced_table/Hooks/useTableActions.ts +1 -1
  7. data/app/pb_kits/playbook/pb_advanced_table/Hooks/useTableState.ts +34 -15
  8. data/app/pb_kits/playbook/pb_advanced_table/SubKits/TableHeader.tsx +7 -3
  9. data/app/pb_kits/playbook/pb_advanced_table/Utilities/VisibilityTree.ts +47 -0
  10. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.scss +11 -10
  11. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.tsx +13 -4
  12. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.html.erb +16 -8
  13. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.rb +9 -0
  14. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.test.jsx +61 -0
  15. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_beta.md +6 -2
  16. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility.jsx +57 -0
  17. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility.md +4 -0
  18. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility_custom.jsx +62 -0
  19. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility_custom.md +1 -0
  20. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility_multi.jsx +82 -0
  21. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility_multi.md +1 -0
  22. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility_with_state.jsx +66 -0
  23. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_visibility_with_state.md +3 -0
  24. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_default.md +1 -1
  25. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_pinned_rows.jsx +57 -0
  26. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_pinned_rows_react.md +5 -0
  27. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_actions_rails.html.erb +137 -0
  28. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_actions_rails.md +3 -0
  29. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_header_rails.html.erb +40 -0
  30. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_header_rails.md +1 -0
  31. data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +7 -0
  32. data/app/pb_kits/playbook/pb_advanced_table/docs/index.js +6 -1
  33. data/app/pb_kits/playbook/pb_advanced_table/index.js +155 -12
  34. data/app/pb_kits/playbook/pb_advanced_table/scss_partials/advanced_table_sticky_mixin.scss +1 -0
  35. data/app/pb_kits/playbook/pb_advanced_table/table_action_bar.html.erb +23 -0
  36. data/app/pb_kits/playbook/pb_advanced_table/table_action_bar.rb +19 -0
  37. data/app/pb_kits/playbook/pb_advanced_table/table_header.rb +4 -0
  38. data/app/pb_kits/playbook/pb_dropdown/_dropdown.scss +1 -1
  39. data/app/pb_kits/playbook/pb_dropdown/_dropdown.tsx +77 -19
  40. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_default_rails.html.erb +31 -0
  41. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_default_rails.md +5 -0
  42. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select.jsx +56 -0
  43. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select.md +3 -0
  44. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_display.jsx +58 -0
  45. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_display.md +3 -0
  46. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_display_rails.html.erb +20 -0
  47. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_display_rails.md +1 -0
  48. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_rails.html.erb +19 -0
  49. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_rails.md +3 -0
  50. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_autocomplete.html.erb +20 -0
  51. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_autocomplete.jsx +57 -0
  52. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_autocomplete.md +1 -0
  53. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_custom_options.html.erb +50 -0
  54. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_custom_options.jsx +105 -0
  55. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_default.html.erb +22 -0
  56. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_multi_select_with_default.jsx +67 -0
  57. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display.jsx +11 -0
  58. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display.md +1 -1
  59. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display_rails.html.erb +33 -2
  60. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display_rails.md +3 -1
  61. data/app/pb_kits/playbook/pb_dropdown/docs/example.yml +11 -1
  62. data/app/pb_kits/playbook/pb_dropdown/docs/index.js +5 -0
  63. data/app/pb_kits/playbook/pb_dropdown/dropdown.html.erb +3 -3
  64. data/app/pb_kits/playbook/pb_dropdown/dropdown.rb +16 -2
  65. data/app/pb_kits/playbook/pb_dropdown/dropdown.test.jsx +108 -2
  66. data/app/pb_kits/playbook/pb_dropdown/dropdown_trigger.html.erb +34 -13
  67. data/app/pb_kits/playbook/pb_dropdown/dropdown_trigger.rb +3 -1
  68. data/app/pb_kits/playbook/pb_dropdown/hooks/useHandleOnKeydown.tsx +0 -6
  69. data/app/pb_kits/playbook/pb_dropdown/index.js +336 -30
  70. data/app/pb_kits/playbook/pb_dropdown/keyboard_accessibility.js +39 -12
  71. data/app/pb_kits/playbook/pb_dropdown/subcomponents/DropdownContainer.tsx +2 -2
  72. data/app/pb_kits/playbook/pb_dropdown/subcomponents/DropdownOption.tsx +16 -12
  73. data/app/pb_kits/playbook/pb_dropdown/subcomponents/DropdownTrigger.tsx +79 -13
  74. data/app/pb_kits/playbook/pb_dropdown/subcomponents/MultiSelectTriggerDisplay.tsx +58 -0
  75. data/app/pb_kits/playbook/pb_file_upload/_file_upload.scss +13 -0
  76. data/app/pb_kits/playbook/pb_file_upload/_file_upload.tsx +11 -1
  77. data/app/pb_kits/playbook/pb_file_upload/docs/_file_upload_error.html.erb +1 -0
  78. data/app/pb_kits/playbook/pb_file_upload/docs/_file_upload_error.jsx +41 -0
  79. data/app/pb_kits/playbook/pb_file_upload/docs/example.yml +2 -0
  80. data/app/pb_kits/playbook/pb_file_upload/docs/index.js +1 -0
  81. data/app/pb_kits/playbook/pb_file_upload/file_upload.html.erb +1 -0
  82. data/app/pb_kits/playbook/pb_file_upload/file_upload.rb +7 -1
  83. data/app/pb_kits/playbook/pb_file_upload/fileupload.test.js +18 -0
  84. data/app/pb_kits/playbook/pb_form/docs/_form_form_with.html.erb +1 -0
  85. data/app/pb_kits/playbook/pb_form/docs/_form_form_with_validate.html.erb +1 -0
  86. data/app/pb_kits/playbook/pb_form_group/_error_state_mixin.scss +2 -2
  87. data/app/pb_kits/playbook/pb_form_pill/_form_pill.scss +19 -12
  88. data/app/pb_kits/playbook/pb_home_address_street/_home_address_street.tsx +13 -7
  89. data/app/pb_kits/playbook/pb_multi_level_select/_multi_level_select.tsx +2 -2
  90. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_color.html.erb +11 -11
  91. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_color.jsx +11 -11
  92. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_default.html.erb +11 -11
  93. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_default.jsx +11 -11
  94. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled.html.erb +11 -11
  95. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled.jsx +11 -11
  96. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options.html.erb +11 -11
  97. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options.jsx +11 -11
  98. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options_default.html.erb +11 -11
  99. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options_default.jsx +11 -11
  100. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options_parent.html.erb +11 -11
  101. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options_parent.jsx +11 -11
  102. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options_parent_default.html.erb +11 -11
  103. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_disabled_options_parent_default.jsx +11 -11
  104. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_error.html.erb +11 -11
  105. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_error.jsx +11 -11
  106. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_label.html.erb +11 -11
  107. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_label.jsx +11 -11
  108. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_react_hook.jsx +11 -11
  109. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_reset.html.erb +11 -11
  110. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_return_all_selected.html.erb +11 -11
  111. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_return_all_selected.jsx +11 -11
  112. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_selected_ids.html.erb +11 -11
  113. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_selected_ids.md +2 -0
  114. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_selected_ids_react.jsx +11 -11
  115. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_selected_ids_react.md +3 -1
  116. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_single.html.erb +22 -22
  117. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_single.jsx +22 -22
  118. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_single_children_only.html.erb +22 -22
  119. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_single_children_only.jsx +22 -22
  120. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_with_children.jsx +11 -11
  121. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_with_children_with_radios.jsx +11 -11
  122. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_with_form.html.erb +11 -11
  123. data/app/pb_kits/playbook/pb_person/_person.tsx +12 -2
  124. data/app/pb_kits/playbook/pb_phone_number_input/_phone_number_input.scss +9 -9
  125. data/app/pb_kits/playbook/pb_phone_number_input/_phone_number_input.tsx +4 -0
  126. data/app/pb_kits/playbook/pb_phone_number_input/docs/_phone_number_input_exclude_countries.html.erb +4 -0
  127. data/app/pb_kits/playbook/pb_phone_number_input/docs/_phone_number_input_exclude_countries.jsx +15 -0
  128. data/app/pb_kits/playbook/pb_phone_number_input/docs/_phone_number_input_exclude_countries.md +1 -0
  129. data/app/pb_kits/playbook/pb_phone_number_input/docs/_phone_number_input_only_countries.jsx +1 -1
  130. data/app/pb_kits/playbook/pb_phone_number_input/docs/example.yml +4 -3
  131. data/app/pb_kits/playbook/pb_phone_number_input/docs/index.js +1 -0
  132. data/app/pb_kits/playbook/pb_phone_number_input/phone_number_input.rb +3 -0
  133. data/app/pb_kits/playbook/pb_section_separator/_section_separator.tsx +2 -2
  134. data/app/pb_kits/playbook/pb_select/docs/_select_custom_select_subheaders.html.erb +12 -0
  135. data/app/pb_kits/playbook/pb_select/docs/_select_custom_select_subheaders.jsx +31 -0
  136. data/app/pb_kits/playbook/pb_select/docs/_select_custom_select_subheaders.md +1 -0
  137. data/app/pb_kits/playbook/pb_select/docs/example.yml +2 -0
  138. data/app/pb_kits/playbook/pb_select/docs/index.js +1 -0
  139. data/app/pb_kits/playbook/pb_text_input/_text_input.scss +4 -2
  140. data/app/pb_kits/playbook/pb_typeahead/_typeahead.tsx +73 -3
  141. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_preserve_input.jsx +23 -0
  142. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_preserve_input.md +1 -0
  143. data/app/pb_kits/playbook/pb_typeahead/docs/example.yml +1 -0
  144. data/app/pb_kits/playbook/pb_typeahead/docs/index.js +1 -0
  145. data/dist/chunks/_typeahead-CRW6dJbW.js +22 -0
  146. data/dist/chunks/_weekday_stacked-yWpUc_c0.js +45 -0
  147. data/dist/chunks/lazysizes-B7xYodB-.js +1 -0
  148. data/dist/chunks/lib-D5R1BjUn.js +29 -0
  149. data/dist/chunks/{pb_form_validation-BioH7DWv.js → pb_form_validation-BZ2AVAi_.js} +1 -1
  150. data/dist/chunks/vendor.js +1 -1
  151. data/dist/playbook-doc.js +2 -2
  152. data/dist/playbook-rails-react-bindings.js +1 -1
  153. data/dist/playbook-rails.js +1 -1
  154. data/dist/playbook.css +1 -1
  155. data/lib/playbook/kit_base.rb +3 -3
  156. data/lib/playbook/version.rb +2 -2
  157. metadata +52 -8
  158. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_default.html.erb +0 -10
  159. data/dist/chunks/_typeahead-D62OcwsT.js +0 -22
  160. data/dist/chunks/_weekday_stacked-Ceh9N0ow.js +0 -45
  161. data/dist/chunks/lazysizes-DHz07jlL.js +0 -1
  162. data/dist/chunks/lib-CeKZrPmu.js +0 -29
@@ -39,7 +39,8 @@ export const TableHeader = ({
39
39
  hasAnySubRows,
40
40
  showActionsBar,
41
41
  selectableRows,
42
- responsive
42
+ responsive,
43
+ headerRef
43
44
  } = useContext(AdvancedTableContext)
44
45
 
45
46
  const classes = classnames(
@@ -62,8 +63,11 @@ export const TableHeader = ({
62
63
  id={id}
63
64
  >
64
65
  {/* Get the header groups (only one in this example) */}
65
- {table.getHeaderGroups().map((headerGroup: HeaderGroup<GenericObject>) => (
66
- <tr key={`${headerGroup.id}-headerGroup`}>
66
+ {table.getHeaderGroups().map((headerGroup: HeaderGroup<GenericObject>, index: number) => (
67
+ <tr
68
+ key={`${headerGroup.id}-headerGroup`}
69
+ ref={index === 0 ? headerRef : null}
70
+ >
67
71
  {!hasAnySubRows && selectableRows && (
68
72
  <th className={customCellClassnames}>
69
73
  <Checkbox
@@ -0,0 +1,47 @@
1
+ export interface VisibilityNode {
2
+ id: string;
3
+ label: string;
4
+ children?: VisibilityNode[];
5
+ }
6
+ export const buildVisibilityTree = (
7
+ defs: any[],
8
+ allowed?: string[] | null
9
+ ): VisibilityNode[] =>
10
+ defs
11
+ .map((def) => {
12
+ const isGroup = Array.isArray(def.columns) && def.columns.length > 0;
13
+
14
+ // No filter at all → keep it
15
+ if (!allowed?.length) {
16
+ return isGroup
17
+ ? {
18
+ id: def.id,
19
+ label: def.label,
20
+ children: buildVisibilityTree(def.columns, allowed),
21
+ }
22
+ : { id: def.id, label: def.label };
23
+ }
24
+
25
+ // 1️⃣ If *this* ID is explicitly allowed → keep it & all its children
26
+ if (allowed.includes(def.id)) {
27
+ return isGroup
28
+ ? {
29
+ id: def.id,
30
+ label: def.label,
31
+ children: buildVisibilityTree(def.columns, null),
32
+ }
33
+ : { id: def.id, label: def.label };
34
+ }
35
+
36
+ // Otherwise, if it’s a group, recurse & keep only if kids survive
37
+ if (isGroup) {
38
+ const kids = buildVisibilityTree(def.columns, allowed).filter(Boolean);
39
+ return kids.length
40
+ ? { id: def.id, label: def.label, children: kids }
41
+ : null;
42
+ }
43
+
44
+ // Leaf not allowed → drop it
45
+ return null;
46
+ })
47
+ .filter(Boolean);
@@ -150,13 +150,6 @@
150
150
  box-sizing: border-box !important;
151
151
  }
152
152
  }
153
- .expand-by-depth-dropdown-wrapper {
154
- position: unset !important;
155
- }
156
- .expand-by-depth-dropdown {
157
- width: unset !important;
158
- text-align: left;
159
- }
160
153
  }
161
154
 
162
155
  .pb_advanced_table_body {
@@ -545,14 +538,12 @@
545
538
  background-color: $white;
546
539
  box-shadow: $shadow_deep;
547
540
  }
541
+
548
542
  @include advanced-table-sticky-mixin(
549
543
  $border_light,
550
544
  $white,
551
545
  lighten($silver, $opacity_7)
552
546
  );
553
-
554
- // Apply border colors for sticky columns
555
- @include apply-sticky-colors("light");
556
547
  }
557
548
 
558
549
  // Responsive Styles
@@ -589,6 +580,11 @@
589
580
  }
590
581
  }
591
582
 
583
+ // Row Pinning - additional inline styles in RegularTableView.tsx
584
+ .pinned-row {
585
+ box-shadow: 0 4px 10px 0 rgba($shadow, 0.16) !important;
586
+ }
587
+
592
588
  &.dark {
593
589
  // Override default border color for dark mode
594
590
  --column-border-color: #{$border_dark};
@@ -818,3 +814,8 @@
818
814
  }
819
815
  }
820
816
  }
817
+
818
+ // Outside of the pb_advanced_table class for popover
819
+ .pb-advanced-table-popover-option:hover {
820
+ background-color: $bg_light;
821
+ }
@@ -35,6 +35,7 @@ type AdvancedTableProps = {
35
35
  className?: string
36
36
  columnDefinitions: GenericObject[]
37
37
  columnGroupBorderColor?: "text_lt_default" | "text_lt_light" | "text_lt_lighter" | "text_dk_default" | "text_dk_light" | "text_dk_lighter"
38
+ columnVisibilityControl?: GenericObject
38
39
  dark?: boolean
39
40
  data?: { [key: string]: string }
40
41
  enableToggleExpansion?: "all" | "header" | "none"
@@ -50,7 +51,8 @@ type AdvancedTableProps = {
50
51
  onRowToggleClick?: (arg: Row<GenericObject>) => void
51
52
  onToggleExpansionClick?: (arg: Row<GenericObject>) => void
52
53
  pagination?: boolean,
53
- paginationProps?: GenericObject
54
+ paginationProps?: GenericObject,
55
+ pinnedRows?: any,
54
56
  responsive?: "scroll" | "none",
55
57
  selectableRows?: boolean,
56
58
  showActionsBar?: boolean,
@@ -73,6 +75,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
73
75
  className,
74
76
  columnDefinitions,
75
77
  columnGroupBorderColor,
78
+ columnVisibilityControl,
76
79
  dark = false,
77
80
  data = {},
78
81
  enableToggleExpansion = "header",
@@ -89,6 +92,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
89
92
  onToggleExpansionClick,
90
93
  pagination = false,
91
94
  paginationProps,
95
+ pinnedRows,
92
96
  responsive = "scroll",
93
97
  showActionsBar = true,
94
98
  selectableRows,
@@ -132,7 +136,9 @@ const AdvancedTable = (props: AdvancedTableProps) => {
132
136
  paginationProps,
133
137
  virtualizedRows,
134
138
  tableOptions,
135
- onRowSelectionChange
139
+ onRowSelectionChange,
140
+ columnVisibilityControl,
141
+ pinnedRows,
136
142
  });
137
143
 
138
144
  // Initialize table actions
@@ -238,7 +244,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
238
244
  maxHeight ? `advanced-table-max-height-${maxHeight}` : '',
239
245
  {
240
246
  'advanced-table-fullscreen': isFullscreen,
241
- 'advanced-table-allow-fullscreen': allowFullScreen
247
+ 'advanced-table-allow-fullscreen': allowFullScreen,
242
248
  },
243
249
  {'advanced-table-sticky-left-columns': stickyLeftColumn && stickyLeftColumn.length > 0},
244
250
  columnGroupBorderColor ? `column-group-border-${columnGroupBorderColor}` : '',
@@ -252,7 +258,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
252
258
  : {};
253
259
 
254
260
  // Visibility flag for action bar
255
- const isActionBarVisible = selectableRows && showActionsBar && selectedRowsLength > 0;
261
+ const isActionBarVisible = (selectableRows && showActionsBar && selectedRowsLength > 0) || columnVisibilityControl;
256
262
 
257
263
  return (
258
264
  <>
@@ -286,6 +292,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
286
292
  <AdvancedTableProvider
287
293
  columnDefinitions={columnDefinitions}
288
294
  columnGroupBorderColor={columnGroupBorderColor}
295
+ columnVisibilityControl={columnVisibilityControl}
289
296
  enableToggleExpansion={enableToggleExpansion}
290
297
  enableVirtualization={virtualizedRows}
291
298
  expandByDepth={expandByDepth}
@@ -298,6 +305,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
298
305
  isFullscreen={isFullscreen}
299
306
  loading={loading}
300
307
  onExpandByDepthClick={onExpandByDepthClick}
308
+ pinnedRows={pinnedRows}
301
309
  responsive={responsive}
302
310
  selectableRows={selectableRows}
303
311
  setExpanded={setExpanded}
@@ -316,6 +324,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
316
324
  actions={actions}
317
325
  isVisible={isActionBarVisible}
318
326
  selectedCount={selectedRowsLength}
327
+ type={columnVisibilityControl ? "column-visibility" : "row-selection"}
319
328
  />
320
329
 
321
330
  {/* Main Table */}
@@ -1,10 +1,18 @@
1
1
  <%= pb_content_tag do %>
2
- <%= pb_rails("table", props: { size: "sm", data_table: true, number_spacing:"tabular", responsive:"none", dark: dark, classname: object.loading ? "content-loading" : "" }.merge(object.table_props)) do %>
3
- <% if content.present? %>
4
- <% content.presence %>
5
- <% else %>
6
- <%= pb_rails("advanced_table/table_header", props: { column_definitions: object.column_definitions, enable_toggle_expansion: object.enable_toggle_expansion, responsive: object.responsive, loading: object.loading, selectable_rows: object.selectable_rows }) %>
7
- <%= pb_rails("advanced_table/table_body", props: { id: object.id, table_data: object.table_data, column_definitions: object.column_definitions, responsive: object.responsive, loading: object.loading, selectable_rows: object.selectable_rows, enable_toggle_expansion: object.enable_toggle_expansion }) %>
8
- <% end %>
9
- <% end %>
2
+ <% if object.id && object.selectable_rows && object.show_actions_bar %>
3
+ <%= pb_rails("advanced_table/table_action_bar", props: {
4
+ actions: object.actions,
5
+ is_visible: false,
6
+ selected_count: 0
7
+ }) %>
8
+ <% end %>
9
+
10
+ <%= pb_rails("table", props: { size: "sm", data_table: true, number_spacing:"tabular", responsive:"none", dark: dark, classname: object.loading ? "content-loading" : "" }.merge(object.table_props)) do %>
11
+ <% if content.present? %>
12
+ <% content.presence %>
13
+ <% else %>
14
+ <%= pb_rails("advanced_table/table_header", props: { column_definitions: object.column_definitions, enable_toggle_expansion: object.enable_toggle_expansion, responsive: object.responsive, loading: object.loading, selectable_rows: object.selectable_rows, show_actions_bar: object.show_actions_bar }) %>
15
+ <%= pb_rails("advanced_table/table_body", props: { id: object.id, table_data: object.table_data, column_definitions: object.column_definitions, responsive: object.responsive, loading: object.loading, selectable_rows: object.selectable_rows, enable_toggle_expansion: object.enable_toggle_expansion }) %>
16
+ <% end %>
17
+ <% end %>
10
18
  <% end %>
@@ -25,6 +25,10 @@ module Playbook
25
25
  default: "auto"
26
26
  prop :selectable_rows, type: Playbook::Props::Boolean,
27
27
  default: false
28
+ prop :show_actions_bar, type: Playbook::Props::Boolean,
29
+ default: true
30
+ prop :actions, type: Playbook::Props::Array,
31
+ default: []
28
32
 
29
33
  def classname
30
34
  additional_classes = [responsive_classname, max_height_classname]
@@ -47,6 +51,11 @@ module Playbook
47
51
  def selected_rows_length
48
52
  selected_rows.length
49
53
  end
54
+
55
+ def is_action_bar_visible
56
+ # Action bar visibility is controlled by JS based on selection
57
+ false
58
+ end
50
59
  end
51
60
  end
52
61
  end
@@ -3,6 +3,12 @@ import { render, screen, waitFor } from "../utilities/test-utils"
3
3
 
4
4
  import { AdvancedTable, Pill } from "playbook-ui"
5
5
 
6
+ global.ResizeObserver = class {
7
+ observe() {}
8
+ unobserve() {}
9
+ disconnect() {}
10
+ };
11
+
6
12
  const MOCK_DATA = [
7
13
  {
8
14
  year: "2021",
@@ -72,6 +78,36 @@ const MOCK_DATA_LOADING = [
72
78
  },
73
79
  ]
74
80
 
81
+ const MOCK_DATA_WITH_ID = [
82
+ {
83
+ id: "1",
84
+ year: "2021",
85
+ quarter: null,
86
+ month: null,
87
+ day: null,
88
+ newEnrollments: "20",
89
+ scheduledMeetings: "10",
90
+ },
91
+ {
92
+ id: "2",
93
+ year: "2022",
94
+ quarter: null,
95
+ month: null,
96
+ day: null,
97
+ newEnrollments: "25",
98
+ scheduledMeetings: "15",
99
+ },
100
+ {
101
+ id: "3",
102
+ year: "2023",
103
+ quarter: null,
104
+ month: null,
105
+ day: null,
106
+ newEnrollments: "30",
107
+ scheduledMeetings: "20",
108
+ },
109
+ ]
110
+
75
111
  const columnDefinitions = [
76
112
  {
77
113
  accessor: "year",
@@ -512,3 +548,28 @@ test("allowFullScreen prop adds fullscreen class", () => {
512
548
  const tableContainer = screen.getByRole("table").closest("div")
513
549
  expect(tableContainer).toHaveClass("advanced-table-allow-fullscreen")
514
550
  })
551
+
552
+ test("pinnedRows prop renders pinned rows at top", () => {
553
+ const pinnedRowsControl = {
554
+ value: { top: ["1", "3"] },
555
+ onChange: jest.fn()
556
+ }
557
+
558
+ render(
559
+ <AdvancedTable
560
+ columnDefinitions={columnDefinitions}
561
+ data={{ testid: testId }}
562
+ pinnedRows={pinnedRowsControl}
563
+ tableData={MOCK_DATA_WITH_ID}
564
+ />
565
+ )
566
+
567
+ const kit = screen.getByTestId(testId)
568
+ const pinnedRows = kit.querySelectorAll(".pinned-row")
569
+
570
+ expect(pinnedRows).toHaveLength(2)
571
+
572
+ const firstPinnedRow = pinnedRows[0]
573
+ expect(firstPinnedRow).toHaveStyle("position: sticky")
574
+ expect(firstPinnedRow).toHaveStyle("background-color: white")
575
+ })
@@ -1,4 +1,8 @@
1
- The AdvancedTable kit accepts tree data and automatically renders expansion controls for nested subrows, to any depth, based on the data it is given. In it's simplest form, __the kit has two required props__:
1
+ The AdvancedTable kit accepts tree data and automatically renders expansion controls for nested subrows, to any depth, based on the data it is given. In it's simplest form, __the kit has three required props__:
2
+
3
+ ### id
4
+
5
+ A unique `id` is required to allow the table functionality to work properly. Without it, certain functions like the action bar will not be able to properly reference the correct table.
2
6
 
3
7
  ### table_data
4
8
 
@@ -13,6 +17,6 @@ Column definitions are the single most important part of building a table as the
13
17
  - `accessor`: this is the key from your data for the value you want rendered in that column
14
18
  - `label`: this is what will be rendered as the column header label
15
19
 
16
- There is also one optional item that is only required if the table has nested data:
20
+ There is also one optional item that is only required if the table has nested data:
17
21
 
18
22
  - `cellAccessors`: This is an array of strings that represent keys from your data object. This is only required for the first column in case of nested data. If you have nested data, the AdvancedTable needs to know what to render in that first column for nested items. This array represents the nested data in the order you want it rendered.
@@ -0,0 +1,57 @@
1
+ import React from "react"
2
+ import AdvancedTable from '../../pb_advanced_table/_advanced_table'
3
+ import MOCK_DATA from "./advanced_table_mock_data.json"
4
+
5
+ const AdvancedTableColumnVisibility = (props) => {
6
+ const columnDefinitions = [
7
+ {
8
+ accessor: "year",
9
+ label: "Year",
10
+ cellAccessors: ["quarter", "month", "day"],
11
+ id: "year"
12
+ },
13
+ {
14
+ accessor: "newEnrollments",
15
+ label: "New Enrollments",
16
+ id: "newEnrollments"
17
+ },
18
+ {
19
+ accessor: "scheduledMeetings",
20
+ label: "Scheduled Meetings",
21
+ id: "scheduledMeetings"
22
+ },
23
+ {
24
+ accessor: "attendanceRate",
25
+ label: "Attendance Rate",
26
+ id: "attendanceRate"
27
+ },
28
+ {
29
+ accessor: "completedClasses",
30
+ label: "Completed Classes",
31
+ id: "completedClasses"
32
+ },
33
+ {
34
+ accessor: "classCompletionRate",
35
+ label: "Class Completion Rate",
36
+ id: "classCompletionRate"
37
+ },
38
+ {
39
+ accessor: "graduatedStudents",
40
+ label: "Graduated Students",
41
+ id: "graduatedStudents"
42
+ },
43
+ ]
44
+
45
+ return (
46
+ <div>
47
+ <AdvancedTable
48
+ columnDefinitions={columnDefinitions}
49
+ columnVisibilityControl={{default: true}}
50
+ tableData={MOCK_DATA}
51
+ {...props}
52
+ />
53
+ </div>
54
+ )
55
+ }
56
+
57
+ export default AdvancedTableColumnVisibility
@@ -0,0 +1,4 @@
1
+ The `columnVisibilityControl` prop allows users to toggle the visibility of table columns dynamically.
2
+
3
+ The default can be enabled simply by passing `{ default:true }` to the prop as shown. This will render the header with the icon enabled dropdown. The dropdown contains all columns present in the Table and any can be toggled on or off via the checkboxes.
4
+ **NOTE**: The first column will not be shown in the dropdown as an option since all the expansion logic/functionality lives there and it should always be visible.
@@ -0,0 +1,62 @@
1
+ import React from "react"
2
+ import AdvancedTable from '../../pb_advanced_table/_advanced_table'
3
+ import MOCK_DATA from "./advanced_table_mock_data.json"
4
+
5
+ const AdvancedTableColumnVisibilityCustom = (props) => {
6
+ const columnDefinitions = [
7
+ {
8
+ accessor: "year",
9
+ label: "Year",
10
+ cellAccessors: ["quarter", "month", "day"],
11
+ id: "year"
12
+ },
13
+ {
14
+ accessor: "newEnrollments",
15
+ label: "New Enrollments",
16
+ id: "newEnrollments"
17
+ },
18
+ {
19
+ accessor: "scheduledMeetings",
20
+ label: "Scheduled Meetings",
21
+ id: "scheduledMeetings"
22
+ },
23
+ {
24
+ accessor: "attendanceRate",
25
+ label: "Attendance Rate",
26
+ id: "attendanceRate"
27
+ },
28
+ {
29
+ accessor: "completedClasses",
30
+ label: "Completed Classes",
31
+ id: "completedClasses"
32
+ },
33
+ {
34
+ accessor: "classCompletionRate",
35
+ label: "Class Completion Rate",
36
+ id: "classCompletionRate"
37
+ },
38
+ {
39
+ accessor: "graduatedStudents",
40
+ label: "Graduated Students",
41
+ id: "graduatedStudents"
42
+ },
43
+ ]
44
+
45
+ const columnVisibilityControl = {
46
+ // This is the list of column ids that will be included in the column visibility control
47
+ includeIds:["newEnrollments", "scheduledMeetings", "attendanceRate", "completedClasses"],
48
+ }
49
+
50
+ return (
51
+ <div>
52
+ <AdvancedTable
53
+ columnDefinitions={columnDefinitions}
54
+ columnVisibilityControl={columnVisibilityControl}
55
+ tableData={MOCK_DATA}
56
+ {...props}
57
+ />
58
+ </div>
59
+ )
60
+ }
61
+
62
+ export default AdvancedTableColumnVisibilityCustom
@@ -0,0 +1 @@
1
+ By using the `includeIds` key/value pair as shown within the `columnVisibilityControl` prop, you can control which columns show up as options in the columnVisibility dropdown.
@@ -0,0 +1,82 @@
1
+ import React from "react"
2
+ import AdvancedTable from '../../pb_advanced_table/_advanced_table'
3
+ import MOCK_DATA from "./advanced_table_mock_data.json"
4
+
5
+ const AdvancedTableColumnVisibilityMulti = (props) => {
6
+ const columnDefinitions = [
7
+ {
8
+ accessor: "year",
9
+ label: "Year",
10
+ id: "year",
11
+ cellAccessors: ["quarter", "month", "day"],
12
+ },
13
+ {
14
+ label: "Enrollment Data",
15
+ id: "enrollmentData",
16
+ columns: [
17
+ {
18
+ label: "Enrollment Stats",
19
+ id: "enrollmentStats",
20
+ columns: [
21
+ {
22
+ accessor: "newEnrollments",
23
+ label: "New Enrollments",
24
+ id: "newEnrollments",
25
+ },
26
+ {
27
+ accessor: "scheduledMeetings",
28
+ label: "Scheduled Meetings",
29
+ id: "scheduledMeetings",
30
+ },
31
+ ],
32
+ },
33
+ ],
34
+ },
35
+ {
36
+ label: "Performance Data",
37
+ id: "performanceData",
38
+ columns: [
39
+ {
40
+ label: "Completion Metrics",
41
+ id: "completionMetrics",
42
+ columns: [
43
+ {
44
+ accessor: "completedClasses",
45
+ label: "Completed Classes",
46
+ id: "completedClasses",
47
+ },
48
+ {
49
+ accessor: "classCompletionRate",
50
+ label: "Class Completion Rate",
51
+ id: "classCompletionRate",
52
+ },
53
+ ],
54
+ },
55
+ {
56
+ label: "Attendance",
57
+ id: "attendance",
58
+ columns: [
59
+ {
60
+ accessor: "attendanceRate",
61
+ label: "Attendance Rate",
62
+ id: "attendanceRate",
63
+ },
64
+ ],
65
+ },
66
+ ],
67
+ },
68
+ ];
69
+
70
+ return (
71
+ <div>
72
+ <AdvancedTable
73
+ columnDefinitions={columnDefinitions}
74
+ columnVisibilityControl={{default: true}}
75
+ tableData={MOCK_DATA}
76
+ {...props}
77
+ />
78
+ </div>
79
+ )
80
+ }
81
+
82
+ export default AdvancedTableColumnVisibilityMulti
@@ -0,0 +1 @@
1
+ The `columnVisibilityControl` prop can also be used with multi-header columns as shown.
@@ -0,0 +1,66 @@
1
+ import React, { useState } from "react"
2
+ import AdvancedTable from '../../pb_advanced_table/_advanced_table'
3
+ import MOCK_DATA from "./advanced_table_mock_data.json"
4
+
5
+ const AdvancedTableColumnVisibilityWithState = (props) => {
6
+ const columnDefinitions = [
7
+ {
8
+ accessor: "year",
9
+ label: "Year",
10
+ cellAccessors: ["quarter", "month", "day"],
11
+ id: "year"
12
+ },
13
+ {
14
+ accessor: "newEnrollments",
15
+ label: "New Enrollments",
16
+ id: "newEnrollments"
17
+ },
18
+ {
19
+ accessor: "scheduledMeetings",
20
+ label: "Scheduled Meetings",
21
+ id: "scheduledMeetings"
22
+ },
23
+ {
24
+ accessor: "attendanceRate",
25
+ label: "Attendance Rate",
26
+ id: "attendanceRate"
27
+ },
28
+ {
29
+ accessor: "completedClasses",
30
+ label: "Completed Classes",
31
+ id: "completedClasses"
32
+ },
33
+ {
34
+ accessor: "classCompletionRate",
35
+ label: "Class Completion Rate",
36
+ id: "classCompletionRate"
37
+ },
38
+ {
39
+ accessor: "graduatedStudents",
40
+ label: "Graduated Students",
41
+ id: "graduatedStudents"
42
+ },
43
+ ]
44
+
45
+ const [columnVisibility, setColumnVisibility] = useState({
46
+ newEnrollments: false
47
+ })
48
+
49
+ const columnVisibilityControl = {
50
+ value: columnVisibility,
51
+ onChange: setColumnVisibility,
52
+ onColumnVisibilityChange: (currentState) => console.log(currentState),
53
+ }
54
+ return (
55
+ <div>
56
+ <AdvancedTable
57
+ columnDefinitions={columnDefinitions}
58
+ columnVisibilityControl={columnVisibilityControl}
59
+ tableData={MOCK_DATA}
60
+ {...props}
61
+ />
62
+ </div>
63
+ )
64
+ }
65
+
66
+ export default AdvancedTableColumnVisibilityWithState
@@ -0,0 +1,3 @@
1
+ The `columnVisibilityControl` prop also allows for greater control over the columnVisibility state. Devs can manage state themselves by passing in `value` and `onChange` as shown.
2
+
3
+ The additional `onColumnVisibilityChange` provides a callback with the current state as the argument if needed.
@@ -3,7 +3,7 @@
3
3
  <br />
4
4
  <br />
5
5
 
6
- The AdvancedTable kit accepts tree data and automatically renders expansion controls for nested subrows, to any depth, based on the data it is given. In it's simplest form, __the kit has two required props__:
6
+ The AdvancedTable kit accepts tree data and automatically renders expansion controls for nested subrows, to any depth, based on the data it is given. In it's simplest form, __the kit has two required props__:
7
7
 
8
8
  ### tableData
9
9