【sgExcelGrid】自定义组件:简单模拟Excel表格拖拽、选中单元格、横行、纵列、拖拽圈选等操作

本文主要是介绍【sgExcelGrid】自定义组件:简单模拟Excel表格拖拽、选中单元格、横行、纵列、拖拽圈选等操作,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

特性:

  1. 可以自定义拖拽过表格
  2. 可以点击某个表格,拖拽右下角小正方形进行任意方向选取单元格
  3. 支持选中某一行、列
  4. 支持监听@selectedGrids、@selectedDatas事件获取选中项的DOM对象和数据数组
  5. 支持props自定义显示label字段别名

 sgExcelGrid源码

<template><div :class="$options.name"><div class="ruler-corner"></div><div class="horizontal-ruler" :style="{ left: `${-rulerPosition.x}px` }"><divclass="tick":hoverGrid="hoverGrid.x === A_Z[i]"@click="(hoverGrid = { x: A_Z[i] }), (mousedownGrid = {})"v-for="(a, i) in A_Z.slice(0, colCount_)":key="i">{{ a }}</div></div><div class="vertical-ruler" :style="{ top: `${-rulerPosition.y}px` }"><divclass="tick":hoverGrid="hoverGrid.y === i"@click="(hoverGrid = { y: i }), (mousedownGrid = {})"v-for="(a, i) in Math.ceil(pageSize / colCount_)":key="i">{{ i + 1 }}</div></div><div class="grids-scroll" ref="scrollContainer"><div class="grids" ref="dragContainer" :selectedGrids="selectedGrids.length > 0"><divclass="grid":hoverGridX="selectedGrids.length == 0 && hoverGrid.x === gridsData[i].x":hoverGridY="selectedGrids.length == 0 && hoverGrid.y === gridsData[i].y":mousedownGrid="mousedownGrid.x === gridsData[i].x && mousedownGrid.y === gridsData[i].y":dragMove="isMouseDragMove":type="a.strong ? 'primary' : ''"v-for="(a, i) in data":key="i"@mouseover="hoverGrid = gridsData[i]"@mouseout="hoverGrid = {}"@click="clickGrid(i)"><span :title="a[label]">{{ a[label] }}</span><i class="el-icon-close" @click="del(a)" /><divclass="position-text":title="`点击复制`"@click="$g.copy($g.stripHTML(getPositionText(gridsData[i])), true)"v-html="getPositionText(gridsData[i])"></div><div class="drag-select-btn" @mousedown.stop="clickResizeHandle"></div></div></div></div><!-- 拖拽 --><sgDragMoveTile :data="dragMoveTileData" @scroll="scroll" /></div>
</template>
<script>
import sgDragMoveTile from "@/vue/components/admin/sgDragMoveTile";
export default {name: "sgExcelGrid",components: {sgDragMoveTile,},data() {return {A_Z: [...Array(26)].map((v, i) => String.fromCharCode(i + 65)),hoverGrid: {}, //移入的宫格标记mousedownGrid: {}, //点击的宫格标记gridsData: [], //记录网格宫格状态selectedGrids: [], //被选中的网格宫格DOMselectedDatas: [], //被选中的网格宫格数据rulerPosition: { x: 0, y: 0 },dragMoveTileData: {},gridWidth: 200,gridHeight: 100,colCount_: 8,pageSize_: 100,isMouseDragMove: false, //鼠标拖拽选中移动label: `label`, //显示文本字段名};},props: ["value","props","data","pageSize", //每页显示多少个宫格"colCount", //列数],computed: {},watch: {props: {handler(newValue, oldValue) {if (newValue && Object.keys(newValue).length) {newValue.label && (this.label = newValue.label);}},deep: true, //深度监听immediate: true, //立即执行},pageSize: {handler(newValue, oldValue) {//console.log('深度监听:', newValue, oldValue);newValue && (this.pageSize_ = newValue);},deep: true, //深度监听immediate: true, //立即执行},data: {handler(newValue, oldValue) {this.init_gridsData();},deep: true, //深度监听immediate: true, //立即执行},colCount: {handler(newValue, oldValue) {//console.log('深度监听:', newValue, oldValue);newValue && (this.colCount_ = newValue);this.$nextTick(() => {this.$el.style.setProperty("--gridWidth", `${this.gridWidth}px`); //js往css传递局部参数this.$el.style.setProperty("--gridHeight", `${this.gridHeight}px`); //js往css传递局部参数this.$el.style.setProperty("--gridsWidth",`${this.colCount_ * this.gridWidth}px`); //js往css传递局部参数});},deep: true, //深度监听immediate: true, //立即执行},selectedGrids: {handler(newValue, oldValue) {this.$emit(`selectedGrids`, newValue || []);},deep: true, //深度监听// immediate: true, //立即执行},selectedDatas: {handler(newValue, oldValue) {this.$emit(`selectedDatas`, newValue || []);},deep: true, //深度监听// immediate: true, //立即执行},},created() {},mounted() {this.init_grid_view();this.addEvents();},destroyed() {this.removeEvents();},methods: {clickGrid(i) {(this.mousedownGrid = this.gridsData[i]),(this.hoverGrid = {}),this.resetSelectGrid();},clickResizeHandle(e) {this.originRect = e.target.parentNode.getBoundingClientRect();this.originRect.bottomRightX = this.originRect.x + this.originRect.width; //右下角坐标.xthis.originRect.bottomRightY = this.originRect.y + this.originRect.height; //右下角坐标.ythis.__addWindowEvents();},__addWindowEvents() {this.__removeWindowEvents();addEventListener("mousemove", this.mousemove_window);addEventListener("mouseup", this.mouseup_window);},__removeWindowEvents() {removeEventListener("mousemove", this.mousemove_window);removeEventListener("mouseup", this.mouseup_window);},mousemove_window(e) {this.isMouseDragMove = true;let { x, y } = e;let minWidth = 0,minHeight = 0,maxWidth = innerWidth,maxHeight = innerHeight;x < 0 && (x = 0),y < 0 && (y = 0),x > maxWidth && (x = maxWidth),y > maxHeight && (y = maxHeight);let style = {};style.x = this.originRect.x;style.y = this.originRect.y;style.width = x - this.originRect.x;style.width <= minWidth &&((style.width = Math.abs(style.width)),((style.x = this.originRect.x - style.width),(style.width = style.width + this.originRect.width)));style.height = y - this.originRect.y;style.height <= minHeight &&((style.height = Math.abs(style.height)),((style.y = this.originRect.y - style.height),(style.height = style.height + this.originRect.height)));style.width > maxWidth && (style.width = maxWidth);style.height > maxHeight && (style.height = maxHeight);this.calcRectGrid(style);},mouseup_window(e) {this.isMouseDragMove = false;this.__removeWindowEvents();},resetAllGridStatus() {this.resetSelectGrid();this.mousedownGrid = {};},resetSelectGrid(d) {this.selectedGrids = [];this.selectedDatas = [];let grids = this.$refs.dragContainer.querySelectorAll(`.grid`);grids.forEach((v) => {v.removeAttribute("selected-left");v.removeAttribute("selected-top");v.removeAttribute("selected-right");v.removeAttribute("selected-bottom");v.removeAttribute("selected");});},// 计算是否选中格子calcRectGrid(rect) {this.resetSelectGrid();this.selectedGrids = this.getSelectedDoms({targetDoms: this.$refs.dragContainer.querySelectorAll(`.grid`),rect,});this.selectedGrids.forEach((grid) => {let grid_rect = grid.getBoundingClientRect();let gridRectScreenWidth = grid_rect.x + grid_rect.width;let gridRectScreenHeight = grid_rect.y + grid_rect.height;grid_rect.x <= rect.x &&rect.x < gridRectScreenWidth &&grid.setAttribute("selected-left", true);grid_rect.y <= rect.y &&rect.y < gridRectScreenHeight &&grid.setAttribute("selected-top", true);let rectScreenWidth = rect.x + rect.width;let rectScreenHeight = rect.y + rect.height;grid_rect.x < rectScreenWidth &&rectScreenWidth <= grid_rect.x + grid_rect.width &&grid.setAttribute("selected-right", true);grid_rect.y < rectScreenHeight &&rectScreenHeight <= grid_rect.y + grid_rect.height &&grid.setAttribute("selected-bottom", true);grid.setAttribute("selected", true);});},// 获取被选中的DOMgetSelectedDoms({ targetDoms, rect } = {}) {this.selectedDatas = [];return [...targetDoms].filter((targetDom, i) => {if (this.$g.isCrash(targetDom, rect)) {this.selectedDatas.push(this.data[i]);return targetDom;}}); // 获取被圈选的内容},// ----------------------------------------del(d) {this.$emit(`del`, d);},addEvents(d) {this.removeEvents();this.__removeWindowEvents();addEventListener("resize", this.resize);},removeEvents(d) {removeEventListener("resize", this.resize);},getPositionText(gridData) {return `<span>第${gridData.y + 1}行</span>&nbsp;<span>第${gridData.x}列</span>`;},init_grid_view() {this.resize();this.$nextTick(() => {this.init_sgDragMoveTile();});},init_gridsData(d) {this.gridsData = [...Array(this.pageSize_)].map((v, i) => ({x: this.A_Z[i % this.colCount_],y: Math.floor(i / this.colCount_),}));this.$nextTick(() => {this.resetAllGridStatus();});},init_sgDragMoveTile() {this.dragMoveTileData = {scrollContainer: this.$refs.scrollContainer,dragContainer: this.$refs.dragContainer,};},resize(d) {this.rulerPosition = {x: 0,y: 0,};},scroll(e) {this.rulerPosition = {x: e.target.scrollLeft,y: e.target.scrollTop,};},},
};
</script>
<style lang="scss" scoped>
.sgExcelGrid {/*禁止选中文本*/user-select: none;overflow: hidden;$tickDis: 40px;$gridWidth: var(--gridWidth);$gridHeight: var(--gridHeight);$gridsWidth: var(--gridsWidth);$scrollbarWidth: 14px;width: 100%;height: 100%;position: relative;.ruler-corner {position: absolute;z-index: 2;left: 0;top: 0;width: $tickDis;height: $tickDis;box-sizing: border-box;border: 1px solid #ebeef5;border-right: none;border-bottom: none;background-color: #eff2f755;/*遮罩模糊*/backdrop-filter: blur(5px);}.horizontal-ruler {position: absolute;z-index: 1;left: 0;top: 0;margin-left: $tickDis;display: flex;flex-wrap: nowrap;border-top: 1px solid #ebeef5;border-left: 1px solid #ebeef5;/*遮罩模糊*/backdrop-filter: blur(5px);// box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);.tick {display: flex;justify-content: center;align-items: center;flex-shrink: 0;width: $gridWidth;height: $tickDis;box-sizing: border-box;border-top: 1px solid transparent;border-left: 1px solid transparent;border-bottom: 1px solid #ebeef5;border-right: 1px solid #ebeef5;font-family: DIN-Black;background-color: #eff2f755;&[hoverGrid] {border-left: 1px solid #409eff;border-right: 1px solid #409eff;background-color: #b3d8ff99;color: #409eff;}}}.vertical-ruler {position: absolute;z-index: 1;left: 0;top: 0;margin-top: $tickDis;display: flex;flex-wrap: wrap;flex-direction: column;border-top: 1px solid #ebeef5;border-left: 1px solid #ebeef5;/*遮罩模糊*/backdrop-filter: blur(5px);// box-shadow: 2px 0 12px 0 rgba(0, 0, 0, 0.1);.tick {display: flex;justify-content: center;align-items: center;flex-shrink: 0;width: $tickDis;height: $gridHeight;box-sizing: border-box;border-top: 1px solid transparent;border-left: 1px solid transparent;border-bottom: 1px solid #ebeef5;border-right: 1px solid #ebeef5;font-family: DIN-Black;background-color: #eff2f7;background-color: #eff2f755;&[hoverGrid] {border-top: 1px solid #409eff;border-bottom: 1px solid #409eff;background-color: #b3d8ff99;color: #409eff;}}}.grids-scroll {width: calc(100% - #{$tickDis});height: calc(100vh - 310px);box-sizing: border-box;overflow: auto;position: relative;margin: $tickDis 0 0 $tickDis;.grids {width: calc(#{$gridsWidth} + #{$scrollbarWidth});min-height: calc(#{$gridHeight} + #{$scrollbarWidth});overflow: auto;display: flex;flex-wrap: wrap;align-content: flex-start;box-sizing: border-box;border-top: 1px solid #ebeef5;border-left: 1px solid #ebeef5;.grid {display: flex;justify-content: center;align-items: center;width: $gridWidth;height: $gridHeight;padding: 20px;box-sizing: border-box;border-top: 1px solid transparent;border-left: 1px solid transparent;border-bottom: 1px solid #ebeef5;border-right: 1px solid #ebeef5;word-wrap: break-word;word-break: break-all;white-space: break-spaces;position: relative;span {/*多行省略号*/overflow: hidden;word-break: break-all;white-space: break-spaces;display: -webkit-box;-webkit-box-orient: vertical;max-height: min-content;-webkit-line-clamp: 3;line-height: 1.2;}// 坐标文本.position-text {position: absolute;height: 22px;z-index: 1;left: 0px;top: 0px;display: none;flex-wrap: nowrap;white-space: nowrap;align-items: center;color: white;background-color: #00000055;box-sizing: border-box;padding: 0 5px;border-radius: 0 0 8px 0;>>> span {font-size: 12px !important;}cursor: cell;&:hover {background-color: #409eff;color: white;}}// 删除i.el-icon-close {z-index: 1;display: none;position: absolute;right: 0;top: 0;font-size: 12px !important;justify-content: center;align-items: center;color: white;background-color: #409eff;box-sizing: border-box;padding: 5px;border-radius: 0 0 0 8px;cursor: pointer;&:hover {background-color: #f56c6c;}}// 拖拽选区.drag-select-btn {position: absolute;height: 9px;width: 9px;z-index: 1;right: -4.5px;bottom: -4.5px;display: none;box-sizing: border-box;border: 2px solid white;background-color: #f56c6c;cursor: crosshair;}&:nth-of-type(2n) {background-color: #eff2f755;}&[hoverGridX] {border-left: 1px solid #409eff;border-right: 1px solid #409eff;background-color: #f2f8fe;}&[hoverGridY] {border-top: 1px solid #409eff;border-bottom: 1px solid #409eff;background-color: #f2f8fe;}&[mousedownGrid] {border: 1px solid #f56c6c;background-color: #f56c6c22;.drag-select-btn {display: block;}}&[selected] {// border: 1px solid #f56c6c;border-top: 1px solid transparent;border-left: 1px solid transparent;border-right: 1px solid #f56c6c22;border-bottom: 1px solid #f56c6c22;background-color: #f56c6c22;.position-text {background-color: #f56c6c66;color: white;&:hover {background-color: #f56c6c;}}i.el-icon-close {background-color: #f56c6c66;&:hover {background-color: #f56c6c;}}.drag-select-btn {display: none;}&:hover:not([dragMove]) {border: 1px solid #f56c6c;background-color: #f56c6c66 !important;}}&[selected-left] {border-left: 1px solid #f56c6c;}&[selected-top] {border-top: 1px solid #f56c6c;}&[selected-right] {border-right: 1px solid #f56c6c;}&[selected-bottom] {border-bottom: 1px solid #f56c6c;}&[mousedownGridX] {border-left: 1px solid #f56c6c;border-right: 1px solid #f56c6c;background-color: #f56c6c22;}&[mousedownGridY] {border-top: 1px solid #f56c6c;border-bottom: 1px solid #f56c6c;background-color: #f56c6c22;}&[dragMove] {}&:hover:not([mousedownGrid]):not([dragMove]) {background-color: #b3d8ff99;i,.position-text {display: flex;}}}&[selectedGrids] {.grid {&:hover:not([mousedownGrid]):not([dragMove]):not([selected]) {border: 1px solid #409eff;}}}}}
}
</style>

应用

<template><sgExcelGrid:props="{ label: `MC` }":data="gridDatas":pageSize="pageSize"@del="delGrid"@selectedDatas="selectedDatas"/>
</template>
<script>
import sgExcelGrid from "@/vue/components/admin/sgExcelGrid";
export default {components: {sgExcelGrid,},data() {return {gridDatas: [{ ID: 1, value: 1, MC: "显示文本1" },{ ID: 2, value: 2, MC: "显示文本2" },{ ID: 3, value: 3, MC: "显示文本3" },{ ID: 4, value: 4, MC: "显示文本4" },{ ID: 5, value: 5, MC: "显示文本5" },],pageSize: 100, //每页显示多少个单元格rectSelectIDS: [], //选中的ID数组};},props: ["value"],computed: {},watch: {},created() {},mounted() {},destroyed() {},methods: {selectedDatas(d) {this.rectSelectIDS = d.map((v) => v.ID); //获取选中项ID数组},delGrid(d) {//删除单元格},},
};
</script>

基于【sgDragMoveTile】自定义组件:拖拽瓦片图、地图、大图,滚动条对应同步滚动_用鼠标拖拽(drag)内容div”,滚动条对应同步滚动 vue-CSDN博客文章浏览阅读140次。【代码】【sgDragMoveTile】自定义组件:拖拽瓦片图、地图、大图,滚动条对应同步滚动。_用鼠标拖拽(drag)内容div”,滚动条对应同步滚动 vuehttps://blog.csdn.net/qq_37860634/article/details/133292981

这篇关于【sgExcelGrid】自定义组件:简单模拟Excel表格拖拽、选中单元格、横行、纵列、拖拽圈选等操作的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/791419

相关文章

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h

usaco 1.2 Transformations(模拟)

我的做法就是一个一个情况枚举出来 注意计算公式: ( 变换后的矩阵记为C) 顺时针旋转90°:C[i] [j]=A[n-j-1] [i] (旋转180°和270° 可以多转几个九十度来推) 对称:C[i] [n-j-1]=A[i] [j] 代码有点长 。。。 /*ID: who jayLANG: C++TASK: transform*/#include<

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

poj 1113 凸包+简单几何计算

题意: 给N个平面上的点,现在要在离点外L米处建城墙,使得城墙把所有点都包含进去且城墙的长度最短。 解析: 韬哥出的某次训练赛上A出的第一道计算几何,算是大水题吧。 用convexhull算法把凸包求出来,然后加加减减就A了。 计算见下图: 好久没玩画图了啊好开心。 代码: #include <iostream>#include <cstdio>#inclu