【前端】实现表格简单操作

news2025/1/14 1:07:03

简言

表格合并基础篇
本篇是在上一章的基础上实现,实现了的功能有添加行、删除行、逆向选区、取消合并功能。
在这里插入图片描述
在这里插入图片描述

功能实现

添加行

添加行分为在上面添加和在下面追加行。
利用 insertAdjacentElement 方法实现,该方法可以实现从前插入元素和从后插入元素。
在这里插入图片描述

删除行

删除当前行就是利用元素remove()方法,从dom树种删除元素。
在这里插入图片描述

逆向选区

逆向选区是指选区从下往上选。
解决思路:记录当前选区时鼠标移动方向,往左上移动则为负,往右下移动则为正。负时在首位插入选中节点,正时从尾部追加选中节点,这样合并只需取第一个选中节点即可。
在这里插入图片描述

取消合并

获取当前元素的rowspan和colspan属性值,然后遍历后面的和下面包含行节点,删除节点的hide类,然后删除当前元素rowspan和colspan属性即可。
在这里插入图片描述

代码

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>表格合并</title>
  <style>
    .zsk-table {
      border-collapse: collapse;
      border: 1px solid;
      font-family: inherit;
      user-select: none;
    }

    .zsk-table tr {
      height: 32px;
    }

    .zsk-table td {
      border: 1px solid;
      height: 32px;
      padding: 16px;
    }


    .amount {
      width: 100px;
    }

    .show-box {
      position: absolute;
      top: -200px;
      left: -200px;
      width: 200px;
      background-color: #eee;
    }

    .show-box>div {
      width: 200px;
      height: 50px;
      line-height: 50px;
      border-bottom: 1px solid #000;
    }

    .show-box>div:hover {
      background-color: #ccc;
      cursor: pointer;

    }

    .select {
      color: #fff;
      background-color: #3987cf;
    }

    .hide {
      display: none;
    }
  </style>
</head>

