Полезная информация

Многие проблемы быстрее решаются поиском по форуму и чтением FAQ, чем созданием новой темы и томительным ожиданием ответа.

№1407617-01-2020 23:56:12

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 71.0

Re: Custom Buttons

func4ptch4
Это куда?

Отсутствует

 

№1407718-01-2020 03:22:21

func4ptch4
Участник
 
Группа: Members
Зарегистрирован: 03-05-2018
Сообщений: 198
UA: Firefox 72.0

Re: Custom Buttons

solombala, /*Initialization Code*/ забыл), мне (кажется) это лучше чем в каждой кнопке отдельно вписывать+можно все коды в одну кнопку и, назвать ну типа Hotkeys.


Dumby, все никак не могу привыкнуть, как без автопопапа люди пользуются кнопками...) ех, все-таки никак нельзя хотя бы в кнопках сделать автопопап? Ну просто смотрите навел на кнопку он раскрылся отвел закрылся, а без этого надо клик, клик по области чтобы закрылся(и это бесит). Ладно гамбургер меню, им наверно мало кто пользуеться, вроде в кнопки интегрировали e.preventDefault(); что если и это интегрировать? или это чисто мой привычки?)

Отредактировано func4ptch4 (18-01-2020 03:52:50)

Отсутствует

 

№1407818-01-2020 05:25:38

xrun1
Участник
 
Группа: Members
Зарегистрирован: 12-12-2013
Сообщений: 586
UA: Firefox 56.0

Re: Custom Buttons

func4ptch4
Если ты об этом автопопапе

скрытый текст
https://www.upload.ee/image/10994260/122.gif

, то работает.

Отсутствует

 

№1407918-01-2020 11:02:36

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 71.0

Re: Custom Buttons

Dumby
Что-то кн. скриншот не того (71- от правки биндинга избавился) Из контекста и иконки пропали и не работает...(текст в файл и вн. редактор)

скрытый текст

Выделить код

Код:

// Save, от 07.03.2017. .............

self.label = "Save";
self._handleClick =()=> menuPopup.openPopup(this, "after_start");
self.image = "";



var folderpath="C:\\Users\\_________\\Desktop";         // папка для сохранения иконок для ярлыков и ярлыков сайтов


// Создать меню для кнопки .............
var array = [
   { label: "Сохранить значок веб-сайта", func: "saveFavicon()", image: ""},
   { label: "Запомнить значок веб-сайта как base64", func: "copyFaviconData()", image: ""},  
   { separator: ''},
   { label: "Сохранить ярлык страницы как…", func: "saveShortcuts()", image: ""},
   { separator: ''},  
   { label: "Кодировать изображение(текст.файл) в base64", func: "copyFaviconbase()", image: ""},
   
   { label: "Сохранить всю страницу как PDF", func: "savePageToPDF()", image: ""},
   { label: "Сохранить всю страницу или выбранное как HTML", func: "savePageToHTML()", image: ""},
   { separator: ''},
   { label: "Сохранить всю страницу как HTML", func: "savePage()", image: ""},
   { label: "Сохранить выделенный текст как txt файл", func: "saveSelectionToTxt()", image: ""},
   { separator: ''},
   { label: "Запомнить изображение как base64, в контекстном меню", value: "CB.Save.WebScreenShotOnImage"},
   { label: "Сохранить выделенный текст в файл, в контекстном меню", value: "CB.Save.SelectionToFile" },
   { label: "Открыть выделенный текст в внешнем редакторе, в контекстном меню", value: "CB.Save.TextToEditor"},
];

var menuPopup = self.appendChild(document.createXULElement("menupopup"));
array.forEach((m,i)=> {
   if ("separator" in m) { menuPopup.appendChild(document.createXULElement("menuseparator")); return };
   var mItem = menuPopup.appendChild(document.createXULElement("menuitem"));
   mItem.setAttribute("label", m.label);
   mItem.setAttribute("class", "menuitem-iconic");
   if ("image" in m) mItem.setAttribute("image", m.image || array[i-1].image); 
   if ("value" in m) { 
       mItem.setAttribute('type', 'checkbox');
       mItem.setAttribute('checked', cbu.getPrefs(m.value) );
       mItem.onclick =()=> cbu.setPrefs(m.value, !cbu.getPrefs(m.value));
       }
   if ("func" in m) mItem.addEventListener("command", ()=> eval(m.func.toString()));
});
menuPopup.setAttribute("onclick", "event.stopPropagation()");


function aDate() {
 var t=new Date();
 var y=1900+t.getYear();
 var min=t.getMinutes(); if (min<10){min="0"+min};
 var h=t.getHours();
 var m=t.getMonth();switch(m){case 0: m="января";break;case 1: m="февраля";break;case 2: m="марта";break;case 3: m="апреля";break;case 4: m="мая";break;case 5: m="июня";break;case 6: m="июля";break;case 7: m="августа";break;case 8: m="сентября";break;case 9: m="октября";break;case 10: m="ноября";break;default: m="декабря";}
 var d=t.getDate();
 var curdate=d+" "+m+" "+y+" "+"г";
 var myfilename=curdate;
 return myfilename;
}
 

function WebScreenShotonImage(image) {
      var canvas = document.createElementNS(xhtmlns, 'canvas');
      canvas.width = image.naturalWidth;
      canvas.height = image.naturalHeight;
      var ctx = canvas.getContext('2d');
      ctx.drawImage(image, 0, 0);
      var base64 = canvas.toDataURL();
      gClipboard.write(base64);
   
      // стиль для изображение в сплывающей подсказке ....
      var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
      var uri = makeURI('data:text/css,'+ encodeURIComponent('#alertImage { height: 25px !important; width: 25px !important; }'));
      sss.loadAndRegisterSheet(uri, 0);
      
     // alertsService.showAlertNotification(base64, self.label, "Запомнил изображение как base64", false, "", (s, t)=> { 
     Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(base64, self.label, "Запомнил изображение как base64", false, "", (s, t)=> {
         if (t == 'alertfinished')
             sss.unregisterSheet(uri, 0); // удалить стиль когда подсказка закрывается
      }, "");
};



var saveToFile = function (fileContent, fileName) {
    var uc = Components.classes['@mozilla.org/intl/scriptableunicodeconverter'].createInstance(Components.interfaces.nsIScriptableUnicodeConverter);
    uc.charset = 'utf-8';
    fileContent = uc.ConvertFromUnicode(fileContent);

    var nsIFilePicker = Components.interfaces.nsIFilePicker;
    var fp = Components.classes['@mozilla.org/filepicker;1'].createInstance(nsIFilePicker);
    fp.init(window, '', fp.modeSave);
    fp.defaultString = fileName;
    fp.appendFilters(fp.filterHTML);
    fp.appendFilters(fp.filterAll);
    fp.open(function (rv) {
  if (rv == nsIFilePicker.returnOK || rv == nsIFilePicker.returnReplace) {
    var stream = Components.classes['@mozilla.org/network/file-output-stream;1'].createInstance(Components.interfaces.nsIFileOutputStream);
    stream.init(fp.file, 0x02|0x20|0x08, 0666, 0);
    stream.write(fileContent, fileContent.length);
    stream.close();
  }
});
};


function savePageToHTML() {

var vert=`javascript:(function(){var getSelWin=function(w){if(w.getSelection().toString())return w;for(var i=0,f,r;f=w.frames[i];i++){try{if(r=getSelWin(f))return r}catch(e){}}};var selWin=getSelWin(window),win=selWin||window,doc=win.document,loc=win.location;var qualifyURL=function(url,base){if(!url||/^([a-z]+:|%23)/.test(url))return url;var a=doc.createElement('a');if(base){a.href=base;a.href=a.protocol+(url.charAt(0)=='/'%3F(url.charAt(1)=='/'%3F'':'//'+a.host):'//'+a.host+a.pathname.slice(0,(url.charAt(0)!='%3F'&&a.pathname.lastIndexOf('/')+1)||a.pathname.length))+url}else{a.href=url};return a.href};var encodeImg=function(src,obj){var canvas,img,ret=src;if(/^https%3F:%5C/%5C//.test(src)){canvas=doc.createElement('canvas');if(!obj||obj.nodeName.toLowerCase()!='img'){img=doc.createElement('img');img.src=src}else{img=obj};if(img.complete)try{canvas.width=img.width;canvas.height=img.height;canvas.getContext('2d').drawImage(img,0,0);ret=canvas.toDataURL((/%5C.jpe%3Fg/i.test(src)%3F'image/jpeg':'image/png'))}catch(e){};if(img!=obj)img.src='about:blank'};return ret};var toSrc=function(obj){var strToSrc=function(str){var chr,ret='',i=0,meta={'%5Cb':'%5C%5Cb','%5Ct':'%5C%5Ct','%5Cn':'%5C%5Cn','%5Cf':'%5C%5Cf','%5Cr':'%5C%5Cr','%5Cx22':'%5C%5C%5Cx22','%5C%5C':'%5C%5C%5C%5C'};while(chr=str.charAt(i++)){ret+=meta[chr]||chr};return'%5Cx22'+ret+'%5Cx22'},arrToSrc=function(arr){var ret=[];for(var i=0;i<arr.length;i++){ret[i]=toSrc(arr[i])||'null'};return'['+ret.join(',')+']'},objToSrc=function(obj){var val,ret=[];for(var prop in obj){if(Object.prototype.hasOwnProperty.call(obj,prop)&&(val=toSrc(obj[prop])))ret.push(strToSrc(prop)+': '+val)};return'{'+ret.join(',')+'}'};switch(Object.prototype.toString.call(obj).slice(8,-1)){case'Array':return arrToSrc(obj);case'Boolean':case'Function':case'RegExp':return obj.toString();case'Date':return'new Date('+obj.getTime()+')';case'Math':return'Math';case'Number':return isFinite(obj)%3FString(obj):'null';case'Object':return objToSrc(obj);case'String':return strToSrc(obj);default:return obj%3F(obj.nodeType==1&&obj.id%3F'document.getElementById('+strToSrc(obj.id)+')':'{}'):'null'}};var ele,pEle,clone,reUrl=/(url%5C(%5Cx22%3F)(.+%3F)(%5Cx22%3F%5C))/g;if(selWin){var rng=win.getSelection().getRangeAt(0);pEle=rng.commonAncestorContainer;ele=rng.cloneContents()}else{pEle=doc.documentElement;ele=(doc.body||doc.getElementsByTagName('body')[0]).cloneNode(true)};while(pEle){if(pEle.nodeType==1){clone=pEle.cloneNode(false);clone.appendChild(ele);ele=clone};pEle=pEle.parentNode};var sel=doc.createElement('div');sel.appendChild(ele);for(var el,all=sel.getElementsByTagName('*'),i=all.length;i--;){el=all[i];if(el.style&&el.style.backgroundImage)el.style.backgroundImage=el.style.backgroundImage.replace(reUrl,function(a,b,c,d){return b+encodeImg(qualifyURL(c))+d});switch(el.nodeName.toLowerCase()){case'link':case'style':case'script':el.parentNode.removeChild(el);break;case'a':case'area':if(el.hasAttribute('href')&&el.getAttribute('href').charAt(0)!='%23')el.href=el.href;break;case'img':case'input':if(el.hasAttribute('src'))el.src=encodeImg(el.src,el);break;case'audio':case'video':case'embed':case'frame':case'iframe':if(el.hasAttribute('src'))el.src=el.src;break;case'object':if(el.hasAttribute('data'))el.data=el.data;break;case'form':if(el.hasAttribute('action'))el.action=el.action;break}};var head=ele.insertBefore(doc.createElement('head'),ele.firstChild);var meta=doc.createElement('meta');meta.httpEquiv='content-type';meta.content='text/html; charset=utf-8';head.appendChild(meta);var title=doc.getElementsByTagName('title')[0];if(title)head.appendChild(title.cloneNode(true));head.copyScript=function(){if('$'in win)return;var f=doc.createElement('iframe');f.src='about:blank';f.setAttribute('style','position:fixed;left:0;top:0;visibility:hidden;width:0;height:0;');doc.documentElement.appendChild(f);var str,script=doc.createElement('script');script.type='text/javascript';for(var name in win){if(name in f.contentWindow||!/^[a-zA-Z_$][0-9a-zA-Z_$]*$/.test(name))continue;try{str=toSrc(win[name]);if(!/%5C{%5Cs*%5C[native code%5C]%5Cs*%5C}/.test(str)){script.appendChild(doc.createTextNode('var '+name+' = '+str.replace(/<%5C/(script>)/ig,'<%5C%5C/$1')+';%5Cn'))}}catch(e){}};f.parentNode.removeChild(f);if(script.childNodes.length)this.nextSibling.appendChild(script)};head.copyScript();head.copyStyle=function(s){if(!s)return;var style=doc.createElement('style');style.type='text/css';if(s.media&&s.media.mediaText)style.media=s.media.mediaText;try{for(var i=0,rule;rule=s.cssRules[i];i++){if(rule.type!=3){if((!rule.selectorText||rule.selectorText.indexOf(':')!=-1)||(!sel.querySelector||sel.querySelector(rule.selectorText))){style.appendChild(doc.createTextNode(rule.cssText.replace(reUrl,function(a,b,c,d){var url=qualifyURL(c,s.href);if(rule.type==1&&rule.style&&rule.style.backgroundImage)url=encodeImg(url);return b+url+d})+'%5Cn'))}}else{this.copyStyle(rule.styleSheet)}}}catch(e){if(s.ownerNode)style=s.ownerNode.cloneNode(false)};this.appendChild(style)};var sheets=doc.styleSheets;for(var j=0;j<sheets.length;j++)head.copyStyle(sheets[j]);head.appendChild(doc.createTextNode('%5Cn'));var doctype='',dt=doc.doctype;if(dt&&dt.name){doctype+='<!DOCTYPE '+dt.name;if(dt.publicId)doctype+=' PUBLIC %5Cx22'+dt.publicId+'%5Cx22';if(dt.systemId)doctype+=' %5Cx22'+dt.systemId+'%5Cx22';doctype+='>%5Cn'};var href = 'data:text/html;charset=utf-8,' + encodeURIComponent(doctype + sel.innerHTML + '\n<!-- This document saved from ' + (loc.protocol != 'data:' ? loc.href : 'data:uri') + ' -->');var a = document.documentElement.appendChild(document.createElement("a"));a.setAttribute("href", href);var name = selWin ? win.getSelection().toString() : (title && title.text ? title.text : loc.pathname.split('/').pop());name=name.replace(/[:\\\/<>?*|"]+/g, '_').replace(/\s+/g, ' ').slice(0, 100).replace(/^\s+|\s+$/g, '');name += (function () {var d = new Date(), z=function(n){return '_' + (n < 10 ? '0' : '') + n};return z(d.getHours()) + z(d.getMinutes()) + z(d.getSeconds());})();a.setAttribute("download", name + ".html");a.click();a.remove();})();`;
gBrowser. loadURI(vert, {triggeringPrincipal: Services.scriptSecurityManager.getSystemPrincipal()});

};

function savePage() {
 saveBrowser(gBrowser.selectedBrowser);
};

function saveShortcuts() {
var file = Components.classes["@mozilla.org/file/local;1"].
           createInstance(Components.interfaces.nsIFile);
file.initWithPath(folderpath);

if( !file.exists() || !file.isDirectory() ) {   file.create(Components.interfaces.nsIFile.DIRECTORY_TYPE, 0x1B6);}

var savetodir=folderpath+"\\"; 
var urllink=gBrowser.currentURI.spec;
var out=getTabLabel();
var filename=savetodir+out+'.url';
var data="[InternetShortcut]\r\nURL="+urllink+"\r\n";

saveToFile(data, filename);
 // стиль для изображение в сплывающей подсказке ....
      var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
      var uri = makeURI('data:text/css,'+ encodeURIComponent('#alertImage { height: 25px !important; width: 25px !important; }'));
      sss.loadAndRegisterSheet(uri, 0);
   // подсказка
   var notific = 'Сохранил в: ' + folderpath;
   var image = gBrowser.selectedBrowser.mIconURL;
   Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(image, filename, notific);
};

// Кодировать изображение или текстовой файл в base64 .............
function copyFaviconbase(){
var fp = window.makeFilePicker();
fp.init(window, "Открыть файл", fp.modeOpen);
fp.appendFilter("Text and images", "*.txt; *.text; *.css; *.js; *.ini; *.rdf; *.xml; *.html; *.htm; *.shtml; *.xhtml; *.jpe; *.jpg; *.jpeg;\
                                    *.gif; *.png; *.bmp; *.ico; *.svg; *.svgz; *.tif; *.tiff; *.ai; *.drw; *.pct; *.psp; *.xcf; *.psd; *.raw");
  fp.open(re=> { 
  if ( re != fp.returnOK ) return;
   var file = fp.file;
   var inputStream = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(Ci.nsIFileInputStream);
   inputStream.init(file, 0x01, 0600, 0);
   var stream = Cc["@mozilla.org/binaryinputstream;1"].createInstance(Ci.nsIBinaryInputStream);
   stream.setInputStream(inputStream);
   var encoded = btoa(stream.readBytes(stream.available()));
   var contentType = Cc["@mozilla.org/mime;1"].getService(Ci.nsIMIMEService).getTypeFromFile(file);
   var dataURI = "data:" + contentType + ";charset=utf-8;base64," + encoded;
   gClipboard.write(dataURI);
   //Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(dataURI, self.label, "Скопировал файл как base64");
    // стиль для изображение в сплывающей подсказке ....
      var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
      var uri = makeURI('data:text/css,'+ encodeURIComponent('#alertImage { height: 25px !important; width: 25px !important; }'));
      sss.loadAndRegisterSheet(uri, 0);
      
     // alertsService.showAlertNotification(base64, self.label, "Запомнил изображение как base64", false, "", (s, t)=> { 
     Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(dataURI, self.label, "Запомнил изображение как base64", false, "", (s, t)=> {
         if (t == 'alertfinished')
             sss.unregisterSheet(uri, 0); // удалить стиль когда подсказка закрывается
      }, "");
});
};

// Сохранить страницу как PDF файл через сервис 'pdfmyurl.com' .............
function savePageToPDF() {
      var loc = gBrowser.currentURI.spec;
   var vert = "http://pdfmyurl.com?url=" + loc;
  
   gBrowser. loadURI(vert, {
   triggeringPrincipal: Services.scriptSecurityManager.getSystemPrincipal()
   });
}; 

// Сохранить иконку текущего сайта с диалогом сохранения .............
function saveFavicon() {
       var uri = gBrowser.currentURI;
       function getSiteName() {
                  try { var domain = uri.host.split('.') } catch(e) { return "" };
                   domain = (domain.length == 2) ? domain[0] : domain[1]
                   return domain.charAt(0).toUpperCase() + domain.slice(1).split('.')[0] + " ";  
            };
      saveImageURL(gBrowser.selectedTab.image, getSiteName(), null, false, false, null, window.document);
};


// Скопировать иконку текущего сайта как base64 код .............
function copyFaviconData() {
   var img = new Image();
   img.src = gBrowser.selectedTab.image;
   WebScreenShotonImage(img);
};


// Сохранить выделенный текст или весь текст на странице как txt файл .............
function saveSelectionToTxt() {

let browserMM = gBrowser.selectedBrowser.messageManager;
        browserMM.addMessageListener('getSelection', function listener(message) {
        var sel = message.data;
       !sel && document.getElementById("cmd_selectAll").doCommand(); 
     
   // создать название файла из заголовка страницы и текущего времени и сохранить текст ....
   var fileTitle = getTabLabel() + '  ' + aDate().replace(/:/g, ".");
   saveURL("data:text/plain," + encodeURIComponent(gBrowser.currentURI.spec + ("\r\n\r\n" + sel)), 
                                fileTitle + ".txt", null, false, false, null, window.document);
   !sel && goDoCommand("cmd_selectNone"); 
 browserMM.removeMessageListener('getSelection', listener, true);
});
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSelection", content.document.getSelection().toString())', false);
};

(popup => addEventListener("popupshowing", {
    handleEvent(e) {
        if (this.shouldHide) return;

        var menuitem = document.createXULElement("menuitem");
        menuitem.id = "content-baseItem";
        menuitem.className = "menuitem-iconic";
        menuitem.setAttribute("oncommand", "copyImageAsBase64()");
        menuitem.setAttribute("label", "Запомнить изображение как base64");
        menuitem.setAttribute("image", "");
        popup.append(menuitem);
        addDestructor(() => menuitem.remove());

        menuitem.copyImageAsBase64 = () => gBrowser.selectedBrowser.messageManager
            .loadFrameScript("data:;charset=utf-8," + encodeURIComponent(this.code()), false);

        this.handleEvent = () => menuitem.hidden = this.shouldHide;
    },
    get shouldHide() {
        return !gContextMenu.onImage || !cbu.getPrefs("CB.Save.WebScreenShotOnImage");
    },
    code: () => `(selectors => {

        var getImage = doc => {
            var elm = doc.querySelector(selectors.shift());
            if (selectors.length) elm = getImage(elm.contentDocument);
            return elm;
        }
        var image = getImage(content.document);

        var canvas = image.ownerDocument.createElementNS("${xhtmlns}", "canvas");
        canvas.width = image.naturalWidth;
        canvas.height = image.naturalHeight;
        var ctx = canvas.getContext("2d");
        ctx.drawImage(image, 0, 0);
        var base64 = canvas.toDataURL();

        Cc["@mozilla.org/widget/clipboardhelper;1"].getService(Ci.nsIClipboardHelper)
            .copyStringToClipboard(base64, Ci.nsIClipboard.kGlobalClipboard);

        Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService)
            .showAlertNotification(base64, "${self.label}", "Запомнил изображение как base64");
    })(${
        JSON.stringify(gContextMenu.targetSelectors)
    })`
}, false, popup || 1))(document.getElementById("contentAreaContextMenu"));


// Добавляем в контекстного меню страницы новые пункты .............
((contextMenu, el)=> {

   // в контекстного меню выделенного текста ....
   var saveItem = contextMenu.insertBefore(document.createXULElement("menuitem"), el);
   saveItem.id = "content-saveItem";
   saveItem.setAttribute("label", "Сохранить выделенный текст в файл");
   saveItem.setAttribute("class", "menu-iconic");
   saveItem.setAttribute("image", ""); 
   saveItem.onclick =()=> saveSelectionToFile();

   var editorItem = contextMenu.insertBefore(document.createXULElement("menuitem"), el);
   editorItem.id = "content-editorItem";
   editorItem.setAttribute("label", "Открыть выделенный текст в внешнем редакторе");
   editorItem.setAttribute("class", "menu-iconic");
   editorItem.setAttribute("image", ""); 
   editorItem.onclick =()=> textToEditor();


    // устанавливаем где и при каких настройках показывать новые пункты ....
   addEventListener('popupshowing', e=> {
      if (e.target != e.currentTarget) return;
      var sel = gContextMenu.isTextSelected;
      saveItem.hidden = !sel || !cbu.getPrefs("CB.Save.SelectionToFile");
      editorItem.hidden = !sel || !cbu.getPrefs("CB.Save.TextToEditor"); 
      }, false, contextMenu);

   // удалять новые пункти при изминениях ....
   addDestructor(()=> {
      saveItem.remove(); editorItem.remove();
   });   
})(document.getElementById("contentAreaContextMenu"), document.getElementById("context-sep-open"));


