分享

原生JS实现日历-CSDN博客

 hzlinhui 2024-07-28
这周写自己的项目发现又用到日历了,加之自己毕业之后的第一个工作中遇到的任务也是需要写个日历(组员写了,我就不用写了)

今天就来好好折腾一下日历是怎么写的。

首先,我们看看 windows 的日历。发现总共有这么几个元素。先实现试试。
  • 1.年份的选择、月份的选择
  • 2.周一 ~ 周日(周日 ~ 周六)
  • 3.日历格子 6*7 = 42
从数据的角度来分析日历的实现是比较简单的
  • 1.我们需要显示一个当前时间的结构 - new Date()
  • 2.我们需要显示当月的信息 - [星期(周一~周日),日期(1-[28,29,30,31])]
    • 其中我们只要知道了每个月的 1日 是星期几,就能很容易地摆放后面的日子(万事开头难)。
    • 我们最多需要 6 行来显示我们的日期,因为要考虑周一是星期日的情况 1+7*4 = 29(五行)显然是不够的
    • 确定了 6 行之后,我们发现我们可能需要获取上个月,和下个月多出来的几天的摆放位置。
    • 不同年份的不同月的 2月份,我们知道它的日期是不同的,所以我们还需要判断 平年还是闰年。
  • 3.显示上个月,下个月的切换。我们发现需要有个函数来帮我们更新日历。

    这里我们就要考虑一下,到底要怎么更新这些 dom 里面的数据了,重新删除插入 dom 肯定是不太好的。

