«Merge Day»custom_buttons-0.0.7.0.0.16-fx-paxmod.xpicustom_buttons-0.0.7.0.0.16-fx-bootstrap.xpi
Отредактировано Dumby (27-07-2020 18:07:44)
А как его поставить? Ругается, что расширение не было проверено и не может быть установлено. Нашел какой-то способ, описанный ранее, но он не работает. Проверка подписей отключена, FF79
Отсутствует
Dumby пишет«Merge Day»custom_buttons-0.0.7.0.0.16-fx-paxmod.xpicustom_buttons-0.0.7.0.0.16-fx-bootstrap.xpi А как его поставить? Ругается, что расширение не было проверено и не может быть установлено. Нашел какой-то способ, описанный ранее, но он не работает. Проверка подписей отключена, FF79
Значит что то не так сделали, проверьте ещё раз...: https://forum.mozilla-russia.org/viewtopic.php?id=70326
И попробуйте установить custom_buttons-0.0.7.0.0.16-fx-paxmod, может в этом дело:
Отредактировано kokoss (31-07-2020 16:12:35)
Win7
Отсутствует
хотелось бы так
Вот так, вроде, похоже на картинку
@-moz-document url(chrome://browser/content/downloads/contentAreaDownloadsView.xhtml) { #downloadsListEmptyDescription { -moz-box-flex: 1 !important; margin: 0 !important; padding: 1px 0 0 6px !important; background-color: white !important; } }
Ругается, что расширение не было проверено
Проверка подписей отключена
Определись.
как его поставить?
Если никогда не устанавливал расширения после их запрета,
а теперь вот, от скуки, захотелось, то нет, не советую.
Отсутствует
Baron_ пишетРугается, что расширение не было проверено
Проверка подписей отключенаОпределись.
Baron_ пишеткак его поставить?
Если никогда не устанавливал расширения после их запрета,
а теперь вот, от скуки, захотелось, то нет, не советую.
Чего определяться, когда-то делал. В about:config проверка подписей отключена, и два файлика в папке pref. Из-за того, что большинство кнопок перестали работать, удалил расширение. Было давно, с тех пор что-то поменялось, устанавливать не хочет.
kokoss, спасибо, все работает. А я в этой ветке искал решение, так и не раскопал.
Отсутствует
Добрый день. По моему , осталась одна рабочая ветка на форуме . Почините кто сможет пожалуйста три кода , два кода часов "цифра"и "аналог", оба варианта часов "цифра" и "аналог" отстают на один час от системных на FF71, как на 32 так и 64, хотя на FF38.0 ESR , FF60.0 ESR ходят нормально,синхронно с системными. Не пойму откуда и как эти коды синхронизируется с системными часами . Крутил "синхрон" по интернету и без ,менял часовой пояс , включал и отключал службы времени ,нет "синхрона" с системными часами. Третий код это переключатель "Proxy", мало того
не переключает выбранный режим прокси на кнопке , сами режимы "без прокси", "Ручная настройка прокси", "URL автоматической настройки прокси" переключаются ,но на кнопке это не видно. Она "кнопка" после перетаскивания на панель рвёт интерфейс панелей в хлам. Хочу завершить переход на FF71 с FF38 .Заранее спасибо.
Код "Аналоговые часы"
/*Initialization Code*/ // Аналоговые часы, от 08.02.2016. ............... (()=> { var dia = 30; // диаметр аналоговых часов var canvas = document.createElementNS(xhtmlns, 'canvas'); canvas.setAttribute("width", dia+"px;"); canvas.setAttribute("height", dia+"px;"); //canvas.style.cssText = "position: fixed !important; top: 30px; right: 218px; min-width: diapx; min-height: diapx; max-width: diapx; max-height: diapx".replace(/dia/g, dia); canvas.style.cssText = "min-width: diapx; min-height: diapx; max-width: diapx; max-height: diapx".replace(/dia/g, dia); var ctx = canvas.getContext("2d"); ctx.scale(dia/122, dia/122); self.parentNode.insertBefore(canvas, self); addDestructor(()=> canvas.remove() ); self.hidden = true; canvas.onclick =()=> self.hidden = !self.hidden; var interval = setInterval(()=> { var ctx = canvas.getContext("2d"); ctx.save(); ctx.clearRect(0, 0, 150, 150); ctx.translate(61, 61); ctx.scale (0.4, 0.4); ctx.fillStyle = "white"; ctx.arc(0, 0, 142, 0, Math. PI * 2, true); ctx.fill(); ctx.rotate(-Math. PI / 2); ctx.strokeStyle = "black"; ctx.fillStyle = "white"; ctx.lineWidth = 12; ctx.lineCap = "round"; ctx.save(); ctx.beginPath(); for ( var i = 0; i < 12; i++ ) { ctx.rotate(Math. PI / 6); ctx.moveTo(100, 0); ctx.lineTo(120, 0); } ctx.stroke(); ctx.restore(); ctx.save(); ctx.lineWidth = 5; ctx.beginPath(); for ( var i = 0; i < 60; i++ ) { if ( i % 5 != 0 ) { ctx. moveTo (117, 0); ctx. lineTo (120, 0); } ctx.rotate(Math. PI / 30); } ctx.stroke(); ctx.restore(); var now = new Date(); var sec = now.getSeconds(); var min = now.getMinutes(); var hr = now.getHours(); self.tooltipText = [hr, min > 9? min: "0" + min, sec > 9? sec: "0" + sec]. join (" : "); hr = hr >= 12? hr - 12: hr; ctx.fillStyle = "black"; ctx.save(); ctx.strokeStyle = "black"; ctx.rotate(hr * (Math. PI / 6) + (Math. PI / 360) * min + (Math. PI / 21600) * sec) ctx.lineWidth = 14; ctx.beginPath(); ctx.moveTo(-20, 0); ctx.lineTo(80, 0); ctx.stroke(); ctx.restore(); ctx.save(); ctx.rotate((Math. PI / 30) * min + (Math. PI / 1800) * sec) ctx.lineWidth = 10; ctx.beginPath(); ctx.moveTo(-28, 0); ctx.lineTo(112, 0); ctx.stroke(); ctx.restore(); ctx.save(); ctx.rotate(sec * Math. PI / 30); ctx.strokeStyle = "#ee0000"; ctx.fillStyle = "#ee0000"; ctx.lineWidth = 6; ctx.beginPath(); ctx.moveTo(-30, 0); ctx.lineTo(93, 0); ctx.stroke(); ctx.fillStyle = "#555"; ctx.arc(0, 0, 3, 0, Math. PI * 2, true); ctx.fill(); ctx.restore(); ctx.beginPath(); ctx.lineWidth = 14; ctx.strokeStyle = '#1581e6'; ctx.arc(0, 0, 142, 0, Math. PI * 2, true); ctx.stroke(); ctx.restore(); }, 1000); addDestructor(()=> clearInterval(interval) ); })(); /*CODE*/ this.leftclick(event);
Код "цифра"
/*Initialization Code*/ this.setAttribute("style", " -moz-appearance: none; font-size: 11pt !important; color: #000000 !important; background-color: !important; border-width: 0px !important; border-color: #4444aa !important;-moz-border-radius: 4px; margin-left: 1px !important; margin-top: 1px !important; margin-bottom: 1px !important; padding-bottom: 0px !important; padding-left: 1px !important; padding-right: 1px !important; padding-top: 0px !important; "); this.style.opacity = "0.99"; this.onmouseover = function(event) { var ttTime = new Date(); var p = (ttTime. getHours () < 12)? "AM": "PM"; var time = ttTime. toLocaleFormat ("%I:%M:%S ") + p; var date = ttTime. toLocaleFormat ("%A, %B %d, %Y"); this.tooltipText = date ; /* var mydatestr = ttTime.toDateString(); var myday = mydatestr.substring(0,3); var mymonth = mydatestr.substring(4,7); var mydate = mydatestr.substring(7,11); var myyear = mydatestr.substring(10,mydatestr.length); this.tooltipText = myday + mydate + mymonth + myyear; */ } // Comment out these next two lines to start as a twelve hour clock var TwelveHourClock = 0; var timesign = ":"; var timetick = ";" var timetick1 = ":"; var timetick2 = "."; this.leftclick = function(event) { if(TwelveHourClock == 0) { TwelveHourClock = 12; } else { TwelveHourClock = 0; } } var beginTime = new Date(); var now = new Date(); var offset = now.getTimezoneOffset(); var myGMToffset = (offset/60)*-1; function count(t) { var millisecond = t.getTime(); var hour = Math.floor(millisecond % 86400000 / 3600000) + myGMToffset; if(TwelveHourClock == 12) { if(hour >= 13) { hour = hour - TwelveHourClock ; } else { morneve = "AM"; } if(hour >= 12) {morneve = "PM"; }; } else { if(hour >= 24) {hour = hour - 24} } var minute = Math.floor(millisecond % 3600000 / 60000); var second = Math.floor(millisecond % 3600000 % 60000 / 1000); hour = (hour < 10 ? "0" : "") + hour; minute = (minute < 10 ? "0" : "") + minute; second = (second < 10 ? "0" : "") + second; if(TwelveHourClock == 0){timesign = timetick1}else{timesign = timetick2} return hour + timesign + minute + timesign + second; } setInterval(function(that) { var currentTime = new Date(); that.label = count(currentTime); }, 10, this); var ios = Components.classes["@mozilla.org/network/io-service;1"]. getService(Components.interfaces.nsIIOService); var sss = Components.classes["@mozilla.org/content/style-sheet-service;1"]. getService(Components.interfaces.nsIStyleSheetService); var css = new String(); css += "#" + this.id + " .toolbarbutton-icon { display: none !important; }\n"; css += "#" + this.id + " .toolbarbutton-text { display: -moz-box !important; }\n"; css += "#" + this.id + " { -moz-box-orient: horizontal !important; }"; var uss = ios.newURI("data:text/css," + encodeURIComponent(css), null, null); // comment out the next line to disable style if (!sss.sheetRegistered(uss, sss.USER_SHEET)) sss.loadAndRegisterSheet(uss, sss.USER_SHEET); this.onDestroy = function(reason) { if (reason == "update") { var uss = ios.newURI("data:text/css," + encodeURIComponent(css), null, null); if (sss.sheetRegistered(uss, sss.USER_SHEET)) sss.unregisterSheet(uss, sss.USER_SHEET); } if (reason == "delete") { var uss = ios.newURI("data:text/css," + encodeURIComponent(css), null, null); if (sss.sheetRegistered(uss, sss.USER_SHEET)) sss.unregisterSheet(uss, sss.USER_SHEET); } } this. onclick = function (event) { custombuttons. gQuot. mHandler (event); } this. ondblclick = function (event) { custombuttons. gQuot. mHandler (event); } this.setAttribute('author','squeaky,Barbiegirl,morat'); this.setAttribute('version','20110408.2.5'); this.setAttribute('homepage', 'http://custombuttons.mozdev.org/drupal/content/button-clock'); /*CODE*/ this.leftclick(event);
Код "Proxy"
// Proxy, от 25.02.2016. // Настройка функций кликов мыши для кнопки ................... this.onmousedown =e=> { this.onmouseup =e=>{ // левый клик if ( e.button ) return; clearTimeout(self.timer); switch( cbu.getPrefs("network.proxy.type") ) { case 0: var data = 1; break; case 1: var data = 2; break; case 2: var data = 0; break; default: var data = 0; } cbu.setPrefs("network.proxy.type", data); } if ( e.button == 0 ) // длинный левый клик self.timer = setTimeout(()=>{ self.onmouseup = ''; cbu.getPrefs("CB.Proxy.connectionsInTab") ? openConnectionsInTab() : openConnections(); }, 500); if ( e.button == 2 ) // правый клик menuPopup.showPopup(self, -1, -1, "popup", "bottomleft", "topleft"); }; self.onclick =e=> e.preventDefault(); // Подсказка для кнопки ................... this.onmouseover =()=> { this.tooltipText = "Proxy \nЛ: Переключить прокси \nДЛ: Открыть настройки прокси" + "\nП: Mеню кнопки \n\nТекущие настройки прокси: " + "\nIP: " + Services.prefs.getComplexValue("network.proxy.http", Ci.nsISupportsString).data + "\nПорт: "+ cbu.getPrefs("network.proxy.http_port"); }; // Создать меню для кнопки ................... var array = [ { label: "Добавление прокси в контекстом меню", value: 'CB.Proxy.inContextMenu' }, { label: "Открывать настройки прокси как вкладку", value: 'CB.Proxy.connectionsInTab' }, { label: "Переключать на режим 'Без прокси' при закрытии браузера ", value: 'CB.Proxy.reset' } ]; var menuPopup = document.getElementById('mainPopupSet').appendChild(document.createElement("menupopup")); array.forEach((m)=> { var mItem = document.createElement("menuitem"); mItem.setAttribute("label", m.label); mItem.setAttribute('type', 'checkbox'); mItem.setAttribute('checked', cbu.getPrefs(m.value) ); mItem.onclick =()=> cbu.setPrefs(m.value, !cbu.getPrefs(m.value)); menuPopup.appendChild(mItem); }); addDestructor(()=> menuPopup.remove() ); // добавить стандартное контекстное меню 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() { for ( var win, nm = Services.wm.getEnumerator(null); win = nm.getNext(); ) if ( win.name == 'Proxy') { win.focus(); break; } 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) {}; }, true ); }; // Функция открывает настройки прокси в вкладке ................... 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"); switch( cbu.getPrefs(s) ) { case 0: icon.src = self.image; break; case 1: icon.src = ''; break; case 2: icon.src = ''; break; default:icon.src = self.image; } }; toggleImage(); gPrefService.addObserver(s, toggleImage, false); addDestructor(()=> gPrefService.removeObserver(s, toggleImage) ); // Переключать на режим 'Без прокси' при закрытии браузера если это разрешено в 'about:config' ................... var switchOffProxy = { observe: function(subject, topic, data) { if ( data == "shutdown" && cbu.getPrefs("CB.Proxy.reset") ) cbu.setPrefs("network.proxy.type", 0); } }; Services.obs.addObserver(switchOffProxy, "quit-application", false); // Создаем меню для добавление прокси в контекстном меню выделенного текста на странице ................... ((contextMenu)=> { // создать новый пункт меню var menuitem = contextMenu.appendChild( document.createElement("menuitem") ); menuitem.setAttribute("label", "Добавить прокси"); menuitem.setAttribute("class", "menuitem-iconic"); menuitem.setAttribute("image", self.image); menuitem.onclick =()=> addNewProxy(); addDestructor(()=> menuitem.remove() ); // устанавливаем где показывать пункт меню addEventListener("popupshowing", ()=>{ menuitem.hidden = !cbu.getPrefs("CB.Proxy.inContextMenu") || !gContextMenu.isContentSelected; // !gContextMenu.isTextSelected; }, false, contextMenu); // добавление прокси function addNewProxy(sel) { 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 < 6 && isFinite(sel) ) { sel = sel.replace(/:/g, ""); var lab = 'порт'; cbu.setPrefs("network.proxy.http_port", +sel); } // если только адрес ... if ( sel.length > 5 && !/:/.test(sel) && sel.split(".").length == 4 ) { var lab = 'адрес'; cbu.setPrefs("network.proxy.http", convertFromUnicode("UTF-8", sel)); } // если адрес и порт ... if ( sel.length > 5 && /:/.test(sel) && sel.split(":").length == 2 && sel.split(".").length == 4 ) { var lab = 'адрес и порт'; var array = sel.split(":"); array.forEach((str)=> addNewProxy(str) ); } if ( lab == undefined ) 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(()=> gBrowser.removeChild(tooltip), 3000[firefox]); }; showTooltip(); }; })(document.getElementById("contentAreaContextMenu"));
Отсутствует
отстают на один час от системных
Может это RFP так выкрутасничает.
В любом случае, можно подвести на час вперёд
цифра
Код "Proxy"
Не души тёмнозелёным, возьми лучше какой-нибудь такой.
Иконки, надеюсь, сам сможешь заменить.
Отсутствует
Dumby. Спасибо большое за отзывчивость и реальную помощь. Всё заработало.
Еще одна хотелочка и можно отказаться от расширений и полностью переходить на FF71
Три кода: Первые два не работают на FF71 а третий код "Двойным левым кликом на папке закладок добавлять закладку в папку закладок" работает но долго 15 секунд висит "транспарант" что закладка добавлена.
Первый "CB drag and go"
// CB drag and go, от 09.01.2017.перетаскивая выделенного текста левой клавишей на странице. .................. ({ link: { U: { name: "Открыть ссылку в новой активной странице", cmd: ()=> { gBrowser.selectedTab = gBrowser.addTab(self.link); }}, //D: { //name: "Открыть ссылку в новой активной странице", cmd: ()=> { //gBrowser.selectedTab = gBrowser.addTab(self.link); //}}, D: { name: "Открыть ссылку в новой фоновой странице", cmd: ()=> { gBrowser.addTab(self.link); }}, }, text: { U: { name: "Поиск текста поисковиком по умолчанию в новой активной странице", cmd: ()=> { var submission = Services.search.currentEngine.getSubmission(self.text, null); gBrowser.loadOneTab(submission.uri.spec, null, null, submission.postData, false, false); }}, D: { name: "Поиск текста поисковиком по умолчанию в новой фоновой странице", cmd: ()=> { var submission = Services.search.currentEngine.getSubmission(self.text, null); gBrowser.loadOneTab(submission.uri.spec, null, null, submission.postData, false, false); }}, }, init: function() { self.this = this; // создать подсказку у кнопки var arr = []; ["link", "text"].forEach(ob=> { arr.push("\n" + ob); for (var key in this[ob]) arr.push(key + ':' + [" ", " ", " "][key.length-1] + this[ob][key].name); }); self.tooltipText = self.label + "\n" + arr.join("\n"); ["dragstart", "dragover", "drop"].forEach(type=> addEventListener(type, this, false, gBrowser)); }, convertFromUnicode: function(charset, str) { var converter = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter); converter.charset = charset; str = converter.ConvertFromUnicode(str); return str + converter.Finish(); }, handleEvent: function(e) { if ( !content.location.protocol.startsWith("http") ) return; // если не http(s) страница switch (e.type) { case "dragstart": { self.lastDirection = ""; self.currentPoint = [e.screenX, e.screenY]; // получить перетаскиваемый элемент, текст или адрес ссылки или изображения self.el = e.target; self.text = e.dataTransfer.getData("text/unicode"); self.img = e.dataTransfer.getData("application/x-moz-file-promise-url").split("\n")[0]; var textLink = /^([a-z]+:\/\/)?([a-z]([a-z0-9\-]*\.)+([a-z]{2}|aero|arpa|biz|com|coop|edu|gov|info|int|jobs|mil|museum|name|nato|net|org|pro|travel)|(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}[0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])(:[0-9]{1,5})?(\/[a-z0-9_\-\.~]+)*(\/([a-z0-9_\-\.]*)(\?[a-z0-9+_\-\.%=&]*)?)?(#[a-z][a-z0-9_]*)?$|^custombutton:\/\/\S+$/.test(self.text); self.link = textLink ? self.text : e.dataTransfer.getData("text/x-moz-url").split("\n")[0]; // получить и обявить тип перетаскиваемого элемента self.type = (self.img || self.link) ? "link" : "text"; break; } case "dragover": { Cc["@mozilla.org/widget/dragservice;1"].getService(Ci.nsIDragService).getCurrentSession().canDrop = true; // получить направление перетаскивания, L налево, R направо, U верх, D вниз var [subX, subY] = [e.screenX - self.currentPoint[0], e.screenY - self.currentPoint[1]]; var [distX, distY] = [(subX > 0 ? subX : (-subX)), (subY > 0 ? subY : (-subY))]; if ( distX < 10 && distY < 10 ) return; var direction = distX > distY ? (subX < 0 ? "L" : "R") : (subY < 0 ? "U" : "D"); self.currentPoint = [e.screenX, e.screenY]; // получить весь жест, указать код жеста и показать подсказку жеста if ( direction != self.lastDirection.split('').pop() ) { self.lastDirection += direction; var dir = self.lastDirection, ob = this[self.type][dir]; this.cmd = ob ? ob.cmd : ""; XULBrowserWindow.statusTextField.label = ob ? "Жест мыши: " + dir + " " + ob.name : "Неизвестный жест мыши: " + dir; } break; } case "drop": { // если перетаскивается в поле текстового ввода или из-за пределов окна браузера if ( !self.currentPoint || e.target instanceof Ci.nsIDOMNSEditableElement ) return; e.preventDefault(); // запустить код жеста и сбросить подсказку жеста this.cmd && this.cmd(); XULBrowserWindow.statusTextField.label = ""; } } } }).init();
Второй Autocopy+3
// Autocopy+3 от 26.12.2013. // Настройка функций кликов мыши для кнопки ................................ this.onclick = function(e) { if ( e.button == 0 ) cbu.setPrefs("Autocopy", !cbu.getPrefs("Autocopy") ); if ( e.button == 2 ) { e.preventDefault(); menuPopup.showPopup( this, -1, -1, "popup", "bottomleft", "topleft"); } }; // Проверить наличие строк в 'about:config' и создать если не существует ................................ if ( !cbu.isPref("Autocopy.saveWithDoubleClick") ) { ["Autocopy", "Autocopy.reset", "Autocopy.selectingTextBlink", "Autocopy.selectWithDoubleClick", "Autocopy.copyWithDoubleClick", "Autocopy.saveWithDoubleClick"].forEach(function(pref) { cbu.setPrefs( pref, false ) }); }; // Создать меню ................................ var array = [ { label: 'Выключать автокопирование при выходе из браузера', value: 'Autocopy.reset' }, { label: 'Выделенный текст мигает при автокопировании', value: 'Autocopy.selectingTextBlink' }, { separator: ''}, { label: 'Двойной правый клик мыши копирует выделенный текст', value: 'Autocopy.copyWithDoubleClick' }, { label: 'Двойной левый клик мыши выделяет все в текстовых полях', value: 'Autocopy.selectWithDoubleClick' }, { label: 'Двойной правый клик мыши сохраняет изображение без запроса', value: 'Autocopy.saveWithDoubleClick' }, { label: 'Средним кликом вставлятъ текст с заменой выделенного текста', value: 'middlemouse.paste' } ]; var menuPopup = self.appendChild( document.createElement("menupopup") ); array.forEach(function( m ) { if ( "separator" in m ) { menuPopup.appendChild( document.createElement("menuseparator") ); return }; var mItem = document.createElement("menuitem"); mItem.setAttribute("label", m.label); mItem.setAttribute('type', 'checkbox'); mItem.setAttribute('checked', custombuttons.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 + "')"); // Установить нужную иконку кнопки при старте браузера или при изменениях настроек в 'about:config' ................................ const s = "Autocopy"; function toggleImage() { self.image = cbu.getPrefs(s) ? "" : ""; }; toggleImage(); gPrefService.addObserver( s, toggleImage, false ); addDestructor(function() { gPrefService.removeObserver( s, toggleImage, false ) }); // Выключать кнопку при закрытии браузера если это разрешено в 'about:config' ................................ var turnOffButton = { observe: function(subject, topic, data) { if ( cbu.getPrefs("Autocopy.reset") && data == "shutdown" ) cbu.setPrefs("Autocopy", false ); } }; Services.obs.addObserver( turnOffButton, "quit-application", false); addDestructor(function() { Services.obs.addObserver( turnOffButton, "quit-application", false ) }); // Функции автоматически копирует выделенный текст на странице, если это разрешено в 'about:config' ................................ function autocopy(e) { if ( e.button == 2 ) return; if ( /input|password|textarea|textbox|searchbar|findbar|tabbrowser/.test( e.target.localName.toLowerCase() ) ) return; if ( e.type == 'mousedown' ) var lastSelection = getBrowserSelection(); if ( e.type !== 'mouseup' ) return; var selection = getBrowserSelection(); if ( cbu.getPrefs("Autocopy") && selection && selection !== lastSelection ) { goDoCommand('cmd_copy'); // выделенный текст мигает .... if ( !cbu.getPrefs("Autocopy.selectingTextBlink") ) return; document.activeElement.blur(); setTimeout(function() { window.content.focus() }, 300); } }; addEventListener("mouseup", autocopy, false, gBrowser ); addEventListener("mousedown", autocopy, false, gBrowser ); // Cредней кнопкой мыши вставить текст из буфера обмена в текстовые поля с заменой выделенного текста ................................ function middleMousePaste(e) { if ( e.button == 1 && cbu.getPrefs('middlemouse.paste') ) { if ( /input|password|textarea|textbox|searchbar|findbar|cbeditor/.test( e.target.localName.toLowerCase() ) && document.commandDispatcher.getControllerForCommand("cmd_paste") ) { e.preventDefault(); e.stopPropagation(); // вставить текст .... var cmd = "cmd_insertText"; var commandDispatcher = ( this.document || document ).commandDispatcher; var controller = commandDispatcher.getControllerForCommand(cmd); if ( controller && controller.isCommandEnabled(cmd) ) { controller = controller.QueryInterface(Components.interfaces.nsICommandController); var params = Components.classes["@mozilla.org/embedcomp/command-params;1"] .createInstance(Components.interfaces.nsICommandParams); params.setStringValue("state_data", gClipboard.read() ); controller.doCommandWithParams(cmd, params); } } } }; addEventListener("click", middleMousePaste, true, document.documentElement ); // Дополнительные возможности для значка идентификации сайта в строке адреса ................................ addEventListener("click", function(e) { e.preventDefault(); e.stopPropagation(); // ЛКМ без запроса открывает информацию о странице в вкладке 'Разрешения' .... if ( e.button == 0 ) { var doc = content.document; BrowserPageInfo( doc, ( (doc.location.protocol).slice(0,4) == "http") ? "permTab" : "generalTab" ); } // ПКМ копирует текущий адрес .... if ( e.button == 2) { gClipboard.write( content.location ); // значок идентификации сайта мигает красным .... document.getElementById("identity-box").setAttribute("style", "background: red;"); setTimeout(function() { document.getElementById("identity-box").removeAttribute("style") }, 500); } }, true, document.getElementById("identity-box") ); // Дополнительные возможности для двойного клика мыши, если это разрешено в 'about:config' ................................ function handleDblClick(e) { var node = e.target; var editor = node.editor; // выделить все в текстовых полях .... if ( e.button == 0 && custombuttons.getPrefs("Autocopy.selectWithDoubleClick") ) { e.preventDefault(); if ( /input|textbox|textarea/.test( node.localName ) ) !editor ? node.select() : editor.selectAll(); } // сохранить изображение без запроса .... if ( e.button == 2 && cbu.getPrefs("Autocopy.saveWithDoubleClick") && node.localName == 'img' ) { saveImageURL( gContextMenu.imageURL, 0, 0, 0, 1, null, content.document ); setTimeout(function() { document.getElementById("contentAreaContextMenu").hidePopup() }, 20); } // скопировать выделенный текст .... if ( e.button == 2 && cbu.getPrefs("Autocopy.copyWithDoubleClick") && !/findbar|tabbrowser/.test( node.localName ) ) { e.preventDefault(); !editor ? goDoCommand("cmd_copy") : editor.copy(); try { var box = ( node.textbox || node ).inputField.parentNode; var popup = box.ownerDocument.getAnonymousElementByAttribute( box, "anonid", "input-box-contextmenu"); setTimeout(function() popup.hidePopup(), 50); } catch(e) { document.getElementById("contentAreaContextMenu").hidePopup() }; } }; addEventListener("dblclick", handleDblClick, false, gBrowser ); // Наблюдатель следит за открытием окон адреса которых указанны в коде и добавляет им обработчики ................................ var observer = { observe: function(subject, topic, data) { subject.addEventListener("load", this, false); }, handleEvent: function(e) { var doc = e.target; var win = doc.defaultView; var href = doc.location.href; win.removeEventListener("load", this, false); // закрывать 'Информацию о странице' или 'Библиотеку' двойным кликом на ней .... if ( /pageInfo.xul|places.xul/.test( href ) ) { win.addEventListener("dblclick", function close() { this.close() }, true); win.addEventListener("unload", function(e) { win.removeEventListener(e.type, arguments.callee, false); win.removeEventListener("dblclick", close, true); }, false); }; // добавлять и удалять обработчики клика для редактора Custom Buttons .... if ( href.substring(0, 41) == "chrome://custombuttons/content/editor.xul" ) { win.addEventListener("click", middleMousePaste, true ); win.addEventListener("dblclick", handleDblClick, false ); win.addEventListener("unload", function(e) { win.removeEventListener(e.type, arguments.callee, false ); win.removeEventListener("click", middleMousePaste, true ); win.removeEventListener("dblclick", handleDblClick, false ); }, false); }; } }; Services.obs.addObserver(observer, "domwindowopened", false); addDestructor(function() { Services.obs.removeObserver(observer, "domwindowopened", false) }); // Подсказка для кнопки ................................ this.tooltipText = "Autocopy \nЛ: Переключить автоматическое копирование \nП: Меню + CB меню";
Третий "Двойным левым кликом на папке закладок добавлять закладку в папку закладок"
// Двойным левым кликом на папке закладок добавлять закладку в папку закладок, от 06.06.2019. ...................... addEventListener("dblclick", (e, targ = e.originalTarget)=> { if ( e.button || !targ._placesNode || !PlacesUtils.nodeIsFolder(targ._placesNode) ) return; var docTitle = gBrowser.selectedTab.label.substr(0, 50); var folderId = PlacesUtils.getConcreteItemId(targ._placesNode); var folderTitle = PlacesUtils.bookmarks.getItemTitle(folderId); var currentURI = Services.io.newURI(gBrowser.currentURI.spec, null, null); PlacesUtils.bookmarks.insertBookmark(folderId, currentURI, 0, docTitle); // всплывающая подсказка .... var favicon = gBrowser.selectedTab.image || "chrome://global/skin/icons/Portrait.png"; Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService) .showAlertNotification(favicon, "Добавил в папку " + folderTitle + ":", docTitle); setTimeout(()=> { document.getElementById("bookmarksMenuPopup").hidePopup() }, 50); });
Отредактировано Duche (02-08-2020 16:27:27)
Отсутствует
две кнопки
Вместо ваших двух кнопок достаточно одной в расширении Async Run Applications https://github.com/VitaliyVstyle/Vitali … xperiments
Отсутствует
egorsemenov06
Это расширение подписать нельзя, только отключать проверку подписи или использовать Firefox Developer Edition.
Актуальный код спросите подскажут для этого.
Отсутствует
egorsemenov06
Ну для user_chrome_files наподобие этого https://forum.mozilla-russia.org/viewto … 54#p782454
можно сделать только с кнопкой, или можете использовать contextmenuopenwith. Ну а с интерфейсом для добавления приложений как в расширении увольте.
Отредактировано Vitaliy V. (02-08-2020 20:45:40)
Отсутствует
Duche
Третий "Двойным левым кликом на папке закладок добавлять закладку в папку закладок"
можно поприкалываться и сделать уведомления вверху
// Двойным левым кликом на папке закладок добавлять закладку в папку закладок, от 06.06.2019. ............................................................. addEventListener("dblclick", (e, targ = e.originalTarget)=> { if ( e.button || !targ._placesNode || !PlacesUtils.nodeIsFolder(targ._placesNode) ) return; var docTitle = gBrowser.selectedTab.label.substr(3, 50); var folderId = PlacesUtils.getConcreteItemId(targ._placesNode); var folderTitle = PlacesUtils.bookmarks.getItemTitle(folderId); var currentURI = Services.io.newURI(gBrowser.currentURI.spec, null, null); PlacesUtils.bookmarks.insertBookmark(folderId, currentURI, 0, docTitle); // всплывающая подсказка .... Components.utils.import('resource://gre/modules/PopupNotifications.jsm'); var notify = new PopupNotifications(gBrowser, document.getElementById("notification-popup"), document.getElementById("notification-popup-box")); var notification = notify.show( // browser gBrowser.selectedBrowser, // popup id "PDES-popup", // message "Добавил в папку " + folderTitle + ":", // anchor id null, // main action null, // secondary action null, // options { // Alternative way to set the popup icon popupIconURL: gBrowser.selectedTab.image || "chrome://global/skin/icons/Portrait.png" } ); setTimeout(function(){ notification.remove(); }, 2000); // Time in milliseconds to disappear the door-hanger popup. });
Насчет Autocopy+3, где-то здесь Dumby выкладывал
/*Initialization Code*/ this.closest("toolbarpaletteitem") || (script => { var id = `CB${_id.slice(20)}:Autocopy`, pid = id + "Parent"; var nsvoStr = `Components.utils.import("resource://gre/modules/Services.jsm", {})`; var nsvo = eval(nsvoStr), {Services} = nsvo, parent = nsvo[pid]; if (!parent) { var cid = id + "Child", u = code => "data:," + encodeURIComponent(code); var pref = "CB.Autocopy.settings", topic = "quit-application-granted"; var PREF_ENABLED = 1, PREF_BLINK = 2, PREF_RESET = 4; (parent = nsvo[pid] = { init() { this.readSettings(); if (!this[PREF_ENABLED]) return; this.initChild(); if (this[PREF_RESET]) this.setObserver(true); }, destroy(reason) { var ud = reason[5] == "e"; if (ud || !this.obsAdded) this.saveSettings(); delete nsvo[pid]; if (reason == "delete") Services.prefs.clearUserPref(pref); if (!this[PREF_ENABLED]) return; this.destroyChild(); if (ud && this[PREF_RESET]) this.setObserver(false); }, get processURL() { delete this.processURL; this.frameURL = u(`${nsvoStr}["${cid}"].init(this);`); return this.processURL = u(script.replace(/%ID%/g, cid) .replace("%NSVO%", nsvoStr) .replace("%PREF%", pref) .replace("%PREF_BLINK%", PREF_BLINK) ); }, get frameURLDestroy() { delete this.frameURLDestroy; this.processURLDestroy = u(`${nsvoStr}["${cid}"].forget();`); return this.frameURLDestroy = u(`${nsvoStr}["${cid}"].destroy(this);`); }, initChild() { Services.ppmm.loadProcessScript(this.processURL, true); Services.mm.loadFrameScript(this.frameURL, true); }, destroyChild() { Services.mm.removeDelayedFrameScript(this.frameURL); Services.mm.loadFrameScript(this.frameURLDestroy, false); Services.ppmm.removeDelayedProcessScript(this.processURL); Services.ppmm.loadProcessScript(this.processURLDestroy, false); }, readSettings() { var val = Services.prefs.getIntPref(pref, 3); for(var setting of [PREF_ENABLED, PREF_BLINK, PREF_RESET]) this[setting] = Boolean(val & setting); }, saveSettings() { var settings = 0; for(var setting of [PREF_ENABLED, PREF_BLINK, PREF_RESET]) if (this[setting]) settings += setting; Services.prefs.setIntPref(pref, settings); }, btns: new Set(), register(btn) { this.btns.add(btn); btn._handleClick = this.click; btn.oncontextmenu = this.context; this.setImg(btn, this[PREF_ENABLED]); }, unregister(btn, reason) { this.btns.delete(btn); if (!this.btns.size) this.destroy(reason); }, setImg(btn, state) { (btn.icon || btn.ownerDocument.getAnonymousElementByAttribute( btn, "class", "toolbarbutton-icon" )).src = state ? "" : ""; }, click() { var newState = parent[PREF_ENABLED] = !parent[PREF_ENABLED]; for(var btn of parent.btns) parent.setImg(btn, newState); newState ? parent.initChild() : parent.destroyChild(); if (parent[PREF_RESET]) parent.setObserver(newState); }, context(e) { if (e.ctrlKey || e.shiftKey) return; var btn = e.target; if (btn.btnPopup && e.detail > 1) return btn.btnPopup.hidePopup(); e.preventDefault(); (btn.btnPopup || parent.getPopup(btn)).openPopup(btn, "after_start"); }, getPopup(btn) { var win = btn.ownerGlobal, doc = win.document; var popup = doc.createElementNS(xulns, "menupopup"); popup.setAttribute("onclick", "event.stopPropagation();"); popup.setAttribute("oncommand", "handleCommand(event.target);"); popup.setAttribute("onpopupshowing", "handlePopupshowing();"); popup.menuitems = []; for(var [lab, pref] of win.Object.entries({ "Выделенный текст мигает при автокопировании": PREF_BLINK, "Выключать автокопирование при выходе из браузера": PREF_RESET })) { var menuitem = popup.appendChild(doc.createElementNS(xulns, "menuitem")); menuitem.setAttribute("label", lab); menuitem.setAttribute("type", "checkbox"); menuitem.pref = pref; popup.menuitems.push(menuitem); } popup.handleCommand = menuitem => { var newState = this[menuitem.pref] = menuitem.hasAttribute("checked"); if (!this[PREF_ENABLED]) return; if (menuitem.pref == PREF_BLINK) this.saveSettings(), Services.ppmm.broadcastAsyncMessage(cid + ":FromParent", {blink: newState}); else if (menuitem.pref == PREF_RESET) this.setObserver(newState); } popup.handlePopupshowing = () => { for(var menuitem of popup.menuitems) this[menuitem.pref] ? menuitem.setAttribute("checked", true) : menuitem.removeAttribute("checked"); } return btn.appendChild(btn.btnPopup = popup); }, obsAdded: false, setObserver(set) {this.obsAdded = set ? Services.obs.addObserver(this, topic, false) : Services.obs.removeObserver(this, topic); }, observe() { Services.obs.removeObserver(this, topic); this[PREF_ENABLED] = false; this.saveSettings(); } }).init(); } parent.register(this); addDestructor(reason => parent.unregister(this, reason), parent); })(`(nsvo => (nsvo["%ID%"] = { x: -1, y: -1, down: false, handleEvent(e) {e.button || this[e.type](e);}, mousedown(e) {this.x = e.screenX; this.y = e.screenY, this.down = true;}, mouseup(e) { var {down} = this; this.down = false; if (!down) return; if (e.screenX == this.x && e.screenY == this.y && (e.detail == 1 || e.target.matches( "textarea[disabled],input[disabled],button,select,summary" ))) return; var name = e.originalTarget.nodeName; if (/^(?:(?:xul:)?(?:slider|scrollbarbutton)|resizer)$/.test(name)) return; this.x = this.y = -1; var win = this.getFocusedWin(e.target.ownerGlobal); var sel = win.getSelection(); if (sel.toString()) { (win.docShell || win.document.docShell).doCommand("cmd_copy", null, win); this.blinkEnabled && this.blink(win, e.detail > 1); } }, blinkEnabled: Boolean( Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService) .getIntPref("%PREF%", 3) & %PREF_BLINK% ), blink(win, pause) { if (pause) return win.setTimeout(() => this.blink(win), 100); var sc = (win.docShell || win.document.docShell) .QueryInterface(Components.interfaces.nsIInterfaceRequestor) .getInterface(Components.interfaces.nsISelectionDisplay) .QueryInterface(Components.interfaces.nsISelectionController); sc.setDisplaySelection(sc.SELECTION_OFF); sc.repaintSelection(sc.SELECTION_NORMAL); win.setTimeout(() => { sc.setDisplaySelection(sc.SELECTION_ON); sc.repaintSelection(sc.SELECTION_NORMAL); }, 150); }, getFocusedWin(win) { var focusedWin = {}; var elm = this.fm.getFocusedElementForWindow(win.top, true, focusedWin); return focusedWin.value; }, get fm() { delete this.fm; return this.fm = Components.classes["@mozilla.org/focus-manager;1"] .getService(Components.interfaces.nsIFocusManager); }, count: 0, init(cfmm) { this.count += 1; cfmm.addEventListener("mousedown", this); cfmm.addEventListener("mouseup", this); cfmm.addEventListener("unload", this); if (this.count == 1) this.cpmm.addMessageListener("%ID%:FromParent", this); }, destroy(cfmm) { this.count -= 1; cfmm.removeEventListener("mousedown", this); cfmm.removeEventListener("mouseup", this); cfmm.removeEventListener("unload", this); if (!this.count) this.cpmm.removeMessageListener("%ID%:FromParent", this); }, receiveMessage(msg) { if ("blink" in msg.data) this.blinkEnabled = msg.data.blink; }, unload(e) {this.destroy(e.target);}, forget: () => delete nsvo["%ID%"] }).cpmm = this)(%NSVO%);`);
а насчет всплывающих подсказок - вверху -там больше возможностей https://developer.mozilla.org/en-US/doc … ifications
Отредактировано Andrey_Krropotkin (03-08-2020 10:34:20)
Отсутствует
Andrey_Krropotkin
Большое спасибо за "Autocopy+3" и "Двойным левым". Меня волновали не столько место всплывающих подсказок (их расположение), сколько время которое оно висит перед главами ,в старом коде 15-20 секунд это перебор. А так и в верху хорошо и в низу хорошо. В Вашем коде интереснее, на мой взгляд.
Ещё бы починить "CB drag and go" и можно окончательно переходить на FF71.
Отсутствует
Обратите пожалуйста внимание на пост
Да видел. Хорошо, попробую. Один код на обе.
Во второй, передвиг пунктов заменил с колеса на перетаскивание.
try {((ext, data) => data.forEach(([id, impl]) => { impl.id = "ucf-cbbtn-" + id; CustomizableUI.createWidget(Object.assign(impl, ext)) }))({ localized: false, onCreated(btn) { btn.owner = this; btn.type = "menu"; btn.setAttribute("image", this.image); btn.openPopup = btn.openMenu; btn.openMenu = this.openMenu; var popup = btn.appendChild(btn.ownerDocument.createXULElement("menupopup")); popup.setAttribute("context", ""); popup.setAttribute("oncommand", "owner.command(event)"); this.init(btn, popup); var {openDelay, closeDelay} = this; this.autoOpenCloseFeature(btn.ownerGlobal, btn, openDelay, closeDelay); }, file: Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile), openMenu(...args) { if (this.parentNode != this.domParent) { this.domParent = this.parentNode; this.owner.setPopupPosition(this); } this.openPopup(...args); }, setPopupPosition(node) { if (node.matches(".widget-overflow-list > :scope")) var pos = "after_start"; else var win = node.ownerGlobal, {width, height, top, bottom, left, right} = node.closest("toolbar").getBoundingClientRect(), pos = width > height ? `${win.innerHeight - bottom > top ? "after" : "before"}_start` : `${win.innerWidth - right > left ? "end" : "start"}_before`; node.firstChild.setAttribute("position", pos); }, // https://github.com/Infocatcher/Custom_Buttons/blob/master/code_snippets/autoOpenCloseMenu.js // Automatically open menu on mouse over (and hide it on mouse out) autoOpenCloseFeature(win, btn, openDelay = 200, closeDelay = 350) { var _openTimer = 0; var _closeTimer = 0; btn.onmouseover = function(e) { win.clearTimeout(_closeTimer); if(e.target == btn && closeOtherMenus()) { btn.open = true; return; } _openTimer = win.setTimeout(function() { btn.open = true; }, openDelay); }; btn.onmouseout = function(e) { win.clearTimeout(_openTimer); _closeTimer = win.setTimeout(function() { if(!isContextOpened()) btn.open = false; }, closeDelay); }; function closeOtherMenus() { return win.Array.prototype.some.call( btn.parentNode.getElementsByTagName("*"), function(node) { if( node != btn && win.XULElement.isInstance(node) // See https://github.com/Infocatcher/Custom_Buttons/issues/28 //&& node.boxObject //&& node.boxObject instanceof Components.interfaces.nsIMenuBoxObject && "open" in node && node.open && node.getElementsByTagName("menupopup").length ) { node.open = false; return true; } return false; } ); } function isContextOpened() { return inBtn(win.document.popupNode); } function inBtn(node) { for(; node; node = node.parentNode) if(node == btn) return true; return false; } } }, Object.entries({ OpenPageInOtherBrowser: { label: "Открыть страницу в другом браузере", image: "", tooltiptext: [ "Л: Открыть меню с браузерами", "С: Добавить в меню новый браузер", "\nФункции кликов мыши для меню:", "\tЛ: Открыть страницу", "\tС: Добавить разделитель", "\tП: Удалить пункт меню или разделитель", "\tCtrl+П: Изменить название пункта меню", "Перетаскиванием можно передвигать пункты меню или разделители" ].join("\n"), init(btn, popup) { btn.onauxclick = this.auxclick; popup.setAttribute("onpopupshowing", "this.shouldRebuild && owner.rebuild(this)"); popup.ondragstart = this.dragstart; popup.shouldRebuild = true; }, get markup() { try {var data = Cu.readUTF8URI(Services.io.newURI( `chrome://user_chrome_files/content/custom_scripts/${this.id}-data.txt` )).split("\n").filter(line => /\S/.test(line));} catch {var data = [ "D:\\Програмное обеспечение\\Флешка загрузочная\\Windows 7 SP1\\Программы\\ChromePortable\\ChromePortable.exe>Chrome", "D:\\Програмное обеспечение\\Флешка загрузочная\\Windows 7 SP1\\Программы\\Opera_1217\\Opera\\launcher.exe>Opera", "C:\\Program Files (x86)\\Maxthon5\\Bin\\Maxthon.exe", "C:\\Program Files (x86)\\Internet Explorer\\iexplore.exe", "D:\\Програмное обеспечение\\Флешка загрузочная\\Windows 7 SP1\\Программы\\Microsoft Edge\\ProgramFiles\\msedge.exe", ];} delete this.markup; return this.markup = this.dataToMarkup(data); }, setMarkup(popup) { this.markup = popup.innerHTML; for(var {node} of CustomizableUI.getWidget(this.id).instances) if (node.firstChild != popup) node.firstChild.shouldRebuild = true; this.write(Array.from(popup.children, node => node.hasAttribute("value") ? node.tooltipText + (node.value == "true" ? ">" + node.label : "") : "separator" ).join("\n")); }, dataToMarkup(data) { var markup = ""; for(var str of data) markup += str == "separator" ? "<menuseparator/>" : this.strToMenuitem(str); return markup; }, repl: [/^./, c => c.toUpperCase()], strToMenuitem(str, ind = str.lastIndexOf(">")) { var name, val, path = str; if ((val = ind != -1)) path = str.slice(0, ind), name = str.slice(ind + 1); else this.file.initWithPath(path), name = this.file.leafName.split(".") .shift().replace(...this.repl); return `<menuitem label="${name}" tooltiptext="${path}" value="${val}"` + ` class="menuitem-iconic" image="moz-icon://file://${path}"/>`; }, append(popup, xul = this.markup) { popup.append(popup.ownerGlobal.MozXULElement.parseXULToFragment(xul)); }, rebuild(popup) { popup.textContent = ""; this.append(popup); delete popup.shouldRebuild; }, auxclick(e) { var trg = e.target, popup = this.firstChild; if (trg == this && e.button == 1) return this.owner.addMenuitem(popup); else if (trg.parentNode != popup) return; if (e.button == 1) { var up = e.screenY < trg.screenY + trg.clientHeight/2; up = up ? trg.previousSibling : !trg.nextSibling; trg[up ? "before" : "after"]( trg.ownerDocument.createXULElement("menuseparator") ); } else { if (e.ctrlKey) { if (trg.nodeName.endsWith("r")) return; var name = this.owner.prompt( "Введите другое название пункта", trg.label, trg.ownerGlobal ); if (name && name != trg.label) trg.label = name, trg.value = true; } else trg.remove(); } this.owner.changeMarkup(popup); }, prompt(msg, value, domWin) { var res = {value}; return Services.prompt.wrappedJSObject.pickPrompter({ domWin, modalType: Ci.nsIPrompt.MODAL_TYPE_WINDOW }).nsIPrompt_prompt(this.label, msg, res, null, {}) ? res.value : null; }, addMenuitem(popup) { var fp = Cc["@mozilla.org/filepicker;1"].createInstance(Ci.nsIFilePicker); fp.appendFilters(fp.filterApps); fp.init(popup.ownerGlobal, "Укажите путь к программе", fp.modeOpen); fp.open(res => { if (res == fp.returnOK) this.append(popup, this.strToMenuitem(fp.file.path, -1)), this.setMarkup(popup); }); }, changeMarkup(popup) { popup.state == "open" ? popup.addEventListener("popuphidden", this, {once: true}) : this.setMarkup(popup); }, handleEvent(e) { this[e.type](e); }, popuphidden(e) { this.setMarkup(e.target); }, dragstart(e) { var trg = e.target; if (trg.parentNode.nodeName != "menupopup") return; var {owner} = this.parentNode; var pn = trg.flattenedTreeParentNode; owner.dragData = {trg, pn, ns: trg.nextSibling}; trg.style.cssText = "font-weight: bold; color: red;" + "outline: 2px solid red; outline-offset: -2px;" .replace(/;/g, " !important;"); var win = trg.ownerGlobal; win.setCursor("grabbing"); pn.addEventListener("mousemove", owner); win.addEventListener("mouseup", owner, {once: true}); }, mousemove(e) { var trg = e.target, dtrg = this.dragData.trg; if (trg == dtrg) return; e.movementY > 0 ? trg.nextSibling != dtrg && trg.after(dtrg) : trg.previousSibling != dtrg && trg.before(dtrg); }, mouseup(e) { e.preventDefault(); var {trg, pn, ns} = this.dragData; delete this.dragData; trg.removeAttribute("style"); trg.ownerGlobal.setCursor("auto"); pn.removeEventListener("mousemove", this); trg.nextSibling != ns && this.changeMarkup(trg.parentNode); }, command(e) { this.file.initWithPath(e.target.tooltipText); if (this.file.exists()) { var process = Cc["@mozilla.org/process/util;1"].createInstance(Ci.nsIProcess); process.init(this.file); return process.run(false, [e.view.gBrowser.currentURI.spec], 1); } Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService) .showAlertNotification(this.image, this.label, "Файл не существует"); }, write(txt) { var file = Services.dirsvc.get("UChrm", Ci.nsIFile), CC = Components.Constructor; ["user_chrome_files", "custom_scripts", this.id + "-data.txt"].forEach(file.append); var te = new (Cu.getGlobalForObject(Cu).TextEncoder)(); var fos = CC("@mozilla.org/network/file-output-stream;1", "nsIFileOutputStream", "init") // MODE_{WRONLY, CREATE, TRUNCATE}, PERMS_FILE .bind(null, file, 0x02 | 0x08 | 0x20, 0o644, 0); var bos = CC("@mozilla.org/binaryoutputstream;1", "nsIBinaryOutputStream", "setOutputStream"); (this.write = txt => { var stream = new fos(); try {new bos(stream).writeByteArray(te.encode(txt));} catch(ex) {Cu.reportError(ex);} finally {stream.close();} })(txt); } }, openExternalWinApplication: { label: "Открыть внешнее win приложение", image: "", data: [ ["IE", "C:\\Program files\\Internet Explorer\\iexplore.exe"], ["Explorer", "C:\\windows\\explorer.exe"], null, ["Cmd Prompt", "C:\\windows\\system32\\cmd.exe"], ["Media Player", "C:\\Program Files\\Windows Media Player\\wmplayer.exe"], ["Task Manager", "C:\\windows\\system32\\taskmgr.exe"], ["Control Panel", "C:\\windows\\system32\\control.exe"], null, ["Notepad", "C:\\windows\\notepad.exe"], ["Calculator", "C:\\windows\\system32\\calc.exe"], ["Virtual Keyboard", "C:\\windows\\system32\\OSK.exe"], ["Character table", "C:\\windows\\system32\\charmap.exe"] ], init(btn) { btn.tooltipText = btn.label; btn.ownerGlobal.Object.defineProperty(btn, "domParent", this); }, configurable: true, get() { delete this.domParent; var doc = this.ownerDocument, df = doc.createDocumentFragment(); for(var arr of this.owner.data) { if (!arr) { df.append(doc.createXULElement("menuseparator")); continue; } var menuitem = df.appendChild(doc.createXULElement("menuitem")); menuitem.className = "menuitem-iconic"; menuitem.setAttribute("label", arr[0]); menuitem.setAttribute("image", "moz-icon://file://" + arr[1]); } this.firstChild.append(df); }, command(e) { this.file.initWithPath(e.target.getAttribute("image").slice(18)); this.file.launch(); } } }))} catch(ex) {Cu.reportError(ex);}
можно Вас попросить пожалуйста адаптировать для user_chrome_files вот эту кнопку
Ещё не легче, совсем поперёк ситуации.
try {CustomizableUI.createWidget(({ label: "XXXXX XXXXXXX", image: "data:image/png;charset=utf-8;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAewQAAHsEBw2lUUwAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAABYSURBVDiNY2CgEjjKwMDwHw0fwaIOJgcHjEgS2AAjGv8/ujgLkS5EtwBuEBORBuAExLoA3atwL8BccBSLJmyBOIgBoXSAU57YdIBTnuJoHDwGEEoHtEsnALmSHj9YVKEQAAAAAElFTkSuQmCC", defaultFavicon: "", id: "ucf-cbbtn-LnkCreator", localized: false, onCreated(btn) { btn.owner = this; btn.tooltipText = this.label; btn.setAttribute("image", this.image); btn.setAttribute("oncommand", "owner.createLnk(this)"); }, init() { this.widget.parent = this; this.widget.contextmenu.destroy = id => { CustomizableUI.destroyWidget(id); delete this.data[id.slice(8)]; this.save(); } try {this.data = JSON.parse(Cu.readUTF8URI(Services.io.newURI( `chrome://user_chrome_files/content/custom_scripts/${this.id}-data.json` )))} catch {this.data = {}; return this;} for(var [id, inf] of Object.entries(this.data)) this.createWidget(id, inf.url, inf.name); return this; }, get fs() { delete this.fs; return this.fs = Cc["@mozilla.org/browser/favicon-service;1"] .getService(Ci.nsIFaviconService); }, createLnk(btn) { var id = Date.now(); var gb = btn.ownerGlobal.gBrowser; var uri = gb.currentURI; var label = gb.contentTitle.slice(0, 75); var widget = this.createWidget(id, uri, label); var {area, position} = CustomizableUI.getPlacementOfWidget(this.id); CustomizableUI.addWidgetToArea(widget.id, area, position + 1); this.data[id] = {name: label, url: uri.spec}; this.save(); }, createWidget(id, url, label) { var obj = { uri: url.spec ? url : Services.io.newURI(url), id: "ucf-lnk-" + id, label, ...this.widget }; var widget = obj.widget = CustomizableUI.createWidget(obj); this.fs.getFaviconDataForPage(obj.uri, obj); return widget; }, tip: "\nShift+ПКМ - Удалить кнопку", widget: { localized: false, onCreated(btn) { btn.uri = this.uri; btn.addTab = this.addTab; btn.oncontextmenu = this.contextmenu; btn.setAttribute("oncommand", "addTab(this)"); btn.tooltipText = this.label + this.parent.tip; this.image && btn.setAttribute("image", this.image); }, addTab(btn) { var gb = btn.ownerGlobal.gBrowser; gb.selectedTab = gb.addTrustedTab(btn.uri.spec, {userContextId: 1}); }, contextmenu: function checkShift(e) { if (e.shiftKey) e.preventDefault(), checkShift.destroy(e.target.id); }, onComplete(uri, len, arr, mmt) { this.image = len ? `data:${mmt};base64,${btoa(String.fromCharCode(...arr))}` : this.parent.defaultFavicon for(var {node} of this.widget.instances) node.hasAttribute("image") || node.setAttribute("image", this.image); delete this.widget; } }, save() { var file = Services.dirsvc.get("UChrm", Ci.nsIFile), CC = Components.Constructor; ["user_chrome_files", "custom_scripts", this.id + "-data.json"].forEach(file.append); var te = new (Cu.getGlobalForObject(Cu).TextEncoder)(); var fos = CC("@mozilla.org/network/file-output-stream;1", "nsIFileOutputStream", "init") // MODE_{WRONLY, CREATE, TRUNCATE}, PERMS_FILE .bind(null, file, 0x02 | 0x08 | 0x20, 0o644, 0); var bos = CC("@mozilla.org/binaryoutputstream;1", "nsIBinaryOutputStream", "setOutputStream"); (this.write = () => { var stream = new fos(); try {new bos(stream).writeByteArray( te.encode(JSON.stringify(this.data)) );} catch(ex) {Cu.reportError(ex);} finally {stream.close();} })(); } }).init())} catch(ex) {Cu.reportError(ex);}
15-20 секунд это перебор
Можно точно так же закрыть по таймауту.
... // всплывающая подсказка .... var favicon = gBrowser.selectedTab.image || "chrome://global/skin/icons/Portrait.png"; var as = Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService); var name = "bookmarks-alert-" + Date.now(); as.showAlertNotification( favicon, "Добавил в папку " + folderTitle + ":", docTitle, false, null, null, name ); setTimeout(() => as.closeAlert(name), 2e3);
Ещё бы починить "CB drag and go" и можно окончательно переходить на FF71.
Ну, перетаскиваемого добра в процессе кнопок нет, однако,
какой-то dataTransfer проброшен, теоретически прицепиться можно.
Но лучше бы WebExtensions поискал, они должны уметь делать то, что в кнопке.
addEventListener("dragstart", ({ link: { U: { name: "Открыть ссылку в новой активной странице", cmd() { openUILinkIn(this.val, "tab", this.opts); } }, D: { name: "Открыть ссылку в новой фоновой странице", cmd() { openUILinkIn(this.val, "tabshifted", this.opts); } } }, text: { U: { name: "Поиск текста поисковиком по умолчанию в новой активной странице", cmd() { this.search("tab"); } }, D: { name: "Поиск текста поисковиком по умолчанию в новой фоновой странице", cmd() { this.search("tabshifted"); } } }, search(where) { var engine = Services.search[`default${this.opts.private ? "Private" : ""}Engine`]; var submission = engine.getSubmission(this.val, null, ""); openUILinkIn(submission.uri.spec, where, {postData: submission.postData, ...this.opts}); }, opts: { //relatedToCurrent: true, triggeringPrincipal: document.nodePrincipal, get userContextId() { return parseInt(gBrowser.selectedBrowser.getAttribute("usercontextid")); }, get private() { return PrivateBrowsingUtils.isWindowPrivate(window); } }, init() { var arr = []; for(var type of ["link", "text"]) arr.push( type + "\n" + Object.entries(this[type]).map(a => a[0] + ":\t" + a[1].name).join("\n") ); self.tooltipText = (self.label ? self.label + "\n\n" : "") + arr.join("\n\n"); this.drop = () => this.drag(); this.handleEvent = e => this[e.type](e); return this; }, dragstart(e) { //if (!gBrowser.currentURI.spec.startsWith("http")) return; if (!e.dataTransfer.mozItemCount || !gBrowser.selectedBrowser.matches(":hover")) return; var dt = e.dataTransfer; this.type = this.link; this.dir = this.val = ""; var url = dt.getData("text/x-moz-url-data"); if (url) this.val = url; else { var txt = dt.getData("text/plain"); if (txt) { this.val = txt; if (!this.textLinkRe.test(txt)) this.type = this.text; } else return; } this.x = e.screenX; this.y = e.screenY; this.drag(true); }, drag(init) { var meth = `${init ? "add" : "remove"}EventListener`; for(var type of this.events) window[meth](type, this, true); init || StatusPanel.panel.setAttribute("inactive", true); }, events: ["dragover", "drop", "dragend"], dragover(e) { var {x, y} = this, cx = e.screenX, cy = e.screenY; var dx = cx - x, ax = Math.abs(dx), dy = cy - y, ay = Math.abs(dy); if (ax < 10 && ay < 10) return; this.x = cx; this.y = cy; var dir = ax > ay ? dx > 0 ? "R" : "L" : dy > 0 ? "D" : "U"; if (this.dir.endsWith(dir)) return; dir = this.dir += dir; var obj = this.type[dir]; var txt = `${ obj ? "Ж" : "Неизвестный ж" }ест мыши: ${dir + (obj ? " " + obj.name : "")}`; StatusPanel._labelElement.value = txt; StatusPanel.panel.removeAttribute("inactive"); }, dragend(e) { var dt = e.dataTransfer; this.drag(); var obj = this.type[this.dir]; if (!obj || dt.mozUserCancelled) return; var x = e.screenX, y = e.screenY; var wx = mozInnerScreenX, wy = mozInnerScreenY; x > wx && y > wy && x < wx + innerWidth && y < wy + innerHeight && obj.cmd.call(this); }, textLinkRe: /^([a-z]+:\/\/)?([a-z]([a-z0-9\-]*\.)+([a-z]{2}|aero|arpa|biz|com|coop|edu|gov|info|int|jobs|mil|museum|name|nato|net|org|pro|travel)|(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}[0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])(:[0-9]{1,5})?(\/[a-z0-9_\-\.~]+)*(\/([a-z0-9_\-\.]*)(\?[a-z0-9+_\-\.%=&]*)?)?(#[a-z][a-z0-9_]*)?$|^custombutton:\/\/\S+$/ }).init(), true, gBrowser.tabpanels || 1);
Отсутствует
Dumby
Видимо, создать числовую настройку permissions.default.persistent-storage со значением 2
Уй,шикардос! А, "иметь доступ к экрану" ? тоже заблокировать на глушняк.
Отредактировано solombala (04-08-2020 11:10:15)
Отсутствует
Добрый день. Поправьте пожалуйста кнопку "Закрыть все вкладки и перейти на домашнюю." Вкладки закрывает а на домашнюю не переходит.
/*CODE*/ [...gBrowser.tabs].forEach((tab)=> !tab.pinned && gBrowser.removeTab(tab)); /*закрыть все вкладки*/ getBrowser (). selectedTab = getBrowser (). addTab ("https://yandex.ru"); /*Закрытие всех вкладок приводило к открытию домашней страницы*/
Отсутствует
Добрый день. Поправьте пожалуйста кнопку "Закрыть все вкладки и перейти на домашнюю." Вкладки закрывает а на домашнюю не переходит.
/*CODE*/ [...gBrowser.tabs].forEach((tab)=> !tab.pinned && gBrowser.removeTab(tab)); /*закрыть все вкладки*/ var url = "https://yandex.ru"; openUILinkIn(url, 'current', { triggeringPrincipal: Services.scriptSecurityManager.getSystemPrincipal() }); /*Закрытие всех вкладок приводило к открытию домашней страницы*/
«The Truth Is Out There»
Отсутствует
Dumby
Возможно ли реализовать такую штуку.
Заходя на любую страницу любого сайта получить массив ссылок.
И, если какие-то из них ссылаются на youtube.com или на youtu.be, то при клике на такие ссылки происходил редирект на invidio.us.
Например:
https://www.youtube.com/watch?v=MdGDMFuT8vU ---> https://invidio.us/watch?v=MdGDMFuT8vU
https://youtu.be/336Z5BhM7h0 ---> https://invidio.us/336Z5BhM7h0
P.S. Желательно, чтобы код работал как на FF52 ESR, так и на FF78 ESR.
«The Truth Is Out There»
Отсутствует
А, "иметь доступ к экрану" ?
То же самое, только persistent-storage заменить на screen
Возможно ли реализовать такую штуку.
Странно, мне казалось ты повнимательнее, и, в отличие от некоторых,
понимаешь, что меня бесполезно спрашивать про контентские дела.
Может попробуй какую-нибудь обезьяну ({Tamper, Grease, Violent}monkey)
под это дело подрядить. Например, чтобы href по клику подменялся, типа
((re, ret) => { var listener = e => { if (e.button > 1) return; var link = e.target.closest("a[href]"); if (link && re.test(link.href)) { var was = link.getAttribute("href"); link.href = "https://invidio.us/" + RegExp.rightContext; setTimeout(ret, 200, link, was); } } addEventListener("click", listener, true); "onauxclick" in document && addEventListener("auxclick", listener, true); })( /^https:\/\/(?:www\.)?youtu(?:be\.com|\.be)\//, (link, href) => link.setAttribute("href", href) );
Отсутствует
Может попробуй какую-нибудь обезьяну ({Tamper, Grease, Violent}monkey)
под это дело подрядить. Например, чтобы href по клику подменялся, типаскрытый текстВыделить кодКод:
((re, ret) => { var listener = e => { if (e.button > 1) return; var link = e.target.closest("a[href]"); if (link && re.test(link.href)) { var was = link.getAttribute("href"); link.href = "https://invidio.us/" + RegExp.rightContext; setTimeout(ret, 200, link, was); } } addEventListener("click", listener, true); "onauxclick" in document && addEventListener("auxclick", listener, true); })( /^https:\/\/(?:www\.)?youtu(?:be\.com|\.be)\//, (link, href) => link.setAttribute("href", href) );
Собственно, у меня Tampermonkey установлен. Однако скрипт не срабатывает.
«The Truth Is Out There»
Отсутствует
Однако скрипт не срабатывает.
Ну да, я же говорю — бесполезно .
Но, всё таки, немного любопытно почему не срабатывает.
Вот, провёл такой эксперимент на Firefox 78.0
Открыл новую пустую вкладку и скормил адресной строке
data: адрес (простая html'ка с двумя ссылками, которые ты предоставил).
Открыл веб-консоль (Ctrl+Shift+K) и запустил с js-терминала код.
ЛКМ по ссылке на странице — GET-запрос на invidio.us есть,
и в адресной строке, в конце концов, invidio.us тоже появляется.
А Ctrl+ЛКМ или СКМ — новая вкладка, и снова с invidio.us
Так же и на Firefox 52.0.
Попробуй повторить, если интересно. Лучше на чистом, по возможности.
В смысле, если так работает, то дело не в самом коде, а в чём-то другом, наверно.
data:text/html;charset=utf-8,%3C!DOCTYPE%20html%3E%0A%3Chtml%3E%0A%09%3Chead%3E%0A%09%09%3Ctitle%3ETest%20yt%20links%3C%2Ftitle%3E%0A%09%09%3Cmeta%20http-equiv%3D%22Content-Type%22%20content%3D%22text%2Fhtml%3B%20charset%3Dutf-8%22%3E%0A%09%3C%2Fhead%3E%0A%09%3Cbody%3E%0A%09%09%3Ccenter%3E%0A%09%09%09%3Cbr%2F%3E%3Cbr%2F%3E%0A%09%09%09%3Ca%20href%3D%22https%3A%2F%2Fwww.youtube.com%2Fwatch%3Fv%3DMdGDMFuT8vU%22%3Ewww.youtube.com%2Fwatch%3Fv%3DMdGDMFuT8vU%3C%2Fa%3E%0A%09%09%09%3Cbr%2F%3E%3Cbr%2F%3E%0A%09%09%09%3Ca%20href%3D%22https%3A%2F%2Fyoutu.be%2F336Z5BhM7h0%22%3Eyoutu.be%2F336Z5BhM7h0%3C%2Fa%3E%0A%09%09%3C%2Fcenter%3E%0A%09%3C%2Fbody%3E%0A%3C%2Fhtml%3E
Отсутствует