Google Translator Tooltip Expanded Fork

Translates the selected text into a tooltip automatically. Based on Google Translation Tooltip MLiteKeysOn.

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name			Google Translator Tooltip Expanded Fork
// @description		Translates the selected text into a tooltip automatically. Based on Google Translation Tooltip MLiteKeysOn.
// @icon           https://raw.githubusercontent.com/Odyseus/GreasemonkeyScripts/master/GoogleTranslatorTooltipExpandedFork/g-translate-logo.png
// @version			1.12
// @include			http*
// @include			https*
// @include			chrome*
// @grant			GM_getValue
// @grant			GM_xmlhttpRequest
// @grant			GM_log
// @grant			GM_deleteValue
// @grant			GM_addStyle
// @grant			GM_openInTab
// @grant			GM_registerMenuCommand
// @grant			GM_setValue
// @namespace https://greasyfork.org/users/5764
// ==/UserScript==
/**
 * ColorPicker - pure JavaScript color picker without using images, external CSS or 1px divs.
 * Copyright © 2011 David Durman, All rights reserved.
 * http://www.daviddurman.com/flexi-color-picker/#
 */
(function (window, document, undefined) {
	var type = (window.SVGAngle || document.implementation.hasFeature("http://www.w3.org/TR/SVG11/feature#BasicStructure", "1.1") ? "SVG" : "VML"),
	picker,
	slide,
	hueOffset = 15,
	svgNS = 'http://www.w3.org/2000/svg';
	// This HTML snippet is inserted into the innerHTML property of the passed color picker element
	// when the no-hassle call to ColorPicker() is used, i.e. ColorPicker(function(hex, hsv, rgb) { ... });
	var colorpickerHTMLSnippet = [
		'<div class="picker-wrapper">',
		'<div class="picker"></div>',
		'<div class="picker-indicator"></div>',
		'</div>',
		'<div class="slide-wrapper">',
		'<div class="slide"></div>',
		'<div class="slide-indicator"></div>',
		'</div>'
	].join('');
	/**
	 * Return mouse position relative to the element el.
	 */
	function mousePosition(evt) {
		// IE:
		if (window.event && window.event.contentOverflow !== undefined) {
			return {
				x : window.event.offsetX,
				y : window.event.offsetY
			};
		}
		// Webkit:
		if (evt.offsetX !== undefined && evt.offsetY !== undefined) {
			return {
				x : evt.offsetX,
				y : evt.offsetY
			};
		}
		// Firefox:
		var wrapper = evt.target.parentNode.parentNode;
		return {
			x : evt.layerX - wrapper.offsetLeft,
			y : evt.layerY - wrapper.offsetTop
		};
	}
	/**
	 * Create SVG element.
	 */
	function $(el, attrs, children) {
		el = document.createElementNS(svgNS, el);
		for (var key in attrs)
			el.setAttribute(key, attrs[key]);
		if (Object.prototype.toString.call(children) != '[object Array]')
			children = [children];
		var i = 0,
		len = (children[0] && children.length) || 0;
		for (; i < len; i++)
			el.appendChild(children[i]);
		return el;
	}
	/**
	 * Create slide and picker markup depending on the supported technology.
	 */
	if (type == 'SVG') {
		slide = $('svg', {
				xmlns : 'http://www.w3.org/2000/svg',
				version : '1.1',
				width : '100%',
				height : '100%'
			},
				[
					$('defs', {},
						$('linearGradient', {
							id : 'gradient-hsv',
							x1 : '0%',
							y1 : '100%',
							x2 : '0%',
							y2 : '0%'
						},
							[
								$('stop', {
									offset : '0%',
									'stop-color' : '#FF0000',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '13%',
									'stop-color' : '#FF00FF',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '25%',
									'stop-color' : '#8000FF',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '38%',
									'stop-color' : '#0040FF',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '50%',
									'stop-color' : '#00FFFF',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '63%',
									'stop-color' : '#00FF40',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '75%',
									'stop-color' : '#0BED00',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '88%',
									'stop-color' : '#FFFF00',
									'stop-opacity' : '1'
								}),
								$('stop', {
									offset : '100%',
									'stop-color' : '#FF0000',
									'stop-opacity' : '1'
								})
							])),
					$('rect', {
						x : '0',
						y : '0',
						width : '100%',
						height : '100%',
						fill : 'url(#gradient-hsv)'
					})
				]);
		picker = $('svg', {
				xmlns : 'http://www.w3.org/2000/svg',
				version : '1.1',
				width : '100%',
				height : '100%'
			},
				[
					$('defs', {},
						[
							$('linearGradient', {
								id : 'gradient-black',
								x1 : '0%',
								y1 : '100%',
								x2 : '0%',
								y2 : '0%'
							},
								[
									$('stop', {
										offset : '0%',
										'stop-color' : '#000000',
										'stop-opacity' : '1'
									}),
									$('stop', {
										offset : '100%',
										'stop-color' : '#CC9A81',
										'stop-opacity' : '0'
									})
								]),
							$('linearGradient', {
								id : 'gradient-white',
								x1 : '0%',
								y1 : '100%',
								x2 : '100%',
								y2 : '100%'
							},
								[
									$('stop', {
										offset : '0%',
										'stop-color' : '#FFFFFF',
										'stop-opacity' : '1'
									}),
									$('stop', {
										offset : '100%',
										'stop-color' : '#CC9A81',
										'stop-opacity' : '0'
									})
								])
						]),
					$('rect', {
						x : '0',
						y : '0',
						width : '100%',
						height : '100%',
						fill : 'url(#gradient-white)'
					}),
					$('rect', {
						x : '0',
						y : '0',
						width : '100%',
						height : '100%',
						fill : 'url(#gradient-black)'
					})
				]);
	} else if (type == 'VML') {
		slide = [
			'<DIV style="position: relative; width: 100%; height: 100%">',
			'<v:rect style="position: absolute; top: 0; left: 0; width: 100%; height: 100%" stroked="f" filled="t">',
			'<v:fill type="gradient" method="none" angle="0" color="red" color2="red" colors="8519f fuchsia;.25 #8000ff;24903f #0040ff;.5 aqua;41287f #00ff40;.75 #0bed00;57671f yellow"></v:fill>',
			'</v:rect>',
			'</DIV>'
		].join('');
		picker = [
			'<DIV style="position: relative; width: 100%; height: 100%">',
			'<v:rect style="position: absolute; left: -1px; top: -1px; width: 101%; height: 101%" stroked="f" filled="t">',
			'<v:fill type="gradient" method="none" angle="270" color="#FFFFFF" opacity="100%" color2="#CC9A81" o:opacity2="0%"></v:fill>',
			'</v:rect>',
			'<v:rect style="position: absolute; left: 0px; top: 0px; width: 100%; height: 101%" stroked="f" filled="t">',
			'<v:fill type="gradient" method="none" angle="0" color="#000000" opacity="100%" color2="#CC9A81" o:opacity2="0%"></v:fill>',
			'</v:rect>',
			'</DIV>'
		].join('');
		if (!document.namespaces['v'])
			document.namespaces.add('v', 'urn:schemas-microsoft-com:vml', '#default#VML');
	}
	/**
	 * Convert HSV representation to RGB HEX string.
	 * Credits to http://www.raphaeljs.com
	 */
	function hsv2rgb(hsv) {
		var R,
		G,
		B,
		X,
		C;
		var h = (hsv.h % 360) / 60;
		C = hsv.v * hsv.s;
		X = C * (1 - Math.abs(h % 2 - 1));
		R = G = B = hsv.v - C;
		h = ~~h;
		R += [C, X, 0, 0, X, C][h];
		G += [X, C, C, X, 0, 0][h];
		B += [0, 0, X, C, C, X][h];
		var r = Math.floor(R * 255);
		var g = Math.floor(G * 255);
		var b = Math.floor(B * 255);
		return {
			r : r,
			g : g,
			b : b,
			hex : "#" + (16777216 | b | (g << 8) | (r << 16)).toString(16).slice(1)
		};
	}
	/**
	 * Convert RGB representation to HSV.
	 * r, g, b can be either in <0,1> range or <0,255> range.
	 * Credits to http://www.raphaeljs.com
	 */
	function rgb2hsv(rgb) {
		var r = rgb.r;
		var g = rgb.g;
		var b = rgb.b;
		if (rgb.r > 1 || rgb.g > 1 || rgb.b > 1) {
			r /= 255;
			g /= 255;
			b /= 255;
		}
		var H,
		S,
		V,
		C;
		V = Math.max(r, g, b);
		C = V - Math.min(r, g, b);
		H = (C == 0 ? null :
			V == r ? (g - b) / C + (g < b ? 6 : 0) :
			V == g ? (b - r) / C + 2 :
			(r - g) / C + 4);
		H = (H % 6) * 60;
		S = C == 0 ? 0 : C / V;
		return {
			h : H,
			s : S,
			v : V
		};
	}
	/**
	 * Return click event handler for the slider.
	 * Sets picker background color and calls ctx.callback if provided.
	 */
	function slideListener(ctx, slideElement, pickerElement) {
		return function (evt) {
			evt = evt || window.event;
			var mouse = mousePosition(evt);
			ctx.h = mouse.y / slideElement.offsetHeight * 360 + hueOffset;
			var pickerColor = hsv2rgb({
					h : ctx.h,
					s : 1,
					v : 1
				});
			var c = hsv2rgb({
					h : ctx.h,
					s : ctx.s,
					v : ctx.v
				});
			pickerElement.style.backgroundColor = pickerColor.hex;
			ctx.callback && ctx.callback(c.hex, {
				h : ctx.h - hueOffset,
				s : ctx.s,
				v : ctx.v
			}, {
				r : c.r,
				g : c.g,
				b : c.b
			}, undefined, mouse);
		}
	};
	/**
	 * Return click event handler for the picker.
	 * Calls ctx.callback if provided.
	 */
	function pickerListener(ctx, pickerElement) {
		return function (evt) {
			evt = evt || window.event;
			var mouse = mousePosition(evt),
			width = pickerElement.offsetWidth,
			height = pickerElement.offsetHeight;
			ctx.s = mouse.x / width;
			ctx.v = (height - mouse.y) / height;
			var c = hsv2rgb(ctx);
			ctx.callback && ctx.callback(c.hex, {
				h : ctx.h - hueOffset,
				s : ctx.s,
				v : ctx.v
			}, {
				r : c.r,
				g : c.g,
				b : c.b
			}, mouse);
		}
	};
	var uniqID = 0;
	/**
	 * ColorPicker.
	 * @param {DOMElement} slideElement HSV slide element.
	 * @param {DOMElement} pickerElement HSV picker element.
	 * @param {Function} callback Called whenever the color is changed provided chosen color in RGB HEX format as the only argument.
	 */
	function ColorPicker(slideElement, pickerElement, callback) {
		if (!(this instanceof ColorPicker))
			return new ColorPicker(slideElement, pickerElement, callback);
		this.h = 0;
		this.s = 1;
		this.v = 1;
		if (!callback) {
			// call of the form ColorPicker(element, funtion(hex, hsv, rgb) { ... }), i.e. the no-hassle call.
			var element = slideElement;
			element.innerHTML = colorpickerHTMLSnippet;
			this.slideElement = element.getElementsByClassName('slide')[0];
			this.pickerElement = element.getElementsByClassName('picker')[0];
			var slideIndicator = element.getElementsByClassName('slide-indicator')[0];
			var pickerIndicator = element.getElementsByClassName('picker-indicator')[0];
			ColorPicker.fixIndicators(slideIndicator, pickerIndicator);
			this.callback = function (hex, hsv, rgb, pickerCoordinate, slideCoordinate) {
				ColorPicker.positionIndicators(slideIndicator, pickerIndicator, slideCoordinate, pickerCoordinate);
				pickerElement(hex, hsv, rgb);
			};
		} else {
			this.callback = callback;
			this.pickerElement = pickerElement;
			this.slideElement = slideElement;
		}
		if (type == 'SVG') {
			// Generate uniq IDs for linearGradients so that we don't have the same IDs within one document.
			// Then reference those gradients in the associated rectangles.
			var slideClone = slide.cloneNode(true);
			var pickerClone = picker.cloneNode(true);
			var hsvGradient = slideClone.getElementById('gradient-hsv');
			var hsvRect = slideClone.getElementsByTagName('rect')[0];
			hsvGradient.id = 'gradient-hsv-' + uniqID;
			hsvRect.setAttribute('fill', 'url(#' + hsvGradient.id + ')');
			var blackAndWhiteGradients = [pickerClone.getElementById('gradient-black'), pickerClone.getElementById('gradient-white')];
			var whiteAndBlackRects = pickerClone.getElementsByTagName('rect');
			blackAndWhiteGradients[0].id = 'gradient-black-' + uniqID;
			blackAndWhiteGradients[1].id = 'gradient-white-' + uniqID;
			whiteAndBlackRects[0].setAttribute('fill', 'url(#' + blackAndWhiteGradients[1].id + ')');
			whiteAndBlackRects[1].setAttribute('fill', 'url(#' + blackAndWhiteGradients[0].id + ')');
			this.slideElement.appendChild(slideClone);
			this.pickerElement.appendChild(pickerClone);
			uniqID++;
		} else {
			this.slideElement.innerHTML = slide;
			this.pickerElement.innerHTML = picker;
		}
		addEventListener(this.slideElement, 'click', slideListener(this, this.slideElement, this.pickerElement));
		addEventListener(this.pickerElement, 'click', pickerListener(this, this.pickerElement));
		enableDragging(this, this.slideElement, slideListener(this, this.slideElement, this.pickerElement));
		enableDragging(this, this.pickerElement, pickerListener(this, this.pickerElement));
	};
	function addEventListener(element, event, listener) {
		if (element.attachEvent) {
			element.attachEvent('on' + event, listener);
		} else if (element.addEventListener) {
			element.addEventListener(event, listener, false);
		}
	}
	/**
	 * Enable drag&drop color selection.
	 * @param {object} ctx ColorPicker instance.
	 * @param {DOMElement} element HSV slide element or HSV picker element.
	 * @param {Function} listener Function that will be called whenever mouse is dragged over the element with event object as argument.
	 */
	function enableDragging(ctx, element, listener) {
		var mousedown = false;
		addEventListener(element, 'mousedown', function (evt) {
			mousedown = true;
		});
		addEventListener(element, 'mouseup', function (evt) {
			mousedown = false;
		});
		addEventListener(element, 'mouseout', function (evt) {
			mousedown = false;
		});
		addEventListener(element, 'mousemove', function (evt) {
			if (mousedown) {
				listener(evt);
			}
		});
	}
	ColorPicker.hsv2rgb = function (hsv) {
		var rgbHex = hsv2rgb(hsv);
		delete rgbHex.hex;
		return rgbHex;
	};
	ColorPicker.hsv2hex = function (hsv) {
		return hsv2rgb(hsv).hex;
	};
	ColorPicker.rgb2hsv = rgb2hsv;
	ColorPicker.rgb2hex = function (rgb) {
		return hsv2rgb(rgb2hsv(rgb)).hex;
	};
	ColorPicker.hex2hsv = function (hex) {
		return rgb2hsv(ColorPicker.hex2rgb(hex));
	};
	ColorPicker.hex2rgb = function (hex) {
		return {
			r : parseInt(hex.substr(1, 2), 16),
			g : parseInt(hex.substr(3, 2), 16),
			b : parseInt(hex.substr(5, 2), 16)
		};
	};
	/**
	 * Sets color of the picker in hsv/rgb/hex format.
	 * @param {object} ctx ColorPicker instance.
	 * @param {object} hsv Object of the form: { h: <hue>, s: <saturation>, v: <value> }.
	 * @param {object} rgb Object of the form: { r: <red>, g: <green>, b: <blue> }.
	 * @param {string} hex String of the form: #RRGGBB.
	 */
	function setColor(ctx, hsv, rgb, hex) {
		ctx.h = hsv.h % 360;
		ctx.s = hsv.s;
		ctx.v = hsv.v;
		var c = hsv2rgb(ctx);
		var mouseSlide = {
			y : (ctx.h * ctx.slideElement.offsetHeight) / 360,
			x : 0 // not important
		};
		var pickerHeight = ctx.pickerElement.offsetHeight;
		var mousePicker = {
			x : ctx.s * ctx.pickerElement.offsetWidth,
			y : pickerHeight - ctx.v * pickerHeight
		};
		ctx.pickerElement.style.backgroundColor = hsv2rgb({
				h : ctx.h,
				s : 1,
				v : 1
			}).hex;
		ctx.callback && ctx.callback(hex || c.hex, {
			h : ctx.h,
			s : ctx.s,
			v : ctx.v
		}, rgb || {
			r : c.r,
			g : c.g,
			b : c.b
		}, mousePicker, mouseSlide);
		return ctx;
	};
	/**
	 * Sets color of the picker in hsv format.
	 * @param {object} hsv Object of the form: { h: <hue>, s: <saturation>, v: <value> }.
	 */
	ColorPicker.prototype.setHsv = function (hsv) {
		return setColor(this, hsv);
	};
	/**
	 * Sets color of the picker in rgb format.
	 * @param {object} rgb Object of the form: { r: <red>, g: <green>, b: <blue> }.
	 */
	ColorPicker.prototype.setRgb = function (rgb) {
		return setColor(this, rgb2hsv(rgb), rgb);
	};
	/**
	 * Sets color of the picker in hex format.
	 * @param {string} hex Hex color format #RRGGBB.
	 */
	ColorPicker.prototype.setHex = function (hex) {
		return setColor(this, ColorPicker.hex2hsv(hex), undefined, hex);
	};
	/**
	 * Helper to position indicators.
	 * @param {HTMLElement} slideIndicator DOM element representing the indicator of the slide area.
	 * @param {HTMLElement} pickerIndicator DOM element representing the indicator of the picker area.
	 * @param {object} mouseSlide Coordinates of the mouse cursor in the slide area.
	 * @param {object} mousePicker Coordinates of the mouse cursor in the picker area.
	 */
	ColorPicker.positionIndicators = function (slideIndicator, pickerIndicator, mouseSlide, mousePicker) {
		if (mouseSlide) {
			slideIndicator.style.top = (mouseSlide.y - slideIndicator.offsetHeight / 2) + 'px';
		}
		if (mousePicker) {
			pickerIndicator.style.top = (mousePicker.y - pickerIndicator.offsetHeight / 2) + 'px';
			pickerIndicator.style.left = (mousePicker.x - pickerIndicator.offsetWidth / 2) + 'px';
		}
	};
	/**
	 * Helper to fix indicators - this is recommended (and needed) for dragable color selection (see enabledDragging()).
	 */
	ColorPicker.fixIndicators = function (slideIndicator, pickerIndicator) {
		pickerIndicator.style.pointerEvents = 'none';
		slideIndicator.style.pointerEvents = 'none';
	};
	window.ColorPicker = ColorPicker;
})(window, window.document);
const HREF_NO = 'javascript:void(0)';
initCrossBrowserSupportForGmFunctions();
var languagesGoogle = '<option value="auto">Detect language</option><option value="af">Afrikaans</option><option value="sq">Albanian</option><option value="ar">Arabic</option><option value="hy">Armenian</option><option value="az">Aerbaijani</option><option value="eu">Basque</option><option value="be">Belarusian</option><option value="bn">Bengali</option><option value="bg">Bulgarian</option><option value="ca">Catalan</option><option value="zh-CN">Chinese (simplified)</option><option value="zh-TW">Chinese (traditional)</option><option value="hr">Croatian</option><option value="cs">Czech</option><option value="da">Danish</option><option value="nl">Dutch</option><option value="en">English</option><option value="et">Estonian</option><option value="tl">Filipino</option><option value="fi">Finnish</option><option value="fr">French</option><option value="gl">Galician</option><option value="ka">Georgian</option><option value="de">German</option><option value="el">Greek</option><option value="ht">Haitian Creole</option><option value="iw">Hebrew</option><option value="hi">Hindi</option><option value="hu">Hungarian</option><option value="is">Icelandic</option><option value="id">Indonesian</option><option value="ga">Irish</option><option value="it">Italian</option><option  value="ja">Japanese</option><option value="ko">Korean</option><option value="lv">Latvian</option><option value="lt">Lithuanian</option><option value="mk">Macedonian</option><option value="ms">Malay</option><option value="mt">Maltese</option><option value="no">Norwegian</option><option value="fa">Persian</option><option value="pl">Polish</option><option value="pt">Portuguese</option><option value="ro">Romanian</option><option value="ru">Russian</option><option value="sr">Serbian</option><option value="sk">Slovak</option><option  value="sl">Slovenian</option><option value="es">Spanish</option><option value="sw">Swahili</option><option value="sv">Swedish</option><option value="th">Thai</option><option value="tr">Turkish</option><option value="uk">Ukrainian</option><option value="ur">Urdu</option><option value="vi">Vietnamese</option><option value="cy">Welsh</option><option value="yi">Yiddish</option>';
var body = getTag('body')[0];
var imgLookup;
var txtSel = encodeURIComponent(txtSel); // text selected
var translation2Element = document.createElement('span');
var currentURL;
var initialized = false;
images();
css();
document.addEventListener('mouseup', showLookupIcon, false);
document.addEventListener('mousedown', mousedownCleaning, false);
function mousedownCleaning(evt) {
	var divDic = getId('divDic');
	var divLookup = getId('divLookup');
	if (divDic) {
		if (!clickedInsideID(evt.target, 'divDic'))
			divDic.parentNode.removeChild(divDic);
	}
	if (divLookup)
		divLookup.parentNode.removeChild(divLookup);
}
function showLookupIcon(evt) {
	if (evt.ctrlKey && evt.altKey && (!GM_getValue('ctrl') || !GM_getValue('alt')))
		return;
	// XOR http://www.howtocreate.co.uk/xor.html
	if (evt.ctrlKey ? !GM_getValue('ctrl') : GM_getValue('ctrl'))
		return;
	if (evt.altKey ? !GM_getValue('alt') : GM_getValue('alt'))
		return;
	if (!initialized) {
		images();
		css();
		initialized = true;
	}
	var divDic = getId('divDic');
	var divLookup = getId('divLookup');
	txtSel = getSelection();
	// Exit if no text is selected
	if (!txtSel || txtSel == "") {
		if (divDic) {
			if (!clickedInsideID(evt.target, 'divDic'))
				divDic.parentNode.removeChild(divDic);
		}
		if (divLookup)
			divLookup.parentNode.removeChild(divLookup);
		return;
	}
	// Possible cleanup
	if (divDic) {
		if (!clickedInsideID(evt.target, 'divDic'))
			divDic.parentNode.removeChild(divDic);
		return;
	}
	// Remove div if exists
	if (divLookup) {
		divLookup.parentNode.removeChild(divLookup);
	}
	// Div container
	divLookup = createElement('div', {
			id : 'divLookup',
			style : 'background-color:transparent; color:#000000; position:absolute; top:' + (evt.clientY + window.pageYOffset + 10) + 'px; left:' + (evt.clientX + window.pageXOffset + 10) + 'px; padding:0px; z-index:10000; border-radius:2px;'
		});
	divLookup.appendChild(imgLookup.cloneNode(false));
	divLookup.addEventListener('mouseover', lookup, false);
	body.appendChild(divLookup);
}
// Create the tooltip and launch the Google Translate request to get the translation
function lookup(evt) {
	var divResult = null;
	var divDic = getId('divDic');
	var divLookup = getId('divLookup');
	var top = divLookup.style.top;
	var left = divLookup.style.left;
	// No text selected
	if (!txtSel || txtSel == "") {
		if (divDic = getId('divDic'))
			divDic.parentNode.removeChild(divDic);
		return;
	}
	// Cleanup divs
	if (divDic = getId('divDic')) {
		divDic.parentNode.removeChild(divDic);
	}
	divLookup.parentNode.removeChild(divLookup);
	// Div container
	divDic = createElement('div', {
			id : 'divDic',
			style : 'opacity: 1; font-family: "Open Sans", Arial, Helvetica, sans-serif !important; font-size: ' + GM_getValue('fontsize', 'small') + '; background-color: ' + GM_getValue('backgroundColor', '#EDF4FC') + '; color: ' + GM_getValue('textcolor', 'Gray') + '; position:absolute; top:' + top + '; left:' + left + '; min-width:250px; min-height:50px; max-width:50%; padding:5px; text-align:left; z-index:10000; border-radius:4px; box-shadow: -2px 0px 9px 5px #898D91'
		});
	divDic.addEventListener('mousedown', dragHandler, false);
	body.appendChild(divDic);
	// Div result
	// This awfull wall of text is the "+" image
	divResult = createElement('div', {
			id : 'divResult',
			style : 'overflow:auto; padding:3px;'
		}, null, '<img src="' +
			'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"/><br/>Loading...');
	divDic.appendChild(divResult);
	// Options link
	var optionLink = createElement('a', {
			id : 'optionsLink',
			href : HREF_NO,
			style : 'opacity:0.2; position:absolute; bottom:3px; right:13px; font-size:18px; text-decoration:none!important;background:#528DDF;padding:1px;color:#fff;border-radius:6px 6px 6px 6px;border:2px solid #EEEEEE;font-weight:bold;width:20px;text-align:center;display:block;'
		}, 'click openCloseOptions false', '+');
	divDic.appendChild(optionLink);
	optionLink.addEventListener('mouseover', function (e) {
		e.target.style.opacity = 1.0
	});
	optionLink.addEventListener('mouseout', function (e) {
		e.target.style.opacity = 0.2
	});
	// Send the Google Translate request
	if ((txtSel + " ").search(/^\s*https?:\/\//) > -1) {
		divResult.innerHTML = '<a href="' + txtSel + '" target="_blank" >' + txtSel + '</a>';
	} else if ((txtSel + " ").search(/^\s*\S+(\.\S+)+/) > -1) // site.dom
	{
		divResult.innerHTML = '<a style="color:#888;" href="http://' + txtSel + '" target="_blank" >' + txtSel + '</a>';
	} else {
		var sl,
		tl,
		lang;
		sl = GM_getValue('from') ? GM_getValue('from') : "auto";
		tl = GM_getValue('to') ? GM_getValue('to') : "auto";
		lang = sl + "|" + tl;
		//currentURL = "http://www.google.com/translate_t?text=" + encodeURIComponent(txtSel) + "&langpair=" + lang; // Basic address, for web page parsing
		//currentURL = "http://translate.google.fr/translate_a/t?client=t&text=" + encodeURIComponent(txtSel) + "&langpair=" + lang; // URL for GET request. This adress return an array as answer
		currentPostData = "client=t&text=" + encodeURIComponent(txtSel) + "&langpair=" + lang; // Data for a POST request, for handling long requests
		GM_xmlhttpRequest({
			/*method: 'GET',
			url: currentURL,*/
			method : 'POST',
			url : 'http://translate.google.fr/translate_a/t',
			data : currentPostData,
			headers : {
				'Content-Type' : 'application/x-www-form-urlencoded'
			},
			onload : function (resp) {
				try {
					extractResult(resp.responseText);
				} catch (e) {
					GM_log(e);
				}
			}
		});
		if (GM_getValue('to2', 'Disabled') != 'Disabled') {
			sl = GM_getValue('from') ? GM_getValue('from') : "auto";
			tl = GM_getValue('to2') ? GM_getValue('to2') : "auto";
			lang = sl + "|" + tl;
			currentPostData = "client=t&text=" + encodeURIComponent(txtSel) + "&langpair=" + lang; // Data for a POST request, for handling long requests
			GM_xmlhttpRequest({
				method : 'POST',
				url : 'http://translate.google.fr/translate_a/t',
				data : currentPostData,
				headers : {
					'Content-Type' : 'application/x-www-form-urlencoded'
				},
				onload : function (resp) {
					try {
						extractResult2(resp.responseText);
					} catch (e) {
						GM_log(e);
					}
				}
			});
		} else {
			translation2Element.innerHTML = '';
		}
	}
}
// Lanched when we select an other language in the setup menu
function quickLookup() {
	getId('divDic').style.fontSize = getId('optFontSize').value;
	getId('divDic').style.color = getId('optTextColor').value;
	getId('divResult').innerHTML = 'Loading...';
	var sl,
	tl,
	lang;
	sl = getId('optSelLangFrom').value;
	tl = getId('optSelLangTo').value;
	lang = sl + "|" + tl;
	currentPostData = "client=t&text=" + encodeURIComponent(txtSel) + "&langpair=" + lang; // Data for a POST request, for handling long requests
	GM_xmlhttpRequest({
		method : 'POST',
		url : 'http://translate.google.fr/translate_a/t',
		data : currentPostData,
		headers : {
			'Content-Type' : 'application/x-www-form-urlencoded'
		},
		onload : function (resp) {
			try {
				extractResult(resp.responseText);
			} catch (e) {
				GM_log(e);
			}
		}
	});
	if (getId('optSelLangTo2').value != 'Disabled') {
		var sl,
		tl,
		lang;
		sl = getId('optSelLangFrom').value;
		tl = getId('optSelLangTo2').value;
		currentPostData = "client=t&text=" + encodeURIComponent(txtSel) + "&langpair=" + lang; // Data for a POST request, for handling long requests
		GM_xmlhttpRequest({
			method : 'POST',
			url : 'http://translate.google.fr/translate_a/t',
			data : currentPostData,
			headers : {
				'Content-Type' : 'application/x-www-form-urlencoded'
			},
			onload : function (resp) {
				try {
					extractResult2(resp.responseText);
				} catch (e) {
					GM_log(e);
				}
			}
		});
	} else {
		translation2Element.innerHTML = '';
	}
}
function extractResult(gTradStringArray) {
	var arr = eval(gTradStringArray); // eval is used to transform the string to an array. I alse made a custom parsing function, but it doesn't handle antislashed characters, so I prefer using eval()
	/*
	Content of the gTrad array :
	0 / 0:Translation 1:Source text
	1 / i:Grammar / 0:Types (word, verb, ...) 1: Other translations
	5 / Array of other translations
	 */
	var translation = '';
	// 0 - Full translation
	translation += '<small><a href="https://translate.google.com/#' + GM_getValue('from', 'auto') + '/' + GM_getValue('to', 'auto') + '/' + txtSel + '">[' + arr[2] + '] ';
	for (var i = 0; i < arr[0].length; i++)
		translation += arr[0][i][1];
	translation += '</a> <span id="texttospeachbuttonfrom"></span></small><br/>';
	translation += '[' + GM_getValue('to', 'auto') + ']<em> ';
	for (var i = 0; i < arr[0].length; i++)
		translation += arr[0][i][0];
	translation += '</em> <span id="texttospeachbuttonto"></span><br/><span id="translation2Element"></span><br/>';
	translation += '<a id="toggleShowDetails" ' + (!GM_getValue('details', 'false') ? 'style="display:none"' : '') + '>Show details</a>';
	translation += '<span id="divDetails" ' + (GM_getValue('details', 'false') ? 'style="display:none"' : '') + '><a id="toggleHideDetails">Hide details</a><br/>';
	// 1 - Grammar
	if (typeof arr[1] != 'undefined') {
		for (var i = 0; i < arr[1].length; i++) {
			translation += '<strong>' + arr[1][i][0] + ' : </strong>';
			for (var j = 0; j < arr[1][i][1].length; j++) {
				translation += ((j == 0) ? '' : ', ') + arr[1][i][1][j];
			}
			translation += '<br/>';
		}
		translation += '<br/>';
	}
	// 5 - Alternative parts
	if (typeof arr[5] != 'undefined') {
		for (var i = 0; i < arr[5].length; i++) {
			if (typeof arr[5][i][2] != 'undefined') { // 5/i/2 array of alternatives, 5/i/0 the part of the text we are studying
				translation += '<strong>' + arr[5][i][0] + ' : </strong>';
				for (var j = 0; j < arr[5][i][2].length; j++) {
					translation += ((j == 0) ? '' : ', ') + arr[5][i][2][j][0];
				}
				translation += '<br/>';
			}
		}
	}
	translation += '</span>'; // Detail end
	getId('divResult').innerHTML = '<p style="margin:0px">' + translation + '</p>';
	getId('translation2Element').appendChild(translation2Element); // Optional second translation
	getId('toggleShowDetails').addEventListener('click', function () {
		getId('toggleShowDetails').style.display = 'none';
		getId('divDetails').style.display = 'block';
	}, false);
	getId('toggleHideDetails').addEventListener('click', function () {
		getId('toggleShowDetails').style.display = 'inline';
		getId('divDetails').style.display = 'none';
	}, false);
	// Create the Text to Speach
	var fromText = '';
	var toText = '';
	for (var i = 0; i < arr[0].length; i++)
		fromText += arr[0][i][1];
	for (var i = 0; i < arr[0].length; i++)
		toText += arr[0][i][0];
	addTextToSpeachLink(getId('texttospeachbuttonfrom'), arr[2], fromText); // arr[2] contains the detected input language
	addTextToSpeachLink(getId('texttospeachbuttonto'), GM_getValue('to', 'auto') == 'auto' ? 'en' : GM_getValue('to', 'auto'), toText); // I cannot find a way to get the detected destination language, so if the requested destination is 'auto', I use the english Text to Speach language
}
function extractResult2(gTradStringArray) {
	var arr = eval(gTradStringArray);
	var translation = '';
	translation += '#[' + GM_getValue('to2', 'auto') + ']<em> ';
	for (var i = 0; i < arr[0].length; i++)
		translation += arr[0][i][0];
	translation += '</em># <span id="texttospeachbuttonto2"></span>';
	translation2Element.innerHTML = translation;
	var toText2 = '';
	for (var i = 0; i < arr[0].length; i++)
		toText2 += arr[0][i][0];
	addTextToSpeachLink(getId('texttospeachbuttonto2'), GM_getValue('to2', 'auto') == 'auto' ? 'en' : GM_getValue('to2', 'auto'), toText2);
}
function addTextToSpeachLink(element, lang, text) {
	if (GM_getValue('tts', false) == false)
		return;
	var speachLink = document.createElement('a');
	speachLink.href = 'http://translate.google.com/translate_tts?tl=' + lang + '&q=' + text.replace(/[«»'"]/g, ' ');
	speachLink.target = '_blank';
	speachLink.innerHTML = '<img src="http://www.trace-de-son.com/images/sound-icon.png" height="16" width="16"/>';
	element.appendChild(speachLink);
}
function getSelection() {
	var txt = null;
	//get selected text
	if (window.getSelection && !window.opera) // window.getSelection() bugs with Opera 12.16 and ViolentMonkey
	{
		txt = window.getSelection();
	} else if (document.getSelection) {
		txt = document.getSelection();
	} else if (document.selection) {
		txt = document.selection.createRange().text;
	}
	return txt;
}
function openCloseOptions(evt) {
	var divOptions = getId('divOpt');
	if (!divOptions) //Show options
	{
		divOptions = createElement('div', {
				id : 'divOpt',
				style : 'border-top:2px solid #5A91D8;position:relative; padding:5px;'
			});
		getId('divDic').appendChild(divOptions);
		getId('optionsLink').style.visibility = 'hidden';
		// color picker, the library doesn't work on Opera
		try {
			if (!window.divColorPicker) {
				window.divColorPicker = createElement('div', {
						id : 'optPicker',
						class : 'cp-small'
					});
				window.colorPicker = ColorPicker(
						window.divColorPicker,
						function (hex, hsv, rgb) {
						getId('divDic').style.backgroundColor = hex;
					});
			}
			window.colorPicker.setHex(GM_getValue('backgroundColor', '#EDF4FC'));
			divOptions.appendChild(window.divColorPicker);
		} catch (err) {
			divOptions.innerHTML += '<p>Error : Cannot load color picker (Known issue on Opera)</p>';
		}
		//fields container
		divOptionsFields = createElement('p');
		divOptions.appendChild(divOptionsFields);
		//from
		divOptionsFields.appendChild(createElement('span', null, null, 'From :'));
		divOptionsFields.appendChild(createElement('select', {
				id : 'optSelLangFrom'
			}, null, languagesGoogle));
		getId('optSelLangFrom').value = GM_getValue('from') ? GM_getValue('from') : 'auto';
		getId('optSelLangFrom').addEventListener('change', quickLookup, false);
		//to
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('span', null, null, ' To :'));
		divOptionsFields.appendChild(createElement('select', {
				id : 'optSelLangTo'
			}, null, languagesGoogle));
		getId('optSelLangTo').value = GM_getValue('to') ? GM_getValue('to') : 'auto';
		getId('optSelLangTo').addEventListener('change', quickLookup, false);
		//to2
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('span', null, null, ' To 2 :'));
		divOptionsFields.appendChild(createElement('select', {
				id : 'optSelLangTo2'
			}, null, '<option value="Disabled">Disabled</option>' + languagesGoogle));
		getId('optSelLangTo2').value = GM_getValue('to2') ? GM_getValue('to2') : 'Disabled';
		getId('optSelLangTo2').addEventListener('change', quickLookup, false);
		//use text to speach
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('input', {
				id : 'checkTTS',
				type : 'checkbox'
			}));
		divOptionsFields.appendChild(createElement('span', null, null, '<span title="The feature has many issues. You often have to refresh the page to launch the .mp3 file. If you use the langage auto-detection, you have to change the langage in the url of the new tab." style="border-bottom:1px dashed">Display Text To Speach</span>'));
		getId('checkTTS').checked = GM_getValue('tts');
		//hide details
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('input', {
				id : 'checkDetails',
				type : 'checkbox'
			}));
		divOptionsFields.appendChild(createElement('span', null, null, 'Hide details by default'));
		getId('checkDetails').checked = GM_getValue('details');
		//font size
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('span', null, null, 'Font size :'));
		divOptionsFields.appendChild(createElement('select', {
				id : 'optFontSize'
			}, null, '<option value="x-small">Extra small</option><option value="small">Small (default)</option><option value="medium">Medium</option><option value="large">Large</option>'));
		getId('optFontSize').value = GM_getValue('fontsize') ? GM_getValue('fontsize') : 'small';
		getId('optFontSize').addEventListener('change', quickLookup, false);
		//text color
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('span', null, null, 'Text color :'));
		divOptionsFields.appendChild(createElement('select', {
				id : 'optTextColor'
			}, null, '<option value="Gray">Gray (default)</option><option value="Black">Black</option><option value="White">White</option><option value="CadetBlue">CadetBlue</option><option value="ForestGreen">ForestGreen</option><option value="FireBrick">FireBrick</option>'));
		getId('optTextColor').value = GM_getValue('textcolor') ? GM_getValue('textcolor') : 'Gray';
		getId('optTextColor').addEventListener('change', quickLookup, false);
		//use ctrl
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('input', {
				id : 'checkCtrl',
				type : 'checkbox'
			}));
		divOptionsFields.appendChild(createElement('span', null, null, 'Use Ctrl key'));
		getId('checkCtrl').checked = GM_getValue('ctrl');
		//use alt
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('input', {
				id : 'checkAlt',
				type : 'checkbox'
			}));
		divOptionsFields.appendChild(createElement('span', null, null, 'Use Alt key'));
		getId('checkAlt').checked = GM_getValue('alt');
		//save
		divOptionsFields.appendChild(createElement('br'));
		divOptionsFields.appendChild(createElement('a', {
				href : HREF_NO,
				class : "gootranslink"
			}, 'click saveOptions false', 'Save'));
		//reset
		divOptionsFields.appendChild(createElement('span', null, null, ' - '));
		divOptionsFields.appendChild(createElement('a', {
				href : HREF_NO,
				class : "gootranslink"
			}, 'click resetOptions false', 'Reset'));
		//cancel
		divOptionsFields.appendChild(createElement('span', null, null, ' - '));
		divOptionsFields.appendChild(createElement('a', {
				href : HREF_NO,
				class : "gootranslink"
			}, 'click openCloseOptions false', 'Cancel'));
	} else // Hide options
	{
		divOptions.parentNode.removeChild(divOptions);
		getId('optionsLink').style.visibility = 'visible';
	}
}
function saveOptions(evt) {
	var backgroundColor = getId('divDic').style.backgroundColor;
	var from = getId('optSelLangFrom').value;
	var to = getId('optSelLangTo').value;
	var to2 = getId('optSelLangTo2').value;
	var tts = getId('checkTTS').checked;
	var details = getId('checkDetails').checked;
	var fontsize = getId('optFontSize').value;
	var textcolor = getId('optTextColor').value;
	var ctrl = getId('checkCtrl').checked;
	var alt = getId('checkAlt').checked;
	GM_setValue('backgroundColor', backgroundColor);
	GM_setValue('from', from);
	GM_setValue('to', to);
	GM_setValue('to2', to2);
	GM_setValue('tts', tts);
	GM_setValue('details', details);
	GM_setValue('fontsize', fontsize);
	GM_setValue('textcolor', textcolor);
	GM_setValue('ctrl', ctrl);
	GM_setValue('alt', alt);
	quickLookup();
	getId('divDic').removeChild(getId('divOpt'));
	getId('optionsLink').style.visibility = 'visible';
}
function resetOptions(evt) {
	GM_deleteValue('backgroundColor');
	GM_deleteValue('from');
	GM_deleteValue('to');
	GM_deleteValue('to2');
	GM_deleteValue('tts');
	GM_deleteValue('fontsize');
	GM_deleteValue('textcolor');
	GM_deleteValue('ctrl');
	GM_deleteValue('alt');
	getId('divDic').parentNode.removeChild(getId('divDic'));
}
function css() {
	var style = createElement('style', {
			type : "text/css"
		}, null, "" +
			'a.gootranslink:link {color: #0000FF !important; text-decoration: underline !important;}' +
			'a.gootranslink:visited {color: #0000FF !important; text-decoration: underline !important;}' +
			'a.gootranslink:hover {color: #0000FF !important; text-decoration: underline !important;}' +
			'a.gootranslink:active {color: #0000FF !important; text-decoration: underline !important;}' +
			'.picker-wrapper,.slide-wrapper{position:relative;float:left}.picker-indicator,.slide-indicator{position:absolute;left:0;top:0;pointer-events:none}.picker,.slide{cursor:crosshair;float:left}.cp-default{background-color:gray;padding:12px;box-shadow:0 0 40px #000;border-radius:15px;float:left}.cp-default .picker{width:200px;height:200px}.cp-default .slide{width:30px;height:200px}.cp-default .slide-wrapper{margin-left:10px}.cp-default .picker-indicator{width:5px;height:5px;border:2px solid darkblue;-moz-border-radius:4px;-o-border-radius:4px;-webkit-border-radius:4px;border-radius:4px;opacity:.5;-ms-filter:"alpha(opacity=50)";filter:alpha(opacity=50);filter:alpha(opacity=50);background-color:white}.cp-default .slide-indicator{width:100%;height:10px;left:-4px;opacity:.6;-ms-filter:"alpha(opacity=60)";filter:alpha(opacity=60);filter:alpha(opacity=60);border:4px solid lightblue;-moz-border-radius:4px;-o-border-radius:4px;-webkit-border-radius:4px;border-radius:4px;background-color:white}.cp-small{padding:5px;background-color:white;float:left;border-radius:5px}.cp-small .picker{width:100px;height:100px}.cp-small .slide{width:15px;height:100px}.cp-small .slide-wrapper{margin-left:5px}.cp-small .picker-indicator{width:1px;height:1px;border:1px solid black;background-color:white}.cp-small .slide-indicator{width:100%;height:2px;left:0;background-color:black}.cp-fancy{padding:10px;background:-webkit-linear-gradient(top,#aaa 0,#222 100%);float:left;border:1px solid #999;box-shadow:inset 0 0 10px white}.cp-fancy .picker{width:200px;height:200px}.cp-fancy .slide{width:30px;height:200px}.cp-fancy .slide-wrapper{margin-left:10px}.cp-fancy .picker-indicator{width:24px;height:24px;background-image:url(http://cdn1.iconfinder.com/data/icons/fugue/bonus/icons-24/target.png)}.cp-fancy .slide-indicator{width:30px;height:31px;left:30px;background-image:url(http://cdn1.iconfinder.com/data/icons/bluecoral/Left.png)}.cp-normal{padding:10px;background-color:white;float:left;border:4px solid #d6d6d6;box-shadow:inset 0 0 10px white}.cp-normal .picker{width:200px;height:200px}.cp-normal .slide{width:30px;height:200px}.cp-normal .slide-wrapper{margin-left:10px}.cp-normal .picker-indicator{width:5px;height:5px;border:1px solid gray;opacity:.5;-ms-filter:"alpha(opacity=50)";filter:alpha(opacity=50);filter:alpha(opacity=50);background-color:white;pointer-events:none}.cp-normal .slide-indicator{width:100%;height:10px;left:-4px;opacity:.6;-ms-filter:"alpha(opacity=60)";filter:alpha(opacity=60);filter:alpha(opacity=60);border:4px solid gray;background-color:white;pointer-events:none}');
	getTag('head')[0].appendChild(style);
}
/*
 * Short functions to replace the document.createElement document.getElementById and document.getElementsByTagName
 */
