您在 contenteditable 光标定位上找到的大多数答案都相当简单,因为它们仅适用于带有纯文本的输入。一旦您在容器中使用 html 元素,输入的文本就会被拆分为节点并在树结构中自由分布。
为了设置光标位置,我有这个函数,它循环提供节点内的所有子文本节点,并设置从初始节点开始到chars.count字符的范围:
function createRange(node, chars, range) {
if (!range) {
range = document.createRange()
range.selectNode(node);
range.setStart(node, 0);
}
if (chars.count === 0) {
range.setEnd(node, chars.count);
} else if (node && chars.count >0) {
if (node.nodeType === Node.TEXT_NODE) {
if (node.textContent.length < chars.count) {
chars.count -= node.textContent.length;
} else {
range.setEnd(node, chars.count);
chars.count = 0;
}
} else {
for (var lp = 0; lp < node.childNodes.length; lp++) {
range = createRange(node.childNodes[lp], chars, range);
if (chars.count === 0) {
break;
}
}
}
}
return range;
};
然后我用这个函数调用例程:
function setCurrentCursorPosition(chars) {
if (chars >= 0) {
var selection = window.getSelection();
range = createRange(document.getElementById("test").parentNode, { count: chars });
if (range) {
range.collapse(false);
selection.removeAllRanges();
selection.addRange(range);
}
}
};
range.collapse(false) 将光标设置到范围的末尾。我已经用最新版本的 Chrome、IE、Mozilla 和 Opera 对其进行了测试,它们都运行良好。
附注。如果有人感兴趣,我可以使用以下代码获取当前光标位置:
function isChildOf(node, parentId) {
while (node !== null) {
if (node.id === parentId) {
return true;
}
node = node.parentNode;
}
return false;
};
function getCurrentCursorPosition(parentId) {
var selection = window.getSelection(),
charCount = -1,
node;
if (selection.focusNode) {
if (isChildOf(selection.focusNode, parentId)) {
node = selection.focusNode;
charCount = selection.focusOffset;
while (node) {
if (node.id === parentId) {
break;
}
if (node.previousSibling) {
node = node.previousSibling;
charCount += node.textContent.length;
} else {
node = node.parentNode;
if (node === null) {
break
}
}
}
}
}
return charCount;
};
该代码执行与 set 函数相反的操作 - 它获取当前 window.getSelection().focusNode 和 focusOffset 并向后计数遇到的所有文本字符,直到它遇到具有 containerId id 的父节点。isChildOf 函数只是在运行之前检查所提供的节点实际上是所提供的parentId的子节点。
该代码应直接工作,没有变化,但我刚刚从插件我已经开发等等都出来砍死一对夫妇一个jQuery采取它这个的-让我知道如果有什么不工作!