<body>

  <h1>表格合并</h1>

  <table tabindex="1" class="zsk-table">
    <tr>
      <td>1-1</td>
      <td>1-2</td>
      <td>1-3</td>
      <td>1-4</td>
      <td>1-5</td>
    </tr>
    <tr>
      <td>2-1</td>
      <td>2-2</td>
      <td>2-3</td>
      <td>2-4</td>
      <td>2-5</td>
    </tr>
    <tr>
      <td>3-1</td>
      <td>3-2</td>
      <td>3-3</td>
      <td>3-4</td>
      <td>3-5</td>
    </tr>
    <tr>
      <td>4-1</td>
      <td>4-2</td>
      <td>4-3</td>
      <td>4-4</td>
      <td>4-5</td>
    </tr>
    <tr>
      <td>5-1</td>
      <td>5-2</td>
      <td>5-3</td>
      <td>5-4</td>
      <td>5-5</td>
    </tr>
  </table>
  <!-- 表格右键 -->
  <div class="show-box">
    <div class="add-down">向下添加一行</div>
    <div class="add-up">向上添加一行</div>
    <div class="delete-cell">删除当前行行</div>
    <div class="merge-cell">合并</div>
    <div class="split-cell">取消合并</div>
  </div>

  <script>
    const table = document.querySelector('.zsk-table')
    const showBox = document.querySelector('.show-box')
    const mergeDiv = document.querySelector('.merge-cell')
    const addDownDiv = document.querySelector('.add-down')
    const addUpDiv = document.querySelector('.add-up')
    const deleteCellDiv = document.querySelector('.delete-cell')
    const cancelDiv = document.querySelector('.split-cell')
    const select = {  // 选中单元格
      value: [[]],
      range: [[], []] //  [start,end]范围
    }
    //  向下添加一行
    addDownDiv.addEventListener('click', (e) => {
      let node = select.value[0][0].parentElement
      let newNode = node.cloneNode(true)
      node.insertAdjacentElement('afterend', newNode)
      clearSelect()

    })
    //  向上添加一行
    addUpDiv.addEventListener('click', (e) => {
      let node = select.value[0][0].parentElement
      let newNode = node.cloneNode(true)
      node.insertAdjacentElement('beforebegin', newNode)
      clearSelect()
    })
    //  删除当前行
    deleteCellDiv.addEventListener('click', () => {
      let node = select.value[0][0].parentElement
      node.remove()
      clearSelect()

    })
    //  取消合并
    cancelDiv.addEventListener('click', () => {
      let node = select.value[0][0]
      let rowspan = node.getAttribute('rowspan')
      let colspan = node.getAttribute('colspan')
      if (!colspan || !rowspan) return
      colspan = Number.parseInt(colspan)
      rowspan = Number.parseInt(rowspan)

      let index = getChildIndex(node)
      let nextNode = node

      for (let i = 0; i < rowspan; i++) {
        let col = colspan
        let temp = nextNode
        while (col--) {

          temp.classList.remove('hide')
          temp = temp.nextElementSibling
        }
        nextNode = getRowXElement(nextNode, 1)

      }
      node.removeAttribute('colspan')
      node.removeAttribute('rowspan')

    })
    //  合并命令
    mergeDiv.addEventListener('click', () => {
      if (select.value.length === 0) return
      //  默认是正向选中,即结尾点比开始点的x和y都大
      select.value.forEach((item, i) => {
        item.forEach((v, k) => {
          if (i === 0 && k === 0) {
            v.setAttribute('colspan', item.length || '1')
            v.setAttribute('rowspan', select.value.length || '1')
          } else {
            v.classList.add('hide')
          }
        })
      })
      clearSelect()
    })
    //  右键
    table.addEventListener('click', (e) => {
      e.target.focus()
    })
    table.addEventListener("contextmenu", (e) => {
      e.preventDefault()
      showBox.style.left = e.clientX + 'px'
      showBox.style.top = e.clientY + 'px'
      if (!select.value[0][0]) {
        select.value[0][0] = e.target
      }
    })
    table.addEventListener('blur', (e) => {
      setTimeout(() => {
        showBox.style.left = -1000 + 'px'
        showBox.style.top = -1000 + 'px'
      }, 150)
    })
    /**
     *  选中逻辑
     * 
     **/

    selectLogic(table, select)
    function selectLogic(table, select) {
      let lastEnd = [0, 0] // 最后选中的单元格位置
      let lastInfo = [0, 0]  //  最后选中单元格的宽高
      let endUp = [0, 0]
      let startRange = [0.0]
      let endRange = [0, 0]
      let run = false
      //  按下
      let timer = 0
      table.addEventListener('mousedown', (e) => {
        if (timer !== 0) {
          clearTimeout(timer)
          timer = 0
        }
        timer = setTimeout(() => {
          //  先清空
          clearSelect()
          run = true
          startRange = [e.clientX - e.offsetX, e.clientY - e.offsetY]
          lastEnd = [startRange[0], startRange[1]]
          lastInfo = [e.target.offsetWidth, e.target.offsetHeight]
          e.target.classList.add('select')
          if (e.target.tagName === 'TD') {
            select.value[0].push(e.target)
            select.range[0] = startRange
            select.range[1] = [startRange[0] + e.target.offsetWidth, startRange[1] + e.target.offsetHeight]
          }
        }, 200)

      })
      //  移动
      table.addEventListener('mousemove', (e) => {
        if (run) {
          end = [e.clientX, e.clientY]


          //  计算范围 然后 判断是否修改选中dom数组
          let x = end[0] - lastEnd[0]
          let y = end[1] - lastEnd[1]
          let xDirection = end[0] - startRange[0] > 0 ? 1 : -1  //  x方向
          let yDirection = end[1] - startRange[1] > 0 ? 1 : -1  //  y方向
          console.log(`x: ${x} y: ${y}  方向x:${xDirection}  方向y:${yDirection}`);

          if ((xDirection === 1 && x >= lastInfo[0]) || (xDirection === -1 && x <= 0)) {
            console.log('横向超出,x扩展');
            //  更新选取范围 x

            if (xDirection === 1) {
              lastEnd = [select.range[1][0], lastEnd[1]]
              lastInfo = [e.target.offsetWidth, lastInfo[1]]
              select.range[1] = [select.range[1][0] + e.target.offsetWidth, select.range[1][1]]

            } else {
              select.range[0] = [select.range[0][0] - e.target.offsetWidth, select.range[0][1]]

              lastEnd = [select.range[0][0], lastEnd[1]]
              lastInfo = [e.target.offsetWidth, lastInfo[1]]
            }
            //  每行横向添加一行
            for (let i = 0; i < select.value.length; i++) {
              //  查找最后一个节点元相邻td元素
              console.log(select.value[i]);
              if (xDirection === 1) {
                let el = getElement(select.value[i][select.value[i].length - 1], xDirection)

                select.value[i].push(el)

              } else {
                let el = getElement(select.value[i][0], xDirection)

                select.value[i].unshift(el)

              }

            }


          } else if ((xDirection === 1 && x <= 0) || (xDirection === -1 && x >= lastInfo[0])) {
            if (select.value[0].length <= 1) return
            console.log(select.value[0].length, '当前个数');

            if (xDirection === 1) {
              select.range[1] = [lastEnd[0], select.range[1][1]]
              lastEnd = [lastEnd[0] - e.target.offsetWidth, lastEnd[1]]
              lastInfo = [e.target.offsetWidth, lastInfo[1]]
            } else {
              select.range[0] = [lastEnd[0] + e.target.offsetWidth, select.range[0][1]]

              lastEnd = [select.range[0][0], lastEnd[1]]
              lastInfo = [e.target.offsetWidth, lastInfo[1]]


            }
            //  减去每行的最后一个
            for (let i = 0; i < select.value.length; i++) {
              if (select.value[i].length > 0) {
                if (xDirection === 1) {
                  select.value[i][select.value[i].length - 1].classList.remove('select')
                  select.value[i].pop()
                } else {
                  select.value[i][0].classList.remove('select')
                  select.value[i].shift()
                }
              }
            }
          }
          if ((yDirection === 1 && y > lastInfo[1]) || (yDirection === -1 && y <= 0)) {
            console.log('纵向超出,y扩展', select.value[0].length);
            if (yDirection === 1) {
              lastEnd = [lastEnd[0], select.range[1][1]]
              lastInfo = [lastInfo[0], e.target.offsetHeight]
              select.range[1] = [select.range[1][0], select.range[1][1] + e.target.offsetHeight]
            } else {
              select.range[0] = [select.range[0][0], select.range[0][1] - e.target.offsetHeight]
              lastEnd = [lastEnd[0], select.range[0][1]]
              lastInfo = [lastInfo[0], e.target.offsetHeight]
            }


            const lastRow = []
            for (let k = 0; k < select.value[0].length; k++) {
              let el = yDirection === 1 ? select.value[select.value.length - 1][k] : select.value[0][k]
              lastRow.push(getRowXElement(el, yDirection))


            }
            if (yDirection === 1) {
              select.value.push(lastRow)
            } else {
              select.value.unshift(lastRow)

            }

            //  更新选区范围
          } else if ((yDirection === 1 && y <= 0) || (yDirection === -1 && y >= lastInfo[1])) {
            if (select.value.length < 1) return
            if (yDirection === 1) {
              select.range[1] = [select.range[1][0], lastEnd[1]]

              lastEnd = [lastEnd[0], lastEnd[1] - e.target.offsetHeight]
              lastInfo = [lastInfo[0], e.target.offsetHeight]
            } else {
              select.range[0] = [select.range[0][0], lastEnd[1] + e.target.offsetHeight]

              lastEnd = [lastEnd[0], lastEnd[1] + e.target.offsetHeight]
              lastInfo = [lastInfo[0], e.target.offsetHeight]
            }

            //  去掉最后一行的class
            if (yDirection === 1) {
              select.value[select.value.length - 1].forEach(el => {
                el.classList.remove('select')
              })
              select.value.pop()
            } else {
              select.value[0].forEach(el => {
                el.classList.remove('select')
              })
              select.value.shift()
            }
          }
          //  选中元素添加class
          for (let i = 0; i < select.value.length; i++) {
            for (let k = 0; k < select.value[i].length; k++) {
              select.value[i][k] && select.value[i][k].classList.add('select')
            }
          }
        }
      })
      //  抬起
      table.addEventListener('mouseup', (e) => {
        run = false
        if (timer !== 0) {
          clearTimeout(timer)
          timer = 0
        }
        console.log(select.value);
      })
    }
    /*
      获取下一行当前横坐标相同位置元素
    */
    function getRowXElement(currentElement, direction = 1) {
      if (!currentElement.parentElement.nextElementSibling && direction == 1 || !currentElement.parentElement.previousElementSibling && direction !== 1) return null
      let nextElement = direction === 1 ? currentElement.parentElement.nextElementSibling.firstElementChild : currentElement.parentElement.previousElementSibling.firstElementChild;
      let childIndex = getChildIndex(currentElement)  //  获取当前元素在父元素中的索引
      if (childIndex !== -1) {
        return nextElement.parentElement.children[childIndex]
      } else {
        let currentLeft = currentElement.offsetLeft;
        let nextElementLeft = nextElement.offsetLeft;
        while (nextElement !== null && nextElementLeft !== currentLeft) {
          nextElement = getElement(nextElement, 1);
          nextElementLeft = nextElement.offsetLeft;
        }

        return nextElement;
      }

    }
    /**
     *  获取下一个兄弟元素
     * direction === 1时默认查找下一个兄弟元素,否则上一个
     **/
    function getElement(element, direction = 1) {
      if (direction === 1) {
        if (element.nextElementSibling) {
          return element.nextElementSibling;
        } else {
          let parent = element.parentElement;
          while (parent && parent.nextElementSibling === null) {
            parent = parent.parentElement;
          }
          return parent ? parent.nextElementSibling.firstElementChild : null;
        }
      } else {

        if (element.previousElementSibling) {
          return element.previousElementSibling;
        } else {
          let parent = element.parentElement;
          while (parent && parent.previousElementSibling === null) {
            parent = parent.parentElement;
          }
          return parent ? parent.previousElementSibling.firstElementChild : null;
        }
      }

    }
    function clearSelect() {
      select.value.forEach((item, index) => {
        item.forEach(v => {
          v.classList.remove('select')
        })
      })
      Object.assign(select, {
        value: [[]],
        range: [[], []] //  [start,end]范围
      })
    }
    /**
     * 获取子元素的索引
     */
    function getChildIndex(child) {
      var parent = child.parentNode;
      for (var i = 0; i < parent.children.length; i++) {
        if (parent.children[i] === child) {
          return i;
        }
      }
      return -1; // 如果找不到子元素,则返回-1
    }

  </script>
