diff --git a/CHANGELOG.md b/CHANGELOG.md index 752eaaeb13..27ef1108da 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -14,6 +14,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Improve typeahead search logic ([#1051](https://github.com/tailwindlabs/headlessui/pull/1051)) - Improve overal codebase, use modern tech like `esbuild` and TypeScript 4! ([#1055](https://github.com/tailwindlabs/headlessui/pull/1055)) - Improve build files ([#1078](https://github.com/tailwindlabs/headlessui/pull/1078)) +- Ensure typeahead stays on same item if it still matches ([#1098](https://github.com/tailwindlabs/headlessui/pull/1098)) ### Added @@ -27,6 +28,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Improve typeahead search logic ([#1051](https://github.com/tailwindlabs/headlessui/pull/1051)) - Improve overal codebase, use modern tech like `esbuild` and TypeScript 4! ([#1055](https://github.com/tailwindlabs/headlessui/pull/1055)) - Improve build files ([#1078](https://github.com/tailwindlabs/headlessui/pull/1078)) +- Ensure typeahead stays on same item if it still matches ([#1098](https://github.com/tailwindlabs/headlessui/pull/1098)) ### Added diff --git a/packages/@headlessui-react/src/components/listbox/listbox.test.tsx b/packages/@headlessui-react/src/components/listbox/listbox.test.tsx index eea9fc118c..2a48fca509 100644 --- a/packages/@headlessui-react/src/components/listbox/listbox.test.tsx +++ b/packages/@headlessui-react/src/components/listbox/listbox.test.tsx @@ -3108,6 +3108,79 @@ describe('Keyboard interactions', () => { assertActiveListboxOption(options[3]) }) ) + + it( + 'should stay on the same item while keystrokes still match', + suppressConsoleLogs(async () => { + render( + + Trigger + + alice + bob + charlie + bob + + + ) + + // Open listbox + await click(getListboxButton()) + + let options = getListboxOptions() + + // --- + + // Reset: Go to first option + await press(Keys.Home) + + // Search for "b" in "bob" + await type(word('b')) + + // We should be on the first `bob` + assertActiveListboxOption(options[1]) + + // Search for "b" in "bob" again + await type(word('b')) + + // We should be on the next `bob` + assertActiveListboxOption(options[3]) + + // --- + + // Reset: Go to first option + await press(Keys.Home) + + // Search for "bo" in "bob" + await type(word('bo')) + + // We should be on the first `bob` + assertActiveListboxOption(options[1]) + + // Search for "bo" in "bob" again + await type(word('bo')) + + // We should be on the next `bob` + assertActiveListboxOption(options[3]) + + // --- + + // Reset: Go to first option + await press(Keys.Home) + + // Search for "bob" in "bob" + await type(word('bob')) + + // We should be on the first `bob` + assertActiveListboxOption(options[1]) + + // Search for "bob" in "bob" again + await type(word('bob')) + + // We should be on the next `bob` + assertActiveListboxOption(options[3]) + }) + ) }) }) diff --git a/packages/@headlessui-react/src/components/listbox/listbox.tsx b/packages/@headlessui-react/src/components/listbox/listbox.tsx index da2741cef6..1931534f6d 100644 --- a/packages/@headlessui-react/src/components/listbox/listbox.tsx +++ b/packages/@headlessui-react/src/components/listbox/listbox.tsx @@ -130,13 +130,16 @@ let reducers: { if (state.disabled) return state if (state.listboxState === ListboxStates.Closed) return state + let wasAlreadySearching = state.searchQuery !== '' + let offset = wasAlreadySearching ? 0 : 1 + let searchQuery = state.searchQuery + action.value.toLowerCase() let reOrderedOptions = state.activeOptionIndex !== null ? state.options - .slice(state.activeOptionIndex + 1) - .concat(state.options.slice(0, state.activeOptionIndex + 1)) + .slice(state.activeOptionIndex + offset) + .concat(state.options.slice(0, state.activeOptionIndex + offset)) : state.options let matchingOption = reOrderedOptions.find( diff --git a/packages/@headlessui-react/src/components/menu/menu.test.tsx b/packages/@headlessui-react/src/components/menu/menu.test.tsx index 1fc853d9b6..f8c36f60b1 100644 --- a/packages/@headlessui-react/src/components/menu/menu.test.tsx +++ b/packages/@headlessui-react/src/components/menu/menu.test.tsx @@ -2698,6 +2698,79 @@ describe('Keyboard interactions', () => { assertMenuLinkedWithMenuItem(items[3]) }) ) + + it( + 'should stay on the same item while keystrokes still match', + suppressConsoleLogs(async () => { + render( + + Trigger + + alice + bob + charlie + bob + + + ) + + // Open menu + await click(getMenuButton()) + + let items = getMenuItems() + + // --- + + // Reset: Go to first item + await press(Keys.Home) + + // Search for "b" in "bob" + await type(word('b')) + + // We should be on the first `bob` + assertMenuLinkedWithMenuItem(items[1]) + + // Search for "b" in "bob" again + await type(word('b')) + + // We should be on the next `bob` + assertMenuLinkedWithMenuItem(items[3]) + + // --- + + // Reset: Go to first item + await press(Keys.Home) + + // Search for "bo" in "bob" + await type(word('bo')) + + // We should be on the first `bob` + assertMenuLinkedWithMenuItem(items[1]) + + // Search for "bo" in "bob" again + await type(word('bo')) + + // We should be on the next `bob` + assertMenuLinkedWithMenuItem(items[3]) + + // --- + + // Reset: Go to first item + await press(Keys.Home) + + // Search for "bob" in "bob" + await type(word('bob')) + + // We should be on the first `bob` + assertMenuLinkedWithMenuItem(items[1]) + + // Search for "bob" in "bob" again + await type(word('bob')) + + // We should be on the next `bob` + assertMenuLinkedWithMenuItem(items[3]) + }) + ) }) }) diff --git a/packages/@headlessui-react/src/components/menu/menu.tsx b/packages/@headlessui-react/src/components/menu/menu.tsx index 962549fdf6..04f34b39b9 100644 --- a/packages/@headlessui-react/src/components/menu/menu.tsx +++ b/packages/@headlessui-react/src/components/menu/menu.tsx @@ -99,13 +99,15 @@ let reducers: { return { ...state, searchQuery: '', activeItemIndex } }, [ActionTypes.Search]: (state, action) => { + let wasAlreadySearching = state.searchQuery !== '' + let offset = wasAlreadySearching ? 0 : 1 let searchQuery = state.searchQuery + action.value.toLowerCase() let reOrderedItems = state.activeItemIndex !== null ? state.items - .slice(state.activeItemIndex + 1) - .concat(state.items.slice(0, state.activeItemIndex + 1)) + .slice(state.activeItemIndex + offset) + .concat(state.items.slice(0, state.activeItemIndex + offset)) : state.items let matchingItem = reOrderedItems.find( diff --git a/packages/@headlessui-vue/src/components/listbox/listbox.test.tsx b/packages/@headlessui-vue/src/components/listbox/listbox.test.tsx index c3cd561fef..4b4c187cfe 100644 --- a/packages/@headlessui-vue/src/components/listbox/listbox.test.tsx +++ b/packages/@headlessui-vue/src/components/listbox/listbox.test.tsx @@ -3231,6 +3231,82 @@ describe('Keyboard interactions', () => { assertActiveListboxOption(options[3]) }) ) + + it( + 'should stay on the same item while keystrokes still match', + suppressConsoleLogs(async () => { + renderTemplate({ + template: html` + + Trigger + + alice + bob + charlie + bob + + + `, + setup: () => ({ value: ref(null) }), + }) + + // Open listbox + await click(getListboxButton()) + + let options = getListboxOptions() + + // --- + + // Reset: Go to first option + await press(Keys.Home) + + // Search for "b" in "bob" + await type(word('b')) + + // We should be on the first `bob` + assertActiveListboxOption(options[1]) + + // Search for "b" in "bob" again + await type(word('b')) + + // We should be on the next `bob` + assertActiveListboxOption(options[3]) + + // --- + + // Reset: Go to first option + await press(Keys.Home) + + // Search for "bo" in "bob" + await type(word('bo')) + + // We should be on the first `bob` + assertActiveListboxOption(options[1]) + + // Search for "bo" in "bob" again + await type(word('bo')) + + // We should be on the next `bob` + assertActiveListboxOption(options[3]) + + // --- + + // Reset: Go to first option + await press(Keys.Home) + + // Search for "bob" in "bob" + await type(word('bob')) + + // We should be on the first `bob` + assertActiveListboxOption(options[1]) + + // Search for "bob" in "bob" again + await type(word('bob')) + + // We should be on the next `bob` + assertActiveListboxOption(options[3]) + }) + ) }) }) diff --git a/packages/@headlessui-vue/src/components/listbox/listbox.ts b/packages/@headlessui-vue/src/components/listbox/listbox.ts index 812192da8f..1032dade3c 100644 --- a/packages/@headlessui-vue/src/components/listbox/listbox.ts +++ b/packages/@headlessui-vue/src/components/listbox/listbox.ts @@ -143,13 +143,16 @@ export let Listbox = defineComponent({ if (props.disabled) return if (listboxState.value === ListboxStates.Closed) return + let wasAlreadySearching = searchQuery.value !== '' + let offset = wasAlreadySearching ? 0 : 1 + searchQuery.value += value.toLowerCase() let reOrderedOptions = activeOptionIndex.value !== null ? options.value - .slice(activeOptionIndex.value + 1) - .concat(options.value.slice(0, activeOptionIndex.value + 1)) + .slice(activeOptionIndex.value + offset) + .concat(options.value.slice(0, activeOptionIndex.value + offset)) : options.value let matchingOption = reOrderedOptions.find( diff --git a/packages/@headlessui-vue/src/components/menu/menu.test.tsx b/packages/@headlessui-vue/src/components/menu/menu.test.tsx index 4d033b457d..3f5b4e7c75 100644 --- a/packages/@headlessui-vue/src/components/menu/menu.test.tsx +++ b/packages/@headlessui-vue/src/components/menu/menu.test.tsx @@ -2792,6 +2792,79 @@ describe('Keyboard interactions', () => { // We should be on the second `bob` assertMenuLinkedWithMenuItem(items[3]) }) + + it( + 'should stay on the same item while keystrokes still match', + suppressConsoleLogs(async () => { + renderTemplate(jsx` + + Trigger + + alice + bob + charlie + bob + + + `) + + // Open menu + await click(getMenuButton()) + + let items = getMenuItems() + + // --- + + // Reset: Go to first item + await press(Keys.Home) + + // Search for "b" in "bob" + await type(word('b')) + + // We should be on the first `bob` + assertMenuLinkedWithMenuItem(items[1]) + + // Search for "b" in "bob" again + await type(word('b')) + + // We should be on the next `bob` + assertMenuLinkedWithMenuItem(items[3]) + + // --- + + // Reset: Go to first item + await press(Keys.Home) + + // Search for "bo" in "bob" + await type(word('bo')) + + // We should be on the first `bob` + assertMenuLinkedWithMenuItem(items[1]) + + // Search for "bo" in "bob" again + await type(word('bo')) + + // We should be on the next `bob` + assertMenuLinkedWithMenuItem(items[3]) + + // --- + + // Reset: Go to first item + await press(Keys.Home) + + // Search for "bob" in "bob" + await type(word('bob')) + + // We should be on the first `bob` + assertMenuLinkedWithMenuItem(items[1]) + + // Search for "bob" in "bob" again + await type(word('bob')) + + // We should be on the next `bob` + assertMenuLinkedWithMenuItem(items[3]) + }) + ) }) }) diff --git a/packages/@headlessui-vue/src/components/menu/menu.ts b/packages/@headlessui-vue/src/components/menu/menu.ts index f154625ca2..7fc505d27e 100644 --- a/packages/@headlessui-vue/src/components/menu/menu.ts +++ b/packages/@headlessui-vue/src/components/menu/menu.ts @@ -106,13 +106,15 @@ export let Menu = defineComponent({ activeItemIndex.value = nextActiveItemIndex }, search(value: string) { + let wasAlreadySearching = searchQuery.value !== '' + let offset = wasAlreadySearching ? 0 : 1 searchQuery.value += value.toLowerCase() let reOrderedItems = activeItemIndex.value !== null ? items.value - .slice(activeItemIndex.value + 1) - .concat(items.value.slice(0, activeItemIndex.value + 1)) + .slice(activeItemIndex.value + offset) + .concat(items.value.slice(0, activeItemIndex.value + offset)) : items.value let matchingItem = reOrderedItems.find(