// Сохранить выделенный текст в файл на рабочем столе .............
function saveSelectionToFile() {

 let browserMM = gBrowser.selectedBrowser.messageManager;
 browserMM.addMessageListener('getSelect', function listener(message) {
   // создать текст для записи
   var url = gBrowser.currentURI.spec;
   if (/\.рф/.test(url.host)) url = convertFromUnicode("UTF-8", url);
   
   var time = convertFromUnicode("UTF-8", aDate().replace(/:/g, "."));
   var text = convertFromUnicode("UTF-8", message.data); 
   var title = convertFromUnicode("UTF-8", getTabLabel());
   
   var text = "..............................................................\n"
            + title + " - " + time + "\n" + url + "\n\n" + text + "\n\n\n";
   var text = text.replace(/\u000A/g, "\u000D\u000A").replace(/\u000D\u000D\u000A/g, "\u000D\u000A");

   // путь к файлу и название файла
   var file = Services.dirsvc.get("Desk", Ci.nsIFile); 
   file.append("Save - " + (aDate().replace(/:/g, ".")) + ".txt");
          
   // создать файл с текстом или добавлять текст в файл
   var foStream = Cc["@mozilla.org/network/file-output-stream;1"].createInstance(Ci.nsIFileOutputStream);
   file.exists() ? foStream.init(file, 0x02 | 0x10, 0664, 0) : foStream.init(file, 0x02|0x08|0x20, 0666, 0);
   foStream.write(text, text.length);
   foStream.close();
    // всплывающая подсказка дает возможность открыть файл если кликнуть на подсказке
       var notificat = 'Сохранил выделенный текст в файл на рабочий стол'; 
   var image = gBrowser.selectedTab.image || self.image;
   Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService)
    .showAlertNotification(image, notificat, "Кликни чтобы открыть файл", true, "", (s, t)=> { 
      if (t == 'alertclickcallback') file.launch();
   }, "");
 browserMM.removeMessageListener('getSelect', listener, true);
});
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSelect", content.document.getSelection().toString())', false);

};


// Создать текстовой файл с выделенным текстом в папке профиля и открыть в редакторе .............
function textToEditor() {


 let browserMM = gBrowser.selectedBrowser.messageManager;
 browserMM.addMessageListener('getSelect', function listener(message) {
   // создать текст для записи
    var text = convertFromUnicode("UTF-8", message.data); 
    var file = Services.dirsvc.get('ProfD', Ci.nsIFile);
   file.append("TextToEditor.txt");
   custombuttonsUtils.writeFile(file.path, text);
   file.launch(); 
          

 browserMM.removeMessageListener('getSelect', listener, true);
});
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSelect", content.document.getSelection().toString())', false);
};




// Получить название вкладки без не сохраняемых символов и лишних пробелов ..............
function getTabLabel() { 
   var label = gBrowser.selectedTab.label;      
   var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " ");
   return label.substring(0, 50);
};
((main, parts) => this.onmousedown = e => {
    if (e.button) return;
    this.onmousedown = null;

    var df = MozXULElement.parseXULToFragment(`
        <menugroup orient="vertical">
            <menuseparator/>
            <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"/>
        </menugroup>
    `);
    var menugroup = df.firstChild;
    menugroup.setAttribute("context", "");
    menugroup.setAttribute("oncommand", "handleCommand(event);");
    menugroup.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));

        (menugroup.handleCommand = e => gBrowser.selectedBrowser.messageManager
            .loadFrameScript(urls[e.target.value], false)
        )(e);
    }
    menuPopup.querySelector('menuitem[label*="ярлык"]').after(df);
})(`
    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));
        }
    }`
}); 

this.oncontextmenu =e=> { e.button && !e.ctrlKey && e.preventDefault() };


// Получить название домена с заглавным первым символом и без приставок( типа .ru и .com ) ..............
function getSiteName() {
   try { var domain = content.document.domain.split('.') } catch(e) { return "" };
   domain = (domain.length == 2) ? domain[0] : domain[1]
   return domain[0].toUpperCase() + domain.slice(1).split('.')[0] + " ";  
};


// Получить название вкладки без не сохраняемых символов и лишних пробелов ..............
function getTabLabel() { 
   var label = gBrowser.selectedTab.label;      
   var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " ");
   return label;
};

    
// Получить выделенный текст из страницы или 'false' ..............
function getSelect() {
   var el = document.commandDispatcher.focusedElement;
   try { return el.value.substring(el.selectionStart, el.selectionEnd) } catch(e) {};
   var sel = document.commandDispatcher.focusedWindow.getSelection();
   return (sel == '') ? false : sel.toString().replace(/^\s+|\s+$/g,"").replace(/\u000A/g, "\u000D\u000A").replace(/\u000D\u000D\u000A/g, "\u000D\u000A");
};

     

Отсутствует

 

№1408018-01-2020 17:09:15

sandro79
Участник
 
Группа: Members
Откуда: Гиперборея
Зарегистрирован: 15-11-2017
Сообщений: 805
UA: Firefox 68.0

Re: Custom Buttons

Dumby
В бытность 72 nightly, вы мне предложили скрипт для изменения сохранения закладок звездочкой в адресной строке в "Меню закладок", а через конт. меню на странице  в "Другие закладки" (штатное поведение).
Тогда в 72 nightly это работало, но теперь в релизной 72, да и в 68 версии проверил, через конт. меню страницы, сохранение идёт тоже в "Меню закладок".
Хотя в нынешней 74 nightly работает как надо. Подправьте пожалуйста, если не сложно под 68 ESR, скорее всего и в 72 сработает.
Хотелось бы на 68 перейти со стиля на скрипт. Спасибо.

Отсутствует

 

№1408118-01-2020 19:34:37

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 1072
UA: Firefox 52.0

Re: Custom Buttons

Andrey_Krropotkin пишет:

Можешь разъяснить по поводу Mozilla Thunderbird 68 и СВ

Чего ж тут разъяснять, в названии написано «-fx», это значит Firefox.
Если была бы поддержка Thunderbird, то было бы написано «-fx-tb».

скрытый текст
Я не пользователь Thunderbird совсем.
Хотя, вобще-то пробовал что-то сделать.
И задачу поставил весьма облегчённую — только Thunderbird
и только Thunderbird 68.

И нет, ничего хорошего не получилось.
Там развели целый вагон тулбоксов, это не для моих куриных мозгов.
Ну и было недоделано, нетестровано, брошено, полный отстой.

Но, кое-где, местами, вроде должно работать.
Если хочешь, можешь попробовать.

put.re что-то не алё, придётся от последнего paxmod'а собрать кодом.
После установки рестарт. Никакой конфиг не требуется.

Выделить код

Код:

// CB-{0.0.7.0.0.9-fx}-{68.0-tb}.js
(async () => {
    var gzip = "H4sIAAAAAAAACuV963YbR5Lmb+Acv0MZ3WsDFnEp3Iu69EiU3M1pSfSKUtu9ag5dAIpkWSCARgEiaZmvsS8wZ/oB9vf+4qPsk2zEF5FZWQWABGX3nLOzOlQByMpLZGZkZNwycng2n55H9eEyWUzPB8vFYjpJ6sPpZBFNFvQ5iqJRvJjOa8MkufmPm38U6e8TP66L39SLw4G8paRC9Xz6c3UQT0bx5HTXW87H5SGq3q1vqlwL1y7Px7/LZKmm7VYeUkvUXg+tBvLDb+KX3+ZHhx9dfvTo7e/C4TAaf4iuKIfnrcJUug9Qpq5FdLkYTC+1ohJguvlHswEomk1+tASyZlfS+vwAtLXzaLIcx8miGlMb8XD9UN0F1mE0PvlTNJ5F87qpT6rL/VwzlrZ97lg4GEcKSMkMbUtGs9Xmn/Vv+GufJ/fmH23pYZt72G7xg0e8zSPe5hFv96SGjuTrcL5OC4PT4UwdGY0Oj0ZH564rrXXb+lMmtqtve/K2p2978rZHb2/+fXgbpo6W5+dXtcvl+OY/qndk3XosfyKMLyeLq3GUeI+feDRxH8O59+npbLZH9S3CySK59h57e2js3SIeJ7X4fDadL8qleZRMl/MhT+jpPKqfT0dLqqXuFqXqz0s73qdrmgdUTCuMapP2auF47D3wtPX3mXKzcbg4mc7Pj7xffjHZx/FkeWkqWs5jqqg0ChfhLqNunWreKVF10YQX1rs3+3tTgnNCg1GmV1zsgpBxeiF9GE/D0eFZFC3eJYSflPtwMacvZap2x3MzPn339k8Hb44P//TixVuqpFio171ny1PPb/cagd/wqt6r+HQeLiIvnsyWiyotIa+efq8ms4h6uZh6obeHmfJejCOag4VX/jaeRyeUvdt6UEG9Z4vFLKHRHCxPf47H47BGSwif0/lpPTmbXhzTm9rwNP5DPHqsABRlOM643cfeaDpccuW102ih7Ty72h+VS/F5eBqVKjUA9m0cjUe1WTin169psB5qJa+mP+/z+2eoSxBLa0m4xnKJ17TtGy+uQi7XKDqJJ1G5WCiU7JrMFtqhd8NxSHhAsxZNRkmm2U9DRoH5ckjkqVzxPiVLQuFy5eH1NZf7pEV2vRJ3uHRdLKyBYTmjCRlFZc7C679S/pQZ3rPTzMDq9+qQCs/DcX0Un5zUm43RoN/uDNvDfq++mE7HH+KFXVy0AnmrKBYIgXe9Hwmwf5mE51EyC4cRtXI+FnLH7VFzFxcXtYsWmvKDIKhfcg6MXaG2hm7xSPAyLAgFDWezKJyHE1pm3oTQ2ftSlh+tEq6hMAiHH07n0+VktOsR+JNEptX70svmE3JMCHkyji53PZ8Tafh+3PlthmZxRoOf1GXlJPXT8XRA+U33ZLTopYzWmm4z+qLTs3DEO0Z1MZ3teq3Z5UM3cTAlKne+mh5PiDZEVaId88Wu11z/NuIhasg76nmBp+GXb26fgvNwfspQU7GVgZ/OR9F8/aRoy+vLORMWT86ieSzpJ4RcmZTffnZAQNfPDV7dPTsrKGnzMg0+Ybry8P+LWfxPmK1psmGuzsMhvZPJ+n8BiYkCP7yDXeGuD8K5JGd/uUz5ZzCVt9XMfGQmhajyF8XrL4pfFP9p7fFMlSrueGqLmazv49G/Pf46y+PK59dH7xdXs+gxtldNKx3trC1/MeelOq9uX4/36Yt/Vqerq0NM3As9/Dv53unHaD4Or/xaR5DhLuZXpJsmeFuaxaLLz3kbOD+PeJZ7CCgqLeGTNtyQAWPWJtOUyzsWmXmk3gYiVNHDd+QqerDU0eyIeEWPnkhYLKhw7hbnbvGbVk+EF3oEIr4UiDcS4cURXExiIKKL/uxwU522I7vQo+eILySlcHNdX6QYLdblzN2uiDKa1uN8Pd/+5Ep7nK9nG+8FJq1v8/U5X79jX3DjfW68H5gsAQ9KwJAGbQNLwI0HtnG/AdRp+Hg28WzhCXG5AXm5AYG50cOzj2cAhBO0Q1kfZX2UJVGbCZa26Ptpa5gwHzPmY8p8zJmPSfMxaz6mzce8+RCN/abtkd9CoRYKtVCohUKYUR9T6mNOfUyq30aDbTtqPoRSH5PrQywl9t++7CB3B+BBPvVpnu1L5MYk+zTLRKTnLOqpkFNjkYGWw3hc8+bT6eLtPIr2F9E5p59/jywPuWQXwHXTHvXQTg89wtz7vY6bpUCSQLgcL54uaHnRQoqIaYecjD500IdOG6jXBu4JphpULZC44V0mLAIQyKNoHJGEtTiLaVVdJryxzKPFcj6xSSyiGnmPEiBgUDffl/7F3YOJcef/tLbn0UO/dFTj4aCWDqP5x5h497JbR0yLfn5CLD19nyT7P7x7ecjlmHE3Y4C56mP0GcNp+5t7ZZakWBygChS8laGoyFBg3Do0bgQFl/oYjqkfpks1jyD7SzheElx3USajz1mOSdouZfK8wCtKZpggdRSslGjlRZWdal4SpWB6ZS6yA7BICt/QF+895zqq6G4vQ9OEsqrpg4r5UBQJ5RPSp7QPxE+on5A/oX9CAIUCipKpibIghE1QwiYtKHygGNZSE2up2bLLoolF1GwjP9Q7zbZQWhQCHjaBh03gYRN42AQeNrGkmh1fKUIThLPZESLdyXQVVLMJstkE5Wh20VgXRbqGqDRBP5tdNIXl1OyiJAhps4eSWFVNrKomsKPJ2KEN9VAISNcE0jX7KNRHoT4Kgbw2QV+bfTTYl50EZfsoG6BsgLJBU/MQWoCfxuN6bY9aILqthhmUFkgu6m+B4qL6VkMS+gbuFihtC5S2BUqbeYncQJGWb8lLC7jRAm60gButZid9idxAipbsjrI9ElLQGpzVvOksmnjlk+VkuIin9G1ullwLqNIiVKEupCu5BZRoASVaQIkWUMLHALeAEi2gRAtUtgUq2wJStIAULeynLRDbFohtC1tqC9jRorG0Q2vaxKbaAla0gBUtYEULWNECVrSAFS1gRQtY0QJWtIRmyCz0UBbI0QJytIAcLYFd3gI5WkCOFpCjBeRoyeT1+/nagBwtIEcLO3ELW3ErQNmg6+QUTkRYkYZNb2NTbgND2tiU20AR87bnDkUbGNKWvbgv31EEeu82aEk73Y7bICKaBYjSBqK0m5KCQiAibRCRNvClDXxptxqKve1WM+Wc0qloYzNuYzNuC4flsFiqIEY7wmmJjhjY0gYBaQNb2sCWNghIoTCZjqKaqDFrtPGOFmdE5yWRaPwzlqOIBd0bx0SD30TDhVeumHwPvNLssvQQlaHxTjszdMCyNrCs3TU7EX/HuIAGtbuAtyucIeAFvrW7dhm2gWhtIFobiNYGorUJ0fCB/MCxNnCs3QdGF+ITrxx9xNYxncckZobjtyRu0ub9+LE3HCwmFWRuplMPJGwDCdtAwrYg4VrEAO61gXtt4F47SNmZNnCvHQiL28DTxxM6euBeB7jXAe51wBB2QJ86DSGIyAOGsAMk7PjNdKXit+VmO9jBOkC+DnYw0MZO0/JaHSBdB0jXER4Qk9EBkeq05LsYENAwtq8OsK8jNImqKl5DUuj0m/KBPBivDsar0xeGHVwTSG6HKHox3dRJPknihGbASylgPNrxQtqzK0bDz9jnKoy9rMaYSzB95PnlrBVPWQwwJqYFvNKKYS9aL5clH+JJHSxZPKyHoxFLoSrTbSfQHUbh+XTyIbp6FcbjAy25lR1EuJ97FTGQGelRVAGb/u5KUtRY/ZV9cUtOsC7Zt5K0oZ7c3699f68/Ia3m47b2WFIqwMAJuuxjifhYInlpqSD51spKBX0JLs/HuPhEpJk0JdEsg+BDEtMXEbHx1gAD3QdlC+chs8jCHHO5w+Xg3kVdwYyfgLAdFHVjzYtn/BTTLsSmFSkNZYSgizgKgu53U8EOlNwHJfdTSu6Dkm8Uz1LhPC+98FM2YBTqO9tM4YQEwWeyTHYzRIXIiVeve2/PonnkhfT/PLzyBhGJEbSISGzQtUV7GW1lCYQiIi0JLT6x/iQxvx3HHyLv6XhcjSfVg0nkHcajaEA0ymZBtcMxKym1xgpBV6+rEPh0Pqf16iVjyHDrSFry7MoRa0rxiOQhgh4zztQw5Bpoyt8fmZRZOInG5yTpxB/j6CKDDX9fRvOrQxJJYaQq/Y42amoJFO1kPL3wnuQKC2IxJc2mi2yL1mijzBPjp5Pp5Op8ukwsVXZ6kK2IRD7TI6AwN8Vbr3SLyPUyOZMUozXXgdP3RPmG4aL8vlar3WPwjlL+uYmdtYmdtYmdNRCZT7Q0ou5qfr6ASH2afheOo8UiEjS8Y3VOJVcNu9xrkXtd8VKwWnexveWcLWaH0SKD3KrF3KeNU3COv02ii2f6QwSK+0ly/BTtnrI5BW0lL3iTCA+gKJUGW/H7hDYt2nIJGwlWrxzOK4LAqr1gpcFyDB2Fs0NrA9AK2CphyAaalMQoHf/MRoN3+yVWWohmqOIt5lciNDVEaIMEIFw85juV9PgJycMXXWVW4DOiBbj5lt918aCF+W5hvtcJfS27K7TAVuVlP6obc9vCntHCntECW+WIetoUuPkWtAKttgh2KNRupXJgKvu1QMF/G9mPn6LAbRrZGTw5y4BWHFwv/Gnnc8IfPx0hrJfK09BDMUmZ7oFe3r5QRAbJrJK8rGgrBnwi4kODsJWQGLhCIrh9UIM2GPVNQmK7ITlFIgCTD/Rq+yL4pdJiG/qE9dIiP1HWD9KcTd+KuvcTGvmJdlviKNRK6wGitVtZ6dEK1NJr4FtbnIxWhEjOk5HpjOxopj8V/fjZyQp/7gxk2um2nPZFEMwKf/xE/4Bv7tiYUkYG5Kc4Q6WCYB5q4E0q3tlktJNq99sANy/Z2ZdZkc4IVlaWswmpENdp2Lo7QIqOqPOlDJCiA6ToACk6QIpUmLOyG7IDGzqi0Rd5zZHmOsAGluasOWVFjHOLgkvtgOx0hOwEovYGlG0zPc4gprpwkyD+ZmK3UcNNJ32J3J00N5hEkRVBYDpd+Y5mMe8d0eQLgJj3jsj+gSijW0qgOph1TEcHk97pyfdgRZO1KqKaoVsZzECe6BMwoBOkc0q0Ah9ihYIZCmSiCzLRbaTo3AWZ6MoupCl9PIM0xRcRvemktFdHvAtE6BIi4AMtARW6gL4L6LsgDF2gQpdQofANXAe7oAtddK2bqpO6wIQuCEIXBKELTOgCE7rtRqZ9rLEu6EAX208X208XmNDF9tMFQeiCIHSBEF0gRBcI0YU+WgxX2H26QIuuWPO6pjVVK3TVloeWgBFdYEQXlKALjOgCI7pgYbrYebrYebrAiW5PDIEoC6ToQpToQpToQpTo9psWJmBHF9jRBXZ0ocDogjB0+0FacWAIED8eIqWd1iBziA2lC7zsAlN6wJQeMKUHTOkBU3rYUHrAlB4wpQei0QOm9IApPV+MmCgrmh/A3AOm9EA7eqAdPdCOHmhHDyjTazZy0PaAN70mS07DMIm81GpzHiVJNDmN5lbLEI5Gc0ocTKcf3O8w4eyKTLWcjKL5IJ6PUCtgAR72gIe9JjaZaEwNMSe3atR592aftXCfC0UF3cvoSGqespMzYcaZaWAvzKdS9BkVfSsZlFsvybhgvLBYelgsPWyiPVDPHqinfBcU6mHN9LBmelgzPVDPXlvwo7BZaVUasgEviaqAk10U2XJFo6npHg9BeBrdWU84SKt4yBXo6Hg8POm8g2PqYfH2sHh7WLw9LN5eW+zfgB2Lt4fF28Pi7XXEYi6mcJTF6u2JRtfZjHvgHXug7T1s6j0Q91634+RB2W4/xUYx0GMp97CUe1jK8l3WVA9LuYel3MNS7onNXufBGft+K99rqf/z5jbo5iHBWu7LWt4AQ7/R+Q1h4HYxdH2h/X3Q/j7WcJ/WMD66eMJHARxAH0310VQfTfXRVB/V99FUn8h/waMlge8oC1Tuoyt96orsHv22+IH3sQP0gUR9IFG/3RNXCRQA/vSBP/2O7O9orCMwAnX6hDoFWrTPD159F84TdukSH1s3zTrcpimsgMh53PLgwuvWK4s2Qey+J9P5MHoxYTGRBIgfnr3U13Yj6wOX+sClPnCpr7wCz1Tf8QARF5B+R36IAwjyYy/oQ7jog1Pog1Pog1Poy14g08B7uI4ibQVQAQbAnYD2AXzAfQRbQIAtIMAWEGALCERFwXyj1hJgCwjALARgHwNsAQG2gABbQIAtIPD74qOC1iC3BsCcAJgTgGsIQK0DoE8A9AmEWgNQYE8A7Ala4uvSKqJSsAwBcCYAzgSEM/zBlA9lU1k1IFTBB/K1JR+wJACWBMCSAFQmAJUJgCoBqEwAKhOARwggoQbqZ9M0Ixuw6RqNglkIQGECMAsBmIUAqy/oGXtwAPoS9Nr2tzjtoFzPBRM8QgB1Y9BfARNMQtDvpaD1gxxohBcCGpAigMQZiI8QGITAkTj9hjoLNXw3TWSERruYZQ8pqWfUrw3xF2qoGktwxO0CpQl7mPaBksTVyE+H1m80c2NLKTq49E0AEs+hRtMFXJTiDdGKN1pN95U0Jm5EuVcCgHgSua+61sPJb7Tb4kclBAfcmaxFSgrklQigDVFZN0Rn3WDzo3aB0CntAtGJ+ZVR1TdEZd0QnXWjK0W72mQ3bZJ+CLBdAbbXcJvs6YcU76Ut92zLPSne077iqVMj2uxGXxNlvLRVkVIa2uG+VNKXSgKBIfDtWAVSRaDwBxvhN25p2ZFLu6Hws2eas7IMxNYxzXcAN75pgpy+nATz5SyYD4pEH335EBjEWU291RqBurm1XFrtp9YXNb+o/YXFNPlsmblkT7WCLQfNhqSLHKPp7TR/283fTvN33PydNH/Hzd9J83cbWZC7MgJdGQEZOTV/qHuar7MgWKSWEDWFqC1EjSGKkn5Pz9TZRsQcAm+ua01p62dXP6WpviwSX8c3kEYEU/xAQA0E1EBLBlISnA7NUoPNpdbfrL737M9s35xZF7Ta5WzhWiujy2hYV6fan7Ywmoqh1dhb+bjXZxSbzaOTZKty6XFGexrPGneJaV4QjxHOtqvJOZ54uwmW+PLNv/VHmlZcU9X6eguFjU2K/bWdywhdTvoafCp/EY3LLTUBf/VrYL+KXkd0KOuBkUTn1bpWuu117ThNmVRxm+1ad11T1rjQtk22hv0ZuJndP8CeHd27hnPT2N86cpta3jyrObBWbaT4gkVqbaMmNZNpTb7C9n1Ms677Jj/ULCjrwPqEqtN5zXNO8WkesW/d5TIpzaRek0VxmMx2UPwm1wyEuExms/Y3ZM35S+aKYabudpzkp3jEdzcO+UZzmxgPrbnNDka+vHhPumXu8J5EHhD6Jisv5V9pFCezcXjFh1/4nNJDj4/CVJP45yg9D+OlR2oellAebYBjbYrFdC2MYoDLwth30CSZTSyOGDcddua2Xr0Zi7VX/jqZhXxE5HHp+DhBajydHOtpkePj0tHXYglSRgQssjhuigbVGvJuVvw3sxa9Dd3REw7Z7tys+HKu2vMKWoGcgVArSzFn1GsZo96N+nVugEIcfR0oNln2NpQXxHXLC+KKO5vj3bmhvCKuOwpOyd4tJfv5ktKyeHZuKOOY7DIlV0x2Ot/CrKhRLjXZie1oa4udbClitWvaARF7XR7QojHVufCt+neu79+qoS5TCzBIDHXSg7VenpvtdE21rq139pR1LAa7NpAob7DbCDaQqN1dAbibyqU6CeoivA4t+LUYiFfmt61qenw1jps6L+uqEbfgW6sR+93t1Yg/8Eq3Vn0yxSalZ3QsE6J2MtHLN2yOjb13XTPXrAuprdmx+C4rLW/U28ZF09SzatvrtDdTPDnusjqsHej2OsLprVj6NtTVya8Rsfp1hPqsLyNeKm6Znm/b6QB30tMo/OxuHG6tUVx6V3vkuqhKjdZFdZP9j59oVeYp2LxZiyVwtVWxBIo1yzEEdiFQiyHwFoTVuv38GHWbluKJhsfBULEI3jZC3Vaeseg20/3UtQveWouQH7eWtl2KYh0UOrxqHXQocLdzSwty1q+7Au2qoVCshY6d0FJ310qopu61naLvRWsjXG3SMRKuL1+tPrmtvDURZipL0/I2waJrDiy6lkDD143ij7VkOJ+Ox8+uyo0dj9ilvYgPhnlVj959N03kXCfbBDdCXTSWOxfYbawWgiu91ubZ67Xza6EHjFm1Scm0bKilk2cwXYvUbfiZt0fxM9/TzB4u1piMPUpSUPZ2exSmvQfK04N2rCeWYWQH1vT68h1FQXh6IDw9kIweCE8PhKcXiNkJzYo5KhDTFMo65qiufIUlQk48YVb6cuSpsXls+uIHt7qZygg7Y9AH7emLyngziepj6+mvoJJrleqJaaq5phIctEVFYmKx9ih+wtwCvqcPvme9VYqfAHWtaUqhlE1FzFLi0ICv/aKrNejIEdmeKSSZumIU6lq9SV/IDV5iy+rLsMnZ4bZbUVHsTlof+usA0ZejHrlW+71cLtf41GuuKSF7lFNCdDUm1417+LeoxqetQzWlXzlwkol6YLUC3f6uV+KwCzMOyDGdjOJwPD0t8VF6TM0WecTJ4a5Mwo7pL9Eny6/C9Q5nWj2nzkwwIztIv4/T/UTH8CGCED/ksBnWBC8DEQuwIIpQoRUK11wiE3tLWxRGyVev1eV8bE/XnocztvGXv+CO0Ysa4ock38eLs/scm62g/FdfSeWu18R2RStSwcl0jgPA59HibDrypicWyPdU71EFP9/L2yP22eBwCDf2NL72Nj0k3xejqn0RmNUbiOXRvBA7GMxgorewL9jICRhs4IOE263pQd6/iP+4QQaEZ9Cixh4hKnZVXlNtRwB5/XF4edNdsxpucMYD76GJupav2ObuPAzPua+52a2XklEi1+8VK82EdYBrPetxtjufNBwM4tfhIv4YvcLMWjW3q97fm56fh5PRS651GzW4PfTU9O91fsnpjplV+6X6U2J7FHOstfG4Nh+d4DfWzXJmKY9QA+CQIQCiEMKMy+I2gQnxHVMW4PtqcAtgzRcOxfjCIResExKKwXoZIRpffC7FYF6fH4EhG1oZTE7Qqk6W5/8j5sNB4XlS418RB3D716dvahz1pEypO9CfvXuzX6PUb+NxVFHo1eOvqHtSGhjDhdiExmAmz4CTxsdgPZYGQIGDxN4wF6VgEi0upvMP9XhaTcS1n2MVgLzU0mAF5b24Nkn29w/0t9Cf2iS6ILDLJUK1+CRKFjSd0wmf/l+Ox7ZXlR3ObLJ8wVHUNCgL1ImMCDQ6CLH3hEPC3DgHrAprQgFSzhpHAxx/jMol19hzXqpwzLlYsiSzSOkdx7O4K+oLLSqkMN3Chng36u+F42gyCu933m/bUInVi3iy5XHF5YTd9u4FxR0hdAwvQHP1SEPzeKzAzYYv4llVJ5/H6QamQaRMS1o8uTMY0hNBUqC4ZTE2UYMbG6kU89tzCMLNSsCbhtAEMBL5HiFA0n+VjlRvnZhVf0xzHCgfNOk/qweP7KBKi/feOI2JapvVmkV5HZ5tDwSvbvTWbKxtR5f0OmYDyDFJ8PFsYU70Gn56b7mWfjn7YKmC2KXCUnIu4Y6+AO3K7u2A67vpOB5ebdV5PWXcGjTvdyrZZDbRVsVO33ZPK8u0ShhXfvsqnISEZDue++u7efyRNr18bFeNGoloWJviuzq1YJA4tBViuw4QhmfIUXcuE9paJ+e1+TLfn4Gzp9WK2eg7MSLtHC5nDEPCMRBgkxr96+HB4KdouICqhZ2NqJRS1+fqw1vWAeLNGmeZ4MmIg1DZjBx5AXsh8z+wm4o/HoRspHHs04GE39k7CyeTaOxh7k1cEXGLXK0WhaxvZc2b8ee3NLaHiznxcjympokdF4aaF34M4zG7VlLlxIxTnwmPQrbK1YkSlNTflZgd74QjLIyvxE1ToCWZRasZjtnF2fhlivMATjoSJ/vhOUfKRTk+RJ910yZp0HE3R4jW1Uyvvs9m4xwSIQoFNvqAZ0s52aSd5HRPfLNvK6VZ0ubQrfPwg54HJab6Qx4Ddjz1UpezmiI55vLAP/78QpHTWWPi/W4r8B5Qvlffl5zIUz+6eI2gvYLgv/8khlgz5t57W8vRdf1HPTNbGE7HbHjVU9WMJYrQmxGZpDbi4BC8S84p44AmpVrj7+qyWOlx5SG7TxNyDc+8Mld+rSPzJQujnnaOWsERUT2bvMMPHmN82UeMTxvrajEYsJzLMmX20PDb8BRjXM5xKRX1MtY6LfTZ6VyBfIdbwqrSsMU1OfCsMm4BkHFlnOs0e5w2HpU0/h6iF3HemgTpWp+fFmM01lP9jZYtwYzguzcvqRCW8tqWJNCy98svG7MMB9VkMeK60nyb1A0InaGMzWxyivMU7CAoC7zfTqOlgSzzqF/NONZamFxNhulxZrz9bj79GNOKOs5n98ohPuVMM7SEjBKSyARGvrHjySi6PKBXe8+Onz5/fvD6+O1fv3tR4WVU9TnUCzwiLGqw91h8ydjxvsSByc4v8BzyMxyUjiqKznOuW06hC5UUhNi4Rryy1vzAK21Yu5i7lfF5drX/fMux4aw0BOagd0Y7YcPz3TgxJ25WovTZg3JZSPZF8M7PVbnC0tb7Iz2oZYJBgMdbGwiCn2hQXCJz8SBu/qHukOIN2RPHzl5LPjryIdq+NHjE2jgQ/ATnKS+Fn+zbBPGXDfw0AQXT2I2igJaDW77r09lsCAuqBxoLEY4X7OpIfOKQ36M4kTR3jD6ZGAI9Hau7w8pdH2H93yFx6ll4sxUl9xYgc2xwVUL13s7TZqLjZI9bmdC3cmMB/8GBWP/0igqVCewPJ33DV/fLyk/CbffldVGOePIjvd0AOhD5oZZXucpAFR/yQzUe8kOsKvKjL0dmNgcMSn1KaWGPo4NsRJ6NjnWEFjfZuzJkB1vMT4nIRLVshCqQ0hWRiChHUYWmO7XcqdpKT01jVbWsCn2D0gr+owKZhFrAtnVGFK12zvtyuW6SH5f/dvGgUq+8948QAcx0yASkMMU8LUdiZ/n93y6qR24hqwczDqWiCsNgbYgCm54YdZRdv5EWZZslsY1SJCtkwTn6vs7NeZnqM4pup8jdKCbbUXCESdShe56KRqkr+B350hVSdwjF7T/luzx/RSSoW0I+Zf/avVxCipeZ1HyCk8aBn/LNrfgit1J/ajEo9/MVigARDj8sZ8xRTE+I03wez8GcrGEriPlFXmShCWQh7nxHTsZJLea2GROtWA/Qi8f/PUMWmz7eFq44Vy5nqblxAhfrir5ZF7PYBemWYFxMEMAk3BKuWKdhPTt0I0GLBQyQpU2Ri/FFdw4cCXFjNG0R5xdZoWDQmCdguSZ3RPh9tUTofETbEdZ/LZNVKGjYJFZNTEY2lko0Jow4CceJhAjm3L2m9sEG33KZrpstYnAZ3suXE0uwBCqS6dETOXni8Fx67ER8BwUJGwJPUyIxYb4N7wU2qdE2UZwkG3Y6uA9JCKcmsK8J7GsC+5riwonNlmM54QPFtozzq0Fmm2m43pyvuiqyto3tm/bXZqGv0wmL4E+NesWN7xS+if6+jBKasxC3z4iShn4dnJwkESfvTZeThYZ5EgeqjrAFll/Ou60rN5qL9IuOro0NhY9sWKibDXF+8SEl7hvd1/qq8xNlV/3Og8DwMA17UHd9xCcbuFeYIHBBbmzfe7qIt5o2JJKN82tcxFspdkisJ1inonkcjolJZz0dB8P64dXLwzStrOGgxKEcraXxW1OvcHxIG/iK7L9hrCcNH5qGeRI3KnyIzx/y3xnl98YJ9GTFyaLxAW+JF6bgxfrAvjdOzCYbBGR9oCYJ/CM8a+rxfVuQpuLNbxefSYL6NoVRbqZ53PhMmlPSJUxC3u1b4zPBUUGjcGzQPu6KggaahhW14AYW4KFTJlUO3hEBIy0ymS7iE9XsficajMe5cO273aBNRK/Zq3Y7oV9tD/on1cGoF1Zp6Qatk2G7O2oOdjO1HjtSxKbOPkyHT3zhZRrcoFX6NkjfitTivt0U+VjzB5n8Gh4rvYug3RE/eWSwgYvNy1zEYpuM3KvRqtKrJzRUMWJfqP4Sysg1UTvzt72kVaTMokQrzoezunGiFdtgVaL5QLKNamUXWTsQd3nZAPIBios3d8Qm7mBJ5WMT32h0q46fxk1aG9bqRmMUd8QteyW61Wqs4ns7wmOiNznCu0GuOm3xC0bZ3M0PaQid4mZP+I7IqSKoiqRqQ1+x/Cq+fijr+MJrCKyW7yriOgiVamIsi4uf4/9uGJIbJ+qV/tajBn3xCmzJDxPzqni7o7u2nQY66wQOtio9EQX+/mTBpIFDzE5HEQkdfoqm3YZxcbdhNMXPHRoZIIZEMehKXAIJsqlhLy3eql97Mw1Y1ezIh7i6W38biW8l25O4s3dbmXPkG6JbmchWojkAPL8yvpVxYV8f4CrrxN4VjYXUmXNi15BU1ofdgLplkCsT0cqSnvXBrczLQNyHAVogDupoJ415xgGrVNnOrMyL+ZyDsLC9efihFk+G4+UoSsrFmy3iWBXFcZ2f8G+2IayUY5DYVQYO3w5DT73RTeyqG401JWsxjTW1bZydTtFG52llgrmLq3qvLU7qKLLBVX01fJKvZ5PF9/fu8Ek94EYPuJGGT3KBAUqkUZP4iSIboyZlvdRNNVu4p/f6RtrR5la802WNiHe6rAvXO12naYN3eh87yu3u6b66sEPkTBdoH+jCvufKWXPQ64SN2LJzPiNOYnzL5TPj6ZB4nJzKejafzqL5Io4SMZ/10xCLaWwlMwj9Zjd9mfqwc2Qlk2xDKvEzzS2cvbir513XW8Zz3Y1OpA7ryGXjJ/Gzk9YPBBKH7j4QqA8E6gOB+hJbUYgLnv2ueOiKnzrKApP69g6kz7jEaHj/O4vgB2ivJ1rjNCHALCffSVBgVwomBtGYx9N7h+bROTG9qQlZMtldiMMnOUY+vE0vO+IKjvlaXljQmbIhQ81L/bK9suGgF2dvpxKiv1JxOFoL6mrbEozLma98OCWNPQQkCIRdwXLJh1ViW2U+gl8A/AmwHIO2hI3CE2QpDbVkgNFAS/ngSn1NSTPKxiT1prytG1JJAkHZiEr8lBQTU4mfqHY1qhKIUAAitCa0Ur+dZbICvW9DMsqRExQVl/LAuQDDwOnEVArkCIeG07F3AQlp03hK9CFaSVVn3h5RiT5EG6phW1WvKXachq8qSqnet6HxG81GStMa4oXe0FN6TTvC2bBK9CHGkI4ZcA2PRB9SMhdVSdoWfWcurJKoPBsgPoIkDkuuYZXoQw+26mHq9Po0iQ20QgUpXUrIgZJGxykh8pEOoAnPodpfaU3nYG1IJQmGsxpSCR+dTJZeJovU0s/U0s/UIgj26yJeAkWVHN4mw94ZwcnGbErvzmsEKaXX6E30IQX1mLto6FVFn+6QvirnVctu1OxyXFnvFDSXCmaCNxlFvW9Yf1tpYTp5N1HX+2iUUUiyZSn1VMJPdnFh5wohi3vvDt8evDp+9u7t24PXh8cvfnj74vXh/sHr4/3nleIq8Xw6HoszmA3e5ChK9eSFHr0Q74a2RpDSwAquqnzJ3oG0NYl7YOnI23z9e949UFwoUWMh85JjaiLhJQEbTVh5xz2of7OyhX1TV+8dNSPoHRt6yYbaEtSYoNYENSfoVRt614ZetqG3baiHhS9rTT0t1NVCfS3U2UINAetCUAX9dqZYsLmYRqDKF8PAsOnJWBQQfnSdW8vD9XXLSdXpx3HNO7lbgZUBoaB7gq9WCTVLqF3CiYaFD2lY45JpB2Q5qnlCnUOcGFn8IWoSMVe4jRp0S8Nk3WhcLPOq2QjcfcgX24Pf9B1vGuuiGR7OoiHU9o4vrqwnJSzpRQnxtMa6Y1uWYdLy4pjpPtNi4kd8GA2X83hxZfGYubErQuHz7+Yks8WzcAw/TpQu5hZSzub0chqO9icnBM7hi73jpy9fHnx/vPfm4PDw+ODN/h/3Xx8/f/r26fH+4fHrdy9forJs//jo2y3VZ/2SPXbWOj54+6cXb8RDyipCfLn20Jd7D/2mrKGmrCEJ5uNLNB9fw/mIRcQXk4jf1AOQzn4lVhFfovn4zZTr8cUy4svVGb7cneGLgcQ3P6SsrBg56uqLpcRnU4kKmLbyntOsYKnc7OXLfYi+mEx8tZk0NeAcUqQ9QVGxmvjNwIiKdnRkt4Ct5HZbv7N1ZY5/rLWFHz0x9vai8UnP/l75A9Oz5V8rbxBPW8inui+N/ww/N54a0ENd5kXWR6Z4+8Gu1dOgYk4oqptMMXWSuZFQakU4DRQLj0TRjdMVDgehXsEcGYCPUP4WnQBF9bzpZCiH9h7be1IixzG6/HUaJvrrHe9rm+fpM83xdaVUf/Irx0OUz2tHBCdA7+/Vdu/DGXIC6R4RAO1ZDOPgoqdNkzo7bkbzaEIUqp4GF6QZ4oHnzm/hE8beM8+W8dgcC8tg9PrlcMdquWYXu2/yqd439fsVMui/vpZ0wu8BYM/fkMfEzGQMUarQd1xmwEOyVH81i6YnHu5o/FPICp05nN4N31nC4WlwjxtyTsGClbyvvtqQq+ad4cuLjzgH4VYOdzp41JmzycylzmkNs7qT3V+h8rT3EHn2ljFa36jP8Ia7WcDiZPL1wgtT9pkqOQuR6AIjB62JBT2mHXsns1OKjtUbMh8+h7t+9Hp5Pojm4lKe8dtJr7nUyy1j9SehkXiRJsNzEJo4J3CTL1HLt+2gaSBhY+GGdv42KWH3glytTuoqwODpp6gpAWOb2QOxxuFKMFOOcKfIKXywxK52HJfuvKbvZuUct9lELTC3uAXdOFfm3cAxSKHrrIMud13ezYqf9J2upbc47RIqLNe75q2EaM1Sn3+Wz11qurk1b/Ymx9/y3keH1KxEJtCD1yYygVEZiQGuqI6l2zvEOaoqI8qriJ9B85Q308DIKpLh2ZMXxgPO8HpppIKWMrQW0X1r9tuE54KoOTTXIhtc4dyjAWI329mI8ll+M70Z8vOP+m+1y2d9c7cqkga+sFECeJbvDBdwY0+l3lj37C24Mz5VK2KXx1vP4+zhuOpP4cdQXpe8ZD58XBLwmtQV4rw+p/CLS5KgJuHY1ICYBeu9vkGXCjj2f0t8E2TI8G9OdBO8W++hzfbxIly0kUu8JlZqcIunIQmkONLE6ldciUogZi0blUAjLxVTV33lKyD8G499lwBkQFwXmkSpQC40ya90izWEQBVyaboqkSV9HU24WesaK89sEJNV2nCzEsjk9l1QYpv07PqtmVgLn3sq2jLg6W5kjELV+YePnxHZ+2xxPv6V7uvbR0TI0ZmtxIlV7ZURo4u6KxXTDepR9mIiiIibfbLqT7xHX1armUt9PI4tV3BIV8GhW0qztgzY8s8JP5FjS+4j8t37UL070eE8Cu1JJFB8ffveRsl5XFrMl1HpiGOFFKrn05+rGkZg11vOx3eGYbqtYT7/rNFJIBz+AwSnaOIg4DD+F86sbbvhXN8Z3WBhApBsN4+5vcPdHPN/3jdf3Pk7n5bPlv5Iczd7m947sDh1bABv3ftsVg1fc2fa5vo2/5lNQkP7/LP+tPYtoC5skwfQykOz9zv2nTRl69lc37o3iuFfrOYqiPU9HLHbKvsOLGv8693bb/ucwqEqOMeJvs2HGeJ08eGAW0H2oIR6Fbihj+726b/B5co3GoheBLM0ED1MSgwiW5DE0ym6RECNdPBuEJxeOyih6ZvgqJpgqVbd/ptt4RMBQjuNrCSR6UWrLUpt0Wk3ERNBYthJ1JNOLhXlOhrGaouw9Kmf/w3ugTYAEJMlnyggXv4C1C1e/jeIR69DIacN+h0DmvX1F5Z0nZ+/vMn6+GvpQHhYMLFyE5IE081d7lyo14XUbnm5s3j8o4T19pdfWU9/W6+93rlwbaNO3azc63xzaxz4G9zrfOMEgned/U2VTgh4cY13bneWgRePfzE6sMe/XWv1b76R2IXGz9/7pi4JWTf/G1zpfGMCTTuXOotUAPCsw7+m8gFZBTH18//3TMCum//4JEu89JFtudNJadcrdfu1RmlnPVFmDVM84lyZ3UnisNBqLyHiV6GULObxcHF8Hk+O8zVb+rUjwJndM03YQpPeSYngXfG9zpNTYSsyHO/7ePRvj7OdMEGQhL0g0InvHC3Odj2/O7v0vhSTdDhZCDZtX9sTD2EeapkSVeY83n8Mx0viarTJ9NqGXGvMj08ksMN0ntQWl8Tfx6Ndb/0cFDngxK6XVQQWi6NIZFCai13vj/HHKPFCbzZNknhAzNXiyltM1SFP6/UyACe14hkN8iw8jd69ebnrnS0WszynVSO6P4o+guwXi6hsOt/1/hpOuPmP0Zh99pLdYuHpZMGuCJnCEEEYU/7l9DyMxzUSQSrFwvNw/qH6PCKGt1g43NujfeTD8znV7YwH1XcwukqiZbLrDZantBFdRkmxMFhORqG/602XC95rEC9Lm/raU9homqfzcCHlZMegktMPH8LzMKl+4JWy6y3OIu9kCnubBNpk57roPPJKb2gbvDKV8gHN3KCXvHDh6V5YnS9pqLElcs7lebEwD3+eTnc92/GvE0T+8MLl4oxhexOeTsJwvpojsVnYposgK6zYpZV25S2T6GQ5Zt86GjnqzMvpcu9qhveZTr4MaW36rd2VAs9jJhHJ2eqbA+r2nFPnCQ8EelFb071y5k1tvqx4NP7eJlTJN4TMCyJRNGMkHS3m4SQZh/mZjpLq0zce+Ir4Z2ha+Jzt/APBPRl7z5cceCb79nn44XA+iH8Kd9nUPojDSS7DuMFtmvawTvYX9H4l43l4NYgiQtBd788kGa+8f8tF14Fw+DGa0NqL5jTEvArxJZdnERFOsCU9yZf+Lp4u5tEHr3y4HI+juLIh295V1KIB2jsjiYnjNB0SLRnHJ3E0quRynoST006n290q8yw6XQ5DbnO+WJ4ukfsZoXDMA5TPfDEOR9MLGujx9GP4IfvyNxJURbp/9Af68HSPeVzyaYP5w5NikcVuvpLrG/7nPXvxx/3X3sv9vRevD194z14e7P1Z3nCWv0jZXe/Vdy89v+azuhz3IkeegpSY+LrC45K86CVLWKuYZjI9eCWo7n23HIzjofcyHtJ6jUzVXB1VzG5HRDT0Jd/3ezVdeoRLMMHQGnDaiC6H0YztMrhRmEd4GHkX8eKM6+JqtJaa91etYzpY0Iomkj6czq4MnTKAhAsupyT74uKi5jLo1O+6dvpwerK44P6N4kQDCY08oXxudQSlm4FJ1sQrPT309g9L3iBM4mSHa/t+/+2fDt699b5/+ubN09dv/+odfOs9ff1X78/7r5/veFEMchJdzqAbIeIJ1ItGNVqekdse18XElZM4tiefbaM9dXK6pM3IO2XNzYSp+zw+PVsI+RjH5/EC6Jak8HM9ZtycWT7QcB3e3nSEvulsunlMSA9PfNFkkiirvYKOsbBs0ODbeB6dTC+9Vq1R0UpyZjlOqn7eLoySbxlTrJIOIyfhsHAxE0K8ySqxECocT7Fx7HqyA/9xHNKinH6krbjiNdlrVnLJsnnx+vnaRcMarKKJQZnYIJTcyjNNbJaKnufRoEySx6UNeHc5GKe5dllPmMl60UIuPwiC+iW/dTJfLscbq/0QXdHmvWBbBY3ihyiCvodQLarFxB9NxuKh5lQ22FwZw/ikmIu4meXzuOvxIjqvyoCXvO0DinJRG4MTddDiiajBQuGRZubvhUcE8O4Zx1BHIIJ1ENjWic6eEiEc4vaIkjcLhx/sL+NvQr3a1cvCqDK5misa7RyzRo0rC4ccY3pHgyKNdoi9GH6IRiXvZBxdEpEFhAoWApHdCRc+SvmG58PHKL5DfHA8otnaoST2ZLG1JwR/xPFriBF/XOpk3oGrNi27bTFrncKaa1RYbpTdCYdDIj2EMDvD+XRW8vj5uAQyUvLOIv6kGrra6qO6mQZMUD2dIRvOdAtU4VVSNYvUzppe6/a4xE0gNO090OiU7frhGJjECATmeKqIa9xnEwCdLK7GUcLR2cUStalu92pN6f6dRRFfaqXco7oDQNGitcc31i2ms+lsSSPODjql9Zi+OmT80hgGcEEhp+RnWb13UhRmjRWv/R3il2mrdXD51hVjsaDZLhG9d+DhtZI2v/2CQOHbV8O2i4CJ4i5g2DxaeO12dkIIR4ip6eF4PL2AQ0iiynV18y1sXjV2TKkFBOqzg7t2oUgfRrS0mD2nfuRhTV+V1HyKBbAeEA6NmkIgQRp1Sh+bqVUohmfxeETvPXPAUSoGzilOO2u4wL/tPgrWQatJ763Xcf/y/R67y74XCBGAkOtMA0yaVuT0PLx68FvOHhU+ade4JAbWDazpRnDEy1LloZP/PObjW6gtFwZzNa7kjtRuyzMc54gZGk1OF2fsQtWomGHermKBKF8xAxbvEGwmAYe2Yqqv8dCLvUee0yolPHhgWjUjkYLGuyAVrXAQg5ETO1U3yJKF1xYlyKmhTOGH9t2AUPOD/XldzH4iHul57IzBhiFIQ4ymgHjvG0emZpk9s5HuMxgMEvu8sXe2ZtNmV8/e0ff6TwnJFKKhqUooegkslA9cTHn3//XwcDk4RN6XyGrOPRRw+YZ5t/lI5b1lrZ8SmnRxi9XRPTrStZ5fH0g94UtjEflUFt2z6eWB+vcZYqH05olzHm9gcknNqGR9ncfp1bSlJzzEK7n1gOiVFnDyr4fg0SmbbHWJ59e4YIqGwUirsoiz8oamf6TnGmVVE9m9Op8uE8WpzKri0hwMDKSZY5YKcbZonDm+6TSi73UmaMCcLjyqmwHYOCQpMfycEVkleyfxPFnsMdE1gF+csZgktI/Du+rSgjYA6+lLXdoChh1OU6/mn9BOfhiTZD05zQ2JS2W3HoYMHrlrNodJyKwX7Ehu2S1Hh+BnZJQ87xFtQvSW72qTXOn2TS8H09GVfqdfmTHkFK9ed0Kar484tmWM8yyl2HBWYzjYXxfQT1ENEP2nwnNHGHZMrAkWjbjnmSnwyjTWAjtDrhhA41y3w/6oLjO4cRVkEWALxM+uRrf4bYhI/PNd9GVNjbz30VZrF4a07W67k+jiL8pClEruC+HVZKN2GQCnOruFGiaEPu/iQQoida3Pa8NZa14HOClG+2EGILMRm/1zTcs7tosZPkbqI5oiRXNogbeVHHnOVc/LdEfgsjVH4yTaolAp7aPkci9K/5jvcTova2eYC7gTF4l7u7N/iD5IPM3LJXwi6oHn/GPgPyI3X4Nj8kpLBDGTdXmulhOoWAJlQ4IWRWWVFYxO7rXDyAxuu7+uMlE5vN+KSc5kXZmXX7FEPYcHyKEbrykDq7dmkd4uHnweX/9P4ur/S/D0KUtva9zAoecJERMAbzOZJHoLyFbWpgzr60MazF+nltxxxyUlpOfxRvLoTtrajOuQZm1GuVMglzGHs7ymaJdfPIsIH6gQ1bOTx2uXFcxXlJuBycoU3EVrRFCvr0rqnCxncuaJFNJfQlEfl6AeKHmzszAh2jQMZ7RMiaU09Cm73J3FTvmdUfpatQxfmxX/yUE8JcHZAODM5ebJkaUWawgVGijndpjcVBlVB6ELOHaT+zorm+kIpAQ6NySD8XK+7YgIHBKzZR0oebnwrraBKsnZ9OLWOdARXWAk3Vs5TLiCNbPgrE45Vvlc9zY+gKXgmlh9t8zNmr0dhb3yZ462QvMZnb2l4+vXVaame04MUSqY5G5ZGFr9cDEf/zm6YjbMJCVn8cmC0jYvDqp+T7tB+V5I4vODV8d/+fPxu+/MRSUFy8/clv/5wfevnRLbNPFt2wU3HAPYFVKvr2kc+MtzOVWaTryN5zmLJq+I7BHzMncY1OvPQg+R+jcO+kb2yFUnLi4X3ipGZ5mfDZQ41VPdwtRS/Xfivv2RZE0jBffat7u8ADKHCMR/63dD2jhPp/OraqaI98Sr2Tfwn/n0BVhcKJdhtahiZxMn8dsuwBkOuDxfgGNdBmuA4L3opTNFSkde7dc1l963Y5ubx8MzqN1oUox32AZPuyN1PHqvKn3nxHN1sJiIS5nalXY98ZafXpqGfpe1TCQkgVcHhEBPvDVvaMaquMWIdfqmPb0RB+aPo93whBUgmSYH7KzFfrc6p7te6f/8z/9V4pQTSqmyCWjX6zX+Gxx/2XmHNs3qODqhnD5J02naYjrb9WotpF1/JuzveSnPk+hoq064IP/vkvjgZS4cvfkPXm4vfvju4M3bF8+PD//66tnBy0O+MqvEAliJr6HF1Xgmqkf+Bjy9EnBTeBsbDATXA6aVGX3tpuq2V/hmLwEvifs1t0JsqL1dTmNf299lzuZmqOFaghe4UeeHdy9/ePaynNaUu7uORRHR7KgXeJmJjuncOOH3tt+iCtdLWTNqbZufZB2vTOzDDq803NjDB4VES41RoYlMr6N+lxAqvHuzf7hgHoev397J5689fff2Twdvjg//9OLF27QbGi6HGpNdeAdyTEUvrZWomrxJZ6QbPfBXQRa5Qg2usdRNye2KsaJYqOzoNl+bXkyi+R9hVxXP9WsPXn7sya71bHBxh4e7SqDiHys3uQi7rmTeBdrlbkkEAHOPdYD707g2SILc/TVgZ/IDAK44HhkoEe9RVLcshzq9M0hRS6XSZ1f7Iy4rvvpckRatpPBrSo11wlrMqIadgxEFbcmVVdDdHa0pcxExo4i58etxyj4CrXZo4Ilx5K2Pd3gz53pPHt/GiNfEUuhl8XmcZ6j+sJKKW9r51e76V1QxjYu9dJLxW6ikufidO4MLz2q1GkehhrGT2cPkCDc0ryLk5fm4mtrw4YZNUFteeWWoasPxdBJxlWW5bWInzezKdwALR0SIPopsXDMZdYYMwmGsKiYjfziNCAfFG4hpJRxNZ2B5y5w1+85ZITUhdUa1UVvOTuehU2jDqEnzaiM+UoS1yyKLwoJOruCfw1pTJkVdKWIqsRCgQQOCDLMFQXIWhNxYWjM0w8zvmI7Hk2XkHmfxWGmB31rSApmbBqVb5m5Zg/4XlozObKz4ksV0IrPy8tarA1fuExNqcGfhbW5gsrWli47ptukcvwF5dZZxGXdY/ujsrdlmf/9Junq9evjzR3u5SobmXqM3yX022CnMIdjCxXgqQ5KH8q4a14d6Kn1eZSyCTldq2aJf9pBipjOKMERHtp5pPY5RylBgKukwB3xF3emkvIZV2GGUpHZvi/9oUyTio3exs0UB4tclvJMpRJSc+Qnt0/lFqXJXNbfGncxXGA7urtCmDOVW2pWEDbAOUfUdla8c3eW6uAalWL9q2V9nuzaKPvKKTnHgIhrolXt1OBDV4N6ZaX0LjMxf2ufgpQAQDqbLxa6IkG7td7DhOZ+tlQuwnXa2J3Ar8mzJVHB3T7Nl13Q0nefpz7QHOIeFzcGRxPman2poFt5/TiVHqmt4nx3qo0rZwkeL3JEHhCLOZSM1OwwzDM52ztTEkU3MjlfGvcFffaUFDPUnhp65LhAQKsif73Pvj8q5FpT0WFGDwGIx/0ChKxsw7axWTW1V2K7FPbZkN4pbqrJxo/6+jBdlGfBMZtGn2vycjZrlj6oTRKRKHA21PBKkua2CLWHHENzWkNM5M3vDQbLjcqU0i8zzOhxxIheQyxXlacxMkbXkpG82UiQCRT4/2Hv36sXrt5Uaq9MgPqbyUrrnQymVET1zF5+XwQMler+WxNSUXzV78fnae8+5nKxGw6VFqh12GOx1/mDl0jelivJt61hMrWaFORcOUdk9y6lzmDLm1JHw9sUPb49fHzx/4f3yy60Z9w5e8eAhr2SsuPkFPWRQnVtrsDbTW975pQ4VLncvrzCJiKqenXGTZC7LNSyaPZbwJjrlUyLhPOtLcq7xTP/7Mppf7RtnkLLBj3xhy/oZJZ/GQ1131txk0Sbgh5K6oZgmsvVUVuBOtJ8PvJITB6WUSjp8//ngJ+76BBeGT/Hj0zUGgim6U9UDz+bhW7j5M61ICkp1JjbdylWbHNnYIv1uRj1zbYH/tPfycP+5E3pOrjLn5cZrHq9ePj08PP726d7bgzd/3fFYg0Tt115hgxM9XX7sCX34WzT/NmT/vqvyPVsBeLl5Y/popuDFhBd6mVdkFei0nJVWai+pV9FOiQbTbQoUyng1cEPitTR8n2ZaM/2pv1He3chOy30H89qdxDXB20qVf/ooq4RttwbqWczxfg8OIfx/v//69duSo9D5EM2JMreaD7PaKayDxf1UlcMFLm2HErEmP0Q8NW3whrAQ4l4yacr+iHS94MVs3tRG0XBMsm+59EfaQcRk9jKeRN8TZlA1tMOHs/g4HMT4OTwL5373eFGjSipK5lJh+xwlIdDu8KXy6r1AXx55za58ffDACukioINelQUoevL9OHPQ8XKlUjNHAh+mdFxKUBt/a9AIi3cCZHLb/oPHqNYR2lGw7uUsGerXXjW8J2uSY3twsvyHXe+X31fqNT4bW9a6K0rtecfli5y8Xx57NKseHzl7dvyKNoXj5/uHT5+9fHG8/3r/7f7Tl/v/4+nb/YPXojrI7ADphBEnZafD2RCsek3YgdkG2vXdfLqYDqcIbXYfAjXTcvzmGlzEzFk3KM/q5hchCeXl4TiJR5hUUN0tVhV3QEtlgHhPdRy5kEgCtGjRBaDYc6fJdG+PXSJN5S7sKF/J0CeqrGJ4KVGWzc5cbXc8ZUJlatboo+mKo4W1drd8EyW5MhhvqryWEN1bDtiStATy5I+iyOIt7WRgoP4Sw1zWzc8s8BKQLL0Q3KhL3A6M4nnyEVxSuUQgnTwvGY7PxHYp5HQtqqzKGdW2hH8Vbm6GYc+2kh30ATG3w8y4S2hgZmrFxPsMOcqlVDGiY3FrYGHeofn7rlfmw2RwoDFcUiG5iFmLpAFt8UpWm9xkMSDJOAo5PIZAxx1/Rml8OVta20OlK07BCaLIUjmnoF7q5pZbLZjAAJItKEaRXFm74lWEA5FRAaMs/HVJr7OuwgRQJSo3OWXpolBweEcmYynriH3eaLTvZQXmaFflT6p22vVKjjSaVUSw3e/ZFS/FN2L0Q1iS1Vsx82FMdoeDdhBErV41aHaDarvRi6r9ZuOkGo2GQThojEbdoL+LyrhPZWc/nVAH2UjyZKOK2GiHU4dA/kdcfLmE/uH8pzrCy67MAo9AZjKOogUJMlXgHRt53ZzyLtGcxonsmD1snssrN7d5n8vtYLdaBB2n1ATmzYzfDmHAX2K54OBU3JGMkwt/JSnrAyyh6lCRpOSbl0NKvcu8Jf9BD5h4uzI+lVo+drGWkvMSYPtcGBfTWTxM1ASLi09kOxJQiNynP5ezES4ekwgtWBsklszKqIIBwhituUb1gbSSlRalFB+5BwQVb6M6QCHnbBkBWyeFFyKh7uGCj3+rxOum1dh9ApUIADkylm8JyTur1eSaXhnl0nLCimUaqHJKxj5rBoU3USH0jqlE3hwDd/vI5lQfzuBqNbnxWZN/0xAp8SO6PqW1sAsIr1ONmxP7uxw5VGD4cco8EZaEHMLloN/D5ZxNOZx6AOlPpztOni2g96Zi9OMNm8WI8+JfxwuV8XPbnk6Z9QeSKiqe/sA1N1Qh5lfWHfOhKW8fagZuA5KR+AChnOGec3VJEYJLbgqSi4I2OaWkNmjoBaMa9+OoHKVD55ALO3Qy/Ey0iW07i0ejiBv+0unhw7VrpFxZHY+nBlxqW/zoFGKlnR+p/VIl7aiaNWyrbqI5Z+qOKDqhRM52IN++2eJTNU8eGmLfQZr3cZ1LSnStuieqZb1Ic4Z3PSxWQWU515+SLCVTT9aJjTVgUY0mb0Zs2iw8FdGiomuAhTQGJ3EY/tQpOoNyytc4HcsyNhZDmN0k3KkSk1baFaBQEZ9RBqe1yqJsGp18caHgqcETVAcOk+cyD+vq3ugiZKrPkQ2XIXNoxpcp1i4mOTcK9TUqVRwmivHG3TvTO7k2ohHrizJjPo/+viTh7+kkPse8fcsnz8oulc5s7rWcty4xEIrRRvxeT0iUlXAWxVxNzg4NxO6eXcJK0OBfqWvoC3EzsFhyDj/chIbI8XQTJztZlcicQ/3sgUENrKaZ87iQtoQhXIvsthPM48eTD8/DRbgr/ejvZu6Fk3h2QTaRr2bRy2DwOsy+3hPrnJtjsLvxtjkznIavpvVrOQRcb5mixpc5XFAnH7P5GNrOQmtGYNWRMc5Wej6M+k2Zf3SXw27dREOu/95BJzNE7xmeWjIb87p9UKq894/et4Mj8E/X9R9TdkK3CWnjgUddot8P05fSyEuBQDMhKHq95GZPC0zkrjF+kyWD5sSZ0zDJDu/evPQeA/7VEnLIoWLo7GqG4aCaLEaI0GBz3WoETd04S85sMF7z3v50Pidh94SKl0+Z+eH9r3aMWDnTS8cykHpPYfJrHPuDLzEtZ5nViuuWIhottFQ7oR/7bEwtC1v/BOk1ddx9bHdvs/C5EBQVWROBvMAkDyNm6nAxuN0KTM+YhZBArvRDrxS0qJU5IeLCJ3vhlE+97CuQloh+OwHgO559XfEeeY0vrG1mzehlfJakdKZtPX7vLvdUNPxWqW+ya1z0lsToeeVwxxuAoA6UhAD7hAZFvLZZNxm672DJL4BD2HXKh7eUz9RdMTQgy9tsvfB1t1mr11i3c1nu3+W8aB0spqen42jNNrZjmqj8egQX9GJ9sjP1VthZnR3rt+vyemsaSl3SGNsFvRtHa0ZrdOL6cQkWG5Mm7aenoukFsKMTo6Wq1WqociMqGterk3SD5KM/ytTeNZnovV0/A2igX01//uHdS9UgiK2Tfr+dWhixmH58hGg5q7GGjFe0jf3SaDRKT4TBeaRahUwpl02Tw6uPS19l9SrMO/Nxh2OcdpaLL7g6DVBjgszUb2tmleva0JhxuOC8nM+296iO0DI/SqjZ9/e6Nny0GIm8X3EQZmUl8FZDksg4dDNs0uuUziL2Ua7i6lJRytTEybQ8UOtl5k5uRQnXIOumS9sOxXJY8lTQhC7tTqcToQK4j9PVV7DDEMnUllhcnKNPE+renONsCqdQsQzil7h/gc0AI/WxqK24GLIfxmPo+FTu0K5xZnOgjV+IgoBF3YwLJ3fyeRyOp+qYXTKBhKWHO8QmjiMalx11DdyR+EwJLd9osnM+HYWOGR+rz2EJQ0ckSyWPFVaEdguzSbJr9ZjV7DlnIPn82+jB7+s7yrH8/qtSJdtqOa3TbZSVpw5Q6wdix0sL72Q3L9ATqsSggYtTlLyCT9Yaum3EBgz4/eyl4t9TE2Xow/8LqYjwYv/4AAA=";

    var sel = "Select";
    try {sel = Services.strings.createBundle("chrome://global/locale/commonDialogs.properties")
        .GetStringFromName(sel);} catch(ex) {}
    var picker = Cc["@mozilla.org/filepicker;1"].createInstance(Ci.nsIFilePicker);
    var ln = "custom_buttons-0.0.7.0.0.9-fx-paxmod.xpi";
    picker.init(window, sel + " " + ln, picker.modeOpen);
    picker.appendFilter(null, ln);
    await new Promise(resolve => picker.open(resolve));
    var {file} = picker;
    if (!file) return;
    var {fileURL} = picker;

    var xpi = file.parent.clone();
    xpi.append(ln = "custom_buttons-68.0-tb.xpi");
    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 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 (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);
            if (val == "-") continue;
        }
        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();
})();


