您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
赤福Firefox版で自動更新しちゃう(実況モードもあるよ!)
// ==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);