将 XML 替换为来自 XMLHttprequest 的响应

Replace XML with response from a XMLHttprequest

本文关键字:XMLHttprequest 响应 XML 替换      更新时间:2023-09-26

>编辑

随着对我的问题的所有编辑,它变得相当冗长。因此,让我尝试将其缩短一点,使其更容易理解。

我正在使用XULRunner构建一个XUL应用程序。我让它加载了一个虚拟的XUL页面,然后我希望使用XMLHttprequest从我的服务器(本地ampps服务器)加载所有内容,使用PHP来完成所有实际工作。我的PHP正在设置XML Content-Type标头,并将所有输出数据格式化为XML。

以下是处理XMLHttprequest和响应的 JavaScript 函数目前的样子。

function RequestData()
{
if (window.XMLHttpRequest)
{
    var url = 'newmenu.xml';
    var request = new XMLHttpRequest();
}
else
{
    var url = 'http://localdomain.prog/';
    var request = Components.classes['@mozilla.org/xmlextras/xmlhttprequest;1'].createInstance(Components.interfaces.nsIXMLHttpRequest);
}
request.onload = function(aEvent)
{
    var xmlDoc = aEvent.target.responseXML;
    var oldmenu = document.getElementById('menubarwrapper');
    oldmenu.parentNode.replaceChild(xmlDoc.documentElement, oldmenu);
};
request.onerror = function(aEvent)
{
    window.alert("Error Status: " + aEvent.target.status);
};
request.open('POST', url, true);
request.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
request.send('pageid=menu');
}

RequestData()与窗口onload事件一起调用。

我的

原始代码看起来什么都不做,但随着我的研究和测试,我最终让XULRunner在错误控制台中发布了一些错误。最终,它导致了我现在假设的工作版本,但我只是不知道。

错误控制台正在发布此消息(现在仍然如此)

警告:窗口元素的XUL框包含一个内联工具箱子项,强制其所有子项包装在一个块中。

为了确定我的代码是否有效,我必须将其放入Firefox。因此,if (window.XMLHttpRequest)的原因,因为它允许我同时使用Firefox和XULRunner进行测试。然后,我采用了我的PHP生成的XML并制作了一个本地文件,因为Firefox不允许XMLHttprequest加载远程文件(即使它在技术上是本地的)。

上面的代码确实导入了 XML 并替换了<menubar id="menubarwrapper">...</menubar>。但是,在Firefox和XULRunner中,菜单都消失了。如果使用 Firebug,我可以看到所有元素,但是为什么它们不再可见超出了我的范围,并且 Firebug 控制台中没有错误。这就是我目前被难倒的地方。

万一有任何用途,下面是我加载的虚拟XUL文件的副本。

<?xml version="1.0"?>
<?xml-stylesheet href="main.css" type="text/css"?>
<window id="main" title="My App" width="1000" height="800" sizemode="maximized" orient="vertical" persist="screenX screenY width height" xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">
<script type="application/javascript" src="main.js"/>
<toolbox id="toolboxwrapper">
    <menubar id="menubarwrapper">
        <menu id="file-menu" label="File" accesskey="F">
            <menupopup id="file-popup">
                <menuitem label="Refresh" funcname="RefreshWin"/>
                <menuitem label="Open Window" funcname="OpenWin" acceltext="Ctrl+O" accesskey="O" />
                <menuseparator/>
                <menuitem label="Exit" funcname="ExitProg"/>
            </menupopup>
        </menu>
        <menu id="edit-menu" label="Edit" accesskey="E">
            <menupopup id="edit-popup">
                <menuitem label="Undo"/>
                <menuitem label="Redo"/>
            </menupopup>
        </menu>
    </menubar>
</toolbox>
</window>

我的PHP生成的XML相当长,但基本上它是<menubar>元素及其所有子元素,与上面类似,但具有更多的<menu><menuitem>元素。

如果没有更多信息,这只是对问题所在的一种猜测。

可能发生的情况是从您的XMLHttpRequest()检索到的XML数据被解释为XML/HTML而不是XUL。 这可能会导致您描述的行为。您可以通过使用 DOM 检查器查看随行插入的元素来检查这一点:

oldmenu.parentNode.replaceChild(xmlDoc.documentElement, oldmenu);

您应该寻找的是Namespace URI为这些元素显示的内容。如果不http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul则元素不会被视为XUL。

解决此问题的一种方法是:

oldmenu.insertAdjacentHTML("afterend", aEvent.target.responseText);
oldmenu.setAttribute("hidden","true");
//Alternately (if you don't want to keep the placeholder <menubar>):
//oldmenu.parentNode.removeChild(oldmenu);

我让它工作了,但代码比它应该需要的长 80+ 行。我尝试了importNodereplaceChildappendChildadoptNoderemoveChild等的所有变体。我能想到。我相当确定我可以写一本关于处理任何给定XML元素或节点的方法的书。

这是一个例子,说明这个问题是多么荒谬。这将起作用

