2 * Copyright 2008 Jacek Caban for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
24 #include "wine/debug.h"
26 WINE_DEFAULT_DEBUG_CHANNEL(jscript
);
28 /* 1601 to 1970 is 369 years plus 89 leap days */
29 #define TIME_EPOCH ((ULONGLONG)(369 * 365 + 89) * 86400 * 1000)
34 /* ECMA-262 3rd Edition 15.9.1.1 */
40 static const WCHAR toStringW
[] = {'t','o','S','t','r','i','n','g',0};
41 static const WCHAR toLocaleStringW
[] = {'t','o','L','o','c','a','l','e','S','t','r','i','n','g',0};
42 static const WCHAR hasOwnPropertyW
[] = {'h','a','s','O','w','n','P','r','o','p','e','r','t','y',0};
43 static const WCHAR propertyIsEnumerableW
[] =
44 {'p','r','o','p','e','r','t','y','I','s','E','n','u','m','e','r','a','b','l','e',0};
45 static const WCHAR isPrototypeOfW
[] = {'i','s','P','r','o','t','o','t','y','p','e','O','f',0};
46 static const WCHAR valueOfW
[] = {'v','a','l','u','e','O','f',0};
47 static const WCHAR toUTCStringW
[] = {'t','o','U','T','C','S','t','r','i','n','g',0};
48 static const WCHAR toDateStringW
[] = {'t','o','D','a','t','e','S','t','r','i','n','g',0};
49 static const WCHAR toTimeStringW
[] = {'t','o','T','i','m','e','S','t','r','i','n','g',0};
50 static const WCHAR toLocaleDateStringW
[] = {'t','o','L','o','c','a','l','e','D','a','t','e','S','t','r','i','n','g',0};
51 static const WCHAR toLocaleTimeStringW
[] = {'t','o','L','o','c','a','l','e','T','i','m','e','S','t','r','i','n','g',0};
52 static const WCHAR getTimeW
[] = {'g','e','t','T','i','m','e',0};
53 static const WCHAR getFullYearW
[] = {'g','e','t','F','u','l','l','Y','e','a','r',0};
54 static const WCHAR getUTCFullYearW
[] = {'g','e','t','U','T','C','F','u','l','l','Y','e','a','r',0};
55 static const WCHAR getMonthW
[] = {'g','e','t','M','o','n','t','h',0};
56 static const WCHAR getUTCMonthW
[] = {'g','e','t','U','T','C','M','o','n','t','h',0};
57 static const WCHAR getDateW
[] = {'g','e','t','D','a','t','e',0};
58 static const WCHAR getUTCDateW
[] = {'g','e','t','U','T','C','D','a','t','e',0};
59 static const WCHAR getDayW
[] = {'g','e','t','D','a','y',0};
60 static const WCHAR getUTCDayW
[] = {'g','e','t','U','T','C','D','a','y',0};
61 static const WCHAR getHoursW
[] = {'g','e','t','H','o','u','r','s',0};
62 static const WCHAR getUTCHoursW
[] = {'g','e','t','U','T','C','H','o','u','r','s',0};
63 static const WCHAR getMinutesW
[] = {'g','e','t','M','i','n','u','t','e','s',0};
64 static const WCHAR getUTCMinutesW
[] = {'g','e','t','U','T','C','M','i','n','u','t','e','s',0};
65 static const WCHAR getSecondsW
[] = {'g','e','t','S','e','c','o','n','d','s',0};
66 static const WCHAR getUTCSecondsW
[] = {'g','e','t','U','T','C','S','e','c','o','n','d','s',0};
67 static const WCHAR getMillisecondsW
[] = {'g','e','t','M','i','l','l','i','s','e','c','o','n','d','s',0};
68 static const WCHAR getUTCMillisecondsW
[] = {'g','e','t','U','T','C','M','i','l','l','i','s','e','c','o','n','d','s',0};
69 static const WCHAR getTimezoneOffsetW
[] = {'g','e','t','T','i','m','e','z','o','n','e','O','f','f','s','e','t',0};
70 static const WCHAR setTimeW
[] = {'s','e','t','T','i','m','e',0};
71 static const WCHAR setMillisecondsW
[] = {'s','e','t','M','i','l','l','i','s','e','c','o','n','d','s',0};
72 static const WCHAR setUTCMillisecondsW
[] = {'s','e','t','U','T','C','M','i','l','l','i','s','e','c','o','n','d','s',0};
73 static const WCHAR setSecondsW
[] = {'s','e','t','S','e','c','o','n','d','s',0};
74 static const WCHAR setUTCSecondsW
[] = {'s','e','t','U','T','C','S','e','c','o','n','d','s',0};
75 static const WCHAR setMinutesW
[] = {'s','e','t','M','i','n','u','t','e','s',0};
76 static const WCHAR setUTCMinutesW
[] = {'s','e','t','U','T','C','M','i','n','u','t','e','s',0};
77 static const WCHAR setHoursW
[] = {'s','e','t','H','o','u','r','s',0};
78 static const WCHAR setUTCHoursW
[] = {'s','e','t','U','T','C','H','o','u','r','s',0};
79 static const WCHAR setDateW
[] = {'s','e','t','D','a','t','e',0};
80 static const WCHAR setUTCDateW
[] = {'s','e','t','U','T','C','D','a','t','e',0};
81 static const WCHAR setMonthW
[] = {'s','e','t','M','o','n','t','h',0};
82 static const WCHAR setUTCMonthW
[] = {'s','e','t','U','T','C','M','o','n','t','h',0};
83 static const WCHAR setFullYearW
[] = {'s','e','t','F','u','l','l','Y','e','a','r',0};
84 static const WCHAR setUTCFullYearW
[] = {'s','e','t','U','T','C','F','u','l','l','Y','e','a','r',0};
86 /*ECMA-262 3th Edition 15.9.1.2 */
87 #define MS_PER_DAY 86400000
88 #define MS_PER_HOUR 3600000
89 #define MS_PER_MINUTE 60000
91 /* ECMA-262 3th Edition 15.9.1.2 */
92 static inline DOUBLE
day(DOUBLE time
)
94 return floor(time
/ MS_PER_DAY
);
97 /* ECMA-262 3th Edition 15.9.1.3 */
98 static inline DOUBLE
days_in_year(DOUBLE year
)
102 if(year
!= (int)year
)
106 if(y
%4 != 0) return 365;
107 if(y
%100 != 0) return 366;
108 if(y
%400 != 0) return 365;
112 /* ECMA-262 3th Edition 15.9.1.3 */
113 static inline DOUBLE
day_from_year(DOUBLE year
)
117 if(year
!= (int)year
)
121 return 365*(y
-1970) + floor((y
-1969)/4)
122 - floor((y
-1901)/100) + floor((y
-1601)/400);
125 /* ECMA-262 3th Edition 15.9.1.3 */
126 static inline DOUBLE
time_from_year(DOUBLE year
)
128 return MS_PER_DAY
*day_from_year(year
);
131 /* ECMA-262 3th Edition 15.9.1.3 */
132 static inline DOUBLE
year_from_time(DOUBLE time
)
139 y
= 1970 + time
/365.25/MS_PER_DAY
;
141 if(time_from_year(y
) > time
)
142 while(time_from_year(y
) > time
) y
--;
144 while(time_from_year(y
+1)<=time
) y
++;
149 /* ECMA-262 3th Edition 15.9.1.3 */
150 static inline int in_leap_year(DOUBLE time
)
152 if(days_in_year(year_from_time(time
))==366)
157 /* ECMA-262 3th Edition 15.9.1.4 */
158 static inline int day_within_year(DOUBLE time
)
160 return day(time
) - day_from_year(year_from_time(time
));
163 /* ECMA-262 3th Edition 15.9.1.4 */
164 static inline DOUBLE
month_from_time(DOUBLE time
)
166 int ily
= in_leap_year(time
);
167 int dwy
= day_within_year(time
);
172 if(0<=dwy
&& dwy
<31) return 0;
173 if(dwy
< 59+ily
) return 1;
174 if(dwy
< 90+ily
) return 2;
175 if(dwy
< 120+ily
) return 3;
176 if(dwy
< 151+ily
) return 4;
177 if(dwy
< 181+ily
) return 5;
178 if(dwy
< 212+ily
) return 6;
179 if(dwy
< 243+ily
) return 7;
180 if(dwy
< 273+ily
) return 8;
181 if(dwy
< 304+ily
) return 9;
182 if(dwy
< 334+ily
) return 10;
186 /* ECMA-262 3th Edition 15.9.1.5 */
187 static inline DOUBLE
date_from_time(DOUBLE time
)
189 int dwy
= day_within_year(time
);
190 int ily
= in_leap_year(time
);
191 int mft
= month_from_time(time
);
196 if(mft
==0) return dwy
+1;
197 if(mft
==1) return dwy
-30;
198 if(mft
==2) return dwy
-58-ily
;
199 if(mft
==3) return dwy
-89-ily
;
200 if(mft
==4) return dwy
-119-ily
;
201 if(mft
==5) return dwy
-150-ily
;
202 if(mft
==6) return dwy
-180-ily
;
203 if(mft
==7) return dwy
-211-ily
;
204 if(mft
==8) return dwy
-242-ily
;
205 if(mft
==9) return dwy
-272-ily
;
206 if(mft
==10) return dwy
-303-ily
;
210 /* ECMA-262 3th Edition 15.9.1.6 */
211 static inline DOUBLE
week_day(DOUBLE time
)
218 ret
= fmod(day(time
)+4, 7);
224 /* ECMA-262 3th Edition 15.9.1.10 */
225 static inline DOUBLE
hour_from_time(DOUBLE time
)
232 ret
= fmod(floor(time
/MS_PER_HOUR
), 24);
238 /* ECMA-262 3th Edition 15.9.1.10 */
239 static inline DOUBLE
min_from_time(DOUBLE time
)
246 ret
= fmod(floor(time
/MS_PER_MINUTE
), 60);
252 /* ECMA-262 3th Edition 15.9.1.10 */
253 static inline DOUBLE
sec_from_time(DOUBLE time
)
260 ret
= fmod(floor(time
/1000), 60);
266 /* ECMA-262 3th Edition 15.9.1.10 */
267 static inline DOUBLE
ms_from_time(DOUBLE time
)
274 ret
= fmod(time
, 1000);
275 if(ret
<0) ret
+= 1000;
280 /* ECMA-262 3rd Edition 15.9.1.14 */
281 static inline DOUBLE
time_clip(DOUBLE time
)
283 if(8.64e15
< time
|| time
< -8.64e15
) {
290 static HRESULT
Date_toString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
291 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
297 static HRESULT
Date_toLocaleString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
298 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
304 static HRESULT
Date_hasOwnProperty(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
305 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
311 static HRESULT
Date_propertyIsEnumerable(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
312 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
318 static HRESULT
Date_isPrototypeOf(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
319 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
325 static HRESULT
Date_valueOf(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
326 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
332 static HRESULT
Date_toUTCString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
333 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
339 static HRESULT
Date_toDateString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
340 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
346 static HRESULT
Date_toTimeString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
347 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
353 static HRESULT
Date_toLocaleDateString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
354 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
360 static HRESULT
Date_toLocaleTimeString(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
361 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
367 /* ECMA-262 3rd Edition 15.9.5.9 */
368 static HRESULT
Date_getTime(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
369 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
373 if(!is_class(dispex
, JSCLASS_DATE
)) {
374 FIXME("throw TypeError\n");
379 DateInstance
*date
= (DateInstance
*)dispex
;
380 num_set_val(retv
, date
->time
);
385 /* ECMA-262 3th Edition 15.9.1.3 */
386 static HRESULT
Date_getFullYear(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
387 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
391 if(!is_class(dispex
, JSCLASS_DATE
)) {
392 FIXME("throw TypeError\n");
397 DateInstance
*date
= (DateInstance
*)dispex
;
398 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
400 num_set_val(retv
, year_from_time(time
));
405 /* ECMA-262 3th Edition 15.9.1.3 */
406 static HRESULT
Date_getUTCFullYear(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
407 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
411 if(!is_class(dispex
, JSCLASS_DATE
)) {
412 FIXME("throw TypeError\n");
417 DateInstance
*date
= (DateInstance
*)dispex
;
418 num_set_val(retv
, year_from_time(date
->time
));
423 /* ECMA-262 3th Edition 15.9.1.4 */
424 static HRESULT
Date_getMonth(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
425 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
429 if(!is_class(dispex
, JSCLASS_DATE
)) {
430 FIXME("throw TypeError\n");
435 DateInstance
*date
= (DateInstance
*)dispex
;
436 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
438 num_set_val(retv
, month_from_time(time
));
443 /* ECMA-262 3th Edition 15.9.1.4 */
444 static HRESULT
Date_getUTCMonth(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
445 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
449 if(!is_class(dispex
, JSCLASS_DATE
)) {
450 FIXME("throw TypeError\n");
455 DateInstance
*date
= (DateInstance
*)dispex
;
456 num_set_val(retv
, month_from_time(date
->time
));
461 /* ECMA-262 3th Edition 15.9.1.5 */
462 static HRESULT
Date_getDate(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
463 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
467 if(!is_class(dispex
, JSCLASS_DATE
)) {
468 FIXME("throw TypeError\n");
473 DateInstance
*date
= (DateInstance
*)dispex
;
474 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
476 num_set_val(retv
, date_from_time(time
));
481 /* ECMA-262 3th Edition 15.9.1.5 */
482 static HRESULT
Date_getUTCDate(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
483 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
487 if(!is_class(dispex
, JSCLASS_DATE
)) {
488 FIXME("throw TypeError\n");
493 DateInstance
*date
= (DateInstance
*)dispex
;
494 num_set_val(retv
, date_from_time(date
->time
));
499 /* ECMA-262 3th Edition 15.9.1.6 */
500 static HRESULT
Date_getDay(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
501 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
505 if(!is_class(dispex
, JSCLASS_DATE
)) {
506 FIXME("throw TypeError\n");
511 DateInstance
*date
= (DateInstance
*)dispex
;
512 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
514 num_set_val(retv
, week_day(time
));
519 /* ECMA-262 3th Edition 15.9.1.6 */
520 static HRESULT
Date_getUTCDay(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
521 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
525 if(!is_class(dispex
, JSCLASS_DATE
)) {
526 FIXME("throw TypeError\n");
531 DateInstance
*date
= (DateInstance
*)dispex
;
532 num_set_val(retv
, week_day(date
->time
));
537 /* ECMA-262 3th Edition 15.9.1.10 */
538 static HRESULT
Date_getHours(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
539 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
543 if(!is_class(dispex
, JSCLASS_DATE
)) {
544 FIXME("throw TypeError\n");
549 DateInstance
*date
= (DateInstance
*)dispex
;
550 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
552 num_set_val(retv
, hour_from_time(time
));
557 /* ECMA-262 3th Edition 15.9.1.10 */
558 static HRESULT
Date_getUTCHours(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
559 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
563 if(!is_class(dispex
, JSCLASS_DATE
)) {
564 FIXME("throw TypeError\n");
569 DateInstance
*date
= (DateInstance
*)dispex
;
570 num_set_val(retv
, hour_from_time(date
->time
));
575 /* ECMA-262 3th Edition 15.9.1.10 */
576 static HRESULT
Date_getMinutes(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
577 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
581 if(!is_class(dispex
, JSCLASS_DATE
)) {
582 FIXME("throw TypeError\n");
587 DateInstance
*date
= (DateInstance
*)dispex
;
588 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
590 num_set_val(retv
, min_from_time(time
));
595 /* ECMA-262 3th Edition 15.9.1.10 */
596 static HRESULT
Date_getUTCMinutes(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
597 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
601 if(!is_class(dispex
, JSCLASS_DATE
)) {
602 FIXME("throw TypeError\n");
607 DateInstance
*date
= (DateInstance
*)dispex
;
608 num_set_val(retv
, min_from_time(date
->time
));
613 /* ECMA-262 3th Edition 15.9.1.10 */
614 static HRESULT
Date_getSeconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
615 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
619 if(!is_class(dispex
, JSCLASS_DATE
)) {
620 FIXME("throw TypeError\n");
625 DateInstance
*date
= (DateInstance
*)dispex
;
626 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
628 num_set_val(retv
, sec_from_time(time
));
633 /* ECMA-262 3th Edition 15.9.1.10 */
634 static HRESULT
Date_getUTCSeconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
635 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
639 if(!is_class(dispex
, JSCLASS_DATE
)) {
640 FIXME("throw TypeError\n");
645 DateInstance
*date
= (DateInstance
*)dispex
;
646 num_set_val(retv
, sec_from_time(date
->time
));
651 /* ECMA-262 3th Edition 15.9.1.10 */
652 static HRESULT
Date_getMilliseconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
653 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
657 if(!is_class(dispex
, JSCLASS_DATE
)) {
658 FIXME("throw TypeError\n");
663 DateInstance
*date
= (DateInstance
*)dispex
;
664 DOUBLE time
= date
->time
- date
->bias
*MS_PER_MINUTE
;
666 num_set_val(retv
, ms_from_time(time
));
671 /* ECMA-262 3th Edition 15.9.1.10 */
672 static HRESULT
Date_getUTCMilliseconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
673 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
677 if(!is_class(dispex
, JSCLASS_DATE
)) {
678 FIXME("throw TypeError\n");
683 DateInstance
*date
= (DateInstance
*)dispex
;
684 num_set_val(retv
, ms_from_time(date
->time
));
689 static HRESULT
Date_getTimezoneOffset(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
690 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
696 static HRESULT
Date_setTime(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
697 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
705 if(!is_class(dispex
, JSCLASS_DATE
)) {
706 FIXME("throw TypeError\n");
711 FIXME("throw ArgumentNotOptional\n");
712 if(retv
) num_set_nan(retv
);
716 hres
= to_number(dispex
->ctx
, get_arg(dp
, 0), ei
, &v
);
720 date
= (DateInstance
*)dispex
;
721 date
->time
= time_clip(num_val(&v
));
724 num_set_val(retv
, date
->time
);
729 static HRESULT
Date_setMilliseconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
730 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
738 if(!is_class(dispex
, JSCLASS_DATE
)) {
739 FIXME("throw TypeError\n");
744 FIXME("throw ArgumentNotOptional\n");
745 if(retv
) num_set_nan(retv
);
749 hres
= to_number(dispex
->ctx
, get_arg(dp
, 0), ei
, &v
);
753 date
= (DateInstance
*)dispex
;
754 date
->time
= time_clip(date
->time
- ms_from_time(date
->time
) + num_val(&v
));
757 num_set_val(retv
, date
->time
);
762 static HRESULT
Date_setUTCMilliseconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
763 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
765 return Date_setMilliseconds(dispex
, lcid
, flags
, dp
, retv
, ei
, caller
);
768 static HRESULT
Date_setSeconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
769 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
777 if(!is_class(dispex
, JSCLASS_DATE
)) {
778 FIXME("throw TypeError\n");
783 FIXME("throw ArgumentNotOptional\n");
784 if(retv
) num_set_nan(retv
);
788 hres
= to_number(dispex
->ctx
, get_arg(dp
, 0), ei
, &v
);
792 date
= (DateInstance
*)dispex
;
793 date
->time
= time_clip(date
->time
- (sec_from_time(date
->time
) - num_val(&v
))*1000.0);
796 num_set_val(retv
, date
->time
);
801 static HRESULT
Date_setUTCSeconds(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
802 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
804 return Date_setSeconds(dispex
, lcid
, flags
, dp
, retv
, ei
, caller
);
807 static HRESULT
Date_setMinutes(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
808 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
816 if(!is_class(dispex
, JSCLASS_DATE
)) {
817 FIXME("throw TypeError\n");
822 FIXME("throw ArgumentNotOptional\n");
823 if(retv
) num_set_nan(retv
);
827 hres
= to_number(dispex
->ctx
, get_arg(dp
, 0), ei
, &v
);
831 date
= (DateInstance
*)dispex
;
832 date
->time
= time_clip(date
->time
- (min_from_time(date
->time
) - num_val(&v
))*MS_PER_MINUTE
);
835 num_set_val(retv
, date
->time
);
840 static HRESULT
Date_setUTCMinutes(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
841 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
843 return Date_setMinutes(dispex
, lcid
, flags
, dp
, retv
, ei
, caller
);
846 static HRESULT
Date_setHours(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
847 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
855 if(!is_class(dispex
, JSCLASS_DATE
)) {
856 FIXME("throw TypeError\n");
861 FIXME("throw ArgumentNotOptional\n");
862 if(retv
) num_set_nan(retv
);
866 hres
= to_number(dispex
->ctx
, get_arg(dp
, 0), ei
, &v
);
870 date
= (DateInstance
*)dispex
;
871 date
->time
= time_clip(date
->time
872 - (hour_from_time(date
->time
- date
->bias
*MS_PER_MINUTE
)
873 - num_val(&v
))*MS_PER_HOUR
);
876 num_set_val(retv
, date
->time
);
881 static HRESULT
Date_setUTCHours(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
882 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
890 if(!is_class(dispex
, JSCLASS_DATE
)) {
891 FIXME("throw TypeError\n");
896 FIXME("throw ArgumentNotOptional\n");
897 if(retv
) num_set_nan(retv
);
901 hres
= to_number(dispex
->ctx
, get_arg(dp
, 0), ei
, &v
);
905 date
= (DateInstance
*)dispex
;
906 date
->time
= time_clip(date
->time
907 - (hour_from_time(date
->time
) - num_val(&v
))*MS_PER_HOUR
);
910 num_set_val(retv
, date
->time
);
915 static HRESULT
Date_setDate(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
916 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
922 static HRESULT
Date_setUTCDate(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
923 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
929 static HRESULT
Date_setMonth(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
930 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
936 static HRESULT
Date_setUTCMonth(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
937 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
943 static HRESULT
Date_setFullYear(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
944 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
950 static HRESULT
Date_setUTCFullYear(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
951 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
957 static HRESULT
Date_value(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
958 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*caller
)
964 static const builtin_prop_t Date_props
[] = {
965 {getDateW
, Date_getDate
, PROPF_METHOD
},
966 {getDayW
, Date_getDay
, PROPF_METHOD
},
967 {getFullYearW
, Date_getFullYear
, PROPF_METHOD
},
968 {getHoursW
, Date_getHours
, PROPF_METHOD
},
969 {getMillisecondsW
, Date_getMilliseconds
, PROPF_METHOD
},
970 {getMinutesW
, Date_getMinutes
, PROPF_METHOD
},
971 {getMonthW
, Date_getMonth
, PROPF_METHOD
},
972 {getSecondsW
, Date_getSeconds
, PROPF_METHOD
},
973 {getTimeW
, Date_getTime
, PROPF_METHOD
},
974 {getTimezoneOffsetW
, Date_getTimezoneOffset
, PROPF_METHOD
},
975 {getUTCDateW
, Date_getUTCDate
, PROPF_METHOD
},
976 {getUTCDayW
, Date_getUTCDay
, PROPF_METHOD
},
977 {getUTCFullYearW
, Date_getUTCFullYear
, PROPF_METHOD
},
978 {getUTCHoursW
, Date_getUTCHours
, PROPF_METHOD
},
979 {getUTCMillisecondsW
, Date_getUTCMilliseconds
, PROPF_METHOD
},
980 {getUTCMinutesW
, Date_getUTCMinutes
, PROPF_METHOD
},
981 {getUTCMonthW
, Date_getUTCMonth
, PROPF_METHOD
},
982 {getUTCSecondsW
, Date_getUTCSeconds
, PROPF_METHOD
},
983 {hasOwnPropertyW
, Date_hasOwnProperty
, PROPF_METHOD
},
984 {isPrototypeOfW
, Date_isPrototypeOf
, PROPF_METHOD
},
985 {propertyIsEnumerableW
, Date_propertyIsEnumerable
, PROPF_METHOD
},
986 {setDateW
, Date_setDate
, PROPF_METHOD
},
987 {setFullYearW
, Date_setFullYear
, PROPF_METHOD
},
988 {setHoursW
, Date_setHours
, PROPF_METHOD
},
989 {setMillisecondsW
, Date_setMilliseconds
, PROPF_METHOD
},
990 {setMinutesW
, Date_setMinutes
, PROPF_METHOD
},
991 {setMonthW
, Date_setMonth
, PROPF_METHOD
},
992 {setSecondsW
, Date_setSeconds
, PROPF_METHOD
},
993 {setTimeW
, Date_setTime
, PROPF_METHOD
},
994 {setUTCDateW
, Date_setUTCDate
, PROPF_METHOD
},
995 {setUTCFullYearW
, Date_setUTCFullYear
, PROPF_METHOD
},
996 {setUTCHoursW
, Date_setUTCHours
, PROPF_METHOD
},
997 {setUTCMillisecondsW
, Date_setUTCMilliseconds
, PROPF_METHOD
},
998 {setUTCMinutesW
, Date_setUTCMinutes
, PROPF_METHOD
},
999 {setUTCMonthW
, Date_setUTCMonth
, PROPF_METHOD
},
1000 {setUTCSecondsW
, Date_setUTCSeconds
, PROPF_METHOD
},
1001 {toDateStringW
, Date_toDateString
, PROPF_METHOD
},
1002 {toLocaleDateStringW
, Date_toLocaleDateString
, PROPF_METHOD
},
1003 {toLocaleStringW
, Date_toLocaleString
, PROPF_METHOD
},
1004 {toLocaleTimeStringW
, Date_toLocaleTimeString
, PROPF_METHOD
},
1005 {toStringW
, Date_toString
, PROPF_METHOD
},
1006 {toTimeStringW
, Date_toTimeString
, PROPF_METHOD
},
1007 {toUTCStringW
, Date_toUTCString
, PROPF_METHOD
},
1008 {valueOfW
, Date_valueOf
, PROPF_METHOD
},
1011 static const builtin_info_t Date_info
= {
1013 {NULL
, Date_value
, 0},
1014 sizeof(Date_props
)/sizeof(*Date_props
),
1020 static HRESULT
create_date(script_ctx_t
*ctx
, BOOL use_constr
, DOUBLE time
, DispatchEx
**ret
)
1024 TIME_ZONE_INFORMATION tzi
;
1026 GetTimeZoneInformation(&tzi
);
1028 date
= heap_alloc_zero(sizeof(DateInstance
));
1030 return E_OUTOFMEMORY
;
1033 hres
= init_dispex_from_constr(&date
->dispex
, ctx
, &Date_info
, ctx
->date_constr
);
1035 hres
= init_dispex(&date
->dispex
, ctx
, &Date_info
, NULL
);
1042 date
->bias
= tzi
.Bias
;
1044 *ret
= &date
->dispex
;
1048 static HRESULT
DateConstr_value(DispatchEx
*dispex
, LCID lcid
, WORD flags
, DISPPARAMS
*dp
,
1049 VARIANT
*retv
, jsexcept_t
*ei
, IServiceProvider
*sp
)
1057 case DISPATCH_CONSTRUCT
:
1058 switch(arg_cnt(dp
)) {
1059 /* ECMA-262 3rd Edition 15.9.3.3 */
1064 GetSystemTimeAsFileTime(&time
);
1065 lltime
= ((LONGLONG
)time
.dwHighDateTime
<<32)
1066 + time
.dwLowDateTime
;
1068 hres
= create_date(dispex
->ctx
, TRUE
, lltime
/10000-TIME_EPOCH
, &date
);
1074 /* ECMA-262 3rd Edition 15.9.3.2 */
1078 hres
= to_primitive(dispex
->ctx
, get_arg(dp
,0), ei
, &prim
);
1082 if(V_VT(&prim
) == VT_BSTR
) {
1083 FIXME("VT_BSTR not supported\n");
1087 hres
= to_number(dispex
->ctx
, &prim
, ei
, &num
);
1088 VariantClear(&prim
);
1092 hres
= create_date(dispex
->ctx
, TRUE
, time_clip(num_val(&num
)), &date
);
1099 FIXME("unimplemented argcnt %d\n", arg_cnt(dp
));
1103 V_VT(retv
) = VT_DISPATCH
;
1104 V_DISPATCH(retv
) = (IDispatch
*)_IDispatchEx_(date
);
1108 FIXME("unimplemented flags %x\n", flags
);
1115 HRESULT
create_date_constr(script_ctx_t
*ctx
, DispatchEx
**ret
)
1120 hres
= create_date(ctx
, FALSE
, 0.0, &date
);
1124 hres
= create_builtin_function(ctx
, DateConstr_value
, PROPF_CONSTR
, date
, ret
);
1126 jsdisp_release(date
);