computedLunarDate.js 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. let lunarYearArr = [
  2. 0x0b557, //1949
  3. 0x06ca0, 0x0b550, 0x15355, 0x04da0, 0x0a5b0, 0x14573, 0x052b0, 0x0a9a8, 0x0e950, 0x06aa0, //1950-1959
  4. 0x0aea6, 0x0ab50, 0x04b60, 0x0aae4, 0x0a570, 0x05260, 0x0f263, 0x0d950, 0x05b57, 0x056a0, //1960-1969
  5. 0x096d0, 0x04dd5, 0x04ad0, 0x0a4d0, 0x0d4d4, 0x0d250, 0x0d558, 0x0b540, 0x0b6a0, 0x195a6, //1970-1979
  6. 0x095b0, 0x049b0, 0x0a974, 0x0a4b0, 0x0b27a, 0x06a50, 0x06d40, 0x0af46, 0x0ab60, 0x09570, //1980-1989
  7. 0x04af5, 0x04970, 0x064b0, 0x074a3, 0x0ea50, 0x06b58, 0x055c0, 0x0ab60, 0x096d5, 0x092e0, //1990-1999
  8. 0x0c960, 0x0d954, 0x0d4a0, 0x0da50, 0x07552, 0x056a0, 0x0abb7, 0x025d0, 0x092d0, 0x0cab5, //2000-2009
  9. 0x0a950, 0x0b4a0, 0x0baa4, 0x0ad50, 0x055d9, 0x04ba0, 0x0a5b0, 0x15176, 0x052b0, 0x0a930, //2010-2019
  10. 0x07954, 0x06aa0, 0x0ad50, 0x05b52, 0x04b60, 0x0a6e6, 0x0a4e0, 0x0d260, 0x0ea65, 0x0d530, //2020-2029
  11. 0x05aa0, 0x076a3, 0x096d0, 0x04afb, 0x04ad0, 0x0a4d0, 0x1d0b6, 0x0d250, 0x0d520, 0x0dd45, //2030-2039
  12. 0x0b5a0, 0x056d0, 0x055b2, 0x049b0, 0x0a577, 0x0a4b0, 0x0aa50, 0x1b255, 0x06d20, 0x0ada0, //2040-2049
  13. 0x14b63, 0x09370, 0x049f8, 0x04970, 0x064b0, 0x168a6, 0x0ea50, 0x06b20, 0x1a6c4, 0x0aae0, //2050-2059
  14. 0x0a2e0, 0x0d2e3, 0x0c960, 0x0d557, 0x0d4a0, 0x0da50, 0x05d55, 0x056a0, 0x0a6d0, 0x055d4, //2060-2069
  15. 0x052d0, 0x0a9b8, 0x0a950, 0x0b4a0, 0x0b6a6, 0x0ad50, 0x055a0, 0x0aba4, 0x0a5b0, 0x052b0, //2070-2079
  16. 0x0b273, 0x06930, 0x07337, 0x06aa0, 0x0ad50, 0x14b55, 0x04b60, 0x0a570, 0x054e4, 0x0d160, //2080-2089
  17. 0x0e968, 0x0d520, 0x0daa0, 0x16aa6, 0x056d0, 0x04ae0, 0x0a9d4, 0x0a2d0, 0x0d150, 0x0f252, //2090-2099
  18. 0x0d520 //2100
  19. ],
  20. lunarMonth = ['正', '二', '三', '四', '五', '六', '七', '八', '九', '十', '冬', '腊'],
  21. lunarDay = ['一', '二', '三', '四', '五', '六', '七', '八', '九', '十', '初', '廿'],
  22. tianGan = ['甲', '乙', '丙', '丁', '戊', '己', '庚', '辛', '壬', '癸'],
  23. diZhi = ['子鼠', '丑牛', '寅虎', '卯兔', '辰龙', '巳蛇', '午马', '未羊', '申猴', '酉鸡', '戌狗', '亥猪'];
  24. // 公历转农历函数
  25. export const sloarToLunar = function (sy, sm, sd) {
  26. // 输入的月份减1处理
  27. sm -= 1;
  28. // debugger
  29. // 计算与公历基准的相差天数
  30. // Date.UTC()返回的是距离公历1970年1月1日的毫秒数,传入的月份需要减1
  31. let daySpan = (Date.UTC(sy, sm, sd) - Date.UTC(1949, 0, 29)) / (24 * 60 * 60 * 1000) + 1;
  32. let ly, lm, ld,lz,nly;
  33. // 确定输出的农历年份
  34. for (let j = 0; j < lunarYearArr.length; j++) {
  35. daySpan -= lunarYearDays(lunarYearArr[j]);
  36. if (daySpan <= 0) {
  37. ly = 1949 + j;
  38. // 获取农历年份确定后的剩余天数
  39. daySpan += lunarYearDays(lunarYearArr[j]);
  40. break
  41. }
  42. }
  43. // 确定输出的农历月份
  44. for (let k = 0; k < lunarYearMonths(lunarYearArr[ly - 1949]).length; k++) {
  45. daySpan -= lunarYearMonths(lunarYearArr[ly - 1949])[k];
  46. if (daySpan <= 0) {
  47. // 有闰月时,月份的数组长度会变成13,因此,当闰月月份小于等于k时,lm不需要加1
  48. if (hasLeapMonth(lunarYearArr[ly - 1949]) && hasLeapMonth(lunarYearArr[ly - 1949]) <= k) {
  49. if (hasLeapMonth(lunarYearArr[ly - 1949]) < k) {
  50. lm = k;
  51. } else if (hasLeapMonth(lunarYearArr[ly - 1949]) === k) {
  52. lm = '闰' + k;
  53. } else {
  54. lm = k + 1;
  55. }
  56. } else {
  57. lm = k + 1;
  58. }
  59. // 获取农历月份确定后的剩余天数
  60. daySpan += lunarYearMonths(lunarYearArr[ly - 1949])[k];
  61. break
  62. }
  63. }
  64. // 确定输出农历哪一天
  65. ld = daySpan;
  66. // 将计算出来的农历月份转换成汉字月份,闰月需要在前面加上闰字
  67. if (hasLeapMonth(lunarYearArr[ly - 1949]) && (typeof (lm) === 'string' && lm.indexOf('闰') > -1)) {
  68. lm = `闰${lunarMonth[/\d/.exec(lm) - 1]}`
  69. } else {
  70. lm = lunarMonth[lm - 1];
  71. }
  72. nly=ly
  73. // 将计算出来的农历年份转换为天干地支年
  74. ly = getTianGan(ly) + getDiZhi(ly);
  75. // 将计算出来的农历天数转换成汉字
  76. if (ld < 11) {
  77. ld = `${lunarDay[10]}${lunarDay[ld-1]}`
  78. } else if (ld > 10 && ld < 20) {
  79. ld = `${lunarDay[9]}${lunarDay[ld-11]}`
  80. } else if (ld === 20) {
  81. ld = `${lunarDay[1]}${lunarDay[9]}`
  82. } else if (ld > 20 && ld < 30) {
  83. ld = `${lunarDay[11]}${lunarDay[ld-21]}`
  84. } else if (ld === 30) {
  85. ld = `${lunarDay[2]}${lunarDay[9]}`
  86. }
  87. return {
  88. lunarYear: ly,
  89. lunarMonth: lm,
  90. lunarDay: ld,
  91. lunarShengXiao:lz,
  92. lunarYearNum:nly,
  93. lunarStr:`${nly}年${lm}月${ld}`,
  94. lunarStr2:`${nly}年${lm}月${ld} (${ly})`
  95. }
  96. };
  97. export const getYearMonthDay = function (sy, sm, sd) {
  98. // 输入的月份减1处理
  99. sm -= 1;
  100. // 计算与公历基准的相差天数
  101. // Date.UTC()返回的是距离公历1970年1月1日的毫秒数,传入的月份需要减1
  102. let daySpan = (Date.UTC(sy, sm, sd) - Date.UTC(1949, 0, 29)) / (24 * 60 * 60 * 1000) + 1;
  103. let ly, lm, ld,lz;
  104. // 确定输出的农历年份
  105. for (let j = 0; j < lunarYearArr.length; j++) {
  106. daySpan -= lunarYearDays(lunarYearArr[j]);
  107. if (daySpan <= 0) {
  108. ly = 1949 + j;
  109. // 获取农历年份确定后的剩余天数
  110. daySpan += lunarYearDays(lunarYearArr[j]);
  111. break
  112. }
  113. }
  114. // 确定输出的农历月份
  115. for (let k = 0; k < lunarYearMonths(lunarYearArr[ly - 1949]).length; k++) {
  116. daySpan -= lunarYearMonths(lunarYearArr[ly - 1949])[k];
  117. if (daySpan <= 0) {
  118. // 有闰月时,月份的数组长度会变成13,因此,当闰月月份小于等于k时,lm不需要加1
  119. if (hasLeapMonth(lunarYearArr[ly - 1949]) && hasLeapMonth(lunarYearArr[ly - 1949]) <= k) {
  120. if (hasLeapMonth(lunarYearArr[ly - 1949]) < k) {
  121. lm = k;
  122. } else if (hasLeapMonth(lunarYearArr[ly - 1949]) === k) {
  123. lm = '闰' + k;
  124. } else {
  125. lm = k + 1;
  126. }
  127. } else {
  128. lm = k + 1;
  129. }
  130. // 获取农历月份确定后的剩余天数
  131. daySpan += lunarYearMonths(lunarYearArr[ly - 1949])[k];
  132. break
  133. }
  134. }
  135. // 确定输出农历哪一天
  136. ld = daySpan;
  137. // 将计算出来的农历月份转换成汉字月份,闰月需要在前面加上闰字
  138. if (hasLeapMonth(lunarYearArr[ly - 1949]) && (typeof (lm) === 'string' && lm.indexOf('闰') > -1)) {
  139. lm = `闰${lunarMonth[/\d/.exec(lm) - 1]}`
  140. } else {
  141. lm = lunarMonth[lm - 1];
  142. }
  143. // 将计算出来的农历年份转换为天干地支年
  144. ly = getTianGan(ly) + getDiZhi(ly);
  145. // 将计算出来的农历天数转换成汉字
  146. if (ld < 11) {
  147. ld = `${lunarDay[10]}${lunarDay[ld-1]}`
  148. } else if (ld > 10 && ld < 20) {
  149. ld = `${lunarDay[9]}${lunarDay[ld-11]}`
  150. } else if (ld === 20) {
  151. ld = `${lunarDay[1]}${lunarDay[9]}`
  152. } else if (ld > 20 && ld < 30) {
  153. ld = `${lunarDay[11]}${lunarDay[ld-21]}`
  154. } else if (ld === 30) {
  155. ld = `${lunarDay[2]}${lunarDay[9]}`
  156. }
  157. return ld;
  158. };
  159. // 计算农历年是否有闰月,参数为存储农历年的16进制
  160. // 农历年份信息用16进制存储,其中16进制的最后1位可以用于判断是否有闰月
  161. function hasLeapMonth(ly) {
  162. // 获取16进制的最后1位,需要用到&与运算符
  163. if (ly & 0xf) {
  164. return ly & 0xf
  165. } else {
  166. return false
  167. }
  168. }
  169. // 如果有闰月,计算农历闰月天数,参数为存储农历年的16进制
  170. // 农历年份信息用16进制存储,其中16进制的第1位(0x除外)可以用于表示闰月是大月还是小月
  171. function leapMonthDays(ly) {
  172. if (hasLeapMonth(ly)) {
  173. // 获取16进制的第1位(0x除外)
  174. return (ly & 0xf0000) ? 30 : 29
  175. } else {
  176. return 0
  177. }
  178. }
  179. // 计算农历一年的总天数,参数为存储农历年的16进制
  180. // 农历年份信息用16进制存储,其中16进制的第2-4位(0x除外)可以用于表示正常月是大月还是小月
  181. function lunarYearDays(ly) {
  182. let totalDays = 0;
  183. // 获取正常月的天数,并累加
  184. // 获取16进制的第2-4位,需要用到>>移位运算符
  185. for (let i = 0x8000; i > 0x8; i >>= 1) {
  186. let monthDays = (ly & i) ? 30 : 29;
  187. totalDays += monthDays;
  188. }
  189. // 如果有闰月,需要把闰月的天数加上
  190. if (hasLeapMonth(ly)) {
  191. totalDays += leapMonthDays(ly);
  192. }
  193. return totalDays
  194. }
  195. // 获取农历每个月的天数
  196. // 参数需传入16进制数值
  197. function lunarYearMonths(ly) {
  198. let monthArr = [];
  199. // 获取正常月的天数,并添加到monthArr数组中
  200. // 获取16进制的第2-4位,需要用到>>移位运算符
  201. for (let i = 0x8000; i > 0x8; i >>= 1) {
  202. monthArr.push((ly & i) ? 30 : 29);
  203. }
  204. // 如果有闰月,需要把闰月的天数加上
  205. if (hasLeapMonth(ly)) {
  206. monthArr.splice(hasLeapMonth(ly), 0, leapMonthDays(ly));
  207. }
  208. return monthArr;
  209. }
  210. // 将农历年转换为天干,参数为农历年
  211. function getTianGan(ly) {
  212. let tianGanKey = (ly - 3) % 10;
  213. if (tianGanKey === 0) tianGanKey = 10;
  214. return tianGan[tianGanKey - 1];
  215. }
  216. // 将农历年转换为地支,参数为农历年
  217. function getDiZhi(ly) {
  218. let diZhiKey = (ly - 3) % 12;
  219. if (diZhiKey === 0) diZhiKey = 12;
  220. return diZhi[diZhiKey - 1];
  221. }