forked from mirrors/gecko-dev
		
	 2c4cc1cdc2
			
		
	
	
		2c4cc1cdc2
		
	
	
	
	
		
			
			MozReview-Commit-ID: BQOgqOwtSdm --HG-- extra : rebase_source : 6a2b2654ed38ba763a88d068cab92a9085e15fdb
		
			
				
	
	
		
			368 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			368 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* -*- Mode: indent-tabs-mode: nil; js-indent-level: 2 -*- */
 | |
| /* vim: set sts=2 sw=2 et tw=80: */
 | |
| "use strict";
 | |
| 
 | |
| Cu.import("resource://gre/modules/ExtensionUtils.jsm");
 | |
| 
 | |
| XPCOMUtils.defineLazyModuleGetter(this, "AppConstants",
 | |
|                                   "resource://gre/modules/AppConstants.jsm");
 | |
| XPCOMUtils.defineLazyModuleGetter(this, "CustomizableUI",
 | |
|                                   "resource:///modules/CustomizableUI.jsm");
 | |
| XPCOMUtils.defineLazyModuleGetter(this, "Services",
 | |
|                                   "resource://gre/modules/Services.jsm");
 | |
| XPCOMUtils.defineLazyModuleGetter(this, "Task",
 | |
|                                   "resource://gre/modules/Task.jsm");
 | |
| 
 | |
| let {
 | |
|   ExtensionError,
 | |
|   IconDetails,
 | |
| } = ExtensionUtils;
 | |
| 
 | |
| const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
 | |
| 
 | |
| // WeakMap[Extension -> SidebarAction]
 | |
| let sidebarActionMap = new WeakMap();
 | |
| 
 | |
| const sidebarURL = "chrome://browser/content/webext-panels.xul";
 | |
| 
 | |
| /**
 | |
|  * Responsible for the sidebar_action section of the manifest as well
 | |
|  * as the associated sidebar browser.
 | |
|  */
 | |
