js实现选择区域添加背景色

用到的基础web API

window.getSelection()

Selection.containsNode()

使用window.getSelection()并不能返回所有选中节点, 通过遍历所有选中节点的最小公共节点下的所有子节点, 并判断遍历到的节点是不是选中的节点, 可以找到所有选中的节点。

Selection.getRangeAt()

Range.commonAncestorContainer

直接遍历document下的所有子节点也可以, 但使用commonAncestorContainer获取所有选中节点的最近公共节点可以缩小遍历范围。

Text.splitText()

选择的开头和结尾部分,可能是半个单词, 即半个文本节点, 需要使用splitText将文本节点切割

源码实现

js 复制代码
import { v4 as uuidV4 } from 'uuid';

// 获取指定节点下的所有子节点
export function getAllNodes(rootNode, callback) {
  if (!callback) {
    var nodes = [];
    getAllNodes(rootNode, function (node) {
      nodes.push(node);
    });
    return nodes;
  }

  if (false === callback(rootNode)) return false;

  if (rootNode.hasChildNodes()) {
    let nodes = rootNode.childNodes;
    for (var i = 0, l = nodes.length; i < l; ++i)
      if (false === getAllNodes(nodes[i], callback)) return;
  }
}
// 监听mouseup事件
export function listenMouseUp() {
  document.addEventListener('mouseup', listenMouseUpCallback);
}

// 移除mouseup事件
export function removeListenMouseUp() {
  document.removeEventListener('mouseup', listenMouseUpCallback);
}

// mouseup事件的回调函数
function listenMouseUpCallback() {
  const selected = window.getSelection();
  const range = selected.getRangeAt(0);
  // range.insertNode(newNode()); // 在Range开头插入一个节点, 这是另外一种分隔方式
  const startContainer = range.startContainer;
  const startOffset = range.startOffset;
  startContainer.splitText(startOffset); // 将选中范围的结束位置处的文本节点分隔为两个节点

  const endContainer = range.endContainer;
  const endOffset = range.endOffset;
  endContainer.splitText(endOffset);

  const nodes = getAllNodes(range.commonAncestorContainer);
  let uuid = uuidV4();
  for (let i = 0; i < nodes.length; i++) {
    const node = nodes[i];
    if (selected.containsNode(node)) {
      addClass(node, uuid);
    }
  }
}

// 给选择区域增加背景色
function addClass(node, uuid) {
  if (node.nodeType === 3) {
    // 将文本节点外面加一层<span class="light-hight"></span>
    let newNode = document.createElement('span'); // 创建一个新的element节点
    newNode.classList.add('light-hight');
    newNode.classList.add('new-span');
    newNode.setAttribute('data-mark-id', uuid);
    node.parentNode.insertBefore(newNode, node); // 将新的节点插入到文本节点前
    newNode.appendChild(node); // 将文本节点移动到新的节点里面
  } else {
    node.classList.add('light-hight');
    node.setAttribute('data-mark-id', uuid);
  }
}
相关推荐
泯泷21 分钟前
Tiptap 深度教程(四):终极定制 - 从零创建你的专属扩展
前端·javascript·架构
局i22 分钟前
vue简介
前端·javascript·vue.js
yqcoder1 小时前
vue2 和 vue3 中,列表中的 key 值作用
前端·javascript·vue.js
U***49831 小时前
前端TypeScript教程汇总,从基础到高级
前端·javascript·typescript
梵得儿SHI1 小时前
Vue 指令系统:事件处理与表单绑定全解析,从入门到精通
前端·javascript·vue.js·v-model·v-on·表单数据绑定·表单双向绑定
二川bro1 小时前
第45节:分布式渲染:Web Workers多线程渲染优化
开发语言·javascript·ecmascript
秋天的一阵风2 小时前
😱一行代码引发的血案:展开运算符(...)竟让图表功能直接崩了!
前端·javascript·vue.js
Hilaku2 小时前
npm scripts的高级玩法:pre、post和--,你真的会用吗?
前端·javascript·vue.js
mCell2 小时前
React 如何处理高频的实时数据?
前端·javascript·react.js
随笔记2 小时前
HbuilderX载入项目,运行后唤起微信开发者工具,提示:Error: Fail to open IDE,唤醒不起来怎么办
javascript·微信小程序·uni-app