kokoss пишет:

Что нужно изменить в коде этой кнопки, что бы работала в [firefox] 72 в многопроцес... режиме ?

В многопрцессном? Ничего.

А чтобы просто работала, заменить
все createElement на createXULElement и первую строку на
this._handleClick = () => menuPopup.openPopup(this, "after_start");

Ну и косметика, заменить некоторые иконки на актуальные или свои,
убрать ненужный LOG, убрать Search plugins и соответствующую проверку.

func4ptch4 пишет:

все-таки никак нельзя хотя бы в кнопках сделать автопопап? Ну просто смотрите навел на кнопку он раскрылся отвел закрылся, а без этого надо клик, клик по области чтобы закрылся(и это бесит).

Кнопки же могут быть сделаны совершенно как угодно,
так что, наверно, нет. Давай самую бесячую из несекретных,
а я попробую в неё какой-нибудь autopopup встроить.

solombala пишет:

Из контекста и иконки пропали и не работает

Иконок нет потому, что неверно указан class,
нужно "menuitem-iconic", а не "menu-iconic".

А не работает потому, что не определена функция convertFromUnicode()
Так что добавь определение, ну что-то типа

скрытый текст

Выделить код

Код:

function convertFromUnicode(...args) {
    var suc = Cc['@mozilla.org/intl/scriptableunicodeconverter']
        .createInstance(Ci.nsIScriptableUnicodeConverter);
    suc.charset = "UTF-8";
    return (convertFromUnicode =
        (a, text) => suc.ConvertFromUnicode(text) + suc.Finish()
    )(...args);
}


sandro79 пишет:

в 68 версии проверил, через конт. меню страницы, сохранение идёт тоже в "Меню закладок".

Как насчёт рискнуть переключить настройку javascript.options.asyncstack
Если неприемлимо, дай знать, подумаю как переделать.

Отсутствует

 

№1408218-01-2020 20:03:45

kokoss
Участник
 
Группа: Members
Зарегистрирован: 15-02-2018
Сообщений: 685
UA: Firefox 52.0

Re: Custom Buttons

Dumby пишет:

и соответствующую проверку

Можно поподробнее?
Add: проехали, и так работает! Благодарю :beer:

Отредактировано kokoss (18-01-2020 23:10:24)

Отсутствует

 

№1408318-01-2020 20:07:53

sandro79
Участник
 
Группа: Members
Откуда: Гиперборея
Зарегистрирован: 15-11-2017
Сообщений: 805
UA: Firefox 68.0

Re: Custom Buttons

Dumby пишет:

Как насчёт рискнуть переключить настройку javascript.options.asyncstack

Рискнул, работает. Негативных последствий вроде не заметил. Переделывать думаю не нужно. Спасибо за подсказку.

Отсутствует

 

№1408418-01-2020 20:18:18

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 71.0

Re: Custom Buttons

Dumby
Норм...Была еще одна шняга -  saveItem.onclick =()=> saveSelectionToTxt(); , а было ToFile
ЭТО Кн. Скриншот !

скрытый текст

Выделить код

Код:

// Save, от 07.03.2017. .............

self.label = "Save";
self._handleClick =()=> menuPopup.openPopup(this, "after_start");
self.image = "";



var folderpath="C:\\Users\\_________\\Desktop";         // папка для сохранения иконок для ярлыков и ярлыков сайтов


// Создать меню для кнопки .............
var array = [
   { label: "Сохранить значок веб-сайта", func: "saveFavicon()", image: ""},
   { label: "Запомнить значок веб-сайта как base64", func: "copyFaviconData()", image: ""},  
   { separator: ''},
   { label: "Сохранить ярлык страницы как…", func: "saveShortcuts()", image: ""},
   { separator: ''},  
   { label: "Кодировать изображение(текст.файл) в base64", func: "copyFaviconbase()", image: ""},
   
   { label: "Сохранить всю страницу как PDF", func: "savePageToPDF()", image: ""},
   { label: "Сохранить всю страницу или выбранное как HTML", func: "savePageToHTML()", image: ""},
   { separator: ''},
   { label: "Сохранить всю страницу как HTML", func: "savePage()", image: ""},
   { label: "Сохранить выделенный текст как txt файл", func: "saveSelectionToTxt()", image: ""},
   { separator: ''},
   { label: "Запомнить изображение как base64, в контекстном меню", value: "CB.Save.WebScreenShotOnImage"},
   { label: "Сохранить выделенный текст в файл, в контекстном меню", value: "CB.Save.SelectionToFile" },
   { label: "Открыть выделенный текст в внешнем редакторе, в контекстном меню", value: "CB.Save.TextToEditor"},
];

var menuPopup = self.appendChild(document.createXULElement("menupopup"));
array.forEach((m,i)=> {
   if ("separator" in m) { menuPopup.appendChild(document.createXULElement("menuseparator")); return };
   var mItem = menuPopup.appendChild(document.createXULElement("menuitem"));
   mItem.setAttribute("label", m.label);
   mItem.setAttribute("class", "menuitem-iconic");
   if ("image" in m) mItem.setAttribute("image", m.image || array[i-1].image); 
   if ("value" in m) { 
       mItem.setAttribute('type', 'checkbox');
       mItem.setAttribute('checked', cbu.getPrefs(m.value) );
       mItem.onclick =()=> cbu.setPrefs(m.value, !cbu.getPrefs(m.value));
       }
   if ("func" in m) mItem.addEventListener("command", ()=> eval(m.func.toString()));
});
menuPopup.setAttribute("onclick", "event.stopPropagation()");


function aDate() {
 var t=new Date();
 var y=1900+t.getYear();
 var min=t.getMinutes(); if (min<10){min="0"+min};
 var h=t.getHours();
 var m=t.getMonth();switch(m){case 0: m="января";break;case 1: m="февраля";break;case 2: m="марта";break;case 3: m="апреля";break;case 4: m="мая";break;case 5: m="июня";break;case 6: m="июля";break;case 7: m="августа";break;case 8: m="сентября";break;case 9: m="октября";break;case 10: m="ноября";break;default: m="декабря";}
 var d=t.getDate();
 var curdate=d+" "+m+" "+y+" "+"г";
 var myfilename=curdate;
 return myfilename;
}
 

function WebScreenShotonImage(image) {
      var canvas = document.createElementNS(xhtmlns, 'canvas');
      canvas.width = image.naturalWidth;
      canvas.height = image.naturalHeight;
      var ctx = canvas.getContext('2d');
      ctx.drawImage(image, 0, 0);
      var base64 = canvas.toDataURL();
      gClipboard.write(base64);
   
      // стиль для изображение в сплывающей подсказке ....
      var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
      var uri = makeURI('data:text/css,'+ encodeURIComponent('#alertImage { height: 25px !important; width: 25px !important; }'));
      sss.loadAndRegisterSheet(uri, 0);
      
     // alertsService.showAlertNotification(base64, self.label, "Запомнил изображение как base64", false, "", (s, t)=> { 
     Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(base64, self.label, "Запомнил изображение как base64", false, "", (s, t)=> {
         if (t == 'alertfinished')
             sss.unregisterSheet(uri, 0); // удалить стиль когда подсказка закрывается
      }, "");
};



