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>
         |