playbook_ui 15.2.0.pre.alpha.PLAY2589advancedtableinlinerowloadingtoggleicon11607 → 15.2.0.pre.alpha.PLAY2589advancedtableinlinerowloadingtoggleicon11641

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 (28) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/pb_advanced_table/Components/TableHeaderCell.tsx +3 -1
  3. data/app/pb_kits/playbook/pb_advanced_table/SubKits/TableHeader.tsx +3 -0
  4. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.tsx +3 -0
  5. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.test.jsx +59 -0
  6. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_inline_row_loading_show_toggle.jsx +59 -0
  7. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_inline_row_loading_show_toggle.md +5 -0
  8. data/app/pb_kits/playbook/pb_advanced_table/docs/_mock_data_inline_loading_empty_children.js +202 -0
  9. data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +1 -0
  10. data/app/pb_kits/playbook/pb_advanced_table/docs/index.js +2 -1
  11. data/app/pb_kits/playbook/pb_phone_number_input/_phone_number_input.tsx +25 -6
  12. data/app/pb_kits/playbook/pb_typeahead/_typeahead.test.jsx +35 -1
  13. data/app/pb_kits/playbook/pb_typeahead/_typeahead.tsx +71 -2
  14. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_options.html.erb +36 -0
  15. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_options.md +1 -0
  16. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_value.jsx +41 -0
  17. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_value.md +1 -0
  18. data/app/pb_kits/playbook/pb_typeahead/docs/example.yml +2 -0
  19. data/app/pb_kits/playbook/pb_typeahead/docs/index.js +1 -0
  20. data/dist/chunks/{_line_graph-BnVgr42C.js → _line_graph-D5MBnrO9.js} +1 -1
  21. data/dist/chunks/{_typeahead-BH_dkgOy.js → _typeahead-BjYBazGq.js} +1 -1
  22. data/dist/chunks/{_weekday_stacked-Duv09EWo.js → _weekday_stacked-B_Uc7-rO.js} +2 -2
  23. data/dist/chunks/vendor.js +1 -1
  24. data/dist/playbook-doc.js +2 -2
  25. data/dist/playbook-rails-react-bindings.js +1 -1
  26. data/dist/playbook-rails.js +1 -1
  27. data/lib/playbook/version.rb +1 -1
  28. metadata +12 -5
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: fcd436ebf1d8c2a0442f88c7bd24280b90e805a2cbf764d36e60a73bf04f8a3e
4
- data.tar.gz: 57f069f6198858056e3217cd8032a03a41a78204bea676892f9cdc4ccd1fb4e8
3
+ metadata.gz: 26ef9154b98ad4d5d762b37915fa398314c6020fc3cf0c0b42b9e074fd55f569
4
+ data.tar.gz: 5a79125ffb800db93c0ebe22ad143167bac0f9c018ebc33beba7a1c2049ec3fc
5
5
  SHA512:
6
- metadata.gz: 39bf03e47685ed9c93b72ca94fc4b47a3decb6de1c61b690db4f84295595a5d3375378f83c37fab02c72bbecd4feea2bb792861c19efe4ce4201901907d2b2b2
7
- data.tar.gz: d95721f238cf35d91464c1efe1e6f7aa0852a5e17facd89dc03c45c45c431eb5f80924ea9e0c261955bc596860d973f11814692b711a33bc95a1a02baff033f4
6
+ metadata.gz: 812c93c1d3667ad67d5e7d810b5d4a399c3ac6d11c003b231b32ed8dafb8742e4097b37b215f8a7ea9d5474626b57d8ead7dc11d60483b72a701efeecc9c3063
7
+ data.tar.gz: d0eeafe285857f375b45699aed9f1d4a25915c99b50e6a50d148937b4de0ce6e2d99a67e986305d83ebf6e36a4d67b3e9daed119a0138ae1f3cf7ca858d04cd2
@@ -30,6 +30,7 @@ type TableHeaderCellProps = {
30
30
  headerChildren?: React.ReactNode | React.ReactNode[]
31
31
  isPinnedLeft?: boolean
32
32
  loading?: boolean
33
+ showToggleWithInlineRowLoading?: boolean
33
34
  sortIcon?: string | string[]
34
35
  table?: Table<GenericObject>
35
36
  } & GlobalProps