</body>

</html>

结语

合并没有做多次合并处理。
添加列没有实现。
边界情况未处理。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1654065.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

嘉楠堪智 CanMV K230 进行 Linux、RT-smart 系统开发

本文记录学习、使用 K230 SDK 进行 Linux、RT-smart 系统的开发的一些关键步骤&#xff0c;如何获取系统源代码&#xff0c;如何配置环境&#xff0c;如何使用 Docker 进行编译&#xff0c;获得系统文件。 具体详细的教程&#xff0c;可以学习 CanMV K230 教程。 目录 一、S…

引入 Redis

简介 Jedis Jedis 是早期的 Redis 的 Java 实现客户端&#xff0c;提供了比较全面的 Redis 命令的支持&#xff0c;其官方网址是&#xff1a;http://tool.oschina.net/uploads/apidocs/redis/clients/jedis/Jedis.html 优点&#xff1a;支持全面的 Redis 操作特性&#xff0…

搜狗输入法 PC端 v14.4.0.9307 去广告绿化版.

软件介绍 搜狗拼音输入法作为众多用户计算机配置的必备工具&#xff0c;其功能的全面性已为众所周知&#xff0c;并且以其高效便捷的输入体验受到广大使用者的青睐。然而&#xff0c;该软件在提供便利的同时&#xff0c;其内置的广告元素常常为用户带来一定的干扰。为此&#…

