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

Хотите узнать больше о расширениях? Посмотрите ролики, рассказывающие о работе с расширениями Firefox.

№10113-10-2023 16:49:50

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

Re: userChrome.js

Dumby
Если возможно, то помогите, пожалуйста с моим вопросом.
xrun1
Вам спасибо за внимание и правки скрипта. :)

Отсутствует

 

№10213-10-2023 20:49:05

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

Re: userChrome.js

rubel пишет

Dumby
Если возможно, то помогите, пожалуйста с моим вопросом.

Ну сколько раз повторять, что проверить мне такое негде.
Хочешь чтобы я что-то из пальца высосал? Хорошо.


Допустим, что речь о переводе выделенного текста в окошке «Google Translate».
Тогда, находишь строку function ujs_google_translate (){
и меняешь на function ujs_google_translate(langTo_google_text) {


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

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

Выделить код

Код:

/*
        {label:"В окне Google", func: ujs_google_translate, image:gticon},
*/
        {label: "В окне Google на русский", func: () => ujs_google_translate("ru"), image: gticon},
        {label: "В окне Google на английский", func: () => ujs_google_translate("en"), image: gticon},

Отсутствует

 

№10314-10-2023 03:30:44

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

Re: userChrome.js

Dumby
Огромное Вам спасибо. Все заработало прекрасно.
Появился выбор языка перевода.
f2aa3ddd0cfddf216b6bd4fc14a12713.png 
Вот окончательный вариант скрипта google_translate.js для Aris-t2

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

Выделить код

Код:

location.href.endsWith("://browser/content/browser.xhtml") && ({
	async init(func) {
		await delayedStartupPromise;
		var code = func.toString();
		code = code.slice(code.indexOf("{") + 1, -1).trim();

		var addEventListener = (...args) => {
			var trg = args[3];
			if (!trg) trg = args[3] = window;
			trg.addEventListener(...args);
			this.handlers.push(args);
		}
		new Function(
			"_id,xhtmlns,addDestructor,addEventListener,gClipboard,LOG", code
		).call(
			this, "ucf-cbinit-google-translate", "http://www.w3.org/1999/xhtml",
			() => {}, addEventListener, {read: readFromClipboard}, Cu.reportError
		);
		window.addEventListener("unload", this, {once: true});
	},
	handlers: [],
	handleEvent() {
		for(var args of this.handlers)
			args.pop().removeEventListener(...args);
		delete this.handlers;
	}
}).init(() => {

	// Здесь код google-translate.js

//Google,
var langFrom_google_text = "auto";//авто
var langTo_google_text = "ru"; 

      
//Назначаем иконки
var mainicon="";
var gticon="";


function GetXmlHttpObject(){
         if (window.XMLHttpRequest){ return new XMLHttpRequest();}
         if (window.ActiveXObject) { return new ActiveXObject("Microsoft.XMLHTTP");}
        return null;
        };

var lc = navigator.lastClick = {};
addEventListener("mouseup", e => {
    if (e.button) return;
    lc.X = e.screenX - mozInnerScreenX;
    lc.Y = e.screenY - mozInnerScreenY;
}, false, gBrowser.tabpanels || 1);

var createWindow = function(text, status, title, id, pos, size){
var win = window, doc = win.document, wId = 'ujs_window'+(id || ''), w = doc.getElementById(wId);
    var keyDown = function(e){if(!e.shiftKey && !e.ctrlKey && !e.altKey && e.keyCode == 27)doc.getElementById(wId).closeWin()};
    if(w)w.closeWin();
    w = doc.createElementNS(xhtmlns, 'div');
      w.setAttribute('style', 'position:fixed;display:block;visibility:hidden;left:0;top:0;width:auto;height:auto;border:1px solid gray;padding:2px;margin:0;z-index:99999;overflow:hidden;cursor:move;'+(typeof w.style.borderRadius === 'string' ? 'background-color:#33343F;padding-top:0px;border-radius:4px;box-shadow:0 0 15px rgba(0,0,0,.4);' : 'background:-o-skin("Window Skin");'));
    w.id = wId;
    w.closeWin = function(){
        doc.removeEventListener('keydown', keyDown, false);
        this.parentNode.removeChild(this);
    };
    w.addEle = function(str, style){
        var ele = doc.createElementNS(xhtmlns, 'div');
        ele.setAttribute('style', style);
        if(str){
            ele.innerHTML = str;
            for(var el, all = ele.getElementsByTagName('*'), i = all.length; i--;){
                el = all[i];
                if(/^(script|frame|iframe|applet|embed|object)$/i.test(el.nodeName)){
                    el.parentNode.removeChild(el);
                }
                else{
                    for(var att = el.attributes, j = att.length; j--;){
                        if(/^on[a-z]+$/i.test(att[j].name))att[j].value = '';
                    }
                }
            }
        };
        return this.appendChild(ele);
    };

    var img = doc.createElementNS(xhtmlns, 'div');
    img.setAttribute('style', 'display:block;float:right;width:16px;height:16px;padding:0;margin-top:2px;margin-right:1px;border:none;cursor:pointer;background-image:url("");background:-o-skin("Caption Close Button Skin");');
    img.title = (win.navigator.language.indexOf('ru') == 0) ? '\u0417\u0430\u043A\u0440\u044B\u0442\u044C' : 'Close';
    img.addEventListener('click', function(){this.parentNode.closeWin()}, false);
    w.appendChild(img);
    var title = w.addEle(title, 'display:table;color:#000;font:17px Times New Roman;width:auto;height:auto;padding:0;margin:0 2px;cursor:text;');
        title.onclick = e => {
        e.preventDefault();
        var url = e.target.href;
        // Здесь открываем url как хотим.
        var ctabpos = gBrowser.selectedTab._tPos +1;
        gBrowser.moveTabTo(gBrowser.selectedTab = gBrowser.addWebTab(url), ctabpos);
        doc.getElementById(wId).closeWin();    
    }
    var cnt = doc.createElement("textarea");
    cnt.style.cssText = `
        color: #000;
        width: 310px;
        height: 160px;
        outline: none;
        padding-left: 3px;
        padding-bottom: 3px;
        border: 1px solid #aaa;
        background-color: #fafcfe;
        font: 17px Times New Roman;
    `;
    if (text) cnt.value = text.trim();
    w.append(cnt);
    w.addEle(status, 'display:table;font:12px Times New Roman;font-weight:bold;color:blue;width:auto;height:auto;padding-top:2px;margin:0 3px;cursor:pointer;');
    w.addEventListener('mousedown', function(e){
        if(e.target == w){
            e.preventDefault();
            var st = w.style;
            var mouseMove = e => {
                st.top = parseInt(st.top) + e.movementY + "px";
                st.left = parseInt(st.left) + e.movementX + "px";
            }
            doc.addEventListener('mousemove', mouseMove, false);
            doc.addEventListener('mouseup', function(){doc.removeEventListener('mousemove', mouseMove, false)}, false);
        }
    }, false);
    doc.documentElement.appendChild(w);
  
    if(size){
        cnt.style.height = size.height;
        cnt.style.width = size.width;
    }
    else{
        for(var i = 3; i < 10; i++){
            if(cnt.scrollHeight > cnt.offsetHeight || cnt.scrollWidth > cnt.offsetWidth){
                cnt.style.height = 80*i+'px';
                cnt.style.width = 160*i+'px';
            }
            else break;
        }
    };

    var docEle = (doc.compatMode == 'CSS1Compat' && win.postMessage) ? doc.documentElement : doc.body;
    var mX = docEle.clientWidth-w.offsetWidth, mY = docEle.clientHeight-w.offsetHeight;
    if(mX < 0){cnt.style.width = parseInt(cnt.style.width)+mX+'px'; mX = 0};
    if(mY < 0){cnt.style.height = parseInt(cnt.style.height)+mY+'px'; mY =0};
    var hW = parseInt(w.offsetWidth/2);
    w.style.left = (pos && pos.X < mX+hW ? (pos.X > hW ? pos.X-hW : 0) : mX)+'px';
    w.style.top = (pos && pos.Y+10 < mY ? pos.Y+10 : mY)+'px';
    w.style.visibility = 'visible';
    doc.addEventListener('keydown', keyDown, false);
	if (text) {
        var st = cnt.style;
        var div = cnt.editor.rootElement;

        var range = new Range();
        range.selectNode(div.firstChild);
        var rect = range.getBoundingClientRect();

        let w = Math.ceil(rect.width);
        if (cnt.scrollTopMax) {
            if (!matchMedia("(-moz-overlay-scrollbars)").matches) // ???
                w += InspectorUtils.getChildrenForNode(div, true, false).at(-1).clientWidth;
        }
        else st.height = Math.max(50, Math.ceil(rect.height) + 1) + "px";

        st.width = Math.max(200, w) + "px";
    }
    return w;
};

var getHash = function (txt) {
    TKK=eval('((function(){var a\x3d817046147;var b\x3d-335196159;return 410049+\x27.\x27+(a+b)})())');
    function sM(a) {
        var b;
        if (null !== yr)
            b = yr;
        else {
            b = wr(String.fromCharCode(84));
            var c = wr(String.fromCharCode(75));
            b = [b(), b()];
            b[1] = c();
            b = (yr = window[b.join(c())] || "") || ""
        }
        var d = wr(String.fromCharCode(116))
            , c = wr(String.fromCharCode(107))
            , d = [d(), d()];
        d[1] = c();
        c = "&" + d.join("") + "=";
        d = b.split(".");
        b = Number(d[0]) || 0;
        for (var e = [], f = 0, g = 0; g < a.length; g++) {
            var l = a.charCodeAt(g);
            128 > l ? e[f++] = l : (2048 > l ? e[f++] = l >> 6 | 192 : (55296 == (l & 64512) && g + 1 < a.length && 56320 == (a.charCodeAt(g + 1) & 64512) ? (l = 65536 + ((l & 1023) << 10) + (a.charCodeAt(++g) & 1023),
                e[f++] = l >> 18 | 240,
                e[f++] = l >> 12 & 63 | 128) : e[f++] = l >> 12 | 224,
                e[f++] = l >> 6 & 63 | 128),
                e[f++] = l & 63 | 128)
        }
        a = b;
        for (f = 0; f < e.length; f++)
            a += e[f],
                a = xr(a, "+-a^+6");
        a = xr(a, "+-3^+b+-f");
        a ^= Number(d[1]) || 0;
        0 > a && (a = (a & 2147483647) + 2147483648);
        a %= 1E6;
        return c + (a.toString() + "." + (a ^ b))
    }

    var yr = null;
    var wr = function(a) {
        return function() {
            return a
        }
    }
        , xr = function(a, b) {
        for (var c = 0; c < b.length - 2; c += 3) {
            var d = b.charAt(c + 2)
                , d = "a" <= d ? d.charCodeAt(0) - 87 : Number(d)
                , d = "+" == b.charAt(c + 1) ? a >>> d : a << d;
            a = "+" == b.charAt(c) ? a + d & 4294967295 : a ^ d
        }
        return a
    };

    return sM(txt);
};

//----------Перевести  текст  из буфера в окне Google------------
var ujs_google_translat = function (dir){
   var lng = 'ru';
   var txt = gClipboard.read(); 
   var l = dir.split('|');
   var encTxt = encodeURIComponent(txt);
   var winWait = function(lng){createWindow('', (lng == 'ru' ? 'Подождите идет перевод' : 'Wait, is going Translating')+'\u2026', 'Google Translate', '_gt', window.navigator.lastClick)};
    if (txt) {
    winWait(lng);
        var xhr = new XMLHttpRequest();
        var url = 'https://translate.google.com/translate_a/single?client=gtx&sl=' + l[0] + '&tl=' + l[1] + '&hl=' + lng + '&eotf=0&dt=at&dt=bd&dt=ex&dt=ld&dt=md&dt=qca&dt=rw&dt=rm&dt=ss&dt=t' + getHash(txt);
        var urlt = "http://translate.google.com/translate_t?text="+encTxt+"&sl=' + l[0] + '&tl=' + l[1] + '&hl=' + lng + '&eotf=0&ujs=gtt";
        xhr.open('POST', url, true);
        xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded;charset=utf-8');
        xhr.onreadystatechange = function() {
            try{
                if (xhr.readyState == 4 && xhr.status == 200) {
                    var result = '', status = '', tmp = JSON.parse(xhr.responseText.replace(/\[(?=,)/g, '[0').replace(/,(?=,|\])/g, ',0').replace(/\\n/g, ""));
                    for(var i = 0, n; n = tmp[0][i]; i++){
                        if(n[0])result += n[0].toString();
                    };
                  //  result = '<span style="background-color:inherit;color:inherit;font-size:inherit;font-family:Times,serif;">' + result + '</span>';
                    status = tmp[8][0][0].toUpperCase() + ' -\u203A ' + l[1].toUpperCase();

                    createWindow(result, status, '<a href="'+urlt.replace(/&/g,'&amp;')+'" target="_blank" style="display:inline;padding:0;margin:0;text-decoration:none;border:none;color:#009;font:16px Times New Roman;">Google Translate</a>', '_gt', window.navigator.lastClick);
                }
            } catch (x){LOG(x)};
        };
        xhr.send('q=' + encodeURIComponent(txt));
    };
};



//----------Перевести выделенный текст в окне Google------------
function ujs_google_translate(langTo_google_text) {
    var lng = 'ru';
    var txt = gContextMenu.selectionInfo.fullText;
    var encTxt = encodeURIComponent(txt);
    var winWait = function(lng){createWindow('', (lng == 'ru' ? 'Подождите идет перевод' : 'Wait, is going Translating')+'\u2026', 'Google Translate', '_gt', window.navigator.lastClick)};
    if (txt) {
    winWait(lng);
        var xhr = new XMLHttpRequest();
       // var url = 'https://translate.google.com/translate_a/single?client=gtx&sl=' + l[0] + '&tl=' + l[1] + '&hl=' + lng + '&eotf=0&dt=at&dt=bd&dt=ex&dt=ld&dt=md&dt=qca&dt=rw&dt=rm&dt=ss&dt=t' + getHash(txt);
        var url = 'https://translate.google.com/translate_a/single?client=gtx&sl=' + langFrom_google_text + '&tl=' + langTo_google_text + '&hl=' + lng + '&eotf=0&dt=at&dt=bd&dt=ex&dt=ld&dt=md&dt=qca&dt=rw&dt=rm&dt=ss&dt=t' + getHash(txt);
        var urlt = "http://translate.google.com/translate_t?text="+encTxt+"&sl='  + langFrom_google_text + '&tl=' + langTo_google_text +'&hl=' + lng + '&eotf=0&ujs=gtt";
      
        xhr.open('POST', url, true);
        xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded;charset=utf-8');
        xhr.onreadystatechange = function() {
            try{
                if (xhr.readyState == 4 && xhr.status == 200) {
                    var result = '', status = '', tmp = JSON.parse(xhr.responseText.replace(/\[(?=,)/g, '[0').replace(/,(?=,|\])/g, ',0').replace(/\\n/g, ""));
                    for(var i = 0, n; n = tmp[0][i]; i++){
                        if(n[0])result += n[0].toString();
                    };
                   // result = '<span style="background-color:inherit;color:inherit;font-size:inherit;font-family:Times,serif;">' + result + '</span>';
                    //status = tmp[8][0][0].toUpperCase() + ' -\u203A ' + l[1].toUpperCase();
                     status = tmp[8][0][0].toUpperCase() + ' -\u203A ' + langTo_google_text.toUpperCase();
                     createWindow(result, status, '<a href="'+urlt.replace(/&/g,'&amp;')+'" target="_blank" style="display:inline;padding:0;margin:0;text-decoration:none;border:none;color:#009;font:16px Times New Roman;">Google Translate</a>', '_gt', window.navigator.lastClick);
                }
            } catch (x){LOG(x)};
        };
        xhr.send('q=' + encodeURIComponent(txt));
     };
};


//----------Заменить текст переводом Google------------
function ujs_google_TexReplace() {
    var lng = 'ru';
    var txt = gContextMenu.selectionInfo.fullText;
    if (txt) {
        var xhr = new XMLHttpRequest();
        var url = 'https://translate.google.com/translate_a/single?client=gtx&sl=' + langFrom_google_text + '&tl=' + langTo_google_text + '&hl=' + lng + '&eotf=0&dt=at&dt=bd&dt=ex&dt=ld&dt=md&dt=qca&dt=rw&dt=rm&dt=ss&dt=t' + getHash(txt);
       
           function gettransdata(){
           xmlhttp=GetXmlHttpObject();
           xmlhttp.onreadystatechange=stateChanged;
           xmlhttp.open('POST', url, true);
           xmlhttp.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded;charset=utf-8');
           xmlhttp.send('q=' + encodeURIComponent(txt));
        }
        function stateChanged() {
            
           if (xmlhttp.readyState == 4 ) {
           var result = '';
           var data = JSON.parse(xmlhttp.responseText.replace(/\[(?=,)/g, '[0').replace(/,(?=,|\])/g, ',0').replace(/\\n/g, "<br />"));
           for(var i = 0, n; n = data[0][i]; i++){
                        if(n[0])result += n[0].toString();
                    };
        var msgName = _id + ":ReplaceSelectionRangeAt0";
        var url = "data:," + encodeURIComponent(
       `addMessageListener("${msgName}", function listener(msg) {
        removeMessageListener("${msgName}", listener);
        var win = {};
        Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager)
            .getFocusedElementForWindow(content, true, win);

        var sel = win.value.document.getSelection();
        if (sel.isCollapsed) return;
        var range = sel.getRangeAt(0);
        range.deleteContents();
        range.insertNode(range.createContextualFragment(msg.data));
    });`
);
function replace(tagString) {
    var mm = gBrowser.selectedBrowser.messageManager;
    mm.loadFrameScript(url, false);
    mm.sendAsyncMessage(msgName, tagString);
}
replace('<span>'+result+'</span>');
                }
        }  
        gettransdata();
    } 
};



//--------Перевести страницу с Google--------------
function ujs_googlePage_translate() {
   var urlt = gBrowser.currentURI.spec;  
   var url = "http://translate.google.com/translate?hl=ru&sl=auto&tl=ru&u="+ encodeURIComponent(urlt) + "&sandbox=1";
   gBrowser. fixupAndLoadURIString(url, {
   triggeringPrincipal: Services.scriptSecurityManager.getSystemPrincipal()
});
};


//Контекстное меню для перевода из буфера-------------------------------------------  

(function () {
 if ( document.getElementById("TranslateBufer") ) return; 
 var contextMenu = document.getElementById("contentAreaContextMenu");  
 var Item = document.createXULElement("menuitem");
       Item.setAttribute("label", "Перевод из буфера");
       Item.setAttribute("class", "menuitem-iconic");
       Item.setAttribute("image", mainicon);
       Item.addEventListener("command", function(){ujs_google_translat('auto|ru')}, false);

    contextMenu.insertBefore(Item, document.getElementById("context-viewpartialsource-selection") ); 
    addDestructor(function() { contextMenu.removeChild( Item ) });
 })();

  

 //Контекстное меню для перевода страниц-------------------------------------------  

(function () {
 if ( document.getElementById("TranslatePage") ) return; 
  
  var menu = document.createXULElement("menu");  
  var menuPopup = document.createXULElement("menupopup");
  var contextMenu = document.getElementById("contentAreaContextMenu");  
  
    menu.id = "TranslatePage";
    menu.setAttribute("label", "Перевести страницу");
    menu.setAttribute("class", "menu-iconic");
    menu.setAttribute("image", mainicon);
   
  contextMenu.insertBefore(menu, document.getElementById("context-viewsource") ); 
  menu.appendChild( menuPopup );
  addDestructor(function() { contextMenu.removeChild( menu ) }); 

    var array = [
        {label:"Google", func: ujs_googlePage_translate, image:gticon},
        
        
        ];
        
   array.forEach(function( m ) {  
       if ( "separator" in m ) { menuPopup.appendChild( document.createXULElement("menuseparator") ); return };
       var mItem = document.createXULElement("menuitem");
       mItem.setAttribute("label", m.label);
       mItem.setAttribute("class", "menuitem-iconic");
       mItem.setAttribute("image", m.image);
       mItem.addEventListener("command", m.func, false);
       menuPopup.appendChild( mItem );

       });
   
     addEventListener("popupshowing", function() {
     menu.hidden = gContextMenu.isTextSelected || gContextMenu.onImage || gContextMenu.onTextInput ; 
  }, true, contextMenu );
})();



 //Контекстное меню для перевода текста-------------------------------------------  
(function () {
 if ( document.getElementById("TranslateSelected") ) return; 
  
  var menu = document.createXULElement("menu");  
  var menuPopup = document.createXULElement("menupopup");
  var contextMenu = document.getElementById("contentAreaContextMenu");  
      
    menu.id = "TranslateSelected";
    menu.setAttribute("label", "Перевести выделенный текст");
    menu.setAttribute("class", "menu-iconic");
    menu.setAttribute("image", mainicon);

  contextMenu.insertBefore(menu, document.getElementById("context-viewpartialsource-selection") ); 
  menu.appendChild( menuPopup );
  addDestructor(function() { contextMenu.removeChild( menu ) });
     
    
  var array = [
        {label: "В окне Google на русский", func: () => ujs_google_translate("ru"), image: gticon},
        {label: "В окне Google на английский", func: () => ujs_google_translate("en"), image:gticon},
       
        { separator: ''},
        {label:"Заменить текст переводом Google", func: ujs_google_TexReplace, image:gticon},
        
        
        
              ];
  array.forEach(function( m ) {  
        if ( "separator" in m ) { menuPopup.appendChild( document.createXULElement("menuseparator") ); return };
       var mItem = document.createXULElement("menuitem");
       mItem.setAttribute("label", m.label);
       mItem.setAttribute("class", "menuitem-iconic");
       mItem.setAttribute("image", m.image);
       mItem.addEventListener("command", m.func, false);
       menuPopup.appendChild( mItem );
       });
 
     addEventListener("popupshowing", function() {
     menu.hidden = !gContextMenu.isTextSelected; 
  }, false, contextMenu ); 
 })();


});

Отсутствует

 

№10414-10-2023 10:00:13

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Dumby, правильно ли я добавил сепаратор?

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

Выделить код

Код:

/** ------------------------- Пункт контекстного меню для перевода страниц ------------------------- **/
(function () {
	if (document.getElementById("TranslatePage")) return;
	var contextMenu = document.getElementById("contentAreaContextMenu");
	var Item = document.createXULElement("menuitem");
	var sep = document.createXULElement("menuseparator");

	Item.setAttribute("id", "TranslatePage");
	Item.setAttribute("label", "Translate Page");
	Item.setAttribute("class", "menuitem-iconic");
	Item.setAttribute("image", rusFlag);
	Item.addEventListener("command", function() {ujs_googlePage_translate()}, false);

	var cvs = document.getElementById("context-viewsource");

	contextMenu.insertBefore(Item, cvs);
	contextMenu.insertBefore(sep, cvs);
	contextMenu.addEventListener("popupshowing", function() {
		Item.hidden = sep.hidden = cvs.hidden;
	}, false);
})();

Отсутствует

 

№10515-10-2023 08:24:20

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

Re: userChrome.js

6e73epo
Правильно. Можно так: contextMenu.insertBefore(Item, cvs).after(sep);


Да и как вообще можно добавить сепаратор неправильно?
Если он не добавился, или добавился не туда, куда ожидалось,
то это сразу будет видно что неправильно.

Отсутствует

 

№10615-10-2023 09:10:41

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Dumby, да просто без sep.hidden сепаратор двоился при первом вызове контекстного меню, в котором пункты скрыты
На скрине, который давал, меню выше затемнено, если буфер обмена пуст и наоборот. Ничего умнее, чем применение readFromClipboard()  - не нашел

Отредактировано 6e73epo (15-10-2023 09:16:46)

Отсутствует

 

№10718-10-2023 10:12:26

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Если кому-то как и мне требуется переводить выделенный текст или текст из буфера на английский только с русского, то можно автоматом определять исходный язык. Вызываю функцию с параметром auto|det, а в коде функции добавляю

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

Выделить код

Код:

if (l[1] === "det") {
	l[1] = "ru";
	var fCodeRus = String.fromCodePoint(1040);
	var lCodeRus = String.fromCodePoint(1103);
	for (var j = 0; j < txt.length && j < 20; j++) {
		if (txt[j] >= fCodeRus && txt[j] <= lCodeRus) {
			l[1] = "en";
			break;
		}
	}
};


Не вижу смысла учитывать ёЁ. Также пробовал сначала задействовать встроенный языковый детектор браузера, но он косячит. Если строку "я хожу в школу" определяет как русскую, то например, строку "мама" определяет как английскую

Отредактировано 6e73epo (18-10-2023 10:18:12)

Отсутствует

 

№10819-10-2023 08:22:19

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

Re: userChrome.js

Подскажите что подправить в google_translate.js для Aris-t2.
В контекстном меню страницы строки переводчика немного сместились вправо от иконки по сравнению с другими пунктами меню
Как бы их выровнять с другими пунктами меню. ?
989297a1f69fe2d0c5ade51a6a444ed9.png

Отсутствует

 

№10920-10-2023 13:34:13

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

Re: userChrome.js

Выровнял с другими пунктами меню этот код:

Выделить код

Код:

menupopup .menu-iconic-left {
    -moz-appearance: none !important;
    appearance: none !important;
    margin-inline-start: 0 !important;
    margin-inline-end: 3px !important;
    visibility: visible !important;
  
}

скрытый текст
1c30d38370a4374bad44f525eedd95a8.png 

Отсутствует

 

№11020-10-2023 18:36:50

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Dumby пишет

Ааа! Речь не о каких не «пустых строках», а о размере <textarea>.
Тут бы хорошо вообще весь разресайз переписать

Вроде бы ничего сложного нет. Можно по детски. Начальный размер окна (320х160) оставляем без изменений, т.к. не очень приятно лицезреть, например, двухстрочную высоту. При переполнении плавно, с сохранением пропорций, в цикле увеличиваем ширину на 40 px, а высоту на 20 px до пропадания скролла, затем в цикле уменьшаем высоту до появления скролла, затем обратно увеличиваем высоту на 20 px и готово.
С учетом того, что скролл может остаться из-за нарушения форматирования текста, то можно увеличить высоту не на 20, а на 40 px, а затем уменьшить на 20px.

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

Выделить код

Код:

if(size) {
	cnt.style.height = size.height;
	cnt.style.width = size.width;
}
else {
	for (var i = 9; i < 33; i++) {
		if (cnt.scrollTopMax) {
			cnt.style.width = 40*i+'px';
			cnt.style.height = 20*i+'px';
		}
		else break;
	}
	if (i > 9) {
		var k = 0;
		while (!cnt.scrollTopMax) {
			cnt.style.height = parseInt(cnt.style.height)-20+'px';
			k++;
		}
		if (k) {
			cnt.style.height = parseInt(cnt.style.height)+40+'px';
		}
	}
};

// ну и ближе к концу, например после строки w.style.top ...

if (k) {
	cnt.style.height = parseInt(cnt.style.height)-20+'px';
}

Отредактировано 6e73epo (20-10-2023 18:40:40)

Отсутствует

 

№11121-10-2023 19:51:54

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

Re: userChrome.js

6e73epo пишет

Код:

Попробовал я это чуть потестировать.
Фейк, конечно, в окошко просто проброшен выделенный текст,
таким, какой он есть, ну просто чтобы было на что смотреть.


И, вобщем-то, весьма неплохо,
в большинстве случаев, подгонка выглядит хорошо.


Впрочем, попалось и такое
Здесь ширина чуть великовата, как мне кажется
(это на странице about:license, где много всякого текста).

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

Выделить код

Код:




И, даже сподобился словить не ожидавшийся скролл.
Это надеюсь показать в следующем посту,
а то в один пост два таких скиншота не поместятся.

Отсутствует

 

№11221-10-2023 20:25:04

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

Re: userChrome.js

Dumby пишет

надеюсь показать в следующем посту

Ага, размечтался, — «Unable to merge post.»

Добавлено 21-10-2023 20:29:58
Во, с третьей попытки получилось

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

Выделить код

Код:



Отредактировано Dumby (21-10-2023 20:29:58)

Отсутствует

 

№11322-10-2023 00:52:04

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Dumby пишет

Фейк, конечно, в окошко просто проброшен выделенный текст,
таким, какой он есть, ну просто чтобы было на что смотреть.

Не понял, имеется ввиду без перевода?


Dumby пишет

ширина чуть великовата

Но пропорции сохранились. Лицезреть на окно, где высота иногда может быть больше ширины, как то не очень приятно


Dumby пишет

И, даже сподобился словить не ожидавшийся скролл

Повторил, текст выглядит один в один как на скрине, вот только скролла не было и последняя строка видна


У меня еще идея появилась как высоту узнать, ведь есть же _moz_dirty. Вроде работает

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

Выделить код

Код:

if (size) {
	cnt.style.height = size.height;
	cnt.style.width = size.width;
}
else {
	for (var i = 9; i < 33; i++) {
		if (cnt.scrollTopMax) {
			cnt.style.width = 40*i+'px';
			cnt.style.height = 20*i+'px';
		}
		else break;
	}
	if (i > 9) {
		cnt.style.height = cnt.editor.rootElement.childNodes[1].offsetTop-4+'px';
	}
};

Отсутствует

 

№11423-10-2023 09:59:02

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

Re: userChrome.js

6e73epo пишет

Не понял, имеется ввиду без перевода?

Да-да, выделенный текст, без перевода.

У меня еще идея появилась как высоту узнать, ведь есть же _moz_dirty. Вроде работает

А, это <br> который идёт после текстовой ноды.
Довольно остроумно. И работает.


Но, вычитать четыре пикселя сразу было понятно,
что у меня не прокатит, скролл стопроцентно возникает.
Убрал этот вычет, но что-то не помогло.
А вот когда, наоборот, прибавил четыре пикселя — стало нормально.
Правда это у меня font так ещё и остался 24px от прошлых экспериментов.


И, вот ещё какой момент, кейс когда текст большой.
У меня написано так, что если, после всех увеличений в цикле, скролл так и не пропал,
то высота не подвергается изменению, остаётся как получилось.
Иначе говоря, если менять, то только в меньшую сторону, для убирания «пустых строк».


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


Вобщем, даже не знаю, как здесь лучше, как должно быть.
Просто отмечу, что небольшое отличие в таком случае может присутствовать.

Отсутствует

 

№11524-10-2023 11:35:39

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Dumby, благодарю, уже сам заметил, что весь цикл может отработать, а скролл остаться, поэтому добавил условие i > 9 && !cnt.scrollTopMax и теперь все ок.


Этот _moz_dirty идеально работает при размере шрифта 17, который как раз меня устраивает, а для большинства размеров offsetTop с плавающим значением. Конечно, можно под определенный размер все настроить, но мне не требуется.


Теперь единственное, что меня не устраивает, это когда при еще не наступившем событии "mouseup" для navigator.lastClick, окошко перевода появляется в правом нижнем углу окна документа браузера, а хочется, чтобы ближе к центру.

Отсутствует

 

№11624-10-2023 18:45:46

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

Re: userChrome.js

6e73epo пишет

Теперь единственное, что меня не устраивает, это когда при еще не наступившем событии "mouseup" для navigator.lastClick, окошко перевода появляется в правом нижнем углу окна документа браузера, а хочется, чтобы ближе к центру.

Функция createWindow() в коде всегда вызывается с аргументом pos
как navigator.lastClick (и, кстати, никогда с аргументом size),
значит pos это гарантированно объект, и это проверять не требуется.


Поэтому, набросал такой вариант. Надеюсь ничего не испортил.

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

Выделить код

Код:

/*
    w.style.left = (pos && pos.X < mX+hW ? (pos.X > hW ? pos.X-hW : 0) : mX)+'px';
    w.style.top = (pos && pos.Y+10 < mY ? pos.Y+10 : mY)+'px';
*/
    var {X, Y} = pos, U = X == undefined;
    w.style.left = (U ? mX/2 : X < mX + hW ? X > hW ? X - hW : 0 : mX) + "px";
    w.style.top  = (U ? mY/2 : Y + 10 < mY ? Y + 10 : mY) + "px";

Отсутствует

 

№11725-10-2023 01:42:44

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

Re: userChrome.js

6e73epo пишет

работает при размере шрифта 17…… Конечно, можно под определенный размер все настроить…

API браузера - почти операционная система, неужели нет способа подгонки окна под произвольный текст (про HTML с графикой уж молчу) ???
Печально, что баги google-translate.js так долго (c 04-10-2023) победить не получается!


Dumby пишет

Поэтому, набросал такой вариант. Надеюсь ничего не испортил.

А если у кого-нибудь на планшете с нестандартным DPI или просто на Линуксе/MacOS размер окна не совпадёт?
Придётся логарифмы ещё добавлять или систему уравнений ! :)

Отсутствует

 

№11826-10-2023 19:29:23

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 118.0

Re: userChrome.js

Dumby пишет

Надеюсь ничего не испортил

Да вроде все ок. Спасибо.


Dobrov пишет

API браузера - почти операционная система, неужели нет способа подгонки окна под произвольный текст (про HTML с графикой уж молчу) ???

Когда текст еще не вписан? Вряд ли это возможно или будет крайне сложно. Даже при уже вписанном тексте при разных способах подгонки, получаются разные результаты, т.е. при одних и тех же размерах окна (например 400х200) в одном случае будет скролл, в другом - нет, а в третьем тоже нет, но с пустой строкой.


Так что в итоге, на случай если буду использовать размер шрифта отличный от 17, переделал все под способ Dumby с небольшой модификацией

Отредактировано 6e73epo (26-10-2023 19:36:03)

Отсутствует

 

№11931-10-2023 14:45:25

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 119.0

Re: userChrome.js

Dumby, поисковик яндекса с результатами поиска. При наведении мыши на ссылки мы видим реальные адреса, но стоит только вызвать контекстное меню, как адрес меняется на редирект, либо добавляется хвост.
Что бы вы посоветовали (без реализации) для избавления от такого изменения, с учетом, что реализация будет выполняться из консоли страницы?

Отсутствует

 

№12031-10-2023 15:17:48

Farby
Участник
 
Группа: Members
Зарегистрирован: 21-11-2012
Сообщений: 306
UA: Google 2.1

Re: userChrome.js

6e73epo

Remove_Fake_Links, но это user.js скрипт


Жизнь иногда такое выкидывает, что хочется подобрать...

Отсутствует

 

№12101-11-2023 05:30:00

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

Re: userChrome.js

Dumby
Помогите, пожалуйста, сделать скрипт cookiesPermissions.js для загрузчика метода Aris-t2.
В userChrome.js прописано так:

Выделить код

Код:

userChrome.import("scripts", "UChrm");

Отсутствует

 

№12201-11-2023 13:56:16

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 119.0

Re: userChrome.js

Farby, как и ожидалось, Remove_Fake_Links не работает, если включены скрипты статистики яндекса

Отсутствует

 

№12301-11-2023 23:11:59

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

Re: userChrome.js

6e73epo пишет

Что бы вы посоветовали

К сожалению, ничего.
У меня странный FF78 с запрещёнными скриптами, куками, и всё такое прочее.
Там даже сам «поисковик яндекса с результатами поиска» просто не возможен.


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


rubel пишет

Помогите, пожалуйста, сделать скрипт cookiesPermissions.js для загрузчика метода Aris-t2.

Что значит «сделать скрипт cookiesPermissions.js»?
Скрипт вполне себе (как-то) сделан, но под CB,
а значит расчитан на (уже) существующую кнопку.


Если нужен скрипт, который такую кнопку создаст,
то клади в scripts рядом с ним что-то типа такого,
а у самого cookiesPermissions.js смени расширение на .txt
чтобы был не для исполнения загрузчиком, а для чтения скриптом.

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

Выделить код

Код:

window.__SSi == "window0" && CustomizableUI.createWidget({
	id: "CookiesPermissions",
	label: "Cookies Permissions",
	localized: false,
	onCreated(btn) {
		var u = Services.io.newURI;
		var code = Cu.readUTF8URI(u(
			u(Components.stack.filename).resolve("cookiesPermissions.txt")
		)).replace(
			'Components.utils.import("resource://gre/modules/Services.jsm", {})',
			"Cu.getGlobalForObject(Cu)"
		);
		var del = function() {
			this.previousSibling.remove();
			this.remove();
		};
		(this.onCreated = btn => {
			btn.defaultContextId = "toolbar-context-menu";
			var win = btn.ownerGlobal;
			var wdp = new win.DOMParser();
			var parser = class {
				parseFromString(...args) {
					var doc = wdp.parseFromSafeString(...args);
					doc.documentElement.lastChild.appendChild = del;
					return doc;
				}
			}
			win.setTimeout(() => new win.Function("DOMParser", code).call(btn, parser), 50);
		})(btn);
	}
});

Отсутствует

 

№12402-11-2023 05:42:56

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

Re: userChrome.js

Dumby

Если нужен скрипт, который такую кнопку создаст...

ОК. Спасибо. Прекрасно работает. :)

Отсутствует

 

№12505-11-2023 13:42:44

6e73epo
Участник
 
Группа: Members
Зарегистрирован: 06-05-2022
Сообщений: 207
UA: Firefox 119.0

Re: userChrome.js

Dumby, можете попробовать во втором приближении перерисовать так, чтобы очистка памяти по левому клику мыши работала без наличия кнопки memoryMinimizationButton, без notifications, статус инфо и прочего мусора, требующегося для сборщиков мусора и без обсервера? Все это у меня отключено в параметрах. Достаточно так, но как на клик прописать?

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

Выделить код

Код:

function doMemMinimize() {
	var gMgr = Cc["@mozilla.org/memory-reporter-manager;1"].getService(Ci.nsIMemoryReporterManager);
	gMgr.minimizeMemoryUsage(() => {});
}


Вопрос решен. Одной строкой прописал и заработало

Отредактировано 6e73epo (05-11-2023 21:11:51)

Отсутствует

 

Board footer

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