Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(react): support react 17 #1680

Merged
merged 37 commits into from
Oct 16, 2024
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
37 commits
Select commit Hold shift + click to select a range
ad9e4b0
feat(react): update to oficially support react 17
scurker Jul 3, 2024
d2a7ad1
add prop-types dependency
scurker Jul 3, 2024
f205cd4
fix combobox test
scurker Jul 3, 2024
9eed663
update clickoutside listener
scurker Jul 17, 2024
f2afda7
test: switch to user-event for clicks in OptionsMenu test
anastasialanz Sep 17, 2024
84323a0
test: typos in popover tests
anastasialanz Sep 17, 2024
b8e7c0c
test: failing test for popover bug
anastasialanz Sep 19, 2024
89ab364
fix: revert click outside changes
anastasialanz Sep 24, 2024
9b75a9c
fix: set capture on event listeners
anastasialanz Sep 30, 2024
9f59231
fix: return if clicking on the target element
anastasialanz Sep 30, 2024
d147b85
test: update test for popover trigger
anastasialanz Oct 1, 2024
aaa242c
refactor: convert options menu item to a function component
anastasialanz Oct 1, 2024
1ea1059
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 1, 2024
cda2956
refactor: convert options menu list to a function component
anastasialanz Oct 2, 2024
098a995
refactor: convert options menu to a function component
anastasialanz Oct 2, 2024
65d6ea2
test: add required onSelect prop to OptionsMenu test
anastasialanz Oct 2, 2024
7158934
fix: linting errors for options menu item
anastasialanz Oct 2, 2024
3ffc129
test: add onSelect prop to OptionsMenuList
anastasialanz Oct 2, 2024
d204969
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 2, 2024
fd74de7
test: add onSelect to options menu in top bar
anastasialanz Oct 2, 2024
952d674
test: fix options menu item tests
anastasialanz Oct 3, 2024
bd25a61
fix: close the options menu if clicking on trigger
anastasialanz Oct 3, 2024
ede47b8
test: fix typescript triggerRef error
anastasialanz Oct 3, 2024
ab96270
docs: fix top bar options menu implementation
anastasialanz Oct 8, 2024
3380ed1
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 8, 2024
4e1abf8
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 8, 2024
f438804
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 8, 2024
250db1f
fix: remove unneeded change
anastasialanz Oct 8, 2024
9df13e6
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 8, 2024
4e58005
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 10, 2024
555e12f
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 14, 2024
aa0642d
fix: use prevent default and internal ref
anastasialanz Oct 14, 2024
fe2a88e
test: revert top bar menu test changes
anastasialanz Oct 14, 2024
0b33e19
fix: revert topbar docs changes
anastasialanz Oct 16, 2024
daad0d2
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 16, 2024
c25ed15
test: fix options menu list onClose outside test
anastasialanz Oct 16, 2024
58287c0
Merge branch 'develop' into fix--react-17-upgrade
anastasialanz Oct 16, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
67 changes: 38 additions & 29 deletions docs/pages/components/TopBar.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ description: A container component that typically contains information about the
source: https://github.com/dequelabs/cauldron/tree/develop/packages/react/src/components/TopBar/TopBar.tsx
---

