字节byte-week-picker(修改-反选bug解决)

<template>
    <div class="weektime" id="timecontainer">
        <div v-if="selectBoxDashed" class="select-box-dashed" ref="selectBoxDashed" @mousemove="handleMousemove"></div>
        <div class="weektime-main">
            <div class="weektime-hd">
                <div class="weektime-hd-title font-bold">星期 / 时间</div>
                <div class="weektime-hd-con">
                    <div class="weektime-hd-con-top">
                        <div class="weektime-date-range font-bold">00:00 - 12:00</div>
                        <div class="weektime-date-range font-bold">12:00 - 24:00</div>
                    </div>
                    <div class="weektime-hd-con-bottom">
                        <span class="weektime-date-cell" v-for="hour in 24" :key="hour">{{hour-1}}</span>
                    </div>
                </div>
            </div>
            <div class="weektime-bd" >
                <div class="week-body">
                    <div v-for="week in weekDays" :key="week" class="week-item">{{week}}</div>
                </div>
                <div class="time-body" @mousedown="handleMousedown" @mouseup="handleMouseup" @mousemove="handleMousemove">
                    <el-tooltip
                            v-for="(i,key) in weekTimes"
                            :key="key"
                            :data-index="key"
                            :content="tiptxt(key)"
                            :open-delay="800"
                            placement="top"
                            effect="light"
                    >
                        <div class="time-cell" :class="{'active':list[key]==='1'}"></div>
                    </el-tooltip>
                </div>
            </div>
        </div>
        <div class="weektime-help">
            <div class="weektime-help-tx">
                <div class="weektime-help-bd">
                    可移动鼠标选择时间段
                </div>
                <div class="weektime-help-ft" @click="initList()">清空</div>
            </div>
            <div class="weektime-help-select">
                <p v-for="(week,key) in weekDays" :key="key" v-show="showTimeText[key]">
                    <span class="weektime-help-week-tx">{{week+":"}}</span>
                    <span>{{showTimeText[key]}}</span>
                </p>
            </div>
        </div>
    </div>
</template>

<script>
import Tooltip from 'element-ui/lib/tooltip.js'
import 'element-ui/lib/theme-chalk/tooltip.css';
import 'element-ui/packages/theme-chalk/src/common/transition.scss';

const DayTimes = 24 * 2;

