Append Tag Searching Tub

『niconico』の検索窓にタグ検索タブを追加 / Adds "tag" search tab above search box in "niconico"

目前為 2014-04-05 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name           Append Tag Searching Tub
// @namespace      http://loda.jp/script/
// @id             niconico-adds-search-tab-347021
// @version        3.0.1
// @description    『niconico』の検索窓にタグ検索タブを追加 / Adds "tag" search tab above search box in "niconico"
// @match          http://www.nicovideo.jp/*
// @match          http://seiga.nicovideo.jp/search/*
// @match          http://live.nicovideo.jp/*
// @match          http://watch.live.nicovideo.jp/*
// @match          http://com.nicovideo.jp/*
// @match          http://blog.nicovideo.jp/en_info/*
// @match          http://tw.blog.nicovideo.jp/*
// @match          http://info.nicovideo.jp/psvita/en/*
// @grant          GM_xmlhttpRequest
// @domain         www.nicovideo.jp
// @domain         seiga.nicovideo.jp
// @domain         live.nicovideo.jp
// @domain         watch.live.nicovideo.jp
// @domain         com.nicovideo.jp
// @domain         blog.nicovideo.jp
// @domain         tw.blog.nicovideo.jp
// @domain         info.nicovideo.jp
// @run-at         document-start
// @icon           
// @author         100の人 https://userscripts.org/users/347021
// @license        Creative Commons Attribution 3.0 Unported License
// ==/UserScript==