function createElement(type, attrArray, evtListener, html) {
	var node = document.createElement(type);
	for (var attr in attrArray)
		if (attrArray.hasOwnProperty(attr)) {
			node.setAttribute(attr, attrArray[attr]);
		}
	if (evtListener) {
		var a = evtListener.split(' ');
		node.addEventListener(a[0], eval(a[1]), eval(a[2]));
	}
	if (html)
		node.innerHTML = html;
	return node;
}
function getId(id, parent) {
	if (!parent)
		return document.getElementById(id);
	return parent.getElementById(id);
}
function getTag(name, parent) {
	if (!parent)
		return document.getElementsByTagName(name);
	return parent.getElementsByTagName(name);
}
/*
 * Drag and drop support adapted from http://www.hunlock.com/blogs/Javascript_Drag_and_Drop
 */
var savedTarget = null; // The target layer (effectively vidPane)
var orgCursor = null; // The original mouse style so we can restore it
var dragOK = false; // True if we're allowed to move the element under mouse
var dragXoffset = 0; // How much we've moved the element on the horozontal
var dragYoffset = 0; // How much we've moved the element on the verticle
var didDrag = false; // Set to true when we do a drag
function moveHandler(e) {
	if (e == null)
		return; // { e = window.event }
	if (e.button <= 1 && dragOK) {
		savedTarget.style.left = e.clientX - dragXoffset + 'px';
		savedTarget.style.top = e.clientY - dragYoffset + 'px';
		return false;
	}
}
function dragCleanup(e) {
	document.removeEventListener('mousemove', moveHandler, false);
	document.removeEventListener('mouseup', dragCleanup, false);
	savedTarget.style.cursor = orgCursor;
	dragOK = false; // Its been dragged now
	didDrag = true;
}
function dragHandler(e) {
	var htype = '-moz-grabbing';
	if (e == null)
		return; // { e = window.event;}  // htype='move';}
	var target = e.target; // != null ? e.target : e.srcElement;
	orgCursor = target.style.cursor;
	if (target.nodeName != 'DIV' && target.nodeName != 'P')
		return;
	if (target = clickedInsideID(target, 'divDic')) {
		savedTarget = target;
		target.style.cursor = htype;
		dragOK = true;
		dragXoffset = e.clientX - target.offsetLeft;
		dragYoffset = e.clientY - target.offsetTop;
		// Set the left before removing the right
		target.style.left = e.clientX - dragXoffset + 'px';
		target.style.right = null;
		document.addEventListener('mousemove', moveHandler, false);
		document.addEventListener('mouseup', dragCleanup, false);
		return false;
	}
}
function clickedInsideID(target, id) {
	if (target.getAttribute('id') == id)
		return getId(id);
	if (target.parentNode) {
		while (target = target.parentNode) {
			try {
				if (target.getAttribute('id') == id)
					return getId(id);
			} catch (e) {}
		}
	}
	return null;
}
// End drag code
/*
 * Images
 */
