yvanui.calendar.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  1. /**
  2. * YvanUI
  3. *
  4. * Copyright (c) 2018 www.yvanui.com. All rights reserved.
  5. * @author luoyifan@qq.com
  6. * @time 2018-11-27 11:16:00
  7. */
  8. (function ($) {
  9. function setSize(target, param) {
  10. var opts = $.data(target, 'calendar').options;
  11. var t = $(target);
  12. if (param) {
  13. $.extend(opts, {
  14. width: param.width,
  15. height: param.height
  16. });
  17. }
  18. t._size(opts, t.parent());
  19. t.find('.calendar-body')._outerHeight(t.height() - t.find('.calendar-header')._outerHeight());
  20. if (t.find('.date-year-list').is(':visible')) {
  21. opts.startYear = opts.year - 7;
  22. showSelectYear(target);
  23. } else if (t.find('.date-month-list').is(':visible')) {
  24. showSelectMonth(target);
  25. }
  26. }
  27. function init(target) {
  28. $(target).addClass('calendar').html(
  29. '<div class="calendar-header">' +
  30. ' <div class="calendar-nav calendar-prevmonth fa fa-angle-left"></div>' +
  31. ' <div class="calendar-nav calendar-nextmonth fa fa-angle-right"></div>' +
  32. ' <div class="calendar-nav calendar-prevyear fa fa-angle-double-left"></div>' +
  33. ' <div class="calendar-nav calendar-nextyear fa fa-angle-double-right"></div>' +
  34. ' <div class="calendar-title">' +
  35. ' <div class="year-range"></div>' +
  36. ' <div class="date-set">' +
  37. ' <span class="dateYear"></span>' +
  38. ' <span class="dateMonth"></span>' +
  39. ' </div>' +
  40. ' </div>' +
  41. '</div>' +
  42. '<div class="calendar-body">' +
  43. ' <ul class="calendar-menu date-month-list" />' +
  44. ' <ul class="calendar-menu date-year-list" />' +
  45. ' <div class="calendar-add-time">' +
  46. ' <ul class="add-time-list">' +
  47. ' <li><p>时</p><ol class="v-hh"></ol></li>' +
  48. ' <li><p>分</p><ol class="v-mm"></ol></li>' +
  49. ' <li><p>秒</p><ol class="v-ss"></ol></li>' +
  50. ' </ul>' +
  51. ' </div>' +
  52. '</div>'
  53. );
  54. var html = [];
  55. for (var i = 0; i < 24; i++) {//时
  56. if (i < 10) {
  57. html.push('<li>0' + i + '</li>');
  58. } else {
  59. html.push('<li>' + i + '</li>');
  60. }
  61. }
  62. $(target).find('.add-time-list .v-hh').html(html.join(''));
  63. html = [];
  64. for (var i = 0; i < 60; i++) {//分-秒
  65. if (i < 10) {
  66. html.push('<li>0' + i + '</li>');
  67. } else {
  68. html.push('<li>' + i + '</li>');
  69. }
  70. }
  71. $(target).find('.add-time-list .v-mm').html(html.join(''));
  72. $(target).find('.add-time-list .v-ss').html(html.join(''));
  73. $(target).bind('_resize', function (e, force) {
  74. if ($(this).hasClass('easyui-fluid') || force) {
  75. setSize(target);
  76. }
  77. return false;
  78. });
  79. }
  80. function restSelect(target) {
  81. $(target).find('.date-set').show();
  82. $(target).find('.year-range').hide();
  83. $(target).find('.calendar-menu').hide();
  84. $(target).find('.calendar-prevmonth').show();
  85. $(target).find('.calendar-nextmonth').show();
  86. $(target).find('.calendar-prevyear').show();
  87. $(target).find('.calendar-nextyear').show();
  88. }
  89. function resetTimePanel(target) {
  90. var opts = $.data(target, 'calendar').options;
  91. var $panel = $(target).find('.calendar-add-time');
  92. var $dom = $(target);
  93. if (!opts.showTime) {
  94. $panel.hide();
  95. $dom.removeClass('show-time');
  96. return;
  97. }
  98. $dom.addClass('show-time');
  99. $dom.find('.calendar-dtable').width(opts.width - 180);
  100. $dom.find('.calendar-menu').width(opts.width - 180);
  101. $panel.show();
  102. }
  103. //显示年份
  104. function showSelectYear(target) {
  105. var opts = $.data(target, 'calendar').options;
  106. restSelect(target);
  107. $(target).find('.calendar-prevmonth').hide();
  108. $(target).find('.calendar-nextmonth').hide();
  109. $(target).find('.date-set').hide();
  110. $(target).find('.year-range').show();
  111. var $ul = $(target).find('.date-year-list');
  112. $ul.show();
  113. var startYear = opts.startYear;
  114. $ul.empty();
  115. for (var i = 0; i < 15; i++) {
  116. $('<li class="calendar-year"></li>').attr('abbr', startYear + i).html(opts.yearFmt(startYear + i)).appendTo($ul);
  117. }
  118. $(target).find('.calendar-title .year-range').html(opts.yearFmt(startYear) + ' - ' + opts.yearFmt(startYear + 14));
  119. if ($ul.find('.date-year-this').attr('abbr') !== opts.year) {
  120. $ul.find('.date-year-this').removeClass('date-year-this');
  121. $ul.find('li[abbr="' + opts.year + '"]').addClass('date-year-this');
  122. }
  123. }
  124. //显示月份
  125. function showSelectMonth(target) {
  126. var opts = $.data(target, 'calendar').options;
  127. restSelect(target);
  128. $(target).find('.calendar-prevmonth').hide();
  129. $(target).find('.calendar-nextmonth').hide();
  130. var $ul = $(target).find('.date-month-list');
  131. $ul.show();
  132. if ($ul.is(':empty')) {
  133. $ul.empty();
  134. for (var i = 0; i < 12; i++) {
  135. $('<li class="calendar-month"></li>').attr('abbr', i + 1).html(opts.months[i]).appendTo($ul);
  136. }
  137. }
  138. if ($ul.find('.date-month-this').attr('abbr') !== opts.month) {
  139. $ul.find('.date-month-this').removeClass('date-month-this');
  140. $ul.find('li[abbr="' + opts.month + '"]').addClass('date-month-this');
  141. }
  142. }
  143. //显示时间
  144. function showSelectTime(target) {
  145. var opts = $.data(target, 'calendar').options;
  146. restSelect(target);
  147. $(target).find('.calendar-prevmonth').hide();
  148. $(target).find('.calendar-nextmonth').hide();
  149. $(target).find('.calendar-prevyear').hide();
  150. $(target).find('.calendar-nextyear').hide();
  151. var $ul = $(target).find('.date-time-list');
  152. $ul.show();
  153. }
  154. function bindEvents(target) {
  155. var opts = $.data(target, 'calendar').options;
  156. var menu = $(target).find('.calendar-menu');
  157. menu.find('.calendar-menu-year').unbind('.calendar').bind('keypress.calendar', function (e) {
  158. if (e.keyCode == 13) {
  159. setDate(true);
  160. }
  161. });
  162. //$('.add-time-list li ol li').click(function () {
  163. // $(this).addClass('date-this').siblings().removeClass('date-this');
  164. //});
  165. //$('.add-time-list li ol li:nth-child(1)').addClass('date-this');
  166. $(target)
  167. .unbind('.calendar')
  168. .bind('mouseover.calendar', function (e) {
  169. var t = toTarget(e.target);
  170. if (t.hasClass('calendar-nav') || t.hasClass('calendar-text') || (t.hasClass('calendar-day') && !t.hasClass('calendar-disabled'))) {
  171. t.addClass('calendar-nav-hover');
  172. }
  173. })
  174. .bind('mouseout.calendar', function (e) {
  175. var t = toTarget(e.target);
  176. if (t.hasClass('calendar-nav') || t.hasClass('calendar-text') || (t.hasClass('calendar-day') && !t.hasClass('calendar-disabled'))) {
  177. t.removeClass('calendar-nav-hover');
  178. }
  179. })
  180. .bind('click.v-mm', function (e) {
  181. console.log('hh', $(e.target).html());
  182. })
  183. .bind('click.calendar', function (e) {
  184. var t = toTarget(e.target);
  185. if (t.hasClass('dateYear')) {
  186. //点击年份
  187. opts.startYear = opts.year - 7;
  188. showSelectYear(target);
  189. } else if (t.hasClass('dateMonth')) {
  190. //点击月份
  191. showSelectMonth(target);
  192. //showSelectTime(target);
  193. } else if (t.hasClass('calendar-menu-next') || t.hasClass('calendar-nextyear')) {
  194. //下一年
  195. if ($(target).find('.date-year-list').is(':visible')) {
  196. //下15年
  197. opts.startYear += 15;
  198. showSelectYear(target);
  199. } else {
  200. showYear(1);
  201. }
  202. } else if (t.hasClass('calendar-menu-prev') || t.hasClass('calendar-prevyear')) {
  203. //上一年
  204. if ($(target).find('.date-year-list').is(':visible')) {
  205. //上15年
  206. opts.startYear -= 15;
  207. showSelectYear(target);
  208. } else {
  209. showYear(-1);
  210. }
  211. } else if (t.hasClass('calendar-prevmonth')) {
  212. //上个月
  213. showMonth(-1);
  214. } else if (t.hasClass('calendar-nextmonth')) {
  215. //下个月
  216. showMonth(1);
  217. } else if (t.hasClass('calendar-month')) {
  218. //月份
  219. $(target).find('.date-month-list').find('.date-month-this').removeClass('date-month-this');
  220. t.addClass('date-month-this');
  221. setDate(true);
  222. } else if (t.hasClass('calendar-year')) {
  223. //年份
  224. $(target).find('.date-year-list').find('.date-year-this').removeClass('date-year-this');
  225. t.addClass('date-year-this');
  226. setDate(true);
  227. } else if (t.hasClass('calendar-day')) {
  228. //选择了日期
  229. if (t.hasClass('calendar-disabled')) {
  230. return;
  231. }
  232. var oldValue = opts.current;
  233. t.closest('div.calendar-body').find('.calendar-selected').removeClass('calendar-selected');
  234. t.addClass('calendar-selected');
  235. var parts = t.attr('abbr').split(',');
  236. var y = parseInt(parts[0]);
  237. var m = parseInt(parts[1]);
  238. var d = parseInt(parts[2]);
  239. opts.current = new Date(y, m - 1, d);
  240. opts.onSelect.call(target, opts.current);
  241. if (!oldValue || oldValue.getTime() != opts.current.getTime()) {
  242. opts.onChange.call(target, opts.current, oldValue);
  243. }
  244. if (opts.year !== y || opts.month !== m) {
  245. opts.year = y;
  246. opts.month = m;
  247. show(target);
  248. }
  249. }
  250. });
  251. function toTarget(t) {
  252. var day = $(t).closest('.calendar-day');
  253. if (day.length) {
  254. return day;
  255. } else {
  256. return $(t);
  257. }
  258. }
  259. function setDate(hideMenu) {
  260. var opts = $.data(target, 'calendar').options;
  261. var $yearUl = $(target).find('.date-year-this');
  262. var $monthUl = $(target).find('.date-month-this');
  263. var year = $yearUl.attr('abbr') || opts.year;
  264. var month = $monthUl.attr('abbr') || opts.month;
  265. if (!isNaN(year)) {
  266. opts.year = parseInt(year);
  267. opts.month = parseInt(month);
  268. show(target);
  269. }
  270. if (hideMenu) {
  271. restSelect(target);
  272. }
  273. }
  274. function showYear(delta) {
  275. opts.year += delta;
  276. show(target);
  277. menu.find('.calendar-menu-year').val(opts.year);
  278. }
  279. function showMonth(delta) {
  280. opts.month += delta;
  281. if (opts.month > 12) {
  282. opts.year++;
  283. opts.month = 1;
  284. } else if (opts.month < 1) {
  285. opts.year--;
  286. opts.month = 12;
  287. }
  288. show(target);
  289. var $ul = menu.parent().find('.date-month-list');
  290. if ($ul.find('.date-month-this').attr('abbr') !== opts.month) {
  291. $ul.find('.date-month-this').removeClass('date-month-this');
  292. $ul.find('li[abbr="' + opts.month + '"]').addClass('date-month-this');
  293. }
  294. }
  295. }
  296. /**
  297. * get weeks data.
  298. */
  299. function getWeeks(target, year, month) {
  300. var opts = $.data(target, 'calendar').options;
  301. var dates = [];
  302. var lastDay = new Date(year, month, 0).getDate();
  303. for (var i = 1; i <= lastDay; i++) dates.push([year, month, i]);
  304. // group date by week
  305. var weeks = [], week = [];
  306. var memoDay = -1;
  307. while (dates.length > 0) {
  308. var date = dates.shift();
  309. week.push(date);
  310. var day = new Date(date[0], date[1] - 1, date[2]).getDay();
  311. if (memoDay == day) {
  312. day = 0;
  313. } else if (day == (opts.firstDay == 0 ? 7 : opts.firstDay) - 1) {
  314. weeks.push(week);
  315. week = [];
  316. }
  317. memoDay = day;
  318. }
  319. if (week.length) {
  320. weeks.push(week);
  321. }
  322. var firstWeek = weeks[0];
  323. if (firstWeek.length < 7) {
  324. while (firstWeek.length < 7) {
  325. var firstDate = firstWeek[0];
  326. var date = new Date(firstDate[0], firstDate[1] - 1, firstDate[2] - 1);
  327. firstWeek.unshift([date.getFullYear(), date.getMonth() + 1, date.getDate()]);
  328. }
  329. } else {
  330. var firstDate = firstWeek[0];
  331. var week = [];
  332. for (var i = 1; i <= 7; i++) {
  333. var date = new Date(firstDate[0], firstDate[1] - 1, firstDate[2] - i);
  334. week.unshift([date.getFullYear(), date.getMonth() + 1, date.getDate()]);
  335. }
  336. weeks.unshift(week);
  337. }
  338. var lastWeek = weeks[weeks.length - 1];
  339. while (lastWeek.length < 7) {
  340. var lastDate = lastWeek[lastWeek.length - 1];
  341. var date = new Date(lastDate[0], lastDate[1] - 1, lastDate[2] + 1);
  342. lastWeek.push([date.getFullYear(), date.getMonth() + 1, date.getDate()]);
  343. }
  344. if (weeks.length < 6) {
  345. var lastDate = lastWeek[lastWeek.length - 1];
  346. var week = [];
  347. for (var i = 1; i <= 7; i++) {
  348. var date = new Date(lastDate[0], lastDate[1] - 1, lastDate[2] + i);
  349. week.push([date.getFullYear(), date.getMonth() + 1, date.getDate()]);
  350. }
  351. weeks.push(week);
  352. }
  353. return weeks;
  354. }
  355. /**
  356. * show the calendar day.
  357. */
  358. function show(target) {
  359. var opts = $.data(target, 'calendar').options;
  360. if (opts.current && !opts.validator.call(target, opts.current)) {
  361. opts.current = null;
  362. }
  363. var now = new Date();
  364. var todayInfo = now.getFullYear() + ',' + (now.getMonth() + 1) + ',' + now.getDate();
  365. var currentInfo = opts.current ? (opts.current.getFullYear() + ',' + (opts.current.getMonth() + 1) + ',' + opts.current.getDate()) : '';
  366. // calulate the saturday and sunday index
  367. var saIndex = 6 - opts.firstDay;
  368. var suIndex = saIndex + 1;
  369. if (saIndex >= 7) saIndex -= 7;
  370. if (suIndex >= 7) suIndex -= 7;
  371. //$(target).find('.calendar-title span').html(opts.months[opts.month - 1] + ' ' + opts.year);
  372. $(target).find('.calendar-title .date-set .dateYear').html(opts.yearFmt(opts.year));
  373. $(target).find('.calendar-title .date-set .dateMonth').html(opts.months[opts.month - 1]);
  374. var body = $(target).find('div.calendar-body');
  375. body.children('table').remove();
  376. var data = ['<table class="calendar-dtable" cellspacing="0" cellpadding="0" border="0">'];
  377. data.push('<thead><tr>');
  378. if (opts.showWeek) {
  379. data.push('<th class="calendar-week">' + opts.weekNumberHeader + '</th>');
  380. }
  381. for (var i = opts.firstDay; i < opts.weeks.length; i++) {
  382. data.push('<th>' + opts.weeks[i] + '</th>');
  383. }
  384. for (var i = 0; i < opts.firstDay; i++) {
  385. data.push('<th>' + opts.weeks[i] + '</th>');
  386. }
  387. data.push('</tr></thead>');
  388. data.push('<tbody>');
  389. var weeks = getWeeks(target, opts.year, opts.month);
  390. for (var i = 0; i < weeks.length; i++) {
  391. var week = weeks[i];
  392. var cls = '';
  393. if (i == 0) {
  394. cls = 'calendar-first';
  395. }
  396. else if (i == weeks.length - 1) {
  397. cls = 'calendar-last';
  398. }
  399. data.push('<tr class="' + cls + '">');
  400. if (opts.showWeek) {
  401. var weekNumber = opts.getWeekNumber(new Date(week[0][0], parseInt(week[0][1]) - 1, week[0][2]));
  402. data.push('<td class="calendar-week">' + weekNumber + '</td>');
  403. }
  404. for (var j = 0; j < week.length; j++) {
  405. var day = week[j];
  406. var s = day[0] + ',' + day[1] + ',' + day[2];
  407. var dvalue = new Date(day[0], parseInt(day[1]) - 1, day[2]);
  408. var d = opts.formatter.call(target, dvalue);
  409. var css = opts.styler.call(target, dvalue);
  410. var classValue = '';
  411. var styleValue = '';
  412. if (typeof css == 'string') {
  413. styleValue = css;
  414. } else if (css) {
  415. classValue = css['class'] || '';
  416. styleValue = css['style'] || '';
  417. }
  418. var cls = 'calendar-day';
  419. if (!(opts.year == day[0] && opts.month == day[1])) {
  420. cls += ' calendar-other-month';
  421. }
  422. if (s == todayInfo) {
  423. cls += ' calendar-today';
  424. }
  425. if (s == currentInfo) {
  426. cls += ' calendar-selected';
  427. }
  428. if (j == saIndex) {
  429. cls += ' calendar-saturday';
  430. }
  431. else if (j == suIndex) {
  432. cls += ' calendar-sunday';
  433. }
  434. if (j == 0) {
  435. cls += ' calendar-first';
  436. }
  437. else if (j == week.length - 1) {
  438. cls += ' calendar-last';
  439. }
  440. cls += ' ' + classValue;
  441. if (!opts.validator.call(target, dvalue)) {
  442. cls += ' calendar-disabled';
  443. }
  444. data.push('<td class="' + cls + '" abbr="' + s + '" style="' + styleValue + '">' + d + '</td>');
  445. }
  446. data.push('</tr>');
  447. }
  448. data.push('</tbody>');
  449. data.push('</table>');
  450. body.append(data.join(''));
  451. body.children('table.calendar-dtable').prependTo(body);
  452. resetTimePanel(target);
  453. opts.onNavigate.call(target, opts.year, opts.month);
  454. }
  455. $.fn.calendar = function (options, param) {
  456. if (typeof options == 'string') {
  457. return $.fn.calendar.methods[options](this, param);
  458. }
  459. options = options || {};
  460. return this.each(function () {
  461. var state = $.data(this, 'calendar');
  462. if (state) {
  463. $.extend(state.options, options);
  464. } else {
  465. state = $.data(this, 'calendar', {
  466. options: $.extend({}, $.fn.calendar.defaults, $.fn.calendar.parseOptions(this), options)
  467. });
  468. init(this);
  469. }
  470. if (state.options.border == false) {
  471. $(this).addClass('calendar-noborder');
  472. }
  473. setSize(this);
  474. bindEvents(this);
  475. show(this);
  476. //$(this).find('.calendar-menu').hide(); // hide the calendar menu
  477. });
  478. };
  479. $.fn.calendar.methods = {
  480. options: function (jq) {
  481. return $.data(jq[0], 'calendar').options;
  482. },
  483. resize: function (jq, param) {
  484. return jq.each(function () {
  485. setSize(this, param);
  486. });
  487. },
  488. moveTo: function (jq, date) {
  489. return jq.each(function () {
  490. if (!date) {
  491. var now = new Date();
  492. $(this).calendar({
  493. year: now.getFullYear(),
  494. month: now.getMonth() + 1,
  495. current: date
  496. });
  497. return;
  498. }
  499. var opts = $(this).calendar('options');
  500. if (opts.validator.call(this, date)) {
  501. var oldValue = opts.current;
  502. $(this).calendar({
  503. year: date.getFullYear(),
  504. month: date.getMonth() + 1,
  505. current: date
  506. });
  507. if (!oldValue || oldValue.getTime() != date.getTime()) {
  508. opts.onChange.call(this, opts.current, oldValue);
  509. }
  510. }
  511. });
  512. }
  513. };
  514. $.fn.calendar.parseOptions = function (target) {
  515. var t = $(target);
  516. return $.extend({}, $.parser.parseOptions(target, [
  517. 'weekNumberHeader', {firstDay: 'number', fit: 'boolean', border: 'boolean', showWeek: 'boolean'}
  518. ]));
  519. };
  520. $.fn.calendar.defaults = {
  521. width: 272,
  522. height: 284,
  523. showTime: false,
  524. fit: false,
  525. border: true,
  526. showWeek: false,
  527. firstDay: 1,
  528. weeks: ['日', '一', '二', '三', '四', '五', '六'],
  529. months: ['一月', '二月', '三月', '四月', '五月', '六月', '七月', '八月', '九月', '十月', '十一月', '十二月'],
  530. //weeks: ['S', 'M', 'T', 'W', 'T', 'F', 'S'],
  531. //months: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'],
  532. year: new Date().getFullYear(),
  533. month: new Date().getMonth() + 1,
  534. current: (function () {
  535. var d = new Date();
  536. return new Date(d.getFullYear(), d.getMonth(), d.getDate());
  537. })(),
  538. weekNumberHeader: '周次',
  539. getWeekNumber: function (date) {
  540. var checkDate = new Date(date.getTime());
  541. checkDate.setDate(checkDate.getDate() + 4 - (checkDate.getDay() || 7));
  542. var time = checkDate.getTime();
  543. checkDate.setMonth(0);
  544. checkDate.setDate(1);
  545. return Math.floor(Math.round((time - checkDate) / 86400000) / 7) + 1;
  546. },
  547. yearFmt: function (v) {
  548. return v + '年';
  549. },
  550. formatter: function (date) {
  551. return date.getDate();
  552. },
  553. styler: function (date) {
  554. return '';
  555. },
  556. validator: function (date) {
  557. return true;
  558. },
  559. onSelect: function (date) {
  560. },
  561. onChange: function (newDate, oldDate) {
  562. },
  563. onNavigate: function (year, month) {
  564. }
  565. };
  566. })(jQuery);