|
@@ -1,77 +1,5 @@
|
|
|
import dayjs from "dayjs";
|
|
|
-
|
|
|
-interface LunarDateDetail {
|
|
|
- solarDate: string; // 阳历日期
|
|
|
- lunarYear: number; // 农历年份
|
|
|
- lunarMon: number; // 农历月份
|
|
|
- lunarDay: number; // 农历日期
|
|
|
- isLeap: boolean; // 是否闰月
|
|
|
- yearCyl: string; // 年柱,天干地支表示的年份
|
|
|
- monCyl: string; // 月柱,天干地支表示的月份
|
|
|
- dayCyl: string; // 日柱,天干地支表示的日期
|
|
|
- zodiac: string; // 生肖
|
|
|
- lunarYearCN: string; // 农历年份的中文写法
|
|
|
- lunarMonCN: string; // 农历月份的中文写法
|
|
|
- lunarDayCN: string; // 农历日期的中文写法
|
|
|
-}
|
|
|
-/**
|
|
|
- * LUNAR_INFO 数组值的计算原理:
|
|
|
- *
|
|
|
- * 每个值使用 16 进制表示,包括以下部分:
|
|
|
- * 1. 前 4 位:表示闰月的月份,如果没有闰月为 0。
|
|
|
- * 2. 中间 12 位:表示 1 到 12 月的大小月,1 为大月(30 天),0 为小月(29 天)。
|
|
|
- * 3. 后 4 位:表示闰月的天数,如果没有闰月为 0。
|
|
|
- *
|
|
|
- * 以 `0x04bd8` 为例:
|
|
|
- * - `0x` 表示这是一个 16 进制数。
|
|
|
- * - `04bd8` 是具体的 16 进制值。
|
|
|
- *
|
|
|
- * 转换为二进制后,`04bd8` 为 `0000 0100 1011 1101 1000`:
|
|
|
- * 1. 前 4 位 `0000`:表示该年份没有闰月(若有闰月,该值为闰月的月份)。
|
|
|
- * 2. 中间 12 位 `0100 1011 1101`:从左到右分别表示 1 到 12 月的天数。`1` 表示大月(30 天),`0` 表示小月(29 天)。
|
|
|
- * - `0`(1月):小月(29 天)
|
|
|
- * - `1`(2月):大月(30 天)
|
|
|
- * - `0`(3月):小月(29 天)
|
|
|
- * - `1`(4月):大月(30 天)
|
|
|
- * - `0`(5月):小月(29 天)
|
|
|
- * - `1`(6月):大月(30 天)
|
|
|
- * - `1`(7月):大月(30 天)
|
|
|
- * - `1`(8月):大月(30 天)
|
|
|
- * - `1`(9月):大月(30 天)
|
|
|
- * - `1`(10月):大月(30 天)
|
|
|
- * - `0`(11月):小月(29 天)
|
|
|
- * - `0`(12月):小月(29 天)
|
|
|
- * 3. 后 4 位 `1000`:表示闰月的天数。如果前 4 位为 `0000`(即没有闰月),则这一部分不使用。
|
|
|
- */
|
|
|
-const LUNAR_INFO: number[] = [
|
|
|
- 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
|
|
|
- 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
|
|
|
-];
|
|
|
-
|
|
|
-const CHINESE_NUMBER = ["零", "一", "二", "三", "四", "五", "六", "七", "八", "九"];
|
|
|
-const NUMBER_MONTH: string[] = ["正", "二", "三", "四", "五", "六", "七", "八", "九", "十", "冬", "腊"];
|
|
|
-const NUMBER_1: string[] = ["甲", "乙", "丙", "丁", "戊", "己", "庚", "辛", "壬", "癸"];
|
|
|
-const NUMBER_2: string[] = ["子", "丑", "寅", "卯", "辰", "巳", "午", "未", "申", "酉", "戌", "亥"];
|
|
|
-const ZODIACS: string[] = ["鼠", "牛", "虎", "兔", "龙", "蛇", "马", "羊", "猴", "鸡", "狗", "猪"];
|
|
|
+import { type LunarDateDetail, LUNAR_INFO, CHINESE_NUMBER, NUMBER_MONTH, NUMBER_1, NUMBER_2, ZODIACS } from './constants'
|
|
|
|
|
|
/**
|
|
|
* 获取指定农历年的天数
|
|
@@ -156,7 +84,7 @@ const getLunarYearText = (lunarYear: number): string => {
|
|
|
* @param endYear 结束农历年份
|
|
|
* @returns 农历年份列表
|
|
|
*/
|
|
|
-const getYears = (startYear: number, endYear: number) => {
|
|
|
+const getLunarYears = (startYear: number, endYear: number) => {
|
|
|
const years = [];
|
|
|
for (let i = startYear; i <= endYear; i++) {
|
|
|
years.push({
|
|
@@ -169,11 +97,11 @@ const getYears = (startYear: number, endYear: number) => {
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * 获取指定年份的所有农历月份
|
|
|
+ * 获取指定阳历年份的闰月
|
|
|
* @param year 年份
|
|
|
* @returns 农历闰月月份
|
|
|
*/
|
|
|
-export const getYearLeapMonth = (year: number) => {
|
|
|
+const getYearLeapMonth = (year: number) => {
|
|
|
const leap = yearLeapMonth(year)
|
|
|
return {
|
|
|
year,
|
|
@@ -250,7 +178,7 @@ export const getLunarDate = (date: dayjs.ConfigType): LunarDateDetail => {
|
|
|
lunarDate[2] = offset + 1; // 农历日期
|
|
|
|
|
|
return {
|
|
|
- solarDate: objDate.format('YYYY-MM-DD'), // 公历日期
|
|
|
+ date: objDate.format('YYYY-MM-DD'), // 公历日期
|
|
|
lunarYear: lunarDate[0], // 农历年份
|
|
|
lunarMon: lunarDate[1] + 1, // 农历月份
|
|
|
lunarDay: lunarDate[2], // 农历日期
|
|
@@ -265,22 +193,76 @@ export const getLunarDate = (date: dayjs.ConfigType): LunarDateDetail => {
|
|
|
};
|
|
|
}
|
|
|
|
|
|
-console.log(Date.now())
|
|
|
-// 2057-9-28
|
|
|
-console.log(getLunarDate('2057-09-28'))
|
|
|
-// 农历丁丑(牛)年八月三十
|
|
|
+/**
|
|
|
+ * 获取范围内所有日期的农历信息
|
|
|
+ * @param startDate 开始日期
|
|
|
+ * @param endDate 结束日期
|
|
|
+ * @returns 范围内所有日期的农历信息
|
|
|
+ */
|
|
|
+export const getLunarDatesInRange = (startDate: dayjs.ConfigType, endDate: dayjs.ConfigType): LunarDateDetail[] => {
|
|
|
+ const start = dayjs(startDate);
|
|
|
+ const end = dayjs(endDate);
|
|
|
+ const lunarDates: LunarDateDetail[] = [];
|
|
|
+
|
|
|
+ for (let date = start; date.isBefore(end) || date.isSame(end, 'day'); date = date.add(1, 'day')) {
|
|
|
+ lunarDates.push(getLunarDate(date));
|
|
|
+ }
|
|
|
+
|
|
|
+ return lunarDates;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * 根据阴历日期查询阳历日期
|
|
|
+ * @param lunarDate 农历日期
|
|
|
+ * @param isLeapMonth 是否闰月
|
|
|
+ * @returns 阳历日期
|
|
|
+ */
|
|
|
+export const getSolarDateFromLunar = (lunarDate: dayjs.ConfigType): {
|
|
|
+ date: string;
|
|
|
+ leapMonthDate?: string;
|
|
|
+} => {
|
|
|
+ const date = dayjs(lunarDate);
|
|
|
+ const lunarYear = date.year();
|
|
|
+ const lunarMonth = date.month() + 1;
|
|
|
+ const lunarDay = date.date();
|
|
|
+
|
|
|
+ // 计算从农历年开始到指定农历日期的总天数
|
|
|
+ let offset = 0;
|
|
|
+ for (let i = 1900; i < lunarYear; i++) {
|
|
|
+ offset += lunarYearDays(i);
|
|
|
+ }
|
|
|
+
|
|
|
+ let leapMonth = yearLeapMonth(lunarYear);
|
|
|
+ for (let i = 1; i < lunarMonth; i++) {
|
|
|
+ offset += monthDays(lunarYear, i);
|
|
|
+ if (i === leapMonth) {
|
|
|
+ offset += yearLeapDays(lunarYear);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ offset += lunarDay - 1;
|
|
|
|
|
|
-// 2097-8-7
|
|
|
-console.log(getLunarDate('2097-08-07'))
|
|
|
-// 农历丁巳(蛇)年七月一
|
|
|
+ // 通过在基准日期上添加偏移量来获取阳历日期
|
|
|
+ const baseDate = dayjs(new Date(1900, 0, 31));
|
|
|
+ const solarDate = baseDate.add(offset, 'day').format('YYYY-MM-DD');
|
|
|
|
|
|
-// 非闰月
|
|
|
-console.log(getLunarDate('2001-04-27'))
|
|
|
-// 闰月
|
|
|
-console.log(getLunarDate('2001-05-27'))
|
|
|
|
|
|
-console.log(getYearLeapMonth(2001))
|
|
|
+ /* 闰月日期 */
|
|
|
+ let leapMonthDateOffset = offset;
|
|
|
+ let solarLeapMonthDate: string | undefined;
|
|
|
+ if (leapMonth === lunarMonth) {
|
|
|
+ leapMonthDateOffset += monthDays(lunarYear, lunarMonth);
|
|
|
+ solarLeapMonthDate = baseDate.add(leapMonthDateOffset, 'day').format('YYYY-MM-DD');
|
|
|
+ }
|
|
|
|
|
|
+ return {
|
|
|
+ date: solarDate,
|
|
|
+ leapMonthDate: solarLeapMonthDate,
|
|
|
+ };
|
|
|
+}
|
|
|
|
|
|
-console.log(getYears(2000, 2003))
|
|
|
-console.log(Date.now())
|
|
|
+export default {
|
|
|
+ getLunarDate,
|
|
|
+ getLunarDatesInRange,
|
|
|
+ getSolarDateFromLunar,
|
|
|
+}
|