calendar.js
上传用户:q2283699q
上传日期:2022-05-17
资源大小:10704k
文件大小:50k
源码类别:

Ftp客户端

开发平台:

Visual C++

  1. /*  Copyright Mihai Bazon, 2002-2005  |  www.bazon.net/mishoo
  2.  * -----------------------------------------------------------
  3.  *
  4.  * The DHTML Calendar, version 1.0 "It is happening again"
  5.  *
  6.  * Details and latest version at:
  7.  * www.dynarch.com/projects/calendar
  8.  *
  9.  * This script is developed by Dynarch.com.  Visit us at www.dynarch.com.
  10.  *
  11.  * This script is distributed under the GNU Lesser General Public License.
  12.  * Read the entire license text here: http://www.gnu.org/licenses/lgpl.html
  13.  */
  14. // $Id: calendar.js,v 1.51 2005/03/07 16:44:31 mishoo Exp $
  15. /** The Calendar object constructor. */
  16. Calendar = function (firstDayOfWeek, dateStr, onSelected, onClose) {
  17. // member variables
  18. this.activeDiv = null;
  19. this.currentDateEl = null;
  20. this.getDateStatus = null;
  21. this.getDateToolTip = null;
  22. this.getDateText = null;
  23. this.timeout = null;
  24. this.onSelected = onSelected || null;
  25. this.onClose = onClose || null;
  26. this.dragging = false;
  27. this.hidden = false;
  28. this.minYear = 1970;
  29. this.maxYear = 2050;
  30. this.dateFormat = Calendar._TT["DEF_DATE_FORMAT"];
  31. this.ttDateFormat = Calendar._TT["TT_DATE_FORMAT"];
  32. this.isPopup = true;
  33. this.weekNumbers = true;
  34. this.firstDayOfWeek = typeof firstDayOfWeek == "number" ? firstDayOfWeek : Calendar._FD; // 0 for Sunday, 1 for Monday, etc.
  35. this.showsOtherMonths = false;
  36. this.dateStr = dateStr;
  37. this.ar_days = null;
  38. this.showsTime = false;
  39. this.time24 = true;
  40. this.yearStep = 2;
  41. this.hiliteToday = true;
  42. this.multiple = null;
  43. // HTML elements
  44. this.table = null;
  45. this.element = null;
  46. this.tbody = null;
  47. this.firstdayname = null;
  48. // Combo boxes
  49. this.monthsCombo = null;
  50. this.yearsCombo = null;
  51. this.hilitedMonth = null;
  52. this.activeMonth = null;
  53. this.hilitedYear = null;
  54. this.activeYear = null;
  55. // Information
  56. this.dateClicked = false;
  57. // one-time initializations
  58. if (typeof Calendar._SDN == "undefined") {
  59. // table of short day names
  60. if (typeof Calendar._SDN_len == "undefined")
  61. Calendar._SDN_len = 3;
  62. var ar = new Array();
  63. for (var i = 8; i > 0;) {
  64. ar[--i] = Calendar._DN[i].substr(0, Calendar._SDN_len);
  65. }
  66. Calendar._SDN = ar;
  67. // table of short month names
  68. if (typeof Calendar._SMN_len == "undefined")
  69. Calendar._SMN_len = 3;
  70. ar = new Array();
  71. for (var i = 12; i > 0;) {
  72. ar[--i] = Calendar._MN[i].substr(0, Calendar._SMN_len);
  73. }
  74. Calendar._SMN = ar;
  75. }
  76. };
  77. // ** constants
  78. /// "static", needed for event handlers.
  79. Calendar._C = null;
  80. /// detect a special case of "web browser"
  81. Calendar.is_ie = ( /msie/i.test(navigator.userAgent) &&
  82.    !/opera/i.test(navigator.userAgent) );
  83. Calendar.is_ie5 = ( Calendar.is_ie && /msie 5.0/i.test(navigator.userAgent) );
  84. /// detect Opera browser
  85. Calendar.is_opera = /opera/i.test(navigator.userAgent);
  86. /// detect KHTML-based browsers
  87. Calendar.is_khtml = /Konqueror|Safari|KHTML/i.test(navigator.userAgent);
  88. // BEGIN: UTILITY FUNCTIONS; beware that these might be moved into a separate
  89. //        library, at some point.
  90. Calendar.getAbsolutePos = function(el) {
  91. var SL = 0, ST = 0;
  92. var is_div = /^div$/i.test(el.tagName);
  93. if (is_div && el.scrollLeft)
  94. SL = el.scrollLeft;
  95. if (is_div && el.scrollTop)
  96. ST = el.scrollTop;
  97. var r = { x: el.offsetLeft - SL, y: el.offsetTop - ST };
  98. if (el.offsetParent) {
  99. var tmp = this.getAbsolutePos(el.offsetParent);
  100. r.x += tmp.x;
  101. r.y += tmp.y;
  102. }
  103. return r;
  104. };
  105. Calendar.isRelated = function (el, evt) {
  106. // related.parentNode causes an error in FF when loaded with the page asynchronously
  107. return (false);
  108. /*
  109. var related = evt.relatedTarget;
  110. if (!related) {
  111. var type = evt.type;
  112. if (type == "mouseover") {
  113. related = evt.fromElement;
  114. } else if (type == "mouseout") {
  115. related = evt.toElement;
  116. }
  117. }
  118. while (related) {
  119. if (related == el) {
  120. return true;
  121. }
  122. related = related.parentNode;
  123. }
  124. return false;
  125. //*/
  126. };
  127. Calendar.removeClass = function(el, className) {
  128. if (!(el && el.className)) {
  129. return;
  130. }
  131. var cls = el.className.split(" ");
  132. var ar = new Array();
  133. for (var i = cls.length; i > 0;) {
  134. if (cls[--i] != className) {
  135. ar[ar.length] = cls[i];
  136. }
  137. }
  138. el.className = ar.join(" ");
  139. };
  140. Calendar.addClass = function(el, className) {
  141. Calendar.removeClass(el, className);
  142. el.className += " " + className;
  143. };
  144. // FIXME: the following 2 functions totally suck, are useless and should be replaced immediately.
  145. Calendar.getElement = function(ev) {
  146. var f = Calendar.is_ie ? window.event.srcElement : ev.currentTarget;
  147. while (f.nodeType != 1 || /^div$/i.test(f.tagName))
  148. f = f.parentNode;
  149. return f;
  150. };
  151. Calendar.getTargetElement = function(ev) {
  152. var f = Calendar.is_ie ? window.event.srcElement : ev.target;
  153. while (f.nodeType != 1)
  154. f = f.parentNode;
  155. return f;
  156. };
  157. Calendar.stopEvent = function(ev) {
  158. ev || (ev = window.event);
  159. if (Calendar.is_ie) {
  160. ev.cancelBubble = true;
  161. ev.returnValue = false;
  162. } else {
  163. ev.preventDefault();
  164. ev.stopPropagation();
  165. }
  166. return false;
  167. };
  168. Calendar.addEvent = function(el, evname, func) {
  169. if (el.attachEvent) { // IE
  170. el.attachEvent("on" + evname, func);
  171. } else if (el.addEventListener) { // Gecko / W3C
  172. el.addEventListener(evname, func, true);
  173. } else {
  174. el["on" + evname] = func;
  175. }
  176. };
  177. Calendar.removeEvent = function(el, evname, func) {
  178. if (el.detachEvent) { // IE
  179. el.detachEvent("on" + evname, func);
  180. } else if (el.removeEventListener) { // Gecko / W3C
  181. el.removeEventListener(evname, func, true);
  182. } else {
  183. el["on" + evname] = null;
  184. }
  185. };
  186. Calendar.createElement = function(type, parent) {
  187. var el = null;
  188. if (document.createElementNS) {
  189. // use the XHTML namespace; IE won't normally get here unless
  190. // _they_ "fix" the DOM2 implementation.
  191. el = document.createElementNS("http://www.w3.org/1999/xhtml", type);
  192. } else {
  193. el = document.createElement(type);
  194. }
  195. if (typeof parent != "undefined") {
  196. parent.appendChild(el);
  197. }
  198. return el;
  199. };
  200. // END: UTILITY FUNCTIONS
  201. // BEGIN: CALENDAR STATIC FUNCTIONS
  202. /** Internal -- adds a set of events to make some element behave like a button. */
  203. Calendar._add_evs = function(el) {
  204. with (Calendar) {
  205. addEvent(el, "mouseover", dayMouseOver);
  206. addEvent(el, "mousedown", dayMouseDown);
  207. addEvent(el, "mouseout", dayMouseOut);
  208. if (is_ie) {
  209. addEvent(el, "dblclick", dayMouseDblClick);
  210. el.setAttribute("unselectable", true);
  211. }
  212. }
  213. };
  214. Calendar.findMonth = function(el) {
  215. if (typeof el.month != "undefined") {
  216. return el;
  217. } else if (typeof el.parentNode.month != "undefined") {
  218. return el.parentNode;
  219. }
  220. return null;
  221. };
  222. Calendar.findYear = function(el) {
  223. if (typeof el.year != "undefined") {
  224. return el;
  225. } else if (typeof el.parentNode.year != "undefined") {
  226. return el.parentNode;
  227. }
  228. return null;
  229. };
  230. Calendar.showMonthsCombo = function () {
  231. var cal = Calendar._C;
  232. if (!cal) {
  233. return false;
  234. }
  235. var cal = cal;
  236. var cd = cal.activeDiv;
  237. var mc = cal.monthsCombo;
  238. if (cal.hilitedMonth) {
  239. Calendar.removeClass(cal.hilitedMonth, "hilite");
  240. }
  241. if (cal.activeMonth) {
  242. Calendar.removeClass(cal.activeMonth, "active");
  243. }
  244. var mon = cal.monthsCombo.getElementsByTagName("div")[cal.date.getMonth()];
  245. Calendar.addClass(mon, "active");
  246. cal.activeMonth = mon;
  247. var s = mc.style;
  248. s.display = "block";
  249. if (cd.navtype < 0)
  250. s.left = cd.offsetLeft + "px";
  251. else {
  252. var mcw = mc.offsetWidth;
  253. if (typeof mcw == "undefined")
  254. // Konqueror brain-dead techniques
  255. mcw = 50;
  256. s.left = (cd.offsetLeft + cd.offsetWidth - mcw) + "px";
  257. }
  258. s.top = (cd.offsetTop + cd.offsetHeight) + "px";
  259. };
  260. Calendar.showYearsCombo = function (fwd) {
  261. var cal = Calendar._C;
  262. if (!cal) {
  263. return false;
  264. }
  265. var cal = cal;
  266. var cd = cal.activeDiv;
  267. var yc = cal.yearsCombo;
  268. if (cal.hilitedYear) {
  269. Calendar.removeClass(cal.hilitedYear, "hilite");
  270. }
  271. if (cal.activeYear) {
  272. Calendar.removeClass(cal.activeYear, "active");
  273. }
  274. cal.activeYear = null;
  275. var Y = cal.date.getFullYear() + (fwd ? 1 : -1);
  276. var yr = yc.firstChild;
  277. var show = false;
  278. for (var i = 12; i > 0; --i) {
  279. if (Y >= cal.minYear && Y <= cal.maxYear) {
  280. yr.innerHTML = Y;
  281. yr.year = Y;
  282. yr.style.display = "block";
  283. show = true;
  284. } else {
  285. yr.style.display = "none";
  286. }
  287. yr = yr.nextSibling;
  288. Y += fwd ? cal.yearStep : -cal.yearStep;
  289. }
  290. if (show) {
  291. var s = yc.style;
  292. s.display = "block";
  293. if (cd.navtype < 0)
  294. s.left = cd.offsetLeft + "px";
  295. else {
  296. var ycw = yc.offsetWidth;
  297. if (typeof ycw == "undefined")
  298. // Konqueror brain-dead techniques
  299. ycw = 50;
  300. s.left = (cd.offsetLeft + cd.offsetWidth - ycw) + "px";
  301. }
  302. s.top = (cd.offsetTop + cd.offsetHeight) + "px";
  303. }
  304. };
  305. // event handlers
  306. Calendar.tableMouseUp = function(ev) {
  307. var cal = Calendar._C;
  308. if (!cal) {
  309. return false;
  310. }
  311. if (cal.timeout) {
  312. clearTimeout(cal.timeout);
  313. }
  314. var el = cal.activeDiv;
  315. if (!el) {
  316. return false;
  317. }
  318. var target = Calendar.getTargetElement(ev);
  319. ev || (ev = window.event);
  320. Calendar.removeClass(el, "active");
  321. if (target == el || target.parentNode == el) {
  322. Calendar.cellClick(el, ev);
  323. }
  324. var mon = Calendar.findMonth(target);
  325. var date = null;
  326. if (mon) {
  327. date = new Date(cal.date);
  328. if (mon.month != date.getMonth()) {
  329. date.setMonth(mon.month);
  330. cal.setDate(date);
  331. cal.dateClicked = false;
  332. cal.callHandler();
  333. }
  334. } else {
  335. var year = Calendar.findYear(target);
  336. if (year) {
  337. date = new Date(cal.date);
  338. if (year.year != date.getFullYear()) {
  339. date.setFullYear(year.year);
  340. cal.setDate(date);
  341. cal.dateClicked = false;
  342. cal.callHandler();
  343. }
  344. }
  345. }
  346. with (Calendar) {
  347. removeEvent(document, "mouseup", tableMouseUp);
  348. removeEvent(document, "mouseover", tableMouseOver);
  349. removeEvent(document, "mousemove", tableMouseOver);
  350. cal._hideCombos();
  351. _C = null;
  352. return stopEvent(ev);
  353. }
  354. };
  355. Calendar.tableMouseOver = function (ev) {
  356. var cal = Calendar._C;
  357. if (!cal) {
  358. return;
  359. }
  360. var el = cal.activeDiv;
  361. var target = Calendar.getTargetElement(ev);
  362. if (target == el || target.parentNode == el) {
  363. Calendar.addClass(el, "hilite active");
  364. Calendar.addClass(el.parentNode, "rowhilite");
  365. } else {
  366. if (typeof el.navtype == "undefined" || (el.navtype != 50 && (el.navtype == 0 || Math.abs(el.navtype) > 2)))
  367. Calendar.removeClass(el, "active");
  368. Calendar.removeClass(el, "hilite");
  369. Calendar.removeClass(el.parentNode, "rowhilite");
  370. }
  371. ev || (ev = window.event);
  372. if (el.navtype == 50 && target != el) {
  373. var pos = Calendar.getAbsolutePos(el);
  374. var w = el.offsetWidth;
  375. var x = ev.clientX;
  376. var dx;
  377. var decrease = true;
  378. if (x > pos.x + w) {
  379. dx = x - pos.x - w;
  380. decrease = false;
  381. } else
  382. dx = pos.x - x;
  383. if (dx < 0) dx = 0;
  384. var range = el._range;
  385. var current = el._current;
  386. var count = Math.floor(dx / 10) % range.length;
  387. for (var i = range.length; --i >= 0;)
  388. if (range[i] == current)
  389. break;
  390. while (count-- > 0)
  391. if (decrease) {
  392. if (--i < 0)
  393. i = range.length - 1;
  394. } else if ( ++i >= range.length )
  395. i = 0;
  396. var newval = range[i];
  397. el.innerHTML = newval;
  398. cal.onUpdateTime();
  399. }
  400. var mon = Calendar.findMonth(target);
  401. if (mon) {
  402. if (mon.month != cal.date.getMonth()) {
  403. if (cal.hilitedMonth) {
  404. Calendar.removeClass(cal.hilitedMonth, "hilite");
  405. }
  406. Calendar.addClass(mon, "hilite");
  407. cal.hilitedMonth = mon;
  408. } else if (cal.hilitedMonth) {
  409. Calendar.removeClass(cal.hilitedMonth, "hilite");
  410. }
  411. } else {
  412. if (cal.hilitedMonth) {
  413. Calendar.removeClass(cal.hilitedMonth, "hilite");
  414. }
  415. var year = Calendar.findYear(target);
  416. if (year) {
  417. if (year.year != cal.date.getFullYear()) {
  418. if (cal.hilitedYear) {
  419. Calendar.removeClass(cal.hilitedYear, "hilite");
  420. }
  421. Calendar.addClass(year, "hilite");
  422. cal.hilitedYear = year;
  423. } else if (cal.hilitedYear) {
  424. Calendar.removeClass(cal.hilitedYear, "hilite");
  425. }
  426. } else if (cal.hilitedYear) {
  427. Calendar.removeClass(cal.hilitedYear, "hilite");
  428. }
  429. }
  430. return Calendar.stopEvent(ev);
  431. };
  432. Calendar.tableMouseDown = function (ev) {
  433. if (Calendar.getTargetElement(ev) == Calendar.getElement(ev)) {
  434. return Calendar.stopEvent(ev);
  435. }
  436. };
  437. Calendar.calDragIt = function (ev) {
  438. var cal = Calendar._C;
  439. if (!(cal && cal.dragging)) {
  440. return false;
  441. }
  442. var posX;
  443. var posY;
  444. if (Calendar.is_ie) {
  445. posY = window.event.clientY + document.body.scrollTop;
  446. posX = window.event.clientX + document.body.scrollLeft;
  447. } else {
  448. posX = ev.pageX;
  449. posY = ev.pageY;
  450. }
  451. cal.hideShowCovered();
  452. var st = cal.element.style;
  453. st.left = (posX - cal.xOffs) + "px";
  454. st.top = (posY - cal.yOffs) + "px";
  455. return Calendar.stopEvent(ev);
  456. };
  457. Calendar.calDragEnd = function (ev) {
  458. var cal = Calendar._C;
  459. if (!cal) {
  460. return false;
  461. }
  462. cal.dragging = false;
  463. with (Calendar) {
  464. removeEvent(document, "mousemove", calDragIt);
  465. removeEvent(document, "mouseup", calDragEnd);
  466. tableMouseUp(ev);
  467. }
  468. cal.hideShowCovered();
  469. };
  470. Calendar.dayMouseDown = function(ev) {
  471. var el = Calendar.getElement(ev);
  472. if (el.disabled) {
  473. return false;
  474. }
  475. var cal = el.calendar;
  476. cal.activeDiv = el;
  477. Calendar._C = cal;
  478. if (el.navtype != 300) with (Calendar) {
  479. if (el.navtype == 50) {
  480. el._current = el.innerHTML;
  481. addEvent(document, "mousemove", tableMouseOver);
  482. } else
  483. addEvent(document, Calendar.is_ie5 ? "mousemove" : "mouseover", tableMouseOver);
  484. addClass(el, "hilite active");
  485. addEvent(document, "mouseup", tableMouseUp);
  486. } else if (cal.isPopup) {
  487. cal._dragStart(ev);
  488. }
  489. if (el.navtype == -1 || el.navtype == 1) {
  490. if (cal.timeout) clearTimeout(cal.timeout);
  491. cal.timeout = setTimeout("Calendar.showMonthsCombo()", 250);
  492. } else if (el.navtype == -2 || el.navtype == 2) {
  493. if (cal.timeout) clearTimeout(cal.timeout);
  494. cal.timeout = setTimeout((el.navtype > 0) ? "Calendar.showYearsCombo(true)" : "Calendar.showYearsCombo(false)", 250);
  495. } else {
  496. cal.timeout = null;
  497. }
  498. return Calendar.stopEvent(ev);
  499. };
  500. Calendar.dayMouseDblClick = function(ev) {
  501. Calendar.cellClick(Calendar.getElement(ev), ev || window.event);
  502. if (Calendar.is_ie) {
  503. document.selection.empty();
  504. }
  505. };
  506. Calendar.dayMouseOver = function(ev) {
  507. var el = Calendar.getElement(ev);
  508. if (Calendar.isRelated(el, ev) || Calendar._C || el.disabled) {
  509. return false;
  510. }
  511. if (el.ttip) {
  512. if (el.ttip.substr(0, 1) == "_") {
  513. el.ttip = el.caldate.print(el.calendar.ttDateFormat) + el.ttip.substr(1);
  514. }
  515. el.calendar.tooltips.innerHTML = el.ttip;
  516. }
  517. if (el.navtype != 300) {
  518. Calendar.addClass(el, "hilite");
  519. if (el.caldate) {
  520. Calendar.addClass(el.parentNode, "rowhilite");
  521. }
  522. }
  523. return Calendar.stopEvent(ev);
  524. };
  525. Calendar.dayMouseOut = function(ev) {
  526. with (Calendar) {
  527. var el = getElement(ev);
  528. if (isRelated(el, ev) || _C || el.disabled)
  529. return false;
  530. removeClass(el, "hilite");
  531. if (el.caldate)
  532. removeClass(el.parentNode, "rowhilite");
  533. if (el.calendar)
  534. el.calendar.tooltips.innerHTML = _TT["SEL_DATE"];
  535. return stopEvent(ev);
  536. }
  537. };
  538. /**
  539.  *  A generic "click" handler :) handles all types of buttons defined in this
  540.  *  calendar.
  541.  */
  542. Calendar.cellClick = function(el, ev) {
  543. var cal = el.calendar;
  544. var closing = false;
  545. var newdate = false;
  546. var date = null;
  547. if (typeof el.navtype == "undefined") {
  548. if (cal.currentDateEl) {
  549. Calendar.removeClass(cal.currentDateEl, "selected");
  550. Calendar.addClass(el, "selected");
  551. closing = (cal.currentDateEl == el);
  552. if (!closing) {
  553. cal.currentDateEl = el;
  554. }
  555. }
  556. cal.date.setDateOnly(el.caldate);
  557. date = cal.date;
  558. var other_month = !(cal.dateClicked = !el.otherMonth);
  559. if (!other_month && !cal.currentDateEl)
  560. cal._toggleMultipleDate(new Date(date));
  561. else
  562. newdate = !el.disabled;
  563. // a date was clicked
  564. if (other_month)
  565. cal._init(cal.firstDayOfWeek, date);
  566. } else {
  567. if (el.navtype == 200) {
  568. Calendar.removeClass(el, "hilite");
  569. cal.callCloseHandler();
  570. return;
  571. }
  572. date = new Date(cal.date);
  573. if (el.navtype == 0)
  574. date.setDateOnly(new Date()); // TODAY
  575. // unless "today" was clicked, we assume no date was clicked so
  576. // the selected handler will know not to close the calenar when
  577. // in single-click mode.
  578. // cal.dateClicked = (el.navtype == 0);
  579. cal.dateClicked = false;
  580. var year = date.getFullYear();
  581. var mon = date.getMonth();
  582. function setMonth(m) {
  583. var day = date.getDate();
  584. var max = date.getMonthDays(m);
  585. if (day > max) {
  586. date.setDate(max);
  587. }
  588. date.setMonth(m);
  589. };
  590. switch (el.navtype) {
  591.     case 400:
  592. Calendar.removeClass(el, "hilite");
  593. var text = Calendar._TT["ABOUT"];
  594. if (typeof text != "undefined") {
  595. text += cal.showsTime ? Calendar._TT["ABOUT_TIME"] : "";
  596. } else {
  597. // FIXME: this should be removed as soon as lang files get updated!
  598. text = "Help and about box text is not translated into this language.n" +
  599. "If you know this language and you feel generous please updaten" +
  600. "the corresponding file in "lang" subdir to match calendar-en.jsn" +
  601. "and send it back to <mihai_bazon@yahoo.com> to get it into the distribution  ;-)nn" +
  602. "Thank you!n" +
  603. "http://dynarch.com/mishoo/calendar.epln";
  604. }
  605. alert(text);
  606. return;
  607.     case -2:
  608. if (year > cal.minYear) {
  609. date.setFullYear(year - 1);
  610. }
  611. break;
  612.     case -1:
  613. if (mon > 0) {
  614. setMonth(mon - 1);
  615. } else if (year-- > cal.minYear) {
  616. date.setFullYear(year);
  617. setMonth(11);
  618. }
  619. break;
  620.     case 1:
  621. if (mon < 11) {
  622. setMonth(mon + 1);
  623. } else if (year < cal.maxYear) {
  624. date.setFullYear(year + 1);
  625. setMonth(0);
  626. }
  627. break;
  628.     case 2:
  629. if (year < cal.maxYear) {
  630. date.setFullYear(year + 1);
  631. }
  632. break;
  633.     case 100:
  634. cal.setFirstDayOfWeek(el.fdow);
  635. return;
  636.     case 50:
  637. var range = el._range;
  638. var current = el.innerHTML;
  639. for (var i = range.length; --i >= 0;)
  640. if (range[i] == current)
  641. break;
  642. if (ev && ev.shiftKey) {
  643. if (--i < 0)
  644. i = range.length - 1;
  645. } else if ( ++i >= range.length )
  646. i = 0;
  647. var newval = range[i];
  648. el.innerHTML = newval;
  649. cal.onUpdateTime();
  650. return;
  651.     case 0:
  652. // TODAY will bring us here
  653. if ((typeof cal.getDateStatus == "function") &&
  654.     cal.getDateStatus(date, date.getFullYear(), date.getMonth(), date.getDate())) {
  655. return false;
  656. }
  657. break;
  658. }
  659. if (!date.equalsTo(cal.date)) {
  660. cal.setDate(date);
  661. newdate = true;
  662. } else if (el.navtype == 0)
  663. newdate = closing = true;
  664. }
  665. if (newdate) {
  666. ev && cal.callHandler();
  667. }
  668. if (closing) {
  669. Calendar.removeClass(el, "hilite");
  670. ev && cal.callCloseHandler();
  671. }
  672. };
  673. // END: CALENDAR STATIC FUNCTIONS
  674. // BEGIN: CALENDAR OBJECT FUNCTIONS
  675. /**
  676.  *  This function creates the calendar inside the given parent.  If _par is
  677.  *  null than it creates a popup calendar inside the BODY element.  If _par is
  678.  *  an element, be it BODY, then it creates a non-popup calendar (still
  679.  *  hidden).  Some properties need to be set before calling this function.
  680.  */
  681. Calendar.prototype.create = function (_par) {
  682. var parent = null;
  683. if (! _par) {
  684. // default parent is the document body, in which case we create
  685. // a popup calendar.
  686. parent = document.getElementsByTagName("body")[0];
  687. this.isPopup = true;
  688. } else {
  689. parent = _par;
  690. this.isPopup = false;
  691. }
  692. this.date = this.dateStr ? new Date(this.dateStr) : new Date();
  693. var table = Calendar.createElement("table");
  694. this.table = table;
  695. table.cellSpacing = 0;
  696. table.cellPadding = 0;
  697. table.calendar = this;
  698. Calendar.addEvent(table, "mousedown", Calendar.tableMouseDown);
  699. var div = Calendar.createElement("div");
  700. this.element = div;
  701. div.className = "calendar";
  702. if (this.isPopup) {
  703. div.style.position = "absolute";
  704. div.style.display = "none";
  705. }
  706. div.appendChild(table);
  707. var thead = Calendar.createElement("thead", table);
  708. var cell = null;
  709. var row = null;
  710. var cal = this;
  711. var hh = function (text, cs, navtype) {
  712. cell = Calendar.createElement("td", row);
  713. cell.colSpan = cs;
  714. cell.className = "button";
  715. if (navtype != 0 && Math.abs(navtype) <= 2)
  716. cell.className += " nav";
  717. Calendar._add_evs(cell);
  718. cell.calendar = cal;
  719. cell.navtype = navtype;
  720. cell.innerHTML = "<div unselectable='on'>" + text + "</div>";
  721. return cell;
  722. };
  723. row = Calendar.createElement("tr", thead);
  724. var title_length = 6;
  725. (this.isPopup) && --title_length;
  726. (this.weekNumbers) && ++title_length;
  727. hh("?", 1, 400).ttip = Calendar._TT["INFO"];
  728. this.title = hh("", title_length, 300);
  729. this.title.className = "title";
  730. if (this.isPopup) {
  731. this.title.ttip = Calendar._TT["DRAG_TO_MOVE"];
  732. this.title.style.cursor = "move";
  733. hh("&#x00d7;", 1, 200).ttip = Calendar._TT["CLOSE"];
  734. }
  735. row = Calendar.createElement("tr", thead);
  736. row.className = "headrow";
  737. this._nav_py = hh("&#x00ab;", 1, -2);
  738. this._nav_py.ttip = Calendar._TT["PREV_YEAR"];
  739. this._nav_pm = hh("&#x2039;", 1, -1);
  740. this._nav_pm.ttip = Calendar._TT["PREV_MONTH"];
  741. this._nav_now = hh(Calendar._TT["TODAY"], this.weekNumbers ? 4 : 3, 0);
  742. this._nav_now.ttip = Calendar._TT["GO_TODAY"];
  743. this._nav_nm = hh("&#x203a;", 1, 1);
  744. this._nav_nm.ttip = Calendar._TT["NEXT_MONTH"];
  745. this._nav_ny = hh("&#x00bb;", 1, 2);
  746. this._nav_ny.ttip = Calendar._TT["NEXT_YEAR"];
  747. // day names
  748. row = Calendar.createElement("tr", thead);
  749. row.className = "daynames";
  750. if (this.weekNumbers) {
  751. cell = Calendar.createElement("td", row);
  752. cell.className = "name wn";
  753. cell.innerHTML = Calendar._TT["WK"];
  754. }
  755. for (var i = 7; i > 0; --i) {
  756. cell = Calendar.createElement("td", row);
  757. if (!i) {
  758. cell.navtype = 100;
  759. cell.calendar = this;
  760. Calendar._add_evs(cell);
  761. }
  762. }
  763. this.firstdayname = (this.weekNumbers) ? row.firstChild.nextSibling : row.firstChild;
  764. this._displayWeekdays();
  765. var tbody = Calendar.createElement("tbody", table);
  766. this.tbody = tbody;
  767. for (i = 6; i > 0; --i) {
  768. row = Calendar.createElement("tr", tbody);
  769. if (this.weekNumbers) {
  770. cell = Calendar.createElement("td", row);
  771. }
  772. for (var j = 7; j > 0; --j) {
  773. cell = Calendar.createElement("td", row);
  774. cell.calendar = this;
  775. Calendar._add_evs(cell);
  776. }
  777. }
  778. if (this.showsTime) {
  779. row = Calendar.createElement("tr", tbody);
  780. row.className = "time";
  781. cell = Calendar.createElement("td", row);
  782. cell.className = "time";
  783. cell.colSpan = 2;
  784. cell.innerHTML = Calendar._TT["TIME"] || "&nbsp;";
  785. cell = Calendar.createElement("td", row);
  786. cell.className = "time";
  787. cell.colSpan = this.weekNumbers ? 4 : 3;
  788. (function(){
  789. function makeTimePart(className, init, range_start, range_end) {
  790. var part = Calendar.createElement("span", cell);
  791. part.className = className;
  792. part.innerHTML = init;
  793. part.calendar = cal;
  794. part.ttip = Calendar._TT["TIME_PART"];
  795. part.navtype = 50;
  796. part._range = [];
  797. if (typeof range_start != "number")
  798. part._range = range_start;
  799. else {
  800. for (var i = range_start; i <= range_end; ++i) {
  801. var txt;
  802. if (i < 10 && range_end >= 10) txt = '0' + i;
  803. else txt = '' + i;
  804. part._range[part._range.length] = txt;
  805. }
  806. }
  807. Calendar._add_evs(part);
  808. return part;
  809. };
  810. var hrs = cal.date.getHours();
  811. var mins = cal.date.getMinutes();
  812. var t12 = !cal.time24;
  813. var pm = (hrs > 12);
  814. if (t12 && pm) hrs -= 12;
  815. var H = makeTimePart("hour", hrs, t12 ? 1 : 0, t12 ? 12 : 23);
  816. var span = Calendar.createElement("span", cell);
  817. span.innerHTML = ":";
  818. span.className = "colon";
  819. var M = makeTimePart("minute", mins, 0, 59);
  820. var AP = null;
  821. cell = Calendar.createElement("td", row);
  822. cell.className = "time";
  823. cell.colSpan = 2;
  824. if (t12)
  825. AP = makeTimePart("ampm", pm ? "pm" : "am", ["am", "pm"]);
  826. else
  827. cell.innerHTML = "&nbsp;";
  828. cal.onSetTime = function() {
  829. var pm, hrs = this.date.getHours(),
  830. mins = this.date.getMinutes();
  831. if (t12) {
  832. pm = (hrs >= 12);
  833. if (pm) hrs -= 12;
  834. if (hrs == 0) hrs = 12;
  835. AP.innerHTML = pm ? "pm" : "am";
  836. }
  837. H.innerHTML = (hrs < 10) ? ("0" + hrs) : hrs;
  838. M.innerHTML = (mins < 10) ? ("0" + mins) : mins;
  839. };
  840. cal.onUpdateTime = function() {
  841. var date = this.date;
  842. var h = parseInt(H.innerHTML, 10);
  843. if (t12) {
  844. if (/pm/i.test(AP.innerHTML) && h < 12)
  845. h += 12;
  846. else if (/am/i.test(AP.innerHTML) && h == 12)
  847. h = 0;
  848. }
  849. var d = date.getDate();
  850. var m = date.getMonth();
  851. var y = date.getFullYear();
  852. date.setHours(h);
  853. date.setMinutes(parseInt(M.innerHTML, 10));
  854. date.setFullYear(y);
  855. date.setMonth(m);
  856. date.setDate(d);
  857. this.dateClicked = false;
  858. this.callHandler();
  859. };
  860. })();
  861. } else {
  862. this.onSetTime = this.onUpdateTime = function() {};
  863. }
  864. var tfoot = Calendar.createElement("tfoot", table);
  865. row = Calendar.createElement("tr", tfoot);
  866. row.className = "footrow";
  867. cell = hh(Calendar._TT["SEL_DATE"], this.weekNumbers ? 8 : 7, 300);
  868. cell.className = "ttip";
  869. if (this.isPopup) {
  870. cell.ttip = Calendar._TT["DRAG_TO_MOVE"];
  871. cell.style.cursor = "move";
  872. }
  873. this.tooltips = cell;
  874. div = Calendar.createElement("div", this.element);
  875. this.monthsCombo = div;
  876. div.className = "combo";
  877. for (i = 0; i < Calendar._MN.length; ++i) {
  878. var mn = Calendar.createElement("div");
  879. mn.className = Calendar.is_ie ? "label-IEfix" : "label";
  880. mn.month = i;
  881. mn.innerHTML = Calendar._SMN[i];
  882. div.appendChild(mn);
  883. }
  884. div = Calendar.createElement("div", this.element);
  885. this.yearsCombo = div;
  886. div.className = "combo";
  887. for (i = 12; i > 0; --i) {
  888. var yr = Calendar.createElement("div");
  889. yr.className = Calendar.is_ie ? "label-IEfix" : "label";
  890. div.appendChild(yr);
  891. }
  892. this._init(this.firstDayOfWeek, this.date);
  893. parent.appendChild(this.element);
  894. };
  895. /** keyboard navigation, only for popup calendars */
  896. Calendar._keyEvent = function(ev) {
  897. var cal = window._dynarch_popupCalendar;
  898. if (!cal || cal.multiple)
  899. return false;
  900. (Calendar.is_ie) && (ev = window.event);
  901. var act = (Calendar.is_ie || ev.type == "keypress"),
  902. K = ev.keyCode;
  903. if (ev.ctrlKey) {
  904. switch (K) {
  905.     case 37: // KEY left
  906. act && Calendar.cellClick(cal._nav_pm);
  907. break;
  908.     case 38: // KEY up
  909. act && Calendar.cellClick(cal._nav_py);
  910. break;
  911.     case 39: // KEY right
  912. act && Calendar.cellClick(cal._nav_nm);
  913. break;
  914.     case 40: // KEY down
  915. act && Calendar.cellClick(cal._nav_ny);
  916. break;
  917.     default:
  918. return false;
  919. }
  920. } else switch (K) {
  921.     case 32: // KEY space (now)
  922. Calendar.cellClick(cal._nav_now);
  923. break;
  924.     case 27: // KEY esc
  925. act && cal.callCloseHandler();
  926. break;
  927.     case 37: // KEY left
  928.     case 38: // KEY up
  929.     case 39: // KEY right
  930.     case 40: // KEY down
  931. if (act) {
  932. var prev, x, y, ne, el, step;
  933. prev = K == 37 || K == 38;
  934. step = (K == 37 || K == 39) ? 1 : 7;
  935. function setVars() {
  936. el = cal.currentDateEl;
  937. var p = el.pos;
  938. x = p & 15;
  939. y = p >> 4;
  940. ne = cal.ar_days[y][x];
  941. };setVars();
  942. function prevMonth() {
  943. var date = new Date(cal.date);
  944. date.setDate(date.getDate() - step);
  945. cal.setDate(date);
  946. };
  947. function nextMonth() {
  948. var date = new Date(cal.date);
  949. date.setDate(date.getDate() + step);
  950. cal.setDate(date);
  951. };
  952. while (1) {
  953. switch (K) {
  954.     case 37: // KEY left
  955. if (--x >= 0)
  956. ne = cal.ar_days[y][x];
  957. else {
  958. x = 6;
  959. K = 38;
  960. continue;
  961. }
  962. break;
  963.     case 38: // KEY up
  964. if (--y >= 0)
  965. ne = cal.ar_days[y][x];
  966. else {
  967. prevMonth();
  968. setVars();
  969. }
  970. break;
  971.     case 39: // KEY right
  972. if (++x < 7)
  973. ne = cal.ar_days[y][x];
  974. else {
  975. x = 0;
  976. K = 40;
  977. continue;
  978. }
  979. break;
  980.     case 40: // KEY down
  981. if (++y < cal.ar_days.length)
  982. ne = cal.ar_days[y][x];
  983. else {
  984. nextMonth();
  985. setVars();
  986. }
  987. break;
  988. }
  989. break;
  990. }
  991. if (ne) {
  992. if (!ne.disabled)
  993. Calendar.cellClick(ne);
  994. else if (prev)
  995. prevMonth();
  996. else
  997. nextMonth();
  998. }
  999. }
  1000. break;
  1001.     case 13: // KEY enter
  1002. if (act)
  1003. Calendar.cellClick(cal.currentDateEl, ev);
  1004. break;
  1005.     default:
  1006. return false;
  1007. }
  1008. return Calendar.stopEvent(ev);
  1009. };
  1010. /**
  1011.  *  (RE)Initializes the calendar to the given date and firstDayOfWeek
  1012.  */
  1013. Calendar.prototype._init = function (firstDayOfWeek, date) {
  1014. var today = new Date(),
  1015. TY = today.getFullYear(),
  1016. TM = today.getMonth(),
  1017. TD = today.getDate();
  1018. this.table.style.visibility = "hidden";
  1019. var year = date.getFullYear();
  1020. if (year < this.minYear) {
  1021. year = this.minYear;
  1022. date.setFullYear(year);
  1023. } else if (year > this.maxYear) {
  1024. year = this.maxYear;
  1025. date.setFullYear(year);
  1026. }
  1027. this.firstDayOfWeek = firstDayOfWeek;
  1028. this.date = new Date(date);
  1029. var month = date.getMonth();
  1030. var mday = date.getDate();
  1031. var no_days = date.getMonthDays();
  1032. // calendar voodoo for computing the first day that would actually be
  1033. // displayed in the calendar, even if it's from the previous month.
  1034. // WARNING: this is magic. ;-)
  1035. date.setDate(1);
  1036. var day1 = (date.getDay() - this.firstDayOfWeek) % 7;
  1037. if (day1 < 0)
  1038. day1 += 7;
  1039. date.setDate(-day1);
  1040. date.setDate(date.getDate() + 1);
  1041. var row = this.tbody.firstChild;
  1042. var MN = Calendar._SMN[month];
  1043. var ar_days = this.ar_days = new Array();
  1044. var weekend = Calendar._TT["WEEKEND"];
  1045. var dates = this.multiple ? (this.datesCells = {}) : null;
  1046. for (var i = 0; i < 6; ++i, row = row.nextSibling) {
  1047. var cell = row.firstChild;
  1048. if (this.weekNumbers) {
  1049. cell.className = "day wn";
  1050. cell.innerHTML = date.getWeekNumber();
  1051. cell = cell.nextSibling;
  1052. }
  1053. row.className = "daysrow";
  1054. var hasdays = false, iday, dpos = ar_days[i] = [];
  1055. for (var j = 0; j < 7; ++j, cell = cell.nextSibling, date.setDate(iday + 1)) {
  1056. iday = date.getDate();
  1057. var wday = date.getDay();
  1058. cell.className = "day";
  1059. cell.pos = i << 4 | j;
  1060. dpos[j] = cell;
  1061. var current_month = (date.getMonth() == month);
  1062. if (!current_month) {
  1063. if (this.showsOtherMonths) {
  1064. cell.className += " othermonth";
  1065. cell.otherMonth = true;
  1066. } else {
  1067. cell.className = "emptycell";
  1068. cell.innerHTML = "&nbsp;";
  1069. cell.disabled = true;
  1070. continue;
  1071. }
  1072. } else {
  1073. cell.otherMonth = false;
  1074. hasdays = true;
  1075. }
  1076. cell.disabled = false;
  1077. cell.innerHTML = this.getDateText ? this.getDateText(date, iday) : iday;
  1078. if (dates)
  1079. dates[date.print("%Y%m%d")] = cell;
  1080. if (this.getDateStatus) {
  1081. var status = this.getDateStatus(date, year, month, iday);
  1082. if (this.getDateToolTip) {
  1083. var toolTip = this.getDateToolTip(date, year, month, iday);
  1084. if (toolTip)
  1085. cell.title = toolTip;
  1086. }
  1087. if (status === true) {
  1088. cell.className += " disabled";
  1089. cell.disabled = true;
  1090. } else {
  1091. if (/disabled/i.test(status))
  1092. cell.disabled = true;
  1093. cell.className += " " + status;
  1094. }
  1095. }
  1096. if (!cell.disabled) {
  1097. cell.caldate = new Date(date);
  1098. cell.ttip = "_";
  1099. if (!this.multiple && current_month
  1100.     && iday == mday && this.hiliteToday) {
  1101. cell.className += " selected";
  1102. this.currentDateEl = cell;
  1103. }
  1104. if (date.getFullYear() == TY &&
  1105.     date.getMonth() == TM &&
  1106.     iday == TD) {
  1107. cell.className += " today";
  1108. cell.ttip += Calendar._TT["PART_TODAY"];
  1109. }
  1110. if (weekend.indexOf(wday.toString()) != -1)
  1111. cell.className += cell.otherMonth ? " oweekend" : " weekend";
  1112. }
  1113. }
  1114. if (!(hasdays || this.showsOtherMonths))
  1115. row.className = "emptyrow";
  1116. }
  1117. this.title.innerHTML = Calendar._MN[month] + ", " + year;
  1118. this.onSetTime();
  1119. this.table.style.visibility = "visible";
  1120. this._initMultipleDates();
  1121. // PROFILE
  1122. // this.tooltips.innerHTML = "Generated in " + ((new Date()) - today) + " ms";
  1123. };
  1124. Calendar.prototype._initMultipleDates = function() {
  1125. if (this.multiple) {
  1126. for (var i in this.multiple) {
  1127. var cell = this.datesCells[i];
  1128. var d = this.multiple[i];
  1129. if (!d)
  1130. continue;
  1131. if (cell)
  1132. cell.className += " selected";
  1133. }
  1134. }
  1135. };
  1136. Calendar.prototype._toggleMultipleDate = function(date) {
  1137. if (this.multiple) {
  1138. var ds = date.print("%Y%m%d");
  1139. var cell = this.datesCells[ds];
  1140. if (cell) {
  1141. var d = this.multiple[ds];
  1142. if (!d) {
  1143. Calendar.addClass(cell, "selected");
  1144. this.multiple[ds] = date;
  1145. } else {
  1146. Calendar.removeClass(cell, "selected");
  1147. delete this.multiple[ds];
  1148. }
  1149. }
  1150. }
  1151. };
  1152. Calendar.prototype.setDateToolTipHandler = function (unaryFunction) {
  1153. this.getDateToolTip = unaryFunction;
  1154. };
  1155. /**
  1156.  *  Calls _init function above for going to a certain date (but only if the
  1157.  *  date is different than the currently selected one).
  1158.  */
  1159. Calendar.prototype.setDate = function (date) {
  1160. if (!date.equalsTo(this.date)) {
  1161. this._init(this.firstDayOfWeek, date);
  1162. }
  1163. };
  1164. /**
  1165.  *  Refreshes the calendar.  Useful if the "disabledHandler" function is
  1166.  *  dynamic, meaning that the list of disabled date can change at runtime.
  1167.  *  Just * call this function if you think that the list of disabled dates
  1168.  *  should * change.
  1169.  */
  1170. Calendar.prototype.refresh = function () {
  1171. this._init(this.firstDayOfWeek, this.date);
  1172. };
  1173. /** Modifies the "firstDayOfWeek" parameter (pass 0 for Synday, 1 for Monday, etc.). */
  1174. Calendar.prototype.setFirstDayOfWeek = function (firstDayOfWeek) {
  1175. this._init(firstDayOfWeek, this.date);
  1176. this._displayWeekdays();
  1177. };
  1178. /**
  1179.  *  Allows customization of what dates are enabled.  The "unaryFunction"
  1180.  *  parameter must be a function object that receives the date (as a JS Date
  1181.  *  object) and returns a boolean value.  If the returned value is true then
  1182.  *  the passed date will be marked as disabled.
  1183.  */
  1184. Calendar.prototype.setDateStatusHandler = Calendar.prototype.setDisabledHandler = function (unaryFunction) {
  1185. this.getDateStatus = unaryFunction;
  1186. };
  1187. /** Customization of allowed year range for the calendar. */
  1188. Calendar.prototype.setRange = function (a, z) {
  1189. this.minYear = a;
  1190. this.maxYear = z;
  1191. };
  1192. /** Calls the first user handler (selectedHandler). */
  1193. Calendar.prototype.callHandler = function () {
  1194. if (this.onSelected) {
  1195. this.onSelected(this, this.date.print(this.dateFormat));
  1196. }
  1197. };
  1198. /** Calls the second user handler (closeHandler). */
  1199. Calendar.prototype.callCloseHandler = function () {
  1200. if (this.onClose) {
  1201. this.onClose(this);
  1202. }
  1203. this.hideShowCovered();
  1204. };
  1205. /** Removes the calendar object from the DOM tree and destroys it. */
  1206. Calendar.prototype.destroy = function () {
  1207. var el = this.element.parentNode;
  1208. el.removeChild(this.element);
  1209. Calendar._C = null;
  1210. window._dynarch_popupCalendar = null;
  1211. };
  1212. /**
  1213.  *  Moves the calendar element to a different section in the DOM tree (changes
  1214.  *  its parent).
  1215.  */
  1216. Calendar.prototype.reparent = function (new_parent) {
  1217. var el = this.element;
  1218. el.parentNode.removeChild(el);
  1219. new_parent.appendChild(el);
  1220. };
  1221. // This gets called when the user presses a mouse button anywhere in the
  1222. // document, if the calendar is shown.  If the click was outside the open
  1223. // calendar this function closes it.
  1224. Calendar._checkCalendar = function(ev) {
  1225. var calendar = window._dynarch_popupCalendar;
  1226. if (!calendar) {
  1227. return false;
  1228. }
  1229. var el = Calendar.is_ie ? Calendar.getElement(ev) : Calendar.getTargetElement(ev);
  1230. for (; el != null && el != calendar.element; el = el.parentNode);
  1231. if (el == null) {
  1232. // calls closeHandler which should hide the calendar.
  1233. window._dynarch_popupCalendar.callCloseHandler();
  1234. return Calendar.stopEvent(ev);
  1235. }
  1236. };
  1237. /** Shows the calendar. */
  1238. Calendar.prototype.show = function () {
  1239. var rows = this.table.getElementsByTagName("tr");
  1240. for (var i = rows.length; i > 0;) {
  1241. var row = rows[--i];
  1242. Calendar.removeClass(row, "rowhilite");
  1243. var cells = row.getElementsByTagName("td");
  1244. for (var j = cells.length; j > 0;) {
  1245. var cell = cells[--j];
  1246. Calendar.removeClass(cell, "hilite");
  1247. Calendar.removeClass(cell, "active");
  1248. }
  1249. }
  1250. this.element.style.display = "block";
  1251. this.hidden = false;
  1252. if (this.isPopup) {
  1253. window._dynarch_popupCalendar = this;
  1254. Calendar.addEvent(document, "keydown", Calendar._keyEvent);
  1255. Calendar.addEvent(document, "keypress", Calendar._keyEvent);
  1256. Calendar.addEvent(document, "mousedown", Calendar._checkCalendar);
  1257. }
  1258. this.hideShowCovered();
  1259. };
  1260. /**
  1261.  *  Hides the calendar.  Also removes any "hilite" from the class of any TD
  1262.  *  element.
  1263.  */
  1264. Calendar.prototype.hide = function () {
  1265. if (this.isPopup) {
  1266. Calendar.removeEvent(document, "keydown", Calendar._keyEvent);
  1267. Calendar.removeEvent(document, "keypress", Calendar._keyEvent);
  1268. Calendar.removeEvent(document, "mousedown", Calendar._checkCalendar);
  1269. }
  1270. this.element.style.display = "none";
  1271. this.hidden = true;
  1272. this.hideShowCovered();
  1273. };
  1274. /**
  1275.  *  Shows the calendar at a given absolute position (beware that, depending on
  1276.  *  the calendar element style -- position property -- this might be relative
  1277.  *  to the parent's containing rectangle).
  1278.  */
  1279. Calendar.prototype.showAt = function (x, y) {
  1280. var s = this.element.style;
  1281. s.left = x + "px";
  1282. s.top = y + "px";
  1283. this.show();
  1284. };
  1285. /** Shows the calendar near a given element. */
  1286. Calendar.prototype.showAtElement = function (el, opts) {
  1287. var self = this;
  1288. var p = Calendar.getAbsolutePos(el);
  1289. if (!opts || typeof opts != "string") {
  1290. this.showAt(p.x, p.y + el.offsetHeight);
  1291. return true;
  1292. }
  1293. function fixPosition(box) {
  1294. if (box.x < 0)
  1295. box.x = 0;
  1296. if (box.y < 0)
  1297. box.y = 0;
  1298. var cp = document.createElement("div");
  1299. var s = cp.style;
  1300. s.position = "absolute";
  1301. s.right = s.bottom = s.width = s.height = "0px";
  1302. document.body.appendChild(cp);
  1303. var br = Calendar.getAbsolutePos(cp);
  1304. document.body.removeChild(cp);
  1305. if (Calendar.is_ie) {
  1306. br.y += document.body.scrollTop;
  1307. br.x += document.body.scrollLeft;
  1308. } else {
  1309. br.y += window.scrollY;
  1310. br.x += window.scrollX;
  1311. }
  1312. var tmp = box.x + box.width - br.x;
  1313. if (tmp > 0) box.x -= tmp;
  1314. tmp = box.y + box.height - br.y;
  1315. if (tmp > 0) box.y -= tmp;
  1316. };
  1317. this.element.style.display = "block";
  1318. Calendar.continuation_for_the_khtml_browser = function() {
  1319. var w = self.element.offsetWidth;
  1320. var h = self.element.offsetHeight;
  1321. self.element.style.display = "none";
  1322. var valign = opts.substr(0, 1);
  1323. var halign = "l";
  1324. if (opts.length > 1) {
  1325. halign = opts.substr(1, 1);
  1326. }
  1327. // vertical alignment
  1328. switch (valign) {
  1329.     case "T": p.y -= h; break;
  1330.     case "B": p.y += el.offsetHeight; break;
  1331.     case "C": p.y += (el.offsetHeight - h) / 2; break;
  1332.     case "t": p.y += el.offsetHeight - h; break;
  1333.     case "b": break; // already there
  1334. }
  1335. // horizontal alignment
  1336. switch (halign) {
  1337.     case "L": p.x -= w; break;
  1338.     case "R": p.x += el.offsetWidth; break;
  1339.     case "C": p.x += (el.offsetWidth - w) / 2; break;
  1340.     case "l": p.x += el.offsetWidth - w; break;
  1341.     case "r": break; // already there
  1342. }
  1343. p.width = w;
  1344. p.height = h + 40;
  1345. self.monthsCombo.style.display = "none";
  1346. fixPosition(p);
  1347. self.showAt(p.x, p.y);
  1348. };
  1349. if (Calendar.is_khtml)
  1350. setTimeout("Calendar.continuation_for_the_khtml_browser()", 10);
  1351. else
  1352. Calendar.continuation_for_the_khtml_browser();
  1353. };
  1354. /** Customizes the date format. */
  1355. Calendar.prototype.setDateFormat = function (str) {
  1356. this.dateFormat = str;
  1357. };
  1358. /** Customizes the tooltip date format. */
  1359. Calendar.prototype.setTtDateFormat = function (str) {
  1360. this.ttDateFormat = str;
  1361. };
  1362. /**
  1363.  *  Tries to identify the date represented in a string.  If successful it also
  1364.  *  calls this.setDate which moves the calendar to the given date.
  1365.  */
  1366. Calendar.prototype.parseDate = function(str, fmt) {
  1367. if (!fmt)
  1368. fmt = this.dateFormat;
  1369. this.setDate(Date.parseDate(str, fmt));
  1370. };
  1371. Calendar.prototype.hideShowCovered = function () {
  1372. if (!Calendar.is_ie && !Calendar.is_opera)
  1373. return;
  1374. function getVisib(obj){
  1375. var value = obj.style.visibility;
  1376. if (!value) {
  1377. if (document.defaultView && typeof (document.defaultView.getComputedStyle) == "function") { // Gecko, W3C
  1378. if (!Calendar.is_khtml)
  1379. value = document.defaultView.
  1380. getComputedStyle(obj, "").getPropertyValue("visibility");
  1381. else
  1382. value = '';
  1383. } else if (obj.currentStyle) { // IE
  1384. value = obj.currentStyle.visibility;
  1385. } else
  1386. value = '';
  1387. }
  1388. return value;
  1389. };
  1390. var tags = new Array("applet", "iframe", "select");
  1391. var el = this.element;
  1392. var p = Calendar.getAbsolutePos(el);
  1393. var EX1 = p.x;
  1394. var EX2 = el.offsetWidth + EX1;
  1395. var EY1 = p.y;
  1396. var EY2 = el.offsetHeight + EY1;
  1397. for (var k = tags.length; k > 0; ) {
  1398. var ar = document.getElementsByTagName(tags[--k]);
  1399. var cc = null;
  1400. for (var i = ar.length; i > 0;) {
  1401. cc = ar[--i];
  1402. p = Calendar.getAbsolutePos(cc);
  1403. var CX1 = p.x;
  1404. var CX2 = cc.offsetWidth + CX1;
  1405. var CY1 = p.y;
  1406. var CY2 = cc.offsetHeight + CY1;
  1407. if (this.hidden || (CX1 > EX2) || (CX2 < EX1) || (CY1 > EY2) || (CY2 < EY1)) {
  1408. if (!cc.__msh_save_visibility) {
  1409. cc.__msh_save_visibility = getVisib(cc);
  1410. }
  1411. cc.style.visibility = cc.__msh_save_visibility;
  1412. } else {
  1413. if (!cc.__msh_save_visibility) {
  1414. cc.__msh_save_visibility = getVisib(cc);
  1415. }
  1416. cc.style.visibility = "hidden";
  1417. }
  1418. }
  1419. }
  1420. };
  1421. /** Internal function; it displays the bar with the names of the weekday. */
  1422. Calendar.prototype._displayWeekdays = function () {
  1423. var fdow = this.firstDayOfWeek;
  1424. var cell = this.firstdayname;
  1425. var weekend = Calendar._TT["WEEKEND"];
  1426. for (var i = 0; i < 7; ++i) {
  1427. cell.className = "day name";
  1428. var realday = (i + fdow) % 7;
  1429. if (i) {
  1430. cell.ttip = Calendar._TT["DAY_FIRST"].replace("%s", Calendar._DN[realday]);
  1431. cell.navtype = 100;
  1432. cell.calendar = this;
  1433. cell.fdow = realday;
  1434. Calendar._add_evs(cell);
  1435. }
  1436. if (weekend.indexOf(realday.toString()) != -1) {
  1437. Calendar.addClass(cell, "weekend");
  1438. }
  1439. cell.innerHTML = Calendar._SDN[(i + fdow) % 7];
  1440. cell = cell.nextSibling;
  1441. }
  1442. };
  1443. /** Internal function.  Hides all combo boxes that might be displayed. */
  1444. Calendar.prototype._hideCombos = function () {
  1445. this.monthsCombo.style.display = "none";
  1446. this.yearsCombo.style.display = "none";
  1447. };
  1448. /** Internal function.  Starts dragging the element. */
  1449. Calendar.prototype._dragStart = function (ev) {
  1450. if (this.dragging) {
  1451. return;
  1452. }
  1453. this.dragging = true;
  1454. var posX;
  1455. var posY;
  1456. if (Calendar.is_ie) {
  1457. posY = window.event.clientY + document.body.scrollTop;
  1458. posX = window.event.clientX + document.body.scrollLeft;
  1459. } else {
  1460. posY = ev.clientY + window.scrollY;
  1461. posX = ev.clientX + window.scrollX;
  1462. }
  1463. var st = this.element.style;
  1464. this.xOffs = posX - parseInt(st.left);
  1465. this.yOffs = posY - parseInt(st.top);
  1466. with (Calendar) {
  1467. addEvent(document, "mousemove", calDragIt);
  1468. addEvent(document, "mouseup", calDragEnd);
  1469. }
  1470. };
  1471. // BEGIN: DATE OBJECT PATCHES
  1472. /** Adds the number of days array to the Date object. */
  1473. Date._MD = new Array(31,28,31,30,31,30,31,31,30,31,30,31);
  1474. /** Constants used for time computations */
  1475. Date.SECOND = 1000 /* milliseconds */;
  1476. Date.MINUTE = 60 * Date.SECOND;
  1477. Date.HOUR   = 60 * Date.MINUTE;
  1478. Date.DAY    = 24 * Date.HOUR;
  1479. Date.WEEK   =  7 * Date.DAY;
  1480. Date.parseDate = function(str, fmt) {
  1481. var today = new Date();
  1482. var y = 0;
  1483. var m = -1;
  1484. var d = 0;
  1485. var a = str.split(/W+/);
  1486. var b = fmt.match(/%./g);
  1487. var i = 0, j = 0;
  1488. var hr = 0;
  1489. var min = 0;
  1490. for (i = 0; i < a.length; ++i) {
  1491. if (!a[i])
  1492. continue;
  1493. switch (b[i]) {
  1494.     case "%d":
  1495.     case "%e":
  1496. d = parseInt(a[i], 10);
  1497. break;
  1498.     case "%m":
  1499. m = parseInt(a[i], 10) - 1;
  1500. break;
  1501.     case "%Y":
  1502.     case "%y":
  1503. y = parseInt(a[i], 10);
  1504. (y < 100) && (y += (y > 29) ? 1900 : 2000);
  1505. break;
  1506.     case "%b":
  1507.     case "%B":
  1508. for (j = 0; j < 12; ++j) {
  1509. if (Calendar._MN[j].substr(0, a[i].length).toLowerCase() == a[i].toLowerCase()) { m = j; break; }
  1510. }
  1511. break;
  1512.     case "%H":
  1513.     case "%I":
  1514.     case "%k":
  1515.     case "%l":
  1516. hr = parseInt(a[i], 10);
  1517. break;
  1518.     case "%P":
  1519.     case "%p":
  1520. if (/pm/i.test(a[i]) && hr < 12)
  1521. hr += 12;
  1522. else if (/am/i.test(a[i]) && hr >= 12)
  1523. hr -= 12;
  1524. break;
  1525.     case "%M":
  1526. min = parseInt(a[i], 10);
  1527. break;
  1528. }
  1529. }
  1530. if (isNaN(y)) y = today.getFullYear();
  1531. if (isNaN(m)) m = today.getMonth();
  1532. if (isNaN(d)) d = today.getDate();
  1533. if (isNaN(hr)) hr = today.getHours();
  1534. if (isNaN(min)) min = today.getMinutes();
  1535. if (y != 0 && m != -1 && d != 0)
  1536. return new Date(y, m, d, hr, min, 0);
  1537. y = 0; m = -1; d = 0;
  1538. for (i = 0; i < a.length; ++i) {
  1539. if (a[i].search(eval("/[a-zA-Z]+/")) != -1) {
  1540. var t = -1;
  1541. for (j = 0; j < 12; ++j) {
  1542. if (Calendar._MN[j].substr(0, a[i].length).toLowerCase() == a[i].toLowerCase()) { t = j; break; }
  1543. }
  1544. if (t != -1) {
  1545. if (m != -1) {
  1546. d = m+1;
  1547. }
  1548. m = t;
  1549. }
  1550. } else if (parseInt(a[i], 10) <= 12 && m == -1) {
  1551. m = a[i]-1;
  1552. } else if (parseInt(a[i], 10) > 31 && y == 0) {
  1553. y = parseInt(a[i], 10);
  1554. (y < 100) && (y += (y > 29) ? 1900 : 2000);
  1555. } else if (d == 0) {
  1556. d = a[i];
  1557. }
  1558. }
  1559. if (y == 0)
  1560. y = today.getFullYear();
  1561. if (m != -1 && d != 0)
  1562. return new Date(y, m, d, hr, min, 0);
  1563. return today;
  1564. };
  1565. /** Returns the number of days in the current month */
  1566. Date.prototype.getMonthDays = function(month) {
  1567. var year = this.getFullYear();
  1568. if (typeof month == "undefined") {
  1569. month = this.getMonth();
  1570. }
  1571. if (((0 == (year%4)) && ( (0 != (year%100)) || (0 == (year%400)))) && month == 1) {
  1572. return 29;
  1573. } else {
  1574. return Date._MD[month];
  1575. }
  1576. };
  1577. /** Returns the number of day in the year. */
  1578. Date.prototype.getDayOfYear = function() {
  1579. var now = new Date(this.getFullYear(), this.getMonth(), this.getDate(), 0, 0, 0);
  1580. var then = new Date(this.getFullYear(), 0, 0, 0, 0, 0);
  1581. var time = now - then;
  1582. return Math.floor(time / Date.DAY);
  1583. };
  1584. /** Returns the number of the week in year, as defined in ISO 8601. */
  1585. Date.prototype.getWeekNumber = function() {
  1586. var d = new Date(this.getFullYear(), this.getMonth(), this.getDate(), 0, 0, 0);
  1587. var DoW = d.getDay();
  1588. d.setDate(d.getDate() - (DoW + 6) % 7 + 3); // Nearest Thu
  1589. var ms = d.valueOf(); // GMT
  1590. d.setMonth(0);
  1591. d.setDate(4); // Thu in Week 1
  1592. return Math.round((ms - d.valueOf()) / (7 * 864e5)) + 1;
  1593. };
  1594. /** Checks date and time equality */
  1595. Date.prototype.equalsTo = function(date) {
  1596. return ((this.getFullYear() == date.getFullYear()) &&
  1597. (this.getMonth() == date.getMonth()) &&
  1598. (this.getDate() == date.getDate()) &&
  1599. (this.getHours() == date.getHours()) &&
  1600. (this.getMinutes() == date.getMinutes()));
  1601. };
  1602. /** Set only the year, month, date parts (keep existing time) */
  1603. Date.prototype.setDateOnly = function(date) {
  1604. var tmp = new Date(date);
  1605. this.setDate(1);
  1606. this.setFullYear(tmp.getFullYear());
  1607. this.setMonth(tmp.getMonth());
  1608. this.setDate(tmp.getDate());
  1609. };
  1610. /** Prints the date in a string according to the given format. */
  1611. Date.prototype.print = function (str) {
  1612. var m = this.getMonth();
  1613. var d = this.getDate();
  1614. var y = this.getFullYear();
  1615. var wn = this.getWeekNumber();
  1616. var w = this.getDay();
  1617. var s = {};
  1618. var hr = this.getHours();
  1619. var pm = (hr >= 12);
  1620. var ir = (pm) ? (hr - 12) : hr;
  1621. var dy = this.getDayOfYear();
  1622. if (ir == 0)
  1623. ir = 12;
  1624. var min = this.getMinutes();
  1625. var sec = this.getSeconds();
  1626. s["%a"] = Calendar._SDN[w]; // abbreviated weekday name [FIXME: I18N]
  1627. s["%A"] = Calendar._DN[w]; // full weekday name
  1628. s["%b"] = Calendar._SMN[m]; // abbreviated month name [FIXME: I18N]
  1629. s["%B"] = Calendar._MN[m]; // full month name
  1630. // FIXME: %c : preferred date and time representation for the current locale
  1631. s["%C"] = 1 + Math.floor(y / 100); // the century number
  1632. s["%d"] = (d < 10) ? ("0" + d) : d; // the day of the month (range 01 to 31)
  1633. s["%e"] = d; // the day of the month (range 1 to 31)
  1634. // FIXME: %D : american date style: %m/%d/%y
  1635. // FIXME: %E, %F, %G, %g, %h (man strftime)
  1636. s["%H"] = (hr < 10) ? ("0" + hr) : hr; // hour, range 00 to 23 (24h format)
  1637. s["%I"] = (ir < 10) ? ("0" + ir) : ir; // hour, range 01 to 12 (12h format)
  1638. s["%j"] = (dy < 100) ? ((dy < 10) ? ("00" + dy) : ("0" + dy)) : dy; // day of the year (range 001 to 366)
  1639. s["%k"] = hr; // hour, range 0 to 23 (24h format)
  1640. s["%l"] = ir; // hour, range 1 to 12 (12h format)
  1641. s["%m"] = (m < 9) ? ("0" + (1+m)) : (1+m); // month, range 01 to 12
  1642. s["%M"] = (min < 10) ? ("0" + min) : min; // minute, range 00 to 59
  1643. s["%n"] = "n"; // a newline character
  1644. s["%p"] = pm ? "PM" : "AM";
  1645. s["%P"] = pm ? "pm" : "am";
  1646. // FIXME: %r : the time in am/pm notation %I:%M:%S %p
  1647. // FIXME: %R : the time in 24-hour notation %H:%M
  1648. s["%s"] = Math.floor(this.getTime() / 1000);
  1649. s["%S"] = (sec < 10) ? ("0" + sec) : sec; // seconds, range 00 to 59
  1650. s["%t"] = "t"; // a tab character
  1651. // FIXME: %T : the time in 24-hour notation (%H:%M:%S)
  1652. s["%U"] = s["%W"] = s["%V"] = (wn < 10) ? ("0" + wn) : wn;
  1653. s["%u"] = w + 1; // the day of the week (range 1 to 7, 1 = MON)
  1654. s["%w"] = w; // the day of the week (range 0 to 6, 0 = SUN)
  1655. // FIXME: %x : preferred date representation for the current locale without the time
  1656. // FIXME: %X : preferred time representation for the current locale without the date
  1657. s["%y"] = ('' + y).substr(2, 2); // year without the century (range 00 to 99)
  1658. s["%Y"] = y; // year with the century
  1659. s["%%"] = "%"; // a literal '%' character
  1660. var re = /%./g;
  1661. if (!Calendar.is_ie5 && !Calendar.is_khtml)
  1662. return str.replace(re, function (par) { return s[par] || par; });
  1663. var a = str.match(re);
  1664. for (var i = 0; i < a.length; i++) {
  1665. var tmp = s[a[i]];
  1666. if (tmp) {
  1667. re = new RegExp(a[i], 'g');
  1668. str = str.replace(re, tmp);
  1669. }
  1670. }
  1671. return str;
  1672. };
  1673. Date.prototype.__msh_oldSetFullYear = Date.prototype.setFullYear;
  1674. Date.prototype.setFullYear = function(y) {
  1675. var d = new Date(this);
  1676. d.__msh_oldSetFullYear(y);
  1677. if (d.getMonth() != this.getMonth())
  1678. this.setDate(28);
  1679. this.__msh_oldSetFullYear(y);
  1680. };
  1681. // END: DATE OBJECT PATCHES
  1682. // global object that remembers the calendar
  1683. window._dynarch_popupCalendar = null;