From d26758a886eb8ffa9d28efe46fa7d90f4d2ad906 Mon Sep 17 00:00:00 2001 From: Kermit Xuan Date: Tue, 8 Sep 2020 10:27:22 +0800 Subject: [PATCH 1/2] chore: optimize hover value experience (#139) --- src/RangePicker.tsx | 8 ++--- src/hooks/useHoverValue.ts | 26 +++++++++++---- tests/picker.spec.tsx | 14 ++++++++ tests/range.spec.tsx | 67 ++++++++++++++++++++++++-------------- 4 files changed, 81 insertions(+), 34 deletions(-) diff --git a/src/RangePicker.tsx b/src/RangePicker.tsx index d994761ce..a8c9d3a43 100644 --- a/src/RangePicker.tsx +++ b/src/RangePicker.tsx @@ -564,9 +564,9 @@ function InnerRangePicker(props: RangePickerProps) { const onDateMouseLeave = () => { setHoverRangedValue(updateValues(selectedValue, null, mergedActivePickerIndex)); if (mergedActivePickerIndex === 0) { - onStartLeave(null); + onStartLeave(); } else { - onEndLeave(null); + onEndLeave(); } }; @@ -1013,9 +1013,9 @@ function InnerRangePicker(props: RangePickerProps) { triggerChange(values, mergedActivePickerIndex); // clear hover value style if (mergedActivePickerIndex === 0) { - onStartLeave(null); + onStartLeave(); } else { - onEndLeave(null); + onEndLeave(); } } else { setSelectedValue(values); diff --git a/src/hooks/useHoverValue.ts b/src/hooks/useHoverValue.ts index b4135f9a0..0ff4f389b 100644 --- a/src/hooks/useHoverValue.ts +++ b/src/hooks/useHoverValue.ts @@ -1,11 +1,23 @@ -import { useState, useEffect } from 'react'; +import { useState, useEffect, useRef } from 'react'; import useValueTexts, { ValueTextConfig } from './useValueTexts'; export default function useHoverValue( valueText: string, { formatList, generateConfig, locale }: ValueTextConfig, -): [string, (date: DateType) => void, (date: DateType) => void] { - const [value, setValue] = useState(null); +): [string, (date: DateType) => void, (immediately?: boolean) => void] { + const [value, internalSetValue] = useState(null); + const raf = useRef(null); + + function setValue(val: DateType, immediately: boolean = false) { + cancelAnimationFrame(raf.current); + if (immediately) { + internalSetValue(val); + return; + } + raf.current = requestAnimationFrame(() => { + internalSetValue(val); + }); + } const [, firstText] = useValueTexts(value, { formatList, @@ -17,13 +29,15 @@ export default function useHoverValue( setValue(date); } - function onLeave() { - setValue(null); + function onLeave(immediately: boolean = false) { + setValue(null, immediately); } useEffect(() => { - onLeave(); + onLeave(true); }, [valueText]); + useEffect(() => () => cancelAnimationFrame(raf.current), []); + return [firstText, onEnter, onLeave]; } diff --git a/tests/picker.spec.tsx b/tests/picker.spec.tsx index 1430fae3b..478fbcf82 100644 --- a/tests/picker.spec.tsx +++ b/tests/picker.spec.tsx @@ -740,14 +740,24 @@ describe('Picker.Basic', () => { }); describe('hover value', () => { + beforeEach(() => { + jest.useFakeTimers(); + }); + afterEach(() => { + jest.useRealTimers(); + }); it('should restore when leave', () => { const wrapper = mount(); const cell = wrapper.findCell(24); cell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect(wrapper.find('input').prop('value')).toBe('2020-07-24'); expect(wrapper.find('.rc-picker-input').hasClass('rc-picker-input-placeholder')).toBeTruthy(); cell.simulate('mouseLeave'); + jest.runAllTimers(); + wrapper.update(); expect(wrapper.find('input').prop('value')).toBe('2020-07-22'); expect(wrapper.find('.rc-picker-input').hasClass('rc-picker-input-placeholder')).toBeFalsy(); }); @@ -757,6 +767,8 @@ describe('Picker.Basic', () => { wrapper.openPicker(); const cell = wrapper.findCell(24); cell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect(wrapper.find('input').prop('value')).toBe('2020-07-24'); expect(wrapper.find('.rc-picker-input').hasClass('rc-picker-input-placeholder')).toBeTruthy(); @@ -770,6 +782,8 @@ describe('Picker.Basic', () => { wrapper.openPicker(); const cell = wrapper.findCell(24); cell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect(wrapper.find('input').prop('value')).toBe('2020-07-24'); expect(wrapper.find('.rc-picker-input').hasClass('rc-picker-input-placeholder')).toBeTruthy(); diff --git a/tests/range.spec.tsx b/tests/range.spec.tsx index c39c38ace..fbefa45a2 100644 --- a/tests/range.spec.tsx +++ b/tests/range.spec.tsx @@ -1430,6 +1430,13 @@ describe('Picker.Range', () => { }); describe('hover placeholder', () => { + beforeEach(() => { + jest.useFakeTimers(); + }); + afterEach(() => { + jest.useRealTimers(); + }); + const defaultValue: [Moment, Moment] = [getMoment('2020-07-22'), getMoment('2020-08-22')]; it('should restore when leave', () => { @@ -1439,6 +1446,8 @@ describe('Picker.Range', () => { wrapper.openPicker(0); const leftCell = wrapper.findCell(24); leftCell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect( wrapper .find('input') @@ -1465,6 +1474,8 @@ describe('Picker.Range', () => { ).toBeFalsy(); leftCell.simulate('mouseLeave'); + jest.runAllTimers(); + wrapper.update(); expect( wrapper .find('input') @@ -1496,6 +1507,8 @@ describe('Picker.Range', () => { wrapper.openPicker(1); const rightCell = wrapper.findCell(24, 1); rightCell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect( wrapper .find('input') @@ -1522,6 +1535,8 @@ describe('Picker.Range', () => { ).toBeTruthy(); rightCell.simulate('mouseLeave'); + jest.runAllTimers(); + wrapper.update(); expect( wrapper .find('input') @@ -1556,6 +1571,8 @@ describe('Picker.Range', () => { wrapper.openPicker(0); const leftCell = wrapper.findCell(24, 0); leftCell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect( wrapper .find('input') @@ -1610,6 +1627,8 @@ describe('Picker.Range', () => { // right const rightCell = wrapper.findCell(24, 1); rightCell.simulate('mouseEnter'); + jest.runAllTimers(); + wrapper.update(); expect( wrapper .find('input') @@ -1661,6 +1680,30 @@ describe('Picker.Range', () => { .hasClass('rc-picker-input-placeholder'), ).toBeFalsy(); }); + + // https://github.com/ant-design/ant-design/issues/26544 + it('should clean hover style when selecting the same value with last value', () => { + const wrapper = mount( + , + ); + + wrapper.openPicker(); + + wrapper.selectCell(24, 0); + expect( + wrapper + .find('input') + .first() + .prop('value'), + ).toBe('2020-07-24'); + expect( + wrapper + .find('input') + .first() + .hasClass('rc-picker-input-placeholder'), + ).toBeFalsy(); + expect(wrapper.isOpen()).toBeTruthy(); + }); }); // https://github.com/ant-design/ant-design/issues/25746 @@ -1699,30 +1742,6 @@ describe('Picker.Range', () => { expect(wrapper.find('.rc-picker-ok button').props().disabled).toBeTruthy(); }); - // https://github.com/ant-design/ant-design/issues/26544 - it('should clean hover style when selecting the same value with last value', () => { - const wrapper = mount( - , - ); - - wrapper.openPicker(); - - wrapper.selectCell(24, 0); - expect( - wrapper - .find('input') - .first() - .prop('value'), - ).toBe('2020-07-24'); - expect( - wrapper - .find('input') - .first() - .hasClass('rc-picker-input-placeholder'), - ).toBeFalsy(); - expect(wrapper.isOpen()).toBeTruthy(); - }); - // https://github.com/ant-design/ant-design/issues/26024 it('panel should keep open when nextValue is empty', () => { const wrapper = mount(); From 8878bb38b273c2055dcfec085c382cd4249de8ce Mon Sep 17 00:00:00 2001 From: afc163 Date: Tue, 8 Sep 2020 11:48:31 +0800 Subject: [PATCH 2/2] 2.0.11 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 5a68fd103..fdc334d3d 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "rc-picker", - "version": "2.0.10", + "version": "2.0.11", "description": "React date & time picker", "keywords": [ "react",