2

我有一个搜索服务器,它提供了一个测试页面,我可以在其中输入查询并以 XML 格式返回结果。我希望能够以更加用户友好的方式查看结果,因此我开始使用 XSLT,现在我有一个简单的样式表,可以将某种臃肿的 XML 转换为仅显示一些数据的简单表格。当我在本地执行此操作时效果很好 - 即,将 XSL 声明添加到 XML,然后在 Firefox 等浏览器中打开 XML。

不过,我想做的是在浏览器中实时应用这种转换,只要我通过该测试页面从服务器获取结果。我调查了一下,发现可以用 javascript 做到这一点

然后我想到了可以将javascript动态注入页面的Greasemonkey用户脚本。当我从测试页面获取 XML 结果时,我只需要一个脚本。但是,我被困在那里,因为 Greasemonkey 似乎不允许脚本在 XML 文件上运行(至少在 Firefox 中)。

我发现的例子很少,并试图将它们用作灵感,但无法使它们发挥作用。(例如,这是一个。)

这是我得到的 XML 的简化示例:

<?xml version="1.0" encoding="utf-8"?>
<Results>
    <Result>
        <Listings total="2">
            <Res>
                <Result index="0">
                    <id>123456</id>
                    <name>My Business</name>
                    <category>Restaurants</category>
                    <phone>9872365</phone>
                </Result>
            </Res>
            <Res>
                <Result index="1">
                    <id>876553</id>
                    <name>Some Other Business</name>
                    <category>Restaurants</category>
                    <phone>9834756</phone>
                </Result>
            </Res>
        </Listings>
    </Result>
</Results>

这是我在 Greasemonkey 中加载的脚本 - 没有发生任何事情:

// ==UserScript==
// @name test xml renderer
// @namespace http://sample.com
// @description stylesheet for xml results
// @include *
// ==/UserScript==

(function () {
    var xsl_str = '<?xml version="1.0" encoding="utf-8"?>\n\
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">\n\
    <xsl:output method="html"/>\n\
    <xsl:template match="/">\n\
        <html>\n\
            <head></head>\n\
            <body>\n\
                <table id="results" border="1" cellspacing="0" cellpadding="0">\n\
                    <thead>\n\
                        <tr>\n\
                            <th class="name">id</th>\n\
                            <th class="name">category ID</th>\n\
                            <th class="name">name</th>\n\
                            <th class="name">phone</th>\n\
                        </tr>\n\
                    </thead>\n\
                    <tbody>\n\
                        <xsl:for-each select="Results/Result/Listings/Res">\n\
                            <tr>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/id"/>\n\
                                </td>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/category"/>\n\
                                </td>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/name"/>\n\
                                </td>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/phone"/>\n\
                                </td>\n\
                            </tr>\n\
                        </xsl:for-each>\n\
                    </tbody>\n\
                </table>\n\
            </body>\n\
        </html>\n\
    </xsl:template>\n\
</xsl:stylesheet>\n\
';

    var processor = new XSLTProcessor();
    var dataXSL = new DOMParser().parseFromString(xsl_str, "text/xml");

    processor.importStylesheet(dataXSL);
    dataXML = document;
    var ownerDocument = document.implementation.createDocument("", "", null);
    var newFragment = processor.transformToFragment(dataXML, ownerDocument);
    dataXML.documentElement.replaceChild(newFragment, dataXML.documentElement.firstChild);
})();

当我在 Greasemonkey 中启用此脚本时,所有页面都成功替换为 XSL 模板中的上述 HTML。但是,它似乎不适用于本地 XML 文件或来自我的服务器的任何 XML。(我知道要使 Greasemonkey 与本地文件一起工作,需要about:config在 Firefox 中更改设置 - extensions.greasemonkey.fileIsGreaseable)。

我对javascript没有任何经验,所以很可能我只是犯了一个非常基本的错误。以防万一,非常感谢所有帮助。

4

2 回答 2

2

该脚本正在核对或添加到document.head. 您想用转换后的内容替换整个文档。您可以通过更改location.href为适当构造的data:URL 来做到这一点。但更简洁的方法是替换整个document.documentElement.

此脚本适用于您的测试/示例 XML 文件:

// ==UserScript==
// @name        _Test XML Renderer
// @description stylesheet for xml results
// @include     http://YOUR_SERVER.COM/YOUR_PATH/*.xml
// @grant       none
// ==/UserScript==

var xsl_str = '<?xml version="1.0" encoding="utf-8"?>\n\
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">\n\
    <xsl:output method="html"/>\n\
    <xsl:template match="/">\n\
        <html>\n\
            <head></head>\n\
            <body>\n\
                <table id="results" border="1" cellspacing="0" cellpadding="0">\n\
                    <thead>\n\
                        <tr>\n\
                            <th class="name">id</th>\n\
                            <th class="name">category ID</th>\n\
                            <th class="name">name</th>\n\
                            <th class="name">phone</th>\n\
                        </tr>\n\
                    </thead>\n\
                    <tbody>\n\
                        <xsl:for-each select="Results/Result/Listings/Res">\n\
                            <tr>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/id"/>\n\
                                </td>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/category"/>\n\
                                </td>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/name"/>\n\
                                </td>\n\
                                <td class="small" width="120">\n\
                                    <xsl:value-of select="Result/phone"/>\n\
                                </td>\n\
                            </tr>\n\
                        </xsl:for-each>\n\
                    </tbody>\n\
                </table>\n\
            </body>\n\
        </html>\n\
    </xsl:template>\n\
</xsl:stylesheet>\n\
';

var processor   = new XSLTProcessor ();
var dataXSL     = new DOMParser ().parseFromString (xsl_str, "text/xml");

processor.importStylesheet (dataXSL);

var newDoc      = processor.transformToDocument (document);

//-- These next lines swap the new, processed doc in for the old one...
window.content  = newDoc;

document.replaceChild (
    document.importNode (newDoc.documentElement, true),
    document.documentElement
);
于 2013-08-02T04:27:13.370 回答
0

我会在评论中添加这个,但我没有相关的声誉。我要检查两件事。特别是如果这是在字符串上工作,而不是像你说的那样来自服务器的文件

  1. 确保你从服务器得到的是一个字符串,而不是一个对象
  2. 确保您从服务器获取的 xml 是正确的。

如果你没有得到一个字符串,你可以做一个 ajax 请求来从 xml 中获取文本。然后将其加载为新的 xml var。

如果您真的只是想让您的 xml 用户友好,我建议您查看
http://code.google.com/p/vkbeautify

http://google-code-prettify.googlecode.com/svn/trunk/自述文件.html

那些应该保持 xml 格式,同时对其进行样式化以使其易于阅读。另外,您不必与greasemonkey 混淆,而可以只使用javascript。

于 2013-08-01T15:43:26.983 回答