|
1 |
| -import { waitFor } from '@testing-library/dom'; |
| 1 | +import { fireEvent, waitFor } from '@testing-library/dom'; |
2 | 2 | import userEvent from '@testing-library/user-event';
|
3 | 3 |
|
4 | 4 | import {
|
@@ -645,6 +645,67 @@ describe('getInputProps', () => {
|
645 | 645 |
|
646 | 646 | expect(environment.clearTimeout).toHaveBeenLastCalledWith(999);
|
647 | 647 | });
|
| 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 | + }); |
648 | 709 | });
|
649 | 710 |
|
650 | 711 | describe('onKeyDown', () => {
|
@@ -1913,6 +1974,85 @@ describe('getInputProps', () => {
|
1913 | 1974 | );
|
1914 | 1975 | });
|
1915 | 1976 | });
|
| 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 | + }); |
1916 | 2056 | });
|
1917 | 2057 |
|
1918 | 2058 | describe('onFocus', () => {
|
|
0 commit comments