Chrome и Safari XSLT с использованием JavaScript

У меня есть следующий код, который применяет стиль XSLT

Test.Xml.xslTransform = function(xml, xsl) {
    try {
        // code for IE
        if (window.ActiveXObject) {
            ex = xml.transformNode(xsl);
            return ex;
        }
        // code for Mozilla, Firefox, Opera, etc.
        else if (document.implementation && document.implementation.createDocument) {
            xsltProcessor = new XSLTProcessor();
            xsltProcessor.importStylesheet(xsl);
            resultDocument = xsltProcessor.transformToFragment(xml, document);
            return resultDocument;
        }
    } catch (exception) {
        if (typeof (exception) == "object") {
            if (exception.message) {
                alert(exception.message);
            }
        } else {
            alert(exception);
        }
    }

Код работает в IE и firefox, но не в Chrome и Safari. Любые идеи, почему?

Обновление

ResultDocument = xsltProcessor.transformToFragment(xml, document);

В приведенной выше строке возвращается значение null. Ошибка не возникает.

Обновление

Код не работает, так как файл xslt содержит xsl: include. Нужно найти способ заставить включить работу, я буду вносить прогресс здесь.

Обновление

Было решено, что я использую плагин http://plugins.jquery.com/project/Transform/. Я пытаюсь использовать клиентскую библиотеку, так как пример include работает здесь (http://daersystems.com/jquery/transform/).

Код работает в IE, но все еще не в хроме.

Test.Xml.xslTransform = function(xml, xsl) {
        try {
                $("body").append("<div id='test' style='display:none;'></div>");
                var a = $("#test").transform({ xmlobj: xml, xslobj: xsl });
                return a.html();
        }
        catch (exception) {
            if (typeof (exception) == "object") {
                if (exception.message) {
                    alert(exception.message);
                }
            } else {
                alert(exception);
            }

        }
    }

xml и xsl - оба передаваемых объекта.

Обновление

Я попытался изменить XSL файл на нечто очень простое без включения, и Chrome по-прежнему не применяет таблицу стилей и IE. XSL, который вводится как объект:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:rs="urn:schemas-microsoft-com:rowset"
    xmlns:z="#RowsetSchema"
    xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"
    xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:xsd="http://www.w3.org/2001/XMLSchema"
    xmlns:spsoap="http://schemas.microsoft.com/sharepoint/soap/"
    >
    <xsl:output method="html"/>
    <xsl:template match="/">
        <h1>test</h1>
    </xsl:template>

</xsl:stylesheet>

Обновление

Конечный результат, который я хочу, - это применить xsl к файлу xml. В него есть файл xsl. Я хочу, чтобы trasnfer выполнялся на клиенте в идеале.

Обновление Руперт мог бы вы обновить вопрос с помощью xml и как вы вызываете Test.Xml.xslTransform?

Я получил xml, используя ie8

<?xml version="1.0"?>
<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"><soap:Body><SearchListItemsResponse xmlns="http://schemas.microsoft.com/sharepoint/soap/"><SearchListItemsResult><listitems xmlns:s="uuid:BDC6E3F0-6DA3-11d1-A2A3-00AA00C14882" xmlns:dt="uuid:C2F41010-65B3-11d1-A29F-00AA00C14882" xmlns:rs="urn:schemas-microsoft-com:rowset" xmlns:z="#RowsetSchema">
                    <rs:data ItemCount="1">
                        <z:row ows_Title="Test" ows_FirstName="Test 4" ows_UniqueId="74;#{1A16CF3E-524D-4DEF-BE36-68A964CC24DF}" ows_FSObjType="74;#0" ows_MetaInfo="74;#" ows_ID="74" ows_owshiddenversion="10" ows_Created="2009-12-29 12:21:01" ows_FileRef="74;#Lists/My List Name/74_.000" ReadOnly="False" VerificationRequired="0"/>
                    </rs:data>
                </listitems></SearchListItemsResult></SearchListItemsResponse></soap:Body></soap:Envelope>

Код вызывается следующим образом:

xsl = Test.Xml.loadXMLDoc("/_layouts/xsl/xsl.xslt");
var doc = Test.Xml.xslTransform(xData.responseXML, xsl);

xData - это xml, возвращаемый веб-службой.

Ответ 1

Если ваш XSLT использует xsl:include, вы можете получить странные необъяснимые ошибки, но всегда с тем же конечным результатом: ваше преобразование не выполняется.

См. отчет об ошибке хрома и, пожалуйста, поддержите его! http://code.google.com/p/chromium/issues/detail?id=8441

Однако ошибка в webkit. Для получения дополнительной информации здесь есть еще одна ссылка, которая более подробно объясняет, почему она не работает.

Единственный способ обойти это - предварительно обработать таблицу стилей так, чтобы она вводила включенные таблицы стилей. Это то, что библиотека XSLT кроссбраузера, такая как Sarissa, сделает для вас автоматически.

Если вы ищете решение jQuery:
http://plugins.jquery.com/project/Transform/ - это кросс-браузерный плагин XSL. Я успешно использовал это, чтобы получить xsl:include в прошлом без особых хлопот. Вам не нужно переписывать свой xsl, этот плагин будет предварительно обрабатывать их для вас. Определенно стоит посмотреть, как он более легкий, чем Сарисса.

UPDATE:

<html>
<head>
<script language="javascript" src="jquery-1.3.2.min.js"></script> 
<script language="javascript" src="jquery.transform.js"></script>  
<script type="text/javascript">
function loadXML(file)
{
    var xmlDoc = null;
    try //Internet Explorer
    {
        xmlDoc=new ActiveXObject("Microsoft.XMLDOM");
        xmlDoc.async=false;
        xmlDoc.load(file);
    }
    catch(e)
    {
        try //Firefox, Mozilla, Opera, etc.
        {
            xmlDoc=document.implementation.createDocument("","",null);
            xmlDoc.async=false;
            xmlDoc.load(file);
        }
        catch(e)
        {
            try //Google Chrome
            {
                var xmlhttp = new window.XMLHttpRequest();
                xmlhttp.open("GET",file,false);
                xmlhttp.send(null);
                xmlDoc = xmlhttp.responseXML.documentElement;
            }
            catch(e)
            {
            error=e.message;
            }
        }
    }
    return xmlDoc;
}
function xslTransform(xmlObject, xslObject)
{
    try 
    {
        $("body").append("<div id='test'></div>");
        var a = $("#test").transform({ xmlobj: xmlObject, xslobj: xslObject });
    }
    catch (exception) 
    {
        if (typeof (exception) == "object" && exception.message) 
            alert(exception.message);
        else alert(exception);
    }
}
var xmlObject = loadXML("input.xml");
var xslObject = loadXML("transform.xsl");
$(document).ready(function()  
{
    xslTransform(xmlObject, xslObject);
});
</script>
</head>
<body>

</body>
</html>

Эта тестовая html-страница работает как в Chrome/FireFox/IE.

input.xml - это простой xml файл, содержащий <root /> transform.xsl - это урезанный xsl, который вы опубликовали.

EDIT

Однако, похоже, что у параметра $.transform есть проблемы с импортом таблиц стилей из включенных файлов:

Вот как это исправить:

Найдите

var safariimportincludefix = function(xObj,rootConfig) {

в jquery.transform.js и замените всю функцию следующим:

var safariimportincludefix = function(xObj,rootConfig) {
    var vals = $.merge($.makeArray(xObj.getElementsByTagName("import")),$.makeArray(xObj.getElementsByTagName("include")));

    for(var x=0;x<vals.length;x++) {
        var node = vals[x];
        $.ajax({
            passData : { node : node, xObj : xObj, rootConfig : rootConfig},
            dataType : "xml",
            async : false,
            url : replaceref(node.getAttribute("href"),rootConfig),
            success : function(xhr) {
                try {
                    var _ = this.passData;
                    xhr = safariimportincludefix(xhr,_.rootConfig);

                    var imports = $.merge(childNodes(xhr.getElementsByTagName("stylesheet")[0],"param"),childNodes(xhr.getElementsByTagName("stylesheet")[0],"template"));
                    var excistingNodes = [];
                    try 
                    {
                        var sheet = _.xObj;
                        var params = childNodes(sheet,"param");
                        var stylesheets = childNodes(sheet,"template");
                        existingNodes = $.merge(params,stylesheets);
                    }
                    catch(exception) 
                    {
                        var x = exception;
                    }
                    var existingNames = [];
                    var existingMatches = [];
                    for(var a=0;a<existingNodes.length;a++) {
                        if(existingNodes[a].getAttribute("name")) {
                            existingNames[existingNodes[a].getAttribute("name")] = true;
                        } else {
                            existingMatches[existingNodes[a].getAttribute("match")] = true;
                        }
                    }

                    var pn = _.node.parentNode;
                    for(var y=0;y<imports.length;y++) {
                        if(!existingNames[imports[y].getAttribute("name")] && !existingMatches[imports[y].getAttribute("match")]) {
                            var clonednode = _.xObj.ownerDocument.importNode(imports[y],true);
                            //pn.insertBefore(clonednode,_.xObj);
                            pn.insertBefore(clonednode,childNodes(_.xObj,"template")[0]);
                        }
                    }
                    pn.removeChild(_.node);
                } catch(ex) { 

                }
            }
        });
    }

    return xObj;
};

Теперь, используя ранее вставленный тест index.html, используйте это для transform.xsl:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    >
        <xsl:include href="include.xsl" />
    <xsl:output method="html"/>
    <xsl:template match="/">
            <xsl:call-template name="giveMeAnIncludedHeader" />
    </xsl:template>
</xsl:stylesheet>

И это для include.xsl

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:template name="giveMeAnIncludedHeader">
        <h1>Test</h1>
    </xsl:template>
</xsl:stylesheet>

С ранее опубликованным исправлением в jquery.transform.js теперь будет вставлен включенный <h1>Test</h1> во все браузеры.

Вы можете увидеть это в действии здесь: http://www.mpdreamz.nl/xsltest

Ответ 2

Это не ответ на исходный вопрос, но во время моего поиска через интернет в поисках примера xslt-преобразования, которое работает на chrome, я нашел ссылки на этот поток много раз. Я искал решение, которое не использует библиотеки с открытым исходным кодом или сторонние библиотеки/плагины и хорошо работает с silverlight.

Проблема с хром и сафари - это ограничение, которое предотвращает загрузку xml файлов напрямую. Предлагаемое решение http://www.mindlence.com/WP/?p=308 - загрузить файл xml любым другим способом и передать его как строку в процессор xslt.

При таком подходе я смог выполнить xsl-преобразования в javascript и передать результат в приложение silverlight через HTML Bridge.