Greasy Fork

Greasy Fork is available in English.

AO3: Fic's Style and Bookmarks

Change font, size, width.. + estimated reading time + full screen mode + bookmarks: save the position you stopped reading a fic

当前为 2015-08-04 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         AO3: Fic's Style and Bookmarks
// @namespace    http://greasyfork.icu/it/users/12632-schegge
// @version      2.0
// @description  Change font, size, width.. + estimated reading time + full screen mode + bookmarks: save the position you stopped reading a fic
// @author       Schegge
// @include      http://archiveofourown.org/*
// @include      https://archiveofourown.org/*
// @grant        none
// @icon         
// ==/UserScript==

(function($) {    

    // BOOKMARKS
    var Bookmarks = {
        getAll: function() {
            var bookmarks = localStorage.getItem("ficstyle_bookmarks");
            if ( !bookmarks ) {
                bookmarks = "";
                localStorage.setItem("ficstyle_bookmarks", bookmarks);
            }
            return bookmarks;
        },
        getSingles: function() {
            var all = this.getAll();
            return all.split("@");
        },
        getElements: function() { // 0 = url, 1 = title, 2 = scrolltop
            var els = [];
            var singles = this.getSingles();
            for(var i = 1; i < singles.length; i++) { // from 1 because the first element is empty (storage starts with a @)
                els.push( singles[i].split("#") );
            }
            return els;
        },
        getUrl: window.location.pathname.split("/works/")[1], // work id
        getTitle: function() {
            var title = $("#workskin .preface.group h2.title.heading").text().trim();            
            title = title.substring(0, 28); // to cut long titles
            if ( /chapters/.test(window.location.pathname) ) { // if chapter by chapter, also storaging the number of the chapter
                var chapter = $("#chapters > .chapter > div.chapter.preface.group > h3 > a").text();
                chapter = chapter.replace("Chapter ", "ch");
                title += " (" + chapter + ")";
            }
            title = title.replace(/#/g, " "); // just in case
            title = title.replace(/@/g, " ");
            return title;
        },
        getNewBook : function() {       
            var newbook = $(document).scrollTop(); // current position of the scroll bar
            var chs = $("dl.stats dd.chapters").text(); // # chapters
            if ( /(\d+)\/\1/.test(chs) || /chapters/.test(window.location.pathname) ) { // if work completed (if number/number is the same) or chapter by chapter view
                newbook = ( newbook / $(document).height() ).toFixed(4) + "%"; // calculate in percent
            }
            return newbook;
        },
        checkIfExist: function(a) {
            var url = this.getUrl;
            var els = this.getElements();

            for(var i = 0; i < els.length; i++) {
                if ( els[i][0] == url ) { // if a bookmark already existed for the current fic
                    if ( a == "book" ) { // retrieve the bookmark
                        var book = els[i][2];
                        if ( book.indexOf("%") !== -1 ) {
                            book = book.replace("%", "");
                            book = parseFloat(book);
                            book = book * $(document).height();
                        }
                        book = parseFloat(book);
                        return book;
                    } else if ( a == "cancel" ) { // delete the old bookmark
                        return "@" + els[i][0] + "#" + els[i][1] + "#" + els[i][2];
                    } else {
                        return true;
                    }
                }
            }
        },
        cancel: function() {
            var newBookmarks = this.getAll();
            var cancel = this.checkIfExist("cancel");
            if ( cancel ) {
                newBookmarks = newBookmarks.replace(cancel, "");
            }
            return newBookmarks;
        },
        getNew: function() {
            var url = this.getUrl;
            var title = this.getTitle();
            var newbook = this.getNewBook();

            var newBookmarks = this.cancel(); // if the the fic was already bookmarked, delete the old bookmark
            newBookmarks += "@" + url + '#' + title + '#' + newbook; // add new bookmark
            localStorage.setItem("ficstyle_bookmarks", newBookmarks);
        }
    };

    // create bookmarks' menu
    $("#header > ul").append(
        '<li id="menu-bookmarks" class="dropdown" aria-haspopup="true">' +
        '<a>Bookmarks</a>' +
        '<ul class="menu dropdown-menu" role="menu"></ul></li>'
    );

    var els = Bookmarks.getElements();
    for(var z = 0; z < els.length; z++) {
        $("#menu-bookmarks > ul.menu").append(
            "<li role='menu-item'>" +
            "<a href='http://archiveofourown.org/works/" + els[z][0] + "'>" + els[z][1] + "</a>" +
            "</li>"
        );
    }

    // add estimated reading time
    $words = $("dl.stats dd.words");
    if ( $words.length ){
        $words.each(function() {
            var numWords = $(this).text();
            numWords = numWords.replace(",", "");
            var timeReading = parseInt( numWords ) / 200; // 200 words per minute
            if ( timeReading < 60 ) {
                timeReading = Math.round ( timeReading ) + "min";
            } else {
                timeReading = ( timeReading / 60 ).toFixed(2);
                timeReading = timeReading.toString();
                timeReading = timeReading.split(".");
                var hours = timeReading[0];
                var minutes = Math.round ( parseInt(timeReading[1]) / 100 * 60 );
                timeReading = hours + "hr & " + minutes.toString() + "min";
            }
            $(this).after("<dt>Time:</dt><dd>" + timeReading + "</dd>");
        });
    }

    //*//*//*//*//*//*//*// everything below is going to be shown only on the fic's page //*//*//*//*//*//*//*//

    var windowUrl = window.location.pathname;
    // include: (whatever)/works/(numbers) and (whatever)/works/(numbers)/chapters/(numbers) and exclude: navigate
    if ( /.*\/works\/\d+(\/chapters\/\d+)?/.test(windowUrl) && !/navigate/.test(windowUrl)) {

        fontNameDefault = $("body").css("font-family"); // the font-family used by AO3
        ////////////////////////////////////////////////////////////////////////////////////////////////////////
        var Options = {
            fontName: [fontNameDefault, "Georgia", "Garamond", "Book Antiqua", "Verdana", "Segoe UI"],
            fontSize: 100, //%
            fontColor: "#000",
            chapterWidth: 90, //% ( min = 10; max = 90 )
            showBackground: "no", //or "yes"
            backgroundColor: "#f6f6f6"
        };
        ////////////////////////////////////////////////////////////////////////////////////////////////////////

        var chaptersCSS;
        numChapters = $("#chapters > .chapter").length; // it counts how many chapters the fic has
        if (numChapters) { // if the fic has several chapters, even if only one chapter is displayed
            chapters = $("#chapters > .chapter");
            chaptersCSS = "#chapters > .chapter";
        } else { // if the fic is a one-shot
            chapters = $("#chapters");
            chaptersCSS = "#chapters";
        }

        // general CSS changes
        var addGlobalCSS = function(css) {
            $head = $("head");
            $style = $("<style type='text/css'>" + css + "</style>");
            $head.append($style);
        };

        addGlobalCSS(
            chaptersCSS + " { " +
            "color: " + Options.fontColor + "; " +
            "margin: auto; " +
            "text-align: justify; " +
            "padding-top: 30px; " +
            "padding-bottom: 30px; " +
            "border-radius: 100% / 60px; " +
            "margin-bottom: 20px; " +
            "padding-right: 3%; " +
            "padding-left: 3%; " +
            "} " +
            ".chapter .preface { " +
            "margin-bottom: 0; " +
            "} " +
            ".chapter .preface[role='complementary'] { " +
            "margin-top: 0; " +
            "padding-top: 0; " +
            "} " +
            "#workskin .notes, #workskin .summary { " +
            "font-size: .91rem; " +
            "} " +
            "#chapters .userstuff p { " +
            "font-family: inherit; " +
            "margin: 0.6em auto; " +
            "text-align: justify; " +
            "} " +
            "#chapters .userstuff blockquote { " +
            "font-family: inherit; " +
            "padding-top: 1px; " +
            "padding-bottom: 1px; " +
            "margin: 0 .5em; " +
            "} " +
            ".userstuff hr { " +
            "width: 100%; " +
            "height: 1px; " +
            "border: 0; " +
            "background-image: linear-gradient(to right, transparent, rgba(0, 0, 0, .5), transparent); " +
            "} " +
            ".userstuff img { " +
            "max-width: 100%; " +
            "height: auto; " +
            "} " +
            "#options, .ficleft { " +
            "position: fixed; " +
            "bottom: 15px; " +
            "margin: 0; " +
            "padding: 0; " +
            "font-family: Consolas, monospace; " +
            "font-size: 16px; " +
            "color: #000; " +
            "text-shadow: 0 0 2px rgba(0, 0, 0, .4); " +
            "z-index: 999; " +
            "} " +
            "#options { " +
            "right: 15px; " +
            "} " +
            ".ficleft { " +
            "left: 15px; " +
            "} " +
            "#options > div { " +
            "margin: 5px 0 0 0; " +
            "padding: 0 5px; " +
            "cursor: pointer; " +
            "} " +
            "#options > div:last-child { " +
            "padding: 2px 5px; " +
            "color: #fff; " +
            "background-color: rgba(0, 0, 0, .2); " +
            "} " +
            ".ficleft a, #options a { " +
            "border: 0; " +
            "} " +
            "div.preface .notes, div.preface .summary, div.preface .series, div.preface .children {" +
            "min-height: 0; " +
            "} " +
            "#notes { " +
            "display: none; " +
            "} " +
            ".notes-hidden { " +
            "cursor: pointer; " +
            "position: fixed; " +
            "width: 50%; " +
            "max-height: 50%; " +
            "left: 50px; " +
            "bottom: 50px; " +
            "background-color: #fff; " +
            "padding: 10px; " +
            "box-shadow: 0 0 2px 1px rgba(0, 0, 0, .4); " +
            "margin: 0; " +
            "overflow: auto; " +
            "z-index: 999; " +
            "display: none; " +
            "} " +
            ".notes-headings { " +
            "cursor: pointer; " +
            "border-bottom-width: 0!important; " +
            "margin-bottom: 0; " +
            "text-align: center; " +
            "color: #666; " +
            "} "
        );

        // CSS changes depending on the user
        var Variables = {
            fontName: function() {
                var fontName = localStorage.getItem("ficstyle_fontName");
                if (!fontName) {
                    fontName = Options.fontName[0];
                    localStorage.setItem("ficstyle_fontName", fontName);
                }
                return fontName;
            },
            fontSize: function() {
                var fontSize = localStorage.getItem("ficstyle_fontSize");
                if (!fontSize) {
                    fontSize = Options.fontSize;
                    localStorage.setItem("ficstyle_fontSize", fontSize);
                }
                return fontSize;
            },
            chapterWidth: function() {
                var chapterWidth = localStorage.getItem("ficstyle_chapterWidth");
                if (!chapterWidth) {
                    chapterWidth = Options.chapterWidth;
                    localStorage.setItem("ficstyle_chapterWidth", chapterWidth);
                }
                return chapterWidth;
            },
            showBackground: function() {
                var showBackground = localStorage.getItem("ficstyle_showBackground");
                if(!showBackground) {
                    showBackground = Options.showBackground;
                    localStorage.setItem("ficstyle_showBackground", showBackground);
                }
                return showBackground;
            },
            hide: function() {
                var hide = localStorage.getItem("ficstyle_hide");
                if (!hide) {
                    hide = "no";
                    localStorage.setItem("ficstyle_hide", hide);
                }
                return hide;
            },
            background: function() {
                chapters.css({
                    "background-color": Options.backgroundColor,
                    "box-shadow": "inset 0 0 8px 0 rgba(0, 0, 0, .2)"
                });
                $(".chapter .preface[role='complementary']").css("border-top-width", "0"); // the chapter title
                $("#chapters .userstuff blockquote").css({
                    "background-image": "linear-gradient(to right, rgba(0, 0, 0, .035), transparent)",
                    "border-width": "0"
                });
            },
            noBackground: function() {
                chapters.css({
                    "background-color": "transparent",
                    "box-shadow": "none"
                });
                $(".chapter .preface[role='complementary']").css("border-top-width", "1px"); // the chapter title
                $("#chapters .userstuff blockquote").css({
                    "background-image": "none",
                    "border-width": "2px",
                    "border-color": "rgba(0, 0, 0, .1)"
                });
            },
            changingCSS: function(a, b) { // a = "Variables" (user changes) or "Options" (default); b = "no" background or "yes"
                var chapterWidth, fontName, fontSize;
                if (a == "Variables") {
                    chapterWidth = this.chapterWidth();
                    fontName = this.fontName();
                    fontSize = this.fontSize();
                } else { // a = "Options"
                    chapterWidth = Options.chapterWidth;
                    fontName = Options.fontName[0];
                    fontSize = Options.fontSize;
                }
                chapters.css({
                    "width": chapterWidth + "%",
                    "font-family": fontName,
                    "font-size": fontSize + "%"
                });
                if (b == "no") {
                    this.noBackground();
                } else { // b = "yes"
                    this.background();
                }
            }
        };

        // more CSS changes
        Variables.changingCSS("Variables", Variables.showBackground()); // saved changes by user
        $("#chapters .userstuff p").each(function() { // it removes empty paragraphs
            if( !$(this).text().trim().length && !$(this).children("img").length ) {
                $(this).remove();
            }
        });
        $("#chapters .userstuff br").after("<p>").remove(); // replace '<br>' with paragraphs

        // the options displayed on the page
        $options = $("<div>", {id: "options"})
        .append( $("<div>", {html: "«", id: "font-name-minus", attr: {"title": "previous font"} }) )
        .append( $("<div>", {html: "»", id: "font-name-plus", attr: {"title": "next font"} }) )
        .append( $("<div>", {html: "-", id: "font-size-minus", attr: {"title": "decrease font size"} }) )
        .append( $("<div>", {html: "+", id: "font-size-plus", attr: {"title": "increase font size"} }) )
        .append( $("<div>", {html: "&#9643;", id: "chapter-width-minus", attr: {"title": "decrease width"} }) )
        .append( $("<div>", {html: "&#9633;", id: "chapter-width-plus", attr: {"title": "increase width"} }) )
        .append( $("<div>", {html: "b", id: "chapter-background", attr: {"title": "add/remove background"} }) )
        .append( $("<div>", {html: "r", id: "reset-local-storage", attr: {"title": "reset"} }) )
        .append( $("<div>", {html: "&#9776;", id: "show-hide", attr: {"title": "show/hide menu"} }) );
        $("body").append($options);

        if ( Variables.hide() == "no" ) {
            $("#options > div").css("display", "block");
        } else {
            $("#options > div:nth-last-child(n+2)").css("display", "none");
        }

        // to remain more or less on in the same position in the text when changes are happening
        var percent,
            checkPosition = function() {
                documentTopB = $(document).scrollTop();
                documentHeightB = $(document).height();
                percent = documentTopB / documentHeightB;
            },
            returnBack = function() {
                documentHeightA = $(document).height();
                var r = percent * documentHeightA;
                if ( documentTopB > $("#chapters").offset().top ) {
                    $("html, body").scrollTop(r);
                }
            };

        // changes triggered by the user
        $("#show-hide").click(function() {
            $("#options > div:nth-last-child(n+2)").slideToggle("300");
            if(localStorage.getItem("ficstyle_hide") == "no") {
                localStorage.setItem("ficstyle_hide", "yes");
            } else { localStorage.setItem("ficstyle_hide", "no"); }
        });

        $("#reset-local-storage").click(function() {
            checkPosition();
            localStorage.setItem("ficstyle_fontName", Options.fontName[0]);
            localStorage.setItem("ficstyle_fontSize", Options.fontSize);
            localStorage.setItem("ficstyle_chapterWidth", Options.chapterWidth);
            localStorage.setItem("ficstyle_showBackground", Options.showBackground);
            Variables.changingCSS("Options", Options.showBackground);
            returnBack();
        });

        $("#chapter-background").click(function() {
            if(localStorage.getItem("ficstyle_showBackground") == "no") {
                localStorage.setItem("ficstyle_showBackground", "yes");
                Variables.background();
            } else {
                localStorage.setItem("ficstyle_showBackground", "no");
                Variables.noBackground();
            }
        });

        var curFont, curFontIncr;
        $("#font-name-minus").click(function() {
            checkPosition();
            curFont = localStorage.getItem("ficstyle_fontName");
            for(var i = 0; i < Options.fontName.length; i++) {
                if(curFont == Options.fontName[i]) {
                    var j = i - 1;
                    if(j === -1) {
                        var u = Options.fontName.length - 1;
                        curFontIncr = Options.fontName[u];
                    } else {
                        curFontIncr = Options.fontName[j];
                    }
                    $(chapters).css("font-family", curFontIncr);
                    localStorage.setItem("ficstyle_fontName", curFontIncr);
                }
            }
            returnBack();
        });
        $("#font-name-plus").click(function() {
            checkPosition();
            curFont = localStorage.getItem("ficstyle_fontName");
            for(var i = 0; i < Options.fontName.length; i++) {
                if(curFont == Options.fontName[i]) {
                    var j = i + 1;
                    if(j === Options.fontName.length) {
                        curFontIncr = Options.fontName[0];
                    } else {
                        curFontIncr = Options.fontName[j];
                    }
                    $(chapters).css("font-family", curFontIncr);
                    localStorage.setItem("ficstyle_fontName", curFontIncr);
                }
            }
            returnBack();
        });

        var curSize;
        $("#font-size-minus").click(function() {
            checkPosition();
            curSize = parseFloat(localStorage.getItem("ficstyle_fontSize")) - 2.5;
            $(chapters).css("font-size", curSize+"%");
            localStorage.setItem("ficstyle_fontSize", curSize);
            returnBack();
        });
        $("#font-size-plus").click(function() {
            checkPosition();
            curSize = parseFloat(localStorage.getItem("ficstyle_fontSize")) + 2.5;
            $(chapters).css("font-size", curSize+"%");
            localStorage.setItem("ficstyle_fontSize", curSize);
            returnBack();
        });

        var curWidth;
        $("#chapter-width-minus").click(function() {
            checkPosition();
            curWidth = parseInt(localStorage.getItem("ficstyle_chapterWidth")) - 10;
            if(curWidth < 10) { curWidth = 10; }
            $(chapters).css("width", curWidth+"%");
            localStorage.setItem("ficstyle_chapterWidth", curWidth);
            returnBack();
        });
        $("#chapter-width-plus").click(function() {
            checkPosition();
            curWidth = parseInt(localStorage.getItem("ficstyle_chapterWidth")) + 10;
            if(curWidth > 90) { curWidth = 90; }
            $(chapters).css("width", curWidth+"%");
            localStorage.setItem("ficstyle_chapterWidth", curWidth);
            returnBack();
        });

        // full screen mode
        $workskin = $("#workskin");

        $divbuttonsFS = $("<div class='actions' style='float: right; margin-bottom: 50px'></div>");

        $fullScreen = $("<div>", {id: "full-screen", html: "<a>Full Screen &#9635;</a>"});
        $gobook = $("<div>", {id: "go-to-book", html: "<a>Go to Bookmark</a>"});
        $deletebook = $("<div>", {id: "delete-book", html: "<a>Delete Bookmark</a>"});
        $divbuttonsFS
        .append($gobook.hide())
        .append($deletebook.hide())
        .append($fullScreen);
        $("#workskin > div:nth-child(1)").prepend($divbuttonsFS);

        $scrollT = $("<div class='ficleft'><a id='arrow'>&#8593;</a> <a id='bookmark' style='font-size: 70%; letter-spacing: -1px'>set bookmark</a></div>");
        $("body").append($scrollT.hide());

        if ( $("#main > div.work > div.wrapper").length ) {
            $wrapper = $("#main > div.work > div.wrapper");
        } else { $wrapper = $("#main > div.wrapper"); }

        // changes to create full screen mode
        var fullscreen = false,
            fullScreenTrue = function() {
                $("#outer").children().hide();
                $("body").append($workskin);

                $("#workskin .preface .userstuff").addClass("notes-hidden");          
                $("#workskin .preface .summary h3, #workskin .preface .notes h3").addClass("notes-headings")
                .each(function() { var text = $(this).text(); text = text.replace(":", ""); $(this).text(text); });
                $("#workskin .preface .notes h3").siblings(".jump").removeClass("jump").addClass("userstuff notes-hidden");
                $("div.preface .module").css({ "width": "10%", "margin": "auto", "padding": "0" });

                $divbuttonsFS.css("font-size", "80%");
                $fullScreen.children("a").prepend("Exit from ");
                $scrollT.show();                
                if ( Bookmarks.checkIfExist() ) {
                    $deletebook.show();
                    $gobook.show();
                }

                $(document).scrollTop(0);

                $workskin.append($("#feedback > ul.actions").css({ "font-size": "80%", "width": "100%", "padding": " 0 0 10px 0" }));
                $("#workskin > ul.actions > li:nth-child(1), #show_comments_link").remove();

                fullscreen = true;
            };

        $("#workskin .preface .module").click(function() { // show/hide summary and notes
            $(this).children(".userstuff.notes-hidden").fadeToggle(300);
        });

        $("#full-screen").click(function() { // open/close full screen mode
            if ( !fullscreen ) {
                fullScreenTrue();
            } else {
                window.location.reload();
            }
        });

        $("#arrow").click(function() { // go to top
            $("html, body").animate({scrollTop:0}, 900);
        });

        $("#bookmark").click(function() { // set new bookmark
            Bookmarks.getNew();
            $deletebook.show();
            $gobook.show();
            $span = $("<span style='margin-left: 5px; font-family: sans-serif; color: #900;'>&#10004;</span>")
            .appendTo( $(this) );
            setTimeout(function() {
                $span.remove();
            }, 1000);
        });

        $("#go-to-book").click(function() { // go to the position of the bookmark
            var book = Bookmarks.checkIfExist("book");
            $("html, body").animate({scrollTop:book}, 900);
        });

        $("#delete-book").click(function() { // delete bookmark
            var newBookmarks = Bookmarks.cancel();
            localStorage.setItem("ficstyle_bookmarks", newBookmarks);
            $deletebook.hide();
            $gobook.hide();
        });

    } // end of regex

})(window.jQuery);