如何突出显示 DOM Range 对象的文本?

IT技术 javascript dom range highlight
2021-01-23 15:12:08

我使用鼠标在 html 页面(在 firefox 中打开)上选择一些文本,并使用 javascript 函数,我创建/获取与所选文本相对应的范围对象。

 userSelection =window.getSelection(); 
 var rangeObject = getRangeObject(userSelection);

现在我想突出显示范围对象下的所有文本。我是这样做的,

  var span = document.createElement("span");
  rangeObject.surroundContents(span);
  span.style.backgroundColor = "yellow";

嗯,这很好用,只有当范围对象(起点和终点)位于同一个文本节点时,它才会突出显示相应的文本。Ex

    <p>In this case,the text selected will be highlighted properly,
       because the selected text lies under a single textnode</p>

但是如果范围对象覆盖多个文本节点,那么它就不能正常工作,它只突出显示位于第一个文本节点中的文本,例如

 <p><h3>In this case</h3>, only the text inside the header(h3) 
  will be highlighted, not any text outside the header</p> 

知道如何制作范围对象下的所有文本,突出显示,独立于范围是位于单个节点还是多个节点?谢谢....

5个回答

我建议使用document's 或TextRange'sexecCommand方法,该方法专为此目的而构建,但通常用于可编辑文档。这是我对一个类似问题的回答:

以下应该做你想做的。在非 IE 浏览器中,它会打开 designMode,应用背景颜色,然后再次关闭 designMode。

更新

固定在 IE 9 中工作。

2013 年 9 月 12 日更新

这是一个链接,详细介绍了删除此方法创建的高光的方法:

https://stackoverflow.com/a/8106283/96100

function makeEditableAndHighlight(colour) {
    var range, sel = window.getSelection();
    if (sel.rangeCount && sel.getRangeAt) {
        range = sel.getRangeAt(0);
    }
    document.designMode = "on";
    if (range) {
        sel.removeAllRanges();
        sel.addRange(range);
    }
    // Use HiliteColor since some browsers apply BackColor to the whole block
    if (!document.execCommand("HiliteColor", false, colour)) {
        document.execCommand("BackColor", false, colour);
    }
    document.designMode = "off";
}

function highlight(colour) {
    var range;
    if (window.getSelection) {
        // IE9 and non-IE
        try {
            if (!document.execCommand("BackColor", false, colour)) {
                makeEditableAndHighlight(colour);
            }
        } catch (ex) {
            makeEditableAndHighlight(colour)
        }
    } else if (document.selection && document.selection.createRange) {
        // IE <= 8 case
        range = document.selection.createRange();
        range.execCommand("BackColor", false, colour);
    }
}
@crizCraig:在以下简单情况下,在 IE 中对我来说效果很好:jsfiddle.net/LPnN2
2021-03-14 15:12:08
@Tim Down :在执行给定的代码后,我们如何撤消选择(突出显示)。?
2021-03-31 15:12:08
@crizCraig:在click事件触发时,选择可能已被破坏您可以改用该mousedown事件。
2021-04-03 15:12:08
如果您试图通过单击高亮按钮来触发此操作,则这在 IE 中将不起作用。您只需要使用此问题的答案将选择保存在 mouseup 上。stackoverflow.com/questions/5767037/...
2021-04-04 15:12:08
@TimDown stackoverflow.com/q/18721005/1503130我在这里遇到了类似的问题
2021-04-08 15:12:08

Rangy是一个跨浏览器的范围和选择库,它的CSS Class Applier module完美地解决了这个问题我正在使用它来在一系列桌面浏览器和 iPad 上实现突出显示,并且效果很好。

Tim Down 的回答很好,但 Rangy 使您不必自己编写和维护所有功能检测代码。

我刚刚注意到Tim Down 是 Rangy 的作者:)
2021-03-14 15:12:08

您能否详细说明此功能的需求。如果您只想更改所选文本的突出显示样式,可以使用 CSS: '::selection'

更多信息:http : //www.quirksmode.org/css/selection.html https://developer.mozilla.org/en/CSS/::selection

我想永久突出显示一些文本(这里永久意味着直到我关闭应用程序,与您的建议完全相反,当您单击其他地方时,“突出显示”消失),并且动态地(意味着我不想打开 html 文件使用文本编辑器并在那里插入一些元素)
2021-04-05 15:12:08
var userSelection = document.getSelection();
var range = userSelection.getRangeAt(0);

您可以这样使用 appendChild 和 extractContents 方法,而不是使用 aroundContent 方法:

let newNode = document.createElement('mark');
newNode.appendChild(range.extractContents());
range.insertNode(newNode);

function markNode() {
if(document.getSelection() && document.getSelection().toString().length){
let range = document.getSelection().getRangeAt(0);
let newNode = document.createElement('mark');
      newNode.appendChild(range.extractContents());
      range.insertNode(newNode);
}
else{
alert('please make selection of text to mark');
}
}

function resetContent() {
      testMe.innerHTML = `Remember: <a href="https://developer.mozilla.org/en-US/docs/Web/API/Node">Read</a> and <strong>stay strong</strong>`;
}
<p id="testMe">Remember: <a href="https://developer.mozilla.org/en-US/docs/Web/API/Node">Read</a> and <strong>stay strong</strong></p>


<div><button onclick="markNode()">markNode</button></div>
<div><button onclick="resetContent()">resetContent</button></div>

您可以尝试为周围的跨度添加一个类并应用分层 CSS 吗?

var span = document.createElement("span");
span.className="selection";
rangeObject.surroundContents(span);

在 CSS 定义中,

span.selection, span.selection * {
   background-color : yellow;  
}

我没有尝试。但只是猜测它会起作用。

它与span.style.background 没有什么不同。但是我提供的分层 CSS “span.selection *”将解决您的问题。
2021-04-01 15:12:08
你觉得这比 span.style.backgroundColor = "yellow" 这个好在哪里?
2021-04-07 15:12:08