Skip to content

Commit fba16e5

Browse files
fix: ignore composition events with option (#1238)
--------- Co-authored-by: Aymeric Giraudet <[email protected]>
1 parent 190e562 commit fba16e5

File tree

9 files changed

+187
-1
lines changed

9 files changed

+187
-1
lines changed

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

+141-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { waitFor } from '@testing-library/dom';
1+
import { fireEvent, waitFor } from '@testing-library/dom';
22
import userEvent from '@testing-library/user-event';
33

44
import {
@@ -645,6 +645,67 @@ describe('getInputProps', () => {
645645

646646
expect(environment.clearTimeout).toHaveBeenLastCalledWith(999);
647647
});
648+
649+
test('stops process if IME composition is in progress and `ignoreCompositionEvents: true`', () => {
650+
const getSources = jest.fn((..._args: any[]) => {
651+
return [
652+
createSource({
653+
getItems() {
654+
return [{ label: '1' }, { label: '2' }];
655+
},
656+
}),
657+
];
658+
});
659+
const { inputElement } = createPlayground(createAutocomplete, {
660+
ignoreCompositionEvents: true,
661+
getSources,
662+
});
663+
664+
// Typing 木 using the Wubihua input method
665+
// see:
666+
// - https://en.wikipedia.org/wiki/Stroke_count_method
667+
// - https://developer.mozilla.org/en-US/docs/Web/API/Element/compositionend_event
668+
const character = '木';
669+
const strokes = ['一', '丨', '丿', '丶', character];
670+
671+
strokes.forEach((stroke, index) => {
672+
const isFirst = index === 0;
673+
const isLast = index === strokes.length - 1;
674+
const query = isLast ? stroke : strokes.slice(0, index + 1).join('');
675+
676+
if (isFirst) {
677+
fireEvent.compositionStart(inputElement);
678+
}
679+
680+
fireEvent.compositionUpdate(inputElement, {
681+
data: query,
682+
});
683+
684+
fireEvent.input(inputElement, {
685+
isComposing: true,
686+
target: {
687+
value: query,
688+
},
689+
});
690+
691+
if (isLast) {
692+
fireEvent.compositionEnd(inputElement, {
693+
data: query,
694+
target: {
695+
value: query,
696+
},
697+
});
698+
}
699+
});
700+
701+
expect(inputElement).toHaveValue(character);
702+
expect(getSources).toHaveBeenCalledTimes(1);
703+
expect(getSources).toHaveBeenLastCalledWith(
704+
expect.objectContaining({
705+
query: character,
706+
})
707+
);
708+
});
648709
});
649710

650711
describe('onKeyDown', () => {
@@ -1913,6 +1974,85 @@ describe('getInputProps', () => {
19131974
);
19141975
});
19151976
});
1977+
1978+
test('stops process if IME composition is in progress`', () => {
1979+
const onStateChange = jest.fn();
1980+
const { inputElement } = createPlayground(createAutocomplete, {
1981+
openOnFocus: true,
1982+
onStateChange,
1983+
initialState: {
1984+
collections: [
1985+
createCollection({
1986+
source: { sourceId: 'testSource' },
1987+
items: [
1988+
{ label: '1' },
1989+
{ label: '2' },
1990+
{ label: '3' },
1991+
{ label: '4' },
1992+
],
1993+
}),
1994+
],
1995+
},
1996+
});
1997+
1998+
inputElement.focus();
1999+
2000+
// 1. Pressing Arrow Down to select the first item
2001+
fireEvent.keyDown(inputElement, { key: 'ArrowDown' });
2002+
expect(onStateChange).toHaveBeenLastCalledWith(
2003+
expect.objectContaining({
2004+
state: expect.objectContaining({
2005+
activeItemId: 0,
2006+
}),
2007+
})
2008+
);
2009+
2010+
// 2. Typing かくてい with a Japanese IME
2011+
const strokes = ['か', 'く', 'て', 'い'];
2012+
strokes.forEach((_stroke, index) => {
2013+
const isFirst = index === 0;
2014+
const query = strokes.slice(0, index + 1).join('');
2015+
2016+
if (isFirst) {
2017+
fireEvent.compositionStart(inputElement);
2018+
}
2019+
2020+
fireEvent.compositionUpdate(inputElement, {
2021+
data: query,
2022+
});
2023+
2024+
fireEvent.input(inputElement, {
2025+
isComposing: true,
2026+
data: query,
2027+
target: {
2028+
value: query,
2029+
},
2030+
});
2031+
});
2032+
2033+
// 3. Checking that activeItemId has reverted to null due to input change
2034+
expect(onStateChange).toHaveBeenLastCalledWith(
2035+
expect.objectContaining({
2036+
state: expect.objectContaining({
2037+
activeItemId: null,
2038+
}),
2039+
})
2040+
);
2041+
2042+
// 4. Selecting the 3rd suggestion on the IME window
2043+
fireEvent.keyDown(inputElement, { key: 'ArrowDown', isComposing: true });
2044+
fireEvent.keyDown(inputElement, { key: 'ArrowDown', isComposing: true });
2045+
fireEvent.keyDown(inputElement, { key: 'ArrowDown', isComposing: true });
2046+
2047+
// 5. Checking that activeItemId has not changed
2048+
expect(onStateChange).toHaveBeenLastCalledWith(
2049+
expect.objectContaining({
2050+
state: expect.objectContaining({
2051+
activeItemId: null,
2052+
}),
2053+
})
2054+
);
2055+
});
19162056
});
19172057

