// ==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 // @downloadURL none // ==/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 Nightly で ES.next で追加されている String.prototype の一部が実装された - hogehoge @teramako */ 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(); } } } })();