From 75a150bc2332ba5c1d4bf37ebb4b5c63b54a6cd0 Mon Sep 17 00:00:00 2001 From: Lukas Tyla Date: Thu, 24 Oct 2024 15:13:55 +0300 Subject: [PATCH] [pickers] Fix `DigitalClock` time options on a `DST` switch day (#10793) --- .../src/DigitalClock/DigitalClock.tsx | 16 ++--- .../tests/timezone.DigitalClock.test.tsx | 65 +++++++++++++++++++ 2 files changed, 73 insertions(+), 8 deletions(-) diff --git a/packages/x-date-pickers/src/DigitalClock/DigitalClock.tsx b/packages/x-date-pickers/src/DigitalClock/DigitalClock.tsx index e07706cf18637..3660a704f409c 100644 --- a/packages/x-date-pickers/src/DigitalClock/DigitalClock.tsx +++ b/packages/x-date-pickers/src/DigitalClock/DigitalClock.tsx @@ -283,13 +283,14 @@ export const DigitalClock = React.forwardRef(function DigitalClock { + const result: TDate[] = []; const startOfDay = utils.startOfDay(valueOrReferenceDate); - return [ - startOfDay, - ...Array.from({ length: Math.ceil((24 * 60) / timeStep) - 1 }, (_, index) => - utils.addMinutes(startOfDay, timeStep * (index + 1)), - ), - ]; + let nextTimeStepOption = startOfDay; + while (utils.isSameDay(valueOrReferenceDate, nextTimeStepOption)) { + result.push(nextTimeStepOption); + nextTimeStepOption = utils.addMinutes(nextTimeStepOption, timeStep); + } + return result; }, [valueOrReferenceDate, timeStep, utils]); const focusedOptionIndex = timeOptions.findIndex((option) => @@ -348,10 +349,9 @@ export const DigitalClock = React.forwardRef(function DigitalClock !readOnly && handleItemSelect(option)} selected={isSelected} disabled={disabled || isTimeDisabled(option)} diff --git a/packages/x-date-pickers/src/DigitalClock/tests/timezone.DigitalClock.test.tsx b/packages/x-date-pickers/src/DigitalClock/tests/timezone.DigitalClock.test.tsx index 3d6a353bf34d5..88e23555c6337 100644 --- a/packages/x-date-pickers/src/DigitalClock/tests/timezone.DigitalClock.test.tsx +++ b/packages/x-date-pickers/src/DigitalClock/tests/timezone.DigitalClock.test.tsx @@ -38,6 +38,71 @@ describe(' - Timezone', () => { expect(actualDate).toEqualDateTime(expectedDate); }); + it('should render correct time options when fall back DST occurs', () => { + render( + , + ); + const oneAM = adapter.setMinutes(adapter.setHours(adapter.date(undefined, 'default'), 1), 0); + const elevenPM = adapter.setMinutes( + adapter.setHours(adapter.date(undefined, 'default'), 23), + 0, + ); + expect( + screen.getAllByText( + adapter.format( + oneAM, + adapter.is12HourCycleInCurrentLocale() ? 'fullTime12h' : 'fullTime24h', + ), + ), + ).to.have.length(adapter.lib === 'dayjs' ? 1 : 2); + expect( + screen.getAllByText( + adapter.format( + elevenPM, + adapter.is12HourCycleInCurrentLocale() ? 'fullTime12h' : 'fullTime24h', + ), + ), + ).to.have.length(1); + }); + + it('should contain time options until the end of day when spring forward DST occurs', () => { + render( + , + ); + const startOfDay = adapter.setMinutes( + adapter.setHours(adapter.date(undefined, 'default'), 0), + 0, + ); + const eleven30PM = adapter.setMinutes( + adapter.setHours(adapter.date(undefined, 'default'), 23), + 30, + ); + expect( + screen.getAllByText( + adapter.format( + startOfDay, + adapter.is12HourCycleInCurrentLocale() ? 'fullTime12h' : 'fullTime24h', + ), + ), + ).to.have.length(1); + expect( + screen.getAllByText( + adapter.format( + eleven30PM, + adapter.is12HourCycleInCurrentLocale() ? 'fullTime12h' : 'fullTime24h', + ), + ), + ).to.have.length(1); + }); + TIMEZONE_TO_TEST.forEach((timezone) => { describe(`Timezone: ${timezone}`, () => { it('should use timezone prop for onChange when no value is provided', () => {