xmlns:xsl='http://www.w3.org/1999/XSL/Transform'
xmlns='http://www.w3.org/2000/svg'
xmlns:saxon='java:de.fzi.xslt.rot'
xmlns:xalan='xalan://de.fzi.xslt.rot'
xmlns:oracle='http://www.oracle.com/XSL/Transform/java/de.fzi.xslt.rot'
exclude-result-prefixes='saxon xalan oracle'>
...
Префикс saxon
соответствует интерфейсу расширений в XSLT-процессоре Saxon, префикс xalan
— процессору Xalan и префикс oracle
— Oracle XSLT Processor.
Теперь осталось только найти поддерживаемый вариант расширения и произвести соответствующий вызов.
<xsl:choose>
<xsl:when test='function-available('saxon:X') '>
<line
x1='{saxon:X($x1, $y1, $alpha) + 100}'
y1='{saxon:Y($x1, $y1, $alpha) + 100}'
x2='{saxon:X($x2, $y2, $alpha) + 100}'
y2='{saxon:Y($x2, $y2, $alpha) + 100}'/>
</xsl:when>
<xsl:when test='function-available('xalan:X')'>
<line
x1='{xalan:X($x1, $y1, $alpha) + 100}'
y1='{xalan:Y($x1, $y1, $alpha) + 100}'
x2='{xalan:X($x2, $y2, $alpha) + 100}'
y2='{xalan:Y($x2, $y2, $alpha) + 100}'/>
</xsl:when>
<xsl:when test='function-available('oracle:X')'>
<line
x1='{oracle:X($x1, $y1, $alpha) + 100}'
y1='{oracle:Y($x1, $y1, $alpha) + 100}'
x2='{oracle:X($x2, $y2, $alpha) + 100}'
y2='{oracle:Y($x2, $y2, $alpha) + 100}'/>
</xsl:when>
<xsl:otherwise>
<xsl:message terminate='yes'>
<xsl:text>Necessary extension function is not available.</xsl:text>
<xsl:text>
Supported processors are:</xsl:text>
<xsl:text>
Saxon, Xalan, Oracle XSLT Processor.</xsl:text>
</xsl:message>
</xsl:otherwise>
</xsl:choose>
В случае, если хотя бы одна из функций saxon:X
, xalan:X
, oracle:X
будет доступна при обработке, она будет использована процессором для создания атрибутов элемента line
. В противном случае, процессор прервет выполнение преобразования и выведет указанное в элементе xsl:message
сообщение.
Нельзя не согласиться с тем, что приведенный выше способ не отличается элегантностью. Реализовывать свой вариант для каждого существующего процессора может быть довольно трудоемкой задачей — но такова уж плата за возможности расширений.
Функция расширения
Одной из самых полезных функций расширения, которая, как правило, уже штатно реализована во многих процессорах (то есть, не требует дополнительного программирования) является функция nodeset
. Эта функция позволяет в обход прямого запрета спецификации конвертировать результирующий фрагмент дерева во множество узлов.
Предположим, что мы создаем в переменной rtf
результирующий фрагмент дерева следующего вида:
<xsl:variable name='rtf'>
<item>1</item>
<item>2</item>
<item>3</item>
</xsl:variable>
При попытке вычислить выражение вида $rtf/item[2]
процессор в соответствии со спецификацией должен вывести ошибку, поскольку в этом фильтрующем выражении ([XP20] FilterExpr
) переменная rtf
должна содержать множество узлов, а не фрагмент дерева.
Текущая спецификация языка XPath совершенно явно говорит о том, что ни один тип данных не может быть преобразован во множество узлов. Функция nodeset
действует в обход этого запрещения: она принимает на вход результирующий фрагмент дерева и возвращает множество, состоящее из корневого узла этого фрагмента.
В разных процессорах эта функция имеет различный синтаксис: она может носить имя nodeset
или node-set
, или nodeSet
, однако семантика ее во всех случаях одинакова:
Функция принимает на вход единственный аргумент, являющийся фрагментом дерева и возвращает множество узлов, состоящее из его корня.
Предположим, что мы обрабатываем входящий документ, содержащий трехбуквенные коды языков.
<items>
<item>ENG</item>
<item>FRE</item>
<item>GER</item>
<item>GRE</item>
<item>ITA</item>
<item>NOR</item>
<item>POR</item>
<item>SPA</item>
</items>
Фрагмент шаблона, обрабатывающий этот список, может выглядеть следующим образом:
<select name='language'>
<xsl:for-each select='items/item'>
<option>