使用一個簡單的 XSL 樣式表就可以將 xml 資料轉換成 HTML。隨著 XML 規範的不斷演進,在新的版本中滿足每個人的需要似乎已經成為必要;不幸的是,進行簡單的轉換一直困擾著規範。
假設我有一個表示一個頁面內容的 XML 數據,現在我想將其內容轉換成佈局。以下是我想要轉換的 XML:
<?xml version='1.0'?> <?xml-stylesheet type="text/xsl" href="article.xsl"?> <xml> <folders> <folder> <text>Folder 1</text> <files> <file> <text>File 1</text> <fields> <field> <data> <type>string</type> <length>50</length> <value>some data</value> </data> </field> </fields> </file> </files> </folder> </folders> </xml>
這個內容表示一組資料夾、檔案和網域。每個資料夾包含文件,每個文件包含用於輸入資料的網域。資料夾組中的每個資料夾都會表現為一個 TABLE 的第一行的一個 TR 元素和一個 TD 元素。檔案組中的每個檔案都會表示為嵌套在資料夾 TR 元素中的一個 TABLE 元素的第一行的一個 TR 元素和一個 TD 元素。域組中的每個域都將在相關的文件中表現為一個 INPUT。
為了實現這個想法,我們需要將遍歷 XML 然後根據XSL 建立一個表格。
下面是用於這個轉換的 XSL:
<?xml version="1.0"?> <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0" xmlns:msxsl="urn:schemas-microsoft-com:xslt" xmlns:fn="http://www.mycompany.com/mynamespace"> <xsl:output method="html"/> <msxsl:script language="JScript" implements-PRefix="fn"> function getElementCount(nodelist, what) { var rtrn = 0; rtrn = nodelist[0].parentNode.selectNodes(what).length; return (rtrn + 1); //1 is added for filler TD } </msxsl:script> <xsl:template match="/"> <TABLE CELLSPACING="0" CELLPADDING="0" WIDTH="100%" BORDER="0" ID="tblRoot" NAME="tblRoot" style="table-layout:fixed;"> <TR> <xsl:for-each select="xml/folders/folder"> <xsl:element name="TD"> <xsl:attribute name="style">width:55px</xsl:attribute> <xsl:value-of select="text"/> </xsl:element> </xsl:for-each> <TD> </TD> </TR> <xsl:for-each select="xml/folders/folder"> <TR> <xsl:element name="TD"> <xsl:attribute name="colspan"> <xsl:value-of select="fn:getElementCount(., 'folder')"/> </xsl:attribute> <TABLE CELLSPACING="0" CELLPADDING="0" WIDTH="100%" BORDER="0" style="table-layout:fixed;"> <TR> <xsl:for-each select="files/file"> <xsl:element name="TD"> <xsl:attribute name="style">width:55px;</xsl:attribute> <xsl:value-of select="text"/> </xsl:element> </xsl:for-each> <TD> </TD> </TR> <xsl:for-each select="files/file"> <TR> <xsl:element name="TD"> <xsl:attribute name="colspan"> <xsl:value-of select="fn:getElementCount(., 'file')"/> </xsl:attribute> <xsl:for-each select="fields/field"> <xsl:element name="INPUT"> <xsl:attribute name="type">text</xsl:attribute> <xsl:attribute name="maxlength"> <xsl:value-of select="data/length"/> </xsl:attribute> <xsl:attribute name="value"> <xsl:value-of select="data/value"/> </xsl:attribute> </xsl:element><BR/> </xsl:for-each> </xsl:element> </TR> </xsl:for-each> </TABLE> </xsl:element> </TR> </xsl:for-each> </TABLE> </xsl:template> </xsl:stylesheet>
在 stylesheet 標記中,我們設定了幾個命名空間,包括定義所有 xsl 轉換標記的 xsl 命名空間。能夠讓我們建立可以在樣式表中使用的使用者函數的msxml 命名空間。我使用它來獲得所有的子元素,以便得到一個 TD 標記的 COLSPAN 屬性集。用於加入一組使用者定義的函數的fn 命名空間,該命名空間由msxml:script element 建立。
然後,我們建立外部 TABLE 和第一個 TR。在 TR 中,我為 XML 中指定的每個資料夾建立一個 TD。我使用了 xsl:element 標記,因為它允許我新增自訂屬性或執行一個函數來為另一個 TD 元素中的 COLSPAN 屬性設定一個屬性。
在為每個資料夾建立了所需的 TD 之後,我開始為每個資料夾建立 TR。我只向這個 TR 添加一個 TD,但是我將它的 COLSPAN 屬性設為等於資料夾組中資料夾標記的數量再加一。多出的一是用來在一個固定佈局樣式的 TABLE 中填入空格用的。
為了得到 COLSPAN,我傳入目前的上下文(這裡由「.」指定)和我想計算的節點的名字。在我的函數中,我得到當前的上下文,paraentNode,和 XPath 查詢中指定的節點的數量。然後函數傳回這個數量加上一個以填充 TD。
有了這個 TD,我在其中嵌入另一個 TABLE,該 TABLE 包含檔案組中的每個檔案。從這一點開始,其流程就與外部 TABLE 轉換一樣。最後的步驟是新增每個文件中的網域。這次我沒有再建立嵌入的 TABLE,只是把這些網域加入到目前的 TD 中。
一旦我完成了通用佈局,我就可以開始添加使用者介面功能了,例如隱藏掉其它的資料夾和檔案行,直到使用者點擊了相關的選項卡。這項功能可以透過編寫支援此功能的腳本來實現,添加一個 onclick xsl:attribute 元素到資料夾和檔案 TD 元素,然後將它的值設為腳本函數的名字。
最後,在通用功能完成之後,你可以加入class xsl:attributes 並在 STYLE 或 CSS 中加入相關的classNames 得到你想要的外觀。
這個範例為在部署 Web 資料解決方案中使用的 File-Folder-Field 視圖建立了一個基礎。存取 MSDN 找到更多有關Microsoft 的 XML 規範的資訊。
以上就是使用XSLT將XML資料轉換成HTML的內容,更多相關內容請關注PHP中文網(m.sbmmt.com)!