export default {
    name: "byte-weektime-picker",
    components: {
        "el-tooltip": Tooltip
    },
    props: {
        value: String
    },
    watch: {
        value(n) {
          if (n.split('') === this.list.join('')) return;
            this.initList(n);
        }
    },
    data() {
        return {
            isMove: false,
            list: [],
            weekTimes: 7 * DayTimes,
            weekDays: ['星期一', '星期二', '星期三', '星期四', '星期五', '星期六', '星期日'],
            timeTextList: [], //显示的时间数组 ["00:00","00:30","01:00",...]
            startIndex: 0,
            axis: {},
            preViewIndex: [],
            showTimeText: [],
            selectBoxDashed: false,
            startX: null,
            startY: null,
            initx: null,
            scrollX:null,
            scrollY:null,
            inity: null
        }
    },
    methods: {
        /**
         * 鼠标停留时提示当前时间段
         */
        tiptxt(index) {
            let timeIndex = index % DayTimes;
            let weekIndex = ~~(index / DayTimes);
            return `${this.weekDays[weekIndex]} ${this.timeTextList[timeIndex]}~${this.timeTextList[timeIndex + 1]}`
        },
        /**
         * 初始化显示的时间数组
         * @return {Array} ["00:00","00:30","01:00",...]
         */
        initTimeText() {
            let timeTextList = [], hours = [], minutes = ['00', '30'];
            for (let i = 0; i <= 24; i++) {
                i < 10 ? hours.push('0' + i) : hours.push(i.toString())
            }
            for (const hour of hours) {
                for (const minute of minutes) {
                    timeTextList.push(`${hour}:${minute}`)
                }
            }
            return timeTextList
        },
        handleMousedown(event) {
            this.isMove = true;
            this.startIndex = event.target.getAttribute('data-index');
            this.axis.startx = this.startIndex % DayTimes;
            this.axis.starty = ~~(this.startIndex / DayTimes);

            this.selectBoxDashed = true
            //  设置选框的初始位置
            this.startX = event.x + this.scrollX || event.clientX + this.scrollX
            this.startY = event.y + this.scrollY || event.clientY + this.scrollY
        },
        handleMouseup(event) {
            this.handleMousemove(event);
            this.resetMousemove()
            this.selectBoxDashed = false
        },
        handleMousemove(event) {
            if (this.selectBoxDashed) {
                //  根据鼠标移动,设置选框的位置、宽高
                this.initx = event.x + this.scrollX || event.clientX + this.scrollX
                this.inity = event.y + this.scrollY || event.clientY + this.scrollY

                //  暂存选框的位置及宽高,用于将 select-item 选中
                this.left = Math.min(this.initx, this.startX)
                this.top = Math.min(this.inity, this.startY)
                this.width = Math.abs(this.initx - this.startX)
                this.height = Math.abs(this.inity - this.startY)

                this.$refs.selectBoxDashed.style.left = `${this.left}px`
                this.$refs.selectBoxDashed.style.top = `${this.top}px`
                this.$refs.selectBoxDashed.style.width = `${this.width}px`
                this.$refs.selectBoxDashed.style.height = `${this.height}px`
            }



            if (!this.isMove) return;
            let index = event.target.getAttribute('data-index');
            if (index !== null) {
              this.axis.endx = index % DayTimes;
              this.axis.endy = ~~(index / DayTimes);
              this.preViewIndex = this.getSelectIndex()
            }
        },
        resetMousemove() {
            if (!this.isMove) return;
            this.setSelectIndex(this.preViewIndex);
            this.isMove = false;
            this.axis = {};
            this.preViewIndex = [];
            this.selectBoxDashed = false
        },
        /**
         * 获取拖动鼠标选择的index数组
         */
        getSelectIndex() {
            let indexList = [],
                newAxis = {
                    startx: Math.min(this.axis.startx, this.axis.endx),
                    starty: Math.min(this.axis.starty, this.axis.endy),
                    endx: Math.max(this.axis.startx, this.axis.endx),
                    endy: Math.max(this.axis.starty, this.axis.endy)
                }
            for (let y = newAxis.starty; y <= newAxis.endy; y++) {
                for (let x = newAxis.startx; x <= newAxis.endx; x++) {
                    indexList.push(x + y * DayTimes)
                }
            }
            return indexList
        },
        /**
         * 设置选择的时间段并赋给绑定的值
         * @param {Array} indexList 选择的index数组
         */
        setSelectIndex(indexList) {
            if (!Array.isArray(indexList)) return;
            let listLength = indexList.length;
            let newData = this.list[this.startIndex] === '1' ? '0' : '1';
            for (let i = 0; i < listLength; i++) {
                this.list.splice(indexList[i], 1, newData);
            }

            this.$emit('input', this.list.join(''));
            this.showSelectTime(this.list);
        },
        /**
         * 展示选择的时间段
         * @param {Array} list 已选择的list数组
         */
        showSelectTime(list) {
            if (!Array.isArray(list)) return;
            let weeksSelect = [], listlength = list.length;
            this.showTimeText = [];
            if (listlength === 0) return;
            // 把 336长度的 list 分成 7 组,每组 48 个
            for (var i = 0; i < listlength; i += DayTimes) {
                weeksSelect.push(list.slice(i, i + DayTimes));
            }
            weeksSelect.forEach(item => {
                this.showTimeText.push(this.getTimeText(item))
            });
        },
        getTimeText(arrIndex) {
            if (!Array.isArray(arrIndex)) return "";

            /*方法一 matchAll 正则匹配 (速度较慢) */
            // let strIndex = arrIndex.join('');
            // let arrMatches = Array.from(strIndex.matchAll(/1+/g));
            // let timeText = "";
            // arrMatches.forEach(value => {
            //   timeText += this.timeTextList[value.index];
            //   timeText += '~' + this.timeTextList[value.index + value[0].length] + '、';
            // })
            /*方法一 end */

            /**方法二 循环 (速度是方法一的十倍+)*/
            let timeLength = arrIndex.length,
                isSelect = false,
                timeText = "";
            arrIndex.forEach((value, index) => {
                if (value === '1') {
                    if (!isSelect) {
                        timeText += this.timeTextList[index]
                        isSelect = true;
                    }
                    if (index === timeLength - 1) timeText += '~' + this.timeTextList[index + 1] + '、';
                } else {
                    if (isSelect) {
                        timeText += '~' + this.timeTextList[index] + '、'
                        isSelect = false;
                    }
                }
            })
            /*方法二 end */

            return timeText.slice(0, -1)
        },
        initList(value) {
            let reg = new RegExp("^[01]{" + this.weekTimes + "}$");
            if (value && reg.test(value)) {
                this.list = value.split('');
                this.showSelectTime(this.list);
                return
            }
            this.list = new Array(this.weekTimes).fill('0');
            this.$emit('input', this.list.join(''));
            this.showSelectTime(this.list);
        },
    },
    destroyed() {
        document.removeEventListener('mouseup', this.resetMousemove)
    },
    created() {
        this.timeTextList = this.initTimeText();
        document.addEventListener('mouseup', this.resetMousemove);
        this.initList(this.value);
    }
}
</script>

