本文发布在我的博客如何写一个日历组件

许可协议: 署名-非商业性使用-禁止演绎 4.0 国际 转载请保留原文链接及作者。众所周知,虽然javascript中关于时间的API有不少,我们可以通过方法单独的获取年、月、日、时、分、秒、毫秒...貌似很多,最近写了一个日历(以前写的,但写得很烂,最近优化一下),所以下面简单的记录一下如何写一个日历,列出了一些我在写日历过程中自己封装的一些方法

效果图

先来一张效果图,由于没有UI设计,所以就自己简单的设计了一个样式,样式不重要,重要的是功能。

思路

一个日历到底是怎样用代码生成的?其实观察一下现有的日历展现形式,可以很快的形成思路,就是:根据计算把日期号数对应到正确的星期几上,并按照顺序逐一输出。

以下是我的思路:取得月份的天数

取得月份第一天是星期几

循环对应号数和星期几返回一个数组对象返回数组对象的每一个子项至少包含:号数,星期几,然后根据情况添加:是否高亮,是否当前月,是否节日...等属性

方法封装

注意,为了保持方便调用javascript的方法,以及保持输出结果符合实际,所有的方法都有如下约定:在计算过程中所有的关于月份都是0~11的数字

所有的关于星期都是0~6的数字

在输出的结果中所有关于月份的输出默认都是1-12的数字

所有关于星期的输出默认都是1-7的数字

所以在向调用方法传递参数过程中,月份以及星期几统统都需要按照实际月份减一

获取月份天数

在javascript中没有直接获取月份天数的方法,但是它提供了一个getDate方法可以获取日期的某一天。那我们只需要获取月份的最后一天(下一个月的第0天)就可以得知这个月的天数:// 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

获取月份有几个星期

要计算月份包含几个星期,需要两个数据:月份天数和月份第一天是星期几,就能得到想要的结果// year是要获取的年份

// month是要获取的月份

// 返回当前月包含几个星期

function getweeksInMonth(year, month){

var days = getMonthDays(year, month);

var FirstDayWeekday = getWeekday(year, month, 1);

return Math.ceil((days + FirstDayWeekday) / 7);

}

循环生成月份对象

有了以上方法之后,就可以通过循环生成一个简单的月份对象了。

在这里需要注意,日历的排序有两种:每一行以星期日开头

每一行以星期开头// 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),如果有大神愿意分享它的经验欢迎,来邮~

php日历如何写,如何写一个好看的实用的日历相关推荐

  1. 写一个构建复杂数据的日历组件 Kalendar

    需求:我们经常会在一些旅游.订票.酒店的页面中看到一些日历,这些日历有公历.农历.节假日.非节假日(调休)的标注,同时还有产品业务上的信息,比如 票价.余票等等.那现在开始造造轮子了,首先这个组件应该 ...

  2. Java 日历的制作 心得 写给自己

    之前已经跟着老师做过一次这个日历,但是时间一久便又拿出来自己再复习一遍.果然不出所料,已经做不出来了,而且因为在学习的时候使用的是Myeclipse,其中话中操作是由软件自己操作的,每写出一句代码软件 ...

  3. thonny python ide_学习用 Thonny 写代码:一个面向初学者的Python IDE

    原标题:学习用 Thonny 写代码:一个面向初学者的Python IDE 编译自: https://fedoramagazine.org/learn-code-thonny-python-ide-b ...

  4. 【重难点】【JUC 03】怎么实现一个线程安全的队列、手写模拟实现一个阻塞队列

    [重难点][JUC 03]怎么实现一个线程安全的队列.手写模拟实现一个阻塞队列 文章目录 [重难点][JUC 03]怎么实现一个线程安全的队列.手写模拟实现一个阻塞队列 一.怎么实现一个线程安全的队列 ...

  5. 写公众号一个月关注量破900,聊聊我的感受

    写公众号一个月关注量破900,聊聊我的感受 前言 写公众号一个多月了,于2019年12月7日零点关注量定格在900人. 最近有好多公众号号主好奇,在微信上问我是怎么做到的,他们说,他们有的做公众号几年 ...

  6. Laravel php 框架的使用写出第一个hello world,Laravel 入门配置

    Laravel 第一次使用,如何写出第一个hello world Laravel php 框架第一次接触,遇到一些困难,在这里记录一下,有需要的童鞋可以看一下 从github上下载下来最新版,地址如下 ...

  7. python写篮球游戏_程序员们把“蔡徐坤打篮球”写成了一个游戏

    原标题:程序员们把"蔡徐坤打篮球"写成了一个游戏 github一直是一个人才辈出的地方 从"杨超越杯编程大赛" 到"清华课程笔记" 无数令人 ...

  8. 用CIL写程序:定义一个叫“慕容小匹夫”的类

    前文回顾: <用CIL写程序:你好,沃尔德> <用CIL写程序:写个函数做加法> 前言: 今天是乙未羊年的第一天,小匹夫先在这里给各位看官拜个年了.不知道各位看官是否和匹夫一样 ...

  9. .验证哥德巴赫猜想,哥德巴赫猜想的内容是:任何一个大于2的偶数都能写成写成两个素数和的形式。 设计一个函数 int isPrime(int n)判断n是否为素数,如果n是素数,函数返回值1,否则返

    .验证哥德巴赫猜想,哥德巴赫猜想的内容是:任何一个大于2的偶数都能写成写成两个素数和的形式. 设计一个函数 int isPrime(int n)判断n是否为素数,如果n是素数,函数返回值1,否则返回0 ...

最新文章

  1. 2022-2028年中国酒行业投资分析及前景预测报告
  2. GCC生成的汇编代码
  3. PyTorch Trick集锦
  4. 学用 TStringGrid [7] - ColWidths[0]、RowHeights[0]、GridLineWidth
  5. Visual Studio 使用 Web Deploy 发布远程站点
  6. 【Python】解决Django Admin管理界面样式表(CSS Style)丢失问题
  7. SpringMVC Ueditor1.4.3 未找到上传数据
  8. Python绘制三维散点图
  9. 使用dotnet Cli向nuget发布包
  10. python程序的name属性作用是什么_python __name__ == ‘__main__’详细解释+例子
  11. linux input输入子系统分析《四》:input子系统整体流程全面分析
  12. 我看Slashdot
  13. 灰狼优化算法与影响力最大化问题
  14. CPU性能的三大主要参数
  15. 富文本TinyMCE
  16. 【AAAI 2021】全部接受论文列表(六)
  17. Python图像处理【5】图像扭曲与逆扭曲详解
  18. 与Lucene 4.10配合的中文分词比较
  19. (连载)Android系统源码分析--Android系统启动流程之Linux内核
  20. HTML <kbd> 标签

热门文章

  1. 求循环小数的表示以及循环节长度
  2. Syncfusion教程:在Xamarin.Forms中创建数据输入表单 (3)
  3. ScrollView 嵌套 ListView 只显示第一行的源码分析
  4. solr5.5.4 添加mysql数据,实现同步更新
  5. 阿里云首席架构师唐洪:拥抱开源的云端更具生命力
  6. windows 2012 exchange server 2013 搭建
  7. POJ-1837 Balance
  8. puppet中master和agent之间实现通信
  9. 邮件发送退信分析大全/SMTP error, RCPT TO: 550 Relay Deny
  10. c语言编程游戏界面,震惊!!!一个关于c语言图形化界面编程的小游戏-Go语言中文社区...