@@ -58,6 +59,7 @@ export const TableHeaderCell = ({
58
59
  selectableRows,
59
60
  hasAnySubRows,
60
61
  showActionsBar,
62
+ showToggleWithInlineRowLoading,
61
63
  stickyLeftColumn,
62
64
  inlineRowLoading,
63
65
  isActionBarVisible,
@@ -220,7 +222,7 @@ const isToggleExpansionEnabled =
220
222
  />
221
223
  )
222
224
  }
223
- {isToggleExpansionEnabled && (hasAnySubRows || inlineRowLoading) && !expandByDepth && (
225
+ {isToggleExpansionEnabled && ((hasAnySubRows) || (inlineRowLoading && showToggleWithInlineRowLoading)) && !expandByDepth && (
224
226
  <ToggleIconButton onClick={handleExpandOrCollapse} />
225
227
  )}
226
228
  {isToggleExpansionEnabled && hasAnySubRows && expandByDepth && (
@@ -39,6 +39,7 @@ export const TableHeader = ({
39
39
  hasAnySubRows,
40
40
  showActionsBar,
41
41
  selectableRows,
42
+ showToggleWithInlineRowLoading,
42
43
  responsive,
43
44
  headerRef,
44
45
  virtualizedRows,
@@ -92,6 +93,7 @@ export const TableHeader = ({
92
93
  isPinnedLeft={isPinnedLeft}
93
94
  key={`${header.id}-header`}
94
95
  loading={loading}
96
+ showToggleWithInlineRowLoading={showToggleWithInlineRowLoading}
95
97
  sortIcon={sortIcon}
96
98
  table={table}
97
99
  />
@@ -136,6 +138,7 @@ export const TableHeader = ({
136
138
  isVirtualized
137
139
  key={`${header.id}-header-virtualized`}
138
140
  loading={loading}
141
+ showToggleWithInlineRowLoading={showToggleWithInlineRowLoading}
139
142
  sortIcon={sortIcon}
140
143
  table={table}
141
144
  />
@@ -63,6 +63,7 @@ type AdvancedTableProps = {
63
63
  scrollBarNone?: boolean,
64
64
  selectableRows?: boolean,
65
65
  showActionsBar?: boolean,
66
+ showToggleWithInlineRowLoading?: boolean,
66
67
  sortControl?: GenericObject
67
68
  tableData: GenericObject[]
68
69
  tableOptions?: GenericObject
@@ -109,6 +110,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
109
110
  scrollBarNone= false,
110
111
  showActionsBar = true,
111
112
  selectableRows,
113
+ showToggleWithInlineRowLoading = false,
112
114
  sortControl,
113
115
  stickyLeftColumn,
114
116
  tableData,
@@ -355,6 +357,7 @@ const AdvancedTable = (props: AdvancedTableProps) => {
355
357
  selectableRows={selectableRows}
356
358
  setExpanded={setExpanded}
357
359
  showActionsBar={showActionsBar}
360
+ showToggleWithInlineRowLoading={showToggleWithInlineRowLoading}
358
361
  sortControl={sortControl}
359
362
  stickyLeftColumn={stickyLeftColumn}
360
363
  subRowHeaders={tableOptions?.subRowHeaders}
@@ -108,6 +108,48 @@ const MOCK_DATA_WITH_ID = [
108
108
  },
109
109
  ]
110
110
 
111
+ const MOCK_DATA_NO_SUBROWS = [
112
+ {
113
+ year: "2021",
114
+ quarter: null,
115
+ month: null,
116
+ day: null,
117
+ newEnrollments: "20",
118
+ scheduledMeetings: "10",
119
+ attendanceRate: "51%",
120
+ completedClasses: "3",
121
+ classCompletionRate: "33%",
122
+ graduatedStudents: "19",
123
+ children: [],
124
+ },
125
+ {
126
+ year: "2022",
127
+ quarter: null,
128
+ month: null,
129
+ day: null,
130
+ newEnrollments: "25",
131
+ scheduledMeetings: "17",
132
+ attendanceRate: "75%",
133
+ completedClasses: "5",
134
+ classCompletionRate: "45%",
135
+ graduatedStudents: "32",
136
+ children: [],
137
+ },
138
+ {
139
+ year: "2023",
140
+ quarter: null,
141
+ month: null,
142
+ day: null,
143
+ newEnrollments: "30",
144
+ scheduledMeetings: "22",
145
+ attendanceRate: "80%",
146
+ completedClasses: "7",
147
+ classCompletionRate: "55%",
148
+ graduatedStudents: "45",
149
+ children: [],
150
+ },
151
+ ]
152
+
111
153
  const columnDefinitions = [
112
154
  {
113
155
  accessor: "year",
@@ -509,6 +551,23 @@ test("inlineRowLoading prop renders inline loading if true", () => {
509
551
  expect(inlineLoading).toBeInTheDocument()
510
552
  })
511
553
 
554
+ test("showToggleWithInlineRowLoading prop shows header toggle when inlineRowLoading is true", () => {
555
+ render(
556
+ <AdvancedTable
557
+ columnDefinitions={columnDefinitions}
558
+ data={{ testid: testId }}
559
+ enableToggleExpansion="all"
560
+ inlineRowLoading
561
+ showToggleWithInlineRowLoading
562
+ tableData={MOCK_DATA_NO_SUBROWS}
563
+ />
564
+ )
565
+
566
+ const kit = screen.getByTestId(testId)
567
+ const headerToggleButton = kit.querySelector(".gray-icon.toggle-all-icon")
568
+ expect(headerToggleButton).toBeInTheDocument()
569
+ })
570
+
512
571
  test("responsive prop functions as expected", () => {
513
572
  render(
514
573
  <AdvancedTable
@@ -0,0 +1,59 @@
1
+ import React from "react"
2
+ import AdvancedTable from '../_advanced_table'
3
+ import { MOCK_DATA_INLINE_LOADING_EMPTY_CHILDREN } from "./_mock_data_inline_loading_empty_children"
4
+
5
+ const AdvancedTableInlineRowLoadingShowToggle = (props) => {
6
+ const columnDefinitions = [
7
+ {
8
+ accessor: "year",
9
+ label: "Year",
10
+ cellAccessors: ["quarter", "month", "day"],
11
+ },
12
+ {
13
+ accessor: "newEnrollments",
14
+ label: "New Enrollments",
15
+ },
16
+ {
17
+ accessor: "scheduledMeetings",
18
+ label: "Scheduled Meetings",
19
+ },
20
+ {
21
+ accessor: "attendanceRate",
22
+ label: "Attendance Rate",
23
+ },
24
+ {
25
+ accessor: "completedClasses",
26
+ label: "Completed Classes",
27
+ },
28
+ {
29
+ accessor: "classCompletionRate",
30
+ label: "Class Completion Rate",
31
+ },
32
+ {
33
+ accessor: "graduatedStudents",
34
+ label: "Graduated Students",
35
+ },
36
+ ]
37
+
38
+ //Render the subRow header rows
39
+ const subRowHeaders = ["Quarter", "Month", "Day"]
40
+
41
+
42
+ return (
43
+ <div>
44
+ <AdvancedTable
45
+ columnDefinitions={columnDefinitions}
46
+ enableToggleExpansion="all"
47
+ inlineRowLoading
48
+ showToggleWithInlineRowLoading
49
+ tableData={MOCK_DATA_INLINE_LOADING_EMPTY_CHILDREN}
50
+ {...props}
51
+ >
52
+ <AdvancedTable.Header />
53
+ <AdvancedTable.Body subRowHeaders={subRowHeaders}/>
54
+ </AdvancedTable>
55
+ </div>
56
+ )
57
+ }
58
+
59
+ export default AdvancedTableInlineRowLoadingShowToggle
@@ -0,0 +1,5 @@
1
+ The `showToggleWithInlineRowLoading` is a boolean prop that renders the toggle-all icon in the top left header cell for complex datasets with enoty `children` arrays and advanced querying logic explained in the preceeding doc example. Your logic may require an additional query helper file to update data specifically from requerying via toggle all buttons.
2
+
3
+ In this code example, all 3 rows have empty children arrays. The toggle all button would not render (prior to an initial row expansion) without `showToggleWithInlineRowLoading` in place.
4
+
5
+ This prop is set to false by default and should only be used in conjunction with `inlineRowLoading`.
@@ -0,0 +1,202 @@
1
+ export const MOCK_DATA_INLINE_LOADING_EMPTY_CHILDREN = [
2
+ {
3
+ year: "2021",
4
+ quarter: null,
5
+ month: null,
6
+ day: null,
7
+ newEnrollments: "20",
8
+ scheduledMeetings: "10",
9
+ attendanceRate: "51%",
10
+ completedClasses: "3",
11
+ classCompletionRate: "33%",
12
+ graduatedStudents: "19",
13
+ children: [],
14
+ },
15
+ {
16
+ year: "2022",
17
+ quarter: null,
18
+ month: null,
19
+ day: null,
20
+ newEnrollments: "25",
21
+ scheduledMeetings: "17",
22
+ attendanceRate: "75%",
23
+ completedClasses: "5",
24
+ classCompletionRate: "45%",
25
+ graduatedStudents: "32",
26
+ children: [],
27
+ // children: [
28
+ // {
29
+ // year: "2022",
30
+ // quarter: "Q1",
31
+ // month: null,
32
+ // day: null,
33
+ // newEnrollments: "2",
34
+ // scheduledMeetings: "35",
35
+ // attendanceRate: "32%",
36
+ // completedClasses: "15",
37
+ // classCompletionRate: "52%",
38
+ // graduatedStudents: "36",
39
+ // children: [
40
+ // {
41
+ // year: "2022",
42
+ // quarter: "Q1",
43
+ // month: "January",
44
+ // day: null,
45
+ // newEnrollments: "16",
46
+ // scheduledMeetings: "20",
47
+ // attendanceRate: "11%",
48
+ // completedClasses: "13",
49
+ // classCompletionRate: "47%",
50
+ // graduatedStudents: "28",
51
+ // children: [
52
+ // {
53
+ // year: "2022",
54
+ // quarter: "Q1",
55
+ // month: "January",
56
+ // day: "15",
57
+ // newEnrollments: "34",
58
+ // scheduledMeetings: "28",
59
+ // attendanceRate: "97%",
60
+ // completedClasses: "20",
61
+ // classCompletionRate: "15%",
62
+ // graduatedStudents: "17",
63
+ // },
64
+ // {
65
+ // year: "2022",
66
+ // quarter: "Q1",
67
+ // month: "January",
68
+ // day: "25",
69
+ // newEnrollments: "43",
70
+ // scheduledMeetings: "23",
71
+ // attendanceRate: "66%",
72
+ // completedClasses: "26",
73
+ // classCompletionRate: "47%",
74
+ // graduatedStudents: "9",
75
+ // },
76
+ // ],
77
+ // },
78
+ // {
79
+ // year: "2022",
80
+ // quarter: "Q1",
81
+ // month: "May",
82
+ // day: null,
83
+ // newEnrollments: "20",
84
+ // scheduledMeetings: "41",
85
+ // attendanceRate: "95%",
86
+ // completedClasses: "26",
87
+ // classCompletionRate: "83%",
88
+ // graduatedStudents: "43",
89
+ // children: [
90
+ // {
91
+ // year: "2011",
92
+ // quarter: "Q1",
93
+ // month: "May",
94
+ // day: "2",
95
+ // newEnrollments: "19",
96
+ // scheduledMeetings: "35",
97
+ // attendanceRate: "69%",
98
+ // completedClasses: "8",
99
+ // classCompletionRate: "75%",
100
+ // graduatedStudents: "23",
101
+ // },
102
+ // ],
103
+ // },
104
+ // ],
105
+ // },
106
+ // ],
107
+ },
108
+ {
109
+ year: "2023",
110
+ quarter: null,
111
+ month: null,
112
+ day: null,
113
+ newEnrollments: "10",
114
+ scheduledMeetings: "15",
115
+ attendanceRate: "65%",
116
+ completedClasses: "4",
117
+ classCompletionRate: "49%",
118
+ graduatedStudents: "29",
119
+ children: [],
120
+ // children: [
121
+ // {
122
+ // year: "2023",
123
+ // quarter: "Q1",
124
+ // month: null,
125
+ // day: null,
126
+ // newEnrollments: "2",
127
+ // scheduledMeetings: "35",
128
+ // attendanceRate: "32%",
129
+ // completedClasses: "15",
130
+ // classCompletionRate: "52%",
131
+ // graduatedStudents: "36",
132
+ // children: [
133
+ // {
134
+ // year: "2023",
135
+ // quarter: "Q1",
136
+ // month: "March",
137
+ // day: null,
138
+ // newEnrollments: "16",
139
+ // scheduledMeetings: "20",
140
+ // attendanceRate: "11%",
141
+ // completedClasses: "13",
142
+ // classCompletionRate: "47%",
143
+ // graduatedStudents: "28",
144
+ // children: [
145
+ // {
146
+ // year: "2023",
147
+ // quarter: "Q1",
148
+ // month: "March",
149
+ // day: "10",
150
+ // newEnrollments: "34",
151
+ // scheduledMeetings: "28",
152
+ // attendanceRate: "97%",
153
+ // completedClasses: "20",
154
+ // classCompletionRate: "15%",
155
+ // graduatedStudents: "17",
156
+ // },
157
+ // {
158
+ // year: "2023",
159
+ // quarter: "Q1",
160
+ // month: "March",
161
+ // day: "11",
162
+ // newEnrollments: "43",
163
+ // scheduledMeetings: "23",
164
+ // attendanceRate: "66%",
165
+ // completedClasses: "26",
166
+ // classCompletionRate: "47%",
167
+ // graduatedStudents: "9",
168
+ // },
169
+ // ],
170
+ // },
171
+ // {
172
+ // year: "2023",
173
+ // quarter: "Q1",
174
+ // month: "April",
175
+ // day: null,
176
+ // newEnrollments: "20",
177
+ // scheduledMeetings: "41",
178
+ // attendanceRate: "95%",
179
+ // completedClasses: "26",
180
+ // classCompletionRate: "83%",
181
+ // graduatedStudents: "43",
182
+ // children: [
183
+ // {
184
+ // year: "2023",
185
+ // quarter: "Q1",
186
+ // month: "April",
187
+ // day: "15",
188
+ // newEnrollments: "19",
189
+ // scheduledMeetings: "35",
190
+ // attendanceRate: "69%",
191
+ // completedClasses: "8",
192
+ // classCompletionRate: "75%",
193
+ // graduatedStudents: "23",
194
+ // },
195
+ // ],
196
+ // },
197
+ // ],
198
+ // },
199
+ // ],
200
+ },
201
+ ]
202
+
@@ -54,6 +54,7 @@ examples:
54
54
  - advanced_table_pagination_with_props: Pagination Props
55
55
  - advanced_table_loading: Loading State
56
56
  - advanced_table_inline_row_loading: Inline Row Loading
57
+ - advanced_table_inline_row_loading_show_toggle: Inline Row Loading with Show Toggle
57
58
  - advanced_table_column_headers: Multi-Header Columns
58
59
  - advanced_table_column_headers_multiple: Multi-Header Columns (Multiple Levels)
59
60
  - advanced_table_column_headers_custom_cell: Multi-Header Columns with Custom Cells
@@ -45,4 +45,5 @@ export { default as AdvancedTableWithCustomHeaderMultiHeader } from './_advanced
45
45
  export { default as AdvancedTableSortPerColumn } from './_advanced_table_sort_per_column.jsx'
46
46
  export { default as AdvancedTableSortPerColumnForMultiColumn } from './_advanced_table_sort_per_column_for_multi_column.jsx'
47
47
  export { default as AdvancedTablePaddingControl } from './_advanced_table_padding_control.jsx'
48
- export { default as AdvancedTablePaddingControlPerRow } from './_advanced_table_padding_control_per_row.jsx'
48
+ export { default as AdvancedTablePaddingControlPerRow } from './_advanced_table_padding_control_per_row.jsx'
49
+ export { default as AdvancedTableInlineRowLoadingShowToggle } from './_advanced_table_inline_row_loading_show_toggle.jsx'
@@ -314,14 +314,18 @@ const PhoneNumberInput = (props: PhoneNumberInputProps, ref?: React.Ref<unknown>
314
314
 
315
315
  const handleOnChange = (evt: React.ChangeEvent<HTMLInputElement>) => {
316
316
  if (!hasTyped) setHasTyped(true)
317
+
317
318
  setInputValue(evt.target.value)
319
+
318
320
  let phoneNumberData
321
+
319
322
  if (formatAsYouType) {
320
323
  const formattedPhoneNumberData = getCurrentSelectedData(itiRef.current, evt.target.value)
321
324
  phoneNumberData = {...formattedPhoneNumberData, number: unformatNumber(formattedPhoneNumberData.number)}
322
325
  } else {
323
326
  phoneNumberData = getCurrentSelectedData(itiRef.current, evt.target.value)
324
327
  }
328
+
325
329
  setSelectedData(phoneNumberData)
326
330
  onChange(phoneNumberData)
327
331
  isValid(itiRef.current.isValidNumber())
@@ -370,11 +374,26 @@ const PhoneNumberInput = (props: PhoneNumberInputProps, ref?: React.Ref<unknown>
370
374
 
371
375
  inputRef.current.addEventListener("open:countrydropdown", () => setDropDownIsOpen(true))
372
376
  inputRef.current.addEventListener("close:countrydropdown", () => setDropDownIsOpen(false))
373
- }
374
- if (formatAsYouType) {
375
- inputRef.current?.addEventListener("input", (evt) => {
376
- handleOnChange(evt as unknown as React.ChangeEvent<HTMLInputElement>);
377
- });
377
+
378
+ // Handle formatAsYouType with input event
379
+ if (formatAsYouType) {
380
+ inputRef.current.addEventListener("input", (evt: Event) => {
381
+ const target = evt.target as HTMLInputElement
382
+ const formattedValue = target.value
383
+
384
+ // Update internal state
385
+ setInputValue(formattedValue)
386
+ setHasTyped(true)
387
+
388
+ // Get phone number data with unformatted number
389
+ const formattedPhoneNumberData = getCurrentSelectedData(telInputInit, formattedValue)
390
+ const phoneNumberData = {...formattedPhoneNumberData, number: unformatNumber(formattedPhoneNumberData.number)}
391
+
392
+ setSelectedData(phoneNumberData)
393
+ onChange(phoneNumberData)
394
+ isValid(telInputInit.isValidNumber())
395
+ })
396
+ }
378
397
  }
379
398
  }, [])
380
399
 
@@ -389,7 +408,7 @@ const PhoneNumberInput = (props: PhoneNumberInputProps, ref?: React.Ref<unknown>
389
408
  label,
390
409
  name,
391
410
  onBlur: validateErrors,
392
- onChange: handleOnChange,
411
+ onChange: formatAsYouType ? undefined : handleOnChange,
393
412
  value: inputValue
394
413
  }
395
414
 
@@ -1,5 +1,5 @@
1
1
  import React from 'react'
2
- import { render, screen } from '../utilities/test-utils'
2
+ import { render, screen, fireEvent, waitFor } from '../utilities/test-utils'
3
3
  import Typeahead from './_typeahead'
4
4
 
5
5
  const options = [
@@ -137,4 +137,38 @@ test('typeahead with colored pills', () => {
137
137
  const kit = screen.getByTestId('pills-color-test')
138
138
  const pill = kit.querySelector(".pb_form_pill_kit.pb_form_pill_neutral")
139
139
  expect(pill).toBeInTheDocument()
140
+ })
141
+
142
+ test('typeahead with defaultValue with focus behavior', async () => {
143
+ render(
144
+ <Typeahead
145
+ data={{ testid: 'default-value-focus-test' }}
146
+ defaultValue={[options[1]]}
147
+ options={options}
148
+ />
149
+ )
150
+
151
+ const kit = screen.getByTestId('default-value-focus-test')
152
+ const inputDiv = kit.querySelector(".typeahead-kit-select__single-value")
153
+ expect(inputDiv).toHaveTextContent("Red")
154
+
155
+ // Test that the control can receive focus
156
+ const control = kit.querySelector('.typeahead-kit-select__control')
157
+ expect(control).toBeInTheDocument()
158
+
159
+ // Simulate opening the menu by clicking the control
160
+ fireEvent.mouseDown(control)
161
+
162
+ // Wait for menu to appear
163
+ await waitFor(() => {
164
+ const menu = kit.querySelector('.typeahead-kit-select__menu')
165
+ expect(menu).toBeInTheDocument()
166
+ })
167
+
168
+ // Check that the correct option has the focused class
169
+ await waitFor(() => {
170
+ const focusedOption = kit.querySelector('.typeahead-kit-select__option--is-focused')
171
+ expect(focusedOption).toBeInTheDocument()
172
+ expect(focusedOption).toHaveTextContent('Red')
173
+ })
140
174
  })
@@ -1,4 +1,4 @@
1
- import React, { useState, useEffect, forwardRef} from 'react'
1
+ import React, { useState, useEffect, forwardRef, useRef} from 'react'
2
2
  import Select from 'react-select'
3
3
  import AsyncSelect from 'react-select/async'
4
4
  import CreateableSelect from 'react-select/creatable'
@@ -106,6 +106,8 @@ const Typeahead = forwardRef<HTMLInputElement, TypeaheadProps>(({
106
106
  const [inputValue, setInputValue] = useState("")
107
107
  // State to track if form has been submitted to control validation display for react rendered rails kit
108
108
  const [formSubmitted, setFormSubmitted] = useState(false)
109
+ // State to track if user has made a selection (to disable defaultValue focus behavior)
110
+ const [hasUserSelected, setHasUserSelected] = useState(false)
109
111
 
110
112
  // If preserveSearchInput is true, we need to control the input value
111
113
  const handleInputChange = preserveSearchInput
@@ -139,6 +141,69 @@ const Typeahead = forwardRef<HTMLInputElement, TypeaheadProps>(({
139
141
  }
140
142
  : props.onBlur
141
143
 
144
+ // Create a ref to access React Select instance
145
+ const selectRef = useRef<any>(null)
146
+
147
+ // Configure focus on selected option using React Select's API
148
+ const handleMenuOpen = () => {
149
+ setTimeout(() => {
150
+ let currentValue = props.value || props.defaultValue
151
+
152
+ // Handle react rendered rails version which passes arrays even for single selects
153
+ if (Array.isArray(currentValue) && currentValue.length > 0) {
154
+ currentValue = currentValue[0]
155
+ }
156
+
157
+ // Only apply custom focus if user has NOT made a selection yet
158
+ if (currentValue && selectRef.current && !hasUserSelected && !props.isMulti) {
159
+
160
+ const options = props.options
161
+ if (options) {
162
+ // Find the index of the current value
163
+ const focusedIndex = options.findIndex((option: any) => {
164
+ const optionValue = props.getOptionValue ? props.getOptionValue(option) : option.value
165
+ const currentOptionValue = props.getOptionValue ? props.getOptionValue(currentValue) : currentValue.value
166
+ return optionValue === currentOptionValue
167
+ })
168
+
169
+ if (focusedIndex >= 0 && options[focusedIndex]) {
170
+ // Use React Select's internal state to set focused option
171
+ if (selectRef.current && selectRef.current.setState) {
172
+ const targetOption = options[focusedIndex]
173
+ selectRef.current.setState({
174
+ focusedOption: targetOption,
175
+ focusedValue: null
176
+ })
177
+
178
+ // Handle scrolling so selected option is visible
179
+ setTimeout(() => {
180
+ if (selectRef.current && selectRef.current.menuListRef) {
181
+ const menuElement = selectRef.current.menuListRef
182
+ if (menuElement && menuElement.children && menuElement.children[focusedIndex]) {
183
+ // Calculate the position of the selected option and scroll the menu container
184
+ const optionElement = menuElement.children[focusedIndex] as HTMLElement
185
+ const optionTop = optionElement.offsetTop
186
+ const optionHeight = optionElement.offsetHeight
187
+ const menuHeight = menuElement.clientHeight
188
+
189
+ // Set the menu's scrollTop to position the selected option in the middle
190
+ const scrollToMiddle = optionTop - (menuHeight / 2) + (optionHeight / 2)
191
+ menuElement.scrollTop = Math.max(0, scrollToMiddle)
192
+ }
193
+ }
194
+ }, 20)
195
+ }
196
+ }
197
+ }
198
+ }
199
+ }, 0)
200
+
201
+ // Call original onMenuOpen if provided
202
+ if (props.onMenuOpen) {
203
+ props.onMenuOpen()
204
+ }
205
+ }
206
+
142
207
  const selectProps = {
143
208
  cacheOptions: true,
144
209
  required,
@@ -172,6 +237,7 @@ const Typeahead = forwardRef<HTMLInputElement, TypeaheadProps>(({
172
237
  ...(preserveSearchInput ? { inputValue } : {}),
173
238
  onInputChange: handleInputChange,
174
239
  onBlur: handleBlur,
240
+ onMenuOpen: handleMenuOpen,
175
241
  ...props,
176
242
  }
177
243
 
@@ -261,6 +327,8 @@ const Typeahead = forwardRef<HTMLInputElement, TypeaheadProps>(({
261
327
  // Reset form submitted state when a selection is made (this is all for react rendered rails kit)
262
328
  if (action === 'select-option') {
263
329
  setFormSubmitted(false)
330
+ // Mark that user has made a selection to disable default value focus behavior
331
+ setHasUserSelected(true)
264
332
  }
265
333
 
266
334
  // If a value is selected and we're preserving input on blur, clear the input
@@ -269,7 +337,7 @@ const Typeahead = forwardRef<HTMLInputElement, TypeaheadProps>(({
269
337
  }
270
338
 
271
339
  if (action === 'select-option') {
272
- if (selectProps.onMultiValueClick) selectProps.onMultiValueClick(option)
340
+ if (selectProps.onMultiValueClick && option) selectProps.onMultiValueClick(option)
273
341
  const multiValueClearEvent = new CustomEvent(`pb-typeahead-kit-${selectProps.id}-result-option-select`, { detail: option ? option : _data })
274
342
  document.dispatchEvent(multiValueClearEvent)
275
343
  }
@@ -317,6 +385,7 @@ const Typeahead = forwardRef<HTMLInputElement, TypeaheadProps>(({
317
385
  error={errorDisplay}
318
386
  isDisabled={disabled}
319
387
  onChange={handleOnChange}
388
+ ref={selectRef}
320
389
  {...selectProps}
321
390
  />
322
391
  </div>