分析完之后,让我们跟着 新增/修改 一些代码。
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <meta http-equiv="X-UA-Compatible" content="ie=edge">
  7. <title>Document</title>
  8. <style>
  9. .week-item {
  10. display: inline-block;
  11. width: 80px;
  12. height: 40px;
  13. line-height: 40px;
  14. border: 1px solid sandybrown;
  15. text-align: center;
  16. }
  17. .date-item {
  18. display: inline-block;
  19. width: 80px;
  20. height: 40px;
  21. line-height: 40px;
  22. border: 1px solid beige;
  23. text-align: center;
  24. }
  25. </style>
  26. </head>
  27. <body>
  28. <div class="wrapper">
  29. <div class="year-line">
  30. <button id="preMonth" class="year-prev">上一月</button>
  31. <button id="nowYear" class="year-now"></button>
  32. <button id="nowMonth"></button>
  33. <button id="nowDate"></button>
  34. <button id="nextMonth" class="year-next">下一月</button>
  35. </div>
  36. <div id="weekLine" class="week-line"></div>
  37. <div id="dateWrap" class="date-wrap"></div>
  38. </div>
  39. </body>
  40. <script>
  41. // 工具方法 - start
  42. // 1.为了获得每个月的日期有多少,我们需要判断 平年闰年[四年一闰,百年不闰,四百年再闰]
  43. const isLeapYear = (year) => {
  44. return (year % 400 === 0) || (year % 100 !== 0 && year % 4 === 0);
  45. };
  46. // 2.获得每个月的日期有多少,注意 month - [0-11]
  47. const getMonthCount = (year, month) => {
  48. let arr = [
  49. 31, null, 31, 30,
  50. 31, 30, 31, 31,
  51. 30, 31, 30, 31
  52. ];
  53. let count = arr[month] || (isLeapYear(year) ? 29 : 28);
  54. return Array.from(new Array(count), (item, value) => value + 1);
  55. };
  56. // 3.获得某年某月的 1号 是星期几,这里要注意的是 JS 的 API-getDay() 是从 [日-六](0-6),返回 number
  57. const getWeekday = (year, month) => {
  58. let date = new Date(year, month, 1);
  59. return date.getDay();
  60. };
  61. // 4.获得上个月的天数
  62. const getPreMonthCount = (year, month) => {
  63. if (month === 0) {
  64. return getMonthCount(year - 1, 11);
  65. } else {
  66. return getMonthCount(year, month - 1);
  67. }
  68. };
  69. // 5.获得下个月的天数
  70. const getNextMonthCount = (year, month) => {
  71. if (month === 11) {
  72. return getMonthCount(year + 1, 0);
  73. } else {
  74. return getMonthCount(year, month + 1);
  75. }
  76. };
  77. // 工具方法 - end
  78. let weekStr = '日一二三四五六';
  79. weekArr = weekStr.split('').map(item => '星期' + item);
  80. // 插入星期 dom
  81. let weekDomStr = '';
  82. let oFragWeek = document.createDocumentFragment();
  83. weekArr.forEach(item => {
  84. let oSpan = document.createElement('span');
  85. let oText = document.createTextNode(item);
  86. oSpan.appendChild(oText);
  87. oSpan.classList.add('week-item');
  88. oFragWeek.appendChild(oSpan);
  89. });
  90. let weekWrap = document.getElementById('weekLine');
  91. weekWrap.appendChild(oFragWeek);
  92. // 这里获得我们第一次的 数据 数组
  93. const updateCalendar = (year, month, day) => {
  94. if (typeof year === 'undefined' && typeof month === 'undefined' && typeof day === 'undefined') {
  95. let nowDate = new Date();
  96. year = nowDate.getFullYear();
  97. month = nowDate.getMonth();
  98. day = nowDate.getDate();
  99. }
  100. // 更新一下顶部的年月显示
  101. document.getElementById('nowYear').innerHTML = year;
  102. document.getElementById('nowMonth').innerHTML = month + 1;
  103. document.getElementById('nowDate').innerHTML = day;
  104. // 生成日历数据,上个月的 x 天 + 当月的 [28,29,30,31]天 + 下个月的 y 天 = 42
  105. let res = [];
  106. let currentMonth = getMonthCount(year, month);
  107. let preMonth = getPreMonthCount(year, month);
  108. let nextMonth = getNextMonthCount(year, month);
  109. let whereMonday = getWeekday(year, month);
  110. let preArr = preMonth.slice(-1 * whereMonday);
  111. let nextArr = nextMonth.slice(0, 42 - currentMonth.length - whereMonday);
  112. res = [].concat(preArr, currentMonth, nextArr);
  113. // 上面经过我本人的测试是没有什么问题,接下来就是更新 dom 的信息的问题
  114. let hadDom = document.getElementsByClassName('date-item');
  115. if (hadDom && hadDom.length) {
  116. let domArr = document.getElementsByClassName('date-item');
  117. for (let i = 0; i < domArr.length; i++) {
  118. domArr[i].innerHTML = res.shift();
  119. }
  120. } else {
  121. // 如果之前没有结构的话
  122. let str = '';
  123. for (let i = 0; i < 6; i++) {
  124. str += '<div class="date-line">';
  125. for (let j = 0; j < 7; j++) {
  126. str += `<span class='date-item'>${res.shift()}</span>`;
  127. if (j === 6) {
  128. str += '</div>';
  129. }
  130. }
  131. }
  132. document.getElementById('dateWrap').innerHTML = str;
  133. }
  134. };
  135. updateCalendar();
  136. // 添加上一月,下一月事件
  137. let oPreButton = document.getElementById('preMonth');
  138. let oNextButton = document.getElementById('nextMonth');
  139. oPreButton.addEventListener('click', function () {
  140. let currentYear = +document.getElementById('nowYear').textContent;
  141. let currentMonth = +document.getElementById('nowMonth').textContent - 1;
  142. let currentDate = +document.getElementById('nowDate').textContent;
  143. if (currentMonth === 0) {
  144. updateCalendar(currentYear - 1, 11, currentDate);
  145. } else {
  146. updateCalendar(currentYear, currentMonth - 1, currentDate);
  147. }
  148. });
  149. oNextButton.addEventListener('click', function () {
  150. let currentYear = +document.getElementById('nowYear').textContent;
  151. let currentMonth = +document.getElementById('nowMonth').textContent - 1;
  152. let currentDate = +document.getElementById('nowDate').textContent;
  153. if (currentMonth === 11) {
  154. updateCalendar(currentYear + 1, 0, currentDate);
  155. } else {
  156. updateCalendar(currentYear, currentMonth + 1, currentDate);
  157. }
  158. });
  159. </script>
  160. </html>
发现用 dom 直接操作而不是通过 mvvm 框架实现确实还是比较蛋疼的,以下是这次实现的效果。

实现一个功能的时候,从数据的层面分析,有时候会比较容易理解

calendar2.png

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约