<style scoped>
    div,
    span,
    p {
        margin: 0;
        padding: 0;
        border: 0;
        font-weight: normal;
        vertical-align: baseline;
        -webkit-tap-highlight-color: transparent;
        -ms-tap-highlight-color: transparent;
        -moz-box-sizing: border-box;
        -webkit-box-sizing: border-box;
        box-sizing: border-box;
    }
    .weektime {
        width: 658px;
        font-size: 14px;
        line-height: 32px;
        color: #515a6e;
        user-select: none;
        margin: 100px auto;
        background: white;
        border-radius: 5px;
    }
    .weektime .weektime-main {
        border: 1px solid #dcdee2;
        position: relative;
        border-radius: 5px 5px 0 0;
        overflow: hidden;
    }
    .weektime .weektime-hd {
        display: flex;
        background: white;
    }
    .weektime .weektime-hd-title {
        display: flex;
        align-items: center;
        padding: 0 6px;
        width: 80px;
        height: 65px;
    }
    .font-bold {
        font-weight: bold;
        font-size: 12px;
        color: black;
    }
    .weektime .weektime-hd-con {
        flex: 1;
        display: flex;
        -webkit-box-orient: vertical;
        flex-direction: column;
    }
    .weektime .weektime-hd-con-top {
        display: flex;
        border-bottom: 1px solid #dcdee2;
    }
    .weektime .weektime-date-range {
        width: 288px;
        height: 42px;
        line-height: 42px;
        text-align: center;
        border-left: 1px solid #dcdee2;
    }
    .weektime .weektime-hd-con-bottom {
        display: flex;
    }
    .weektime .weektime-date-cell {
        width: 24px;
        height: 32px;
        line-height: 32px;
        text-align: center;
        border-left: 1px solid #dcdee2;
    }
    .weektime .weektime-bd {
        display: flex;
    }
    .weektime .week-body {
        width: 80px;
        flex-shrink: 0;
        background: white;
    }
    .weektime .week-item {
        border-top: 1px solid #dcdee2;
        text-align: center;
        height: 30px;
        line-height: 30px;
    }
    .weektime .time-body {
        width: 576px;
        height: 210px;
        display: flex;
        flex-wrap: wrap;
        align-items: flex-start;
        position: relative;
    }
    .weektime .time-cell {
        position: relative;
        width: 12px;
        height: 30px;
        border-left: 1px solid #efefef;
        border-top: 1px solid #efefef;
        overflow: hidden;
        transition: background 0.3s ease;
        outline-width: 0;
    }
    .time-cell:hover {
        background: #ebebeb;
    }
    .weektime .time-cell.active {
        background: #2d8cf0;
    }
    .weektime .time-cell::after {
        content: "";
        position: absolute;
        top: 0;
        left: 0;
        right: 0;
        bottom: 0;
        background: transparent;
        opacity: 0.5;
        transition: background 888ms ease;
        z-index: 9000;
    }
    .weektime .pre-active::after {
        background: #113860;
    }
    .time-area {
        width: 576px;
        height: 210px;
        position: absolute;
        top: 0;
        left: 0;
        z-index: 100;
        background: transparent;
    }
    .weektime .weektime-help {
        width: 658px;
        border: 1px solid #dcdee2;
        border-top: none;
        padding: 5px 15px;
        border-radius: 0 0 5px 5px;
    }
    .weektime .weektime-help-tx {
        display: flex;
        align-items: center;
        justify-content: space-between;
    }

    .weektime .weektime-help-week-tx {
        color: #999;
    }

    .weektime .weektime-help-bd {
        display: flex;
        align-items: center;
        -webkit-box-pack: start;
        -ms-flex-pack: start;
        justify-content: flex-start;
        padding: 4px 0;
    }
    .weektime .weektime-help .color-box {
        width: 14px;
        height: 20px;
        background: #fff;
        border: 1px solid #dddddd;
        display: block;
        margin-right: 6px;
    }
    .weektime .weektime-help-bd .color-box.color-active {
        background: #2d8cf0;
    }
    .weektime .weektime-help .text-box {
        margin-right: 15px;
    }
    .weektime .weektime-help .weektime-help-ft {
        color: #2d8cf0;
        cursor: pointer;
    }
    .select-box-dashed{
        position: absolute;
        width: 0;
        height: 0;
        padding: 0px;
        margin: 0px;
        border: 1px dashed #0099ff;
        background-color: #c3d5ed;
        opacity: 0.5;
        filter: alpha(opacity=50);
        font-size: 0px;
        z-index: 99999;
    }
</style>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容