j-calendar.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. <template>
  2. <!-- 打卡日历页面 -->
  3. <view class='all'>
  4. <view class="bar acenter">
  5. <!-- 上一个月 -->
  6. <view class="previous" @click="changeMonth(-1)">
  7. <u-icon name="play-left-fill"></u-icon>
  8. <!-- <button class="barbtn">{{langType=='ch'?'上一月':'Last'}}</button> -->
  9. </view>
  10. <!-- 显示年月 -->
  11. <view class="date" style="font-weight: bold;">{{nowYear || "--"}} 年 {{nowMonth || "--"}} 月</view>
  12. <!-- 下一个月 -->
  13. <view class="next" @click="changeMonth(1)">
  14. <u-icon name="play-right-fill"></u-icon>
  15. <!-- <button class="barbtn">{{langType=='ch'?'下一月':'Nex/'}}</button> -->
  16. </view>
  17. </view>
  18. <!-- 显示星期 -->
  19. <view class="week-area" >
  20. <view class="week-txt" v-for="(item,index) in weeksTxt[langType]" :key="index">{{item}}</view>
  21. </view>
  22. <view class="myDateTable hflex acenter jbetween fwrap">
  23. <view v-for="(item,j) in calendarDays" :key="j" class='dateCell' :class="item.date==today&&nowMonth==toMonth&&nowYear==toYear ? 'bgBlue' : ''">
  24. <view v-if="item.date==undefined||item.date == null" class='cell'></view>
  25. <template v-else>
  26. <!-- 已签到日期 -->
  27. <view v-if="item.isSign == true" class='cell whiteColor'>
  28. <u-icon name="checkmark-circle-fill" color="#00b0b0"></u-icon>
  29. {{item.date}}
  30. </view>
  31. <!-- 漏签 -->
  32. <view class="cell outSignStyle" v-else-if="item.isBeforeToday&&item.isThisMonth">
  33. <!-- redColor bgGray -->
  34. <u-icon name="close-circle-fill" color="#b7b8bd"></u-icon>
  35. {{item.date}}
  36. <!-- <view class="redDot"></view> -->
  37. </view>
  38. <!-- 今日未签到-->
  39. <view @click="clickSign(item.date,1)" class="cell whiteColor" v-else-if="item.date==today&&nowMonth==toMonth&&nowYear==toYear">
  40. {{item.date}}
  41. </view>
  42. <!-- 当前日期之后 -->
  43. <view class="cell" v-else>
  44. <view class="circle hflex acenter jcenter">
  45. <view class="circle2 hflex acenter jcenter"></view>
  46. </view>
  47. <!-- <u-icon name="play-left-fill"></u-icon> -->
  48. {{item.date}}
  49. </view>
  50. </template>
  51. </view>
  52. </view>
  53. </view>
  54. </template>
  55. <script>
  56. export default {
  57. data() {
  58. return {
  59. calendarDays: [],
  60. SignData: [],// 已经签到的数据
  61. nowYear: 0, //当前选的年
  62. nowMonth: 0, //当前选的月
  63. today: parseInt(new Date().getDate()), //系统本日
  64. toMonth: parseInt(new Date().getMonth() + 1), //系统本月
  65. toYear: parseInt(new Date().getFullYear()), //系统本年
  66. weeksTxt: {
  67. ch:['日', '一', '二', '三', '四', '五', '六'],
  68. en: ['Sun', 'Mon', 'Tues', 'Wed', 'Thur', 'Fri', 'Sat'],
  69. },
  70. };
  71. },
  72. props: {
  73. isReplenishSign: { // 是否允许过期补签
  74. type: Boolean,
  75. default: false
  76. },
  77. isFullCalendar: { // 是否需要填满日历,前后空的格子填入上/下个月的日期
  78. type: Boolean,
  79. default: true
  80. },
  81. yearMonth: { // 2022-01 这种格式,默认当前年月
  82. type: String,
  83. default: new Date().getFullYear()+'-'+ Number(new Date().getMonth() + 1)
  84. },
  85. dataSource: { //已签到的数据源,例: 5、6号已签到: ["2022-01-05","2022-01-06"],兼容个位数前可加可不加0
  86. type: Array,
  87. default: () => {
  88. return []
  89. }
  90. },
  91. langType: { //只是示例一个翻译而已,要想所有都翻译自己可以再加加
  92. type: String,
  93. default: "ch" //en
  94. },
  95. },
  96. created() {
  97. if(!(/en|ch/g.test(this.langType))){
  98. this.langType='ch'; // 非中英,则固定中文
  99. }
  100. const ymArr=this.yearMonth.split('-');
  101. this.buildCalendar(ymArr[0], ymArr[1]);
  102. this.onSignDataChange(this.dataSource);
  103. },
  104. watch: {
  105. dataSource: 'onSignDataChange',
  106. },
  107. methods: {
  108. clickSign(date, type) { //type=0补签,type=1当日签到
  109. return
  110. var strTip = "签到";
  111. if (type == 0) {
  112. if(!this.isReplenishSign){ // 未开启补签,阻止继续执行
  113. console.log("————补签功能未开启————");
  114. return;
  115. }
  116. strTip = "补签";
  117. }
  118. uni.showToast({
  119. title: date + "号" + strTip + "成功",
  120. icon: 'success',
  121. position:"bottom",
  122. });
  123. this.$emit('clickChange', this.nowYear + "-" + this.nowMonth + "-" + date); //传给调用模板页面
  124. // 父页面要在clickChange里去修改输入的数据源dataSource,否则视图不更新的!
  125. },
  126. //构建日历数据
  127. buildCalendar(y, m){
  128. this.nowYear = y;
  129. this.nowMonth = m;
  130. this.calculateEmptyGrids(y, m);
  131. this.calculateDays(y, m);
  132. if(this.isFullCalendar){
  133. this.fullCell()
  134. }
  135. },
  136. // 监听到签到数据源改变
  137. onSignDataChange(newData, oldData = []) {
  138. this.SignData = newData;
  139. this.matchSign();
  140. },
  141. //匹配标记已经签到的日子
  142. matchSign() {
  143. var signs = this.SignData;
  144. var daysArr = this.calendarDays;
  145. console.log('daysArr',daysArr);
  146. for (var i = 0; i < signs.length; i++) {
  147. var current = new Date(this.toIOSDate(signs[i].date)).getTime(); // ios只认/格式的日期
  148. for (var j = 0; j < daysArr.length; j++) {
  149. if(current == new Date(this.toIOSDate(daysArr[j].fullDate)).getTime()){
  150. daysArr[j].isSign = signs[i].is_sign;
  151. }
  152. }
  153. }
  154. this.calendarDays = daysArr;
  155. // console.log(this.calendarDays );
  156. },
  157. // 计算当月1号前空了几个格子,把它填充在calendarDays数组的前面
  158. calculateEmptyGrids(year, month) {
  159. //计算每个月时要清零
  160. this.calendarDays = [];
  161. const firstDayOfWeek = this.getFirstDayOfWeek(year, month);
  162. if (firstDayOfWeek > 0) {
  163. for (let i = 0; i < firstDayOfWeek; i++) {
  164. this.calendarDays.push({
  165. date: null, // 显示的日期
  166. fullDate: null, // 日期yyyy-mm-dd格式
  167. isBeforeToday: true, // 今日之前
  168. isSign: false, // 是否签到
  169. isThisMonth:false, // 是本月
  170. });
  171. }
  172. }
  173. },
  174. // 绘制当月天数占的格子,并把它放到days数组中
  175. calculateDays(year, month) {
  176. const thisMonthDays = this.getMonthDayLength(year, month);
  177. const toDate= new Date(this.toYear+'/'+this.toMonth+'/'+this.today);
  178. for (let i = 1; i <= thisMonthDays; i++) {
  179. const fullDate=year+'-'+month+'-'+ i;
  180. const isBeforeToday=new Date(this.toIOSDate(fullDate)) < toDate;
  181. this.calendarDays.push({
  182. date: i,
  183. fullDate,
  184. isBeforeToday,
  185. isSign: false,
  186. isThisMonth:true,
  187. });
  188. }
  189. //console.log(this.calendarDays);
  190. },
  191. // 切换控制年月,上一个月,下一个月
  192. changeMonth(type) {
  193. const nowYear = parseInt(this.nowYear);
  194. const nowMonth = parseInt(this.nowMonth);
  195. const newObj=this.getOperateMonthDate(nowYear,nowMonth,type);
  196. this.buildCalendar(newObj.year, newObj.month); // 重新构建日历数据
  197. let m = this.nowMonth
  198. if(m<10) {
  199. m = '0' + m
  200. }
  201. this.$emit('dateChange', this.nowYear + "-" + m); //传给调用模板页面去拿新数据
  202. },
  203. // 获取当月共多少天,也就是获取月的最后一天
  204. getMonthDayLength(year, month) {
  205. return new Date(year, month, 0).getDate()
  206. },
  207. // 获取当月第一天星期几
  208. getFirstDayOfWeek(year, month, day=1) {
  209. return new Date(Date.UTC(year, month - 1, day)).getDay();
  210. },
  211. toIOSDate(strDate){ // iso不认识"-"拼接的日期,所以转/
  212. return strDate?strDate.replace(/-/g,'/'):strDate;
  213. },
  214. // 需要填满格子,上/下个月的日期拉出来填满格子
  215. fullCell(){
  216. const endDay= this.getMonthDayLength(this.nowYear, this.nowMonth);
  217. const beforeEmptyLength = this.getFirstDayOfWeek(this.nowYear, this.nowMonth);
  218. const afterEmptyLength = 6 - this.getFirstDayOfWeek(this.nowYear, this.nowMonth, endDay);
  219. const last=this.getOperateMonthDate(this.nowYear,this.nowMonth,-1);
  220. const lastMonthEndDay=this.getMonthDayLength(last.year, last.month);
  221. for (let i = 0; i < beforeEmptyLength; i++) {
  222. const date=lastMonthEndDay - beforeEmptyLength + i + 1;
  223. this.calendarDays[i].date=date;
  224. this.calendarDays[i].fullDate=last.year+"-"+last.month+"-"+date;
  225. }
  226. const next=this.getOperateMonthDate(this.nowYear,this.nowMonth,1);
  227. for (let i = 1; i <= afterEmptyLength; i++) {
  228. this.calendarDays.push({
  229. date: i, // 显示的日期
  230. fullDate: next.year+"-"+next.month+"-"+i, // 日期yyyy-mm-dd格式
  231. isBeforeToday: false, // 今日之前
  232. isSign: false, // 是否签到
  233. isThisMonth:false, // 是本月
  234. });
  235. }
  236. // console.log(beforeEmptyLength,afterEmptyLength,lastMonthEndDay);
  237. // console.log(this.calendarDays);
  238. },
  239. // 获取加/减一个月的日期
  240. getOperateMonthDate(yy,mm,num){
  241. let month=parseInt(mm) + parseInt(num);
  242. let year=parseInt(yy);
  243. if(month>12){
  244. month=1;
  245. year++;
  246. }else if(month<1){
  247. month=12;
  248. year--;
  249. }
  250. return {
  251. month,
  252. year,
  253. }
  254. },
  255. }
  256. }
  257. </script>
  258. <style lang="scss">
  259. .all {
  260. margin-top: 20rpx;
  261. background: #FFFFFF;
  262. border-radius: 20rpx;
  263. overflow: hidden;
  264. z-index: 99;
  265. }
  266. .all .bar {
  267. display: flex;
  268. flex-direction: row;
  269. justify-content: space-between;
  270. /* margin: 30rpx 20rpx; */
  271. /* padding: 10rpx; */
  272. width: 100%;
  273. height: 88rpx;
  274. background: #C7E8E8;
  275. padding: 34rpx;
  276. box-sizing: border-box;
  277. }
  278. .bar .barbtn {
  279. height: 30px;
  280. line-height: 30px;
  281. font-size: 12px;
  282. }
  283. .all .week-area {
  284. display: flex;
  285. justify-content: space-between;
  286. padding: 10px 1vw;
  287. box-sizing: border-box;
  288. width: 91vw;
  289. margin: 10px auto;
  290. border-radius: 10px;
  291. }
  292. .all .week-txt {
  293. text-align: center;
  294. width: 76rpx;
  295. }
  296. .myDateTable {
  297. margin: 0 auto;
  298. width: 91vw;
  299. padding: 2vw;
  300. border-radius: 10px;
  301. // background: linear-gradient(#74AADA, #94db98);
  302. }
  303. .myDateTable .dateCell {
  304. width: 76rpx;
  305. padding: 1vw;
  306. display: inline-block;
  307. text-align: center;
  308. font-size: 16px;
  309. box-sizing: border-box;
  310. overflow: hidden;
  311. background: #F7F7F7;
  312. border-radius: 6rpx;
  313. margin: 0 20rpx 16rpx 0;
  314. }
  315. .dateCell:nth-child(7n+7) {
  316. margin: 0 0 16rpx;
  317. }
  318. .dateCell .cell {
  319. display: flex;
  320. border-radius: 50%;
  321. height: 11vw;
  322. justify-content: center;
  323. align-items: center;
  324. box-sizing: border-box;
  325. flex-direction: column;
  326. color: #999999;
  327. }
  328. .circle {
  329. width: 32rpx;
  330. height: 32rpx;
  331. background: #CFCFCF;
  332. border-radius: 50%;
  333. }
  334. .circle2 {
  335. width: 25rpx;
  336. height: 25rpx;
  337. background: #969A9A;
  338. box-shadow: inset 0rpx 1rpx 1rpx 0rpx #717575;
  339. border-radius: 50%;
  340. }
  341. .whiteColor {
  342. color: #333333 !important;
  343. }
  344. .greenColor {
  345. color: #999999;
  346. font-weight: bold;
  347. }
  348. .bgWhite {
  349. background-color: #F7F7F7;
  350. }
  351. .bgGray {
  352. background-color: rgba(255, 255, 255, 0.42);
  353. }
  354. .bgBlue {
  355. background: rgba(0, 176, 176, .1) !important;
  356. }
  357. .redColor {
  358. color: #ff0000;
  359. }
  360. .outSignStyle{
  361. border:1px solid #ffffff;
  362. color: #999999;
  363. }
  364. .redDot{
  365. width: 3px;
  366. height: 3px;
  367. border-radius: 50%;
  368. background-color: red;
  369. }
  370. </style>