Greasy Fork

来自缓存

Greasy Fork is available in English.

Google 検索窓を複製

インスタント検索無効時、検索窓をページ下部にも表示

当前为 2014-04-05 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name           Google 検索窓を複製
// @namespace      http://userscripts.org/users/347021
// @version        2.1.0
// @description    インスタント検索無効時、検索窓をページ下部にも表示
// @include        http://www.google.*/search*
// @include        https://www.google.*/search*
// @run-at         document-start
// @grant          none
// @icon           
// @screenshot     
// @author         100の人
// @license        Creative Commons Attribution 3.0 Unported License
// ==/UserScript==

(function() {
'use strict';

// @include 補助
if (!(/^www\.google\.(?:com|(?:com?\.)?[a-z]{2})$/.test(document.domain) && window.location.pathname === '/search')) {
	return;
}



// String::contains (Opera / Google Chrome)
if (!('contains' in String.prototype)) {
	/**
	 * Determines whether one string may be found within another string, returning true or false as appropriate.
	 * 
	 * @param {string} searchString A string to be searched for within this string.
	 * @param {number} [position] The position in this string at which to begin searching for searchString; defaults to 0.
	 * @returns {boolean}
	 * @see <a href="http://d.hatena.ne.jp/teramako/20120807/p1">Nightly で ES.next で追加されている String.prototype の一部が実装された - hogehoge @teramako</a>
	 */
	String.prototype.contains = function (searchString, position) {
		return (typeof position === "number" && position ? this.substr(position) : this).indexOf(searchString) >= 0;
	};
}

// DOMTokenList::add/remove (Firefox / Opera)
if ('mozFullScreenEnabled' in document || 'opera' in window) {
	var _add = DOMTokenList.prototype.add, _remove = DOMTokenList.prototype.remove;
	DOMTokenList.prototype.add = function() {
		for (var i = 0, l = arguments.length; i < l; i++) {
			_add.call(this, arguments[i]);
		}
	};
	DOMTokenList.prototype.remove = function() {
		for (var i = 0, l = arguments.length; i < l; i++) {
			_remove.call(this, arguments[i]);
		}
	};
}



// body要素挿入時に実行し、Google検索のバージョンを判別する
var textBoxId, inputNodeId, inputParentNodesClassName, textBoxBorderClass, classOnfocuse, previousSiblingId;
startScript(function() {
	var functionsAccordingToBrowsers, body = document.body;
	
	if (body.id) {
		if (window.location.search.contains('tbm=isch')) {
			// 画像検索ページなら実行しない
			return;
		}
		if (body.getAttribute('marginheight')) {
			// User-AgentがFirefox
			textBoxId = 'tsf';
			inputNodeId = 'lst-ib';
			inputParentNodesClassName = 'lst-d';
			textBoxBorderClass = 'lst-td';
			classOnfocuse = ['lst-d-f'];
		} else {
			// User-AgentがOpera、Google Chrome、IE8以降
			textBoxId = 'gbqf';
			inputNodeId = 'gbqfq';
			inputParentNodesClassName = 'gbqfqwc';
			textBoxBorderClass = 'gbqfqw';
			classOnfocuse = ['gbqfqwf', 'gsfe_b'];
		}
		previousSiblingId = 'xjs';
		functionsAccordingToBrowsers = {
			firefox: {
				isTargetParent: function (parent) {
					return parent.classList.contains('mw');
				},
				isTarget: function (target) {
					var firstElementChild = target.firstElementChild;
					return firstElementChild && firstElementChild.id === 'foot';
				},
			},
			google: {
				isTargetParent: function (parent) {
					return parent.id === 'foot';
				},
				isTarget: function (target) {
					return target.id === 'xjs';
				},
			},
		};
	} else {
		// その他のUser-Agent、又はJavaScriptが無効
		textBoxId = 'tsf';
		previousSiblingId = 'nav';
		functionsAccordingToBrowsers = {
			firefox: {
				isTargetParent: function (parent) {
					return parent.localName === 'tbody' && parent.parentNode.id === 'mn';
				},
				isTarget: function (target) {
					var cells = target.cells;
					return cells && cells[0] && cells[0].id === 'leftnav';
				},
			},
			google: {
				isTargetParent: function (parent) {
					return parent.id === 'foot';
				},
				isTarget: function (target) {
					return target.id === 'nav';
				},
			},
		};
	}
	startScript(main, null, null, function() {
		return document.getElementById(previousSiblingId);
	}, functionsAccordingToBrowsers);
}, function(parent) {
	return parent.localName === 'html';
}, function(target) {
	return target.localName === 'body';
}, function() {
	return document.body;
}, {});

function main() {
	var style, sheet, cssRules, original, previousSibling,
			bottomForm, textBoxBorder, textBoxBorderClassList, inputParentNodes, submitButton, submitButtonClassList;

	// スタイルの設定
	style = document.createElement('style');
	document.head.appendChild(style);
	sheet = style.sheet;
	cssRules = sheet.cssRules;
	[
		'#foot form {'
				+ 'margin-top: 13px;'
				+ '}',
		// 以降 Firefox
		'#foot .nojsv {'
				+ 'display: none;'
				+ '}',
		'#foot .tsf-p {'
				+ 'width: 631px;'
				+ 'padding-left: 8px;'
				+ '}',
	].forEach(function(rule) {
		sheet.insertRule(rule, cssRules.length);
	});

	// 検索ボックスを取得
	original = document.getElementById(textBoxId);
	if (!original) {
		return;
	}

	// 複製
	bottomForm = original.cloneNode(true);
	
	// 移動先を取得
	previousSibling = document.getElementById(previousSiblingId);
	
	// 挿入
	previousSibling.parentNode.insertBefore(bottomForm, previousSibling.nextSibling);
	
	// ページ描画後のスクリプトによる書き換えを待機
	if (inputParentNodesClassName) {
		inputParentNodes = document.getElementsByClassName(inputParentNodesClassName);
		startScript(function() {
			// 後から挿入された検索窓を複製
			var table = inputParentNodes[0].firstElementChild.cloneNode(true), input = table.getElementsByTagName('input')[0];
			// オートコンプリートを有効に
			input.removeAttribute('autocomplete');
			// 下の検索窓を置き換え
			inputParentNodes[1].replaceChild(table, inputParentNodes[1].firstElementChild);
		}, function(parent) {
			return parent.id === 'gs_lc0';
		}, function(target) {
			return target.id === inputNodeId;
		}, function() {
			return document.querySelector('#' + inputNodeId + '[style]');
		});
	}
	
	// 検索窓にフォーカスが移った時
	if (textBoxBorderClass) {
		textBoxBorder = bottomForm.getElementsByClassName(textBoxBorderClass)[0];
		textBoxBorderClassList = textBoxBorder.classList;
		textBoxBorder.addEventListener('focus', function() {
			DOMTokenList.prototype.add.apply(textBoxBorderClassList, classOnfocuse);
		}, true);

		textBoxBorder.addEventListener('blur', function() {
			DOMTokenList.prototype.remove.apply(textBoxBorderClassList, classOnfocuse);
		}, true);
		
		// 検索窓をクリックしたとき
		textBoxBorder.addEventListener('click', function(event) {
			if (event.target.localName !== 'input') {
				bottomForm.elements.namedItem('q').focus();
			}
		});
	}
	
	// 検索窓にマウスが載ったとき
	submitButton = bottomForm.getElementsByClassName('gbqfb')[0];
	if (submitButton) {
		submitButtonClassList = submitButton.classList;
		bottomForm.addEventListener('mouseover', function(event) {
			var target = event.target;

			if (textBoxBorder.contains(target)) {
				// 検索窓
				textBoxBorderClassList.add('gbqfqw-hvr', 'gsfe_a');
			} else if (submitButton.contains(target)) {
				// 検索ボタン
				submitButtonClassList.add('gbqfb-hvr');
			}
		});

		bottomForm.addEventListener('mouseout', function(event) {
			var relatedTarget = event.relatedTarget;

			if (!textBoxBorder.contains(relatedTarget)) {
				// 検索窓
				textBoxBorderClassList.remove('gbqfqw-hvr', 'gsfe_a');
			}
			if (!submitButton.contains(relatedTarget)) {
				// 検索ボタン
				submitButtonClassList.remove('gbqfb-hvr');
			}
		});
	}
}



/**
 * 指定した要素が挿入された直後に関数を実行する
 * @param {Function} main 実行する関数
 * @param {Function} isTargetParent 挿入された要素の親要素が、指定した要素の親要素か否かを返す関数。functionsAccordingToBrowsersを指定していれば省略する
 * @param {Function} isTarget 挿入された要素が、指定した要素か否かを返す関数。functionsAccordingToBrowsersを指定していれば省略する
 * @param {Function} existsTarget 指定した要素が存在するか否かを返す関数
 * @param {Object} [functionsAccordingToBrowsers] DOMContentLoaded前のタイミングで1回だけスクリプトを起動させる場合に設定
 * @param {Function} functionsAccordingToBrowsers.firefox.isTargetParent 挿入された要素の親要素が、指定した要素の親要素か否かを返す関数(Firefoxの場合)
 * @param {Function} functionsAccordingToBrowsers.firefox.isTarget 挿入された要素が、指定した要素か否かを返す関数(Firefoxの場合)
 * @param {Function} functionsAccordingToBrowsers.google.isTargetParent 挿入された要素の親要素が、指定した要素の親要素か否かを返す関数(Google Chromeの場合)
 * @param {Function} functionsAccordingToBrowsers.google.isTarget 挿入された要素が、指定した要素か否かを返す関数(Google Chromeの場合)
 * @version 2013-06-08
 */
function startScript(main, isTargetParent, isTarget, existsTarget, functionsAccordingToBrowsers) {
	var observer, flag;
	
	// ブラウザによってDOMContentLoaded前のMutationObserverの挙動が異なるため、isTargetParent、isTargetをブラウザに合わせて変更
	if (functionsAccordingToBrowsers) {
		if ('chrome' in window) {
			// Google Chromeなら
			if (functionsAccordingToBrowsers.google) {
				isTargetParent = functionsAccordingToBrowsers.google.isTargetParent;
				isTarget = functionsAccordingToBrowsers.google.isTarget;
			}
		} else {
			// Firefoxなら
			if (functionsAccordingToBrowsers.firefox) {
				isTargetParent = functionsAccordingToBrowsers.firefox.isTargetParent;
				isTarget = functionsAccordingToBrowsers.firefox.isTarget;
			}
		}
	}
	
	// 指定した要素が既に存在していれば、即実行
	startMain();
	if (flag) {
		return;
	}
	
	if (typeof MutationObserver !== 'undefined') {
		// MutationObserverが利用できる場合
		observer = new MutationObserver(mutationCallback);
		observer.observe(document, {
			childList: true,
			subtree: true,
		});
	} else {
		// MutationObserverが利用できない場合 (Opera)
		checkExistingTarget(0);
	}
	
	if (functionsAccordingToBrowsers) {
		// DOMContentLoadedまでにスクリプトを実行できなかった場合、監視を停止(指定した要素が存在するか確認し、存在すれば実行)
		document.addEventListener('DOMContentLoaded', function stopScript(event) {
			event.target.removeEventListener('DOMContentLoaded', stopScript);
			if (observer) {
				observer.disconnect();
			}
			startMain();
			flag = true;
		});
	}
	
	/**
	 * 指定された要素が挿入されたら、監視を停止し、{@link checkExistingTarget}を実行する
	 * 
	 * @param {MutationRecord[]} mutations a list of MutationRecord objects
	 * @param {MutationObserver} observer the constructed MutationObserver object
	 */
	function mutationCallback(mutations, observer) {
		var mutation, target, addedNodes, addedNode,
				i, j, l, l2;
		for (i = 0, l = mutations.length; i < l; i++) {
			mutation = mutations[i];
			target = mutation.target;
			if (target.nodeType === Node.ELEMENT_NODE && isTargetParent(target)) {
				// 子が追加されたノードがElementノードで、かつそのノードについてisTargetParentが真を返せば
				addedNodes = Array.prototype.slice.call(mutation.addedNodes);
				for (j = 0, l2 = addedNodes.length; j < l2; j++) {
					addedNode = addedNodes[j];
					if (addedNode.nodeType === Node.ELEMENT_NODE && isTarget(addedNode)) {
						// 追加された子がElementノードで、かつそのノードについてisTargetが真を返せば
						observer.disconnect();
						checkExistingTarget(0);
						return;
					}
				}
			}
		}
	}
	
	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければ再度実行
	 * 
	 * @param {integer} count {@link startMain}を実行した回数
	 */
	function checkExistingTarget(count) {
		var LIMIT = 500, INTERVAL = 10;
		startMain();
		if (!flag && count < LIMIT) {
			window.setTimeout(checkExistingTarget, INTERVAL, count + 1);
		}
	}
	
	/**
	 * 指定した要素が存在するか確認し、存在すれば監視を停止しスクリプトを実行
	 */
	function startMain() {
		if (!flag && existsTarget()) {
			flag = true;
			main();
		}
	}
}

})();