import { useRef } from 'react'
scurker marked this conversation as resolved.
Show resolved Hide resolved
import {
TopBar,
MenuBar,
Expand Down Expand Up @@ -60,35 +61,43 @@ import {
In addition to navigational items, the `MenuBar` can also contain a dropdown menu with items when composed with `TopBarMenu` and `OptionsMenuList`:

```jsx example
<TopBar>
<MenuBar>
<TopBarItem>
<a className="MenuItem__logo" href="#" tabIndex={-1}>
<img src="/public/fonts/logo.svg" alt="" />
Cauldron
</a>
</TopBarItem>
<TopBarItem>
<a href="#" tabIndex={-1}>
Item 1
</a>
</TopBarItem>
<TopBarMenu>
Menu <Icon type="chevron-down" />
<OptionsMenuList>
<OptionsMenuItem>
Menu Item 1
</OptionsMenuItem>
<OptionsMenuItem>
Menu Item 2
</OptionsMenuItem>
<OptionsMenuItem>
Menu Item 3
</OptionsMenuItem>
</OptionsMenuList>
</TopBarMenu>
</MenuBar>
</TopBar>
function TopBarWithOptionsMenuList () {
const menuTriggerRef = useRef(null);

return (
<TopBar>
<MenuBar>
<TopBarItem>
<a className="MenuItem__logo" href="#" tabIndex={-1}>
<img src="/public/fonts/logo.svg" alt="" />
Cauldron
</a>
</TopBarItem>
<TopBarItem>
<a href="#" tabIndex={-1}>
Item 1
</a>
</TopBarItem>
<TopBarMenu menuItemRef={(el) => {
menuTriggerRef.current = el
}}>
Menu <Icon type="chevron-down" />
<OptionsMenuList triggerRef={menuTriggerRef}>
<OptionsMenuItem>
Menu Item 1
</OptionsMenuItem>
<OptionsMenuItem>
Menu Item 2
</OptionsMenuItem>
<OptionsMenuItem>
Menu Item 3
</OptionsMenuItem>
</OptionsMenuList>
</TopBarMenu>
</MenuBar>
</TopBar>
)
}
```


Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@
"prettier": "^2",
"puppeteer": "^22.12.1",
"react": "^17",
"react-dom": "^16.13.1",
"react-dom": "^17",
"react-helmet": "^5.2.1",
"react-router-dom": "^5.3.0",
"remark-frontmatter": "^5.0.0",
Expand Down
7 changes: 4 additions & 3 deletions packages/react/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@
"dependencies": {
"@popperjs/core": "^2.5.4",
"classnames": "^2.2.6",
"focus-trap-react": "8",
"focus-trap-react": "^10.2.3",
"focusable": "^2.3.0",
"keyname": "^0.1.0",
"react-id-generator": "^3.0.1",
Expand Down Expand Up @@ -65,8 +65,9 @@
"postcss-cli": "^7.1.1",
"postcss-import": "^12.0.1",
"postcss-loader": "^3.0.0",
"react": "^16.13.1",
"react-dom": "^16.13.1",
"prop-types": "^15.8.1",
"react": "^17",
"react-dom": "^17",
"rollup": "^2.23.0",
"sinon": "^10.0.0",
"ts-node": "^10.9.2",
Expand Down
scurker marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -60,15 +60,15 @@ function ClickOutsideListener(

useEffect(() => {
typeof mouseEvent === 'string' &&
document.addEventListener(mouseEvent, handleEvent);
document.addEventListener(mouseEvent, handleEvent, true);
typeof touchEvent === 'string' &&
document.addEventListener(touchEvent, handleEvent);
document.addEventListener(touchEvent, handleEvent, true);

return () => {
typeof mouseEvent === 'string' &&
document.removeEventListener(mouseEvent, handleEvent);
document.removeEventListener(mouseEvent, handleEvent, true);
typeof touchEvent === 'string' &&
document.removeEventListener(touchEvent, handleEvent);
document.removeEventListener(touchEvent, handleEvent, true);
};
}, [mouseEvent, touchEvent]);

Expand Down
6 changes: 6 additions & 0 deletions packages/react/src/components/Combobox/Combobox.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -515,6 +515,9 @@ test('should prevent default with enter keypress and open listbox', () => {
// rtl doesn't let us mock preventDefault
// see: https://github.com/testing-library/react-testing-library/issues/572
event.preventDefault = preventDefault;
if (type === 'focus') {
fireEvent.focusIn(combobox);
}
fireEvent(combobox, event);
};

Expand Down Expand Up @@ -545,6 +548,9 @@ test('should not prevent default with enter keypress and closed listbox', () =>
// rtl doesn't let us mock preventDefault
// see: https://github.com/testing-library/react-testing-library/issues/572
event.preventDefault = preventDefault;
if (type === 'focus') {
scurker marked this conversation as resolved.
Show resolved Hide resolved
fireEvent.focusIn(combobox);
}
fireEvent(combobox, event);
};

Expand Down
92 changes: 63 additions & 29 deletions packages/react/src/components/OptionsMenu/OptionsMenu.test.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import React from 'react';
import { render, screen, fireEvent } from '@testing-library/react';
import { userEvent } from '@testing-library/user-event';
import OptionsMenu from './';
import axe from '../../axe';

Expand All @@ -10,8 +11,10 @@ const trigger = (props: React.HTMLAttributes<HTMLButtonElement>) => (
);

test('should render menu', () => {
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li>option 1</li>
<li>option 2</li>
</OptionsMenu>
Expand All @@ -22,8 +25,10 @@ test('should render menu', () => {
});

test('should render children', () => {
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li>option 1</li>
<li>option 2</li>
</OptionsMenu>
Expand All @@ -33,8 +38,10 @@ test('should render children', () => {
});

test('should render trigger prop', () => {
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li className="foo">option 1</li>
</OptionsMenu>
);
Expand All @@ -43,8 +50,10 @@ test('should render trigger prop', () => {
});

test('should support className prop', () => {
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger} className="bananas">
<OptionsMenu onSelect={onSelect} trigger={trigger} className="bananas">
<li>option 1</li>
<li>option 2</li>
</OptionsMenu>
Expand All @@ -56,13 +65,15 @@ test('should support className prop', () => {
});

test('should support align prop', () => {
const onSelect = jest.fn();

render(
<>
<OptionsMenu trigger={trigger} align="left">
<OptionsMenu onSelect={onSelect} trigger={trigger} align="left">
<li>option 1</li>
<li>option 2</li>
</OptionsMenu>
<OptionsMenu trigger={trigger} align="right">
<OptionsMenu onSelect={onSelect} trigger={trigger} align="right">
<li>option 1</li>
<li>option 2</li>
</OptionsMenu>
Expand All @@ -79,8 +90,10 @@ test('should support align prop', () => {

test('should support menuRef prop', () => {
const menuRef = React.createRef<HTMLUListElement>();
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger} menuRef={menuRef}>
<OptionsMenu onSelect={onSelect} trigger={trigger} menuRef={menuRef}>
<li>option 1</li>
<li>option 2</li>
</OptionsMenu>
Expand All @@ -90,81 +103,101 @@ test('should support menuRef prop', () => {
expect(menuRef.current).toEqual(screen.getByRole('menu'));
});

test('should toggle menu on trigger clicks', () => {
test('should toggle menu on trigger clicks', async () => {
const user = userEvent.setup();
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li className="foo">option 1</li>
</OptionsMenu>
);

fireEvent.click(screen.getByRole('button'));
expect(screen.getByRole('button')).toHaveAttribute('aria-expanded', 'true');
const button = screen.getByRole('button', { name: 'trigger' });

await user.click(button);
expect(button).toHaveAttribute('aria-expanded', 'true');
expect(screen.getByRole('menu')).toHaveAttribute('aria-expanded', 'true');

fireEvent.click(screen.getByRole('button'));
expect(screen.getByRole('button')).toHaveAttribute('aria-expanded', 'false');
await user.click(button);

expect(button).toHaveAttribute('aria-expanded', 'false');
expect(screen.getByRole('menu')).toHaveAttribute('aria-expanded', 'false');
});

test('should click trigger with down key on trigger', () => {
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li className="foo">option 1</li>
</OptionsMenu>
);

fireEvent.keyDown(screen.getByRole('button'), { keyCode: 40 });
expect(screen.getByRole('button')).toHaveAttribute('aria-expanded', 'true');
const button = screen.getByRole('button');

fireEvent.keyDown(button, { keyCode: 40 });
expect(button).toHaveAttribute('aria-expanded', 'true');
});

test('should focus trigger on close', () => {
test('should focus trigger on close', async () => {
const user = userEvent.setup();
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li className="foo">option 1</li>
</OptionsMenu>
);

const button = screen.getByRole('button');

fireEvent.click(button);
fireEvent.click(button); // to close
await user.click(button); // opens menu
await user.click(button); // closes menu
expect(button).toHaveFocus();
});

test('should call onClose when closed', () => {
test('should call onClose when closed', async () => {
const user = userEvent.setup();
const onClose = jest.fn();
const onSelect = jest.fn();

render(
<OptionsMenu trigger={trigger} onClose={onClose}>
<OptionsMenu onSelect={onSelect} trigger={trigger} onClose={onClose}>
<li className="foo">option 1</li>
</OptionsMenu>
);

const option = screen.getByRole('menuitem');

fireEvent.click(option);
await user.click(option);

expect(onClose).toBeCalled();
});

test('should close menu when click outside event occurs', () => {
test('should close menu when click outside event occurs', async () => {
const user = userEvent.setup();
const onSelect = jest.fn();

render(
<>
<button>Click me!</button>
<OptionsMenu trigger={trigger}>
<OptionsMenu onSelect={onSelect} trigger={trigger}>
<li className="foo">option 1</li>
</OptionsMenu>
</>
);
fireEvent.click(screen.getByRole('button', { name: 'trigger' }));

await user.click(screen.getByRole('button', { name: 'trigger' }));
expect(screen.getByRole('menu')).toHaveAttribute('aria-expanded', 'true');
fireEvent.click(screen.getByRole('button', { name: 'Click me!' }));
await user.click(screen.getByRole('button', { name: 'Click me!' }));
expect(screen.getByRole('menu')).toHaveAttribute('aria-expanded', 'false');
});

test('should return no axe violations when hidden', async () => {
const onSelect = jest.fn();
const { container } = render(
<OptionsMenu trigger={trigger} show={false}>
<OptionsMenu onSelect={onSelect} trigger={trigger} show={false}>
<li className="foo">option 1</li>
</OptionsMenu>
);
Expand All @@ -173,8 +206,9 @@ test('should return no axe violations when hidden', async () => {
});

test('should return no axe violations when shown', async () => {
const onSelect = jest.fn();
const { container } = render(
<OptionsMenu trigger={trigger} show={true}>
<OptionsMenu onSelect={onSelect} trigger={trigger} show={true}>
<li className="foo">option 1</li>
</OptionsMenu>
);
Expand Down
Loading
Loading