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

Юристы зарабатывают огромные деньги и славу, оперируя хорошим знанием законов. Правила форума — простой путь к успешному общению.

№1352619-07-2019 00:11:56

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

Re: Custom Buttons

Dumby
Отлично, модификаторы работают. Спасибо.

Не понял куда добавить. Код очистки quantumbar'а:

Я просто хотел, чтобы модификатор вставлялся в чистый quantumbar. Спасибо за код, кнопку для себя поправил.
А со второй кнопкой всё нормально оказалось, просто её код шёл в инициализации за предыдущим. Поэтому и не работал.

Отсутствует

 

№1352719-07-2019 09:59:05

Erik010
Участник
 
Группа: Members
Зарегистрирован: 19-07-2019
Сообщений: 1
UA: Firefox 68.0

Re: Custom Buttons

Dumby
А есть ли кнопка очистки searchbar'a?

Отсутствует

 

№1352819-07-2019 14:21:02

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

Re: Custom Buttons

Erik010
Есть такие кнопки "очистка поля поиска колесом мышки, проверяйте у себя. Я использую вторую ссылку.
https://forum.mozilla-russia.org/viewtopic.php?pid=661897#p661897
https://forum.mozilla-russia.org/viewtopic.php?pid=734731#p734731

Отредактировано xrun1 (19-07-2019 14:22:35)

Отсутствует

 

№1352919-07-2019 17:14:48

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby опять запнулся, вроде уже есть примеры, но до сих пор не могу понять для многопроцессорного режима, как объект вызывать.   
Есть у меня такая кнопка (фрагмент):

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

Выделить код

Код:

var defaultURL = "data:text/html;base64," + window.btoa(unescape(encodeURIComponent( this.Help ))); // файл HTML

this.onclick = e => {
        if (e.button == 0) {   
             // запомнить текст из буфера обмена и скопировать текст на странице ....
               var clip = gClipboard.read();
               goDoCommand("cmd_copy");
             
             // открыть блокнот в новой вкладке
               gBrowser.selectedTab = gBrowser.addTrustedTab(defaultURL); 
              
               gBrowser.addEventListener("pageshow", function(event) {
               this.removeEventListener("pageshow", arguments.callee, true);
               content.document.getElementById("req_message").value = gClipboard.read(); 
               if (clip) gClipboard.write(clip);
               }, true);    
               } 
        if (e.button == 1)   SidebarUI.toggle(id);
     }


Как в многопроцессорном режиме вызвать document.getElementById("req_message") ) и вставить в него из буфера текст.
И второе есть одна примерно в том же ключе
скрытый текст

Выделить код

Код:

var XML = window.XML || {};

var alertsService = Components.classes["@mozilla.org/alerts-service;1"].getService(Components.interfaces.nsIAlertsService);
var buttImage = this.image; 

var menu = '<menupopup xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"\n\
              xmlns:cb="http://xsms.nm.ru/custombuttons/">\n\
     <menuitem\n\
       class="menuitem-iconic"\n\
       image=""\n\
       label="Все буквы прописные"\n\
       onclick="' + e4xConv_encodeHTML("document.getElementById('" + this.id + "').allLowerCaseLetters(event)", true) + '"/>  \n\
     <menuitem\n\
       class="menuitem-iconic"\n\
       image=""\n\
       label="Заглавная только первая буква"\n\
       onclick="' + e4xConv_encodeHTML("document.getElementById('" + this.id + "').capitalizeFirstLetter(event)", true) + '"/>          \n\
     <menuitem\n\
       class="menuitem-iconic"\n\
       image=""\n\
       label="Заглавные все буквы"\n\
       onclick="' + e4xConv_encodeHTML("document.getElementById('" + this.id + "').capitalizeAllLetters(event)", true) + '"/>  \n\
     <menuitem\n\
       class="menuitem-iconic"\n\
       image=""\n\
       label="Заглавные для выбранных слов"\n\
       onclick="' + e4xConv_encodeHTML("document.getElementById('" + this.id + "').capitalizeSelectedWords(event)", true) + '"/>                \n\
</menupopup>';

menu = menu.replace(/>\s+</g, "><");
var bool = XML.prettyPrinting;
XML.prettyPrinting = false;
var el = new DOMParser().parseFromString(menu, "application/xml").documentElement;
XML.prettyPrinting = bool;
this.appendChild(el);
this.type = "menu";
this.orient = "horizontal";
this.menuClick = function(event) {
  event.preventDefault();
  event.stopPropagation();
  this.open = false;
}


 /*------------------Регистр букв начало------------------*/
 /*------------------Все буквы прописные------------------*/

this.allLowerCaseLetters = function(event) 
{
  if ( !this.intro.initMeth( this ) ) {
    return;
  } // End if ( !this.intro.initMeth( this ) )
  var x = this.intro;
  with(x){
    text += Highlight.toLowerCase();
    nLen = text.length;
    text += eBlock;
    theBox.value = text;
    theBox.selectionStart = theBox.selectionEnd = nLen ;
    theBox.scrollTop = oPosition + nHeight;
  }
}; 

//Заглавная только первая буква------------------

this.capitalizeFirstLetter = function(event) 
{
  if ( !this.intro.initMeth( this ) ) {
    return;
  } // End if ( !this.intro.initMeth( this ) )
  var x = this.intro;
  with(x){
    text += Highlight.substr( 0, 1 ).toUpperCase();
    text += Highlight.substr( 1 ).toLowerCase();
    nLen = text.length;
    text += eBlock;
    theBox.value = text;
    theBox.selectionStart = theBox.selectionEnd = nLen ;
    theBox.scrollTop = oPosition + nHeight;
  }
}; 

//Заглавные все буквы------------------

this.capitalizeAllLetters = function(event) 
{
  if ( !this.intro.initMeth( this ) ) {
    return;
  } // End if ( !this.intro.initMeth( this ) )
  var x = this.intro;
  with(x){
    text += Highlight.toUpperCase();
    nLen = text.length;
    text += eBlock;
    theBox.value = text;
    theBox.selectionStart = theBox.selectionEnd = nLen ;
    theBox.scrollTop = oPosition + nHeight;
  }
}; 
 
//Заглавные для выбранных слов------------------

this.capitalizeSelectedWords = function(event) 
{
  if ( !this.intro.initMeth( this ) ) {
    return;
  } // End if ( !this.intro.initMeth( this ) )
  var x = this.intro;
  with(x){
var tmp = "";
var tmpfront = "";

    tmp += Highlight.substr( 0, 1 ).toUpperCase() + Highlight.substring(1, Highlight.length);

    // capitalise remaining words
  if(tmp.indexOf(" ") > 0)
  {
    while(tmp.indexOf(" ") > 0) //that's a SPACE not a null
    {
      var spaceswap = tmp.indexOf(" ")
      tmpfront += tmp.substr(0, spaceswap + 1); 
      tmp = Highlight.substr(tmpfront.length,  Highlight.length);
      tmp = tmp.substr(0, 1).toUpperCase() + tmp.substr(1,  Highlight.length);
    }
  }
    text = tmpfront + tmp;    
    gClipboard.write(text);
    goDoCommand("cmd_paste");
  }
}; 

// Button Code Block Start ------------------------
this.intro = {
  theBox: {},
  startPos: 0,
  endPos: 0,
  oPosition: 0,
  oHeight: 0,
  text: '',
  nHeight: 0,
  Highlight: '',
  eBlock: '',
  cBoard: '',
  n: '',
  nLen: 0,
  arr: [],
  initMeth: function( oBtn ){
    with( this ) {
     theBox  = document.commandDispatcher.focusedElement;
        if ( theBox ) {
        startPos = theBox.selectionStart;
        endPos = theBox.selectionEnd;
        oPosition = theBox.scrollTop;
        oHeight = theBox.scrollHeight;
        text = theBox.value.substring(0,startPos);
        nHeight = theBox.scrollHeight - oHeight;
        Highlight = theBox.value.substring(startPos, endPos);
        eBlock = theBox.value.substring(endPos, theBox.value.length);
        n = '';
        return true;
      } // End if ( this.theBox )
    } // End with( this )
    return false;
  } // End initMeth
};


function e4xConv_encodeHTML(s, isAttr) {
    s = String(s)
        .replace(/&/g, "&amp;")
        .replace(/</g, "&lt;")
        .replace(/>/g, "&gt;")
        .replace(/"/g, "&quot;");
    if(isAttr) {
        s = s
            .replace(/\t/g, "&#x9;")
            .replace(/\n/g, "&#xA;")
            .replace(/\r/g, "&#xD;");
    }
    return s;
}


Тут все пункты меню завязаны на this.intro, а вней на theBox  = document.commandDispatcher.focusedElement; то есть вызвать объект - theBox

Отсутствует

 

№1353020-07-2019 06:32:13

Dobrov
Участник
 
Группа: Members
Откуда: Irkutsk
Зарегистрирован: 04-10-2011
Сообщений: 59
UA: Firefox 60.0

Re: Custom Buttons

Есть стиль от Vitaliy V: прокрутка вкладок колёсиком мыши
Просьба к экспертам: переделать этот код, чтобы он работал не в userChrome.css, а в кнопке CustonButtons

Выделить код

Код:

<?xml version="1.0" encoding="UTF-8"?>
<bindings xmlns="http://www.mozilla.org/xbl">
    <binding id="wheeltabs">
        <implementation>
            <field name="_eventListeners">[];</field>
            <field name="initialized">false;</field>
            <constructor><![CDATA[
                if (!window.gBrowser)
                    window.addEventListener("load", this, false);
                else
                    this.init();
            ]]></constructor>
            <destructor><![CDATA[
                if (!this.initialized)
                    return;
                this._eventListeners.forEach((args) => args[0].removeEventListener(args[1], args[2], args[3]));
                this.initialized = false;
            ]]></destructor>
            <method name="init">
                <body><![CDATA[
                    if (this.initialized || !("tabContainer" in gBrowser))
                        return;
                    this.initialized = true;
                    var tabContain = gBrowser.tabContainer;
                    this._addEventListener(tabContain, "wheel", function(event) {
                        if (event.shiftKey || event.ctrlKey || event.altKey || event.metaKey)
                            return;
                        event.stopPropagation();
                        event.preventDefault();
                        this.advanceSelectedTab(event.deltaY < 0 ? 1 : -1, true);
                    }, true);
                ]]></body>
            </method>
            <method name="handleEvent">
                <parameter name="event"/>
                <body><![CDATA[
                    window.removeEventListener("load", this, false);
                    this.init();
                ]]></body>
            </method>
            <method name="_addEventListener">
                <body><![CDATA[
                    arguments[0].addEventListener(arguments[1], arguments[2], arguments[3]);
                    this._eventListeners.push(arguments);
                ]]></body>
            </method>
        </implementation>
    </binding>
</bindings>

подключение кода в userChrome.css

Выделить код

Код:

@-moz-document url-prefix("chrome://browser/content/browser.x") {
  vbox#browser-border-end {
    -moz-binding: url("./bindings/wheeltabs.xml#wheeltabs") !important;
    display: -moz-box !important;
  }
  vbox#browser-border-end[hidden="true"] {
    visibility: collapse !important;
    width: 0 !important;
    padding: 0 !important;
    border: none !important;
  }
}

