Merge remote-tracking branch 'pootle/master'
[phpmyadmin/tyronm.git] / js / date.js
blob9a923408ad3ea884b0e94f36a5be8e7833fa0f4d
1 // ===================================================================
2 // Author: Matt Kruse <matt@mattkruse.com>
3 // WWW: http://www.mattkruse.com/
4 //
5 // NOTICE: You may use this code for any purpose, commercial or
6 // private, without any further permission from the author. You may
7 // remove this notice from your final code if you wish, however it is
8 // appreciated by the author if at least my web site address is kept.
9 //
10 // You may *NOT* re-distribute this code in any way except through its
11 // use. That means, you can include it in your product, or your web
12 // site, or any other form where the code is actually being used. You
13 // may not put the plain javascript up on your site for download or
14 // include it in your javascript libraries for download. 
15 // If you wish to share this code with others, please just point them
16 // to the URL instead.
17 // Please DO NOT link directly to my .js files from your site. Copy
18 // the files to your server and use them there. Thank you.
19 // ===================================================================
21 // HISTORY
22 // ------------------------------------------------------------------
23 // May 17, 2003: Fixed bug in parseDate() for dates <1970
24 // March 11, 2003: Added parseDate() function
25 // March 11, 2003: Added "NNN" formatting option. Doesn't match up
26 //                 perfectly with SimpleDateFormat formats, but 
27 //                 backwards-compatability was required.
29 // ------------------------------------------------------------------
30 // These functions use the same 'format' strings as the 
31 // java.text.SimpleDateFormat class, with minor exceptions.
32 // The format string consists of the following abbreviations:
33 // 
34 // Field        | Full Form          | Short Form
35 // -------------+--------------------+-----------------------
36 // Year         | yyyy (4 digits)    | yy (2 digits), y (2 or 4 digits)
37 // Month        | MMM (name or abbr.)| MM (2 digits), M (1 or 2 digits)
38 //              | NNN (abbr.)        |
39 // Day of Month | dd (2 digits)      | d (1 or 2 digits)
40 // Day of Week  | EE (name)          | E (abbr)
41 // Hour (1-12)  | hh (2 digits)      | h (1 or 2 digits)
42 // Hour (0-23)  | HH (2 digits)      | H (1 or 2 digits)
43 // Hour (0-11)  | KK (2 digits)      | K (1 or 2 digits)
44 // Hour (1-24)  | kk (2 digits)      | k (1 or 2 digits)
45 // Minute       | mm (2 digits)      | m (1 or 2 digits)
46 // Second       | ss (2 digits)      | s (1 or 2 digits)
47 // AM/PM        | a                  |
49 // NOTE THE DIFFERENCE BETWEEN MM and mm! Month=MM, not mm!
50 // Examples:
51 //  "MMM d, y" matches: January 01, 2000
52 //                      Dec 1, 1900
53 //                      Nov 20, 00
54 //  "M/d/yy"   matches: 01/20/00
55 //                      9/2/00
56 //  "MMM dd, yyyy hh:mm:ssa" matches: "January 01, 2000 12:30:45AM"
57 // ------------------------------------------------------------------
59 var MONTH_NAMES=new Array('January','February','March','April','May','June','July','August','September','October','November','December','Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec');
60 var DAY_NAMES=new Array('Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday','Sun','Mon','Tue','Wed','Thu','Fri','Sat');
61 function LZ(x) {return(x<0||x>9?"":"0")+x}
63 // ------------------------------------------------------------------
64 // isDate ( date_string, format_string )
65 // Returns true if date string matches format of format string and
66 // is a valid date. Else returns false.
67 // It is recommended that you trim whitespace around the value before
68 // passing it to this function, as whitespace is NOT ignored!
69 // ------------------------------------------------------------------
70 function isDate(val,format) {
71         var date=getDateFromFormat(val,format);
72         if (date==0) { return false; }
73         return true;
74         }
76 // -------------------------------------------------------------------
77 // compareDates(date1,date1format,date2,date2format)
78 //   Compare two date strings to see which is greater.
79 //   Returns:
80 //   1 if date1 is greater than date2
81 //   0 if date2 is greater than date1 of if they are the same
82 //  -1 if either of the dates is in an invalid format
83 // -------------------------------------------------------------------
84 function compareDates(date1,dateformat1,date2,dateformat2) {
85         var d1=getDateFromFormat(date1,dateformat1);
86         var d2=getDateFromFormat(date2,dateformat2);
87         if (d1==0 || d2==0) {
88                 return -1;
89                 }
90         else if (d1 > d2) {
91                 return 1;
92                 }
93         return 0;
94         }
96 // ------------------------------------------------------------------
97 // formatDate (date_object, format)
98 // Returns a date in the output format specified.
99 // The format string uses the same abbreviations as in getDateFromFormat()
100 // ------------------------------------------------------------------
101 function formatDate(date,format) {
102         format=format+"";
103         var result="";
104         var i_format=0;
105         var c="";
106         var token="";
107         var y=date.getYear()+"";
108         var M=date.getMonth()+1;
109         var d=date.getDate();
110         var E=date.getDay();
111         var H=date.getHours();
112         var m=date.getMinutes();
113         var s=date.getSeconds();
114         var yyyy,yy,MMM,MM,dd,hh,h,mm,ss,ampm,HH,H,KK,K,kk,k;
115         // Convert real date parts into formatted versions
116         var value=new Object();
117         if (y.length < 4) {y=""+(y-0+1900);}
118         value["y"]=""+y;
119         value["yyyy"]=y;
120         value["yy"]=y.substring(2,4);
121         value["M"]=M;
122         value["MM"]=LZ(M);
123         value["MMM"]=MONTH_NAMES[M-1];
124         value["NNN"]=MONTH_NAMES[M+11];
125         value["d"]=d;
126         value["dd"]=LZ(d);
127         value["E"]=DAY_NAMES[E+7];
128         value["EE"]=DAY_NAMES[E];
129         value["H"]=H;
130         value["HH"]=LZ(H);
131         if (H==0){value["h"]=12;}
132         else if (H>12){value["h"]=H-12;}
133         else {value["h"]=H;}
134         value["hh"]=LZ(value["h"]);
135         if (H>11){value["K"]=H-12;} else {value["K"]=H;}
136         value["k"]=H+1;
137         value["KK"]=LZ(value["K"]);
138         value["kk"]=LZ(value["k"]);
139         if (H > 11) { value["a"]="PM"; }
140         else { value["a"]="AM"; }
141         value["m"]=m;
142         value["mm"]=LZ(m);
143         value["s"]=s;
144         value["ss"]=LZ(s);
145         while (i_format < format.length) {
146                 c=format.charAt(i_format);
147                 token="";
148                 while ((format.charAt(i_format)==c) && (i_format < format.length)) {
149                         token += format.charAt(i_format++);
150                         }
151                 if (value[token] != null) { result=result + value[token]; }
152                 else { result=result + token; }
153                 }
154         return result;
155         }
156         
157 // ------------------------------------------------------------------
158 // Utility functions for parsing in getDateFromFormat()
159 // ------------------------------------------------------------------
160 function _isInteger(val) {
161         var digits="1234567890";
162         for (var i=0; i < val.length; i++) {
163                 if (digits.indexOf(val.charAt(i))==-1) { return false; }
164                 }
165         return true;
166         }
167 function _getInt(str,i,minlength,maxlength) {
168         for (var x=maxlength; x>=minlength; x--) {
169                 var token=str.substring(i,i+x);
170                 if (token.length < minlength) { return null; }
171                 if (_isInteger(token)) { return token; }
172                 }
173         return null;
174         }
175         
176 // ------------------------------------------------------------------
177 // getDateFromFormat( date_string , format_string )
179 // This function takes a date string and a format string. It matches
180 // If the date string matches the format string, it returns the 
181 // getTime() of the date. If it does not match, it returns 0.
182 // ------------------------------------------------------------------
183 function getDateFromFormat(val,format) {
184         val=val+"";
185         format=format+"";
186         var i_val=0;
187         var i_format=0;
188         var c="";
189         var token="";
190         var token2="";
191         var x,y;
192         var now=new Date();
193         var year=now.getYear();
194         var month=now.getMonth()+1;
195         var date=1;
196         var hh=now.getHours();
197         var mm=now.getMinutes();
198         var ss=now.getSeconds();
199         var ampm="";
200         
201         while (i_format < format.length) {
202                 // Get next token from format string
203                 c=format.charAt(i_format);
204                 token="";
205                 while ((format.charAt(i_format)==c) && (i_format < format.length)) {
206                         token += format.charAt(i_format++);
207                         }
208                 // Extract contents of value based on format token
209                 if (token=="yyyy" || token=="yy" || token=="y") {
210                         if (token=="yyyy") { x=4;y=4; }
211                         if (token=="yy")   { x=2;y=2; }
212                         if (token=="y")    { x=2;y=4; }
213                         year=_getInt(val,i_val,x,y);
214                         if (year==null) { return 0; }
215                         i_val += year.length;
216                         if (year.length==2) {
217                                 if (year > 70) { year=1900+(year-0); }
218                                 else { year=2000+(year-0); }
219                                 }
220                         }
221                 else if (token=="MMM"||token=="NNN"){
222                         month=0;
223                         for (var i=0; i<MONTH_NAMES.length; i++) {
224                                 var month_name=MONTH_NAMES[i];
225                                 if (val.substring(i_val,i_val+month_name.length).toLowerCase()==month_name.toLowerCase()) {
226                                         if (token=="MMM"||(token=="NNN"&&i>11)) {
227                                                 month=i+1;
228                                                 if (month>12) { month -= 12; }
229                                                 i_val += month_name.length;
230                                                 break;
231                                                 }
232                                         }
233                                 }
234                         if ((month < 1)||(month>12)){return 0;}
235                         }
236                 else if (token=="EE"||token=="E"){
237                         for (var i=0; i<DAY_NAMES.length; i++) {
238                                 var day_name=DAY_NAMES[i];
239                                 if (val.substring(i_val,i_val+day_name.length).toLowerCase()==day_name.toLowerCase()) {
240                                         i_val += day_name.length;
241                                         break;
242                                         }
243                                 }
244                         }
245                 else if (token=="MM"||token=="M") {
246                         month=_getInt(val,i_val,token.length,2);
247                         if(month==null||(month<1)||(month>12)){return 0;}
248                         i_val+=month.length;}
249                 else if (token=="dd"||token=="d") {
250                         date=_getInt(val,i_val,token.length,2);
251                         if(date==null||(date<1)||(date>31)){return 0;}
252                         i_val+=date.length;}
253                 else if (token=="hh"||token=="h") {
254                         hh=_getInt(val,i_val,token.length,2);
255                         if(hh==null||(hh<1)||(hh>12)){return 0;}
256                         i_val+=hh.length;}
257                 else if (token=="HH"||token=="H") {
258                         hh=_getInt(val,i_val,token.length,2);
259                         if(hh==null||(hh<0)||(hh>23)){return 0;}
260                         i_val+=hh.length;}
261                 else if (token=="KK"||token=="K") {
262                         hh=_getInt(val,i_val,token.length,2);
263                         if(hh==null||(hh<0)||(hh>11)){return 0;}
264                         i_val+=hh.length;}
265                 else if (token=="kk"||token=="k") {
266                         hh=_getInt(val,i_val,token.length,2);
267                         if(hh==null||(hh<1)||(hh>24)){return 0;}
268                         i_val+=hh.length;hh--;}
269                 else if (token=="mm"||token=="m") {
270                         mm=_getInt(val,i_val,token.length,2);
271                         if(mm==null||(mm<0)||(mm>59)){return 0;}
272                         i_val+=mm.length;}
273                 else if (token=="ss"||token=="s") {
274                         ss=_getInt(val,i_val,token.length,2);
275                         if(ss==null||(ss<0)||(ss>59)){return 0;}
276                         i_val+=ss.length;}
277                 else if (token=="a") {
278                         if (val.substring(i_val,i_val+2).toLowerCase()=="am") {ampm="AM";}
279                         else if (val.substring(i_val,i_val+2).toLowerCase()=="pm") {ampm="PM";}
280                         else {return 0;}
281                         i_val+=2;}
282                 else {
283                         if (val.substring(i_val,i_val+token.length)!=token) {return 0;}
284                         else {i_val+=token.length;}
285                         }
286                 }
287         // If there are any trailing characters left in the value, it doesn't match
288         if (i_val != val.length) { return 0; }
289         // Is date valid for month?
290         if (month==2) {
291                 // Check for leap year
292                 if ( ( (year%4==0)&&(year%100 != 0) ) || (year%400==0) ) { // leap year
293                         if (date > 29){ return 0; }
294                         }
295                 else { if (date > 28) { return 0; } }
296                 }
297         if ((month==4)||(month==6)||(month==9)||(month==11)) {
298                 if (date > 30) { return 0; }
299                 }
300         // Correct hours value
301         if (hh<12 && ampm=="PM") { hh=hh-0+12; }
302         else if (hh>11 && ampm=="AM") { hh-=12; }
303         var newdate=new Date(year,month-1,date,hh,mm,ss);
304         return newdate.getTime();
305         }
307 // ------------------------------------------------------------------
308 // parseDate( date_string [, prefer_euro_format] )
310 // This function takes a date string and tries to match it to a
311 // number of possible date formats to get the value. It will try to
312 // match against the following international formats, in this order:
313 // y-M-d   MMM d, y   MMM d,y   y-MMM-d   d-MMM-y  MMM d
314 // M/d/y   M-d-y      M.d.y     MMM-d     M/d      M-d
315 // d/M/y   d-M-y      d.M.y     d-MMM     d/M      d-M
316 // A second argument may be passed to instruct the method to search
317 // for formats like d/M/y (european format) before M/d/y (American).
318 // Returns a Date object or null if no patterns match.
319 // ------------------------------------------------------------------
320 function parseDate(val) {
321         var preferEuro=(arguments.length==2)?arguments[1]:false;
322         generalFormats=new Array('y-M-d','MMM d, y','MMM d,y','y-MMM-d','d-MMM-y','MMM d');
323         monthFirst=new Array('M/d/y','M-d-y','M.d.y','MMM-d','M/d','M-d');
324         dateFirst =new Array('d/M/y','d-M-y','d.M.y','d-MMM','d/M','d-M');
325         var checkList=new Array('generalFormats',preferEuro?'dateFirst':'monthFirst',preferEuro?'monthFirst':'dateFirst');
326         var d=null;
327         for (var i=0; i<checkList.length; i++) {
328                 var l=window[checkList[i]];
329                 for (var j=0; j<l.length; j++) {
330                         d=getDateFromFormat(val,l[j]);
331                         if (d!=0) { return new Date(d); }
332                         }
333                 }
334         return null;
335         }