1 /* Digital Mars DMDScript source code.
2  * Copyright (c) 2000-2002 by Chromium Communications
3  * D version Copyright (c) 2004-2010 by Digital Mars
4  * Distributed under the Boost Software License, Version 1.0.
5  * (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6  * written by Walter Bright
7  * http://www.digitalmars.com
8  *
9  * D2 port by Dmitry Olshansky 
10  *
11  * DMDScript is implemented in the D Programming Language,
12  * http://www.digitalmars.com/d/
13  *
14  * For a C++ implementation of DMDScript, including COM support, see
15  * http://www.digitalmars.com/dscript/cppscript.html
16  */
17 
18 module dmdscript.ddate;
19 
20 import undead.date;
21 import std.math;
22 
23 debug
24 {
25     import std.stdio;
26 }
27 
28 import dmdscript.script;
29 import dmdscript.dobject;
30 import dmdscript.value;
31 import dmdscript.threadcontext;
32 import dmdscript.dfunction;
33 import dmdscript.dnative;
34 import dmdscript.property;
35 import dmdscript.text;
36 import dmdscript.errmsgs;
37 
38 
39 version = DATETOSTRING;                 // use DateToString
40 
41 enum TIMEFORMAT
42 {
43     String,
44     DateString,
45     TimeString,
46     LocaleString,
47     LocaleDateString,
48     LocaleTimeString,
49     UTCString,
50 }
51 
52 d_time parseDateString(CallContext *cc, string s)
53 {
54     return parse(s);
55 }
56 
57 string dateToString(CallContext *cc, d_time t, TIMEFORMAT tf)
58 {
59     string p;
60 
61     if(t == d_time_nan)
62         p = "Invalid Date";
63     else
64     {
65         switch(tf)
66         {
67         case TIMEFORMAT.String:
68             t = localTimetoUTC(t);
69             p = UTCtoString(t);
70             break;
71 
72         case TIMEFORMAT.DateString:
73             t = localTimetoUTC(t);
74             p = toDateString(t);
75             break;
76 
77         case TIMEFORMAT.TimeString:
78             t = localTimetoUTC(t);
79             p = toTimeString(t);
80             break;
81 
82         case TIMEFORMAT.LocaleString:
83             //p = toLocaleString(t);
84             p = UTCtoString(t);
85             break;
86 
87         case TIMEFORMAT.LocaleDateString:
88             //p = toLocaleDateString(t);
89             p = toDateString(t);
90             break;
91 
92         case TIMEFORMAT.LocaleTimeString:
93             //p = toLocaleTimeString(t);
94             p = toTimeString(t);
95             break;
96 
97         case TIMEFORMAT.UTCString:
98             p = toUTCString(t);
99             //p = toString(t);
100             break;
101 
102         default:
103             assert(0);
104         }
105     }
106     return p;
107 }
108 
109 
110 /* ===================== Ddate.constructor functions ==================== */
111 
112 void* Ddate_parse(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
113 {
114     // ECMA 15.9.4.2
115     string s;
116     d_time n;
117 
118     if(arglist.length == 0)
119         n = d_time_nan;
120     else
121     {
122         s = arglist[0].toString();
123         n = parseDateString(cc, s);
124     }
125 
126     ret.putVtime(n);
127     return null;
128 }
129 
130 void* Ddate_UTC(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
131 {
132     // ECMA 15.9.4.3 - 15.9.4.10
133 
134     d_time n;
135 
136     d_time year;
137     d_time month = 0;
138     d_time date = 0;
139     d_time hours = 0;
140     d_time minutes = 0;
141     d_time seconds = 0;
142     d_time ms = 0;
143 
144     d_time day;
145     d_time time = 0;
146 
147     switch(arglist.length)
148     {
149     default:
150     case 7:
151         ms = arglist[6].toDtime();
152         goto case;
153     case 6:
154         seconds = arglist[5].toDtime();
155         goto case;
156     case 5:
157         minutes = arglist[4].toDtime();
158         goto case;
159     case 4:
160         hours = arglist[3].toDtime();
161         time = makeTime(hours, minutes, seconds, ms);
162         goto case;
163     case 3:
164         date = arglist[2].toDtime();
165         goto case;
166     case 2:
167         month = arglist[1].toDtime();
168         goto case;
169     case 1:
170         year = arglist[0].toDtime();
171 
172         if(year != d_time_nan && year >= 0 && year <= 99)
173             year += 1900;
174         day = makeDay(year, month, date);
175         n = timeClip(makeDate(day, time));
176         break;
177 
178     case 0:
179         n = getUTCtime();
180         break;
181     }
182     ret.putVtime(n);
183     return null;
184 }
185 
186 /* ===================== Ddate_constructor ==================== */
187 
188 class DdateConstructor : Dfunction
189 {
190     this()
191     {
192         super(7, Dfunction_prototype);
193         name = "Date";
194 
195         static enum NativeFunctionData[] nfd =
196         [
197             { TEXT_parse, &Ddate_parse, 1 },
198             { TEXT_UTC, &Ddate_UTC, 7 },
199         ];
200 
201         DnativeFunction.initialize(this, nfd, DontEnum);
202     }
203 
204     override void *Construct(CallContext *cc, Value *ret, Value[] arglist)
205     {
206         // ECMA 15.9.3
207         Dobject o;
208         d_time n;
209 
210         d_time year;
211         d_time month;
212         d_time date = 0;
213         d_time hours = 0;
214         d_time minutes = 0;
215         d_time seconds = 0;
216         d_time ms = 0;
217 
218         d_time day;
219         d_time time = 0;
220         //generate NaN check boilerplate code
221         static d_string breakOnNan(d_string var)
222         {
223             return "if(" ~ var ~ " == d_time_nan){
224 			n = d_time_nan;
225 			break;
226 		}";
227         }
228         //writefln("Ddate_constructor.Construct()");
229         switch(arglist.length)
230         {
231         default:
232         case 7:
233             ms = arglist[6].toDtime();
234             mixin (breakOnNan("ms"));
235             goto case;
236         case 6:
237             seconds = arglist[5].toDtime();
238             mixin (breakOnNan("seconds"));
239             goto case;
240         case 5:
241             minutes = arglist[4].toDtime();
242             mixin (breakOnNan("minutes"));
243             goto case;
244         case 4:
245             hours = arglist[3].toDtime();
246             mixin (breakOnNan("hours"));
247             time = makeTime(hours, minutes, seconds, ms);
248             goto case;
249         case 3:
250             date = arglist[2].toDtime();
251             goto case;
252         case 2:
253             month = arglist[1].toDtime();
254             year = arglist[0].toDtime();
255 
256             if(year != d_time_nan && year >= 0 && year <= 99)
257                 year += 1900;
258             day = makeDay(year, month, date);
259             n = timeClip(localTimetoUTC(makeDate(day, time)));
260             break;
261 
262         case 1:
263             arglist[0].toPrimitive(ret, null);
264             if(ret.getType() == TypeString)
265             {
266                 n = parseDateString(cc, ret..string);
267             }
268             else
269             {
270                 n = ret.toDtime();
271                 n = timeClip(n);
272             }
273             break;
274 
275         case 0:
276             n = getUTCtime();
277             break;
278         }
279         //writefln("\tn = %s", n);
280         o = new Ddate(n);
281         ret.putVobject(o);
282         return null;
283     }
284 
285     override void *Call(CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
286     {
287         // ECMA 15.9.2
288         // return string as if (new Date()).toString()
289         immutable(char)[] s;
290         d_time t;
291 
292         version(DATETOSTRING)
293         {
294             t = getUTCtime();
295             t = UTCtoLocalTime(t);
296             s = dateToString(cc, t, TIMEFORMAT.String);
297         }
298         else
299         {
300             t = time();
301             s = toString(t);
302         }
303         ret.putVstring(s);
304         return null;
305     }
306 }
307 
308 
309 /* ===================== Ddate.prototype functions =============== */
310 
311 void *checkdate(Value* ret, d_string name, Dobject othis)
312 {
313     ret.putVundefined();
314     ErrInfo errinfo;
315     return Dobject.RuntimeError(&errinfo, errmsgtbl[ERR_FUNCTION_WANTS_DATE],
316                                 name, othis.classname);
317 }
318 
319 int getThisTime(Value* ret, Dobject othis, out d_time n)
320 {
321     d_number x;
322 
323     n = cast(d_time)othis.value.number;
324     ret.putVtime(n);
325     return (n == d_time_nan) ? 1 : 0;
326 }
327 
328 int getThisLocalTime(Value* ret, Dobject othis, out d_time n)
329 {
330     int isn = 1;
331 
332     n = cast(d_time)othis.value.number;
333     if(n != d_time_nan)
334     {
335         isn = 0;
336         n = UTCtoLocalTime(n);
337     }
338     ret.putVtime(n);
339     return isn;
340 }
341 
342 void* Ddate_prototype_toString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
343 {
344     // ECMA 15.9.5.2
345     d_time n;
346     immutable(char)[] s;
347 
348     //writefln("Ddate_prototype_toString()");
349     if(!othis.isDdate())
350         return checkdate(ret, TEXT_toString, othis);
351 
352     version(DATETOSTRING)
353     {
354         getThisLocalTime(ret, othis, n);
355         s = dateToString(cc, n, TIMEFORMAT.String);
356     }
357     else
358     {
359         getThisTime(ret, othis, n);
360         s = toString(n);
361     }
362     ret.putVstring(s);
363     return null;
364 }
365 
366 void* Ddate_prototype_toDateString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
367 {
368     // ECMA 15.9.5.3
369     d_time n;
370     immutable(char)[] s;
371 
372     if(!othis.isDdate())
373         return checkdate(ret, TEXT_toDateString, othis);
374 
375     version(DATETOSTRING)
376     {
377         getThisLocalTime(ret, othis, n);
378         s = dateToString(cc, n, TIMEFORMAT.DateString);
379     }
380     else
381     {
382         getThisTime(ret, othis, n);
383         s = toDateString(n);
384     }
385     ret.putVstring(s);
386     return null;
387 }
388 
389 void* Ddate_prototype_toTimeString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
390 {
391     // ECMA 15.9.5.4
392     d_time n;
393     immutable(char)[] s;
394 
395     if(!othis.isDdate())
396         return checkdate(ret, TEXT_toTimeString, othis);
397 
398     version(DATETOSTRING)
399     {
400         getThisLocalTime(ret, othis, n);
401         s = dateToString(cc, n, TIMEFORMAT.TimeString);
402     }
403     else
404     {
405         getThisTime(ret, othis, n);
406         s = toTimeString(n);
407     }
408     //s = toTimeString(n);
409     ret.putVstring(s);
410     return null;
411 }
412 
413 void* Ddate_prototype_valueOf(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
414 {
415     // ECMA 15.9.5.3
416     d_time n;
417 
418     if(!othis.isDdate())
419         return checkdate(ret, TEXT_valueOf, othis);
420     getThisTime(ret, othis, n);
421     return null;
422 }
423 
424 void* Ddate_prototype_getTime(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
425 {
426     // ECMA 15.9.5.4
427     d_time n;
428 
429     if(!othis.isDdate())
430         return checkdate(ret, TEXT_getTime, othis);
431     getThisTime(ret, othis, n);
432     return null;
433 }
434 
435 void* Ddate_prototype_getYear(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
436 {
437     // ECMA 15.9.5.5
438     d_time n;
439 
440     if(!othis.isDdate())
441         return checkdate(ret, TEXT_getYear, othis);
442 
443     if(getThisLocalTime(ret, othis, n) == 0)
444     {
445         n = yearFromTime(n);
446         if(n != d_time_nan)
447         {
448             n -= 1900;
449             version(all)  // emulate jscript bug
450             {
451                 if(n < 0 || n >= 100)
452                     n += 1900;
453             }
454         }
455         ret.putVtime(n);
456     }
457     return null;
458 }
459 
460 void* Ddate_prototype_getFullYear(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
461 {
462     // ECMA 15.9.5.6
463     d_time n;
464 
465     if(!othis.isDdate())
466         return checkdate(ret, TEXT_getFullYear, othis);
467 
468     if(getThisLocalTime(ret, othis, n) == 0)
469     {
470         n = yearFromTime(n);
471         ret.putVtime(n);
472     }
473     return null;
474 }
475 
476 void* Ddate_prototype_getUTCFullYear(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
477 {
478     // ECMA 15.9.5.7
479     d_time n;
480 
481     if(!othis.isDdate())
482         return checkdate(ret, TEXT_getUTCFullYear, othis);
483     if(getThisTime(ret, othis, n) == 0)
484     {
485         n = yearFromTime(n);
486         ret.putVtime(n);
487     }
488     return null;
489 }
490 
491 void* Ddate_prototype_getMonth(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
492 {
493     // ECMA 15.9.5.8
494     d_time n;
495 
496     if(!othis.isDdate())
497         return checkdate(ret, TEXT_getMonth, othis);
498 
499     if(getThisLocalTime(ret, othis, n) == 0)
500     {
501         n = monthFromTime(n);
502         ret.putVtime(n);
503     }
504     return null;
505 }
506 
507 void* Ddate_prototype_getUTCMonth(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
508 {
509     // ECMA 15.9.5.9
510     d_time n;
511 
512     if(!othis.isDdate())
513         return checkdate(ret, TEXT_getUTCMonth, othis);
514 
515     if(getThisTime(ret, othis, n) == 0)
516     {
517         n = monthFromTime(n);
518         ret.putVtime(n);
519     }
520     return null;
521 }
522 
523 void* Ddate_prototype_getDate(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
524 {
525     // ECMA 15.9.5.10
526     d_time n;
527 
528     if(!othis.isDdate())
529         return checkdate(ret, TEXT_getDate, othis);
530 
531     if(getThisLocalTime(ret, othis, n) == 0)
532     {
533         //printf("LocalTime = %.16g\n", n);
534         //printf("DaylightSavingTA(n) = %d\n", daylightSavingTA(n));
535         n = dateFromTime(n);
536         ret.putVtime(n);
537     }
538     return null;
539 }
540 
541 void* Ddate_prototype_getUTCDate(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
542 {
543     // ECMA 15.9.5.11
544     d_time n;
545 
546     if(!othis.isDdate())
547         return checkdate(ret, TEXT_getUTCDate, othis);
548 
549     if(getThisTime(ret, othis, n) == 0)
550     {
551         n = dateFromTime(n);
552         ret.putVtime(n);
553     }
554     return null;
555 }
556 
557 void* Ddate_prototype_getDay(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
558 {
559     // ECMA 15.9.5.12
560     d_time n;
561 
562     if(!othis.isDdate())
563         return checkdate(ret, TEXT_getDay, othis);
564 
565     if(getThisLocalTime(ret, othis, n) == 0)
566     {
567         n = weekDay(n);
568         ret.putVtime(n);
569     }
570     return null;
571 }
572 
573 void* Ddate_prototype_getUTCDay(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
574 {
575     // ECMA 15.9.5.13
576     d_time n;
577 
578     if(!othis.isDdate())
579         return checkdate(ret, TEXT_getUTCDay, othis);
580 
581     if(getThisTime(ret, othis, n) == 0)
582     {
583         n = weekDay(n);
584         ret.putVtime(n);
585     }
586     return null;
587 }
588 
589 void* Ddate_prototype_getHours(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
590 {
591     // ECMA 15.9.5.14
592     d_time n;
593 
594     if(!othis.isDdate())
595         return checkdate(ret, TEXT_getHours, othis);
596 
597     if(getThisLocalTime(ret, othis, n) == 0)
598     {
599         n = hourFromTime(n);
600         ret.putVtime(n);
601     }
602     return null;
603 }
604 
605 void* Ddate_prototype_getUTCHours(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
606 {
607     // ECMA 15.9.5.15
608     d_time n;
609 
610     if(!othis.isDdate())
611         return checkdate(ret, TEXT_getUTCHours, othis);
612 
613     if(getThisTime(ret, othis, n) == 0)
614     {
615         n = hourFromTime(n);
616         ret.putVtime(n);
617     }
618     return null;
619 }
620 
621 void* Ddate_prototype_getMinutes(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
622 {
623     // ECMA 15.9.5.16
624     d_time n;
625 
626     if(!othis.isDdate())
627         return checkdate(ret, TEXT_getMinutes, othis);
628 
629     if(getThisLocalTime(ret, othis, n) == 0)
630     {
631         n = minFromTime(n);
632         ret.putVtime(n);
633     }
634     return null;
635 }
636 
637 void* Ddate_prototype_getUTCMinutes(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
638 {
639     // ECMA 15.9.5.17
640     d_time n;
641 
642     if(!othis.isDdate())
643         return checkdate(ret, TEXT_getUTCMinutes, othis);
644 
645     if(getThisTime(ret, othis, n) == 0)
646     {
647         n = minFromTime(n);
648         ret.putVtime(n);
649     }
650     return null;
651 }
652 
653 void* Ddate_prototype_getSeconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
654 {
655     // ECMA 15.9.5.18
656     d_time n;
657 
658     if(!othis.isDdate())
659         return checkdate(ret, TEXT_getSeconds, othis);
660 
661     if(getThisLocalTime(ret, othis, n) == 0)
662     {
663         n = secFromTime(n);
664         ret.putVtime(n);
665     }
666     return null;
667 }
668 
669 void* Ddate_prototype_getUTCSeconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
670 {
671     // ECMA 15.9.5.19
672     d_time n;
673 
674     if(!othis.isDdate())
675         return checkdate(ret, TEXT_getUTCSeconds, othis);
676 
677     if(getThisTime(ret, othis, n) == 0)
678     {
679         n = secFromTime(n);
680         ret.putVtime(n);
681     }
682     return null;
683 }
684 
685 void* Ddate_prototype_getMilliseconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
686 {
687     // ECMA 15.9.5.20
688     d_time n;
689 
690     if(!othis.isDdate())
691         return checkdate(ret, TEXT_getMilliseconds, othis);
692 
693     if(getThisLocalTime(ret, othis, n) == 0)
694     {
695         n = msFromTime(n);
696         ret.putVtime(n);
697     }
698     return null;
699 }
700 
701 void* Ddate_prototype_getUTCMilliseconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
702 {
703     // ECMA 15.9.5.21
704     d_time n;
705 
706     if(!othis.isDdate())
707         return checkdate(ret, TEXT_getUTCMilliseconds, othis);
708 
709     if(getThisTime(ret, othis, n) == 0)
710     {
711         n = msFromTime(n);
712         ret.putVtime(n);
713     }
714     return null;
715 }
716 
717 void* Ddate_prototype_getTimezoneOffset(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
718 {
719     // ECMA 15.9.5.22
720     d_time n;
721 
722     if(!othis.isDdate())
723         return checkdate(ret, TEXT_getTimezoneOffset, othis);
724 
725     if(getThisTime(ret, othis, n) == 0)
726     {
727         n = (n - UTCtoLocalTime(n)) / (60 * 1000);
728         ret.putVtime(n);
729     }
730     return null;
731 }
732 
733 void* Ddate_prototype_setTime(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
734 {
735     // ECMA 15.9.5.23
736     d_time n;
737 
738     if(!othis.isDdate())
739         return checkdate(ret, TEXT_setTime, othis);
740 
741     if(!arglist.length)
742         n = d_time_nan;
743     else
744         n = arglist[0].toDtime();
745     n = timeClip(n);
746     othis.value.putVtime(n);
747     ret.putVtime(n);
748     return null;
749 }
750 
751 void* Ddate_prototype_setMilliseconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
752 {
753     // ECMA 15.9.5.24
754 
755     d_time ms;
756     d_time t;
757     d_time time;
758     d_time n;
759 
760     if(!othis.isDdate())
761         return checkdate(ret, TEXT_setMilliseconds, othis);
762 
763     if(getThisLocalTime(ret, othis, t) == 0)
764     {
765         if(!arglist.length)
766             ms = d_time_nan;
767         else
768             ms = arglist[0].toDtime();
769         time = makeTime(hourFromTime(t), minFromTime(t), secFromTime(t), ms);
770         n = timeClip(localTimetoUTC(makeDate(day(t), time)));
771         othis.value.putVtime(n);
772         ret.putVtime(n);
773     }
774     return null;
775 }
776 
777 void* Ddate_prototype_setUTCMilliseconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
778 {
779     // ECMA 15.9.5.25
780     d_time ms;
781     d_time t;
782     d_time time;
783     d_time n;
784 
785     if(!othis.isDdate())
786         return checkdate(ret, TEXT_setUTCMilliseconds, othis);
787 
788     if(getThisTime(ret, othis, t) == 0)
789     {
790         if(!arglist.length)
791             ms = d_time_nan;
792         else
793             ms = arglist[0].toDtime();
794         time = makeTime(hourFromTime(t), minFromTime(t), secFromTime(t), ms);
795         n = timeClip(makeDate(day(t), time));
796         othis.value.putVtime(n);
797         ret.putVtime(n);
798     }
799     return null;
800 }
801 
802 void* Ddate_prototype_setSeconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
803 {
804     // ECMA 15.9.5.26
805     d_time ms;
806     d_time seconds;
807     d_time t;
808     d_time time;
809     d_time n;
810 
811     if(!othis.isDdate())
812         return checkdate(ret, TEXT_setSeconds, othis);
813 
814     if(getThisLocalTime(ret, othis, t) == 0)
815     {
816         switch(arglist.length)
817         {
818         default:
819         case 2:
820             ms = arglist[1].toDtime();
821             seconds = arglist[0].toDtime();
822             break;
823 
824         case 1:
825             ms = msFromTime(t);
826             seconds = arglist[0].toDtime();
827             break;
828 
829         case 0:
830             ms = msFromTime(t);
831             seconds = d_time_nan;
832             break;
833         }
834         time = makeTime(hourFromTime(t), minFromTime(t), seconds, ms);
835         n = timeClip(localTimetoUTC(makeDate(day(t), time)));
836         othis.value.putVtime(n);
837         ret.putVtime(n);
838     }
839     return null;
840 }
841 
842 void* Ddate_prototype_setUTCSeconds(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
843 {
844     // ECMA 15.9.5.27
845     d_time ms;
846     d_time seconds;
847     d_time t;
848     d_time time;
849     d_time n;
850 
851     if(!othis.isDdate())
852         return checkdate(ret, TEXT_setUTCSeconds, othis);
853 
854     if(getThisTime(ret, othis, t) == 0)
855     {
856         switch(arglist.length)
857         {
858         default:
859         case 2:
860             ms = arglist[1].toDtime();
861             seconds = arglist[0].toDtime();
862             break;
863 
864         case 1:
865             ms = msFromTime(t);
866             seconds = arglist[0].toDtime();
867             break;
868 
869         case 0:
870             ms = msFromTime(t);
871             seconds = d_time_nan;
872             break;
873         }
874         time = makeTime(hourFromTime(t), minFromTime(t), seconds, ms);
875         n = timeClip(makeDate(day(t), time));
876         othis.value.putVtime(n);
877         ret.putVtime(n);
878     }
879     return null;
880 }
881 
882 void* Ddate_prototype_setMinutes(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
883 {
884     // ECMA 15.9.5.28
885     d_time ms;
886     d_time seconds;
887     d_time minutes;
888     d_time t;
889     d_time time;
890     d_time n;
891 
892     if(!othis.isDdate())
893         return checkdate(ret, TEXT_setMinutes, othis);
894 
895     if(getThisLocalTime(ret, othis, t) == 0)
896     {
897         switch(arglist.length)
898         {
899         default:
900         case 3:
901             ms = arglist[2].toDtime();
902             seconds = arglist[1].toDtime();
903             minutes = arglist[0].toDtime();
904             break;
905 
906         case 2:
907             ms = msFromTime(t);
908             seconds = arglist[1].toDtime();
909             minutes = arglist[0].toDtime();
910             break;
911 
912         case 1:
913             ms = msFromTime(t);
914             seconds = secFromTime(t);
915             minutes = arglist[0].toDtime();
916             break;
917 
918         case 0:
919             ms = msFromTime(t);
920             seconds = secFromTime(t);
921             minutes = d_time_nan;
922             break;
923         }
924         time = makeTime(hourFromTime(t), minutes, seconds, ms);
925         n = timeClip(localTimetoUTC(makeDate(day(t), time)));
926         othis.value.putVtime(n);
927         ret.putVtime(n);
928     }
929     return null;
930 }
931 
932 void* Ddate_prototype_setUTCMinutes(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
933 {
934     // ECMA 15.9.5.29
935     d_time ms;
936     d_time seconds;
937     d_time minutes;
938     d_time t;
939     d_time time;
940     d_time n;
941 
942     if(!othis.isDdate())
943         return checkdate(ret, TEXT_setUTCMinutes, othis);
944 
945     if(getThisTime(ret, othis, t) == 0)
946     {
947         switch(arglist.length)
948         {
949         default:
950         case 3:
951             ms = arglist[2].toDtime();
952             seconds = arglist[1].toDtime();
953             minutes = arglist[0].toDtime();
954             break;
955 
956         case 2:
957             ms = msFromTime(t);
958             seconds = arglist[1].toDtime();
959             minutes = arglist[0].toDtime();
960             break;
961 
962         case 1:
963             ms = msFromTime(t);
964             seconds = secFromTime(t);
965             minutes = arglist[0].toDtime();
966             break;
967 
968         case 0:
969             ms = msFromTime(t);
970             seconds = secFromTime(t);
971             minutes = d_time_nan;
972             break;
973         }
974         time = makeTime(hourFromTime(t), minutes, seconds, ms);
975         n = timeClip(makeDate(day(t), time));
976         othis.value.putVtime(n);
977         ret.putVtime(n);
978     }
979     return null;
980 }
981 
982 void* Ddate_prototype_setHours(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
983 {
984     // ECMA 15.9.5.30
985     d_time ms;
986     d_time seconds;
987     d_time minutes;
988     d_time hours;
989     d_time t;
990     d_time time;
991     d_time n;
992 
993     if(!othis.isDdate())
994         return checkdate(ret, TEXT_setHours, othis);
995 
996     if(getThisLocalTime(ret, othis, t) == 0)
997     {
998         switch(arglist.length)
999         {
1000         default:
1001         case 4:
1002             ms = arglist[3].toDtime();
1003             seconds = arglist[2].toDtime();
1004             minutes = arglist[1].toDtime();
1005             hours = arglist[0].toDtime();
1006             break;
1007 
1008         case 3:
1009             ms = msFromTime(t);
1010             seconds = arglist[2].toDtime();
1011             minutes = arglist[1].toDtime();
1012             hours = arglist[0].toDtime();
1013             break;
1014 
1015         case 2:
1016             ms = msFromTime(t);
1017             seconds = secFromTime(t);
1018             minutes = arglist[1].toDtime();
1019             hours = arglist[0].toDtime();
1020             break;
1021 
1022         case 1:
1023             ms = msFromTime(t);
1024             seconds = secFromTime(t);
1025             minutes = minFromTime(t);
1026             hours = arglist[0].toDtime();
1027             break;
1028 
1029         case 0:
1030             ms = msFromTime(t);
1031             seconds = secFromTime(t);
1032             minutes = minFromTime(t);
1033             hours = d_time_nan;
1034             break;
1035         }
1036         time = makeTime(hours, minutes, seconds, ms);
1037         n = timeClip(localTimetoUTC(makeDate(day(t), time)));
1038         othis.value.putVtime(n);
1039         ret.putVtime(n);
1040     }
1041     return null;
1042 }
1043 
1044 void* Ddate_prototype_setUTCHours(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1045 {
1046     // ECMA 15.9.5.31
1047     d_time ms;
1048     d_time seconds;
1049     d_time minutes;
1050     d_time hours;
1051     d_time t;
1052     d_time time;
1053     d_time n;
1054 
1055     if(!othis.isDdate())
1056         return checkdate(ret, TEXT_setUTCHours, othis);
1057 
1058     if(getThisTime(ret, othis, t) == 0)
1059     {
1060         switch(arglist.length)
1061         {
1062         default:
1063         case 4:
1064             ms = arglist[3].toDtime();
1065             seconds = arglist[2].toDtime();
1066             minutes = arglist[1].toDtime();
1067             hours = arglist[0].toDtime();
1068             break;
1069 
1070         case 3:
1071             ms = msFromTime(t);
1072             seconds = arglist[2].toDtime();
1073             minutes = arglist[1].toDtime();
1074             hours = arglist[0].toDtime();
1075             break;
1076 
1077         case 2:
1078             ms = msFromTime(t);
1079             seconds = secFromTime(t);
1080             minutes = arglist[1].toDtime();
1081             hours = arglist[0].toDtime();
1082             break;
1083 
1084         case 1:
1085             ms = msFromTime(t);
1086             seconds = secFromTime(t);
1087             minutes = minFromTime(t);
1088             hours = arglist[0].toDtime();
1089             break;
1090 
1091         case 0:
1092             ms = msFromTime(t);
1093             seconds = secFromTime(t);
1094             minutes = minFromTime(t);
1095             hours = d_time_nan;
1096             break;
1097         }
1098         time = makeTime(hours, minutes, seconds, ms);
1099         n = timeClip(makeDate(day(t), time));
1100         othis.value.putVtime(n);
1101         ret.putVtime(n);
1102     }
1103     return null;
1104 }
1105 
1106 void* Ddate_prototype_setDate(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1107 {
1108     // ECMA 15.9.5.32
1109     d_time date;
1110     d_time t;
1111     d_time day;
1112     d_time n;
1113 
1114     if(!othis.isDdate())
1115         return checkdate(ret, TEXT_setDate, othis);
1116 
1117     if(getThisLocalTime(ret, othis, t) == 0)
1118     {
1119         if(!arglist.length)
1120             date = d_time_nan;
1121         else
1122             date = arglist[0].toDtime();
1123         day = makeDay(yearFromTime(t), monthFromTime(t), date);
1124         n = timeClip(localTimetoUTC(makeDate(day, timeWithinDay(t))));
1125         othis.value.putVtime(n);
1126         ret.putVtime(n);
1127     }
1128     return null;
1129 }
1130 
1131 void* Ddate_prototype_setUTCDate(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1132 {
1133     // ECMA 15.9.5.33
1134     d_time date;
1135     d_time t;
1136     d_time day;
1137     d_time n;
1138 
1139     if(!othis.isDdate())
1140         return checkdate(ret, TEXT_setUTCDate, othis);
1141 
1142     if(getThisTime(ret, othis, t) == 0)
1143     {
1144         if(!arglist.length)
1145             date = d_time_nan;
1146         else
1147             date = arglist[0].toDtime();
1148         day = makeDay(yearFromTime(t), monthFromTime(t), date);
1149         n = timeClip(makeDate(day, timeWithinDay(t)));
1150         othis.value.putVtime(n);
1151         ret.putVtime(n);
1152     }
1153     return null;
1154 }
1155 
1156 void* Ddate_prototype_setMonth(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1157 {
1158     // ECMA 15.9.5.34
1159     d_time date;
1160     d_time month;
1161     d_time t;
1162     d_time day;
1163     d_time n;
1164 
1165     if(!othis.isDdate())
1166         return checkdate(ret, TEXT_setMonth, othis);
1167 
1168     if(getThisLocalTime(ret, othis, t) == 0)
1169     {
1170         switch(arglist.length)
1171         {
1172         default:
1173         case 2:
1174             month = arglist[0].toDtime();
1175             date = arglist[1].toDtime();
1176             break;
1177 
1178         case 1:
1179             month = arglist[0].toDtime();
1180             date = dateFromTime(t);
1181             break;
1182 
1183         case 0:
1184             month = d_time_nan;
1185             date = dateFromTime(t);
1186             break;
1187         }
1188         day = makeDay(yearFromTime(t), month, date);
1189         n = timeClip(localTimetoUTC(makeDate(day, timeWithinDay(t))));
1190         othis.value.putVtime(n);
1191         ret.putVtime(n);
1192     }
1193     return null;
1194 }
1195 
1196 void* Ddate_prototype_setUTCMonth(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1197 {
1198     // ECMA 15.9.5.35
1199     d_time date;
1200     d_time month;
1201     d_time t;
1202     d_time day;
1203     d_time n;
1204 
1205     if(!othis.isDdate())
1206         return checkdate(ret, TEXT_setUTCMonth, othis);
1207 
1208     if(getThisTime(ret, othis, t) == 0)
1209     {
1210         switch(arglist.length)
1211         {
1212         default:
1213         case 2:
1214             month = arglist[0].toDtime();
1215             date = arglist[1].toDtime();
1216             break;
1217 
1218         case 1:
1219             month = arglist[0].toDtime();
1220             date = dateFromTime(t);
1221             break;
1222 
1223         case 0:
1224             month = d_time_nan;
1225             date = dateFromTime(t);
1226             break;
1227         }
1228         day = makeDay(yearFromTime(t), month, date);
1229         n = timeClip(makeDate(day, timeWithinDay(t)));
1230         othis.value.putVtime(n);
1231         ret.putVtime(n);
1232     }
1233     return null;
1234 }
1235 
1236 void* Ddate_prototype_setFullYear(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1237 {
1238     // ECMA 15.9.5.36
1239     d_time date;
1240     d_time month;
1241     d_time year;
1242     d_time t;
1243     d_time day;
1244     d_time n;
1245 
1246     if(!othis.isDdate())
1247         return checkdate(ret, TEXT_setFullYear, othis);
1248 
1249     if(getThisLocalTime(ret, othis, t))
1250         t = 0;
1251 
1252     switch(arglist.length)
1253     {
1254     default:
1255     case 3:
1256         date = arglist[2].toDtime();
1257         month = arglist[1].toDtime();
1258         year = arglist[0].toDtime();
1259         break;
1260 
1261     case 2:
1262         date = dateFromTime(t);
1263         month = arglist[1].toDtime();
1264         year = arglist[0].toDtime();
1265         break;
1266 
1267     case 1:
1268         date = dateFromTime(t);
1269         month = monthFromTime(t);
1270         year = arglist[0].toDtime();
1271         break;
1272 
1273     case 0:
1274         date = dateFromTime(t);
1275         month = monthFromTime(t);
1276         year = d_time_nan;
1277         break;
1278     }
1279     day = makeDay(year, month, date);
1280     n = timeClip(localTimetoUTC(makeDate(day, timeWithinDay(t))));
1281     othis.value.putVtime(n);
1282     ret.putVtime(n);
1283     return null;
1284 }
1285 
1286 void* Ddate_prototype_setUTCFullYear(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1287 {
1288     // ECMA 15.9.5.37
1289     d_time date;
1290     d_time month;
1291     d_time year;
1292     d_time t;
1293     d_time day;
1294     d_time n;
1295 
1296     if(!othis.isDdate())
1297         return checkdate(ret, TEXT_setUTCFullYear, othis);
1298 
1299     getThisTime(ret, othis, t);
1300     if(t == d_time_nan)
1301         t = 0;
1302     switch(arglist.length)
1303     {
1304     default:
1305     case 3:
1306         month = arglist[2].toDtime();
1307         date = arglist[1].toDtime();
1308         year = arglist[0].toDtime();
1309         break;
1310 
1311     case 2:
1312         month = monthFromTime(t);
1313         date = arglist[1].toDtime();
1314         year = arglist[0].toDtime();
1315         break;
1316 
1317     case 1:
1318         month = monthFromTime(t);
1319         date = dateFromTime(t);
1320         year = arglist[0].toDtime();
1321         break;
1322 
1323     case 0:
1324         month = monthFromTime(t);
1325         date = dateFromTime(t);
1326         year = d_time_nan;
1327         break;
1328     }
1329     day = makeDay(year, month, date);
1330     n = timeClip(makeDate(day, timeWithinDay(t)));
1331     othis.value.putVtime(n);
1332     ret.putVtime(n);
1333     return null;
1334 }
1335 
1336 void* Ddate_prototype_setYear(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1337 {
1338     // ECMA 15.9.5.38
1339     d_time date;
1340     d_time month;
1341     d_time year;
1342     d_time t;
1343     d_time day;
1344     d_time n;
1345 
1346     if(!othis.isDdate())
1347         return checkdate(ret, TEXT_setYear, othis);
1348 
1349     if(getThisLocalTime(ret, othis, t))
1350         t = 0;
1351     switch(arglist.length)
1352     {
1353     default:
1354     case 1:
1355         month = monthFromTime(t);
1356         date = dateFromTime(t);
1357         year = arglist[0].toDtime();
1358         if(0 <= year && year <= 99)
1359             year += 1900;
1360         day = makeDay(year, month, date);
1361         n = timeClip(localTimetoUTC(makeDate(day, timeWithinDay(t))));
1362         break;
1363 
1364     case 0:
1365         n = d_time_nan;
1366         break;
1367     }
1368     othis.value.putVtime(n);
1369     ret.putVtime(n);
1370     return null;
1371 }
1372 
1373 void* Ddate_prototype_toLocaleString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1374 {
1375     // ECMA 15.9.5.39
1376     immutable(char)[] s;
1377     d_time t;
1378 
1379     if(!othis.isDdate())
1380         return checkdate(ret, TEXT_toLocaleString, othis);
1381 
1382     if(getThisLocalTime(ret, othis, t))
1383         t = 0;
1384 
1385     s = dateToString(cc, t, TIMEFORMAT.LocaleString);
1386     ret.putVstring(s);
1387     return null;
1388 }
1389 
1390 void* Ddate_prototype_toLocaleDateString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1391 {
1392     // ECMA 15.9.5.6
1393     immutable(char)[] s;
1394     d_time t;
1395 
1396     if(!othis.isDdate())
1397         return checkdate(ret, TEXT_toLocaleDateString, othis);
1398 
1399     if(getThisLocalTime(ret, othis, t))
1400         t = 0;
1401 
1402     s = dateToString(cc, t, TIMEFORMAT.LocaleDateString);
1403     ret.putVstring(s);
1404     return null;
1405 }
1406 
1407 void* Ddate_prototype_toLocaleTimeString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1408 {
1409     // ECMA 15.9.5.7
1410     immutable(char)[] s;
1411     d_time t;
1412 
1413     if(!othis.isDdate())
1414         return checkdate(ret, TEXT_toLocaleTimeString, othis);
1415 
1416     if(getThisLocalTime(ret, othis, t))
1417         t = 0;
1418     s = dateToString(cc, t, TIMEFORMAT.LocaleTimeString);
1419     ret.putVstring(s);
1420     return null;
1421 }
1422 
1423 void* Ddate_prototype_toUTCString(Dobject pthis, CallContext *cc, Dobject othis, Value* ret, Value[] arglist)
1424 {
1425     // ECMA 15.9.5.40
1426     immutable(char)[] s;
1427     d_time t;
1428 
1429     if(!othis.isDdate())
1430         return checkdate(ret, TEXT_toUTCString, othis);
1431 
1432     if(getThisTime(ret, othis, t))
1433         t = 0;
1434     s = dateToString(cc, t, TIMEFORMAT.UTCString);
1435     ret.putVstring(s);
1436     return null;
1437 }
1438 
1439 /* ===================== Ddate_prototype ==================== */
1440 
1441 class DdatePrototype : Ddate
1442 {
1443     this()
1444     {
1445         super(Dobject_prototype);
1446 
1447         Dobject f = Dfunction_prototype;
1448 
1449         Put(TEXT_constructor, Ddate_constructor, DontEnum);
1450 
1451         static enum NativeFunctionData[] nfd =
1452         [
1453             { TEXT_toString, &Ddate_prototype_toString, 0 },
1454             { TEXT_toDateString, &Ddate_prototype_toDateString, 0 },
1455             { TEXT_toTimeString, &Ddate_prototype_toTimeString, 0 },
1456             { TEXT_valueOf, &Ddate_prototype_valueOf, 0 },
1457             { TEXT_getTime, &Ddate_prototype_getTime, 0 },
1458             //{	TEXT_getVarDate, &Ddate_prototype_getVarDate, 0 },
1459             { TEXT_getYear, &Ddate_prototype_getYear, 0 },
1460             { TEXT_getFullYear, &Ddate_prototype_getFullYear, 0 },
1461             { TEXT_getUTCFullYear, &Ddate_prototype_getUTCFullYear, 0 },
1462             { TEXT_getMonth, &Ddate_prototype_getMonth, 0 },
1463             { TEXT_getUTCMonth, &Ddate_prototype_getUTCMonth, 0 },
1464             { TEXT_getDate, &Ddate_prototype_getDate, 0 },
1465             { TEXT_getUTCDate, &Ddate_prototype_getUTCDate, 0 },
1466             { TEXT_getDay, &Ddate_prototype_getDay, 0 },
1467             { TEXT_getUTCDay, &Ddate_prototype_getUTCDay, 0 },
1468             { TEXT_getHours, &Ddate_prototype_getHours, 0 },
1469             { TEXT_getUTCHours, &Ddate_prototype_getUTCHours, 0 },
1470             { TEXT_getMinutes, &Ddate_prototype_getMinutes, 0 },
1471             { TEXT_getUTCMinutes, &Ddate_prototype_getUTCMinutes, 0 },
1472             { TEXT_getSeconds, &Ddate_prototype_getSeconds, 0 },
1473             { TEXT_getUTCSeconds, &Ddate_prototype_getUTCSeconds, 0 },
1474             { TEXT_getMilliseconds, &Ddate_prototype_getMilliseconds, 0 },
1475             { TEXT_getUTCMilliseconds, &Ddate_prototype_getUTCMilliseconds, 0 },
1476             { TEXT_getTimezoneOffset, &Ddate_prototype_getTimezoneOffset, 0 },
1477             { TEXT_setTime, &Ddate_prototype_setTime, 1 },
1478             { TEXT_setMilliseconds, &Ddate_prototype_setMilliseconds, 1 },
1479             { TEXT_setUTCMilliseconds, &Ddate_prototype_setUTCMilliseconds, 1 },
1480             { TEXT_setSeconds, &Ddate_prototype_setSeconds, 2 },
1481             { TEXT_setUTCSeconds, &Ddate_prototype_setUTCSeconds, 2 },
1482             { TEXT_setMinutes, &Ddate_prototype_setMinutes, 3 },
1483             { TEXT_setUTCMinutes, &Ddate_prototype_setUTCMinutes, 3 },
1484             { TEXT_setHours, &Ddate_prototype_setHours, 4 },
1485             { TEXT_setUTCHours, &Ddate_prototype_setUTCHours, 4 },
1486             { TEXT_setDate, &Ddate_prototype_setDate, 1 },
1487             { TEXT_setUTCDate, &Ddate_prototype_setUTCDate, 1 },
1488             { TEXT_setMonth, &Ddate_prototype_setMonth, 2 },
1489             { TEXT_setUTCMonth, &Ddate_prototype_setUTCMonth, 2 },
1490             { TEXT_setFullYear, &Ddate_prototype_setFullYear, 3 },
1491             { TEXT_setUTCFullYear, &Ddate_prototype_setUTCFullYear, 3 },
1492             { TEXT_setYear, &Ddate_prototype_setYear, 1 },
1493             { TEXT_toLocaleString, &Ddate_prototype_toLocaleString, 0 },
1494             { TEXT_toLocaleDateString, &Ddate_prototype_toLocaleDateString, 0 },
1495             { TEXT_toLocaleTimeString, &Ddate_prototype_toLocaleTimeString, 0 },
1496             { TEXT_toUTCString, &Ddate_prototype_toUTCString, 0 },
1497 
1498             // Map toGMTString() onto toUTCString(), per ECMA 15.9.5.41
1499             { TEXT_toGMTString, &Ddate_prototype_toUTCString, 0 },
1500         ];
1501 
1502         DnativeFunction.initialize(this, nfd, DontEnum);
1503         assert(proptable.get("toString", Value.calcHash("toString")));
1504     }
1505 }
1506 
1507 
1508 /* ===================== Ddate ==================== */
1509 
1510 class Ddate : Dobject
1511 {
1512     this(d_number n)
1513     {
1514         super(Ddate.getPrototype());
1515         classname = TEXT_Date;
1516         value.putVnumber(n);
1517     }
1518 
1519     this(d_time n)
1520     {
1521         super(Ddate.getPrototype());
1522         classname = TEXT_Date;
1523         value.putVtime(n);
1524     }
1525 
1526     this(Dobject prototype)
1527     {
1528         super(prototype);
1529         classname = TEXT_Date;
1530         value.putVnumber(d_number.nan);
1531     }
1532 
1533     static void initialize()
1534     {
1535         Ddate_constructor = new DdateConstructor();
1536         Ddate_prototype = new DdatePrototype();
1537 
1538         Ddate_constructor.Put(TEXT_prototype, Ddate_prototype,
1539                                  DontEnum | DontDelete | ReadOnly);
1540 
1541         assert(Ddate_prototype.proptable.table.length != 0);
1542     }
1543 
1544     static Dfunction getConstructor()
1545     {
1546         return Ddate_constructor;
1547     }
1548 
1549     static Dobject getPrototype()
1550     {
1551         return Ddate_prototype;
1552     }
1553 }
1554 
1555