Отсутствует

 

№1353120-07-2019 09:58:19

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

Re: Custom Buttons

Andrey_Krropotkin пишет:

но до сих пор не могу понять для многопроцессорного режима, как объект вызывать

Совет: возьми бумажку, напиши на ней крупными буквами
«ЭТОГО ЗДЕСЬ НЕТ» и повесь куда-нибудь на видное место.
Когда захочется опять что-нибудь вызвать, сразу смотри на эту памятку.

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

Но можно посылать туда скрипты на исполнение.
И можно посылать туда сообщения и принимать сообщения оттуда.

Возвращаю твои коды, надеюсь не сильно испортил

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

Выделить код

Код:

var defaultURL = 'data:text/html;charset=utf-8,<textarea id="req_message" style="width: 400px; height: 300px;"/>';

var leftClick = () => {
    var msg = _id + ":NotepadPageShowAndPaste";
    var url = "data:," + encodeURIComponent(`addEventListener("pageshow", () => {
        content.document.getElementById("req_message").focus();
        docShell.doCommand("cmd_paste");
        sendAsyncMessage("${msg}");
    }, {once: true})`);

    var busy = 0, clip, listener = () => {
        if (clip) clip = gClipboard.write(clip);
        busy = 0;
    }
    messageManager.addMessageListener(msg, listener);
    addDestructor(() => messageManager.removeMessageListener(msg, listener));

    var openTab = () => gBrowser.getBrowserForTab(
        gBrowser.selectedTab = gBrowser.addTrustedTab(defaultURL)
    ).messageManager.loadFrameScript(url, false);

    (leftClick = () => {
        if (busy && Date.now() - busy < 4e3) return;
        busy = Date.now();

        // запомнить текст из буфера обмена и скопировать текст на странице ....
        clip = gClipboard.read();
        goDoCommand("cmd_copy");
    
        // открыть блокнот в новой вкладке
        setTimeout(openTab, 50);
    })();
}
this.onclick = e => {
    if (!e.button) leftClick();
    else if (e.button == 1) SidebarUI.toggle(id);
}


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

Выделить код

Код:

(this.type != "menu" && (this.type = "menu") && !this.hasAttribute("is")) || (intro => {
    var popup = this.appendChild(document.createXULElement("menupopup"));
    this.onmousedown = e => {
        if (e.button) return;
        this.onmousedown = null;
        popup.setAttribute("context", "");
        popup.setAttribute("oncommand", "handleCommand(event);");
        popup.append(MozXULElement.parseXULToFragment(`
            <menuitem class="menuitem-iconic"
                image=""
                label="Все буквы прописные"
                value="allLowerCaseLetters"/>

            <menuitem class="menuitem-iconic"
                image=""
                label="Заглавная только первая буква"
                value="capitalizeFirstLetter"/>

            <menuitem class="menuitem-iconic"
                image=""
                label="Заглавные все буквы"
                value="capitalizeAllLetters"/>

            <menuitem class="menuitem-iconic"
                image=""
                label="Заглавные для выбранных слов"\n\
                value="capitalizeSelectedWords"/> 
        `));
        popup.handleCommand = e => {
            var br = document.activeElement;
            if (!br) return;
            var url = urls[e.target.value];
            br.localName == "browser" && br.isRemoteBrowser
                ? br.messageManager.loadFrameScript(url, false)
                : Services.scriptloader.loadSubScript(url, window);
        }
    }
    var u = code => "data:," + encodeURIComponent(
        `(function(intro) {\n\tif (intro.initMeth()) with(intro) {${code}}\n})({${intro}})`
    );
    var urls = {
        // Все буквы прописные --------------------------
        get allLowerCaseLetters() {
            delete this.allLowerCaseLetters; return this.allLowerCaseLetters = u(`
            text += Highlight.toLowerCase();
            nLen = text.length;
            text += eBlock;
            theBox.value = text;
            theBox.selectionStart = theBox.selectionEnd = nLen;
            theBox.scrollTop = oPosition + nHeight;
        `)},
        // Заглавная только первая буква ----------------
        get capitalizeFirstLetter() {
            delete this.capitalizeFirstLetter; return this.capitalizeFirstLetter = u(`
            text += Highlight.substr(0, 1).toUpperCase();
            text += Highlight.substr(1).toLowerCase();
            nLen = text.length;
            text += eBlock;
            theBox.value = text;
            theBox.selectionStart = theBox.selectionEnd = nLen ;
            theBox.scrollTop = oPosition + nHeight;
        `)},
        // Заглавные все буквы --------------------------
        get capitalizeAllLetters() {
            delete this.capitalizeAllLetters; return this.capitalizeAllLetters = u(`
            text += Highlight.toUpperCase();
            nLen = text.length;
            text += eBlock;
            theBox.value = text;
            theBox.selectionStart = theBox.selectionEnd = nLen ;
            theBox.scrollTop = oPosition + nHeight;
        `)},
        // Заглавные для выбранных слов -----------------
        get capitalizeSelectedWords() {
            delete this.capitalizeSelectedWords; return this.capitalizeSelectedWords = u(`
            var tmp = "", tmpfront = "";
            tmp += Highlight.substr(0, 1).toUpperCase() + Highlight.substring(1, Highlight.length);
            
            // capitalise remaining words
            if (tmp.indexOf(" ") > 0) while(tmp.indexOf(" ") > 0) { //that's a SPACE not a null

                var spaceswap = tmp.indexOf(" ");
                tmpfront += tmp.substr(0, spaceswap + 1);
                tmp = Highlight.substr(tmpfront.length, Highlight.length);
                tmp = tmp.substr(0, 1).toUpperCase() + tmp.substr(1,  Highlight.length);
            }
            text = tmpfront + tmp;
            theBox.editor.QueryInterface(Ci.nsIPlaintextEditor).insertText(text);
        `)}
    };
})(`
    theBox: {}, arr: [],
    text: "", Highlight: "", eBlock: "", cBoard: "", n: "",
    startPos: 0, endPos: 0, oPosition: 0, oHeight: 0, nHeight: 0, nLen: 0,
    initMeth() {
        with(this) {
            theBox = Cc["@mozilla.org/focus-manager;1"]
                .getService(Ci.nsIFocusManager).focusedElement;
            if (!theBox) return false;

            startPos = theBox.selectionStart;
            endPos = theBox.selectionEnd;
            oPosition = theBox.scrollTop;
            oHeight = theBox.scrollHeight;
            text = theBox.value.substring(0, startPos);
            nHeight = theBox.scrollHeight - oHeight;
            Highlight = theBox.value.substring(startPos, endPos);
            eBlock = theBox.value.substring(endPos, theBox.value.length);
            n = "";
            return true;
        }
    }
`);


Dobrov пишет:

к экспертам

А что, простая копипаста не работает что ли?

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

Выделить код

Код:

(tc => addEventListener("wheel", event => {
    if (event.shiftKey || event.ctrlKey || event.altKey || event.metaKey)
        return;
    event.stopPropagation();
    event.preventDefault();
    tc.advanceSelectedTab(event.deltaY < 0 ? 1 : -1, true);
}, true, tc || 1))(gBrowser.tabContainer);

Отсутствует

 

№1353220-07-2019 12:49:32

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby спасибо. Скажи так правильно будет

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

Выделить код

Код:

if (e.button == 0) {  
              if (gMultiProcessBrowser){  leftClick(); } 
              else {   
             // запомнить текст из буфера обмена и скопировать текст на странице ....
               var clip = gClipboard.read();
               goDoCommand("cmd_copy");
             
             // открыть блокнот в новой вкладке
               gBrowser.selectedTab = gBrowser.addTrustedTab(defaultURL); 
              
               gBrowser.addEventListener("pageshow", function(event) {
               this.removeEventListener("pageshow", arguments.callee, true);
               content.document.getElementById("req_message").value = gClipboard.read(); 
               if (clip) gClipboard.write(clip);
               }, true);
                } ;   
               } 


а то мне кажется в однопроцессорном больше памями жрет и дольше загружается

Отредактировано Andrey_Krropotkin (20-07-2019 12:50:14)

Отсутствует

 

№1353320-07-2019 20:16:13

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

Re: Custom Buttons

Andrey_Krropotkin
gMultiProcessBrowser это nsILoadContext.useRemoteTabs, а он getter-only,
значит менятся в этом окне уже не будет. Можно сделать типа
var leftClick = () => {
    if (!gMultiProcessBrowser) return (leftClick = () => {
        .......
    })();
    .......
}
Но и так вполне нормально.

Отсутствует

 

№1353422-07-2019 20:20:43

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby объясни еще раз пожалуйста как отправить в другой процесс скрипты или сообщения на исполнение
на примере вот этой функции
// Сохранить иконку текущего сайта с диалогом сохранения .............
function getSiteName() {
   try { var domain = content.document.domain.split('.') } catch(e) { return "" };
   domain = (domain.length == 2) ? domain[0] : domain[1]
   return domain.charAt(0).toUpperCase() + domain.slice(1).split('.')[0] + " "; 
};
function saveFavicon() { saveImageURL(gBrowser.selectedTab.image, getSiteName(), null, false, false, null, content.document) };

Некоторые простые функции я пустил через javascript и gBrowser. loadURI, но сложные не переводятся в бурмаклеты, вот например:

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

