微信小程序自定义可滑动日历界面

 更新时间:2018年12月28日 17:23:26   作者:white_poland  
这篇文章主要为大家详细介绍了微信小程序自定义可滑动日历界面,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了微信小程序可滑动日历界面的具体代码,供大家参考,具体内容如下

参考某个博主的自定义控件做了一些改动,希望这篇博客能帮助需要的人。

WXML

<view class='container'>
<view class='month flex m-around'>
 <view class='arrow' bindtap='prevMonth'>《 </view>
 <view class='year-and-month'>
  <picker mode="date" value="{{date}}" start="2015-09" end="2020-09" fields='month' bindchange="bindDateChange">
  <view>
   {{date}}
  </view>
  </picker>
 </view>
 <view class='arrow' bindtap='nextMonth'> 》</view>
</view>
<view class='calendar flex column s-center'>
 <view class='week-row flex m-around'>
 <view class='grid' wx:for="{{week}}" wx:key='item'>{{item}}</view>
 </view>
 <swiper class='swpier-box' circular="true" current="{{swiperIndex}}" bindchange='swiperChange'>
 <swiper-item class='flex m-around days-table '>
  <view wx:for="{{calendar.first}}" wx:for-item='x' wx:key='x.date'
  class='grid {{x.month === month?"":"notCurrent"}} {{x.date === today?"today":""}} {{x.date == beSelectDate ? "choice":""}}' data-month='{{x.month}}' data-day='{{x.day}}' data-date='{{x.date}}' bindtap='bindDayTap'>
  <view>{{x.date === today?'今天':x.day}}</view>
  </view>
 </swiper-item>
 <swiper-item class='flex m-around days-table '>
  <view wx:for="{{calendar.second}}" wx:for-item='x' wx:key='x.date'
  class='grid {{x.month === month?"":"notCurrent"}} {{x.date === today?"today":""}} {{x.date == beSelectDate ? "choice":""}}' data-month='{{x.month}}' data-day='{{x.day}}' data-date='{{x.date}}' data-test='{{(year + "-" +month + "-" + day)}}' bindtap='bindDayTap'>
  <view>{{x.date === today?'今天':x.day}}</view>
  </view>
 </swiper-item>
 <swiper-item class='flex m-around days-table'>
  <view wx:for="{{calendar.third}}" wx:for-item='x' wx:key='x.date'
  class='grid {{x.month === month?"":"notCurrent"}} {{x.date === today?"today":""}} {{x.date == beSelectDate ? "choice":""}}' data-month='{{x.month}}' data-day='{{x.day}}' data-date='{{x.date}}' bindtap='bindDayTap'>
  <view>{{x.date === today?'今天':x.day}}</view>
  </view>
 </swiper-item>
 <swiper-item class='flex m-around days-table '>
  <view wx:for="{{calendar.fourth}}" wx:for-item='x' wx:key='x.date'
  class='grid {{x.month === month?"":"notCurrent"}} {{x.date === today?"today":""}} {{x.date == beSelectDate ? "choice":""}}' data-month='{{x.month}}' data-day='{{x.day}}' data-date='{{x.date}}' bindtap='bindDayTap'>
  <view>{{x.date === today?'今天':x.day}}</view>
  </view>
 </swiper-item>
 </swiper>
</view>
</view>

CSS

/* pages/calendar/calendar.wxss */
 .container {
 height: 100vh;

 background-color: #393E44;
}
.days-table {
 flex-wrap: wrap;
 align-content: flex-start;
}
.calendar{
 position: fixed;
 z-index:10000;
 background: #393E44;
 
}
.grid {
 width: 107.14rpx;
 height: 100rpx;
 text-align: center;
 line-height: 100rpx;
 font-size:.7rem;
 color:#fff;
}
.today {
 color: #88a1fd;
}
.grid view {
 height:85rpx;
 line-height: 85rpx;
 width:85rpx;
}
.choice view{
 border-radius: 50%;
 background: #88a1fd;
 background-position:center;
 color: white;
}
/* 非本月日期 */
.notCurrent {
 color: silver;
}
.day-hover {
 background: red;
}
.swpier-box {
 height: 550rpx;
 width: 100%;
}
.arrow {
 width: 100rpx;
 color: #88a1fd;
 text-align: center;
}
.year-and-month{
 color: #88a1fd;
}

.flex {
 display: flex;
}
/* 轴向 */
.column {
 flex-direction: column;
}
/* 主轴方向 */
.m-start {
 justify-content: flex-start;
}

.m-end {
 justify-content: flex-end;
}

.m-around {
 justify-content: space-around;
}
.m-between {
 justify-content: space-between;
}
.m-center {
 justify-content: center;
}
/* 侧轴方向 */
.s-start {
 align-items: flex-start;
}
.s-end {
 align-items: flex-end;
}

