如何在Firefox中使用XUL创建底端面板?

问题描述 投票:1回答:2

对于我的插件,我想创建一个显示信息的底端“面板”。 Firefox的Web控制台面板就是此类示例,可以在Web Developer Tools下进行切换。

我试图深入研究代码,但无法弄清楚如何实现。有人可以给我一个基本的解释,说明如何使用XUL创建它,或者为我指明正确的方向吗?

javascript firefox-addon xul
2个回答
4
投票

Web控制台不是sidebar。在Fireox中,只有一个侧边栏,它可以位于浏览器内容的左侧或右侧。即使更改选项卡,侧边栏也是UI的恒定部分。它通常用于内容,历史记录,书签或其他此类信息,这些信息不会随所查看的选项卡而改变。

为了调查类似的东西,如果您尚未安装DOM Inspector,我建议您使用它。

Web控制台包含在<iframe>所在的选项卡下的<tabbrowser>中。

iframe的XUL为:<iframe xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul" class="devtools-toolbox-bottom-iframe" height="436" tooltip="aHTMLTooltip" src="chrome://browser/content/devtools/framework/toolbox.xul"/>

首先在其中包含选项卡的<splitter>之后插入<iframe>

通过调用<hbox class="browserSidebarContainer">打开Web控制台

[gDevToolsBrowser.selectToolCommand(gBrowser, "webconsole");是从gDevToolsBrowser的内容在chrome://browser/content/browser.js中定义的

它们实际上是在resource:///modules/devtools/gDevTools.jsm中的函数SH_create中创建的>]

所有这些resource:///modules/devtools/framework/toolbox-hosts.jschrome:// URL都应在Firefox中运行。 Firefox安装中有大量文件打包在三个文件目录中[。这些文件位于 /omni.ja /browser/omni.ja /webart/omni.ja中。 omni.ja文件只是扩展名为zip的文件。为了方便地访问这些文件,我通常将它们解压缩到目录中(在Firefox安装目录树的外部)。我发现当我想弄清楚如何完成操作时,这使搜索和操作文件变得更加容易。如果您只是在寻找将创建用于Web控制台的框的代码,那么复杂性取决于您所操作的上下文。以下内容几乎可以在任何地方工作:

resource:///

注意:

toolbox-hosts.js

中的代码使用方法/** * Creates an <iframe> based panel within the current tab, * or opens a window, for use as an user interface box. * If it is not a window, it is associated with the current * browser tab. * @param location * Placement of the panel [right|left|top|bottom|window] * The default location is "right". * @param size * Width if on left or right. Height if top or bottom. * Both width and height if location="window" unless * features is a string. * Default is 400. * @param id * The ID to assign to the iframe. Default is * "makyen-interface-panel" * The <splitter> will be assigned the * ID = id + "-splitter" * @param chromeUrl * This is the chrome:// URL to use for the contents * of the iframe or the window. * the default is: * "chrome://browser/content/devtools/framework/toolbox.xul" * @param features * The features string for the window. See: * https://developer.mozilla.org/en-US/docs/Web/API/Window.open * returns [splitterEl, iframeEl] * The elements for the <splitter> and <iframe> * * Copyright 2014 by Makyen. * Released under the MPL 2.0. http://mozilla.org/MPL/2.0/. **/ function createInterfacePanelIframe(location,size,id,chromeUrl,features) { //defaults size = ( (typeof size !== "number") || size<1) ? 400 : size; id = typeof id !== "string" ? "makyen-interface-panel" : id; chromeUrl = typeof chromeUrl !== "string" ? "chrome://browser/content/devtools/framework/toolbox.xul" : chromeUrl; //Create some common variables if they do not exist. // This should work from any Firefox context. // Depending on the context in which the function is being run, // this could be simplified. if (typeof window === "undefined") { //If there is no window defined, get the most recent. var window=Components.classes["@mozilla.org/appshell/window-mediator;1"] .getService(Components.interfaces.nsIWindowMediator) .getMostRecentWindow("navigator:browser"); } if (typeof document === "undefined") { //If there is no document defined, get it var document = window.content.document; } if (typeof gBrowser === "undefined") { //If there is no gBrowser defined, get it var gBrowser = window.gBrowser; } //Get the current tab & notification box (container for tab UI). let tab = gBrowser.selectedTab; let browserForTab = gBrowser.getBrowserForTab( tab ); let notificationBox = gBrowser.getNotificationBox( browserForTab ); let ownerDocument = gBrowser.ownerDocument; //Create the <iframe> use //ownerDocument for the XUL namespace. let iframeEl = ownerDocument.createElement("iframe"); iframeEl.id = id; iframeEl.setAttribute("src",chromeUrl); iframeEl.setAttribute("height", size.toString()); iframeEl.setAttribute("width", size.toString()); //Call createInterfacePanel, pass the size if it is to be a window. let splitterEl; if(location == "window" ) { splitterEl = createInterfacePanel(location, size, size ,id + "-splitter", chromeUrl, features); return [splitterEl, null]; } else { splitterEl = createInterfacePanel(location, iframeEl, iframeEl ,id + "-splitter", chromeUrl, features); } return [splitterEl, iframeEl]; } /** * Creates a panel within the current tab, or opens a window, for use as a * user interface box. If not a window, it is associated with the current * browser tab. * @param location * Placement of the panel [right|left|top|bottom|window] * The default location is "right". * @param objectEl * The element of an XUL object that will be inserted into * the DOM such that it is within the current tab. * Some examples of possible objects are <iframe>, * <browser>, <box>, <hbox>, <vbox>, etc. * If the location="window" and features is not a string * and this is a number then it is used as the width of the * window. * If features is a string, it is assumed the width is * set in that, or elsewhere (e.g. in the XUL). * @param sizeEl * The element that contains attributes of "width" and * "height". If location=="left"|"right" then the * "height" attribute is removed prior to the objectEl * being inserted into the DOM. * A spearate reference for the size element in case the * objectEl is a documentFragment containing multiple elements. * However, normal usage is for objectEl === sizeEl when * location != "window". * When location == "window" and features is not a string, * and sizeEl is a number then it is used as the height * of the window. * If features is a string, it is assumed the height is * set in that, or elsewhere (e.g. in the XUL). * @param id * The ID to assign to the <splitter>. The default is: * "makyen-interface-panel-splitter". * @param chromeUrl * This is the chrome:// URL to use for the contents * of the window. * the default is: * "chrome://browser/content/devtools/framework/toolbox.xul" * @param features * The features string for the window. See: * https://developer.mozilla.org/en-US/docs/Web/API/Window.open * returns * if location != "window": * splitterEl, The element for the <splitter>. * if location == "window": * The windowObjectReference returned by window.open(). * * Copyright 2014 by Makyen. * Released under the MPL 2.0. http://mozilla.org/MPL/2.0/. **/ function createInterfacePanel(location,objectEl,sizeEl,id,chromeUrl,features) { //Set location default: location = typeof location !== "string" ? "right" : location; if(location == "window") { if(typeof features !== "string") { let width = ""; let height = ""; if(typeof objectEl == "number") { width = "width=" + objectEl.toString() + ","; } if(typeof sizeEl == "number") { height = "height=" + sizeEl.toString() + ","; } features = width + height + "menubar=no,toolbar=no,location=no,personalbar=no" + ",status=no,chrome=yes,resizable,centerscreen"; } } id = typeof id !== "string" ? "makyen-interface-panel-splitter" : id; chromeUrl = typeof chromeUrl !== "string" ? "chrome://browser/content/devtools/framework/toolbox.xul" : chromeUrl; //Create some common variables if they do not exist. // This should work from any Firefox context. // Depending on the context in which the function is being run, // this could be simplified. if (typeof window === "undefined") { //If there is no window defined, get the most recent. var window=Components.classes["@mozilla.org/appshell/window-mediator;1"] .getService(Components.interfaces.nsIWindowMediator) .getMostRecentWindow("navigator:browser"); } if (typeof document === "undefined") { //If there is no document defined, get it var document = window.content.document; } if (typeof gBrowser === "undefined") { //If there is no gBrowser defined, get it var gBrowser = window.gBrowser; } //Get the current tab & notification box (container for tab UI). let tab = gBrowser.selectedTab; let browserForTab = gBrowser.getBrowserForTab( tab ); let notificationBox = gBrowser.getNotificationBox( browserForTab ); let ownerDocument = gBrowser.ownerDocument; //Create a Document Fragment. //If doing multiple DOM additions, we should be in the habit // of doing things in a way which causes the least number of reflows. // We know that we are going to add more than one thing, so use a // document fragment. let docFrag = ownerDocument.createDocumentFragment(); //ownerDocument must be used here in order to have the XUL namespace // or the splitter causes problems. // createElementNS() does not work here. let splitterEl = ownerDocument.createElement("splitter"); splitterEl.id = id ; //Look for the child element with class="browserSidebarContainer". //It is the element in procimity to which the <splitter> //and objectEl will be placed. let theChild = notificationBox.firstChild; while (!theChild.hasAttribute("class") || !theChild.getAttribute("class").contains("browserSidebarContainer") ) { theChild = theChild.nextSibling; if(!theChild) { //We failed to find the correct node. //This implies that the structure Firefox // uses has changed significantly and it should // be assumed that the extension is no longer compatible. return null; } } let toReturn = null; switch(location) { case "window" : return window.open(chromeUrl,"_blank",features); break; case "top" : if(sizeEl) { sizeEl.removeAttribute("width"); } docFrag.appendChild(objectEl); docFrag.appendChild(splitterEl); //Inserting the document fragment results in the same // DOM structure as if you Inserted each child of the // fragment separately. (i.e. the document fragment // is just a temporary container). //Insert the interface prior to theChild. toReturn = notificationBox.insertBefore(docFrag,theChild); break; case "bottom" : if(sizeEl) { sizeEl.removeAttribute("width"); } docFrag.appendChild(splitterEl); docFrag.appendChild(objectEl); //Insert the interface just after theChild. toReturn = notificationBox.insertBefore(docFrag,theChild.nextSibling); break; case "left" : if(sizeEl) { sizeEl.removeAttribute("height"); } docFrag.appendChild(objectEl); //Splitter is second in this orientaiton. docFrag.appendChild(splitterEl); //Insert the interface as the first child of theChild. toReturn = theChild.insertBefore(docFrag,theChild.firstChild); break; case "right" : default : //Right orientaiton, the default. if(sizeEl) { sizeEl.removeAttribute("height"); } docFrag.appendChild(splitterEl); docFrag.appendChild(objectEl); //Insert the interface as the last child of theChild. toReturn = theChild.appendChild(docFrag); break; } return splitterEl; } 查找要附加到的容器。该方法没有文档,因此我使用了getSidebarContainer()

2
投票
它称为侧边栏。我在这里右侧制作了一个:getNotificationBox()
© www.soinside.com 2019 - 2024. All rights reserved.