19182058
describe('onFocus', () => {

packages/autocomplete-core/src/getDefaultProps.ts

+1
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ export function getDefaultProps<TItem extends BaseItem>(
2828
debug: false,
2929
openOnFocus: false,
3030
enterKeyHint: undefined,
31+
ignoreCompositionEvents: false,
3132
placeholder: '',
3233
autoFocus: false,
3334
defaultActiveItemId: null,

packages/autocomplete-core/src/getPropGetters.ts

+27
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import {
2121
getAutocompleteElementId,
2222
isOrContainsNode,
2323
isSamsung,
24+
getNativeEvent,
2425
} from './utils';
2526

2627
interface GetPropGettersOptions<TItem extends BaseItem>
@@ -219,6 +220,28 @@ export function getPropGetters<
219220
maxLength,
220221
type: 'search',
221222
onChange: (event) => {
223+
const value = (
224+
(event as unknown as Event).currentTarget as HTMLInputElement
225+
).value;
226+
227+
if (
228+
props.ignoreCompositionEvents &&
229+
getNativeEvent(event as unknown as InputEvent).isComposing
230+
) {
231+
setters.setQuery(value);
232+
return;
233+
}
234+
235+
onInput({
236+
event,
237+
props,
238+
query: value.slice(0, maxLength),
239+
refresh,
240+
store,
241+
...setters,
242+
});
243+
},
244+
onCompositionEnd: (event) => {
222245
onInput({
223246
event,
224247
props,
@@ -231,6 +254,10 @@ export function getPropGetters<
231254
});
232255
},
233256
onKeyDown: (event) => {
257+
if (getNativeEvent(event as unknown as InputEvent).isComposing) {
258+
return;
259+
}
260+
234261
onKeyDown({
235262
event: event as unknown as KeyboardEvent,
236263
props,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
export function getNativeEvent<TEvent>(event: TEvent) {
2+
return (event as unknown as { nativeEvent: TEvent }).nativeEvent || event;
3+
}

packages/autocomplete-core/src/utils/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -8,3 +8,4 @@ export * from './getAutocompleteElementId';
88
export * from './isOrContainsNode';
99
export * from './isSamsung';
1010
export * from './mapToAlgoliaResponse';
11+
export * from './getNativeEvent';

packages/autocomplete-js/src/utils/setProperties.ts

+3
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,9 @@ function getNormalizedName(name: string): string {
110110
switch (name) {
111111
case 'onChange':
112112
return 'onInput';
113+
// see: https://github.com/preactjs/preact/issues/1978
114+
case 'onCompositionEnd':
115+
return 'oncompositionend';
113116
default:
114117
return name;
115118
}

packages/autocomplete-shared/src/core/AutocompleteOptions.ts

+9
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,14 @@ export interface AutocompleteOptions<TItem extends BaseItem> {
8888
* @link https://www.algolia.com/doc/ui-libraries/autocomplete/api-reference/autocomplete-js/autocomplete/#param-enterkeyhint
8989
*/
9090
enterKeyHint?: AutocompleteEnterKeyHint;
91+
/**
92+
* Whether to update the search input value in the middle of a
93+
* composition session.
94+
*
95+
* @default false
96+
* @link https://www.algolia.com/doc/ui-libraries/autocomplete/api-reference/autocomplete-js/autocomplete/#param-ignorecompositionevents
97+
*/
98+
ignoreCompositionEvents?: boolean;
9199
/**
92100
* The placeholder text to show in the search input when there's no query.
93101
*
@@ -200,6 +208,7 @@ export interface InternalAutocompleteOptions<TItem extends BaseItem>
200208
id: string;
201209
onStateChange(props: OnStateChangeProps<TItem>): void;
202210
enterKeyHint: AutocompleteEnterKeyHint | undefined;
211+
ignoreCompositionEvents: boolean;
203212
placeholder: string;
204213
autoFocus: boolean;
205214
defaultActiveItemId: number | null;

packages/autocomplete-shared/src/core/AutocompletePropGetters.ts

+1
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,7 @@ export type GetInputProps<TEvent, TMouseEvent, TKeyboardEvent> = (props: {
8383
'aria-controls': string | undefined;
8484
'aria-labelledby': string;
8585
onChange(event: TEvent): void;
86+
onCompositionEnd(event: TEvent): void;
8687
onKeyDown(event: TKeyboardEvent): void;
8788
onFocus(event: TEvent): void;
8889
onBlur(): void;

test/utils/createPlayground.ts

+1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ export function createPlayground<TItem extends Record<string, unknown>>(
2424
const formProps = autocomplete.getFormProps({ inputElement });
2525
inputElement.addEventListener('blur', inputProps.onBlur);
2626
inputElement.addEventListener('input', inputProps.onChange);
27+
inputElement.addEventListener('compositionend', inputProps.onCompositionEnd);
2728
inputElement.addEventListener('click', inputProps.onClick);
2829
inputElement.addEventListener('focus', inputProps.onFocus);
2930
inputElement.addEventListener('keydown', inputProps.onKeyDown);

0 commit comments

Comments
 (0)