代码随想录算法训练营DAY48|C++动态规划Part9|121.买卖股票的最佳时机、122.买卖股票的最佳时机II、123.买卖股票的最佳时机III

文章目录 121.买卖股票的最佳时机思路CPP代码 122.买卖股票的最佳时机II思路CPP代码 123.买卖股票的最佳时机III思路CPP代码 121.买卖股票的最佳时机 力扣题目链接 文章讲解&#xff1a;121.买卖股票的最佳时机 视频讲解&#xff1a;动态规划之 LeetCode&#xff1a;121.买卖股…

由树形解空间入手,深入分析回溯、动态规划、分治算法的共同点和不同点

一、回溯、动态规划、分治 其实这三个算法就可以直接认为是一类算法&#xff0c;它们的共同点都是会涉及到递归。 更需要清楚明白的一点是&#xff0c;它们的解&#xff0c;都是基于一颗递归形式的树上得来的&#xff01;也就是——回溯、动态规划、分治的解空间是 一棵树&am…

ssm+vue的私人健身和教练预约管理系统(有报告)。Javaee项目,ssm vue前后端分离项目。

演示视频&#xff1a; ssmvue的私人健身和教练预约管理系统(有报告)。Javaee项目&#xff0c;ssm vue前后端分离项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;通…

深度学习常用优化算法笔记介绍,各种梯度下降法详细介绍

