vue3结合element-plus实现标签手动标注效果

发布时间:2022-5-06 14:37

先上效果图

在这里插入图片描述 功能描述:默认首选标签为第一个“时间”【读者可以根据代码修改默认的标签】,能够增加标签。 点击不同的标签可以进行标签切换。在正文部分能够根据输入的文本txt,或者内置的html文件进行标注,对选中的词语或文段打上标签【体现在背景颜色和文本节点的‘title’属性】。可以再次点击已经标注的内容进行取消标注。最终标注的结果将会以一个对象数组的形式保存,读者可以自行对被标注的内容进行一系列操作。标注结果形如:

Proxy {0: {…}, 1: {…}, 2: {…}, 3: {…}, 4: {…}, 5: {…}, 6: {…}, 7: {…}, 8: {…}, 9: {…}}
[[Handler]]: Object
[[Target]]: Array(10)
0: {name: '时间', comment: '公铁立交'}
1: {name: '人物', comment: '省道S30'}
2: {name: '颜色', comment: '上部结构'}
3: {name: '颜色', comment: '下部结构'}
4: {name: ' 部位', comment: '中心'}
5: {name: ' 部位', comment: '桩基础'}
6: {name: ' 部位', comment: '层,设置异型钢'}
7: {name: ' 部位', comment: '过各项无损'}
8: {name: ' 部位', comment: '土双柱'}
9: {name: ' 部位', comment: '土空心'}
length: 10
[[Prototype]]: Array(0)
[[IsRevoked]]: false

实现原理

首先:监听鼠标在文本上动作,根据时间间隔来区分是选中了文本?还是只是点击了一下?

useMouse() {
let last = new Date().getTime();
//松开鼠标后,获得时间
function mousedown() {
last = new Date().getTime();
}
//根据时间间隔判断是选中还是点击事件
function getMouseEvent() {
const d = new Date().getTime() - last;
return d > 200 ? "select" : "click";
}
//暴露出的接口
return {
mousedown,
getMouseEvent,
};
},

通过对鼠标活动时间的判断,我们可以区分点击和选中这两个动作,显然我们对于点击事件不会做任何处理,需要对选中动作做出一系列操作。

监听既然有了,下一步肯定是选择监听对象了。根据vue生命周期的不同钩子函数所处不同的vue实例状态前提,我们将在vue实例完成对data和methods属性初始化,vDOM挂载到真正得DOM树上后的mounted()钩子函数中将我们的监听挂载到某个DOM节点上。关于vue生命周期的相关内容这里不多讲,可以参考博客vue生命周期的理解

这里的DOM节点在本文中为正文下所对应的div标签并设置id为text,也是方便通过id获取DOM节点。

