Пожалуйста, предложите, как добраться до внутреннего mrow/mo, содержащего фигурные скобки.
В данных примерах фигурные скобки первого уровня правильно преобразуются в «MFENCED» при mrow (если его первый и последний дочерние элементы должны быть «MO» со знаками фигурных скобок). Но невозможно изменить скобки второго уровня, где они также имеют «MFRAC» в качестве потомка. Шаблон должен совпадать с 'MROW' (как указано).
Ввод XML:
<article>
<body>
<math id="m1">
<mrow>
<mo>(</mo><!--first level braces open-->
<mi>u</mi>
<mo>+</mo>
<mi>g</mi>
<mi>=</mi>
<mrow>
<mo>(</mo><!--second level braces open-->
<mfrac>
<mrow><mn>1</mn></mrow>
<mrow><mn>2</mn></mrow>
</mfrac>
<mo>)</mo><!--second level braces close-->
</mrow>
<mo>)</mo><!--first level braces close-->
</mrow>
</math>
<math id="m2">
<mrow>
<mo>(</mo>
<mrow>
<mfrac>
<mn>8</mn>
<mn>9</mn>
</mfrac>
</mrow>
<mo>)</mo>
</mrow>
</math>
</body>
</article>
XSLT 2.0:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="2.0">
<xsl:template match="node()|@*">
<xsl:copy><xsl:apply-templates select="@*|node()"/></xsl:copy>
</xsl:template>
<xsl:template match="mrow[matches(child::*[1][name()='mo'], '^(\(|\[|\{)$')]
[matches(child::*[position()=last()][name()='mo'], '^(\)|\]|\})$')]">
<xsl:choose>
<xsl:when test="descendant::mfrac">
<xsl:copy>
<xsl:element name="mfenced">
<xsl:attribute name="open"><xsl:value-of select="child::*[1][name()='mo']"/></xsl:attribute>
<xsl:attribute name="close"><xsl:value-of select="child::*[position()=last()][name()='mo']"/></xsl:attribute>
<xsl:for-each select="*">
<xsl:if test="position()=1"/>
<xsl:if test="not(position()=1) and not(position()=last())">
<xsl:copy><xsl:apply-templates select="@* | node()"/></xsl:copy>
</xsl:if>
<xsl:if test="position()=last()"/>
</xsl:for-each>
</xsl:element>
</xsl:copy>
</xsl:when>
<xsl:otherwise>
<xsl:copy><xsl:apply-templates select="@* | node()"/></xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
Необходимый результат:
<article>
<body>
<math id="m1">
<mrow>
<mfenced open="(" close=")"><mi>u</mi><mo>+</mo><mi>g</mi><mi>=</mi>
<mrow>
<mfenced open="(" close=")"><!-- this node or modification required, because, within this MFRAC presents, then it should convert to 'MFENCED' -->
<mfrac>
<mrow><mn>1</mn></mrow>
<mrow><mn>2</mn></mrow>
</mfrac>
</mfenced>
</mrow>
</mfenced>
</mrow>
</math>
<math id="m2">
<mrow>
<mfenced open="(" close=")">
<mrow>
<mfrac>
<mn>8</mn>
<mn>9</mn>
</mfrac>
</mrow>
</mfenced>
</mrow>
</math>
</body>
</article>