Как сохранить объект диапазона (из getSelection), чтобы воспроизвести его при другой загрузке страницы?

0

Я пытаюсь создать веб-приложение, которое позволяет пользователю выбирать текст на странице, а затем щелчком другой кнопки выделять его. Когда пользователь вернется на страницу, я хочу, чтобы подсветка отображалась на том же месте.

Поэтому я дошел до:

var selectedRange = document.getSelection().getRangeAt(0);
highlightRange(selectedRange);

Где highlightRange - это функция, которая выделяет диапазон. Это работает до сих пор.

Проблема в том, что мне нужен способ сохранить выбранный Range в базе данных, чтобы его можно было получить позже. После этого мне нужно заново создать диапазон из этих данных и снова выделить его. Я нашел этот метод:

document.createRange();

На этой странице: https://developer.mozilla.org/en-US/docs/Web/API/range

Но я не уверен, как это сделать

ОБНОВИТЬ:

Я знаю, что мне нужно будет снова создать диапазон с нуля. И для этого я буду использовать что-то вроде этого:

var range = document.createRange();

range.setStart(startNode,startOffset);
range.setEnd(endNode,endOffset);

Я могу легко хранить startOffset и endOffset, потому что это просто цифры. Но startNode и endNode являются узловыми объектами. Я не знаю, как хранить это в базе данных?

В частности, мне нужно сохранить ссылку на узел в базе данных.

  • 0
    @JayBlanchard Я обновил свой пост более подробно, спасибо.
Теги:
dom

2 ответа

2
Лучший ответ

Я решил эту проблему, сохранив 5 единиц информации из диапазона в базу данных:

var saveNode = range.startContainer;

var startOffset = range.startOffset;  // where the range starts
var endOffset = range.endOffset;      // where the range ends

var nodeData = saveNode.data;                       // the actual selected text
var nodeHTML = saveNode.parentElement.innerHTML;    // parent element innerHTML
var nodeTagName = saveNode.parentElement.tagName;   // parent element tag name

И затем, чтобы построить диапазон из базы данных, у меня есть эта функция:

function buildRange(startOffset, endOffset, nodeData, nodeHTML, nodeTagName){
    var cDoc = document.getElementById('content-frame').contentDocument;
    var tagList = cDoc.getElementsByTagName(nodeTagName);

    // find the parent element with the same innerHTML
    for (var i = 0; i < tagList.length; i++) {
        if (tagList[i].innerHTML == nodeHTML) {
            var foundEle = tagList[i];
        }
    }

    // find the node within the element by comparing node data
    var nodeList = foundEle.childNodes;
    for (var i = 0; i < nodeList.length; i++) {
        if (nodeList[i].data == nodeData) {
            var foundNode = nodeList[i];
        }
    }

    // create the range
    var range = cDoc.createRange();

    range.setStart(startNode, startOffset);
    range.setEnd(endNode, endOffset);
    return range;
}

Оттуда, я могу просто использовать свою highlightRange функцию еще раз, чтобы выделить текст.

  • 0
    это будет работать только для одной платформы .. в кроссплатформенной не может работать ..
  • 0
    @Amitraj Что по этому поводу мешает ему быть кроссплатформенным?
0

запустите следующий скрипт, когда пользователь выбирает тексты

let sel = window.getSelection();
let range = sel.getRangeAt(0);
let startNode = range.startContainer;
let endNode = range.endContainer;

if (startNode.nodeType == 3) {
  var startIsText = true;
  var startFlag = startNode.parentNode;
  startNode = startNode.nodeValue;
} else {
  var startIsText = false;
  var startFlag = startNode;
}
if (endNode.nodeType == 3) {
  var endIsText = true;
  var endFlag = endNode.parentNode;
  endNode = endNode.nodeValue;
} else {
  var endIsText = false;
  var endFlag = endNode;
}

let startOffset = range.startOffset; 
let endOffset = range.endOffset; 

let startTagName = startFlag.nodeName;
let startHTML = startFlag.innerHTML;

let endTagName = endFlag.nodeName;
let endHTML = endFlag.innerHTML;

//you can store this in database and use it
let rInfo = {
  startNode: startNode,
  startOffset: startOffset,
  startIsText: startIsText,
  startTagName: startTagName,
  startHTML: startHTML,

  endNode: endNode,
  endOffset: endOffset,
  endIsText: endIsText,
  endTagName: endTagName,
  endHTML: endHTML
};
window.localStorage.setItem("r", JSON.stringify(rInfo));

затем используйте следующие скрипты, когда пользователь вернется на страницу

function findEle(tagName, innerHTML) {
  let list = document.getElementsByTagName(tagName);
  for (let i = 0; i < list.length; i++) {
    if (list[i].innerHTML == innerHTML) {
      return list[i];
    }
  }
}

function show(startNode,startIsText,startOffset,
          endNode,endIsText,endOffset,sP,eP) {
  var s, e;
  if (startIsText) {
    let childs = sP.childNodes;
    console.log(childs);
    for (let i = 0; i < childs.length; i++) {
      console.log(childs[i].nodeValue);
      console.log(startNode);
      if (childs[i].nodeType == 3 && childs[i].nodeValue == startNode)
        s = childs[i];
      console.log(s);
    }
  } else {
    s = startNode;
  }
  if (endIsText) {
    let childs = eP.childNodes;
    console.log(childs);
    for (let i = 0; i < childs.length; i++) {
      if (childs[i].nodeType == 3 && childs[i].nodeValue == endNode)
        e = childs[i];
      console.log(e);
    }
  } else {
    e = startNode;
  }
  let range = document.createRange();
  range.setStart(s, startOffset);
  range.setEnd(e, endOffset);

  let sel = window.getSelection();
  sel.removeAllRanges();
  sel.addRange(range);
}

function use(obj) {
  let sP = findEle(obj.startTagName, obj.startHTML);
  let eP = findEle(obj.endTagName, obj.endHTML);
  show(
    obj.startNode,
    obj.startIsText,
    obj.startOffset,
    obj.endNode,
    obj.endIsText,
    obj.endOffset,
    sP,
    eP
  );
}
let a = window.localStorage.getItem("r");
use(JSON.parse(a));

Ещё вопросы

Сообщество Overcoder
Наверх
Меню