var that = this;
const ele = document.getElementById("text");
const { mousedown, getMouseEvent } = this.useMouse();
/* 中间部分有省略,最终代码请向下看*/
 function mouseup() {
if (getMouseEvent() !== "select") {
return;
}
else{

}
ele.addEventListener("mousedown", mousedown);
ele.addEventListener("mouseup", mouseup);

**再下一步:**我们需要对选中文本做一系列操作。首先应该是能够获得选中文本的信息。这里需要知道的是: Range 接口表示一个包含节点与文本节点的一部分的文档片段。可以用 Document 对象的 Document.createRange 方法创建 Range,也可以用 Selection 对象的 getRangeAt 方法获取 Range。我们采用 const e = window.getSelection();获得鼠标选中对象。Range中包含了四个属性endContainer,startContainer, startOffset, endOffset。我们通过 const{endContainer,startContainer, startOffset, endOffset } =e.getRangeAt(0);解析出来这四个属性。 他们代表的含义分别是

  1. Range.startContainer是只读属性,返回Range开始的节点
  2. Range.endContainer 是一个只读属性。它会返回Range对象结束的Node
  3. Range.startOffset 是一个只读属性,用于返回一个表示 Range 在 startContainer 中的起始位置的数字。如果 startContainer 是一个文本(Text)、注释(Comment)或者CDATA区块(CDATASection)节点,那么返回的偏移量是从 startContainer 开始到 Range 的边界点的字符数量。对于其他的节点类型, startOffset 返回 startContainer 到边界点的子节点数量。
  4. Range.endOffset 返回代表 Range 结束位置在 Range.endContainer 中的偏移值的数字。**如果 endContainer 的 Node 类型为 Text, Comment,或 CDATASection,偏移值是 endContainer 节点开头到 Range 末尾的总字符个数。**对其他类型的 Node , endOffset 指 endContainer 开头到 Range 末尾的总 Node 个数。

为了便于理解这四个属性值,请看下面图例

在这里插入图片描述

在这里插入图片描述

可以对比得知,我们标注的内容【文本节点】,startContainer和endContainer 为其父结点,startOffset 为从父结点开始到range对象开始的字符偏移,endOffset 为从父结点开始到range对象结束的偏移,两者相减等于标注对象的字符数量

上一步我们获得了选中对象range相对于父结点的位置。那么我们怎么在整个文本节点中获得代操作的标注对象呢?我们首先通过遍历根结点id="text"下的子节点,

cutrrentNodes() {
function getNodes(ele, count = { value: 0 }) {
const list = [];
const items = ele.childNodes;
for (let index in items) {
const _item = items[index];
if (_item.nodeName === "#text") {
const item = _item;
const value = item.nodeValue || "";
const len = value.length;
list.push({
item,
index: +index,
len,
value,
});
count.value += len;
} else {
list.push(...getNodes(_item, count));
}
}
return list;
}
return {
getNodes,
};
},

在这里插入图片描述

通过findIndex()搜索,我们可以得到包含标签结点的父结点在根结点的孩子结点的数组中的索引。需要注意的是getNodes的返回值是**每一次操作之前的孩子结点数组。**这是由于我们标签修改节点生效的操作在查询之前导致的,并不是错误。

const nodes = getNodes(ele);
const start = nodes.findIndex((x) => x.item === startContainer);
const end = nodes.findIndex((x) => x.item === endContainer);

注:-1表示没有找到,其他值表示对应所处数组的下标值。

startContainer和endContainer相同的情况即没有重叠的情况下 比较理想化的情况,不存在语义标签重叠和多语义元素,这也是我们的不足之处

if (start > -1 && end > -1) {
 //startContainer和endContainer相同的情况即没有重叠的情况下比较理想化的情况,不存在语义标签重叠和多语义元素
if (start === end) {
const { item, value } = nodes[start];
const left = getText(value.slice(0, startOffset));
const center = getSpan(value.slice(startOffset, endOffset));
const right = getText(value.slice(endOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(center, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else {
//不相同的情况处理:
for (let i = start; i <= end; i++) {
const { item, value } = nodes[i];
if (i === start) {
const left = getText(value.slice(0, startOffset));
const right = getSpan(value.slice(startOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else if (i === end) {
const left = getSpan(value.slice(0, endOffset));
const right = getText(value.slice(endOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else {
item.parentNode?.replaceChild(getSpan(value), item);
}
}
}
}

比较重要的函数是getSpan()和getText()

function getText(text) {
return document.createTextNode(text);
}

getText(text)方法的作用是创建并返回一个内容为text(参数)的文本节点。 我们这么做的原因可以参照下图来理解:选中某一文本片段后,【文本内容是根据之前的偏移量进行截取的】我们将该节点分割成三个文本重新装回DOM树,左右两侧不添加特殊标签,选中的文本我们给节点外加span标签方便添加样式。

 //以这一段为例将节点分成三份
const left = getText(value.slice(0, startOffset));
const center = getSpan(value.slice(startOffset, endOffset));
const right = getText(value.slice(endOffset));

在这里插入图片描述

function getSpan(text) {
const span = document.createElement("span");
//span.classList.add("mytest");
//设置节点背景颜色
span.style.backgroundColor = that.dynamicTags[that.selectedIndex].color;
//设置节点的title属性,并赋予对应标签名
span.setAttribute("title", that.dynamicTags[that.selectedIndex].name);
let tempText = Object();
tempText.name = that.dynamicTags[that.selectedIndex].name;
tempText.comment = text;
that.addTagText(tempText);
span.addEventListener("click", function ($event) {
let tempText = Object();
tempText.name = that.dynamicTags[that.selectedIndex].name;
tempText.comment = this.innerText;
that.deleteTagText(tempText);
console.log(that.showSelectedText());
var temp = document.createTextNode(this.innerText);
var sour = $event.currentTarget;
sour.parentNode.replaceChild(temp, sour);
ele.normalize();
});
span.innerText = text.replace(/\n/g, "");
return span;
}

getSpan(text)方法中参数是我们选中的标签文本内容,我们需要再增加的操作包括,对该文本节点附带上我们所选的标签的背景颜色,并绑定title属性为标签名。

data中存放的标签类型和当前选中标签索引: dynamicTags: [ { name: “时间”, color: “red”, }, { name: “人物”, color: “yellow”, }, { name: “天气”, color: “blue”, }, ], selectedIndex: 0,

利用上述两个数据变量,可以实现对选中文本改为span标签下的文本节点,并添加上样式属性。后面就是通过对新的文本节点绑定监听再次点击事件,用于删除节点样式回复到未操作前的状态。在监听事件中我们简单(深拷贝)复制一遍节点信息【内容和节点标签名】。然后在我们选中的所有标签容器中找到对应项并删除,内部通过replaceChild()替换掉即可。最后也是官方提供的方法,多个文本节点在渲染上是看不出的,但是你通过F12可以查看源码为分割的片段,他们在DOM树上是兄弟关系。还是结合下图来理解,虽然你返还了选中内容,在外表上没有区别,但是他们依然是分割成3个片段而不是一个整体。为解决这个问题,我们使用了 ele.normalize();

在这里插入图片描述

注意:replaceChild()是通过当前选中的节点的父结点来执行的! normalize(),其作用是处理文档树中的文本节点。当在某个节点上调用这个方法时,就会在该节点的后代节点中查找。如果找到了空文本节点,则删除它;如果找到相邻的文本节点,则将它们合并为一个文本节点。

最后就是抽出我们选中的文本进行操作:

//选中好的文本添加到数组中
addTagText(tempText) {
this.selectedText.push(tempText);
},

deleteTagText(tempText) {
// 先找到
const deleteIndex = this.selectedText.findIndex((item) => {
// 不写return返回的是-1,谜
return item.comment === tempText.comment;
});
console.log(deleteIndex); // 2
this.selectedText.splice(deleteIndex, 1);
},
showSelectedText() {
return this.selectedText;
},

整体代码(html通过iframe本地引用)

代码略微修改,样式太难看了,就小动了一下。关于iframe的使用需要注意的是路径问题,如果是本地存储则静态html文件应该在public目录下,如果是网络资源加载请确保具有读权限和在线预览。

<!-- 文本框 -->
<div id="text" >
 <iframe src="/html/test2.html" height="500px" width="100%"></iframe>
</div>

在这里插入图片描述

<template>
<div class="card">
<!-- 标签值单选框-->
<div class="container" >
<el-tag
:key="tag"
v-for="(tag, index) in dynamicTags"
closable
:disable-transitions="false"
@close="handleClose(tag)"
@click="selectTag(tag, index)"
:color="tag.color"
>
{{ tag.name }}
</el-tag>
<el-input
class="input-new-tag"
v-if="inputVisible"
v-model="inputValue"
ref="saveTagInput"
size="small"
@keyup.enter="handleInputConfirm"
@blur="handleInputConfirm"
>
</el-input>
<el-button v-else class="button-new-tag" size="small" @click="showInput"
>+ New Tag</el-button
>
</div>
<el-divider content-position="center">正文</el-divider>
<!-- 文本框 -->
<div id="text" >
 <iframe src="/html/test2.html" height="500px" width="100%"></iframe>
</div>

</div>
</template>


<script>
export default {
// 数据源
data() {
return {
color1: "#409EFF",
dynamicTags: [
{
name: "时间",
color: "red",
},
{
name: "人物",
color: "yellow",
},
{
name: "天气",
color: "blue",
},
],
selectedIndex: 0,
inputVisible: false,
inputValue: "",
selectedText: [],
};
},
//生命周期函数---
mounted() {
var that = this;
const ele = document.getElementById("text");
const { mousedown, getMouseEvent } = that.useMouse();
const { getNodes } = that.cutrrentNodes();
var index = that.getSelectTag();
function getSpan(text) {
const span = document.createElement("span");
 // span.classList.add("mytest");
span.style.backgroundColor = that.dynamicTags[that.selectedIndex].color;
span.setAttribute("title", that.dynamicTags[that.selectedIndex].name);
let tempText = Object();
tempText.name = that.dynamicTags[that.selectedIndex].name;
tempText.comment = text;
that.addTagText(tempText);
span.addEventListener("click", function ($event) {
let tempText = Object();
tempText.name = that.dynamicTags[that.selectedIndex].name;
tempText.comment = this.innerText;
that.deleteTagText(tempText);
// console.log(that.showSelectedText());
var temp = document.createTextNode(this.innerText);
var sour = $event.currentTarget;
sour.parentNode.replaceChild(temp, sour);
ele.normalize();
});
span.innerText = text.replace(/\n/g, "");
return span;
}
function getText(text) {
return document.createTextNode(text);
}
function mouseup() {
if (getMouseEvent() !== "select") {
return;
}
const e = window.getSelection();
console.log(e)
if (e && e.type === "Range") {
try {
const { endContainer, startContainer, startOffset, endOffset } =
e.getRangeAt(0);
const nodes = getNodes(ele);
const start = nodes.findIndex((x) => x.item === startContainer);
const end = nodes.findIndex((x) => x.item === endContainer);
if (start > -1 && end > -1) {
if (start === end) {
const { item, value } = nodes[start];
const left = getText(value.slice(0, startOffset));
const center = getSpan(value.slice(startOffset, endOffset));
const right = getText(value.slice(endOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(center, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else {
for (let i = start; i <= end; i++) {
const { item, value } = nodes[i];
if (i === start) {
const left = getText(value.slice(0, startOffset));
const right = getSpan(value.slice(startOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else if (i === end) {
const left = getSpan(value.slice(0, endOffset));
const right = getText(value.slice(endOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else {
item.parentNode?.replaceChild(getSpan(value), item);
}
}
}
}
} catch (error) {
console.error(error);
} finally {
e.removeAllRanges();
}
}
}
ele.addEventListener("mousedown", mousedown);
ele.addEventListener("mouseup", mouseup);
},
//基本方法
methods: {
handleClose(tag) {
this.dynamicTags.splice(this.dynamicTags.indexOf(tag), 1);
},
showInput() {
this.inputVisible = true;
this.$nextTick((_) => {
this.$refs.saveTagInput.$refs.input.focus();
});
},
handleInputConfirm() {
let inputValue = new Object();
inputValue.name = this.inputValue;
inputValue.color = this.rdmRgbColor();
if (inputValue) {
this.dynamicTags.push(inputValue);
}
this.inputVisible = false;
this.inputValue = " ";
},
//选中标签
selectTag(tag, index) {
ElMessage({
message: "当前标签为:" + tag.name,
type: "success",
});
this.selectedIndex = index;
},
getSelectTag() {
return this.selectedIndex;
},
//所有色
rdmRgbColor() {
//随机生成RGB颜色
let arr = [];
for (var i = 0; i < 3; i++) {
arr.push(Math.floor(Math.random() * 256));
}
let [r, g, b] = arr;
// rgb颜色
// var color=`rgb(${r},${g},${b})`;
// 16进制颜色
var color = `#${
r.toString(16).length > 1 ? r.toString(16) : "0" + r.toString(16)
}${g.toString(16).length > 1 ? g.toString(16) : "0" + g.toString(16)}${
b.toString(16).length > 1 ? b.toString(16) : "0" + b.toString(16)
}`;
return color;
},
// 监听鼠标动作判断
useMouse() {
let last = new Date().getTime();
function mousedown() {
last = new Date().getTime();
}
function getMouseEvent() {
const d = new Date().getTime() - last;
return d > 200 ? "select" : "click";
}
return {
mousedown,
getMouseEvent,
};
},
cutrrentNodes() {
function getNodes(ele, count = { value: 0 }) {
const list = [];
const items = ele.childNodes;
for (let index in items) {
const _item = items[index];
if (_item.nodeName === "#text") {
const item = _item;
const value = item.nodeValue || "";
const len = value.length;
list.push({
item,
index: +index,
len,
value,
});
count.value += len;
} else {
list.push(...getNodes(_item, count));
}
}
return list;
}
return {
getNodes,
};
},

getTags() {
return this.dynamicTags;
},
//选中好的文本添加到数组中
addTagText(tempText) {
this.selectedText.push(tempText);
},

deleteTagText(tempText) {
// 先找到
const deleteIndex = this.selectedText.findIndex((item) => {
// 不写return返回的是-1,谜
return item.comment === tempText.comment;
});
console.log(deleteIndex); // 2
this.selectedText.splice(deleteIndex, 1);
},
showSelectedText() {
return this.selectedText;
},
},
//计算类型方法和其他方法的区别:有缓存。以内存换时间,适用于频繁使用的计算
computed: {},
};
</script>

<stylelang="scss" scoped>
.card {
width: 600px;
 
margin: 0 auto; /* 这里的0表示上面边距,左右边距需要设置为auto才能水平居中 */
padding-top: 50px;
padding-bottom: 50px;
}
.container {
display: flex;
align-items: center;
justify-content: center;
height: 100px;
}
.el-tag + .el-tag {
margin-left: 10px;
}
.button-new-tag {
margin-left: 10px;
height: 32px;
line-height: 30px;
padding-top: 0;
padding-bottom: 0;
}
.input-new-tag {
width: 90px;
margin-left: 10px;
vertical-align: bottom;
}

</style>

整体代码(html文件直接在页面中)

<template>
<div class="card">
<!-- 标签值单选框-->
<div class="container" >
<el-tag
:key="tag"
v-for="(tag, index) in dynamicTags"
closable
:disable-transitions="false"
@close="handleClose(tag)"
@click="selectTag(tag, index)"
:color="tag.color"
>
{{ tag.name }}
</el-tag>
<el-input
class="input-new-tag"
v-if="inputVisible"
v-model="inputValue"
ref="saveTagInput"
size="small"
@keyup.enter="handleInputConfirm"
@blur="handleInputConfirm"
>
</el-input>
<el-button v-else class="button-new-tag" size="small" @click="showInput"
>+ New Tag</el-button
>
</div>
<el-divider content-position="center">正文</el-divider>
<!-- 文本框 -->
<div style=" overflow-y: auto; -webkit-overflow-scrolling: touch; white-space: nowrap; width: 100%; height: 530px;">
<div id="text" >
<p><a id="_Toc467345721"></a></p>
<h1>
<a id="报告标题A"></a><a id="_Toc342212070"></a
><a id="_Toc361148894"></a><a id="_Toc487438895"></a
><a id="_Toc342212083"></a><a id="_Toc341261013"></a
><a id="_Toc181668622"></a><a id="_Toc101548670"></a>概况
</h1>
<h2><a id="_Toc487438896"></a><a id="_Toc101548671"></a>桥梁概况</h2>
<p>
太西公铁立交桥位于省道S301。路线技术等级为二级,桥梁中心桩号为K39+319,设计荷载等级为公路-I级,2008年建成通车。
</p>
<p>桥梁全长245.2m,共12跨。桥梁全宽12.50m,现阶段桥梁未设置限重标志。</p>
<p>
上部结构:12×20m预应力混凝土空心板梁,支座类型为板式橡胶支座,伸缩缝为4道异型钢伸缩缝,第6跨为下跨铁路。
</p>
<p>
下部结构:钢筋混凝土双柱框架式桥墩、桩基础;钢筋混凝土埋入式桥台、桩基础。
</p>
<p>桥面系:沥青混凝土铺装层,设置异型钢伸缩缝装置。</p>
<p>
受宁夏公路管理局的委托,中铁大桥科学研究院有限公司于2017年6月24日对太西公铁立交桥进行了定期检查。
</p>
<p>桥梁地理位置图如所示,桥梁现阶段正、立面照如所示。</p>
<img
src=""
/>
<p><a id="_Ref467309844"></a>图 ‑1 桥梁地理位置图</p>
<h2>
<a id="_Toc487438897"></a><a id="_Toc101548672"></a>上次检测及维修情况
</h2>
<p>未搜集到上次检测报告。</p>
<h2><a id="_Toc487438898"></a><a id="_Toc101548673"></a>检测目的</h2>
<p>通过对桥梁的全面检查,达到下列目的:</p>
<p>
(1)通过对桥梁缺损和病害的调查,记录其位置、大小、范围和程度,全面掌握桥梁运营现状,评定桥梁的使用功能。
</p>
<p>
(2)通过各项无损检测,掌握桥梁主要构件的材质状况,对桥梁结构的耐久性和受力性能进行综合分析和评价。
</p>
<p>
(3)通过检查和检测,分析桥梁主要病害产生的原因,判断病害的性质、发展趋势及对桥梁承载能力、耐久性和使用性能的影响程度,提出需进行特殊检查的项目,并对桥梁病害提出处治建议。
</p>
<p>
(4)对桥梁技术状况进行综合评定,更新和完善桥梁管理系统数据库,为桥梁日后的养护管理、维修加固设计提供依据。
</p>
<h2><a id="_Toc487438899"></a><a id="_Toc101548674"></a>检测依据</h2>
<p>(1)《公路桥梁技术状况评定标准》(JTG/T H21-2011);</p>
<p>(2)《公路桥涵养护规范》(JTG H11-2004);</p>
<p>(3)《公路桥梁承载能力检测评定规程》(JTG/T J21-2011);</p>
<p>(4)《公路桥涵设计通用规范》(JTG D60-2004、JTG D60-2015);</p>
<p>(5)《公路钢筋混凝土及预应力混凝土桥涵设计规范》(JTG D62-2004);</p>
<p>(6)《公路桥涵地基与基础设计规范》(JTG D63-2007);</p>
<p>
(7)《<a href="http://www.docin.com/p-248286935.html"
>回弹法检测混凝土抗压强度技术规程</a
>》(JGJ/T 23-2011);
</p>
<p>(8)《混凝土中钢筋检测技术规程》(JGJ/T 152-2008);</p>
<p>(9)《公路桥梁板式橡胶支座技术标准》JT/T 4-2004</p>
<p>(10)《公路桥梁伸缩缝装置》 JTT 327-2004</p>
<p>(11)《公路养护安全作业规程》(JTG H30-2004);</p>
<p>(12)桥梁施工、设计、竣工资料及养护、维修、加固资料;</p>
<p>(13)检测项目招、投标及合同文件。</p>
<h1>
<a id="_Toc487438902"></a
><a id="_Toc101548675"></a>桥梁部件、构件划分及编号
</h1>
<h2>
<a id="_Toc487438903"></a><a id="_Toc101548676"></a>构件划分及数量
</h2>
<p>
根据桥梁结构特点,参照《公路桥梁技术状况评定标准》(JTG/T
H21-2011),该桥部件划分及构件数量见。
</p>
<p><a id="_Ref466586631"></a>表‑1 桥梁部件划分及构件数量表</p>
<table>
<thead>
<tr>
<th>
<p><strong>序号</strong></p>
</th>
<th>
<p><strong>桥梁结构</strong></p>
</th>
<th>
<p><strong>桥梁部件</strong></p>
</th>
<th>
<p><strong>构件数量</strong></p>
</th>
<th>
<p><strong>备注</strong></p>
</th>
</tr>
</thead>
<tbody>
<tr>
<td><p>1</p></td>
<td rowspan="3"><p>上部结构</p></td>
<td><p>上部承重构件</p></td>
<td><p>132</p></td>
<td><p>空心板梁</p></td>
</tr>
<tr>
<td><p>2</p></td>
<td><p>上部一般构件</p></td>
<td><p>120</p></td>
<td><p>铰缝</p></td>
</tr>
<tr>
<td><p>3</p></td>
<td><p>支座</p></td>
<td><p>528</p></td>
<td><p>板式橡胶支座</p></td>
</tr>
<tr>
<td><p>4</p></td>
<td rowspan="7"><p>下部结构</p></td>
<td><p>翼墙、耳墙</p></td>
<td><p>4</p></td>
<td><p>/</p></td>
</tr>
<tr>
<td><p>5</p></td>
<td><p>锥坡、护坡</p></td>
<td><p>6</p></td>
<td><p>4个锥坡,2个护坡</p></td>
</tr>
<tr>
<td><p>6</p></td>
<td><p>桥墩</p></td>
<td><p>44</p></td>
<td><p>22根墩柱,11片盖梁,,11个系梁</p></td>
</tr>
<tr>
<td><p>7</p></td>
<td><p>桥台</p></td>
<td><p>4</p></td>
<td><p>2个台身,2个台帽</p></td>
</tr>
<tr>
<td><p>8</p></td>
<td><p>墩台基础</p></td>
<td><p>24</p></td>
<td><p>桩基础</p></td>
</tr>
<tr>
<td><p>9</p></td>
<td><p>河床</p></td>
<td><p>/</p></td>
<td><p>/</p></td>
</tr>
<tr>
<td><p>10</p></td>
<td><p>调治构造物</p></td>
<td><p>/</p></td>
<td><p>/</p></td>
</tr>
<tr>
<td><p>11</p></td>
<td rowspan="6"><p>桥面系</p></td>
<td><p>桥面铺装</p></td>
<td><p>1</p></td>
<td><p>沥青混凝土</p></td>
</tr>
<tr>
<td><p>12</p></td>
<td><p>伸缩缝装置</p></td>
<td><p>4</p></td>
<td><p>异型钢伸缩缝</p></td>
</tr>
<tr>
<td><p>13</p></td>
<td><p>人行道</p></td>
<td><p>/</p></td>
<td><p>/</p></td>
</tr>
<tr>
<td><p>14</p></td>
<td><p>栏杆、护栏</p></td>
<td><p>2</p></td>
<td><p>/</p></td>
</tr>
<tr>
<td><p>15</p></td>
<td><p>排水系统</p></td>
<td><p>1</p></td>
<td><p>泄水孔</p></td>
</tr>
<tr>
<td><p>16</p></td>
<td><p>照明、标志</p></td>
<td><p>1</p></td>
<td><p>标志</p></td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
</template>


<script>
export default {
// 数据源
data() {
return {
color1: "#409EFF",
dynamicTags: [
{
name: "时间",
color: "red",
},
{
name: "人物",
color: "yellow",
},
{
name: "天气",
color: "blue",
},
],
selectedIndex: 0,
inputVisible: false,
inputValue: "",
selectedText: [],
};
},
//生命周期函数---
mounted() {
var that = this;
const ele = document.getElementById("text");
const { mousedown, getMouseEvent } = this.useMouse();
const { getNodes } = this.cutrrentNodes();
var index = this.getSelectTag();
function getSpan(text) {
const span = document.createElement("span");
 // span.classList.add("mytest");
span.style.backgroundColor = that.dynamicTags[that.selectedIndex].color;
span.setAttribute("title", that.dynamicTags[that.selectedIndex].name);
let tempText = Object();
tempText.name = that.dynamicTags[that.selectedIndex].name;
tempText.comment = text;
that.addTagText(tempText);
span.addEventListener("click", function ($event) {
let tempText = Object();
tempText.name = that.dynamicTags[that.selectedIndex].name;
tempText.comment = this.innerText;
that.deleteTagText(tempText);
console.log(that.showSelectedText());
var temp = document.createTextNode(this.innerText);
var sour = $event.currentTarget;
sour.parentNode.replaceChild(temp, sour);
ele.normalize();
});
span.innerText = text.replace(/\n/g, "");
return span;
}
function getText(text) {
return document.createTextNode(text);
}
function mouseup() {
if (getMouseEvent() !== "select") {
return;
}
const e = window.getSelection();
console.log(e)
if (e && e.type === "Range") {
try {
const { endContainer, startContainer, startOffset, endOffset } =
e.getRangeAt(0);
const nodes = getNodes(ele);
const start = nodes.findIndex((x) => x.item === startContainer);
const end = nodes.findIndex((x) => x.item === endContainer);
if (start > -1 && end > -1) {
if (start === end) {
const { item, value } = nodes[start];
const left = getText(value.slice(0, startOffset));
const center = getSpan(value.slice(startOffset, endOffset));
const right = getText(value.slice(endOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(center, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else {
for (let i = start; i <= end; i++) {
const { item, value } = nodes[i];
if (i === start) {
const left = getText(value.slice(0, startOffset));
const right = getSpan(value.slice(startOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else if (i === end) {
const left = getSpan(value.slice(0, endOffset));
const right = getText(value.slice(endOffset));
item.parentNode?.insertBefore(left, item);
item.parentNode?.insertBefore(right, item);
item.parentNode?.removeChild(item);
} else {
item.parentNode?.replaceChild(getSpan(value), item);
}
}
}
}
} catch (error) {
console.error(error);
} finally {
e.removeAllRanges();
}
}
}
ele.addEventListener("mousedown", mousedown);
ele.addEventListener("mouseup", mouseup);
},
//基本方法
methods: {
handleClose(tag) {
this.dynamicTags.splice(this.dynamicTags.indexOf(tag), 1);
},
showInput() {
this.inputVisible = true;
this.$nextTick((_) => {
this.$refs.saveTagInput.$refs.input.focus();
});
},
handleInputConfirm() {
let inputValue = new Object();
inputValue.name = this.inputValue;
inputValue.color = this.rdmRgbColor();
if (inputValue) {
this.dynamicTags.push(inputValue);
}
this.inputVisible = false;
this.inputValue = " ";
},
//选中标签
selectTag(tag, index) {
ElMessage({
message: "当前标签为:" + tag.name,
type: "success",
});
this.selectedIndex = index;
},
getSelectTag() {
return this.selectedIndex;
},
//所有色
rdmRgbColor() {
//随机生成RGB颜色
let arr = [];
for (var i = 0; i < 3; i++) {
arr.push(Math.floor(Math.random() * 256));
}
let [r, g, b] = arr;
// rgb颜色
// var color=`rgb(${r},${g},${b})`;
// 16进制颜色
var color = `#${
r.toString(16).length > 1 ? r.toString(16) : "0" + r.toString(16)
}${g.toString(16).length > 1 ? g.toString(16) : "0" + g.toString(16)}${
b.toString(16).length > 1 ? b.toString(16) : "0" + b.toString(16)
}`;
return color;
},
// 监听鼠标动作判断
useMouse() {
let last = new Date().getTime();
function mousedown() {
last = new Date().getTime();
}
function getMouseEvent() {
const d = new Date().getTime() - last;
return d > 200 ? "select" : "click";
}
return {
mousedown,
getMouseEvent,
};
},
cutrrentNodes() {
function getNodes(ele, count = { value: 0 }) {
const list = [];
const items = ele.childNodes;
for (let index in items) {
const _item = items[index];
if (_item.nodeName === "#text") {
const item = _item;
const value = item.nodeValue || "";
const len = value.length;
list.push({
item,
index: +index,
len,
value,
});
count.value += len;
} else {
list.push(...getNodes(_item, count));
}
}
return list;
}
return {
getNodes,
};
},

getTags() {
return this.dynamicTags;
},
//选中好的文本添加到数组中
addTagText(tempText) {
this.selectedText.push(tempText);
},

deleteTagText(tempText) {
// 先找到
const deleteIndex = this.selectedText.findIndex((item) => {
// 不写return返回的是-1,谜
return item.comment === tempText.comment;
});
console.log(deleteIndex); // 2
this.selectedText.splice(deleteIndex, 1);
},
showSelectedText() {
return this.selectedText;
},
},
//计算类型方法和其他方法的区别:有缓存。以内存换时间,适用于频繁使用的计算
computed: {},
};
</script>

<stylelang="scss" scoped>
.card {
width: 600px;
height: 600px;
margin: 0 auto; /* 这里的0表示上面边距,左右边距需要设置为auto才能水平居中 */
padding-top: 50px;
padding-bottom: 50px;
}
.container {
display: flex;
align-items: center;
justify-content: center;
height: 100px;
}
.el-tag + .el-tag {
margin-left: 10px;
}
.button-new-tag {
margin-left: 10px;
height: 32px;
line-height: 30px;
padding-top: 0;
padding-bottom: 0;
}
.input-new-tag {
width: 90px;
margin-left: 10px;
vertical-align: bottom;
}

.text {
padding-top: 40px;
}
</style>

不足之处

  1. 没有实现标签嵌套,不能在一个标签标注的内容内再标注其他标签。
  2. 后面将会单独整理出一个小Demo,放到gitee上供大家访问。
Vue3全局实例上挂载属性方法案例讲解 网站建设

Vue3全局实例上挂载属性方法案例讲解

在大多数开发需求中,我们有时需要将某个数据,或者某个函数方法,挂载到,全局实例身上,以便于,在项目全局的任何位置都能够调用其方法,或读取其数据。 在Vue2 中,我们是在 main.js 中 直...
Vue3异步组件suspense使用详解 网站建设

Vue3异步组件suspense使用详解

vue在解析我们的组件时, 是通过打包成一个 js 文件,当我们的一个组件 引入过多子组件是,页面的首屏加载时间 由最后一个组件决定 优化的一种方式就是采用异步组件 ,先给慢的组件一个提示语或者 骨架...