(function () {
'use strict';

polyfill();

// L10N
setLocalizedTexts({
	'en': {
		'静画を検索': 'Search Image',
		'静画': 'Images',
		'生放送を検索': 'Search Live Program',
		'生放送': 'Live',
		'タグ': 'Tags',
		'マンガ': 'Comics',
	},
	'zh': {
		'静画を検索': '搜尋靜畫',
		'静画': '靜畫',
		'生放送を検索': '搜尋生放送',
		'生放送': '生放送',
		'タグ': '標籤',
		'マンガ': '漫畫',
	},
});

/**
 * 検索窓の最大幅
 * @constant {string}
 */
var MAX_SEARCH_BOX_WIDTH = '268px';



var host = window.location.host, pathname = window.location.pathname, pageType,
		targetParentIdFirefox, isTargetParentFirefox, isTargetFirefox;

// 検索ページの種類を取得
switch (host) {
	case 'www.nicovideo.jp':
		if (window.location.pathname === '/') {
			pageType = 'top';
		} else if (/^\/(?:search\/|mylist_search(?:\/|$))/.test(pathname)) {
			pageType = 'video';
		} else if (/^\/(?:(?:tag|related_tag|watch|mylist)\/|(?:recent|newarrival|hotlist|video_top|openlist|playlist|recommendations)(?:\/|$))/.test(pathname)) {
			pageType = 'tag';
		}
		break;
	case 'seiga.nicovideo.jp':
		pageType = 'image';
		break;
	case 'live.nicovideo.jp':
	case 'watch.live.nicovideo.jp':
		if (pathname.startsWith('/search')) {
			pageType = 'live';
		}
		break;
	case 'info.nicovideo.jp':
		 if (pathname.startsWith('/psvita/en/')) {
			 // 英語版PS Vita紹介ページ
			 startScript(prepare,
					function (parent) { return parent.localName === 'body'; },
					function (target) { return target.id === 'header'; },
					function () { return document.getElementById('header'); },
					{
						isTargetParent: function (parent) { return parent.localName === 'html'; },
						isTarget: function (target) { return target.localName === 'body'; },
					});
		 }
		 return;
}

// 上部メニューが追加されるまで待機
switch (host) {
	case 'seiga.nicovideo.jp':
		// 静画
		isTargetFirefox = function (target) { return target.id === 'wrapper'; };
		break;
	case 'live.nicovideo.jp':
		// 生放送
		targetParentIdFirefox = 'body_header';
		break;
	case 'blog.nicovideo.jp':
		// 英語版ニコニコインフォ
		targetParentIdFirefox = 'container-inner';
		break;
	case 'tw.blog.nicovideo.jp':
		// 台湾版ニコニコインフォ
		targetParentIdFirefox = 'header';
		break;
	case 'info.nicovideo.jp':
		break;
}
if (!isTargetParentFirefox) {
	if (targetParentIdFirefox) {
		isTargetParentFirefox = function (parent) { return parent.id === targetParentIdFirefox; };
	} else {
		isTargetParentFirefox = function (parent) { return parent.localName === 'body'; };
	}
}
startScript(prepare,
		function (parent) { return parent.classList.contains('siteHeaderGlovalNavigation'); },
		function (target) { return target.id === 'siteHeaderLeftMenu'; },
		function () { return document.getElementById('siteHeaderLeftMenu'); },
		{
			isTargetParent: isTargetParentFirefox,
			isTarget: isTargetFirefox || function (target) { return target.id === 'siteHeader'; },
		});

function prepare () {
	var parentId, parentIdFirefox, targetId, targetIdFirefox, isTargetParent, isTargetParentFirefox,
			textVideo, harajuku, itemLive, item;

	// ニコニコ生放送ではlang属性値が常にja-JPのため、ニコニコ動画へのリンク文字によって、ページの言語を判定する
	textVideo = document.querySelector('[href^="http://www.nicovideo.jp/video_top"]').textContent;
	if (textVideo.contains('Video')) {
		setlang('en');
	} else if (textVideo.contains('動畫')) {
		setlang('zh');
	}
	
	if (!document.querySelector(pageType === 'image' ? '#siteHeader [href="/?header"], #siteHeader [href="/"]' : '#siteHeader [href^="http://seiga.nicovideo.jp/"], #globalNav [href^="http://seiga.nicovideo.jp/"]')) {
		// ヘッダに静画へのリンクが無ければ
		// 生放送へのリンクを取得
		itemLive = document.querySelector('#siteHeader [href^="http://live.nicovideo.jp/"], #globalNav [href^="http://live.nicovideo.jp/"]').parentNode;
		// 生放送リンクの複製
		item = itemLive.cloneNode(true);
		// リンク文字を変更
		(item.getElementsByTagName('span')[0] || item.getElementsByTagName('a')[0]).textContent = _('静画');
		// アドレスを変更
		item.getElementsByTagName('a')[0].host = 'seiga.nicovideo.jp';
		// ヘッダに静画へのリンクを追加
		itemLive.parentNode.insertBefore(item, itemLive);
	}
	
	// スクリプトを起動
	if (!pageType) {
		return;
	}
	harajuku = document.doctype.publicId;
	switch (pageType) {
		case 'video':
			if (harajuku) {
				// マイリスト検索、キーワード検索
				parentId = 'form_search';
				targetId = 'search_united_form';
				parentIdFirefox = 'PAGEMAIN';
				targetIdFirefox = 'PAGEBODY';
			} else {
				// GINZAバージョンのキーワード検索
				startScript(main,
						function (parent) { return parent.classList.contains('formSearch'); },
						function (target) { return target.id === 'search_united_form'; },
						function () { return document.getElementById('search_united_form'); },
						{
							isTargetParent: function (parent) { return parent.localName === 'body'; },
							isTarget: function (target) { return target.localName === 'section'; },
						});
				return;
			}
			break;
			
		case 'top':
			// トップページ
			main = mainTop;
			parentId = 'searchFormInner';
			targetId = 'searchForm';
			isTargetParentFirefox = function (parent) {
				return parent.id === 'main_container' || parent.localName === 'body';
			};
			targetIdFirefox = 'searchFormWrap';
			break;
			
		case 'image':
			// 静画
			parentId = 'usearch_form';
			targetId = 'usearch_form_input';
			parentIdFirefox = 'wrapper';
			targetIdFirefox = 'main';
			break;
			
		case 'live':
			// 生放送
			isTargetParentFirefox = isTargetParent = function (target) {
				return target.classList.contains('container');
			};
			targetIdFirefox = targetId = 'form_frm_btm';
			break;
			
		case 'tag':
			if (harajuku) {
				// タグ検索等
				main = mainTag;
				parentId = 'search_tab';
				targetId = 'target_m';
				parentIdFirefox = 'PAGEMAIN';
				targetIdFirefox = 'PAGEBODY';
			} else {
				// GINZAバージョンのタグ検索
				startScript(mainTag,
						function (parent) { return parent.classList.contains('videoSearchOption'); },
						function (target) { return target.classList.contains('optMylist'); },
						function () { return document.getElementsByClassName('optMylist')[0]; },
						{
							isTargetParent: function (parent) { return parent.localName === 'body'; },
							isTarget: function (target) { return target.localName === 'header'; },
						});
				return;
			}
	}
	startScript(main,
			isTargetParent || function (parent) { return parent.id === parentId; },
			function (target) { return target.id === targetId; },
			function () { return document.getElementById(targetId); },
			{
				isTargetParent: isTargetParentFirefox || function (parent) { return parent.id === parentIdFirefox; },
				isTarget: function (target) { return target.id === targetIdFirefox; },
			});
}



// タグ検索
function mainTag () {
	var mylistTab, tabList, styleSheet, cssRules, script;
	
	// スタイルの設定
	styleSheet = document.head.appendChild(document.createElement('style')).sheet;
	cssRules = styleSheet.cssRules;
	[
		'#PAGEHEADER > div {'
				+ 'display: flex;'
				+ '}',
		'#head_search {'
				+ 'max-width: ' + MAX_SEARCH_BOX_WIDTH + ';'
				+ 'flex-grow: 1;'
				+ '}',
		'#search_input {'
				+ 'width: 100%;'
				+ 'display: flex;'
				+ '}',
		'#search_input .typeText {'
				+ 'flex-grow: 1;'
				+ '}',
		'#head_ads {'
				+ 'margin-right: -26px;'
				+ '}',
		'#search_input #bar_search {'
				+ '-moz-box-sizing: border-box;'
				+ 'box-sizing: border-box;'
				+ 'width: 100% !important;'
				+ '}',
		// GINZAバージョン
		'.siteHeader > .inner {'
				+ 'display: flex;'
				+ '}',
		'.videoSearch {'
				+ 'max-width: ' + MAX_SEARCH_BOX_WIDTH + ';'
				+ 'flex-grow: 1;'
				+ 'padding-left: 4px;'
				+ 'padding-right: 4px;'
				+ '}',
		'.videoSearch form {'
				+ 'display: flex;'
				+ '}',
		'.videoSearch form .inputText {'
				+ 'flex-grow: 1;'
				+ '}',
	].forEach(function (rule) {
		styleSheet.insertRule(rule, cssRules.length);
	});
	
	// タブリストの取得
	mylistTab = document.querySelector('#target_m, .optMylist');
	tabList = mylistTab.parentNode;
	
	// タブの複製・追加
	[
		{
			type: 'image',
			title: _('静画を検索'),
			uri: 'http://seiga.nicovideo.jp/search',
			text: _('静画'),
		},
		{
			type: 'live',
			title: _('生放送を検索'),
			uri: 'http://live.nicovideo.jp/search',
			text: _('生放送'),
		},
	].forEach(function (option) {
		var tab = mylistTab.cloneNode(true);
		if (mylistTab.classList.contains('optMylist')) {
			// GINZAバージョン
			tab.classList.remove('optMylist');
			tab.classList.add('opt' + option.type[0].toUpperCase() + option.type.slice(1));
			tab.dataset.type = option.type;
			tab.getElementsByTagName('a')[0].textContent = option.text;
		} else {
			// 原宿バージョン
			tab.id = 'target_' + option.type[0];
			tab.title = option.title;
			tab.setAttribute('onclick', tab.getAttribute('onclick').replace(/'.+?'/, '\'' + option.uri + '\''));
			tab.textContent = option.text;
		}
		tabList.appendChild(tab);
	});
	
	if (mylistTab.classList.contains('optMylist')) {
		// GINZAバージョン
		script = document.createElement('script');
		script.text = '(' + (function () {
			eval('Nico.Navigation.HeaderSearch.Controller.search = ' + Nico.Navigation.HeaderSearch.Controller.search.toString().replace(/(switch.+?{.+?)(})/, '$1; break;'
					+ 'case "image":'
						+ 'd = "http://seiga.nicovideo.jp/search/" + e; break;'
					+ 'case "live":'
						+ 'd = "http://live.nicovideo.jp/search/" + e; break;'
					+ '$2'));
		}).toString() + ')();';
		document.head.appendChild(script);
	}
}



// トップページ
function mainTop() {
	var styleSheet, cssRules, refItem, item, anchor;
	
	fixPrototypeJavaScriptFramework();
	
	// スタイルの設定
	styleSheet = document.head.appendChild(document.createElement('style')).sheet;
	cssRules = styleSheet.cssRules;
	[
		'#searchFormInner {'
				+ 'width: auto;'
				+ 'margin-left: 136px;'
				+ '}',
	].forEach(function (rule) {
		styleSheet.insertRule(rule, cssRules.length);
	});
	
	// マイリスト検索ボタンの取得
	refItem = document.getElementsByClassName('sMylist')[0].parentNode;
	
	// マイリスト検索ボタンの複製
	item = refItem.cloneNode(true);
	
	// ボタン名を変更
	anchor = item.getElementsByTagName('a')[0];
	anchor.textContent = _('タグ');
	
	// クラス名を変更
	anchor.className = 'sVideo';
	
	// アドレスを変更
	anchor.href = 'http://www.nicovideo.jp/tag/';
	
	// タグ検索ボタンを追加
	refItem.parentNode.insertBefore(item, refItem);
	
	if (!document.getElementsByClassName('sSeiga')[0]) {
		// 静画検索ボタンが存在しなければ
		// 生放送検索の取得
		refItem = document.getElementsByClassName('sLive')[0].parentNode;
		// 生放送検索の複製
		item = refItem.cloneNode(true);
		// ボタン名を変更
		anchor = item.getElementsByTagName('a')[0];
		anchor.textContent = _('静画');
		// クラス名を変更
		anchor.className = 'sSeiga';
		// アドレスを変更
		anchor.href = 'http://seiga.nicovideo.jp/search/';
		// 静画検索を追加
		refItem.parentNode.insertBefore(item, refItem);
		
		startScript(function () {
			var list, item, anchor;
			// メニューの生放送リンクの取得
			list = document.querySelector('.service_main .live').parentNode.parentNode;
			// 生放送リンクの複製
			item = list.cloneNode(true);
			// リンク文字を変更
			anchor = item.getElementsByTagName('a')[0];
			anchor.title = anchor.textContent = _('静画');
			// クラス名を変更
			anchor.classList.remove('live');
			anchor.classList.add('seiga');
			// アドレスを変更
			item.getElementsByTagName('a')[0].href = 'http://seiga.nicovideo.jp/';
			// メニューに静画へのリンクを追加
			list.parentNode.insertBefore(item, list);

			// サブメニューの複製
			item = document.getElementsByClassName('service_sub')[0].cloneNode(true);
			// 2つ目以降の要素を削除
			Array.prototype.forEach.call(item.querySelectorAll('li:first-child ~ li'), function (item) {
				item.parentNode.removeChild(item);
			});
			// リンク文字を変更
			anchor = item.getElementsByTagName('a')[0];
			anchor.title = anchor.textContent = _('マンガ');
			// アドレスを変更
			item.getElementsByTagName('a')[0].href = 'http://seiga.nicovideo.jp/manga/';
			// メニューに静画のサブメニューへのリンクを追加
			list.parentNode.insertBefore(item, list);
		},
				function (parent) { return parent.id === 'sideNav'; },
				function (target) { return target.id === 'trendyTags'; },
				function () { return document.querySelector('#menuService [href="http://live.nicovideo.jp/timetable/"]'); },
				{
					isTarget: function (target) { return target.id === 'NewServiceList'; },
				});
	}
}



// キーワード検索、マイリスト検索、静画検索、生放送検索
function main() {
	var inactiveTab, mylistTab, tagTab, tabNameNode, searchCount, anchor, searchWords = '', searchWordsPattern;
	
	// マイリスト検索タブの取得
	mylistTab = document.querySelector('.tab_table td:nth-of-type(2), #search_frm_a a:nth-of-type(2), .seachFormA a:nth-of-type(2)');
	
	// マイリスト検索タブの複製
	tagTab = mylistTab.cloneNode(true);
	
	// タブ名を変更
	anchor = tagTab.tagName.toLowerCase() === 'a' ? tagTab : tagTab.getElementsByTagName('a')[0];
	tabNameNode = anchor.getElementsByTagName('div');
	tabNameNode = (tabNameNode.length > 0  ? tabNameNode[0].firstChild : anchor.firstChild);
	tabNameNode.data = _('タグ') + (pageType === 'live' ? '(' : ' ( ');
	
	// クラス名を変更・動画件数をリセット
	searchCount = tagTab.querySelector('strong, span');
	if (pageType === 'image') {
		searchCount.classList.remove('search_value_em');
		searchCount.classList.add('search_value');
	} else if (pageType === 'live') {
		searchCount.classList.remove('Redtxt');
	} else{
		searchCount.style.removeProperty('color');
	}
	searchCount.textContent = '-';
	
	if (searchCount.id) {
		// 生放送
		searchCount.id = 'search_count_tag';
	}

	// 検索語句を取得
	searchWordsPattern = /(?:\/(?:search|tag|mylist_search)\/|[?&]keyword=)([^?&#]+)/g;
	if (searchWords = window.location.href.match(searchWordsPattern)) {
		searchWords = searchWordsPattern.exec(searchWords[pageType === 'live' ? searchWords.length - 1 : 0])[1];
	}
	
	// タグが付いた動画件数を取得・表示
	if (searchWords) {
		GM_xmlhttpRequest({
			method: 'GET',
			url: 'http://www.nicovideo.jp/tag/' + searchWords,
			onload: function (response) {
				var responseDocument, total, trimmedThousandsSep;
				responseDocument = new DOMParser().parseFromString(response.responseText, 'text/html');
				if (!responseDocument) {
					// Blink
					// Issue 265379: DOMParser + text/html does not work <http://code.google.com/p/chromium/issues/detail?id=265379>
					responseDocument = document.implementation.createHTMLDocument();
					responseDocument.documentElement.innerHTML = response.responseText;
				}
				total = responseDocument.title.contains('(原宿)')
						// 原宿バージョン
						? /[,0-9]+/.exec(responseDocument.getElementsByClassName('searchTagTotal')[0].textContent)[0]
						// GINZAバージョン
						: responseDocument.querySelector('.tagCaption .dataValue .num').textContent;
				trimmedThousandsSep = total.replace(/,/g, '');
				if (trimmedThousandsSep >= 100) {
					if (pageType === 'image') {
						searchCount.classList.remove('search_value');
						searchCount.classList.add('search_value_em');
					} else if (pageType === 'live') {
						searchCount.classList.add('Redtxt');
					} else {
						searchCount.style.color = '#CC0000';
					}
				}
				searchCount.textContent = pageType === 'live' ? trimmedThousandsSep : (pageType === 'image' ? total : ' ' + total + ' ');
			}
		});
	}
	
	// 非アクティブタブを取得
	inactiveTab = document.querySelector('.tab_0, .tab1');
	
	// クラス名を変更
	anchor.className = inactiveTab.className;
	
	// アドレスを変更
	anchor.href = 'http://www.nicovideo.jp/tag/' + searchWords + inactiveTab.search;
	
	// タグ検索タブを追加
	mylistTab.parentNode.insertBefore(tagTab, mylistTab);
	if (inactiveTab.classList.contains('tab1')) {
		// GINZAバージョン
		mylistTab.parentNode.insertBefore(tagTab.previousSibling.cloneNode(true), mylistTab);
	}
}



/**
 * 挿入された節の親節が、目印となる節の親節か否かを返すコールバック関数
 * @callback isTargetParent
 * @param {(Document|Element)} parent
 * @returns {boolean}
 */

/**
 * 挿入された節が、目印となる節か否かを返すコールバック関数
 * @callback isTarget
 * @param {(DocumentType|Element)} target
 * @returns {boolean}
 */

/**
 * 目印となる節が文書に存在するか否かを返すコールバック関数
 * @callback existsTarget
 * @returns {boolean}
 */

/**
 * 目印となる節が挿入された直後に関数を実行する
 * @param {Function} main - 実行する関数
 * @param {isTargetParent} isTargetParent
 * @param {isTarget} isTarget
 * @param {existsTarget} existsTarget
 * @param {Object} [callbacksForFirefox] - DOMContentLoaded前のタイミングで1回だけスクリプトを起動させる場合に設定
 * @param {isTargetParent} [callbacksForFirefox.isTargetParent] - FirefoxにおけるisTargetParent
 * @param {isTarget} [callbacksForFirefox.isTarget] - FirefoxにおけるisTarget
 * @version 2013-09-23
 */
function startScript(main, isTargetParent, isTarget, existsTarget, callbacksForFirefox) {
	var observer, flag;
	
	// FirefoxのDOMContentLoaded前のMutationObserverは、要素をまとめて挿入したと見なすため、isTargetParent、isTargetを変更
	if (callbacksForFirefox && window.navigator.userAgent.contains(' Firefox/')) {
		if (callbacksForFirefox.isTargetParent) {
			isTargetParent = callbacksForFirefox.isTargetParent;
		}
		if (callbacksForFirefox.isTarget) {
			isTarget = callbacksForFirefox.isTarget;
		}
	}
	
	// 指定した節が既に存在していれば、即実行
	startMain();
	if (flag) {
		return;
	}
	
	observer = new MutationObserver(mutationCallback);
	observer.observe(document, {
		childList: true,
		subtree: true,
	});
	
	if (callbacksForFirefox) {
		// 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, nodeType, addedNodes, addedNode, i, j, l, l2;
		for (i = 0, l = mutations.length; i < l; i++) {
			mutation = mutations[i];
			target = mutation.target;
			nodeType = target.nodeType;
			if ((nodeType === Node.ELEMENT_NODE || nodeType === Node.DOCUMENT_NODE) && isTargetParent(target)) {
				// 子が追加された節が要素節か文書節で、かつそのノードについてisTargetParentが真を返せば
				addedNodes = Array.prototype.slice.call(mutation.addedNodes);
				for (j = 0, l2 = addedNodes.length; j < l2; j++) {
					addedNode = addedNodes[j];
					nodeType = addedNode.nodeType;
					if ((nodeType === Node.ELEMENT_NODE || nodeType === Node.DOCUMENT_TYPE_NODE) && isTarget(addedNode)) {
						// 追加された子が要素節か文書型節で、かつそのノードについてisTargetが真を返せば
						observer.disconnect();
						checkExistingTarget(0);
						return;
					}
				}
			}
		}
	}
	
	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければ再度実行
	 * @param {number} 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();
		}
	}
}

/**
 * prototype汚染が行われる Prototype JavaScript Framework (prototype.js) 1.5.1.1 のバグを修正(Tampermonkey用)
 */
function fixPrototypeJavaScriptFramework() {
	[
		[document, 'getElementsByClassName'],
	].forEach(function (objectProperty) {
		delete objectProperty[0][objectProperty[1]];
	});
}

/**
 * 国際化・地域化関数の読み込み、ECMAScript仕様のPolyfill
 */
function polyfill() {
// i18n
(function () {
	/**
	 * 翻訳対象文字列 (msgid) の言語
	 * @constant {string}
	 */
	var ORIGINAL_LOCALE = 'ja';
	
	/**
	 * クライアントの言語の翻訳リソースが存在しないとき、どの言語に翻訳するか
	 * @constant {string}
	 */
	var DEFAULT_LOCALE = 'en';
	
	/**
	 * 以下のような形式の翻訳リソース
	 * {
	 *     'IETF言語タグ': {
	 *         '翻訳前 (msgid)': '翻訳後 (msgstr)',
	 *         ……
	 *     },
	 *     ……
	 * }
	 * @typedef {Object} LocalizedTexts
	 */
	
	/**
	 * クライアントの言語。{@link setlang}から変更される
	 * @type {string}
	 * @access private
	 */
	var langtag = 'ja';
	
	/**
	 * クライアントの言語のlanguage部分。{@link setlang}から変更される
	 * @type {string}
	 * @access private
	 */
	var language = 'ja';
	
	/**
	 * 翻訳リソース。{@link setLocalizedTexts}から変更される
	 * @type {LocalizedTexts}
	 * @access private
	 */
	var multilingualLocalizedTexts = {};
	multilingualLocalizedTexts[ORIGINAL_LOCALE] = {};
	
	/**
	 * テキストをクライアントの言語に変換する
	 * @param {string} message - 翻訳前
	 * @returns {string} 翻訳後
	 */
	window._ = window.gettext = function (message) {
		// クライアントの言語の翻訳リソースが存在すれば、それを返す
		return langtag in multilingualLocalizedTexts && multilingualLocalizedTexts[langtag][message]
				// 地域下位タグを取り除いた言語タグの翻訳リソースが存在すれば、それを返す
				|| language in multilingualLocalizedTexts && multilingualLocalizedTexts[language][message]
				// デフォルト言語の翻訳リソースが存在すれば、それを返す
				|| DEFAULT_LOCALE in multilingualLocalizedTexts && multilingualLocalizedTexts[DEFAULT_LOCALE][message]
				// そのまま返す
				|| message;
	};
	
	/**
	 * {@link gettext}から参照されるクライアントの言語を設定する
	 * @param {string} lang - IETF言語タグ(「language」と「language-REGION」にのみ対応)
	 */
	window.setlang = function (lang) {
		lang = lang.split('-', 2);
		language = lang[0].toLowerCase();
		langtag = language + (lang[1] ? '-' + lang[1].toUpperCase() : '');
	};
	
	/**
	 * {@link gettext}から参照される翻訳リソースを追加する
	 * @param {LocalizedTexts} localizedTexts
	 */
	window.setLocalizedTexts = function (localizedTexts) {
		var localizedText, lang, language, langtag, msgid;
		for (lang in localizedTexts) {
			localizedText = localizedTexts[lang];
			lang = lang.split('-');
			language = lang[0].toLowerCase();
			langtag = language + (lang[1] ? '-' + lang[1].toUpperCase() : '');
			
			if (langtag in multilingualLocalizedTexts) {
				// すでに該当言語の翻訳リソースが存在すれば、統合する(同じmsgidがあれば上書き)
				for (msgid in localizedText) {
					multilingualLocalizedTexts[langtag][msgid] = localizedText[msgid];
				}
			} else {
				multilingualLocalizedTexts[langtag] = localizedText;
			}
			
			if (language !== langtag) {
				// 言語タグに地域下位タグが含まれていれば
				// 地域下位タグを取り除いた言語タグも翻訳リソースとして追加する
				if (language in multilingualLocalizedTexts) {
					// すでに該当言語の翻訳リソースが存在すれば、統合する(同じmsgidがあれば無視)
					for (msgid in localizedText) {
						if (!(msgid in multilingualLocalizedTexts[language])) {
							multilingualLocalizedTexts[language][msgid] = localizedText[msgid];
						}
					}
				} else {
					multilingualLocalizedTexts[language] = localizedText;
				}
			}
			
			// msgidの言語の翻訳リソースを生成
			for (msgid in localizedText) {
				multilingualLocalizedTexts[ORIGINAL_LOCALE][msgid] = msgid;
			}
		}
	};
})();

// Polyfill for Blink
if (!String.prototype.hasOwnProperty('startsWith')) {
	/**
	 * Determines whether a string begins with the characters of another string, returning true or false as appropriate.
	 * @param {string} searchString - The characters to be searched for at the start of this string.
	 * @param {number} [position=0] - The position in this string at which to begin searching for searchString.
	 * @returns {boolean}
	 * @see {@link http://people.mozilla.org/~jorendorff/es6-draft.html#sec-string.prototype.startswith 21.1.3.18 String.prototype.startsWith (searchString [, position ] )}
	 * @see {@link https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith String.startsWith - JavaScript | MDN}
	 * @version polyfill-2013-11-05
	 * @name String.prototype.startsWith
	 */
	Object.defineProperty(String.prototype, 'startsWith', {
		writable: true,
		enumerable: false,
		configurable: true,
		value: function (searchString) {
			var position = arguments[1];
			return this.indexOf(searchString, position) === Math.max(Math.floor(position) || 0, 0);
		},
	});
}

if (!String.prototype.hasOwnProperty('contains')) {
	/**
	 * 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=0] - The position in this string at which to begin searching for searchString.
	 * @returns {boolean}
	 * @see {@link http://people.mozilla.org/~jorendorff/es6-draft.html#sec-string.prototype.contains 21.1.3.6 String.prototype.contains (searchString, position = 0 )}
	 * @see {@link https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String/contains String.contains - JavaScript | MDN}
	 * @version polyfill-2013-11-05
	 * @name String.prototype.contains
	 */
	Object.defineProperty(String.prototype, 'contains', {
		writable: true,
		enumerable: false,
		configurable: true,
		value: function (searchString) {
			return this.indexOf(searchString, arguments[1]) !== -1;
		},
	});
}

}

})();