Vue2 elementUI年份区间选择组件

一、显现效果

二、组件封装 yearPicker.vue

复制代码
<template>
  <div class="yearPicker" ref="yearPicker" :style="{ width: width + 'px' }">
    <div class="_inner labelText" :style="{ width: labelWidth + 'px' }">{{ labelText }}</div>
    <input class="_inner" ref="inputLeft" v-model.number="startShowYear" @focus="onFocus" type="text" @click="clickInput"
      name="yearInput" @input="checkStartInput($event)" placeholder="选择年份" />
    <span>{{ sp }}</span>
    <input class="_inner" ref="inputRight" v-model.number="endShowYear" @focus="onFocus" type="text" @click="clickInput"
      name="yearInput" @input="checkEndInput($event)" placeholder="选择年份" />
    <div class="_inner floatPanel" v-if="showPanel">
      <div class="_inner leftPanel">
        <div class="_inner panelHead">
          <i class="_inner el-icon-d-arrow-left" @click="onClickLeft"></i>
          {{ leftYearList[0] + "-" + leftYearList[9] }}
        </div>
        <div class="_inner panelContent">
          <div v-for="item in leftYearList" :class="{
            disabled: checkValidYear(item) != 0,
            oneSelected: item === startYear && oneSelected,
            startSelected: item === startYear,
            endSelected: item === endYear,
            _inner: true,
            betweenSelected: item > startYear && item < endYear
          }" :key="item">
            <a :class="{
              cell: true,
              _inner: true,
              selected: item === startYear || item === endYear
            }" @click="onClickItem(item)" @mouseover="onHoverItem(item)">
              {{ item }}
            </a>
          </div>
        </div>
      </div>
      <div class="_inner rightPanel">
        <div class="_inner panelHead">
          <i class="_inner el-icon-d-arrow-right" @click="onClickRight"></i>
          {{ rightYearList[0] + "-" + rightYearList[9] }}
        </div>
        <div class="_inner panelContent">
          <div :class="{
            disabled: checkValidYear(item) != 0,
            startSelected: item === startYear,
            endSelected: item === endYear,
            betweenSelected: item > startYear && item < endYear
          }" v-for="item in rightYearList" :key="item">
            <a :class="{
              cell: true,
              _inner: true,
              selected: item === endYear || item === startYear
            }" @click="onClickItem(item)" @mouseover="onHoverItem(item)">
              {{ item }}
            </a>
          </div>
        </div>
      </div>
    </div>
  </div>
</template>