var saveToFile = function (fileContent, fileName) {
    var uc = Components.classes['@mozilla.org/intl/scriptableunicodeconverter'].createInstance(Components.interfaces.nsIScriptableUnicodeConverter);
    uc.charset = 'utf-8';
    fileContent = uc.ConvertFromUnicode(fileContent);

    var nsIFilePicker = Components.interfaces.nsIFilePicker;
    var fp = Components.classes['@mozilla.org/filepicker;1'].createInstance(nsIFilePicker);
    fp.init(window, '', fp.modeSave);
    fp.defaultString = fileName;
    fp.appendFilters(fp.filterHTML);
    fp.appendFilters(fp.filterAll);
    fp.open(function (rv) {
  if (rv == nsIFilePicker.returnOK || rv == nsIFilePicker.returnReplace) {
    var stream = Components.classes['@mozilla.org/network/file-output-stream;1'].createInstance(Components.interfaces.nsIFileOutputStream);
    stream.init(fp.file, 0x02|0x20|0x08, 0666, 0);
    stream.write(fileContent, fileContent.length);
    stream.close();
  }
});
};


function savePageToHTML() {

var vert=`javascript:(function(){var getSelWin=function(w){if(w.getSelection().toString())return w;for(var i=0,f,r;f=w.frames[i];i++){try{if(r=getSelWin(f))return r}catch(e){}}};var selWin=getSelWin(window),win=selWin||window,doc=win.document,loc=win.location;var qualifyURL=function(url,base){if(!url||/^([a-z]+:|%23)/.test(url))return url;var a=doc.createElement('a');if(base){a.href=base;a.href=a.protocol+(url.charAt(0)=='/'%3F(url.charAt(1)=='/'%3F'':'//'+a.host):'//'+a.host+a.pathname.slice(0,(url.charAt(0)!='%3F'&&a.pathname.lastIndexOf('/')+1)||a.pathname.length))+url}else{a.href=url};return a.href};var encodeImg=function(src,obj){var canvas,img,ret=src;if(/^https%3F:%5C/%5C//.test(src)){canvas=doc.createElement('canvas');if(!obj||obj.nodeName.toLowerCase()!='img'){img=doc.createElement('img');img.src=src}else{img=obj};if(img.complete)try{canvas.width=img.width;canvas.height=img.height;canvas.getContext('2d').drawImage(img,0,0);ret=canvas.toDataURL((/%5C.jpe%3Fg/i.test(src)%3F'image/jpeg':'image/png'))}catch(e){};if(img!=obj)img.src='about:blank'};return ret};var toSrc=function(obj){var strToSrc=function(str){var chr,ret='',i=0,meta={'%5Cb':'%5C%5Cb','%5Ct':'%5C%5Ct','%5Cn':'%5C%5Cn','%5Cf':'%5C%5Cf','%5Cr':'%5C%5Cr','%5Cx22':'%5C%5C%5Cx22','%5C%5C':'%5C%5C%5C%5C'};while(chr=str.charAt(i++)){ret+=meta[chr]||chr};return'%5Cx22'+ret+'%5Cx22'},arrToSrc=function(arr){var ret=[];for(var i=0;i<arr.length;i++){ret[i]=toSrc(arr[i])||'null'};return'['+ret.join(',')+']'},objToSrc=function(obj){var val,ret=[];for(var prop in obj){if(Object.prototype.hasOwnProperty.call(obj,prop)&&(val=toSrc(obj[prop])))ret.push(strToSrc(prop)+': '+val)};return'{'+ret.join(',')+'}'};switch(Object.prototype.toString.call(obj).slice(8,-1)){case'Array':return arrToSrc(obj);case'Boolean':case'Function':case'RegExp':return obj.toString();case'Date':return'new Date('+obj.getTime()+')';case'Math':return'Math';case'Number':return isFinite(obj)%3FString(obj):'null';case'Object':return objToSrc(obj);case'String':return strToSrc(obj);default:return obj%3F(obj.nodeType==1&&obj.id%3F'document.getElementById('+strToSrc(obj.id)+')':'{}'):'null'}};var ele,pEle,clone,reUrl=/(url%5C(%5Cx22%3F)(.+%3F)(%5Cx22%3F%5C))/g;if(selWin){var rng=win.getSelection().getRangeAt(0);pEle=rng.commonAncestorContainer;ele=rng.cloneContents()}else{pEle=doc.documentElement;ele=(doc.body||doc.getElementsByTagName('body')[0]).cloneNode(true)};while(pEle){if(pEle.nodeType==1){clone=pEle.cloneNode(false);clone.appendChild(ele);ele=clone};pEle=pEle.parentNode};var sel=doc.createElement('div');sel.appendChild(ele);for(var el,all=sel.getElementsByTagName('*'),i=all.length;i--;){el=all[i];if(el.style&&el.style.backgroundImage)el.style.backgroundImage=el.style.backgroundImage.replace(reUrl,function(a,b,c,d){return b+encodeImg(qualifyURL(c))+d});switch(el.nodeName.toLowerCase()){case'link':case'style':case'script':el.parentNode.removeChild(el);break;case'a':case'area':if(el.hasAttribute('href')&&el.getAttribute('href').charAt(0)!='%23')el.href=el.href;break;case'img':case'input':if(el.hasAttribute('src'))el.src=encodeImg(el.src,el);break;case'audio':case'video':case'embed':case'frame':case'iframe':if(el.hasAttribute('src'))el.src=el.src;break;case'object':if(el.hasAttribute('data'))el.data=el.data;break;case'form':if(el.hasAttribute('action'))el.action=el.action;break}};var head=ele.insertBefore(doc.createElement('head'),ele.firstChild);var meta=doc.createElement('meta');meta.httpEquiv='content-type';meta.content='text/html; charset=utf-8';head.appendChild(meta);var title=doc.getElementsByTagName('title')[0];if(title)head.appendChild(title.cloneNode(true));head.copyScript=function(){if('$'in win)return;var f=doc.createElement('iframe');f.src='about:blank';f.setAttribute('style','position:fixed;left:0;top:0;visibility:hidden;width:0;height:0;');doc.documentElement.appendChild(f);var str,script=doc.createElement('script');script.type='text/javascript';for(var name in win){if(name in f.contentWindow||!/^[a-zA-Z_$][0-9a-zA-Z_$]*$/.test(name))continue;try{str=toSrc(win[name]);if(!/%5C{%5Cs*%5C[native code%5C]%5Cs*%5C}/.test(str)){script.appendChild(doc.createTextNode('var '+name+' = '+str.replace(/<%5C/(script>)/ig,'<%5C%5C/$1')+';%5Cn'))}}catch(e){}};f.parentNode.removeChild(f);if(script.childNodes.length)this.nextSibling.appendChild(script)};head.copyScript();head.copyStyle=function(s){if(!s)return;var style=doc.createElement('style');style.type='text/css';if(s.media&&s.media.mediaText)style.media=s.media.mediaText;try{for(var i=0,rule;rule=s.cssRules[i];i++){if(rule.type!=3){if((!rule.selectorText||rule.selectorText.indexOf(':')!=-1)||(!sel.querySelector||sel.querySelector(rule.selectorText))){style.appendChild(doc.createTextNode(rule.cssText.replace(reUrl,function(a,b,c,d){var url=qualifyURL(c,s.href);if(rule.type==1&&rule.style&&rule.style.backgroundImage)url=encodeImg(url);return b+url+d})+'%5Cn'))}}else{this.copyStyle(rule.styleSheet)}}}catch(e){if(s.ownerNode)style=s.ownerNode.cloneNode(false)};this.appendChild(style)};var sheets=doc.styleSheets;for(var j=0;j<sheets.length;j++)head.copyStyle(sheets[j]);head.appendChild(doc.createTextNode('%5Cn'));var doctype='',dt=doc.doctype;if(dt&&dt.name){doctype+='<!DOCTYPE '+dt.name;if(dt.publicId)doctype+=' PUBLIC %5Cx22'+dt.publicId+'%5Cx22';if(dt.systemId)doctype+=' %5Cx22'+dt.systemId+'%5Cx22';doctype+='>%5Cn'};var href = 'data:text/html;charset=utf-8,' + encodeURIComponent(doctype + sel.innerHTML + '\n<!-- This document saved from ' + (loc.protocol != 'data:' ? loc.href : 'data:uri') + ' -->');var a = document.documentElement.appendChild(document.createElement("a"));a.setAttribute("href", href);var name = selWin ? win.getSelection().toString() : (title && title.text ? title.text : loc.pathname.split('/').pop());name=name.replace(/[:\\\/<>?*|"]+/g, '_').replace(/\s+/g, ' ').slice(0, 100).replace(/^\s+|\s+$/g, '');name += (function () {var d = new Date(), z=function(n){return '_' + (n < 10 ? '0' : '') + n};return z(d.getHours()) + z(d.getMinutes()) + z(d.getSeconds());})();a.setAttribute("download", name + ".html");a.click();a.remove();})();`;
gBrowser. loadURI(vert, {triggeringPrincipal: Services.scriptSecurityManager.getSystemPrincipal()});

};

function savePage() {
 saveBrowser(gBrowser.selectedBrowser);
};

function saveShortcuts() {
var file = Components.classes["@mozilla.org/file/local;1"].
           createInstance(Components.interfaces.nsIFile);
file.initWithPath(folderpath);

if( !file.exists() || !file.isDirectory() ) {   file.create(Components.interfaces.nsIFile.DIRECTORY_TYPE, 0x1B6);}

var savetodir=folderpath+"\\"; 
var urllink=gBrowser.currentURI.spec;
var out=getTabLabel();
var filename=savetodir+out+'.url';
var data="[InternetShortcut]\r\nURL="+urllink+"\r\n";

saveToFile(data, filename);
 // стиль для изображение в сплывающей подсказке ....
      var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
      var uri = makeURI('data:text/css,'+ encodeURIComponent('#alertImage { height: 25px !important; width: 25px !important; }'));
      sss.loadAndRegisterSheet(uri, 0);
   // подсказка
   var notific = 'Сохранил в: ' + folderpath;
   var image = gBrowser.selectedBrowser.mIconURL;
   Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(image, filename, notific);
};

// Кодировать изображение или текстовой файл в base64 .............
function copyFaviconbase(){
var fp = window.makeFilePicker();
fp.init(window, "Открыть файл", fp.modeOpen);
fp.appendFilter("Text and images", "*.txt; *.text; *.css; *.js; *.ini; *.rdf; *.xml; *.html; *.htm; *.shtml; *.xhtml; *.jpe; *.jpg; *.jpeg;\
                                    *.gif; *.png; *.bmp; *.ico; *.svg; *.svgz; *.tif; *.tiff; *.ai; *.drw; *.pct; *.psp; *.xcf; *.psd; *.raw");
  fp.open(re=> { 
  if ( re != fp.returnOK ) return;
   var file = fp.file;
   var inputStream = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(Ci.nsIFileInputStream);
   inputStream.init(file, 0x01, 0600, 0);
   var stream = Cc["@mozilla.org/binaryinputstream;1"].createInstance(Ci.nsIBinaryInputStream);
   stream.setInputStream(inputStream);
   var encoded = btoa(stream.readBytes(stream.available()));
   var contentType = Cc["@mozilla.org/mime;1"].getService(Ci.nsIMIMEService).getTypeFromFile(file);
   var dataURI = "data:" + contentType + ";charset=utf-8;base64," + encoded;
   gClipboard.write(dataURI);
   //Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(dataURI, self.label, "Скопировал файл как base64");
    // стиль для изображение в сплывающей подсказке ....
      var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
      var uri = makeURI('data:text/css,'+ encodeURIComponent('#alertImage { height: 25px !important; width: 25px !important; }'));
      sss.loadAndRegisterSheet(uri, 0);
      
     // alertsService.showAlertNotification(base64, self.label, "Запомнил изображение как base64", false, "", (s, t)=> { 
     Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(dataURI, self.label, "Запомнил изображение как base64", false, "", (s, t)=> {
         if (t == 'alertfinished')
             sss.unregisterSheet(uri, 0); // удалить стиль когда подсказка закрывается
      }, "");
});
};

// Сохранить страницу как PDF файл через сервис 'pdfmyurl.com' .............
function savePageToPDF() {
      var loc = gBrowser.currentURI.spec;
   var vert = "http://pdfmyurl.com?url=" + loc;
  
   gBrowser. loadURI(vert, {
   triggeringPrincipal: Services.scriptSecurityManager.getSystemPrincipal()
   });
}; 

// Сохранить иконку текущего сайта с диалогом сохранения .............
function saveFavicon() {
       var uri = gBrowser.currentURI;
       function getSiteName() {
                  try { var domain = uri.host.split('.') } catch(e) { return "" };
                   domain = (domain.length == 2) ? domain[0] : domain[1]
                   return domain.charAt(0).toUpperCase() + domain.slice(1).split('.')[0] + " ";  
            };
      saveImageURL(gBrowser.selectedTab.image, getSiteName(), null, false, false, null, window.document);
};


// Скопировать иконку текущего сайта как base64 код .............
function copyFaviconData() {
   var img = new Image();
   img.src = gBrowser.selectedTab.image;
   WebScreenShotonImage(img);
};


// Сохранить выделенный текст или весь текст на странице как txt файл .............
function saveSelectionToTxt() {

let browserMM = gBrowser.selectedBrowser.messageManager;
        browserMM.addMessageListener('getSelection', function listener(message) {
        var sel = message.data;
       !sel && document.getElementById("cmd_selectAll").doCommand(); 
     
   // создать название файла из заголовка страницы и текущего времени и сохранить текст ....
   var fileTitle = getTabLabel() + '  ' + aDate().replace(/:/g, ".");
   saveURL("data:text/plain," + encodeURIComponent(gBrowser.currentURI.spec + ("\r\n\r\n" + sel)), 
                                fileTitle + ".txt", null, false, false, null, window.document);
   !sel && goDoCommand("cmd_selectNone"); 
 browserMM.removeMessageListener('getSelection', listener, true);
});
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSelection", content.document.getSelection().toString())', false);
};

(popup => addEventListener("popupshowing", {
    handleEvent(e) {
        if (this.shouldHide) return;

        var menuitem = document.createXULElement("menuitem");
        menuitem.id = "content-baseItem";
        menuitem.className = "menuitem-iconic";
        menuitem.setAttribute("oncommand", "copyImageAsBase64()");
        menuitem.setAttribute("label", "Запомнить изображение как base64");
        menuitem.setAttribute("image", "");
        popup.append(menuitem);
        addDestructor(() => menuitem.remove());

        menuitem.copyImageAsBase64 = () => gBrowser.selectedBrowser.messageManager
            .loadFrameScript("data:;charset=utf-8," + encodeURIComponent(this.code()), false);

        this.handleEvent = () => menuitem.hidden = this.shouldHide;
    },
    get shouldHide() {
        return !gContextMenu.onImage || !cbu.getPrefs("CB.Save.WebScreenShotOnImage");
    },
    code: () => `(selectors => {

        var getImage = doc => {
            var elm = doc.querySelector(selectors.shift());
            if (selectors.length) elm = getImage(elm.contentDocument);
            return elm;
        }
        var image = getImage(content.document);

        var canvas = image.ownerDocument.createElementNS("${xhtmlns}", "canvas");
        canvas.width = image.naturalWidth;
        canvas.height = image.naturalHeight;
        var ctx = canvas.getContext("2d");
        ctx.drawImage(image, 0, 0);
        var base64 = canvas.toDataURL();

        Cc["@mozilla.org/widget/clipboardhelper;1"].getService(Ci.nsIClipboardHelper)
            .copyStringToClipboard(base64, Ci.nsIClipboard.kGlobalClipboard);

        Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService)
            .showAlertNotification(base64, "${self.label}", "Запомнил изображение как base64");
    })(${
        JSON.stringify(gContextMenu.targetSelectors)
    })`
}, false, popup || 1))(document.getElementById("contentAreaContextMenu"));


// Добавляем в контекстного меню страницы новые пункты .............
((contextMenu, el)=> {

   // в контекстного меню выделенного текста ....
   var saveItem = contextMenu.insertBefore(document.createXULElement("menuitem"), el);
   saveItem.id = "content-saveItem";
   saveItem.setAttribute("label", "Сохранить выделенный текст в файл");
   saveItem.setAttribute("class", "menuitem-iconic");
   saveItem.setAttribute("image", ""); 
   saveItem.onclick =()=> saveSelectionToTxt();

   var editorItem = contextMenu.insertBefore(document.createXULElement("menuitem"), el);
   editorItem.id = "content-editorItem";
   editorItem.setAttribute("label", "Открыть выделенный текст в внешнем редакторе");
   editorItem.setAttribute("class", "menuitem-iconic");
   editorItem.setAttribute("image", ""); 
   editorItem.onclick =()=> textToEditor();


    // устанавливаем где и при каких настройках показывать новые пункты ....
   addEventListener('popupshowing', e=> {
      if (e.target != e.currentTarget) return;
      var sel = gContextMenu.isTextSelected;
      saveItem.hidden = !sel || !cbu.getPrefs("CB.Save.SelectionToFile");
      editorItem.hidden = !sel || !cbu.getPrefs("CB.Save.TextToEditor"); 
      }, false, contextMenu);

   // удалять новые пункти при изминениях ....
   addDestructor(()=> {
      saveItem.remove(); editorItem.remove();
   });   
})(document.getElementById("contentAreaContextMenu"), document.getElementById("context-sep-open"));


// Сохранить выделенный текст в файл на рабочем столе .............
function saveSelectionToFile() {

 let browserMM = gBrowser.selectedBrowser.messageManager;
 browserMM.addMessageListener('getSelect', function listener(message) {
   // создать текст для записи
   var url = gBrowser.currentURI.spec;
   if (/\.рф/.test(url.host)) url = convertFromUnicode("UTF-8", url);
   
   var time = convertFromUnicode("UTF-8", aDate().replace(/:/g, "."));
   var text = convertFromUnicode("UTF-8", message.data); 
   var title = convertFromUnicode("UTF-8", getTabLabel());
   
   var text = "..............................................................\n"
            + title + " - " + time + "\n" + url + "\n\n" + text + "\n\n\n";
   var text = text.replace(/\u000A/g, "\u000D\u000A").replace(/\u000D\u000D\u000A/g, "\u000D\u000A");

   // путь к файлу и название файла
   var file = Services.dirsvc.get("Desk", Ci.nsIFile); 
   file.append("Save - " + (aDate().replace(/:/g, ".")) + ".txt");
          
   // создать файл с текстом или добавлять текст в файл
   var foStream = Cc["@mozilla.org/network/file-output-stream;1"].createInstance(Ci.nsIFileOutputStream);
   file.exists() ? foStream.init(file, 0x02 | 0x10, 0664, 0) : foStream.init(file, 0x02|0x08|0x20, 0666, 0);
   foStream.write(text, text.length);
   foStream.close();
    // всплывающая подсказка дает возможность открыть файл если кликнуть на подсказке
       var notificat = 'Сохранил выделенный текст в файл на рабочий стол'; 
   var image = gBrowser.selectedTab.image || self.image;
   Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService)
    .showAlertNotification(image, notificat, "Кликни чтобы открыть файл", true, "", (s, t)=> { 
      if (t == 'alertclickcallback') file.launch();
   }, "");
 browserMM.removeMessageListener('getSelect', listener, true);
});
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSelect", content.document.getSelection().toString())', false);

};


// Создать текстовой файл с выделенным текстом в папке профиля и открыть в редакторе .............
function textToEditor() {


 let browserMM = gBrowser.selectedBrowser.messageManager;
 browserMM.addMessageListener('getSelect', function listener(message) {
   // создать текст для записи
    function convertFromUnicode(...args) {
    var suc = Cc['@mozilla.org/intl/scriptableunicodeconverter']
        .createInstance(Ci.nsIScriptableUnicodeConverter);
    suc.charset = "UTF-8";
    return (convertFromUnicode =
        (a, text) => suc.ConvertFromUnicode(text) + suc.Finish()
    )(...args);
}

    var text = convertFromUnicode("UTF-8", message.data); 
    var file = Services.dirsvc.get('ProfD', Ci.nsIFile);
   file.append("TextToEditor.txt");
   custombuttonsUtils.writeFile(file.path, text);
   file.launch(); 
          

 browserMM.removeMessageListener('getSelect', listener, true);
});
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSelect", content.document.getSelection().toString())', false);
};




// Получить название вкладки без не сохраняемых символов и лишних пробелов ..............
function getTabLabel() { 
   var label = gBrowser.selectedTab.label;      
   var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " ");
   return label.substring(0, 50);
};
((main, parts) => this.onmousedown = e => {
    if (e.button) return;
    this.onmousedown = null;

    var df = MozXULElement.parseXULToFragment(`
        <menugroup orient="vertical">
            <menuseparator/>
            <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"/>
        </menugroup>
    `);
    var menugroup = df.firstChild;
    menugroup.setAttribute("context", "");
    menugroup.setAttribute("oncommand", "handleCommand(event);");
    menugroup.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));

        (menugroup.handleCommand = e => gBrowser.selectedBrowser.messageManager
            .loadFrameScript(urls[e.target.value], false)
        )(e);
    }
    menuPopup.querySelector('menuitem[label*="ярлык"]').after(df);
})(`
    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));
        }
    }`
}); 

this.oncontextmenu =e=> { e.button && !e.ctrlKey && e.preventDefault() };


// Получить название домена с заглавным первым символом и без приставок( типа .ru и .com ) ..............
function getSiteName() {
   try { var domain = content.document.domain.split('.') } catch(e) { return "" };
   domain = (domain.length == 2) ? domain[0] : domain[1]
   return domain[0].toUpperCase() + domain.slice(1).split('.')[0] + " ";  
};


// Получить название вкладки без не сохраняемых символов и лишних пробелов ..............
function getTabLabel() { 
   var label = gBrowser.selectedTab.label;      
   var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " ");
   return label;
};

    
// Получить выделенный текст из страницы или 'false' ..............
function getSelect() {
   var el = document.commandDispatcher.focusedElement;
   try { return el.value.substring(el.selectionStart, el.selectionEnd) } catch(e) {};
   var sel = document.commandDispatcher.focusedWindow.getSelection();
   return (sel == '') ? false : sel.toString().replace(/^\s+|\s+$/g,"").replace(/\u000A/g, "\u000D\u000A").replace(/\u000D\u000D\u000A/g, "\u000D\u000A");
};

     


Кстати, можно ли рихтануть по-взрослому 71 , (за 72 молчу) насчет SSl/TLS , а то прокси не берет от Frigate , только Socks могу, а там тормоза...

Отредактировано solombala (18-01-2020 20:31:32)

Отсутствует

 

№1408518-01-2020 21:16:31

rubel
Участник
 
Группа: Members
Откуда: г.Самара
Зарегистрирован: 10-05-2005
Сообщений: 392
UA: Firefox 68.0

Re: Custom Buttons

solombala
А как бы убрать лишнее вот такое окно при сохранении  "Сохранить всю страницу или выбранное как HTML"
В кнопке Save, от 07.03.2017
https://i111.fastpic.ru/big/2020/0118/c6/4baada21edf8e71cdd55757dd99733c6.png

Отсутствует

 

№1408618-01-2020 21:42:14

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 71.0

Re: Custom Buttons

rubel
Аддон поставь и "сохранить"
https://addons.mozilla.org/ru/firefox/a … src=search

Отсутствует

 

