代码git地址:
https://github.com/BaoMax/Task/tree/gh-pages/%E6%97%A5%E5%8E%86
工作主要分为以下几个步骤:
- 公历转农历的计算
- 24节气和节日的判断
- 界面html+css
- 交互事件
公历转农历
根据网上的资料和别人的代码,公历转农历代码如下:
/**
* 农历1900-2100的润大小信息表
* @Array Of Property
* @return Hex
*
* 5位16进制数代表20位2进制
* 比如1900年代表0x04bd8二进制
* 0000 0100 1011 1101 1000
* 其中0-3位代表此年的闰月是大月还是小月,1为大月30天,0为小月29天
* 4-15位代表1-12月每个月是大月还是小月,1为大月30天,0为小月29天
* 16-20位代表此年是否闰月,如果全0代表不闰月,否则代表润的月份
*/
lunarInfo: [0x04bd8, 0x04ae0, 0x0a570, 0x054d5, 0x0d260, 0x0d950, 0x16554, 0x056a0, 0x09ad0, 0x055d2, //1900-1909
0x04ae0, 0x0a5b6, 0x0a4d0, 0x0d250, 0x1d255, 0x0b540, 0x0d6a0, 0x0ada2, 0x095b0, 0x14977, //1910-1919
0x04970, 0x0a4b0, 0x0b4b5, 0x06a50, 0x06d40, 0x1ab54, 0x02b60, 0x09570, 0x052f2, 0x04970, //1920-1929
0x06566, 0x0d4a0, 0x0ea50, 0x06e95, 0x05ad0, 0x02b60, 0x186e3, 0x092e0, 0x1c8d7, 0x0c950, //1930-1939
0x0d4a0, 0x1d8a6, 0x0b550, 0x056a0, 0x1a5b4, 0x025d0, 0x092d0, 0x0d2b2, 0x0a950, 0x0b557, //1940-1949
0x06ca0, 0x0b550, 0x15355, 0x04da0, 0x0a5b0, 0x14573, 0x052b0, 0x0a9a8, 0x0e950, 0x06aa0, //1950-1959
0x0aea6, 0x0ab50, 0x04b60, 0x0aae4, 0x0a570, 0x05260, 0x0f263, 0x0d950, 0x05b57, 0x056a0, //1960-1969
0x096d0, 0x04dd5, 0x04ad0, 0x0a4d0, 0x0d4d4, 0x0d250, 0x0d558, 0x0b540, 0x0b6a0, 0x195a6, //1970-1979
0x095b0, 0x049b0, 0x0a974, 0x0a4b0, 0x0b27a, 0x06a50, 0x06d40, 0x0af46, 0x0ab60, 0x09570, //1980-1989
0x04af5, 0x04970, 0x064b0, 0x074a3, 0x0ea50, 0x06b58, 0x055c0, 0x0ab60, 0x096d5, 0x092e0, //1990-1999
0x0c960, 0x0d954, 0x0d4a0, 0x0da50, 0x07552, 0x056a0, 0x0abb7, 0x025d0, 0x092d0, 0x0cab5, //2000-2009
0x0a950, 0x0b4a0, 0x0baa4, 0x0ad50, 0x055d9, 0x04ba0, 0x0a5b0, 0x15176, 0x052b0, 0x0a930, //2010-2019
0x07954, 0x06aa0, 0x0ad50, 0x05b52, 0x04b60, 0x0a6e6, 0x0a4e0, 0x0d260, 0x0ea65, 0x0d530, //2020-2029
0x05aa0, 0x076a3, 0x096d0, 0x04afb, 0x04ad0, 0x0a4d0, 0x1d0b6, 0x0d250, 0x0d520, 0x0dd45, //2030-2039
0x0b5a0, 0x056d0, 0x055b2, 0x049b0, 0x0a577, 0x0a4b0, 0x0aa50, 0x1b255, 0x06d20, 0x0ada0, //2040-2049
/**Add By [email protected]**/
0x14b63, 0x09370, 0x049f8, 0x04970, 0x064b0, 0x168a6, 0x0ea50, 0x06b20, 0x1a6c4, 0x0aae0, //2050-2059
0x0a2e0, 0x0d2e3, 0x0c960, 0x0d557, 0x0d4a0, 0x0da50, 0x05d55, 0x056a0, 0x0a6d0, 0x055d4, //2060-2069
0x052d0, 0x0a9b8, 0x0a950, 0x0b4a0, 0x0b6a6, 0x0ad50, 0x055a0, 0x0aba4, 0x0a5b0, 0x052b0, //2070-2079
0x0b273, 0x06930, 0x07337, 0x06aa0, 0x0ad50, 0x14b55, 0x04b60, 0x0a570, 0x054e4, 0x0d160, //2080-2089
0x0e968, 0x0d520, 0x0daa0, 0x16aa6, 0x056d0, 0x04ae0, 0x0a9d4, 0x0a2d0, 0x0d150, 0x0f252, //2090-2099
0x0d520
], //2100
根据上面的数据,我们就可以计算出农历年的年长,是否闰月,闰几月,以1900年1月31号正月初一为起点,来进行公历到农历的转换
/**
* 参数为农历日期
* @method 根据年份计算有多少天
* @param {int} 年
* @return {int} 天数
* **/
lunarYearLength: function(y) {
var sum = 12 * 29;
for (var i = 0x8000; i > 0x8; i = i >> 1) {
sum += (ChineseCalendar.lunarInfo[y - 1900] & i ? 1 : 0);
}
sum += ChineseCalendar.leapMonthLengths(y);
return sum;
},
/**
* 参数为农历日期
* @method 根据该年份的月份有多少天
* @param {int} 年
* @param {int} 月
* @return {int} 月天数
* **/
lunarMonthLength: function(y, m) {
return ChineseCalendar.lunarInfo[y - 1900] & (0x10000 >> m) ? 30 : 29;
},
/**
* 参数为农历日期
* @method 根据年份判断是否闰月
* @param {int} 年
* @return {boolean} 是否闰月
* **/
isLeapMonth: function(y) {
return ChineseCalendar.lunarInfo[y - 1900] & 0xf ? true : false;
},
/**
* 参数为农历日期
* @method 根据年份判断闰几月
* @param {int} 年
* @return {boolean} 闰几月
* **/
leapMonth: function(y) {
if (ChineseCalendar.isLeapMonth(y)) {
return ChineseCalendar.lunarInfo[y - 1900] & 0xf;
}
return 0;
},
/**
* 参数为农历日期
* @method 根据年份判断闰月天数,不闰月返回0
* @param {int} 年
* @return {int} 闰月天数
* **/
leapMonthLengths: function(y) {
if (ChineseCalendar.isLeapMonth(y)) {
return ChineseCalendar.lunarInfo[y - 1900] & 0x10000 ? 30 : 29;
}
return 0;
}
下面是公历=>农历的过程
date2lunar: function(date) {
date = date || new Date();
var result = {
year: date.getFullYear(),
month: date.getMonth() + 1,
day: date.getDate()
};
//date距离我们的base时间1900-1-31日有多少天
var offset = (Date.UTC(result.year, result.month - 1, result.day) - Date.UTC(1900, 0, 31)) / (60 * 60 * 24 * 1000),
temp = 0;
//干支记日
result.gzD = ChineseCalendar.day2GanZhe(offset);
//获取农历年
for (var i = 1900; i < 2101 && offset > 0; i++) {
temp = ChineseCalendar.lunarYearLength(i);
offset -= temp;
}
if (offset < 0) {
offset += temp;
i--;
}
result.lunarYear = i;
//获取农历月
var isLear = false,
lunarMonth = ChineseCalendar.leapMonth(result.lunarYear);
for (var i = 1; i < 13 && offset >= 0; i++) {
temp = ChineseCalendar.lunarMonthLength(result.lunarYear, i);
offset -= temp;
isLear = false;
if (i === lunarMonth) {
if (offset < 0) {
i--;
isLear = true;
} else {
temp = ChineseCalendar.leapMonthLengths(result.lunarYear);
offset -= temp;
}
}
}
if (offset < 0) {
if (i === lunarMonth && isLear) {
offset += temp;
isLear = false;
} else if (i === (lunarMonth + 1)) {
offset += temp;
isLear = true;
i--;
} else {
offset += temp;
i--;
}
}
result.lunarMonth = i;
result.lunarMonthChiness = ChineseCalendar.toLunarMonth(result.lunarMonth, isLear);
//获取农历日
result.lunarDay = offset + 1;
result.lunarDayChiness = ChineseCalendar.toLunarDay(result.lunarDay);
//获取生肖
result.animal = ChineseCalendar.getAnimal(result.year);
//获取星期
result.week = "星期" + ChineseCalendar.weekend[date.getDay()];
//获取星座
result.start = ChineseCalendar.getStar(result.month, result.day);
//获取干支年
result.gzY = ChineseCalendar.year2GanZhe(result.lunarYear);
result.isTody = ChineseCalendar.isTody(date);
//获取干支月
result.gzM = ChineseCalendar.month2GanZhe(result.lunarYear, result.lunarMonth);
//是否是节气,是返回节气名,不是返回false
result.isTerm = ChineseCalendar.isTerm(result.year, result.month, result.day);
if (result.isTerm) {
result.term = result.isTerm;
result.isTerm = true;
}
//是否是节日,是返回节日名,不是返回false
result.isFestive = ChineseCalendar.isFestive(result.year, result.month, result.day, result.lunarYear, result.lunarMonth, result.lunarDay);
if (result.isFestive) {
result.festive = result.isFestive;
result.isFestive = true;
}
return result;
}
干支纪年、干支记月、干支记日的方法和原
/* @{array} 天干对应表
* **/
Gan: ["甲", "乙", "丙", "丁", "戊", "己", "庚", "辛", "壬", "癸"],
/**
* @{array} 地支对应表
* **/
Zhe: ["子", "丑", "寅", "卯", "辰", "巳", "午", "未", "申", "酉", "戌", "亥"],
/**
* 参数为农历日期
* @method 根据年份判断天干地支年
* @param {int} 年
* @return {string} 天干地支
*
* 干支纪年方法:
* 十天干:甲、乙、丙、丁、戊、己、庚、辛、壬、癸。
* 十二地支:子、丑、寅、卯、辰、巳、午、未、申、酉、戌、亥。
* 干支还是阴阳之分:
* 甲、丙、戊、庚、壬为阳干,
* 乙、丁、己、辛、癸为阴干;
* 子、寅、辰、午、申、戌为阳支,
* 丑、卯、巳、未、酉、亥为阴支。
* 以一个天干和一个地支相配,排列起来,
* 天干在前,地支在后,
* 天干由甲起,地支由子起,
* 阳干配阳支,阴干配阴支(阳干不配阴支,阴干不配阳支),共有六十个组合,称为“六十甲子”。
* **/
year2GanZhe: function(y) {
var gan = (y - 3) % 10;
if (gan === 0) {
gan = 10;
}
var zhe = (y - 3) % 12;
if (zhe === 0) {
zhe = 12;
}
return ChineseCalendar.Gan[gan - 1] + ChineseCalendar.Zhe[zhe - 1];
},
/**
* 参数为农历日期
* @method 根据年份,月份判断天干地支月
* @param {int} 年
* @param {int} 月
* @return {string} 天干地支
*
*
* 干支记月的方法:
* 如果年干为甲、己,正月从丙寅开始,其余月按照六十甲子计算
* 如果年干为乙、庚,正月从戊寅开始,其余月按照六十甲子计算
* 如果年干为丙、辛,正月从庚寅开始,其余月按照六十甲子计算
* 如果年干为丁、壬,正月从壬寅开始,其余月按照六十甲子计算
* 如果年干为戊、癸,正月从甲寅开始,其余月按照六十甲子计算
* **/
month2GanZhe: function(y, m) {
var ganY = (y - 3) % 10;
if (ganY === 0) {
ganY = 10;
}
var month = ((ganY - 1) % 5) * 12 + m + 1;
return ChineseCalendar.Gan[month % 10] + ChineseCalendar.Zhe[month % 12];
},
/**
* 参数为农历日期
* @method 按照1900.1.31为甲辰日做起点,根据偏移日,判断天干地支日
* @param {int} 偏移
* @return {string} 天干地支
*
*
* 干支记月的方法:
* 如果年干为甲、己,正月从丙寅开始,其余月按照六十甲子计算
* 如果年干为乙、庚,正月从戊寅开始,其余月按照六十甲子计算
* 如果年干为丙、辛,正月从庚寅开始,其余月按照六十甲子计算
* 如果年干为丁、壬,正月从壬寅开始,其余月按照六十甲子计算
* 如果年干为戊、癸,正月从甲寅开始,其余月按照六十甲子计算
* **/
day2GanZhe: function(offset) {
var temp = 40 + offset;
return ChineseCalendar.Gan[temp % 10] + ChineseCalendar.Zhe[temp % 12];
}
24节气的计算
24节气是根据太阳的公转计算的,跟公历有关,原理如下:
/**
* @{array} 24节气对应表
* **/
solarTerm: ["小寒", "大寒", "立春", "雨水", "惊蛰", "春分",
"清明", "谷雨","立夏", "小满", "芒种", "夏至",
"小暑", "大暑", "立秋", "处暑", "白露", "秋分",
"寒露", "霜降", "立冬", "小雪", "大雪", "冬至"],
/**
* 参数均为公历日期
* @method 某年的第n个节气为几日(从0小寒起算)
* @param {int} 年份
* @param {int} 第几个节气
* @return {date} 公历日期
* **/
sTerm: function(y, n) {
//ms代表y年第n个节气到1900年第一个节气的毫秒数
var ms = 31556925974.7 * (y - 1900) + ChineseCalendar.sTermInfo[n] * 60000;
//1900年一月六日两点五分是正小寒点,此点到1970年1.1 00:00:00的毫秒数
var base = Date.UTC(1900, 0, 6, 2, 5);
//对应的公历日期
var date = new Date(ms + base);
return date;
},
/**
* 参数为公历日期
* @method 判断是不是节气
* @param {int} 年份
* @param {int} 月份
* @param {date} 日子
* @return {boolean} 是否是节气,是返回节气名,不是返回false
* **/
isTerm: function(y, m, d) {
var date = new Date(y, m - 1, d);
var n = (m - 1) * 2;
var dateTerm = ChineseCalendar.sTerm(y, n);
if (ChineseCalendar.isSameDay(dateTerm, date)) {
return ChineseCalendar.solarTerm[n];
}
n = n + 1;
dateTerm = ChineseCalendar.sTerm(y, n);
if (ChineseCalendar.isSameDay(dateTerm, date)) {
return ChineseCalendar.solarTerm[n];
}
return false;
}
节日计算,统计一下常见的节日,然后按着一定的编码规则编码,然后解码
/**
* 前两位代表月份,第三位代表第几个星期,第四位代表星期几,其中0代表星期7
* **/
wFestive: [
'0520-母亲节',
'0630-父亲节',
'1144-感恩节'
],
/**公历节日
* 前两位代表月份,后两位代表日期
**/
cFestive: [
'0101-元旦',
'0214-情人节',
'0305#1963-雷锋日',
'0308-妇女节',
'0312-植树节',
'0401-愚人节',
'0501-劳动日',
'0504-青年节',
'0512-护士节',
'0601#1925-儿童节',
'0701#1997-建党节',
'0801#1927-建军节',
'0910-教师节',
'1001#1949-国庆节',
'1224-平安夜',
'1225-圣诞节',
],
/**农历节日
* 前两位代表月份,后两位代表日期
**/
lFestive: [
'0101-春节',
'0115-元宵节',
'0505-端午节',
'0707-七夕节',
'0715-中元节',
'0815-中秋节',
'0909-重阳节',
'1208-腊八节',
'1224-小年',
'0100-除夕'
],
/**
* 参数为日期
* @method 判断是不是节日
* @param {int} 公历年份
* @param {int} 公历月份
* @param {int} 公历日子
* @param {int} 农历年份
* @param {int} 农历月份
* @param {int} 农历日子
* @return {boolean} 是否是节日,是返回节日名,不是返回false
* **/
isFestive: function(y, m, d, ly, lm, ld) {
var lstr = lm > 9 ? (lm + '') : '0' + lm;
lstr += ld > 9 ? (ld + '') : '0' + ld;
var cstr = m > 9 ? (m + '') : '0' + m;
cstr += d > 9 ? (d + '') : '0' + d;
//先算农历节日
var festive = ChineseCalendar.lFestive;
for (var i = 0, l = festive.length; i < l; i += 1) {
var test = festive[i].match(/(\d*)-(.*)/);
if (test[1] === lstr) {
return test[2];
}
}
//算阳历
festive = ChineseCalendar.cFestive;
for (var i = 0, l = festive.length; i < l; i += 1) {
var test = festive[i].match(/(\d*#?\d*)-(.*)/);
var year = test[1].split('#');
if (year[1] && y > year[1]) {
if (year[0] === cstr) {
return test[2];
}
} else if (!year[1]) {
if (test[1] === cstr) {
return test[2];
}
}
}
//算星期
festive = ChineseCalendar.wFestive;
var date = new Date(y, m - 1, d);
for (var i = 0, l = festive.length; i < l; i += 1) {
var test = festive[i].match(/(\d*)-(.*)/);
var month = test[1].substring(0, 2);
var day = test[1].substring(3, 4);
var nWeek = test[1].substring(2, 3);
var str = m > 9 ? (m + '') : '0' + m;
if (str === month && date.getDay() === parseInt(day)) {
var temp = ChineseCalendar.theNoWeek(y, m, d);
if (temp === parseInt(nWeek)) {
return test[2];
}
}
}
return false;
},
/**
* 参数为日期
* @method 判断是第几周
* @param {int} 公历年份
* @param {int} 公历月份
* @param {int} 公历日子
* @return {init} 返回第几周
* **/
theNoWeek: function(y, m, d) {
var first = new Date(y, m - 1, d);
first.setDate(1);
var date = new Date(y, m - 1, d);
var day = first.getDay(),
current = date.getDate();
if (day === 0) {
day = 7;
}
var head = 1,
tail = 8 - day,
result = 1;
while (1) {
if (current >= head && current <= tail) {
return result;
}
result += 1;
head = tail + 1;
tail = head + 6;
}
}
界面略
交互事件
主要分为年份、月份、日子的选择事件,以及返回今天。
主要通过把每个月的日期转换成一个数组,数组存
{
day: date.getDate(),//公历日子
lunarDay: ChineseCalendar.lunarTime(date),//农历日子
state: 'selected weekDay'//类型
}
其中类型分为:selected:当前天;weekDay:周六日;normal:平常
然后按照数组进行日历的渲染