Существующий XSLT-код необходимо усовершенствовать следующими условиями:
1 - Наряду с алгоритмом рекурсии добавить проверку того, что каждая сравниваемая строка (элемент) имеет:
- статус атрибута = 0
- , а атрибут status2 равен нулю (здесь это означает, что в элементе нет атрибута status2, фактически) (цепочка может быть создана только при выполнении этих двух условий)
2 - Порядок ранга. Наконец, все восстановленные (или восстанавливающие) цепочки должны быть упорядочены по атрибуту 'rank' (он может совпадать с нативной рекурсией, но иногда нет. В любом случае атрибут 'rank' имеет более высокий приоритет. Rank не имеет стабильной номенклатуры (например, 1-2- 3-4) но его цепочка может быть рассчитана путем сравнения «больше» - «меньше». (1 <3 <7). </p>
исходный код
<A>
<X id="top" text="first" text2="*" status="0" rank="1"/>
<X id="middle" id-parent="top" text="second" text2="**" status="0" rank="3"/>
<X id="bottom" id-parent="middle" text="third" text2="***" status="0" rank="6"/>
<X id="bottom2" id-parent="middle" text="fourth" text2="****" status="0" rank="2"/> <!--note rank! bottom and middle should be switching because "bottom2 has a "higher" rank-->
<X id="bottom3" id-parent="middle" text="fifth" text2="*****" status="2" rank="6" status2="any-value-make-its-status-not_null"/>
</A>
XSLT-преобразование
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:template match="@* | node()">
<xsl:copy>
<xsl:apply-templates select="@* | node()"/>
</xsl:copy>
</xsl:template>
<xsl:key name="ref" match="X" use="@id"/>
<xsl:template match="X">
<xsl:copy>
<xsl:copy-of select="@*"/>
<xsl:variable name="Xs">
<xsl:call-template name="XATT">
<xsl:with-param name="currentX" select="."/>
<xsl:with-param name="X" select=".">
</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:if test="$Xs[node()]">
<xsl:attribute name="chain-text">
<xsl:for-each select="$Xs/X">
<xsl:if test="position() ne 1"><xsl:text> | </xsl:text></xsl:if>
<xsl:value-of select="concat(@text, ' ', @text2)"/>
</xsl:for-each>
</xsl:attribute>
<xsl:for-each select="$Xs/X">
<xsl:attribute name="level-{position()}"><xsl:value-of select="concat(@text, ' | ', @text2)"/></xsl:attribute>
</xsl:for-each>
</xsl:if>
</xsl:copy>
</xsl:template>
<xsl:template name="XATT">
<xsl:param name="currentX"/>
<xsl:param name="X"/>
<xsl:choose>
<xsl:when test="$currentX/@id-parent">
<xsl:call-template name="XATT">
<xsl:with-param name="currentX" select="key('ref', $currentX/@id-parent)"/>
<xsl:with-param name="X">
<xsl:copy-of select="key('ref', $currentX/@id-parent)"/>
<xsl:copy-of select="$X"/>
</xsl:with-param>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:copy-of select="$X"/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
https://xsltfiddle.liberty-development.net/94AcskR
взаимосвязь схема
желаемый вывод
<?xml version="1.0" encoding="UTF-8"?><A>
<X id="top" text="first" text2="*" chain-text="first *" level-1="first | *"/>
<X id="middle" id-parent="top" text="second" text2="**" chain-text="first * | second **" level-1="first | *" level-2="second | **"/>
<X id="bottom" id-parent="middle" text="third" text2="***" chain-text="first * | second ** | third ***" level-1="first | *" level-2="second | **" level-3="third | ***"/>
<!--new conditions are clearly visible on the following lines-->
<X id="bottom2" id-parent="middle" text="fourth" text2="****" chain-text="first * | fourth **** | second **" level-1="first | *" level-2="fourth | ****" level-3="second | **"/> <!-- bottom and middle are switching because of 'rank' -->
<X id="bottom3" id-parent="middle" text="fifth" text2="*****" status="1" rank="6" status2="any-value-here-make-its-status-not_null"/> <!--nothing happens with this line because of 'status' and 'status2' attributes -->
</A>