<script>
const SELECT_STATE = {
  unselect: 0, //未选择
  selecting: 1, //选中一个
  selected: 2, //全部选中
};
export default {
  name: "YearPicker",
  computed: {
    oneSelected () {
      return (
        this.curState === SELECT_STATE.selecting && (this.startYear === this.endYear || this.endYear == null)
      );
    },
    startDate () {
      return this.startYear;
    },
    leftYearList () {
      return this.yearList.slice(0, 10);
    },
    rightYearList () {
      return this.yearList.slice(10, 20);
    }
  },
  props: {
    width: {
      default: 200,
    },
    labelWidth: {
      default: 80,
    },
    labelText: {
      default: "时间标签",
    },
    sp: {
      default: "至",
    },
    initYear: {
      default: null,
    },
  },
  data () {
    return {
      startShowYear: null,//输入框 开始年份
      endShowYear: null,//输入框 结束年份
      yearList: [], //选择框里可选择的年份
      showPanel: false, //选择框是否显示
      startYear: null, //选择框选中的开始年份
      endYear: null, //选择框选中的结束年份
      curYear: 0,
      curSelectedYear: 0,
      curState: SELECT_STATE.unselect,
    };
  },
  methods: {
    // 重置时间范围
    onReset () {
      this.startYear = null;
      this.endYear = null;
      this.startShowYear = null;
      this.endShowYear = null;
      this.$emit("updateTimeRange", {
        startYear: null,
        endYear: null,
      });
    },
    // 检查年份是否在可选区间内  是否禁用
    checkValidYear (iYear) {
      if (this.initYear) {
        if (iYear > this.initYear.endYear) {
          return 1
        } else if (iYear < this.initYear.startYear) {
          return -1
        }
      }
      return 0
    },
    // 开始年份 输入框校验  并将选择框对应开始年份选中
    checkStartInput (event) {
      if (isNaN(this.startShowYear)) {
        this.startShowYear = this.startYear;
      } else {
        this.startYear = this.startShowYear * 1;
      }
    },
    // 结束年份 输入框校验  并将选择框对应结束年份选中
    checkEndInput () {
      if (isNaN(this.endShowYear)) {
        this.endShowYear = this.endYear;
      } else {
        this.endYear = this.endShowYear * 1;
      }
    },
    // 选择框 鼠标悬停事件  选中年份范围
    onHoverItem (iYear) {
      if (this.checkValidYear(iYear) != 0) {
        return;
      }
      if (this.curState === SELECT_STATE.selecting) {
        let tmpStart = this.curSelectedYear;
        this.endYear = Math.max(tmpStart, iYear);
        this.startYear = Math.min(tmpStart, iYear);
      }
    },
    // 选择框 点击事件  选中年份范围
    onClickItem (iYear) {
      if (this.checkValidYear(iYear) != 0) {
        // 点击禁用年份  直接返回
        return;
      }
      if (this.curState === SELECT_STATE.unselect || this.curState === SELECT_STATE.selected) {
        // 点击未选择或全部选中状态  开始年份直接选中当前年份
        this.startYear = iYear;
        this.curSelectedYear = iYear;
        this.endYear = null;
        this.curState = SELECT_STATE.selecting;
      } else if (this.curState === SELECT_STATE.selecting) {
        // 处于选中一个,然后再点击 
        // 如果 开始年份和结束年份是一样  但 结束年份值为空 则 结束年份 == 开始年份
        if(this.startYear && !this.endYear){
          this.endYear = this.startYear;
        }
        this.endShowYear = this.endYear;
        this.startShowYear = this.startYear;
        this.curState = SELECT_STATE.selected;
        this.$emit("updateTimeRange", {
          startYear: this.startYear,
          endYear: this.endYear,
        });
        setTimeout(() => {
          //为动画留的时间,可优化
          this.showPanel = false;
        }, 300);
      }
    },
    //使 input 获取焦点
    onFocus () {
      this.$nextTick(() => {
        this.showPanel = true;
        if(this.startShowYear && this.endShowYear){
          // 点击选择框  已选择年份范围  则将选择框对应年份选中
          this.endYear = this.endShowYear;
          this.startYear = this.startShowYear;
          this.curState = SELECT_STATE.selected;
        }
      });
    },
    // 点击 input 阻止事件冒泡
    clickInput (e) {
      e.stopPropagation();
      return false;
    },
    // 点击空白区域关闭选择框
    closePanel (e) {
      if (!this.showPanel) {
        return;
      }
      if (typeof e.target.className !== "string" || e.target.className === "") {
        this.$nextTick(() => {
          this.changeYear();
          this.showPanel = false;
        });
        return;
      }
      if (
        e.target.className.indexOf("_inner") === -1 ||
        (e.target.name === "yearInput" &&
          e.target !== this.$refs.inputLeft &&
          e.target !== this.$refs.inputRight)
      ) {
        this.$nextTick(() => {
          this.changeYear();
          this.showPanel = false;
        });
      }

      e.stopPropagation();
      return false;
    },
    // 点击空白区域关闭选择框   时间检验回显
    changeYear () {
      if (!this.startYear || !this.endYear) {
        return;
      }
      if (this.startYear > this.endYear) {
        let tmp = this.endYear;
        this.endYear = this.startYear;
        this.startYear = tmp;
      }

      if (this.initYear) {
        this.startYear = Math.max(this.startYear, this.initYear.startYear)
        this.endYear = Math.min(this.endYear, this.initYear.endYear)
      }
      this.startShowYear = this.startYear;
      this.endShowYear = this.endYear;


      if (this.startYear && this.endYear) {
        this.$emit("updateTimeRange", {
          startYear: this.startYear,
          endYear: this.endYear + ""
        });
      } else {
        console.warn("WARN:年份不合法", this.startYear, this.endYear);
      }
    },
    // 点击左箭头  切换到上一个十年
    onClickLeft () {
      this.curYear = this.curYear * 1 - 10;
      this.updateYearList();
    },
    // 点击右箭头  切换到下一个十年
    onClickRight () {
      this.curYear = this.curYear * 1 + 10;
      this.updateYearList();
    },
    // 更新年份列表
    updateYearList () {
      let iStart = Math.floor(this.curYear / 10) * 10 - 10;
      iStart = iStart < 0 ? 0 : iStart;
      this.yearList = [];
      for (let index = 0; index < 20; index++) {
        this.yearList.push(iStart + index);
      }
    },
    //------------------对外接口------------------------
    // //直接传时间戳
    // setYear(startYearStamp, endYearStamp) {
    //     if (!isNaN(startYearStamp) && !isNaN(endYearStamp)) {
    //         let startYear = moment(startYearStamp).format("yyyy");
    //         let endYear = moment(endYearStamp).format("yyyy");
    //         this.startYear = startYear * 1;
    //         this.endYear = endYear * 1;
    //         this.endShowYear = endYear * 1;
    //         this.startShowYear = startYear * 1;
    //     }
    // },
  },
  created () {
    this.curYear = new Date().getFullYear();
    this.updateYearList();
  },
  beforeUnmount () {
    // 组件销毁时移除事件监听
    // 点击空白区域关闭选择框
    document.removeEventListener("click", this.closePanel.bind(this));
  },
  mounted () {
    // 组件挂载时添加事件监听
    // 点击空白区域关闭选择框
    document.addEventListener("click", this.closePanel.bind(this));
  },
};
</script>
<style lang="scss" scoped>
.yearPicker {
  font-size: 14px;
  display: flex;
  position: relative;
  transition: all 0.3s;

  input {
    text-align: center;
  }

  input:first-child {
    text-align: right;
  }

  background-color: #fff;

  .labelText {
    text-align: center;
  }

  span {
    padding: 0 8px;
    height: 38px;
    line-height: 38px;
  }

  border: 1px solid #eff1f3;
  height: 40px;
  line-height: 40px;
  border-radius: 4px;
  padding: 0 28px 0 8px;
  box-sizing: border-box;

  .floatPanel {
    >div {
      width: 50%;
    }

    padding: 0 16px;
    position: absolute;
    display: flex;
    background-color: #fff;
    z-index: 2000;
    border-radius: 4px;
    width: 650px;
    height: 250px;
    top: 40px;
    left: -10px;
    box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);

    .panelContent {
      display: flex;
      flex-wrap: wrap;
      width: 100%;
      height: calc(100% - 70px);

      .disabled {
        color: #ccc;
      }

      .oneSelected {
        border-top-right-radius: 24px;
        border-bottom-right-radius: 24px;
      }

      .startSelected {
        background-color: #f6f6f7;
        border-top-left-radius: 24px;
        border-bottom-left-radius: 24px;
      }

      .endSelected {
        background-color: #f6f6f7;
        border-top-right-radius: 24px;
        border-bottom-right-radius: 24px;
      }

      .betweenSelected {
        background-color: #f6f6f7;
      }

      >div {
        width: 75px;
        height: 48px;
        line-height: 48px;
        margin: 3px 0;
        // border-radius: 24px;
        text-align: center;

        a {
          display: inline-block;
          width: 60px;
          height: 36px;
          cursor: pointer;
          line-height: 36px;
          border-radius: 18px;
        }

        .selected {
          background-color: #3e77fc;
          color: #fff;
        }
      }
    }

    .panelHead {
      position: relative;
      height: 46px;
      line-height: 46px;
      text-align: center;

      i {
        position: absolute;
        cursor: pointer;

        &:hover {
          color: #3e77fc;
        }
      }
    }

    .rightPanel {
      padding-left: 8px;
    }

    .leftPanel .panelHead i {
      left: 20px;
      top: 15px;
    }

    .rightPanel .panelHead i {
      right: 20px;
      top: 15px;
    }
  }

  .floatPanel::before {
    content: "";
    height: 100%;
    position: absolute;
    left: 50%;
    width: 1px;
    border-left: 1px solid #e4e4e4;
  }
}

