0

what I'm trying to achieve is when I split the inner contents of an element, I get each item seperate, but the html element needs to be 1 element in the split.

For example:

<p id="name1" class=""> We deliver
      <span class="color-secondary">software</span> &
      <span class="color-secondary">websites</span> for your organization<span class="color-secondary">.</span>
</p>

Like in the example above, I want to make anything inside the <span> 1 array item after splitting the inner contents of #name1.

So in other words, I want the split array to look like this:

[
 'we',
 'deliver',
 '<span class="color-secondary">software</span>',
 '&',
 '<span class="color-secondary">websites</span>'

 ... etc.
]

Currently this is what I have. But this does not work since it ignores the text inside of the html element and therefore splits it halfway through the element. I would also like it to be any html element, and not just limited to <span>.

let sentence = el.innerHTML; // el being #name1 in this case
let words = sentence.split(/\s(?=<span)/i);

How would I be able to achieve this with regex? Is this possible? Thank you for any help.

3
  • You should be using a DOMParser, not regex for this. Commented May 21, 2022 at 11:44
  • What do you expect if a span contains multiple words? Commented May 21, 2022 at 11:44
  • A two step method can be a solution. Step one using RegExp to pick out span pairs. Step two to split each word for others., Commented May 21, 2022 at 11:49

2 Answers 2

1

Here is a DOMParser based solution which parses the HTML and then iterates over the top node's children, pushing the HTML into the result array if the node is an element, or splitting the text on space (if it is a text element) and adding those values to the result array:

const html = `<p id="name1" class=""> We deliver
      <span class="color-secondary">software</span> &
      <span class="color-secondary">websites</span> for your organization<span class="color-secondary">.</span>
</p>`

const parser = new DOMParser();
const s = parser.parseFromString(html, 'text/html');
let result = [];
for (el of s.body.firstChild.childNodes) {
  if (el.nodeType == 3 /* TEXT_NODE */ ) {
    result = result.concat(el.nodeValue.trim().split(' ').filter(Boolean));
  } 
  else if (el.nodeType == 1 /* ELEMENT_NODE */ ) {
    result.push(el.outerHTML);
  }
}

console.log(result);

Sign up to request clarification or add additional context in comments.

Comments

1

Details are commented in example below

const nodeSplitter = (mainNode) => {
  let scan;
  /*
  Check if initial node has text or elements
  */
  if (mainNode.hasChildNodes) {
    scan =
      /*
      Collect all elements, text, and comments
      into an array
      */
      Array.from(mainNode.childNodes)
      /*
      If node is an element, return it...
      ...if node is text, use `.matchAll()` to
      find each word and add to array...
      .filter() any falsy values and flatten
      the array and then return it
      */
      .flatMap(node => {
        if (node.nodeType === 1) {
          return node;
        } else if (node.nodeType === 3) {
          const rgx = new RegExp(/[\w\\\-\.\]\&]+/, 'g');
          let strings = [...node.textContent.matchAll(rgx)]
            .filter(node => node).flat()
          return strings;
        } else {
          /*
          Otherwise, return empty array which is 
          basically nothing since .flatMap() 
          flattens an array as default
          */
          return [];
        }
      });
  } else {
    // Return if mainNode is empty
    return;
  }
  // return results
  return scan;
}

const main = document.getElementById('name1');
console.log(nodeSplitter(main));
<p id="name1" class=""> We deliver
  <span class="color-secondary">software</span> &
  <span class="color-secondary">websites</span> for your organization
  <span class="color-secondary">.</span>
</p>

Comments

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.