№1408718-01-2020 22:06:58

rubel
Участник
 
Группа: Members
Откуда: г.Самара
Зарегистрирован: 10-05-2005
Сообщений: 392
UA: Firefox 68.0

Re: Custom Buttons

solombala пишет:

rubel
Аддон поставь и "сохранить"

Поставил, ничего не изменилось. Вываливается тоже окошко. :/

Отсутствует

 

№1408818-01-2020 23:09:36

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 1072
UA: Firefox 52.0

Re: Custom Buttons

kokoss пишет:

Можно поподробнее?

Можно

скрытый текст
Вот смотри, три фрагмента кода.

В первом два объекта с данными для создания
пунктов меню Search plugins.

Во втором вычисляется номер версии Firefox.

В третьем, собственно, проверка. Если версия Firefox больше 40
и название пункта меню должно начинаться с «Search»,
то такой пункт не создаётся.

Таким образом, эти три фрагмента можно удалить,
ведь 72 больше чем 40, и намного.

А можно и не удалять, работать будет и так,
просто ненужная лишняя обработка.

Выделить код

Код:

     {
      label: "Search plugins(default)",
      style: "chrome://global/skin/icons/Search-glass.png",
      value: "CurProcD,searchplugins" 
     },
     {
      label: "Search plugins(user-defined)",
      style: "chrome://global/skin/icons/Search-glass.png",
      value: "UsrSrchPlugns"
     },
Выделить код

Код:

var version = parseInt(Services.appinfo.version);
Выделить код

Код:

   if ( version > 40 && m.label.startsWith("Search") ) return;

Отсутствует

 

№1408918-01-2020 23:24:37

kokoss
Участник
 
Группа: Members
Зарегистрирован: 15-02-2018
Сообщений: 685
UA: Firefox 52.0

Re: Custom Buttons

Dumby пишет:

Код:

var version = parseInt(Services.appinfo.version);


Код:

   if ( version > 40 && m.label.startsWith("Search") ) return;

Понятно!

Отсутствует

 

№1409019-01-2020 14:03:29

func4ptch4
Участник
 
Группа: Members
Зарегистрирован: 03-05-2018
Сообщений: 198
UA: Firefox 72.0

Re: Custom Buttons

xrun1, это tooltipText, autopopup когда меню раскрывается, как при клике(в общем экономит 2клика), ну и легко между несколькими кнопками не надо на каждый кликать... может на топбаре это и не удобно, но в вертикальной очень даже, если там еще закладками пользуетесь, на топбаре просто при любом наведений будет мешать там ведь юрлбар и т.д. + я там не держу CB-кнопки, удобно) там только аддоны uc.js скрипты, 1-2 возле вертикальной гамбургер-меню тамже..


Dumby, вроде это отвечает this._handleClick=()=>menuPopup.openPopup(this,"after_start"); за открытие? хотел еще пару кодов но тут ограничение в 64кб, в основном там с этой секцией, может пример а там я сам попробую изменить, или не только там надо править?
скрытый текст

Выделить код

Код:

//Quick toggle for about:config preferences [Fx]
// Быстрое переключение параметров about:config от 24.07.2016
this._handleClick=()=>menuPopup.openPopup(this,"after_start"); //(this,-1,-1,"popup","bottomleft","topleft");
var menuPopup=document.createXULElement("menupopup");self.prepend(menuPopup); //var menuPopup=self.appendChild(document.createXULElement("menupopup")); [old autopopup]
menuPopup.id='quick-aboutconfig-menupopup';

// Изменить иконку при несоответствие любого параметра пользовательскому предпочтению (см.ниже)||Иконка меняется только при изменеии параметров через меню кнопки, либо после его открытия.
  var s='CB.hasNotUserChoice';function toggleImage(){var val=custombuttons.getPrefs(s);self.image=val
//var s='CB.hasNotUserChoice';function toggleImage(){        custombuttons.getPrefs(s)?self.style.cssText='':self.style.cssText='filter:grayscale(100%)';};
? ''
: '';};
    toggleImage();Services.prefs.addObserver(s,toggleImage,false);
addDestructor(()=>Services.prefs.removeObserver(s,toggleImage));

// nodeName: menuitem - для логических(boolean) параметров, menu - для целых(integer) и строковых(string). menuseparator - для разделителя.
// pref - параметр about:config.
// Параметры имеющие значения отличные от дефолтных - выделены жирным стилем текста.
// restart (задавать с пустым значением. т.е., restart: "") - добавляет возможность перезапуска браузера
// (с подтверждением в диалоговом окне) после изменения параметра.
// key - задает accesskey - клавиши для быстрой навигации по меню.
// userChoice - задает предпочитаемое значение и если текущее значение с ним не совпадает, пункт меню/название меню помечаются красным цветом.
// Также можно установить предупреждающую иконку для таких пунктов. См. стиль в посте ккнопки.
// А также меняется иконка самой кнопки (см. выше).
// strValues - значения и отображаемое в меню название значения. Задавать для целых(integer) и строковых(string) параметров.
// Задается в виде: значение,,,название,,,accesskey|||значение2,,,название2,,,accesskey2|||значение3,,,название3 и т.д. (accesskey - задается опционально)
// Полное значение отображается в подсказках, при наведении на название подменю/пункт подменю.
// Для логических(boolean) - отображается сразу после самого параметра (значение true - также ставит галочку для него).
// ЛКМ по пунктам меню - перключает значения для логических(boolean) параметров,
// любая кнопка по пунктам в субменю - задает это значение для целых(integer) и строковых(string) параметров.
// ПКМ по пунктам меню и названию субменю - сбрасывает значение параметра в дефолтное.
// Клавиатура: Enter - переключение параметра.
// Спецклавиша вызова контекстного меню / Shift+Enter - сброс в дефолтное значение.
// Alt + M - открыть меню кнопки. (Сочетание можно сменить на свое. См. в конце кода)
[ //{nodeName:"menuitem", name:"Откл. дискового кэша", pref:"browser.cache.disk.enable", userChoice:"false"},
  //{nodeName:"menuitem", name:"Откл. кэша в оперативной памяти", pref:"browser.cache.memory.enable", userChoice:"false"},
  //{nodeName:"menuseparator"},
  //{nodeName:"menuitem", name:"Откл. локального хранилища DB (Storage)", pref:"dom.indexedDB.enabled", key:'d'},
  //{nodeName:"menuitem", name:"Откл. локального хранилища", pref:"dom.storage.enabled", key:'s'},
  //{nodeName:"menuseparator"},
  //{nodeName:"menu", name:"Back-story-cash [Tessssttt]", pref:"browser.sessionhistory.max_total_viewers", strValues:"0,,,0"},
  //{nodeName:"menu", pref:"image.animation_mode", key:'i', userChoice:"none", strValues:"normal,,,Анимация картинок вкл.,,,|||none,,,Анимация картинок выкл.,,,"},
  //{nodeName:"menu", pref:"network.cookie.cookieBehavior", key:'k', userChoice:"1", strValues:"2,,,Не принимать куки с сайтов,,,|||0,,,Принимать куки со сторонних сайтов всегда,,,|||3,,,Принимать куки со сторонних посещённых сайтов,,,|||1,,,Принимать куки со сторонних сайтов никогда,,,"},
  //{nodeName:"menuseparator"},
  //{nodeName:"menu", pref:"general.useragent.locale", key:'l', restart:"", strValues:"en-US,,,English,,,e|||ru,,,Русский,,,r"},
  //{nodeName:"menu", name:"language", pref:"intl.accept_languages", strValues:"en-US, en;q=0.5,,,en-US, en;q=0.5,,,e|||en-US, en, ru-RU, ru,,,en-US, en, ru-RU, ru,,,r"},
  //{nodeName:"menu", pref:"browser.display.document_color_use", key:'c', userChoice:"0", strValues:"0,,,Automatic,,,0|||1,,,Always,,,1|||2,,,Never,,,2"},
  //{nodeName:"menu", pref:"CB.TEST", key:'t', userChoice:"C:\\Downloads\\TEST1", strValues:"C:\\Downloads\\TEST1,,,TEST1,,,1|||C:\\Downloads\\TEST2,,,TEST2,,,2"}
  //{nodeName:"menuseparator"},
  //{nodeName:"menuitem", name:"Откл. инфу начало/конец загрузки стр", pref:"dom.enable_performance", userChoice:"false"},
  //{nodeName:"menu", name:"Вкл/Выкл Referer", pref:"network.http.sendRefererHeader", strValues:"0,,,0"},
  //{nodeName:"menuitem", name:"В качестве реферера корень сайта", pref:"network.http.referer.spoofSource", userChoice:"true"},
  //{nodeName:"menu", name:"referer.trimmingPolicy", pref:"network.http.referer.trimmingPolicy", strValues:"2,,,2"},
  //{nodeName:"menuseparator"},
  //{nodeName:"menu", name:"On/Off Image", pref:"permissions.default.image", userChoice:1, strValues:"2,,,Off"},

  //{nodeName:"menuitem", name:"On/Off javascript", pref:"javascript.enabled", key:'j', userChoice:"true"},
  {nodeName:"menuitem", name:"On/Off useragentS", pref:"general.useragent.site_specific_overrides", userChoice:"true"},
  //{nodeName:"menuitem", pref:"dom.workers.enabled", key:'w', userChoice:"false"},
  //{nodeName:"menu", name:"Stopautoplay", pref:"media.autoplay.default", userChoice:0, strValues:"0,,,Stop,,,0|||1,,,Play,,,1"},
  //{nodeName:"menuitem", pref:"xpinstall.signatures.required"},                       //Check is compatibility
  //{nodeName:"menuitem", pref:"browser.bookmarks.autoExportHTML"},                    //BookmarksHtml [false=places.sqlite]
  //{nodeName:"menuitem", pref:"media.peerconnection.enabled"},                        //WebRTC false=off!

  {nodeName:"menuitem", name:"Drag&Drop(Safe) !Manual restart required!", pref:"browser.launcherProcess.enabled", userChoice:"true"},   //Drag&Drop(Safe) Manual restart required! - forum.mozilla-russia.org/viewtopic.php?pid=774980#p774980
  {nodeName:"menuitem", name:"On/Off media.play-stand-alone", pref:"media.play-stand-alone"},   //сразу скачивать медиафайлы, без перехода на новую страницу.
  {nodeName:"menuseparator"},
  {nodeName:"menu", name:"On/Off DoH", pref:"network.trr.mode", strValues:"3,,,3"},
  {nodeName:"menu", name:"Configure Proxies", pref:"network.proxy.type", userChoice:5, strValues:"1,,,httpP,,,1|||2,,,AutomaticP,,,2|||5,,,Use systemP,,,5"},
  {nodeName:"menu", name:"AutomaticP", pref:"network.proxy.autoconfig_url", strValues:"resource://FF/..pac,,,resource://FF/..pac,,,1|||https://antizapret.prostovpn.org/proxy.pac,,,https://antizapret.prostovpn.org/proxy.pac,,,2"},
  {nodeName:"menuseparator"}, 
  {nodeName:"menu", name:"User Agent", pref:"general.useragent.override", key:'u', strValues:"Mozilla/5.0 (Macintosh; Intel Mac OS X 10.13; rv:60.0) Gecko/20100101 Firefox/60.0,,,Firefox 60/MacOSX 10.13|||Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/66.0.3359.181 Safari/537.36,,,Chrome 66/MacOSX 10.13.5|||Mozilla/5.0 (Macintosh; Intel Mac OS X 10_11_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.133 Safari/537.36,,,Chrome 57/MacOSX|||Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_4) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/10.1 Safari/603.1.30,,,Safari Generic/MacOSX|||Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.133 Safari/537.36,,,Chrome57/W7|||Opera/9.80 (Windows NT 6.2; Win64; x64) Presto/2.12 Version/12.16,,,Opera12/W8|||Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/61.0.3163.98 Safari/537.36,,,Chrome61/W10|||Mozilla/5.0 (Linux; Android 7.0; PLUS Build/NRD90M) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/61.0.3163.98 Mobile Safari/537.36,,,Chrome61/Android7|||Mozilla/5.0 (compatible; Googlebot/2.1; +http://www.google.com/bot.html),,,GoogleBot|||Mozilla/5.0 (compatible; YandexBot/3.0; +http://yandex.com/bots),,,YandexBot|||Mozilla/5.0 (compatible; Yahoo! Slurp; http://help.yahoo.com/help/us/ysearch/slurp),,,YahooBot|||Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm),,,BingBot|||DuckDuck bot/1.0; (+http://duckduckgo.com/duckduckbot.html),,,DuckDuckBot|||Mozilla/5.0 (compatible; Baiduspider/2.0; +http://www.baidu.com/search/spider.html),,,BaiduspiderBot|||ia_archiver (+http://www.alexa.com/site/help/webmasters; crawler@alexa.com),,,AlexaCrawlerBot|||Mozilla/5.0 (Linux; Android 5.1.1; SM-G928X Build/LMY47X) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/47.0.2526.83 Mobile Safari/537.36,,,Samsung Galaxy S6 Edge Plus|||Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 950) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/46.0.2486.0 Mobile Safari/537.36 Edge/13.10586,,,Microsoft Lumia 950|||Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Xbox; Xbox One) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/46.0.2486.0 Mobile Safari/537.36 Edge/13.10586,,,Xbox One|||Mozilla/5.0 (PlayStation 4 3.11) AppleWebKit/537.73 (KHTML, like Gecko),,,Playstation 4|||,,,Пустое значение"},
]

.forEach(function(m){var mItem=document.createElementNS(xulns,m.nodeName);var browserRestart='';
if("restart"in m)browserRestart='if(custombuttons.confirmBox(null,"Restart?","Yes","Cancel"))Services.startup.quit(Services.startup.eAttemptQuit|Services.startup.eRestart);';
if("name"in m) mItem.setAttribute('name',m.name);
if("pref"in m){mItem.setAttribute('closemenu','none');
mItem.setAttribute('oncontextmenu','event.preventDefault();custombuttons.clearPrefs("'+m.pref+'");'+browserRestart);}
if("key"in m)  mItem.setAttribute('accesskey',m.key);
if(m.nodeName==="menuitem"){mItem.setAttribute('type','checkbox');
mItem.setAttribute('oncommand','custombuttons.setPrefs("'+m.pref+'",!custombuttons.getPrefs("'+m.pref+'"));if(event.shiftKey&&event.keyCode==event.DOM_VK_RETURN){event.preventDefault();custombuttons.clearPrefs("'+m.pref+'")};'+browserRestart);}
if(m.nodeName==="menu"){mItem.setAttribute('class','menu-iconic');

var subMenu=mItem.appendChild(document.createXULElement("menupopup"));

for (var value of m.strValues.split('|||')){var submItem=document.createXULElement("menuitem");
var smVal=value.split(',,,')[0];
var smValConv=convertFromUnicode("UTF-8",smVal);
var smName=value.split(',,,')[1];
var key=value.split(',,,')[2];

key&&submItem.setAttribute('accesskey',key);submItem.setAttribute('type','radio');
submItem.setAttribute('label',smName);submItem.setAttribute('tooltiptext',smVal);
submItem.setAttribute('closemenu','none');
submItem.setAttribute('oncommand','try{custombuttons.setPrefs("'+m.pref+'","'+smValConv.replace(/\\/g,'\\\\')+'")}catch(e){Services.prefs.setIntPref("'+m.pref+'","'+smValConv+'")};'+browserRestart);
subMenu.appendChild(submItem);}}
menuPopup.appendChild(mItem);

  // Листенеры отслеживающие переключение параметров
  // и устанавливающие соответствующие названия и чекбоксы для пунктов меню при открытии меню и кликах
  for (var type of ['command','popupshowing','contextmenu']){addEventListener(type,(e)=>{setTimeout(()=>{if ("pref" in m){var val,def;
   def=Services.prefs.prefHasUserValue(m.pref);
   try {val=Services.prefs.getComplexValue(m.pref,Ci.nsISupportsString).data;} catch(e) {
   if (Services.prefs.getPrefType(m.pref)==64) val=custombuttons.getPrefs(m.pref).toString();else val=custombuttons.getPrefs(m.pref);}
   def ? mItem.style.setProperty('font-weight', 'bold', 'important') : mItem.style.removeProperty('font-weight');}

     if (m.nodeName==='menuitem'){mItem.setAttribute('checked',val);mItem.label=(mItem.hasAttribute('name') ? mItem.getAttribute('name') : m.pref)+' - "'+val+'"';
     if ("userChoice" in m){try {var usrChc=(val.toString()===m.userChoice)} catch(e) {usrChc=false};

     mItem.setAttribute('user-choice',usrChc);usrChc ? mItem.style.removeProperty('color') : mItem.style.setProperty('color','orangered','important');}}
     if (subMenu){for (var smitem of subMenu.getElementsByTagName('menuitem')) {var smval=smitem.getAttribute('tooltiptext');smitem.setAttribute('checked',(val===smval) ? true : false);}}
     if (m.nodeName==="menu") {var vname;
     try {vname=subMenu.getElementsByAttribute('checked','true')[0].getAttribute('label');} catch(e) {if (!Services.prefs.prefHasUserValue(m.pref)) vname='Default';else vname='Other';}

     mItem.setAttribute('label',(mItem.hasAttribute('name') ? mItem.getAttribute('name') : m.pref)+' - "'+vname+'"');
     mItem.setAttribute('tooltiptext',val || 'This preferences has null value or does not exist.');

     if ("userChoice" in m) {var smUsrChc=(val===m.userChoice.toString());mItem.setAttribute('user-choice',smUsrChc);
     smUsrChc ? mItem.style.removeProperty('color') : mItem.style.setProperty('color','orangered','important');}}

     if ("userChoice" in m) {var hasNotUserChoice=menuPopup.getElementsByAttribute('user-choice','false')[0];custombuttons.setPrefs(s,hasNotUserChoice ? true : false);}}, 0)},false,menuPopup)}});

// Листенер позволяющий сброс параметров с субменю по Shift+Enter||За код спасибо Dumby
addEventListener("popupshown",{handleEvent:function(e){this[e.type](e);},
  popupshown:function(e){if(e.target!=menuPopup)return;menuPopup.addEventListener   ("popuphidden",this,false);window.addEventListener   ("keydown",this,true);},
 popuphidden:function(e){if(e.target!=menuPopup)return;menuPopup.removeEventListener("popuphidden",this,false);window.removeEventListener("keydown",this,true);},
popupshowing:function(e){e.target.parentNode.removeEventListener("popupshowing",this,false);e.preventDefault();},get old(){delete this.old;
this.e=new MouseEvent("contextmenu",{});return this.old=parseInt(Services.appinfo.platformVersion)<25;},get prop(){delete this.prop;
if("key"in KeyboardEvent.prototype)this.prop="key",this.val="Enter";else this.prop="keyCode",this.val=KeyboardEvent.DOM_VK_RETURN;return this.prop;},
keydown:function(e){if(!e.shiftKey||e.ctrlKey||e.altKey||e[this.prop]!=this.val)return;var target=menuPopup.querySelector("menu[_moz-menuactive]:not([open])");
if(!target)return;this.old ? target.addEventListener("popupshowing",this,false) : e.stopPropagation();target.dispatchEvent(this.e);menuPopup.dispatchEvent(this.e);}},false,menuPopup);

// Открыть меню кнопки по сочетанию клавиш Alt + M (не зависит от текущей раскладки клавиатуры)
// Посмотреть коды клавиш можно здесь: https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode#Constants_for_keyCode_value
addEventListener('keyup',(e)=>{if(e.altKey&&!e.shiftKey&&!e.ctrlKey&&e.keyCode==77){e.preventDefault();e.stopPropagation();
menuPopup.showPopup(this,-1,-1,"popup","bottomleft","topleft");}},false,window);

// Конвертировать текст в юникод .............
function convertFromUnicode(charset,str){var converter=Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter);
converter.charset=charset;str=converter.ConvertFromUnicode(str);return str+converter.Finish();};

все кнопки в вертикальном тулбаре, если можно реализовать как с этим кодом (this,-1,-1,"popup","bottomleft","topleft"); он открывает не как на навбаре


еще есть такой код, в uc.js, "//Define button, //Define menu" отвечают походу за открытие, но не знаю как реализовать.
QuickOpen.uc.js

Выделить код

Код:

//Define button
CustomizableUI.createWidget({id:'QuickOpen',defaultArea:CustomizableUI.AREA_NAVBAR,label:'Quick tools',/*tooltiptext:'Quickly open the specified options',*/onCreated:function(aNode){aNode.setAttribute('type','menu');    
 //Define menu
var myMenuJson=['xul:menupopup',{id:'QuickOpen_pop',position:'after_end'},
................................
................................
];aNode.appendChild(jsonToDOM(myMenuJson,aNode.ownerDocument,{}));aNode.setAttribute('menupopup','QuickOpen_pop');
}
});

весь код не вошел, поэтому только часть

Отредактировано func4ptch4 (19-01-2020 15:34:07)

Отсутствует

 

№1409119-01-2020 15:12:12

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 72.0

Re: Custom Buttons

Dumby
Больше не к кому...Такая поганка в 72 (нет там ничего нового после 70, тем более 71), но и морду попортили опять...
Megabar-Urlbar  - И при запуске расширяется и при клике на  самом и на панели вкладок...Это при дефолтной теме , никакие стили не помогают...Это скрипт где-то..
https://forum.mozilla-russia.org/viewto … 98#p776998
Там местные деляги  просто вянут, короче, есть мысль?
browser.urlbar.update1 = true , кажись это влияет

Отредактировано solombala (19-01-2020 15:52:52)

Отсутствует

 

№1409219-01-2020 18:03:47

kokoss
Участник
 
Группа: Members
Зарегистрирован: 15-02-2018
Сообщений: 685
UA: Firefox 52.0

Re: Custom Buttons

Dumby
Как убрать эти иконки:

скрытый текст
https://d.radikal.ru/d43/2001/47/17237f6eff56.png

а то всё равно другие не получается добавить ?

Отсутствует

 

№1409319-01-2020 20:14:26

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 1072
UA: Firefox 52.0

Re: Custom Buttons

func4ptch4
Здесь можно сделать так:

1. Переделываем кнопку под [type=menu].
Для этого, удаляем третью строку (начинающуюся с this._handleClick),
а в конец кода кладём this.type = "menu";

Если нужно, чтобы на вертикальном тулбаре
меню открывалось вбок, дописываем
this.matches("toolbar[orient=vertical] > :scope") && menuPopup.setAttribute("position", "end_before");

2. Идём к Infocatcher'у за сниппетом, открываем ссылку «Raw»,
копируем код от комментария // Autoopen/close feature и до конца,
и добавляем в код кнопки, тоже в конец.

Всё. Встроенный autopopup готов.

Кстати, у тебя там открытие меню кнопки по Alt+M висит непеределанным.
Так что, заодно, можно либо удалить, раз не пользуешься,
либо, к контексте данной переделки, записать так:

//menuPopup.showPopup(this,-1,-1,"popup","bottomleft","topleft");}},false,window);
self.open = true;}},false,window);

solombala пишет:

никакие стили не помогают

Вот специально скачал Firefox 72.0.1, собрал чистый портабл,
тема «Стандартная», ничего не трогал,
включил browser.urlbar.update1, включил userChrome.css,
скормил ему стиль проекта «AntiMegabar» от Vitaliy V., рестарт.

И мегабар ни при запуске, ни при каких кликах, никуда не расширяется.
Ни на один пиксель. Стоит как вкопанный. Так что...

kokoss пишет:

Как убрать эти иконки

Удалить этот фрагмент кода.
Ну и все свойства image и style из объектов массива array (не обязательно).

скрытый текст

Выделить код

Код:

   mItem.setAttribute("class", "menuitem-iconic");
   ("image" in m) && mItem.setAttribute("image", m.image);    
   ("style" in m) && mItem.setAttribute("style", "list-style-image: url('"+ m.style +"'); -moz-image-region: rect(0, 16px, 16px, 0)");

