最近工作中遇到这样一个需求对富文本内容进行翻译成英语,放到另一个富文本中。然而,当富文本中的内容非常多时,翻译速度会变得很慢,甚至接口可能报错。为了解决这个问题,我们可以遍历富文本中的所有节点,将每个节点的内容提取为字符串数组,翻译完成后再替换回去。
模拟两个富文本编辑器的翻译效果:
下面是对应源代码:
<body>
<div id="chinese">
<div class="document">
<div class="section">
<p>
窗前明月光,疑是<strong>地上</strong>霜;<br />举头望<span
style="color: rgb(45, 194, 107)"
data-mce-style="color: #2dc26b;"
>明月</span
>,<span
style="
text-decoration: underline;
background-color: rgb(224, 62, 45);
"
data-mce-style="text-decoration: underline; background-color: #e03e2d;"
>低头</span
>思故乡;
</p>
<p>
<img
src=""
width="69"
height="89"
/>
</p>
</div>
</div>
</div>
<div id="translateButton"><button>Translate</button></div>
<div id="english"></div>
</body>
<style>
#chinese {
width: 500px;
height: 300px;
background: pink;
margin: 0 auto;
}
#english {
width: 500px;
height: 300px;
background: yellow;
margin: 0 auto;
}
#translateButton {
margin: 0 auto;
width: 200px;
text-align: center;
}
</style>
以看到,粉色的富文本中包含一张体积较大的base64图片。如果直接将其与其他内容一同发送给后端,传输的数据量会大幅增加,从而导致接口响应变慢,甚至可能报错。
function extractText(node, textArray = []) {
if (node.nodeType === Node.TEXT_NODE) {
const trimmedText = node.textContent.trim()
if (trimmedText) {
textArray.push({ node, text: trimmedText })
}
} else if (node.nodeType === Node.ELEMENT_NODE) {
node.childNodes.forEach((child) => {
extractText(child, textArray)
})
}
console.log(textArray)
return textArray
}
因此,我们可以传入富文本中的整个 DOM 节点,然后通过判断 node.nodeType
是否为 Node.TEXT_NODE
或 Node.ELEMENT_NODE
来确定每个节点是文本节点还是元素节点。对于文本节点,可以通过 node.textContent
提取其中的文本内容。
Node.ELEMENT_NODE
(值为1
): 表示元素节点,例如<div>
、<p>
等。Node.TEXT_NODE
(值为3
): 表示文本节点,即元素或属性中的文本内容。如果节点是文本节点
这样就将文本节点提取到一个数组中:
接下来我模拟一个翻译好的对应数组
// 模拟翻译内容数组
const translations = [
'The bright moonlight before my bed, I suspect',
'on the ground',
'is frost;',
'I raise my head to gaze at',
'the bright moon',
',',
'Lower your head',
'thinking of hometown;'
]
拿到翻译好的内容后,再写一个替换函数,按顺序替换掉将提取出来的文本节点
function replaceText(textArray, translations) {
textArray.forEach((item, index) => {
item.node.textContent = translations[index] || item.text
})
}
document.getElementById('translateButton').addEventListener('click', () => {
const documentDiv = document.querySelector('.document')
const clonedDiv = documentDiv.cloneNode(true)
const allTextArray = extractText(clonedDiv)
replaceText(allTextArray, translations)
const englishDiv = document.getElementById('english')
englishDiv.innerHTML = ''
englishDiv.appendChild(clonedDiv)
})
- 点击翻译按钮时,获取
.document
类对应的元素,假设这是第一个富文本编辑器。 - 克隆该元素及其子节点(深度克隆),以便在不修改原始内容的情况下操作。
- 使用
extractText
函数提取克隆元素中的文本节点。 - 调用
replaceText
函数,用翻译后的文本替换克隆元素中的文本节点。 - 清空
#english
元素的内容,然后将翻译后的克隆内容添加到该元素中。