// ==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  // @downloadURL none // ==/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( '' ); var els = Bookmarks.getElements(); if ( els.length ) { for(var z = 0; z < els.length; z++) { $("#menu-bookmarks > ul.menu").append( "
  • " + "" + els[z][1] + "" + "
  • " ); } } else { $("#menu-bookmarks > ul.menu").append("
  • No bookmarks yet.
  • "); } // 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("
    Time:
    " + countTime(numWords) + "
    "); }); } 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 = $(""); $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("

    ").remove(); // replace '
    ' 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

    Chapter Text

    $(this).siblings(".chapter.preface.group[role='complementary']").before( "
    " + "this chapter has " + numWords + " words (time: " + countTime(numWords) + ")
    " ); }); } // the options displayed on the page $options = $("
    ", {id: "options"}) .append( $("
    ", {html: "«", id: "font-name-minus", attr: {"title": "previous font"} }) ) .append( $("
    ", {html: "»", id: "font-name-plus", attr: {"title": "next font"} }) ) .append( $("
    ", {html: "-", id: "font-size-minus", attr: {"title": "decrease font size"} }) ) .append( $("
    ", {html: "+", id: "font-size-plus", attr: {"title": "increase font size"} }) ) .append( $("
    ", {html: "▫", id: "padding-plus", attr: {"title": "decrease width"} }) ) .append( $("
    ", {html: "□", id: "padding-minus", attr: {"title": "increase width"} }) ) .append( $("
    ", {html: "▪", id: "workskin-colors", attr: {"title": "change background and color"} }) ) .append( $("
    ", {html: "r", id: "reset-local-storage", attr: {"title": "reset"} }) ) .append( $("
    ", {html: "☰", 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 = $("
    "); $fullScreen = $("
    ", {id: "full-screen", html: "Full Screen ▣"}); $gobook = $("
    ", {id: "go-to-book", html: "Go to Bookmark"}); $divbuttonsFS .append($gobook.hide()) .append($fullScreen); $workskin.prepend($divbuttonsFS); $scrollT = $(""); $deletebook = $(""); $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 = $("
    "); $(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);