【vue自造轮子】日期范围选择器

组件介绍

1.样式类似element ui的日期范围选择器,但父组件中不强制使用input框来打开日期选择器。可以任意定义一个元素来控制日期选择器组件的显示隐藏,使用更为自由
2.返回的数据格式为开始和结束日期的Date对象,可以根据自己的需要转成任意格式

引用组件

在父组件中使用组件时,要绑定三个参数和两个事件

<choose-date-range
                :monthNum="6"
                :startDateStr="form.startDate"
                :endDateStr="form.endDate"
                @chooseDate="chooseDate" 
                @close="showDateRange = false">
</choose-date-range>

各参数和事件的含义为:
monthNum: 日期的范围从当前月份往后数monthNum个月,目前仅支持偶数,若传奇数,则默认加1
startDateStr:默认的开始日期,格式为"yyyy-mm-dd"
endDateStr: 默认的结束日期,格式为"yyyy-mm-dd"

chooseDate事件: 接收选择结果,参数为chooseDate (startDate, endDate),两个参数均为new Date()对象
close事件: 关闭日期选择器,根据自己的项目情况写

组件代码

<template>
  <div class="choose-date-range-components">
    <div class="opt-btns">
      <el-link @click="determine">确定</el-link>
      <el-link @click="cancel">取消</el-link>
    </div>
    <div class="tow-month">
      <div class="month-bar">
        <div class="header">
          <div class="pre-month">
            <i class="arrow el-icon-caret-left" @click="goPreMonth"></i>
          </div>
          <div class="month-date">{{curLeftYear}}年{{curLeftMonth}}月</div>
          <div class="next-month"></div>
        </div>
        <div class="weekdays">
          <div class="weekday">日</div>
          <div class="weekday">一</div>
          <div class="weekday">二</div>
          <div class="weekday">三</div>
          <div class="weekday">四</div>
          <div class="weekday">五</div>
          <div class="weekday">六</div>
        </div>
        <div class="days">
          <div class="day"
               @click="chooseDay(curLeftYear, curLeftMonth, day)"
               @mouseover="preChoose(curLeftYear, curLeftMonth, day)"
               @mouseout="preNotChoose(curLeftYear, curLeftMonth, day)"
               :class="{
                 'cannot-choose': judgeDisable(curLeftYear, curLeftMonth, day),
                 'in-range': judgeInRange(curLeftYear, curLeftMonth, day),
                 'in-pre-range': judgeInPreRange(curLeftYear, curLeftMonth, day),
                 'is-start': judgeStart(curLeftYear, curLeftMonth, day),
                 'is-end': judgeEnd(curLeftYear, curLeftMonth, day),
                 'is-today': judgeToday(curLeftYear, curLeftMonth, day)
               }"
               v-for="(day, index) in curLeftDays" :key="index">
            <div :class="{'day-inner': true,'chosen-day': judgeChosen(curLeftYear, curLeftMonth, day)}">
              {{day}}
            </div>
          </div>
        </div>
      </div>

      <div class="month-bar">
        <div class="header">
          <div class="pre-month"></div>
          <div class="month-date">{{curRightYear}}年{{curRightMonth}}月</div>
          <div class="next-month">
            <i class="arrow el-icon-caret-right" @click="goNextMonth"></i>
          </div>
        </div>
        <div class="weekdays">
          <div class="weekday">日</div>
          <div class="weekday">一</div>
          <div class="weekday">二</div>
          <div class="weekday">三</div>
          <div class="weekday">四</div>
          <div class="weekday">五</div>
          <div class="weekday">六</div>
        </div>
        <div class="days">
          <div class="day"
               @click="chooseDay(curRightYear, curRightMonth, day)"
               @mouseover="preChoose(curRightYear, curRightMonth, day)"
               @mouseout="preNotChoose(curRightYear, curRightMonth, day)"
               :class="{
                 'cannot-choose': judgeDisable(curRightYear, curRightMonth, day),
                 'in-range': judgeInRange(curRightYear, curRightMonth, day),
                 'in-pre-range': judgeInPreRange(curRightYear, curRightMonth, day),
                 'is-start': judgeStart(curRightYear, curRightMonth, day),
                 'is-end': judgeEnd(curRightYear, curRightMonth, day),
                 'is-today': judgeToday(curRightYear, curRightMonth, day)
               }"
               v-for="(day, index) in curRightDays" :key="index">
            <div :class="{'chosen-day': judgeChosen(curRightYear, curRightMonth, day)}">
              {{day}}
            </div>
          </div>
          <!--        <div class="day" v-for="(day, index) in curRightDays" :key="index">-->
          <!--          {{day}}-->
          <!--        </div>-->
        </div>
      </div>
    </div>
  </div>