| class SidebarAction {
 | |
|   constructor(options, extension) {
 | |
|     this.extension = extension;
 | |
| 
 | |
|     // Add the extension to the sidebar menu.  The sidebar widget will copy
 | |
|     // from that when it is viewed, so we shouldn't need to update that.
 | |
|     let widgetId = makeWidgetId(extension.id);
 | |
|     this.id = `${widgetId}-sidebar-action`;
 | |
|     this.menuId = `menu_${this.id}`;
 | |
| 
 | |
|     this.defaults = {
 | |
|       enabled: true,
 | |
|       title: options.default_title || extension.name,
 | |
|       icon: IconDetails.normalize({path: options.default_icon}, extension),
 | |
|       panel: options.default_panel || "",
 | |
|     };
 | |
| 
 | |
|     this.tabContext = new TabContext(tab => Object.create(this.defaults),
 | |
|                                      extension);
 | |
| 
 | |
|     // We need to ensure our elements are available before session restore.
 | |
|     this.windowOpenListener = (window) => {
 | |
|       this.createMenuItem(window, this.defaults);
 | |
|     };
 | |
|     windowTracker.addOpenListener(this.windowOpenListener);
 | |
|   }
 | |
| 
 | |
|   build() {
 | |
|     this.tabContext.on("tab-select", // eslint-disable-line mozilla/balanced-listeners
 | |
|                        (evt, tab) => { this.updateWindow(tab.ownerGlobal); });
 | |
| 
 | |
|     let install = this.extension.startupReason === "ADDON_INSTALL";
 | |
|     for (let window of windowTracker.browserWindows()) {
 | |
|       this.updateWindow(window);
 | |
|       if (install) {
 | |
|         let {SidebarUI} = window;
 | |
|         SidebarUI.show(this.id);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Bug 1331507: UX review/analysis of sidebar-button injection.
 | |
|     if (AppConstants.RELEASE_OR_BETA) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (install && !Services.prefs.prefHasUserValue("extensions.sidebar-button.shown")) {
 | |
|       Services.prefs.setBoolPref("extensions.sidebar-button.shown", true);
 | |
|       // If the sidebar button has never been moved to the toolbar, move it now
 | |
|       // so the user can see/access the sidebars.
 | |
|       let widget = CustomizableUI.getWidget("sidebar-button");
 | |
|       if (!widget.areaType) {
 | |
|         CustomizableUI.addWidgetToArea("sidebar-button", CustomizableUI.AREA_NAVBAR, 0);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   sidebarUrl(panel) {
 | |
|     if (this.extension.remote) {
 | |
|       return `${sidebarURL}?remote=1&panel=${encodeURIComponent(panel)}`;
 | |
|     }
 | |
|     return `${sidebarURL}?&panel=${encodeURIComponent(panel)}`;
 | |
|   }
 | |
| 
 | |
|   createMenuItem(window, details) {
 | |
|     let {document} = window;
 | |
| 
 | |
|     // Use of the broadcaster allows browser-sidebar.js to properly manage the
 | |
|     // checkmarks in the menus.
 | |
|     let broadcaster = document.createElementNS(XUL_NS, "broadcaster");
 | |
|     broadcaster.setAttribute("id", this.id);
 | |
|     broadcaster.setAttribute("autoCheck", "false");
 | |
|     broadcaster.setAttribute("type", "checkbox");
 | |
|     broadcaster.setAttribute("group", "sidebar");
 | |
|     broadcaster.setAttribute("label", details.title);
 | |
|     broadcaster.setAttribute("sidebarurl", this.sidebarUrl(details.panel));
 | |
|     // oncommand gets attached to menuitem, so we use the observes attribute to
 | |
|     // get the command id we pass to SidebarUI.
 | |
|     broadcaster.setAttribute("oncommand", "SidebarUI.toggle(this.getAttribute('observes'))");
 | |
| 
 | |
|     let menuitem = document.createElementNS(XUL_NS, "menuitem");
 | |
|     menuitem.setAttribute("id", this.menuId);
 | |
|     menuitem.setAttribute("observes", this.id);
 | |
|     menuitem.setAttribute("class", "menuitem-iconic webextension-menuitem");
 | |
| 
 | |
|     this.setMenuIcon(menuitem, details);
 | |
| 
 | |
|     document.getElementById("mainBroadcasterSet").appendChild(broadcaster);
 | |
|     document.getElementById("viewSidebarMenu").appendChild(menuitem);
 | |
| 
 | |
|     return menuitem;
 | |
|   }
 | |
| 
 | |
|   setMenuIcon(menuitem, details) {
 | |
|     let getIcon = size => IconDetails.escapeUrl(
 | |
|       IconDetails.getPreferredIcon(details.icon, this.extension, size).icon);
 | |
| 
 | |
|     menuitem.setAttribute("style", `
 | |
|       --webextension-menuitem-image: url("${getIcon(16)}");
 | |
|       --webextension-menuitem-image-2x: url("${getIcon(32)}");
 | |
|     `);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Update the broadcaster and menuitem `node` with the tab context data
 | |
|    * in `tabData`.
 | |
|    *
 | |
|    * @param {ChromeWindow} window
 | |
|    *        Browser chrome window.
 | |
|    * @param {object} tabData
 | |
|    *        Tab specific sidebar configuration.
 | |
|    */
 | |
|   updateButton(window, tabData) {
 | |
|     let {document, SidebarUI} = window;
 | |
|     let title = tabData.title || this.extension.name;
 | |
|     let menu = document.getElementById(this.menuId);
 | |
|     if (!menu) {
 | |
|       menu = this.createMenuItem(window, tabData);
 | |
|     }
 | |
| 
 | |
|     // Update the broadcaster first, it will update both menus.
 | |
|     let broadcaster = document.getElementById(this.id);
 | |
|     broadcaster.setAttribute("tooltiptext", title);
 | |
|     broadcaster.setAttribute("label", title);
 | |
| 
 | |
|     let url = this.sidebarUrl(tabData.panel);
 | |
|     let urlChanged = url !== broadcaster.getAttribute("sidebarurl");
 | |
|     if (urlChanged) {
 | |
|       broadcaster.setAttribute("sidebarurl", url);
 | |
|     }
 | |
| 
 | |
|     this.setMenuIcon(menu, tabData);
 | |
| 
 | |
|     // Update the sidebar if this extension is the current sidebar.
 | |
|     if (SidebarUI.currentID === this.id) {
 | |
|       SidebarUI.title = title;
 | |
|       if (SidebarUI.isOpen && urlChanged) {
 | |
|         SidebarUI.show(this.id);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Update the broadcaster and menuitem for a given window.
 | |
|    *
 | |
|    * @param {ChromeWindow} window
 | |
|    *        Browser chrome window.
 | |
|    */
 | |
|   updateWindow(window) {
 | |
|     let nativeTab = window.gBrowser.selectedTab;
 | |
|     this.updateButton(window, this.tabContext.get(nativeTab));
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Update the broadcaster and menuitem when the extension changes the icon,
 | |
|    * title, url, etc. If it only changes a parameter for a single
 | |
|    * tab, `tab` will be that tab. Otherwise it will be null.
 | |
|    *
 | |
|    * @param {XULElement|null} nativeTab
 | |
|    *        Browser tab, may be null.
 | |
|    */
 | |
|   updateOnChange(nativeTab) {
 | |
|     if (nativeTab) {
 | |
|       if (nativeTab.selected) {
 | |
|         this.updateWindow(nativeTab.ownerGlobal);
 | |
|       }
 | |
|     } else {
 | |
|       for (let window of windowTracker.browserWindows()) {
 | |
|         this.updateWindow(window);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Set a default or tab specific property.
 | |
|    *
 | |
|    * @param {XULElement|null} nativeTab
 | |
|    *        Webextension tab object, may be null.
 | |
|    * @param {string} prop
 | |
|    *        String property to retrieve ["icon", "title", or "panel"].
 | |
|    * @param {string} value
 | |
|    *        Value for property.
 | |
|    */
 | |
|   setProperty(nativeTab, prop, value) {
 | |
|     if (nativeTab === null) {
 | |
|       this.defaults[prop] = value;
 | |
|     } else if (value !== null) {
 | |
|       this.tabContext.get(nativeTab)[prop] = value;
 | |
|     } else {
 | |
|       delete this.tabContext.get(nativeTab)[prop];
 | |
|     }
 | |
| 
 | |
|     this.updateOnChange(nativeTab);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Retrieve a property from the tab or defaults if tab is null.
 | |
|    *
 | |
|    * @param {XULElement|null} nativeTab
 | |
|    *        Browser tab object, may be null.
 | |
|    * @param {string} prop
 | |
|    *        String property to retrieve ["icon", "title", or "panel"]
 | |
|    * @returns {string} value
 | |
|    *          Value for prop.
 | |
|    */
 | |
|   getProperty(nativeTab, prop) {
 | |
|     if (nativeTab === null) {
 | |
|       return this.defaults[prop];
 | |
|     }
 | |
|     return this.tabContext.get(nativeTab)[prop];
 | |
|   }
 | |
| 
 | |
|   shutdown() {
 | |
|     this.tabContext.shutdown();
 | |
|     for (let window of windowTracker.browserWindows()) {
 | |
|       let {document, SidebarUI} = window;
 | |
|       if (SidebarUI.currentID === this.id) {
 | |
|         SidebarUI.hide();
 | |
|       }
 | |
|       let menu = document.getElementById(this.menuId);
 | |
|       if (menu) {
 | |
|         menu.remove();
 | |
|       }
 | |
|       let broadcaster = document.getElementById(this.id);
 | |
|       if (broadcaster) {
 | |
|         broadcaster.remove();
 | |
|       }
 | |
|     }
 | |
|     windowTracker.removeOpenListener(this.windowOpenListener);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Triggers this sidebar action for the given window, with the same effects as
 | |
|    * if it were toggled via menu or toolbarbutton by a user.
 | |
|    *
 | |
|    * @param {ChromeWindow} window
 | |
|    */
 | |
|   triggerAction(window) {
 | |
|     let {SidebarUI} = window;
 | |
|     if (SidebarUI) {
 | |
|       SidebarUI.toggle(this.id);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| SidebarAction.for = (extension) => {
 | |
|   return sidebarActionMap.get(extension);
 | |
| };
 | |
| 
 | |
| global.sidebarActionFor = SidebarAction.for;
 | |
| 
 | |
| /* eslint-disable mozilla/balanced-listeners */
 | |
| extensions.on("manifest_sidebar_action", (type, directive, extension, manifest) => {
 | |
|   let sidebarAction = new SidebarAction(manifest.sidebar_action, extension);
 | |
|   sidebarActionMap.set(extension, sidebarAction);
 | |
| });
 | |
| 
 | |
| extensions.on("ready", (type, extension) => {
 | |
|   // We build sidebars during ready to ensure the background scripts are ready.
 | |
|   if (sidebarActionMap.has(extension)) {
 | |
|     sidebarActionMap.get(extension).build();
 | |
|   }
 | |
| });
 | |
| 
 | |
| extensions.on("shutdown", (type, extension) => {
 | |
|   if (sidebarActionMap.has(extension)) {
 | |
|     // Don't remove everything on app shutdown so session restore can handle
 | |
|     // restoring open sidebars.
 | |
|     if (extension.shutdownReason !== "APP_SHUTDOWN") {
 | |
|       sidebarActionMap.get(extension).shutdown();
 | |
|     }
 | |
|     sidebarActionMap.delete(extension);
 | |
|   }
 | |
| });
 | |
| /* eslint-enable mozilla/balanced-listeners */
 | |
| 
 | |
| extensions.registerSchemaAPI("sidebarAction", "addon_parent", context => {
 | |
|   let {extension} = context;
 | |
| 
 | |
|   function getTab(tabId) {
 | |
|     if (tabId !== null) {
 | |
|       return tabTracker.getTab(tabId);
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     sidebarAction: {
 | |
|       async setTitle(details) {
 | |
|         let nativeTab = getTab(details.tabId);
 | |
| 
 | |
|         let title = details.title;
 | |
|         // Clear the tab-specific title when given a null string.
 | |
|         if (nativeTab && title === "") {
 | |
|           title = null;
 | |
|         }
 | |
|         SidebarAction.for(extension).setProperty(nativeTab, "title", title);
 | |
|       },
 | |
| 
 | |
|       getTitle(details) {
 | |
|         let nativeTab = getTab(details.tabId);
 | |
| 
 | |
|         let title = SidebarAction.for(extension).getProperty(nativeTab, "title");
 | |
|         return Promise.resolve(title);
 | |
|       },
 | |
| 
 | |
|       async setIcon(details) {
 | |
|         let nativeTab = getTab(details.tabId);
 | |
| 
 | |
|         let icon = IconDetails.normalize(details, extension, context);
 | |
|         SidebarAction.for(extension).setProperty(nativeTab, "icon", icon);
 | |
|       },
 | |
| 
 | |
|       async setPanel(details) {
 | |
|         let nativeTab = getTab(details.tabId);
 | |
| 
 | |
|         let url;
 | |
|         // Clear the tab-specific url when given a null string.
 | |
|         if (nativeTab && details.panel === "") {
 | |
|           url = null;
 | |
|         } else if (details.panel !== "") {
 | |
|           url = context.uri.resolve(details.panel);
 | |
|         } else {
 | |
|           throw new ExtensionError("Invalid url for sidebar panel.");
 | |
|         }
 | |
| 
 | |
|         SidebarAction.for(extension).setProperty(nativeTab, "panel", url);
 | |
|       },
 | |
| 
 | |
|       getPanel(details) {
 | |
|         let nativeTab = getTab(details.tabId);
 | |
| 
 | |
|         let panel = SidebarAction.for(extension).getProperty(nativeTab, "panel");
 | |
|         return Promise.resolve(panel);
 | |
|       },
 | |
|     },
 | |
|   };
 | |
| });
 |