您需要先安装一个扩展,例如 篡改猴、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.8.0
- // @grant GM_addStyle
- // @grant GM_xmlhttpRequest
- // @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 = "76"; //実況モード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, 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; //実況モード有効フラグ
- if(!isFileNotFound()){
- setNormalReload();
- }
- soudane();
- makeFormClearButton();
- reset_title();
- make_live_button();
- addCss();
- setWindowFocusEvent();
- observeInserted();
- showFindNextThread();
- //通常リロード開始
- 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 make_live_button() {
- //通常モードボタン
- 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+" + String.fromCharCode(LIVE_TOGGLE_KEY) + ")]",
- 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");
- }
- var $input = $("input[value$='信する']");
- $input.after($notificationButton);
- $input.after($liveButton);
- if(SHOW_NORMAL_BUTTON){
- $input.after($normalButton);
- }
- //実況モードトグルショートカットキー
- window.addEventListener("keydown",function(e) {
- if ( e.altKey && e.keyCode == LIVE_TOGGLE_KEY ) {
- liveMode();
- }
- }, false);
- /*
- * 通常モード切り替え
- */
- function toggleNormalMode() {
- if(normal_flag) {
- clearNormalReload();
- $normalButton.css("background" , "none");
- normal_flag = false;
- } else {
- setNormalReload();
- $normalButton.css("background-color" , "#ea8");
- normal_flag = true;
- }
- }
- /*
- * 通知切り替え
- */
- function toggleNotification() {
- 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;
- });
- }
- }
- }
- /*
- * 実況モード
- * 呼出ごとにON/OFFトグル
- */
- function liveMode() {
- var live_button = $("#GM_FAR_relButton_live");
- if (!live_flag) {
- //実況モード時リロード
- timerLiveReload = setInterval(rel_scroll, RELOAD_INTERVAL_LIVE);
- //自動スクロール
- timerLiveScroll = setInterval(live_scroll, LIVE_SCROLL_INTERVAL);
- live_button.css("backgroundColor", "#ffa5f0");
- startspin();
- console.log(script_name + ": Start live mode @" + url);
- live_flag = true;
- } else {
- clearInterval(timerLiveReload);
- clearInterval(timerLiveScroll);
- live_button.css("background", "none");
- stopspin();
- console.log(script_name + ": Stop live mode @" + url);
- live_flag = false;
- }
- //リロード+新着スクロール
- function rel_scroll() {
- $('html, body').animate(
- {scrollTop:window.scrollMaxY},"fast"
- );
- rel();
- }
- function live_scroll() {
- window.scrollBy( 0, LIVE_SCROLL_SPEED );
- }
- function startspin() {
- $("#akahuku_throp_menu_opener").css(
- "animation", "spin 2s infinite steps(8)"
- );
- }
- function stopspin() {
- $("#akahuku_throp_menu_opener").css(
- "animation", "none"
- );
- }
- }
- /*
- * 新着レスをリセット
- */
- function reset_title() {
- //ページ末尾でホイールダウンした時
- window.addEventListener("DOMMouseScroll",function scroll(event) {
- var window_y = Math.ceil(window.scrollY);
- var window_ymax = window.scrollMaxY;
- if (event.detail > 0 && window_y >= window_ymax) {
- reset_titlename();
- }
- return;
- } ,false);
- //F5キー押された時
- window.addEventListener("keydown",function(e) {
- if ( e.keyCode == "116" ) {
- reset_titlename();
- }
- }, false);
- 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);
- if(relbutton){
- var e = document.createEvent("MouseEvents");
- e.initEvent("click", false, true);
- relbutton.dispatchEvent(e);
- }
- setTimeout(function(){
- soudane();
- if (!isWindowActive && isNotificationEnable) {
- getNewResContent();
- }
- if(isAkahukuNotFound()) {
- //404時
- 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 ( !USE_SOUDANE ) return;
- clearTimeout(timerSoudane);
- timerSoudane = setTimeout(function() {
- var coloredNode = $(".rtd[style]");
- coloredNode.each(function() {
- $(this).removeAttr("style");
- });
- $("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 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 makeFormClearButton() {
- 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);
- }
- function clearForm() {
- $("#ftxa").val("");
- }
- }
- function addCss() {
- GM_addStyle(
- "@keyframes spin {" +
- " 0% { transform: rotate(0deg); }" +
- " 100% { transform: rotate(359deg); }" +
- "}"
- );
- }
- // タブのアクティブ状態を取得
- 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().substr(0, 4);
- 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, 4) == threadTitle) {
- 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();
- }
- });
- }
- })(jQuery);