Skip to content

Commit caa3ef8

Browse files
authored
fix(behaviour): clear completion on "reset" (#1215)
* chore(test): add real attributes to playground input * test(debouncing): prevent unrealistic behaviour * fix(behaviour): clear completion on "reset" * chore: use right word in comment
1 parent ce8c78e commit caa3ef8

File tree

7 files changed

+102
-6
lines changed

7 files changed

+102
-6
lines changed

packages/autocomplete-core/src/__tests__/completion.test.ts

+31
Original file line numberDiff line numberDiff line change
@@ -161,4 +161,35 @@ describe('completion', () => {
161161
})
162162
);
163163
});
164+
165+
test('clears completion on "reset"', () => {
166+
const { inputElement, resetElement } = createPlayground(
167+
createAutocomplete,
168+
{
169+
openOnFocus: true,
170+
initialState: {
171+
collections: [
172+
createCollection({
173+
source: {
174+
getItemInputValue({ item }) {
175+
return item.label;
176+
},
177+
},
178+
items: [{ label: '1' }, { label: '2' }],
179+
}),
180+
],
181+
},
182+
}
183+
);
184+
inputElement.focus();
185+
186+
userEvent.type(inputElement, 'Some text to make sure reset shows');
187+
expect(inputElement).toHaveValue('Some text to make sure reset shows');
188+
189+
userEvent.type(inputElement, '{arrowdown}');
190+
expect(inputElement).toHaveValue('1');
191+
192+
resetElement.click();
193+
expect(inputElement).toHaveValue('');
194+
});
164195
});

packages/autocomplete-core/src/__tests__/debouncing.test.ts

+23-2
Original file line numberDiff line numberDiff line change
@@ -42,15 +42,31 @@ describe('debouncing', () => {
4242
);
4343
});
4444

45-
test('triggers subsequent queries after reopening the panel', async () => {
45+
test('triggers subsequent queries after closing and reopening the panel', async () => {
4646
const onStateChange = jest.fn();
4747
const getItems = jest.fn(({ query }) => [{ label: query }]);
4848
const { inputElement } = createPlayground(createAutocomplete, {
4949
onStateChange,
50+
openOnFocus: true,
5051
getSources: () => debounced([createSource({ getItems })]),
5152
});
5253

53-
userEvent.type(inputElement, 'abc{esc}');
54+
inputElement.focus();
55+
userEvent.type(inputElement, 'ab');
56+
await defer(noop, delay);
57+
58+
expect(onStateChange).toHaveBeenLastCalledWith(
59+
expect.objectContaining({
60+
state: expect.objectContaining({
61+
status: 'idle',
62+
isOpen: true,
63+
}),
64+
})
65+
);
66+
expect(getItems).toHaveBeenCalledTimes(1);
67+
expect(inputElement).toHaveValue('ab');
68+
69+
userEvent.type(inputElement, 'c{esc}');
5470

5571
expect(onStateChange).toHaveBeenLastCalledWith(
5672
expect.objectContaining({
@@ -60,8 +76,11 @@ describe('debouncing', () => {
6076
}),
6177
})
6278
);
79+
expect(getItems).toHaveBeenCalledTimes(1);
80+
expect(inputElement).toHaveValue('abc');
6381

6482
userEvent.type(inputElement, 'def');
83+
expect(inputElement).toHaveValue('abcdef');
6584

6685
await defer(noop, delay);
6786

@@ -78,6 +97,8 @@ describe('debouncing', () => {
7897
}),
7998
})
8099
);
100+
expect(getItems).toHaveBeenCalledTimes(2);
101+
expect(inputElement).toHaveValue('abcdef');
81102
});
82103
});
83104

packages/autocomplete-core/src/__tests__/metadata.test.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ describe('metadata', () => {
7373
).content
7474
)
7575
).toEqual({
76-
options: { 'autocomplete-core': ['environment'] },
76+
options: { 'autocomplete-core': ['environment', 'onStateChange'] },
7777
plugins: [],
7878
ua: [{ segment: 'autocomplete-core', version }],
7979
});
@@ -111,7 +111,12 @@ describe('metadata', () => {
111111
).content
112112
).options
113113
).toEqual({
114-
'autocomplete-core': ['openOnFocus', 'placeholder', 'environment'],
114+
'autocomplete-core': [
115+
'openOnFocus',
116+
'placeholder',
117+
'environment',
118+
'onStateChange',
119+
],
115120
});
116121
});
117122