function AddNewElem()
{
    var menubar = document.getElementById('menubarwrapper');
    var menuitem = '<menu id="help-menu" label="Help" accesskey="H" xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">';
    menuitem += '<menupopup id="help-popup">';
    menuitem += '<menuitem id="test-menu" label="Test" acceltext="Ctrl+T" accesskey="T" />';
    menuitem += '<menuseparator/>';
    menuitem += '<menuitem id="about-menu" label="About" acceltext="Ctrl+A" accesskey="A" />';
    menuitem += '</menupopup>';
    menuitem += '</menu>';
    var parser = new DOMParser();
    var xmlDoc = parser.parseFromString(menuitem, 'text/xml').documentElement;
    menubar.appendChild(xmlDoc);
}

但这不会

function RequestData()
{
    var url = 'newmenu.txt';
    var request = new XMLHttpRequest();
    request.onload = function(aEvent)
    {
        var menubar = document.getElementById('menubarwrapper');
        var responsetxt = aEvent.target.responseText;
        var parser = new DOMParser();
        var xmlDoc = parser.parseFromString(responsetxt, 'text/xml').documentElement;
        menubar.appendChild(xmlDoc);
    };
    request.onerror = function(aEvent)
    {
        alert("Error Status: " + aEvent.target.status);
    };
    request.open('POST', url, true);
    request.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
    request.send('pageid=menu');
}

newmenu.txt只是上一段代码中menuitem的内容。我什至将xmlDoc序列化回字符串,然后再次解析它,但仍然不起作用。我尝试了很多愚蠢的想法,我知道很多想法都行不通,万一我可能会得到关于问题的线索。当你崩溃Firefox时,你知道你太用力了。

但我离题了。因此,为了其他人尝试使用XULrunner,下面是我必须做的。

function RequestData()
{
    var url = 'http://localdomain.prog/';
    var request = Components.classes['@mozilla.org/xmlextras/xmlhttprequest;1'].createInstance(Components.interfaces.nsIXMLHttpRequest);
    request.onload = function(aEvent)
    {
        var xmlDoc = aEvent.target.responseXML;
        var toolbar = document.getElementById('toolboxwrapper');
        var menubar = document.getElementById('menubarwrapper');
        toolbar.removeChild(menubar);
        var menubar = document.createElement('menubar');
        menubar.setAttribute('id', 'menubarwrapper');
        var docfrag = document.createDocumentFragment();
        docfrag.appendChild(menubar);
        var newmenus = xmlDoc.getElementsByTagName('menu');
        var menuslen = newmenus.length;
        for (var i = 0; i < menuslen; i++)
        {
            docfrag = CreateMenu(newmenus[i], docfrag);
        }
        toolbar.appendChild(docfrag);
    };
    request.onerror = function(aEvent)
    {
        window.alert("Error Status: " + aEvent.target.status);
    };
    request.open('POST', url, true);
    request.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
    request.send('pageid=menu');
}
function CreateMenu(obj, docfrag)
{
    var fragbar = docfrag.getElementById('menubarwrapper');
    var menu = document.createElement('menu');
    menu = SetMenuAttr(obj, 'id', menu);
    menu = SetMenuAttr(obj, 'label', menu);
    menu = SetMenuAttr(obj, 'accesskey', menu);
    var fragmenu = fragbar.appendChild(menu);
    var popup = obj.getElementsByTagName('menupopup')[0];
    var menupopup = document.createElement('menupopup');
    menupopup = SetMenuAttr(popup, 'id', menupopup);
    var fragpopup = fragmenu.appendChild(menupopup);
    if (popup.hasChildNodes())
    {
        var childmenu = popup.childNodes;
        var menulen = childmenu.length;
        for (var i = 0; i < menulen; i++)
        {
            if (childmenu[i].nodeName == 'menuitem' || childmenu[i].nodeName == 'menu')
            {
                if (childmenu[i].nodeName == 'menuitem')
                {
                    var menuitem = CreateMenuitem(childmenu[i]);
                }
                if (childmenu[i].nodeName == 'menu')
                {
                    var menuitem = CreateMenu(childmenu[i]);
                }
                fragpopup.appendChild(menuitem);
            }
        }
    }
    return docfrag;
}
function CreateMenuitem(obj)
{
    var menuitem = document.createElement('menuitem');
    menuitem = SetMenuAttr(obj, 'id', menuitem);
    menuitem = SetMenuAttr(obj, 'label', menuitem);
    menuitem = SetMenuAttr(obj, 'accesskey', menuitem);
    menuitem = SetMenuAttr(obj, 'acceltext', menuitem);
    menuitem = SetMenuAttr(obj, 'disabled', menuitem);
    return menuitem;
}
function SetMenuAttr(obj, attr, newobj)
{
    if (obj.hasAttribute(attr))
    {
        newobj.setAttribute(attr, obj.getAttribute(attr));
    }
    return newobj;
}

基本上必须检索每个元素及其属性并创建新元素。到目前为止,它不是首选解决方案。对于我自己来说,如果我必须多写一行代码,而不是绝对必要的,那是错误的。解决方案应该是 2 行。