我有一个XML数据集(从SharePoint 2007提供给DVWP),结构类似于:
<Rows>
<Row ID="1" Spanoffset="0" Span="55" Spantail="55"/>
<Row ID="2" Spanoffset="30" Span="31" Spantail="61"/>
<Row ID="3" Spanoffset="61" Span="20" Spantail="81"/>
<Row ID="4" Spanoffset="82" Span="30" Spantail="112"/>
</Rows>
假设每行代表一个从@Spanoffset
开始并且@Span
宽的栏,@Spantail
就在那里,所以如果需要,我不需要计算它。我试图有效地将行打包在一起,以便不会重叠的行组合在一起。数据集按@Spanoffset
预先排序。这基本上是一个背包问题,因为每行可以适合多个可能的组。我想要做的是一个简单的贪心解决方案,并且知道如何在c#或java中编写代码,但因为我无法将节点标记为已访问(我可以,但是当我回到递归树时我会丢失它)当我访问节点时,我似乎无法移除节点,我对如何处理它感到茫然。
例如,上面的数据看起来像这样:
<div style="clear:both">
<div style="width: 110px; margin-left: 0px; float:left;">1</div>
<div style="width: 40px; margin-left: 12px; float:left;">3</div>
<div style="width: 60px; margin-left: 2px; float:left;">4</div>
</div>
<div style="clear:both">
<div style="width: 62px; margin-left: 60px; float:left;">2</div>
</div>
我一直没有费心去尝试使浮动工作正常,因为我还没有能够让Row
个节点按正确的顺序每次只出现一次。一旦我得到它们,我相当肯定我可以得到格式化。
到目前为止,我提出的最好的XSLT是:
<xsl:template match="row">
<xsl:variable name="tail" select="@Spantail"/>
<div style="width:{2*@Span}px;
left:{2*(@Spanoffset)}px;">
<xsl:value-of select="@ID"/>
</div>
<xsl:apply-templates select="(following-sibling::row)[@Spanoffset>=$tail][1]"/>
</xsl:template>
生成
<div style="width: 110px;left: 0px">1</div>
<div style="width: 40px; left: 122px">3</div>
<div style="width: 60px; left: 164px">4</div>
<div style="width: 62px; left: 60px">2</div>
<div style="width: 40px; left: 122px">3</div>
<div style="width: 60px; left: 164px">4</div>
<div style="width: 40px; left: 122px">3</div>
<div style="width: 60px; left: 164px">4</div>
<div style="width: 60px; left: 164px">4</div>
所以我的问题是2(我看到),我认为它们是交织在一起的。
1)如何修复/重新计算我的模板只发出一次每行。
和
2)如何将分组包装在容器<div>
元素中。
对这个问题持续了2天,任何人都可以帮忙吗?
编辑:
好吧,经过一段时间的休眠后,我通过在我的模板中添加一个布尔参数来使用包装容器,并使用一些CDATA
标签在其真实时发出<div>
标签。布尔值默认为true,当我调用嵌套的apply-templates时,我将其设置为false,从而将组包装在容器中。我仍然无法看到将<Row>
标记为已访问的方式。
答案 0 :(得分:1)
我认为只使用核心XSLT非常棘手,但使用node-set()(XSLT扩展函数)会更容易:
<xsl:template name="add-row">
<xsl:param name="row"/>
<xsl:param name="prev-group" />
<xsl:if test="$row and not($row/@ID = $prev-group/Row/@ID)">
<xsl:copy-of select="$row" />
<xsl:call-template name="add-row">
<xsl:with-param name="row" select="$row/following-sibling::Row[@Spanoffset > $row/@Spantail][1]" />
<xsl:with-param name="prev-group" select="$prev-group" />
</xsl:call-template>
</xsl:if>
</xsl:template>
<xsl:template name="add-group">
<xsl:param name="first-row" />
<xsl:param name="prev-group" select="exsl:node-set(/)" />
<xsl:if test="$first-row">
<xsl:variable name="group">
<xsl:call-template name="add-row">
<xsl:with-param name="row" select="$first-row" />
<xsl:with-param name="prev-group" select="$prev-group" />
</xsl:call-template>
</xsl:variable>
<div clear="both">
<xsl:for-each select="exsl:node-set($group)/Row">
<div style="width: {2*@Span}px; left: {2*@Spanoffset}px"><xsl:value-of select="@ID"/></div>
</xsl:for-each>
</div>
<xsl:call-template name="add-group">
<xsl:with-param name="first-row" select="$first-row/following-sibling::Row[@Spanoffset < preceding-sibling::Row/@Spantail][1]" />
<xsl:with-param name="prev-group" select="exsl:node-set($group)" />
</xsl:call-template>
</xsl:if>
</xsl:template>
<xsl:template match="Rows">
<xsl:call-template name="add-group">
<xsl:with-param name="first-row" select="Row[1]" />
</xsl:call-template>
</xsl:template>
不要忘记在样式表标记中声明扩展名前缀和名称空间:
<xsl:stylesheet
version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
extension-element-prefixes="exsl"
xmlns:exsl="http://exslt.org/common">
http://exslt.org/common是Java XSLT处理器的有效命名空间,例如Xalan或Saxon;如果您使用的是MSXML,请改用urn:schemas-microsoft-com:xslt。