Skip to content

Commit

Permalink
T1185479: Add broken timezones list to exclude issues with daylight s…
Browse files Browse the repository at this point in the history
…aving time with GMT-12 timezones.
  • Loading branch information
Ilya Vinogradov committed Sep 6, 2023
1 parent 59d1951 commit b0cfecf
Show file tree
Hide file tree
Showing 2 changed files with 44 additions and 8 deletions.
12 changes: 11 additions & 1 deletion packages/devextreme/js/core/utils/date.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
/* globals Intl */
import { isObject, isString, isDate, isDefined, isNumeric } from './type';
import { adjust } from './math';
import { each } from './iterator';
Expand Down Expand Up @@ -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,

Expand Down Expand Up @@ -740,7 +748,9 @@ const dateUtils = {

getDatesOfInterval: getDatesOfInterval,

createDateWithFullYear: createDateWithFullYear
createDateWithFullYear: createDateWithFullYear,

getMachineTimezoneName: getMachineTimezoneName,
};

dateUtils.sameView = function(view, date1, date2) {
Expand Down
40 changes: 33 additions & 7 deletions packages/devextreme/js/ui/scheduler/recurrence.js
Original file line number Diff line number Diff line change
Expand Up @@ -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() {
Expand Down Expand Up @@ -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,
]);
Expand All @@ -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;
}
Expand Down

0 comments on commit b0cfecf

Please sign in to comment.