Выделить код

Код:

function SaveHTML(event) {
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();
  }
});

};
var resolveURL = function (url, base) {
    try {
        var ioService = Components.classes['@mozilla.org/network/io-service;1'].getService(Components.interfaces.nsIIOService);
        var baseURI = ioService.newURI(base, null, null);
        var absURI = ioService.newURI(url, null, baseURI);
        return absURI.spec;
    } catch (e) {}
};

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 encodeImg = function (src, obj) {
    var canvas, img, ret = src;
    if (/^https?:\/\//.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((/\.jpe?g/i.test(src) ? '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 = {'\b': '\\b', '\t': '\\t', '\n': '\\n', '\f': '\\f', '\r': '\\r', '\x22' : '\\\x22', '\\': '\\\\'};
        while (chr = str.charAt(i++)) {
            ret += meta[chr] || chr;
        };
        return '\x22' + ret + '\x22';
    },
    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 (obj.hasOwnProperty(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) ? String(obj) : 'null';
        case 'Object': return objToSrc(obj);
        case 'String': return strToSrc(obj);
        default: return obj ? (obj.nodeType == 1 && obj.id ? 'document.getElementById(' + strToSrc(obj.id) + ')' : '{}') : 'null';
    }
};

var mainWin = document.commandDispatcher.focusedWindow.top == content ? document.commandDispatcher.focusedWindow : content
var selWin = getSelWin(mainWin), win = selWin || mainWin, doc = win.document, loc = win.location;
var ele, pEle, clone, reUrl = /(url\(\x22)(.+?)(\x22\))/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, prev, url, next) {
        if (!/^[a-z]+:/.test(url)) url = resolveURL(url, loc.href);
        return prev + encodeImg(url) + next;
    });
    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) != '#') 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 (unsafeWin) {
    if ('$' in unsafeWin) 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 unsafeWin) {
        if (name in f.contentWindow || !/^[a-zA-Z_$][0-9a-zA-Z_$]*$/.test(name)) continue;
        try {
            str = toSrc(unsafeWin[name]);
            if (!/\{\s*\[native code\]\s*\}/.test(str)) {
                script.appendChild(doc.createTextNode('var ' + name + ' = ' + str.replace(/<\/(script>)/ig, '<\\/$1') + ';\n'));
            }
        } catch (e) {};
    };
    f.parentNode.removeChild(f);
    if (script.childNodes.length) this.nextSibling.appendChild(script);
};
head.copyScript(win.wrappedJSObject || win);

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))) {
                    var css = !rule.cssText ? '' : rule.cssText.replace(reUrl, function (a, prev, url, next) {
                        if (!/^[a-z]+:/.test(url)) url = resolveURL(url, s.href || loc.href);
                        if(rule.type == 1 && rule.style && rule.style.backgroundImage) url = encodeImg(url);
                        return prev + url + next;
                    });
                    style.appendChild(doc.createTextNode(css + '\n'));
                }
            } 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('\n'));

var doctype = '', dt = doc.doctype;
if (dt && dt.name) {
    doctype += '<!DOCTYPE ' + dt.name;
    if (dt.publicId) doctype += ' PUBLIC \x22' + dt.publicId + '\x22';
    if (dt.systemId) doctype += ' \x22' + dt.systemId + '\x22';
    doctype += '>\n';
};
var fileName = selWin ? win.getSelection().toString() : (title && title.text ? title.text : loc.pathname.split('/').pop());
fileName = fileName.replace(/[:\\\/<>?*|"]+/g, '_').replace(/\s+/g, ' ').slice(0, 100).replace(/^\s+|\s+$/g, '');
fileName += (function () {
    var d = new Date(), z = function(n){return '_' + (n < 10 ? '0' : '') + n};
    return z(d.getHours()) + z(d.getMinutes()) + z(d.getSeconds());
})();
if(!/\.html?$/.test(fileName))fileName += '.html';

saveToFile(doctype + sel.innerHTML + '\n<!-- This document saved from ' + (loc.protocol != 'data:' ? loc.href : 'data:uri') + ' -->', fileName);
 }


, хотя есть еще старый бурмаклет
скрытый текст

Выделить код

Код:

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'};loc.href='data:text/phf;charset=utf-8,'+encodeURIComponent(doctype+sel.innerHTML+'%5Cn<!-- This document saved from '+(loc.protocol!='data:'%3Floc.href:'data:uri')+' -->')})();


Он работает, но файл сохраняется без имени и без типа файла -html (просто существует много расширений сохранения, но здесь в этом коде можно сохранять и выделенное) (пытался допилить бурмаклет - не получилось). Был бы признателен если бы Вы  бурмаклет подправили или по другому через messageManager (для меня это пока сложно, еще не понимаю как) .
  И еще интересует ( в качестве затрат ресурсов) что лучше пускать через javascript, javascript в base64 или messageManager

Отредактировано Andrey_Krropotkin (22-07-2019 23:53:03)

Отсутствует

 

№1353523-07-2019 20:12:50

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

Re: Custom Buttons

Andrey_Krropotkin пишет:

объясни еще раз пожалуйста как отправить в другой процесс скрипты или сообщения на исполнение
на примере вот этой функции

Не, функция кривовата для примера, давай упростим,
раз уж затруднение вызывает сама концепция.

скрытый текст
Итак, допустим, зачем-то, понадобился content.document.domain
Определяем что будет происходить по первому клику:

Задаём имя для сообщения, что-нибудь с претензией на уникальность,
чтобы не пересекалось с другими кодами.

Пишем код для исполнения там, в другом процессе, в ContentFrameMessageManager'е.
В этом окружении предопределён метод sendAsyncMessage() для отправки сообщений,
и есть content, который, так же как здесь в однопрцессье, ссылается на контентское окно.
Код состоит из одной строки и отправляет в сообщении наш domain.

Теперь собираем адрес frame script'а. Ставим в начало «data:,» и присоединяем код.
Текст скрипта не энкодим, здесь он простой, будет работать и так.

Затем определяем листенер. В первом агументе будет пришедшее сообщение.
Далее получаем ссылку на оконный message manager и просим его
зарегистрировать наш листенер. Ну и добавляем деструктор для удаления листенера.

Наконец, переопределяем что будет происходить по клику:
просим message manager <browser>'а активной вкладки загрузить туда скрипт с нашим адресом.
Ну и сразу вызываем, потому что уже кликнуто.

Скрипт отработает там, отправит сообщение сюда,
листенер подхватит сообщение и выведет алертом результат.

Выделить код

Код:

this._handleClick = () => {

    var messageName = "CB:Test:GetContentDocumentDomain";
    var code = `
        sendAsyncMessage("${messageName}", content.document.domain);
    `;
    var scriptURL = "data:," + /*encodeURIComponent*/(code);

    var listener = message => {
        Services.prompt.alert(window, message.name, message.data);
    }
    var wmm = window.messageManager;
    wmm.addMessageListener(messageName, listener);
    addDestructor(() => wmm.removeMessageListener(messageName, listener));

    (this._handleClick = () => gBrowser.selectedBrowser
        .messageManager.loadFrameScript(scriptURL, false)
    )();
}


не переводятся в бурмаклеты, вот например

Ну ещё бы, где букмарклеты и где Components

Был бы признателен если бы Вы  бурмаклет подправили

Странно меня про них спрашивать. Не знаю, может попробуй сменить концовку

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

Выделить код

Код:

    /*
    loc.href = 'data:text/phf;charset=utf-8,' + encodeURIComponent(doctype + sel.innerHTML + '\n<!-- This document saved from ' + (loc.protocol != 'data:' ? loc.href : 'data:uri') + ' -->')
    */
    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 = "Some Name";
    a.setAttribute("download", name + ".html");
    a.click();
    a.remove();
})();

Отсутствует

 

№1353623-07-2019 20:24:45

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby Большое спасибо, буду разбираться.

Отсутствует

 

№1353725-07-2019 19:35:37

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby вроде несколько кнопок по Вашему уроку подправил для многопроцессорного. Сейчас запнулся вот на такой кнопке:

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

Выделить код

Код:

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

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


var pref = "CB.Shortcuts.pathToSaveShortcuts";
var folderpath="C:\\Users\\vieva\\Desktop";         // папка для сохранения иконок для ярлыков и ярлыков сайтов
var alertsService = Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService);


