如何写一个日历组件
众所周知,虽然
javascript
中关于时间的API有不少,我们可以通过方法单独的获取年、月、日、时、分、秒、毫秒…貌似很多,最近写了一个日历(以前写的,但写得很烂,最近优化一下),所以下面简单的记录一下如何写一个日历,列出了一些我在写日历过程中自己封装的一些方法
效果图
先来一张效果图,由于没有UI设计,所以就自己简单的设计了一个样式(好歹我也是设计专业的,虽然已不做设计很多年),虽然略丑,但重要的是功能!!!
思路
一个日历到底是怎样用代码生成的?其实观察一下现有的日历展现形式,可以很快的形成思路,就是:根据计算把日期号数对应到正确的星期几上,并按照顺序逐一输出。
以下是我的思路:
- 取得月份的天数
- 取得月份第一天是星期几
- 循环对应号数和星期几返回一个数组对象
- 返回数组对象的每一个子项至少包含:号数,星期几,然后根据情况添加:是否高亮,是否当前月,是否节日…等属性
方法封装
注意,为了保持方便调用javascript
的方法,以及保持输出结果符合实际,所有的方法都有如下约定:
- 在计算过程中
- 所有的关于月份都是0~11的数字
- 所有的关于星期都是0~6的数字
- 在输出的结果中
- 所有关于月份的输出默认都是1-12的数字
- 所有关于星期的输出默认都是1-7的数字
所以在向调用方法传递参数过程中,月份以及星期几统统都需要按照实际月份减一
获取月份天数
在javascript
中没有直接获取月份天数的方法,但是它提供了一个getDate
方法可以获取日期的某一天。那我们只需要获取月份的最后一天(下一个月的第0天)就可以得知这个月的天数:
1 | // year是要获取的年份,闰年不一样 |
获取星期几
1 | // year是要获取的年份 |
获取月份有几个星期
要计算月份包含几个星期,需要两个数据:月份天数和月份第一天是星期几,就能得到想要的结果
1 | // year是要获取的年份 |
循环生成月份对象
有了以上方法之后,就可以通过循环生成一个简单的月份对象了。
在这里需要注意,日历的排序有两种:
- 每一行以星期日开头
- 每一行以星期开头
1 | // year是要获取的年份 |
格式化时间
涉及到时间时,常常需要把时间格式进行转换,为了应对多中需求,所以自己封装了一个
1 | // 参数fmt必须 |
最后
附上这些方法的源码datepicker
基于vue实现的一个日历:
- demovue-datepicker
- 源码datePickerPanel.vue
当然这只是最简单的日历输出,思路也是超级简单(感觉有点Low),如果有大神愿意分享它的经验欢迎,来邮~