packages/autocomplete-core/src/onKeyDown.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,7 @@ export function onKeyDown<TItem extends BaseItem>({
114114
} else if (event.key === 'Tab') {
115115
store.dispatch('blur', null);
116116

117-
// Hitting the `Escape` key signals the end of a user interaction with the
117+
// Hitting the `Tab` key signals the end of a user interaction with the
118118
// autocomplete. At this point, we should ignore any requests that are still
119119
// pending and could reopen the panel once they resolve, because that would
120120
// result in an unsolicited UI behavior.

packages/autocomplete-core/src/stateReducer.ts

+1
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,7 @@ export const stateReducer: Reducer = (state, action) => {
129129
? action.props.defaultActiveItemId
130130
: null,
131131
status: 'idle',
132+
completion: null,
132133
query: '',
133134
};
134135
}

packages/autocomplete-plugin-algolia-insights/src/__tests__/createAlgoliaInsightsPlugin.test.ts

+9
Original file line numberDiff line numberDiff line change
@@ -242,6 +242,9 @@ describe('createAlgoliaInsightsPlugin', () => {
242242
<body>
243243
<form>
244244
<input />
245+
<button
246+
type="reset"
247+
/>
245248
</form>
246249
</body>
247250
`);
@@ -264,6 +267,9 @@ describe('createAlgoliaInsightsPlugin', () => {
264267
<body>
265268
<form>
266269
<input />
270+
<button
271+
type="reset"
272+
/>
267273
</form>
268274
</body>
269275
`);
@@ -286,6 +292,9 @@ describe('createAlgoliaInsightsPlugin', () => {
286292
/>
287293
<form>
288294
<input />
295+
<button
296+
type="reset"
297+
/>
289298
</form>
290299
</body>
291300
`);

test/utils/createPlayground.ts

+30-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import {
2+
AutocompleteApi,
23
AutocompleteOptions,
34
createAutocomplete as createAutocompleteCore,
45
} from '@algolia/autocomplete-core';
@@ -7,9 +8,18 @@ export function createPlayground<TItem extends Record<string, unknown>>(
78
createAutocomplete: typeof createAutocompleteCore,
89
props: AutocompleteOptions<TItem>
910
) {
10-
const autocomplete = createAutocomplete<TItem>(props);
1111
const inputElement = document.createElement('input');
12+
const resetElement = document.createElement('button');
13+
resetElement.type = 'reset';
1214
const formElement = document.createElement('form');
15+
let autocomplete: AutocompleteApi<TItem> | null = null;
16+
autocomplete = createAutocomplete<TItem>({
17+
...props,
18+
onStateChange(p) {
19+
props.onStateChange?.(p);
20+
simplifiedRender();
21+
},
22+
});
1323
const inputProps = autocomplete.getInputProps({ inputElement });
1424
const formProps = autocomplete.getFormProps({ inputElement });
1525
inputElement.addEventListener('blur', inputProps.onBlur);
@@ -20,11 +30,30 @@ export function createPlayground<TItem extends Record<string, unknown>>(
2030
formElement.addEventListener('reset', formProps.onReset);
2131
formElement.addEventListener('submit', formProps.onSubmit);
2232
formElement.appendChild(inputElement);
33+
formElement.appendChild(resetElement);
2334
document.body.appendChild(formElement);
2435

36+
function simplifiedRender() {
37+
// early exit if the autocomplete instance is not ready yet (eg. through plugins)
38+
if (!autocomplete) {
39+
return;
40+
}
41+
42+
Object.entries(autocomplete.getInputProps({ inputElement })).forEach(
43+
([key, value]) => {
44+
if (key.startsWith('on')) {
45+
return;
46+
}
47+
48+
inputElement[key as any] = value;
49+
}
50+
);
51+
}
52+
2553
return {
2654
...autocomplete,
2755
inputElement,
56+
resetElement,
2857
formElement,
2958
inputProps,
3059
formProps,

0 commit comments

Comments
 (0)