// Создать меню для кнопки .............
var array = [
   { label: "Сохранить значок веб-сайта", func: "saveFavicon()", image: ""},
   { label: "Запомнить значок веб-сайта как base64", func: "copyFaviconData()", image: ""},  
   { separator: ''},
   { label: "Сохранить ярлык страницы как…", func: "saveShortcuts()", image: ""},
   { separator: ''},
   { label: "Сохранить всю страницу как PNG", func: "WebScreenShotoncaptureAll()", image: ""},
   { label: "Сохранить видимую часть страницы как PNG", func: "WebScreenShotoncapturePage()", image: false},
   { label: "Сохранить выбранный элемент страницы как PNG", func: "WebScreenShotByClick.init()", image: ""},
   { label: "Сохранить выбранную область страницы как PNG", func: "WebScreenShotByClipping.init()", image: ""},
   { separator: ''},
   { label: "Кодировать изображение или текстовой файл в base64", func: "copyFaviconbase()", 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;
}





// Сохранить как PNG страницу или части страницы .............
function WebScreenShotoncapture(win, x, y, width, height){
var canvas = document.createElementNS(xhtmlns, 'canvas');
      canvas.width = width;
      canvas.height = height;
      var ctx = canvas.getContext("2d");
      ((i = 17)=> { 
         try { ctx.drawWindow(win, x, y, canvas.width, canvas.height, "white") }
         catch(e) { canvas.height = canvas.width*i; arguments.callee(--i) };
      })();
      var url = makeURI(canvas.toDataURL("image/png"));
      var fp = window.makeFilePicker();
      fp.init(window, "Сохранить как…", fp.modeSave);
      fp.appendFilter("", "*.png");
           fp.defaultString = getTabLabel() + ".png";   
      fp.open(res => {
         if (res == fp.returnCancel || !fp.file) return;

        makeWebBrowserPersist().saveURI(
    makeURI(canvas.toDataURL("image/png")), document.nodePrincipal,
    null, null, null, null, null, fp.file, null
     );   
            
      });
   };
function WebScreenShotoncaptureAll(){
        var win = content;
        WebScreenShotoncapture(win, 0, 0, win.innerWidth + win.scrollMaxX, win.innerHeight + win.scrollMaxY);
   };
function WebScreenShotoncapturePage(){
      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;
      WebScreenShotoncapture(win, scrX, scrY, win.innerWidth, win.innerHeight);
   };   


// Сохранить выбранную область страницы как PNG ....
WebScreenShotByClipping = {
   capture: WebScreenShotoncapture,
   handleEvent: function(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: function() {
      this.win = document.commandDispatcher.focusedWindow;
      if (this.win == window) this.win = content;
      this.doc = this.win.document;
      this.body = this.doc.body;
      if (!this.body instanceof HTMLBodyElement){
          Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService).showAlertNotification(self.image, 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 = getComputedStyle(this.body, '').cursor;
      this.body.style.cursor = 'crosshair';
      ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.addEventListener(type, this, true));
   },
   uninit: function() {
      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.capture.apply(this, pos);
      ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.removeEventListener(type, this, true));
   },
};

// Сохранить выбранный элемент на странице как PNG ....
  WebScreenShotByClick = {
   capture: WebScreenShotoncapture,
   getPosition: function() {
      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: function() {
      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: function(e) {
      if (this.orgStyle) this.target.style.cssText = this.orgStyle;
      else this.target.removeAttribute('style'); 
   },
   handleEvent: function(e) {
      switch(e.type){
             case 'click':
                if (e.button) return;
                e.preventDefault();
                e.stopPropagation();
                this.lowlight();
                this.capture.apply(this, this.getPosition());
                this.uninit();
                break;
             case 'mouseover':
                if (this.target) this.lowlight();
                this.target = e.target;
                this.highlight();
                break;
      }
   },
   init: function() {
      this.win = content;
      this.doc = content.document;
      ["click", "mouseover"].forEach(type=> this.doc.addEventListener(type, this, true));
   },
   uninit: function() {
      this.target = false;
      ["click", "mouseover"].forEach(type=> this.doc.removeEventListener(type, this, true));
   },
};


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 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 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); // удалить стиль когда подсказка закрывается
      }, "");
});
};



// Сохранить иконку текущего сайта с диалогом сохранения .............
function saveFavicon() {
        let browserMM = gBrowser.selectedBrowser.messageManager;
        browserMM.addMessageListener('getSdomain', function listener(message) {
           
           function getSiteName() {
                   try { var domain = message.data.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);
            browserMM.removeMessageListener('getSdomain', listener, true);
        });
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSdomain", content.document.domain)', false);
};


// Скопировать иконку текущего сайта как 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);
};

 
// Добавляем в контекстного меню страницы новые пункты .............
((contextMenu, el)=> {
   // в контекстного меню изображений ....
   var baseItem = contextMenu.appendChild(document.createXULElement("menuitem"));
   baseItem.id = "content-baseItem";
   baseItem.setAttribute("label", "Запомнить изображение как base64");
   baseItem.setAttribute("class", "menuitem-iconic");
   baseItem.setAttribute("image", "");
   baseItem.onclick =()=> WebScreenShotonImage(gContextMenu.target);
   

   


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

   var editorItem = contextMenu.insertBefore(document.createXULElement("menuitem"), el);
   editorItem.id = "content-editorItem";
   editorItem.setAttribute("label", "Открыть выделенный текст в внешнем редакторе");
   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"); 
      baseItem.hidden =  !gContextMenu.onImage || !cbu.getPrefs("CB.Save.WebScreenShotOnImage"); 
   }, false, contextMenu);

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


// Сохранить выделенный текст в файл на рабочем столе .............
function saveSelectionToFile() {
   // создать текст для записи
   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", gContextMenuContentData.selectionInfo.fullText); 
   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();
   }, "");
};


// Создать текстовой файл с выделенным текстом в папке профиля и открыть в редакторе .............
function textToEditor() {
   var text = convertFromUnicode("UTF-8", gContextMenuContentData.selectionInfo.fullText); 
   var file = Services.dirsvc.get('ProfD', Ci.nsIFile);
   file.append("TextToEditor.txt");
   custombuttonsUtils.writeFile(file.path, text);
   file.launch(); 
};


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

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


Здесь пока не могу побороть вроде бы простой пункт - Запомнить изображение как base64, в контекстном меню (в частности сколько не пытался разными способами заменить в "baseItem.onclick = ( )=> WebScreenShotonImage(gContextMenu.target);" - "gContextMenu.target" (через сообщения и другие способы пробавал, все равно в function WebScreenShotonImage(image) - image = null))
И осталось здесь запустить функции function WebScreenShotoncaptureAll(), function WebScreenShotoncapturePage(), WebScreenShotByClipping, WebScreenShotByClick, которые все завязаны на function WebScreenShotoncapture(win, x, y, width, height) и на content   в другой процесс, но пока даже не знаю как. Подскажите пожайлуста

Отредактировано Andrey_Krropotkin (25-07-2019 19:38:40)

Отсутствует

 

№1353826-07-2019 21:36:23

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

Re: Custom Buttons

Andrey_Krropotkin пишет:

через сообщения и другие способы пробавал, все равно в function WebScreenShotonImage(image) - image = null

Не-не, нельзя отправить из процесса в процесс никакой живой реальный объект
типа элемента или XPCOM объекта, можно отправлять только данные.
Но в этом конкретном случае всё можно сделать прямо там, на месте.

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

Выделить код

Код:

(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;
    },
    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"));


И осталось здесь запустить функции

Ага, всего-то ничего :).

Отсутствует

 

№1353926-07-2019 22:42:49

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby пишет:

Ага, всего-то ничего :).

и что все так плохо? И ничего придумать нельзя или полностью код менять? В принципе  есть штатная сохранялка. И наверно последняя кнопка которую мне надо исправить - это переключатель раскладки. Тут много вариантов есть, У меня стоит вот такая кнопка

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

Выделить код

Код:

var keybUtils = {
    //== Options
    noSelBehavior: { // Shift+Home
        ctrlKey:  false,
        altKey:   false,
        shiftKey: true,
        metaKey:  false,
        keyCode:  KeyEvent.DOM_VK_HOME,
        charCode: 0
    },
    // 0 - do nothing
    // 1 - convert all text
    // Or use object like following to simulate "keypress" event:
    /*
    noSelBehavior: { // Ctrl+Shift+Left
        ctrlKey:  true,
        altKey:   false,
        shiftKey: true,
        metaKey:  false,
        keyCode:  KeyEvent.DOM_VK_LEFT,
        charCode: 0
    }
    */
    convTableForward: { // ru -> en
        "\"": "@",
        ":": "^",
        ";": "$",
        "?": "&",
        ",": "?",
        "/": "|",
        ".": "/",
        "э": "'",
        "б": ",",
        "ю": ".",
        "Ж": ":",
        "ж": ";",
        "Б": "<",
        "Ю": ">",
        "Э": "\"",
        "х": "[",
        "ъ": "]",
        "ё": "`",
        "Х": "{",
        "Ъ": "}",
        "Ё": "~",
        "№": "#",
        "Ф": "A",
        "ф": "a",
        "И": "B",
        "и": "b",
        "С": "C",
        "с": "c",
        "В": "D",
        "в": "d",
        "У": "E",
        "у": "e",
        "А": "F",
        "а": "f",
        "П": "G",
        "п": "g",
        "Р": "H",
        "р": "h",
        "Ш": "I",
        "ш": "i",
        "О": "J",
        "о": "j",
        "Л": "K",
        "л": "k",
        "Д": "L",
        "д": "l",
        "Ь": "M",
        "ь": "m",
        "Т": "N",
        "т": "n",
        "Щ": "O",
        "щ": "o",
        "З": "P",
        "з": "p",
        "Й": "Q",
        "й": "q",
        "К": "R",
        "к": "r",
        "Ы": "S",
        "ы": "s",
        "Е": "T",
        "е": "t",
        "Г": "U",
        "г": "u",
        "М": "V",
        "м": "v",
        "Ц": "W",
        "ц": "w",
        "Ч": "X",
        "ч": "x",
        "Н": "Y",
        "н": "y",
        "Я": "Z",
        "я": "z",
        __proto__: null
    },
    //== End of options
    button: this,
    get convTableBackward() {
        var ctb = { __proto__: null };
        var ctf = this.convTableForward;
        for(var c in ctf)
            ctb[ctf[c]] = c;
        delete this.convTableBackward;
        return this.convTableBackward = ctb;
    },
    inPrimaryLayout: function(s) {
        for(var i = 0, l = s.length; i < l; ++i) {
            var c = s.charAt(i);
            if(c in this.convTableForward)
                return true;
            if(c in this.convTableBackward)
                return false;
        }
        return false;
    },
    switchKeybLayout: function(s, convTable) {
        var res = "";
        for(var i = 0, l = s.length; i < l; ++i) {
            var c = s.charAt(i);
            res += c in convTable ? convTable[c] : c;
        }
        return res;
    },
    switchSelKeybLayout: function(_subCall, _focusFixed) {
        if(
            !_focusFixed
            && "closeMenus" in window
            && document.commandDispatcher.focusedElement == this.button
        ) {
            closeMenus(this.button);
            setTimeout(function(_this) {
                _this.switchSelKeybLayout(_subCall, true);
            }, 0, this);
            return;
        }
        var fe = document.commandDispatcher.focusedElement;
        if(!fe)
            return;
        if(fe instanceof HTMLInputElement || fe instanceof HTMLTextAreaElement) {
            var ta = fe;
            try {
                var val = ta.value;
                var sel = val.substring(ta.selectionStart, ta.selectionEnd);
            }
            catch(e) { // Non-text HTMLInputElement
                return;
            }
            if(!sel && val && this.noSelBehavior && !_subCall) {
                if(this.noSelBehavior == 1) {
                    ta.selectionStart = 0;
                    ta.selectionEnd = val.length;
                    sel = val;
                }
                else {
                    this.handleNoSel(ta);
                    return;
                }
            }
            if(!sel)
                return;
            var res = this.switchKeybLayout(
                sel,
                this.inPrimaryLayout(sel)
                    ? this.convTableForward
                    : this.convTableBackward
            );
            if(res != sel)
                this.insertText(ta, res);
        }
        else if(fe.contentEditable == "true") {
            var doc = fe.ownerDocument;

            var docURI = doc.documentURI;
            if(
                docURI.substr(0, 5) == "data:"
                && docURI.indexOf("chrome://browser/skin/devtools/") != -1
            ) {
                //~ todo: seems like we only can use paste from clipboard here...
                return;
            }

            var sel = doc.defaultView.getSelection();
            var rng = sel.rangeCount && sel.getRangeAt(0);
            var tmpNode;
            if(!rng || rng.collapsed) {
                if(!this.noSelBehavior || _subCall)
                    return;
                if(this.noSelBehavior == 1) {
                    var r = doc.createRange();
                    r.selectNodeContents(fe);
                    sel.removeAllRanges();
                    sel.addRange(r);
                    tmpNode = fe.cloneNode(true);
                }
                else {
                    this.handleNoSel(fe);
                    return;
                }
            }
            else {
                tmpNode = doc.createElementNS("http://www.w3.org/1999/xhtml", "div");
                tmpNode.appendChild(rng.cloneContents());
            }

            var orig = tmpNode.innerHTML;
            var convTable = this.inPrimaryLayout(tmpNode.textContent)
                ? this.convTableForward
                : this.convTableBackward;

            var _this = this;
            var parseChildNodes = function(node) {
                if(node instanceof Element) {
                    var childNodes = node.childNodes;
                    for(var i = childNodes.length - 1; i >= 0; --i)
                        parseChildNodes(childNodes[i]);
                }
                else if(node.nodeType == node.TEXT_NODE) {
                    var text = node.nodeValue;
                    var newText = _this.switchKeybLayout(node.nodeValue, convTable);
                    if(newText != text)
                        node.parentNode.replaceChild(doc.createTextNode(newText), node);
                }
            }
            parseChildNodes(tmpNode);

            var res = tmpNode.innerHTML;
            if(res != orig)
                doc.execCommand("insertHTML", false, res);
        }
    },
    handleNoSel: function(node) {
        this.select(node);
        this.switchSelKeybLayout(true);
    },
    select: function(node) {
        var e = this.noSelBehavior;
        if(!e || typeof e != "object")
            return;
        var evt = document.createEvent("KeyboardEvent");
        evt.initKeyEvent(
            "keypress", true /*bubbles*/, true /*cancelable*/, node.ownerDocument.defaultView,
            e.ctrlKey, e.altKey, e.shiftKey, e.metaKey,
            e.keyCode, e.charCode
        );
        node.dispatchEvent(evt);
    },
    insertText: function(ta, text) {
        //var editor = ta.QueryInterface(Components.interfaces.nsIDOMNSEditableElement).editor
        var editor = ta.editor
            .QueryInterface(Components.interfaces.nsIPlaintextEditor);
        if(editor.flags & editor.eEditorReadonlyMask)
            return;

        var sTop = ta.scrollTop;
        var sHeight = ta.scrollHeight;
        var sLeft = ta.scrollLeft;
        // var sWidth = ta.scrollWidth;

        if(text)
            editor.insertText(text);
        else
            editor.deleteSelection(0, 0);

        ta.scrollTop = sTop + (ta.scrollHeight - sHeight);
        ta.scrollLeft = sLeft; // + (ta.scrollWidth - sWidth);
    }
};