优化算法 mini-batch梯度下降法 当一个数据集其数据量非常大的时候&#xff0c;比如上百万上千万的数据集&#xff0c;如果采用普通的梯度下降法&#xff0c;那么运算速度会非常慢&#xff0c;因为如果使用梯度下降法在每一次迭代的时候&#xff0c;都需要将这整个上百万的数…

机器学习:基于TF-IDF算法、决策树,使用NLTK库对亚马逊美食评论进行情绪分析

前言 系列专栏&#xff1a;机器学习&#xff1a;高级应用与实践【项目实战100】【2024】✨︎ 在本专栏中不仅包含一些适合初学者的最新机器学习项目&#xff0c;每个项目都处理一组不同的问题&#xff0c;包括监督和无监督学习、分类、回归和聚类&#xff0c;而且涉及创建深度学…

第一天复习Qt文件读取

Qt文件操作&#xff1a; 1、QFile QTextStream操作文件案例&#xff1a; 1、打开文件 QFile file(absolute filepath | relative path); file.readLine()返回内容长度&#xff0c;如果为-1就是读取失败 file. Close()读取后关闭 file.errorString()返回文件打开发生的错误2、…

数据仓库实验三:分类规则挖掘实验

目录 一、实验目的二、实验内容和要求三、实验步骤1、创建数据库和表2、决策树分类规则挖掘&#xff08;1&#xff09;新建一个 Analysis Services 项目 jueceshu&#xff08;2&#xff09;建立数据源视图&#xff08;3&#xff09;建立挖掘结构 DST.dmm&#xff08;4&#xff…

16.接口自动化学习-编码处理与装饰器

1.编码和解码 编码&#xff1a;将自然语言翻译成计算机可以识别的语言 hello–01010 解码&#xff1a;将机器识别的语言翻译成自然语言 2.编码格式 UTF-8 GBK unicode 3.编码操作 #编码操作str1"hello呀哈哈哈"str2str1.encode(gbk)print(str2)print(type(str2))…

FPGA ov5640视频以太网传输