Отсутствует

 

№1409419-01-2020 21:24:20

kokoss
Участник
 
Группа: Members
Зарегистрирован: 15-02-2018
Сообщений: 685
UA: Firefox 52.0

Re: Custom Buttons

Dumby
Благодарю!

Отсутствует

 

№1409520-01-2020 14:25:36

egorsemenov06
Участник
 
Группа: Members
Зарегистрирован: 12-06-2018
Сообщений: 91
UA: Firefox 72.0

Re: Custom Buttons

Dumby
не гляните вот эту кнопку в ней не работает правый клик проверка обновлений доролнений

скрытый текст

Выделить код

Код:

// http://infocatcher.ucoz.net/js/cb/toggleRestartlessAddons.js
    // https://forum.mozilla-russia.org/viewtopic.php?id=57948
    // https://github.com/Infocatcher/Custom_Buttons/tree/master/Toggle_Restartless_Add-ons
    
    // Toggle Restartless Add-ons button for Custom Buttons
    // (code for "initialization" section)
    // Also the code can be used from main window context (as Mouse Gestures code, for example)
    
    // Also you can check for add-ons updates using right-click:
    // copy all code from
    // https://github.com/Infocatcher/Custom_Buttons/blob/master/Check_for_Addons_Updates/checkForAddonsUpdates.js
    // after "//== Check for Addons Updates begin"
    
    // See "var style = " to modify styles for specific add-ons
    
    // (c) Infocatcher 2013-2017
    // version 0.1.3pre3 - 2017-10-23
      
    var options = {
        addonTypes: ["extension"],
        // Possible values: "extension", "plugin"
        // From extensions: "userstyle" (Stylish), "greasemonkey-user-script" (Greasemonkey), "userscript" (Scriptish)
        // (swap to reorder in the menu)
        showVersions: 1,
        // 0 - don't show versions
        // 1 - show after name: "Addon Name 1.2"
        // 2 - show as "acceltext" (in place for hotkey text)
        showHidden: 0,
        // 0  - don't show hidden add-ons
        // -1 - show only enabled hidden add-ons (e.g. to track new items)
        // 1  - show all hidden add-ons
        sort: {
            enabled:     0,
            clickToPlay: 0,
            disabled:    0
            // Sort order:
            // 0, 0, 0 - sort add-ons of each type alphabetically
            // 0, 0, 1 - show enabled add-ons (of each type) first
            // 0, 1, 2 - enabled add-ons, then click-to-play and then disabled
        },
        closeMenu: false, // Close menu after left-click
        closeMenuClickToPlay: false // Close menu after left-click, for click to play plugins
        // Use Shift+click to invert closeMenu* behavior
    };
    
    var mp = document.createXULElement("menupopup");
    mp.setAttribute("onpopupshowing", "this.updateMenu();");
    mp.setAttribute("oncommand", "this.handleEvent(event);");
    mp.setAttribute("onmousedown", "if(event.button == 0) this.handleEvent(event);");
    mp.setAttribute("onclick", "if(event.button > 0) this.handleEvent(event);");
    mp.setAttribute("oncontextmenu", "return false;");
    mp.setAttribute("onpopuphidden", "this.destroyMenu();");
    
    var tb = this.parentNode;
    if(tb && tb.getAttribute("orient") == "vertical") {
        // https://addons.mozilla.org/firefox/addon/vertical-toolbar/
        var isRight = tb.parentNode.getAttribute("placement") == "right";
        mp.setAttribute("position", isRight ? "start_before" : "end_before");
    }
    
    var cleanupTimer = 0;
    mp.updateMenu = function() {
        clearTimeout(cleanupTimer);
        addStyle();
        getRestartlessAddons(options.addonTypes, function(addons) {
            var df = document.createDocumentFragment();
            var prevType;
            function sortPosition(addon) {
                if("STATE_ASK_TO_ACTIVATE" in AddonManager && addon.userDisabled == AddonManager.STATE_ASK_TO_ACTIVATE)
                    return options.sort.clickToPlay;
                if(addon.isActive)
                    return options.sort.enabled;
                return options.sort.disabled;
            }
            function key(addon) {
                return options.addonTypes.indexOf(addon.type)
                    + "\n" + sortPosition(addon)
                    + "\n" + addon.name.toLowerCase();
            }
            addons.sort(function(a, b) {
                var ka = key(a);
                var kb = key(b);
                return ka == kb ? 0 : ka < kb ? -1 : 1;
            }).forEach(function(addon) {
                var type = addon.type;
                if(prevType && type != prevType)
                    df.appendChild(document.createXULElement("menuseparator"));
                prevType = type;
                var icon = addon.iconURL || addon.icon64URL;
                var mi = document.createXULElement("menuitem");
                mi.className = "menuitem-iconic";
                var label = addon.name;
                if(options.showVersions == 1)
                    label += " " + addon.version;
                else if(options.showVersions == 2)
                    mi.setAttribute("acceltext", addon.version);
                mi.setAttribute("label", label);
                mi.setAttribute("image", icon || mp.icons[type] || "");
                if(!icon && mp.icons.useSVG)
                    mi.style.fill = "#15c";
                var tip = addon.description || "";
                var delay = "delayedStartupAddons" in Services
                    && Services.delayedStartupAddons[addon.id] || null;
                var isDelayed = delay !== null;
                mi.classList.toggle("toggleRestartlessAddons-isDelayed", isDelayed);
                if(isDelayed)
                    tip = "[Delayed Startup: " + delay.toLocaleString() + "]" + (tip ? "\n" + tip : "");
                tip && mi.setAttribute("tooltiptext", tip);
                mi.classList.toggle("toggleRestartlessAddons-isHidden", addon.hidden || false);
                setDisabled(mi, addon.userDisabled);
                mi._cbAddon = addon;
                df.appendChild(mi);
            });
            mp.textContent = "";
            mp.appendChild(df);
        });
    };
    mp.handleEvent = function(e) {
        var mi = e.target;
        if(!("_cbAddon" in mi))
            return;
        var addon = mi._cbAddon;
        if(e.type == "mousedown") {
            var closeMenu = isAskToActivateAddon(addon)
                ? options.closeMenuClickToPlay
                : options.closeMenu;
            if(e.shiftKey)
                closeMenu = !closeMenu;
            mi.setAttribute("closemenu", closeMenu ? "auto" : "none");
            return;
        }
        var hasMdf = hasModifier(e);
        if(e.type == "command" && (!hasMdf || e.shiftKey)) {
            let newDis = setNewDisabled(addon);
            setDisabled(mi, newDis);
        }
        else if(e.type == "command" && hasMdf || e.type == "click" && e.button == 1) {
            openAddonPage(addon);
            closeMenus(mi);
        }
        else if(e.type == "click" && e.button == 2) {
            if(openAddonOptions(addon))
                closeMenus(mi);
        }
    };
    mp.destroyMenu = function() {
        removeStyle();
        clearTimeout(cleanupTimer);
        cleanupTimer = setTimeout(function() {
            mp.textContent = "";
        }, 5000);
    };
    mp.icons = {
        get useSVG() {
            delete this.useSVG;
            return this.useSVG = Services.appinfo.name == "Firefox"
                && parseFloat(Services.appinfo.version) >= 57;
        },
        get plugin() {
            delete this.plugin;
            return this.plugin = this.useSVG
                ? "chrome://mozapps/skin/plugins/pluginGeneric.svg"
                : "chrome://mozapps/skin/plugins/pluginGeneric-16.png";
        },
        get extension() {
            delete this.extension;
            return this.extension = this.useSVG
                ? "chrome://mozapps/skin/extensions/extensionGeneric-16.svg"
                : "chrome://mozapps/skin/extensions/extensionGeneric-16.png";
        }
    };
    function isAskToActivateAddon(addon) {
        return addon.type == "plugin"
            && "STATE_ASK_TO_ACTIVATE" in AddonManager
            && Services.prefs.getBoolPref("plugins.click_to_play");
    }
    function setNewDisabled(addon) {
        var newDis = getNewDisabled(addon);
        var oldDis = addon.userDisabled;
        try {
            addon.userDisabled = newDis;
        }
        catch(e) { // Error: Cannot disable hidden add-on firefox@getpocket.com
            _log("Can't set addon.userDisabled to " + newDis + ", error:\n" + e);
            if(addon.hidden)
                setNewDisabledRaw(addon, newDis);
        }
        var realDis = addon.userDisabled;
        if(realDis != newDis && addon.type == "extension") { // Firefox 62+? Weird things happens
            setNewDisabledRaw(addon, newDis);
            realDis = addon.userDisabled;
        }
        if(realDis != newDis) { // We can't enable vulnerable plugins
            let err = "Can't set addon.userDisabled to " + newDis + ", real value: " + realDis;
            if(newDis) {
                _log(err + "\nSTATE_ASK_TO_ACTIVATE not supported?");
                newDis = false;
            }
            else {
                _log(err + "\nVulnerable plugin?");
                if(oldDis == AddonManager.STATE_ASK_TO_ACTIVATE)
                    newDis = true;
                else
                    newDis = AddonManager.STATE_ASK_TO_ACTIVATE;
            }
            addon.userDisabled = newDis;
        }
        return addon.userDisabled;
    }
    function getNewDisabled(addon) {
        // disabled -> STATE_ASK_TO_ACTIVATE -> enabled -> ...
        var curDis = addon.userDisabled;
        var newDis;
        if("STATE_ASK_TO_ACTIVATE" in AddonManager && curDis == AddonManager.STATE_ASK_TO_ACTIVATE)
            newDis = false;
        else if(!curDis)
            newDis = true;
        else {
            if(isAskToActivateAddon(addon))
                newDis = AddonManager.STATE_ASK_TO_ACTIVATE;
            else
                newDis = false;
        }
        return newDis;
    }
    function setNewDisabledRaw(addon, newDis) {
        _log("Let's try set addon.userDisabled using raw hack");
        let g = Components.utils.import("resource://gre/modules/addons/XPIProvider.jsm", {});
        if("XPIDatabase" in g && "updateAddonDisabledState" in g.XPIDatabase) { // Firefox 61+
            let rawAddon = g.XPIDatabase.getAddons().find(function(rawAddon) {
                return rawAddon.id == addon.id;
            });
            g.XPIDatabase.updateAddonDisabledState(rawAddon, newDis);
        }
        else if("eval" in g) { // See "set userDisabled(val)"
            let addonFor = g.eval("addonFor");
            let rawAddon = addonFor(addon);
            //rawAddon.userDisabled = newDis;
            g.XPIProvider.updateAddonDisabledState(rawAddon, newDis);
        }
        else { // Firefox 57+? See https://forum.mozilla-russia.org/viewtopic.php?pid=745272#p745272
            updateAddonDisabledState(addon, newDis);
        }
    }
    function updateAddonDisabledState(addon, newDis) {
        var nsvo = Components.utils.import("resource://gre/modules/addons/XPIProvider.jsm", {});
        var key = "_cbToggleRestartlessAddonsData";
        var url = URL.createObjectURL(new Blob([
            "XPIProvider.updateAddonDisabledState(addonFor(this." + key + "[0]), this." + key + "[1]); delete this." + key + ";"
        ]));
        addDestructor(function() {
            URL.revokeObjectURL(url);
        });
        (updateAddonDisabledState = function(addon, newDis) {
            nsvo[key] = [addon, newDis];
            Services.scriptloader.loadSubScript(url, nsvo);
        })(addon, newDis);
    }
    function setDisabled(mi, disabled) {
        var askToActivate = "STATE_ASK_TO_ACTIVATE" in AddonManager && disabled == AddonManager.STATE_ASK_TO_ACTIVATE;
        var cl = mi.classList;
        cl.toggle("toggleRestartlessAddons-askToActivate", askToActivate);
        cl.toggle("toggleRestartlessAddons-disabled", disabled && !askToActivate);
    }
    
    if(
        this instanceof XULElement // Custom Buttons
        && typeof event == "object"
        && !("type" in event) && typeof _phase == "string" && _phase == "init" // Initialization
    ) {
        this.type = "menu";
        this.orient = "horizontal";
        this.appendChild(mp);
    
        this.onmouseover = function(e) {
            if(e.target != this)
                return;
            Array.prototype.some.call(
                this.parentNode.getElementsByTagName("*"),
                function(node) {
                    if(
                        node != this
                        && node.namespaceURI == xulns
                        && node.boxObject
                        // See https://github.com/Infocatcher/Custom_Buttons/issues/28
                        //&& node.boxObject instanceof Components.interfaces.nsIMenuBoxObject
                        && "open" in node
                        && node.open
                        && node.getElementsByTagName("menupopup").length
                    ) {
                        node.open = false;
                        this.open = true;
                        return true;
                    }
                    return false;
                },
                this
            );
        };
        this.onmousedown = function(e) {
            if(e.target == this && e.button == 0 && hasModifier(e))
                e.preventDefault();
        };
        this.oncontextmenu = function(e) {
            if(e.target == this && !hasModifier(e) && hasUpdater())
                e.preventDefault();
        };
        this.onclick = function(e) {
            if(e.target != this)
                return;
            if(e.button == 0 && hasModifier(e) || e.button == 1)
                openAddonsManager();
            else if(e.button == 2 && !hasModifier(e) && hasUpdater())
                checkForAddonsUpdates.call(this);
        };
    }
    else { // Mouse gestures or something other...
        let e;
        if(typeof event == "object" && event instanceof Event && "screenX" in event) // FireGestures
            e = event;
        else if(
            this instanceof Components.interfaces.nsIDOMChromeWindow
            && "mgGestureState" in window && "endEvent" in mgGestureState // Mouse Gestures Redox
        )
            e = mgGestureState.endEvent;
        else {
            let anchor = this instanceof XULElement && this
                || window.gBrowser && gBrowser.selectedBrowser
                || document.documentElement;
            if("boxObject" in anchor) {
                let bo = anchor.boxObject;
                e = {
                    screenX: bo.screenX,
                    screenY: bo.screenY
                };
                if(this instanceof XULElement)
                    e.screenY += bo.height;
            }
        }
        if(!e || !("screenX" in e))
            throw new Error("[Toggle Restartless Add-ons]: Can't get event object");
        document.documentElement.appendChild(mp);
        mp.addEventListener("popuphidden", function destroy(e) {
            mp.removeEventListener(e.type, destroy, false);
            setTimeout(function() {
                mp.destroyMenu();
                mp.parentNode.removeChild(mp);
            }, 0);
        }, false);
        mp.openPopupAtScreen(e.screenX, e.screenY);
    }
    
    function getRestartlessAddons(addonTypes, callback, context) {
        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(addonTypes, then = function(addons) {
            var restartless = addons.filter(function(addon) {
                var ops = addon.operationsRequiringRestart;
                return !addon.appDisabled
                    && !(ops & AddonManager.OP_NEEDS_RESTART_ENABLE || ops & AddonManager.OP_NEEDS_RESTART_DISABLE)
                    && (
                        !addon.hidden
                        || options.showHidden > 0
                        || options.showHidden == -1 && !addon.userDisabled
                    );
            });
            callback.call(context, restartless);
        });
        promise && typeof promise.then == "function" && promise.then(then, Components.utils.reportError); // Firefox 61+
    }
    function openAddonOptions(addon) {
        // Based on code from chrome://mozapps/content/extensions/extensions.js
        // Firefox 21.0a1 (2013-01-27)
        var optionsURL = addon.optionsURL;
        if(!addon.isActive || !optionsURL)
            return false;
        if(addon.type == "plugin") // No options for now!
            return false;
        if(
            addon.optionsType == AddonManager.OPTIONS_TYPE_INLINE
            || addon.optionsType == (AddonManager.OPTIONS_TYPE_INLINE_INFO || NaN)
            || addon.optionsType == (AddonManager.OPTIONS_TYPE_INLINE_BROWSER || NaN)
        )
            openAddonPage(addon, true);
        else if(addon.optionsType == AddonManager.OPTIONS_TYPE_TAB && "switchToTabHavingURI" in window)
            switchToTabHavingURI(optionsURL, true);
        else {
            let windows = Services.wm.getEnumerator(null);
            while(windows.hasMoreElements()) {
                let win = windows.getNext();
                if(win.document.documentURI == optionsURL) {
                    win.focus();
                    return true;
                }
            }
            // Note: original code checks browser.preferences.instantApply and may open modal windows
            window.openDialog(optionsURL, "", "chrome,titlebar,toolbar,centerscreen,dialog=no");
        }
        return true;
    }
    function openAddonsManager(view) {
        var openAddonsMgr = window.BrowserOpenAddonsMgr // Firefox
            || window.openAddonsMgr // Thunderbird
            || window.toEM; // SeaMonkey
        openAddonsMgr(view);
    }
    function openAddonPage(addon, scrollToPreferences) {
        var platformVersion = parseFloat(
            Services.appinfo.name == "Pale Moon"
                ? Services.appinfo.version
                : Services.appinfo.platformVersion
        );
        scrollToPreferences = scrollToPreferences && platformVersion >= 12
            ? "/preferences"
            : "";
        openAddonsManager("addons://detail/" + encodeURIComponent(addon.id) + scrollToPreferences);
    }
    
    function hasModifier(e) {
        return e.ctrlKey || e.shiftKey || e.altKey || e.metaKey;
    }
    
    function addStyle() {
        if(addStyle.hasOwnProperty("_style"))
            return;
        var style = '\
            .toggleRestartlessAddons-isDelayed > .menu-iconic-text {\n\
                opacity: 0.75;\n\
                color: #070;\n\
            }\n\
            .toggleRestartlessAddons-isHidden > .menu-iconic-text {\n\
                color: #609;\n\
            }\n\
            .toggleRestartlessAddons-disabled > .menu-iconic-left {\n\
                opacity: 0.4;\n\
            }\n\
            .toggleRestartlessAddons-disabled > .menu-iconic-text,\n\
            .toggleRestartlessAddons-disabled > .menu-accel-container {\n\
                opacity: 0.5;\n\
            }\n\
            .toggleRestartlessAddons-askToActivate {\n\
                color: -moz-nativehyperlinktext;\n\
            }';
        addStyle._style = document.insertBefore(
            document.createProcessingInstruction(
                "xml-stylesheet",
                'href="' + "data:text/css,"
                    + encodeURIComponent(style) + '" type="text/css"'
            ),
            document.documentElement
        );
    }
    function removeStyle() {
        if(!addStyle.hasOwnProperty("_style"))
            return;
        var s = addStyle._style;
        s.parentNode.removeChild(s);
        delete addStyle._style;
    }
    function closeMenus(node) {
        // Based on function closeMenus from chrome://browser/content/utilityOverlay.js
        for(; node && "tagName" in node; node = node.parentNode) {
            if(
                node.namespaceURI == "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"
                && (node.localName == "menupopup" || node.localName == "popup")
            )
                node.hidePopup();
        }
    }
    function _log(s) {
        if(typeof LOG == "function") // Custom Buttons
            LOG(s);
        else // Or something else
            Services.console.logStringMessage("Toggle Restartless Add-ons: " + s);
    }
    
    function hasUpdater() {
        var has = checkForAddonsUpdates.toString().indexOf("about:addons") != -1;
        hasUpdater = function() {
            return has;
        };
        return has;
    }
    
    
    
    
    
    
    
    
    
    function checkForAddonsUpdates() {
// http://infocatcher.ucoz.net/js/cb/checkForAddonsUpdates.js
// https://forum.mozilla-russia.org/viewtopic.php?id=57958
// https://github.com/Infocatcher/Custom_Buttons/tree/master/Check_for_Addons_Updates

// Check for Addons Updates button for Custom Buttons
// (code for "code" section)

// (c) Infocatcher 2012-2014
// version 0.1.5 - 2014-10-13

// Button just open hidden tab with about:addons and trigger built-in "Check for Updates" function.
// And show tab, if found updates.

(function() {
var btn = this instanceof XULElement
    ? this
    : { // Launched not from custom button
        image: "", // Base64-encoded icon (if empty, will be used "imgLoading")
        label: "Check for Addons Updates",
        tooltipText: ""
    };
if("_cb_disabled" in btn)
    return;
btn._cb_disabled = true;

if(!("Services" in window))
    Components.utils.import("resource://gre/modules/Services.jsm");
var app = Services.appinfo.name;

var ADDONS_URL = "about:addons";

var progressIcon = new ProgressIcon(btn);
var image = btn.image || progressIcon.imgLoading;
var tip = btn.tooltipText;
btn.tooltipText = "Open " + ADDONS_URL + "…";

var tab, browser, gBrowser;
var tbTabInfo, tbTab;

var trgWindow = Services.wm.getMostRecentWindow("navigator:browser")
    || app == "Thunderbird" && Services.wm.getMostRecentWindow("mail:3pane")
    || window;
var trgDocument = trgWindow.document;
var tabmail = trgDocument.getElementById("tabmail");

if(tabmail && app == "Thunderbird") { // Note: SeaMonkey doesn't support content tabs in mail window
    let addonsWin;
    let receivePong = function(subject, topic, data) {
        addonsWin = subject;
    };
    Services.obs.addObserver(receivePong, "EM-pong", false);
    Services.obs.notifyObservers(null, "EM-ping", "");
    Services.obs.removeObserver(receivePong, "EM-pong");
    if(addonsWin) {
        let rootWindow = addonsWin
            .QueryInterface(Components.interfaces.nsIInterfaceRequestor)
            .getInterface(Components.interfaces.nsIWebNavigation)
            .QueryInterface(Components.interfaces.nsIDocShellTreeItem)
            .rootTreeItem
            .QueryInterface(Components.interfaces.nsIInterfaceRequestor)
            .getInterface(Components.interfaces.nsIDOMWindow);
        tabmail = rootWindow.document.getElementById("tabmail");
        tbTabInfo = tabmail.getBrowserForDocument(addonsWin);
        tbTab = tab = tbTabInfo.tabNode;
        processAddonsTab(addonsWin);
    }
    else {
        Services.obs.addObserver(function observer(subject, topic, data) {
            Services.obs.removeObserver(observer, topic);
            if(subject.document.readyState == "complete")
                processAddonsTab(subject);
            else {
                subject.addEventListener("load", function onLoad(e) {
                    subject.removeEventListener(e.type, onLoad, false);
                    processAddonsTab(subject);
                }, false);
            }
        }, "EM-loaded", false);
        // See openAddonsMgr() -> openContentTab()
        tbTabInfo = tabmail.openTab("contentTab", {
            contentPage: ADDONS_URL,
            clickHandler: "specialTabs.siteClickHandler(event, /addons\.mozilla\.org/);",
            background: true
        });
        tbTab = tab = tbTabInfo.tabNode;
        tbTab.collapsed = true;
        // Note: dontSelectHiddenTab() not implemented
    }
}
else if("gBrowser" in trgWindow && trgWindow.gBrowser.tabs) {
    let isPending = false;
    let ws = Services.wm.getEnumerator("navigator:browser");
    windowsLoop:
    while(ws.hasMoreElements()) {
        let w = ws.getNext();
        let tabs = w.gBrowser.tabs;
        for(let i = 0, l = tabs.length; i < l; ++i) {
            let t = tabs[i];
            if(
                !t.closing
                && t.linkedBrowser
                && t.linkedBrowser.currentURI.spec == ADDONS_URL
            ) {
                tab = t;
                break windowsLoop;
            }
        }
    }

    gBrowser = trgWindow.gBrowser;
    if(!tab) {
        tab = gBrowser.addTab(ADDONS_URL, {
            triggeringPrincipal: "Services" in window // Firefox 63+
                && Services.scriptSecurityManager
                && Services.scriptSecurityManager.getSystemPrincipal()
        });
        tab.collapsed = true;
        tab.closing = true; // See "visibleTabs" getter in chrome://browser/content/tabbrowser.xml
        trgWindow.addEventListener("TabSelect", dontSelectHiddenTab, false);
    }
    else if(
        tab.getAttribute("pending") == "true" // Gecko >= 9.0
        || tab.linkedBrowser.contentDocument.readyState == "uninitialized"
        // || tab.linkedBrowser.__SS_restoreState == 1
    )
        isPending = true;

    browser = tab.linkedBrowser;
    if(isPending || browser.webProgress.isLoadingDocument) {
        browser.addEventListener("load", processAddonsTab, true);
        if(isPending) {
            if(parseFloat(Services.appinfo.platformVersion) >= 41) {
                // Workaround to correctly restore pending tab
                // See https://github.com/Infocatcher/Custom_Buttons/issues/39
                let selTab = gBrowser.selectedTab;
                gBrowser.selectedTab = tab;
                gBrowser.selectedTab = selTab;
            }
            else {
                browser.reload();
            }
        }
    }
    else {
        processAddonsTab();
    }
}
else {
    progressIcon.restore();
    btn.tooltipText = tip;
    delete btn._cb_disabled;
    Services.prompt.alert(window, btn.label, "Error: Can't find supported window!");
    return;
}

function processAddonsTab(e) {
    var doc;
    if(e && e instanceof Components.interfaces.nsIDOMWindow) {
        doc = e.document;
    }
    else if(e) {
        doc = e.target;
        if(doc.location != ADDONS_URL)
            return;
        browser.removeEventListener(e.type, processAddonsTab, true);
    }
    else {
        doc = browser.contentDocument;
    }

    progressIcon.loading();
    var inProgress = $("updates-progress");
    btn.tooltipText = inProgress.getAttribute("value");

    var origIcon = tab.image;
    tab.image = image;

    var updEnabledPref = "extensions.update.enabled";
    var updEnabled = Services.prefs.getBoolPref(updEnabledPref);
    if(!updEnabled)
        Services.prefs.setBoolPref(updEnabledPref, true);

    var notFound = $("updates-noneFound");
    var updated = $("updates-installed");
    // Avoid getting false results from the past update check (may not be required for "noneFound")
    notFound.hidden = updated.hidden = true;

    $("cmd_findAllUpdates").doCommand();

    var waitTimer = setInterval(function() {
        if(!doc.defaultView || doc.defaultView.closed) {
            stopWait();
            notify("Tab with add-ons manager was closed!");
            return;
        }
        if(!inProgress.hidden)
             return;
        var autoUpdate = $("utils-autoUpdateDefault");
        var autoUpdateChecked = autoUpdate.getAttribute("checked") == "true";

        var found = $("updates-manualUpdatesFound-btn");
        if(
            autoUpdateChecked
                ? notFound.hidden && updated.hidden
                : notFound.hidden && found.hidden
        ) // Too early?
            return;

        stopWait();
        if(!tbTab)
            tab.closing = false;
        function removeTab() {
            if(!tab.collapsed)
                return;
            if(tbTab)
                tabmail.closeTab(tbTabInfo, true /*aNoUndo*/);
            else {
                gBrowser.removeTab(tab);
                (function forgetClosedTab(isSecondTry) {
                    var ss = "nsISessionStore" in Components.interfaces
                        ? (
                            Components.classes["@mozilla.org/browser/sessionstore;1"]
                            || Components.classes["@mozilla.org/suite/sessionstore;1"]
                        ).getService(Components.interfaces.nsISessionStore)
                        : SessionStore; // Firefox 61+ https://bugzilla.mozilla.org/show_bug.cgi?id=1450559
                    if(!("forgetClosedTab" in ss))
                        return;
                    var closedTabs = JSON.parse(ss.getClosedTabData(window));
                    for(let i = 0, l = closedTabs.length; i < l; ++i) {
                        let closedTab = closedTabs[i];
                        let state = closedTab.state;
                        if(state.entries[state.index - 1].url == ADDONS_URL) {
                            ss.forgetClosedTab(window, i);
                            return;
                        }
                    }
                    if(!isSecondTry) // May be needed in SeaMonkey
                        setTimeout(forgetClosedTab, 0, true);
                })();
            }
        }

        if(!updEnabled)
            Services.prefs.setBoolPref(updEnabledPref, false);

        if(!notFound.hidden) {
            removeTab();
            notify(notFound.getAttribute("value"));
            return;
        }
        if(autoUpdateChecked) {
            removeTab();
            notify(updated.getAttribute("value"));
            return;
        }

        tab.collapsed = false;
        $("categories").selectedItem = $("category-availableUpdates");
        var tabWin = tab.ownerDocument.defaultView;
        if(tbTab)
            tabmail.switchToTab(tbTabInfo);
        else
            tabWin.gBrowser.selectedTab = tab;
        setTimeout(function() {
            tabWin.focus();
            doc.defaultView.focus();
            $("addon-list").focus();
        }, 0);
    }, 50);
    function $(id) {
        return doc.getElementById(id);
    }
    function stopWait() {
        clearInterval(waitTimer);
        progressIcon.restore();
        btn.tooltipText = tip;
        if(tab.image == image)
            tab.image = origIcon;
        trgWindow.removeEventListener("TabSelect", dontSelectHiddenTab, false);
        setTimeout(function() {
            delete btn._cb_disabled;
        }, 500);
    }
    function notify(msg) {
        Components.classes["@mozilla.org/alerts-service;1"]
            .getService(Components.interfaces.nsIAlertsService)
            .showAlertNotification(
                Services.appinfo.name == "Firefox" && parseFloat(Services.appinfo.version) >= 57
                    ? "chrome://mozapps/skin/extensions/extensionGeneric.svg"
                    : "chrome://mozapps/skin/extensions/extensionGeneric.png",
                btn.label,
                msg, false, "", null
            );
    }
}
function dontSelectHiddenTab(e) {
    // <tab /><tab collapsed="true" />
    // Close first tab: collapsed tab becomes selected
    var trgTab = e.originalTarget || e.target;
    if(trgTab != tab)
        return;

    if(/\n(?:BrowserOpenAddonsMgr|toEM)@chrome:\/\//.test(new Error().stack)) {
        // User open Add-ons Manager, show tab
        trgWindow.removeEventListener("TabSelect", dontSelectHiddenTab, false);
        setTimeout(function() { // Hidden tab can't be selected, so select it manually...
            tab.collapsed = tab.closing = false;
            gBrowser.selectedTab = tab;
        }, 0);
    }

    function done(t) {
        if(!t.hidden && !t.closing) {
            e.preventDefault();
            e.stopPropagation();
            return gBrowser.selectedTab = t;
        }
        return false;
    }
    for(var t = tab.nextSibling; t; t = t.nextSibling)
        if(done(t))
            return;
    for(var t = tab.previousSibling; t; t = t.previousSibling)
        if(done(t))
            return;
}
function ProgressIcon(btn) {
    if(!(btn instanceof XULElement)) {
        this.loading = this.restore = function() {};
        return;
    }
    var app = Services.appinfo.name;
    var pv = parseFloat(Services.appinfo.platformVersion);
    if(app == "SeaMonkey")
        this.imgConnecting = this.imgLoading = "chrome://communicator/skin/icons/loading.gif";
    else if(app == "Thunderbird") {
        this.imgConnecting = "chrome://messenger/skin/icons/connecting.png";
        this.imgLoading = "chrome://messenger/skin/icons/loading.png";
    }
    else {
        this.imgConnecting = "chrome://browser/skin/tabbrowser/connecting.png";
        this.imgLoading = app == "Firefox" && pv >= 48
            ? "chrome://global/skin/icons/loading.png"
            : "chrome://browser/skin/tabbrowser/loading.png";
    }
    var useAnimation = app == "Firefox" && pv >= 32;
    var btnIcon = btn.ownerDocument.getAnonymousElementByAttribute(btn, "class", "toolbarbutton-icon")
                    || btn.getElementsByClassName("toolbarbutton-icon")[0];
    var origIcon = btnIcon.src;
    btnIcon.src = this.imgConnecting;
    if(useAnimation) {
        let cs = btnIcon.ownerDocument.defaultView.getComputedStyle(btnIcon, null);
        let s = btnIcon.style;
        s.margin = [cs.marginTop, cs.marginRight, cs.marginBottom, cs.marginLeft].join(" ");
        s.padding = [cs.paddingTop, cs.paddingRight, cs.paddingBottom, cs.paddingLeft].join(" ");
        s.width = cs.width;
        s.height = cs.height;
        s.boxShadow = "none";
        s.borderColor = s.background = "transparent";
        btnIcon.setAttribute("fadein", "true");
        btnIcon.setAttribute("busy", "true");
        btnIcon.classList.add("tab-throbber");
        btnIcon._restore = function() {
            delete btnIcon._restore;
            btnIcon.removeAttribute("busy");
            btnIcon.removeAttribute("progress");
            setTimeout(function() {
                btnIcon.classList.remove("tab-throbber");
                btnIcon.removeAttribute("style");
                btnIcon.removeAttribute("fadein");
            }, 0);
        };
    }
    this.loading = function() {
        btnIcon.src = this.imgLoading;
        if(useAnimation)
            btnIcon.setAttribute("progress", "true");
    };
    this.restore = function() {
        btnIcon.src = origIcon;
        if(useAnimation)
            btnIcon._restore();
    };
}
}).call(this);
//== Check for Addons Updates end
}


