Vue实现类似于word插入表格时选中行列的效果

首先看一下效果:

实现的思路:
  1. 先设置一个容器的,容器中放上10×10的小格子,同时监听容器的进入和离开方法。
  2. 每个小格子上设置鼠标进入的方法,同时传入当前的序号,计算出当前的行和列,改变背景颜色。
  3. 监听容器的是从那个位置进入,只有从左边和上边进入有效。
相关的核心代码:
  • 判断鼠标移入元素的方向——上下左右,核心代码:
var direction = Math.round((((Math.atan2(y, x) * (180 / Math.PI)) + 180) / 90) + 3) % 4;

相关解释参考这篇文章:

https://www.cnblogs.com/liuyanxia/p/5666892.html

相关代码:

// 判断鼠标是否从左上进入
direct(e) {
    const x = e.offsetX - 230 / 2; //鼠标进入的X坐标-盒子宽度的一半
    const y = e.offsetY - 230 / 2; //鼠标进入的y坐标-盒子宽度的一半
    const direction =
          (Math.round((Math.atan2(y, x) * (180 / Math.PI) + 180) / 90) + 3) % 4;
    if (direction === 0 || direction === 3) {
        this.isLeftTop = true;
    } else {
        this.isLeftTop = false;
    }
}
  • 判断当前滑入到那个小窗格,改变背景颜色
// 获取鼠标进入方格的位置,进行背景渲染
getMousePlace(index) {
    this.isOn = true;
    if (!this.isLeftTop || this.isSelected) {
        return;
    }
    const x = Math.floor(index % 10);
    const y = Math.floor(index / 10);
    const children = this.$refs.wallRow.children;
    for (let i = 0; i < this.divList.length; i++) {
        const childrenx = Math.floor(i % 10);
        const childreny = Math.floor(i / 10);
        if (childrenx < x + 1 && childreny < y + 1) {
            children[i].className = "bgColor";
            this.rows = x + 1;
            this.cols = y + 1;
        } else {
            children[i].className = "";
        }
    }
}
  • 点击时关闭绘制状态
// 选定行列
selected(index) {
    this.isSelected = true;
    const data = {
        rows: this.rows,
        cols: this.cols
    };
}
整体代码如下:
<template>
  <div class="custom-style">
    <div class="word">
      <span>{{ rows }}×{{ cols }}</span>
    </div>
    <ul class="wall-row" ref="wallRow" @mouseenter="direct($event)" @mouseleave="clearSelect">
      <li
        v-for="(item, index) in divList"
        :key="index"
        @mouseenter="getMousePlace(index)"
        @click="selected"
      ></li>
    </ul>
    <div class="wall-cancel">
      <button @click="cancel">重置</button>
    </div>
  </div>
</template>

<script>
export default {
  props: {
    isBlur: {
      type: Boolean
    }
  },
  data() {
    return {
      divList: [], // 方格列表数组
      rows: 1, // 行
      cols: 1, // 列
      isLeftTop: true, // 鼠标是否从左上方进入
      isSelected: false, // 是否选中电视墙行列
      isOn: false
    };
  },
  created() {
    this.divList.length = 100; // 方格个数 10*10
  },
  mounted() {
    this.autoSelect();
  },
  methods: {
    // 自动根据vuex中行列选中对应方格
    autoSelect() {
      this.rows = 1;
      this.cols = 1;
      this.selectRowCol(this.rows, this.cols);
    },
    // 获取鼠标进入方格的位置,进行背景渲染
    getMousePlace(index) {
      this.isOn = true;
      if (!this.isLeftTop || this.isSelected) {
        return;
      }
      const x = Math.floor(index % 10);
      const y = Math.floor(index / 10);
      const children = this.$refs.wallRow.children;
      for (let i = 0; i < this.divList.length; i++) {
        const childrenx = Math.floor(i % 10);
        const childreny = Math.floor(i / 10);
        if (childrenx < x + 1 && childreny < y + 1) {
          children[i].className = "bgColor";
          this.rows = x + 1;
          this.cols = y + 1;
        } else {
          children[i].className = "";
        }
      }
    },
    // 当鼠标移出选择区域,如未选定行列,则重置方格 1*1
    clearSelect() {
      this.isOn = false;
      if (this.isSelected) {
        return;
      }
      const children = this.$refs.wallRow.children;
      for (let i = 0; i < this.divList.length; i++) {
        if (i != 0) {
          children[i].className = "";
        }
      }
      this.rows = 1;
      this.cols = 1;
    },
    // 选定行列
    selected(index) {
      this.isSelected = true;
      const data = {
        rows: this.rows,
        cols: this.cols
      };
    },
    // 判断鼠标是否从左上进入
    direct(e) {
      const x = e.offsetX - 230 / 2; //鼠标进入的X坐标-盒子宽度的一半
      const y = e.offsetY - 230 / 2; //鼠标进入的y坐标-盒子宽度的一半
      const direction =
        (Math.round((Math.atan2(y, x) * (180 / Math.PI) + 180) / 90) + 3) % 4;
      if (direction === 0) {
        this.isLeftTop = true;
      } else if (direction === 1) {
        this.isLeftTop = false;
      } else if (direction === 2) {
        this.isLeftTop = false;
      } else {
        this.isLeftTop = true;
      }
    },
    // 点击取消,重置方格 1*1
    cancel() {
      const children = this.$refs.wallRow.children;
      for (let i = 0; i < this.divList.length; i++) {
        if (i != 0) {
          children[i].className = "";
        }
      }
      this.rows = 1;
      this.cols = 1;
      this.isSelected = false;
    },
    // 改变行列数
    handleTypeChange() {
      this.selectRowCol(this.rows, this.cols);
      this.isSelected = true;
    },
    // 渲染方格
    selectRowCol(rows, cols) {
      const children = this.$refs.wallRow.children;
      for (let i = 0; i < this.divList.length; i++) {
        const childrenx = Math.floor(i % 10);
        const childreny = Math.floor(i / 10);
        if (childrenx < rows && childreny < cols) {
          children[i].className = "bgColor";
        } else {
          children[i].className = "";
        }
      }
    },
    // 点击保存,将数据同步到vuex,并关闭下拉框
    save() {
      const data = {
        rows: this.rows,
        cols: this.cols
      };
    }
  }
};
</script>

<style scoped lang="css">
.custom-style {
  position: absolute;
  margin: 0;
  padding: 0;
  height: 440px;
  width: 400px;
  left: 50%;
  top: 50%;
  margin-left: -200px;
  margin-top: -210px;
}

.word {
  display: block;
  width: 100%;
  height: 20px;
  font-size: 13px;
  background-color: #e2eef6;
}
.word span {
  line-height: 20px;
  text-align: center;
}

.wall-row {
  width: 400px;
  height: 400px;
  padding: 0;
  margin: 0;
  display: block;
  position: relative;
}

.wall-row li {
  list-style: none;
  width: 39px;
  height: 40px;
  border: 0.5px solid #fff;
  float: left;
  background-color: #ebf0f4;
}

.wall-row .bgColor {
  background-color: #00b4ff;
}

.word-cancel {
  width: 100%;
  height: 20px;
  display: flex;
  justify-content: center;
  align-items: center;
}

</style>

你可能感兴趣的:(CSS3,Web前端)