input {
  width: 60px;
  border: none;
  height: 40px;
  line-height: 40px;
  box-sizing: border-box;
  background-color: transparent;
}

input:focus {
  outline: none;
  background-color: transparent;
}

.yearPicker:hover {
  border-color: #3e77fc;
}

.dateIcon {
  position: absolute;
  right: 16px;
  top: 9px;
  color: #adb2bc;
}
</style>

三、组件使用

<template>

<div class="moneyType">

<div class="moneyType_con">

<el-form :inline="true" :model="formInline" class="demo-form-inline">

<el-form-item label="年度">

<YearPicker ref="yearPicker" labelText="" :label-width="0" :initYear="initYear" @updateTimeRange="updateStatisticYear"></YearPicker>

</el-form-item>

<el-form-item>

<el-button type="primary" icon="el-icon-search" @click="getData(1)">查询</el-button>

<el-button type="success" icon="el-icon-refresh" @click="onReset(1)">重置</el-button>

</el-form-item>

</el-form>

</div>

</div>

</template>

<script>

import YearPickerfrom "../components/yearPicker.vue";

export default {

components: {

YearPicker

},

data () {

return {

initYear: {

startYear: 1945,

endYear: new Date().getFullYear()

},//年份区间选择框初始时间

formInline: {

nd: '',

nds: '',

},

};

},

mounted () {

},

methods: {

// 更新时间范围

updateStatisticYear({startYear, endYear}){

this.formInline.nd = startYear

this.formInline.nds = endYear

},

// 查询

getData () {

},

// 重置

onReset () {

this.formInline = {

nd: '',

nds: '',

}

this.$refs.yearPicker.onReset()

this.getData();

},

},

};

</script>

四、实现功能

1、可选年份区间(区间外禁止点击和选择,同时校正输入)。

2、实现选择开始年和结束年一样,如2024-2024。

3、实现弹框显示时回显当前选择的开始年和结束年。

相关推荐
山塘小鱼儿3 小时前
JavaScript 性能优化实战大纲
javascript
笨笨狗吞噬者3 小时前
【uniapp】小程序体积优化,分包异步化
前端·微信小程序·uni-app
该用户已不存在3 小时前
Golang 上传文件到 MinIO?别瞎折腾了,这 5 个库拿去用
前端·后端·go
snows_l3 小时前
JavaScript 性能优化实战大纲
前端
asfdsfgas3 小时前
Angular CDK 响应式工具实操指南:自适应布局构建技巧
javascript·ecmascript·angular.js
文心快码BaiduComate3 小时前
文心快码3.5S开发古风射覆小游戏,它帅到我了!
前端·后端·程序员
佛系菜狗3 小时前
防抖和节流-防抖鸿蒙版本实现
前端
不一样的少年_4 小时前
老板问我:AI真能一键画广州旅游路线图?我用 MCP 现场开图
前端·人工智能·后端
东方石匠4 小时前
Javascript常见面试题
前端·javascript·面试