Greasy Fork

AO3: Fic's Style and Bookmarks

Change font, size, width, background.. + number of words for every chapter + estimated reading time + full screen mode + bookmarks: save the position you stopped reading a fic

目前为 2015-08-21 提交的版本。查看 最新版本

// ==UserScript==
// @name         AO3: Fic's Style and Bookmarks
// @namespace    https://greasyfork.org/it/users/12632-schegge
// @version      2.1
// @description  Change font, size, width, background.. + number of words for every chapter + 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==

// for full screen mode
window.computerFullScreen = false; // basically F11 key

(function($) {

    /*////////////////////////////////////////////////////////////////////////////////////////////
    function debugging(varName, variable) {
        var message = "{FICSTYLE} [" + varName + "]";
        if ( variable !== undefined ) message += " (" + typeof variable + ") " + variable;
        console.log(message);
    }
    /*/////////////////////////////////////////////////////////////////////////////////////////////

    // to delete old unused storage
    // debugging("ficstyle_version", localStorage.getItem("ficstyle_version"));
    if ( !localStorage.getItem("ficstyle_version") ||
        ( localStorage.getItem("ficstyle_version") !== "2.1" ) ) {
        localStorage.setItem("ficstyle_version", "2.1");
        // debugging("ficstyle_version", localStorage.getItem("ficstyle_version"));
        localStorage.removeItem("ficstyle_showBackground");
        localStorage.removeItem("ficstyle_chapterWidth");
        localStorage.removeItem("ficstyle_hide");
    }

    // BOOKMARKS
    var Bookmarks = {
        getAll: function() {
            var bookmarks = localStorage.getItem("ficstyle_bookmarks");
            if ( !bookmarks ) {
                bookmarks = "";
                localStorage.setItem("ficstyle_bookmarks", bookmarks);
            }
            // debugging("getAll", 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("#") );
            }
            // debugging("getElements", els);
            return els;
        },
        getUrl: window.location.pathname.split("/works/")[1], // work id
        getTitle: function() {
            var title = $("#workskin .preface.group h2.title.heading").text().trim();
            // debugging("getTitle heading", title);
            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 + ")";
                // debugging("getTitle chapter", chapter);
            }
            title = title.replace(/[#@]/g, " "); // just in case
            // debugging("getTitle final", title);
            return title;
        },
        getNewBook : function() {       
            var newbook = $(document).scrollTop(); // current position of the scroll bar
            // debugging("getNewBook px", newbook);
            var chs = $("dl.stats dd.chapters").text(); // # chapters
            // debugging("getNewBook chapters", chs);
            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
                // debugging("getNewBook %", newbook);
            }
            // debugging("getNewBook final", newbook);
            return newbook;
        },
        checkIfExist: function(a) {
            var url = this.getUrl;
            var els = this.getElements();
            // debugging("getUrl", url);

            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);
                        // debugging("checkIfExist('book')", book);
                        return book;
                    } else if ( a == "cancel" ) { // delete the old bookmark
                        // debugging("checkIfExist('cancel')", els[i]);
                        return "@" + els[i][0] + "#" + els[i][1] + "#" + els[i][2];
                    } else {                        
                        // debugging("checkIfExist()", true);
                        return true;
                    }
                }
            }
            // debugging("checkIfExist", false);
        },
        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
            // debugging("getNew", newBookmarks);
            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();
    if ( els.length ) {
        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>"
            );
        }
    } else {
        $("#menu-bookmarks > ul.menu").append("<li role='menu-item'><a>No bookmarks yet.</a></li>");
    }

    // add estimated reading time
    $words = $("dl.stats dd.words");
    if ( $words.length ){
        $words.each(function() {
            var numWords = $(this).text();
            numWords = numWords.replace(",", "");
            // debugging("numWorkWords", numWords);
            $(this).after("<dt>Time:</dt><dd>" + countTime(numWords) + "</dd>");
        });
    }
    function countTime(num) {
        var timeReading = parseInt( num ) / 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";
        }
        return timeReading;
    }

    /**
     * BELOW 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)) {

        $workskin = $("#workskin");

        // default values
        var Options = {
            fontName: [
                "inherit", // default (AO3 font)
                "Georgia",
                "Garamond",
                "Book Antiqua",
                "Verdana",
                "Segoe UI"
            ],
            fontSize: 100, //(%)
            padding: 7, //(%) (min = 0; max = 40) to change text's width
            colors: {//background, font color
                light: ["#ffffff", "#000000"], // default
                grey:  ["#eeeeee", "#111111"],
                sepia: ["#fbf0d9", "#54331b"],
                dark:  ["#3c3c3c", "#d2d2d2"]
            }
        };

        // CSS changes
        var addCSS = function(id, css) {
            // debugging("addCSS "+ id + ".length", $("style#" + id).length);
            if ( !$("style#" + id).length ) {
                $head = $("head");
                $style = $("<style id='" + id + "' type='text/css'>" + css + "</style>");
                $head.append($style);
            } else {
                $("style#" + id).html(css);
            }
            // debugging("addCSS "+ id, css);
        };

        addCSS(
            "ficstyle-general",
            "#workskin {\n" +
            "   margin: 0;\n" +
            "   text-align: justify;\n" +
            "   max-width: none!important;\n" +
            "}\n" +
            "#main > div.wrapper, #main > div.work > div.wrapper {\n" +
            "   margin-bottom: 1em;\n" +
            "}\n" +
            ".actions {\n" +
            "   font-family: 'Lucida Grande', 'Lucida Sans Unicode', 'GNU Unifont', Verdana, Helvetica, sans-serif;\n" +
            "   font-size: 14px;\n" +
            "}\n" +
            ".chapter .preface {\n" +
            "   margin-bottom: 0;\n" +
            "}\n" +
            ".chapter .preface[role='complementary'] {\n" +
            "   margin-top: 0;\n" +
            "   padding-top: 0;\n" +
            "}\n" +
            "#workskin .notes, #workskin .summary {\n" +
            "   font-family: inherit;\n" +
            "   font-size: 14px\n" +
            "}\n" +
            "#chapters .userstuff p {\n" +
            "   font-family: inherit;\n" +
            "   margin: 0.6em auto;\n" +
            "   text-align: justify;\n" +
            "}\n" +
            "div.afterword {\n" +
            "   font-size: 14px\n" +
            "}\n" +
            "#chapters a, #chapters a:link, #chapters a:visited {\n" +
            "   color: inherit;\n" +
            "}\n" +
            "blockquote {\n" +
            "   font-family: inherit;\n" +
            "}\n" +
            "#chapters .userstuff blockquote {\n" +
            "   padding-top: 1px;\n" +
            "   padding-bottom: 1px;\n" +
            "   margin: 0 .5em;\n" +
            "}\n" +
            ".userstuff hr {\n" +
            "   width: 100%;\n" +
            "   height: 1px;\n" +
            "   border: 0;\n" +
            "   background-image: linear-gradient(to right, transparent, rgba(0, 0, 0, .5), transparent);\n" +
            "}\n" +
            ".userstuff img {\n" +
            "   max-width: 100%;\n" +
            "   height: auto;\n" +
            "}\n" +
            "#options, .ficleft {\n" +
            "   position: fixed;\n" +
            "   bottom: 10px;\n" +
            "   margin: 0;\n" +
            "   padding: 0;\n" +
            "   font-family: Consolas, monospace;\n" +
            "   font-size: 16px;\n" +
            "   line-height: 18px;\n" +
            "   color: #000;\n" +
            "   text-shadow: 0 0 2px rgba(0, 0, 0, .4);\n" +
            "   z-index: 999;\n" +
            "}\n" +
            "#options {\n" +
            "   right: 10px;\n" +
            "}\n" +
            ".ficleft {\n" +
            "   display: none;\n" +
            "   left: 10px;\n" +
            "}\n" +
            "#options > div {\n" +
            "   display: none;\n" +
            "   margin: 5px 0 0 0;\n" +
            "   padding: 0 5px;\n" +
            "   cursor: pointer;\n" +
            "}\n" +
            "#options > div:last-child {\n" +
            "   display: block;\n" +
            "   padding: 2px 5px;\n" +
            "   color: #fff;\n" +
            "   background-color: rgba(0, 0, 0, .2);\n" +
            "}\n" +
            ".ficleft a, #options a {\n" +
            "   border: 0;\n" +
            "}\n" +
            "div.preface .notes, div.preface .summary, div.preface .series, div.preface .children {\n" +
            "   min-height: 0;\n" +
            "}\n" +
            ".notes-hidden {\n" +
            "   cursor: pointer;\n" +
            "   position: fixed;\n" +
            "   width: 50%;\n" +
            "   max-height: 50%;\n" +
            "   left: 50px;\n" +
            "   bottom: 50px;\n" +
            "   color: rgb(42, 42, 42);\n" +
            "   background-color: #fff;\n" +
            "   padding: 10px;\n" +
            "   box-shadow: 0 0 2px 1px rgba(0, 0, 0, .4);\n" +
            "   margin: 0;\n" +
            "   overflow: auto;\n" +
            "   z-index: 999;\n" +
            "   display: none;\n" +
            "}\n" +
            ".notes-headings {\n" +
            "   cursor: pointer;\n" +
            "   border-bottom-width: 0!important;\n" +
            "   margin: 0;\n" +
            "   text-align: center;\n" +
            "   color: #666;\n" +
            "}"
        );

        // 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;
            },
            padding: function() {
                var padding = localStorage.getItem("ficstyle_padding");
                if (!padding) {
                    padding = Options.padding;
                    localStorage.setItem("ficstyle_padding", padding);
                }
                return padding;
            },
            colors: function() {
                var colors = localStorage.getItem("ficstyle_colors");
                if (!colors) {
                    colors = Object.keys(Options.colors)[0];
                    localStorage.setItem("ficstyle_colors", colors);
                }
                return colors;
            },
            colorsDo: function() {
                for(var i in Options.colors) {
                    if ( i == this.colors() ) {
                        return [Options.colors[i][0], Options.colors[i][1]];
                    }
                }
            },
            changingCSS: function() {
                addCSS(
                    "ficstyle-user-changes",
                    "#workskin {\n" +
                    "   padding: 0 " + this.padding() + "%;\n" +
                    "   font-family: " + this.fontName() + ";\n" +
                    "   font-size: " + this.fontSize() + "%;\n" +
                    "   background-color: " + this.colorsDo()[0] + ";\n" +
                    "   color: " + this.colorsDo()[1] + ";\n" +
                    "}"
                );

            }
        };

        Variables.changingCSS(); // 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     

        // # words and time for every chapter
        numChapters = $("#chapters > .chapter").length; // if the fic has chapters
        // debugging("numChapters", numChapters);
        if (numChapters) {
            var chTexts = $("#chapters > .chapter > div.userstuff.module")
            .each(function() {
                var text = $(this).text();
                text = text.replace(/\s-\s/g, "");
                text = text.replace(/-/g, "");
                text = text.replace(/[\."“”?!\)\(]/g, " ");
                var words = text.match(/\S+\s/g);
                // debugging("wordsChapter", text);
                // debugging("wordsChapter", words.join(" | "));
                var numWords = words.length;
                numWords = numWords - 2; // because of <h3 class="landmark heading" id="work">Chapter Text</h3>
                $(this).siblings(".chapter.preface.group[role='complementary']").before(
                    "<div style='font-size: .9em; color: inherit; font-family: verdana, sans-serif; font-variant: small-caps; text-align: center; margin: 2em 0 .6em'>" +
                    "this chapter has " + numWords +
                    " words (time: " + countTime(numWords) +
                    ")</div>"
                );
            });
        }

        // 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: "padding-plus", attr: {"title": "decrease width"} }) )
        .append( $("<div>", {html: "&#9633;", id: "padding-minus", attr: {"title": "increase width"} }) )
        .append( $("<div>", {html: "&#9642;", id: "workskin-colors", attr: {"title": "change background and color"} }) )
        .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);

        $("#show-hide").click(function() {
            $("#options > div:nth-last-child(n+2)").slideToggle("300");
        });

        // 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;
                $("html, body").scrollTop(r);
            };

        // changes triggered by the user
        $("#reset-local-storage").click(function() {
            checkPosition();
            localStorage.setItem("ficstyle_fontName", Options.fontName[0]);
            localStorage.setItem("ficstyle_fontSize", Options.fontSize);
            localStorage.setItem("ficstyle_padding", Options.padding);
            localStorage.setItem("ficstyle_colors", Object.keys(Options.colors)[0]);
            Variables.changingCSS();
            returnBack();
        });

        var curColors, curColorIncr;
        $("#workskin-colors").click(function() {
            curColors = localStorage.getItem("ficstyle_colors");
            for(var i = 0; i < Object.keys(Options.colors).length; i++) {
                // debugging("Object.keys(Options.colors)[i]", Object.keys(Options.colors)[i]);
                if(curColors === Object.keys(Options.colors)[i]) {
                    // debugging("found", Object.keys(Options.colors)[i]);
                    var j = i + 1;
                    if(j === Object.keys(Options.colors).length) {
                        curColorIncr = Object.keys(Options.colors)[0];
                    } else {
                        curColorIncr = Object.keys(Options.colors)[j];
                    }
                    localStorage.setItem("ficstyle_colors", curColorIncr);
                    Variables.changingCSS();
                }
            }
        });

        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];
                    }
                    localStorage.setItem("ficstyle_fontName", curFontIncr);
                    Variables.changingCSS();
                }
            }
            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];
                    }
                    localStorage.setItem("ficstyle_fontName", curFontIncr);
                    Variables.changingCSS();
                }
            }
            returnBack();
        });

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

        var curPadding;
        $("#padding-plus").click(function() {
            checkPosition();
            curPadding = parseInt(localStorage.getItem("ficstyle_padding")) + 1;
            if(curPadding > 40) { curPadding = 40; }
            localStorage.setItem("ficstyle_padding", curPadding);
            Variables.changingCSS();
            returnBack();
        });
        $("#padding-minus").click(function() {
            checkPosition();
            curPadding = parseInt(localStorage.getItem("ficstyle_padding")) - 1;
            if(curPadding < 0) { curPadding = 0; }
            localStorage.setItem("ficstyle_padding", curPadding);
            Variables.changingCSS();
            returnBack();
        });

        // full screen mode
        $divbuttonsFS = $("<div class='actions' style='float: right; margin: 1.5em 0;'></div>");
        $fullScreen = $("<div>", {id: "full-screen", html: "<a>Full Screen &#9635;</a>"});
        $gobook = $("<div>", {id: "go-to-book", html: "<a>Go to Bookmark</a>"});
        $divbuttonsFS
        .append($gobook.hide())
        .append($fullScreen);
        $workskin.prepend($divbuttonsFS);

        $scrollT = $("<div class='ficleft'><a id='arrow'>&#8679;</a> <a id='bookmark'>&#9989;</a> </div>");
        $deletebook = $("<a id='delete-book'>&#9003;</a>");
        $scrollT.append( $deletebook.hide() );
        $("body").append( $scrollT.hide() );

        // changes to create full screen mode
        var fullscreen = false,
            fullscreenF11 = function(){
                var element = document.documentElement;
                if (element.mozRequestFullScreen) {
                    element.mozRequestFullScreen();
                } else if (element.webkitRequestFullScreen) {
                    element.webkitRequestFullScreen();
                }
            },
            fullScreenTrue = function() {
                // debugging("fullScreen");

                if ( window.computerFullScreen ) fullscreenF11();

                $("#outer").children().hide();
                $("body").append($workskin);

                $("#workskin .preface").css({"margin": "0", "padding-bottom": "0"});
                $("#workskin div.afterword").css("margin-bottom", "2.5em");
                $("#workskin .preface .summary .userstuff, #work_endnotes .userstuff").addClass("notes-hidden");
                $("#workskin .preface .notes").each(function() {
                    var $notes = $("<div class='notes-hidden'></div>");
                    $(this).children("h3.heading").siblings().appendTo($notes);
                    $(this).append($notes);
                });
                $("#workskin .preface .summary h3, #workskin .preface .notes h3").addClass("notes-headings")
                .each(function() { var text = $(this).text(); text = text.replace(":", ""); $(this).text(text); });

                $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(".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}, 600);
        });

        $("#bookmark").click(function() { // set new bookmark
            // debugging("setBookmark");
            Bookmarks.getNew();
            $gobook.show();
            $deletebook.show();
            $("#bookmark").css("color", "#900");
            setTimeout(function() {
                $("#bookmark").css("color", "inherit");
            }, 1000);
        });

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

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

    } // end of regex

})(window.jQuery);