skip to Main Content

this is my setup, the solution can be in JS or CSS:

<p>
  "Text"
  <br>
  <br>
 "Text2"
  <br>
  <br>
  <br> -> should be hidden
  <br> -> should be hidden
"Text3"

</p>

I want to check every paragraph for br, if there are more than 2 br DIRECTLY following each other, then I want to hide every br (in that break section) except for the first and second br.

In other words: There is a paragraph. This p contains text, followed by two breaks – thats okay. Then there is more text in the same p, this time followed by 3 breaks. That is not okay, because the 3rd break should be hidden there. There could be more text with breaks following after that.

So in this example only the last two breaks should be hidden. Targeting the childs is not working, because then only the first and second breaks are visible but every break after that is hidden.

4

Answers


  1. Let’s try with regexp. I was thinking along the line of /<br>s*<br>s*((<br>s*)*)/

    Then I looked up for negative lookbehind something. It seems to be working.

    var str = `
    <p>
      "...."
      <br>
      <br>
     "...."
      <br>
      <br>
      <br>
      <br>
    </p>`;
    
    
    var reg = /(?<=<br>s*<br>s*)((<br>s*)+)/gm;
    
    console.log(str.replace(reg, ''));
    .as-console-wrapper {max-height: 100% !important; top: 0;}
    Login or Signup to reply.
  2. Using RegEx for processing HTML is generally not recommended but in this case, I think it’s the only answer.

    According to the other answer and using JS:

    const regex = /(?<=<br>s*<br>s*)((<br>s*)*)/gm;
    const paragraph = document.querySelector('p')
    paragraph.innerHTML = paragraph.innerHTML.replace(regex, '')
    <p>
      Hi!
      <br>
      <br>
      Welcome to StackOverflow!
      <br>
      Thanks!
      <br>
      <br>
      <br>
      <br>
      Bye!
    </p>

    If wrapping up the text elements in span tags is possible, as epascarello commented, we can simply do this:

    p>br+br+br {
      display: none;
    }
    <p>
      <span>Hi!</span>
      <br>
      <br>
      <span>Welcome to StackOverflow!</span>
      <br>
      <span>Thanks!</span>
      <br>
      <br>
      <br>
      <br>
      <span>Bye!</span>
    </p>
    Login or Signup to reply.
  3. Assuming you are just going to have BR and text nodes, you can select all the children and loop over them and count the occurrences of the BR tag. If you are over 2, start removing.

    // Get all of the nodes in the p element into an array
    const nodes = [...document.querySelector("p").childNodes];
    
    // loop over the nodes so we can inspect each one
    nodes.reduce((count, node) => {
      // Do we have text node?
      if(node.nodeType === 3) {
        // is the text node just whitespace
        const isEmpty = node.nodeValue.trim().length === 0;
    
        // if whitespace ignore
        // if text, reset count
        return isEmpty ? count : 0; 
      }
      // update the count for number of brs encountered
      count++;
      // if we are at number 3 or greater remove it
      if (count>2) node.remove();
      // return the count
      return count;
    }, 0);
    <p>
      "Text"
      <br>
      <br> 
      "Text2"
      <br>
      <br>
      <br>
      <br>
      "Text3"
    </p>

    And if you have more than one you have to select all the paragraphs and loop over that and process each one.

    const paragraphs = document.querySelectorAll("p");
    
    paragraphs.forEach( elem => {
      const nodes = [...elem.childNodes];
      nodes.reduce((count, node) => {
        if(node.nodeType === 3) {
          const isEmpty = node.nodeValue.trim().length === 0;
          return isEmpty ? count : 0; 
        }
        count++;
        if (count>2) node.remove();
        return count;
      }, 0);
    });
    <p>
      "Text"
      <br>
      <br> 
      "Text2"
      <br>
      <br>
      <br>
      <br>
      "Text3"
    </p>
    <p>
      "Text"
      <br>
      <br> 
      <br>
      <br> 
      <br>
      <br> 
      "Text2"
      <br>
      <br>
      <br>
      <br>
      "Text3"
    </p>
    Login or Signup to reply.
  4. You can wrap your text nodes in a span, using the standard find-text-nodes trick, then use .nextUntil to hide the extra nodes:

    $("span + br + br").nextUntil("span").hide();
    

    Updated snippet:

    $("p")
      .contents()
      .filter(function() {
        return this.nodeType === 3 && this.textContent.trim() != ""; //Node.TEXT_NODE
      }).wrap("<span>");
      
    $("span + br + br").nextUntil("span").hide();
    <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
    <p>
      "Text"
      <br>
      <br>
     "Text2"
      <br>
      <br>
      <br> 
      <br>
      "Text3"
      <br>
      <br>
      <br> 
      <br>
      "Text3"
      <br>
      <br>
      <br> 
      <br>
      <br>
      <br>
      <br> 
      <br>
      "Text3"
    </p>

    If you don’t want to use .nextUntil but are still ok with wrapping the text in a span (or can change the HTML to wrap the text in a span), then you can also use css:

    Hide all <br>, then show the ones you want span + br, span + br + br

    Updated snippet:

    $("p")
      .contents()
      .filter(function() {
        return this.nodeType === 3 && this.textContent.trim() != ""; //Node.TEXT_NODE
      }).wrap("<span>");
     
    span { color: red; } /* just to show they've been wrapped in a span */
    
    br { display:none; }
    p > span + br,
    p > span + br + br { display:inline; }
    <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
    <p>
      "Text"
      <br>
      <br>
     "Text2"
      <br>
      <br>
      <br> 
      <br>
      "Text3"
      <br>
      <br>
      <br> 
      <br>
      "Text3"
      <br>
      <br>
      <br> 
      <br>
      <br>
      <br>
      <br> 
      <br>
      "Text3"
      
    
    </p>
    Login or Signup to reply.
Please signup or login to give your own answer.
Back To Top
Search