将 XML 替换为来自 XMLHttprequest 的响应
Replace XML with response from a XMLHttprequest
>编辑
随着对我的问题的所有编辑,它变得相当冗长。因此,让我尝试将其缩短一点,使其更容易理解。
我正在使用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+ 行。我尝试了importNode
、replaceChild
、appendChild
、adoptNode
、removeChild
等的所有变体。我能想到。我相当确定我可以写一本关于处理任何给定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 行。
- XMLHttpRequest未返回值-状态202
- XMLHttpRequest在移动设备上的chrome上不起作用
- AJAX简单错误.XMLHttpRequest无法加载http://localhost/mpl/getPage.php.
- 检查xmlhttprequest问题的消息
- 主线程上的同步XMLHttpRequest已弃用
- 如何定期发出xmlhttprequest
- javascript第三个XMLHttpRequest被拒绝.但前两个是允许的
- 上传带有XMLHttprequest的文件-多部分/表单数据中缺少边界
- 对象#<XMLHttpRequest>没有方法'完成'
- 在XMLHttpRequest之后重新初始化jQuery
- 如何将GreaseMonkey中的XMLHttpRequest延迟到目标页面加载完成
- Javascript XMLHttpRequest——只有第一个POST请求有效
- XMLHttpRequest - difference between Chrome & Firefox
- IE上的新XMLHttpRequest()出现JS Ajax未指定错误
- 非常简单的XMLHttpRequest不起作用
- 如何使用XMLHttpRequest下载文件
- XMLHttpRequest在$.ajax工作的地方给出了CORS错误
- XMLHttpRequest.open()AJAX中的参数url
- XMLHTTPRequest脚本中没有internet连接和超时
- 为什么当async标志设置为false时,xmlhttprequest中的代码可以工作,而当它设置为true时却不能工作