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.
- checksums.yaml +4 -4
- data/app/pb_kits/playbook/pb_advanced_table/Components/TableHeaderCell.tsx +3 -1
- data/app/pb_kits/playbook/pb_advanced_table/SubKits/TableHeader.tsx +3 -0
- data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.tsx +3 -0
- data/app/pb_kits/playbook/pb_advanced_table/advanced_table.test.jsx +59 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_inline_row_loading_show_toggle.jsx +59 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_inline_row_loading_show_toggle.md +5 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_mock_data_inline_loading_empty_children.js +202 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +1 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/index.js +2 -1
- data/app/pb_kits/playbook/pb_phone_number_input/_phone_number_input.tsx +25 -6
- data/app/pb_kits/playbook/pb_typeahead/_typeahead.test.jsx +35 -1
- data/app/pb_kits/playbook/pb_typeahead/_typeahead.tsx +71 -2
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_options.html.erb +36 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_options.md +1 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_value.jsx +41 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_value.md +1 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/example.yml +2 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/index.js +1 -0
- data/dist/chunks/{_line_graph-BnVgr42C.js → _line_graph-D5MBnrO9.js} +1 -1
- data/dist/chunks/{_typeahead-BH_dkgOy.js → _typeahead-BjYBazGq.js} +1 -1
- data/dist/chunks/{_weekday_stacked-Duv09EWo.js → _weekday_stacked-B_Uc7-rO.js} +2 -2
- data/dist/chunks/vendor.js +1 -1
- data/dist/playbook-doc.js +2 -2
- data/dist/playbook-rails-react-bindings.js +1 -1
- data/dist/playbook-rails.js +1 -1
- data/lib/playbook/version.rb +1 -1
- metadata +12 -5
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: 26ef9154b98ad4d5d762b37915fa398314c6020fc3cf0c0b42b9e074fd55f569
|
|
4
|
+
data.tar.gz: 5a79125ffb800db93c0ebe22ad143167bac0f9c018ebc33beba7a1c2049ec3fc
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
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 &&
|
|
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
|
data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_inline_row_loading_show_toggle.jsx
ADDED
|
@@ -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
|
data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_inline_row_loading_show_toggle.md
ADDED
|
@@ -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
|
-
|
|
375
|
-
|
|
376
|
-
|
|
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>
|