solombala
Вот парочка вариантов, которые работают.
Может ещё как-нибудь можно. А вот как лучше — без понятия.
addon-card[addon-id^="custombutton://buttons/"] .addon-icon { width: 16px !important; height: 16px !important; /* padding-top: 3px !important; */ }
addon-list[type=custombuttons] .addon-icon { width: 16px !important; height: 16px !important; /* padding-top: 3px !important; */ }
Отсутствует
Отсутствует
как тут выровнять? Значок могу шевелить , а текст то не опустится...
Попробуй так:
Отредактировано unter_officer (12-12-2019 14:02:02)
«The Truth Is Out There»
Отсутствует
Жаль размер текста никак...
addon-list[type=custombuttons] .addon-name {
font-size: 22px !important;
}
«The Truth Is Out There»
Отсутствует
И самих аддонов тоже не могу сделать.
В этом коде font-size не для всех аддонов, а только для СВ.
Не срабатывает.
Не срабатывает возможно из-за версии , я на 68 ESR проверял, там срабатывает. А 71+ я пока не устанавливал, поэтому проверить не могу.
«The Truth Is Out There»
Отсутствует
Dumby
В 71 отрихтованая Вами Flip-Close_Tab ....ПКМ - на prev Tab - не пашет ( это я па правый переделал , но сам код не трогал) В 70 - океу...
var {interfaces: Ci, utils: Cu} = Components; Cu.import("resource://gre/modules/Services.jsm"); var flipclosetabStatic = { customjsm: false, initialized: false, get styleURI() { delete this.styleURI; return this.styleURI = Services.io.newURI("chrome://flip_close_tab/content/button.css", null, null); }, get buttonStrings() { delete this.buttonStrings; return this.buttonStrings = Services.strings.createBundle("chrome://flip_close_tab/locale/button.properties"); }, get CustomizableUI() { delete this.CustomizableUI; var _CustomizableUI = null; try { _CustomizableUI = Cu.import("resource:///modules/CustomizableUI.jsm", {}).CustomizableUI; } catch(e) { try { _CustomizableUI = Cu.import("chrome://flip_close_tab/content/customizable.jsm", {}).CustomizableUI; this.customjsm = true; } catch(e) {} } return this.CustomizableUI = _CustomizableUI; }, start: function(win) { var onLoad = () => { win.removeEventListener("load", onLoad); this.loadIntoWindow(win); }; win.addEventListener("load", onLoad); }, get loadIntoWindow() { delete this.loadIntoWindow; this.init(); return this.loadIntoWindow = function(win) { var obj = new FlipCloseTab(); obj.load(win); Object.defineProperty(win, "_Flip_Close_Tab_Prototype", { value: obj, writable: false, configurable: true, enumerable: false }); }; }, init: function() { if (this.initialized || !this.CustomizableUI) return; this.initialized = true; var cWidget = { id: "f-flip-tabs", type: "custom", label: flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.label"), tooltiptext: flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.tooltip"), defaultArea: this.CustomizableUI.AREA_NAVBAR, onBuild: function(document) { var window = document.defaultView; flipclosetabStatic.stylifyButton(window, true); var toolbarbutton_0 = (document.createXULElement || document.createElement).call(document, "toolbarbutton"); toolbarbutton_0.id = "f-flip-tabs"; toolbarbutton_0.setAttribute("label", flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.label")); toolbarbutton_0.setAttribute("tooltiptext", flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.tooltip")); toolbarbutton_0.setAttribute("context", false); toolbarbutton_0.classList.add("toolbarbutton-1"); toolbarbutton_0.classList.add("chromeclass-toolbar-additional"); toolbarbutton_0.addEventListener("click", function(event) { var gtBrowser = window.gBrowser; if (event.button == 2) { var old = gtBrowser.selectedTab; var tabs = Array.filter(old.parentNode.getElementsByAttribute("flipselectedID", "*"), tab => !tab.hidden && !tab.closing); var last = 2; var prevTab = null; tabs.forEach((tab)=> { let s = +tab.getAttribute("flipselectedID"); if (s && s > last && old != tab) { prevTab = tab; last = s; } }); if (prevTab !== null) gtBrowser.selectedTab = prevTab; } else if (event.button == 1) { gtBrowser.removeAllTabsBut(gtBrowser.selectedTab); } else if (event.button == 0) { event.preventDefault(); event.stopPropagation(); gtBrowser.removeTab(gtBrowser.selectedTab); } }, false); return toolbarbutton_0; } }; if (this.customjsm) { this.CustomizableUI.addListener(); cWidget.onloadStylePalette = function(window) { flipclosetabStatic.stylifyButton(window, true); }; } try { this.CustomizableUI.createWidget(cWidget); } catch(e) {} }, getWindowUtils: function(window) { var has = "windowUtils" in window && window.windowUtils instanceof Ci.nsIDOMWindowUtils; return (this.getWindowUtils = has ? window => window.windowUtils : window => window.QueryInterface(Ci.nsIInterfaceRequestor) .getInterface(Ci.nsIDOMWindowUtils) )(window); }, stylifyButton: function(window, stylify) { var method = stylify ? "loadSheet" : "removeSheet"; try { this.getWindowUtils(window)[method]( this.styleURI, Ci.nsIDOMWindowUtils.USER_SHEET ); } catch(ex) { Cu.reportError(ex); } }, uninit: function() { if (!this.initialized) return; try { this.CustomizableUI.destroyWidget("f-flip-tabs"); } catch(e) {} if (this.customjsm) { this.CustomizableUI.removeListener(); Cu.unload("chrome://flip_close_tab/content/customizable.jsm"); } var windows = Services.wm.getEnumerator("navigator:browser"); while (windows.hasMoreElements()) { var win = windows.getNext(); if (win.QueryInterface) win = win.QueryInterface(Ci.nsIDOMWindow); try { if ("_Flip_Close_Tab_Prototype" in win) { win._Flip_Close_Tab_Prototype.unload(true); delete win._Flip_Close_Tab_Prototype; } this.stylifyButton(win); } catch(e) {} } } }; var filterTabs = node => { var func = tab => !tab.hidden && !tab.closing; return (filterTabs = "from" in Array ? node => Array.from(node.getElementsByAttribute("flipselectedID", "*")).filter(func) : node => Array.filter(node.getElementsByAttribute("flipselectedID", "*"), func) )(node); } function FlipCloseTab() { this.gtBrowser = null; this.removeListeners = null; this.blurTab = null; this.eventTabClose = false; } FlipCloseTab.prototype = { load: function(win) { var gtBrowser = this.gtBrowser = win.gBrowser || win.getBrowser(); var tabid = 1; var TabSelect = (event) => { var tab = event.target; tab.setAttribute("flipselectedID", tabid++); }; var TabClose = (event) => { var old = event.target; if (!old.selected) return; var tabs = filterTabs(old.parentNode); var last = 0; var prevTab = null; tabs.forEach((tab) => { var s = +tab.getAttribute("flipselectedID"); if (s && s > last && old != tab) { prevTab = tab; last = s; } }); if (prevTab !== null && !prevTab.closing) gtBrowser.selectedTab = prevTab; }; gtBrowser.tabContainer.addEventListener("TabSelect", TabSelect, false); if ("_blurTab" in gtBrowser) { var blurTab = this.blurTab = gtBrowser._blurTab; gtBrowser._blurTab = function _blurTab() { var old = arguments[0]; if (!old.selected) return; var tabs = filterTabs(old.parentNode); var last = 0; var prevTab = null; tabs.forEach((tab) => { var s = +tab.getAttribute("flipselectedID"); if (s && s > last && old != tab) { prevTab = tab; last = s; } }); if (prevTab !== null && !prevTab.closing) { this.selectedTab = prevTab; return; } return blurTab.apply(this, arguments); }; } else { gtBrowser.tabContainer.addEventListener("TabClose", TabClose, true); this.eventTabClose = true; } this.removeListeners = function() { gtBrowser.tabContainer.removeEventListener("TabSelect", TabSelect, false); if (this.eventTabClose) gtBrowser.tabContainer.removeEventListener("TabClose", TabClose, true); }; var sel = gtBrowser.selectedTab; if (sel && !sel.hidden) sel.setAttribute("flipselectedID", tabid++); }, unload: function(shutdown = false) { this.removeListeners(); if (shutdown) { if (this.blurTab !== null) this.gtBrowser._blurTab = this.blurTab; var tabs = filterTabs(this.gtBrowser.tabContainer); tabs.forEach((tab) => { tab.removeAttribute("flipselectedID"); }); } } }; var WindowListener = { onOpenWindow: function(aWindow) { var win = aWindow.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindow); function onWindowLoad() { win.removeEventListener("load", onWindowLoad); if (win.document.documentElement.getAttribute("windowtype") == "navigator:browser") flipclosetabStatic.loadIntoWindow(win); } win.addEventListener("load", onWindowLoad); }, onCloseWindow: function(aWindow) { var win = aWindow.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindow); if (win.document.documentElement.getAttribute("windowtype") == "navigator:browser" && "_Flip_Close_Tab_Prototype" in win) win._Flip_Close_Tab_Prototype.unload(); }, onWindowTitleChange: function(aWindow, newTitle) { } }; function startup(data, reason) { var wm = Services.wm; var windows = wm.getEnumerator(null); while (windows.hasMoreElements()) { var win = windows.getNext(); if (win.QueryInterface) win = win.QueryInterface(Ci.nsIDOMWindow); try { var type = win.document.documentElement.getAttribute("windowtype"); if (type == "navigator:browser") flipclosetabStatic.loadIntoWindow(win); else if (type == "navigator:blank") flipclosetabStatic.start(win); } catch(e) {} } wm.addListener(WindowListener); } function shutdown(data, reason) { if (reason == APP_SHUTDOWN) return; flipclosetabStatic.uninit(); Services.wm.removeListener(WindowListener); } function install(data, reason) { } function uninstall(data, reason) { }
Отсутствует
ПКМ - на prev Tab - не пашет ( это я па правый переделал , но сам код не трогал)
toolbarbutton_0.addEventListener("click", function(event) { var gtBrowser = window.gBrowser; if (event.button == 2) { event.preventDefault(); event.stopPropagation(); var old = gtBrowser.selectedTab; var tabs = filterTabs(old.parentNode); var last = 0; var prevTab = null; tabs.forEach((tab)=> { let s = +tab.getAttribute("flipselectedID"); if (s && s > last && old != tab) { prevTab = tab; last = s; } }); if (prevTab !== null) gtBrowser.selectedTab = prevTab; } else if (event.button == 1) { gtBrowser.removeAllTabsBut(gtBrowser.selectedTab); } else if (event.button == 0) { gtBrowser.removeTab(gtBrowser.selectedTab); } }, false);
Отсутствует
Есть кнопка Proxy. В ней есть функция:
// Функция открывает настройки прокси в окне ................... function openConnections() { self.win && self.win.close(); self.win = openDialog("chrome://browser/content/preferences/connection.xul", "Proxy", "centerscreen"); // добавить атрибут "prefwindow" self.win.addEventListener("load", function f(e) { this.removeEventListener("load", f, true); e.target.documentElement.setAttribute("type", "prefwindow"); }, true); // закрыть настройки прокси по клику на странице gBrowser.addEventListener("click", function c() { this.removeEventListener("click", c); try { self.win.close() } catch(e) {}; }, true); };
В целом функция работает нормально - открывает настройки прокси в окне, но при этом в консоли появляется ошибка: TypeError: window.opener.gSubDialog is undefined
Поскольку всё работает, ошибка особо не напрягает, но всё же, возможно ли это поправить под 68 ESR?
«The Truth Is Out There»
Отсутствует
Dumby
Что с мульти в 71? Код config.js ? В портабл в ini это MOZ_FORCE_DISABLE_E10S=1
Раньше обходилось config.js и кнопкой cbu.setPrefs(s, cbu.getPrefs(s) == true ? false : true);
Теперь наглухо только MOZ_FORCE_DISABLE_E10S .... " Проблема небольшая с раскрытием подробностей в about:addons (кнопки СВ - редактирование) и аддоны тоже..
Шняга вылазит ..Типа, не может протокол обработать ...
https://forum.mozilla-russia.org/viewto … 94#p775894
Отредактировано solombala (14-12-2019 17:29:03)
Отсутствует
поправить под 68 ESR
Может так
function openConnections() { var url = "chrome://browser/content/preferences/connection.xul"; var win = [...Services.wm.getEnumerator(null)].find(w => w.location == url); if (win) return win.focus(); win = openDialog(url, "Proxy", "centerscreen"); win.opener = {gSubDialog: {_dialogs: [{ _frame: {get contentDocument() { var args = ["click", win.close.bind(win), true]; var unload = () => gBrowser.removeEventListener(...args); gBrowser.addEventListener(...args); win.addEventListener("unload", unload, {once: true}); delete this.contentDocument; return this.contentDocument = win.document; }}, resizeVertically: () => win.sizeToContent() }]}}; }
Раньше обходилось config.js и кнопкой
Это похерили ещё в 70. Я пробовал тогда раскопать,
и мне показалось, что вот здесь расклад запрашивается
для решения, запускать ли socket-процесс, а будучи запрошенным,
всё, поздно уже что-то в config.js переставлять.
Но, разумеется, это гадание на кофейной гуще, факт лишь в том,
что работать перестало, и с этим ничего не поделаешь.
Шняга вылазит
Да, у меня воспроизводится, если отключить многопроцессность,
и кликать по названию аддона.
Как любителю правки omni.ja, могу предложить попробовать
добавить e.preventDefault(); в aboutaddons.js, вот сюда
default: // Handle a click on the card itself. if ( !this.expanded && (e.target === this.addonNameEl || !e.target.closest("a")) ) { e.preventDefault(); // <= ДОБАВИТЬ loadViewFn(`detail/${this.addon.id}`, e); } else if (
Отсутствует
Может так
скрытый текстВыделить кодКод:
function openConnections() { var url = "chrome://browser/content/preferences/connection.xul"; var win = [...Services.wm.getEnumerator(null)].find(w => w.location == url); if (win) return win.focus(); win = openDialog(url, "Proxy", "centerscreen"); win.opener = {gSubDialog: {_dialogs: [{ _frame: {get contentDocument() { var args = ["click", win.close.bind(win), true]; var unload = () => gBrowser.removeEventListener(...args); gBrowser.addEventListener(...args); win.addEventListener("unload", unload, {once: true}); delete this.contentDocument; return this.contentDocument = win.document; }}, resizeVertically: () => win.sizeToContent() }]}}; }
То, что надо! Спасибо!
«The Truth Is Out There»
Отсутствует
А на кнопках СВ - также...Может там рихтануть?
Да, точно, в CB тоже e.preventDefault() надо дописать.
Я добавлю в addons-html.js, спасибо.
click(e) { if (this.cbAddonList && !e.button && !e.target.matches(this.ignoreClickSelectors)) { var addon = a(e); if (addon) e.stopPropagation(), e.preventDefault(), e.detail == 1 && this.editButton(addon); } },
Отсутствует
Dumby
Курвье маjка ! Три дня мудохался , пока мульти определил... Благо есть такие врачи ...Хвала, брате! Чува тебе Бог!
browser.tabs.remote.autostart - true // На СВ - влияет , только true !
Очередная поганка! https://forum.mozilla-russia.org/viewto … 53#p775953
Нужен код в config.js ///Эта штука в ini MOZ_FORCE_DISABLE_E10S= все портит...
Отредактировано solombala (15-12-2019 20:45:42)
Отсутствует
Dumby скажи пожалуйста в вновь создаваемых кнопках в секциях Код и Инициализация отсутствуют /*CODE*/ и /*Initialization Code*/ так должно быть или только у меня?
Версия 71 и custom_buttons-0.0.7.0.0.8-fx-paxmod.xpi, кеш сброшен
В кнопке
/*Initialization Code*/ ((main, parts) => this._handleClick = () => { var df = MozXULElement.parseXULToFragment(` <menupopup> <menuitem class="menuitem-iconic" image="" label="Сохранить всю страницу как PNG" value="all"/> <menuitem class="menuitem-iconic" image="" label="Сохранить видимую часть страницы как PNG" value="page"/> <menuitem class="menuitem-iconic" image="" label="Сохранить выбранный элемент страницы как PNG" value="click"/> <menuitem class="menuitem-iconic" image="" label="Сохранить выбранную область страницы как PNG" value="clipping"/> </menupopup> `); var popup = df.firstChild; popup.setAttribute("context", ""); popup.setAttribute("oncommand", "handleCommand(event);"); popup.handleCommand = e => { var name = _id + ":DataURLReady"; main = main.replace("%MESSAGE_NAME%", name); var urls = {}, configurable = true, enumerable = true; Object.entries(parts).forEach(([key, part]) => Object.defineProperty(urls, key, { configurable, enumerable, get() { var value = `data:;charset=utf-8,({${ encodeURIComponent(main + part) }%0A}).init("${key}")`; Object.defineProperty(urls, key, {configurable, enumerable, value}); return value; }})); var getTabLabel = () => { var label = gBrowser.selectedTab.label; var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " "); return label.substring(0, 50); } var listener = msg => { var fp = makeFilePicker(); fp.init(window, "Сохранить как…", fp.modeSave); fp.appendFilter("", "*.png"); fp.defaultString = getTabLabel() + ".png"; fp.open(res => { if (res == fp.returnCancel || !fp.file) return; var wbp = makeWebBrowserPersist(), args = [ Services.io.newURI(msg.data), document.nodePrincipal, null, null, null, null, fp.file, null ]; wbp.saveURI.length == 9 && args.splice(2, 0, null); wbp.saveURI(...args); }); } messageManager.addMessageListener(name, listener); addDestructor(() => messageManager.removeMessageListener(name, listener)); (popup.handleCommand = e => gBrowser.selectedBrowser.messageManager .loadFrameScript(urls[e.target.value], false) )(e); } this.append(df); (this._handleClick = () => popup.openPopup(this, "after_start"))(); })(` init(cmd) { cmd.startsWith("c") ? this[cmd].init(this[cmd].parent = this) : this[cmd](); }, capture(win, x, y, width, height) { var canvas = win.document.createElementNS("${xhtmlns}", "canvas"); canvas.width = width; canvas.height = height; var ctx = canvas.getContext("2d"); var tryDraw = ind => { try {ctx.drawWindow(win, x, y, canvas.width, canvas.height, "white")} catch(ex) {canvas.height = ind * canvas.width; tryDraw(--ind);} } tryDraw(17); sendAsyncMessage("%MESSAGE_NAME%", canvas.toDataURL("image/png")); }, `, { all: `all() { var win = content; this.capture(win, 0, 0, win.innerWidth + win.scrollMaxX, win.innerHeight + win.scrollMaxY); }`, page: `page() { var win = content, doc = win.document, body = doc.body, html = doc.documentElement; var scrX = (body.scrollLeft || html.scrollLeft) - html.clientLeft; var scrY = (body.scrollTop || html.scrollTop) - html.clientTop; this.capture(win, scrX, scrY, win.innerWidth, win.innerHeight); }`, clipping: `clipping: { handleEvent(e) { if (e.button) return false; e.preventDefault(); e.stopPropagation(); switch(e.type) { case "mousedown": this.downX = e.pageX; this.downY = e.pageY; this.bs.left = this.downX + "px"; this.bs.top = this.downY + "px"; this.body.appendChild(this.box); this.flag = true; break; case "mousemove": if (!this.flag) return; this.moveX = e.pageX; this.moveY = e.pageY; if (this.downX > this.moveX) this.bs.left = this.moveX + "px"; if (this.downY > this.moveY) this.bs.top = this.moveY + "px"; this.bs.width = Math.abs(this.moveX - this.downX) + "px"; this.bs.height = Math.abs(this.moveY - this.downY) + "px"; break; case "mouseup": this.uninit(); break; } }, init() { var win = {}; Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager) .getFocusedElementForWindow(content, true, win); this.win = win.value; this.doc = this.win.document; this.body = this.doc.body; if (!HTMLBodyElement.isInstance(this.body)) { Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService) .showAlertNotification("${self.image}", ${JSON.stringify(self.label)}, "Не удается захватить!"); return false; } this.flag = null; this.box = this.doc.createElement("div"); this.bs = this.box.style; this.bs.border = "#0f0 dashed 2px"; this.bs.position = "absolute"; this.bs.zIndex = "2147483647"; this.defaultCursor = this.win.getComputedStyle(this.body, "").cursor; this.body.style.cursor = "crosshair"; ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.addEventListener(type, this, true)); }, uninit() { var pos = [this.win, parseInt(this.bs.left), parseInt(this.bs.top), parseInt(this.bs.width), parseInt(this.bs.height)]; this.body.style.cursor = this.defaultCursor; this.body.removeChild(this.box); this.parent.capture.apply(this, pos); ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.removeEventListener(type, this, true)); } }`, click: `click: { getPosition() { var html = this.doc.documentElement; var body = this.doc.body; var rect = this.target.getBoundingClientRect(); return [ this.win, Math.round(rect.left) + (body.scrollLeft || html.scrollLeft) - html.clientLeft, Math.round(rect.top) + (body.scrollTop || html.scrollTop) - html.clientTop, parseInt(rect.width), parseInt(rect.height) ]; }, highlight() { this.orgStyle = this.target.hasAttribute("style") ? this.target.style.cssText : false; this.target.style.cssText += "outline: red 2px solid; outline-offset: 2px; -moz-outline-radius: 2px;"; }, lowlight() { if (this.orgStyle) this.target.style.cssText = this.orgStyle; else this.target.removeAttribute("style"); }, handleEvent(e) { switch(e.type){ case "click": if (e.button) return; e.preventDefault(); e.stopPropagation(); this.lowlight(); this.parent.capture.apply(this, this.getPosition()); this.uninit(); break; case "mouseover": if (this.target) this.lowlight(); this.target = e.target; this.highlight(); break; } }, init() { this.win = content; this.doc = content.document; ["click", "mouseover"].forEach(type=> this.doc.addEventListener(type, this, true)); }, uninit() { this.target = false; ["click", "mouseover"].forEach(type=> this.doc.removeEventListener(type, this, true)); } }` });
Доп пункты в about;addons
обсуждалось на 556 стр. со значками и без. Мой вариант
//Дополнительные пункты контекстного меню на странице about:addons для аддонов, плагинов, тем, CB..................................................................................... ((id, g, iconizer) => addDestructor(r => r[5] == "e" && id in g && g[id].destroy()) + addEventListener("shown", { //------------------------------------------------------------------ "Копировать имя_i": "", "Копировать имя"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.name); }, //------------------------------------------------------------------ "Копировать ID_i": "", "Копировать ID"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.id); }, //------------------------------------------------------------------ "Копировать версию_i": "", "Копировать версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; gClipboard.write(addon.version); }, //------------------------------------------------------------------ "Копировать имя и версию_i": "", "Копировать имя и версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; gClipboard.write(addon.name + " " + addon.version); }, //------------------------------------------------------------------ "Домашняя страница_i": "", "Домашняя страница"(addon, hideOn) { if (hideOn) return !addon.homepageURL; var url = addon.homepageURL; if (!url) { if (addon.reviewURL) { url = addon.reviewURL.replace(/\/reviews\/.*$/, "/"); } } openURL(url); }, //------------------------------------------------------------------ "Поиск на АМО_i": "", "Поиск на АМО"(addon, hideOn) { if (hideOn) return ["custombuttons","theme","plugin"]; var url = addon.homepageURL; if (!url) { url = "https://addons.mozilla.org/search/?q=" + encodeURIComponent(addon.name); } openURL(url); }, //------------------------------------------------------------------ "Папка установки_i": "", "Папка установки"(addon, hideOn) { if (hideOn) return ["custombuttons","theme","plugin"]; switch (addon.type) { case "plugin": var pathes = addon.pluginFullpath; for (var i = 0; i < pathes.length; i++) { this.revealPath(pathes[i]); } return; case "userchromejs": var file = addon._script.file; if (file.exists()) file.reveal(); return; } var gecko = parseInt(Services.appinfo.platformVersion); var nsLocalFile = Components.Constructor("@mozilla.org/file/local;1", (gecko >= 14) ? "nsIFile" : "nsILocalFile", "initWithPath"); var dir = Services.dirsvc.get("ProfD", Ci.nsIFile); dir.append("extensions"); dir.append(addon.id); var fileOrDir = dir.path + (dir.exists() ? "" : ".xpi"); try { (new nsLocalFile(fileOrDir)).reveal(); } catch (ex) { var addonDir = /.xpi$/.test(fileOrDir) ? dir.parent : dir; try { if (addonDir.exists()) { addonDir.launch(); } } catch (ex) { var uri = Services.io.newFileURI(addonDir); var protSvc = Cc["@mozilla.org/uriloader/external-protocol-service;1"]. getService(Ci.nsIExternalProtocolService); protSvc.loadUrl(uri); } } }, //------------------------------------------------------------------ "Файл установки_i": "", "Файл установки"(addon, hideOn) { if (hideOn) return ["custombuttons","theme","plugin"]; var gecko = parseInt(Services.appinfo.platformVersion); var nsLocalFile = Components.Constructor("@mozilla.org/file/local;1", (gecko >= 14) ? "nsIFile" : "nsILocalFile", "initWithPath"); var dir = Services.dirsvc.get("ProfD", Ci.nsIFile); dir.append('extensions'); dir.append(addon.id); if ( dir.exists() ) dir.launch(); var file = Components.classes['@mozilla.org/file/directory_service;1'] .getService(Components.interfaces.nsIProperties) .get('ProfD', Components.interfaces.nsIFile); file.append('extensions'); file.append( addon.id + '.xpi' ) if ( file.exists() ) file.launch(); return; }, //------------------------------------------------------------------ url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { if (e.target.baseURI != this.url) return; var item = this.getItem(e.target.ownerDocument); var addon = item.addon = e.target.closest("addon-card").addon; for(var child of item.children) { var res = this[child.textContent](addon, true); child.hidden = Array.isArray(res) ? res.includes(addon.type) : res; } e.target.contains(item) || requestAnimationFrame( () => e.target.prepend(item) ); }, click(e) { e.stopPropagation(); iconizer.item.parentNode.hide(); this[e.target.textContent](iconizer.item.addon); }, getItem(doc) { if (iconizer.item) { if (iconizer.item.ownerDocument == doc) return iconizer.item; iconizer.handleEvent(); } var item = doc.createElement("div"); item.id = id; for(var lab of this.labels) item.appendChild( doc.createElement("panel-item") ).append(lab); item.onclick = this.click; doc.ownerGlobal.addEventListener("unload", iconizer); return iconizer.item = item; }, get labels() { delete this.labels; this.click = this.click.bind(this); if (id in g) return this.labels = (iconizer = g[id]).labs; g[id] = iconizer; var css = "", ind = 0, arr = []; var push = (ind, icon) => { var chromeImg = `chrome://custombuttons/content/${id + ind}`; arr.push(["override", chromeImg, icon]); return chromeImg; } var labs = iconizer.labs = Object.keys(this).filter(key => { var res = String(this[key]).startsWith('"'); if (!res) return false; ind++; var icon = this[key + "_i"]; if (icon) css += `\n\t#${ id } > panel-item:nth-child(${ind}) {\n\t\t--icon: url(${ push(ind, icon) }) !important;\n\t}`; return true; }); var ams = Cc["@mozilla.org/addons/addon-manager-startup;1"].getService(Ci.amIAddonManagerStartup); var mUri = Services.io.getProtocolHandler("resource").getSubstitution("custombuttons-modules"); iconizer.iconHelper = ams.registerChrome(mUri, arr); var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService); var md = "@-moz-document url(chrome://mozapps/content/extensions/aboutaddons.html) {"; var uri = Services.io.newURI("data:text/css;charset=utf-8," + encodeURIComponent(md + css + "\n}")); var args = [uri, sss.USER_SHEET]; sss.loadAndRegisterSheet(...args); iconizer.handleEvent = function() { if (!this.item) return; this.item.ownerGlobal.removeEventListener("unload", this); this.item.remove(); this.item = null; } iconizer.destroy = function() { delete g[id]; this.handleEvent(); sss.unregisterSheet(...args); this.iconHelper.destruct(); } return this.labels = labs; }, revealPath: function(path){ var file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsIFile); file.initWithPath(path); if(file.exists()) file.reveal(); } }, true, gBrowser.tabpanels || 1))("CBAddonsMenuExt", Cu.import("resource://gre/modules/Services.jsm", {}), {}); //Дополнительные кнопки на странице about:addons для аддонов, плагинов, тем, CB..................................................................................... ((cn, flag, css) => addEventListener("update", ({ btnActions: ["preferences", "toggle-disabled", "remove", "install-update"], every(reason) { reason || addDestructor(this.every, this); var g = this.g || (this.g = Cu.import( "resource://gre/modules/AddonManager.jsm", {} )); var has = flag in g; if (reason) { if (!has || reason[5] != "e") return; } else if (has) return this; var method, wins = Array.from( g.AddonManagerInternal.addonListeners, Cu.getGlobalForObject ).filter( w => w.docShell && w.docShell.name == "html-view-browser" ); if (reason) method = "destroyDoc", delete g[flag]; else method = "initDoc", g[flag] = true; for(var win of new Set(wins)) this[method](win.document); return this; }, initDoc(doc, onlyStyle) { if (doc[flag]) return; doc[flag] = true; doc.ownerGlobal.windowUtils.loadSheetUsingURIString(this.url, this.type); if (onlyStyle) return; for(var card of doc.getElementsByTagName("addon-card")) this.onCard(card); }, destroyDoc(doc) { if (!doc[flag]) return; delete doc[flag]; for(var span of Array.from(doc.getElementsByClassName(cn))) span.remove(); doc.ownerGlobal.windowUtils.removeSheetUsingURIString(this.url, this.type); }, get url() { this.type = windowUtils.USER_SHEET; this.btnActions = this.btnActions.map( action => `panel-list > panel-item[action="${action}"]` ); delete this.url; return this.url = "data:text/css;charset=utf-8," + encodeURIComponent(css.replace(/;/g, " !important;")); }, handleEvent(e) { var card = e.target; card.nodeName == "ADDON-CARD" && this.onCard(card); }, onCard(card, again) { var div = card.querySelector("div.more-options-menu"); if (!div) return again || card.ownerGlobal.requestAnimationFrame( () => this.onCard(card, true) ); var [span] = card.getElementsByClassName(cn); if (span) span.textContent = ""; else { var doc = card.ownerDocument; doc[flag] || this.initDoc(doc, true); div.before(span = doc.createElement("span")); span.className = cn; span.onclick = this.onclick; } var btns = this.btnActions.map(sel => { var item = div.querySelector(sel); return item && item.cloneNode(true); }); var {addon} = card, cbbtn = addon.type == "custombuttons"; for(var item of btns) if (item) { span.append(item); item.shadowRoot.querySelector("button").classList.add("cb-cloned-button"); if (cbbtn) item.closest = this.closest; } if (addon.id != "custombuttons@xsms.org") return; var item = span.querySelector("[action=preferences]"); if (item) item.removeAttribute("hidden"), item.onclick = this.cbprefs; }, onclick(e) { InspectorUtils.removeContentState(e.target, 4, true); Services.focus.clearFocus(e.target.ownerGlobal); }, closest(sel) { return sel == ".more-options-menu" || this.ownerGlobal .Element.prototype.closest.call(this, sel); }, cbprefs(e) { var url = "chrome://custombuttons/content/prefs.xul"; for(var win of Services.wm.getEnumerator(null)) if (!win.closed && win.document.documentURI == url) return win.focus(); e.view.requestAnimationFrame(() => openDialog( url, "", "chrome,titlebar,toolbar,centerscreen,modal" )); } }).every(), true))("cb-cloned-buttons-container", "AAcbButtonizedFlag", ` addon-card:not([expanded]) .card-contents { width: 1px; } span.cb-cloned-buttons-container { display: flex; } button.cb-cloned-button { width: auto; padding: 1px 6px 3px 6px; margin: 0 1px; background-image: none; //background: #CCFFFF ; border: 1px solid var(--in-content-box-border-color); //border-style: solid; //border-color: #99FFFF #99FFFF; border-radius: 0px; font-size: 13px; font-family: Segoe UI; } button.cb-cloned-button:hover { background: gold ; } button.cb-cloned-button:after { display: none; } `);
Отредактировано Andrey_Krropotkin (16-12-2019 12:43:55)
Отсутствует
solombala а хз, не задавался этим вопросом
Отредактировано Andrey_Krropotkin (16-12-2019 17:38:21)
Отсутствует
А, как кнопочку эту отцентровать?
Как-то так, наверное.
@-moz-document url-prefix(chrome://mozapps/content/extensions/) { .more-options-button.ghost-button { margin: 10px 6px 0 2px !important; } }
Цифры свои подберешь.
«The Truth Is Out There»
Отсутствует
solombala
В теме по стилям Vitaliy V. вам ответил как это исправить.
У меня нет 71, поэтому проверить не могу.
А на 68 ESR стиль от Виталия не совсем подошел. Поэтому пришлось немного переделать под себя:
addon-card .cb-cloned-buttons-container, addon-card .more-options-menu { display: flex !important; align-self: center !important; margin: 0 0 0 1px !important; } addon-card panel-list[valign="top"] { bottom: 20px !important; }
Отредактировано unter_officer (17-12-2019 00:54:51)
«The Truth Is Out There»
Отсутствует