.s-around {
 align-items: space-around;
}
.s-between {
 align-items: space-between;
}
.s-center {
 align-items: center;
}

JS

// pages/calendar/calendar.js

'use strict';

let choose_year = null,
 choose_month = null;
const conf = {
 data: {
 day: '',
 year: '',
 month: '',
 date: '2017-01',
 today: '',
 week: ['日', '一', '二', '三', '四', '五', '六'],
 calendar: {
  first: [],
  second: [],
  third: [],
  fourth: []
 },
 swiperMap: ['first', 'second', 'third', 'fourth'],
 swiperIndex: 1,
 showCaldenlar: false
 },
 onLoad() {
 const date = new Date()
  , month = this.formatMonth(date.getMonth() + 1)
  , year = date.getFullYear()
  , day = this.formatDay(date.getDate())
  , today = `${year}-${month}-${day}`
 let calendar = this.generateThreeMonths(year, month)

 this.setData({
  calendar,
  month,
  year,
  day,
  today,
  beSelectDate: today,
  date: `${year}-${month}`
 })
 },

 showCaldenlar() {
 this.setData({
  showCaldenlar: !this.data.showCaldenlar
 })
 },
 /**
 * 
 * 左右滑动
 * @param {any} e 
 */
 swiperChange(e) {
 const lastIndex = this.data.swiperIndex
  , currentIndex = e.detail.current
 let flag = false
  , { year, month, day, today, date, calendar, swiperMap } = this.data
  , change = swiperMap[(lastIndex + 2) % 4]
  , time = this.countMonth(year, month)
  , key = 'lastMonth'

 if (lastIndex > currentIndex) {
  lastIndex === 3 && currentIndex === 0
  ? flag = true
  : null
 } else {
  lastIndex === 0 && currentIndex === 3
  ? null
  : flag = true
 }
 if (flag) {
  key = 'nextMonth'
 }

 year = time[key].year
 month = time[key].month
 date = `${year}-${month}`
 day = ''
 if (today.indexOf(date) !== -1) {
  day = today.slice(-2)
 }

 time = this.countMonth(year, month)
 calendar[change] = null
 calendar[change] = this.generateAllDays(time[key].year, time[key].month)

 this.setData({
  swiperIndex: currentIndex,
  //文档上不推荐这么做,但是滑动并不会改变current的值,所以随之而来的计算会出错
  year,
  month,
  date,
  day,
  calendar
 })
 },
 /**
 * 
 * 点击切换月份,生成本月视图以及临近两个月的视图
 * @param {any} year 
 * @param {any} month 
 * @returns {object} calendar
 */
 generateThreeMonths(year, month) {
 let { swiperIndex, swiperMap, calendar } = this.data
  , thisKey = swiperMap[swiperIndex]
  , lastKey = swiperMap[swiperIndex - 1 === -1 ? 3 : swiperIndex - 1]
  , nextKey = swiperMap[swiperIndex + 1 === 4 ? 0 : swiperIndex + 1]
  , time = this.countMonth(year, month)
 delete calendar[lastKey]
 calendar[lastKey] = this.generateAllDays(time.lastMonth.year, time.lastMonth.month)
 delete calendar[thisKey]
 calendar[thisKey] = this.generateAllDays(time.thisMonth.year, time.thisMonth.month)
 delete calendar[nextKey]
 calendar[nextKey] = this.generateAllDays(time.nextMonth.year, time.nextMonth.month)
 return calendar
 },
 bindDayTap(e) {
 let { month, year } = this.data
  , time = this.countMonth(year, month)
  , tapMon = e.currentTarget.dataset.month
  , day = e.currentTarget.dataset.day
 if (tapMon == time.lastMonth.month) {
  this.changeDate(time.lastMonth.year, time.lastMonth.month)
 } else if (tapMon == time.nextMonth.month) {
  this.changeDate(time.nextMonth.year, time.nextMonth.month)
 } else {
  this.setData({
  day
  })
 }
 let beSelectDate = e.currentTarget.dataset.date;
 this.setData({
  beSelectDate,
  showCaldenlar: false
 })
 },
 bindDateChange(e) {
 if (e.detail.value === this.data.date) {
  return
 }

 const month = e.detail.value.slice(-2)
  , year = e.detail.value.slice(0, 4)

 this.changeDate(year, month)
 },
 prevMonth(e) {
 let { year, month } = this.data
  , time = this.countMonth(year, month)
 this.changeDate(time.lastMonth.year, time.lastMonth.month)
 },
 nextMonth(e) {
 let { year, month } = this.data
  , time = this.countMonth(year, month)
 this.changeDate(time.nextMonth.year, time.nextMonth.month)
 },
 /**
 * 
 * 直接改变日期
 * @param {any} year 
 * @param {any} month 
 */
 changeDate(year, month) {
 let { day, today } = this.data
  , calendar = this.generateThreeMonths(year, month)
  , date = `${year}-${month}`
 date.indexOf(today) === -1
  ? day = '01'
  : day = today.slice(-2)

 this.setData({
  calendar,
  day,
  date,
  month,
  year,
 })
 },
 /**
 * 
 * 月份处理
 * @param {any} year 
 * @param {any} month 
 * @returns 
 */
 countMonth(year, month) {
 let lastMonth = {
  month: this.formatMonth(parseInt(month) - 1)
 }
  , thisMonth = {
  year,
  month,
  num: this.getNumOfDays(year, month)
  }
  , nextMonth = {
  month: this.formatMonth(parseInt(month) + 1)
  }

 lastMonth.year = parseInt(month) === 1 && parseInt(lastMonth.month) === 12
  ? `${parseInt(year) - 1}`
  : year + ''
 lastMonth.num = this.getNumOfDays(lastMonth.year, lastMonth.month)
 nextMonth.year = parseInt(month) === 12 && parseInt(nextMonth.month) === 1
  ? `${parseInt(year) + 1}`
  : year + ''
 nextMonth.num = this.getNumOfDays(nextMonth.year, nextMonth.month)
 return {
  lastMonth,
  thisMonth,
  nextMonth
 }
 },
 currentMonthDays(year, month) {
 const numOfDays = this.getNumOfDays(year, month)
 return this.generateDays(year, month, numOfDays)
 },
 /**
 * 生成上个月应显示的天
 * @param {any} year 
 * @param {any} month 
 * @returns 
 */
 lastMonthDays(year, month) {
 const lastMonth = this.formatMonth(parseInt(month) - 1)
  , lastMonthYear = parseInt(month) === 1 && parseInt(lastMonth) === 12
  ? `${parseInt(year) - 1}`
  : year
  , lastNum = this.getNumOfDays(lastMonthYear, lastMonth) //上月天数
 let startWeek = this.getWeekOfDate(year, month - 1, 1) //本月1号是周几
  , days = []
 if (startWeek == 7) {
  return days
 }

 const startDay = lastNum - startWeek

 return this.generateDays(lastMonthYear, lastMonth, lastNum, { startNum: startDay, notCurrent: true })
 },
 /**
 * 生成下个月应显示天
 * @param {any} year 
 * @param {any} month
 * @returns 
 */
 nextMonthDays(year, month) {
 const nextMonth = this.formatMonth(parseInt(month) + 1)
  , nextMonthYear = parseInt(month) === 12 && parseInt(nextMonth) === 1
  ? `${parseInt(year) + 1}`
  : year
  , nextNum = this.getNumOfDays(nextMonthYear, nextMonth) //下月天数
 let endWeek = this.getWeekOfDate(year, month)  //本月最后一天是周几
  , days = []
  , daysNum = 0
 if (endWeek == 6) {
  return days
 } else if (endWeek == 7) {
  daysNum = 6
 } else {
  daysNum = 6 - endWeek
 }
 return this.generateDays(nextMonthYear, nextMonth, daysNum, { startNum: 1, notCurrent: true })
 },
 /**
 * 
 * 生成一个月的日历
 * @param {any} year 
 * @param {any} month 
 * @returns Array
 */
 generateAllDays(year, month) {
 let lastMonth = this.lastMonthDays(year, month)
  , thisMonth = this.currentMonthDays(year, month)
  , nextMonth = this.nextMonthDays(year, month)
  , days = [].concat(lastMonth, thisMonth, nextMonth)
 return days
 },
 /**
 * 
 * 生成日详情
 * @param {any} year 
 * @param {any} month 
 * @param {any} daysNum 
 * @param {boolean} [option={
 * startNum:1,
 * grey: false
 * }] 
 * @returns Array 日期对象数组
 */
 generateDays(year, month, daysNum, option = {
 startNum: 1,
 notCurrent: false
 }) {
 const weekMap = ['一', '二', '三', '四', '五', '六', '日']
 let days = []
 for (let i = option.startNum; i <= daysNum; i++) {
  let week = weekMap[new Date(year, month - 1, i).getUTCDay()]
  let day = this.formatDay(i)
  days.push({
  date: `${year}-${month}-${day}`,
  event: false,
  day,
  week,
  month,
  year
  })
 }
 return days
 },
 /**
 * 
 * 获取指定月第n天是周几 |
 * 9月第1天: 2017, 08, 1 |
 * 9月第31天:2017, 09, 0 
 * @param {any} year 
 * @param {any} month 
 * @param {number} [day=0] 0为最后一天,1为第一天
 * @returns number 周 1-7, 
 */
 getWeekOfDate(year, month, day = 0) {
 let dateOfMonth = new Date(year, month, 0).getUTCDay() + 1;
 dateOfMonth == 7 ? dateOfMonth = 0 : '';
 return dateOfMonth;
 },
 /**
 * 
 * 获取本月天数
 * @param {number} year 
 * @param {number} month 
 * @param {number} [day=0] 0为本月0最后一天的
 * @returns number 1-31
 */
 getNumOfDays(year, month, day = 0) {
 return new Date(year, month, day).getDate()
 },
 /**
 * 
 * 月份处理
 * @param {number} month 
 * @returns format month MM 1-12
 */
 formatMonth(month) {
 let monthStr = ''
 if (month > 12 || month < 1) {
  monthStr = Math.abs(month - 12) + ''
 } else {
  monthStr = month + ''
 }
 monthStr = `${monthStr.length > 1 ? '' : '0'}${monthStr}`
 return monthStr
 },
 formatDay(day) {
 return `${(day + '').length > 1 ? '' : '0'}${day}`
 }
}
Page(conf)