1 实验任务 使用DFZU4EV MPSoC 开发板及双目OV5640摄像头其中一个摄像头实现图像采集&#xff0c;并通过开发板上的以太网接口发送给上位机实时显示。 2 Verilog代码 2.1 顶层模块 timescale 1ns / 1ps //以太网传输视频顶层模块module ov5640_udp_pc (input sys_cl…

在WPS表格(Excel)中,每10行增加一个特定的值

注&#xff1a;如下为WPS表格操作演示 例如1&#xff0d;15的数值是1&#xff0c;16-30就变为2&#xff0c;31-45就变为3&#xff0c;类推&#xff01; 1、在B1单元格输入一个起始值&#xff0c;B2单元格输入公式IF(MOD(ROW(),15)0,B11,B1) 然后鼠标放到B2单元格右下角小点处&…

开机弹窗找不到OpenCL.dll是怎么回事,哪种修复方法更推荐

当用户在操作电脑过程中遇到系统提示“OpenCL.dll丢失”时&#xff0c;这究竟是怎么一回事呢&#xff1f;OpenCL.dll&#xff0c;作为Open Computing Language&#xff08;开放计算语言&#xff09;的重要动态链接库文件&#xff0c;它在图形处理器&#xff08;GPU&#xff09;…

springboot整合rabbitmq的不同工作模式详解

前提是已经安装并启动了rabbitmq&#xff0c;并且项目已经引入rabbitmq&#xff0c;完成了配置。 不同模式所需参数不同&#xff0c;生产者可以根据参数不同使用重载的convertAndSend方法。而消费者均是直接监听某个队列。 不同的交换机是实现不同工作模式的关键组件.每种交换…

县供电公司员工向媒体投稿发文章用亲身经历告诉你并不难

在县供电公司的日子里,我肩负着一项至关重要的使命——信息宣传工作。这不仅仅是一份职责,更是连接公司与外界的桥梁,通过新闻稿件传递我们的声音,展示我们的成果。然而,回忆起刚刚踏入这个领域的时光,那段经历至今让我感慨万千。 初涉投稿,步履维艰 刚接手这项工作时,我的投稿…

信息化飞速发展下,源代码防泄密方案该如何选择?常见四种有效方案分享

信息化时代发展迅速&#xff0c;数据防泄露一词也频繁的出现在我们身边。无论企业或政府单位&#xff0c;无纸化办公场景越来越多&#xff0c;数据泄露的时间也层出不穷。例如&#xff1a;世界最大职业中介网站Monster遭到黑客大规模攻击&#xff0c;黑客窃取在网站注册的数百万…

Dockerfile实践java项目

目的&#xff1a;用java项目测试dockerfil部署&#xff08;前提是安装好了docker&#xff09; 部署准备文件如下 1. java项目 java项目demo地址 https://gitee.com/xiaoqu_12/dockerfileDemo.git 或者百度网盘直接下载打包好的jar包 链接&#xff1a;https://pan.baidu.com/s/…

PostgreSQL的学习心得和知识总结(一百四十一)|深入理解PostgreSQL数据库数据库角色的使用及预定义角色的原理

目录结构 注&#xff1a;提前言明 本文借鉴了以下博主、书籍或网站的内容&#xff0c;其列表如下&#xff1a; 1、参考书籍&#xff1a;《PostgreSQL数据库内核分析》 2、参考书籍&#xff1a;《数据库事务处理的艺术&#xff1a;事务管理与并发控制》 3、PostgreSQL数据库仓库…

【计算机毕业设计】基于SSM++jsp的电子竞技管理平台【源码+lw+部署文档+讲解】

目录 1 绪论 1.1 研究背景 1.2 目的和意义 1.3 论文结构安排 2 相关技术 2.1 SSM框架介绍 2.2 B/S结构介绍 2.3 Mysql数据库介绍 3 系统分析 3.1 系统可行性分析 3.1.1 技术可行性分析 3.1.2 经济可行性分析 3.1.3 运行可行性分析 3.2 系统性能分析 3.2.1 易用性指标 3.2.2 可…