VUE时间范围选择的组件

最近手写了一个框选时间的组件,可以按半个小时为单位选择时间

 

代码部分

<template>
    <div>
      <div class="byted-schedule">
        <table :key="itemKey" class="byted-schedule-calendar-table">
          <thead>
            <tr>
              <th rowspan="8" class="byted-schedule-week-td">星期/事件</th>
              <th colspan="24">00:00 - 12:00</th>
              <th colspan="24">12:00 - 24:00</th>
            </tr>
            <tr>
              <td v-for="index of 24" :key="index" colspan="2">
                {{ index - 1 }}
              </td>
            </tr>
          </thead>
          <tbody @mouseleave="handelMouseLeave">
            <div v-show="is_show_mask" class="mask" :style="'width:' + mask_width + 'left:' + mask_left + 'height:' + mask_height + 'top:' + mask_top" />
            <tr v-for="(item, weekIndex) in week" :key="item">
              <td @mouseenter="handelMouseEnter">{{ item }}</td>
              <td
                v-for="timeIndex of timeLength"
                :ref="`td${weekIndex * timeLength + timeIndex - 1}`"
                :key="timeIndex"
                :colspan="48 / timeLength"
                :data-week="weekIndex"
                :data-time="timeIndex"
                :class="
                  timeArray[weekIndex * timeLength + timeIndex - 1] === '1'
                    ? 'selected'
                    : ''
                "
                class="byted-schedule-calendar-atom-time"
                @mousedown="handelMouseDown(weekIndex, timeIndex - 1,$event)"
                @mousemove="handelMouseMove(weekIndex, timeIndex - 1,$event)"
                @mouseup="handelMouseUp(weekIndex, timeIndex - 1,$event)"
              />
            </tr>
            <tr @mouseenter="handelMouseEnter">
              <td colspan="49">
                <div style="display: flex; justify-content: space-between">
                  <span style="margin-left: 10px">已选择时间段</span>
                  <a
                    :underline="false"
                    style="margin-right: 10px;color: #1890FF;cursor:pointer"
                    @click="clearAllTime"
                  >清空</a>
                </div>
                <div style="text-align: start; padding: 0 10px">
                  <p v-for="(item, key) in timeObject" :key="key">
                    <span v-if="item.length" style="margin-right: 10px">{{
                      key
                    }}</span>
                    <span v-if="item.length">{{ item.join("、") }}</span>
                  </p>
                </div>
              </td>
            </tr>
          </tbody>
        </table>
      </div>
    </div>
  </template>
  
  <script>
  export default {
    name:'time-choice-range',
    props: {
      value: {
        type: String,
        default: ''
      },
      timeWidth: {
        type: Number,
        default: 7
      },
      timeLength: {
        type: Number,
        default: 48
      }
    },
    data() {
      return {
        is_show_mask: false,
        startWeekIndex: null,
        startTimeIndex: null,
        endWeekIndex: null,
        endTimeIndex: null,
        week: [
          '星期一',
          '星期二',
          '星期三',
          '星期四',
          '星期五',
          '星期六',
          '星期日'
        ],
        cellStyle: {
          // color: 'red'
        },
        timeArray: [],
        itemKey: '',
        timeObject: {
          星期一: [],
          星期二: [],
          星期三: [],
          星期四: [],
          星期五: [],
          星期六: [],
          星期日: []
        },
        box_screen_left: 0, // 定位td的left位置
        box_screen_top: 0, // 定位td的top位置
        start_x: 0,
        start_y: 0,
        end_x: 0,
        end_y: 0
      }
    },
    created() {
      if (this.value) {
        this.timeArray = this.value.split('')
        for (var weekIndex = 0; weekIndex < this.timeWidth; weekIndex++) {
          this.handelClick(weekIndex)
        }
      } else {
        this.timeArray = []
        for (var i = 0; i < this.timeWidth * this.timeLength; i++) {
          this.timeArray[i] = '0'
        }
      }
    },
    computed: {
      mask_width() {
        return `${Math.abs(this.end_x - this.start_x)}px;`
      },
      mask_height() {
        return `${Math.abs(this.end_y - this.start_y)}px;`
      },
      mask_left() {
        return `${Math.min(this.start_x, this.end_x) - this.box_screen_left}px;`
      },
      mask_top() {
        return `${Math.min(this.start_y, this.end_y) - this.box_screen_top}px;`
      }
    },
    mounted() {
      window.addEventListener('scroll', this.handleScroll)
    },
    methods: {
      handleScroll() {
        const dom_box = document.querySelector('.byted-schedule-calendar-atom-time')
        this.box_screen_left = dom_box.getBoundingClientRect().left - 76
        this.box_screen_top = dom_box.getBoundingClientRect().top
      },
      handelClick(weekIndex, timeIndex) {
        var rowStartIndex = weekIndex * this.timeLength
        var i = 0
        var startAndEndTimeList = [] // 记录开始和结束时间
        while (i < this.timeLength) {
          if (this.timeArray[rowStartIndex + i] === '1') {
            var item = {
              startTime: null,
              endTime: null
            }
            item.startTime = i * (24 / this.timeLength)
            var obj = this.findEndTime(weekIndex, i)
            i = obj.index
            item.endTime = obj.endTime
            startAndEndTimeList.push(item)
          }
          i++
        }
        var checkedTimeList = []
        for (let k = 0; k < startAndEndTimeList.length; k++) {
          var start = this.numToTime(startAndEndTimeList[k].startTime)
          var end = this.numToTime(startAndEndTimeList[k].endTime)
          var str = start + '~' + end
          checkedTimeList.push(str)
        }
        this.timeObject[this.week[weekIndex]] = JSON.parse(
          JSON.stringify(checkedTimeList)
        )
      },
      findEndTime(weekIndex, timeIndex) {
        var index = weekIndex * this.timeLength + timeIndex
        var obj = {}
        obj.index = timeIndex
        if (this.timeArray[index] === '0') {
          obj.endTime = timeIndex * (24 / this.timeLength)
          return obj
        }
        // 如果选择了当天最后时段
        if (timeIndex === this.timeLength - 1 && this.timeArray[index] === '1') {
          obj.endTime = this.timeLength * (24 / this.timeLength)
          return obj
        }
        return this.findEndTime(weekIndex, timeIndex + 1)
      },
      numToTime(num) {
        var hour = parseInt(num)
        var minute = num % 1
        hour = hour < 10 ? '0' + hour : hour + ''
        minute = minute === 0 ? '00' : minute * 60 + ''
        return hour + ':' + minute
      },
      findStartAndEndTime(weekIndex, timeIndex) {
        var startX
        var startY
        var endX
        var endY
        startX =
          this.startWeekIndex < weekIndex ? this.startWeekIndex : weekIndex
        startY =
          this.startTimeIndex < timeIndex ? this.startTimeIndex : timeIndex
        endX = this.startWeekIndex < weekIndex ? weekIndex : this.startWeekIndex
        endY = this.startTimeIndex < timeIndex ? timeIndex : this.startTimeIndex
        return {
          startX,
          startY,
          endX,
          endY
        }
      },
      handelMouseDown(weekIndex, timeIndex, event) {
        this.handleScroll()// 获取一下组件在视窗的位置
        this.startWeekIndex = weekIndex
        this.startTimeIndex = timeIndex
        // 选择框
        this.is_show_mask = true
        this.start_x = event.clientX
        this.start_y = event.clientY
        this.end_x = event.clientX
        this.end_y = event.clientY
      },
      handelMouseMove(weekIndex, timeIndex, event) {
        this.endWeekIndex = weekIndex
        this.endTimeIndex = timeIndex
        if (this.startWeekIndex !== null) {
          // event.path[0].style.background='#7594ef'
          const { startX, endX, startY, endY } = this.findStartAndEndTime(
            weekIndex,
            timeIndex
          )
          for (var i = startX; i <= endX; i++) {
            for (var j = startY; j <= endY; j++) {
              this.$refs[`td${i * this.timeLength + j}`][0].style.background =
                '#7594ef'
            }
          }
        }
        this.end_x = event.clientX
        this.end_y = event.clientY
      },
      handelMouseLeave(event) {
        if (this.startWeekIndex !== null) {
          this.handelMouseUp(this.endWeekIndex, this.endTimeIndex)
        }
      },
      handelMouseEnter(event) {
        if (this.startWeekIndex !== null) {
          this.handelMouseUp(this.endWeekIndex, this.endTimeIndex)
        }
      },
      handelMouseUp(weekIndex, timeIndex) {
        if (this.startWeekIndex !== null) {
          const { startX, endX, startY, endY } = this.findStartAndEndTime(
            weekIndex,
            timeIndex
          )
          var value =
            this.timeArray[
              this.startWeekIndex * this.timeLength + this.startTimeIndex
            ]
          for (var i = startX; i <= endX; i++) {
            for (var j = startY; j <= endY; j++) {
              this.timeArray[i * this.timeLength + j] = value === '1' ? '0' : '1'
              this.itemKey = Math.random()
              this.handelClick(i, j)
            }
          }
        }
        this.startWeekIndex = null
        this.startTimeIndex = null
        this.endWeekIndex = null
        this.endTimeIndex = null
        this.$emit('input', this.timeArray.join(''))
        this.is_show_mask = false
        this.resSetXY()
      },
      resSetXY() {
        this.start_x = 0
        this.start_y = 0
        this.end_x = 0
        this.end_y = 0
      },
      clearAllTime() {
        for (var i = 0; i < this.timeWidth * this.timeLength; i++) {
          this.timeArray[i] = '0'
        }
        this.timeObject = {
          星期一: [],
          星期二: [],
          星期三: [],
          星期四: [],
          星期五: [],
          星期六: [],
          星期日: []
        }
        this.$emit('input', '')
      }
    }
  }
  </script>
  
  <style scoped src="../assets/index.css">
  
  </style>
  

