Dumby раньше Вы мне показывали как создать в контекстном меню аддонов на странице about:addons.
var copyToClipboard = function (aString) { Cc["@mozilla.org/widget/clipboardhelper;1"]. getService(Ci.nsIClipboardHelper).copyString(aString); }; (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Домашняя страница"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { var url = addon.homepageURL; if (!url) { if (addon.reviewURL) { url = addon.reviewURL.replace(/\/reviews\/.*$/, "/"); } else { url = "https://addons.mozilla.org/search/?q=" + encodeURIComponent(addon.name); } } openURL(url); } }, true, gBrowser.tabpanels || 1); })(); (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Папка установки"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { 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); } } } }, true, gBrowser.tabpanels || 1); })(); (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Файл установки"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { 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; } }, true, gBrowser.tabpanels || 1); })(); (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Копировать имя и версию"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { copyToClipboard(addon.name + " " + addon.version); } }, true, gBrowser.tabpanels || 1); })(); (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Копировать версию"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { copyToClipboard(addon.version); } }, true, gBrowser.tabpanels || 1); })(); (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Копировать ID"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { copyToClipboard("ID: " + addon.id); } }, true, gBrowser.tabpanels || 1); })(); (function () { addEventListener("shown", { url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { e.target.baseURI == this.url && !e.target.contains(this.item) && requestAnimationFrame(() => e.target.prepend(this.item)); }, get item() { var item = document.createElementNS(xhtmlns, "panel-item"); item.append("Копировать имя"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this.onAddon(item.closest("addon-card").addon); } addDestructor(() => item.remove()); delete this.item; return this.item = item; }, onAddon(addon) { copyToClipboard(addon.name); } }, true, gBrowser.tabpanels || 1); })();
/*CODE*/ var hid = false; //false - скрыть скрытые адоны, true - показать скрытые адоны var br = gBrowser; var img2 = ""; var img1 = ""; var trigger, trigger1, dialog, drives, count; var icon1 = ""; var icon2 = ""; var icon3 = ""; var icon4 = ""; var icon5 = ""; function jarLoaded(e) { var val = gURLBar.value; if(val.search(/(file|jar):/) == 0 && val.search(/\.(ja|jar|xpi|zip)$/i) > -1) loadURI("jar:" + val + "!/"); } function jarClicked(e) { if(e.button == 0) { var tabUrl = gURLBar.value; if(tabUrl.search(/(file|jar):/) == 0 && tabUrl.search(/\/$/) > -1 || tabUrl.search(/view-source:/) == 0) { var node = e.target.href; if(node && node.search(/\.(exe|dll|sqlite|sqlite-shm|sqlite-wal)$/i) == -1) { if(tabUrl.search(/view-source:/) == 0 && node.search(/view-source:/) == 0 || node.search(/\.(ja|jar|xpi|zip)$/i) > -1) { e.preventDefault(); e.stopPropagation(); br.selectedTab = br.addTab(node); } else if(node.search(/\/$/) == -1) { e.preventDefault(); e.stopPropagation(); br.selectedTab = br.addTab("view-source:" + un(node)); }}}}} function intf(drives, count) { var data = '<?xml version="1.0"?>'; data += '<?xml-stylesheet href="chrome://global/skin/" type="text/css"?>'; data += '<window title="Firefox Explorer" onload="self.load()" xmlns="' + xulns + '">'; data += '<keyset>'; data += '<key keycode="VK_ESCAPE" oncommand="close()"/>'; data += '</keyset>'; data += '<vbox flex="1">'; data += '<richlistbox id="listbox" flex="1" >'; data += '<listcols>'; data += '<listcol/>'; data += '<listcol flex="1" />'; data += '</listcols>'; data += '</richlistbox>'; data += '<hbox>'; data += '<button image="' + icon2 + '" label=" Add-on" oncommand="self.addon(event)" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + img2 + '" label=" 1 omni.ja" oncommand="self.omni(event)" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + img2 + '" label=" 2 omni.ja" oncommand="self.omni1(event)" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + icon4 + '" label=" Firefox" oncommand="self.folder(event,' + "'GreD'" + ')" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + icon3 + '" label=" Профиль" oncommand="self.folder(event,' + "'ProfD'" + ')" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button label="Закрыть" oncommand="self.close()"/>'; data += '</hbox>'; data += '<hbox>'; for(i = 0; i < count; i++) data += '<button image="' + icon1 + '" label=" ' + drives[i] + '" oncommand="self.mydrives(event,' + "'" + drives[i] + "'" + ')" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '</hbox>'; data += '</vbox>'; data += '</window>'; return data.replace(/self/g, "opener.document.getElementById("" + self.id + "")"); } this.load = function() { if(!("AddonManager" in window)) Components.utils.import("resource://gre/modules/AddonManager.jsm"); if(!("Services" in window)) Components.utils.import("resource://gre/modules/Services.jsm"); var then, promise = AddonManager.getAddonsByTypes(["extension"], then = function(addons) { var list = new Array(); addons.forEach( function(addon) { if (addon.hidden == false || addon.hidden == hid) { list.push(addon); } }); function key(addon) { return addon.name.toLowerCase(); } list.sort( function( a, b){ var ka = key(a); var kb = key(b); return ka == kb ? 0 : ka < kb ? -1 : 1; }); var {document} = dialog; for(var addon of list){ var item = document.createXULElement("richlistitem"); item.setAttribute("selected", "false"); var cell1 = document.createXULElement("vbox"); cell1.setAttribute("class", "icon-container"); item.appendChild(cell1); var cell2 = document.createXULElement("image"); cell2.setAttribute("anoid", "icon"); cell2.setAttribute("class", "icon"); cell2.setAttribute("src", addon.iconURL); if (addon.iconURL == null) cell2.setAttribute("src", icon5); cell2.style.height="16px"; cell2.style.width="16px"; cell1.appendChild(cell2); var cell = document.createXULElement("label"); cell.setAttribute("value", addon.name); cell.style.height="16px"; item.appendChild(cell); var cell = document.createXULElement("label"); cell.setAttribute("value", addon.version); item.appendChild(cell); var box = dialog.document.getElementById("listbox"); box.appendChild(item).addon = addon; box.focus(); } }); promise && typeof promise.then == "function" && promise.then(then, Components.utils.reportError); // Firefox 61+ } this.addon = e => { var item1 = dialog.document.querySelector("richlistitem[selected]"); var uri = item1.addon.getResourceURI(); if (uri.schemeIs("jar")) uri = uri.spec else { var file = uri.QueryInterface(Ci.nsIFileURL).file; uri = file.isDirectory() ? uri.spec : "jar:" + uri.spec + "!/"; } if (!e.ctrlKey) { dialog.close(); dialog = null; } br.selectedTab = br.addTrustedTab(uri); } this.omni1 = function(e) { var profileDir = Components.classes["@mozilla.org/file/directory_service;1"] .getService(Components.interfaces.nsIProperties) .get("GreD", Components.interfaces.nsIFile) .path; var file = profileDir + "\\browser\\omni.ja"; var vert="jar:" + "file:///" + file + "!/"; br.selectedTab = br.addTrustedTab(vert); if(!e.ctrlKey) dialog.close(); } this.omni = function(e) { var file = Services.dirsvc.get("GreD", Ci.nsIFile); file.append("omni.ja"); var uri = Services.io.newFileURI(file); br.selectedTab = br.addTrustedTab("jar:" + uri.spec + "!/"); if(!e.ctrlKey) dialog.close(); } this.folder = function(e, shortcut) { var uri = Services.io.newFileURI(Services.dirsvc.get(shortcut, Ci.nsIFile)); br.selectedTab = br.addTrustedTab(uri.spec); if(!e.ctrlKey) dialog.close(); } this.mydrives = function(e, letter) { br.selectedTab = br.addTrustedTab("file:///" + letter + "/"); if(!e.ctrlKey) dialog.close(); } this.close = function() { dialog.close(); } Cu.import("resource://gre/modules/FileUtils.jsm"); var root = new FileUtils.File("\\\\."); var drivesEnum = root.directoryEntries; drives = []; while (drivesEnum.hasMoreElements()) { drives.push(drivesEnum.getNext().QueryInterface(Ci.nsIFile).path); } count = drives.length; //var url = "data:application/vnd.mozilla.xul+xml;text/plain," + encodeURIComponent(intf(drives, count)); var url = "data:application/vnd.mozilla.xul+xml;text/plain," + encodeURIComponent(intf(drives, count)); if (parseInt(Services.appinfo.platformVersion) >= 69 && Services.appinfo.browserTabsRemoteAutostart) { var chromeURL = `chrome://custombuttons/content/cbdialog${Date.now()}.xul`; Cc["@mozilla.org/addons/addon-manager-startup;1"].getService(Ci.amIAddonManagerStartup).registerChrome( Services.io.newFileURI(Services.dirsvc.get("ProfD", Ci.nsIFile)), [["override", chromeURL, url]] ); url = chromeURL; } var feature = "chrome,centerscreen,width=580,height=410,alwaysRaised"; dialog = window.openDialog(url, "", feature);
Отредактировано Andrey_Krropotkin (25-10-2019 14:10:20)
Отсутствует
bunda1
Похоже, теперь в адресной строке не работает прокрутка. Пробовал интерфейс и со строкой поиска и без. В "urlbar" никак.
searchbar [144.083×26]
namespaceURI: XUL
margin: 0
border: 1px
padding: 0
id = searchbar
flex = 1
src = moz-extension://243005c3-3d6a-465d-9496-850fde7893be/yandex-ru.ico
Отсутствует
bunda1
Похоже, теперь в адресной строке не работает прокрутка. Пробовал интерфейс и со строкой поиска и без. В "urlbar" никак.
Добавлено 24-10-2019 20:13:59
addEventListener('wheel', (e, trg = e.target)=> { if ( trg.id != 'urlbar' ) return; trg.value = ''; });
Отредактировано bunda1 (24-10-2019 20:13:59)
Отсутствует
bunda1
1-й вариант работает. Спасибо!
// Очистить панель адреса прокруткой колёсиком мыши на панели адреса (()=>{ document.getElementById("urlbar").onwheel =e=> { e.target.value = ''; }; })(); // Очистить панель поиска прокруткой колёсиком мыши на панели поиска (()=>{ document.getElementById("searchbar").onwheel =e=> { e.target.value = ''; }; })();
Отсутствует
А так работает? зачем их разделять если надо в обойх.
(()=>{document.getElementById("urlbar","searchbar").onwheel=e=>{e.target.value='';};})();
без (()=>{ })();
document.getElementById("urlbar","searchbar").onwheel=e=>{e.target.value='';};
Отредактировано func4ptch4 (25-10-2019 14:18:47)
Отсутствует
Где это расширение взять для FireFox Quantum ?
Отсутствует
Нет так не работает, но вот должно работать:
Нет так не работает, код document.getElementById("urlbar","searchbar") отдает только первый элемент, но вот так должно работать:
["urlbar", "searchbar"].forEach(el=> document.getElementById(el).onwheel=e=> e.target.value = "");
Отсутствует
Вопрос такой нельзя ли все упростить и второе как скрыть некоторые из этих пунктов
Так достаточно просто?
(lst => { addEventListener("shown", lst, true, gBrowser.tabpanels || 1); addDestructor(() => lst.item && lst.item.remove(lst.item = null)); })({ //------------------------------------------------------------------ "Копировать имя"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.name); }, //------------------------------------------------------------------ "Копировать ID"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.id); }, //------------------------------------------------------------------ "Копировать версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; gClipboard.write(addon.version); }, //------------------------------------------------------------------ url: "chrome://mozapps/content/extensions/aboutaddons.html", handleEvent(e) { if (e.target.baseURI != this.url) return; var item = this.getItem(e.target.ownerDocument); var type = e.target.closest("addon-list").getAttribute("type"); for(var child of item.children) { var res = this[child.textContent](null, type); child.hidden = res && res.includes(type); } e.target.contains(item) || requestAnimationFrame( () => e.target.prepend(item) ); }, item: null, getItem(doc) { if (this.item) { if (this.item.ownerDocument == doc) return this.item; this.item.remove(); } var item = doc.createElement("div"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this[e.target.textContent]( e.target.closest("addon-card").addon ); } for(var lab of this.labels) item.appendChild( doc.createElement("panel-item") ).append(lab); return this.item = item; }, get labels() { delete this.labels; return this.labels = Object.entries(this).map( ([key, val]) => String(val).startsWith('"') && key ).filter(Boolean); } });
Отсутствует
Dumby
Так достаточно просто?
да все отлично, а можно еще условие туда засунуть item.hidden если аddon.homepageURL отсутствует (т.е. если нет домашней страницы - пункт не показывать.)
//Контекстного меню на странице about:addons для addons.................................................................................................................. (lst => { addEventListener("shown", lst, true, gBrowser.tabpanels || 1); addDestructor(() => lst.item && lst.item.remove(lst.item = null)); })({ //------------------------------------------------------------------ "Копировать имя"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.name); }, //------------------------------------------------------------------ "Копировать ID"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.id); }, //------------------------------------------------------------------ "Копировать версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; gClipboard.write(addon.version); }, //------------------------------------------------------------------ "Копировать имя и версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; copyToClipboard(addon.name + " " + addon.version); }, //------------------------------------------------------------------ "Домашняя страница"(addon, hideOn) { if (hideOn) return ["custombuttons","theme"]; var url = addon.homepageURL; if (!url) { if (addon.reviewURL) { url = addon.reviewURL.replace(/\/reviews\/.*$/, "/"); } else { url = "https://addons.mozilla.org/search/?q=" + encodeURIComponent(addon.name); } } openURL(url); }, //------------------------------------------------------------------ "Папка установки"(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); } } }, //------------------------------------------------------------------ "Файл установки"(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 type = e.target.closest("addon-list").getAttribute("type"); for(var child of item.children) { var res = this[child.textContent](null, type); child.hidden = res && res.includes(type); } e.target.contains(item) || requestAnimationFrame( () => e.target.prepend(item) ); }, item: null, getItem(doc) { if (this.item) { if (this.item.ownerDocument == doc) return this.item; this.item.remove(); } var item = doc.createElement("div"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this[e.target.textContent]( e.target.closest("addon-card").addon ); } for(var lab of this.labels) item.appendChild( doc.createElement("panel-item") ).append(lab); return this.item = item; }, get labels() { delete this.labels; return this.labels = Object.entries(this).map( ([key, val]) => String(val).startsWith('"') && key ).filter(Boolean); } });
/*CODE*/ var hid = false; //false - скрыть скрытые адоны, true - показать скрытые адоны var br = gBrowser; var img2 = ""; var img1 = ""; var trigger, trigger1, dialog, drives, count; var icon1 = ""; var icon2 = ""; var icon3 = ""; var icon4 = ""; var icon5 = ""; function jarLoaded(e) { var val = gURLBar.value; if(val.search(/(file|jar):/) == 0 && val.search(/\.(ja|jar|xpi|zip)$/i) > -1) loadURI("jar:" + val + "!/"); } function jarClicked(e) { if(e.button == 0) { var tabUrl = gURLBar.value; if(tabUrl.search(/(file|jar):/) == 0 && tabUrl.search(/\/$/) > -1 || tabUrl.search(/view-source:/) == 0) { var node = e.target.href; if(node && node.search(/\.(exe|dll|sqlite|sqlite-shm|sqlite-wal)$/i) == -1) { if(tabUrl.search(/view-source:/) == 0 && node.search(/view-source:/) == 0 || node.search(/\.(ja|jar|xpi|zip)$/i) > -1) { e.preventDefault(); e.stopPropagation(); br.selectedTab = br.addTab(node); } else if(node.search(/\/$/) == -1) { e.preventDefault(); e.stopPropagation(); br.selectedTab = br.addTab("view-source:" + un(node)); }}}}} function intf(drives, count) { var data = '<?xml version="1.0"?>'; data += '<?xml-stylesheet href="chrome://global/skin/" type="text/css"?>'; data += '<window title="Firefox Explorer" onload="self.load()" xmlns="' + xulns + '">'; data += '<keyset>'; data += '<key keycode="VK_ESCAPE" oncommand="close()"/>'; data += '</keyset>'; data += '<vbox flex="1">'; data += '<richlistbox id="listbox" flex="1" >'; data += '<listcols>'; data += '<listcol/>'; data += '<listcol flex="1" />'; data += '</listcols>'; data += '</richlistbox>'; data += '<hbox>'; data += '<button image="' + icon2 + '" label=" Add-on" oncommand="self.addon(event)" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + img2 + '" label=" 1 omni.ja" oncommand="self.omni(event)" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + img2 + '" label=" 2 omni.ja" oncommand="self.omni1(event)" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + icon4 + '" label=" Firefox" oncommand="self.folder(event,' + "'GreD'" + ')" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button image="' + icon3 + '" label=" Профиль" oncommand="self.folder(event,' + "'ProfD'" + ')" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '<button label="Закрыть" oncommand="self.close()"/>'; data += '</hbox>'; data += '<hbox>'; for(i = 0; i < count; i++) data += '<button image="' + icon1 + '" label=" ' + drives[i] + '" oncommand="self.mydrives(event,' + "'" + drives[i] + "'" + ')" tooltiptext="Ctrl+ = Без закрытия диалога"/>'; data += '</hbox>'; data += '</vbox>'; data += '</window>'; return data.replace(/self/g, "opener.document.getElementById("" + self.id + "")"); } this.load = function() { if(!("AddonManager" in window)) Components.utils.import("resource://gre/modules/AddonManager.jsm"); if(!("Services" in window)) Components.utils.import("resource://gre/modules/Services.jsm"); var then, promise = AddonManager.getAddonsByTypes(["extension"], then = function(addons) { var list = new Array(); addons.forEach( function(addon) { if (addon.hidden == false || addon.hidden == hid) { list.push(addon); } }); function key(addon) { return addon.name.toLowerCase(); } list.sort( function( a, b){ var ka = key(a); var kb = key(b); return ka == kb ? 0 : ka < kb ? -1 : 1; }); var {document} = dialog; for(var addon of list){ var item = document.createXULElement("richlistitem"); item.setAttribute("selected", "false"); var cell1 = document.createXULElement("vbox"); cell1.setAttribute("class", "icon-container"); item.appendChild(cell1); var cell2 = document.createXULElement("image"); cell2.setAttribute("anoid", "icon"); cell2.setAttribute("class", "icon"); cell2.setAttribute("src", addon.iconURL); if (addon.iconURL == null) cell2.setAttribute("src", icon5); cell2.style.height="16px"; cell2.style.width="16px"; cell1.appendChild(cell2); var cell = document.createXULElement("label"); cell.setAttribute("value", addon.name); cell.style.height="16px"; item.appendChild(cell); var cell = document.createXULElement("label"); cell.setAttribute("value", addon.version); item.appendChild(cell); var box = dialog.document.getElementById("listbox"); box.appendChild(item).addon = addon; box.focus(); } }); promise && typeof promise.then == "function" && promise.then(then, Components.utils.reportError); // Firefox 61+ } this.addon = e => { var item1 = dialog.document.querySelector("richlistitem[selected]"); var uri = item1.addon.getResourceURI(); if (uri.schemeIs("jar")) uri = uri.spec else { var file = uri.QueryInterface(Ci.nsIFileURL).file; uri = file.isDirectory() ? uri.spec : "jar:" + uri.spec + "!/"; } if (!e.ctrlKey) { dialog.close(); dialog = null; } br.selectedTab = br.addTrustedTab(uri); } this.omni1 = function(e) { var profileDir = Components.classes["@mozilla.org/file/directory_service;1"] .getService(Components.interfaces.nsIProperties) .get("GreD", Components.interfaces.nsIFile) .path; var file = profileDir + "\\browser\\omni.ja"; var vert="jar:" + "file:///" + file + "!/"; br.selectedTab = br.addTrustedTab(vert); if(!e.ctrlKey) dialog.close(); } this.omni = function(e) { var file = Services.dirsvc.get("GreD", Ci.nsIFile); file.append("omni.ja"); var uri = Services.io.newFileURI(file); br.selectedTab = br.addTrustedTab("jar:" + uri.spec + "!/"); if(!e.ctrlKey) dialog.close(); } this.folder = function(e, shortcut) { var uri = Services.io.newFileURI(Services.dirsvc.get(shortcut, Ci.nsIFile)); br.selectedTab = br.addTrustedTab(uri.spec); if(!e.ctrlKey) dialog.close(); } this.mydrives = function(e, letter) { br.selectedTab = br.addTrustedTab("file:///" + letter + "/"); if(!e.ctrlKey) dialog.close(); } this.close = function() { dialog.close(); } Cu.import("resource://gre/modules/FileUtils.jsm"); var root = new FileUtils.File("\\\\."); var drivesEnum = root.directoryEntries; drives = []; while (drivesEnum.hasMoreElements()) { drives.push(drivesEnum.getNext().QueryInterface(Ci.nsIFile).path); } count = drives.length; //var url = "data:application/vnd.mozilla.xul+xml;text/plain," + encodeURIComponent(intf(drives, count)); var url = "data:application/vnd.mozilla.xul+xml;text/plain," + encodeURIComponent(intf(drives, count)); if (parseInt(Services.appinfo.platformVersion) >= 69 && Services.appinfo.browserTabsRemoteAutostart) { var chromeURL = `chrome://custombuttons/content/cbdialog${Date.now()}.xul`; Cc["@mozilla.org/addons/addon-manager-startup;1"].getService(Ci.amIAddonManagerStartup).registerChrome( Services.io.newFileURI(Services.dirsvc.get("ProfD", Ci.nsIFile)), [["override", chromeURL, url]] ); url = chromeURL; } var feature = "chrome,centerscreen,width=580,height=410,alwaysRaised"; dialog = window.openDialog(url, "", feature);
Отредактировано Andrey_Krropotkin (25-10-2019 16:33:40)
Отсутствует
а можно еще условие туда засунуть item.hidden если аddon.homepageURL отсутствует (т.е. если нет домашней страницы - пункт не показывать.)
Можно, наверно
(lst => { addEventListener("shown", lst, true, gBrowser.tabpanels || 1); addDestructor(() => lst.item && lst.item.remove(lst.item = null)); })({ //------------------------------------------------------------------ "Копировать имя"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.name); }, //------------------------------------------------------------------ "Копировать ID"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.id); }, //------------------------------------------------------------------ "Копировать версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; gClipboard.write(addon.version); }, //------------------------------------------------------------------ "Домашняя страница"(addon, hideOn) { if (hideOn) return !addon.homepageURL; alert(addon.homepageURL); }, //------------------------------------------------------------------ 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 = e.target.closest("addon-card").addon; for(var child of item.children) { var res = this[child.textContent](child.addon = addon, true); child.hidden = Array.isArray(res) ? res.includes(addon.type) : res; } e.target.contains(item) || requestAnimationFrame( () => e.target.prepend(item) ); }, item: null, getItem(doc) { if (this.item) { if (this.item.ownerDocument == doc) return this.item; this.item.remove(); } var item = doc.createElement("div"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this[e.target.textContent](e.target.addon); } for(var lab of this.labels) item.appendChild( doc.createElement("panel-item") ).append(lab); return this.item = item; }, get labels() { delete this.labels; return this.labels = Object.keys(this).filter( key => String(this[key]).startsWith('"') ); } });
начиная с третьего и далее открывается как пункт 2
Вот так, вроде, помогает
box.appendChild(item).addon = addon; //box.focus(); } box.focus(); }); promise && typeof promise.then == "function" && promise.then(then, Components.utils.reportError); // Firefox 61+ } this.addon = e => { //var item1 = dialog.document.querySelector("richlistitem[selected]"); var item1 = dialog.document.querySelector("richlistitem[current]");
Так в 71b CB работает или нет?
Разумеется нет. Интересно, в эти последние времена,
такое вообще бывало, чтобы после прохождения Nightly-цикла,
CB, без обновления, продолжал бы работать на новорожденной бете?
Как обычно, в Merge Date, подготовил новую версию.
Открываю anonfile.com и вижу, что он переметнулся.
Другой сервис, подобный такому, как этот (был), мне не известен.
Таким образом, я иду лесом, выложить некуда.
Отредактировано Dumby (25-10-2019 19:39:24)
Отсутствует
Dumby
А, это?https://www.upload.ee/
Хотя, смысл? Судя по 70, эта такая же лажа будет...
Отсутствует
Другой сервис, подобный такому, как этот (был), мне не известен.
Таким образом, я иду лесом, выложить некуда.
Может выложи без трекера на torrent тут, а народ будет сидировать.
Отсутствует
Dumby спасибо.
Дополнительные пункты контекстного меню на странице about:addons для аддонов, плагинов, тем, CB
/*Initialization Code*/ //Дополнительные пункты контекстного меню на странице about:addons для аддонов, плагинов, тем, CB.................................................................................................................. (lst => { addEventListener("shown", lst, true, gBrowser.tabpanels || 1); addDestructor(() => lst.item && lst.item.remove(lst.item = null)); })({ //------------------------------------------------------------------ "Копировать имя"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.name); }, //------------------------------------------------------------------ "Копировать ID"(addon, hideOn) { if (hideOn) return false; gClipboard.write(addon.id); }, //------------------------------------------------------------------ "Копировать версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; gClipboard.write(addon.version); }, //------------------------------------------------------------------ "Копировать имя и версию"(addon, hideOn) { if (hideOn) return ["custombuttons"]; copyToClipboard(addon.name + " " + addon.version); }, //------------------------------------------------------------------ "Домашняя страница"(addon, hideOn) { if (hideOn) return !addon.homepageURL; var url = addon.homepageURL; if (!url) { if (addon.reviewURL) { url = addon.reviewURL.replace(/\/reviews\/.*$/, "/"); } } openURL(url); }, //------------------------------------------------------------------ "Поиск на АМО"(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); }, //------------------------------------------------------------------ "Папка установки"(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); } } }, //------------------------------------------------------------------ "Файл установки"(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 = e.target.closest("addon-card").addon; for(var child of item.children) { var res = this[child.textContent](child.addon = addon, true); child.hidden = Array.isArray(res) ? res.includes(addon.type) : res; } e.target.contains(item) || requestAnimationFrame( () => e.target.prepend(item) ); }, item: null, getItem(doc) { if (this.item) { if (this.item.ownerDocument == doc) return this.item; this.item.remove(); } var item = doc.createElement("div"); item.onclick = e => { e.stopPropagation(); item.parentNode.hide(); this[e.target.textContent](e.target.addon); } for(var lab of this.labels) item.appendChild( doc.createElement("panel-item") ).append(lab); return this.item = item; }, get labels() { delete this.labels; return this.labels = Object.keys(this).filter( key => String(this[key]).startsWith('"') ); } });
Отсутствует
А, это?https://www.upload.ee/
ERROR: Javascript must be allowed.
Может выложи без трекера на torrent
Да, звучит логично, но полностью противоречит моей вздорной позиции.
Вобщем, решил выложить кодом.
Требуется: наличие на диске xpi-файла CB 0.0.7.0.0.6
(в смысле один из этих).
xpi должен быть расположен в хорошей папке, такой,
где лисе будут позволительны файловые операции.
Как задумано: запускаем код, откроется диалог выбора файла,
выбираем xpi, и код запишет рядом новую версию CB 0.0.7.0.0.7
Ну, надеюсь, запишет.
(async re => { var gzip = "H4sIAAAAAAAACs19+3LbRtLv31SV3wHhtxWTG4ri/SJFybFlZeNdx/ax5M13ynG8EAlJiEmAS4CWvY6eZmvPg/jJTvevZwYzAHiRk3x1qiSQBHpmenp6evo2g8n1Mp4HB5NVksbzi1WaxlFyMImjNIhS+pwGwTRM42VzkiSf/vPp/+7R30e+3O79+WCPvrTb+N3u0h3+HMjPEV/GdOm0+NKWp50hnnb4Qbcl97pd3Ov2+TKQe70e7vXw8+DP9LXf4vbos4Mn/R5fuEh/wJchX7jN/lhqGAjcgOEGfbpHHww0GOH+kLEaKqyG0tpQNT4c4+dIPR3J05F6OpKnY3r66d+TTcQTwnWavwjhuHvcp73a5SqapGEcebUwvkjq3kfv4MC7TtNFcnhwcLG6+lc4a878g3a/2+n223uV8FJDLoN0tYyO9irv/KU3S7xj7yaMpvHNyzScJU1vFvvTs+sgSF8mYXT18sXjs3RJXxqeXwB98PL8+2cv3px9f3p6TvVRXbWq9IZwKO/PWTC7/D6YLYLlQRq8Ty/i9/j0l4HP3FHlZupHe3sV6s3D1ZXX7re73U7f2/deBPP4XeCpUt4FIUJoebXvwmVwSTeG7a/qKOYSYeY357F8xsurg+Q6vnlDT5qTq/DbcHqs6hf6VKPk8dk14fI8nLwNllUvjLyT0FBsDYW77U6v08kh8tnUmK9mRNwwChxy3NZrVX86fTydBc8ukmD5TmEX+e/CK584hIF4trRkNtGlIxOKLsy+7b7MK7oMrakl04ounb1KpVLjbwzdYejOQCYbXUbWfKMLsdNtA7zY25MJR5ehvtljmJ6B6XGFvb7MRLpwhT2usDeWKUmXtkxKunR1MWdKqnuDtkxK/ZMrHQxkVqp7mJO4NzRww75MTP1gJPOTpmFLg4y6Mknp0te4jLjxkWl8zI2P+cGYgccMPGbgMVc6ZuAx1zweYxwwEC2MRAtD0cJYtDAYLWK4yq1uqN0yjbRl7GTwZPRk+GT8ZABlBGUIIR7bHdORdgeFMIptDGMb49jGQLYxkm0MZRtjSdyLqyFWGzK0jTFtd1Gom6HXA3QP6PWEuXrZQ0BjbNsY3DZGt43hbUtX+yjUF5ZESxjoNka6Denb7qPsIOvTAIUghtsDYeRBDgTj3h6ipSFaGqLQEIXAAe2hsD9aAhO0wQXtEcqOUHaEsmCHNvihDYZoj1CWWELGrT1CyXELjYMr2mCLNviiDcZogzPaYI32WGacrGZYzsAcHTBHh5iDK1a96bQw61qYdi0UacsKiCLgjg64owPu6IA7OuCODrijA+7oyOIpq6ea5yhLTIIPFJN5LhO9Y4a6A8bodAHfRVtdtAX+6IA/OuCPThdtddEWJn8HbEJSUVGrAxnQgRDokBSwuwoW6YBFOmCRDlik0xc5pCdKBwzSAYN0wCCdAUpCJnQGKAk+6YBPOgMRYQPT0ACFwCcd8EkHfNIBn3TAJx3wSWcogg8Ngk864JMO+KQDPumATzqjroIh4cl9kstteY8gQDpjQxRwitQPRpHqx3JjbEQsGKQL6dGF9HAejnCFaG6b2dAFb3TBG13wRrc9yB4CGkzRBVN0wRRdSI4uSY5KBboU+KLb0SzflQUAUgPd6mZCoyuKlywEYIpuVyYYAMEUXTBFtyd30BbYogu26GJx6EKEdCFCuhAhXfAHNBhNXN0mBEcXfNEFX3TBF13wRRd80QVfdMEXXfBFdyAq4kANGX9HWbBHF+zRBXt0BXd5Cvbogj26Q1nu0O5Qno7ztYE9umCPLsRIF2KkCzHSxcqiIdHuGO2O29l9SJMueKQLadIFk+inI5sUPfBIT1aYoXzHgtvCigtp0ssWmR7EiAIBq/TAKr223EEhiJEexEgPHNMDx/Q6bcUPvY5Zq3udbCh6WGJ6YJYemKWX6Qs9iJAeuKUHbumBW3rglh5ESE/0B3BLD9zSA7f0RJHo9Z2eg0l6YJJeP5P+PUy5Xl90DzQHdukJjmCXXt/Mox74pAc+6YFPegPRWAbyGPBgkR5YpDcEQ0JrrAXvSJdrevEyvAojf3buL6+C1Pv1V089SHGj7h0fe5OLNKqjfNcMZg9s1RuKdgT8hK1Khxrc1AM39cBNvVG27PbATT1wUw/c1MPa1AM39cBNPXBTD9zUw9rUG4s6JgISRhHWpj7Yqo+1Sc09/DbKVr8t2huAsSqJEtfuZRDQ4MBGfdFVhgInKp987+KKhjuifkH5U6KH/m6PUFlXVESAg5P64KQ+OKkPTuqDk/rgpD44qQ9V9AjfULIHpbUPgdOHwOkT8fhDlFHRRqUn/a48QDOikvYNVvxrIDyXcV4f7NMfiFmJErAZ+4OR6Q1KDqVqDH5/KNouADH4fSw5fawJfVpy9rbYiWQaLWf+B8tO3PS37ZYaveIv98EGSExc96ncWlNP7u+3Pr/Tn8gz/XHH9vQTrJl/1F8etbXjswPOljDZQIscACTEnrZrttSxEYmyBrd2d31zFQe2sjti5QO+M3jWiBh2W5syhdej2BXzDor3nWi6frDvQIOSGspR+S3Ni7rzW/5Ep9mxb0VGUlpaGRp8szA4DgmyJ7Y6Zys1pm7RbvK488LooCDqz1YsPksEb5IS5Uybrzo//vnnsq7noGzalnRmE/rZ11G/iJ7djig9Gfdq+aQG+bZ0lkkFtkqyroPlzLHz95Jul1B7C/FNb+4k0Uolp9KfPuevUKP+G0A7zc2xwTDPzrs2sm7tzQzrox1rG3bEI1jK4Lq2+q61yZ89cddiXobkUHyUZWQZQpsdQpvNITaEITKEdjzsyHc4C6AdSweHcNjYnR3C2iptdAjVdKjA/pB2uTkhRdbJz+EFV40qASgfVhMuGLYGo7a3751Gl/FyEngPkiRYpqfv/NnTWEIaZx+SNJg/X4bRJFz4My+OvCiO9h8FF1TBw1U4myb5qELlM+IKQGRP2YgcWXg8JUX95NqPomAmoYV4vogjupk0Q9Lil5f+JNgUySHTYjQuomb63hsMx8MW9f1llKwWi3iZepNlnCTeNJ78ZRmvFp4/jRcSNCqrRLd5feV0TX3fnxCmS392sAzeHUxGfmfSGfrtdmc06Y1arUn7MricjC/9y2lr0Gn1gt4omHR8YyRXF/4yCb4jC+bMvwwkpAQiPHr2w3N+tGx6i2WcxumHRSAYVThw6JHNTB/NZbCYEXlq1dvbasOrptdh0syqRHXHuZumnSMqA/4ewUE0YrexTNvRQHvjRgOxWfkbnHEw2kYQcqPB2Ezu0dA4M0YwlUdkwOED8LDdRhJcgO02grtwBH/QSKIMsOAHHbk/2Kv8GTHJEUcaECOE9T4atyVQiCIw3Ecw3EfiK4ThPsKyN2J/oKplDMN9DMN9jDV+DJ/hGM6eMXSnMfxB49ZIohotXFEMJvwYHqExPEJjeITG0EvHWDfHbTg7Eb2EQ2jckbgIirJfGZ8oAEkxhj9o3EGsacyuIJTtGufRuDuQD8B1BQ7enzEsxnFPwi3AB77CMUz3MUz3MfweY7iBxrDfx7Dfx2S/a5r0e6pRGPBjuIDGcAGNwRBjeAzHA+3ZGkMMjpkj1G+UA0eMBzaasPvHEK7jYQHNoQSGhhlqw3EONeILQQ1MMYbaM4ZWOR5JRClzEo7BGGNxEqo7aFOchLgzkLEda6ftWEejJBwl3GEjT/ewWMgQ9yQ6Ba8yfWREbbfaOarSHUVW+jaQj6F8ZCi3Wx0p3VGBsI79SBqTIFXukSDQGecfcf9UBKYlwUf6xISAH0VmId0ayyMJV7UkXtXqqQgc4a26wLEI04VKJV1+UM7vdkuciS0VoJOwVauvmuxnTdIPQVaiVq1By25yoD6k+CBreWBaHkjxgRQXQDU0EshqDdVNoZdqVYJZLdVhCWe1JJ7VkoBWa9Q2tJKAVmuk8B+tx3/cKqFc1g2N/3hgzymNsQp76rinIK46pUKf7ZZEMlsSymxJLFM4rg2Oa7dVFFtioS0JzHE01NKF2xwFFQqqGGhbWK3dactzdk8rgA58xqoch7jU/W7Lut/N4Ls2fDeD79nwvQy+Z8P3Mvh+y0W5r0K5QgGhnIp8qtCnin22hYva4l1sC/lVGFTFQRVLqkjoMGtEIqFtrFPqTk99DtSnNDWUSaICnyryqUKfKvapgp8c/ZRPKSmzlKOfn/59EcdpQnrBwrgeOa3k4YtnP56dvnjz8sWThnf+7NmThw9ePHx5fv7s6ZvHZ29OXp6dP/vhzemT0x9On543vP8+ffrs0bOTM/r28MmbR6cPHtn2yt7vouH8YfqNRp40lR2UHKPjHO2JYqSL1z2oL5x980x8uVShSfKxbteor6s54drwVssZqYu8HjPN6T4VQS1XQfpIQdUIqOHpMnVutkKacY1LRKxgxZfeg+XS/9C8JLy58ubkmnTgp/QsqYsmxnBNvjz15wGHMarv57P9JP0wCxLOFKp6X34JQN0M6bOsdT+kAVgGNS6ZodC8DJcJ6cHUBhJ9tqKjC57OApQHessgUsgxEYFgOFXEEHLQWBA0UcQ0TFRRdTz88HhqCh1JGa5HytQ9jRGayqGEYhkOnr9YBNFUFW1IEWibWIQrlWCWBBrIoJLrUwN9Bya3fKlhFN+vZoZxqB9RcJMxUq1eb8LAOY38i1nw3y+f0NSpSbsi4zFFJaODLxuSgJBft4ckII7j72VJQCK9UKvOAdqr6DlKKFG3/h4sE+bQb469YedIAsx8kQjxHnKEssQ85AnxYq5S9BCiQoRKQnx7WaqQijKOxKos5gll+XvKyYFMIcnbU3lCdGE4nb9XsRP46MI91zl8yBdiW5bLi0qPpdVJFkI+H12M0WAShtSCuicJQ6J6iWK9t2vSkF6qVcZglimkfsoqLZJXrY9qgZJlR5YPWROUtBYpjR+igGGEsxQhMXjKE4TUM5MexNeuuevmCOm7QK6XJeU4GUL8DQVMmhA4spAhJClA2xKEeGbVwc1WZsAnnSRkF8eAt0XLEiVr2Fpb3E4Usls3iUKFgp/cPCFTnyKBqFNgg13ThAQdFj83tHRAfjXjmyhY/mUWX/gzyEx+SovbYFyYhIOxXlnwvL5t9S3OYnTy11+9h3E8C/yoRlg0JeiopBXWl1o1pecX/lIikdV6HZOfM5nwgUya1kDhGkvSImeQCtpUP/tEnsRXYfRglcbsC5kFaXAW+MvJtcq8XJtx2e21JAnKpEghjPQHu4Ek3Uo6NLkOJm+5uiL92noADIy1SM/9DxfBg+n09H2A1Xq5gibApof3MfeQCR9Q8Vr10qdFpFo/uvUmfjq5rgXvqUp0mHv8KNgnmaxbIHSHnXLZbPvJ2j2y0olAJ3FEekXq+ctlfJOQNjWb2em1aez5noy+F8jwlzrHque0YnmEZuwtkaibkCYSeBd+EnCImqrVVTY8oj6NCC9x3o0vcDM/Se02g/cprZdemDarn+d9Q/c0SVYRV/1DEK1mYZIKDxKNiAfX8bZLjWo9T7w75SR/Fv6SlWzjf/LwFMngNv7lGcubM5W5Vu2XzFGGarR5UOtTjANRhBp+9VooUVLYUb44HWEZwn2nddhaRMojAVFFzcUqua69EvBKVQOzY29LwnS2l2C/6n3lcZX0UVW50tWpn/qHPBoHdKNRlQZe07yRb7qhWnWukK5q7S9H4i+/9DLgyYU0WdVqnaFAGbEqWQerCm10zO7PPh68T7MeHwRUxUH1tcaHiZj40ZTZSlS/k1XzTG7UFOHoDjHrs4tfgklq5FjtCqsET8d6w/t4uSQVnYWrv0yZvw8hchQ9dGMnk1fV/2Xz4y/JlMXgVbA8aldfNyfLwE+Dx1GS+jRzaydh8/Ffzx4piHrTn05PaAInNYWw3YfpxZXCXz1s6nI1NakqlZzYMwxDk9MQFZXRpOL18ILsN/9tIMuhdP9FcBmQ/k3IcSHVfiWb3UL3JhN5lTJXUlX61+NIamoyB9ToAYa5J+5FNdiYhcuQCp7ROhZOgqQZxkz952zPTeLZ99S7GXWqSvSOV7TiVJUTm4HOVhdJGqYr9CrHCfN4uiIzSjMi5hU1FCpik+1B5KZZ/tcHL16+eKwqLaJCFKbHteov/vKQZwZBNKnId+EsaCYLoivNki8OtCCrZEzSFP57EVzJRgSqt8AONMA8CfGxP/cjn4aPzD9iqdWCGcTqqaDELOLPHz/gAj8I/JmA15vL4IomXrA8Qbvc2QYLF+k/GDPHD85PklCGP7AGZgv/MRmm6n9PG9fKVKl71xifhyzVg+Wj2OEzUrCYz+ys6RwWtZwOJsjOiOW3CSxHSXJ/NX9hmVVWM/cKZmNOi/gDOuE0pbW0z2gJGc3ieuuICSk2pEkxFxOgI2YkHLsd2YAwFqtzfZ753rYU8yzp2mSY8xX527IbYST27LZU8z07z1xp9x21ycBOMte/dYo5X1EMOX6VbTxhT6dkP5woVtgTkR7O2a+VCZJ8NVpiqPI9YaOPtywaxFK389ozyu+Y1S6pAh1YMx2YtZ2R7AGCwdYZS5a6yWXPUtk7sGYkl70z1lmSQKkLv3AXbuEsoR0fcAzAGdyFLzif0s7XtrgV3KR2NT5duDK6EnTYmNou/hhJZdep7XxFSSu5vdvNrLWuuDDEh7Emyb0LtlyX5D6WjPa+Mvr21qW38xWl+jlc+9oVkk9y1wnnVoq74G9S3LM7ZSnuugGFiN6WsVdMbuePkYSdyvLadUKTLrRnEtqlBqLkXmkuu9BmPFKe6l5L8tKzZPYenCD5ZHZxDIFDem3ZbIaCMAl7iBOwbBPW43R25Wtys9j5SvxbFyhIqx6kFWezS4lcEjtfxVcF3H6XXHa+omxvmEkyCTVLPnuPuEKwEQGVZbDv5Za41RKbOaGiswcNTbM/wsj3La4AJKYPulJap7/zFS0OMLBZ+jtfiX6vgSpiU5zIDqrv6RT2HkRMDyKmBw7qgYN64KAsl137+mRvoJXJnocYY4ewcYuTXjp9KGJRe8ehMUJZvIhZl1k1Gejl+Xcj0o9ExTtZNcp0J2pjrdA9UJ86zfr9alYV3lGKqT+1l0hCAga9bBxlVGp54/1Whu+LbPwUcK3YKeir/rQuKrM0SPcOPDIVc+qkUjSuCLGN9dmoKpcG9Uk1VKtrFwbdM4jRd5OGsQwa3n0vTMqbr/7py/syA7km4d41+w36WBl4v4Gmh4mHqGbzAQ0beRUA4QpkMuIrculbxsPdh6Qo2auABaWPBaUPz2kfvvEsQVAJRFDnaZDyrmdhqZI1+moZHOiVWcHSyjyXpbmp7hBNtnTHCtXYA6PHgyCaULyTH8P0mhCo1pWJBOLVsjHMT4u6tfG7cutBJqCgFFGMrbFKXlFLr49MxeCDzBmQpNTEnHNzLtAZldUkHRB75USUnefxLJx8aJ7/n+enb2hYX0LONONFENWUxbOhfbqvCKcsDO7Y42ixSs+AwHksMbaa4NNQeDX9d34447BIra6tMIKb1+qGfw+++Sn56uuDK7K/v/m6mtke2vO6sIMuJmpjKc9NKxLDpaVAeUimoqeiAiqJEvLAES7+YkHk8pkZDt7PZ0BM7xfZsEuGr+B0LFh92aInya5my0xFTRTEtrmLF2mUNEiC0GU91zhyQM5MQLVYy7L9NHzt25p4HzpOX7Z4m2016pkEb1T0xt1Zw1eUFBc/wBHN6Q9kcw2KFrbYjPUOG75mJbFK9WVbDtThvuxuMntuNFzZjhu+oiTWqj5Wor5ZifqjYTY0Y3W+AlEwoVWJbNAHjx7RCvvo8dmDh09Os5l8SeY4ezD81G/Cwp/NnvvptabyF/zcchaLmQ9oLWlooTKzcpOfQJW13ACbizHEyxdP6hpFLnipSvGgnszZzfNQB/sXwZTs+vAySNInsTBsDcgfqUgclL3+uK8pJMdqjLdtY7IORMi+shzVx4bYOZ+D0aFXjaNp6M/iq2rjHt8btume9sdqLx+bWOHEM368e1QBcll5zWDE1G9RI3ep2lQh64b+MTI/yOiiQlEcL+CmaDB8Mb76yT1kgYFo8uJjW/yUFWfGFnYAl3DCqBVpcbcTFxBJVSWKJ6GoB8WYqn6gDmBQUVX2WssDJ6qKA1LsoxjExEAINjuKgYtJqiT2HNoagB4MneLApD26B1rp4Cu+I2dtW+iVIdcf1nBvT8KvirAqVUkUitIw7Cec2CBDrhhEeUnu7X1m1EAV/JyowT0lTUCmdyoEdMyBKXZrf7HulJP6xqDBvfx4aOzVcOi5lRsencqQQdc+WqvoIQdgbuve8TdCqorJgcEJNAl7cxvZpKsf5YAmy3CRlgDZA+IKgdJx+Z1CYarSPy4U9plMwd3bwhQ51+BOMbF6gSdy8racFQpAn8MRbiWbGaME1uYPzbkWZ+TryVGHgzZZSIgKkJi45Vr/rdb05nJ6aS1Y1kel8nUwP1T0/6bVbDWHTVy/PrDu7+X3O+xt2v2QXxi3g6yrqqf2rPXUbirZVv//z03Rwf4H299Ku3+rKEhORWEOQx4kfLQ6DH/sVa0Rr6rVSy723701n2sByp7t9kd6xcZW9JaifAF326Km2J5SVOy6bu/dEaXyv4pM2zUPbzc9dSE3wpUhu+2Etgm7hWbBM3cD/noEGtYPaPnpkpWaoFlyfkRTjo4o2azW7mzqxyeLSXeBLEdPI6m3EKpbuxxYxw4y0AHk/PpbN1/Uu6ZF8/gOsX4twhHn9zh99riqw/zVb7+599tb0AJ9bQvE2uiM8D8y/fY41zJLtfy045lrn8xxhjAJYBkoewDfy09cg0lQAR4bDAIArMmqlGdbj19DJ/ulNdjFM1NA3NS455oCaxIsy02BNYmW6mi2nL2iMy131fc/43y2TO9XWyey+zLgalfD7ie1SWYlypZmY36644FtkpEJZYa5YfO0dGOO1+l8tk1Slf7dK37dJni15jmZkbnB6TXiY6p7NLO8r46huN3/KZXkgom/nL6Sr+H0Z9e7DI1XelV97S38KJjth2kw30+Chb/kQxAbP0X3va9+e42vfPhjj4P3dG/62vvo1PtTOg2TBcn6Q04YDI5+in5KbwkCS5CclYeZpo5BvFcyi5RZfk+m4T2LwfG9/FxSGNvYaiuVajtcGeH3rGmnLFc97bRBrqZdpS4/7WmnjW932mUJzqjKOhHx3mdMu3vOlNNWdn66DXX1eStb3zaT7Z4z0eSXPg7xnppgulBuft1Tc+teybxCETOtkOasa5FDzUye86c1ByLy1bhBsnxn/Co9ElGD5hKeM0xkP5BsB5LdQLkzEXUNpdnOn9acjKgdHLsfjIiihgGyUxH5OtJDzUnPeGwnPGdDseZUxAzCOhNR39DZxJ+ssxEF+exkRNKmwiS8CGdhSlNzQmaqv0gCZZ99sg5G1L/10Yn8+x/QIHd2TpadXaulqe1dtHQEWInCRXt3UhC40L/nyttKUjuOPv3nI6Rupaosxuqha1o0SuR0mRg3DFb+p/hjqwpsEcY1tnM/seTUJLdoqU18VoFxUNo5KV0IYFcvY2qHcyMvZqslfy7ixWrBng3eUMXmdzwngkz569vgw2IZSPpnyKGh6usjVSs3HLDZVSMVu8G2WOZXQC52kF7HHLelJ963Hh+we8qIPEE59owRWcWF496HtyHDmaM+p/7kulZjvbHhhZE0RK2+kjZeqyeq6wDxvvY6de0+qOTbrlVXEXsgqFu1PNYmjdkEpv65CpYfaBSCCTsy7r8qBmZ1qX1Wczk8ta/cIa/vi/PEVPvll6aJJi10BBVRtcH0xJ/NLvzJ25rg3PA+xtEkUPmkdRizEht8Gy5+iP+FMN1DuPyqV/FZkJ7ImElwbAqvoxwmzdXlC5Hxk3NBTYPLMApqjKrx6OzP43/tY9D32SMFN/2EM1GV0yzJVwI/lluo7n1kC456AdrRz2S1QFTvFu6hykdVF/HCNRfg0bK6y+nPD/0kOFGuvWOP+mEatJ78EL4PI+BvA2Tf5fl/v3zCNap7ddVWpZwYu45xw3OpkscZvJWjArQdTYrMu/YGlPsuDGZT7d97k4DxguljUp04ZLqazVCCKVVJUp9mv8cWbRhdB0vSr6YP0nQZEtZBYlpSbkT5Uak2i31h2h96VZU53SDWBCM1OGwZR7MPDUgN4qyGrmMNdZgoXFOynByHc/8qaNC3rFQJFSFWqMQ7f7YKjmf+RTDL2icYYuPgvUGkpKrpMl7M/SV72zmutAiirLxsHFTIH+tOoIpbQ8WSWWj7R0kCYn9hs8kMRTyk90oi1Ey/z+Pvlv4V9oMKcv/4ml3bZDRHySHZ6sdVdiqTPnxzc+O4k0m8zn1O1z248tPgbRBg9aNxDJoIyswkF+YbqbXyNf045GkiHHdcLR1HHfqwpqF3OQveH1fbVc+fhVcRia+Aj0SpklY+eZv9ug7Cq+v0uNrpVb0/KWapFKQHyXKW3fdDGwF32t9XZW815oI6+EHjvol/DtxyCaEZ8IataXp9XO3bj9nWOkTD62kiHGbRgAwLzu1R92nM4xusN1Sa5W1W/9cHmuLOEGQMV2jU4kXl5+BHpsavD7iuf+BH3dqrm215taRBGIVpSAP2r8BM6ccyybFRQIlJlVLBa7iWGdwkbkhl2HvLPzVXowSmm7Wl2bRRk5mo3f6yLIasO6ASa6Nv8vCDVQozl6QhSuvC2MfB/v4gukqvOSbSqitS7lKnYJKrkxEKG4ST+k0KglfjeHzryAtp4c/ao99ffWV6bZBhK7QW1t1N10w0flDN4AlFzpDPShzpBxckjt7qX7fyeZv1+It5mPWyvJPn/hU3XbPafdV6rarEmOQEP1VE67YR/6o50cIQ+2lAWahni0O5fqF6ZxU0RfhDlhK9Chlw9Cq/RtUtbEuWLldvqr5S8w6ArzV72REuq46sF4yQYej8ilbWziH1ehF432Slqk4PZb5Pz5DxUaPVKVcnNmi8T+ZJM5o3l6u8Vo4s/nAS8NYM6X9uawYHgOX4hKTevJF0jr+eyU6aZrHxPPHtYS/trqMRWKpAvqxsjTSjV6ZNHHsAUqlrljiJgpu/K/lQpekni7LMwkymWA1UtDShz43CBDWVwon40HAWAlKEeN9q/jaTkklJaw3TgXqGrtRDCjiKee5I4GHdZmjPrtirVaHGCDKqUs7y21qgWnVFumbw5rt896pVWxPMzX6CPjKjg8XKtlBkCxlMnFoVH4luVk7J5qVEPZaKqyItbJkhDbP3jRNpBRql6+5kBOKlnGnjLIT5dsPyIhCHXilhXL6WJnMMLeLIHUYGKePnTcvjZ6xvFleVL253XNvqGSbWwrbrunbXZc1Z1TxrWbNXtVtrnkkD9eIgo55sMnhFqUCiBM3lWDVTou0FUM/9ebhmVmekKgEqjk8JEFRQF0hGSSv5VKShBs4+SmX9uhxpChh7gq33zKw1m2Fhxt/Wa0wbcVWoOWblRbwKmqw1vm5OiK9rOkLZnMziJEhSe2dtwwuUq4AusGtMRdnkuPaTrO/3lfVzv27PIt2G98Vxfiba0tCan9KYLTNcEhtL0RNtOhMfwJW9TjU7FUR8QCUVZP2zPVSmmxVDHjHynuJAQ+kFiXmAZGwn/pFHSrohmRitKDYs6b6uY0P/JcVFqnbIX4JaCZZ1m4QOU5nYtG5EO+JyrUzS5exvwQcJZyfX4WVKvywBKHCKWlTHiWqdoMB8nGj95u9/e/PyOVWxHezRsx+f7gT4XQ+JTwGJZ8aIC9RVKeoGQz4KLv3VLK3VGxkR2HRn70nNLEroO2hfs/pdoJjmCAT836deYSytRaV8Ehu1et0yTdVmfMzz2PZoY3u52Xf66T+yiYSXgLIN0jwe//gpsl4y96ePNNosspPbOsfD/vSRg+e3P0W39X+QMFktZ4f2eaZJSpo8Eoh5iz5hw8jgHFAZeLj/lgHeRsfzi+81V1HiXwaPgqXZzF1vckb8h1osMUPeKHwFY8DI8mpK/bvR1UF3IukUX6JePMMaE0O3xYouN83jslaPtqZ8lGYmfPpP03EShBMMOjwPKqHvkHdp1bakJeSc5u/ns//azdEH8b27r17nPBx8+s9Xdw99OC+q+/QfdfuVua08Fa/X0OC3vChQtkbcqa92HXwKwbpIxCxJynbA0/eDX5JkdZEgI3CfXfNyOALPuJx5RcCP/3p2trqQ9MEngK1LQqG+yTPV3T8uW/IN7vIzyyqixUrc5ZA62F3GCbuvvv3229cqj3XUHQ/4zJsfAhLMHo8ti6qsCpokaezlK26UnjWMurhvG5EstuGi6W55g7a7W6BDH8Cm84ZpQpcc42Y7KUz6r1qy1DjfNYKxxtGuay940m1v61386PPHBWfE3BzvkpP188dXUbwMTjiBwgDLve94pectZHxAkFVksoyNMSE+TZ2nLaqA/ulPJrRk8/p8rK273+K1d/y6xmPeQPBr7r8X0yBzfSfhv4LMf4+dWdfxjCZL5lQXBGkdb0Sx8jZKdNBVaDc4yNfomlUjqKpG26yYXHW1ey2jinqmzvHb0cleuV/i/XVplHP1HnyjHNN1R6Hfwc/quhyKs0zHVE0IMm9WF4uo6OsdSsxjVljjmygrtrkAkoJcYKMm0SwyIaaJTxrxNJszf3f9PATqjmGJekQwiqTTgE8a216zRc83ZEA8DW6YXXDSC2v5Blmj8LkMFswX6Qec8lO31GiLy8td21YxmxqhmbP07SuvqvgnM8tfqc0Fr8uPvHQtewXLPhap0vIXqGd1T33JGU/5ohmlgaKpyqhod0CGCiv/wRrLTjWxowdZttxZIjbnvZVj8pQlJZDCLo7buJRDWOaWqu2uYFeSPHtckk6ww9wsLbVtepYW2jJDS8uUTNLdnfEWPWyqlqx/d3bGW3Xs6v+7oxNvU1HL5WkanorJ+Pdd2rdhXTScWrZjY4OXIQWvUyk2pe5tttTzksrFTyq0ECtxsTiuLr2Vv4Ca1cQ2cq3Fxq5jO7Fc8VtACML3nIRvKTprcciKrcFgS7OsIO0wQgxmOavQNIrm/M74WTIoKG97FiGjy11cqq21Q0fq3BNI6m0DZwBdvLPy2wfNwJbOOaVNbp1vCi4313TpPDuXlLSDHJvIrOFz/sVN1DZFNlCcdOLHrBPnOwr18zEpnWUco/Toaj3PNrqyDf3WMLv22zS2K4sZ5NZ3mk2EbSPLMG7vUGo7XzFYGUuxSfKMTJJtDWs4t3FTegNpNcyupM1yi3ZmKVNkA3WxssuSmzyY7bJI5EpU8VZbQcmlQr7qHUVUvv7d+1vAbH231YbPEmVFTHBHUTl19jVbYyi1lKglp9kDpSUtA+qlm7LlJLcUdAFx8FhoqN1J6dKPEtmdoM5VbE5mgW98CwaRpTFixAXjwS1/ayOb+QzUoDHpaus4Vj92RvllEiyhhNVEbV5btgCZ1yGVDUDqZcG7kK/NAs0JSFqAd1uTMsictEEnibg4rXKr3HGgcxLIrWmDKHAhlUBYLw819Gm0dcWzYddgx7XsgttpNF2Pmar1TAO/8KOroOafOR1reNmNU06QlhYPDrwHk0m81LvdzTs6eIcSH1zavLn205sreGH/axrvR3G6j6BAQ5cvNl33wsRj2eeZk4D4eG/fm4Vz9mRxEXYpil8mRQ63kQQFXqHHIuRgk1tz0uXwu3U/G49S14JlwFpuEFcUR1LI5FEoR4jJHxGRYfwi8Et6UfL4+cwPkUNrJJukBOXuK4GCED9OoeIYSxHMzV07+DlQfUkOaJYlae195l3LUCRyvgiuTt8vmkvO5zyRnN5mGj+Jb4LliZ/wyU1ZMUcE/m82Ex/r9+XVSjAyxSrNyCUsW+QF+FfvdVJbJZc2pxPm3Py5zOn4Bseryrjay6eJaFa0DWw5cPXw5NYrHcvNAsIs5skWV77l47XR360Spbh02fMaddhsreicZN2yJ7qKLa8RGHbkHSSyY+kFaHlogH+P9AMTBijLPngnx6pxXWt8diawn0vRSW5CWUQRQpZGda8mvtJJDoWIJJK+i5c3/nLqoTocHDJJV/7M+/78hycidcpHzklc2D1zwWFbg45VscLMdNqOHujpsiWskPnzyrg3m6yW4mJrVup0mF9/9ezfHI5YxPw2hmzSbuLBTJ+x1Bk9Oz+PXEr1ysbuGWe334RJ0JAEPC/mG55Of+achqum9/LFE++CT1GTkGOdB5nXZ7SYb+QzM0sYihAiLf4+rVIkKa+Qzu2KjZtrwknYkwWcdYKcCsQlmt2BldxjmSDfSOYutCBRd8JEjub+UUDzeSrbxBJDW/7uHEdpxVSO5tHOSGdqro15rZmqilK2yzEr64ieNbhkrJNbw1sNr2UtrmoeWCLAJPOojJPSh477yWTYFMTuBikqSTzcHYd5CssPCJvP/Ng9DcCcgbAuOB5GWVxMPW7a94hHzq6D2YxY6DxePCFizNTDR+rJkarp3JwHVfpmGfu8JxyjMpkEs7fBh7+pHKPvNW5CjbLFQuXrRH+TEjuvF0LsiinosKa8+uViqlMENuRgY+PM4rNSsI9MU28RmKUGuTR13s52BUsCgHTT07OTB89PswxK4cN8ItNRbhkQG/QLDnJaApyfXJwiwB1Mf5B38NoLgWu/mvhoxUrpcrcaFGvc0lo2MUtrvS10//zBw6wH69osq3BrZ0tpVVbTJopbgcQNbSELeBlchu+dMX+OW87IOwOginzFzKKcJBtQkWfEqovny3jhX8nRjZaoubfjZkM9JddlRugZjkHZcW/heuJsifGjruaaRI9Sa81aSOJI7SnEPihSFcrp52TJFyPZ1j7kNdLqbrtfpEvr01c2Bew+Axderxzyl2/scEBcn4JbenN0yIF13MXljGB8pLsg87tGANfYCZMLtZ355WLqp3YKCVaI+bQ8CI/3TpZ5VQvVWeIHLxmz5Q5V79DTWxNyyOntXMztRRZ1sKvO6aSeE53Iy7NtUdHJhXRJ9a+mtfZ7+SR0LUdUIpbZX507GsDe7q9PB5BMCLxKidU2HACgSlvvzPrt4km9HsW4YxK8qw6qondsryfBVRg9Nzm+1F7245IjHiU5Xpd+QmSMzAoIG4ssubMgSkJWuN1GH5IcYAuzNPdrtWQlk0M3kWn3DfIa1dcAOzbstASDViHzMZiTVshm2sGSO8qAa94JhVOEHdfWZ2aRaT10X/OETiTbPekLa2lpJufC5Pkm+zvsk+PVV92pZzAPiRZkg1bBTXwAR+JuNtf2q54weJkjKXk/8v5gwmq/nalwC2zfvvzeT9hHLkH/qgE2U9MuvoD8eByljJ0DnFVbIrQ2V1oilkorzsqQMtRid5ZzZ79dUrfur6YD3bIe83O9Z7u6zw+qDnPeLenuj9zZ/o6rNru09TZxza6sRvghHxDibg7XSaymYD5H2iQBaoYvVK0fmIrVvnKdEp1PLtT3C1vJcUTecfXmmubiPnatH7IG+aePTPZb7wt5s4EfpUf2HvYDqxu5vvEbrayb1l2221TjLDXod9W7DqcklVS3s0LWuQG57f+aZhasAMumRDmRofqlMV3/uSIrmZtT2NrdWDseGs1/JprOdqHiFntDEelmnk4uSeTmu+yO2mGvTol4tS4RfhaTyRro8xSn6VQJKWdTfmEDmFqUd9uWvwz+uSIr+EEUzmEH0EyaBzUrDfuOwUcb2J7+5ruC26HdnevKcnY53dsJsedca2B94wK81bqMoYbJOeeAT3xpaR4qUl1Q+p1DfX4/xX4rImVKfzku/MFog47PlP5S8/W34u5rreMYzxRvlqme+OyC5GOSaKrAHarhqmsUJNNERhWFtYuKYyTYFbinyHCpB+aFNWUldFjF3GnaZWx66FAxn9K+w2Z9nYeoiryu5qymfybnd6/MSJpCdbncBKjATgZCCavbmQgGuDxcJlqNCpTdoUQ+xuaYY1b7O1hgugN3TcB0yrl2Bn+UBKZ0iR2iUho0F5KqMMue0Eqa1vjbk+AybXgFj2DxYIniIGUKEDTxzWkEAhesgdPhfAmnJqx1JcGOrr8NeQfr4/6666SkUXOHnvGOiUWCI+GN65w4J/57GNw4zi89wgaqluTC40lJcoBDygK+ZXQrLyG9KE+H4PUkw8oVLpnNVI6ELfuSYC2sGS9suhRz0Q7RZ5Csf8OefJDWWpbtMI0nuSLauWCxy0K/LF6dtSCK8tOzmq0B33Sh/LbH4/HBe1Yh2W4mPTDSCyNVk3PWyLrJb1fWx56GESuvR94lKbz7nER3qO27I29BqwQ2+bWO1lcZwlHx5o3p+BtxgS/fvDGJ3kyGZkK2bLyKpiqMkNSoLmt2rqLy4cvoYdNsGcfaEZM/MY8p8IpVwHKc9CF54ltmHMpOcEBshxmN+psFerIRioL3+jF/PQsvOCfBGeVLsmXWjLQaVv2GLGP1KMxUYW0sNZtNbgdRS8YiMWqOYKZ0CKiOGVH1wzCiNTR9GJAmEtRUzQ3g78JF8XIOPTO/2T4bEj2TGsb54KaQgd0TZ+R24faLePph/fBmcBPiHiY6w1vU0KdU2Gam486xJpLKKbI6mDl4clCeA6YdP6VApe2CdEgYY3yz6GMZHKduCRT66NapMNxYnwWT1dVy60EX8rWY9gpQJTgZ5Q8p8HaWjIRk/bfBC/VSNlUl2aazQOmjbMnV4OCqu7FS5k8k3CuFGDWcZwwqFEvjhVu5cktZt21kCuLkN4htkV/6FNqaNYLGd2ejulZV4bPt8sj+wQjZSpc1AJZSZA/aDgTcbcyURGCMHkRTXb+jbGnxooVGXuhw2ec+W5MpJ5JWVRtVpbNl508cWVv3vrBIIW/jy9JUbP7LiXw8L+2usaYWZNgvUtWTulfLYuXfqvSLFySz7FfLeId6+zSOmniEV14px9a2HfPyfqzkgDtN9y7D5RweLH0c5ZuLmR+9FT85V9SYx1N/1lgGtIKze/84ihtTHDFDtR1/CBK1xQ/FS/KvswntkGQtJ1/Zy4KzUu+kZhmVeIuStdZkaPL2ffW+yIQVTnchKnCP0PMcdnV+otjYE02cFM1S+cEn1M91vgJRimo99KwW5LbNhIeqZvXImi6HfOKZ3HVCBIclYQMF53CignPuOQiYfh7mOy4avXTrf4ideQrK99/MzRgEm52LM/9YgJybFqw1CgbUumdHYXLBG4F17lrQzlgYaOduAWM7BmQjbe5n7B0FN8LeNaZmgY3XC1AGJ8K5pSxVuEDAUimZX0S2CAoGVxmUrvjHiUJlGVbGB2OfDeW4MLnOp7z22GhsT9ow+Oje//4IWdt274CP9GXDYBaihDZsJpcuA+fosy1nUGUjfxncsa8u4e3UmdtN7GTVso6Js5VpU5PbWY7kGWnp3E1/Jg3tottkeHtl88BEZzfoL5/tFsDLx7LYslLoldFtninFbxZHrikDzIt9VguBFN9JX6y4M525xh6gNZHro0Kp0nh3icSUs6vSFzbJVBUcgK4VuKhRiNc3NO0aVozepouuf8M4FkZJTGqhMYju2Eh5aH86VaC6KVvCb/ap5Xd5gYcZzDqGrDyvIJ/y6ix9uemcP9oAk8QSV0r7ERTPsU1GbSUxiJ+YvVrNs9Mnpyfnj589ffPg/Pz0KX/bQB6rIM2XR+J5KvEecrN1mwYPC65dhwRmcwwtMeq+LQec266Sd+duPn324ocHT37XPpaOaS6F8PJiy4EKKhhrjlS4vMh55yRIm0+aL2MOK/9zixXLs61A2XVLVT6hJScXfocOF4KDqs/GlFGDvkmtX7v4uds44KQPZu6KuMZFvcW0WTcGBGkDGEGQVxwkOnjuXyQ7qjHFMIaTZZszquWBwmGVsuJdIJq1Tmu3wBq4nVShQul8QHNd9GADrfMnCRm3RfWntLpegbF6U6vKJw5Xpv+s3OZZUiTa79KdElf1C2YbUqjUNjyvVq397H1sNYa3P6X1X/l7uzG6dXNedd8UlmxPvQjyx0evlwDrVjQtC8o3PJcKyzVVKbbaXTY37lDiwdOT75+9ePPi9C/0Sxdk+ShfNWfmdydZfT+Paxw2ebqaX2SRd7uv0N8iWoqfXWovv60iXiwLfhLxaygDtHbw6uefotcHV9aY4FRx06j3DVeiT1gqIzetf1njxpu9dURC+uY/oXZ+IKkqanfDs7Xv25yODh+vhdn+PuHWdiTIhuZmuiU5ILyg5m/RnrYwKopxZ0qGy0pg0+tCeb4cMnZb1WwU+JlK30jjq6tZ8CPd4FaSfDighFl2UyLcUXAPT78THrnFL9+ks67hFF365G1jylYgkUMKxXk4D0hU1GrT8F1DjBA7wUfOPE+ytIPJhcqzbObSG6lyxu95zApAzHpJP4tSoJ6Pabxgb9q783hxy/Zs+A5Zmhw5pOl0MguJm19kxxeoMFP47jkQkILeV4zPn7nsBAW+x7tmDtqtVkljhBSVaagX0uC3wN8ai2s7BlToBHlusoYLEqamg44CZZSEKx5+4D1pWbF91QlDjtuG12o4o2ZFxVDStSV3dd3G5zFmhD0PohUcmpBB7JK8ta2z0tzbLJdYPFt2Am5Z6i3D6ORb5QxTQl68iiTo/hKnsceoNZtNXmpPQReeOowfzZ1DftXCat5AQlFD7xlVoyBdMAfo0M+m/R6XMPmOU+kCflA3poOZoHw3o1FhNuVERpYLS8Nmj1Auo630UTHxTbsSSsVPI4POwsPi4lnwGwMTFYMJp0SyQpbvDjLU6jFn9iM07+Re5xSMWvXng4NSbUIqMcpEDmotR6riko3Int2Ui/9yli7/uMjC7t7+i8A0nAs9eJjAiQU745cPXHBImswjnLf0jITeTyYhYxZ+9ZVy6ei2GToLZszCuogNc8Ili2ahjBxcISRWq6t+IDSTQxqkcJ2TsLV2x7nYSostrVQ90+4ijRqPQA4b6WQEksj9Jlm6YWp3kmF+SWaqgiw6X9EJVqyEY7WTcD7qK4BdPnkcJRqkXYq27p7RfoCwtY2x0FRLPXAq1zd13zSMPizDeieGg2rxbU/WKADyVfjaPvoCkkz1lx4ZXU8xvFH0iph/ZZVSr//Yxx2HbKoJhlLNZPVd4t0e1vuwckRwG9TlimdlZNwhtPglDiMMPjEKZqwr0bIZNQuzMF3BGlM8m40+3uRZYFDiaWasQ7MHodgMv4eaSv/Z4h9uV910e+nGDJ8GNyzsZatNcafw7uJnvcC4s7yQ+aL2/uQlBfWpXW/OYaMc/Fx75f2Uvv5z/duD7M1eRaMX4yQ1al/7ettcCecs1M/hfduK5d9FoyJf4ihnL9iwcnxvNnzZLn1sYOVv7laRcL6QbycINFqprWp3WMN7ZefF6uTk16h13W5YteOAVi5djVKm7nbCua5nhyPOTZMqF8ZserjjIee39Zp+11eShFeR/iU5SjVWlOoNYFm2sV504A+LgDVdkRYYWLrThJsv+TFMr2vVt9XcW1zkRRvyIpB5kPr594AoML2nmxQByYCn7wnx+nU1YwrsZ/wie7kIqWb6dAYu7+DxYMkviKirt31oZ5VAAxezhx47TkQ91Cyq2n/LUwCQORw4oWYa8MlIzKEGYbltwdrdVDNMx84Jr+IrRywli6BfbTzyJmML58ybc//CjJnzGha0b7sCszLfdTcUMWHHwMmjkZL048GaN7+gg66TsOQdK7ZJmqmUdv1/KcHNHVSjomf2il3Bd+UV5IKLa3r3/YbCxaQlp+iPuyNeZpJLPZgC5RS2X9Lj+tkbZaxVUbtv6fIWk4NTu51DlppZ1rt4Eaz+SJEXbKCuK6P8P1ah02TiL4LS9w9tDiNbFpEdmSntl4vhy4WLnhvEz0M/wiE0eXiThGCBQ0rkepJ/rZG3rsXynpd42R1NJadl6LmS29+yBfN76vShjBV3oL/hSx1XM3SQNx7dfUB1dY9VeVDFEqc7BCDKOq+OyMH+78/sYiEiiJN0+Ntnv2Vmf+5P4uS984IcoDYnvMLo0Ost3nNLWc48fm97rY3adii33V9ydA98d6JnuNuSJxcnnAYgb/jyInxYr5YXMHpy/2unVg/hMmRVqQ2rxRfMyye/wAJDWjN1bd22K5D2tsa6is+pdVBNqtrOfZHMb35qJ7fI1laGV026nmOtkH3kN1apN4neslbGVTd9s229LjW7vherjBuiB6ys5vUa16RZKzcyD6ZTecfVefzcJ/02RaBTvX6rdpFGtmx6AeoI/HfEKKoEg5lXOQBwIQ/0pgB+PtFE02mtGp+7qa1UlaOxOqFXTgrlkGedwe78Ch5J29evqcGoOu8okw99oAQvNHmVCG0fqarMIWOllcXRD/xc5P+2quTlUMojCRdZuEjlVVCFtzvpjhCX+ijmnj9ReuyVPee0h2Lr2RTuC5rWvLDsQu6wJSQvaYK6mmmZv+1onZL3usjo8KlA8m0zuP1WCvPdeaNFKQdklMl1Z+MGXOBQ/qKLTVhvfZ3Gzog/CtbgfasZbWFEwLpDZKoPSWshM/VceEZJgOzENLUJHW4+qYvWOS0O3M1J8J0ElwQq87zc4GUcy0V+QzO5e56Lw8zcO+Am6XKCnFGXqImaNfnnsfId/7BKsT/c7BYWBIvdYBuFK5nHlrSp1WXm0k21J7imGiZEsRWIx9GSPSShCfy+g/mrcPrzmqXu9au1q+Dr+/zGQ/fli5/1kj2t/xQ0CLxQj/htuS8upUNsaebu8ud+cu0TJx7S2hWppDUU4HXA57NLAoG/1DtTJ6tlEi+zSnjCXGHn3f4knvETlDc7WQs3VfbzNtXFenOi8V6gb8YNYjp3lzcmmrqK70rMWsQQMy3+Sx+K9Ts0ljsGTdVT34EUG9TGWRitNmiNncX7gubI9/i/69zfl52a+zALtbr5GxC7CaM/Dq3/B9P99stF5AAA"; var s = (path, name, def) => {try { return Services.strings.createBundle(`chrome://${path}.properties`) .GetStringFromName(name); } catch(ex) {return def;}} var picker = Cc["@mozilla.org/filepicker;1"].createInstance(Ci.nsIFilePicker); var title = s("global/locale/commonDialogs", "Select", "Select") + " CB 0.0.7.0.0.6"; picker.init(window, title, picker.modeOpen); var filter = s("mozapps/locale/extensions/extensions", "installFromFile.filterName", "Add-ons"); picker.appendFilter(filter, "*.xpi"); await new Promise(resolve => picker.open(resolve)); var {file} = picker; if (!file) return; var ln = file.leafName; if (!re.test(ln)) return alert("???\n" + ln); var {fileURL} = picker; var xpi = file.parent.clone(); xpi.append(ln = ln.replace("6-", "7-")); file.copyTo(file.parent, ln); var obs = {}, data; var td = new TextDecoder(), te = new TextEncoder(); var scs = Cc["@mozilla.org/streamConverters;1"].getService(Ci.nsIStreamConverterService); var sis = Cc["@mozilla.org/io/string-input-stream;1"].createInstance(Ci.nsIStringInputStream); var sl = Cc["@mozilla.org/network/stream-loader;1"].createInstance(Ci.nsIStreamLoader); sis.data = atob(gzip); obs.onStreamComplete = (a, b, c, d, result) => data = td.decode(new Uint8Array(result)); sl.init(obs); var converter = scs.asyncConvertData("gzip", "uncompressed", sl, null); converter.onStartRequest(null, null); var args = [null, null, sis, 0, sis.data.length]; if (converter.onDataAvailable.length == 4) args.shift(); converter.onDataAvailable(...args); converter.onStopRequest(null, null, null); var zw = Cc['@mozilla.org/zipwriter;1'].createInstance(Ci.nsIZipWriter); var mt = Date.now() * 1000, cp = zw.COMPRESSION_DEFAULT; var bootstrap = ln.includes("bootstrap"); var prefix = "jar:" + fileURL.spec + "!/"; var lr = /^¦(?:\d+)?$/, sep1 = "£", sep2 = "¥"; zw.open(xpi, 0x04); // PR_RDWR for(var item of data.split(sep1)) { var [entry, val] = item.split(sep2); if (bootstrap && (entry == "manifest.json" || entry == "startup.jsm")) continue; if (val == "+") { zw.addEntryDirectory(entry, mt, false); continue; } if (zw.hasEntry(entry)) { if (val.includes("¦")) { var lines = val.split("\n"); var oldLines = (await (await fetch(prefix + entry)).text()).split("\n"); lines.forEach((line, ind) => { if (lr.test(line)) lines[ind] = oldLines[ line.length == 1 ? ind : +line.slice(1) ]; }); val = lines.join("\n"); } zw.removeEntry(entry, false); } var stream = Cc["@mozilla.org/io/string-input-stream;1"] .createInstance(Ci.nsISupportsCString); stream.data = String.fromCharCode(...new Uint8Array(te.encode(val))); zw.addEntryStream(entry, mt, cp, stream, false); stream.close(); }; zw.close(); xpi.reveal(); })( /^custom_buttons-0\.0\.7\.0\.0\.6-fx-(?:paxmod|bootstrap)\.xpi$/ );
Можно из Вэб-консоли на странице about:addons ( Ctrl+Shift+K ).
Можно из CB-кнопки ( вкладка Код ).
Можно, пока ещё не выпилили, запустить со Scratchpad'а
( devtools.chrome.enabled=true, Shift+F4, Окружение > Браузер
или, лучше, открыть во вкладке:
chrome://devtools/content/scratchpad/index.xul
)
Отсутствует
Dumby
https://www.upload.ee/files/10644103/cu … p.rar.html
Хорошая вещь , но 71 кое-какие кнопки отвалились , там много чего отвалилось .... Зато 69 перестала падать...
Нужен код " Переключиться на последнюю использованную вкладку" Этот код не то...
Отредактировано solombala (27-10-2019 11:33:26)
Отсутствует
А есть способ на FF70 отключить подпись, но чтоб при этом устанавливались расширения без ID?
Во-первых, расширений без ID в манфесте не бывает, только WebExtensions.
Если WebExtensions неподписан, то что мешает указать ID?
Ведь это всё равно значит либо писал сам,
либо изменил существующий подписанный.
То есть ответ будет в стиле «Совсем обленились!».
Правда есть вариант, что кто-то написал WebExtensions,
не указал в манфесте ID, не подписал, и предлагает к установке.
Это значит, что автор алень и не заботится о пользователях.
Следует, по возможности, указать ему на это.
Или, опять же, дописать ID в манифест самостоятельно,
это вовсе не сложная процедура.
А если WebExtensions подписан и не устанавливается,
то тогда совсем другое дело.
В этом случае, просьба предоставить примеры
(ссылки на такие WebExtensions).
Отсутствует
Правда есть вариант, что кто-то написал WebExtensions,
не указал в манфесте ID, не подписал, и предлагает к установке
именно этот вариант. И таких расширений в репозитарии куча. Первые попавшиеся:
https://addons.mozilla.org/ru/firefox/a … bextension
https://addons.mozilla.org/ru/firefox/a … che-button
https://addons.mozilla.org/ru/firefox/a … leaner-pro
Просто не удобно каждый раз для установки таких расширений переименовывать config.js и перезагружаться, а потом назад.
Отсутствует
именно этот вариант. И таких расширений в репозитарии куча.
Какой ещё репозитарий? Если имеется в виду AMO,
то там не может быть неподписанных WebExtensions,
а значит это никакой не «именно этот вариант».
Все трое, Cache Cleaner 0.1.1, Empty Cache Button 3.4 и Browser Cleaner Pro 0.1.0,
установились в релизный Firefox 70.0 сразу сходу и без проблем.
«RESOLVED WORKSFORME»
P.S.
Конфиг этот.
Но, так ли это всё важно, ведь приговор лисе
находится уже в стадии рассмотрения.
Это чисто формальная процедура, не знаю чего Крис
тянет резину за хвост, наверно занят чем-то.
Отсутствует
Dumby
Можно эту кнопку заставить работать в многопроцессорном режиме?
// Двойным левым кликом на папке закладок добавлять закладку в папку закладок, от 09.05.2014 addEventListener("dblclick", function(e) { var target = e.originalTarget; if ( e.button !== 0 || !target._placesNode || !PlacesUtils.nodeIsFolder( target._placesNode ) ) return; var Host = window.gBrowser.selectedBrowser.contentWindow.location.hostname; Host = '[url]'+ Host +'[url]'; // выделение адреса var docTitle = ( content.document.title || gBrowser.mCurrentTab.label).substr(0, 60); // ограничить длину имени закладки var folderId = PlacesUtils.getConcreteItemId( target._placesNode ); var dirTitle = PlacesUtils.bookmarks.getItemTitle( folderId ); // не добавлять имя папки dirTitle = ''; if (dirTitle.length > 0 ) { dirTitle = ' ['+ dirTitle +']' } else dirTitle = ''; var parentFolderTitle = PlacesUtils.bookmarks.getItemTitle( folderId ); var currentURI = Services.io.newURI( content.location, null, null ); var selTxt = ''; var getSel = document.commandDispatcher.focusedWindow.content.document.getSelection().toString(); if (getSel) selTxt = '[txt]:: ' + getSel + ' ::[txt]'; Host = ''; // отключить жирный адрес справа от имени закладки PlacesUtils.bookmarks.insertBookmark( folderId, currentURI, 0, Host + docTitle + selTxt + dirTitle); var favicon = ( !!gBrowser.mCurrentTab.image ) ? gBrowser.mCurrentTab.image : "chrome://global/skin/icons/Portrait.png"; window.show_tooltip(favicon, "Добавил в папку "+ dirTitle +":", docTitle, 7000); });
На форуме
Можно эту кнопку заставить работать в многопроцессорном режиме?
Не, вот прям именно эту неохота, тем более описания нет.
Но могу попробовать набросать что-нибудь приблизительно аналогичное.
addEventListener("dblclick", { handleEvent(e) { if (this.skip(e)) return; var name = _id + ":SelectionForDblclickFolderBkm"; messageManager.addMessageListener(name, this); addDestructor(() => messageManager.removeMessageListener(name, this)); var url = "data:," + encodeURIComponent(`(fw => { Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager) .getFocusedElementForWindow(content, true, fw); sendAsyncMessage("${name}", fw.value.document.getSelection().toString()); })({});`); var lfs = () => gBrowser.selectedBrowser.messageManager.loadFrameScript(url, false); this.handleEvent = e => this.skip(e) || lfs(); lfs(); }, skip(e) { if (e.button) return true; var trg = e.target; var pn = trg._placesNode || (trg._placesView && trg._placesView.result.root); if (!pn) return true; var res = PlacesUtils.nodeIsFolder(pn); if (res) this.guid = PlacesUtils.getConcreteItemGuid(pn); return !res; }, receiveMessage(msg) { var title = (gBrowser.contentTitle || gBrowser.selectedTab.label).slice(0, 60); if (msg.data) title += `[txt]:: ${msg.data} ::[txt]`; var url = gBrowser.currentURI.spec, index = 0; PlacesUtils.bookmarks.insert({url, title, index, parentGuid: this.guid}); } });
Нужен код " Переключиться на последнюю использованную вкладку"
Я так понимаю, что это связано с «FlipClosetab приказал долго жить».
Если да, то лучше подправить расширение. Я скачал Flip Close Tab 1.2.0
и вижу, что там, вроде, ничего особо непоправимого быть не должно.
Вот, подпаял чуть-чуть bootstrap.js, можешь попробовать. Надеюсь Автор не будет против.
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 == 0) { 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 == 2) { 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) { }
Отсутствует