From 3a98d2946fac2639a2207a41999a66c02ba6fe61 Mon Sep 17 00:00:00 2001 From: Mateus Rodrigues <34143412+mateus037@users.noreply.github.com> Date: Fri, 27 Oct 2023 12:13:08 -0300 Subject: [PATCH 01/10] Removing the title inside the tab source (#15003) * Removing the title inside the tab source * fix(tabs Component): oorrecting error in formatting step --- packages/react/src/components/Tabs/Tabs.tsx | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/packages/react/src/components/Tabs/Tabs.tsx b/packages/react/src/components/Tabs/Tabs.tsx index e3b3b01b443a..92d6eaf1599d 100644 --- a/packages/react/src/components/Tabs/Tabs.tsx +++ b/packages/react/src/components/Tabs/Tabs.tsx @@ -903,11 +903,7 @@ const Tab = forwardRef(function Tab( {} )} - - {children} - + {children} {/* always rendering dismissIcon so we don't lose reference to it, otherwise events do not work when switching from/to dismissable state */}
Date: Fri, 27 Oct 2023 13:00:48 -0500 Subject: [PATCH 02/10] fix(slider): update rem() usage to to-rem() (#15049) --- packages/styles/scss/components/slider/_slider.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/styles/scss/components/slider/_slider.scss b/packages/styles/scss/components/slider/_slider.scss index 4bcddcdd71ba..67ff6f1f6c60 100644 --- a/packages/styles/scss/components/slider/_slider.scss +++ b/packages/styles/scss/components/slider/_slider.scss @@ -43,7 +43,7 @@ min-inline-size: convert.to-rem(200px); .#{$prefix}--slider-container--two-handles & { - margin-inline: rem(4px); + margin-inline: convert.to-rem(4px); } } From 8473d795ca3deb78e8ed2cd84ecaa629099c0318 Mon Sep 17 00:00:00 2001 From: Dominik Brugger Date: Mon, 30 Oct 2023 15:40:21 +0100 Subject: [PATCH 03/10] fix(ContentSwitcher): selectionMode optional type (#15061) --- .../react/src/components/ContentSwitcher/ContentSwitcher.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/react/src/components/ContentSwitcher/ContentSwitcher.tsx b/packages/react/src/components/ContentSwitcher/ContentSwitcher.tsx index 3f3129543476..384b978973d2 100644 --- a/packages/react/src/components/ContentSwitcher/ContentSwitcher.tsx +++ b/packages/react/src/components/ContentSwitcher/ContentSwitcher.tsx @@ -56,7 +56,7 @@ export interface ContentSwitcherProps /** * Choose whether or not to automatically change selection on focus */ - selectionMode: 'automatic' | 'manual'; + selectionMode?: 'automatic' | 'manual'; /** * Specify the size of the Content Switcher. Currently supports either `sm`, 'md' (default) or 'lg` as an option. From 3434c930c437acaf19502778e0f7b3c8875e9fa3 Mon Sep 17 00:00:00 2001 From: Taylor Jones Date: Tue, 31 Oct 2023 13:16:59 -0500 Subject: [PATCH 04/10] ci(cache): turn off caching in the merge queue (#15077) Co-authored-by: Alison Joseph --- .github/workflows/ci.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index e4efe9794a85..f9709559ed73 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -62,6 +62,7 @@ jobs: with: node-version: '20.x' - uses: actions/cache@v3 + if: github.event_name != 'merge_group' id: cache with: path: | @@ -88,6 +89,7 @@ jobs: with: node-version: '20.x' - uses: actions/cache@v3 + if: github.event_name != 'merge_group' id: cache with: path: | @@ -127,6 +129,7 @@ jobs: with: node-version: '20.x' - uses: actions/cache@v3 + if: github.event_name != 'merge_group' id: cache with: path: | @@ -186,6 +189,7 @@ jobs: with: node-version: '20.x' - uses: actions/cache@v3 + if: github.event_name != 'merge_group' id: cache with: path: | From 736de0869f1637e2e8f598cfcceb3ee4a70692b9 Mon Sep 17 00:00:00 2001 From: Alex Lewitt <48691328+alewitt2@users.noreply.github.com> Date: Tue, 31 Oct 2023 14:46:49 -0400 Subject: [PATCH 05/10] fix(bug): slider.scss uses deprecated `/` (#15063) * fix(bug): slider.scss uses deprecated `/` an accidental deprecated division `/` snuck in and is causing sass loader errors downstream. * Update packages/styles/scss/components/slider/_slider.scss Co-authored-by: Taylor Jones * Update packages/styles/scss/components/slider/_slider.scss --------- Co-authored-by: Taylor Jones --- packages/styles/scss/components/slider/_slider.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/styles/scss/components/slider/_slider.scss b/packages/styles/scss/components/slider/_slider.scss index 67ff6f1f6c60..212a6c8fca27 100644 --- a/packages/styles/scss/components/slider/_slider.scss +++ b/packages/styles/scss/components/slider/_slider.scss @@ -146,7 +146,7 @@ block-size: convert.to-rem(2px); content: ''; inline-size: convert.to-rem(6px); - inset-block-start: calc(50% - #{convert.to-rem(2px) / 2}); + inset-block-start: calc(50% - #{convert.to-rem(2px) * 0.5}); inset-inline-end: 0; } From e30238bd01aa196dffe759525fb1e3c2bf668be0 Mon Sep 17 00:00:00 2001 From: Guilherme Datilio Ribeiro Date: Tue, 31 Oct 2023 15:43:57 -0300 Subject: [PATCH 06/10] test: added keyboard navigation (#14916) Co-authored-by: Taylor Jones --- .../Toggletip/Toggletip-test.avt.e2e.js | 57 +++++++++++++++++++ .../Toggletip/Toggletip-test.e2e.js | 17 +----- 2 files changed, 60 insertions(+), 14 deletions(-) create mode 100644 e2e/components/Toggletip/Toggletip-test.avt.e2e.js diff --git a/e2e/components/Toggletip/Toggletip-test.avt.e2e.js b/e2e/components/Toggletip/Toggletip-test.avt.e2e.js new file mode 100644 index 000000000000..26525fc2519b --- /dev/null +++ b/e2e/components/Toggletip/Toggletip-test.avt.e2e.js @@ -0,0 +1,57 @@ +/** + * Copyright IBM Corp. 2016, 2023 + * + * This source code is licensed under the Apache-2.0 license found in the + * LICENSE file in the root directory of this source tree. + */ + +'use strict'; + +import { expect, test } from '@playwright/test'; +import { visitStory } from '../../test-utils/storybook'; + +test.describe('Toggletip @avt', () => { + test('@avt-default-state Toggletip', async ({ page }) => { + await visitStory(page, { + component: 'Toggletip', + id: 'components-toggletip--default', + globals: { + theme: 'white', + }, + }); + await expect(page).toHaveNoACViolations('Toggletip'); + }); + + test('@avt-keyboard-nav Toggletip', async ({ page }) => { + await visitStory(page, { + component: 'Toggletip', + id: 'components-toggletip--default', + globals: { + theme: 'white', + }, + }); + + // Checking if the defaultOpen is working + await expect(page.locator('.cds--popover--open')).toBeVisible(); + + // Checking first Toggletip interaction + await page.keyboard.press('Tab'); + await expect(page.getByLabel('Show information').first()).toBeFocused(); + await page.keyboard.press('Enter'); + await expect(page.locator('.cds--popover--open')).toBeVisible(); + // Tabbing inside the popover + await page.keyboard.press('Tab'); + await expect(page.locator('.cds--link').first()).toBeFocused(); + await page.keyboard.press('Tab'); + await expect(page.getByRole('button', { name: 'Button' })).toBeFocused(); + await page.keyboard.press('Tab'); + await expect(page.locator('.cds--popover--open')).not.toBeVisible(); + + // Checking second Toggletip interaction and close on Escape key + await expect(page.getByLabel('Show information').last()).toBeFocused(); + await page.keyboard.press('Enter'); + await expect(page.locator('.cds--popover--open')).toBeVisible(); + await page.keyboard.press('Escape'); + await expect(page.locator('.cds--popover--open')).not.toBeVisible(); + }); +}); diff --git a/e2e/components/Toggletip/Toggletip-test.e2e.js b/e2e/components/Toggletip/Toggletip-test.e2e.js index df57ebbc70fb..b64f2a196073 100644 --- a/e2e/components/Toggletip/Toggletip-test.e2e.js +++ b/e2e/components/Toggletip/Toggletip-test.e2e.js @@ -7,9 +7,9 @@ 'use strict'; -const { expect, test } = require('@playwright/test'); -const { themes } = require('../../test-utils/env'); -const { snapshotStory, visitStory } = require('../../test-utils/storybook'); +import { test } from '@playwright/test'; +import { themes } from '../../test-utils/env'; +import { snapshotStory } from '../../test-utils/storybook'; test.describe('Toggletip', () => { themes.forEach((theme) => { @@ -23,15 +23,4 @@ test.describe('Toggletip', () => { }); }); }); - - test('accessibility-checker @avt', async ({ page }) => { - await visitStory(page, { - component: 'Toggletip', - id: 'components-toggletip--default', - globals: { - theme: 'white', - }, - }); - await expect(page).toHaveNoACViolations('Toggletip'); - }); }); From 384dca0b68c74833b07743dc33c54ff2277d6773 Mon Sep 17 00:00:00 2001 From: TJ Egan Date: Tue, 31 Oct 2023 15:12:53 -0400 Subject: [PATCH 07/10] feat(Combobox): allow custom value (#14935) * feat(Combobox): add allowCustomValue prop * test(Combobox): add tests to cover new functionality * docs(Combobox): add some docs around the new prop * docs(Combobox): add test story * test(snapshot): update snapshots * chore(storybook): remove commented out code --------- Co-authored-by: Taylor Jones --- .../__snapshots__/PublicAPI-test.js.snap | 3 ++ .../src/components/ComboBox/ComboBox-test.js | 15 ++++++++ .../src/components/ComboBox/ComboBox.mdx | 14 +++++++ .../components/ComboBox/ComboBox.stories.js | 26 +++++++++++++ .../src/components/ComboBox/ComboBox.tsx | 37 ++++++++++++++++++- 5 files changed, 93 insertions(+), 2 deletions(-) diff --git a/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap b/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap index 0e53df0816fd..610ba8e301f9 100644 --- a/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap +++ b/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap @@ -1054,6 +1054,9 @@ Map { "ComboBox" => Object { "$$typeof": Symbol(react.forward_ref), "propTypes": Object { + "allowCustomValue": Object { + "type": "bool", + }, "aria-label": Object { "type": "string", }, diff --git a/packages/react/src/components/ComboBox/ComboBox-test.js b/packages/react/src/components/ComboBox/ComboBox-test.js index 36e0d8ee8374..dca836a0b8c8 100644 --- a/packages/react/src/components/ComboBox/ComboBox-test.js +++ b/packages/react/src/components/ComboBox/ComboBox-test.js @@ -101,6 +101,21 @@ describe('ComboBox', () => { }); }); + it('should retain value if custom value is entered and `allowCustomValue` is set', async () => { + render(); + + expect(findInputNode()).toHaveDisplayValue(''); + + await userEvent.type(findInputNode(), 'Apple'); + // Should close menu and keep value in input, even though it is not in the item list + await userEvent.keyboard('[Enter]'); + assertMenuClosed(); + expect(findInputNode()).toHaveDisplayValue('Apple'); + // Should retain value on blur + await userEvent.keyboard('[Tab]'); + expect(findInputNode()).toHaveDisplayValue('Apple'); + }); + describe('should display initially selected item found in `initialSelectedItem`', () => { it('using an object type for the `initialSelectedItem` prop', () => { render( diff --git a/packages/react/src/components/ComboBox/ComboBox.mdx b/packages/react/src/components/ComboBox/ComboBox.mdx index 956c9dcacdcf..06fae50fa306 100644 --- a/packages/react/src/components/ComboBox/ComboBox.mdx +++ b/packages/react/src/components/ComboBox/ComboBox.mdx @@ -21,6 +21,7 @@ import ComboBox from '../ComboBox'; - [itemToElement](#itemtoelement) - [itemToString](#itemtostring) - [shouldFilterItem](#shouldfilteritem) +- [allowCustomValue](#allowcustomvalue) - [With Layer](#with-layer) - [Feedback](#feedback) @@ -147,6 +148,19 @@ const filterItems = (menu) => { />; ``` +## `allowCustomValue` + +By default, if text is entered into the `Combobox` and it does not match an +item, it will be cleared on blur. However, you can change this behavior by +passing in the `allowCustomValue` prop. This will allow a user to close the menu +and accept a custom value by pressing `Enter` as well as retain the value on +blur. The `inputValue` is provided as a second argument to the `onChange` +callback. + +```js +{selectedItem: undefined, inputValue: 'Apple'} +``` + ## With Layer diff --git a/packages/react/src/components/ComboBox/ComboBox.stories.js b/packages/react/src/components/ComboBox/ComboBox.stories.js index dc6197a1d912..b987255bf215 100644 --- a/packages/react/src/components/ComboBox/ComboBox.stories.js +++ b/packages/react/src/components/ComboBox/ComboBox.stories.js @@ -79,6 +79,32 @@ export const Default = () => (
); +export const AllowCustomValue = () => { + const filterItems = (menu) => { + return menu?.item?.toLowerCase().includes(menu?.inputValue?.toLowerCase()); + }; + return ( +
+ { + console.log(e); + }} + id="carbon-combobox" + items={['Apple', 'Orange', 'Banana', 'Pineapple', 'Raspberry', 'Lime']} + downshiftProps={{ + onStateChange: () => { + console.log('the state has changed'); + }, + }} + titleText="ComboBox title" + helperText="Combobox helper text" + /> +
+ ); +}; + export const _WithLayer = () => ( {(layer) => ( diff --git a/packages/react/src/components/ComboBox/ComboBox.tsx b/packages/react/src/components/ComboBox/ComboBox.tsx index 61914df39043..00060d69c922 100644 --- a/packages/react/src/components/ComboBox/ComboBox.tsx +++ b/packages/react/src/components/ComboBox/ComboBox.tsx @@ -55,6 +55,7 @@ const { clickButton, blurButton, changeInput, + blurInput, } = Downshift.stateChangeTypes; const defaultItemToString = (item: ItemType | null) => { @@ -127,13 +128,20 @@ const getInstanceId = setupGetInstanceId(); type ExcludedAttributes = 'id' | 'onChange' | 'onClick' | 'type' | 'size'; interface OnChangeData { - selectedItem: ItemType | null; + selectedItem: ItemType | null | undefined; + inputValue?: string | null; } type ItemToStringHandler = (item: ItemType | null) => string; export interface ComboBoxProps extends Omit, ExcludedAttributes> { + /** + * Specify whether or not the ComboBox should allow a value that is + * not in the list to be entered in the input + */ + allowCustomValue?: boolean; + /** * Specify a label to be read by screen readers on the container node * 'aria-label' of the ListBox component. @@ -329,6 +337,7 @@ const ComboBox = forwardRef( translateWithId, warn, warnText, + allowCustomValue = false, ...rest } = props; const prefix = usePrefix(); @@ -447,6 +456,14 @@ const ComboBox = forwardRef( case changeInput: updateHighlightedIndex(getHighlightedIndex(changes)); break; + case blurInput: + if (allowCustomValue) { + setInputValue(inputValue); + if (onChange) { + onChange({ selectedItem, inputValue }); + } + } + break; } }; @@ -571,8 +588,18 @@ const ComboBox = forwardRef( event.stopPropagation(); } - if (match(event, keys.Enter) && !inputValue) { + if ( + match(event, keys.Enter) && + (!inputValue || allowCustomValue) + ) { toggleMenu(); + + // Since `onChange` does not normally fire when the menu is closed, we should + // manually fire it when `allowCustomValue` is provided, the menu is closing, + // and there is a value. + if (allowCustomValue && isOpen && inputValue) { + onChange({ selectedItem, inputValue }); + } } if (match(event, keys.Escape) && inputValue) { @@ -744,6 +771,12 @@ const ComboBox = forwardRef( ComboBox.displayName = 'ComboBox'; ComboBox.propTypes = { + /** + * Specify whether or not the ComboBox should allow a value that is + * not in the list to be entered in the input + */ + allowCustomValue: PropTypes.bool, + /** * 'aria-label' of the ListBox component. * Specify a label to be read by screen readers on the container node From 91ce3750f88c8d2dcc25a719ed446cb65fc18cf5 Mon Sep 17 00:00:00 2001 From: Parth Chawande <101104958+parthrc@users.noreply.github.com> Date: Wed, 1 Nov 2023 00:42:54 +0530 Subject: [PATCH 08/10] Fixed Incorrect typescript for renderSelectedItem property of the Dropdown component (#14893) * Fixed Incorrect typescript for renderSelectedItem property of the Dropdown component * Fixed incorrect typescript for renderSelectedItem property of dropdown component * chore: format --------- Co-authored-by: Taylor Jones Co-authored-by: Alison Joseph --- packages/react/src/components/Dropdown/Dropdown.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/packages/react/src/components/Dropdown/Dropdown.tsx b/packages/react/src/components/Dropdown/Dropdown.tsx index 4284fc96c448..25206b4a6b24 100644 --- a/packages/react/src/components/Dropdown/Dropdown.tsx +++ b/packages/react/src/components/Dropdown/Dropdown.tsx @@ -184,7 +184,9 @@ export interface DropdownProps * An optional callback to render the currently selected item as a react element instead of only * as a string. */ - renderSelectedItem?(item: ItemType): string; + renderSelectedItem?( + item: ItemType + ): React.JSXElementConstructor | null; /** * In the case you want to control the dropdown selection entirely. From 083740a439759b1c6780fabc345c0e42c631c02e Mon Sep 17 00:00:00 2001 From: TJ Egan Date: Tue, 31 Oct 2023 15:14:12 -0400 Subject: [PATCH 09/10] test(Menu): add AVT tests for Menu (#14949) Co-authored-by: Taylor Jones --- e2e/components/Menu/Menu-test.avt.e2e.js | 69 ++++++++++++++++++++++++ e2e/components/Menu/Menu-test.e2e.js | 15 +----- 2 files changed, 71 insertions(+), 13 deletions(-) create mode 100644 e2e/components/Menu/Menu-test.avt.e2e.js diff --git a/e2e/components/Menu/Menu-test.avt.e2e.js b/e2e/components/Menu/Menu-test.avt.e2e.js new file mode 100644 index 000000000000..ecf249ecb777 --- /dev/null +++ b/e2e/components/Menu/Menu-test.avt.e2e.js @@ -0,0 +1,69 @@ +/** + * Copyright IBM Corp. 2023 + * + * This source code is licensed under the Apache-2.0 license found in the + * LICENSE file in the root directory of this source tree. + */ + +'use strict'; + +const { expect, test } = require('@playwright/test'); +const { visitStory } = require('../../test-utils/storybook'); + +test.describe('Menu @avt', () => { + test('@avt-default-state', async ({ page }) => { + await visitStory(page, { + component: 'Menu', + id: 'components-menu--playground', + globals: { + theme: 'white', + }, + }); + await expect(page).toHaveNoACViolations('Menu @avt-default-state'); + }); + + test('@avt-keyboard-nav Menu', async ({ page }) => { + await visitStory(page, { + component: 'Menu', + id: 'components-menu--playground', + globals: { + theme: 'white', + }, + }); + + const firstItem = page.getByRole('menuitem', { name: 'Share with' }); + const LastItem = page.getByRole('menuitem', { name: 'Delete' }); + const nestedMenu = page.getByRole('menu', { name: 'Share with' }); + const nestedMenuItem = page + .getByRole('menuitemradio', { + name: 'None', + }) + .first(); + + await expect(firstItem).toBeVisible(); + await expect(LastItem).toBeVisible(); + await expect(nestedMenu).not.toBeVisible(); + await expect(firstItem).toBeFocused(); + + // Should go to last item when focused on the first item and arrow up is pressed + await page.keyboard.press('ArrowUp'); + await expect(LastItem).toBeFocused(); + + // Should open menu with ArrowRight and focus on first item + await page.keyboard.press('ArrowDown'); + await expect(firstItem).toBeFocused(); + await page.keyboard.press('ArrowRight'); + await expect(nestedMenu).toBeVisible(); + await expect(nestedMenuItem).toBeVisible(); + await expect(nestedMenuItem).toBeFocused(); + await expect(nestedMenuItem).not.toBeChecked(); + + // Should select item with enter key + await page.keyboard.press('Enter'); + await expect(nestedMenuItem).toBeChecked(); + + // Should close menu with ArrowLeft + await page.keyboard.press('ArrowLeft'); + await expect(nestedMenu).not.toBeVisible(); + }); +}); diff --git a/e2e/components/Menu/Menu-test.e2e.js b/e2e/components/Menu/Menu-test.e2e.js index cf3309d3dd24..7541bc740e50 100644 --- a/e2e/components/Menu/Menu-test.e2e.js +++ b/e2e/components/Menu/Menu-test.e2e.js @@ -7,9 +7,9 @@ 'use strict'; -const { expect, test } = require('@playwright/test'); +const { test } = require('@playwright/test'); const { themes } = require('../../test-utils/env'); -const { snapshotStory, visitStory } = require('../../test-utils/storybook'); +const { snapshotStory } = require('../../test-utils/storybook'); test.describe('Menu', () => { themes.forEach((theme) => { @@ -23,15 +23,4 @@ test.describe('Menu', () => { }); }); }); - - test('accessibility-checker @avt', async ({ page }) => { - await visitStory(page, { - component: 'Menu', - id: 'components-menu--playground', - globals: { - theme: 'white', - }, - }); - await expect(page).toHaveNoACViolations('Menu'); - }); }); From cb2e75aa12a063b0c50261199e90c46ad8f72610 Mon Sep 17 00:00:00 2001 From: Rajat Date: Wed, 1 Nov 2023 00:44:39 +0530 Subject: [PATCH 10/10] Feat: Forwarding Ref in `TableExpandRow` component (#14950) * feat: allow forwardingref * fix: update snapshot --------- Co-authored-by: Andrea N. Cardona Co-authored-by: Taylor Jones --- .../__snapshots__/PublicAPI-test.js.snap | 4 + .../components/DataTable/TableExpandRow.tsx | 102 ++++++++++-------- 2 files changed, 59 insertions(+), 47 deletions(-) diff --git a/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap b/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap index 610ba8e301f9..4fb469ca7068 100644 --- a/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap +++ b/packages/react/__tests__/__snapshots__/PublicAPI-test.js.snap @@ -1948,6 +1948,7 @@ Map { }, }, "TableExpandRow": Object { + "$$typeof": Symbol(react.forward_ref), "propTypes": Object { "aria-controls": Object { "type": "string", @@ -1982,6 +1983,7 @@ Map { "type": "func", }, }, + "render": [Function], }, "TableExpandedRow": Object { "propTypes": Object { @@ -7560,6 +7562,7 @@ Map { }, }, "TableExpandRow" => Object { + "$$typeof": Symbol(react.forward_ref), "propTypes": Object { "aria-controls": Object { "type": "string", @@ -7594,6 +7597,7 @@ Map { "type": "func", }, }, + "render": [Function], }, "TableExpandedRow" => Object { "propTypes": Object { diff --git a/packages/react/src/components/DataTable/TableExpandRow.tsx b/packages/react/src/components/DataTable/TableExpandRow.tsx index 9c2e604a76a6..24356f76cc2f 100644 --- a/packages/react/src/components/DataTable/TableExpandRow.tsx +++ b/packages/react/src/components/DataTable/TableExpandRow.tsx @@ -54,66 +54,73 @@ interface TableExpandRowProps extends PropsWithChildren { onExpand: MouseEventHandler; } -const TableExpandRow = ({ - ['aria-controls']: ariaControls, - ['aria-label']: ariaLabel, - ariaLabel: deprecatedAriaLabel, - className: rowClassName, - children, - isExpanded, - onExpand, - expandIconDescription, - isSelected, - expandHeader = 'expand', - ...rest -}: TableExpandRowProps) => { - const prefix = usePrefix(); - const className = cx( +const TableExpandRow = React.forwardRef( + ( { - [`${prefix}--parent-row`]: true, - [`${prefix}--expandable-row`]: isExpanded, - [`${prefix}--data-table--selected`]: isSelected, - }, - rowClassName - ); - const previousValue = isExpanded ? 'collapsed' : undefined; - - return ( - - - - - {children} - - ); -}; + ['aria-controls']: ariaControls, + ['aria-label']: ariaLabel, + ariaLabel: deprecatedAriaLabel, + className: rowClassName, + children, + isExpanded, + onExpand, + expandIconDescription, + isSelected, + expandHeader = 'expand', + ...rest + }: TableExpandRowProps, + ref: React.Ref + ) => { + const prefix = usePrefix(); + const className = cx( + { + [`${prefix}--parent-row`]: true, + [`${prefix}--expandable-row`]: isExpanded, + [`${prefix}--data-table--selected`]: isSelected, + }, + rowClassName + ); + const previousValue = isExpanded ? 'collapsed' : undefined; + + return ( + + + + + {children} + + ); + } +); TableExpandRow.propTypes = { /** * Space separated list of one or more ID values referencing the TableExpandedRow(s) being controlled by the TableExpandRow * TODO: make this required in v12 */ + /**@ts-ignore*/ ['aria-controls']: PropTypes.string, /** * Specify the string read by a voice reader when the expand trigger is * focused */ + /**@ts-ignore*/ ['aria-label']: PropTypes.string, /** @@ -152,4 +159,5 @@ TableExpandRow.propTypes = { onExpand: PropTypes.func.isRequired, }; +TableExpandRow.displayName = 'TableExpandRow'; export default TableExpandRow;