var btn = this;
if(btn instanceof XULElement && addEventListener.length > 3) {
    addEventListener("command", function(e) {
        if(e.target != btn)
            return;
        e.preventDefault();
        e.stopPropagation();
        keybUtils.switchSelKeybLayout();
    }, true, this.parentNode);
}
keybUtils.switchSelKeybLayout();


Как с ней боротсься не знаю. Есть у меня в HTML блокнотe такой вариант
скрытый текст

Выделить код

Код:

function replaceSelectedText(obj,cbFunc)
{
  obj.focus();
  if (document.selection) 
  {
     var s = document.selection.createRange(); 
     if (s.text)
     {
       eval("s.text="+cbFunc+"(s.text);");
           s.select();
           return true;
     }
  }
  else if (typeof(obj.selectionStart)=="number")
  {
     if (obj.selectionStart!=obj.selectionEnd)
     {
        var start = obj.selectionStart;
        var end = obj.selectionEnd;

        eval("var rs = "+cbFunc+"(obj.value.substr(start,end-start));");
        obj.value = obj.value.substr(0,start)+rs+obj.value.substr(end);
        obj.setSelectionRange(end,end);
     }
     return true;
  }
  return false;
} 
function layoutRU(str) {
        replacer = {
            "q":"й", "w":"ц", "e":"у", "r":"к", "t":"е", "y":"н", "u":"г", 
            "i":"ш", "o":"щ", "p":"з", "[":"х", "]":"ъ", "a":"ф", "s":"ы", 
            "d":"в", "f":"а", "g":"п", "h":"р", "j":"о", "k":"л", "l":"д", 
            ";":"ж", "'":"э", "z":"я", "x":"ч", "c":"с", "v":"м", "b":"и", 
            "n":"т", "m":"ь", ",":"б", ".":"ю", "/":".",
        };  
        for(i=0; i < str.length; i++){                        
            if( replacer[ str[i].toLowerCase() ] != undefined){
                if(str[i] == str[i].toLowerCase()){
                    replace = replacer[ str[i].toLowerCase() ];    
                } else if(str[i] == str[i].toUpperCase()){
                    replace = replacer[ str[i].toLowerCase() ].toUpperCase();
                } 
                str = str.replace(str[i], replace);
            }
        }
       return str;
}
   
function layoutEN(str, reverse) {
   replacer = {
            "й":"q", "ц":"w", "у":"e", "к":"r", "е":"t", "н":"y", "г":"u", 
            "ш":"i", "щ":"o", "з":"p", "х":"[", "ъ":"]", "ф":"a", "ы":"s", 
            "в":"d", "а":"f", "п":"g", "р":"h", "о":"j", "л":"k", "д":"l", 
            "ж":";", "э":"'", "я":"z", "ч":"x", "с":"c", "м":"v", "и":"b", 
            "т":"n", "ь":"m", "б":",", "ю":".", ".":"/",
        };  
        for(i=0; i < str.length; i++){                        
            if( replacer[ str[i].toLowerCase() ] != undefined){
                if(str[i] == str[i].toLowerCase()){
                    replace = replacer[ str[i].toLowerCase() ];    
                } else if(str[i] == str[i].toUpperCase()){
                    replace = replacer[ str[i].toLowerCase() ].toUpperCase();
                } 
                str = str.replace(str[i], replace);
            }
        }
       return str;

и кнопки HTML:
<input type="button" value="RU" title="Меняет раскладку клавиатуры" onClick="replaceSelectedText(document.getElementById('req_message'),'layoutRU');">
<input type="button" value="EN" title="Меняет раскладку клавиатуры" onClick="replaceSelectedText(document.getElementById('req_message'),'layoutEN');">
    
где     req_message - это textarea                               


Можно ли для многопроцессорного простенькую кнопку создать для  полей ввода  - переключатель раскладки EN→RU и RU→EN

Отредактировано Andrey_Krropotkin (27-07-2019 12:12:19)

Отсутствует

 

№1354027-07-2019 12:56:35

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

Re: Custom Buttons

Andrey_Krropotkin пишет:

и что все так плохо? И ничего придумать нельзя

Ладно, уговорил, перерисую тяп-ляп отдельной кнопкой, дорабатывай.

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

Выделить код

Код:

((main, parts) => this._handleClick = () => {
    var df = MozXULElement.parseXULToFragment(`
        <menupopup>
            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить всю страницу как PNG"
                value="all"/>
    
            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить видимую часть страницы как PNG"
                value="page"/>
    
            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить выбранный элемент страницы как PNG"
                value="click"/>
    
            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить выбранную область страницы как PNG"
                value="clipping"/>
        </menupopup>
    `);
    var popup = df.firstChild;
    popup.setAttribute("context", "");
    popup.setAttribute("oncommand", "handleCommand(event);");
    popup.handleCommand = e => {
        var name = _id + ":DataURLReady";
        main = main.replace("%MESSAGE_NAME%", name);

        var urls = {}, configurable = true, enumerable = true;
        Object.entries(parts).forEach(([key, part]) => Object.defineProperty(urls, key, {
            configurable, enumerable, get() {
                var value = `data:;charset=utf-8,({${
                    encodeURIComponent(main + part)
                }%0A}).init("${key}")`;
                Object.defineProperty(urls, key, {configurable, enumerable, value});
                return value;
        }}));
        var getTabLabel = () => {
            var label = gBrowser.selectedTab.label;      
            var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " ");
            return label.substring(0, 50);
        }
        var listener = msg => {
            var fp = makeFilePicker();
            fp.init(window, "Сохранить как…", fp.modeSave);
            fp.appendFilter("", "*.png");
            fp.defaultString = getTabLabel() + ".png";   
            fp.open(res => res == fp.returnCancel || !fp.file || makeWebBrowserPersist().saveURI(
                Services.io.newURI(msg.data), document.nodePrincipal,
                null, null, null, null, null, fp.file, null
            ));
        }
        messageManager.addMessageListener(name, listener);
        addDestructor(() => messageManager.removeMessageListener(name, listener));

        (popup.handleCommand = e => gBrowser.selectedBrowser.messageManager
            .loadFrameScript(urls[e.target.value], false)
        )(e);
    }
    this.append(df);
    (this._handleClick = () => popup.openPopup(this, "after_start"))();
})(`
    init(cmd) {
        cmd.startsWith("c")
            ? this[cmd].init(this[cmd].parent = this)
            : this[cmd]();
    },
    capture(win, x, y, width, height) {
        var canvas = win.document.createElementNS("${xhtmlns}", "canvas");
        canvas.width = width;
        canvas.height = height;
        var ctx = canvas.getContext("2d");
        var tryDraw = ind => {
            try {ctx.drawWindow(win, x, y, canvas.width, canvas.height, "white")}
            catch(ex) {canvas.height = ind * canvas.width; tryDraw(--ind);}
        }
        tryDraw(17);
        sendAsyncMessage("%MESSAGE_NAME%", canvas.toDataURL("image/png"));
    },
    `, {

    all: `all() {
        var win = content;
        this.capture(win, 0, 0, win.innerWidth + win.scrollMaxX, win.innerHeight + win.scrollMaxY);
    }`,
    page: `page() {
        var win = content, doc = win.document, body = doc.body, html = doc.documentElement;
        var scrX = (body.scrollLeft || html.scrollLeft) - html.clientLeft;
        var scrY = (body.scrollTop || html.scrollTop) - html.clientTop;
        this.capture(win, scrX, scrY, win.innerWidth, win.innerHeight);
    }`,
    clipping: `clipping: {
        handleEvent(e) {
            if (e.button) return false;
            e.preventDefault();
            e.stopPropagation();
            switch(e.type) {
                case "mousedown":
                    this.downX = e.pageX;
                    this.downY = e.pageY;
                    this.bs.left = this.downX + "px";
                    this.bs.top = this.downY + "px";
                    this.body.appendChild(this.box);
                    this.flag = true;
                    break;
                case "mousemove":
                    if (!this.flag) return;
                    this.moveX = e.pageX;
                    this.moveY = e.pageY;
                    if (this.downX > this.moveX) this.bs.left = this.moveX + "px";
                    if (this.downY > this.moveY) this.bs.top  = this.moveY + "px";
                    this.bs.width = Math.abs(this.moveX - this.downX) + "px";
                    this.bs.height = Math.abs(this.moveY - this.downY) + "px";
                    break;
                case "mouseup":
                    this.uninit();
                    break;
            }
        },
        init() {
            var win = {};
            Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager)
                .getFocusedElementForWindow(content, true, win);
            this.win = win.value;

            this.doc = this.win.document;
            this.body = this.doc.body;
            if (!HTMLBodyElement.isInstance(this.body)) {
                Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService)
                    .showAlertNotification("${self.image}", ${JSON.stringify(self.label)}, "Не удается захватить!");
                return false;
            }
            this.flag = null;
            this.box = this.doc.createElement("div");
            this.bs = this.box.style;
            this.bs.border = "#0f0 dashed 2px";
            this.bs.position = "absolute";
            this.bs.zIndex = "2147483647";
            this.defaultCursor = this.win.getComputedStyle(this.body, "").cursor;
            this.body.style.cursor = "crosshair";
            ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.addEventListener(type, this, true));
        },
        uninit() {
            var pos = [this.win, parseInt(this.bs.left), parseInt(this.bs.top), parseInt(this.bs.width), parseInt(this.bs.height)];
            this.body.style.cursor = this.defaultCursor;
            this.body.removeChild(this.box);
            this.parent.capture.apply(this, pos);
            ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.removeEventListener(type, this, true));
        }
    }`,
    click: `click: {
        getPosition() {
            var html = this.doc.documentElement;
            var body = this.doc.body;
            var rect = this.target.getBoundingClientRect();
            return [
                this.win,
                Math.round(rect.left) + (body.scrollLeft || html.scrollLeft) - html.clientLeft,
                Math.round(rect.top) + (body.scrollTop || html.scrollTop) - html.clientTop,
                parseInt(rect.width),
                parseInt(rect.height)
            ];
        },
        highlight() {
            this.orgStyle = this.target.hasAttribute("style") ? this.target.style.cssText : false;
            this.target.style.cssText += "outline: red 2px solid; outline-offset: 2px; -moz-outline-radius: 2px;";
        },
        lowlight() {
            if (this.orgStyle) this.target.style.cssText = this.orgStyle;
            else this.target.removeAttribute("style");
        },
        handleEvent(e) {
            switch(e.type){
                case "click":
                    if (e.button) return;
                    e.preventDefault();
                    e.stopPropagation();
                    this.lowlight();
                    this.parent.capture.apply(this, this.getPosition());
                    this.uninit();
                    break;
                case "mouseover":
                    if (this.target) this.lowlight();
                    this.target = e.target;
                    this.highlight();
                    break;
            }
        },
        init() {
            this.win = content;
            this.doc = content.document;
            ["click", "mouseover"].forEach(type=> this.doc.addEventListener(type, this, true));
        },
        uninit() {
            this.target = false;
            ["click", "mouseover"].forEach(type=> this.doc.removeEventListener(type, this, true));
        }
    }`
});