效果图

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • JavaScript中 this 的绑定指向规则

    JavaScript中 this 的绑定指向规则

    这篇文章主要介绍了JavaScript中 this 的绑定指向规则,this的指向问题存在各种各样的,关于如何绑定指向,下面文章作简单介绍需要的小伙伴可以参考一下
    2022-06-06
  • JavaScript学习笔记之基于定时器实现图片无缝滚动功能详解

    JavaScript学习笔记之基于定时器实现图片无缝滚动功能详解

    这篇文章主要介绍了JavaScript学习笔记之基于定时器实现图片无缝滚动功能,结合实例形式分析了javascript定时器与页面元素属性动态设置等相关操作技巧,需要的朋友可以参考下
    2019-01-01
  • 深入分析javascript中console命令

    深入分析javascript中console命令

    console对象是JavaScript的原生对象,它有点像Unix系统的标准输出stdout和标准错误stderr,可以输出各种信息用来调试程序,而且还提供了很多额外的方法,供开发者调用。它的常见用途有两个。显示网页代码运行时的错误信息。提供了一个命令行接口,用来与网页代码互动。
    2016-08-08
  • JS表单验证方法实例小结【电话、身份证号、Email、中文、特殊字符、身份证号等】

    JS表单验证方法实例小结【电话、身份证号、Email、中文、特殊字符、身份证号等】

    这篇文章主要介绍了JS表单验证方法,结合实例形式总结分析了常用的表单验证技巧,包括电话、身份证号、Email、中文、特殊字符、身份证号等验证方法,需要的朋友可以参考下
    2017-02-02
  • 图解JavaScript作用域链底层原理

    图解JavaScript作用域链底层原理

    当代码在一个环境中执行时,会创建变量对象的一个作用域链,作用域链的用途是保证对执行环境有权访问的所有变量和函数的有序访问,下面这篇文章主要给大家介绍了关于JavaScript作用域链底层原理的相关资料,需要的朋友可以参考下
    2021-11-11
  • 详解JavaScript中的属性和特性

    详解JavaScript中的属性和特性

    本文对JavaScript中对象的本质、对象与类的关系、对象与引用类型的关系;对象属性如何进行分类;属性中特性进行介绍。感兴趣的朋友可以看下
    2016-12-12
  • 能在网页中写字和能擦写的js程序

    能在网页中写字和能擦写的js程序

    从经典论坛发现的一个可以在网页中写字和有擦除功能的javascript代码
    2008-04-04
  • hash特点、hashchange事件介绍及其常见应用场景

    hash特点、hashchange事件介绍及其常见应用场景

    浅析hash特点、hashchange事件介绍及其常见应用场景(不同hash对应不同事件处理、移动端大图展示状态后退页面问题、原生轻应用头部后退问题、移动端自带返回按钮二次确认问题),hashchange和popstate事件触发条件
    2023-11-11
  • JS按字节截取字符长度实例

    JS按字节截取字符长度实例

    这篇文章主要介绍了JS按字节截取字符长度实例,有需要的朋友可以参考一下
    2013-11-11
  • JS实现在线Excel的附件上传与下载

    JS实现在线Excel的附件上传与下载

    在本地使用Excel时,经常会有需要在Excel中添加一些附件文件的需求,今天小编将为大家介绍如何使用前端HTML+JS+CSS技术通过超链接单元格的形式实现在线Excel附件上传、下载和修改的操作,需要的可以参考下
    2023-08-08

最新评论