Необходимо объединить 3-х уровневые дочерние и родительские отношения двумя параллельными способами.
1 - Выход с конкатенацией полных уровней
2 - Выход с делением по уровню
исходный код
<A>
<X id="top" text="first" text2="*"/>
<X id="middle" id-parent="top" text="second" text2="**"/>
<X id="bottom" id-parent='middle"' text="third" text2="***"/>
</A>
Текущий XSLT-код частично решает второй вариант. Он делится в соответствии с уровнем, но вычисление постфикса уровня не привязано к вычислениям иерархии. Нижний (нижний) уровень не отображается. Также порядок выходных уровней обратен желаемому.
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:apply-templates select="key('ref', @id-parent)" mode="att"/>
</xsl:copy>
</xsl:template>
<xsl:template match="X" mode="att">
<xsl:param name="pos" select="count(@*) + 1"/>
<xsl:attribute name="level-{$pos}">
<xsl:value-of select="concat(@text,' ', '| ', @text2)"/>
</xsl:attribute>
<xsl:apply-templates select="key('ref', @id-parent)" mode="att">
<xsl:with-param name="pos" select="$pos + 1"/>
</xsl:apply-templates>
</xsl:template>
</xsl:stylesheet>
взаимосвязь схема
https://xsltfiddle.liberty-development.net/bFWRAoS/1
желаемый выход
<A> <!-- KIND OF-->
<X id="top" text="first" text2="*" chain-text="first *" level-1="first | *" /> <!-- however, this upper level may remain as in the base source and don't be changed -->
<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 | ***"/>
</A>