| Index: runtime/lib/date_patch.dart
|
| diff --git a/runtime/lib/date_patch.dart b/runtime/lib/date_patch.dart
|
| index 6c5c0c242ddc94184576bb3a6ad8f433922220c0..4697901739fb7b71363e9c8f1cd84358c5ec6d07 100644
|
| --- a/runtime/lib/date_patch.dart
|
| +++ b/runtime/lib/date_patch.dart
|
| @@ -7,28 +7,39 @@
|
| patch class DateTime {
|
| // Natives.
|
| // The natives have been moved up here to work around Issue 10401.
|
| - static int _getCurrentMs() native "DateNatives_currentTimeMillis";
|
| + static int _getCurrentMicros() native "DateTime_currentTimeMicros";
|
|
|
| static String _timeZoneNameForClampedSeconds(int secondsSinceEpoch)
|
| - native "DateNatives_timeZoneName";
|
| + native "DateTime_timeZoneName";
|
|
|
| static int _timeZoneOffsetInSecondsForClampedSeconds(int secondsSinceEpoch)
|
| - native "DateNatives_timeZoneOffsetInSeconds";
|
| + native "DateTime_timeZoneOffsetInSeconds";
|
|
|
| static int _localTimeZoneAdjustmentInSeconds()
|
| - native "DateNatives_localTimeZoneAdjustmentInSeconds";
|
| -
|
| - static const _MILLISECOND_INDEX = 0;
|
| - static const _SECOND_INDEX = 1;
|
| - static const _MINUTE_INDEX = 2;
|
| - static const _HOUR_INDEX = 3;
|
| - static const _DAY_INDEX = 4;
|
| - static const _WEEKDAY_INDEX = 5;
|
| - static const _MONTH_INDEX = 6;
|
| - static const _YEAR_INDEX = 7;
|
| + native "DateTime_localTimeZoneAdjustmentInSeconds";
|
| +
|
| + static const _MICROSECOND_INDEX = 0;
|
| + static const _MILLISECOND_INDEX = 1;
|
| + static const _SECOND_INDEX = 2;
|
| + static const _MINUTE_INDEX = 3;
|
| + static const _HOUR_INDEX = 4;
|
| + static const _DAY_INDEX = 5;
|
| + static const _WEEKDAY_INDEX = 6;
|
| + static const _MONTH_INDEX = 7;
|
| + static const _YEAR_INDEX = 8;
|
|
|
| List __parts;
|
|
|
| + /* patch */ DateTime.fromMillisecondsSinceEpoch(int millisecondsSinceEpoch,
|
| + {bool isUtc: false})
|
| + : this._withValue(
|
| + millisecondsSinceEpoch * Duration.MICROSECONDS_PER_MILLISECOND,
|
| + isUtc: isUtc);
|
| +
|
| + /* patch */ DateTime.fromMicrosecondsSinceEpoch(int microsecondsSinceEpoch,
|
| + {bool isUtc: false})
|
| + : this._withValue(microsecondsSinceEpoch, isUtc: isUtc);
|
| +
|
| /* patch */ DateTime._internal(int year,
|
| int month,
|
| int day,
|
| @@ -36,27 +47,29 @@ patch class DateTime {
|
| int minute,
|
| int second,
|
| int millisecond,
|
| + int microsecond,
|
| bool isUtc)
|
| : this.isUtc = isUtc,
|
| - this.millisecondsSinceEpoch = _brokenDownDateToMillisecondsSinceEpoch(
|
| - year, month, day, hour, minute, second, millisecond, isUtc) {
|
| - if (millisecondsSinceEpoch == null) throw new ArgumentError();
|
| + this._value = _brokenDownDateToValue(
|
| + year, month, day, hour, minute, second, millisecond, microsecond,
|
| + isUtc) {
|
| + if (_value == null) throw new ArgumentError();
|
| if (isUtc == null) throw new ArgumentError();
|
| }
|
|
|
| /* patch */ DateTime._now()
|
| : isUtc = false,
|
| - millisecondsSinceEpoch = _getCurrentMs() {
|
| + _value = _getCurrentMicros() {
|
| }
|
|
|
| /* patch */ String get timeZoneName {
|
| if (isUtc) return "UTC";
|
| - return _timeZoneName(millisecondsSinceEpoch);
|
| + return _timeZoneName(microsecondsSinceEpoch);
|
| }
|
|
|
| /* patch */ Duration get timeZoneOffset {
|
| if (isUtc) return new Duration();
|
| - int offsetInSeconds = _timeZoneOffsetInSeconds(millisecondsSinceEpoch);
|
| + int offsetInSeconds = _timeZoneOffsetInSeconds(microsecondsSinceEpoch);
|
| return new Duration(seconds: offsetInSeconds);
|
| }
|
|
|
| @@ -66,7 +79,7 @@ patch class DateTime {
|
| const [const [0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334],
|
| const [0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335]];
|
|
|
| - static List _computeUpperPart(int localMs) {
|
| + static List _computeUpperPart(int localMicros) {
|
| const int DAYS_IN_4_YEARS = 4 * 365 + 1;
|
| const int DAYS_IN_100_YEARS = 25 * DAYS_IN_4_YEARS - 1;
|
| const int DAYS_IN_400_YEARS = 4 * DAYS_IN_100_YEARS + 1;
|
| @@ -80,8 +93,8 @@ patch class DateTime {
|
| int resultDay = 0;
|
|
|
| // Always round down.
|
| - final int daysSince1970 = _flooredDivision(localMs,
|
| - Duration.MILLISECONDS_PER_DAY);
|
| + final int daysSince1970 = _flooredDivision(localMicros,
|
| + Duration.MICROSECONDS_PER_DAY);
|
| int days = daysSince1970;
|
| days += DAYS_OFFSET;
|
| resultYear = 400 * (days ~/ DAYS_IN_400_YEARS) - YEARS_OFFSET;
|
| @@ -110,16 +123,20 @@ patch class DateTime {
|
| }
|
| resultDay = days - daysUntilMonth[resultMonth - 1] + 1;
|
|
|
| - int resultMillisecond = localMs % Duration.MILLISECONDS_PER_SECOND;
|
| + int resultMicrosecond = localMicros % Duration.MICROSECONDS_PER_MILLISECOND;
|
| + int resultMillisecond =
|
| + _flooredDivision(localMicros, Duration.MICROSECONDS_PER_MILLISECOND) %
|
| + Duration.MILLISECONDS_PER_SECOND;
|
| int resultSecond =
|
| - _flooredDivision(localMs, Duration.MILLISECONDS_PER_SECOND) %
|
| + _flooredDivision(localMicros, Duration.MICROSECONDS_PER_SECOND) %
|
| Duration.SECONDS_PER_MINUTE;
|
|
|
| - int resultMinute = _flooredDivision(localMs,
|
| - Duration.MILLISECONDS_PER_MINUTE);
|
| + int resultMinute = _flooredDivision(localMicros,
|
| + Duration.MICROSECONDS_PER_MINUTE);
|
| resultMinute %= Duration.MINUTES_PER_HOUR;
|
|
|
| - int resultHour = _flooredDivision(localMs, Duration.MILLISECONDS_PER_HOUR);
|
| + int resultHour =
|
| + _flooredDivision(localMicros, Duration.MICROSECONDS_PER_HOUR);
|
| resultHour %= Duration.HOURS_PER_DAY;
|
|
|
| // In accordance with ISO 8601 a week
|
| @@ -129,6 +146,7 @@ patch class DateTime {
|
| DateTime.DAYS_PER_WEEK) + DateTime.MONDAY;
|
|
|
| List list = new List(_YEAR_INDEX + 1);
|
| + list[_MICROSECOND_INDEX] = resultMicrosecond;
|
| list[_MILLISECOND_INDEX] = resultMillisecond;
|
| list[_SECOND_INDEX] = resultSecond;
|
| list[_MINUTE_INDEX] = resultMinute;
|
| @@ -142,11 +160,32 @@ patch class DateTime {
|
|
|
| get _parts {
|
| if (__parts == null) {
|
| - __parts = _computeUpperPart(_localDateInUtcMs);
|
| + __parts = _computeUpperPart(_localDateInUtcMicros);
|
| }
|
| return __parts;
|
| }
|
|
|
| + /* patch */ DateTime add(Duration duration) {
|
| + return new DateTime._withValue(
|
| + _value + duration.inMicroseconds, isUtc: isUtc);
|
| + }
|
| +
|
| + /* patch */ DateTime subtract(Duration duration) {
|
| + return new DateTime._withValue(
|
| + _value - duration.inMicroseconds, isUtc: isUtc);
|
| + }
|
| +
|
| + /* patch */ Duration difference(DateTime other) {
|
| + return new Duration(microseconds: _value - other._value);
|
| + }
|
| +
|
| + /* patch */ int get millisecondsSinceEpoch =>
|
| + _value ~/ Duration.MICROSECONDS_PER_MILLISECOND;
|
| +
|
| + /* patch */ int get microsecondsSinceEpoch => _value;
|
| +
|
| + /* patch */ int get microsecond => _parts[_MICROSECOND_INDEX];
|
| +
|
| /* patch */ int get millisecond => _parts[_MILLISECOND_INDEX];
|
|
|
| /* patch */ int get second => _parts[_SECOND_INDEX];
|
| @@ -164,26 +203,26 @@ patch class DateTime {
|
| /* patch */ int get year => _parts[_YEAR_INDEX];
|
|
|
| /**
|
| - * Returns the amount of milliseconds in UTC that represent the same values
|
| + * Returns the amount of microseconds in UTC that represent the same values
|
| * as [this].
|
| *
|
| - * Say [:t:] is the result of this function, then
|
| - * * [:this.year == new DateTime.fromMillisecondsSinceEpoch(t, true).year:],
|
| - * * [:this.month == new DateTime.fromMillisecondsSinceEpoch(t, true).month:],
|
| - * * [:this.day == new DateTime.fromMillisecondsSinceEpoch(t, true).day:],
|
| - * * [:this.hour == new DateTime.fromMillisecondsSinceEpoch(t, true).hour:],
|
| + * Say `t` is the result of this function, then
|
| + * * `this.year == new DateTime.fromMicrosecondsSinceEpoch(t, true).year`,
|
| + * * `this.month == new DateTime.fromMicrosecondsSinceEpoch(t, true).month`,
|
| + * * `this.day == new DateTime.fromMicrosecondsSinceEpoch(t, true).day`,
|
| + * * `this.hour == new DateTime.fromMicrosecondsSinceEpoch(t, true).hour`,
|
| * * ...
|
| *
|
| * Daylight savings is computed as if the date was computed in [1970..2037].
|
| * If [this] lies outside this range then it is a year with similar
|
| * properties (leap year, weekdays) is used instead.
|
| */
|
| - int get _localDateInUtcMs {
|
| - int ms = millisecondsSinceEpoch;
|
| - if (isUtc) return ms;
|
| + int get _localDateInUtcMicros {
|
| + int micros = _value;
|
| + if (isUtc) return micros;
|
| int offset =
|
| - _timeZoneOffsetInSeconds(ms) * Duration.MILLISECONDS_PER_SECOND;
|
| - return ms + offset;
|
| + _timeZoneOffsetInSeconds(micros) * Duration.MICROSECONDS_PER_SECOND;
|
| + return micros + offset;
|
| }
|
|
|
| static int _flooredDivision(int a, int b) {
|
| @@ -204,9 +243,10 @@ patch class DateTime {
|
| return (y % 4 == 0) && ((y % 16 == 0) || (y % 100 != 0));
|
| }
|
|
|
| - /* patch */ static int _brokenDownDateToMillisecondsSinceEpoch(
|
| + /// Converts the given broken down date to microseconds.
|
| + /* patch */ static int _brokenDownDateToValue(
|
| int year, int month, int day,
|
| - int hour, int minute, int second, int millisecond,
|
| + int hour, int minute, int second, int millisecond, int microsecond,
|
| bool isUtc) {
|
| // Simplify calculations by working with zero-based month.
|
| --month;
|
| @@ -225,20 +265,21 @@ patch class DateTime {
|
| int days = day - 1;
|
| days += _DAYS_UNTIL_MONTH[_isLeapYear(year) ? 1 : 0][month];
|
| days += _dayFromYear(year);
|
| - int millisecondsSinceEpoch =
|
| - days * Duration.MILLISECONDS_PER_DAY +
|
| - hour * Duration.MILLISECONDS_PER_HOUR +
|
| - minute * Duration.MILLISECONDS_PER_MINUTE +
|
| - second * Duration.MILLISECONDS_PER_SECOND +
|
| - millisecond;
|
| + int microsecondsSinceEpoch =
|
| + days * Duration.MICROSECONDS_PER_DAY +
|
| + hour * Duration.MICROSECONDS_PER_HOUR +
|
| + minute * Duration.MICROSECONDS_PER_MINUTE +
|
| + second * Duration.MICROSECONDS_PER_SECOND +
|
| + millisecond * Duration.MICROSECONDS_PER_MILLISECOND +
|
| + microsecond;
|
|
|
| // Since [_timeZoneOffsetInSeconds] will crash if the input is far out of
|
| // the valid range we do a preliminary test that weeds out values that can
|
| // not become valid even with timezone adjustments.
|
| // The timezone adjustment is always less than a day, so adding a security
|
| // margin of one day should be enough.
|
| - if (millisecondsSinceEpoch.abs() >
|
| - (_MAX_MILLISECONDS_SINCE_EPOCH + Duration.MILLISECONDS_PER_DAY)) {
|
| + if (microsecondsSinceEpoch.abs() >
|
| + _MAX_MILLISECONDS_SINCE_EPOCH * 1000 + Duration.MICROSECONDS_PER_DAY) {
|
| return null;
|
| }
|
|
|
| @@ -246,15 +287,17 @@ patch class DateTime {
|
| // Note that we need to remove the local timezone adjustement before
|
| // asking for the correct zone offset.
|
| int adjustment = _localTimeZoneAdjustmentInSeconds() *
|
| - Duration.MILLISECONDS_PER_SECOND;
|
| + Duration.MICROSECONDS_PER_SECOND;
|
| +
|
| int zoneOffset =
|
| - _timeZoneOffsetInSeconds(millisecondsSinceEpoch - adjustment);
|
| - millisecondsSinceEpoch -= zoneOffset * Duration.MILLISECONDS_PER_SECOND;
|
| + _timeZoneOffsetInSeconds(microsecondsSinceEpoch - adjustment);
|
| + microsecondsSinceEpoch -= zoneOffset * Duration.MICROSECONDS_PER_SECOND;
|
| }
|
| - if (millisecondsSinceEpoch.abs() > _MAX_MILLISECONDS_SINCE_EPOCH) {
|
| + if (microsecondsSinceEpoch.abs() >
|
| + _MAX_MILLISECONDS_SINCE_EPOCH * Duration.MICROSECONDS_PER_MILLISECOND) {
|
| return null;
|
| }
|
| - return millisecondsSinceEpoch;
|
| + return microsecondsSinceEpoch;
|
| }
|
|
|
| static int _weekDay(y) {
|
| @@ -304,25 +347,25 @@ patch class DateTime {
|
| // According to V8 this fast case works for dates from 1970 to 2099.
|
| return 1970 + (4 * days + 2) ~/ DAYS_IN_4_YEARS;
|
| }
|
| - int ms = secondsSinceEpoch * Duration.MILLISECONDS_PER_SECOND;
|
| - return _computeUpperPart(ms)[_YEAR_INDEX];
|
| + int micros = secondsSinceEpoch * Duration.MICROSECONDS_PER_SECOND;
|
| + return _computeUpperPart(micros)[_YEAR_INDEX];
|
| }
|
|
|
| /**
|
| * Returns a date in seconds that is equivalent to the current date. An
|
| - * equivalent date has the same fields ([:month:], [:day:], etc.) as the
|
| - * [this], but the [:year:] is in the range [1970..2037].
|
| + * equivalent date has the same fields (`month`, `day`, etc.) as the
|
| + * [this], but the `year` is in the range [1970..2037].
|
| *
|
| * * The time since the beginning of the year is the same.
|
| * * If [this] is in a leap year then the returned seconds are in a leap
|
| * year, too.
|
| * * The week day of [this] is the same as the one for the returned date.
|
| */
|
| - static int _equivalentSeconds(int millisecondsSinceEpoch) {
|
| + static int _equivalentSeconds(int microsecondsSinceEpoch) {
|
| const int CUT_OFF_SECONDS = 2100000000;
|
|
|
| - int secondsSinceEpoch = _flooredDivision(millisecondsSinceEpoch,
|
| - Duration.MILLISECONDS_PER_SECOND);
|
| + int secondsSinceEpoch = _flooredDivision(microsecondsSinceEpoch,
|
| + Duration.MICROSECONDS_PER_SECOND);
|
|
|
| if (secondsSinceEpoch < 0 || secondsSinceEpoch >= CUT_OFF_SECONDS) {
|
| int year = _yearsFromSecondsSinceEpoch(secondsSinceEpoch);
|
| @@ -335,13 +378,13 @@ patch class DateTime {
|
| return secondsSinceEpoch;
|
| }
|
|
|
| - static int _timeZoneOffsetInSeconds(int millisecondsSinceEpoch) {
|
| - int equivalentSeconds = _equivalentSeconds(millisecondsSinceEpoch);
|
| + static int _timeZoneOffsetInSeconds(int microsecondsSinceEpoch) {
|
| + int equivalentSeconds = _equivalentSeconds(microsecondsSinceEpoch);
|
| return _timeZoneOffsetInSecondsForClampedSeconds(equivalentSeconds);
|
| }
|
|
|
| - static String _timeZoneName(int millisecondsSinceEpoch) {
|
| - int equivalentSeconds = _equivalentSeconds(millisecondsSinceEpoch);
|
| + static String _timeZoneName(int microsecondsSinceEpoch) {
|
| + int equivalentSeconds = _equivalentSeconds(microsecondsSinceEpoch);
|
| return _timeZoneNameForClampedSeconds(equivalentSeconds);
|
| }
|
| }
|
|
|