У меня стоит вот такая кнопка

Тоже самое.

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

Выделить код

Код:

(keybUtils => {
    var btn = this;
    var listener = {
        handleEvent(e) {
            if(e.target != btn)
                return;
            e.preventDefault();
            e.stopPropagation();
            this.switch();
        },
        switch() {
            var br = document.activeElement;
            br && br.localName == "browser" && br.isRemoteBrowser
                ? br.messageManager.loadFrameScript(this.url, false)
                : this.keybUtils.switchSelKeybLayout();
        },
        get url() {
            delete this.url;
            return this.url = `data:;charset=utf-8,(${
                encodeURIComponent(keybUtils)
            }).switchSelKeybLayout()`;
        },
        get keybUtils() {
            delete this.keybUtils;
            var url = "data:;charset=utf-8,this.keybUtils = " + encodeURIComponent(keybUtils);
            Services.scriptloader.loadSubScript(url, this);
            this.keybUtils.button = btn;
            this.keybUtils.getFocusedElement = function(_subCall, _focusFixed) {
                if(
                    !_focusFixed
                    && "closeMenus" in window
                    && document.commandDispatcher.focusedElement == this.button
                ) {
                    closeMenus(this.button);
                    setTimeout(function(_this) {
                        _this.switchSelKeybLayout(_subCall, true);
                    }, 0, this);
                    return;
                }
                return document.commandDispatcher.focusedElement;
            }
            return this.keybUtils;
        }
    };
    if(btn instanceof XULElement && addEventListener.length > 3) {
        addEventListener("command", listener, true, this.parentNode);
    }
    listener.switch();
})(`{
    //== Options
    noSelBehavior: { // Shift+Home
        ctrlKey:  false,
        altKey:   false,
        shiftKey: true,
        metaKey:  false,
        keyCode:  KeyEvent.DOM_VK_HOME,
        charCode: 0
    },
    // 0 - do nothing
    // 1 - convert all text
    // Or use object like following to simulate "keypress" event:
    /*
    noSelBehavior: { // Ctrl+Shift+Left
        ctrlKey:  true,
        altKey:   false,
        shiftKey: true,
        metaKey:  false,
        keyCode:  KeyEvent.DOM_VK_LEFT,
        charCode: 0
    }
    */
    convTableForward: { // ru -> en
        "\\"": "@",
        ":": "^",
        ";": "$",
        "?": "&",
        ",": "?",
        "/": "|",
        ".": "/",
        "э": "'",
        "б": ",",
        "ю": ".",
        "Ж": ":",
        "ж": ";",
        "Б": "<",
        "Ю": ">",
        "Э": "\\"",
        "х": "[",
        "ъ": "]",
        "ё": "\`",
        "Х": "{",
        "Ъ": "}",
        "Ё": "~",
        "№": "#",
        "Ф": "A",
        "ф": "a",
        "И": "B",
        "и": "b",
        "С": "C",
        "с": "c",
        "В": "D",
        "в": "d",
        "У": "E",
        "у": "e",
        "А": "F",
        "а": "f",
        "П": "G",
        "п": "g",
        "Р": "H",
        "р": "h",
        "Ш": "I",
        "ш": "i",
        "О": "J",
        "о": "j",
        "Л": "K",
        "л": "k",
        "Д": "L",
        "д": "l",
        "Ь": "M",
        "ь": "m",
        "Т": "N",
        "т": "n",
        "Щ": "O",
        "щ": "o",
        "З": "P",
        "з": "p",
        "Й": "Q",
        "й": "q",
        "К": "R",
        "к": "r",
        "Ы": "S",
        "ы": "s",
        "Е": "T",
        "е": "t",
        "Г": "U",
        "г": "u",
        "М": "V",
        "м": "v",
        "Ц": "W",
        "ц": "w",
        "Ч": "X",
        "ч": "x",
        "Н": "Y",
        "н": "y",
        "Я": "Z",
        "я": "z",
        __proto__: null
    },
    //== End of options

    get convTableBackward() {
        var ctb = { __proto__: null };
        var ctf = this.convTableForward;
        for(var c in ctf)
            ctb[ctf[c]] = c;
        delete this.convTableBackward;
        return this.convTableBackward = ctb;
    },
    inPrimaryLayout: function(s) {
        for(var i = 0, l = s.length; i < l; ++i) {
            var c = s.charAt(i);
            if(c in this.convTableForward)
                return true;
            if(c in this.convTableBackward)
                return false;
        }
        return false;
    },
    switchKeybLayout: function(s, convTable) {
        var res = "";
        for(var i = 0, l = s.length; i < l; ++i) {
            var c = s.charAt(i);
            res += c in convTable ? convTable[c] : c;
        }
        return res;
    },
    getFocusedElement: function() {
        return Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager)
            .getFocusedElementForWindow(content, true, {});
    },
    switchSelKeybLayout: function(_subCall, _focusFixed) {
        var fe = this.getFocusedElement(_subCall, _focusFixed);
        if(!fe)
            return;
        if(fe instanceof HTMLInputElement || fe instanceof HTMLTextAreaElement) {
            var ta = fe;
            try {
                var val = ta.value;
                var sel = val.substring(ta.selectionStart, ta.selectionEnd);
            }
            catch(e) { // Non-text HTMLInputElement
                return;
            }
            if(!sel && val && this.noSelBehavior && !_subCall) {
                if(this.noSelBehavior == 1) {
                    ta.selectionStart = 0;
                    ta.selectionEnd = val.length;
                    sel = val;
                }
                else {
                    this.handleNoSel(ta);
                    return;
                }
            }
            if(!sel)
                return;
            var res = this.switchKeybLayout(
                sel,
                this.inPrimaryLayout(sel)
                    ? this.convTableForward
                    : this.convTableBackward
            );
            if(res != sel)
                this.insertText(ta, res);
        }
        else if(fe.contentEditable == "true") {
            var doc = fe.ownerDocument;

            var docURI = doc.documentURI;
            if(
                docURI.substr(0, 5) == "data:"
                && docURI.indexOf("chrome://browser/skin/devtools/") != -1
            ) {
                //~ todo: seems like we only can use paste from clipboard here...
                return;
            }

            var sel = doc.defaultView.getSelection();
            var rng = sel.rangeCount && sel.getRangeAt(0);
            var tmpNode;
            if(!rng || rng.collapsed) {
                if(!this.noSelBehavior || _subCall)
                    return;
                if(this.noSelBehavior == 1) {
                    var r = doc.createRange();
                    r.selectNodeContents(fe);
                    sel.removeAllRanges();
                    sel.addRange(r);
                    tmpNode = fe.cloneNode(true);
                }
                else {
                    this.handleNoSel(fe);
                    return;
                }
            }
            else {
                tmpNode = doc.createElementNS("http://www.w3.org/1999/xhtml", "div");
                tmpNode.appendChild(rng.cloneContents());
            }

            var orig = tmpNode.innerHTML;
            var convTable = this.inPrimaryLayout(tmpNode.textContent)
                ? this.convTableForward
                : this.convTableBackward;

            var _this = this;
            var parseChildNodes = function(node) {
                if(node instanceof Element) {
                    var childNodes = node.childNodes;
                    for(var i = childNodes.length - 1; i >= 0; --i)
                        parseChildNodes(childNodes[i]);
                }
                else if(node.nodeType == node.TEXT_NODE) {
                    var text = node.nodeValue;
                    var newText = _this.switchKeybLayout(node.nodeValue, convTable);
                    if(newText != text)
                        node.parentNode.replaceChild(doc.createTextNode(newText), node);
                }
            }
            parseChildNodes(tmpNode);

            var res = tmpNode.innerHTML;
            if(res != orig)
                doc.execCommand("insertHTML", false, res);
        }
    },
    handleNoSel: function(node) {
        this.select(node);
        this.switchSelKeybLayout(true);
    },
    select: function(node) {
        var e = this.noSelBehavior;
        if(!e || typeof e != "object")
            return;
        var evt = node.ownerDocument.createEvent("KeyboardEvent");
        evt.initKeyEvent(
            "keypress", true /*bubbles*/, true /*cancelable*/, node.ownerDocument.defaultView,
            e.ctrlKey, e.altKey, e.shiftKey, e.metaKey,
            e.keyCode, e.charCode
        );
        node.dispatchEvent(evt);
    },
    insertText: function(ta, text) {
        //var editor = ta.QueryInterface(Components.interfaces.nsIDOMNSEditableElement).editor
        var editor = ta.editor
            .QueryInterface(Components.interfaces.nsIPlaintextEditor);
        if(editor.flags & editor.eEditorReadonlyMask)
            return;

        var sTop = ta.scrollTop;
        var sHeight = ta.scrollHeight;
        var sLeft = ta.scrollLeft;
        // var sWidth = ta.scrollWidth;

        if(text)
            editor.insertText(text);
        else
            editor.deleteSelection(0, 0);

        ta.scrollTop = sTop + (ta.scrollHeight - sHeight);
        ta.scrollLeft = sLeft; // + (ta.scrollWidth - sWidth);
    }
}`);

