>Форум Mozilla Россия http://forum.mozilla-russia.org/index.php >Сustom Buttons http://forum.mozilla-russia.org/viewforum.php?id=34 >[CB]Context Search http://forum.mozilla-russia.org/viewtopic.php?id=52666 |
bunda1 > 03-12-2011 16:47:15 |
Context Search( Firefox24+ ) Клики на пункте контекстного меню или на пункте подменю: Использование: положите код в любую Custom Buttons кнопку, в инициализацию. Не обязательно создавать новую CB кнопку, можно использовать уже существующую. Выделить код Код:// Context Search, от 19.12.2015. ........................ ((contextMenu, searchSelect, searchService)=> { searchSelect.collapsed = true; // удалить стандартный пункт меню для поиска // Создать новый пункт меню для поиска .... var menu = contextMenu.insertBefore(document.createElement('menu'), searchSelect); menu.setAttribute("class", "menu-iconic"); addEventListener("popupshowing", (e)=> menu.hidden = searchSelect.hidden, false, contextMenu); function setMenu() { menu.engine = searchService.currentEngine; menu.setAttribute("label", "Искать в " + menu.engine.name + " или в ..."); menu.setAttribute("image", menu.engine.iconURI.spec ); }; setMenu(); // Создать подменю с поисковиками .... var menuPopup = menu.appendChild(document.createElement("menupopup")); menuPopup.setAttribute('style', 'overflow: scroll'); function setItemsToMenuPopup(e) { menuPopup.textContent = ""; var engines = searchService.getVisibleEngines({}); engines.forEach((engine)=> { var mItem = document.createElement("menuitem"); mItem.setAttribute("label", engine.name); mItem.setAttribute("class", "menuitem-iconic"); mItem.setAttribute("src", engine.iconURI.spec); mItem.engine = engine; menuPopup.appendChild(mItem); }) }; setItemsToMenuPopup(); // установить действие для клика на меню и подменю menu.onmouseup =e=> { var background = (e.button == 0) ? false : true; var clip = gClipboard.read(); goDoCommand('cmd_copy'); setTimeout(()=> { contextMenu.hidePopup(); var submission = e.target.engine.getSubmission(gClipboard.read(), null); gBrowser.loadOneTab(submission.uri.spec, null, null, submission.postData, background, false); gClipboard.write(clip); }, 0); }; // Наблюдатель за изменениями в поисковиках пересоздаёт меню и подменю .... var getEngineModified = { observe:(subject, topic, data)=> { if ( /changed|removed|current/.test(data) ) { setMenu(), setItemsToMenuPopup() }; } }; Services.obs.addObserver(getEngineModified, "browser-search-engine-modified", false); // Удалять наблюдатели и меню, показать стандартный пункт .... addDestructor(()=> { menu.remove(); searchSelect.collapsed = false; Services.obs.removeObserver(getEngineModified, "browser-search-engine-modified"); }); })(document.getElementById("contentAreaContextMenu"), document.getElementById("context-searchselect"), Services.search); |
iDev.Pi > 18-12-2011 23:02:20 |
Аналогичное Context Search-у дополнение есть и в виде джетпака (его установка не требует перезапуска браузера): Context Search RG. |
vitalii201 > 18-12-2011 23:53:38 |
iDev.Pi пишет
Здесь также |
bunda1 > 19-12-2011 01:44:38 |
vitalii201 пишет
измени в коде на или на |
villa7 > 23-09-2012 00:44:41 |
bunda1 |
bunda1 > 23-09-2012 09:22:04 |
Как без подменю, объясни подробней. |
villa7 > 23-09-2012 10:13:26 |
bunda1 пишет
Выделяю слово, жму правую кнопку, и у меня сразу вместе с пунктами копировать, выделить и т.д., все поисковики, не надо еще открывать подпункт с поисковиками, у меня как-то так с самого начала было, привык, меньше телодвижений. |
bunda1 > 23-09-2012 10:27:34 |
Если ты используешь кнопку CB Mouse Gestures то там есть жест RLRL - [Popup] Поиск текста в выбранном поисковике. Это практически аналог Context Search, попробуй. |
villa7 > 23-09-2012 11:11:42 |
bunda1 пишет
Попробовал, там выходит один поисковик, а у меня их с десяток для разных целей, надо дополнительно переключаться, получается еще более неудобнее чем с подменю, если трудоемко убрать подменю то ладно, буду дальше старым расширением пользоваться, пока не сломается в новых версиях лисы, что теперь поделаешь. |
voqabuhe > 23-09-2012 11:48:03 |
villa7 пишет
Попробуй FireGestures :: Дополнения Firefox, там весь список поисковиков выдаёт. |
bunda1 > 23-09-2012 12:52:59 |
villa7 пишет
У меня этот жест выдаёт весь список поисковиков. |
villa7 > 23-09-2012 16:09:08 |
bunda1 Всем спасибо за участие , добавил эту команду и стало то что нужно, до этого неправильно копировал из списка удаленных команд. А еще такой вопрос,у знакомого ссылки на код кнопки кликабельны, а у меня нет, ставлю кнопки перетаскиванием в адресную строку. Скрипт текстовых ссылок Linkification стоит, если это от него зависит, или как. |
villa7 > 24-09-2012 00:11:10 |
LongLogin У меня раньше стояло, сменил на кнопку Drag and go, и потом там нужно сначала выделить потом дернуть, а Linkification сразу клацнул и все, пока браузер не крашился, тьфу,тфу. Все равно спасибо. |
Kamui > 24-09-2012 07:09:07 |
LongLogin |
villa7 > 24-09-2012 14:28:34 |
Kamui пишет
Да, именно эту кнопку поставил. |
villa7 > 24-09-2012 16:15:11 |
LongLogin LongLogin пишет
Иеется ввиду Quick Search Bar?, я пробовал ставить, но у меня по правой кнопке вообще ни какие поисковики не появляются, может какие-то настройки поменять, а может на 15 лисе не работает. |
Lenya1995 > 29-10-2012 14:40:22 |
bunda1 |
LBra > 06-11-2012 11:27:50 |
villa7 пишет
Без подменю есть расширение Context Search X |
lokiju > 22-12-2012 21:40:07 |
Griever обновил contextSearcher до версии 0.0.7 слегка русифицированный contextSearcher Выделить код Код:// ==UserScript== // @name contextSearcher.uc.js // @namespace http://d.hatena.ne.jp/Griever/ // @description 右クリック→検索の強化 // @include main // @compatibility Firefox 4 // @version 0.0.7 // @note 0.0.7 Изменен чтобы получить слово, даже на ссылку // @note 0.0.7 Оптимизация приобретение слова // @note 0.0.6 カタカナの正規表現のミスを修正 // @note 0.0.6 splitmenu をやめた(menu 部分をクリックして検索可能) // @note 0.0.6 Mac でカーソル下の単語をうまく拾えてなかったらしいのを修正したかも // @note 0.0.5 サブメニューを中クリックすると2回実行される問題を修正 // @note 0.0.5 メニューの検索エンジン名を非表示にした // @note 0.0.5 カーソル下の単語の取得を調整 // @note 0.0.5 "・"をカタカナとして処理していたのを修正 // @note 0.0.4 アイコンの無い検索エンジンがあるとエラーになるのを修正 // ==/UserScript== // http://f.hatena.ne.jp/Griever/20100918161044 // ホイールで既定のエンジン変更、サブメニューから他の検索エンジンの利用 // 右クリックの位置により選択文字列、カーソル下の単語を検索可能 if (window.contextSearcher) { window.contextSearcher.destroy(); delete window.contextSearcher; } window.contextSearcher = { NEW_TAB: true, _regexp: { hiragana: "[\\u3040-\\u309F]+", katakana: "[\\u30A0-\\u30FA\\u30FC]+", kanji : "[\\u4E00-\\u9FA0]+", suuji : "[0-9_./,%-]+", eisu_han: "\\w[\\w\\-]*", eisu_zen: "[\\uFF41-\\uFF5A\\uFF21-\\uFF3A\\uFF10-\\uFF19]+", hankaku : "[\\uFF00-\\uFFEF]+", hangul : "[\\u1100-\\u11FF\\uAC00-\\uD7AF\\u3130-\\u318F]+", }, get startReg() { let reg = {}; for(let n in this._regexp) { reg[n] = new RegExp('^' + this._regexp[n]); } delete this.startReg; return this.startReg = reg; }, get endReg() { let reg = {}; for(let n in this._regexp) { reg[n] = new RegExp(this._regexp[n] + '$'); } delete this.endReg; return this.endReg = reg; }, getCharType: function(aChar) { var c = aChar.charCodeAt(0); //if (c >= 0x30 && c <= 0x39) return "suuji"; if (c >= 0x30 && c <= 0x39 || c >= 0x41 && c <= 0x5A || c >= 0x61 && c <= 0x7A || c === 0x5F) return "eisu_han"; if (c >= 0x30A0 && c <= 0x30FA || c === 0x30FC) return "katakana"; if (c >= 0x3040 && c <= 0x309F) return "hiragana"; if (c >= 0x4E00 && c <= 0x9FA0) return "kanji"; if (c >= 0xFF41 && c <= 0x9F5A || c >= 0xFF21 && c <= 0xFF3A || c >= 0xFF10 && c <= 0xFF19) return "eisu_zen"; if (c >= 0xFF00 && c <= 0xFFEF) return "hankaku"; if (c >= 0x1100 && c <= 0x11FF || c >= 0xAC00 && c <= 0xD7AF || c >= 0x3130 && c <= 0x318F) return "hangul"; return ""; }, searchText: '', searchEngines: [], init: function(){ this.isMac = navigator.platform.indexOf("Mac") == 0; this.searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); this.context = document.getElementById('contentAreaContextMenu'); var searchselect = document.getElementById('context-searchselect'); searchselect.style.display = 'none'; this.menu = this.context.insertBefore(document.createElement('menu'), searchselect); this.menu.setAttribute('id', 'context-searcher'); this.menu.setAttribute('class', 'menu-iconic'); this.menu.setAttribute('accesskey', searchselect.accessKey); this.menu.setAttribute('onclick', 'if (event.target == this) { contextSearcher.command(event); closeMenus(this); }'); this.popup = this.menu.appendChild( document.createElement('menupopup') ); this.context.addEventListener('popupshowing', this, false); this.menu.addEventListener('DOMMouseScroll', this, false); gBrowser.mPanelContainer.addEventListener(this.isMac ? 'mousedown' : 'click', this, false); window.addEventListener('unload', this, false); }, uninit: function() { this.context.removeEventListener('popupshowing', this, false); this.menu.removeEventListener('DOMMouseScroll', this, false); gBrowser.mPanelContainer.removeEventListener('click', this, false); gBrowser.mPanelContainer.removeEventListener('mousedown', this, false); window.removeEventListener('unload', this, false); }, destroy: function(){ this.uninit(); document.getElementById('context-searchselect').style.removeProperty('display'); var m = document.getElementById('context-searcher'); if (m) m.parentNode.removeChild(m); }, handleEvent: function(event) { if (this[event.type]) this[event.type](event); }, unload: function(e){ this.uninit(); }, DOMMouseScroll: function(e) { if (this.searchEngines.length === 0) this.searchEngines = this.searchService.getVisibleEngines({}); if (!this.searchEngines || this.searchEngines.length == 0) return; var index = this.searchEngines.indexOf(this.searchService.currentEngine); // var newEngine = e.detail > 0? // this.searchEngines[index+1] || this.searchEngines[0]: // this.searchEngines[index-1] || this.searchEngines[this.searchEngines.length -1]; var newEngine = e.detail > 0? this.searchEngines[index+1] : this.searchEngines[index-1]; if (!newEngine) return; this.searchService.currentEngine = newEngine; this.setMenuitem(); }, command: function(e){ var target = e.target; var engine = e.target.engine || this.menu.engine; var submission = engine.getSubmission(this.searchText, null); if (!submission) return; var newtab = this.NEW_TAB || e.button === 1 || e.shiftKey || e.ctrlKey; if (!newtab) { loadURI(submission.uri.spec, null, submission.postData, false); } else { gBrowser.selectedTab = gBrowser.addTab(submission.uri.spec, { postData: submission.postData, ownerTab: gBrowser.mCurrentTab, }); } }, click: function(event) { if (event.button === 2) { this._clickNode = event.rangeParent; this._clickOffset = event.rangeOffset; this._clientX = event.clientX; } else { this._clickNode = null; this._clickOffset = 0; this._clientX = 0; } }, mousedown: function(event) { this.click(event); }, setMenuitem: function() { var currentEngine = this.searchService.currentEngine; var l = this.searchText.length > 16? this.searchText.substr(0, 16) + '...' : this.searchText; this.menu.engine = currentEngine; this.menu.setAttribute('label', gNavigatorBundle.getFormattedString("contextMenuSearchText", [currentEngine.name, l])); //this.menu.setAttribute('label', 'Искать "' + l + '"'); //this.menu.setAttribute('tooltiptext', currentEngine.name); if (currentEngine.iconURI) this.menu.style.listStyleImage = 'url("' + currentEngine.iconURI.spec + '")'; else this.menu.style.removeProperty('list-style-image'); }, popupshowing: function(e){ if (e.target != this.context) return; this.searchText = gContextMenu.isTextSelected? this.getBrowserSelection() : gContextMenu.onImage? gContextMenu.target.getAttribute('alt') : //gContextMenu.onLink? gContextMenu.linkText() : gContextMenu.onTextInput? this.getTextInputSelection() : this.getCursorPositionText(); if (!this.searchText || !/\S/.test(this.searchText)) { this.menu.hidden = true; return; } if (this.searchText.length > 256) this.searchText = this.searchText.substr(0, 256); this.menu.hidden = false; if (!this.popup.hasChildNodes() || e.ctrlKey) this.createMenuitem(); this.setMenuitem(); }, createMenuitem: function(){ this.searchEngines = this.searchService.getVisibleEngines({}); if (!this.searchEngines || this.searchEngines.length == 0) return; var f; while (f = this.popup.firstChild) { this.popup.removeChild(f); } this.menu.engine = this.searchService.currentEngine; if (this.menu.engine.iconURI) this.menu.style.listStyleImage = 'url("' + this.menu.engine.iconURI.spec + '")'; else this.menu.style.removeProperty('list-style-image'); for (var i = 0, s = this.searchEngines, l = s.length; i < l; i++) { var engine = s[i]; var m = document.createElement('menuitem'); m.setAttribute('label', engine.name); m.setAttribute('class', 'menuitem-iconic bookmark-item'); if (engine.iconURI) { m.setAttribute('image', engine.iconURI.spec); } m.setAttribute('oncommand', 'contextSearcher.command(event);'); m.setAttribute('onclick', 'checkForMiddleClick(this, event);'); m.engine = engine; this.popup.appendChild(m); } }, getBrowserSelection: function () { var win = document.commandDispatcher.focusedWindow; var sel = win.getSelection(); var str = ''; if (sel.isCollapsed) return str; for(var i = 0, l = sel.rangeCount; i < l; i++) { str += sel.getRangeAt(i) + ' '; } return str.replace(/^\s*|\s*$/g, '').replace(/\s+/g, ' '); }, getTextInputSelection: function () { var elem = document.commandDispatcher.focusedElement; var str = elem.value.slice(elem.selectionStart, elem.selectionEnd); return str.replace(/^\s*|\s*$/g, '').replace(/\s+/g, ' '); }, getCursorPositionText: function() { var node = this._clickNode; var offset = this._clickOffset; if (!node || node.nodeType !== Node.TEXT_NODE) return ""; var text = node.nodeValue; // 文字の右半分をクリック時に次の文字を取得する対策 var range = node.ownerDocument.createRange(); range.setStart(node, offset); var rect = range.getBoundingClientRect(); range.detach(); if (rect.left >= this._clientX) offset--; if (!text[offset]) return ""; var type = this.getCharType(text[offset]); if (!type) return ""; var mae = text.substr(0, offset); var ato = text.substr(offset); // text[offset] はこっちに含まれる var ato_word = (this.startReg[type].exec(ato) || [""])[0]; var str = this.endReg[type].test(mae) ? RegExp.lastMatch + ato_word : ato_word; if (str.length === 1) { if (type === "kanji") { if (this.startReg["hiragana"].test(ato.substr(ato_word.length))) str += RegExp.lastMatch; } else { return ""; } } return str; }, log: function() { Application.console.log("[contextSearcher] " + Array.slice(arguments)); } } window.contextSearcher.init(); |
PEAKTOP > 19-09-2013 20:47:18 |
Обновлённый и русифицированный вариант contextSearcher версии 0.0.8 Выделить код Код:// ==UserScript== // @name contextSearcher.uc.js // @namespace http://d.hatena.ne.jp/Griever/ // @description 右クリック→検索の強化 // @include main // @compatibility Firefox 4 // @version 0.0.8 // @note 0.0.8 Firefox 19 で入力欄で使えなくなったのを修正 // @note 0.0.8 NEW_TAB の初期値を browser.search.openintab にした // @note 0.0.7 リンク上でも単語を取得するように変更 // @note 0.0.7 単語の取得を効率化 // @note 0.0.6 カタカナの正規表現のミスを修正 // @note 0.0.6 splitmenu をやめた(menu 部分をクリックして検索可能) // @note 0.0.6 Mac でカーソル下の単語をうまく拾えてなかったらしいのを修正したかも // @note 0.0.5 サブメニューを中クリックすると2回実行される問題を修正 // @note 0.0.5 メニューの検索エンジン名を非表示にした // @note 0.0.5 カーソル下の単語の取得を調整 // @note 0.0.5 "・"をカタカナとして処理していたのを修正 // @note 0.0.4 アイコンの無い検索エンジンがあるとエラーになるのを修正 // ==/UserScript== // http://f.hatena.ne.jp/Griever/20100918161044 // ホイールで既定のエンジン変更、サブメニューから他の検索エンジンの利用 // 右クリックの位置により選択文字列、カーソル下の単語を検索可能 if (window.contextSearcher) { window.contextSearcher.destroy(); delete window.contextSearcher; } window.contextSearcher = { NEW_TAB: Services.prefs.getBoolPref("browser.search.openintab"), _regexp: { hiragana: "[\\u3040-\\u309F]+", katakana: "[\\u30A0-\\u30FA\\u30FC]+", kanji : "[\\u4E00-\\u9FA0]+", suuji : "[0-9_./,%-]+", eisu_han: "\\w[\\w\\-]*", eisu_zen: "[\\uFF41-\\uFF5A\\uFF21-\\uFF3A\\uFF10-\\uFF19]+", hankaku : "[\\uFF00-\\uFFEF]+", hangul : "[\\u1100-\\u11FF\\uAC00-\\uD7AF\\u3130-\\u318F]+", }, get startReg() { let reg = {}; for(let n in this._regexp) { reg[n] = new RegExp('^' + this._regexp[n]); } delete this.startReg; return this.startReg = reg; }, get endReg() { let reg = {}; for(let n in this._regexp) { reg[n] = new RegExp(this._regexp[n] + '$'); } delete this.endReg; return this.endReg = reg; }, getCharType: function(aChar) { var c = aChar.charCodeAt(0); //if (c >= 0x30 && c <= 0x39) return "suuji"; if (c >= 0x30 && c <= 0x39 || c >= 0x41 && c <= 0x5A || c >= 0x61 && c <= 0x7A || c === 0x5F) return "eisu_han"; if (c >= 0x30A0 && c <= 0x30FA || c === 0x30FC) return "katakana"; if (c >= 0x3040 && c <= 0x309F) return "hiragana"; if (c >= 0x4E00 && c <= 0x9FA0) return "kanji"; if (c >= 0xFF41 && c <= 0x9F5A || c >= 0xFF21 && c <= 0xFF3A || c >= 0xFF10 && c <= 0xFF19) return "eisu_zen"; if (c >= 0xFF00 && c <= 0xFFEF) return "hankaku"; if (c >= 0x1100 && c <= 0x11FF || c >= 0xAC00 && c <= 0xD7AF || c >= 0x3130 && c <= 0x318F) return "hangul"; return ""; }, searchText: '', searchEngines: [], init: function(){ this.isMac = navigator.platform.indexOf("Mac") == 0; this.searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); this.context = document.getElementById('contentAreaContextMenu'); var searchselect = document.getElementById('context-searchselect'); searchselect.style.display = 'none'; this.menu = this.context.insertBefore(document.createElement('menu'), searchselect); this.menu.setAttribute('id', 'context-searcher'); this.menu.setAttribute('class', 'menu-iconic'); this.menu.setAttribute('accesskey', searchselect.accessKey); this.menu.setAttribute('onclick', 'if (event.target == this) { contextSearcher.command(event); closeMenus(this); }'); this.popup = this.menu.appendChild( document.createElement('menupopup') ); this.context.addEventListener('popupshowing', this, false); this.menu.addEventListener('DOMMouseScroll', this, false); gBrowser.mPanelContainer.addEventListener(this.isMac ? 'mousedown' : 'click', this, false); window.addEventListener('unload', this, false); }, uninit: function() { this.context.removeEventListener('popupshowing', this, false); this.menu.removeEventListener('DOMMouseScroll', this, false); gBrowser.mPanelContainer.removeEventListener('click', this, false); gBrowser.mPanelContainer.removeEventListener('mousedown', this, false); window.removeEventListener('unload', this, false); }, destroy: function(){ this.uninit(); document.getElementById('context-searchselect').style.removeProperty('display'); var m = document.getElementById('context-searcher'); if (m) m.parentNode.removeChild(m); }, handleEvent: function(event) { if (this[event.type]) this[event.type](event); }, unload: function(e){ this.uninit(); }, DOMMouseScroll: function(e) { if (this.searchEngines.length === 0) this.searchEngines = this.searchService.getVisibleEngines({}); if (!this.searchEngines || this.searchEngines.length == 0) return; var index = this.searchEngines.indexOf(this.searchService.currentEngine); // var newEngine = e.detail > 0? // this.searchEngines[index+1] || this.searchEngines[0]: // this.searchEngines[index-1] || this.searchEngines[this.searchEngines.length -1]; var newEngine = e.detail > 0? this.searchEngines[index+1] : this.searchEngines[index-1]; if (!newEngine) return; this.searchService.currentEngine = newEngine; this.setMenuitem(); }, command: function(e){ var target = e.target; var engine = e.target.engine || this.menu.engine; var submission = engine.getSubmission(this.searchText, null); if (!submission) return; var where = whereToOpenLink(e); if (this.NEW_TAB && where === 'current' || where === 'save') where = 'tab'; openLinkIn(submission.uri.spec, where, { postData: submission.postData, relatedToCurrent: true }); }, click: function(event) { if (event.button === 2) { this._clickNode = event.rangeParent; this._clickOffset = event.rangeOffset; this._clientX = event.clientX; } else { this._clickNode = null; this._clickOffset = 0; this._clientX = 0; } }, mousedown: function(event) { this.click(event); }, setMenuitem: function() { var currentEngine = this.searchService.currentEngine; var l = this.searchText.length > 16? this.searchText.substr(0, 16) + '...' : this.searchText; this.menu.engine = currentEngine; //this.menu.setAttribute('label', gNavigatorBundle.getFormattedString("contextMenuSearchText", [currentEngine.name, l])); this.menu.setAttribute('label', 'Искать "' + l + '"' +' в ' + currentEngine.name); this.menu.setAttribute('tooltiptext', currentEngine.name); this.menu.setAttribute('tooltiptext', currentEngine.name); if (currentEngine.iconURI) this.menu.style.listStyleImage = 'url("' + currentEngine.iconURI.spec + '")'; else this.menu.style.removeProperty('list-style-image'); }, popupshowing: function(e){ if (e.target != this.context) return; this.searchText = gContextMenu.onTextInput? this.getTextInputSelection() : gContextMenu.isTextSelected? this.getBrowserSelection() : gContextMenu.onImage? gContextMenu.target.getAttribute('alt') : //gContextMenu.onLink? gContextMenu.linkText() : this.getCursorPositionText(); if (!this.searchText || !/\S/.test(this.searchText)) { this.menu.hidden = true; return; } if (this.searchText.length > 256) this.searchText = this.searchText.substr(0, 256); this.menu.hidden = false; if (!this.popup.hasChildNodes() || e.ctrlKey) this.createMenuitem(); this.setMenuitem(); }, createMenuitem: function(){ this.searchEngines = this.searchService.getVisibleEngines({}); if (!this.searchEngines || this.searchEngines.length == 0) return; var f; while (f = this.popup.firstChild) { this.popup.removeChild(f); } this.menu.engine = this.searchService.currentEngine; if (this.menu.engine.iconURI) this.menu.style.listStyleImage = 'url("' + this.menu.engine.iconURI.spec + '")'; else this.menu.style.removeProperty('list-style-image'); for (var i = 0, s = this.searchEngines, l = s.length; i < l; i++) { var engine = s[i]; var m = document.createElement('menuitem'); m.setAttribute('label', engine.name); m.setAttribute('class', 'menuitem-iconic bookmark-item'); if (engine.iconURI) { m.setAttribute('image', engine.iconURI.spec); } m.setAttribute('oncommand', 'contextSearcher.command(event);'); m.setAttribute('onclick', 'checkForMiddleClick(this, event);'); m.engine = engine; this.popup.appendChild(m); } }, getBrowserSelection: function () { var win = document.commandDispatcher.focusedWindow; var sel = win.getSelection(); var str = ''; if (sel.isCollapsed) return str; for(var i = 0, l = sel.rangeCount; i < l; i++) { str += sel.getRangeAt(i) + ' '; } return str.replace(/^\s*|\s*$/g, '').replace(/\s+/g, ' '); }, getTextInputSelection: function () { var elem = document.commandDispatcher.focusedElement; var str = elem.value.slice(elem.selectionStart, elem.selectionEnd); return str.replace(/^\s*|\s*$/g, '').replace(/\s+/g, ' '); }, getCursorPositionText: function() { var node = this._clickNode; var offset = this._clickOffset; if (!node || node.nodeType !== Node.TEXT_NODE) return ""; var text = node.nodeValue; // 文字の右半分をクリック時に次の文字を取得する対策 var range = node.ownerDocument.createRange(); range.setStart(node, offset); var rect = range.getBoundingClientRect(); range.detach(); if (rect.left >= this._clientX) offset--; if (!text[offset]) return ""; var type = this.getCharType(text[offset]); if (!type) return ""; var mae = text.substr(0, offset); var ato = text.substr(offset); // text[offset] はこっちに含まれる var ato_word = (this.startReg[type].exec(ato) || [""])[0]; var str = this.endReg[type].test(mae) ? RegExp.lastMatch + ato_word : ato_word; if (str.length === 1) { if (type === "kanji") { if (this.startReg["hiragana"].test(ato.substr(ato_word.length))) str += RegExp.lastMatch; } else { return ""; } } return str; }, log: function() { Application.console.log("[contextSearcher] " + Array.slice(arguments)); } } window.contextSearcher.init(); |
SendInfo > 19-09-2013 22:31:29 |
PEAKTOP |
PEAKTOP > 20-09-2013 00:12:20 |
SendInfo, в том-то и дело, что код из поста №19 не совсем корректно работал - не отображал текст в пункте контекстного меню Попутно два вопроса к знатокам: |
bunda1 > 21-09-2013 13:48:55 |
PEAKTOP пишет
Выделить код Код:// Context Search mini, от 21.09.2013. ................................ (function () { var searchSelect = document.getElementById('context-searchselect'); searchSelect.collapsed = true; var contextMenu = document.getElementById("contentAreaContextMenu"); var menu = contextMenu.insertBefore( document.createElement('menu'), searchSelect ); menu.setAttribute("label", "Искать в Google или в ..."); menu.setAttribute("class", "menu-iconic"); menu.style.listStyleImage = 'url("https://www.google.lv/favicon.ico")'; menu.setAttribute("onclick", "\ if ( !!event.target.engine ) return;\ gBrowser.loadOneTab('http://www.google.com/search?q=' + getBrowserSelection(), null, null, null, false, false);\ setTimeout(function() { document.getElementById('contentAreaContextMenu').hidePopup() }, 0);\ "); var observeStatus = new MutationObserver(function() { menu.hidden = searchSelect.hidden; }); observeStatus.observe( searchSelect, { attributes: true, attributeFilter: ["hidden"] } ); addDestructor(function() { contextMenu.removeChild( menu ); observeStatus.disconnect(); searchSelect.collapsed = false }); var menuPopup = menu.appendChild(document.createElement("menupopup")); var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); addEventListener('popupshowing', function(e) { menuPopup.textContent = ""; var engines = searchService.getVisibleEngines({}); engines.forEach(function( engine ) { var mItem = document.createElement("menuitem"); mItem.setAttribute("label", engine.name ); mItem.setAttribute("class", "menuitem-iconic"); mItem.setAttribute("src", engine.iconURI.spec ); mItem.engine = engine; menuPopup.appendChild( mItem ); }); menuPopup.setAttribute("oncommand", "\ var submission = event.target.engine.getSubmission( getBrowserSelection(), null );\ gBrowser.loadOneTab( submission.uri.spec, null, null, submission.postData, false, false );\ "); }, false, menu ); })(); PEAKTOP пишет
Выделить код Код:var menu = contextMenu.insertBefore( document.createElement('menu'), searchSelect ); Тут вместо searchSelect укажи пункт контекстного меню над которым пункт Искать в Google должен находится, например пункт Копировать: |
PEAKTOP > 21-09-2013 19:30:55 |
bunda1, БОЛЬШОЕ ПРЕБОЛЬШОЕ СПАСИБО! |
dedmazai1870 > 21-09-2013 21:28:28 |
PEAKTOP скрытый текст https://www.google.com/search?q=%D0%BF%D0%BE%D0%B8%D1%81%D0%BA+%D0%BF%D0%BE+%D1%81%D0%B0%D0%B9%D1%82%D1%83&filter=0&as_q=site%3Aforum.mozilla-russia.org Эта строка в виде ссылки. Это пример, сайт должен подставляться тот, на котором я выделил какую-то фразу. Можно в кнопке это реализовать? P.S. Я перешёл на с . Там поиск по форуму отломан и человек сделал javascript, который реализовал поиск по форуму через google именно такой функцией. Я не знаю javascript и самостоятельно прикрутить к Вашей кнопке не смогу. Может этот код Вам пригодится? скрытый текст Выделить код Код:document.addEventListener('DOMContentLoaded', function() { if (typeof(searchform) != 'undefined') { var all_f = location.host + location.pathname.substring(0, location.pathname.lastIndexOf('/')); //opera.postError(all_f); var google_search = '<div>'; google_search += '<input type="text" name="q" size="30" value="поиск в google" onblur="if (this.value == \'\') this.value = \'поиск в google\'" onfocus="if (this.value == \'поиск в google\') this.value = \'\'">'; google_search += '<input type="submit" value=" Поиск ">'; google_search += '<input type="hidden" name="filter" value="0">'; if (location.pathname.indexOf('topic.dml') != '-1') { var topic_id = location.search.substring(location.search.indexOf('id=')); if (topic_id.indexOf('&') != '-1') topic_id = topic_id.substring(0, topic_id.indexOf('&')); //opera.postError(topic_id); google_search += '<p>'; google_search += '<input type="radio" name="as_q" value="site:' + all_f + ' inurl:' + topic_id + '" checked id="this-forum">'; google_search += '<label for="this-topic">В этой теме</label>'; google_search += '<input type="radio" name="as_q" value="site:' + all_f + '" id="all-forums">'; google_search += '<label for="all-forums">Во всех форумах</label>'; google_search += '</p>'; } else google_search += '<input type="hidden" name="as_q" value="site:' + all_f + '">'; google_search += '</div>'; searchform.action = 'http://www.google.com/search'; searchform.innerHTML = google_search; } }, false) P.P.S. Ещё раз спасибо. |
Zaycoff > 21-09-2013 21:38:15 |
dedmazai1870 пишет
Для FF тоже такое есть - Search in Website+ Хотя, наверное, возможность, при зажатом Ctrl или чего-то такого, производить поиск по сайту в выбранной поисковой системе, была бы интересной, правда не знаю поддерживают ли всякие дакдаки и яндексы такую фичу... |
2k1dmg > 21-09-2013 22:20:08 |
PEAKTOP пишет
скрытый текст Выделить код Код:// 2013-09-22 (function () { function _localize(sid) { let strings = { en : { siteSearch : "Site Search" }, ru : { siteSearch : "Поиск по сайту" } }; let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } var searchSelect = document.getElementById('context-searchselect'); searchSelect.collapsed = true; var contextMenu = document.getElementById("contentAreaContextMenu"); var menu = contextMenu.insertBefore(document.createElement('menu'), searchSelect); menu.setAttribute("class", "menu-iconic"); var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var engines = searchService.getVisibleEngines({}); var isCustomEngine = false; var customName = "Google"; if (customName != "") { for (let i in engines) { if (engines[i].name != customName) continue; menu.setAttribute("label", engines[i].name); menu.style.listStyleImage = 'url("' + engines[i].iconURI.spec + '")'; menu.mEngine = engines[i]; isCustomEngine = true; break; } } if (!isCustomEngine) { menu.setAttribute("label", searchService.defaultEngine.name); menu.style.listStyleImage = 'url("' + searchService.defaultEngine.iconURI.spec + '")'; menu.mEngine = searchService.defaultEngine; } menu.setAttribute("onclick", "\ if (!event.target.mEngine) return;\ var submission = event.target.mEngine.getSubmission( getBrowserSelection(), null );\ gBrowser.loadOneTab( submission.uri.spec, {relatedToCurrent: true, postData: submission.postData, inBackground: false} );\ setTimeout(function() { document.getElementById('contentAreaContextMenu').hidePopup() }, 0);\ "); var observeStatus = new MutationObserver(function () { menu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); addDestructor(function () { contextMenu.removeChild(menu); observeStatus.disconnect(); searchSelect.collapsed = false; }); var menuPopup = menu.appendChild(document.createElement("menupopup")); addEventListener('popupshowing', function (e) { if (e.target != contextMenu) return; var selectedText = getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (e) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; var engineName = isCustomEngine ? customName : searchService.defaultEngine.name; var menuLabel = engineName; try { menuLabel = gNavigatorBundle.getFormattedString("contextMenuSearch", [engineName, selectedText]); } catch (e) { menuLabel = gNavigatorBundle.getFormattedString("contextMenuSearchText", [engineName, selectedText]); } menu.label = menuLabel; function clearMenuLabel(e) { if (e.target != contextMenu) return; removeEventListener(e.type, clearMenuLabel, false, contextMenu); menu.label = engineName; } addEventListener('popuphiding', clearMenuLabel, false, contextMenu); }, false, contextMenu); var siteSearchGoogle; for (let i in engines) { if (engines[i].name != "Google") continue; siteSearchGoogle = engines[i]; break; } addEventListener('popupshowing', function (e) { menuPopup.textContent = ""; function mItemCreate(trg, ssg/*, dm*/) { var mItem = document.createElement("menuitem"); mItem.setAttribute("class", "menuitem-iconic"); mItem.engine = trg; if (ssg == "siteSearchGoogle") { menuPopup.appendChild(document.createElement("menuseparator")); mItem.setAttribute("label", _localize("siteSearch") /*+ ": " + dm*/); //mItem.setAttribute("src", gBrowser.selectedTab.image); mItem.setAttribute(ssg, "true"); } else { mItem.setAttribute("label", trg.name); mItem.setAttribute("src", trg.iconURI.spec); } menuPopup.appendChild(mItem); } for (let i in engines) { if (isCustomEngine && engines[i].name == customName || !isCustomEngine && engines[i].name == searchService.defaultEngine.name) continue; mItemCreate(engines[i]); } function isDomain() { var domain; try { domain = gBrowser.currentURI.host; } catch (e) {} return domain; } if (siteSearchGoogle && isDomain()) { mItemCreate(siteSearchGoogle, "siteSearchGoogle"/*, isDomain()*/); } menuPopup.setAttribute("oncommand", "\ if (!event.target.engine) return;\ var ssg = '';\ if (event.target.hasAttribute('siteSearchGoogle')) {\ ssg = ' site:' + gBrowser.currentURI.host;\ }\ var submission = event.target.engine.getSubmission(getBrowserSelection() + ssg, null);\ gBrowser.loadOneTab( submission.uri.spec, {relatedToCurrent: true, postData: submission.postData, inBackground: false} );\ "); }, false, menu); })(); upd: + "Поиск по сайту" upd: исправил ошибки |
PEAKTOP > 21-09-2013 22:44:32 |
dedmazai1870, в основной версии Context Search так и сделано, там стоит - в пункт меню выводится не более 16 символов из выделенного текста. 21-09-2013 23:10:33 |
dedmazai1870 > 21-09-2013 23:33:25 |
Zaycoff |
PEAKTOP > 22-09-2013 00:31:54 |
Соглашусь с dedmazai1870, пункт "Поиск по текущему сайту" с помощью этой кнопки был бы очень кстати. |
2k1dmg > 22-09-2013 10:21:38 |
dedmazai1870 |
dedmazai1870 > 22-09-2013 11:06:42 |
villa7 > 22-09-2013 13:25:54 |
2k1dmg |
PEAKTOP > 23-09-2013 03:29:45 |
Добавил "Поиск по текущему сайту" через Яндекс Код инициализации Выделить код Код://Context Search - искать поисковыми системами через контекстное меню............................... (function () { function _localize(sid) { let strings = { en : { siteSearch : "Search this site - Google" }, ru : { siteSearch : "Поиск по этому сайту - Google" } }; let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } function _localize2(sid) { let strings = { en : { siteSearch2 : "Search this site - Yandex" }, ru : { siteSearch2: "Поиск по этому сайту - Яндекс" } }; let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; _localize2 = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize2.apply(this, arguments); } var searchSelect = document.getElementById('context-searchselect'); searchSelect.collapsed = true; var contextMenu = document.getElementById("contentAreaContextMenu"); var menu = contextMenu.insertBefore(document.createElement('menu'), searchSelect ); menu.setAttribute("class", "menu-iconic"); var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var engines = searchService.getVisibleEngines({}); var isCustomEngine = false; var customName = "Google"; if (customName != "") { for (let i in engines) { if (engines[i].name != customName) continue; menu.setAttribute("label", engines[i].name); menu.style.listStyleImage = 'url("' + engines[i].iconURI.spec + '")'; menu.mEngine = engines[i]; isCustomEngine = true; break; } } if (!isCustomEngine) { menu.setAttribute("label", searchService.defaultEngine.name); menu.style.listStyleImage = 'url("' + searchService.defaultEngine.iconURI.spec + '")'; menu.mEngine = searchService.defaultEngine; } menu.setAttribute("onclick", "\ if ( !!event.target.engine || !event.target.mEngine) return;\ var submission = event.target.mEngine.getSubmission( getBrowserSelection(), null );\ gBrowser.loadOneTab( submission.uri.spec, {relatedToCurrent: true, postData: submission.postData, inBackground: false} );\ setTimeout(function() { document.getElementById('contentAreaContextMenu').hidePopup() }, 0);\ "); var observeStatus = new MutationObserver(function () { menu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); addDestructor(function () { contextMenu.removeChild(menu); observeStatus.disconnect(); searchSelect.collapsed = false; }); var menuPopup = menu.appendChild(document.createElement("menupopup")); addEventListener('popupshowing', function (e) { if (e.target != contextMenu) return; var selectedText = getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (e) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; var engineName = isCustomEngine ? customName : searchService.defaultEngine.name; var menuLabel = engineName; try { menuLabel = gNavigatorBundle.getFormattedString("contextMenuSearch", [engineName, selectedText]); } catch (e) { menuLabel = gNavigatorBundle.getFormattedString("contextMenuSearchText", [engineName, selectedText]); } menu.label = menuLabel; function clearMenuLabel(e) { if (e.target != contextMenu) return; removeEventListener(e.type, clearMenuLabel, false, contextMenu); menu.label = engineName; } addEventListener('popuphiding', clearMenuLabel, false, contextMenu); }, false, contextMenu); addEventListener('popupshowing', function (e) { menuPopup.textContent = ""; for (let i in engines) { if (isCustomEngine && engines[i].name == customName || !isCustomEngine && engines[i].name == searchService.defaultEngine.name) continue; var mItem = document.createElement("menuitem"); mItem.setAttribute("label", engines[i].name); mItem.setAttribute("class", "menuitem-iconic"); mItem.setAttribute("src", engines[i].iconURI.spec); mItem.engine = engines[i]; menuPopup.appendChild(mItem); } menuPopup.setAttribute("oncommand", "\ if ( !event.target.engine ) return;\ var submission = event.target.engine.getSubmission( getBrowserSelection(), null );\ gBrowser.loadOneTab( submission.uri.spec, {relatedToCurrent: true, postData: submission.postData, inBackground: false} );\ "); menuPopup.appendChild(document.createElement("menuseparator")); var mItemSbS = menuPopup.appendChild(document.createElement("menuitem")); mItemSbS.setAttribute("class", "menuitem-iconic"); mItemSbS.setAttribute("image", "") mItemSbS.setAttribute("label", _localize("siteSearch")); mItemSbS.setAttribute("oncommand", "\ var selectedText = getBrowserSelection(); if (!selectedText) return;\ try {var domain = content.document.domain;} catch (e) {return;}\ selectedText = encodeURIComponent(selectedText);\ gBrowser.loadOneTab( 'https://www.google.com/search?q=' + selectedText + '&filter=0&as_q=site:' + domain, {relatedToCurrent: true, inBackground: false} );\ "); //menuPopup.appendChild(document.createElement("menuseparator")); var mItemSbS = menuPopup.appendChild(document.createElement("menuitem")); mItemSbS.setAttribute("class", "menuitem-iconic"); mItemSbS.setAttribute("image", "data:application/file;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAABCklEQVR4nKVTuwrCUAw9vXRwEnF0kI4iTuLg5NRB/BRx6IeIk4M4iJ/SoWM/QMTByUnED5ATh+AjrdqqgXC5JDn3JOfGIyn4wpxznrnnMuZzYDgEwhAIAsD3gSi6h3MPkpS7z2bCRkN4Oul9NBJWq2JyMgCWwWIB9HpArfaxjWcQC3A+A5XKx+KsWYBWC9jvSxXeWFiAKALSFJhOgculHIXsgLhaCZ0TttvCZvPlEJ89L+N6ra0kCdDpFBKwAEmiPpkUKgHop7IAcaxnGBYWv2aQpipjEJR6PQ+w2Wj/vv8Dg+MR2O2Afv8RfSOlWSiSwsNBOBgI63XhdqsSLZfCbld9PH67C96/63wFqBHGA4wLrxUAAAAASUVORK5CYII=") mItemSbS.setAttribute("label", _localize2("siteSearch2")); mItemSbS.setAttribute("oncommand", "\ var selectedText = getBrowserSelection(); if (!selectedText) return;\ try {var domain = content.document.domain;} catch (e) {return;}\ selectedText = encodeURIComponent(selectedText);\ gBrowser.loadOneTab( 'http://yandex.ru/yandsearch?text=' + selectedText + '&site=' + domain, {relatedToCurrent: true, inBackground: false} );\ "); }, false, menu); })(); // Библиотека - Открывать библиотеку в вкладке и последнюю активную папку библиотеки, от 08.09.2013. ................................ (function () { var command; const pn = "CB.lastUsedLibraryFolder"; const library = "chrome://browser/content/places/places.xul"; // Перехватывать открытие библиотеки и открывать её в вкладке рядом с текущей .... function libraryInTab(e) { var oncommand = e.target.getAttribute("oncommand"); if ( !/PlacesCommandHook.showPlacesOrganizer|DownloadsUI|DownloadsPanel/.test( oncommand ) ) return; if ( /UnfiledBookmarks/.test( oncommand ) && e.type == 'click' ) return; e.stopPropagation(); command = oncommand; setTimeout(function() { document.getElementById("appmenu-popup").hidePopup() }, 0); // закрыть все вкладки библиотеки кроме первой .... var libTab = null; Array.slice( gBrowser.tabs ).forEach(function(tab) { if ( tab.linkedBrowser.currentURI.spec !== library ) return; !libTab ? libTab = tab : gBrowser.removeTab(tab); }) // переместить или открыть библиотеку рядом с текущей вкладкой .... var selTab = gBrowser.selectedTab; if ( libTab !== selTab ) { if ( libTab && gBrowser.visibleTabs.indexOf(libTab) == -1 ) TabView.moveTabTo( libTab, TabView._window.GroupItems._activeGroupItem.id ); libTab = libTab || gBrowser.addTab( library ); var pos = selTab._tPos + ( libTab._tPos > selTab._tPos ); gBrowser.moveTabTo( libTab, pos ); gBrowser.selectedTab = libTab; // если библиотека уже открыта if ( !libTab.hasAttribute('busy') ) openLastUsedFolder( false, content.document.defaultView ); } }; addEventListener("command", libraryInTab, true, window ); addEventListener("click", libraryInTab, true, document.getElementById('appmenuSecondaryPane') ); // Открывать и запоминать последнюю активную папку если библиотека открыта как вкладка .... function openLastUsedFolder( e, win ) { if ( e !== false ) { var win = e.target.defaultView; if ( win.location != library ) return; } // получить нужную папку try { var value = cbu.getPrefs( pn ) } catch(e) { var value = '' }; if ( /History/.test( command ) ) value = 'place:sort=4&type=3'; if ( /UnfiledBookmarks/.test( command ) ) value = 'place:folder=UNFILED_BOOKMARKS&excludeItems=1&expandQueries=0'; if ( /DownloadsUI|DownloadsPanel/.test( command ) ) value = 'place:transition=7&sort=4'; if ( /DownloadsPanel/.test( command ) ) setTimeout(function() { document.getElementById("downloadsPanel").hidePopup() }, 10); // открыть нужную папку используя рекурсию (function f() { try { win.PlacesOrganizer.location = value; win.PlacesOrganizer._places.focus(); } catch(e) { setTimeout(function() { f() }, 0) }; })(); // запоминать последнюю папку при закрытии библиотеки if ( e == false ) return; win.addEventListener("unload", function(e) { this.removeEventListener( e.type, arguments.callee, false ); var loc = win.PlacesOrganizer.location; if ( loc == 'place:transition=7&sort=4') return; // исключить папки загрузки if ( /place:beginTime|place:sort=4&type=3/.test( loc ) ) return; // исключить папки истории cbu.setPrefs( pn, loc ); }, false ); }; addEventListener("DOMContentLoaded", openLastUsedFolder, false, gBrowser ); })(); |
Vladik > 23-09-2013 15:21:02 |
Хорошая кнопка, а когда будет готовый вариант, а то там собери, тут доложи, оттуда добавь... |
2k1dmg > 23-09-2013 20:44:46 |
Context Search 2013-09-23 Выделить код Код:// 2013-09-23 (function () { var options = { customName: "Google", // оставить пустым если не нужно закреплять // определенный поисковик // примеры customName: "Google" или customName: "Яндекс" или customName: "" hideDefaultEngineInList: false, // true - не допбовлять поисковик по умолчанию в выпадающий список поисковиков siteSearchIconsOn: true // true - добавить иконки для поисковиков по сайту }; function _localize(sid) { let strings = { en: { siteSearchGoogle: "Search this site - Google", siteSearchYandex: "Search this site - Yandex", siteSearchYahoo: "Search this site - Yahoo", siteSearchBing: "Search this site - Bing", siteSearchDuckDuckGo: "Search this site - DuckDuckGo" }, ru: { siteSearchGoogle: "Поиск по этому сайту - Google", siteSearchYandex: "Поиск по этому сайту - Яндекс", siteSearchYahoo: "Поиск по этому сайту - Yahoo", siteSearchBing: "Поиск по этому сайту - Bing", siteSearchDuckDuckGo: "Поиск по этому сайту - DuckDuckGo" } }; let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } var contextSearcherObj = { commandHandler: function (event, trg) { var inBg = false; if (event.type == "click" && event.button && event.button == 1) { inBg = true; } var siteSearch = ''; if (event.target.hasAttribute('siteSearch')) { siteSearch = ' site:' + gBrowser.currentURI.host; } var submission = event.target.engine.getSubmission(getBrowserSelection() + siteSearch, null); gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); if (trg == "menu") { setTimeout(function () { document.getElementById('contentAreaContextMenu').hidePopup(); }, 0); } } } var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var engines = searchService.getVisibleEngines({}); var contextMenu = document.getElementById("contentAreaContextMenu"); var searchSelect = document.getElementById('context-searchselect'); searchSelect.style.display = 'none'; var menu = contextMenu.insertBefore(document.createElement('menu'), searchSelect); menu.setAttribute("id", "Context_Search_mini_By_bunda1"); menu.setAttribute("class", "menu-iconic"); var isCustomEngine = false; var customName = options.customName; if (customName != "") { for (let i in engines) { if (engines[i].name != customName) continue; menu.setAttribute("label", engines[i].name); menu.style.listStyleImage = 'url("' + engines[i].iconURI.spec + '")'; menu.engine = engines[i]; isCustomEngine = true; break; } } if (!isCustomEngine) { menu.setAttribute("label", searchService.defaultEngine.name); menu.style.listStyleImage = 'url("' + searchService.defaultEngine.iconURI.spec + '")'; menu.engine = searchService.defaultEngine; } menu.searchWith = contextSearcherObj.commandHandler; menu.setAttribute("onclick", "if (event.target == this && event.target.engine) this.searchWith(event, 'menu');"); addEventListener('popupshowing', function (e) { if (e.target != contextMenu) return; var selectedText = getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (e) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; var engineName = isCustomEngine ? customName : searchService.defaultEngine.name; var menuLabel = engineName; try { menuLabel = gNavigatorBundle.getFormattedString("contextMenuSearch", [engineName, selectedText]); } catch (e) { menuLabel = gNavigatorBundle.getFormattedString("contextMenuSearchText", [engineName, selectedText]); } menu.label = menuLabel; function clearMenuLabel(e) { if (e.target != contextMenu) return; removeEventListener(e.type, clearMenuLabel, false, contextMenu); menu.label = engineName; } addEventListener('popuphiding', clearMenuLabel, false, contextMenu); }, false, contextMenu); var menuPopup = menu.appendChild(document.createElement("menupopup")); //menuPopup.textContent = ""; var mSeparator; function mItemCreate(trg, ssl) { var mItem = document.createElement("menuitem"); mItem.setAttribute("class", "menuitem-iconic"); mItem.engine = trg; if (ssl) { if (!mSeparator ) { mSeparator = document.createElement("menuseparator"); mSeparator.setAttribute("siteSearch", "true"); menuPopup.appendChild(mSeparator); } mItem.setAttribute("label", _localize(ssl)); if (options.siteSearchIconsOn) { mItem.setAttribute("src", trg.iconURI.spec); } mItem.setAttribute("siteSearch", "true"); } else { mItem.setAttribute("label", trg.name); mItem.setAttribute("src", trg.iconURI.spec); } menuPopup.appendChild(mItem); } for (let i in engines) { if (options.hideDefaultEngineInList && isCustomEngine && engines[i].name == customName || options.hideDefaultEngineInList && !isCustomEngine && engines[i].name == searchService.defaultEngine.name) { LOG(engines[i].name); continue; } mItemCreate(engines[i]); } var siteSearchArr = []; [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ].forEach(function(elem) { for (let i in engines) { if (elem[1].indexOf(engines[i].name) == -1) continue; siteSearchArr.push([elem[0], engines[i]]); break; } }); if (siteSearchArr[0]) { siteSearchArr.forEach(function(elem) { mItemCreate(elem[1], "siteSearch" + elem[0]); }); } menuPopup.searchWith = contextSearcherObj.commandHandler; menuPopup.setAttribute("oncommand", "if (!event.target.engine) return; this.searchWith(event);"); menuPopup.setAttribute("onclick", "checkForMiddleClick(this, event);"); addEventListener('popupshowing', function (e) { var siteSearchDomain; function isDomain() { try { siteSearchDomain = gBrowser.currentURI.host; } catch (e) { return false; } return true; } var mItems; if (!menuPopup.mItems) { var mItemsObg = menuPopup.getElementsByTagName("menuitem"); menuPopup.mItems = []; Array.slice(mItemsObg).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.hasAttribute("siteSearch")) menuPopup.mItems.push(elem); }); mItems = menuPopup.mItems; } else { mItems = menuPopup.mItems; } if (isDomain()) { mItems.forEach(function(elem) { elem.removeAttribute("disabled"); elem.setAttribute("tooltiptext", siteSearchDomain); }); } else { mItems.forEach(function(elem) { elem.setAttribute("disabled", "true"); elem.removeAttribute("tooltiptext"); }); } }, false, menu); var observeStatus = new MutationObserver(function () { menu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); addDestructor(function () { contextMenu.removeChild(menu); observeStatus.disconnect(); document.getElementById('context-searchselect').style.removeProperty('display'); }); })(); |
bunda1 > 23-09-2013 22:06:41 |
2k1dmg пишет
А зачем ты изменяешь меню при закрытии : Выделить код Код:function clearMenuLabel(e) { if (e.target != contextMenu) return; removeEventListener(e.type, clearMenuLabel, false, contextMenu); menu.label = engineName; } addEventListener('popuphiding', clearMenuLabel, false, contextMenu); если потом пункт меню при открытии меню опять меняется: Выделить код Код:addEventListener('popupshowing', function (e) { .................... menu.label = menuLabel; |
PEAKTOP > 24-09-2013 23:32:32 |
2k1dmg, есть одно неудобство. Если я зафиксирую в пункте меню "Google", то он будет дублироваться и в подменю. Понятно, что ты берешь список поисковиков из "Панели поиска". Есть возможность исключить "зафиксированный" поисковик из подменю? |
2k1dmg > 25-09-2013 11:07:48 |
PEAKTOP
bunda1 скрытый текст Выделить код Код:// 2013-09-25 (function () { var options = { pinnedEngineName: "", // оставить пустым если не нужно закреплять // определенный поисковик // примеры pinnedEngineName: "Google" или pinnedEngineName: "Яндекс" или pinnedEngineName: "" hideDefaultEngineInList: false, // true - не добавлять поисковик по умолчанию в выпадающий список (подменю) поисковиков siteSearch: { enable: true, // true - добавить поисковики по сайту iconsOn: true, // true - добавить иконки для поисковиков по сайту useDefaultNamesList: true, // true - использовать список поисковиков по умолчанию addCustomNames: false, // true - добавить свои поисковики по сайту customNamesList: [ ["Rambler", ["Rambler", "Рамблер"]] ] // например ["Rambler", ["Rambler", "Рамблер"]] // первая ячека название на английском // вторая ячека массив с предполагаемыми именами поисковика // может состоять из нескольких значений // ["Name1", ["Name1"]], // ["Name2", ["Name2", "Имя2"]], // ["Name3", ["Name3", "Имя3", "Név3"]] }, addImageSearch: true // true - добавить поиск изображений }; function _localize(sid) { let strings = { en: { siteSearch: "Search this site", searchImage: "Search by image" }, ru: { siteSearch: "Поиск по этому сайту", searchImage: "Поиск по изображению" } }; let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } var contextSearcherObj = { init: function() { var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var contextMenu = document.getElementById("contentAreaContextMenu"); var searchSelect = document.getElementById("context-searchselect"); searchSelect.style.display = "none"; var menu = contextMenu.insertBefore(document.createElement("menu"), searchSelect); this.searchService = searchService; this.contextMenu = contextMenu; this.menu = menu; var observeStatus = new MutationObserver(function () { menu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); var prefService = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService); this.branch = prefService.getBranch("browser.search."); if (!("addObserver" in this.branch)) this.branch.QueryInterface(Components.interfaces.nsIPrefBranch2); this.branch.addObserver("", this, false); Services.ww.registerNotification(this); this.createMenu(); if (options.addImageSearch) this.creatImageMenu(); addDestructor(function () { observeStatus.disconnect(); searchSelect.style.removeProperty("display"); Services.ww.unregisterNotification(this); this.branch.removeObserver("", this); this.destroy(); }, this); }, destroy: function() { var contextMenu = this.contextMenu; var menu = this.menu; if (menu) contextMenu.removeChild(menu); var imageMenu = this.imageMenu; if (imageMenu) contextMenu.removeChild(imageMenu); }, update: function() { this.menu.textContent = ""; this.createMenu(); }, observe: function(subject, topic, data) { switch (topic) { case "domwindowclosed": if (subject.document.documentElement.getAttribute("windowtype") == "Browser:SearchManager") { this.update(); } break; case "nsPref:changed": if (data == "defaultenginename") this.update(); break; } }, handleEvent: function(e) { var contextMenu = this.contextMenu; var menu = this.menu; var imageMenu = this.imageMenu; switch (e.type) { case "popupshowing": if(e.target == contextMenu) { this.contextMenuPopup(e); if (imageMenu) this.imageMenuPopup(e); } else if (e.target == menu) this.contextSearcherMenuPopup(e); break; } }, commandHandler: function(event, trg) { var inBg = false; if (event.type == "click" && event.button && event.button == 1) { inBg = true; } var siteSearch = ""; if (event.target.hasAttribute("siteSearch")) { siteSearch = " site:" + gBrowser.currentURI.host; } var submission = event.target.engine.getSubmission(getBrowserSelection() + siteSearch, null); gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); if (trg == "menu") { setTimeout(function () { document.getElementById("contentAreaContextMenu").hidePopup(); }, 0); } }, contextMenuPopup: function(e) { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.menu; var selectedText = getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (e) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; var engineName = this.isCustomEngine ? customName : searchService.defaultEngine.name; var menuLabel; var searchFormattedString = [ "contextMenuSearch", "contextMenuSearchText" ]; function getMenuLabel(str) { try { return gNavigatorBundle.getFormattedString(str, [engineName, selectedText]); } catch (e) {} } for(let i in searchFormattedString) { menuLabel = getMenuLabel(searchFormattedString[i]); if (menuLabel) break; else if (!menuLabel && i == searchFormattedString.length-1) menuLabel = engineName; } menu.label = menuLabel; }, contextSearcherMenuPopup: function(e) { var siteSearchDomain; function isDomain() { try { siteSearchDomain = gBrowser.currentURI.host; } catch (e) { return false; } return true; } var mItems; if (!menuPopup.mItems) { var mItemsObg = menuPopup.getElementsByTagName("menuitem"); menuPopup.mItems = []; Array.slice(mItemsObg).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.hasAttribute("siteSearch")) menuPopup.mItems.push(elem); }); mItems = menuPopup.mItems; } else { mItems = menuPopup.mItems; } if (isDomain()) { mItems.forEach(function(elem) { elem.removeAttribute("disabled"); elem.setAttribute("tooltiptext", siteSearchDomain); }); } else { mItems.forEach(function(elem) { elem.setAttribute("disabled", "true"); elem.removeAttribute("tooltiptext"); }); } }, createMenu: function() { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.menu; var engines = searchService.getVisibleEngines({}); menu.setAttribute("id", "Context_Search_mini_By_bunda1"); menu.setAttribute("class", "menu-iconic"); this.isCustomEngine = false; var customName = options.pinnedEngineName; if (customName != "") { for (let i in engines) { if (engines[i].name != customName) continue; menu.setAttribute("label", engines[i].name); menu.setAttribute("image", engines[i].iconURI.spec); menu.engine = engines[i]; this.isCustomEngine = true; break; } } if (!this.isCustomEngine) { menu.setAttribute("label", searchService.defaultEngine.name); menu.setAttribute("image", searchService.defaultEngine.iconURI.spec); menu.engine = searchService.defaultEngine; } menu.gObj = this; menu.setAttribute("onclick", "if (event.target == this && event.target.engine) this.gObj.commandHandler(event, 'menu');"); addEventListener("popupshowing", this, false, contextMenu); var menuPopup = menu.appendChild(document.createElement("menupopup")); var mSeparator; function mItemCreate(engine, siteSearch) { var mItem = document.createElement("menuitem"); mItem.setAttribute("class", "menuitem-iconic"); mItem.engine = engine; if (siteSearch) { if (!mSeparator ) { mSeparator = document.createElement("menuseparator"); mSeparator.setAttribute(siteSearch, "true"); menuPopup.appendChild(mSeparator); } mItem.setAttribute("label", _localize(siteSearch) + " - " + engine.name); if (options.siteSearch.iconsOn) { mItem.setAttribute("src", engine.iconURI.spec); } mItem.setAttribute(siteSearch, "true"); } else { mItem.setAttribute("label", engine.name); mItem.setAttribute("src", engine.iconURI.spec); } menuPopup.appendChild(mItem); } for (let i in engines) { if (options.hideDefaultEngineInList && this.isCustomEngine && engines[i].name == customName || options.hideDefaultEngineInList && !this.isCustomEngine && engines[i].name == searchService.defaultEngine.name) { continue; } mItemCreate(engines[i]); } function siteSearch_mItemCreate() { var namesList = []; if (options.siteSearch.useDefaultNamesList) { let defaultNamesList = [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ]; namesList = namesList.concat(defaultNamesList); } if (options.siteSearch.addCustomNames && Array.isArray(options.siteSearch.customNamesList)) { namesList = namesList.concat(options.siteSearch.customNamesList); } let i_skips = []; namesList.forEach(function(elem) { for (let i in engines) { if (i_skips.indexOf(i) != -1 || elem[1].indexOf(engines[i].name) == -1) continue; mItemCreate(engines[i], "siteSearch"); i_skips.push(i); break; } }); } if (options.siteSearch.enable) { siteSearch_mItemCreate(); } menuPopup.gObj = this; menuPopup.setAttribute("oncommand", "if (!event.target.engine) return; this.gObj.commandHandler(event);"); menuPopup.setAttribute("onclick", "checkForMiddleClick(this, event);"); addEventListener("popupshowing", this, false, menu); }, commandHandlerSearchImage: function(event) { var inBg = false; if (event.type == "click" && event.button && event.button == 1) { inBg = true; } var imageMenu = this.imageMenu; var link = event.target.link; var linkURI = link + encodeURIComponent(imageMenu.imageSrc); gBrowser.loadOneTab(linkURI, { relatedToCurrent: true, inBackground: inBg }); }, imageMenuPopup: function(e) { var imageMenu = this.imageMenu; imageMenu.hidden = !gContextMenu.onImage; var src = gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL; if (!src || !(/^(ht|f)tps?:\/\//.test(src))) { imageMenu.hidden = true; return; } imageMenu.imageSrc = src; }, creatImageMenu: function() { var menu = document.createElement("menu"); this.imageMenu = menu; menu.setAttribute("id", "context-image-search"); menu.setAttribute("class", "menu-iconic"); menu.setAttribute("label", _localize("searchImage")); menu.setAttribute("image", ""); var mInsNode = document.getElementById("context-sep-copyimage"); mInsNode.parentNode.insertBefore(menu, mInsNode); var menuPopup = menu.appendChild(document.createElement("menupopup")); var mItemsList = [ [ "Google", "https://www.google.com/searchbyimage?image_url=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?rpt=imagedups&text=&img_url=", "" ], [ "TinEye", "http://www.tineye.com/search/?pluginver=firefox-1.0&url=", "" ] ]; function mItemCreate(name, link, image) { var mItem = document.createElement("menuitem"); mItem.setAttribute("class", "menuitem-iconic"); mItem.setAttribute("src", image); mItem.setAttribute("label", name); mItem.link = link; menuPopup.appendChild(mItem); } mItemsList.forEach(function(elem) { mItemCreate(elem[0], elem[1], elem[2]); }); var contextMenu = this.contextMenu; menuPopup.gObj = this; menuPopup.setAttribute("oncommand", "this.gObj.commandHandlerSearchImage(event);"); menuPopup.setAttribute("onclick", "checkForMiddleClick(this, event);"); addEventListener("popupshowing", this, false, contextMenu); } } contextSearcherObj.init(); })(); |
bunda1 > 25-09-2013 20:48:36 |
2k1dmg пишет
Получается отличная кнопка, но было бы не плохо если меню поисковиков открывалась и в текстовых полях, для этого можно поставить наблюдатель на пункт Копировать в контекстном меню страницы: Выделить код Код:var copy = document.getElementById('context-copy'); var observeStatus = new MutationObserver(function() { menu.hidden = copy.hidden || copy.disabled; }); observeStatus.observe( copy, { attributes: true, attributeFilter: ["hidden", "disabled"] } ); addDestructor(function() { observeStatus.disconnect() }); И мне кажется код для отслеживания изменения поисковиков в about:config можно сделать компактней: Выделить код Код:var s = 'browser.search.'; gPrefService.addObserver( s, this, false ); addDestructor(function() { gPrefService.removeObserver( s, this, false ) }); И наверно можно обойтись одним обработчиком: |
PEAKTOP > 26-09-2013 07:08:19 |
2k1dmg пишет
Извини, не заметил. |
Northtech > 08-10-2013 18:02:11 |
Создана новая кнопка и по умолчанию скрыта = скрипт не работает. upd: Выделить код Код:} function siteSearch_mItemCreate() { var namesList = []; if (options.siteSearch.useDefaultNamesList) { let defaultNamesList = [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ]; namesList = namesList.concat(defaultNamesList); } + |
PEAKTOP > 09-10-2013 15:13:28 |
Northtech, если у тебя в панели поиска не присутствуют эти поисковики или их названия отличаются от указанных в коде , то они не будут отображаться в контекстном меню. |
Northtech > 09-10-2013 16:09:54 |
PEAKTOP пишет
Ага, спасибо, поисковики присутствуют, но названия другие. Но насчет настройки (browser.search.context.loadInBackground) пожелание все же остаётся. |
2k1dmg > 09-10-2013 18:17:57 |
Northtech Выделить код Код:(function () { // Context Search mini 2013-10-09 var options = { loadInBackground: false, // true - открывать вкладки в фоне pinnedEngineName: "", // оставить пустым если не нужно закреплять // определенный поисковик // примеры pinnedEngineName: "Google" или pinnedEngineName: "Яндекс" или pinnedEngineName: "" hideDefaultEngineInPopupSubMenu: false, // true - не добавлять поисковик по умолчанию в выпадающий список (подменю) поисковиков campactMenu: true, // true - поисковики в выпадающем списке (подменю) без назаваний campactMenuLengthPerLine: 5, // максимальное количество элементов в одной строке // от 3 до 10, по умолчанию 5 searchBySite: { enable: true, // true - добавить поисковики по сайту iconsOn: true, // true - добавить иконки для поисковиков по сайту useDefaultNamesList: true, // true - использовать список поисковиков по умолчанию addCustomNames: false, // true - добавить свои поисковики по сайту customNamesList: [ ["Rambler", ["Rambler", "Рамблер"]] ] // например ["Rambler", ["Rambler", "Рамблер"]] // первая ячека название на английском // вторая ячека массив с предполагаемыми именами поисковика // может состоять из нескольких значений // ["Name1", ["Name1"]], // ["Name2", ["Name2", "Имя2"]], // ["Name3", ["Name3", "Имя3", "Név3"]] }, searchImageByText: { enable: true, // true - добавить поиск изображения по тексту useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchByImage: { enable: true, // true - добавить поиск по изображению useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchInCache: { enable: true, // true - добавить поиск в кэше // нужно доработать useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] } }; function _localize(sid) { let strings = { en: { searchBySite: "Search this site", searchImageByText: "Search image by text", searchByImage: "Search by image", searchInCache: "Search this page in cache" }, ru: { searchBySite: "Поиск по этому сайту", searchImageByText: "Поиск изображения по тексту", searchByImage: "Поиск по изображению", searchInCache: "Поиск этой страницы в кэше" } }; //let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; function getBrowserUILocale() { // Browser UI locale return Components.classes["@mozilla.org/chrome/chrome-registry;1"] .getService(Components.interfaces.nsIXULChromeRegistry) .getSelectedLocale("global"); } let locale = ""; // ru, en if (!locale || locale == "") locale = getBrowserUILocale().match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } let XULNS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; let XHTMLNS = "http://www.w3.org/1999/xhtml" var contextSearcherObj = { initialized: false, init: function() { if(this.initialized) return; this.initialized = true; this.nodeIds = { searchMenu: "contextSearchMiniByBunda1-menu", searchMenuPopup: "contextSearchMiniByBunda1-popup", searchByImageMenu: "contextSearchMiniByBunda1-image-search-menu", searchByImageMenuPopup: "contextSearchMiniByBunda1-image-search-popup" }; if (document.getElementById(this.nodeIds.searchMenu)) return; var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var contextMenu = document.getElementById("contentAreaContextMenu"); var searchSelect = document.getElementById("context-searchselect"); searchSelect.style.display = "none"; var searchMenu = contextMenu.insertBefore(document.createElementNS(XULNS, "menu"), searchSelect); this.searchService = searchService; this.contextMenu = contextMenu; this.searchSelect = searchSelect; this.searchMenu = searchMenu; this.param = { isPinnedEngine: false, pinnedEngineName: "" }; try { this.createSearchMenu(); if (options.searchByImage.enable) this.createSearchByImageMenu(); } catch (ex) { this.destroy(); Components.utils.reportError(ex); return; } if (options.campactMenu) this.loadStyles(); var observeStatus = new MutationObserver(function () { searchMenu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); /*var prefService = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService); this.branch = prefService.getBranch("browser.search."); if (!("addObserver" in this.branch)) this.branch.QueryInterface(Components.interfaces.nsIPrefBranch2); this.branch.addObserver("", this, false);*/ /* var searchPref = "browser.search."; gPrefService.addObserver(searchPref, this, false);*/ let os = Cc["@mozilla.org/observer-service;1"]. getService(Ci.nsIObserverService); os.addObserver(this, "browser-search-engine-modified", false); Services.ww.registerNotification(this); addEventListener("popupshowing", this, false, contextMenu); addDestructor(function (reason) { observeStatus.disconnect(); Services.ww.unregisterNotification(this); //this.branch.removeObserver("", this); //gPrefService.removeObserver(searchPref, this, false); os.removeObserver(this, "browser-search-engine-modified", false); this.destroy(); if (reason != "destructor") this.unloadStyles(); }, this); }, destroy: function() { if(!this.initialized) return; this.initialized = false; var contextMenu = this.contextMenu; if (!contextMenu) return; if (this.searchMenu) contextMenu.removeChild(this.searchMenu); if (this.searchByImageMenu) contextMenu.removeChild(this.searchByImageMenu); if (this.searchSelect) this.searchSelect.style.removeProperty("display"); }, update: function() { this.searchMenu.textContent = ""; this.createSearchMenu(); }, get isSeaMonkey() { delete this.isSeaMonkey; return this.isSeaMonkey = Services.appinfo.name == "SeaMonkey"; }, observe: function(subject, topic, data) { switch (topic) { case "domwindowclosed": if (subject.document.documentElement.getAttribute("windowtype") == "Browser:SearchManager") { this.update(); } break; /*case "nsPref:changed": switch (data) { case "defaultenginename": case "browser.search.defaultenginename": this.update(); break; } break;*/ case "browser-search-engine-modified": switch (data) { case "engine-current": // engine-default case "engine-changed": case "engine-removed": this.update(); break; } break; } }, handleEvent: function(e) { //var contextMenu = this.contextMenu; //var menu = this.menu; //var imageMenu = this.imageMenu; switch (e.type) { case "popupshowing": this.popupshowingEvent(e); break; } }, popupshowingEvent: function(e) { var trgId = e.target.id; if (trgId) { if (trgId == "contentAreaContextMenu") { this.contextMenuPopup(e); if (this.searchByImageMenu) this.searchByImageMenuPopup(e); } else if (trgId == this.nodeIds.searchMenuPopup) this.searchMenuPopup(e); } }, _stylesLoaded: false, loadStyles: function() { if(this._stylesLoaded) return; this._stylesLoaded = true; var sss = this.sss; var cssURI = this.cssURI = this.makeCSSURI(); if(!sss.sheetRegistered(cssURI, sss.USER_SHEET)) sss.loadAndRegisterSheet(cssURI, sss.USER_SHEET); }, unloadStyles: function() { if(!this._stylesLoaded) return; this._stylesLoaded = false; var sss = this.sss; if(sss.sheetRegistered(this.cssURI, sss.USER_SHEET)) sss.unregisterSheet(this.cssURI, sss.USER_SHEET); }, get sss() { delete this.sss; return this.sss = Components.classes["@mozilla.org/content/style-sheet-service;1"] .getService(Components.interfaces.nsIStyleSheetService); }, makeCSSURI: function() { var cssStr = '\ @namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n\ @-moz-document url("' + window.location.href + '") {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ /*width: 2em !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] * {\n\ /*padding-left: 0 !important;\n\ padding-right: 0 !important;\n\ margin-left: 0 !important;\n\ margin-right: 0 !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n\ width: 2em !important;\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ width: 3em !important;\n\ height: 2em !important;\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox >.menuitem-iconic[campactMenu="true"] > image {\n\ margin-left: .8em !important;\n\ }\n\ }\n\ '; return Services.io.newURI("data:text/css," + encodeURIComponent(cssStr), null, null); }, searchCommandHandler: function(event, trg) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchBySite = ""; if (event.target.hasAttribute("searchBySite")) { searchBySite = " site:" + gBrowser.currentURI.host; } else if (event.target.hasAttribute("searchInCache")) { searchBySite = content.location.href; } let selectedText = this.isSeaMonkey ? content.getSelection() : getBrowserSelection(); if (event.target.engine.imageLink) { gBrowser.loadOneTab(event.target.engine.imageLink + /*getBrowserSelection()*/ selectedText, { relatedToCurrent: true, inBackground: inBg }); } else if (event.target.engine.cacheLink) { gBrowser.loadOneTab(event.target.engine.cacheLink + /*getBrowserSelection()*/ searchBySite, { relatedToCurrent: true, inBackground: inBg }); } else { let submission = event.target.engine.getSubmission(/*getBrowserSelection()*/ selectedText + searchBySite, null); gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); } if (trg == "menu") { setTimeout(function () { document.getElementById("contentAreaContextMenu").hidePopup(); }, 0); } }, contextMenuPopup: function(e) { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; //var selectedText = getBrowserSelection(16); let selectedText = this.isSeaMonkey ? content.getSelection(16) : getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (ex) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; var engineName = this.param.isPinnedEngine ? this.param.pinnedEngineName : searchService.defaultEngine.name; var menuLabel; var searchFormattedString = [ "contextMenuSearch", "contextMenuSearchText", "searchSelected" ]; var _this = this; function getMenuLabel(str) { try { if (_this.isSeaMonkey) { let bundle = document.getElementById("contentAreaCommandsBundle"); return bundle.getFormattedString(str, [engineName, selectedText]); } else return gNavigatorBundle.getFormattedString(str, [engineName, selectedText]); } catch (ex) { return false; } } for(let i in searchFormattedString) { menuLabel = getMenuLabel(searchFormattedString[i]); if (menuLabel) break; else if (!menuLabel && i == searchFormattedString.length-1) menuLabel = engineName; } menu.label = menuLabel; }, searchMenuPopup: function(e) { var popup = e.target; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } var menus; if (!popup.menus) { var itemsObj = popup.getElementsByTagName("menu"); popup.menus = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menu" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.menus.push(elem); }); menus = popup.menus; } else { menus = popup.menus; } var siteSearchDomain; var gBrowser = gBrowser || getBrowser(); function isDomain() { try { siteSearchDomain = gBrowser.currentURI.host; } catch (ex) { return false; } return true; } if (isDomain()) { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.setAttribute("tooltiptext", siteSearchDomain); elem.removeAttribute("disabled"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite")) elem.setAttribute("tooltiptext", siteSearchDomain); else if (elem.hasAttribute("searchInCache")) elem.setAttribute("tooltiptext", content.document.title); elem.removeAttribute("disabled"); }); } else { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); } }, createSearchMenu: function() { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; var engines = searchService.getVisibleEngines({}); menu.setAttribute("id", this.nodeIds.searchMenu); menu.setAttribute("class", "menu-iconic"); this.param.isPinnedEngine = false; this.param.pinnedEngineName = ""; var pinnedEngineName = options.pinnedEngineName; if (pinnedEngineName != "") { for (let i in engines) { if (engines[i].name != pinnedEngineName) continue; menu.setAttribute("label", engines[i].name); menu.setAttribute("image", engines[i].iconURI.spec); menu.engine = engines[i]; this.param.isPinnedEngine = true; this.param.pinnedEngineName = pinnedEngineName; break; } } if (!this.param.isPinnedEngine) { menu.setAttribute("label", searchService.defaultEngine.name); menu.setAttribute("image", searchService.defaultEngine.iconURI.spec); menu.engine = searchService.defaultEngine; } menu.gObj = this; menu.setAttribute("onclick", "if (event.target == this && event.target.engine) this.gObj.searchCommandHandler(event, 'menu');"); menu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchMenu.popup; popup.setAttribute("id", this.nodeIds.searchMenuPopup); var counterMaxLength = options.campactMenuLengthPerLine; if (typeof counterMaxLength != "number" || 3 > counterMaxLength || counterMaxLength > 10) counterMaxLength = 5; var counter = 0; var hBox; function itemCreateCampact(engine, last) { if(counter == 0) { hBox = document.createElementNS(XULNS, "hbox"); let item = document.createElementNS(XULNS, "spacer"); //item.setAttribute("style", "width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); } counter++; var item = document.createElementNS(XULNS, "menuitem"); item.engine = engine; item.setAttribute("class", "menuitem-iconic"); item.setAttribute("tooltiptext", engine.name); //item.setAttribute("src", engine.iconURI.spec); var image = document.createElementNS(XULNS, "image"); image.setAttribute("src", engine.iconURI.spec); item.appendChild(image); //item.setAttribute("style", "max-width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); if (counter == counterMaxLength || last) { popup.appendChild(hBox); counter = 0; } } var separator = {}; function itemCreate(engine, reason) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.engine = engine; if (reason && !separator[reason]) { /*let sep = separator[reason] = document.createElementNS(XULNS, "menuseparator"); sep.setAttribute(reason, "true"); popup.appendChild(sep); let item = document.createElementNS(XULNS, "label"); //item.setAttribute("class", "menuitem-iconic"); item.setAttribute("value", _localize(reason) + ":"); item.setAttribute("style", "padding-left: 2em"); popup.appendChild(item);*/ if (!separator["menuseparator"]) { let sep = separator["menuseparator"] = document.createElementNS(XULNS, "menuseparator"); popup.appendChild(sep); } let subMenu = separator[reason] = document.createElementNS(XULNS, "menu"); subMenu.setAttribute("class", "menu-iconic"); subMenu.setAttribute("label", _localize(reason)); subMenu.setAttribute(reason, "true"); subMenu.popup = subMenu.appendChild(document.createElementNS(XULNS, "menupopup")); popup.appendChild(subMenu); } if (reason == "searchBySite") { item.setAttribute("label", engine.name); if (options.searchBySite.iconsOn) { item.setAttribute("src", engine.iconURI.spec); } item.setAttribute(reason, "true"); } else if (reason == "searchImageByText" || reason == "searchInCache") { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI_spec); item.setAttribute(reason, "true"); } else { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI.spec); } //popup.appendChild(item); if (reason) { let subMenu = separator[reason]; subMenu.popup.appendChild(item); } else popup.appendChild(item); } for (let i in engines) { if (options.hideDefaultEngineInPopupSubMenu && this.param.isPinnedEngine && engines[i].name == pinnedEngineName || options.hideDefaultEngineInPopupSubMenu && !this.param.isPinnedEngine && engines[i].name == searchService.defaultEngine.name) { continue; } if (options.campactMenu) itemCreateCampact(engines[i], i == engines.length-1); else itemCreate(engines[i]); } function searchBySite_itemCreate() { var namesList = []; if (options.searchBySite.useDefaultNamesList) { let defaultNamesList = [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchBySite.addCustomNames && Array.isArray(options.searchBySite.customNamesList)) { namesList = namesList.concat(options.searchBySite.customNamesList); } //let i_skips = []; function isNameExist(names, engine) { for (let i in names) { if (names[i] != engine && i != names.length-1) continue; else if (names[i] == engine) return true; return false; } } namesList.forEach(function(elem) { for (let i in engines) { if (/*i_skips.indexOf(i) != -1 ||*/ !(elem[1].some(function(listName) listName == engines[i].name)) /*!isNameExist(elem[1], engines[i].name)*/ /*elem[1].indexOf(engines[i].name) == -1*/) continue; itemCreate(engines[i], "searchBySite"); //i_skips.push(i); break; } }); } if (options.searchBySite.enable) { searchBySite_itemCreate(); } function searchImageByText_itemCreate() { var namesList = []; if (options.searchImageByText.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.ru/search?tbm=isch&q=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?text=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchImageByText.addCustomNames && Array.isArray(options.searchImageByText.customNamesList)) { namesList = namesList.concat(options.searchImageByText.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.imageLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchImageByText"); }); } if (options.searchImageByText.enable) { searchImageByText_itemCreate(); } function searchInCache() { var namesList = []; if (options.searchInCache.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/search?q=cache:", "" ], [ "Archive.org", "http://web.archive.org/web/*/", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchInCache.addCustomNames && Array.isArray(options.searchInCache.customNamesList)) { namesList = namesList.concat(options.searchInCache.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.cacheLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchInCache"); }); } if (options.searchInCache.enable) { searchInCache(); } popup.gObj = this; //popup.setAttribute("oncommand", "if (!event.target.engine) return; var csObj = this.gObj || document.getElementById('contextSearchMiniByBunda1-context-popup').gObj; csObj.searchCommandHandler(event);"); popup.setAttribute("oncommand", "if (!event.target.engine) return; this.gObj.searchCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); }, searchByImageCommandHandler: function(event) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchByImageMenu = this.searchByImageMenu; var link = event.target.link; var imageLink = searchByImageMenu.imageSrc; if (imageLink.indexOf("data:") == 0) { this.searchByImageByData(imageLink, inBg); return; } else if (imageLink.indexOf("file:") == 0) { this.searchByImageByFile(imageLink, inBg); return; } else { imageLink = encodeURIComponent(imageLink) } var gBrowser = gBrowser || getBrowser(); gBrowser.loadOneTab(link + imageLink, { relatedToCurrent: true, inBackground: inBg }); }, // searchByImageByData() and searchByImageByFile() based on // Google Image Search 0.5 by Nishan Naseer // https://addons.mozilla.org/ru/firefox/addon/google-similar-images searchByImageByData: function(src, inBg) { var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byData(e) { newTabBrowser.removeEventListener(e.type, byData, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = src; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageByFile: function(src, inBg) { //Canvas to get contents in base-64 var canvas = gContextMenu.target.ownerDocument.createElementNS(XHTMLNS, "canvas"); var image = new Image(); image.src = src; canvas.width = image.width; canvas.height = image.height; var ctx = canvas.getContext("2d"); ctx.drawImage(image, 0, 0); var dataURL = canvas.toDataURL("image/png"); var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byFile(e) { newTabBrowser.removeEventListener(e.type, byFile, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = dataURL; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageMenuPopup: function(e) { var searchByImageMenu = this.searchByImageMenu; searchByImageMenu.hidden = !gContextMenu.onImage; var src = gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL; if (!src || !(/^((ht|f)tps?:\/\/|data:image|file:)/.test(src))) { searchByImageMenu.hidden = true; return; } //var popup = document.getElementById("contextSearchMiniByBunda1-context-image-search-popup"); var popup = searchByImageMenu.popup; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.getAttribute("label") != "Google") popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } if (/^(data:image|file:)/.test(src)) { items.forEach(function(elem) { elem.setAttribute("disabled", "true"); }); } else { items.forEach(function(elem) { elem.removeAttribute("disabled"); }); } searchByImageMenu.imageSrc = src; }, createSearchByImageMenu: function() { var menu = document.createElement("menu"); var insertMenu = document.getElementById("context-sep-copyimage"); insertMenu.parentNode.insertBefore(menu, insertMenu); this.searchByImageMenu = menu; menu.setAttribute("id", this.nodeIds.searchByImageMenu); menu.setAttribute("class", "menu-iconic"); menu.setAttribute("label", _localize("searchByImage")); menu.setAttribute("image", ""); this.searchByImageMenu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchByImageMenu.popup; popup.setAttribute("id", this.nodeIds.searchByImageMenuPopup); popup.googleURL = "https://www.google.com/searchbyimage"; var namesList = []; if (options.searchByImage.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/searchbyimage?image_url=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?rpt=imagedups&text=&img_url=", "" ], [ "TinEye", "http://www.tineye.com/search/?pluginver=firefox-1.0&url=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchByImage.addCustomNames && Array.isArray(options.searchByImage.customNamesList)) { namesList = namesList.concat(options.searchByImage.customNamesList); } function itemCreate(name, link, image) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.setAttribute("src", image); item.setAttribute("label", name); item.link = link; popup.appendChild(item); } namesList.forEach(function(elem) { itemCreate(elem[0], elem[1], elem[2]); }); var contextMenu = this.contextMenu; popup.gObj = this; popup.setAttribute("oncommand", "this.gObj.searchByImageCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); } } contextSearcherObj.init(); })(); |
Northtech > 09-10-2013 19:09:19 |
2k1dmg upd: что-то "на лету" в контекстном меню не меняется поисковик по умолчанию (если изменять выбор в панели поиска), а только после перезагрузки браузера. |
bunda1 > 09-10-2013 23:18:51 |
2k1dmg Выделить код Код:let os = Cc["@mozilla.org/observer-service;1"]. getService(Ci.nsIObserverService); os.addObserver(this, "browser-search-engine-modified", false); можно проще сделать: 09-10-2013 23:24:06 без: Выделить код Код:let XULNS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; let XHTMLNS = "http://www.w3.org/1999/xhtml" |
2k1dmg > 10-10-2013 08:52:40 |
Northtech Выделить код Код:(function () { // Context Search mini 2013-10-10 var options = { loadInBackground: false, // true - открывать вкладки в фоне pinnedEngineName: "", // оставить пустым если не нужно закреплять // определенный поисковик // примеры pinnedEngineName: "Google" или pinnedEngineName: "Яндекс" или pinnedEngineName: "" hideDefaultEngineInPopupSubMenu: false, // true - не добавлять поисковик по умолчанию в выпадающий список (подменю) поисковиков campactMenu: true, // true - поисковики в выпадающем списке (подменю) без назаваний campactMenuLengthPerLine: 5, // максимальное количество элементов в одной строке // от 3 до 10, по умолчанию 5 searchBySite: { enable: true, // true - добавить поисковики по сайту iconsOn: true, // true - добавить иконки для поисковиков по сайту useDefaultNamesList: true, // true - использовать список поисковиков по умолчанию addCustomNames: false, // true - добавить свои поисковики по сайту customNamesList: [ ["Rambler", ["Rambler", "Рамблер"]] ] // например ["Rambler", ["Rambler", "Рамблер"]] // первая ячека название на английском // вторая ячека массив с предполагаемыми именами поисковика // может состоять из нескольких значений // ["Name1", ["Name1"]], // ["Name2", ["Name2", "Имя2"]], // ["Name3", ["Name3", "Имя3", "Név3"]] }, searchImageByText: { enable: true, // true - добавить поиск изображения по тексту useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchByImage: { enable: true, // true - добавить поиск по изображению useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchInCache: { enable: true, // true - добавить поиск в кэше // нужно доработать useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] } }; function _localize(sid) { let strings = { en: { searchBySite: "Search this site", searchImageByText: "Search image by text", searchByImage: "Search by image", searchInCache: "Search this page in cache" }, ru: { searchBySite: "Поиск по этому сайту", searchImageByText: "Поиск изображения по тексту", searchByImage: "Поиск по изображению", searchInCache: "Поиск этой страницы в кэше" } }; //let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; function getBrowserUILocale() { // Browser UI locale return Components.classes["@mozilla.org/chrome/chrome-registry;1"] .getService(Components.interfaces.nsIXULChromeRegistry) .getSelectedLocale("global"); } let locale = ""; // ru, en if (!locale || locale == "") locale = getBrowserUILocale().match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } let XULNS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; let XHTMLNS = "http://www.w3.org/1999/xhtml" var contextSearcherObj = { initialized: false, init: function() { if(this.initialized) return; this.initialized = true; this.nodeIds = { searchMenu: "contextSearchMiniByBunda1-menu", searchMenuPopup: "contextSearchMiniByBunda1-popup", searchByImageMenu: "contextSearchMiniByBunda1-image-search-menu", searchByImageMenuPopup: "contextSearchMiniByBunda1-image-search-popup" }; if (document.getElementById(this.nodeIds.searchMenu)) return; var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var contextMenu = document.getElementById("contentAreaContextMenu"); var searchSelect = document.getElementById("context-searchselect"); searchSelect.style.display = "none"; var searchMenu = contextMenu.insertBefore(document.createElementNS(XULNS, "menu"), searchSelect); this.searchService = searchService; this.contextMenu = contextMenu; this.searchSelect = searchSelect; this.searchMenu = searchMenu; this.param = { isPinnedEngine: false, pinnedEngineName: "" }; try { this.createSearchMenu(); if (options.searchByImage.enable) this.createSearchByImageMenu(); } catch (ex) { this.destroy(); Components.utils.reportError(ex); return; } if (options.campactMenu) this.loadStyles(); var observeStatus = new MutationObserver(function () { searchMenu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); /*var prefService = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService); this.branch = prefService.getBranch("browser.search."); if (!("addObserver" in this.branch)) this.branch.QueryInterface(Components.interfaces.nsIPrefBranch2); this.branch.addObserver("", this, false);*/ /* var searchPref = "browser.search."; gPrefService.addObserver(searchPref, this, false);*/ /*let os = Cc["@mozilla.org/observer-service;1"]. getService(Ci.nsIObserverService); os.addObserver(this, "browser-search-engine-modified", false);*/ Services.obs.addObserver(this, "browser-search-engine-modified", false); Services.ww.registerNotification(this); addEventListener("popupshowing", this, false, contextMenu); addDestructor(function (reason) { observeStatus.disconnect(); Services.ww.unregisterNotification(this); //this.branch.removeObserver("", this); //gPrefService.removeObserver(searchPref, this, false); //os.removeObserver(this, "browser-search-engine-modified", false); Services.obs.removeObserver(this, "browser-search-engine-modified", false); this.destroy(); if (reason != "destructor") this.unloadStyles(); }, this); }, destroy: function() { if(!this.initialized) return; this.initialized = false; var contextMenu = this.contextMenu; if (!contextMenu) return; if (this.searchMenu) contextMenu.removeChild(this.searchMenu); if (this.searchByImageMenu) contextMenu.removeChild(this.searchByImageMenu); if (this.searchSelect) this.searchSelect.style.removeProperty("display"); }, update: function() { this.searchMenu.textContent = ""; this.createSearchMenu(); delete this.updateTimeoutID; }, get isSeaMonkey() { delete this.isSeaMonkey; return this.isSeaMonkey = Services.appinfo.name == "SeaMonkey"; }, observe: function(subject, topic, data) { switch (topic) { case "domwindowclosed": if (subject.document.documentElement.getAttribute("windowtype") == "Browser:SearchManager") { this.update(); } break; /*case "nsPref:changed": switch (data) { case "defaultenginename": case "browser.search.defaultenginename": this.update(); break; } break;*/ case "browser-search-engine-modified": /*switch (data) { case "engine-default": case "engine-current": // engine-default case "engine-changed": case "engine-removed":*/ if (typeof this.updateTimeoutID == "number") { window.clearTimeout(this.updateTimeoutID); delete this.updateTimeoutID; } let _this = this; this.updateTimeoutID = window.setTimeout(function () { _this.update(); }, 1000); /*break; }*/ break; } }, handleEvent: function(e) { //var contextMenu = this.contextMenu; //var menu = this.menu; //var imageMenu = this.imageMenu; switch (e.type) { case "popupshowing": this.popupshowingEvent(e); break; } }, popupshowingEvent: function(e) { var trgId = e.target.id; if (trgId) { if (trgId == "contentAreaContextMenu") { this.contextMenuPopup(e); if (this.searchByImageMenu) this.searchByImageMenuPopup(e); } else if (trgId == this.nodeIds.searchMenuPopup) this.searchMenuPopup(e); } }, _stylesLoaded: false, loadStyles: function() { if(this._stylesLoaded) return; this._stylesLoaded = true; var sss = this.sss; var cssURI = this.cssURI = this.makeCSSURI(); if(!sss.sheetRegistered(cssURI, sss.USER_SHEET)) sss.loadAndRegisterSheet(cssURI, sss.USER_SHEET); }, unloadStyles: function() { if(!this._stylesLoaded) return; this._stylesLoaded = false; var sss = this.sss; if(sss.sheetRegistered(this.cssURI, sss.USER_SHEET)) sss.unregisterSheet(this.cssURI, sss.USER_SHEET); }, get sss() { delete this.sss; return this.sss = Components.classes["@mozilla.org/content/style-sheet-service;1"] .getService(Components.interfaces.nsIStyleSheetService); }, makeCSSURI: function() { var cssStr = '\ @namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n\ @-moz-document url("' + window.location.href + '") {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ /*width: 2em !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] * {\n\ /*padding-left: 0 !important;\n\ padding-right: 0 !important;\n\ margin-left: 0 !important;\n\ margin-right: 0 !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n\ width: 2em !important;\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ width: 3em !important;\n\ height: 2em !important;\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox >.menuitem-iconic[campactMenu="true"] > image {\n\ margin-left: .8em !important;\n\ }\n\ }\n\ '; return Services.io.newURI("data:text/css," + encodeURIComponent(cssStr), null, null); }, searchCommandHandler: function(event, trg) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchBySite = ""; if (event.target.hasAttribute("searchBySite")) { searchBySite = " site:" + gBrowser.currentURI.host; } else if (event.target.hasAttribute("searchInCache")) { searchBySite = content.location.href; } let selectedText = this.isSeaMonkey ? content.getSelection().toString() : getBrowserSelection(); if (event.target.engine.imageLink) { gBrowser.loadOneTab(event.target.engine.imageLink + /*getBrowserSelection()*/ selectedText, { relatedToCurrent: true, inBackground: inBg }); } else if (event.target.engine.cacheLink) { gBrowser.loadOneTab(event.target.engine.cacheLink + /*getBrowserSelection()*/ searchBySite, { relatedToCurrent: true, inBackground: inBg }); } else { let submission = event.target.engine.getSubmission(/*getBrowserSelection()*/ selectedText + searchBySite, null); gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); } if (trg == "menu") { setTimeout(function () { document.getElementById("contentAreaContextMenu").hidePopup(); }, 0); } }, contextMenuPopup: function(e) { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; //var selectedText = getBrowserSelection(16); var selectedText = this.isSeaMonkey ? content.getSelection(16).toString() : getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (ex) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; let rEngine = this.isSeaMonkey ? "currentEngine" : "defaultEngine"; var engineName = this.param.isPinnedEngine ? this.param.pinnedEngineName : searchService[rEngine].name; var menuLabel; var searchFormattedString = [ "contextMenuSearch", "contextMenuSearchText", "searchSelected" ]; var thisIsSM = this.isSeaMonkey; function getMenuLabel(str) { try { if (thisIsSM) { let bundle = document.getElementById("contentAreaCommandsBundle"); return bundle.getFormattedString(str, [engineName, selectedText]); } else return gNavigatorBundle.getFormattedString(str, [engineName, selectedText]); } catch (ex) { return false; } } for(let i in searchFormattedString) { menuLabel = getMenuLabel(searchFormattedString[i]); if (menuLabel) break; else if (!menuLabel && i == searchFormattedString.length-1) menuLabel = engineName; } menu.label = menuLabel; }, searchMenuPopup: function(e) { var popup = e.target; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } var menus; if (!popup.menus) { var itemsObj = popup.getElementsByTagName("menu"); popup.menus = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menu" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.menus.push(elem); }); menus = popup.menus; } else { menus = popup.menus; } var siteSearchDomain; var gBrowser = gBrowser || getBrowser(); function isDomain() { try { siteSearchDomain = gBrowser.currentURI.host; } catch (ex) { return false; } return true; } if (isDomain()) { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.setAttribute("tooltiptext", siteSearchDomain); elem.removeAttribute("disabled"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite")) elem.setAttribute("tooltiptext", siteSearchDomain); else if (elem.hasAttribute("searchInCache")) elem.setAttribute("tooltiptext", content.document.title); elem.removeAttribute("disabled"); }); } else { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); } }, createSearchMenu: function() { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; var engines = searchService.getVisibleEngines({}); menu.setAttribute("id", this.nodeIds.searchMenu); menu.setAttribute("class", "menu-iconic"); this.param.isPinnedEngine = false; this.param.pinnedEngineName = ""; var pinnedEngineName = options.pinnedEngineName; if (pinnedEngineName != "") { for (let i in engines) { if (engines[i].name != pinnedEngineName) continue; menu.setAttribute("label", engines[i].name); menu.setAttribute("image", engines[i].iconURI.spec); menu.engine = engines[i]; this.param.isPinnedEngine = true; this.param.pinnedEngineName = pinnedEngineName; break; } } let rEngine = this.isSeaMonkey ? "currentEngine" : "defaultEngine"; if (!this.param.isPinnedEngine) { menu.setAttribute("label", searchService[rEngine].name); menu.setAttribute("image", searchService[rEngine].iconURI.spec); menu.engine = searchService[rEngine]; } menu.gObj = this; menu.setAttribute("onclick", "if (event.target == this && event.target.engine) this.gObj.searchCommandHandler(event, 'menu');"); menu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchMenu.popup; popup.setAttribute("id", this.nodeIds.searchMenuPopup); var counterMaxLength = options.campactMenuLengthPerLine; if (typeof counterMaxLength != "number" || 3 > counterMaxLength || counterMaxLength > 10) counterMaxLength = 5; var counter = 0; var hBox; function itemCreateCampact(engine, last) { if(counter == 0) { hBox = document.createElementNS(XULNS, "hbox"); let item = document.createElementNS(XULNS, "spacer"); //item.setAttribute("style", "width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); } counter++; var item = document.createElementNS(XULNS, "menuitem"); item.engine = engine; item.setAttribute("class", "menuitem-iconic"); item.setAttribute("tooltiptext", engine.name); //item.setAttribute("src", engine.iconURI.spec); var image = document.createElementNS(XULNS, "image"); image.setAttribute("src", engine.iconURI.spec); item.appendChild(image); //item.setAttribute("style", "max-width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); if (counter == counterMaxLength || last) { popup.appendChild(hBox); counter = 0; } } var separator = {}; function itemCreate(engine, reason) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.engine = engine; if (reason && !separator[reason]) { /*let sep = separator[reason] = document.createElementNS(XULNS, "menuseparator"); sep.setAttribute(reason, "true"); popup.appendChild(sep); let item = document.createElementNS(XULNS, "label"); //item.setAttribute("class", "menuitem-iconic"); item.setAttribute("value", _localize(reason) + ":"); item.setAttribute("style", "padding-left: 2em"); popup.appendChild(item);*/ if (!separator["menuseparator"]) { let sep = separator["menuseparator"] = document.createElementNS(XULNS, "menuseparator"); popup.appendChild(sep); } let subMenu = separator[reason] = document.createElementNS(XULNS, "menu"); subMenu.setAttribute("class", "menu-iconic"); subMenu.setAttribute("label", _localize(reason)); subMenu.setAttribute(reason, "true"); subMenu.popup = subMenu.appendChild(document.createElementNS(XULNS, "menupopup")); popup.appendChild(subMenu); } if (reason == "searchBySite") { item.setAttribute("label", engine.name); if (options.searchBySite.iconsOn) { item.setAttribute("src", engine.iconURI.spec); } item.setAttribute(reason, "true"); } else if (reason == "searchImageByText" || reason == "searchInCache") { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI_spec); item.setAttribute(reason, "true"); } else { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI.spec); } //popup.appendChild(item); if (reason) { let subMenu = separator[reason]; subMenu.popup.appendChild(item); } else popup.appendChild(item); } for (let i in engines) { if (options.hideDefaultEngineInPopupSubMenu && this.param.isPinnedEngine && engines[i].name == pinnedEngineName || options.hideDefaultEngineInPopupSubMenu && !this.param.isPinnedEngine && engines[i].name == searchService[rEngine].name) { continue; } if (options.campactMenu) itemCreateCampact(engines[i], i == engines.length-1); else itemCreate(engines[i]); } function searchBySite_itemCreate() { var namesList = []; if (options.searchBySite.useDefaultNamesList) { let defaultNamesList = [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchBySite.addCustomNames && Array.isArray(options.searchBySite.customNamesList)) { namesList = namesList.concat(options.searchBySite.customNamesList); } //let i_skips = []; function isNameExist(names, engine) { for (let i in names) { if (names[i] != engine && i != names.length-1) continue; else if (names[i] == engine) return true; return false; } } namesList.forEach(function(elem) { for (let i in engines) { if (/*i_skips.indexOf(i) != -1 ||*/ !(elem[1].some(function(listName) listName == engines[i].name)) /*!isNameExist(elem[1], engines[i].name)*/ /*elem[1].indexOf(engines[i].name) == -1*/) continue; itemCreate(engines[i], "searchBySite"); //i_skips.push(i); break; } }); } if (options.searchBySite.enable) { searchBySite_itemCreate(); } function searchImageByText_itemCreate() { var namesList = []; if (options.searchImageByText.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.ru/search?tbm=isch&q=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?text=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchImageByText.addCustomNames && Array.isArray(options.searchImageByText.customNamesList)) { namesList = namesList.concat(options.searchImageByText.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.imageLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchImageByText"); }); } if (options.searchImageByText.enable) { searchImageByText_itemCreate(); } function searchInCache() { var namesList = []; if (options.searchInCache.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/search?q=cache:", "" ], [ "Archive.org", "http://web.archive.org/web/*/", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchInCache.addCustomNames && Array.isArray(options.searchInCache.customNamesList)) { namesList = namesList.concat(options.searchInCache.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.cacheLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchInCache"); }); } if (options.searchInCache.enable) { searchInCache(); } popup.gObj = this; //popup.setAttribute("oncommand", "if (!event.target.engine) return; var csObj = this.gObj || document.getElementById('contextSearchMiniByBunda1-context-popup').gObj; csObj.searchCommandHandler(event);"); popup.setAttribute("oncommand", "if (!event.target.engine || event.target.nodeName == 'menu') return; this.gObj.searchCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); }, searchByImageCommandHandler: function(event) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchByImageMenu = this.searchByImageMenu; var link = event.target.link; var imageLink = searchByImageMenu.imageSrc; if (imageLink.indexOf("data:") == 0) { this.searchByImageByData(imageLink, inBg); return; } else if (imageLink.indexOf("file:") == 0) { this.searchByImageByFile(imageLink, inBg); return; } else { imageLink = encodeURIComponent(imageLink) } var gBrowser = gBrowser || getBrowser(); gBrowser.loadOneTab(link + imageLink, { relatedToCurrent: true, inBackground: inBg }); }, // searchByImageByData() and searchByImageByFile() based on // Google Image Search 0.5 by Nishan Naseer // https://addons.mozilla.org/ru/firefox/addon/google-similar-images searchByImageByData: function(src, inBg) { var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byData(e) { newTabBrowser.removeEventListener(e.type, byData, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = src; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageByFile: function(src, inBg) { //Canvas to get contents in base-64 var canvas = gContextMenu.target.ownerDocument.createElementNS(XHTMLNS, "canvas"); var image = new Image(); image.src = src; canvas.width = image.width; canvas.height = image.height; var ctx = canvas.getContext("2d"); ctx.drawImage(image, 0, 0); var dataURL = canvas.toDataURL("image/png"); var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byFile(e) { newTabBrowser.removeEventListener(e.type, byFile, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = dataURL; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageMenuPopup: function(e) { var searchByImageMenu = this.searchByImageMenu; searchByImageMenu.hidden = !gContextMenu.onImage; var src = gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL; if (!src || !(/^((ht|f)tps?:\/\/|data:image|file:)/.test(src))) { searchByImageMenu.hidden = true; return; } //var popup = document.getElementById("contextSearchMiniByBunda1-context-image-search-popup"); var popup = searchByImageMenu.popup; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.getAttribute("label") != "Google") popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } if (/^(data:image|file:)/.test(src)) { items.forEach(function(elem) { elem.setAttribute("disabled", "true"); }); } else { items.forEach(function(elem) { elem.removeAttribute("disabled"); }); } searchByImageMenu.imageSrc = src; }, createSearchByImageMenu: function() { var menu = document.createElement("menu"); var insertMenu = document.getElementById("context-sep-copyimage"); insertMenu.parentNode.insertBefore(menu, insertMenu); this.searchByImageMenu = menu; menu.setAttribute("id", this.nodeIds.searchByImageMenu); menu.setAttribute("class", "menu-iconic"); menu.setAttribute("label", _localize("searchByImage")); menu.setAttribute("image", ""); this.searchByImageMenu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchByImageMenu.popup; popup.setAttribute("id", this.nodeIds.searchByImageMenuPopup); popup.googleURL = "https://www.google.com/searchbyimage"; var namesList = []; if (options.searchByImage.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/searchbyimage?image_url=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?rpt=imagedups&text=&img_url=", "" ], [ "TinEye", "http://www.tineye.com/search/?pluginver=firefox-1.0&url=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchByImage.addCustomNames && Array.isArray(options.searchByImage.customNamesList)) { namesList = namesList.concat(options.searchByImage.customNamesList); } function itemCreate(name, link, image) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.setAttribute("src", image); item.setAttribute("label", name); item.link = link; popup.appendChild(item); } namesList.forEach(function(elem) { itemCreate(elem[0], elem[1], elem[2]); }); var contextMenu = this.contextMenu; popup.gObj = this; popup.setAttribute("oncommand", "this.gObj.searchByImageCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); } } contextSearcherObj.init(); })(); bunda1 |
Northtech > 10-10-2013 16:34:29 |
2k1dmg |
PEAKTOP > 10-10-2013 18:17:07 |
2k1dmg и bunda1 совместными усилиями вы сделали очень удобную и функциональную кнопку. БОЛЬШОЕ ВАМ СПАСИБО! |
villa7 > 10-10-2013 19:23:05 |
2k1dmg |
2k1dmg > 11-10-2013 10:35:21 |
PEAKTOP пишет
как-нибудь потом villa7 скрытый текст можно попробовать добавить после |
villa7 > 11-10-2013 20:31:01 |
2k1dmg
Спасибо, как и хотелось, только значение другое подставить пришлось. |
>ORG@niZM< > 12-12-2013 12:31:14 |
bunda1 пишет
отвалилось на ESR 24.2.0, но если убрать/закомментировать то работает, но в списке поисковиков не отображаются иконки. |
bunda1 > 12-12-2013 19:56:37 |
У меня Context Search на ESR 24.2.0 работает |
>ORG@niZM< > 13-12-2013 08:31:40 |
bunda1 |
bunda1 > 13-12-2013 09:23:15 |
Evybr пишет
Хорошо. |
rubel > 13-12-2013 17:09:42 |
bunda1 |
Kiril__777 > 13-12-2013 20:33:30 |
на моем фф вот так работает кнопка. |
2k1dmg > 14-12-2013 06:39:29 |
rubel Context Search mini 2013-11-22 Выделить код Код:(function () { // Context Search mini 2013-11-22 var options = { loadInBackground: false, // true - открывать вкладки в фоне pinnedEngineName: "", // оставить пустым если не нужно закреплять // определенный поисковик // примеры pinnedEngineName: "Google" или pinnedEngineName: "Яндекс" или pinnedEngineName: "" hideDefaultEngineInPopupSubMenu: false, // true - не добавлять поисковик по умолчанию в выпадающий список (подменю) поисковиков campactMenu: true, // true - поисковики в выпадающем списке (подменю) без назаваний campactMenuLengthPerLine: 5, // максимальное количество элементов в одной строке // от 3 до 10, по умолчанию 5 searchBySite: { enable: true, // true - добавить поисковики по сайту iconsOn: true, // true - добавить иконки для поисковиков по сайту useDefaultNamesList: true, // true - использовать список поисковиков по умолчанию addCustomNames: false, // true - добавить свои поисковики по сайту customNamesList: [ ["Rambler", ["Rambler", "Рамблер"]] ] // например ["Rambler", ["Rambler", "Рамблер"]] // первая ячека название на английском // вторая ячека массив с предполагаемыми именами поисковика // может состоять из нескольких значений // ["Name1", ["Name1"]], // ["Name2", ["Name2", "Имя2"]], // ["Name3", ["Name3", "Имя3", "Név3"]] }, searchImageByText: { enable: true, // true - добавить поиск изображения по тексту useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchByImage: { enable: true, // true - добавить поиск по изображению useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchInCache: { enable: true, // true - добавить поиск в кэше // нужно доработать useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] } }; function _localize(sid) { let strings = { en: { searchBySite: "Search this site", searchImageByText: "Search image by text", searchByImage: "Search by image", searchInCache: "Search this page in cache" }, ru: { searchBySite: "Поиск по этому сайту", searchImageByText: "Поиск изображения по тексту", searchByImage: "Поиск по изображению", searchInCache: "Поиск этой страницы в кэше" } }; //let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; function getBrowserUILocale() { let gPrefService = Services.prefs; // SeaMonkey if (!gPrefService.getBoolPref("intl.locale.matchOS")) { try { var locale = gPrefService.getCharPref("general.useragent.locale"); if (locale.substr(0, 9) == "chrome://") { return gPrefService.getComplexValue("general.useragent.locale", Components.interfaces.nsIPrefLocalizedString).data; } return locale; } catch (ex) {} } try { return Components.classes["@mozilla.org/chrome/chrome-registry;1"] .getService(Components.interfaces.nsIXULChromeRegistry) .getSelectedLocale("global"); } catch (ex) { return "en-US"; } } let locale = ""; // ru, en if (!locale) locale = getBrowserUILocale().match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } let XULNS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; let XHTMLNS = "http://www.w3.org/1999/xhtml"; var contextSearcherObj = { initialized: false, init: function() { if(this.initialized) return; this.initialized = true; this.nodeIds = { searchMenu: "contextSearchMiniByBunda1-menu", searchMenuPopup: "contextSearchMiniByBunda1-popup", searchByImageMenu: "contextSearchMiniByBunda1-image-search-menu", searchByImageMenuPopup: "contextSearchMiniByBunda1-image-search-popup" }; if (document.getElementById(this.nodeIds.searchMenu)) return; var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var contextMenu = document.getElementById("contentAreaContextMenu"); var searchSelect = document.getElementById("context-searchselect"); searchSelect.style.display = "none"; var searchMenu = contextMenu.insertBefore(document.createElementNS(XULNS, "menu"), searchSelect); this.searchService = searchService; this.contextMenu = contextMenu; this.searchSelect = searchSelect; this.searchMenu = searchMenu; this.param = { isPinnedEngine: false, pinnedEngineName: "" }; try { this.createSearchMenu(); if (options.searchByImage.enable) this.createSearchByImageMenu(); } catch (ex) { this.destroy(); Components.utils.reportError(ex); return; } if (options.campactMenu) this.loadStyles(); var observeStatus = new MutationObserver(function () { searchMenu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); /*var prefService = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService); this.branch = prefService.getBranch("browser.search."); if (!("addObserver" in this.branch)) this.branch.QueryInterface(Components.interfaces.nsIPrefBranch2); this.branch.addObserver("", this, false);*/ /* var searchPref = "browser.search."; gPrefService.addObserver(searchPref, this, false);*/ /*let os = Cc["@mozilla.org/observer-service;1"]. getService(Ci.nsIObserverService); os.addObserver(this, "browser-search-engine-modified", false);*/ Services.obs.addObserver(this, "browser-search-engine-modified", false); Services.ww.registerNotification(this); addEventListener("popupshowing", this, false, contextMenu); addDestructor(function (reason) { observeStatus.disconnect(); Services.ww.unregisterNotification(this); //this.branch.removeObserver("", this); //gPrefService.removeObserver(searchPref, this, false); //os.removeObserver(this, "browser-search-engine-modified", false); Services.obs.removeObserver(this, "browser-search-engine-modified", false); this.destroy(); if (reason != "destructor") this.unloadStyles(); }, this); }, destroy: function() { if(!this.initialized) return; this.initialized = false; var contextMenu = this.contextMenu; if (!contextMenu) return; if (this.searchMenu) contextMenu.removeChild(this.searchMenu); if (this.searchByImageMenu) contextMenu.removeChild(this.searchByImageMenu); if (this.searchSelect) this.searchSelect.style.removeProperty("display"); }, update: function() { this.searchMenu.textContent = ""; this.createSearchMenu(); delete this.updateTimeoutID; }, get isSeaMonkey() { delete this.isSeaMonkey; return this.isSeaMonkey = Services.appinfo.name == "SeaMonkey"; }, observe: function(subject, topic, data) { switch (topic) { /*case "domwindowclosed": if (subject.document.documentElement.getAttribute("windowtype") == "Browser:SearchManager") { this.update(); } break;*/ /*case "nsPref:changed": switch (data) { case "defaultenginename": case "browser.search.defaultenginename": this.update(); break; } break;*/ case "browser-search-engine-modified": /*switch (data) { case "engine-default": case "engine-current": // engine-default case "engine-changed": case "engine-removed":*/ if (typeof this.updateTimeoutID == "number") { window.clearTimeout(this.updateTimeoutID); delete this.updateTimeoutID; } let _this = this; this.updateTimeoutID = window.setTimeout(function () { _this.update(); }, 1000); /*break; }*/ break; } }, handleEvent: function(e) { //var contextMenu = this.contextMenu; //var menu = this.menu; //var imageMenu = this.imageMenu; switch (e.type) { case "popupshowing": this.popupshowingEvent(e); break; } }, popupshowingEvent: function(e) { var trgId = e.target.id; if (trgId) { if (trgId == "contentAreaContextMenu") { this.contextMenuPopup(e); if (this.searchByImageMenu) this.searchByImageMenuPopup(e); } else if (trgId == this.nodeIds.searchMenuPopup) this.searchMenuPopup(e); } }, _stylesLoaded: false, loadStyles: function() { if(this._stylesLoaded) return; this._stylesLoaded = true; var sss = this.sss; var cssURI = this.cssURI = this.makeCSSURI(); if(!sss.sheetRegistered(cssURI, sss.USER_SHEET)) sss.loadAndRegisterSheet(cssURI, sss.USER_SHEET); }, unloadStyles: function() { if(!this._stylesLoaded) return; this._stylesLoaded = false; var sss = this.sss; if(sss.sheetRegistered(this.cssURI, sss.USER_SHEET)) sss.unregisterSheet(this.cssURI, sss.USER_SHEET); }, get sss() { delete this.sss; return this.sss = Components.classes["@mozilla.org/content/style-sheet-service;1"] .getService(Components.interfaces.nsIStyleSheetService); }, makeCSSURI: function() { var cssStr = '\ @namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n\ @-moz-document url("' + window.location.href + '") {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ /*width: 2em !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] * {\n\ /*padding-left: 0 !important;\n\ padding-right: 0 !important;\n\ margin-left: 0 !important;\n\ margin-right: 0 !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n\ width: /*2em*/26px !important;\n\ @media (min-resolution: 2dppx) {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n\ width: 52px !important;\n\ }\n\ }\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ /*width: 3em !important;\n\ height: 2em !important;*/\n\ width: 36px !important;\n\ height: 26px !important;\n\ -moz-box-pack: center !important;\n\ @media (min-resolution: 2dppx) {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ width: 72px !important;\n\ height: 52px !important;\n\ }\n\ }\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox >.menuitem-iconic[campactMenu="true"] > image {\n\ /*margin-left: .8em !important;*/\n\ }\n\ }\n\ '; return Services.io.newURI("data:text/css," + encodeURIComponent(cssStr), null, null); }, searchCommandHandler: function(event, trg) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchBySite = ""; if (event.target.hasAttribute("searchBySite")) { searchBySite = " site:" + gBrowser.currentURI.host; } else if (event.target.hasAttribute("searchInCache")) { searchBySite = content.location.href; } let selectedText = this.isSeaMonkey ? content.getSelection().toString() : getBrowserSelection(); if (event.target.engine.imageLink) { gBrowser.loadOneTab(event.target.engine.imageLink + /*getBrowserSelection()*/ selectedText, { relatedToCurrent: true, inBackground: inBg }); } else if (event.target.engine.cacheLink) { gBrowser.loadOneTab(event.target.engine.cacheLink + /*getBrowserSelection()*/ searchBySite, { relatedToCurrent: true, inBackground: inBg }); } else { let submission = event.target.engine.getSubmission(/*getBrowserSelection()*/ selectedText + searchBySite, null); gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); } if (trg == "menu") { setTimeout(function () { document.getElementById("contentAreaContextMenu").hidePopup(); }, 0); } }, contextMenuPopup: function(e) { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; //var selectedText = getBrowserSelection(16); var selectedText = this.isSeaMonkey ? content.getSelection(16).toString() : getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (ex) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; let rEngine = this.isSeaMonkey ? "currentEngine" : "defaultEngine"; var engineName = this.param.isPinnedEngine ? this.param.pinnedEngineName : searchService[rEngine].name; var menuLabel; var searchFormattedString = [ "contextMenuSearch", "contextMenuSearchText", "searchSelected" ]; var thisIsSM = this.isSeaMonkey; function getMenuLabel(str) { try { if (thisIsSM) { let bundle = document.getElementById("contentAreaCommandsBundle"); return bundle.getFormattedString(str, [engineName, selectedText]); } else return gNavigatorBundle.getFormattedString(str, [engineName, selectedText]); } catch (ex) { return false; } } for(let i in searchFormattedString) { menuLabel = getMenuLabel(searchFormattedString[i]); if (menuLabel) break; else if (!menuLabel && i == searchFormattedString.length-1) menuLabel = engineName; } menu.label = menuLabel; }, searchMenuPopup: function(e) { var popup = e.target; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } var menus; if (!popup.menus) { var itemsObj = popup.getElementsByTagName("menu"); popup.menus = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menu" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.menus.push(elem); }); menus = popup.menus; } else { menus = popup.menus; } var siteSearchDomain; var gBrowser = gBrowser || getBrowser(); function isDomain() { try { siteSearchDomain = gBrowser.currentURI.host; } catch (ex) { return false; } return true; } if (isDomain()) { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.setAttribute("tooltiptext", siteSearchDomain); elem.removeAttribute("disabled"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite")) elem.setAttribute("tooltiptext", siteSearchDomain); else if (elem.hasAttribute("searchInCache")) elem.setAttribute("tooltiptext", content.document.title); elem.removeAttribute("disabled"); }); } else { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); } }, createSearchMenu: function() { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; var engines = searchService.getVisibleEngines({}); menu.setAttribute("id", this.nodeIds.searchMenu); menu.setAttribute("class", "menu-iconic"); this.param.isPinnedEngine = false; this.param.pinnedEngineName = ""; var pinnedEngineName = options.pinnedEngineName; if (pinnedEngineName != "") { for (let i in engines) { if (engines[i].name != pinnedEngineName) continue; menu.setAttribute("label", engines[i].name); menu.setAttribute("image", engines[i].iconURI.spec); menu.engine = engines[i]; this.param.isPinnedEngine = true; this.param.pinnedEngineName = pinnedEngineName; break; } } let rEngine = this.isSeaMonkey ? "currentEngine" : "defaultEngine"; if (!this.param.isPinnedEngine) { menu.setAttribute("label", searchService[rEngine].name); menu.setAttribute("image", searchService[rEngine].iconURI.spec); menu.engine = searchService[rEngine]; } menu.gObj = this; menu.setAttribute("onclick", "if (event.target == this && event.target.engine) this.gObj.searchCommandHandler(event, 'menu');"); menu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchMenu.popup; popup.setAttribute("id", this.nodeIds.searchMenuPopup); var counterMaxLength = options.campactMenuLengthPerLine; if (typeof counterMaxLength != "number" || 3 > counterMaxLength || counterMaxLength > 10) counterMaxLength = 5; var counter = 0; var hBox; function itemCreateCampact(engine, last) { if(counter == 0) { hBox = document.createElementNS(XULNS, "hbox"); let item = document.createElementNS(XULNS, "spacer"); //item.setAttribute("style", "width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); } counter++; var item = document.createElementNS(XULNS, "menuitem"); item.engine = engine; item.setAttribute("class", "menuitem-iconic"); item.setAttribute("tooltiptext", engine.name); //item.setAttribute("src", engine.iconURI.spec); var image = document.createElementNS(XULNS, "image"); image.setAttribute("src", engine.iconURI.spec); item.appendChild(image); //item.setAttribute("style", "max-width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); if (counter == counterMaxLength || last) { popup.appendChild(hBox); counter = 0; } } var separator = {}; function itemCreate(engine, reason) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.engine = engine; if (reason && !separator[reason]) { /*let sep = separator[reason] = document.createElementNS(XULNS, "menuseparator"); sep.setAttribute(reason, "true"); popup.appendChild(sep); let item = document.createElementNS(XULNS, "label"); //item.setAttribute("class", "menuitem-iconic"); item.setAttribute("value", _localize(reason) + ":"); item.setAttribute("style", "padding-left: 2em"); popup.appendChild(item);*/ if (!separator["menuseparator"]) { let sep = separator["menuseparator"] = document.createElementNS(XULNS, "menuseparator"); popup.appendChild(sep); } let subMenu = separator[reason] = document.createElementNS(XULNS, "menu"); subMenu.setAttribute("class", "menu-iconic"); subMenu.setAttribute("label", _localize(reason)); subMenu.setAttribute(reason, "true"); subMenu.popup = subMenu.appendChild(document.createElementNS(XULNS, "menupopup")); popup.appendChild(subMenu); } if (reason == "searchBySite") { item.setAttribute("label", engine.name); if (options.searchBySite.iconsOn) { item.setAttribute("src", engine.iconURI.spec); } item.setAttribute(reason, "true"); } else if (reason == "searchImageByText" || reason == "searchInCache") { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI_spec); item.setAttribute(reason, "true"); } else { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI.spec); } //popup.appendChild(item); if (reason) { let subMenu = separator[reason]; subMenu.popup.appendChild(item); } else popup.appendChild(item); } for (let i in engines) { if (options.hideDefaultEngineInPopupSubMenu && this.param.isPinnedEngine && engines[i].name == pinnedEngineName || options.hideDefaultEngineInPopupSubMenu && !this.param.isPinnedEngine && engines[i].name == searchService[rEngine].name) { continue; } if (options.campactMenu) itemCreateCampact(engines[i], i == engines.length-1); else itemCreate(engines[i]); } function searchBySite_itemCreate() { var namesList = []; if (options.searchBySite.useDefaultNamesList) { let defaultNamesList = [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchBySite.addCustomNames && Array.isArray(options.searchBySite.customNamesList)) { namesList = namesList.concat(options.searchBySite.customNamesList); } //let i_skips = []; function isNameExist(names, engine) { for (let i in names) { if (names[i] != engine && i != names.length-1) continue; else if (names[i] == engine) return true; return false; } } namesList.forEach(function(elem) { for (let i in engines) { if (/*i_skips.indexOf(i) != -1 ||*/ !(elem[1].some(function(listName) listName == engines[i].name)) /*!isNameExist(elem[1], engines[i].name)*/ /*elem[1].indexOf(engines[i].name) == -1*/) continue; itemCreate(engines[i], "searchBySite"); //i_skips.push(i); break; } }); } if (options.searchBySite.enable) { searchBySite_itemCreate(); } function searchImageByText_itemCreate() { var namesList = []; if (options.searchImageByText.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.ru/search?tbm=isch&q=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?text=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchImageByText.addCustomNames && Array.isArray(options.searchImageByText.customNamesList)) { namesList = namesList.concat(options.searchImageByText.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.imageLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchImageByText"); }); } if (options.searchImageByText.enable) { searchImageByText_itemCreate(); } function searchInCache() { var namesList = []; if (options.searchInCache.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/search?q=cache:", "" ], [ "Archive.org", "http://web.archive.org/web/*/", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchInCache.addCustomNames && Array.isArray(options.searchInCache.customNamesList)) { namesList = namesList.concat(options.searchInCache.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.cacheLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchInCache"); }); } if (options.searchInCache.enable) { searchInCache(); } popup.gObj = this; //popup.setAttribute("oncommand", "if (!event.target.engine) return; var csObj = this.gObj || document.getElementById('contextSearchMiniByBunda1-context-popup').gObj; csObj.searchCommandHandler(event);"); popup.setAttribute("oncommand", "if (!event.target.engine || event.target.nodeName == 'menu') return; this.gObj.searchCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); }, searchByImageCommandHandler: function(event) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchByImageMenu = this.searchByImageMenu; var link = event.target.link; var imageLink = searchByImageMenu.imageSrc; if (imageLink.indexOf("data:") == 0) { this.searchByImageByData(imageLink, inBg); return; } else if (imageLink.indexOf("file:") == 0) { this.searchByImageByFile(imageLink, inBg); return; } else { imageLink = encodeURIComponent(imageLink) } var gBrowser = gBrowser || getBrowser(); gBrowser.loadOneTab(link + imageLink, { relatedToCurrent: true, inBackground: inBg }); }, // searchByImageByData() and searchByImageByFile() based on // Google Image Search 0.5 by Nishan Naseer // https://addons.mozilla.org/ru/firefox/addon/google-similar-images searchByImageByData: function(src, inBg) { var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byData(e) { newTabBrowser.removeEventListener(e.type, byData, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = src; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageByFile: function(src, inBg) { //Canvas to get contents in base-64 var canvas = gContextMenu.target.ownerDocument.createElementNS(XHTMLNS, "canvas"); var image = new Image(); image.src = src; canvas.width = image.width; canvas.height = image.height; var ctx = canvas.getContext("2d"); ctx.drawImage(image, 0, 0); var dataURL = canvas.toDataURL("image/png"); var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byFile(e) { newTabBrowser.removeEventListener(e.type, byFile, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = dataURL; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageMenuPopup: function(e) { var searchByImageMenu = this.searchByImageMenu; searchByImageMenu.hidden = !gContextMenu.onImage; var src = gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL; if (!src || !(/^((ht|f)tps?:\/\/|data:image|file:)/.test(src))) { searchByImageMenu.hidden = true; return; } //var popup = document.getElementById("contextSearchMiniByBunda1-context-image-search-popup"); var popup = searchByImageMenu.popup; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.getAttribute("label") != "Google") popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } if (/^(data:image|file:)/.test(src)) { items.forEach(function(elem) { elem.setAttribute("disabled", "true"); }); } else { items.forEach(function(elem) { elem.removeAttribute("disabled"); }); } searchByImageMenu.imageSrc = src; }, createSearchByImageMenu: function() { var menu = document.createElement("menu"); var insertMenu = document.getElementById("context-sep-copyimage"); insertMenu.parentNode.insertBefore(menu, insertMenu); this.searchByImageMenu = menu; menu.setAttribute("id", this.nodeIds.searchByImageMenu); menu.setAttribute("class", "menu-iconic"); menu.setAttribute("label", _localize("searchByImage")); menu.setAttribute("image", ""); this.searchByImageMenu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchByImageMenu.popup; popup.setAttribute("id", this.nodeIds.searchByImageMenuPopup); popup.googleURL = "https://www.google.com/searchbyimage"; var namesList = []; if (options.searchByImage.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/searchbyimage?image_url=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?rpt=imagedups&text=&img_url=", "" ], [ "TinEye", "http://www.tineye.com/search/?pluginver=firefox-1.0&url=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchByImage.addCustomNames && Array.isArray(options.searchByImage.customNamesList)) { namesList = namesList.concat(options.searchByImage.customNamesList); } function itemCreate(name, link, image) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.setAttribute("src", image); item.setAttribute("label", name); item.link = link; popup.appendChild(item); } namesList.forEach(function(elem) { itemCreate(elem[0], elem[1], elem[2]); }); var contextMenu = this.contextMenu; popup.gObj = this; popup.setAttribute("oncommand", "this.gObj.searchByImageCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); } } contextSearcherObj.init(); })(); |
Kiril__777 > 14-12-2013 08:21:48 |
2k1dmg |
rubel > 14-12-2013 08:47:44 |
2k1dmg |
villi311 > 14-12-2013 09:04:28 |
2k1dmg пишет
Мне ваш вариант понравился больше. Огромное спасибо. |
Ader > 15-12-2013 14:45:14 |
villa7 пишет
А можно узнать, какое значение Вы поставили, и как это выглядит? |
bunda1 > 15-12-2013 15:42:36 |
2k1dmg |
villa7 > 15-12-2013 18:02:01 |
Ader |
LBra > 20-12-2013 09:07:54 |
2k1dmg пишет
2k1dmg, Я добавил: Пожелания: |
1fesFFFF > 22-01-2014 16:43:08 |
bunda1 |
bunda1 > 22-01-2014 23:02:32 |
Что значит не срабатывает поиск, когда из контекстного меню я щёлкаю по другому поисковику, вообще ничего не происходит или что? |
feas > 22-01-2014 23:21:37 |
bunda1 пишет
Возможно именно это имеет ввиду, однако я не вижу такой проблемы, хоть убей и левой и правой и все поисковики перебрал. 22-01-2014 23:22:01 |
bunda1 > 22-01-2014 23:33:44 |
feas |
1fesFFFF > 23-01-2014 04:12:35 |
Да, ничего не происходит. Ладно, буду дальше копать. |
>ORG@niZM< > 23-01-2014 06:40:24 |
1fesFFFF |
1fesFFFF > 29-01-2014 17:35:55 |
В 90% случаев не срабатывает при таком алгоритме: открывается список поисковиков, нажимаю левой кнопкой по одному из них, но кнопку не отпускаю, потом вожу курсор вверх-вниз (ищу нужный) и отпускаю кнопку напротив того, который нужен, но не над тем, над которым я зажал кнопку. В итоге не срабатывает поиск. |
bunda1 > 29-01-2014 20:33:33 |
1fesFFFF пишет
Попробуй: Выделить код Код:// Context Search, от 29.01.2014. ................................ (function () { var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var searchSelect = document.getElementById('context-searchselect'); searchSelect.collapsed = true; // удалить стандартный пункт меню для поиска // Создать новый пункт меню для поиска .... var contextMenu = document.getElementById("contentAreaContextMenu"); var menu = contextMenu.insertBefore( document.createElement('menu'), searchSelect ); // над каким пунктом меню показывать menu.setAttribute("class", "menu-iconic"); // устанавливать иконку, название и поисковик для нового пункта меню function setMenu() { menu.engine = searchService.currentEngine; menu.setAttribute("label", "Искать в " + menu.engine.name + " или в ..."); menu.setAttribute("image", menu.engine.iconURI.spec ); }; setMenu(); // наблюдатель за стандартным пунктом меню 'Копировать' прячет меню поиска var copy = document.getElementById('context-copy'); var setHiddenMenu = new MutationObserver(function() { menu.hidden = copy.hidden || copy.disabled; }); setHiddenMenu.observe( copy, { attributes: true, attributeFilter: ["hidden", "disabled"] } ); // Создать подменю с поисковиками .... var menuPopup = menu.appendChild( document.createElement("menupopup") ); menuPopup.setAttribute('style', 'overflow: scroll'); // создать пункты в подменю function setItemsToMenuPopup(e) { menuPopup.textContent = ""; var engines = searchService.getVisibleEngines({}); engines.forEach(function( engine ) { var mItem = document.createElement("menuitem"); mItem.setAttribute("label", engine.name ); mItem.setAttribute("class", "menuitem-iconic"); mItem.setAttribute("src", engine.iconURI.spec ); mItem.engine = engine; menuPopup.appendChild( mItem ); }); }; setItemsToMenuPopup(); // Установить действие для клика на меню и подменю .... menu.setAttribute("onmouseup", "\ var background = ( event.button == 0 ) ? false : true;\ var clip = gClipboard.read();\ goDoCommand('cmd_copy');\ setTimeout(function() {\ document.getElementById('contentAreaContextMenu').hidePopup();\ var submission = event.target.engine.getSubmission( gClipboard.read(), null );\ gBrowser.loadOneTab( submission.uri.spec, null, null, submission.postData, background, false );\ gClipboard.write( clip );\ }, 0);\ "); // Наблюдатель за изменениями в поисковиках пересоздаёт меню и подменю .... var getEngineModified = { observe: function(subject, topic, data) { if ( /changed|removed|current/.test( data ) ) { setMenu(); setItemsToMenuPopup() }; } }; Services.obs.addObserver( getEngineModified, "browser-search-engine-modified", false ); // Удалять наблюдатели и меню, показать стандартный пункт .... addDestructor(function() { contextMenu.removeChild( menu ); setHiddenMenu.disconnect(); Services.obs.removeObserver( getEngineModified, "browser-search-engine-modified", false ); searchSelect.collapsed = false; }); })(); |
1fesFFFF > 30-01-2014 18:47:08 |
bunda1
У меня так было, когда я кнопку поместил в строку заголовка (на самый верх). Но когда переместил эту кнопку вниз (панель дополнений), то всё стало как надо - один пункт в меню. |
mssign > 09-02-2015 21:21:30 |
Не работает. В меню присутствует, но по клику на поисковике ничего не происходит. firefox 35.0.1 |
bunda1 > 09-02-2015 22:33:21 |
mssign пишет
Сейчас проверил и всё работает, возможно конфликт с установленными расширениями, попробуй их отключать по одному и проверять работу кода. |
mssign > 09-02-2015 23:15:20 |
bunda1 |
Northtech > 31-05-2015 17:42:55 |
2k1dmg Выделить код Код:// Скрыть кнопку с панели. self.style.display = 'none'; (function () { // Context Search mini 2013-11-22 var options = { loadInBackground: true, // true - открывать вкладки в фоне pinnedEngineName: "", // оставить пустым если не нужно закреплять // определенный поисковик // примеры pinnedEngineName: "Google" или pinnedEngineName: "Яндекс" или pinnedEngineName: "" hideDefaultEngineInPopupSubMenu: true, // true - не добавлять поисковик по умолчанию в выпадающий список (подменю) поисковиков campactMenu: false, // true - поисковики в выпадающем списке (подменю) без назаваний campactMenuLengthPerLine: 5, // максимальное количество элементов в одной строке // от 3 до 10, по умолчанию 5 searchBySite: { enable: true, // true - добавить поисковики по сайту iconsOn: true, // true - добавить иконки для поисковиков по сайту useDefaultNamesList: true, // true - использовать список поисковиков по умолчанию addCustomNames: false, // true - добавить свои поисковики по сайту customNamesList: [ ["Rambler", ["Rambler", "Рамблер"]] ] // например ["Rambler", ["Rambler", "Рамблер"]] // первая ячека название на английском // вторая ячека массив с предполагаемыми именами поисковика // может состоять из нескольких значений // ["Name1", ["Name1"]], // ["Name2", ["Name2", "Имя2"]], // ["Name3", ["Name3", "Имя3", "Név3"]] }, searchImageByText: { enable: true, // true - добавить поиск изображения по тексту useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchByImage: { enable: true, // true - добавить поиск по изображению useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] }, searchInCache: { enable: true, // true - добавить поиск в кэше // нужно доработать useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ "name", "link", "image" ] ] } }; function _localize(sid) { let strings = { en: { searchBySite: "Search this site", searchImageByText: "Search image by text", searchByImage: "Search by image", searchInCache: "Search this page in cache" }, ru: { searchBySite: "Поиск по этому сайту", searchImageByText: "Поиск изображения по тексту", searchByImage: "Поиск по изображению", searchInCache: "Поиск этой страницы в кэше" } }; //let locale = (Application.prefs.getValue("general.useragent.locale", false) || "en").match(/^[a-z]*/)[0]; function getBrowserUILocale() { let gPrefService = Services.prefs; // SeaMonkey if (!gPrefService.getBoolPref("intl.locale.matchOS")) { try { var locale = gPrefService.getCharPref("general.useragent.locale"); if (locale.substr(0, 9) == "chrome://") { return gPrefService.getComplexValue("general.useragent.locale", Components.interfaces.nsIPrefLocalizedString).data; } return locale; } catch (ex) {} } try { return Components.classes["@mozilla.org/chrome/chrome-registry;1"] .getService(Components.interfaces.nsIXULChromeRegistry) .getSelectedLocale("global"); } catch (ex) { return "en-US"; } } let locale = ""; // ru, en if (!locale) locale = getBrowserUILocale().match(/^[a-z]*/)[0]; _localize = function (sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize.apply(this, arguments); } let XULNS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"; let XHTMLNS = "http://www.w3.org/1999/xhtml"; var contextSearcherObj = { initialized: false, init: function() { if(this.initialized) return; this.initialized = true; this.nodeIds = { searchMenu: "contextSearchMiniByBunda1-menu", searchMenuPopup: "contextSearchMiniByBunda1-popup", searchByImageMenu: "contextSearchMiniByBunda1-image-search-menu", searchByImageMenuPopup: "contextSearchMiniByBunda1-image-search-popup" }; if (document.getElementById(this.nodeIds.searchMenu)) return; var searchService = Cc["@mozilla.org/browser/search-service;1"].getService(Ci.nsIBrowserSearchService); var contextMenu = document.getElementById("contentAreaContextMenu"); var searchSelect = document.getElementById("context-searchselect"); searchSelect.style.display = "none"; var searchMenu = contextMenu.insertBefore(document.createElementNS(XULNS, "menu"), searchSelect); this.searchService = searchService; this.contextMenu = contextMenu; this.searchSelect = searchSelect; this.searchMenu = searchMenu; this.param = { isPinnedEngine: false, pinnedEngineName: "" }; try { this.createSearchMenu(); if (options.searchByImage.enable) this.createSearchByImageMenu(); } catch (ex) { this.destroy(); Components.utils.reportError(ex); return; } if (options.campactMenu) this.loadStyles(); var observeStatus = new MutationObserver(function () { searchMenu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ["hidden"] }); /*var prefService = Components.classes["@mozilla.org/preferences-service;1"] .getService(Components.interfaces.nsIPrefService); this.branch = prefService.getBranch("browser.search."); if (!("addObserver" in this.branch)) this.branch.QueryInterface(Components.interfaces.nsIPrefBranch2); this.branch.addObserver("", this, false);*/ /* var searchPref = "browser.search."; gPrefService.addObserver(searchPref, this, false);*/ /*let os = Cc["@mozilla.org/observer-service;1"]. getService(Ci.nsIObserverService); os.addObserver(this, "browser-search-engine-modified", false);*/ Services.obs.addObserver(this, "browser-search-engine-modified", false); Services.ww.registerNotification(this); addEventListener("popupshowing", this, false, contextMenu); addDestructor(function (reason) { observeStatus.disconnect(); Services.ww.unregisterNotification(this); //this.branch.removeObserver("", this); //gPrefService.removeObserver(searchPref, this, false); //os.removeObserver(this, "browser-search-engine-modified", false); Services.obs.removeObserver(this, "browser-search-engine-modified", false); this.destroy(); if (reason != "destructor") this.unloadStyles(); }, this); }, destroy: function() { if(!this.initialized) return; this.initialized = false; var contextMenu = this.contextMenu; if (!contextMenu) return; if (this.searchMenu) contextMenu.removeChild(this.searchMenu); if (this.searchByImageMenu) contextMenu.removeChild(this.searchByImageMenu); if (this.searchSelect) this.searchSelect.style.removeProperty("display"); }, update: function() { this.searchMenu.textContent = ""; this.createSearchMenu(); delete this.updateTimeoutID; }, get isSeaMonkey() { delete this.isSeaMonkey; return this.isSeaMonkey = Services.appinfo.name == "SeaMonkey"; }, observe: function(subject, topic, data) { switch (topic) { /*case "domwindowclosed": if (subject.document.documentElement.getAttribute("windowtype") == "Browser:SearchManager") { this.update(); } break;*/ /*case "nsPref:changed": switch (data) { case "defaultenginename": case "browser.search.defaultenginename": this.update(); break; } break;*/ case "browser-search-engine-modified": /*switch (data) { case "engine-default": case "engine-current": // engine-default case "engine-changed": case "engine-removed":*/ if (typeof this.updateTimeoutID == "number") { window.clearTimeout(this.updateTimeoutID); delete this.updateTimeoutID; } let _this = this; this.updateTimeoutID = window.setTimeout(function () { _this.update(); }, 1000); /*break; }*/ break; } }, handleEvent: function(e) { //var contextMenu = this.contextMenu; //var menu = this.menu; //var imageMenu = this.imageMenu; switch (e.type) { case "popupshowing": this.popupshowingEvent(e); break; } }, popupshowingEvent: function(e) { var trgId = e.target.id; if (trgId) { if (trgId == "contentAreaContextMenu") { this.contextMenuPopup(e); if (this.searchByImageMenu) this.searchByImageMenuPopup(e); } else if (trgId == this.nodeIds.searchMenuPopup) this.searchMenuPopup(e); } }, _stylesLoaded: false, loadStyles: function() { if(this._stylesLoaded) return; this._stylesLoaded = true; var sss = this.sss; var cssURI = this.cssURI = this.makeCSSURI(); if(!sss.sheetRegistered(cssURI, sss.USER_SHEET)) sss.loadAndRegisterSheet(cssURI, sss.USER_SHEET); }, unloadStyles: function() { if(!this._stylesLoaded) return; this._stylesLoaded = false; var sss = this.sss; if(sss.sheetRegistered(this.cssURI, sss.USER_SHEET)) sss.unregisterSheet(this.cssURI, sss.USER_SHEET); }, get sss() { delete this.sss; return this.sss = Components.classes["@mozilla.org/content/style-sheet-service;1"] .getService(Components.interfaces.nsIStyleSheetService); }, makeCSSURI: function() { var cssStr = '\ @namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n\ @-moz-document url("' + window.location.href + '") {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ /*width: 2em !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] * {\n\ /*padding-left: 0 !important;\n\ padding-right: 0 !important;\n\ margin-left: 0 !important;\n\ margin-right: 0 !important;*/\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n\ width: /*2em*/26px !important;\n\ @media (min-resolution: 2dppx) {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n\ width: 52px !important;\n\ }\n\ }\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ /*width: 3em !important;\n\ height: 2em !important;*/\n\ width: 36px !important;\n\ height: 26px !important;\n\ -moz-box-pack: center !important;\n\ @media (min-resolution: 2dppx) {\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n\ width: 72px !important;\n\ height: 52px !important;\n\ }\n\ }\n\ }\n\ #' + this.nodeIds.searchMenuPopup + ' > hbox >.menuitem-iconic[campactMenu="true"] > image {\n\ /*margin-left: .8em !important;*/\n\ }\n\ }\n\ '; return Services.io.newURI("data:text/css," + encodeURIComponent(cssStr), null, null); }, searchCommandHandler: function(event, trg) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchBySite = ""; if (event.target.hasAttribute("searchBySite")) { searchBySite = " site:" + gBrowser.currentURI.host; } else if (event.target.hasAttribute("searchInCache")) { searchBySite = content.location.href; } let selectedText = this.isSeaMonkey ? content.getSelection().toString() : getBrowserSelection(); if (event.target.engine.imageLink) { gBrowser.loadOneTab(event.target.engine.imageLink + /*getBrowserSelection()*/ selectedText, { relatedToCurrent: true, inBackground: inBg }); } else if (event.target.engine.cacheLink) { gBrowser.loadOneTab(event.target.engine.cacheLink + /*getBrowserSelection()*/ searchBySite, { relatedToCurrent: true, inBackground: inBg }); } else { let submission = event.target.engine.getSubmission(/*getBrowserSelection()*/ selectedText + searchBySite, null); gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); } if (trg == "menu") { setTimeout(function () { document.getElementById("contentAreaContextMenu").hidePopup(); }, 0); } }, contextMenuPopup: function(e) { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; //var selectedText = getBrowserSelection(16); var selectedText = this.isSeaMonkey ? content.getSelection(16).toString() : getBrowserSelection(16); if (!selectedText) return; var ellipsis = "\u2026"; try { ellipsis = gPrefService.getComplexValue("intl.ellipsis", Ci.nsIPrefLocalizedString).data; } catch (ex) { } if (selectedText.length > 15) selectedText = selectedText.substr(0,15) + ellipsis; let rEngine = this.isSeaMonkey ? "currentEngine" : "defaultEngine"; var engineName = this.param.isPinnedEngine ? this.param.pinnedEngineName : searchService[rEngine].name; var menuLabel; var searchFormattedString = [ "contextMenuSearch", "contextMenuSearchText", "searchSelected" ]; var thisIsSM = this.isSeaMonkey; function getMenuLabel(str) { try { if (thisIsSM) { let bundle = document.getElementById("contentAreaCommandsBundle"); return bundle.getFormattedString(str, [engineName, selectedText]); } else return gNavigatorBundle.getFormattedString(str, [engineName, selectedText]); } catch (ex) { return false; } } for(let i in searchFormattedString) { menuLabel = getMenuLabel(searchFormattedString[i]); if (menuLabel) break; else if (!menuLabel && i == searchFormattedString.length-1) menuLabel = engineName; } menu.label = menuLabel; }, searchMenuPopup: function(e) { var popup = e.target; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } var menus; if (!popup.menus) { var itemsObj = popup.getElementsByTagName("menu"); popup.menus = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menu" && elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) popup.menus.push(elem); }); menus = popup.menus; } else { menus = popup.menus; } var siteSearchDomain; var gBrowser = gBrowser || getBrowser(); function isDomain() { try { siteSearchDomain = gBrowser.currentURI.host; } catch (ex) { return false; } return true; } if (isDomain()) { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.setAttribute("tooltiptext", siteSearchDomain); elem.removeAttribute("disabled"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite")) elem.setAttribute("tooltiptext", siteSearchDomain); else if (elem.hasAttribute("searchInCache")) elem.setAttribute("tooltiptext", content.document.title); elem.removeAttribute("disabled"); }); } else { items.forEach(function(elem) { //if (elem.hasAttribute("searchBySite")) // elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); menus.forEach(function(elem) { if (elem.hasAttribute("searchBySite") || elem.hasAttribute("searchInCache")) elem.removeAttribute("tooltiptext"); elem.setAttribute("disabled", "true"); }); } }, createSearchMenu: function() { var searchService = this.searchService; var contextMenu = this.contextMenu; var menu = this.searchMenu; var engines = searchService.getVisibleEngines({}); menu.setAttribute("id", this.nodeIds.searchMenu); menu.setAttribute("class", "menu-iconic"); this.param.isPinnedEngine = false; this.param.pinnedEngineName = ""; var pinnedEngineName = options.pinnedEngineName; if (pinnedEngineName != "") { for (let i in engines) { if (engines[i].name != pinnedEngineName) continue; menu.setAttribute("label", engines[i].name); menu.setAttribute("image", engines[i].iconURI.spec); menu.engine = engines[i]; this.param.isPinnedEngine = true; this.param.pinnedEngineName = pinnedEngineName; break; } } let rEngine = this.isSeaMonkey ? "currentEngine" : "defaultEngine"; if (!this.param.isPinnedEngine) { menu.setAttribute("label", searchService[rEngine].name); menu.setAttribute("image", searchService[rEngine].iconURI.spec); menu.engine = searchService[rEngine]; } menu.gObj = this; menu.setAttribute("onclick", "if (event.target == this && event.target.engine) this.gObj.searchCommandHandler(event, 'menu');"); menu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchMenu.popup; popup.setAttribute("id", this.nodeIds.searchMenuPopup); var counterMaxLength = options.campactMenuLengthPerLine; if (typeof counterMaxLength != "number" || 3 > counterMaxLength || counterMaxLength > 10) counterMaxLength = 5; var counter = 0; var hBox; function itemCreateCampact(engine, last) { if(counter == 0) { hBox = document.createElementNS(XULNS, "hbox"); let item = document.createElementNS(XULNS, "spacer"); //item.setAttribute("style", "width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); } counter++; var item = document.createElementNS(XULNS, "menuitem"); item.engine = engine; item.setAttribute("class", "menuitem-iconic"); item.setAttribute("tooltiptext", engine.name); //item.setAttribute("src", engine.iconURI.spec); var image = document.createElementNS(XULNS, "image"); image.setAttribute("src", engine.iconURI.spec); item.appendChild(image); //item.setAttribute("style", "max-width: 2em"); item.setAttribute("campactMenu", "true"); hBox.appendChild(item); if (counter == counterMaxLength || last) { popup.appendChild(hBox); counter = 0; } } var separator = {}; function itemCreate(engine, reason) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.engine = engine; if (reason && !separator[reason]) { /*let sep = separator[reason] = document.createElementNS(XULNS, "menuseparator"); sep.setAttribute(reason, "true"); popup.appendChild(sep); let item = document.createElementNS(XULNS, "label"); //item.setAttribute("class", "menuitem-iconic"); item.setAttribute("value", _localize(reason) + ":"); item.setAttribute("style", "padding-left: 2em"); popup.appendChild(item);*/ if (!separator["menuseparator"]) { let sep = separator["menuseparator"] = document.createElementNS(XULNS, "menuseparator"); popup.appendChild(sep); } let subMenu = separator[reason] = document.createElementNS(XULNS, "menu"); subMenu.setAttribute("class", "menu-iconic"); subMenu.setAttribute("label", _localize(reason)); subMenu.setAttribute(reason, "true"); subMenu.popup = subMenu.appendChild(document.createElementNS(XULNS, "menupopup")); popup.appendChild(subMenu); } if (reason == "searchBySite") { item.setAttribute("label", engine.name); if (options.searchBySite.iconsOn) { item.setAttribute("src", engine.iconURI.spec); } item.setAttribute(reason, "true"); } else if (reason == "searchImageByText" || reason == "searchInCache") { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI_spec); item.setAttribute(reason, "true"); } else { item.setAttribute("label", engine.name); item.setAttribute("src", engine.iconURI.spec); } //popup.appendChild(item); if (reason) { let subMenu = separator[reason]; subMenu.popup.appendChild(item); } else popup.appendChild(item); } for (let i in engines) { if (options.hideDefaultEngineInPopupSubMenu && this.param.isPinnedEngine && engines[i].name == pinnedEngineName || options.hideDefaultEngineInPopupSubMenu && !this.param.isPinnedEngine && engines[i].name == searchService[rEngine].name) { continue; } if (options.campactMenu) itemCreateCampact(engines[i], i == engines.length-1); else itemCreate(engines[i]); } function searchBySite_itemCreate() { var namesList = []; if (options.searchBySite.useDefaultNamesList) { let defaultNamesList = [ ["Google", ["Google"]], ["Yandex", ["Yandex", "Яндекс"]], ["Yahoo", ["Yahoo"]], ["Bing", ["Bing"]], ["DuckDuckGo", ["DuckDuckGo"]] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchBySite.addCustomNames && Array.isArray(options.searchBySite.customNamesList)) { namesList = namesList.concat(options.searchBySite.customNamesList); } //let i_skips = []; function isNameExist(names, engine) { for (let i in names) { if (names[i] != engine && i != names.length-1) continue; else if (names[i] == engine) return true; return false; } } namesList.forEach(function(elem) { for (let i in engines) { if (/*i_skips.indexOf(i) != -1 ||*/ !(elem[1].some(function(listName) listName == engines[i].name)) /*!isNameExist(elem[1], engines[i].name)*/ /*elem[1].indexOf(engines[i].name) == -1*/) continue; itemCreate(engines[i], "searchBySite"); //i_skips.push(i); break; } }); } if (options.searchBySite.enable) { searchBySite_itemCreate(); } function searchImageByText_itemCreate() { var namesList = []; if (options.searchImageByText.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.ru/search?tbm=isch&q=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?text=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchImageByText.addCustomNames && Array.isArray(options.searchImageByText.customNamesList)) { namesList = namesList.concat(options.searchImageByText.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.imageLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchImageByText"); }); } if (options.searchImageByText.enable) { searchImageByText_itemCreate(); } function searchInCache() { var namesList = []; if (options.searchInCache.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/search?q=cache:", "" ], [ "Archive.org", "http://web.archive.org/web/*/", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchInCache.addCustomNames && Array.isArray(options.searchInCache.customNamesList)) { namesList = namesList.concat(options.searchInCache.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.cacheLink = elem[1]; engine.iconURI_spec = elem[2]; itemCreate(engine, "searchInCache"); }); } if (options.searchInCache.enable) { searchInCache(); } popup.gObj = this; //popup.setAttribute("oncommand", "if (!event.target.engine) return; var csObj = this.gObj || document.getElementById('contextSearchMiniByBunda1-context-popup').gObj; csObj.searchCommandHandler(event);"); popup.setAttribute("oncommand", "if (!event.target.engine || event.target.nodeName == 'menu') return; this.gObj.searchCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); }, searchByImageCommandHandler: function(event) { var inBg = options.loadInBackground || false; if (event.type == "click" && event.button && event.button == 1) { inBg = !inBg; } var searchByImageMenu = this.searchByImageMenu; var link = event.target.link; var imageLink = searchByImageMenu.imageSrc; if (imageLink.indexOf("data:") == 0) { this.searchByImageByData(imageLink, inBg); return; } else if (imageLink.indexOf("file:") == 0) { this.searchByImageByFile(imageLink, inBg); return; } else { imageLink = encodeURIComponent(imageLink) } var gBrowser = gBrowser || getBrowser(); gBrowser.loadOneTab(link + imageLink, { relatedToCurrent: true, inBackground: inBg }); }, // searchByImageByData() and searchByImageByFile() based on // Google Image Search 0.5 by Nishan Naseer // https://addons.mozilla.org/ru/firefox/addon/google-similar-images searchByImageByData: function(src, inBg) { var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byData(e) { newTabBrowser.removeEventListener(e.type, byData, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = src; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageByFile: function(src, inBg) { //Canvas to get contents in base-64 var canvas = gContextMenu.target.ownerDocument.createElementNS(XHTMLNS, "canvas"); var image = new Image(); image.src = src; canvas.width = image.width; canvas.height = image.height; var ctx = canvas.getContext("2d"); ctx.drawImage(image, 0, 0); var dataURL = canvas.toDataURL("image/png"); var gBrowser = gBrowser || getBrowser(); //var tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); var tab = gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); var newTabBrowser = gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener("load", function byFile(e) { newTabBrowser.removeEventListener(e.type, byFile, true); if (src) { try { var image_url; var sub; var form = newTabBrowser.contentDocument.getElementById("qbf"); var inputs = form.getElementsByTagName("input"); for (var k = 0; k < inputs.length; k++) { var i = inputs[k]; if (i.name && i.name == "image_url") { image_url = i; } if (i.type && i.type == "submit") { sub = i; } } image_url.value = dataURL; sub.click(); src = null; } catch (ex) { } } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageMenuPopup: function(e) { var searchByImageMenu = this.searchByImageMenu; searchByImageMenu.hidden = !gContextMenu.onImage; var src = gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL; if (!src || !(/^((ht|f)tps?:\/\/|data:image|file:)/.test(src))) { searchByImageMenu.hidden = true; return; } //var popup = document.getElementById("contextSearchMiniByBunda1-context-image-search-popup"); var popup = searchByImageMenu.popup; var items; if (!popup.items) { var itemsObj = popup.getElementsByTagName("menuitem"); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == "object" && elem.nodeName == "menuitem" && elem.getAttribute("label") != "Google") popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } if (/^(data:image|file:)/.test(src)) { items.forEach(function(elem) { elem.setAttribute("disabled", "true"); }); } else { items.forEach(function(elem) { elem.removeAttribute("disabled"); }); } searchByImageMenu.imageSrc = src; }, createSearchByImageMenu: function() { var menu = document.createElement("menu"); var insertMenu = document.getElementById("context-sep-copyimage"); insertMenu.parentNode.insertBefore(menu, insertMenu); this.searchByImageMenu = menu; menu.setAttribute("id", this.nodeIds.searchByImageMenu); menu.setAttribute("class", "menu-iconic"); menu.setAttribute("label", _localize("searchByImage")); menu.setAttribute("image", ""); this.searchByImageMenu.popup = menu.appendChild(document.createElementNS(XULNS, "menupopup")); var popup = this.searchByImageMenu.popup; popup.setAttribute("id", this.nodeIds.searchByImageMenuPopup); popup.googleURL = "https://www.google.com/searchbyimage"; var namesList = []; if (options.searchByImage.useDefaultNamesList) { let defaultNamesList = [ [ "Google", "https://www.google.com/searchbyimage?image_url=", "" ], [ "Yandex", "http://images.yandex.ru/yandsearch?rpt=imagedups&text=&img_url=", "" ], [ "TinEye", "http://www.tineye.com/search/?pluginver=firefox-1.0&url=", "" ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchByImage.addCustomNames && Array.isArray(options.searchByImage.customNamesList)) { namesList = namesList.concat(options.searchByImage.customNamesList); } function itemCreate(name, link, image) { var item = document.createElementNS(XULNS, "menuitem"); item.setAttribute("class", "menuitem-iconic"); item.setAttribute("src", image); item.setAttribute("label", name); item.link = link; popup.appendChild(item); } namesList.forEach(function(elem) { itemCreate(elem[0], elem[1], elem[2]); }); var contextMenu = this.contextMenu; popup.gObj = this; popup.setAttribute("oncommand", "this.gObj.searchByImageCommandHandler(event);"); popup.setAttribute("onclick", "checkForMiddleClick(this, event);"); } } contextSearcherObj.init(); })(); |
bunda1 > 31-05-2015 18:20:57 |
Northtech на: Выделить код Код:https://yandex.ru/images/search?uinfo=sw-1024-sh-768-ww-1024-wh-612-pd-1-wp-4x3_1024x768&viewport=wide&_=1433085221683&rpt=imageview&img_url= |
voqabuhe > 31-05-2015 18:30:54 |
bunda1 |
Northtech > 31-05-2015 18:31:53 |
bunda1 |
bunda1 > 31-05-2015 18:43:48 |
voqabuhe пишет
А мне на FF38 без проблем, попробуй искать изображения в Гугле на FF41 не используя код Context Search mini и если поиск работает скопируй правильный адрес из адресной строки браузера в код Context Search mini. И все дела. |
emlen > 18-12-2015 21:59:52 |
Перечитал ветку несколько раз, так и не пойму, рабочий готовый код или установочная ссылка тут есть? |
bunda1 > 19-12-2015 13:32:48 |
emlen пишет
А кто его знает, на Seamonkey их никто не проверял. |
emlen > 19-12-2015 15:03:52 |
я бы проверил))) было бы что проверить... |
bunda1 > 19-12-2015 17:23:32 |
Проверил Context Search на SeaMonkey, работает. |
emlen > 19-12-2015 23:25:08 |
bunda1 пишет
Очень благодарен, тоже заработало! |
bezuma > 20-12-2015 19:07:33 |
bunda1 |
Mishania > 21-12-2015 14:02:58 |
bunda1, вы как всегда радуете, уже не один аддон вашими кнопками заменил. Ну и конечно просьба,
Как кнопки поменять? Не могу привыкнуть, мне бы ЛКМ в фоне, а ПКМ - активная. |
difabor > 21-12-2015 17:09:02 |
Уважаемый bunda1, нельзя ли добавить в кнопку следующие возможности: |
bunda1 > 21-12-2015 20:51:23 |
Mishania пишет
В: Выделить код Код:// установить действие для клика на меню и подменю menu.onmouseup =e=> { var background = (e.button == 0) ? false : true; поменяй false и true местами. |
Mishania > 21-12-2015 21:31:19 |
bunda1 |
bunda1 > 23-12-2015 14:12:07 |
bezuma пишет
Сделаем. |
2k1dmg > 28-12-2015 14:48:51 |
Context Search 2 (mini) скрытый текст Выделить код Код:(function() { // Context Search 2 2015-12-28v2 'use strict'; let options = { loadInBackground: false, // true - открывать вкладки в фоне pinnedEngineName: '', // оставить пустым если не нужно закреплять // определенный поисковик // примеры pinnedEngineName: 'Google' или pinnedEngineName: 'Яндекс' или pinnedEngineName: '' hideDefaultEngineInPopupSubMenu: false, // true - не добавлять поисковик по умолчанию в выпадающий список (подменю) поисковиков campactMenu: true, // true - поисковики в выпадающем списке (подменю) без назаваний campactMenuLengthPerLine: 5, // максимальное количество элементов в одной строке // от 3 до 10, по умолчанию 5 searchBySite: { enable: true, // true - добавить поисковики по сайту iconsOn: true, // true - добавить иконки для поисковиков по сайту useDefaultNamesList: true, // true - использовать список поисковиков по умолчанию addCustomNames: false, // true - добавить свои поисковики по сайту customNamesList: [ ['Rambler', ['Rambler', 'Рамблер']] ] // например ['Rambler', ['Rambler', 'Рамблер']] // первая ячека название на английском // вторая ячека массив с предполагаемыми именами поисковика // может состоять из нескольких значений // ['Name1', ['Name1']], // ['Name2', ['Name2', 'Имя2']], // ['Name3', ['Name3', 'Имя3', 'Nev3']] }, searchImageByText: { enable: true, // true - добавить поиск изображения по тексту useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ 'name', 'link', 'image' ] ] }, searchByImage: { enable: true, // true - добавить поиск по изображению useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ 'name', 'link', 'image' ] ] }, searchInCache: { enable: true, // true - добавить поиск в кэше // нужно доработать useDefaultNamesList: true, addCustomNames: false, customNamesList: [ [ 'name', 'link', 'image' ] ] }, defaultIcon: 'chrome://mozapps/skin/places/defaultFavicon.png' }; if (Services.appinfo.OS == 'Darwin' && devicePixelRatio >= 2) { options.defaultIcon = 'chrome://mozapps/skin/places/defaultFavicon@2x.png'; } function _localize(sid) { let strings = { en: { searchBySite: 'Search this site', searchImageByText: 'Search image by text', searchByImage: 'Search by image', searchInCache: 'Search this page in cache' }, ru: { searchBySite: 'Поиск по этому сайту', searchImageByText: 'Поиск изображения по тексту', searchByImage: 'Поиск по изображению', searchInCache: 'Поиск этой страницы в кэше' } }; let getBrowserUILocale = function() { let gPrefService = Services.prefs; // SeaMonkey if (!gPrefService.getBoolPref('intl.locale.matchOS')) { try { let locale = gPrefService.getCharPref('general.useragent.locale'); if (locale.substr(0, 9) == 'chrome://') { return gPrefService.getComplexValue('general.useragent.locale', Components.interfaces.nsIPrefLocalizedString).data; } return locale; } catch (ex) {} } try { return Components.classes['@mozilla.org/chrome/chrome-registry;1'] .getService(Components.interfaces.nsIXULChromeRegistry) .getSelectedLocale('global'); } catch (ex) { return 'en-US'; } }; let locale = ''; // ru, en if (!locale) locale = getBrowserUILocale().match(/^[a-z]*/)[0]; _localize = function(sid) { return strings[locale] && strings[locale][sid] || strings.en[sid] || sid; }; return _localize(sid); } let XULNS = 'http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul'; let XHTMLNS = 'http://www.w3.org/1999/xhtml'; let contextSearcherObj = { initialized: false, init: function() { if (this.initialized) return; this.initialized = true; this.nodeIds = { searchMenu: 'context-search-2-by-2k1dmg-menu', searchMenuPopup: 'context-search-2-by-2k1dmg-popup', searchByImageMenu: 'context-search-2-by-2k1dmg-image-search-menu', searchByImageMenuPopup: 'context-search-2-by-2k1dmg-image-search-popup' }; if (document.getElementById(this.nodeIds.searchMenu)) return; let searchService = Cc['@mozilla.org/browser/search-service;1'] .getService(Ci.nsIBrowserSearchService); let contextMenu = document.getElementById('contentAreaContextMenu'); let searchSelect = document.getElementById('context-searchselect'); searchSelect.style.display = 'none'; let searchMenu = contextMenu .insertBefore(document.createElementNS(XULNS, 'menu'), searchSelect); this.searchService = searchService; this.contextMenu = contextMenu; this.searchSelect = searchSelect; this.searchMenu = searchMenu; this.param = { isPinnedEngine: false, pinnedEngineName: '' }; try { this.createSearchMenu(); if (options.searchByImage.enable) this.createSearchByImageMenu(); } catch (ex) { this.destroy(); Components.utils.reportError(ex); return; } if (options.campactMenu) this.loadSheet(window, this.makeCSS()); let observeStatus = this.observeStatus = new MutationObserver(function() { searchMenu.hidden = searchSelect.hidden; }); observeStatus.observe(searchSelect, { attributes: true, attributeFilter: ['hidden'] }); Services.obs.addObserver(this, 'browser-search-engine-modified', false); contextMenu.addEventListener('popupshowing', this, false); if (typeof addDestructor == 'function' && // userChromeJS/uc addDestructor != ('addDestructor' in window && window.addDestructor)) { addDestructor(function(reason) { this.destroy(reason); }, this); } }, destroy: function(reason) { if (!this.initialized) return; this.initialized = false; if (this.observeStatus) this.observeStatus.disconnect(); Services.obs.removeObserver(this, 'browser-search-engine-modified', false); let contextMenu = this.contextMenu; if (!contextMenu) return; contextMenu.removeEventListener('popupshowing', this, false); if (this.searchMenu) contextMenu.removeChild(this.searchMenu); if (this.searchByImageMenu) contextMenu.removeChild(this.searchByImageMenu); if (this.searchSelect) this.searchSelect.style.removeProperty('display'); this.removeSheet(window, this.makeCSS()); }, update: function() { this.searchMenu.textContent = ''; this.createSearchMenu(); this.updateTimeoutID = null; }, get isSeaMonkey() { delete this.isSeaMonkey; return this.isSeaMonkey = Services.appinfo.name == 'SeaMonkey'; }, observe: function(subject, topic, data) { switch (topic) { case 'browser-search-engine-modified': if (typeof this.updateTimeoutID == 'number') { window.clearTimeout(this.updateTimeoutID); this.updateTimeoutID = null; } this.updateTimeoutID = window.setTimeout(function() { this.update(); }.bind(this), 1000); break; } }, handleEvent: function(e) { switch (e.type) { case 'popupshowing': this.popupshowingEvent(e); break; } }, popupshowingEvent: function(e) { let trgId = e.target.id; if (trgId) { if (trgId == 'contentAreaContextMenu') { this.contextMenuPopup(e); if (this.searchByImageMenu) this.searchByImageMenuPopup(e); } else if (trgId == this.nodeIds.searchMenuPopup) this.searchMenuPopup(e); } }, makeCSS: function() { return '@namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n' + '#' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n' + ' width: 26px !important;\n' + '}\n' + '@media (min-resolution: 2dppx) {\n' + ' #' + this.nodeIds.searchMenuPopup + ' > hbox > spacer[campactMenu="true"] {\n' + ' width: 52px !important;\n' + ' }\n' + '}\n' + '#' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n' + ' width: 36px !important;\n' + ' height: 26px !important;\n' + ' -moz-box-pack: center !important;\n' + '}\n' + '@media (min-resolution: 2dppx) {\n' + ' #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic[campactMenu="true"] {\n' + ' width: 72px !important;\n' + ' height: 52px !important;\n' + ' }\n' + '}\n' + '#' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic > image {\n' + ' width: 16px !important;\n' + ' height: 16px !important;\n' + '}\n' + '@media (min-resolution: 2dppx) {\n' + ' #' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic > image {\n' + ' width: 32px !important;\n' + ' height: 32px !important;\n' + ' }\n' + '}\n' + '#' + this.nodeIds.searchMenuPopup + ' > hbox > .menuitem-iconic > *:not(image) {\n' + ' display: none !important;\n' + '}'; }, SHEET_TYPE: { 'agent': 'AGENT_SHEET', 'user': 'USER_SHEET', 'author': 'AUTHOR_SHEET' }, isTypeValid: function(type) { return type in SHEET_TYPE; }, makeCSSURI: function(url) { if (!/css$/.test(url)) url = 'data:text/css,' + encodeURIComponent(url); return this.ios.newURI(url, null, null); }, get ios() { delete this.ios; return this.ios = Cc['@mozilla.org/network/io-service;1']. getService(Ci.nsIIOService); }, getDOMWindowUtils: function(window) { return window.QueryInterface(Ci.nsIInterfaceRequestor). getInterface(Ci.nsIDOMWindowUtils); }, loadSheet: function(window, url, type) { if (!(type && type in SHEET_TYPE)) type = 'author'; type = this.SHEET_TYPE[type]; if (!(url instanceof Ci.nsIURI)) url = this.makeCSSURI(url); let winUtils = this.getDOMWindowUtils(window); try { winUtils.loadSheet(url, winUtils[type]); } catch (e) {}; }, removeSheet: function(window, url, type) { if (!(type && type in SHEET_TYPE)) type = 'author'; type = this.SHEET_TYPE[type]; if (!(url instanceof Ci.nsIURI)) url = this.makeCSSURI(url); let winUtils = this.getDOMWindowUtils(window); try { winUtils.removeSheet(url, winUtils[type]); } catch (e) {}; }, searchCommandHandler: function(event, trg) { let inBg = options.loadInBackground || false; if (event.type == 'click' && event.button && event.button == 1) { inBg = !inBg; } let searchBySite = ''; let _gBrowser = gBrowser || getBrowser(); if (event.target.hasAttribute('searchBySite')) { searchBySite = ' site:' + _gBrowser.currentURI.host; } else if (event.target.hasAttribute('searchInCache')) { searchBySite = content.location.href; } let selectedText = this.isSeaMonkey ? content.getSelection().toString() : getBrowserSelection(); if (event.target.engine.imageLink) { _gBrowser.loadOneTab(event.target.engine.imageLink + selectedText, { relatedToCurrent: true, inBackground: inBg }); } else if (event.target.engine.cacheLink) { _gBrowser.loadOneTab(event.target.engine.cacheLink + searchBySite, { relatedToCurrent: true, inBackground: inBg }); } else { let submission = event.target.engine.getSubmission(selectedText + searchBySite, null); _gBrowser.loadOneTab(submission.uri.spec, { relatedToCurrent: true, postData: submission.postData, inBackground: inBg }); } if (trg == 'menu') { setTimeout(function() { document.getElementById('contentAreaContextMenu').hidePopup(); }, 0); } }, contextMenuPopup: function(e) { let searchService = this.searchService; let contextMenu = this.contextMenu; let menu = this.searchMenu; let selectedText = this.isSeaMonkey ? content.getSelection(16).toString() : getBrowserSelection(16); if (!selectedText) return; let ellipsis = '\u2026'; try { ellipsis = gPrefService.getComplexValue('intl.ellipsis', Ci.nsIPrefLocalizedString).data; } catch (ex) {} if (selectedText.length > 15) selectedText = selectedText.substr(0, 15) + ellipsis; let currentBrowserEngine = this.isSeaMonkey ? 'currentEngine' : 'defaultEngine'; let engineName = this.param.isPinnedEngine ? this.param.pinnedEngineName : searchService[currentBrowserEngine].name; let menuLabel; let searchFormattedString = [ 'contextMenuSearch', 'contextMenuSearchText', 'searchSelected' // SeaMonkey ]; let thisIsSM = this.isSeaMonkey; let getMenuLabel = function(str) { try { if (thisIsSM) { let bundle = document.getElementById('contentAreaCommandsBundle'); return bundle.getFormattedString(str, [engineName, selectedText]); } else return gNavigatorBundle.getFormattedString(str, [engineName, selectedText]); } catch (ex) { return false; } }; for (let i = 0, len = searchFormattedString.length; i < len; i++) { menuLabel = getMenuLabel(searchFormattedString[i]); if (menuLabel) break; else if (!menuLabel && i == searchFormattedString.length - 1) menuLabel = engineName; } menu.label = menuLabel; }, searchMenuPopup: function(e) { let popup = e.target; let items; if (!popup.items) { let itemsObj = popup.getElementsByTagName('menuitem'); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == 'object' && elem.nodeName == 'menuitem' && elem.hasAttribute('searchBySite') || elem.hasAttribute('searchInCache')) popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } let menus; if (!popup.menus) { let itemsObj = popup.getElementsByTagName('menu'); popup.menus = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == 'object' && elem.nodeName == 'menu' && elem.hasAttribute('searchBySite') || elem.hasAttribute('searchInCache')) popup.menus.push(elem); }); menus = popup.menus; } else { menus = popup.menus; } let siteSearchDomain; let _gBrowser = gBrowser || getBrowser(); let isDomain = function() { try { siteSearchDomain = _gBrowser.currentURI.host; } catch (ex) { return false; } return true; }; if (isDomain()) { items.forEach(function(elem) { elem.removeAttribute('disabled'); }); menus.forEach(function(elem) { if (elem.hasAttribute('searchBySite')) elem.setAttribute('tooltiptext', siteSearchDomain); else if (elem.hasAttribute('searchInCache')) elem.setAttribute('tooltiptext', content.document.title); elem.removeAttribute('disabled'); }); } else { items.forEach(function(elem) { elem.setAttribute('disabled', 'true'); }); menus.forEach(function(elem) { if (elem.hasAttribute('searchBySite') || elem.hasAttribute('searchInCache')) elem.removeAttribute('tooltiptext'); elem.setAttribute('disabled', 'true'); }); } }, createSearchMenu: function() { let searchService = this.searchService; let contextMenu = this.contextMenu; let menu = this.searchMenu; let engines = searchService.getVisibleEngines({}); menu.setAttribute('id', this.nodeIds.searchMenu); menu.setAttribute('class', 'menu-iconic'); this.param.isPinnedEngine = false; this.param.pinnedEngineName = ''; let pinnedEngineName = options.pinnedEngineName; if (pinnedEngineName != '') { for (let i in engines) { if (engines[i].name != pinnedEngineName) continue; menu.setAttribute('label', engines[i].name); menu.setAttribute('image', engines[i].iconURI.spec); menu.engine = engines[i]; this.param.isPinnedEngine = true; this.param.pinnedEngineName = pinnedEngineName; break; } } let currentBrowserEngine = this.isSeaMonkey ? 'currentEngine' : 'defaultEngine'; if (!this.param.isPinnedEngine) { menu.setAttribute('label', searchService[currentBrowserEngine].name); menu.setAttribute('image', (searchService[currentBrowserEngine].iconURI ? searchService[currentBrowserEngine].iconURI.spec : options.defaultIcon)); menu.engine = searchService[currentBrowserEngine]; } menu.gObj = this; menu.setAttribute('onclick', 'if (event.target == this && event.target.engine) this.gObj.searchCommandHandler(event, "menu");'); menu.popup = menu.appendChild(document.createElementNS(XULNS, 'menupopup')); let popup = this.searchMenu.popup; popup.setAttribute('id', this.nodeIds.searchMenuPopup); let counterMaxLength = options.campactMenuLengthPerLine; if (typeof counterMaxLength != 'number' || 3 > counterMaxLength || counterMaxLength > 10) counterMaxLength = 5; let counter = 0; let hBox; let itemCreateCampact = function(engine, last) { if (counter == 0) { hBox = document.createElementNS(XULNS, 'hbox'); let item = document.createElementNS(XULNS, 'spacer'); item.setAttribute('campactMenu', 'true'); hBox.appendChild(item); } counter++; let item = document.createElementNS(XULNS, 'menuitem'); item.engine = engine; item.setAttribute('class', 'menuitem-iconic'); item.setAttribute('tooltiptext', engine.name); let image = document.createElementNS(XULNS, 'image'); image.setAttribute('src', (engine.iconURI ? engine.iconURI.spec : options.defaultIcon)); item.appendChild(image); item.setAttribute('campactMenu', 'true'); hBox.appendChild(item); if (counter == counterMaxLength || last) { popup.appendChild(hBox); counter = 0; } }; let separator = {}; let itemCreate = function(engine, reason) { let item = document.createElementNS(XULNS, 'menuitem'); item.setAttribute('class', 'menuitem-iconic'); item.engine = engine; if (reason && !separator[reason]) { if (!separator['menuseparator']) { let sep = separator['menuseparator'] = document.createElementNS(XULNS, 'menuseparator'); popup.appendChild(sep); } let subMenu = separator[reason] = document.createElementNS(XULNS, 'menu'); subMenu.setAttribute('class', 'menu-iconic'); subMenu.setAttribute('label', _localize(reason)); subMenu.setAttribute(reason, 'true'); subMenu.popup = subMenu.appendChild(document.createElementNS(XULNS, 'menupopup')); popup.appendChild(subMenu); } if (reason == 'searchBySite') { item.setAttribute('label', engine.name); if (options.searchBySite.iconsOn) { item.setAttribute('src', (engine.iconURI ? engine.iconURI.spec : options.defaultIcon)); } item.setAttribute(reason, 'true'); } else if (reason == 'searchImageByText' || reason == 'searchInCache') { item.setAttribute('label', engine.name); item.setAttribute('src', (engine.iconURI_spec != '' ? engine.iconURI_spec : options.defaultIcon)); item.setAttribute(reason, 'true'); } else { item.setAttribute('label', engine.name); item.setAttribute('src', (engine.iconURI ? engine.iconURI.spec : options.defaultIcon)); } if (reason) { let subMenu = separator[reason]; subMenu.popup.appendChild(item); } else popup.appendChild(item); }; for (let i in engines) { if (options.hideDefaultEngineInPopupSubMenu && this.param.isPinnedEngine && engines[i].name == pinnedEngineName || options.hideDefaultEngineInPopupSubMenu && !this.param.isPinnedEngine && engines[i].name == searchService[currentBrowserEngine].name) { continue; } if (options.campactMenu) itemCreateCampact(engines[i], i == engines.length - 1); else itemCreate(engines[i]); } let itemCreateSearchBySite = function() { let namesList = []; if (options.searchBySite.useDefaultNamesList) { let defaultNamesList = [ ['Google', ['Google']], ['Yandex', ['Yandex', 'Яндекс']], ['Yahoo', ['Yahoo']], ['Bing', ['Bing']], ['DuckDuckGo', ['DuckDuckGo']] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchBySite.addCustomNames && Array.isArray(options.searchBySite.customNamesList)) { namesList = namesList.concat(options.searchBySite.customNamesList); } namesList.forEach(function(elem) { for (let i in engines) { if (!(elem[1].some(function(listName) { return listName == engines[i].name }) ) ) continue; itemCreate(engines[i], 'searchBySite'); break; } }); }; if (options.searchBySite.enable) { itemCreateSearchBySite(); } let itemCreateSearchImageByText = function() { let namesList = []; if (options.searchImageByText.useDefaultNamesList) { let defaultNamesList = [ [ 'Google', 'https://www.google.ru/search?tbm=isch&q=', '' ], [ 'Yandex', 'http://images.yandex.ru/yandsearch?text=', '' ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchImageByText.addCustomNames && Array.isArray(options.searchImageByText.customNamesList)) { namesList = namesList.concat(options.searchImageByText.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.imageLink = elem[1]; engine.iconURI_spec = elem[2]; // itemCreate(engine, 'searchImageByText'); }); }; if (options.searchImageByText.enable) { itemCreateSearchImageByText(); } let itemCreateSearchInCache = function() { let namesList = []; if (options.searchInCache.useDefaultNamesList) { let defaultNamesList = [ [ 'Google', 'https://www.google.com/search?q=cache:', '' ], [ 'Archive.org', 'http://web.archive.org/web/*/', '' ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchInCache.addCustomNames && Array.isArray(options.searchInCache.customNamesList)) { namesList = namesList.concat(options.searchInCache.customNamesList); } namesList.forEach(function(elem) { let engine = {}; engine.name = elem[0]; engine.cacheLink = elem[1]; engine.iconURI_spec = elem[2]; // !!! это не engine.iconURI.spec itemCreate(engine, 'searchInCache'); }); }; if (options.searchInCache.enable) { itemCreateSearchInCache(); } popup.gObj = this; popup.setAttribute('oncommand', 'if (!event.target.engine || event.target.nodeName == "menu") return; this.gObj.searchCommandHandler(event);'); popup.setAttribute('onclick', 'checkForMiddleClick(this, event);'); }, searchByImageCommandHandler: function(event) { let inBg = options.loadInBackground || false; if (event.type == 'click' && event.button && event.button == 1) { inBg = !inBg; } let searchByImageMenu = this.searchByImageMenu; let link = event.target.link; let imageLink = searchByImageMenu.imageSrc; if (imageLink.indexOf('data:') == 0) { this.searchByImageByData(imageLink, inBg); return; } else if (imageLink.indexOf('file:') == 0) { this.searchByImageByFile(imageLink, inBg); return; } else { imageLink = encodeURIComponent(imageLink) } let _gBrowser = gBrowser || getBrowser(); _gBrowser.loadOneTab(link + imageLink, { relatedToCurrent: true, inBackground: inBg }); }, // searchByImageByData() and searchByImageByFile() based on // Google Image Search 0.5 by Nishan Naseer // https://addons.mozilla.org/ru/firefox/addon/google-similar-images searchByImageByData: function(src, inBg) { let _gBrowser = gBrowser || getBrowser(); //let tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); let tab = _gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); let newTabBrowser = _gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener('load', function byData(e) { newTabBrowser.removeEventListener(e.type, byData, true); if (src) { try { let image_url; let sub; let form = newTabBrowser.contentDocument.getElementById('qbf'); let inputs = form.getElementsByTagName('input'); for (let k = 0; k < inputs.length; k++) { let i = inputs[k]; if (i.name && i.name == 'image_url') { image_url = i; } if (i.type && i.type == 'submit') { sub = i; } } image_url.value = src; sub.click(); src = null; } catch (ex) {} } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageByFile: function(src, inBg) { //Canvas to get contents in base-64 let canvas = gContextMenu.target.ownerDocument.createElementNS(XHTMLNS, 'canvas'); let image = new Image(); image.src = src; canvas.width = image.width; canvas.height = image.height; let ctx = canvas.getContext('2d'); ctx.drawImage(image, 0, 0); let dataURL = canvas.toDataURL('image/png'); let _gBrowser = gBrowser || getBrowser(); //let tab = gBrowser.addTab(this.searchByImageMenu.popup.googleURL); let tab = _gBrowser.loadOneTab(this.searchByImageMenu.popup.googleURL, { relatedToCurrent: true }); let newTabBrowser = _gBrowser.getBrowserForTab(tab); newTabBrowser.addEventListener('load', function byFile(e) { newTabBrowser.removeEventListener(e.type, byFile, true); if (src) { try { let image_url; let sub; let form = newTabBrowser.contentDocument.getElementById('qbf'); let inputs = form.getElementsByTagName('input'); for (let k = 0; k < inputs.length; k++) { let i = inputs[k]; if (i.name && i.name == 'image_url') { image_url = i; } if (i.type && i.type == 'submit') { sub = i; } } image_url.value = dataURL; sub.click(); src = null; } catch (ex) {} } }, true); if (!inBg) { getBrowser().selectedTab = tab; } }, searchByImageMenuPopup: function(e) { let searchByImageMenu = this.searchByImageMenu; searchByImageMenu.hidden = !gContextMenu.onImage; let src = gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL; if (!src || !(/^((ht|f)tps?:\/\/|data:image|file:)/.test(src))) { searchByImageMenu.hidden = true; return; } let popup = searchByImageMenu.popup; let items; if (!popup.items) { let itemsObj = popup.getElementsByTagName('menuitem'); popup.items = []; Array.slice(itemsObj).forEach(function(elem) { if (typeof elem == 'object' && elem.nodeName == 'menuitem' && elem.getAttribute('label') != 'Google') popup.items.push(elem); }); items = popup.items; } else { items = popup.items; } if (/^(data:image|file:)/.test(src)) { items.forEach(function(elem) { elem.setAttribute('disabled', 'true'); }); } else { items.forEach(function(elem) { elem.removeAttribute('disabled'); }); } searchByImageMenu.imageSrc = src; }, createSearchByImageMenu: function() { let menu = document.createElement('menu'); let insertMenu = document.getElementById('context-sep-copyimage'); insertMenu.parentNode.insertBefore(menu, insertMenu); this.searchByImageMenu = menu; menu.setAttribute('id', this.nodeIds.searchByImageMenu); menu.setAttribute('class', 'menu-iconic'); menu.setAttribute('label', _localize('searchByImage')); menu.setAttribute('image', ''); this.searchByImageMenu.popup = menu.appendChild(document.createElementNS(XULNS, 'menupopup')); let popup = this.searchByImageMenu.popup; popup.setAttribute('id', this.nodeIds.searchByImageMenuPopup); popup.googleURL = 'https://www.google.com/searchbyimage'; let namesList = []; if (options.searchByImage.useDefaultNamesList) { let defaultNamesList = [ [ 'Google', 'https://www.google.com/searchbyimage?image_url=', '' ], [ 'Yandex', 'http://images.yandex.ru/yandsearch?rpt=imageview&&img_url=', '' ], [ 'TinEye', 'http://www.tineye.com/search/?pluginver=firefox-1.0&url=', '' ] ]; namesList = namesList.concat(defaultNamesList); } if (options.searchByImage.addCustomNames && Array.isArray(options.searchByImage.customNamesList)) { namesList = namesList.concat(options.searchByImage.customNamesList); } let itemCreate = function(name, link, image) { let item = document.createElementNS(XULNS, 'menuitem'); item.setAttribute('class', 'menuitem-iconic'); item.setAttribute('src', image); item.setAttribute('label', name); item.link = link; popup.appendChild(item); }; namesList.forEach(function(elem) { itemCreate(elem[0], elem[1], elem[2]); }); let contextMenu = this.contextMenu; popup.gObj = this; popup.setAttribute('oncommand', 'this.gObj.searchByImageCommandHandler(event);'); popup.setAttribute('onclick', 'checkForMiddleClick(this, event);'); } }; contextSearcherObj.init(); }()); |
bunda1 > 28-12-2015 17:15:11 |
2k1dmg Выделить код Код:alert( Cc['@mozilla.org/network/io-service;1'].getService(Ci.nsIIOService) == Services.io ); //== true Кстати, красивый код у тебя получился. |
Mishania > 28-12-2015 20:44:48 |
2k1dmg |
villa7 > 01-01-2016 14:28:58 |
Как в этой кнопке сделать чтобы убрать надпись "Поиск <бла-бла> в Google, оставив только Google. |
Dumby > 01-01-2016 17:11:41 |
villa7 пишет
Ломать не строить . скрытый текст |
villa7 > 01-01-2016 17:20:20 |
Dumby |
2k1dmg > 13-01-2016 19:10:22 |
Context Search 2 2016-01-10 |
difabor > 14-01-2016 01:22:43 |
2k1dmg пишет
Уважаемый 2k1dmg, |
2k1dmg > 14-01-2016 18:21:39 |
difabor скрытый текст Выделить код Код:let counterMaxLength = options.campactMenuLengthPerLine; if (typeof counterMaxLength != 'number' || 3 > counterMaxLength || counterMaxLength > 10) counterMaxLength = 5; let counter = 0; let hBox; let itemCreateCampact = function(engine, last) { А по всему остальному: ничего нового добавлять не планирую, сохранить бы в рабочем состоянии что есть. |
bunda1 > 14-01-2016 21:51:31 |
difabor пишет
Другой пример: Выделить код Код:custombutton://%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%0D%0A%3Ccustombutton%20xmlns%3Acb%3D%22http%3A//xsms.nm.ru/custombuttons/%22%3E%0A%20%20%3Cname%3EDoubleM%3C/name%3E%0A%20%20%3Cimage%3E%3C%21%5BCDATA%5Bdata%3Aimage/png%3Bbase64%2CiVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAIAAACQkWg2AAAAQElEQVR42mNkIBEwAvG/f/+IVM3ExATVAGQRVA1RhtCwf/9+ZGlHR8d//2YgmZ0xqmGYaCAtaRBUDbMHrIEkAABT+ooR2wR10QAAAABJRU5ErkJggg%3D%3D%5D%5D%3E%3C/image%3E%0A%20%20%3Cmode%3E0%3C/mode%3E%0A%20%20%3Cinitcode%3E%3C%21%5BCDATA%5B//%20%u041D%u0430%u0441%u0442%u0440%u043E%u0439%u043A%u0430%20%u0444%u0443%u043D%u043A%u0446%u0438%u0439%20%u043A%u043B%u0438%u043A%u043E%u0432%20%u043C%u044B%u0448%u0438%20....................%20%0Athis._handleClick%20%3D%28%29%3D%3E%20popup.openPopup%28this%29%3B%0A%0A%0A//%20%u0421%u043E%u0437%u0434%u0430%u0442%u044C%20%u0434%u0432%u0443%u0445%u0441%u0435%u043A%u0446%u0438%u043E%u043D%u043D%u043E%u0435%20%u043C%u0435%u043D%u044E%20......................%20%0Avar%20popup%20%3D%20addElement%28%22menupopup%22%2C%20%7B%0A%20%20%20position%3A%20%22after_start%22%2C%0A%20%20%20oncontextmenu%3A%20%22return%20false%22%2C%0A%20%20%20style%3A%20%22-moz-appearance%3A%20none%3B%20border%3A%201px%20solid%22%0A%7D%2C%20self%29%3B%0A%0Avar%20mainBox%20%3D%20addElement%28%22hbox%22%2C%20%7B%7D%2C%20popup%29%3B%0Avar%20leftBox%20%3D%20addElement%28%22vbox%22%2C%20%7B%7D%2C%20%20mainBox%29%3B%20%20%20%20%20//%20%u041B%u0435%u0432%u043E%u0435%20%u043C%u0435%u043D%u044E%0Avar%20rightBox%20%3D%20addElement%28%22vbox%22%2C%20%7B%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20//%20%u041F%u0440%u0430%u0432%u043E%u0435%20%u043C%u0435%u043D%u044E%0A%20%20%20style%3A%20%22background-color%3A%20rgb%28241%2C%20245%2C%20251%29%3B%20box-shadow%3A%201px%200px%202px%20rgb%28204%2C%20214%2C%20234%29%20inset%3B%22%20//%20%u0441%u0442%u0438%u043B%u044C%20%u043F%u0440%u0430%u0432%u043E%u0433%u043E%20%u043C%u0435%u043D%u044E%0A%7D%2C%20mainBox%29%3B%0A%0A%0A%0A//%20%u041B%u0435%u0432%u044B%u0435%20%u043F%u0443%u043D%u043A%u0442%u044B%20%u043C%u0435%u043D%u044E%20......................%20%0Avar%20leftItem0%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u041D%u043E%u0432%u043E%u0435%20%u043F%u0440%u0438%u0432%u0430%u0442%u043D%u043E%u0435%20%u043E%u043A%u043D%u043E%22%2C%0A%20%20%20oncommand%3A%20%22OpenBrowserWindow%28%7Bprivate%3A%20true%7D%29%22%2C%0A%20%20%20image%3A%20%22chrome%3A//browser/skin/Privacy-16.png%22%0A%7D%2C%20leftBox%29%3B%0A%0Avar%20leftItem1%20%3D%20addElement%28%22menuseparator%22%2C%20%7B%7D%2C%20%20leftBox%29%3B%0A%0Avar%20leftItem2%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u041E%u0442%u043A%u0440%u044B%u0442%u044C%20%u0444%u0430%u0439%u043B%22%2C%0A%20%20%20oncommand%3A%20%22BrowserOpenFileWindow%28%29%22%0A%7D%2C%20leftBox%29%3B%0A%0Avar%20leftItem3%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u0421%u043E%u0445%u0440%u0430%u043D%u0438%u0442%u044C%20%u043A%u0430%u043A%u2026%22%2C%0A%20%20%20oncommand%3A%20%22saveDocument%28window.content.document%29%22%0A%7D%2C%20leftBox%29%3B%0A%0Avar%20leftItem4%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u041F%u0435%u0447%u0430%u0442%u044C%u2026%22%2C%0A%20%20%20oncommand%3A%20%22PrintUtils.print%28%29%22%0A%7D%2C%20leftBox%29%3B%0A%0Avar%20leftItem5%20%3D%20addElement%28%22menuseparator%22%2C%20%7B%7D%2C%20%20leftBox%29%3B%0A%0Avar%20leftItem6%20%3D%20addElement%28%22menu%22%2C%20%7B%0A%20%20%20label%3A%20%22%u0412%u0435%u0431-%u0440%u0430%u0437%u0440%u0430%u0431%u043E%u0442%u043A%u0430%u2026%22%2C%0A%20%20%20onclick%3A%20%22gDevToolsBrowser.toggleToolboxCommand%28gBrowser%29%22%0A%7D%2C%20leftBox%29%0A.appendChild%28document.getElementById%28%27menuWebDeveloperPopup%27%29.cloneNode%28true%29%29%3B%0A%0Avar%20leftItem7%20%3D%20addElement%28%22menuseparator%22%2C%20%7B%7D%2C%20%20leftBox%29%3B%0A%0Avar%20leftItem8%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u041F%u043E%u043B%u043D%u044B%u0439%20%u044D%u043A%u0440%u0430%u043D%22%2C%0A%20%20%20oncommand%3A%20%22BrowserFullScreen%28%29%22%0A%7D%2C%20leftBox%29%3B%0A%0Avar%20leftItem9%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u041D%u0430%u0441%u0442%u0440%u043E%u0438%u0442%u044C%20Sync%u2026%22%2C%0A%20%20%20oncommand%3A%20%22gSyncUI.openSetup%28%29%22%0A%7D%2C%20leftBox%29%3B%0A%0Avar%20leftItem10%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22%u0417%u0430%u043A%u0440%u044B%u0442%u044C%20%u0431%u0440%u0430%u0443%u0437%u0435%u0440%22%2C%0A%20%20%20oncommand%3A%20%22Application.quit%28%29%22%0A%7D%2C%20leftBox%29%3B%0A%0A%0A%0A//%20%u041F%u0440%u0430%u0432%u044B%u0435%20%u043F%u0443%u043D%u043A%u0442%u044B%20%u043C%u0435%u043D%u044E%20......................%20%0Avar%20rightItem0%20%3D%20addElement%28%22menu%22%2C%20%7B%0A%20%20%20label%3A%20%22%u0417%u0430%u043A%u043B%u0430%u0434%u043A%u0438%22%2C%0A%20%20%20image%3A%20%22chrome%3A//custombuttons/skin/button.png%22%2C%0A%7D%2C%20rightBox%29%0A.appendChild%28document.getElementById%28%27bookmarksMenuPopup%27%29.cloneNode%28true%29%29%3B%0A%0Avar%20rightItem1%20%3D%20addElement%28%22menuitem%22%2C%20%7B%0A%20%20%20label%3A%20%22menuitem%20R1%22%2C%0A%20%20%20image%3A%20%22chrome%3A//global/skin/icons/information-16.png%22%2C%0A%20%20%20oncommand%3A%20%22alert%28this.label%29%3B%22%0A%7D%2C%20rightBox%29%3B%0A%0Avar%20rightItem2%20%3D%20addElement%28%22menuseparator%22%2C%20%7B%7D%2C%20%20rightBox%29%3B%0A%0A%0A%0A//%20%u0421%u043E%u0437%u0434%u0430%u0432%u0430%u0442%u044C%20%u044D%u043B%u0435%u043C%u0435%u043D%u0442%u044B%20%u0441%20%u0443%u043A%u0430%u0437%u0430%u043D%u043D%u044B%u043C%u0438%20%u0430%u0442%u0440%u0438%u0431%u0443%u0442%u0430%u043C%u0438%20......................%20%0Afunction%20addElement%28elementName%2C%20attributes%2C%20parent%29%20%7B%0A%20%20%20var%20element%20%3D%20document.createElement%28elementName%29%3B%0A%20%20%20%0A%20%20%20element.id%20%3D%20_id%20+%20%22-%22%20+%20elementName%20+%20%22-%22%20+%20Date.now%28%29%3B%20%20%20%0A%20%20%20for%20%28%20var%20attribute%20in%20attributes%20%29%20element.setAttribute%28attribute%2C%20attributes%5Battribute%5D%29%3B%0A%20%20%20if%20%28%20%5B%22menu%22%2C%20%22menuitem%22%5D.indexOf%28elementName%29%20%21%3D%20-1%20%29%20element.classList.add%28elementName%20+%20%22-iconic%22%29%3B%0A%20%20%20%0A%20%20%20return%20parent.appendChild%28element%29%3B%0A%7D%3B%0AaddDestructor%28%28%29%3D%3E%20popup.remove%28%29%20%29%3B%5D%5D%3E%3C/initcode%3E%0A%20%20%3Ccode%3E%3C%21%5BCDATA%5B/*CODE*/%5D%5D%3E%3C/code%3E%0A%20%20%3Caccelkey%3E%3C%21%5BCDATA%5B%5D%5D%3E%3C/accelkey%3E%0A%20%20%3Chelp%3E%3C%21%5BCDATA%5B%5D%5D%3E%3C/help%3E%0A%20%20%3Cattributes/%3E%0A%3C/custombutton%3E |
PEAKTOP > 18-01-2016 17:54:48 |
2k1dmg, а какие кардинальные отличия от версии 2013-10-10? Я визуальных не нашёл. |
xrun1 > 18-01-2016 18:30:34 |
PEAKTOP скрытый текст |
Mishania > 14-02-2016 20:16:58 |
|
Злой Буратино > 20-11-2016 12:36:15 |
del |
1fesFFFF > 24-01-2017 14:56:24 |
bunda1 |
oleg.sgh > 24-01-2017 15:06:56 |
1fesFFFF |
1fesFFFF > 24-01-2017 15:17:35 |
bunda1 И дабы не плодить темы, ещё у меня сломалась функция, которая работала из папки chrome - ScrollSearchEngines.uc - позволяла менять поисковые движки колёсиком мыши. Вот сам файл ссылка , а вот его содержимое: скрытый текст Выделить код Код:// ==UserScript== // @name Scroll Search Engines // @namespace http://amb.vis.ne.jp/mozilla/ // @description Change the selected search engine by scroll wheel on 'Search xxx for yyy' menu and do search by middle-clicking the menu. // @include main // @compatibility Firefox 2.0 3.0 3.1 3.2a1 // @author Gomita // @permalink http://amb.vis.ne.jp/mozilla/?p=71 // @contributor Alice0775, . // @Note http://space.geocities.yahoo.co.jp/gl/alice0775 // @version 2009/02/20 18:00 works without searchbar by alice0775 // @version 2009/02/18 21:00 works with full screen mode by alice0775 // ==/UserScript== // @version 2007/10/21 03:50 compatibility with Fx3 and conqueryModoki2.uc.xul and performance improved by alice0775 // @version 2007.01.18 (function() { const nsIBSS = Components.interfaces.nsIBrowserSearchService; const searchService =Components.classes["@mozilla.org/browser/search-service;1"] .getService(nsIBSS); var timer,index; try{ var searchBar = BrowserSearch.getSearchBar(); }catch(e){ var searchBar = BrowserSearch.searchBar; //fx3 } var mouseScrollHandler = function(event) { var engineName,flg; var engines = searchService.getVisibleEngines({ }); // make sure that search bar is visible //if (!searchBar) // return; event.stopPropagation(); // Find the new index if(!index) index = engines.indexOf(searchService.currentEngine); // change search engine var i = index; if(event.detail > 0){ do{ i = i + 1; if(i >= engines.length) { i= index; break; } engineName = engines[i].name; flg = engineName.match(/-{2,}|\u2015{2,}/) || (engineName.match(/^{/) && !engineName.match(/}/)) || (engineName.match(/}/) && !engineName.match(/{/)) }while(flg) }else{ do{ i = i - 1; if(i < 0 ) { i= index; break; } engineName = engines[i].name; flg = engineName.match(/-{2,}|\u2015{2,}/) || (engineName.match(/^{/) && !engineName.match(/}/)) || (engineName.match(/}/) && !engineName.match(/{/)) }while(flg) } index = i; if(!gContextMenu){ searchService.currentEngine = engines[i]; //指示されたエンジンにする return; } // update context menu label if(timer) clearTimeout(timer); engineName = engines[i].name; var label = gNavigatorBundle.getFormattedString("contextMenuSearchText", [engineName, getBrowserSelection(16)]); var menuitem = event.originalTarget; menuitem.engine = engines[i]; menuitem.setAttribute("label", label); // update context menu icon timer = setTimeout(function(){ searchService.currentEngine = engines[i]; var iconURI = engines[i].iconURI; if (iconURI) menuitem.setAttribute("src", iconURI.spec); else menuitem.removeAttribute("src"); },100); }; if (searchBar) { // enable to change search engine by mouse-wheel on engine button document.getAnonymousElementByAttribute(searchBar, "anonid", "searchbar-engine-button") .addEventListener("DOMMouseScroll", mouseScrollHandler, false); } document.getElementById("cmd_CustomizeToolbars").addEventListener("DOMAttrModified", function(e) { if (e.attrName == "disabled" && !e.newValue){ try{ var searchBar = BrowserSearch.getSearchBar(); }catch(e){ var searchBar = BrowserSearch.searchBar; //fx3 } if(searchBar) document.getAnonymousElementByAttribute(searchBar, "anonid", "searchbar-engine-button").addEventListener("DOMMouseScroll", mouseScrollHandler, false); } }, false); var searchMenu = document.getElementById("context-searchselect"); if(!searchMenu) return; // enables to change search engine by mouse-wheel on context menu searchMenu.addEventListener("DOMMouseScroll", mouseScrollHandler, false); // update context menu icon on showing popup document.getElementById("contentAreaContextMenu").addEventListener("popupshowing",function(aEvent) { var ss = Cc["@mozilla.org/browser/search-service;1"]. getService(Ci.nsIBrowserSearchService); if(aEvent.originalTarget != document.getElementById('contentAreaContextMenu')) return; if (!gContextMenu || !gContextMenu.isTextSelected) return; // update context menu icon var menuitem = document.getElementById("context-searchselect"); var iconURI = ss.currentEngine.iconURI; if (iconURI) menuitem.setAttribute("src", iconURI.spec); else menuitem.removeAttribute("src"); }, false); // enable to search by middle-click searchMenu.addEventListener("click", function(event) { if (event.button == 1) { searchMenu.loadSearch(getBrowserSelection(), true, searchMenu.engine); event.originalTarget.parentNode.hidePopup(); } }, false); searchMenu.loadSearch = function(searchText, useNewTab, engine) { var ss = Cc["@mozilla.org/browser/search-service;1"]. getService(Ci.nsIBrowserSearchService); if (typeof engine == "undefined"){ // If the search bar is visible, use the current engine, otherwise, fall // back to the default engine. //if (isElementVisible(this.searchBar)) engine = ss.currentEngine; //else // engine = ss.defaultEngine; } var submission = engine.getSubmission(searchText, null); // HTML response // getSubmission can return null if the engine doesn't have a URL // with a text/html response type. This is unlikely (since // SearchService._addEngineToStore() should fail for such an engine), // but let's be on the safe side. if (!submission) return; if (useNewTab) { gBrowser.loadOneTab(submission.uri.spec, null, null, submission.postData, window.fullScreen?false:null, false); } else loadURI(submission.uri.spec, null, submission.postData, false); } searchMenu.setAttribute("oncommand", "this.loadSearch(getBrowserSelection(), true, this.engine);"); // ready to show icon in search menu document.getElementById("context-searchselect").className = "menuitem-iconic"; }());
какая-нибудь рабочая бета-версия CB есть? |
oleg.sgh > 24-01-2017 15:38:33 |
1fesFFFF
Так ссылка в сообщении |
sonyas75 > 24-01-2017 15:44:58 |
oleg.sgh |
oleg.sgh > 24-01-2017 15:55:57 |
Здесь умельцы выкладывали исправленную |
Quartz1t > 26-01-2017 12:40:49 |
Скажите пожалуйста, можно ли сделать данную кнопка как на скриншоте? скрытый текст |
Артик > 27-01-2017 13:38:05 |
oleg.sgh пишет
куда следует кинуть вот этот файл? custombuttons@xsms.org.xpi ? |
oleg.sgh > 27-01-2017 13:55:13 |
Артик
Скачать дополнение. На вкладке дополнения - шестеренка - установить дополнение из файла. |
Артик > 27-01-2017 18:11:27 |
oleg.sgh пишет
большое спасибо, помогло |
momo2000 > 26-02-2018 08:07:02 |
Можно ли сделать, чтобы контекстное меню не закрывалось после жмакания по поисковику, т.е. когда поиск открывается в фоне, можно было ещё жмакнуть на 2 и 3 поисковик? |
Coroner > 24-03-2018 12:38:27 |
Исходный код из первого сообщения (https://forum.mozilla-russia.org/viewto … 72#p533572) не захотел работать в 59.0.1 и вставить её, допустим, 59-й строчкой. Чуточку портит радость показ всех поисковых систем, вместо только включенных. |
Dumby > 24-03-2018 16:21:33 |
Coroner скрытый текст Выделить код Код:(async (contextMenu, searchSelect, searchService) => { var fullText = true; Services.search.isInitialized || await new Promise( resolve => Services.search.init(resolve) ); searchSelect.collapsed = true; // удалить стандартный пункт меню для поиска // Создать новый пункт меню для поиска .... var menu = contextMenu.insertBefore(document.createElement('menu'), searchSelect); menu.setAttribute("class", "menu-iconic"); addEventListener("popupshowing", (e)=> menu.hidden = searchSelect.hidden, false, contextMenu); function setMenu() { menu.engine = searchService.currentEngine; menu.setAttribute("label", "Искать в " + menu.engine.name + " или в ..."); menu.setAttribute("image", menu.engine.iconURI.spec ); }; setMenu(); // Создать подменю с поисковиками .... var menuPopup = menu.appendChild(document.createElement("menupopup")); menuPopup.setAttribute('style', 'overflow: scroll'); function setItemsToMenuPopup(e) { menuPopup.textContent = ""; var engines = searchService.getVisibleEngines({}) .filter(engine => engine != searchService.currentEngine); var hiddenList = Services.prefs.getStringPref("browser.search.hiddenOneOffs").split(","); engines.forEach((engine)=> { if (hiddenList.includes(engine.name)) return; var mItem = document.createElement("menuitem"); mItem.setAttribute("label", engine.name); mItem.setAttribute("class", "menuitem-iconic"); mItem.setAttribute("src", engine.iconURI.spec); mItem.engine = engine; menuPopup.appendChild(mItem); }) }; setItemsToMenuPopup(); // установить действие для клика на меню и подменю menu.onmouseup = e => { var background = e.button != 0; var text = gContextMenuContentData.selectionInfo[fullText ? "fullText" : "text"]; contextMenu.hidePopup(); var submission = e.target.engine.getSubmission(text, null); gBrowser.loadOneTab(submission.uri.spec, null, null, submission.postData, background, false); } // Наблюдатель за изменениями в поисковиках пересоздаёт меню и подменю .... var getEngineModified = { observe:(subject, topic, data)=> { if ( /changed|removed|current/.test(data) ) { setMenu(), setItemsToMenuPopup() }; } }; Services.obs.addObserver(getEngineModified, "browser-search-engine-modified", false); Services.prefs.addObserver("browser.search.hiddenOneOffs", setItemsToMenuPopup); // Удалять наблюдатели и меню, показать стандартный пункт .... addDestructor(()=> { menu.remove(); searchSelect.collapsed = false; Services.obs.removeObserver(getEngineModified, "browser-search-engine-modified"); Services.prefs.removeObserver("browser.search.hiddenOneOffs", setItemsToMenuPopup); }); })(document.getElementById("contentAreaContextMenu"), document.getElementById("context-searchselect"), Services.search); |
Coroner > 24-03-2018 20:58:04 |
Dumby, спасибо добрый человек. Работает как надо. |
Coroner > 24-10-2018 01:17:10 |
И этот чудный код отвалился. Кому помешал? Если так, то не ли способа вроде https://forum.mozilla-russia.org/viewto … 86#p761786 , только для loadOneTab? |
Dumby > 24-10-2018 04:55:37 |
Coroner скрытый текст Выделить код Код:/* contextMenu.hidePopup(); var submission = e.target.engine.getSubmission(text, null); gBrowser.loadOneTab(submission.uri.spec, null, null, submission.postData, background, false); */ var {principal} = gContextMenu; contextMenu.hidePopup(); var submission = e.target.engine.getSubmission(text, null); gBrowser.loadOneTab(submission.uri.spec, { postData: submission.postData, inBackground: background, triggeringPrincipal: principal }); |
Coroner > 24-10-2018 11:07:56 |
Dumby, нет, поведение не изменилось. Может кто-нибудь проверит? |
Yeesha > 24-10-2018 11:32:47 |
Coroner скрытый текст https://addons.mozilla.org/ru/firefox/a … xt-search/ То же, что делает этот скрипт + можно настроить особое контекстное меню для ссылки (например, поиск в Archive.org), для изображения (поиск в Google image search), для текущей страницы (например, перевести страницу в Google Translate), поиск в Google по текущему сайту и др. |
Coroner > 24-10-2018 13:16:05 |
Yeesha, он у меня установлен (про запас). Просто привык минимумом дополнений обходиться. |
_backup > 16-02-2020 19:10:59 |
Странное поведение кода на FF 56, а именно ищется выделенный текст, который искался до него. Как это можно исправить? Проблему решил кнопкой Context Search 2. |
Maxut > 04-03-2020 00:57:56 |
_backup пишет
Поделись |
_backup > 16-03-2020 19:35:39 |
Maxut пишет
|