Skip to content

Instantly share code, notes, and snippets.

@2k1dmg 2k1dmg/context-search-2.js
Last active Jan 28, 2016

Embed
What would you like to do?
(function() {
// Context Search 2 2016-01-19
'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 prefsServices = Services.prefs;
if (!prefsServices.getBoolPref('intl.locale.matchOS')) {
try {
let locale = prefsServices.getCharPref('general.useragent.locale');
if (locale.substr(0, 9) == 'chrome://') {
return prefsServices.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 = Services.search;
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 = Services.io;
},
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) {};
},
getSelection: function() {
if ('gBrowser' in window &&
gBrowser.selectedTab.linkedBrowser.isRemoteBrowser) {
return gContextMenuContentData.selectionInfo.text;
}
else if ('BrowserUtils' in window) {
return BrowserUtils.getSelectionDetails(window).text;
}
var fe = document.commandDispatcher.focusedElement;
if(fe) try {
return fe.value.substring(fe.selectionStart, fe.selectionEnd);
}
catch(e) {}
return document.commandDispatcher.focusedWindow.getSelection();
},
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.getSelection();
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.getSelection();
if (!selectedText)
return;
let ellipsis = '\u2026';
try {
ellipsis = prefsServices.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 getMenuLabel = function(str) {
let bundle;
try {
if (this.isSeaMonkey) {
bundle = document.getElementById('contentAreaCommandsBundle');
}
else {
bundle = gNavigatorBundle;
}
return bundle.getFormattedString(str, [engineName, selectedText]);
}
catch (ex) {
return false;
}
};
for (let i = 0, len = searchFormattedString.length; i < len; i++) {
menuLabel = getMenuLabel.call(this, 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();
}());
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
You can’t perform that action at this time.