Отредактировано Dumby (27-07-2019 12:58:04)

Отсутствует

 

№1354127-07-2019 14:48:18

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

Dumby очень большое спасибо.

Отредактировано Andrey_Krropotkin (27-07-2019 14:48:32)

Отсутствует

 

№1354227-07-2019 16:29:43

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

Re: Custom Buttons

Господа форумчане можно ли подправить кнопку Savefrom для мультипроцесорнного режима


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

Выделить код

Код:

/*Initialization Code*/
// Настройка функций кликов мыши для кнопки ..............
this._handleClick =()=> gBrowser.selectedTab = gBrowser.addTrustedTab('http://savefrom.net/?url=' + escape(content.location.href));


// Получаем из вкладки Справка массив с доменами закачку из которые подерживает savefrom ..............
var hosts = self.Help.split('\n').filter((host)=> /\S/.test(host)).map((host)=> host.trim());


// Если домен сайта совпадает с массивом меняем иконку и подсказку ..............
var [img] = document.getAnonymousNodes(self);
addEventListener("TabAttrModified", (e, tab = e.target)=> {
   if ( !tab.selected ) return;   
   
   if ( ~hosts.indexOf(content.document.domain.replace("www.", "")) ) 
        self.tooltipText = "Есть видео для загрузки",      
        img.src = "";
   else 
        self.tooltipText = "Нет видео для загрузки",
        img.src = self.image;
        
}, true, gBrowser.tabContainer );


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

Выделить код

Код:

filefactory.com
sendspace.com
youtube.com
metacafe.com
break.com
dailymotion.com
vimeo.com
sevenload.com
mail.ru
smotri.com
yandex.ru
rambler.ru
tvigle.ru
intv.ru
facebook.com
livejournal.com
vk.com
odnoklassniki.ru
soundcloud.com
liveinternet.ru
guitar-tube.com
gametrailers.com
twitch.tv
zaycev.net
1tv.ru
rutv.ru
ntv.ru
vesti.ru
mreporter.ru
karusel-tv.ru
autoplustv.ru
russia.ru
amik.ru
life.ru
a1tv.ru
kiwi.kz

Отредактировано egorsemenov06 (27-07-2019 19:02:28)

Отсутствует

 

№1354327-07-2019 18:56:33

psihkakihmalo
Участник
 
Группа: Members
Зарегистрирован: 28-01-2016
Сообщений: 24
UA: Firefox 50.0

Re: Custom Buttons

Подскажите пожалуйста код кнопки, которая при нажатии просто вызывает диалоговое окно "Параметры соединения" ("Connection Settings"), где задаются параметры прокси. Версия Фокса старая v.50, если это имеет значение.

Отсутствует

 

№1354427-07-2019 19:58:54

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

Re: Custom Buttons

пкм - параметры соединения
лкм - переключает системные настройки - свой урл прокси

скрытый текст
// Proxy от 11.12.2013.

// Настройка функций кликов мыши для кнопки ................................
this.onclick = function(e) {     
   
   if ( e.button == 0 ) {
        cbu.setPrefs("network.proxy.type", (cbu.getPrefs("network.proxy.type") == 2 ) ? 5 : 2 );
        }
                     
   if ( e.button == 1 ) menuPopup.showPopup( self, -1, -1, "popup", "bottomleft", "topleft");
         
   if ( e.button == 2 && !e.ctrlKey && !e.shiftKey && !e.altKey && !e.metaKey ) {
        e.preventDefault();
        cbu.getPrefs("Proxy.connectionsInTab") ? openConnectionsInTab(): openConnections();
        }
};



// Проверить наличие строк в about:config и создать если не существует ................................
if ( !cbu.isPref("Proxy.reset") ) {
     ["Proxy.reset", "Proxy.connectionsInTab", "Proxy.inContextMenu"].forEach(function( pref ) { cbu.setPrefs( pref, false ) });
};



// Подсказка для кнопки ................................
this.onmouseover = function() {
     var address = Application.prefs.getValue("network.proxy.http", 0);
     var port = Application.prefs.getValue("network.proxy.http_port", 0);
                   
     this.tooltipText = "Proxy \nЛ: Переключить прокси \nС: Mеню кнопки"
                            + "\nП: Открыть настройки прокси \n\nТекущие настройки прокси: "
                            + "\nПорт: "+ port + "\nАдрес: " + address;                                                   
};



// Создать меню для кнопки ................................
var array = [
    { label: "Добавление прокси в контекстом меню", value: 'Proxy.inContextMenu' },
    { label: "Открывать настройки прокси как вкладку", value: 'Proxy.connectionsInTab' },
    { label: "Переключать на режим 'Без прокси' при закрытии браузера ", value: 'Proxy.reset' }   
];

var menuPopup = self.appendChild( document.createElement("menupopup") );
array.forEach(function( m ) {
   var mItem = document.createElement("menuitem");
   mItem.setAttribute("label", m.label);
   mItem.setAttribute('type', 'checkbox');
   mItem.setAttribute('checked', cbu.getPrefs( m.value ) );
   mItem.setAttribute('onclick', 'custombuttons.setPrefs("' + m.value + '", !custombuttons.getPrefs("' + m.value + '"))'); 
   menuPopup.appendChild( mItem );
});
menuPopup.setAttribute("onclick", "event.stopPropagation()");

// добавить стандартное контекстное меню ....
menuPopup.appendChild( document.createElement("menuseparator") );
menuPopup.appendChild( document.createElement("menu") ).setAttribute("label", "Меню кнопки");
var clone = menuPopup.lastChild.appendChild( document.getElementById("custombuttons-contextpopup").cloneNode(true) );
clone.setAttribute("onpopupshowing", "document.popupNode = document.getElementById('" + _id + "')");



// Функция открывает настройки прокси в окне ................................
function openConnections() {
   var windows = Cc['@mozilla.org/appshell/window-mediator;1'].getService(Ci.nsIWindowMediator).getEnumerator(null);
   while( windows.hasMoreElements() ) {
          var win = windows.getNext();
          if ( win.name == 'Proxy') {   
               win.focus();
               return;
               }
   };   
   var win = openDialog("chrome://browser/content/preferences/connection.xul", "Proxy", "centerscreen");   

   // добавить атрибут "prefwindow"
   win.addEventListener("load", function f(e) {
       this.removeEventListener("load", f, true);
       e.target.documentElement.setAttribute("type", "prefwindow");
   }, true);
   
   // закрыть настройки прокси по клику на странице
   gBrowser.addEventListener("click", function c() {
      this.removeEventListener("click", c );
      try { win.close() } catch(e) {};
   });
};



// Функция открывает настройки прокси в вкладке ................................
function openConnectionsInTab() {
   var connections = gBrowser.getBrowserForTab( gBrowser.selectedTab = gBrowser.addTab("chrome://browser/content/preferences/connection.xul") );
         
   // oбработчик ждет пока откроется прокси, удаляет себя и добавляет атрибут
   connections.addEventListener("pageshow", function c(e) {         
      this.removeEventListener( e.type, c );         
      e.originalTarget.documentElement.setAttribute("type", "prefwindow");
   })   
};



