Search code examples
phpregexxpathdomxpath

Remove <p><br/></p> with DOMxpath or regex?


I use DOMxpath to remove html tags that have empty text node but to keep <br/> tags,

$xpath = new DOMXPath($dom);

while(($nodeList = $xpath->query('//*[not(text()) and not(node()) and not(self::br)]')) && $nodeList->length > 0) 
{
    foreach ($nodeList as $node) 
    {
        $node->parentNode->removeChild($node);
    }
}

it works perfectly until I came across another problem,

$content = '<p><br/><br/><br/><br/></p>'; 

How do remove this kind of messy <br/>and<p>? which means I don't want to allow <br/> alone with <p> but I allow <br/> with proper text like this only,

$content = '<p>first break <br/> second break <br/> the last line</p>'; 

Is that possible?

Or is it better with a regular expression?

I tried something like this,

$nodeList = $xpath->query("//p[text()=<br\s*\/?>\s*]");
    foreach($nodeList as $node) 
    {
        $node->parentNode->removeChild($node);
    }

but it return this error,

Warning: DOMXPath::query() [domxpath.query]: Invalid expression in...

Solution

  • You can select the unwanted p using XPath:

    "//p[count(*)=count(br) and br and normalize-space(.)='']"
    

    Note to select empty-text nodes shouldn't you better use (?):

    "//*[normalize-space(.)='' and not(self::br)]"
    

    This will select any element (but br) whithout text nodes, nodes like:

    <p><b/><i/></p>
    

    or

    <p> <br/>   <br/>
    </p>
    

    included.