this.tooltipText = "Переключатель джетпаков" 
                   + "\n\nУправление:\nЛКМ – открыть меню" 
                   + "\nПКМ – проверить обновления"
                   + "\nСКМ – открыть страницу дополнений"
                   + "\nShift+ПКМ – меню кнопки"
                   + "\n\nВ меню: \nЛКМ – включить/выключить дополнение"
                   + "\nShift+ЛКМ – включить/выключить дополнение без закрытия меню"   
                   + "\nСКМ – открыть страницу дополнения в управлении дополнениями"                    
                   + "\nПКМ – открыть настройки дополнения (если есть)";    


и если не сложно как сделать autopopup на эту кнопку
скрытый текст

Выделить код

Код:

(async obj => {
    Services.search.isInitialized || await Services.search.init();
    obj.observe();  
    this._handleClick = () => obj.menupopup.openPopup(this);
    var topic = "browser-search-engine-modified";
    Services.obs.addObserver(obj, topic, false);
    addDestructor(() => Services.obs.removeObserver(obj, topic));
    if (!obj.excludeHiddenOneOffs) return;
    var obs = () => obj.upd = true;
    Services.prefs.addObserver(obj.pref, obs);
    addDestructor(() => Services.prefs.removeObserver(obj.pref, obs));
})({
    excludeHiddenOneOffs: false,

    get menupopup() {
        this.upd && this.rebuild();
        return this.popup;
    },
    get popup() {
        var popup = self.appendChild(document.createXULElement("menupopup"));
        popup.setAttribute("context", "");
        popup.setAttribute("position", "after_start");
        popup.setAttribute("oncommand", "Services.search.defaultEngine = event.target.engine");
        delete this.popup; return this.popup = popup;
    },
    async rebuild() {
        this.popup.textContent = "";
        var df = document.createDocumentFragment();
        var de = Services.search.defaultEngine.wrappedJSObject, jsde = this.json(de);
        if (this.excludeHiddenOneOffs)
            var ex = Services.prefs.getStringPref(this.pref, "").split(",");
        var check = true;
        for(var engine of await Services.search.getVisibleEngines()) {
            if (check && engine.name == de.name && this.json(engine) == jsde) {
                check = false; continue;
            }
            if (this.excludeHiddenOneOffs && ex.includes(engine.name)) continue;
            var menuitem = df.appendChild(document.createXULElement("menuitem"));
            menuitem.engine = engine;
            menuitem.label = engine.name;
            menuitem.image = this.img(engine);
            menuitem.className = "menuitem-iconic";
        }
        this.upd = this.popup.append(df);
    },
    observe() {
        var engine = Services.search.defaultEngine;
        (self.icon || document.getAnonymousElementByAttribute(
            self, "class", "toolbarbutton-icon"
        )).src = this.img(engine);
        self.tooltipText = engine.name;
        this.upd = true;
    },
    pref: "browser.search.hiddenOneOffs",
    json: e => JSON.stringify(e.toJSON()),
    img: e => e.iconURI ? e.iconURI.spec : "chrome://browser/skin/search-engine-placeholder.png"
});


[firefox] 72.0.1

Отредактировано egorsemenov06 (20-01-2020 15:27:26)

Отсутствует

 

№1409620-01-2020 22:13:35

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 70.0

Re: Custom Buttons

Отсутствует

 

№1409721-01-2020 07:43:17

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 1072
UA: Firefox 52.0

Re: Custom Buttons

egorsemenov06 пишет:

не гляните вот эту кнопку в ней не работает правый клик проверка обновлений доролнений

Не, не моя тема, совсем. Но можно приглядывать за страницей
Check_for_Addons_Updates, прогресс возможен, наверно.

egorsemenov06 пишет:

сделать autopopup на эту кнопку

Хорошо, попробую

скрытый текст

Выделить код

Код:

(async obj => {
    Services.search.isInitialized || await Services.search.init();
    obj.observe();
    this.type = "menu";
    obj.popup = this.appendChild(document.createXULElement("menupopup"));
    addEventListener("popupshowing", obj, false, obj.popup);

    // https://github.com/Infocatcher/Custom_Buttons/blob/master/code_snippets/autoOpenCloseMenu.js
    // Automatically open menu on mouse over (and hide it on mouse out)

    //=======[ Start Autoopen/close feature ]=======
    var openDelay = 200;
    var closeDelay = 350;

    var _openTimer = 0;
    var _closeTimer = 0;
    this.onmouseover = function(e) {
        clearTimeout(_closeTimer);
        if(e.target == this && closeOtherMenus()) {
            this.open = true;
            return;
        }
        _openTimer = setTimeout(function() {
            self.open = true;
        }, openDelay);
    };
    this.onmouseout = function(e) {
        clearTimeout(_openTimer);
        _closeTimer = setTimeout(function() {
            if(!isContextOpened())
                self.open = false;
        }, closeDelay);
    };
    function closeOtherMenus() {
        return Array.prototype.some.call(
            self.parentNode.getElementsByTagName("*"),
            function(node) {
                if(
                    node != self
                    && node.namespaceURI == xulns
                    // See https://github.com/Infocatcher/Custom_Buttons/issues/28
                    //&& node.boxObject
                    //&& node.boxObject instanceof Components.interfaces.nsIMenuBoxObject
                    && "open" in node
                    && node.open
                    && node.getElementsByTagName("menupopup").length
                ) {
                    node.open = false;
                    return true;
                }
                return false;
            }
        );
    }
    function isContextOpened() {
        return inBtn(document.popupNode);
    }
    function inBtn(node) {
        for(; node; node = node.parentNode)
            if(node == self)
                return true;
        return false;
    }
    //=======[ End Autoopen/close feature ]=======

    var topic = "browser-search-engine-modified";
    Services.obs.addObserver(obj, topic, false);
    addDestructor(() => Services.obs.removeObserver(obj, topic));
    if (!obj.excludeHiddenOneOffs) return;
    var obs = () => obj.upd = true;
    Services.prefs.addObserver(obj.pref, obs);
    addDestructor(() => Services.prefs.removeObserver(obj.pref, obs));
})({
    excludeHiddenOneOffs: false,

    handleEvent() {
        var {popup} = this;
        popup.setAttribute("context", "");
        popup.setAttribute("position", "after_start");
        popup.setAttribute("oncommand", "Services.search.defaultEngine = event.target.engine");
        (this.handleEvent = () => this.upd && this.rebuild())();
    },
    async rebuild() {
        this.popup.textContent = "";
        var df = document.createDocumentFragment();
        var de = Services.search.defaultEngine.wrappedJSObject, jsde = this.json(de);
        if (this.excludeHiddenOneOffs)
            var ex = Services.prefs.getStringPref(this.pref, "").split(",");
        var check = true;
        for(var engine of await Services.search.getVisibleEngines()) {
            if (check && engine.name == de.name && this.json(engine) == jsde) {
                check = false; continue;
            }
            if (this.excludeHiddenOneOffs && ex.includes(engine.name)) continue;
            var menuitem = df.appendChild(document.createXULElement("menuitem"));
            menuitem.engine = engine;
            menuitem.label = engine.name;
            menuitem.image = this.img(engine);
            menuitem.className = "menuitem-iconic";
        }
        this.upd = this.popup.append(df);
    },
    observe() {
        var engine = Services.search.defaultEngine;
        (self.icon || document.getAnonymousElementByAttribute(
            self, "class", "toolbarbutton-icon"
        )).src = this.img(engine);
        self.tooltipText = engine.name;
        this.upd = true;
    },
    pref: "browser.search.hiddenOneOffs",
    json: e => JSON.stringify(e.toJSON()),
    img: e => e.iconURI ? e.iconURI.spec : "chrome://browser/skin/search-engine-placeholder.png"
});


solombala пишет:

Может знаете...?

WFM

Отсутствует

 

№1409821-01-2020 09:19:15

solombala
Участник
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 350
UA: Firefox 72.0

Re: Custom Buttons

Dumby

Dumby пишет:

WFM

Это что ? Радио-УКВ ? Кстати, странно эта 72 ведет, после очистки профиля , кэша запуска и т.д. Зависает прилично...

Отредактировано solombala (21-01-2020 09:23:38)

Отсутствует

 

№1409921-01-2020 11:16:23

egorsemenov06
Участник
 
Группа: Members
Зарегистрирован: 12-06-2018
Сообщений: 91
UA: Firefox 72.0

Re: Custom Buttons

Dumby

СПАСИБО!
посмотрите пожалуйста еще кнопку в ней в консоли выдает ошибку   NS_ERROR_FILE_NOT_FOUND: Component returned failure code: 0x80520012 (NS_ERROR_FILE_NOT_FOUND) [nsIProcess.init]

скрытый текст

Выделить код

Код:

// Запустить QTranslate и жмакнуть(программно) дважды CTRL(код Dumby)
// - стандартн. комбин. для захвата текста из окна в QTranslate.
// интервал можно увеличить или уменьшить, зависит от железа  
 setTimeout(dblCtrlSend, 100);
 quickTranslate();

 // ----------------------------------- =  QuickTranslate  = ----------------------------------------------
   function quickTranslate() {
//  скопировать выделенный текст в буфер. 
//       gClipboard.write(document.commandDispatcher.focusedWindow.getSelection().toString());
     var profile = Services.dirsvc.get('ProfD', Ci.nsIFile);
     profile.initWithPath(profile.path + "\\_QTranslate\\QTranslate.exe");
      var process = Cc["@mozilla.org/process/util;1"].createInstance(Ci.nsIProcess);
      var arg = [];
   process.init(profile);
   process.run(false, [arg], 1);
//   setTimeout(function() window.content.focus(), 500);
   };


function dblCtrlSend() {

// See https://gist.github.com/Noitidart/0de3be2442a0295eb386
// vk codes: https://msdn.microsoft.com/en-us/library/windows/desktop/dd375731%28v=vs.85%29.aspx

// Double Ctrl [Ctrl-Down, Ctrl-Up, Ctrl-Down, Ctrl-Up]
var data = [[0x11, false], [0x11, true], [0x11, false], [0x11, true]];

// Ctrl+Q [Ctrl-Down, Q-Down, Q-Up, Ctrl-Up]
//var data = [[0x11, false], [0x51, false], [0x51, true], [0x11, true]];

var {ctypes} = Cu.import("resource://gre/modules/ctypes.jsm", {});
var KEYBDINPUT = ctypes.StructType("tagKEYBDINPUT", [
    {wVk: ctypes.unsigned_short},
    {wScan: ctypes.unsigned_short},
    {dwFlags: ctypes.unsigned_long},
    {time: ctypes.unsigned_long},
    {dwExtraInfo: ctypes.voidptr_t.size == 8 ? ctypes.uint64_t : ctypes.unsigned_long},
    {padding0: ctypes.uint8_t.array(8)}
]);
var INPUT = ctypes.StructType("tagINPUT", [{type: ctypes.unsigned_long}, {ki: KEYBDINPUT}]);
var pInputs = INPUT.array()(data.map(([vkCode, keyup]) => 
    INPUT(1, KEYBDINPUT(vkCode, 0, keyup ? 2 : 0, 0, 0, ctypes.uint8_t.array(8)()))
));
var user32 = ctypes.open("user32"); try {
    var SendInput = user32.declare("SendInput", ctypes.winapi_abi, ctypes.unsigned_int, ctypes.unsigned_int, INPUT.ptr, ctypes.int);
    SendInput(pInputs.length, pInputs, INPUT.size);
 } finally {
    user32.close();
 }
};


и в этой кнопке   TypeError: gURLBar.reset is not a function
инициализация
скрытый текст

Выделить код

Код:

// Настройка функций кликов мыши для кнопки ...........
this.onclick =e=> {
   if ( e.button == 0 ) gBrowser.selectedTab = gBrowser.addTrustedTab(notepad); // открыть блокнот    
     
   if ( e.button == 1 ) {  
        // запомнить текст из буфера обмена и скопировать текст на странице
        var clip = gClipboard.read();
        window.content.focus();
        goDoCommand("cmd_copy");            

        // открыть блокнот и вставить текст из буфера обмена
        var browser = gBrowser.getBrowserForTab(gBrowser.selectedTab = gBrowser.addTrustedTab(notepad)); 
        browser.addEventListener("pageshow", function c(e) {      
           this.removeEventListener(e.type, c);           

           setTimeout(()=> {              
              content.document.getElementsByTagName("textarea")[0].value = gClipboard.read();
              clip && gClipboard.write(clip); // вернуть текст в буфер обмена 
           }, 50);      
        });       
        };
};


// Обновить блокнот в текущей вкладке после обновления кнопки ...........
if ( gBrowser.currentURI.spec == notepad ) {
     var val = content.document.getElementsByTagName("textarea")[0].value;
     loadURI(notepad);
     setTimeout(()=> content.document.getElementsByTagName("textarea")[0].value = val, 500);     
     };

        
// Очистить адресную строку в кладке блокнота ...........
addEventListener("TabAttrModified", ()=> gBrowser.currentURI.spec == notepad && gURLBar.reset());


// Получаем адрес блокнота как base64 из вкладки Справка и добавляем иконку для вкладки блокнота ...........
var notepad = "data:text/html;base64," + window.btoa(self.getAttribute('Help').replace('selfImage', self.image));


// Подсказка у кнопки ...........
this.tooltipText = "Блокнот в вкладке \nЛ: Открыть блокнот \nС: Открыть блокнот с текстом( выделенным или из буфера ) \nП: CB меню";


Справка
скрытый текст

Выделить код

Код:

<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/><title>Notepad</title>
<head>
 <link rel="shortcut icon" href=""type="image/x-icon">
</head>

<TEXTAREA></TEXTAREA>
<style>

html {
  background-color: #696969;
  overflow: hidden;
}
textarea {
  background: white;
  overflow-y: auto;
  width: 100%;
  height: 100%;
  border-radius: 2px;
  border: 2px inset #170202;
  color: black;
}
</style>


[firefox] 72.0.2

Отредактировано egorsemenov06 (21-01-2020 21:46:45)

Отсутствует

 

№1410021-01-2020 12:01:07

sonyas75
Участник
 
Группа: Members
Откуда: Ставрополь
Зарегистрирован: 22-03-2011
Сообщений: 385
UA: Firefox 72.0

Re: Custom Buttons

ничего не понимаю, поэтому вопрос, вот такой код работает в принципе?

Выделить код

Код:

addEventListener("click", function(e) {
   if ( e.button == 0 && e.target.nodeName == "tab" ) BrowserReload();
}, true, gBrowser.mTabContainer );

у меня не фурычит, а хотелось бы :blush:

На форуме

 

Board footer

Powered by PunBB
Modified by Mozilla Russia
Copyright © 2004–2011 Mozilla Russia
Язык отображения форума: [Русский] [English]