// Установливать нужную иконку кнопки при старте баузера или при изменениях в 'about:config' ................................
var  s = "network.proxy.type";
function toggleImage() {
    var icon = self.ownerDocument.getAnonymousElementByAttribute( self, "class", "toolbarbutton-icon");
    icon.src = ( cbu.getPrefs(s) == 2 )
    ? self.image
    : '';
}; 
toggleImage();
gPrefService.addObserver( s, toggleImage, false );
addDestructor(function() { gPrefService.removeObserver( s, toggleImage, false ) });



// Переключать на режим 'Без прокси' при закрытии браузера если это разрешено в 'about:config' ................................
var switchOffProxy = {
    observe: function(subject, topic, data) {
       if ( data.toString() == "shutdown" && cbu.getPrefs("Proxy.reset") ) cbu.setPrefs("network.proxy.type", 0); 
    }
};
Services.obs.addObserver( switchOffProxy, "quit-application", false );
addDestructor(function() { Services.obs.addObserver( switchOffProxy, "quit-application", false ) }); 



// Создаем меню для добавление прокси в контекстном меню выделенного текста на странице ................................
(function func() {
  var contextMenu = document.getElementById("contentAreaContextMenu"); 
 
  // создать новый пункт меню
  var menuitem = document.createElement("menuitem");     
  menuitem.setAttribute("label", "Добавить прокси");
  menuitem.setAttribute("class", "menuitem-iconic");
  menuitem.setAttribute("image", self.image );
  addEventListener("command", function(e) { addNewProxy() }, false, menuitem );
 
  // устанавливаем где показывать пункт меню
  addEventListener("popupshowing", function() {
     menuitem.hidden = !cbu.getPrefs("Proxy.inContextMenu") || !gContextMenu.isContentSelected;
  }, false, contextMenu );
  addDestructor(function() { contextMenu.removeChild( menuitem ) });
  contextMenu.appendChild( menuitem ); // как последний пункт меню
 
 
  // добавление прокси
  function addNewProxy( sel, i ) { 
     var selection = document.commandDispatcher.focusedWindow.getSelection().toString();
     var sel = ( sel == undefined ) ? selection : sel.toString();
     sel = sel.replace(/^\s+|\s+$/g, ""); // удалить пробелы, слева и справа от строки
     sel = sel.replace(/\s+/g,":"); // заменить пробелы внутри строки
     
     // если порт ...
     if ( sel.length < 5 && isFinite(sel) ) { 
          sel = sel.replace(/:/g, "");
          sel = +sel;  // порт как число
          var lab = 'порт';
          var pref = 'network.proxy.http_port';
          }
     
     // если адрес ...
     if ( sel.length > 4 && !/:/.test(sel) && sel.split(".").length == 4 ) { 
          var lab = 'адрес';
          var pref = 'network.proxy.http';
          }     
     
     // если адрес и порт ...   
     if ( sel.length > 4 && /:/.test(sel) && sel.split(":").length == 2 && sel.split(".").length == 4 ) {
          var lab = 'адрес и порт';
          var array = sel.split(":"); 
          array.forEach(function(sel, i) { addNewProxy( sel, i ) });         
          }     
     
     Application.prefs.setValue( pref, sel ); // переключить настройки прокси     

     if ( lab == undefined || i !== undefined || i == 0 ) return;
     
     // всплывающая подсказка рядом с выделенным текстом ...     
     function showTooltip() {
        var tooltip = gBrowser.appendChild( document.createElement("tooltip") );
        tooltip.style.cssText = "color: red !important; font-weight: bold !important; font-size: 14px !important; -moz-box-orient: horizontal; text-align: center;";
   
        var image = tooltip.appendChild( document.createElement("image") );
        image.setAttribute("src", self.image );
   
        var label = tooltip.appendChild( document.createElement("label") );
        label.setAttribute("value", "Установлен " + lab + " прокси: " + sel );
   
        var focused = document.commandDispatcher.focusedWindow;
        var selection = focused.getSelection().getRangeAt(0).getBoundingClientRect();
        var posX = focused.mozInnerScreenX + selection.left;
        var posY = focused.mozInnerScreenY + selection.bottom - 5;   

        tooltip.showPopup( gBrowser, posX, posY );
        setTimeout(function() { gBrowser.removeChild( tooltip ) }, 2000 );
     };
     showTooltip();   
  };
   
})();

только сама кнопка древняя, не помню что я в ней менял, или не я, или не менял, или кто-то менял. у меня пашет

Отсутствует

 

№1354527-07-2019 20:55:07

Andrey_Krropotkin
Участник
 
Группа: Members
Зарегистрирован: 11-11-2011
Сообщений: 321
UA: Firefox 68.0

Re: Custom Buttons

egorsemenov06 попробуй замени

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

Выделить код

Код:

// Настройка функций кликов мыши для кнопки ..............
this._handleClick =()=> gBrowser.selectedTab = gBrowser.addTrustedTab('http://savefrom.net/?url=' + escape(gBrowser.currentURI.spec));


// Получаем из вкладки Справка массив с доменами закачку из которые подерживает savefrom ..............
var hosts = self.Help.split('\n').filter((host)=> /\S/.test(host)).map((host)=> host.trim());

 
// Если домен сайта совпадает с массивом меняем иконку и подсказку ..............
var [img] = document.getAnonymousNodes(self);
addEventListener("TabAttrModified", (e, tab = e.target)=> {
   if ( !tab.selected ) return;   
   
   let browserMM = gBrowser.selectedBrowser.messageManager;
        browserMM.addMessageListener('getSdomain', function listener(message) {
           
           if ( ~hosts.indexOf(message.data.replace("www.", "")) )
           self.tooltipText = "Есть видео для загрузки",      
        img.src = "";
   else 
        self.tooltipText = "Нет видео для загрузки",
        img.src = self.image;
            browserMM.removeMessageListener('getSdomain', listener, true);
        });
        browserMM.loadFrameScript('data:,sendAsyncMessage("getSdomain", content.document.domain)', false); 
        
        
}, true, gBrowser.tabContainer );

Отсутствует

 

№1354627-07-2019 21:26:46

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

Re: Custom Buttons

Andrey_Krropotkin пишет:

попробуй замени

Если уж пробовать, то может сначала так

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

Выделить код

Код:

/*Initialization Code*/

// Настройка функций кликов мыши для кнопки ..............
this._handleClick = () => gBrowser.selectedTab = gBrowser.addTrustedTab(
    "http://savefrom.net/?url=" + escape(gBrowser.currentURI.spec)
);

// Получаем из вкладки Справка массив с доменами закачку из которые подерживает savefrom ..............
var hosts = this.Help.split("\n").filter(host => /\S/.test(host)).map(host => host.trim());

// Если хост сайта совпадает с массивом меняем иконку и подсказку ..............
var img = this.icon || document.getAnonymousNodes(this)[0];
var listener = e => {
    if (!e.target.selected) return;   
    var uri = gBrowser.currentURI;
    if (
        uri.scheme.startsWith("http") &&
        hosts.includes(uri.host.replace(/^www\./, ""))
    )
        self.tooltipText = "Есть видео для загрузки",     
        img.src = "";
    else
        self.tooltipText = "Нет видео для загрузки",
        img.src = this.image;
}
addEventListener("TabAttrModified", listener, false, gBrowser.tabContainer);
listener({target: gBrowser.selectedTab});

Отредактировано Dumby (27-07-2019 21:29:00)

Отсутствует

 

№1354727-07-2019 21:45:37

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

Re: Custom Buttons

Dumby пишет:

Andrey_Krropotkin пишет:

попробуй замени

Если уж пробовать, то может сначала так

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

Выделить код

Код:

/*Initialization Code*/

// Настройка функций кликов мыши для кнопки ..............
this._handleClick = () => gBrowser.selectedTab = gBrowser.addTrustedTab(
    "http://savefrom.net/?url=" + escape(gBrowser.currentURI.spec)
);

// Получаем из вкладки Справка массив с доменами закачку из которые подерживает savefrom ..............
var hosts = this.Help.split("\n").filter(host => /\S/.test(host)).map(host => host.trim());

// Если хост сайта совпадает с массивом меняем иконку и подсказку ..............
var img = this.icon || document.getAnonymousNodes(this)[0];
var listener = e => {
    if (!e.target.selected) return;   
    var uri = gBrowser.currentURI;
    if (
        uri.scheme.startsWith("http") &&
        hosts.includes(uri.host.replace(/^www\./, ""))
    )
        self.tooltipText = "Есть видео для загрузки",     
        img.src = "";
    else
        self.tooltipText = "Нет видео для загрузки",
        img.src = this.image;
}
addEventListener("TabAttrModified", listener, false, gBrowser.tabContainer);
listener({target: gBrowser.selectedTab});


Спасибо работает

Отсутствует

 

№1354828-07-2019 06:03:10

psihkakihmalo
Участник
 
Группа: Members
Зарегистрирован: 28-01-2016
Сообщений: 24
UA: Firefox 50.0

Re: Custom Buttons

sonyas75 спасибо, работает

Отсутствует

 

№1354928-07-2019 09:39:42

grh1
Участник
 
Группа: Members
Зарегистрирован: 25-12-2016
Сообщений: 14
UA: Chrome 75.0

Re: Custom Buttons

del

Отредактировано grh1 (28-07-2019 12:27:06)

Отсутствует

 

№1355028-07-2019 16:56:38

psihkakihmalo
Участник
 
Группа: Members
Зарегистрирован: 28-01-2016
Сообщений: 24
UA: Firefox 50.0

Re: Custom Buttons

Подскажите, нет ли такой кнопки, которая бы при просмотре любого видео на любой странице могла определить и показать (ну или скопировать в буфер) прямую ссылку на это видео? Например, на seasonvar.ru запускаешь в плейере видео, потом жмёшь кнопку и она показывает ссылку на текущее .mp4, такого вида "http://data11-cdn.datalock.ru/fi2lm/64ce511c626e9a1e7167ccfa5eecc00c/7f_Stranger.Things.S01E01.720p.WEB.rus.LostFilm.TV.a1.19.07.16.mp4". Эту ссыль можно увидеть в Инструментах разработчика > Сеть , но это не удобно каждый раз туда лазить. Нельзя ли это кнопкой реализовать?

Отсутствует

 

Board footer

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