From b0cfecfa81b28d32f0083cd57a4c24ce33aa6cf5 Mon Sep 17 00:00:00 2001 From: Ilya Vinogradov Date: Wed, 6 Sep 2023 14:26:24 +1300 Subject: [PATCH] T1185479: Add broken timezones list to exclude issues with daylight saving time with GMT-12 timezones. --- packages/devextreme/js/core/utils/date.js | 12 +++++- .../devextreme/js/ui/scheduler/recurrence.js | 40 +++++++++++++++---- 2 files changed, 44 insertions(+), 8 deletions(-) diff --git a/packages/devextreme/js/core/utils/date.js b/packages/devextreme/js/core/utils/date.js index 9988f56c0bff..0360cfa222ed 100644 --- a/packages/devextreme/js/core/utils/date.js +++ b/packages/devextreme/js/core/utils/date.js @@ -1,3 +1,4 @@ +/* globals Intl */ import { isObject, isString, isDate, isDefined, isNumeric } from './type'; import { adjust } from './math'; import { each } from './iterator'; @@ -680,6 +681,13 @@ const createDateWithFullYear = function(year) { return result; }; +const getMachineTimezoneName = () => { + const hasIntl = typeof Intl !== 'undefined'; + return hasIntl + ? Intl.DateTimeFormat().resolvedOptions().timeZone + : null; +}; + const dateUtils = { dateUnitIntervals: dateUnitIntervals, @@ -740,7 +748,9 @@ const dateUtils = { getDatesOfInterval: getDatesOfInterval, - createDateWithFullYear: createDateWithFullYear + createDateWithFullYear: createDateWithFullYear, + + getMachineTimezoneName: getMachineTimezoneName, }; dateUtils.sameView = function(view, date1, date2) { diff --git a/packages/devextreme/js/ui/scheduler/recurrence.js b/packages/devextreme/js/ui/scheduler/recurrence.js index 6aceaceef562..edd1c72265a2 100644 --- a/packages/devextreme/js/ui/scheduler/recurrence.js +++ b/packages/devextreme/js/ui/scheduler/recurrence.js @@ -13,6 +13,16 @@ const loggedWarnings = []; const MS_IN_HOUR = 1000 * 60 * 60; const MS_IN_DAY = MS_IN_HOUR * 24; +const RRULE_BROKEN_TIMEZONES = [ + 'Etc/GMT-13', + 'MIT', + 'Pacific/Apia', + 'Pacific/Enderbury', + 'Pacific/Tongatapu', + 'Etc/GMT-14', + 'Pacific/Kiritimati', +]; + let recurrence = null; export function getRecurrenceProcessor() { @@ -84,15 +94,9 @@ class RecurrenceProcessor { } _convertRruleResult(rruleIntervalParams, options, rruleDate) { - const localTimezoneOffset = timeZoneUtils.getClientTimezoneOffset(rruleDate); - // NOTE: Workaround for the RRule bug with timezones greater than GMT+12 (e.g. Apia Standard Time GMT+13) - // GitHub issue: https://github.com/jakubroztocil/rrule/issues/555 - const additionalWorkaroundOffsetForRrule = - localTimezoneOffset / MS_IN_HOUR <= -13 ? -MS_IN_DAY : 0; const convertedBackDate = timeZoneUtils.setOffsetsToDate( rruleDate, [ - localTimezoneOffset, - additionalWorkaroundOffsetForRrule, + this._getLocalMachineOffset(rruleDate), -options.appointmentTimezoneOffset, rruleIntervalParams.startIntervalDateDSTShift, ]); @@ -108,6 +112,28 @@ class RecurrenceProcessor { return resultDate; } + _getLocalMachineOffset(rruleDate) { + const machineTimezoneOffset = timeZoneUtils.getClientTimezoneOffset(rruleDate); + const machineTimezoneName = dateUtils.getMachineTimezoneName(); + const result = [machineTimezoneOffset]; + + // NOTE: Workaround for the RRule bug with timezones greater than GMT+12 (e.g. Apia Standard Time GMT+13) + // GitHub issue: https://github.com/jakubroztocil/rrule/issues/555 + // UPD: 05.09.2023 - The issue still hasn't been fixed in the Rule package. + // RRule returns results that are one day greater than expected. + // Therefore, for broken from RRule point of view timezones, we subtract one day from the result. + const brokenTimezonesOffset = -13; + const isTimezoneOffsetInBrokenRange = machineTimezoneOffset / MS_IN_HOUR <= brokenTimezonesOffset; + const isTimezoneNameInBrokenNames = !machineTimezoneName + || RRULE_BROKEN_TIMEZONES.some((timezone) => machineTimezoneName.includes(timezone)); + + if(isTimezoneOffsetInBrokenRange && isTimezoneNameInBrokenNames) { + result.push(MS_IN_DAY); + } + + return result; + } + hasRecurrence(options) { return !!this.generateDates(options).length; }