Skip to content

Commit 934d722

Browse files
author
dujiaqi
committed
fix: ensure consistency between controlled and uncontrolled logic
1 parent f6c6803 commit 934d722

File tree

2 files changed

+44
-4
lines changed

2 files changed

+44
-4
lines changed

src/hooks/useRangeOpen.ts

+23-3
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,8 @@ export default function useRangeOpen(
4242
firstTimeOpen: boolean,
4343
triggerOpen: (open: boolean, activeIndex: 0 | 1 | false, source: SourceType) => void,
4444
] {
45+
const rafRef = React.useRef<number>(null);
46+
4547
const [firstTimeOpen, setFirstTimeOpen] = React.useState(false);
4648

4749
const [afferentOpen, setAfferentOpen] = useMergedState(defaultOpen || false, {
@@ -61,13 +63,26 @@ export default function useRangeOpen(
6163

6264
const [nextActiveIndex, setNextActiveIndex] = React.useState<0 | 1>(null);
6365

66+
const queryNextIndex = (index: number) => (index === 0 ? 1 : 0);
67+
6468
React.useEffect(() => {
6569
if (mergedOpen) {
6670
setFirstTimeOpen(true);
6771
}
6872
}, [mergedOpen]);
6973

70-
const queryNextIndex = (index: number) => (index === 0 ? 1 : 0);
74+
React.useEffect(() => {
75+
if (!afferentOpen && rafRef.current !== null) {
76+
// Unfocus
77+
raf.cancel(rafRef.current);
78+
rafRef.current = null;
79+
80+
// Since the index will eventually point to the next one, it needs to be reset.
81+
if (mergedActivePickerIndex !== null) {
82+
setMergedActivePickerIndex(queryNextIndex(mergedActivePickerIndex));
83+
}
84+
}
85+
}, [afferentOpen]);
7186

7287
const triggerOpen = useEvent((nextOpen: boolean, index: 0 | 1 | false, source: SourceType) => {
7388
if (index === false) {
@@ -102,12 +117,17 @@ export default function useRangeOpen(
102117
setFirstTimeOpen(false);
103118
setMergedActivePickerIndex(customNextActiveIndex);
104119
}
105-
120+
106121
setNextActiveIndex(null);
107122

108123
// Focus back
109124
if (customNextActiveIndex !== null && !disabled[customNextActiveIndex]) {
110-
raf(() => {
125+
// Trigger closure to ensure consistency between controlled and uncontrolled logic.
126+
if (afferentOpen && !firstTimeOpen && nextActiveIndex === null) {
127+
setMergedOpen(false);
128+
}
129+
130+
rafRef.current = raf(() => {
111131
const ref = [startInputRef, endInputRef][customNextActiveIndex];
112132
ref.current?.focus();
113133
});

tests/range.spec.tsx

+21-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ import moment from 'moment';
44
import KeyCode from 'rc-util/lib/KeyCode';
55
import { spyElementPrototypes } from 'rc-util/lib/test/domHook';
66
import { resetWarned } from 'rc-util/lib/warning';
7-
import React from 'react';
7+
import React, { useState } from 'react';
88
import type { PickerMode } from '../src/interface';
99
import zhCN from '../src/locale/zh_CN';
1010
import type { RangePickerProps } from '../src/RangePicker';
@@ -1985,4 +1985,24 @@ describe('Picker.Range', () => {
19851985

19861986
expect(onOpenChange).toHaveBeenCalledWith(false);
19871987
});
1988+
1989+
it('controlled open logic should be consistent with the uncontrolled logic', () => {
1990+
const Demo: React.FC = () => {
1991+
const [open, setOpen] = useState(false);
1992+
1993+
return <MomentRangePicker open={open} onOpenChange={(bool) => setOpen(bool)} />;
1994+
};
1995+
1996+
const { container, baseElement } = render(<Demo />);
1997+
1998+
openPicker(container);
1999+
2000+
expect(baseElement.querySelector('.rc-picker-dropdown-hidden')).toBeFalsy();
2001+
2002+
selectCell(2);
2003+
selectCell(4);
2004+
2005+
expect(baseElement.querySelector('.rc-picker-dropdown-hidden')).toBeTruthy();
2006+
expect(baseElement.querySelectorAll('.rc-picker-input')[1]).toHaveClass('rc-picker-input-active');
2007+
});
19882008
});

0 commit comments

Comments
 (0)