From 6b9667b55b754a84c2bfa23e7d87d8c75abd7cdb Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Wed, 15 Jun 2022 14:43:49 +0200 Subject: [PATCH 01/12] test: perform actual user action instead of triggering blur event manually --- packages/autocomplete-js/src/__tests__/positioning.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/autocomplete-js/src/__tests__/positioning.test.ts b/packages/autocomplete-js/src/__tests__/positioning.test.ts index a47bf59b8..bfb68a09d 100644 --- a/packages/autocomplete-js/src/__tests__/positioning.test.ts +++ b/packages/autocomplete-js/src/__tests__/positioning.test.ts @@ -191,7 +191,7 @@ describe('Panel positioning', () => { right: '1020px', }); - input.blur(); + userEvent.click(document.body); // Move the root vertically root.getBoundingClientRect = jest From 3257bc5c0d5bb246b10cae1c486ff498340d535a Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Thu, 16 Jun 2022 11:20:26 +0200 Subject: [PATCH 02/12] test: make concurrency tests more exhaustive and precise --- .../src/__tests__/concurrency.test.ts | 76 ++++++++++++++++++- 1 file changed, 72 insertions(+), 4 deletions(-) diff --git a/packages/autocomplete-core/src/__tests__/concurrency.test.ts b/packages/autocomplete-core/src/__tests__/concurrency.test.ts index de7dde305..bc8ecc100 100644 --- a/packages/autocomplete-core/src/__tests__/concurrency.test.ts +++ b/packages/autocomplete-core/src/__tests__/concurrency.test.ts @@ -134,7 +134,7 @@ describe('concurrency', () => { expect(getSources).toHaveBeenCalledTimes(3); }); - test('keeps the panel closed on blur', async () => { + test('keeps the panel closed on Enter', async () => { const onStateChange = jest.fn(); const { timeout, delayedGetSources } = createDelayedGetSources({ sources: [100, 200], @@ -188,7 +188,74 @@ describe('concurrency', () => { expect(getSources).toHaveBeenCalledTimes(2); }); - test('keeps the panel closed on touchstart blur', async () => { + test('keeps the panel closed on click outside', async () => { + const onStateChange = jest.fn(); + const { timeout, delayedGetSources } = createDelayedGetSources({ + sources: [100, 200], + }); + const getSources = jest.fn(delayedGetSources); + + const { + inputElement, + getEnvironmentProps, + formElement, + } = createPlayground(createAutocomplete, { + onStateChange, + getSources, + }); + + const panelElement = document.createElement('div'); + + const { onMouseDown } = getEnvironmentProps({ + inputElement, + formElement, + panelElement, + }); + window.addEventListener('mousedown', onMouseDown); + + userEvent.type(inputElement, 'a'); + + await runAllMicroTasks(); + + // The search request is triggered + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + status: 'loading', + query: 'a', + }), + }) + ); + + userEvent.click(document.body); + + // The status is immediately set to "idle" and the panel is closed + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + status: 'idle', + isOpen: false, + query: 'a', + }), + }) + ); + + await defer(noop, timeout); + + // Once the request is settled, the state remains unchanged + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + status: 'idle', + isOpen: false, + }), + }) + ); + + expect(getSources).toHaveBeenCalledTimes(1); + }); + + test('keeps the panel closed on touchstart', async () => { const onStateChange = jest.fn(); const { timeout, delayedGetSources } = createDelayedGetSources({ sources: [100, 200], @@ -227,8 +294,9 @@ describe('concurrency', () => { }) ); - const customEvent = new CustomEvent('touchstart', { bubbles: true }); - window.document.dispatchEvent(customEvent); + window.document.dispatchEvent( + new CustomEvent('touchstart', { bubbles: true }) + ); // The status is immediately set to "idle" and the panel is closed expect(onStateChange).toHaveBeenLastCalledWith( From cb71f9cacf0ebc8a7351cd5b308dbe0eaf6021e0 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Thu, 16 Jun 2022 14:21:01 +0200 Subject: [PATCH 03/12] fix: move blur handling to environment mousedown event --- .../src/__tests__/getEnvironmentProps.test.ts | 261 +++++++++++++++++- .../src/__tests__/getInputProps.test.ts | 102 ++----- .../autocomplete-core/src/getPropGetters.ts | 103 ++++--- packages/autocomplete-core/src/onKeyDown.ts | 8 + 4 files changed, 344 insertions(+), 130 deletions(-) diff --git a/packages/autocomplete-core/src/__tests__/getEnvironmentProps.test.ts b/packages/autocomplete-core/src/__tests__/getEnvironmentProps.test.ts index 788688ab6..739e5d928 100644 --- a/packages/autocomplete-core/src/__tests__/getEnvironmentProps.test.ts +++ b/packages/autocomplete-core/src/__tests__/getEnvironmentProps.test.ts @@ -1,3 +1,5 @@ +import userEvent from '@testing-library/user-event'; + import { createPlayground, createSource, @@ -29,6 +31,202 @@ describe('getEnvironmentProps', () => { ); }); + describe('onMouseDown', () => { + test('is a noop when panel is not open and status is idle', () => { + const onStateChange = jest.fn(); + const { + getEnvironmentProps, + inputElement, + formElement, + } = createPlayground(createAutocomplete, { onStateChange }); + const panelElement = document.createElement('div'); + + const { onMouseDown } = getEnvironmentProps({ + inputElement, + formElement, + panelElement, + }); + window.addEventListener('mousedown', onMouseDown); + + // Dispatch MouseDown event on window + const customEvent = new CustomEvent('mousedown', { bubbles: true }); + window.dispatchEvent(customEvent); + + expect(onStateChange).not.toHaveBeenCalled(); + + window.removeEventListener('mousedown', onMouseDown); + }); + + test('is a noop when the event target is the input element', async () => { + const onStateChange = jest.fn(); + const { + getEnvironmentProps, + inputElement, + formElement, + } = createPlayground(createAutocomplete, { + onStateChange, + openOnFocus: true, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + const panelElement = document.createElement('div'); + + const { onMouseDown } = getEnvironmentProps({ + inputElement, + formElement, + panelElement, + }); + window.addEventListener('mousedown', onMouseDown); + + // Click input (focuses it, which opens the panel) + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + }), + }) + ); + + onStateChange.mockClear(); + + // Dispatch MouseDown event on the input (bubbles to window) + const customEvent = new CustomEvent('mousedown', { bubbles: true }); + inputElement.dispatchEvent(customEvent); + + await runAllMicroTasks(); + + expect(onStateChange).not.toHaveBeenCalled(); + + window.removeEventListener('mousedown', onMouseDown); + }); + + test('closes panel and resets `activeItemId` if the target is outside Autocomplete', async () => { + const onStateChange = jest.fn(); + const { + getEnvironmentProps, + inputElement, + formElement, + } = createPlayground(createAutocomplete, { + onStateChange, + openOnFocus: true, + defaultActiveItemId: 1, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + const panelElement = document.createElement('div'); + + const { onMouseDown } = getEnvironmentProps({ + inputElement, + formElement, + panelElement, + }); + window.addEventListener('mousedown', onMouseDown); + + // Click input (focuses it, which opens the panel) + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + }), + }) + ); + + onStateChange.mockClear(); + + // Dispatch MouseDown event on window (so, outside of Autocomplete) + const customEvent = new CustomEvent('mousedown', { bubbles: true }); + window.document.dispatchEvent(customEvent); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: false, + activeItemId: null, + }), + }) + ); + + window.removeEventListener('mousedown', onMouseDown); + }); + + test('does not close panel nor reset `activeItemId` if the target is outside Autocomplete in debug mode', async () => { + const onStateChange = jest.fn(); + const { + getEnvironmentProps, + inputElement, + formElement, + } = createPlayground(createAutocomplete, { + onStateChange, + openOnFocus: true, + defaultActiveItemId: 1, + debug: true, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + const panelElement = document.createElement('div'); + + const { onMouseDown } = getEnvironmentProps({ + inputElement, + formElement, + panelElement, + }); + window.addEventListener('mousedown', onMouseDown); + + // Click input (focuses it, which opens the panel) + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + }), + }) + ); + + onStateChange.mockClear(); + + // Dispatch MouseDown event on window (so, outside of Autocomplete) + const customEvent = new CustomEvent('mousedown', { bubbles: true }); + window.document.dispatchEvent(customEvent); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + + window.removeEventListener('mousedown', onMouseDown); + }); + }); + describe('onTouchStart', () => { test('is a noop when panel is not open and status is idle', () => { const onStateChange = jest.fn(); @@ -107,7 +305,7 @@ describe('getEnvironmentProps', () => { window.removeEventListener('touchstart', onTouchStart); }); - test('closes panel if the target is outside Autocomplete', async () => { + test('closes panel and resets `activeItemId` if the target is outside Autocomplete', async () => { const onStateChange = jest.fn(); const { getEnvironmentProps, @@ -116,6 +314,7 @@ describe('getEnvironmentProps', () => { } = createPlayground(createAutocomplete, { onStateChange, openOnFocus: true, + defaultActiveItemId: 1, getSources() { return [ createSource({ @@ -156,6 +355,66 @@ describe('getEnvironmentProps', () => { expect.objectContaining({ state: expect.objectContaining({ isOpen: false, + activeItemId: null, + }), + }) + ); + + window.removeEventListener('touchstart', onTouchStart); + }); + + test('does not close panel nor reset `activeItemId` if the target is outside Autocomplete in debug mode', async () => { + const onStateChange = jest.fn(); + const { + getEnvironmentProps, + inputElement, + formElement, + } = createPlayground(createAutocomplete, { + onStateChange, + openOnFocus: true, + defaultActiveItemId: 1, + debug: true, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + const panelElement = document.createElement('div'); + + const { onTouchStart } = getEnvironmentProps({ + inputElement, + formElement, + panelElement, + }); + window.addEventListener('touchstart', onTouchStart); + + // Focus input (opens the panel) + inputElement.focus(); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + }), + }) + ); + + onStateChange.mockClear(); + + // Dispatch TouchStart event on window (so, outside of Autocomplete) + const customEvent = new CustomEvent('touchstart', { bubbles: true }); + window.document.dispatchEvent(customEvent); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, }), }) ); diff --git a/packages/autocomplete-core/src/__tests__/getInputProps.test.ts b/packages/autocomplete-core/src/__tests__/getInputProps.test.ts index e94827e79..481af09d6 100644 --- a/packages/autocomplete-core/src/__tests__/getInputProps.test.ts +++ b/packages/autocomplete-core/src/__tests__/getInputProps.test.ts @@ -1034,6 +1034,7 @@ describe('getInputProps', () => { openOnFocus: true, initialState: { completion: 'a', + isOpen: true, collections: [ createCollection({ items: [{ label: '1' }, { label: '2' }], @@ -1042,7 +1043,6 @@ describe('getInputProps', () => { }, }); - inputElement.focus(); userEvent.type(inputElement, '{esc}'); expect(onStateChange).toHaveBeenLastCalledWith( @@ -1155,6 +1155,31 @@ describe('getInputProps', () => { expect(event.preventDefault).toHaveBeenCalledTimes(1); }); + test('closes the panel when no item was selected', async () => { + const onStateChange = jest.fn(); + const { inputElement } = createPlayground(createAutocomplete, { + onStateChange, + initialState: { + isOpen: true, + collections: [ + createCollection({ + items: [{ label: '1' }, { label: '2' }], + }), + ], + }, + }); + + userEvent.type(inputElement, '{enter}'); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ isOpen: false }), + }) + ); + }); + describe('Plain Enter', () => { test('calls onSelect with item URL', () => { const onSelect = jest.fn(); @@ -1893,81 +1918,6 @@ describe('getInputProps', () => { }); }); - describe('onBlur', () => { - test('resets activeItemId and isOpen', async () => { - const onStateChange = jest.fn(); - const { inputElement } = createPlayground(createAutocomplete, { - onStateChange, - defaultActiveItemId: 1, - openOnFocus: true, - }); - - inputElement.focus(); - inputElement.blur(); - - await runAllMicroTasks(); - - expect(onStateChange).toHaveBeenLastCalledWith( - expect.objectContaining({ - state: expect.objectContaining({ - activeItemId: null, - isOpen: false, - }), - }) - ); - }); - - test('does not reset activeItemId and isOpen when debug is true', () => { - const onStateChange = jest.fn(); - const { inputElement } = createPlayground(createAutocomplete, { - onStateChange, - debug: true, - defaultActiveItemId: 1, - openOnFocus: true, - shouldPanelOpen: () => true, - }); - - inputElement.focus(); - inputElement.blur(); - - expect(onStateChange).toHaveBeenLastCalledWith( - expect.objectContaining({ - state: expect.objectContaining({ - activeItemId: 1, - isOpen: true, - }), - }) - ); - }); - - test('does not reset activeItemId and isOpen on touch devices', () => { - const environment = { - ...global, - ontouchstart: () => {}, - }; - const onStateChange = jest.fn(); - const { inputElement } = createPlayground(createAutocomplete, { - environment, - onStateChange, - defaultActiveItemId: 1, - openOnFocus: true, - shouldPanelOpen: () => true, - }); - - inputElement.focus(); - inputElement.blur(); - - expect(onStateChange).toHaveBeenLastCalledWith( - expect.objectContaining({ - state: expect.objectContaining({ - activeItemId: 1, - isOpen: true, - }), - }) - ); - }); - }); - describe('onClick', () => { test('is a noop when the input is not focused', () => { const onStateChange = jest.fn(); diff --git a/packages/autocomplete-core/src/getPropGetters.ts b/packages/autocomplete-core/src/getPropGetters.ts index f4793af18..586acc2e2 100644 --- a/packages/autocomplete-core/src/getPropGetters.ts +++ b/packages/autocomplete-core/src/getPropGetters.ts @@ -1,3 +1,5 @@ +import { noop } from '@algolia/autocomplete-shared'; + import { onInput } from './onInput'; import { onKeyDown } from './onKeyDown'; import { @@ -31,46 +33,53 @@ export function getPropGetters< const getEnvironmentProps: GetEnvironmentProps = (providedProps) => { const { inputElement, formElement, panelElement, ...rest } = providedProps; - return { - // On touch devices, we do not rely on the native `blur` event of the - // input to close the panel, but rather on a custom `touchstart` event - // outside of the autocomplete elements. - // This ensures a working experience on mobile because we blur the input - // on touch devices when the user starts scrolling (`touchmove`). - // @TODO: support cases where there are multiple Autocomplete instances. - // Right now, a second instance makes this computation return false. - onTouchStart(event) { - // The `onTouchStart` event shouldn't trigger the `blur` handler when - // it's not an interaction with Autocomplete. We detect it with the - // following heuristics: - // - the panel is closed AND there are no pending requests - // (no interaction with the autocomplete, no future state updates) - // - OR the touched target is the input element (should open the panel) - const isAutocompleteInteraction = - store.getState().isOpen || !store.pendingRequests.isEmpty(); - - if (!isAutocompleteInteraction || event.target === inputElement) { - return; + function onMouseDownOrTouchStart(event: MouseEvent | TouchEvent) { + // The `onTouchStart`/`onMouseDown` events shouldn't trigger the `blur` + // handler when it's not an interaction with Autocomplete. + // We detect it with the following heuristics: + // - the panel is closed AND there are no pending requests + // (no interaction with the autocomplete, no future state updates) + // - OR the touched target is the input element (should open the panel) + const isAutocompleteInteraction = + store.getState().isOpen || !store.pendingRequests.isEmpty(); + + if (!isAutocompleteInteraction || event.target === inputElement) { + return; + } + + const isTargetWithinAutocomplete = [formElement, panelElement].some( + (contextNode) => { + return isOrContainsNode(contextNode, event.target as Node); } + ); - const isTargetWithinAutocomplete = [formElement, panelElement].some( - (contextNode) => { - return isOrContainsNode(contextNode, event.target as Node); - } - ); - - if (isTargetWithinAutocomplete === false) { - store.dispatch('blur', null); - - // If requests are still pending when the user closes the panel, they - // could reopen the panel once they resolve. - // We want to prevent any subsequent query from reopening the panel - // because it would result in an unsolicited UI behavior. - if (!props.debug) { - store.pendingRequests.cancelAll(); - } + if (isTargetWithinAutocomplete === false) { + store.dispatch('blur', null); + + // If requests are still pending when the user closes the panel, they + // could reopen the panel once they resolve. + // We want to prevent any subsequent query from reopening the panel + // because it would result in an unsolicited UI behavior. + if (!props.debug) { + store.pendingRequests.cancelAll(); } - }, + } + } + + return { + // We do not rely on the native `blur` event of the input to close the + // panel, but rather on a custom `touchstart`/`mousedown` event outside + // of the autocomplete elements. + // This ensures we don't mistakenly interpret interactions within the + // autocomplete (but outside of the input) as a signal to close the panel. + // For example, clicking reset button causes an input blur, but if + // `openOnFocus=true`, it shouldn't close the panel. + // On touch devices, scrolling results (`touchmove`) causes an input blur + // but shouldn't close the panel. + // @TODO: support cases where there are multiple Autocomplete instances. + // Right now, a second instance makes this computation return false. + onTouchStart: onMouseDownOrTouchStart, + onMouseDown: onMouseDownOrTouchStart, // When scrolling on touch devices (mobiles, tablets, etc.), we want to // mimic the native platform behavior where the input is blurred to // hide the virtual keyboard. This gives more vertical space to @@ -158,7 +167,6 @@ export function getPropGetters< store.dispatch('focus', null); } - const isTouchDevice = 'ontouchstart' in props.environment; const { inputElement, maxLength = 512, ...rest } = providedProps || {}; const activeItem = getActiveItem(store.getState()); @@ -207,21 +215,10 @@ export function getPropGetters< }); }, onFocus, - onBlur: () => { - // We do rely on the `blur` event on touch devices. - // See explanation in `onTouchStart`. - if (!isTouchDevice) { - store.dispatch('blur', null); - - // If requests are still pending when the user closes the panel, they - // could reopen the panel once they resolve. - // We want to prevent any subsequent query from reopening the panel - // because it would result in an unsolicited UI behavior. - if (!props.debug) { - store.pendingRequests.cancelAll(); - } - } - }, + // We don't rely on the `blur` event. + // See explanation in `onTouchStart`/`onMouseDown`. + // @MAJOR See if we need to keep this handler. + onBlur: noop, onClick: (event) => { // When the panel is closed and you click on the input while // the input is focused, the `onFocus` event is not triggered diff --git a/packages/autocomplete-core/src/onKeyDown.ts b/packages/autocomplete-core/src/onKeyDown.ts index 65b9135c3..a6b119240 100644 --- a/packages/autocomplete-core/src/onKeyDown.ts +++ b/packages/autocomplete-core/src/onKeyDown.ts @@ -114,6 +114,14 @@ export function onKeyDown({ .getState() .collections.every((collection) => collection.items.length === 0) ) { + // If requests are still pending when the panel closes, they could reopen + // the panel once they resolve. + // We want to prevent any subsequent query from reopening the panel + // because it would result in an unsolicited UI behavior. + if (!props.debug) { + store.pendingRequests.cancelAll(); + } + return; } From 157424a0a44c385006e9100f258573f9e9038c9b Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Thu, 16 Jun 2022 14:21:30 +0200 Subject: [PATCH 04/12] feat: expose onMouseDown on GetEnvironmentProps --- packages/autocomplete-core/src/types/AutocompletePropGetters.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/autocomplete-core/src/types/AutocompletePropGetters.ts b/packages/autocomplete-core/src/types/AutocompletePropGetters.ts index cef3c00c6..79e5d5a1b 100644 --- a/packages/autocomplete-core/src/types/AutocompletePropGetters.ts +++ b/packages/autocomplete-core/src/types/AutocompletePropGetters.ts @@ -25,6 +25,7 @@ export type GetEnvironmentProps = (props: { }) => { onTouchStart(event: TouchEvent): void; onTouchMove(event: TouchEvent): void; + onMouseDown(event: MouseEvent): void; }; export type GetRootProps = (props?: { From f236fe0e22823c16819ce1a53712990335b022a2 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Thu, 16 Jun 2022 14:42:53 +0200 Subject: [PATCH 05/12] test: test clearing in detached mode scenarios --- .../src/__tests__/detached.test.ts | 119 +++++++++++++++++- 1 file changed, 118 insertions(+), 1 deletion(-) diff --git a/packages/autocomplete-js/src/__tests__/detached.test.ts b/packages/autocomplete-js/src/__tests__/detached.test.ts index c0cd09780..ea90fe79c 100644 --- a/packages/autocomplete-js/src/__tests__/detached.test.ts +++ b/packages/autocomplete-js/src/__tests__/detached.test.ts @@ -1,8 +1,17 @@ -import { fireEvent, waitFor } from '@testing-library/dom'; +import { + fireEvent, + waitFor, + waitForElementToBeRemoved, +} from '@testing-library/dom'; +import userEvent from '@testing-library/user-event'; import { createMatchMedia } from '../../../../test/utils'; import { autocomplete } from '../autocomplete'; +beforeEach(() => { + document.body.innerHTML = ''; +}); + describe('detached', () => { beforeAll(() => { Object.defineProperty(window, 'matchMedia', { @@ -144,4 +153,112 @@ describe('detached', () => { expect(document.body).not.toHaveClass('aa-Detached'); }); }); + + test('stays open after clear when `openOnFocus` is `true`', async () => { + const container = document.createElement('div'); + document.body.appendChild(container); + autocomplete<{ label: string }>({ + id: 'autocomplete', + detachedMediaQuery: '', + openOnFocus: true, + container, + initialState: { + query: 'a', + isOpen: true, + }, + getSources() { + return [ + { + sourceId: 'testSource', + getItems() { + return [ + { label: 'Item 1' }, + { label: 'Item 2' }, + { label: 'Item 3' }, + ]; + }, + templates: { + item({ item }) { + return item.label; + }, + }, + }, + ]; + }, + }); + + // Wait for the panel to open + await waitFor(() => { + expect( + document.querySelector('.aa-Panel') + ).toBeInTheDocument(); + }); + + // Clear the query + userEvent.click( + document.querySelector('.aa-ClearButton') + ); + + // Ensures the overlay never disappears + await waitForElementToBeRemoved( + document.querySelector('.aa-DetachedOverlay') + ).catch(() => {}); + + await waitFor(() => { + expect(document.querySelector('.aa-DetachedOverlay')).toBeInTheDocument(); + expect(document.body).toHaveClass('aa-Detached'); + }); + }); + + test('closes after clear when `openOnFocus` is `false`', async () => { + const container = document.createElement('div'); + document.body.appendChild(container); + autocomplete<{ label: string }>({ + id: 'autocomplete', + detachedMediaQuery: '', + container, + initialState: { + query: 'a', + isOpen: true, + }, + getSources() { + return [ + { + sourceId: 'testSource', + getItems() { + return [ + { label: 'Item 1' }, + { label: 'Item 2' }, + { label: 'Item 3' }, + ]; + }, + templates: { + item({ item }) { + return item.label; + }, + }, + }, + ]; + }, + }); + + // Wait for the panel to open + await waitFor(() => { + expect( + document.querySelector('.aa-Panel') + ).toBeInTheDocument(); + }); + + // Clear the query + userEvent.click( + document.querySelector('.aa-ClearButton') + ); + + await waitFor(() => { + expect( + document.querySelector('.aa-DetachedOverlay') + ).not.toBeInTheDocument(); + expect(document.body).not.toHaveClass('aa-Detached'); + }); + }); }); From ac73f63af4c61ac286da11ca75fd3d3e8088c230 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Thu, 16 Jun 2022 16:46:06 +0200 Subject: [PATCH 06/12] fix: avoid triggering custom logic on escape on detached mode --- .../autocomplete-js/src/createAutocompleteDom.ts | 6 ------ packages/autocomplete-js/src/elements/Input.ts | 15 +-------------- 2 files changed, 1 insertion(+), 20 deletions(-) diff --git a/packages/autocomplete-js/src/createAutocompleteDom.ts b/packages/autocomplete-js/src/createAutocompleteDom.ts index 3ce303773..bdbd3da39 100644 --- a/packages/autocomplete-js/src/createAutocompleteDom.ts +++ b/packages/autocomplete-js/src/createAutocompleteDom.ts @@ -101,12 +101,6 @@ export function createAutocompleteDom({ getInputProps: propGetters.getInputProps, getInputPropsCore: autocomplete.getInputProps, autocompleteScopeApi, - onDetachedEscape: isDetached - ? () => { - autocomplete.setIsOpen(false); - setIsModalOpen(false); - } - : undefined, }); const inputWrapperPrefix = createDomElement('div', { diff --git a/packages/autocomplete-js/src/elements/Input.ts b/packages/autocomplete-js/src/elements/Input.ts index 8d06b9259..5a2957b89 100644 --- a/packages/autocomplete-js/src/elements/Input.ts +++ b/packages/autocomplete-js/src/elements/Input.ts @@ -14,7 +14,6 @@ type InputProps = { environment: AutocompleteEnvironment; getInputProps: AutocompletePropGetters['getInputProps']; getInputPropsCore: AutocompleteCoreApi['getInputProps']; - onDetachedEscape?(): void; state: AutocompleteState; }; @@ -24,7 +23,6 @@ export const Input: AutocompleteElement = ({ classNames, getInputProps, getInputPropsCore, - onDetachedEscape, state, ...props }) => { @@ -37,18 +35,7 @@ export const Input: AutocompleteElement = ({ ...autocompleteScopeApi, }); - setProperties(element, { - ...inputProps, - onKeyDown(event: KeyboardEvent) { - if (onDetachedEscape && event.key === 'Escape') { - event.preventDefault(); - onDetachedEscape(); - return; - } - - inputProps.onKeyDown(event); - }, - }); + setProperties(element, inputProps); return element; }; From 77bb200a4b513637e67d2de5131ac1bcc9bd1379 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Fri, 17 Jun 2022 09:43:09 +0200 Subject: [PATCH 07/12] refactor: no need to inline --- packages/autocomplete-core/src/__tests__/concurrency.test.ts | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/packages/autocomplete-core/src/__tests__/concurrency.test.ts b/packages/autocomplete-core/src/__tests__/concurrency.test.ts index bc8ecc100..14616d59f 100644 --- a/packages/autocomplete-core/src/__tests__/concurrency.test.ts +++ b/packages/autocomplete-core/src/__tests__/concurrency.test.ts @@ -294,9 +294,8 @@ describe('concurrency', () => { }) ); - window.document.dispatchEvent( - new CustomEvent('touchstart', { bubbles: true }) - ); + const customEvent = new CustomEvent('touchstart', { bubbles: true }); + window.document.dispatchEvent(customEvent); // The status is immediately set to "idle" and the panel is closed expect(onStateChange).toHaveBeenLastCalledWith( From c5002daa9cf3d38d03c4a65c9dc3ced41b620863 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Fri, 17 Jun 2022 10:15:39 +0200 Subject: [PATCH 08/12] chore: move comment where more relevant --- packages/autocomplete-core/src/getPropGetters.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/autocomplete-core/src/getPropGetters.ts b/packages/autocomplete-core/src/getPropGetters.ts index 586acc2e2..c4921689a 100644 --- a/packages/autocomplete-core/src/getPropGetters.ts +++ b/packages/autocomplete-core/src/getPropGetters.ts @@ -47,6 +47,8 @@ export function getPropGetters< return; } + // @TODO: support cases where there are multiple Autocomplete instances. + // Right now, a second instance makes this computation return false. const isTargetWithinAutocomplete = [formElement, panelElement].some( (contextNode) => { return isOrContainsNode(contextNode, event.target as Node); @@ -76,8 +78,6 @@ export function getPropGetters< // `openOnFocus=true`, it shouldn't close the panel. // On touch devices, scrolling results (`touchmove`) causes an input blur // but shouldn't close the panel. - // @TODO: support cases where there are multiple Autocomplete instances. - // Right now, a second instance makes this computation return false. onTouchStart: onMouseDownOrTouchStart, onMouseDown: onMouseDownOrTouchStart, // When scrolling on touch devices (mobiles, tablets, etc.), we want to From fa20108ae4b3d1611970e80da4e15661428753c2 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Fri, 17 Jun 2022 10:19:29 +0200 Subject: [PATCH 09/12] test: explicitly set parameter --- packages/autocomplete-js/src/__tests__/detached.test.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/autocomplete-js/src/__tests__/detached.test.ts b/packages/autocomplete-js/src/__tests__/detached.test.ts index ea90fe79c..97e3dc635 100644 --- a/packages/autocomplete-js/src/__tests__/detached.test.ts +++ b/packages/autocomplete-js/src/__tests__/detached.test.ts @@ -216,6 +216,7 @@ describe('detached', () => { autocomplete<{ label: string }>({ id: 'autocomplete', detachedMediaQuery: '', + openOnFocus: false, container, initialState: { query: 'a', From 80b799a42a4bac39c4cde48c11cafa9a3d0fef7e Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Tue, 21 Jun 2022 11:23:02 +0200 Subject: [PATCH 10/12] test: assume element exists --- packages/autocomplete-js/src/__tests__/detached.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/autocomplete-js/src/__tests__/detached.test.ts b/packages/autocomplete-js/src/__tests__/detached.test.ts index 97e3dc635..98c1782e7 100644 --- a/packages/autocomplete-js/src/__tests__/detached.test.ts +++ b/packages/autocomplete-js/src/__tests__/detached.test.ts @@ -196,7 +196,7 @@ describe('detached', () => { // Clear the query userEvent.click( - document.querySelector('.aa-ClearButton') + document.querySelector('.aa-ClearButton')! ); // Ensures the overlay never disappears From 4b80cc1fefc2d6640c94bc4770b5dbf5a06a05f1 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Tue, 21 Jun 2022 14:41:50 +0200 Subject: [PATCH 11/12] feat: trigger a blur when hitting Tab or Shift+Tab --- .../src/__tests__/getInputProps.test.ts | 174 ++++++++++++++++++ packages/autocomplete-core/src/onKeyDown.ts | 8 + .../src/__tests__/autocomplete.test.ts | 109 +++++++++++ .../src/__tests__/detached.test.ts | 112 ++++++++++- .../src/createAutocompleteDom.ts | 3 + .../autocomplete-js/src/elements/Input.ts | 15 +- 6 files changed, 419 insertions(+), 2 deletions(-) diff --git a/packages/autocomplete-core/src/__tests__/getInputProps.test.ts b/packages/autocomplete-core/src/__tests__/getInputProps.test.ts index 481af09d6..eca8bb8d4 100644 --- a/packages/autocomplete-core/src/__tests__/getInputProps.test.ts +++ b/packages/autocomplete-core/src/__tests__/getInputProps.test.ts @@ -1689,6 +1689,180 @@ describe('getInputProps', () => { }); }); }); + + describe('Tab', () => { + test('closes the panel and resets `activeItemId`', async () => { + const onStateChange = jest.fn(); + const { inputElement } = createPlayground(createAutocomplete, { + onStateChange, + openOnFocus: true, + defaultActiveItemId: 1, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + + userEvent.tab(); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: false, + activeItemId: null, + }), + }) + ); + }); + + test('does not close closes the panel nor reset `activeItemId` in debug mode', async () => { + const onStateChange = jest.fn(); + const { inputElement } = createPlayground(createAutocomplete, { + onStateChange, + debug: true, + openOnFocus: true, + defaultActiveItemId: 1, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + + userEvent.tab(); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + }); + }); + + describe('Tab+Shift', () => { + test('closes the panel and resets `activeItemId`', async () => { + const onStateChange = jest.fn(); + const { inputElement } = createPlayground(createAutocomplete, { + onStateChange, + openOnFocus: true, + defaultActiveItemId: 1, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + + userEvent.tab({ shift: true }); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: false, + activeItemId: null, + }), + }) + ); + }); + + test('does not close closes the panel nor reset `activeItemId` in debug mode', async () => { + const onStateChange = jest.fn(); + const { inputElement } = createPlayground(createAutocomplete, { + onStateChange, + debug: true, + openOnFocus: true, + defaultActiveItemId: 1, + getSources() { + return [ + createSource({ + getItems: () => [{ label: '1' }], + }), + ]; + }, + }); + + userEvent.click(inputElement); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + + userEvent.tab({ shift: true }); + + await runAllMicroTasks(); + + expect(onStateChange).toHaveBeenLastCalledWith( + expect.objectContaining({ + state: expect.objectContaining({ + isOpen: true, + activeItemId: 1, + }), + }) + ); + }); + }); }); describe('onFocus', () => { diff --git a/packages/autocomplete-core/src/onKeyDown.ts b/packages/autocomplete-core/src/onKeyDown.ts index a6b119240..c0b50a34c 100644 --- a/packages/autocomplete-core/src/onKeyDown.ts +++ b/packages/autocomplete-core/src/onKeyDown.ts @@ -100,6 +100,14 @@ export function onKeyDown({ store.dispatch(event.key, null); + // Hitting the `Escape` key signals the end of a user interaction with the + // autocomplete. At this point, we should ignore any requests that are still + // pending and could reopen the panel once they resolve, because that would + // result in an unsolicited UI behavior. + store.pendingRequests.cancelAll(); + } else if (event.key === 'Tab') { + store.dispatch('blur', null); + // Hitting the `Escape` key signals the end of a user interaction with the // autocomplete. At this point, we should ignore any requests that are still // pending and could reopen the panel once they resolve, because that would diff --git a/packages/autocomplete-js/src/__tests__/autocomplete.test.ts b/packages/autocomplete-js/src/__tests__/autocomplete.test.ts index 4b01c21bc..39269e595 100644 --- a/packages/autocomplete-js/src/__tests__/autocomplete.test.ts +++ b/packages/autocomplete-js/src/__tests__/autocomplete.test.ts @@ -1,5 +1,6 @@ import * as autocompleteShared from '@algolia/autocomplete-shared'; import { fireEvent, waitFor } from '@testing-library/dom'; +import userEvent from '@testing-library/user-event'; import { castToJestMock, @@ -17,6 +18,10 @@ jest.mock('@algolia/autocomplete-shared', () => { }; }); +beforeEach(() => { + document.body.innerHTML = ''; +}); + describe('autocomplete-js', () => { test('renders with default options', () => { const container = document.createElement('div'); @@ -562,4 +567,108 @@ describe('autocomplete-js', () => { expect(input).toHaveValue('a'); }); + + test('closes the panel and focuses the next focusable element on `Tab`', async () => { + const container = document.createElement('div'); + document.body.appendChild(container); + autocomplete<{ label: string }>({ + id: 'autocomplete', + container, + initialState: { + query: 'a', + isOpen: true, + }, + getSources() { + return [ + { + sourceId: 'testSource', + getItems() { + return [ + { label: 'Item 1' }, + { label: 'Item 2' }, + { label: 'Item 3' }, + ]; + }, + templates: { + item({ item }) { + return item.label; + }, + }, + }, + ]; + }, + }); + + // Wait for the panel to open + await waitFor(() => { + expect( + document.querySelector('.aa-Panel') + ).toBeInTheDocument(); + }); + + userEvent.click(document.querySelector('.aa-Input')!); + userEvent.tab(); + + await waitFor(() => { + expect( + document.querySelector('.aa-DetachedOverlay') + ).not.toBeInTheDocument(); + expect(document.body).not.toHaveClass('aa-Detached'); + expect(document.activeElement).toEqual( + document.querySelector('.aa-ClearButton') + ); + }); + }); + + test('closes the panel and focuses the next focusable element on `Shift+Tab`', async () => { + const container = document.createElement('div'); + document.body.appendChild(container); + autocomplete<{ label: string }>({ + id: 'autocomplete', + container, + initialState: { + query: 'a', + isOpen: true, + }, + getSources() { + return [ + { + sourceId: 'testSource', + getItems() { + return [ + { label: 'Item 1' }, + { label: 'Item 2' }, + { label: 'Item 3' }, + ]; + }, + templates: { + item({ item }) { + return item.label; + }, + }, + }, + ]; + }, + }); + + // Wait for the panel to open + await waitFor(() => { + expect( + document.querySelector('.aa-Panel') + ).toBeInTheDocument(); + }); + + userEvent.click(document.querySelector('.aa-Input')!); + userEvent.tab({ shift: true }); + + await waitFor(() => { + expect( + document.querySelector('.aa-DetachedOverlay') + ).not.toBeInTheDocument(); + expect(document.body).not.toHaveClass('aa-Detached'); + expect(document.activeElement).toEqual( + document.querySelector('.aa-SubmitButton') + ); + }); + }); }); diff --git a/packages/autocomplete-js/src/__tests__/detached.test.ts b/packages/autocomplete-js/src/__tests__/detached.test.ts index 98c1782e7..4f92eacdc 100644 --- a/packages/autocomplete-js/src/__tests__/detached.test.ts +++ b/packages/autocomplete-js/src/__tests__/detached.test.ts @@ -252,7 +252,7 @@ describe('detached', () => { // Clear the query userEvent.click( - document.querySelector('.aa-ClearButton') + document.querySelector('.aa-ClearButton')! ); await waitFor(() => { @@ -262,4 +262,114 @@ describe('detached', () => { expect(document.body).not.toHaveClass('aa-Detached'); }); }); + + test('stays open and focuses the next focusable element on `Tab`', async () => { + const container = document.createElement('div'); + document.body.appendChild(container); + autocomplete<{ label: string }>({ + id: 'autocomplete', + detachedMediaQuery: '', + container, + initialState: { + query: 'a', + isOpen: true, + }, + getSources() { + return [ + { + sourceId: 'testSource', + getItems() { + return [ + { label: 'Item 1' }, + { label: 'Item 2' }, + { label: 'Item 3' }, + ]; + }, + templates: { + item({ item }) { + return item.label; + }, + }, + }, + ]; + }, + }); + + // Wait for the panel to open + await waitFor(() => { + expect( + document.querySelector('.aa-Panel') + ).toBeInTheDocument(); + }); + + userEvent.tab(); + + // Ensures the overlay never disappears + await waitForElementToBeRemoved( + document.querySelector('.aa-DetachedOverlay') + ).catch(() => {}); + + await waitFor(() => { + expect(document.querySelector('.aa-DetachedOverlay')).toBeInTheDocument(); + expect(document.body).toHaveClass('aa-Detached'); + expect(document.activeElement).toEqual( + document.querySelector('.aa-ClearButton') + ); + }); + }); + + test('stays open and focuses the previous focusable element on `Shift+Tab`', async () => { + const container = document.createElement('div'); + document.body.appendChild(container); + autocomplete<{ label: string }>({ + id: 'autocomplete', + detachedMediaQuery: '', + container, + initialState: { + query: 'a', + isOpen: true, + }, + getSources() { + return [ + { + sourceId: 'testSource', + getItems() { + return [ + { label: 'Item 1' }, + { label: 'Item 2' }, + { label: 'Item 3' }, + ]; + }, + templates: { + item({ item }) { + return item.label; + }, + }, + }, + ]; + }, + }); + + // Wait for the panel to open + await waitFor(() => { + expect( + document.querySelector('.aa-Panel') + ).toBeInTheDocument(); + }); + + userEvent.tab({ shift: true }); + + // Ensures the overlay never disappears + await waitForElementToBeRemoved( + document.querySelector('.aa-DetachedOverlay') + ).catch(() => {}); + + await waitFor(() => { + expect(document.querySelector('.aa-DetachedOverlay')).toBeInTheDocument(); + expect(document.body).toHaveClass('aa-Detached'); + expect(document.activeElement).toEqual( + document.querySelector('.aa-SubmitButton') + ); + }); + }); }); diff --git a/packages/autocomplete-js/src/createAutocompleteDom.ts b/packages/autocomplete-js/src/createAutocompleteDom.ts index bdbd3da39..2bfe84019 100644 --- a/packages/autocomplete-js/src/createAutocompleteDom.ts +++ b/packages/autocomplete-js/src/createAutocompleteDom.ts @@ -4,6 +4,7 @@ import { AutocompleteScopeApi, BaseItem, } from '@algolia/autocomplete-core'; +import { noop } from '@algolia/autocomplete-shared'; import { ClearIcon, Input, LoadingIcon, SearchIcon } from './elements'; import { getCreateDomElement } from './getCreateDomElement'; @@ -101,6 +102,8 @@ export function createAutocompleteDom({ getInputProps: propGetters.getInputProps, getInputPropsCore: autocomplete.getInputProps, autocompleteScopeApi, + // In detached mode we don't want to close the panel when hittin `Tab`. + onDetachedTab: isDetached ? noop : undefined, }); const inputWrapperPrefix = createDomElement('div', { diff --git a/packages/autocomplete-js/src/elements/Input.ts b/packages/autocomplete-js/src/elements/Input.ts index 5a2957b89..897889675 100644 --- a/packages/autocomplete-js/src/elements/Input.ts +++ b/packages/autocomplete-js/src/elements/Input.ts @@ -14,6 +14,7 @@ type InputProps = { environment: AutocompleteEnvironment; getInputProps: AutocompletePropGetters['getInputProps']; getInputPropsCore: AutocompleteCoreApi['getInputProps']; + onDetachedTab?(): void; state: AutocompleteState; }; @@ -23,6 +24,7 @@ export const Input: AutocompleteElement = ({ classNames, getInputProps, getInputPropsCore, + onDetachedTab, state, ...props }) => { @@ -35,7 +37,18 @@ export const Input: AutocompleteElement = ({ ...autocompleteScopeApi, }); - setProperties(element, inputProps); + setProperties(element, { + ...inputProps, + onKeyDown(event: KeyboardEvent) { + if (onDetachedTab && event.key === 'Tab') { + onDetachedTab(); + + return; + } + + inputProps.onKeyDown(event); + }, + }); return element; }; From 22ff47857af4cddc1d8e797b5e83bf9cb5f3fce4 Mon Sep 17 00:00:00 2001 From: Sarah Dayan <5370675+sarahdayan@users.noreply.github.com> Date: Tue, 21 Jun 2022 15:16:03 +0200 Subject: [PATCH 12/12] refactor: forward detached status to input component --- packages/autocomplete-js/src/createAutocompleteDom.ts | 4 +--- packages/autocomplete-js/src/elements/Input.ts | 9 ++++----- 2 files changed, 5 insertions(+), 8 deletions(-) diff --git a/packages/autocomplete-js/src/createAutocompleteDom.ts b/packages/autocomplete-js/src/createAutocompleteDom.ts index 2bfe84019..7803d71d4 100644 --- a/packages/autocomplete-js/src/createAutocompleteDom.ts +++ b/packages/autocomplete-js/src/createAutocompleteDom.ts @@ -4,7 +4,6 @@ import { AutocompleteScopeApi, BaseItem, } from '@algolia/autocomplete-core'; -import { noop } from '@algolia/autocomplete-shared'; import { ClearIcon, Input, LoadingIcon, SearchIcon } from './elements'; import { getCreateDomElement } from './getCreateDomElement'; @@ -102,8 +101,7 @@ export function createAutocompleteDom({ getInputProps: propGetters.getInputProps, getInputPropsCore: autocomplete.getInputProps, autocompleteScopeApi, - // In detached mode we don't want to close the panel when hittin `Tab`. - onDetachedTab: isDetached ? noop : undefined, + isDetached, }); const inputWrapperPrefix = createDomElement('div', { diff --git a/packages/autocomplete-js/src/elements/Input.ts b/packages/autocomplete-js/src/elements/Input.ts index 897889675..b687231a0 100644 --- a/packages/autocomplete-js/src/elements/Input.ts +++ b/packages/autocomplete-js/src/elements/Input.ts @@ -14,7 +14,7 @@ type InputProps = { environment: AutocompleteEnvironment; getInputProps: AutocompletePropGetters['getInputProps']; getInputPropsCore: AutocompleteCoreApi['getInputProps']; - onDetachedTab?(): void; + isDetached: boolean; state: AutocompleteState; }; @@ -24,7 +24,7 @@ export const Input: AutocompleteElement = ({ classNames, getInputProps, getInputPropsCore, - onDetachedTab, + isDetached, state, ...props }) => { @@ -40,9 +40,8 @@ export const Input: AutocompleteElement = ({ setProperties(element, { ...inputProps, onKeyDown(event: KeyboardEvent) { - if (onDetachedTab && event.key === 'Tab') { - onDetachedTab(); - + // In detached mode we don't want to close the panel when hittin `Tab`. + if (isDetached && event.key === 'Tab') { return; }