Chrome и Safari XSLT с использованием JavaScript - PullRequest
11 голосов
/ 11 января 2010

У меня есть следующий код, который применяет стиль 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);

Строка выше возвращает ноль. Ошибка не выдается.

Обновление

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

Обновление

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

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

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 имеет в нем включает. Я хочу, чтобы в идеале, клиент проходил на клиенте.

Обновлено Руперт, не могли бы вы обновить вопрос с помощью 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, возвращаемый веб-службой.

Ответы [ 2 ]

14 голосов
/ 11 января 2010

Если ваш 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, который вы опубликовали.

РЕДАКТИРОВАТЬ

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

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

Найти

var safariimportincludefix = function(xObj,rootConfig) {

in 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

5 голосов
/ 23 ноября 2010

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

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

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

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...