Интересно, можно ли и как можно зарегистрировать функцию пользовательского пространства PHP с XSLT-процессором, способным не только взять массив узлов, но и вернуть его?
Прямо сейчас PHP жалуется на преобразование массива в строковое преобразование с использованием общей настройки:
function all_but_first(array $nodes) { array_shift($nodes); shuffle($nodes); return $nodes; }; $proc = new XSLTProcessor(); $proc->registerPHPFunctions(); $proc->importStylesheet($xslDoc); $buffer = $proc->transformToXML($xmlDoc);
XMLDocument ( $xmlDoc
) для преобразования может, например, быть:
<p> <name>Name-1</name> <name>Name-2</name> <name>Name-3</name> <name>Name-4</name> </p>
В таблице стилей он называется так:
<xsl:template name="listing"> <xsl:apply-templates select="php:function('all_but_first', /p/name)"> </xsl:apply-templates> </xsl:template>
Уведомление следующее:
Примечание. Преобразование массива в строку
Я не понимаю, почему, если функция получает массив в качестве входных данных, не может также вернуть массив?
Я также пытался использовать другие имена функций, поскольку я видел, что есть php:functionString
но все попытки до сих пор ( php:functionArray
, php:functionSet
и php:functionList
) не работали.
В руководстве PHP написано, что я могу вернуть другие элементы, содержащие DOMDocument
, однако эти элементы больше не из исходного документа. Это не имеет большого значения для меня.
Что-то, что работает для меня, – это вернуть экземпляр DOMDocumentFragment
вместо массива. Поэтому, чтобы попробовать это на вашем примере, я сохранил ваш вход как foo.xml
. Затем я сделал foo.xslt
следующим образом:
<xsl:stylesheet version="1.0" xmlns:xsl='http://www.w3.org/1999/XSL/Transform' xmlns:php="http://php.net/xsl"> <xsl:template match="/"> <xsl:call-template name="listing" /> </xsl:template> <xsl:template match="name"> <bar> <xsl:value-of select="text()" /> </bar> </xsl:template> <xsl:template name="listing"> <foo> <xsl:for-each select="php:function('all_but_first', /p/name)"> <xsl:apply-templates /> </xsl:for-each> </foo> </xsl:template> </xsl:stylesheet>
(Это всего лишь ваш пример с оболочкой xsl:stylesheet
чтобы вызвать его.) И в самом деле, foo.php
:
<?php function all_but_first($nodes) { if (($nodes == null) || (count($nodes) == 0)) { return ''; // Not sure what the right "nothing" return value is } $returnValue = $nodes[0]->ownerDocument->createDocumentFragment(); array_shift($nodes); shuffle($nodes); foreach ($nodes as $node) { $returnValue->appendChild($node); } return $returnValue; }; $xslDoc = new SimpleXMLElement('./foo.xslt', 0, true); $xmlDoc = new SimpleXMLElement('./foo.xml', 0, true); $proc = new XSLTProcessor(); $proc->registerPHPFunctions(); $proc->importStylesheet($xslDoc); $buffer = $proc->transformToXML($xmlDoc); echo $buffer; ?>
Важной частью является вызов ownerDocument->createDocumentFragment()
чтобы объект возвращался из функции.