</template>

<script>
export default {
  name: "ChooseDateRange",
  props: ['monthNum', 'startDateStr', 'endDateStr'],
  data () {
    return {
      leftDate: {},
      rightDate: {},
      chooseStartDateStr: '',
      chooseEndDateStr: '',
      preChooseDateStr: '',
      curLeftYear: '',
      curLeftMonth: '',
      curRightYear: '',
      curRightMonth: '',
      curLeftDays: [],
      curRightDays: []
    }
  },
  methods: {
    addMonth (date, monthNum = this.monthNum) {
      let year = date.getFullYear()
      let month = date.getMonth() + 1
      let year2 = year
      let month2 = month + monthNum
      if (month2 > 12) {
        year2 += 1
        month2 %= 12
      }
      return new Date (year2 + '-' + ('0' + month2).slice(-2) + '-01')
    },
    minusMonth (date, monthNum) {
      let year = date.getFullYear()
      let month = date.getMonth() + 1
      let year2 = year
      let month2 = month - monthNum
      if (month2 <= 0) {
        year2 -= 1
        month2 += 12
      }
      let dayNum = new Date(year2, month2, 0).getDate()
      return new Date (year2 + '-' + ('0' + month2).slice(-2) + '-' + dayNum + ' 00:00:00')
    },
    getDays (year, month) {
      let dayNum = new Date(year, month, 0).getDate()
      let days = Array.apply(null, new Array(new Date(year, month - 1, 1).getDay())).map(() => '')
      days = [...days, ...Array.apply(null, new Array(dayNum)).map((cur, index) => index + 1)]
      // console.log(days)
      return days
    },
    judgeChosen (year, month, day) {
      let tmpDateStr = year + '-' + ('0' + month).slice(-2) + '-' + ('0' + day).slice(-2)
      // console.log(tmpDateStr)
      if (tmpDateStr === this.chooseStartDateStr || tmpDateStr === this.chooseEndDateStr) {
        return true
      }
      return false
    },
    judgeDisable (year, month, day) {
      if (day === '') {
        return true
      }
      let tmpDate = new Date(year + '-' + month + '-' + day)
      if (tmpDate < this.leftDate || tmpDate > this.endDate) {
        return true
      }
      return false
    },
    judgeInRange(year, month, day) {
      if (day === '') {
        return false
      }
      let tmpDate = new Date(year + '-' + month + '-' + day)
      // console.log('.....................')
      // console.log(tmpDate)
      // console.log(new Date(this.chooseStartDateStr))
      // console.log(new Date(this.chooseEndDateStr))
      if (tmpDate >= new Date(this.chooseStartDateStr + ' 00:00:00') && tmpDate <= new Date(this.chooseEndDateStr + ' 00:00:00')) {
        return true
      }
      return false
    },
    judgeStart (year, month, day) {
      let tmpCurDateStr = year + '-' + ('0' + month).slice(-2) + '-' + ('0' + day).slice(-2)
      if (this.chooseStartDateStr && this.chooseEndDateStr) {
        // 开始日期和结束日期都选中
        if (tmpCurDateStr === this.chooseStartDateStr) {
          return true
        }
      } else if (this.chooseStartDateStr || this.chooseEndDateStr) {
        // 只选中一个, 要考虑hover的预选日期情况
        let tmpChooseDateStr = this.chooseStartDateStr || this.chooseEndDateStr
        if (!this.preChooseDateStr) {
          return true
        } else {
          if (this.preChooseDateStr >= tmpChooseDateStr) {
            if (tmpCurDateStr === tmpChooseDateStr) {
              return true
            }
          } else {
            if (tmpCurDateStr === this.preChooseDateStr) {
              return true
            }
          }
        }
      }
      return false
    },
    judgeEnd (year, month, day) {
      let tmpCurDateStr = year + '-' + ('0' + month).slice(-2) + '-' + ('0' + day).slice(-2)
      if (this.chooseStartDateStr && this.chooseEndDateStr) {
        // 开始日期和结束日期都选中
        if (tmpCurDateStr === this.chooseEndDateStr) {
          return true
        }
      } else if (this.chooseStartDateStr || this.chooseEndDateStr) {
        // 只选中一个, 要考虑hover的预选日期情况
        let tmpChooseDateStr = this.chooseStartDateStr || this.chooseEndDateStr
        if (!this.preChooseDateStr) {
          return true
        } else {
          if (this.preChooseDateStr <= tmpChooseDateStr) {
            if (tmpCurDateStr === tmpChooseDateStr) {
              return true
            }
          } else {
            if (tmpCurDateStr === this.preChooseDateStr) {
              return true
            }
          }
        }
      }
      return false
    },
    judgeToday (year, month, day) {
      let today = new Date()
      if (year === today.getFullYear() && month === today.getMonth() + 1 && day === today.getDate()) {
        return true
      }
      return false
    },
    chooseDay (year, month, day) {
      if (this.judgeDisable(year, month, day)) {
        return
      }
      let tmpDateStr = year + '-' + ('0' + month).slice(-2) + '-' + ('0' + day).slice(-2)
      console.log(tmpDateStr)
      if (this.chooseStartDateStr && this.chooseEndDateStr) {
        // 开始和结束日期都已选中
        if (tmpDateStr === this.chooseStartDateStr) {
          this.chooseStartDateStr = this.chooseEndDateStr
          this.chooseEndDateStr = ''
        } else if (tmpDateStr === this.chooseEndDateStr) {
          this.chooseEndDateStr = ''
        } else if (tmpDateStr > this.chooseEndDateStr) {
          this.chooseEndDateStr = tmpDateStr
        } else if (tmpDateStr < this.startDateStr) {
          this.chooseStartDateStr = tmpDateStr
        } else {
          this.chooseEndDateStr = tmpDateStr
        }
      } else if (this.chooseStartDateStr || this.chooseEndDateStr) {
        // 开始和结束日期仅选中一个
        if (tmpDateStr > this.chooseStartDateStr) {
          this.chooseEndDateStr = tmpDateStr
        } else if (tmpDateStr < this.chooseStartDateStr) {
          this.chooseEndDateStr = this.chooseStartDateStr
          this.chooseStartDateStr = tmpDateStr
        } else {
          this.chooseStartDateStr = ''
        }
      } else {
        // 开始和结束日期均未选中
        this.chooseStartDateStr = tmpDateStr
      }
    },
    preChoose (year, month, day) {
      if (!this.judgeDisable(year, month, day)) {
        this.preChooseDateStr = year + '-' + ('0' + month).slice(-2) + '-' + ('0' + day).slice(-2)
        // console.log(this.preChooseDateStr)
        this.$forceUpdate()
      }
    },
    preNotChoose (year, month, day) {
      this.preChooseDateStr = ''
    },
    judgeInPreRange (year, month, day) {
      // console.log(year, month, day)
      if (!this.preChooseDateStr) {
        return false
      }
      if (this.judgeDisable(year, month, day)) {
        return false
      }
      if ((this.chooseStartDateStr && this.chooseEndDateStr) || (!this.chooseStartDateStr && !this.chooseEndDateStr)) {
        return false
      } else {
        let tmpCurDateStr = year + '-' + ('0' + month).slice(-2) + '-' + ('0' + day).slice(-2)
        let tmpDateStr = this.chooseStartDateStr || this.chooseEndDateStr
        let left, right
        if (tmpDateStr < this.preChooseDateStr) {
          left = tmpDateStr
          right = this.preChooseDateStr
        } else {
          left = this.preChooseDateStr
          right = tmpDateStr
        }
        console.log(left, right, tmpCurDateStr)
        if (tmpCurDateStr <= right && tmpCurDateStr >= left) {
          // console.log(tmpCurDateStr)
          return true
        } else {
          return false
        }
      }
    },

    goPreMonth () {
      // console.log(new Date(this.curLeftYear, this.curLeftMonth - 1, 1))
      // console.log(this.minusMonth(new Date(this.curLeftYear, this.curLeftMonth - 1, 1),2))
      // console.log(this.leftDate)
      if (this.minusMonth(new Date(this.curLeftYear, this.curLeftMonth - 1, 1), 2) >= this.leftDate ) {
        let curLeftDate = this.minusMonth(new Date(this.curLeftYear, this.curLeftMonth - 1, 1), 2)
        let curRightDate = this.minusMonth(new Date(this.curLeftYear, this.curLeftMonth - 1, 1), 1)
        this.curLeftYear = curLeftDate.getFullYear()
        this.curLeftMonth = curLeftDate.getMonth() + 1
        this.curRightYear = curRightDate.getFullYear()
        this.curRightMonth = curRightDate.getMonth() + 1
        this.curLeftDays = this.getDays(this.curLeftYear, this.curLeftMonth)
        this.curRightDays = this.getDays(this.curRightYear, this.curRightMonth)
      }
    },
    goNextMonth () {
      if (this.addMonth(new Date(this.curRightYear, this.curRightMonth - 1, 1), 2) <= this.rightDate ) {
        let curLeftDate = this.addMonth(new Date(this.curRightYear, this.curRightMonth - 1, 1), 1)
        let curRightDate = this.addMonth(new Date(this.curRightYear, this.curRightMonth - 1, 1), 2)
        this.curLeftYear = curLeftDate.getFullYear()
        this.curLeftMonth = curLeftDate.getMonth() + 1
        this.curRightYear = curRightDate.getFullYear()
        this.curRightMonth = curRightDate.getMonth() + 1
        this.curLeftDays = this.getDays(this.curLeftYear, this.curLeftMonth)
        this.curRightDays = this.getDays(this.curRightYear, this.curRightMonth)
      }
    },
    determine () {
      if (this.chooseStartDateStr && this.chooseEndDateStr) {
        this.$emit('chooseDate', new Date(this.chooseStartDateStr), new Date(this.chooseEndDateStr))
      } else {
        this.$notify.warning({
          title: '提醒',
          message: '开始和结束日期都要选择!'
        })
      }
    },
    cancel () {
      this.$emit('close')
    }
  },
  created() {
    if (this.monthNum % 2) {
      this.monthNum += 1
    }
    let leftDate = new Date(new Date(new Date().getTime()).setHours(0,0,0,0))
    let rightDate = this.addMonth(leftDate, this.monthNum)
    this.leftDate = leftDate
    this.rightDate = rightDate
    console.log(leftDate, rightDate)
    let startDate = new Date(this.startDateStr)
    let endDate = new Date(this.endDateStr)
    console.log(startDate, endDate)
    if (startDate >= endDate || startDate < leftDate || endDate > rightDate) {
      this.chooseStartDateStr = ''
      this.chooseEndDateStr = ''
      // console.log('超范围', this.leftDate)
      this.curLeftYear = this.leftDate.getFullYear()
      this.curLeftMonth = this.leftDate.getMonth() + 1
      this.curRightYear = this.addMonth(this.leftDate, 1).getFullYear()
      this.curRightMonth = this.addMonth(this.leftDate, 1).getMonth() + 1
      this.$notify.warning({
        title: '提醒',
        message: '日期超出范围,请重新选择'
      })
    } else {
      this.chooseStartDateStr = this.startDateStr
      this.chooseEndDateStr = this.endDateStr
      // console.log('没超', this.chooseStartDateStr)
      let chooseStartDate = new Date(this.chooseStartDateStr)
      // console.log(chooseStartDate)
      // let chooseEndDate = new Date(this.chooseEndDateStr)
      this.curLeftYear = chooseStartDate.getFullYear()
      this.curLeftMonth = chooseStartDate.getMonth() + 1
      this.curRightYear = this.addMonth(chooseStartDate, 1).getFullYear()
      this.curRightMonth = this.addMonth(chooseStartDate, 1).getMonth() + 1
      // console.log(this.addMonth(chooseStartDate, 1))
    }
    this.curLeftDays = this.getDays(this.curLeftYear, this.curLeftMonth)
    this.curRightDays = this.getDays(this.curRightYear, this.curRightMonth)
  }
}
</script>

