可以将文章内容翻译成中文,广告屏蔽插件可能会导致该功能失效(如失效,请关闭广告屏蔽插件后再试):
问题:
The add-on SDK attaches panels to widgets as seen here. I would like to achieve the same effect using the add-on SDK with a toolbar button instead.
The toolbar button I'm using is of the type menu-button, which means that the left side is an icon and has an oncommand
listener. The right side is a drop-down arrow which shows its contents on click. Here's the code to create such a button with the add-on SDK:
const doc = require('sdk/window/utils').getMostRecentBrowserWindow().document; var navBar = doc.getElementById('nav-bar') var btn = doc.createElement('toolbarbutton'); btn.setAttribute('id', 'hylytit'); btn.setAttribute('type', 'menu-button'); btn.setAttribute('class', 'toolbarbutton-1'); btn.setAttribute('image', data.url('resources/hylyt_off.png')); btn.setAttribute('orient', 'horizontal'); btn.setAttribute('label', 'Hylyt.it'); btn.addEventListener('command', function(event) { if (event.button===0) btnClick(); console.log(TAG+'button clicked'); }, false); var panel = doc.createElement('panel'); panel.setAttribute('id', 'search-panel'); panel.addEventListener('command', function(event) { console.log(TAG+'dropdown clicked'); }, false); var label = doc.createElement('label'); label.setAttribute('control', 'name'); label.setAttribute('value', 'Article List'); var textbox = doc.createElement('textbox'); textbox.setAttribute('id', 'name'); panel.appendChild(label); panel.appendChild(textbox); btn.appendChild(panel); navBar.appendChild(btn);
The panel
above is not an add-on SDK panel, it's a XUL panel and is severely limited in that it can't be styled with CSS. On top of this, the panel's onCommand
listener never fires despite the fact that the btn
's onCommand
fires as expected. The XUL panel shows itself when I click the dropdown button (as long as it has children), but because I can't access its click handler, I can't just create an add-on SDK panel on click.
So my question is this. Is there a way to access the toolbar button's menu portion's click handler or is there a way to append an add-on SDK panel as a child of a toolbar button?
回答1:
You have 900+ rep, you should know better. It is common knowledge to create another question topic rather then ask how to do something different in a comment especially after solution acceptance.
Nonetheless, this is what you do to accomplish the Pocket toolbarbutton effect. Based on code supplied by contributor above. Ask another question and I'll move this there and you can accept my solution there.
var doc = document; var navBar = doc.getElementById('nav-bar') var btn = doc.createElement('toolbarbutton'); btn.setAttribute('id', 'hylytit'); btn.setAttribute('type', 'menu-button'); btn.setAttribute('class', 'toolbarbutton-1'); btn.setAttribute('image', ''); btn.setAttribute('orient', 'horizontal'); btn.setAttribute('label', 'Hylyt.it'); //// var toolbarbuttonPanel = doc.createElement('panel'); toolbarbuttonPanel.setAttribute('id', 'toolbarbutton-panel'); toolbarbuttonPanel.setAttribute('type', 'arrow'); var toolbarbuttonLabel = doc.createElement('label'); toolbarbuttonLabel.setAttribute('value', 'toolbarbutton panel'); toolbarbuttonPanel.appendChild(toolbarbuttonLabel); //// //// var dropmarkerPanel = doc.createElement('panel'); dropmarkerPanel.setAttribute('id', 'dropmarker-panel'); dropmarkerPanel.setAttribute('type', 'arrow'); var dropmarkerLabel = doc.createElement('label'); dropmarkerLabel.setAttribute('value', 'dropmarker panel'); dropmarkerPanel.appendChild(dropmarkerLabel); //// navBar.appendChild(btn); var mainPopupSet = document.querySelector('#mainPopupSet'); mainPopupSet.appendChild(dropmarkerPanel); mainPopupSet.appendChild(toolbarbuttonPanel); btn.addEventListener('click',function(event) { console.log('event.originalTarget',event.originalTarget); if (event.originalTarget.nodeName == 'toolbarbutton') { dropmarkerPanel.openPopup(btn); } else if (event.originalTarget.nodeName == 'xul:toolbarbutton') { toolbarbuttonPanel.openPopup(btn); } }, false);
回答2:
Panels don't have an onCommand method see MDN - Panels Article
You can make your panel stylized, give it type arrow like panel.setAttribute('type', 'arrow')
and then to attach to your button. I didn't give it type arrow below.
Heres the working code. Copy paste to scratchpad and set Environment > Browser then run it.
var doc = document; //to put this back in sdk do const doc = require('sdk/window/utils').getMostRecentBrowserWindow().document; var navBar = doc.getElementById('nav-bar') var btn = doc.createElement('toolbarbutton'); btn.setAttribute('id', 'hylytit'); btn.setAttribute('type', 'menu-button'); btn.setAttribute('class', 'toolbarbutton-1'); btn.setAttribute('image', ''); //i made this image blank because i dont have the image and im running from scratchpad btn.setAttribute('orient', 'horizontal'); btn.setAttribute('label', 'Hylyt.it'); var panel = doc.createElement('panel'); btn.addEventListener('command', function(event) { //moved this below var panel = doc.createElement because panel needs to be crated before we write this function //if (event.button===0) btnClick(); //console.log(TAG+'button clicked'); //what is TAG? its undefeined for me panel.openPopup(btn); }, false); panel.setAttribute('id', 'search-panel'); /* panel.addEventListener('command', function(event) { console.log(TAG+'dropdown clicked'); //what is TAG? its undefeined for me }, false); */ var label = doc.createElement('label'); label.setAttribute('control', 'name'); label.setAttribute('value', 'Article List'); var textbox = doc.createElement('textbox'); textbox.setAttribute('id', 'name'); panel.appendChild(label); panel.appendChild(textbox); btn.appendChild(panel); navBar.appendChild(btn);
回答3:
You can create a Toolbarbutton and Panel using the Addon SDK and some Jetpack modules. Try toolbarwidget-jplib and browser-action-jplib by Rob--W.
You can easy add a button to the toolbar and style the panel whatever you want with css / html:
var badge = require('browserAction').BrowserAction({ default_icon: 'images/icon19.png', // optional default_title: 'Badge title', // optional; shown in tooltip default_popup: 'popup.html' // optional });