futaba auto reloader

赤福Firefox版で自動更新しちゃう(実況モードもあるよ!)

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name           futaba auto reloader
// @namespace      https://github.com/himuro-majika
// @description    赤福Firefox版で自動更新しちゃう(実況モードもあるよ!)
// @author         himuro_majika
// @include        http://*.2chan.net/*/res/*
// @include        https://*.2chan.net/*/res/*
// @include        http://board.futakuro.com/*/res/*
// @require        http://ajax.googleapis.com/ajax/libs/jquery/2.0.3/jquery.min.js
// @version        1.9.0
// @grant          GM_addStyle
// @grant          GM_xmlhttpRequest
// @grant          GM_getValue
// @grant          GM_setValue
// @license        MIT
// @icon         
// ==/UserScript==
this.$ = this.jQuery = jQuery.noConflict(true);

(function ($) {
	/*
	 *	設定
	 */
	// =====================================================
	var USE_SOUDANE = true;								//そうだねをハイライト表示する
	var USE_CLEAR_BUTTON = true;					//フォームにクリアボタンを表示する
	var USE_TITLE_NAME = true;						//新着レス数・スレ消滅状態をタブに表示する
	var RELOAD_INTERVAL_NORMAL = 60000;		//リロード間隔[ミリ秒](通常時)
	var RELOAD_INTERVAL_LIVE = 5000;			//リロード間隔[ミリ秒](実況モード時)
	var LIVE_SCROLL_INTERVAL = 12;				//実況モードスクロール間隔[ミリ秒]
	var LIVE_SCROLL_SPEED = 2;						//実況モードスクロール幅[px]
	var LIVE_TOGGLE_KEY = "l";						//実況モードON・OFF切り替えキー(With Alt)
	var SHOW_NORMAL_BUTTON = true;				//通常モードボタンを表示する
	var USE_NOTIFICATION_DEFAULT = false;	// 新着レスの通知をデフォルトで有効にする
	var USE_SAVE_MHT = false;							// スレ消滅時にMHTで保存する
	// =====================================================


	var res = 0;	//新着レス数
	var timerNormal, timerLiveReload, timerLiveJump, timerLiveScroll, timerSoudane;
	var url = location.href;
	var script_name = "futaba_auto_reloader";
	var isWindowActive = true;	// タブのアクティブ状態
	var isNotificationEnable = USE_NOTIFICATION_DEFAULT;	// 通知の有効フラグ
	var normal_flag = true;	//通常モード有効フラグ
	var live_flag = false;	//実況モード有効フラグ
	var isSoudane = true;

	init();

	function init() {
		if(isFileNotFound()) return;
		
		loadSettings();
		setNormalReload();
		soudane();
		makeButton();
		addCss();
		setWindowFocusEvent();
		observeInserted();
		showFindNextThread();
		setWheelEvent();
		setKeyEvent();
	}

	// 設定ロード
	function loadSettings() {
		isSoudane = GM_getValue("soudane", true);
		console.log(isSoudane);
	}
	//通常リロード開始
	function setNormalReload() {
		timerNormal = setInterval(rel, RELOAD_INTERVAL_NORMAL);
		console.log(script_name + ": Start auto reloading @" + url);
	}
	//通常リロード停止
	function clearNormalReload() {
		clearInterval(timerNormal);
		console.log(script_name + ": Stop auto reloading @" + url);
	}
	/*
	 * 404チェック
	 */
	function isFileNotFound() {
		if(document.title == "404 File Not Found") {
			return true;
		}
		else {
			return false;
		}
	}

	/*
	 * ボタン作成
	 */
	function makeButton() {
		//通常モードボタン
		var normalButton = $("<a>", {
			id: "GM_FAR_relButton_normal",
			class: "GM_FAR_relButton",
			text: "[通常]",
			title: (RELOAD_INTERVAL_NORMAL / 1000) + "秒毎のリロード",
			css: {
				cursor: "pointer",
				"background-color": "#ea8",
			},
			click: function() {
				toggleNormalMode();
			}
		});

		//実況モードボタン
		var liveButton = $("<a>", {
			id: "GM_FAR_relButton_live",
			class: "GM_FAR_relButton",
			text: "[実況(Alt+" + LIVE_TOGGLE_KEY.toUpperCase() + ")]",
			title: (RELOAD_INTERVAL_LIVE / 1000) + "秒毎のリロード + スクロール",
			css: {
				cursor: "pointer",
			},
			click: function() {
				liveMode();
			}
		});
		// 通知ボタン
		var notificationButton = $("<a>", {
			id: "GM_FAR_notificationButton",
			text: "[通知]",
			title: "新着レスのポップアップ通知",
			css: {
				cursor: "pointer",
			},
			click: function() {
				toggleNotification();
			}
		});
		if (isNotificationEnable) {
			notificationButton.css("background-color", "#a9d8ff");
		}
		// フォームクリアボタン
		if ( USE_CLEAR_BUTTON ) {
			var formClearButton = $("<div>", {
				id: "formClearButton",
				text: "[クリア]",
				css: {
					cursor: "pointer",
					margin: "0 5px"
				},
				click: function() {
					clearForm();
				}
			});
			var comeTd = $(".ftdc b:contains('コメント')");
			comeTd.after(formClearButton);
		}
		// そボタン
		var soudaneButton = $("<a>", {
			id: "GM_FAR_soudaneButton",
			text: "[そ]",
			title: "そうだねに色を付ける",
			css: {
				cursor: "pointer",
			},
			click: function() {
				toggleSoudane();
			}
		});
		if (isSoudane) soudaneButton.css("background-color" , "#ea8");


		var input = $("input[value$='信する']");
		input.after(soudaneButton);
		input.after(notificationButton);
		input.after(liveButton);
		if(SHOW_NORMAL_BUTTON){
			input.after(normalButton);
		}
	}
	/*
		* 通常モード切り替え
		*/
	function toggleNormalMode() {
		var normalButton = $("#GM_FAR_relButton_normal");
		if(normal_flag) {
			clearNormalReload();
			normalButton.css("background" , "none");
			normal_flag = false;
		} else {
			setNormalReload();
			normalButton.css("background-color" , "#ea8");
			normal_flag = true;
		}
	}
	/*
	 * 実況モード
	 * 呼出ごとにON/OFFトグル
	 */
	function liveMode() {
		var live_button = $("#GM_FAR_relButton_live");
		if (!live_flag) {
			setLiveReload();
			setLiveJump();
			setLiveScroll();
			startspin();
			live_flag = true;
			live_button.css("backgroundColor", "#ffa5f0");
			console.log(script_name + ": Start live mode @" + url);
		} else {
			clearLiveReload();
			clearLiveJump();
			clearLiveScroll();
			stopspin();
			live_flag = false;
			live_button.css("background", "none");
			console.log(script_name + ": Stop live mode @" + url);
		}
	}
	function setLiveReload() {
		clearInterval(timerLiveReload);
		timerLiveReload = setInterval(rel, RELOAD_INTERVAL_LIVE);
	}
	function clearLiveReload() {
		clearInterval(timerLiveReload);
	}
	function setLiveJump() {
		clearInterval(timerLiveJump);
		timerLiveJump = setInterval(liveJump, RELOAD_INTERVAL_LIVE);
	}
	function clearLiveJump() {
		clearInterval(timerLiveJump);
	}
	function setLiveScroll() {
		clearInterval(timerLiveScroll);
		timerLiveScroll = setInterval(live_scroll, LIVE_SCROLL_INTERVAL);
	}
	function clearLiveScroll() {
		clearInterval(timerLiveScroll);
	}
	//自動スクロール
	function live_scroll() {
		window.scrollBy( 0, LIVE_SCROLL_SPEED );
	}
	//新着ジャンプ
	function liveJump() {
		$('html, body').animate({scrollTop:document.body.scrollHeight},"fast");
	}
	function startspin() {
		$("#akahuku_throp_menu_opener").css(
			"animation", "spin 2s infinite steps(8)"
		);
	}
	function stopspin() {
		$("#akahuku_throp_menu_opener").css(
			"animation", "none"
		);
	}
	/**
	 * 新着レスをリセット
	 */
	function reset_titlename() {
		res = 0;
		var title_char = title_name();
		document.title = title_char;
	}
	/**
	 * 赤福の続きを読むボタンをクリック
	 */
	function rel() {
		// if(isAkahukuNotFound()) {
		// 	return;
		// }
		var relbutton = $("#akahuku_reload_button").get(0) ? $("#akahuku_reload_button").get(0) : $("#contres a").get(0);
		if (relbutton){
			var e = document.createEvent("MouseEvents");
			e.initEvent("click", false, true);
			relbutton.dispatchEvent(e);
		} else {
			return;
		}
		setTimeout(function(){
			soudane();

			if (!isWindowActive && isNotificationEnable) {
				getNewResContent();
			}
			var res = $(".rsc");
			if(isAkahukuNotFound() || res.length >= 1000) {
				//404 or 1000レス時
				if (live_flag) {
					liveMode();
				}

				changeTitleWhenExpired();
				clearNormalReload();
				if (USE_SAVE_MHT) {
					saveMHT();
				}
				findNextThread();
				console.log(script_name + ": Page not found, Stop auto reloading @" + url);
			}
		}, 1000);
	}
	/**
	 * MHTで保存
	 */
	function saveMHT() {
		var saveMHTButton = $("#akahuku_throp_savemht_button").get(0);
		if (saveMHTButton) {
			var e = document.createEvent("MouseEvents");
			e.initEvent("click", false, true);
			saveMHTButton.dispatchEvent(e);
		}
	}
	/*
	 * そうだねの数に応じてレスを着色
	 */
	function soudane() {
		if ( !isSoudane ) return;

		clearTimeout(timerSoudane);
		timerSoudane = setTimeout(function() {
			clearSoudane();

			$("td > .sod").each(function(){
				var sodnum = $(this).text().match(/\d+/);
				if (sodnum){
					var col = "rgb(180, 240," + (Math.round(10 * sodnum + 180)) + ")";
					$(this).parent().css("background-color", col);
				}
			});
		}, 100);
	}
	function clearSoudane() {
		var coloredNode = $(".rtd[style]");
		coloredNode.each(function() {
			$(this).removeAttr("style");
		});
	}
	function toggleSoudane() {
		var soudaneButton = $("#GM_FAR_soudaneButton");
		if (isSoudane) {
			isSoudane = false;
			clearSoudane();
			soudaneButton.css("background" , "none");
		} else {
			isSoudane = true;
			soudane();
			soudaneButton.css("background-color" , "#ea8");
		}
		GM_setValue("soudane", isSoudane);
	}
	// 続きを読むで挿入される要素を監視
	function observeInserted() {
		var target = $(".thre").length ?
			$(".thre").get(0) :
			$("html > body > form[action]:not([enctype])").get(0);
		var observer = new MutationObserver(function(mutations) {
			soudane();

			mutations.forEach(function(mutation) {
				var $nodes = $(mutation.addedNodes);
				replaceNodeInserted($nodes);
			});
		});
		observer.observe(target, { childList: true });
	}
	// 挿入されたレス
	function replaceNodeInserted($nodes) {
		var insertedRes = $nodes.find(".rtd");
		if( insertedRes.length ) {
			changetitle();
		}
	}

	/*
	 * タブタイトルに新着レス数・スレ消滅状態を表示
	 */
	function changetitle() {
		if ( !USE_TITLE_NAME ) return;
		var title_char = title_name();
		if (isAkahukuNotFound()) return;
		res++;
		document.title = "(" + res + ")" + title_char;
	}

	function changeTitleWhenExpired() {
		if (!isAkahukuNotFound()) return;
		if(document.title.substr(0,1) !== "#"){
			document.title = "#" + document.title;
		}
	}

	// 新着レスの内容を取得
	function getNewResContent() {
		var $newrestable = $("#akahuku_new_reply_header ~ table:not([id])");
		if ($newrestable.length) {
			var restexts = [];
			$newrestable.each(function() {
				var texts = [];
				$(this).find("blockquote").contents().each(function() {
					if ($(this).text() !== "") {
						texts.push($(this).text());
					}
				});
				restexts.push(texts.join("\r\n"));
			});
			var popupText = restexts.join("\r\n===============\r\n");
			showNotification(popupText);
		}
	}
	/*
	 * 赤福のステータスからスレ消滅状態をチェック
	 */
	function isAkahukuNotFound() {
		var statustext = $("#akahuku_reload_status").text();
		if (statustext.match(/(No Future)|((M|N)ot Found)/)) {
			return true;
		}
		else {
			return false;
		}
	}

	function title_name() {
		var title = document.title;
		var title_num = title.match(/^(#|\(\d+\))/);
		var title_num_length;
		if(!title_num){
			title_num_length = 0;
		}
		else {
			title_num_length = title_num[0].length;
		}
		var act_title_name = title.substr(title_num_length);
		return act_title_name;
	}

	function clearForm() {
		$("#ftxa").val("");
	}

	function addCss() {
		GM_addStyle(
			"@keyframes spin {" +
			"  0% { transform: rotate(0deg); }" +
			"  100% { transform: rotate(359deg); }" +
			"}"
		);
	}
	/**
	 * 通知切り替え
	 */
	 function toggleNotification() {
		 var notificationButton = $("#GM_FAR_notificationButton");
		if(isNotificationEnable) {
			notificationButton.css("background" , "none");
			isNotificationEnable = false;
		} else {
			Notification.requestPermission(function(result) {
				if (result == "denied") {
					notificationButton.attr("title",
						"通知はFirefoxの設定でブロックされています\n" +
						"ロケーションバー(URL)の左のアイコンをクリックして\n" +
						"「サイトからの通知の表示」を「許可」に設定してください");
					return;
				} else if (result == "default") {
					console.log("default");
					return;
				}
				notificationButton.attr("title", "新着レスのポップアップ通知");
				notificationButton.css("background-color" , "#a9d8ff");
				isNotificationEnable = true;
			});
		}
	}
	// タブのアクティブ状態を取得
	function setWindowFocusEvent() {
		$(window).bind("focus", function() {
			// タブアクティブ時
			isWindowActive = true;
		}).bind("blur", function() {
			// タブ非アクティブ時
			isWindowActive = false;
		});
	}
	// 新着レスをポップアップでデスクトップ通知する
	function showNotification(body) {
		Notification.requestPermission();
		var icon = $("#akahuku_thumbnail").attr("src");
		var instance = new Notification(
			document.title, {
				body: body,
				icon: icon,
			}
		);
	}
	/**
	 * 次スレ候補検索ボタン表示
	 */
	function showFindNextThread() {
		$("body").append(
			$("<div>", {
				id: "GM_FAR_next_thread_area",
				class: "GM_FAR"
			}).append(
				$("<div>").append(
					$("<a>", {
						id: "GM_FAR_find_next_thread",
						class: "GM_FAR_Button",
						text: "[次スレ候補検索]",
						css: {
							cursor: "pointer",
							"font-size": "9pt"
						},
						click: function() {
							findNextThread();
						}
					}),
					$("<span>", {
						id: "GM_FAR_next_thread_search_result",
						css: {
							"display": "none",
							"font-size": "9pt"
						}
					}).append(
						$("<span>", {
							text: "検索結果:",
						}),
						$("<span>", {
							id: "GM_FAR_next_thread_search_result_count",
							text: "0"
						})
					)
				),
				$("<ul>", {
					"id": "GM_FAR_next_thread_found"
				}).append(
					$("<span>", {
						id: "GM_FAR_next_thread_search_status",
						text: "次スレ候補検索中...",
						css: {
							"display": "none"
						}
					})
				)
			)
		)
	}

	/**
	 * 次スレ候補検索
	 */
	function findNextThread() {
		var foundList = $("#GM_FAR_next_thread_found");
		foundList.empty()
		var statusMessage = $("#GM_FAR_next_thread_search_status")
		statusMessage.show();
		var dir = location.href.substr(0, location.href.lastIndexOf('/') - 3);
		var threadTitle = $("#akahuku_thread_text").text();
		var catalogURL = dir + "futaba.php?mode=cat&sort=1"
		var resultCount = 0;
		GM_xmlhttpRequest({
			method: "GET",
			url: catalogURL,
			onload: function(res) {
				statusMessage.hide();
				var catalog = $($.parseHTML(res.response));
				var cattable = catalog.filter("#cattable");
				var td = cattable.find("td small");
				td.each(function() {
					var tdText = $(this).text()
					if (tdText.substr(0, 3) != threadTitle.substr(0, 3) || tdText.substr(0, 3) == "キタ━") return;
					resultCount++;
					var foundThread = $(this).parent().find("a");
					var foundThreadResCount = $(this).parent().find("font").text();
					var href = foundThread.attr("href");
					foundThread.attr("href", dir + href);
					foundList.append(
						$("<li>").append(
							$(this),
							$("<span>", {
								text: foundThreadResCount + "レス",
								css: {
									"margin-left": "2em"
								}
							}),
							foundThread
						)
					);
				});
				$("#GM_FAR_next_thread_search_result_count").text(resultCount);
				$("#GM_FAR_next_thread_search_result").show();
			}
		});
	}

	/**
	 * マウスホイールイベント
	 */
	function setWheelEvent() {
		var wheelNum = 3;	// ホイールダウン回数
		var timerWheel;
		var n = 0;
		window.addEventListener("wheel", (e) => {
			var y = window.pageYOffset;
			var ym = getPageBottom();
			//新着レス数をリセット
			if (e.deltaY > 0 && y >= ym) {
				reset_titlename();
			}
			// スレ落ち後の手動次スレ検索
			var res = $(".rsc");
			if(isAkahukuNotFound() || res.length >= 1000) {
				if (e.deltaY > 0 && y >= ym) {
					n++;
					if (n >= wheelNum) {
						clearTimeout(timerWheel);
						n = 0;
						timerWheel = setTimeout(() => {
							findNextThread();
						}, 200);
					}
				}
			}
			// 実況モード時上スクロールで自動スクロールの一時停止
			if (live_flag && e.deltaY < 0) {
				clearLiveScroll();
				clearLiveJump();
				$("#GM_FAR_relButton_live").css("backgroundColor", "#c0ffa5");
			} else if (live_flag && y >= ym) {
				setLiveScroll();
				setLiveJump();
				$("#GM_FAR_relButton_live").css("backgroundColor", "#ffa5f0");
			}
		} ,false);
		/* ページ末尾 */
		function getPageBottom() {
			var pageBottom = document.body.scrollHeight - window.innerHeight;
			return pageBottom;
		}
	}
	/**
	 * キーボードイベント
	 */
	function setKeyEvent() {
		//実況モードトグルショートカットキー
		window.addEventListener("keydown",function(e) {
			if ( e.altKey && e.key == LIVE_TOGGLE_KEY ) {
				liveMode();
			}
		}, false);
	}

})(jQuery);