本文发布在个人博客 如何写一个日历组件
许可协议: 署名-非商业性使用-禁止演绎 4.0 国际 转载请保留原文连接及做者。
众所周知,虽然
javascript
中关于时间的API有很多,咱们能够经过方法单独的获取年、月、日、时、分、秒、毫秒...貌似不少,最近写了一个日历(之前写的,但写得很烂,最近优化一下),因此下面简单的记录一下如何写一个日历,列出了一些我在写日历过程当中本身封装的一些方法
先来一张效果图,因为没有UI设计,因此就本身简单的设计了一个样式(好歹我也是设计专业的,虽然已不作设计不少年),虽然略丑,但重要的是功能!!!javascript
一个日历究竟是怎样用代码生成的?其实观察一下现有的日历展示形式,能够很快的造成思路,就是:根据计算把日期号数对应到正确的星期几上,并按照顺序逐一输出。
如下是个人思路:vue
循环对应号数和星期几返回一个数组对象java
注意,为了保持方便调用javascript
的方法,以及保持输出结果符合实际,全部的方法都有以下约定:git
在计算过程当中github
在输出的结果中数组
因此在向调用方法传递参数过程当中,月份以及星期几通通都须要按照实际月份减一优化
在javascript
中没有直接获取月份天数的方法,可是它提供了一个getDate
方法能够获取日期的某一天。那咱们只须要获取月份的最后一天(下一个月的第0天)就能够得知这个月的天数:ui
// year是要获取的年份,闰年不同 // month是要获取的月份 // 返回当前月天数 function getMonthDays(year, month){ return new Date(year, month, 0).getDate(); } getMonthDays(2016,2) //29 getMonthDays(2017,2) //28
// year是要获取的年份 // month是要获取的月份 // 返回数字几则是星期几 function getWeekday(year, month, day){ return new Date(year, month-1, day).getDay(); } getWeekday(2016,10,9) //输出4,表示2016年11月9是星期4 getWeekday(2017,10,9) //输出5,表示2017年11月9是星期5
要计算月份包含几个星期,须要两个数据:月份天数和月份第一天是星期几,就能获得想要的结果this
// year是要获取的年份 // month是要获取的月份 // 返回当前月包含几个星期 function getweeksInMonth(year, month){ var days = getMonthDays(year, month); var FirstDayWeekday = getWeekday(year, month, 1); return Math.ceil((days + FirstDayWeekday) / 7); }
有了以上方法以后,就能够经过循环生成一个简单的月份对象了。
在这里须要注意,日历的排序有两种:spa
// year是要获取的年份 // month是要获取的月份 // day天,用来判断是不是当前天 // type代表要星期几开头,0为星期一开头,1为星期日开头,默认为0 // 返回当前月包含几个星期 const WEEKTABLE = [{ cn: ['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六'], cns: ['日', '一', '二', '三', '四', '五', '六'], en: ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'] },{ cn: ['星期一', '星期二', '星期三', '星期四', '星期五', '星期六', '星期日'], cns: ['一', '二', '三', '四', '五', '六', '日'], en: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'] }] getMonthDaysArray(year, month, day, type) { if (typeof day === 'undefined' && year === YEAR && month === MONTH) day = DAY; var dayArrays = []; var days = this.getMonthDays(year, month), preDays = this.getMonthDays(year, month - 1); var thisMonthFirstDayInWeek = this.getWeekday(year, month, 1), thisMonthLastDayInWeek = this.getWeekday(year, month, days); type = !type || type !== 1 ? 0 : 1; //上月在当月日历面板中的排列 for (var i = 0; i < thisMonthFirstDayInWeek; i++) { dayArrays.push({ dayNum: (preDays - thisMonthFirstDayInWeek + i + 1), weekDay: WEEKTABLE[type].cn[i] }) } //当月日历面板中的排列 for (var i = 1; i <= days; i++) { var weekDayFlag = (thisMonthFirstDayInWeek + i - 1) % 7 dayArrays.push({ dayNum: i, weekDay: WEEKTABLE[type].cn[weekDayFlag], selected: i === +day, isThisMonth: true }) }; //下月在当月日历面板中的排列 for (var i = 1; i <= (6 - thisMonthLastDayInWeek); i++) { var weekDayFlag = (thisMonthFirstDayInWeek + days + i - 1) % 7 dayArrays.push({ dayNum: i, weekDay: WEEKTABLE[type].cn[weekDayFlag] }) }; return dayArrays; }
涉及到时间时,经常须要把时间格式进行转换,为了应对多中需求,因此本身封装了一个
// 参数fmt必须 // date参数没必要须,容许字符串和时间对象,不传或者传没法转换成合法时间对象的字符串则默认当前时间, // 年(YYYY/yyyy)固定四个占位符 // 月(M)、日(d)、小时(h)、分(m)、秒(s)能够用 1-2个占位符,严格区分大小写, // 毫秒(ms/mss)最多三个占位符,分别对应56,056这种类型 // 例子: // (Format("yyyy-MM-dd hh:mm:ss:ms") ==> 2006-07-02 08:09:04:23 // (Format("yyyy-MM-dd hh:mm:ss:mss") ==> 2006-07-02 08:09:04:023 // (Format("yyyy-M-d h:m:s:ms") ==> 2006-7-2 8:9:4.180 function formate(fmt, date){ date = new Date(date).toString() === 'Invalid Date' ? new Date() : new Date(date); var _rules = [{ rule: '[yY]{4}', value: _date.getFullYear() }, { rule: 'M+', value: _date.getMonth() + 1 }, { rule: '[dD]+', value: _date.getDate() }, { rule: 'h+', value: _date.getHours() }, { rule: 'm+', value: _date.getMinutes() }, { rule: 's+', value: _date.getSeconds() }, { rule: 'ms{1,2}', value: _date.getMilliseconds() }]; _rules.forEach(function (_r){ const rule = _r.rule, val = _r.value; fmt = fmt.replace(new RegExp(rule), function ($1) { const rLen = val.toString().length, fLen = $1.length; return (fLen !== 2 || rLen >= fLen) ? val : ['00', val].join().substr(rLen); }); }); return fmt; } //调用: var time1 = formate("YYYY/MM/DD hh:mm:ss", new Date()); //2017/11/2 11:09:20 var time2 = formate("YYYY-MM-DD", time1); //2017-11-2 var time3 = formate("MM-DD-YYYY", time2); //11-2-2017
附上这些方法的源码datepicker
基于vue实现的一个日历:
固然这只是最简单的日历输出,思路也是超级简单(感受有点Low),若是有大神愿意分享它的经验欢迎,来邮~