123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243 |
- import {mountWithTheme} from 'sentry-test/enzyme';
- import DropdownMenu from 'sentry/components/dropdownMenu';
- jest.useFakeTimers();
- describe('DropdownMenu', function () {
- let wrapper;
- beforeEach(function () {
- wrapper = mountWithTheme(
- <DropdownMenu>
- {({getRootProps, getActorProps, getMenuProps, isOpen}) => (
- <span {...getRootProps({})}>
- <button {...getActorProps({})}>Open Dropdown</button>
- {isOpen && (
- <ul {...getMenuProps({})}>
- <li>Dropdown Menu Item 1</li>
- </ul>
- )}
- </span>
- )}
- </DropdownMenu>
- );
- });
- it('renders', function () {
- expect(wrapper).toSnapshot();
- });
- it('can toggle dropdown menu with actor', function () {
- wrapper.find('button').simulate('click');
- expect(wrapper.state('isOpen')).toBe(true);
- expect(wrapper.find('ul')).toHaveLength(1);
- wrapper.find('button').simulate('click');
- expect(wrapper.state('isOpen')).toBe(false);
- expect(wrapper.find('ul')).toHaveLength(0);
- });
- it('closes dropdown when clicking on anything in menu', function () {
- wrapper.find('button').simulate('click');
- wrapper.find('li').simulate('click');
- expect(wrapper.state('isOpen')).toBe(false);
- expect(wrapper.find('ul')).toHaveLength(0);
- });
- it('closes dropdown when clicking outside of menu', async function () {
- wrapper.find('button').simulate('click');
- // Simulate click on document
- const event = document.createEvent('HTMLEvents');
- event.initEvent('click', false, true);
- document.body.dispatchEvent(event);
- jest.runAllTimers();
- await Promise.resolve();
- wrapper.update();
- expect(wrapper.find('ul')).toHaveLength(0);
- });
- it('closes dropdown when pressing escape', function () {
- wrapper.find('button').simulate('click');
- expect(wrapper.state('isOpen')).toBe(true);
- wrapper.simulate('keyDown', {key: 'Escape'});
- wrapper.find('button').simulate('keyDown', {key: 'Escape'});
- expect(wrapper.state('isOpen')).toBe(false);
- expect(wrapper.find('ul')).toHaveLength(0);
- });
- it('ignores "Escape" key if `closeOnEscape` is false', function () {
- wrapper = mountWithTheme(
- <DropdownMenu closeOnEscape={false}>
- {({getRootProps, getActorProps, getMenuProps, isOpen}) => (
- <span {...getRootProps({})}>
- <button {...getActorProps({})}>Open Dropdown</button>
- {isOpen && (
- <ul {...getMenuProps({})}>
- <li>Dropdown Menu Item 1</li>
- </ul>
- )}
- </span>
- )}
- </DropdownMenu>
- );
- wrapper.find('button').simulate('click');
- expect(wrapper.state('isOpen')).toBe(true);
- wrapper.find('button').simulate('keyDown', {key: 'Escape'});
- expect(wrapper.find('ul')).toHaveLength(1);
- expect(wrapper.state('isOpen')).toBe(true);
- });
- it('keeps dropdown open when clicking on anything in menu with `keepMenuOpen` prop', function () {
- wrapper = mountWithTheme(
- <DropdownMenu keepMenuOpen>
- {({getRootProps, getActorProps, getMenuProps, isOpen}) => (
- <span {...getRootProps({})}>
- <button {...getActorProps({})}>Open Dropdown</button>
- {isOpen && (
- <ul {...getMenuProps({})}>
- <li>Dropdown Menu Item 1</li>
- </ul>
- )}
- </span>
- )}
- </DropdownMenu>
- );
- wrapper.find('button').simulate('click');
- wrapper.find('li').simulate('click');
- expect(wrapper.state('isOpen')).toBe(true);
- expect(wrapper.find('ul')).toHaveLength(1);
- });
- it('render prop getters all extend props and call original onClick handlers', function () {
- const rootClick = jest.fn();
- const actorClick = jest.fn();
- const menuClick = jest.fn();
- const addSpy = jest.spyOn(document, 'addEventListener');
- const removeSpy = jest.spyOn(document, 'removeEventListener');
- wrapper = mountWithTheme(
- <DropdownMenu keepMenuOpen>
- {({getRootProps, getActorProps, getMenuProps, isOpen}) => (
- <span
- {...getRootProps({
- className: 'root',
- onClick: rootClick,
- })}
- >
- <button
- {...getActorProps({
- className: 'actor',
- onClick: actorClick,
- })}
- >
- Open Dropdown
- </button>
- {isOpen && (
- <ul
- {...getMenuProps({
- className: 'menu',
- onClick: menuClick,
- })}
- >
- <li>Dropdown Menu Item 1</li>
- </ul>
- )}
- </span>
- )}
- </DropdownMenu>
- );
- expect(wrapper.find('ul')).toHaveLength(0);
- wrapper.find('span').simulate('click');
- expect(rootClick).toHaveBeenCalled();
- wrapper.find('button').simulate('click');
- expect(actorClick).toHaveBeenCalled();
- wrapper.find('li').simulate('click');
- expect(menuClick).toHaveBeenCalled();
- // breaks in jest22
- // expect(wrapper).toSnapshot();
- expect(wrapper.find('ul')).toHaveLength(1);
- expect(document.addEventListener).toHaveBeenCalled();
- wrapper.unmount();
- expect(document.removeEventListener).toHaveBeenCalled();
- addSpy.mockRestore();
- removeSpy.mockRestore();
- });
- it('always rendered menus should attach document event listeners only when opened', function () {
- const addSpy = jest.spyOn(document, 'addEventListener');
- const removeSpy = jest.spyOn(document, 'removeEventListener');
- wrapper = mountWithTheme(
- <DropdownMenu alwaysRenderMenu>
- {({getRootProps, getActorProps, getMenuProps}) => (
- <span
- {...getRootProps({
- className: 'root',
- })}
- >
- <button
- {...getActorProps({
- className: 'actor',
- })}
- >
- Open Dropdown
- </button>
- <ul
- {...getMenuProps({
- className: 'menu',
- })}
- >
- <li>Dropdown Menu Item 1</li>
- </ul>
- </span>
- )}
- </DropdownMenu>
- );
- // Make sure this is only called when menu is open
- expect(document.addEventListener).not.toHaveBeenCalled();
- wrapper.find('button').simulate('click');
- expect(wrapper.state('isOpen')).toBe(true);
- expect(document.addEventListener).toHaveBeenCalled();
- expect(document.removeEventListener).not.toHaveBeenCalled();
- wrapper.find('button').simulate('click');
- expect(wrapper.state('isOpen')).toBe(false);
- expect(document.removeEventListener).toHaveBeenCalled();
- addSpy.mockRestore();
- removeSpy.mockRestore();
- });
- it('does not close nested dropdown on actor clicks', function () {
- wrapper = mountWithTheme(
- <DropdownMenu isNestedDropdown>
- {({getRootProps, getActorProps, getMenuProps}) => (
- <span {...getRootProps({})}>
- <button {...getActorProps({})}>Open Dropdown</button>
- {
- <ul {...getMenuProps({})}>
- <li data-test-id="menu-item">Dropdown Menu Item 1</li>
- </ul>
- }
- </span>
- )}
- </DropdownMenu>
- );
- wrapper.find('button').simulate('mouseEnter');
- expect(wrapper.find('[data-test-id="menu-item"]')).toHaveLength(1);
- wrapper.find('button').simulate('click');
- // Should still be visible.
- expect(wrapper.find('[data-test-id="menu-item"]')).toHaveLength(1);
- });
- });
|