css

.byted-schedule {
    user-select: none;
  }
  .byted-schedule table {
    border: 1px solid #ebebeb;
    border-collapse: collapse;
    table-layout: fixed;
    width: 100%;
    font-size: 12px;
  }
  .byted-schedule-calendar-table {
    border-collapse: separate;
    border-radius: 4px;
  }
  .byted-schedule-calendar-table tr td,
  .byted-schedule-calendar-table tr th {
    border-left: none;
    border-top: none;
    border-bottom: 1px solid;
    border-bottom-color: #ebebeb;
    border-right: 1px solid;
    border-right-color: #ebebeb;
    font-size: 14px;
    text-align: center;
    min-width: 11px;
    line-height: 1.8em;
    /* transition: background 200ms ease; */
    color: #333333;
    background: transparent;
    padding: 0;
  }
  .byted-schedule-week-td {
    width: 75px;
    padding: 20px 0;
  }
  .byted-schedule-calendar-table .selected {
    background: #1890ff !important;
  }
  .byted-schedule-calendar-atom-time:hover {
    background: #edf1fd;
  }
  tbody{
    position: relative;
      overflow: hidden;
  }
  .mask {
      position: absolute;
      background: red;
      opacity: 0.4;
      z-index: 999;
  }

npm

1、npm install time-choice-range

2、main.js:引入import time-choice-range from 'time-choice-range'

Vue.use(time-choice-range)

3.    <time-choice-range></time-choice-range>

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值