calendar.js
上传用户:stephen_wu
上传日期:2008-07-05
资源大小:1757k
文件大小:50k
源码类别:

网络

开发平台:

Unix_Linux

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