PHP Dom Удалить элемент оставить содержимое - PullRequest
7 голосов
/ 13 января 2011

Я пытаюсь удалить определенные ссылки в зависимости от их идентификатора, но оставляю содержание ссылки. Например я хочу включить

Some text goes <a href="http://www.domain.tdl/" id="remove">here</a>

до

Some text goes here

Я пытался использовать ниже.

$dom = new DOMDocument;
$dom->loadHtml(mb_convert_encoding($html, 'HTML-ENTITIES', "UTF-8"));
$xp = new DOMXPath($dom);

foreach($xp->query('//a[contains(@id="remove")]') as $oldNode) {
$revised = strip_tags($oldNode);
}

$revised = mb_substr($dom->saveXML($xp->query('//body')->item(0)), 6, -7, "UTF-8");
echo $revised;

примерно взято из здесь , но он просто выплевывает то же содержимое, что и $html.

Есть идеи, как мне этого добиться?

Ответы [ 3 ]

14 голосов
/ 13 января 2011

Это моя функция для этого:

function DOMRemove(DOMNode $from) {
    $sibling = $from->firstChild;
    do {
        $next = $sibling->nextSibling;
        $from->parentNode->insertBefore($sibling, $from);
    } while ($sibling = $next);
    $from->parentNode->removeChild($from);    
}

Так вот:

$dom->loadHTML('Hello <a href="foo"><span>World</span></a>');
$a = $dom->getElementsByTagName('a')->item(0); // get first
DOMRemove($a);

Должно дать вам:

Hello <span>World</span>

Чтобы получить узлы с определенным идентификатором, используйте XPath:

$xpath = new DOMXpath($dom);
$node = $xpath->query('//a[@id="something"]')->item(0); // get first
DOMRemove($node);
2 голосов
/ 27 января 2012

Подход, аналогичный ответу @ netcoder, но с использованием другой структуры цикла и методов DOMElement.

$html = '<html><body>This <a href="http://www.domain.tdl/" id="remove">link</a> was removed.</body></html>';
$dom = new DOMDocument();
$dom->loadHTML($html);
$xpath = new DOMXPath($dom);
foreach ($xpath->query('//a[@id="remove"]') as $link) {
  // Move all link tag content to its parent node just before it.
  while($link->hasChildNodes()) {
    $child = $link->removeChild($link->firstChild);
    $link->parentNode->insertBefore($child, $link);
  }
  // Remove the link tag.
  $link->parentNode->removeChild($link);
}
$html = $dom->saveXML();
1 голос
/ 13 января 2011

Использовать :

 //a[@id='remove']/node() 
| 
 //*[a[@id='remove']]/node()[not(self::a[@id=''remove])]

Выбирает всех потомков любого a с атрибутом id со значением "remove" и всех предшествующих и последующих братьев и сестер этого aкоторые сами по себе не являются a, имеющими атрибут id со значением "remove"

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...