forked from aminomancer/uc.css.js
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdebugExtensionInToolbarContextMenu.uc.js
396 lines (390 loc) · 15.7 KB
/
debugExtensionInToolbarContextMenu.uc.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
// ==UserScript==
// @name Debug Extension in Toolbar Context Menu
// @version 1.4.5
// @author aminomancer
// @homepageURL https://github.com/aminomancer/uc.css.js
// @long-description
// @description
/*
Adds a new context menu when right-clicking an add-on's button in the toolbar or urlbar, any time the "Manage Extension" and "Remove Extension" items are available. The new "Debug Extension" menu contains several items: "Extension Manifest" opens the extension's manifest directly in a new tab. Aside from reading the manifest, from there you can also view the whole contents of the extension within Firefox by removing `/manifest.json` from the URL.
In the "View Documents" submenu there are several options for viewing, debugging and modding an addon's main HTML contents.
* "Browser Action" opens the extension's toolbar button popup URL (if it has one) in a regular browser window. The popup URL is whatever document it displays in its panel view, the popup that opens when you click the addon's toolbar button. This is the one you're most likely to want to modify with CSS.
* "Page Action" opens the extension's page action popup URL in the same manner. A page action is an icon on the right side of the urlbar whose behavior is specific to the page in the active tab.
* "Sidebar Action" opens the extension's sidebar document, so this would let you debug Tree Style Tab for example.
* "Extension Options" opens the document that the extension uses for configuration, also in a regular browser window. This could be the page that displays in its submenu on about:addons, or a separate page.
* "Inspect Extension" opens a devtools tab targeting the extension background. This is the same page you'd get if you opened about:debugging and clicked the "Inspect" button next to an extension.
* "View Source" opens the addon's .xpi archive.
* As you'd expect, "Copy ID" copies the extension's ID to your clipboard.
* "Copy URL" copies the extension's base URL, so it can be used in CSS rules like `@-moz-document`.
The menu items' labels are not localized automatically since Firefox doesn't include any similar strings. If you need to change the language or anything, modify the strings in the script under `config`. As usual, icons for the new menu are included in [uc-context-menu-icons.css][]
[uc-context-menu-icons.css]: https://github.com/aminomancer/uc.css.js/blob/master/uc-context-menu-icons.css
*/
// @downloadURL https://cdn.jsdelivr.net/gh/aminomancer/uc.css.js@master/JS/debugExtensionInToolbarContextMenu.uc.js
// @updateURL https://cdn.jsdelivr.net/gh/aminomancer/uc.css.js@master/JS/debugExtensionInToolbarContextMenu.uc.js
// @license This Source Code Form is subject to the terms of the Creative Commons Attribution-NonCommercial-ShareAlike International License, v. 4.0. If a copy of the CC BY-NC-SA 4.0 was not distributed with this file, You can obtain one at http://creativecommons.org/licenses/by-nc-sa/4.0/ or send a letter to Creative Commons, PO Box 1866, Mountain View, CA 94042, USA.
// ==/UserScript==
class DebugExtension {
// you can modify the menu items' labels and access keys here, e.g. if you
// prefer another language. an access key is the letter highlighted in a
// menuitem's label. if the letter highlighted is "D" for example, and you
// press D on your keyboard while the context menu is open, it will
// automatically select the menu item with that access key. if two menu items
// have the same access key and are both visible, then instead of selecting
// one menu item it will just cycle between the two. however, the access key
// does not need to be a character in the label. if the access key isn't in
// the label, then instead of underlining the letter in the label, it will add
// the access key to the end of the label in parentheses.
// e.g. "Debug Extension (Q)" instead of "_D_ebug Extension".
config = {
menuLabel: "Debug Extension", // menu label
menuAccessKey: "D",
// individual menu items
Manifest: {
label: "Extension Manifest",
accesskey: "M",
},
ViewDocs: {
label: "View Documents",
accesskey: "D",
},
BrowserAction: {
label: "Browser Action",
accesskey: "B",
},
PageAction: {
label: "Page Action",
accesskey: "P",
},
SidebarAction: {
label: "Sidebar Action",
accesskey: "S",
},
Options: {
label: "Extension Options",
accesskey: "O",
},
Inspector: {
label: "Inspect Extension",
accesskey: "I",
},
ViewSource: {
label: "View Source",
accesskey: "V",
},
CopyID: {
label: "Copy ID",
accesskey: "C",
},
CopyURL: {
label: "Copy URL",
accesskey: "U",
},
};
actionTypes = ["BrowserAction", "PageAction", "SidebarAction", "Options"];
constructor() {
this.setupUpdate();
this.toolbarContext = document.getElementById("toolbar-context-menu");
this.overflowContext = document.getElementById(
"customizationPanelItemContextMenu"
);
this.pageActionContext = document.getElementById("pageActionContextMenu");
this.toolbarMenu = this.makeMainMenu(this.toolbarContext);
this.toolbarMenupopup = this.toolbarMenu.appendChild(
document.createXULElement("menupopup")
);
this.toolbarMenupopup.addEventListener("popupshowing", this);
this.overflowMenu = this.makeMainMenu(this.overflowContext);
this.overflowMenupopup = this.overflowMenu.appendChild(
document.createXULElement("menupopup")
);
this.overflowMenupopup.addEventListener("popupshowing", this);
this.pageActionMenu = this.makeMainMenu(this.pageActionContext);
this.pageActionMenupopup = this.pageActionMenu.appendChild(
document.createXULElement("menupopup")
);
this.pageActionMenupopup.addEventListener("popupshowing", this);
// make a menu item for each type of page within each context
[
"Manifest",
{ name: "ViewDocs", children: this.actionTypes },
"Inspector",
"ViewSource",
"CopyID",
"CopyURL",
].forEach(type =>
["toolbar", "overflow", "pageAction"].forEach(context => {
if (typeof type === "string") {
this.makeMenuitem(type, this[`${context}Menupopup`]);
} else if (typeof type === "object") {
this.makeMenu(type, this[`${context}Menupopup`]);
}
})
);
}
/**
* set a bunch of attributes on a node
* @param {object} element (a DOM node)
* @param {object} attrs (an object containing properties — keys are turned into attributes on the DOM node)
*/
maybeSetAttributes(element, attrs) {
for (let [name, value] of Object.entries(attrs)) {
if (value === void 0) element.removeAttribute(name);
else element.setAttribute(name, value);
}
}
// enable/disable menu items depending on whether the clicked extension has pages available to open.
handleEvent(e) {
if (e.target !== e.currentTarget) return;
let popup = e.target;
let id = this.getExtensionId(popup);
if (!id) return;
let extension = WebExtensionPolicy.getByID(id).extension;
let actions = new Map();
for (let type of this.actionTypes) {
actions.set(type, this.getActionURL(extension, type));
}
if (popup.className.includes("Submenu-Popup")) {
actions.forEach((url, type) => {
popup.querySelector(`.customize-context-${type}`).disabled = !url;
});
} else {
popup.querySelector(".customize-context-ViewDocs-Submenu").disabled = [
...actions.values(),
].every(url => !url);
popup.querySelector(".customize-context-ViewSource").disabled =
extension.addonData.isSystem ||
extension.addonData.builtIn ||
extension.addonData.temporarilyInstalled;
}
}
makeMainMenu(popup) {
let menu = document.createXULElement("menu");
this.maybeSetAttributes(menu, {
class: "customize-context-debugExtension",
label: this.config.menuLabel,
accesskey: this.config.menuAccessKey,
contexttype: popup === this.pageActionContext ? void 0 : "toolbaritem",
});
popup
.querySelector(
popup === this.pageActionContext
? ".manageExtensionItem"
: ".customize-context-manageExtension"
)
.after(menu);
return menu;
}
/**
* make a menu item that opens a given type of page, with label & accesskey
* corresponding to those defined in the "config" property
* @param {string} type (which menuitem to make)
* @param {object} popup (where to put the menuitem)
* @returns a menuitem DOM node
*/
makeMenuitem(type, popup) {
let item = document.createXULElement("menuitem");
this.maybeSetAttributes(item, {
class: `customize-context-${type}`,
label: this.config[type].label,
accesskey: this.config[type].accesskey,
oncommand: `debugExtensionMenu.onCommand(event, this.parentElement, "${type}")`,
contexttype: popup.closest("#pageActionContextMenu")
? void 0
: "toolbaritem",
});
popup.appendChild(item);
return item;
}
/**
* make a submenu in a given popup
* @param {string} type (which menu to make)
* @param {object} popup (where to put the menuitem)
* @returns a menu DOM node
*/
makeMenu(type, popup) {
let { name, children } = type;
if (!name || !children) return;
let menu = document.createXULElement("menu");
this.maybeSetAttributes(menu, {
class: `customize-context-${name}-Submenu`,
label: this.config[name].label,
accesskey: this.config[name].accesskey,
contexttype: popup.closest("#pageActionContextMenu")
? void 0
: "toolbaritem",
});
let menupopup = menu.appendChild(document.createXULElement("menupopup"));
menupopup.className = `customize-context-${name}-Submenu-Popup`;
menupopup.addEventListener("popupshowing", this);
children.forEach(item => this.makeMenuitem(item, menupopup));
popup.appendChild(menu);
return menu;
}
// get the ID for the button the context menu was opened on
getExtensionId(popup) {
return popup.closest("#pageActionContextMenu")
? BrowserPageActions.actionForNode(popup.triggerNode).extensionID
: ToolbarContextMenu._getExtensionId(popup);
}
matchesActionNode(elt) {
return (
elt.localName === "toolbarbutton" ||
elt.localName === "toolbaritem" ||
elt.localName === "toolbarpaletteitem" ||
elt.classList?.contains("urlbar-page-action")
);
}
getActionNode(elt) {
while (elt && !this.matchesActionNode(elt)) {
if (elt.parentNode.localName === "toolbar") return null;
elt = elt.parentNode;
}
return elt;
}
// get the URL for a given type of extension page, e.g. the popup that appears
// when you click the addon's toolbar button, or the addon's sidebar panel.
getActionURL(extension, type = this.actionTypes[0]) {
if (!extension) return;
let url;
let { global } = extension.apiManager;
let { manifest } = extension;
switch (type) {
case "BrowserAction":
url =
manifest.browser_action?.default_popup ||
global.browserActionFor(extension)?.action.globals.popup;
break;
case "PageAction":
url =
manifest.page_action?.default_popup ||
global.pageActionFor(extension)?.action.globals.popup;
break;
case "SidebarAction":
url =
manifest.sidebar_action?.default_panel ||
global.sidebarActionFor(extension)?.globals.panel;
break;
case "Options":
url = manifest.options_ui?.page;
break;
default:
}
return url;
}
// click callback
onCommand(event, popup, type) {
let id = this.getExtensionId(popup);
if (!id) return;
// this contains information about an extension with a given ID.
let extension = WebExtensionPolicy.getByID(id).extension;
// use extension's principal if it's available.
let triggeringPrincipal = extension.principal;
let url;
// which type of page to open. the "type" value passed is different for each menu item.
switch (type) {
case "Manifest":
url = `${extension.baseURL}manifest.json`;
break;
case "BrowserAction":
case "PageAction":
case "SidebarAction":
case "Options":
url = this.getActionURL(extension, type);
break;
case "Inspector":
url = `about:devtools-toolbox?id=${encodeURIComponent(
id
)}&type=extension`;
// use the system principal for about:devtools-toolbox
triggeringPrincipal = Services.scriptSecurityManager.getSystemPrincipal();
break;
case "ViewSource":
this.openArchive(id);
return;
case "CopyID":
case "CopyURL":
Cc["@mozilla.org/widget/clipboardhelper;1"]
.getService(Ci.nsIClipboardHelper)
.copyString(type === "CopyID" ? id : extension.baseURL);
let actionNode = this.getActionNode(popup.triggerNode);
if (
actionNode &&
windowUtils.getBoundsWithoutFlushing(actionNode)?.width
) {
window.CustomHint?.show(actionNode, "Copied");
}
return;
}
if (!url) return;
// if the extension's principal isn't available for some reason, make a content principal.
if (!triggeringPrincipal) {
triggeringPrincipal = Services.scriptSecurityManager.createContentPrincipal(
Services.io.newURI(url),
{}
);
}
// whether to open in the current tab or a new tab. only opens in the
// current tab if the current tab is on the new tab page or home page.
let where = new RegExp(
`(${BROWSER_NEW_TAB_URL}|${HomePage.get(window)})`,
"i"
).test(gBrowser.currentURI.spec)
? "current"
: "tab";
openLinkIn(url, where, {
triggeringPrincipal,
// only open in the background if the shift key was pressed when the menu item was clicked
inBackground: event.shiftKey,
});
}
/**
* open a given addon's source xpi file
* @param {string} id (an addon's ID)
*/
openArchive(id) {
let dir = Services.dirsvc.get("ProfD", Ci.nsIFile);
dir.append("extensions");
dir.append(`${id}.xpi`);
dir.launch();
}
// modify the internal functions that updates the visibility of the built-in
// "remove extension," "manage extension" items, etc. that's based on whether
// the button that was clicked is an extension or not, so it also updates the
// visibility of our menu by the same parameter.
setupUpdate() {
eval(
`ToolbarContextMenu.updateExtension = async function ${ToolbarContextMenu.updateExtension
.toSource()
.replace(/async updateExtension/, "")
.replace(
/let separator/,
`let debugExtension = popup.querySelector(\".customize-context-debugExtension\");\n let separator`
)
.replace(
/\[removeExtension, manageExtension,/,
`[removeExtension, manageExtension, debugExtension,`
)}`
);
eval(
`BrowserPageActions.onContextMenuShowing = async function ${BrowserPageActions.onContextMenuShowing
.toSource()
.replace(/async onContextMenuShowing/, "")
.replace(
/(let removeExtension.*);/,
`$1, debugExtension = popup.querySelector(".customize-context-debugExtension");`
)
.replace(/(removeExtension.hidden =)/, `$1 debugExtension.hidden =`)}`
);
}
}
if (gBrowserInit.delayedStartupFinished) {
window.debugExtensionMenu = new DebugExtension();
} else {
let delayedListener = (subject, topic) => {
if (topic == "browser-delayed-startup-finished" && subject == window) {
Services.obs.removeObserver(delayedListener, topic);
window.debugExtensionMenu = new DebugExtension();
}
};
Services.obs.addObserver(delayedListener, "browser-delayed-startup-finished");
}