человек сделал javascript, который реализовал поиск по форуму через google именно такой функцией.
Для FF тоже такое есть - Search in Website+
Хотя, наверное, возможность, при зажатом Ctrl или чего-то такого, производить поиск по сайту в выбранной поисковой системе, была бы интересной, правда не знаю поддерживают ли всякие дакдаки и яндексы такую фичу...
Отсутствует
Можешь добавить в текст пункта меню выделенное/искомое слово?
// 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); })();
Отредактировано 2k1dmg (22-09-2013 19:42:14)
Отсутствует
dedmazai1870, в основной версии Context Search так и сделано, там стоит - в пункт меню выводится не более 16 символов из выделенного текста.
Добавлено 21-09-2013 23:10:33
2k1dmg, вот теперь именно то, что я хотел.
Отредактировано PEAKTOP (22-09-2013 19:38:59)
1. Приму из добрых рук щедрый дар - инвайт на Хабр и Лепру
2. Бессмысленно осмысливать смысл неосмысленными мыслями!
3. Прежде чем подумать - подумай!
Отсутствует
Zaycoff
А вот объединить 2 кнопки как? Чтобы с списку моих поисков добавлялся гуглопоиск последней строкой "Включает строку поиска выделенного текста на текущем сайте". Вот чего хочется.
Отсутствует
Отсутствует
Добавил "Поиск по текущему сайту" через Яндекс
//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 ); })();
Отредактировано PEAKTOP (24-09-2013 19:40:26)
1. Приму из добрых рук щедрый дар - инвайт на Хабр и Лепру
2. Бессмысленно осмысливать смысл неосмысленными мыслями!
3. Прежде чем подумать - подумай!
Отсутствует
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'); }); })();
Отсутствует
Context Search 2013-09-23
А зачем ты изменяешь меню при закрытии :
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;
Отредактировано bunda1 (23-09-2013 22:10:44)
Отсутствует
2k1dmg, есть одно неудобство. Если я зафиксирую в пункте меню "Google", то он будет дублироваться и в подменю. Понятно, что ты берешь список поисковиков из "Панели поиска". Есть возможность исключить "зафиксированный" поисковик из подменю?
Отредактировано PEAKTOP (25-09-2013 00:40:20)
1. Приму из добрых рук щедрый дар - инвайт на Хабр и Лепру
2. Бессмысленно осмысливать смысл неосмысленными мыслями!
3. Прежде чем подумать - подумай!
Отсутствует
PEAKTOP
Там же есть options (с описанием настроек), где можно изменить:
hideDefaultEngineInList: true
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(); })();
Отредактировано 2k1dmg (25-09-2013 16:47:42)
Отсутствует
bunda1
Убрал изменение меню при закрытии и
добавил "Поиск по изображени"
Получается отличная кнопка, но было бы не плохо если меню поисковиков открывалась и в текстовых полях, для этого можно поставить наблюдатель на пункт Копировать в контекстном меню страницы:
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 ) });
И наверно можно обойтись одним обработчиком:
Отредактировано bunda1 (25-09-2013 20:50:30)
Отсутствует
hideDefaultEngineInList: true
Извини, не заметил.
1. Приму из добрых рук щедрый дар - инвайт на Хабр и Лепру
2. Бессмысленно осмысливать смысл неосмысленными мыслями!
3. Прежде чем подумать - подумай!
Отсутствует
Создана новая кнопка и по умолчанию скрыта = скрипт не работает.
А теперь вопрос: можно ли сделать, чтобы она работала или это ограничения дополнения/браузера?
upd:
в http://forum.mozilla-russia.org/viewtop … 09#p628009 показываются только 3 поиска по сайту (Google, Yandex, Bing), должно же быть пять или нет?
} 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); }
+
Сделайте пожалуйста, чтобы учитывало эту настройку - user_pref("browser.search.context.loadInBackground", true);
Отредактировано Northtech (08-10-2013 18:56:35)
Отсутствует
Northtech, если у тебя в панели поиска не присутствуют эти поисковики или их названия отличаются от указанных в коде , то они не будут отображаться в контекстном меню.
1. Приму из добрых рук щедрый дар - инвайт на Хабр и Лепру
2. Бессмысленно осмысливать смысл неосмысленными мыслями!
3. Прежде чем подумать - подумай!
Отсутствует
Northtech, если у тебя в панели поиска не присутствуют эти поисковики или их названия отличаются от указанных в коде , то они не будут отображаться в контекстном меню.
Ага, спасибо, поисковики присутствуют, но названия другие.
Но насчет настройки (browser.search.context.loadInBackground) пожелание все же остаётся.
Отсутствует
Northtech
только вместо browser.search.context.loadInBackground
добавил options.loadInBackground
(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(); })();
Отсутствует
2k1dmg
Я в полнейшем восторге Спасибо!
upd: что-то "на лету" в контекстном меню не меняется поисковик по умолчанию (если изменять выбор в панели поиска), а только после перезагрузки браузера.
На всякий случай: использую hideDefaultEngineInPopupSubMenu: true, но pinnedEngineName: "" не трогаю.
p.s. если не сильно понятно расписал, могу потом показать с картинками.
Отредактировано Northtech (09-10-2013 21:07:00)
Отсутствует
2k1dmg
Крутой код получился, пора переименовать в Context Search full
Кстати это:
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"
Отредактировано bunda1 (09-10-2013 23:30:00)
Отсутствует
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
а XULNS и XHTMLNS пусть будут
Отсутствует
2k1dmg и bunda1 совместными усилиями вы сделали очень удобную и функциональную кнопку. БОЛЬШОЕ ВАМ СПАСИБО!
Вот ещё бы для пункта отображаемого при клике на картинке сделать возможность задать поисковик по умолчанию и сделать его активным - осуществлять поиск этим поисковиком при клике на пункте контекстного меню (без захода в подменю). И тогда кнопка будет универсальная на все 100%.
Отредактировано PEAKTOP (10-10-2013 21:27:04)
1. Приму из добрых рук щедрый дар - инвайт на Хабр и Лепру
2. Бессмысленно осмысливать смысл неосмысленными мыслями!
3. Прежде чем подумать - подумай!
Отсутствует