function images() {
	imgLookup = createElement('img', {
			border : 0
		});
	imgLookup.src = '';
}
if (typeof GM_deleteValue == 'undefined') {
	GM_addStyle = function (css) {
		var style = document.createElement('style');
		style.textContent = css;
		document.getElementsByTagName('head')[0].appendChild(style);
	}
	GM_deleteValue = function (name) {
		localStorage.removeItem(name);
	}
	GM_getValue = function (name, defaultValue) {
		var value = localStorage.getItem(name);
		if (!value)
			return defaultValue;
		var type = value[0];
		value = value.substring(1);
		switch (type) {
		case 'b':
			return value == 'true';
		case 'n':
			return Number(value);
		default:
			return value;
		}
	}
	GM_log = function (message) {
		console.log(message);
	}
	GM_openInTab = function (url) {
		return window.open(url, "_blank");
	}
	GM_registerMenuCommand = function (name, funk) {
		//todo
	}
	GM_setValue = function (name, value) {
		value = (typeof value)[0] + value;
		localStorage.setItem(name, value);
	}
}
/*
 * Cross browser support for GM functions
 * http://userscripts.org/topics/41177
 */
function initCrossBrowserSupportForGmFunctions() {
	if (typeof GM_deleteValue == 'undefined') {
		GM_addStyle = function (css) {
			var style = document.createElement('style');
			style.textContent = css;
			document.getElementsByTagName('head')[0].appendChild(style);
		}
		GM_deleteValue = function (name) {
			localStorage.removeItem(name);
		}
		GM_getValue = function (name, defaultValue) {
			var value = localStorage.getItem(name);
			if (!value)
				return defaultValue;
			var type = value[0];
			value = value.substring(1);
			switch (type) {
			case 'b':
				return value == 'true';
			case 'n':
				return Number(value);
			default:
				return value;
			}
		}
		GM_log = function (message) {
			console.log(message);
		}
		GM_openInTab = function (url) {
			return window.open(url, "_blank");
		}
		GM_registerMenuCommand = function (name, funk) {
			//todo
		}
		GM_setValue = function (name, value) {
			value = (typeof value)[0] + value;
			localStorage.setItem(name, value);
		}
	}
}