/media/system/js/calendar-uncompressed.js

https://bitbucket.org/kraymitchell/fcd · JavaScript · 1805 lines · 1540 code · 126 blank · 139 comment · 476 complexity · ec63acd3c05812c7edbd7d414efd95eb MD5 · raw file

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