<style lang="less" scoped>
.choose-date-range-components {
  color: #000000;
  display: flex;
  flex-direction: column;
  box-shadow: 0 0 4px 0 rgba(117,117,117,0.5);
  //opacity: 1!important;
  border-radius: 2px;
  .opt-btns {
    padding: 10px;
    display: flex;
    justify-content: space-between;
    border-bottom: 1px solid #dddddd;
  }
  .tow-month {
    display: flex;
    .month-bar {
      width: 50%;
      border-right: 1px solid #dddddd;
      padding: 0 10px;
      .header {
        display: flex;
        justify-content: space-between;
        padding: 5px 0;
        .arrow {
          cursor: pointer;
        }
      }
      .weekdays {
        display: flex;
        .weekday {
          width: 40px;
          text-align: center;
        }
      }
      .days {
        display: flex;
        flex-wrap: wrap;
        .day {
          width: 40px;
          text-align: center;
          height: 40px;
          line-height: 30px;
          cursor: pointer;
          padding: 5px;
          .day-inner {
            width: 100%;
            height: 100%;
          }
          .chosen-day {
            //width: 100%;
            //height: 100%;
            border-radius: 20px;
            //border: 1px solid red;
            background-color: #409eff;
            color: #FFFFFF;
          }
        }
        .cannot-choose {
          color: #dddddd;
          cursor: auto;
        }
        .in-range {
          background-color: #f2f6fc;
        }
        .in-pre-range {
          background-color: #f2f6fc;
        }
        .is-start {
          border-radius: 20px 0 0 20px;
        }
        .is-end {
          border-radius: 0 20px 20px 0!important;
        }
        .is-today {
          color: #409eff;
        }
      }
    }
    .month-bar:last-child {
      border-right: none;
    }
  }
}
</style>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容