calendar.vue 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. <template>
  2. <view class="wrap">
  3. <view class="header-cal">
  4. <view v-for="(item, index) in date" :key="index">{{ item }}<view></view> </view>
  5. </view>
  6. <view class="date-box">
  7. <view v-for="(item, index) in dateArr" :key="index" @tap="calendarTap(item, index)">
  8. <view
  9. class="date-head1"
  10. :class="[todayIndex === index ? 'operationInProgress' : showBusiness ? item.styleClass : '', { showOverdue: item.showOverdue, 'date-class': item.dateClass }]"
  11. >
  12. <view>{{ item.dateNum }}</view>
  13. </view>
  14. </view>
  15. </view>
  16. </view>
  17. </template>
  18. <script>
  19. export default {
  20. props: {
  21. year: {
  22. type: Number,
  23. value: 0
  24. },
  25. month: {
  26. type: Number,
  27. value: 0
  28. },
  29. showBusiness: {
  30. // 是否显示业务标记
  31. type: Boolean,
  32. default: false
  33. }
  34. },
  35. data() {
  36. return {
  37. date: ['日', '一', '二', '三', '四', '五', '六'],
  38. dateArr: [],
  39. todayIndex: -1 // 操作索引,默认指向当天
  40. }
  41. },
  42. onLoad(options) {},
  43. methods: {
  44. calendarTap(item, index) {
  45. this.todayIndex = index
  46. this.$emit('changeDate', item.date)
  47. },
  48. selCalendar(year, month) {
  49. this.dateInit(year, month - 1)
  50. },
  51. setAttendance(attendance) {
  52. // 设置日历考勤
  53. const start = this.dateArr.findIndex(item => item.dateNum)
  54. attendance.map((item, index) => {
  55. if (item !== 'stateless') {
  56. const pos = start + index
  57. const dateObj = this.dateArr[pos]
  58. this.$set(dateObj, 'styleClass', item)
  59. this.$set(this.dateArr, pos, dateObj)
  60. }
  61. })
  62. },
  63. setClass(days) {
  64. //区分有课日期
  65. this.dateArr.map(item => {
  66. this.$set(item, 'dateClass', days.includes(item.dateNum))
  67. })
  68. },
  69. dateInit(setYear, setMonth) {
  70. // 1.获取当前时间
  71. var date = new Date()
  72. var currentYear = date.getFullYear() // 获取完整的年份(4位)
  73. var currentMonth = date.getMonth() + 1 // 获取当前月份(0-11,0代表1月)
  74. var currentDate = date.getDate() // 获取当前日(1-31)
  75. var currentTime = '' + currentYear + (currentMonth > 9 ? currentMonth : '0' + currentMonth) + (currentDate > 9 ? currentDate : '0' + currentDate)
  76. // 全部时间的月份都是按0~11基准,显示月份才+1
  77. const dateArr = [] // 需要遍历的日历数组数据
  78. let arrLen = 0 // dateArr的数组长度
  79. const now = setYear ? new Date(setYear, setMonth) : new Date()
  80. const year = setYear || now.getFullYear()
  81. let nextYear = 0
  82. const month = setMonth || now.getMonth() // 没有+1方便后面计算当月总天数
  83. const nextMonth = month + 1 > 11 ? 1 : month + 1
  84. const startWeek = new Date(year + '/' + (month + 1) + '/' + 1).getDay() // 目标月1号对应的星期
  85. let dayNums = new Date(year, nextMonth, 0).getDate() // 获取目标月有多少天
  86. let obj = {}
  87. let num = 0
  88. let showOverdue = 0
  89. if (month + 1 > 11) {
  90. nextYear = year + 1
  91. dayNums = new Date(nextYear, nextMonth, 0).getDate()
  92. }
  93. arrLen = startWeek + dayNums // 渲染日期格子数量包括空格子
  94. for (let i = 0; i < arrLen; i++) {
  95. if (i >= startWeek) {
  96. num = i - startWeek + 1
  97. // 计算 -- 需要渲染的过期时间
  98. const renderingTime = '' + year + (month + 1 > 9 ? month + 1 : '0' + (month + 1)) + (num > 9 ? num : '0' + num)
  99. if (Number(currentTime) > Number(renderingTime)) {
  100. showOverdue = 1 // 过期
  101. } else {
  102. showOverdue = 0 // 有效
  103. }
  104. obj = {
  105. year: year,
  106. month: month + 1,
  107. dateNum: num,
  108. date: renderingTime,
  109. showOverdue: showOverdue
  110. }
  111. } else {
  112. obj = {}
  113. }
  114. dateArr[i] = obj
  115. }
  116. this.dateArr = dateArr
  117. }
  118. }
  119. }
  120. </script>
  121. <style lang="scss" scoped>
  122. .date-show {
  123. position: relative;
  124. width: 250rpx;
  125. font-family: PingFang-SC-Regular;
  126. font-size: 40rpx;
  127. color: #282828;
  128. text-align: center;
  129. margin: 59rpx auto 10rpx;
  130. }
  131. .lt-arrow,
  132. .rt-arrow {
  133. position: absolute;
  134. top: 1rpx;
  135. width: 60rpx;
  136. height: 60rpx;
  137. }
  138. .lt-arrow image,
  139. .rt-arrow image {
  140. width: 14rpx;
  141. height: 26rpx;
  142. }
  143. .lt-arrow {
  144. left: -110rpx;
  145. transform: rotate(180deg);
  146. }
  147. .rt-arrow {
  148. right: -100rpx;
  149. }
  150. .header-cal {
  151. font-size: 0;
  152. /* padding: 0 24rpx; */
  153. }
  154. .header-cal > view {
  155. display: inline-block;
  156. width: 14.285%;
  157. color: #5fd0e4;
  158. font-size: 30rpx;
  159. text-align: center;
  160. /* border-bottom: 1px solid #D0D0D0; */
  161. padding: 39rpx 0;
  162. }
  163. .weekMark {
  164. position: relative;
  165. }
  166. .weekMark view {
  167. position: absolute;
  168. bottom: 0;
  169. left: 0;
  170. width: 100%;
  171. border-bottom: 1px solid #22a7f6;
  172. }
  173. .date-box {
  174. font-size: 0;
  175. padding: 10rpx 0;
  176. }
  177. .date-box > view {
  178. position: relative;
  179. display: inline-block;
  180. width: 14.285%;
  181. color: #020202;
  182. font-size: 40rpx;
  183. text-align: center;
  184. vertical-align: middle;
  185. margin: 15rpx 0;
  186. }
  187. .date-head1 {
  188. position: relative;
  189. height: 60rpx;
  190. line-height: 60rpx;
  191. font-size: 26rpx;
  192. border-radius: 12rpx;
  193. color: #ccc;
  194. }
  195. .date-head2 {
  196. background: #20ba96;
  197. color: #fff;
  198. height: 60rpx;
  199. line-height: 60rpx;
  200. font-size: 26rpx;
  201. border-radius: 12rpx;
  202. }
  203. .date-class {
  204. color: #000;
  205. }
  206. .date-weight {
  207. font-size: 22rpx;
  208. padding: 15rpx 0;
  209. }
  210. .nowDay .date-weight {
  211. color: #22a7f6;
  212. }
  213. /* 过期样式*/
  214. .showOverdue {
  215. color: #5fd0e4;
  216. }
  217. /* 正在操作样式*/
  218. .operationInProgress {
  219. // background: #20ba96;
  220. // color: #fff;
  221. color: #20ba96;
  222. }
  223. /* 正常样式*/
  224. .normal {
  225. // background: #4db8e4;
  226. // color: #fff;
  227. &:after {
  228. content: '';
  229. position: absolute;
  230. bottom: 0;
  231. right: 0;
  232. width: 0;
  233. height: 0;
  234. border-color: #39b54a transparent;
  235. border-width: 0 0 12px 12px;
  236. border-style: solid;
  237. }
  238. }
  239. /* 异常样式*/
  240. .abnormal {
  241. // background: #ff7361;
  242. // color: #fff;
  243. &:after {
  244. content: '';
  245. position: absolute;
  246. bottom: 0;
  247. right: 0;
  248. width: 0;
  249. height: 0;
  250. border-color: #ff7361 transparent;
  251. border-width: 0 0 12px 12px;
  252. border-style: solid;
  253. }
  254. }
  255. .leave {
  256. // background: #3390f5;
  257. // color: #fff;
  258. &:after {
  259. content: '';
  260. position: absolute;
  261. bottom: 0;
  262. right: 0;
  263. width: 0;
  264. height: 0;
  265. border-color: #3390f5 transparent;
  266. border-width: 0 0 12px 12px;
  267. border-style: solid;
  268. }
  269. }
  270. .late {
  271. // background: #fbbd08;
  272. // color: #fff;
  273. &:after {
  274. content: '';
  275. position: absolute;
  276. bottom: 0;
  277. right: 0;
  278. width: 0;
  279. height: 0;
  280. border-color: #fbbd08 transparent;
  281. border-width: 0 0 12px 12px;
  282. border-style: solid;
  283. }
  284. }
  285. .absenteeism {
  286. background: #e54d42;
  287. color: #fff;
  288. }
  289. </style>