// ==UserScript== // @name AO3: Fic's Style and Bookmarks // @namespace https://greasyfork.org/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  // @downloadURL none // ==/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( '' ); var els = Bookmarks.getElements(); for(var z = 0; z < els.length; z++) { $("#menu-bookmarks > ul.menu").append( "
  • " + "" + els[z][1] + "" + "
  • " ); } // 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("
    Time:
    " + timeReading + "
    "); }); } //*//*//*//*//*//*//*// 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 = $(""); $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("

    ").remove(); // replace '
    ' with paragraphs // 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: "chapter-width-minus", attr: {"title": "decrease width"} }) ) .append( $("
    ", {html: "□", id: "chapter-width-plus", attr: {"title": "increase width"} }) ) .append( $("
    ", {html: "b", id: "chapter-background", attr: {"title": "add/remove background"} }) ) .append( $("
    ", {html: "r", id: "reset-local-storage", attr: {"title": "reset"} }) ) .append( $("
    ", {html: "☰", 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 = $("
    "); $fullScreen = $("
    ", {id: "full-screen", html: "Full Screen ▣"}); $gobook = $("
    ", {id: "go-to-book", html: "Go to Bookmark"}); $deletebook = $("
    ", {id: "delete-book", html: "Delete Bookmark"}); $divbuttonsFS .append($gobook.hide()) .append($deletebook.hide()) .append($fullScreen); $("#workskin > div:nth-child(1)").prepend($divbuttonsFS); $scrollT = $(""); $("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 = $("") .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);