| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 // TODO(jmesserly): the native class should be the real JS Date. | |
| 6 // TODO(jimhug): Making the date value non-lazy might be a good path there. | |
| 7 class DateImplementation implements Date { | |
| 8 final int value; | |
| 9 final TimeZone timeZone; | |
| 10 | |
| 11 factory DateImplementation(int years, | |
| 12 [int month = 1, | |
| 13 int day = 1, | |
| 14 int hours = 0, | |
| 15 int minutes = 0, | |
| 16 int seconds = 0, | |
| 17 int milliseconds = 0, | |
| 18 bool isUtc = false]) { | |
| 19 return new DateImplementation.withTimeZone( | |
| 20 years, month, day, | |
| 21 hours, minutes, seconds, milliseconds, | |
| 22 isUtc ? const TimeZone.utc() | |
| 23 : new TimeZone.local()); | |
| 24 } | |
| 25 | |
| 26 DateImplementation.withTimeZone(int years, | |
| 27 int month, | |
| 28 int day, | |
| 29 int hours, | |
| 30 int minutes, | |
| 31 int seconds, | |
| 32 int milliseconds, | |
| 33 TimeZone timeZone) | |
| 34 : this.timeZone = timeZone, | |
| 35 value = _valueFromDecomposed(years, month, day, | |
| 36 hours, minutes, seconds, milliseconds, | |
| 37 timeZone.isUtc) { | |
| 38 if (value === null) throw new IllegalArgumentException(); | |
| 39 _asJs(); | |
| 40 } | |
| 41 | |
| 42 DateImplementation.now() | |
| 43 : timeZone = new TimeZone.local(), | |
| 44 value = _now() { | |
| 45 _asJs(); | |
| 46 } | |
| 47 | |
| 48 factory DateImplementation.fromString(String formattedString) { | |
| 49 // Read in (a subset of) ISO 8601. | |
| 50 // Examples: | |
| 51 // - "2012-02-27 13:27:00" | |
| 52 // - "2012-02-27 13:27:00.423z" | |
| 53 // - "20120227 13:27:00" | |
| 54 // - "20120227T132700" | |
| 55 // - "20120227" | |
| 56 // - "2012-02-27T14Z" | |
| 57 // - "-123450101 00:00:00 Z" // In the year -12345. | |
| 58 final RegExp re = const RegExp( | |
| 59 @'^([+-]?\d?\d\d\d\d)-?(\d\d)-?(\d\d)' // The day part. | |
| 60 @'(?:[ T](\d\d)(?::?(\d\d)(?::?(\d\d)(.\d{1,6})?)?)? ?([zZ])?)?$'); | |
| 61 Match match = re.firstMatch(formattedString); | |
| 62 if (match !== null) { | |
| 63 int parseIntOrZero(String matched) { | |
| 64 // TODO(floitsch): we should not need to test against the empty string. | |
| 65 if (matched === null || matched == "") return 0; | |
| 66 return Math.parseInt(matched); | |
| 67 } | |
| 68 | |
| 69 double parseDoubleOrZero(String matched) { | |
| 70 // TODO(floitsch): we should not need to test against the empty string. | |
| 71 if (matched === null || matched == "") return 0.0; | |
| 72 return Math.parseDouble(matched); | |
| 73 } | |
| 74 | |
| 75 int years = Math.parseInt(match[1]); | |
| 76 int month = Math.parseInt(match[2]); | |
| 77 int day = Math.parseInt(match[3]); | |
| 78 int hours = parseIntOrZero(match[4]); | |
| 79 int minutes = parseIntOrZero(match[5]); | |
| 80 int seconds = parseIntOrZero(match[6]); | |
| 81 bool addOneMillisecond = false; | |
| 82 int milliseconds = (parseDoubleOrZero(match[7]) * 1000).round().toInt(); | |
| 83 if (milliseconds == 1000) { | |
| 84 addOneMillisecond = true; | |
| 85 milliseconds = 999; | |
| 86 } | |
| 87 // TODO(floitsch): we should not need to test against the empty string. | |
| 88 bool isUtc = (match[8] !== null) && (match[8] != ""); | |
| 89 int epochValue = _valueFromDecomposed( | |
| 90 years, month, day, hours, minutes, seconds, milliseconds, isUtc); | |
| 91 if (epochValue === null) { | |
| 92 throw new IllegalArgumentException(formattedString); | |
| 93 } | |
| 94 if (addOneMillisecond) epochValue++; | |
| 95 return new DateImplementation.fromEpoch(epochValue, isUtc); | |
| 96 } else { | |
| 97 throw new IllegalArgumentException(formattedString); | |
| 98 } | |
| 99 } | |
| 100 | |
| 101 DateImplementation.fromEpoch(this.value, [bool isUtc = false]) | |
| 102 : this.timeZone = isUtc ? const TimeZone.utc() : new TimeZone.local(); | |
| 103 | |
| 104 bool operator ==(other) { | |
| 105 if (other is !DateImplementation) return false; | |
| 106 return (value == other.value); | |
| 107 } | |
| 108 | |
| 109 bool operator <(Date other) => value < other.value; | |
| 110 | |
| 111 bool operator <=(Date other) => value <= other.value; | |
| 112 | |
| 113 bool operator >(Date other) => value > other.value; | |
| 114 | |
| 115 bool operator >=(Date other) => value >= other.value; | |
| 116 | |
| 117 int compareTo(Date other) { | |
| 118 return value.compareTo(other.value); | |
| 119 } | |
| 120 | |
| 121 int hashCode() { | |
| 122 return value; | |
| 123 } | |
| 124 | |
| 125 Date toLocal() { | |
| 126 if (isUtc()) return changeTimeZone(new TimeZone.local()); | |
| 127 return this; | |
| 128 } | |
| 129 | |
| 130 Date toUtc() { | |
| 131 if (isUtc()) return this; | |
| 132 return changeTimeZone(const TimeZone.utc()); | |
| 133 } | |
| 134 | |
| 135 Date changeTimeZone(TimeZone targetTimeZone) { | |
| 136 if (targetTimeZone == null) { | |
| 137 targetTimeZone = new TimeZone.local(); | |
| 138 } | |
| 139 return new Date.fromEpoch(value, targetTimeZone.isUtc); | |
| 140 } | |
| 141 | |
| 142 String get timeZoneName() { throw "Unimplemented"; } | |
| 143 Duration get timeZoneOffset() { throw "Unimplemented"; } | |
| 144 | |
| 145 int get year() native | |
| 146 '''return this.isUtc() ? this._asJs().getUTCFullYear() : | |
| 147 this._asJs().getFullYear();''' { | |
| 148 isUtc(); | |
| 149 _asJs(); | |
| 150 } | |
| 151 | |
| 152 int get month() native | |
| 153 '''return this.isUtc() ? this._asJs().getUTCMonth() + 1 : | |
| 154 this._asJs().getMonth() + 1;''' { | |
| 155 isUtc(); | |
| 156 _asJs(); | |
| 157 } | |
| 158 | |
| 159 int get day() native | |
| 160 '''return this.isUtc() ? this._asJs().getUTCDate() : | |
| 161 this._asJs().getDate();''' { | |
| 162 isUtc(); | |
| 163 _asJs(); | |
| 164 } | |
| 165 | |
| 166 int get hours() native | |
| 167 '''return this.isUtc() ? this._asJs().getUTCHours() : | |
| 168 this._asJs().getHours();''' { | |
| 169 isUtc(); | |
| 170 _asJs(); | |
| 171 } | |
| 172 | |
| 173 int get minutes() native | |
| 174 '''return this.isUtc() ? this._asJs().getUTCMinutes() : | |
| 175 this._asJs().getMinutes();''' { | |
| 176 isUtc(); | |
| 177 _asJs(); | |
| 178 } | |
| 179 | |
| 180 int get seconds() native | |
| 181 '''return this.isUtc() ? this._asJs().getUTCSeconds() : | |
| 182 this._asJs().getSeconds();''' { | |
| 183 isUtc(); | |
| 184 _asJs(); | |
| 185 } | |
| 186 | |
| 187 int get milliseconds() native | |
| 188 '''return this.isUtc() ? this._asJs().getUTCMilliseconds() : | |
| 189 this._asJs().getMilliseconds();''' { | |
| 190 isUtc(); | |
| 191 _asJs(); | |
| 192 } | |
| 193 | |
| 194 // Adjust by one because JS weeks start on Sunday. | |
| 195 int get weekday() native ''' | |
| 196 var day = this.isUtc() ? this._asJs().getUTCDay() : this._asJs().getDay(); | |
| 197 return (day + 6) % 7;'''; | |
| 198 | |
| 199 bool isUtc() { | |
| 200 return timeZone.isUtc; | |
| 201 } | |
| 202 | |
| 203 String toString() { | |
| 204 String fourDigits(int n) { | |
| 205 int absN = n.abs(); | |
| 206 String sign = n < 0 ? "-" : ""; | |
| 207 if (absN >= 1000) return "$n"; | |
| 208 if (absN >= 100) return "${sign}0$absN"; | |
| 209 if (absN >= 10) return "${sign}00$absN"; | |
| 210 if (absN >= 1) return "${sign}000$absN"; | |
| 211 } | |
| 212 String threeDigits(int n) { | |
| 213 if (n >= 100) return "${n}"; | |
| 214 if (n >= 10) return "0${n}"; | |
| 215 return "00${n}"; | |
| 216 } | |
| 217 String twoDigits(int n) { | |
| 218 if (n >= 10) return "${n}"; | |
| 219 return "0${n}"; | |
| 220 } | |
| 221 | |
| 222 String y = fourDigits(year); | |
| 223 String m = twoDigits(month); | |
| 224 String d = twoDigits(day); | |
| 225 String h = twoDigits(hours); | |
| 226 String min = twoDigits(minutes); | |
| 227 String sec = twoDigits(seconds); | |
| 228 String ms = threeDigits(milliseconds); | |
| 229 if (timeZone.isUtc) { | |
| 230 return "$y-$m-$d $h:$min:$sec.${ms}Z"; | |
| 231 } else { | |
| 232 return "$y-$m-$d $h:$min:$sec.$ms"; | |
| 233 } | |
| 234 } | |
| 235 | |
| 236 // TODO(jimhug): Why not use operators here? | |
| 237 // Adds the [duration] to this Date instance. | |
| 238 Date add(Duration duration) { | |
| 239 return new DateImplementation.fromEpoch(value + duration.inMilliseconds, | |
| 240 timeZone.isUtc); | |
| 241 } | |
| 242 | |
| 243 // Subtracts the [duration] from this Date instance. | |
| 244 Date subtract(Duration duration) { | |
| 245 return new DateImplementation.fromEpoch(value - duration.inMilliseconds, | |
| 246 timeZone.isUtc); | |
| 247 } | |
| 248 | |
| 249 // Returns a [Duration] with the difference of [this] and [other]. | |
| 250 Duration difference(Date other) { | |
| 251 return new Duration(milliseconds: value - other.value); | |
| 252 } | |
| 253 | |
| 254 // TODO(floitsch): Use real exception object. | |
| 255 static int _valueFromDecomposed(int years, int month, int day, | |
| 256 int hours, int minutes, int seconds, | |
| 257 int milliseconds, bool isUtc) native | |
| 258 '''var jsMonth = month - 1; | |
| 259 var date = new Date(0); | |
| 260 if (isUtc) { | |
| 261 date.setUTCFullYear(years, jsMonth, day); | |
| 262 date.setUTCHours(hours, minutes, seconds, milliseconds); | |
| 263 } else { | |
| 264 date.setFullYear(years, jsMonth, day); | |
| 265 date.setHours(hours, minutes, seconds, milliseconds); | |
| 266 } | |
| 267 var value = date.valueOf(); | |
| 268 if (isNaN(value)) return (void 0); | |
| 269 return value;'''; | |
| 270 | |
| 271 static int _valueFromString(String str) native | |
| 272 '''var value = Date.parse(str); | |
| 273 if (isNaN(value)) throw Error("Invalid Date"); | |
| 274 return value;'''; | |
| 275 | |
| 276 static int _now() native "return new Date().valueOf();"; | |
| 277 | |
| 278 // Lazily keep a JS Date stored in the dart object. | |
| 279 _asJs() native ''' | |
| 280 if (!this.date) { | |
| 281 this.date = new Date(this.value); | |
| 282 } | |
| 283 return this.date;'''; | |
| 284 } | |
| 285 | |
| 286 class TimeZone { | |
| 287 const TimeZone.utc() : this.isUtc = true; | |
| 288 const TimeZone.local() : this.isUtc = false; | |
| 289 | |
| 290 bool operator ==(other) { | |
| 291 if (other is !TimeZone) return false; | |
| 292 return isUtc == other.isUtc; | |
| 293 } | |
| 294 | |
| 295 String toString() { | |
| 296 if (isUtc) return "TimeZone (UTC)"; | |
| 297 return "TimeZone (Local)"; | |
| 298 } | |
| 299 | |
| 300 final bool isUtc; | |
| 301 } | |
| OLD | NEW |