Clone Turning Page Button in nicovideo

ニコニコ生放送の検索結果と「放送中の番組」ページにおいて、ページ送りボタンを上部にも表示 / Also shows the pagination on the top of Nico Live search result and Lives by Category page.

目前為 2014-07-24 提交的版本,檢視 最新版本

// ==UserScript==
// @name        Clone Turning Page Button in nicovideo
// @namespace   http://pc12.2ch.net/test/read.cgi/streaming/1275642556/
// @version     3.3.0
// @description ニコニコ生放送の検索結果と「放送中の番組」ページにおいて、ページ送りボタンを上部にも表示 / Also shows the pagination on the top of Nico Live search result and Lives by Category page.
// @match       http://live.nicovideo.jp/search?*
// @match       http://live.nicovideo.jp/recent
// @match       http://live.nicovideo.jp/recent?*
// @match       http://live.nicovideo.jp/recent#*
// @match       http://watch.live.nicovideo.jp/search?*
// @match       http://watch.live.nicovideo.jp/recent
// @match       http://watch.live.nicovideo.jp/recent?*
// @match       http://watch.live.nicovideo.jp/recent#*
// @run-at      document-start
// @grant       dummy
// @icon        
// @author      100の人
// @homepage    https://greasyfork.org/ja/scripts/275-clone-turning-page-button-in-nicovideo
// @license     Creative Commons Attribution 4.0 International Public License; http://creativecommons.org/licenses/by/4.0/
// ==/UserScript==

(function () {
'use strict';

switch (location.pathname) {
	case '/search':
		// ニコニコ生放送 検索結果
		startScript(function () {
			var pager = document.getElementsByClassName('pager')[0];
			if (pager) {
				// 41件以上ヒットしていれば
				// スタイルシートの設定
				document.head.insertAdjacentHTML('beforeend', '<style> \
					.result_list { \
						border-top: 1px solid #888888; \
					} \
				</style>');
				// 数字リンクの複製
				var ref = document.getElementsByClassName('result_list')[0];
				ref.parentElement.insertBefore(pager.cloneNode(true), ref);
			}
		},
				function (parent) { return parent.classList.contains('container'); },
				function (target) { return target.id === 'search_right'; },
				function () { return document.getElementById('search_right'); });
		break;

	case '/recent':
		// ニコニコ生放送 放送中の番組
		startScript(function () {
			var parent = document.getElementById('onair_stream_list');

			clonePager();

			var observer = new MutationObserver(function (mutations) {
				if (mutations[0].addedNodes.length > 1) {
					// 数字リンクの複製でなければ
					clonePager();
				}
			});
			observer.observe(parent, {
				childList: true,
			});

			function clonePager() {
				parent.insertBefore(document.getElementById('pagerFrame').cloneNode(true), document.getElementById('userFrame'));
			}
		},
				function (parent) { return parent.id === 'left'; },
				function (target) { return target.id === 'pickupFrame'; },
				function () { return document.getElementById('pickupFrame'); });
		break;
}



/**
 * 挿入された節の親節が、目印となる節の親節か否かを返すコールバック関数。
 * @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]
 * @param {isTargetParent} [callbacksForFirefox.isTargetParent] - Firefoxにおける{@link isTargetParent}。
 * @param {isTarget} [callbacksForFirefox.isTarget] - Firefoxにおける{@link isTarget}。
 * @param {boolean} [timeoutSinceStopParsingDocument=0] - DOM構築完了後に監視を続けるミリ秒数。
 * @version 2014-07-06
 */
function startScript(main, isTargetParent, isTarget, existsTarget) {
	/**
	 * {@link checkExistingTarget}で{@link startMain}を実行する間隔(ミリ秒)。
	 * @constant {number}
	 */
	var INTERVAL = 10;
	/**
	 * {@link checkExistingTarget}で{@link startMain}を実行する回数。
	 * @constant {number}
	 */
	var LIMIT = 500;

	/**
	 * 実行済みなら真。
	 * @type {boolean}
	 */
	var alreadyCalled = false;

	// 指定した節が既に存在していれば、即実行
	startMain();
	if (alreadyCalled) {
		return;
	}

	// FirefoxのMutationObserverは、HTMLのDOM構築に関して要素をまとめて挿入したと見なすため、isTargetParent、isTargetを変更
	var callbacksForFirefox = arguments[4];
	if (callbacksForFirefox && typeof sidebar !== 'undefined') {
		if (callbacksForFirefox.isTargetParent) {
			isTargetParent = callbacksForFirefox.isTargetParent;
		}
		if (callbacksForFirefox.isTarget) {
			isTarget = callbacksForFirefox.isTarget;
		}
	}

	var observer = new MutationObserver(mutationCallback);
	observer.observe(document, {
		childList: true,
		subtree: true,
	});

	var timeoutSinceStopParsingDocument = arguments[5] || 0;
	if (document.readyState === 'complete') {
		// DOMの構築が完了していれば
		onDOMContentLoaded();
	} else {
		document.addEventListener('DOMContentLoaded', onDOMContentLoaded);
	}

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければさらに{@link timeoutSinceStopParsingDocument}ミリ秒待機し、
	 * スクリプトが開始されていなければ{@link stopObserving}を実行する。
	 */
	function onDOMContentLoaded() {
		startMain();
		if (timeoutSinceStopParsingDocument === 0) {
			if (!alreadyCalled) {
				stopObserving();
			}
		} else {
			window.setTimeout(function () {
				if (!alreadyCalled) {
					stopObserving();
				}
			}, timeoutSinceStopParsingDocument);
		}
	}

	/**
	 * 目印となる節が挿入されたら、監視を停止し、{@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) && 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) && isTarget(addedNode)) {
						// 追加された子が要素節で、かつその節についてisTargetが真を返せば
						observer.disconnect();
						checkExistingTarget(0);
						return;
					}
				}
			}
		}
	}

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければ再度実行。
	 * @param {number} count - {@link startMain}を実行した回数。
	 */
	function checkExistingTarget(count) {
		startMain();
		if (!alreadyCalled && count < LIMIT) {
			window.setTimeout(checkExistingTarget, INTERVAL, count + 1);
		}
	}

	/**
	 * 指定した節が存在するか確認し、存在すれば{@link stopObserving}を実行しスクリプトを開始。
	 */
	function startMain() {
		if (!alreadyCalled && existsTarget()) {
			stopObserving();
			main();
		}
	}

	/**
	 * 監視を停止する。
	 */
	function stopObserving() {
		alreadyCalled = true;
		if (observer) {
			observer.disconnect();
		}
		document.removeEventListener('DOMContentLoaded', onDOMContentLoaded);
	}
}

})();