// ==UserScript== // @name AO3: Fic's Style, Blacklist, Bookmarks // @namespace https://github.com/Schegge // @version 3.0 // @description Change font, size, width, background... of a work + number of words for each chapter and estimated reading time + blacklist/savior: hide works that contain certains tags + fullscreen reading mode + bookmarks: save the position you stopped reading a fic // @author Schegge // @include http*://archiveofourown.org/* // @grant none // @require https://ajax.googleapis.com/ajax/libs/jquery/1.11.3/jquery.min.js // @icon  // @downloadURL none // ==/UserScript== (function($) { /*function debugging(varName, variable) { var message = 'FS\t[' + varName + ']'; if (variable) { message += '\t(' + typeof variable + ') ' + JSON.stringify(variable); } console.log(message); }*/ // BOOKMARKS var Bookmarks = { getBooks: function() { var bookmarks = localStorage.getItem('ficstyle_bookmarks'); if (!bookmarks || bookmarks.charAt(0) !== '[') { bookmarks = '[]'; localStorage.setItem('ficstyle_bookmarks', bookmarks); } //debugging('getBooks', JSON.parse(bookmarks)); return JSON.parse(bookmarks.trim()); }, 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); } //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, b) { var books = this.getBooks(); var url = b || this.getUrl; for (var i = 0; i < books.length; i++) { // if a bookmark already existed for the current chapter if (books[i][0] === url) { //debugging('same chapter'); if (a === 'book') { // retrieve the bookmark var book = books[i][2]; if (book.toString().indexOf('%') !== -1) { book = book.replace('%', ''); book = parseFloat(book); book = book * $(document).height(); } //debugging('checkIfExist(book)', book); return book; } else if (a === 'cancel') { // delete the old bookmark //debugging('checkIfExist(cancel)', i); //debugging('checkIfExist(cancel)', books[i]); return i; } else { //debugging('checkIfExist()', true); return true; } // if a bookmark already existed for the current fic } else if (a === 'cancel' && books[i][0].split('/chapters/')[0] === url.split('/chapters/')[0]) { // delete the old bookmark //debugging('same fic'); //debugging('checkIfExist(cancel)', i); //debugging('checkIfExist(cancel)', books[i]); return i; } } //debugging('checkIfExist', false); return false; }, cancel: function(b) { var newBookmarks = this.getBooks(); var cancel = this.checkIfExist('cancel', b); //debugging('cancel', cancel); if (cancel || cancel === 0) { newBookmarks.splice(cancel, 1); } return newBookmarks; }, getNew: function() { var newBookmarks = this.cancel(); // if the the fic was already bookmarked, delete the old bookmark newBookmarks.push([this.getUrl, this.getTitle(), this.getNewBook()]); // add new bookmark //debugging('getNew', newBookmarks); localStorage.setItem('ficstyle_bookmarks', JSON.stringify(newBookmarks)); } }; // create bookmarks' menu addCSS('ficstyle-menu', '#menu-bookmarks ul li { display: flex!important; align-items: center; justify-content: space-between; } ' + '#menu-bookmarks ul li a:first-child { flex-grow: 1; font-size: .9em; } ' + 'a.delete-book-menu { color: #900!important; } ' ); $('#header > ul').append(''); var books = Bookmarks.getBooks(); if (books.length) { for (var z = 0; z < books.length; z++) { $('#menu-bookmarks > ul.menu').append('
  • ' + books[z][1] + ' x
  • '); } } else { $('#menu-bookmarks > ul.menu').append('
  • No bookmark yet.
  • '); } $('.delete-book-menu').on('click', function() { // delete bookmark //debugging('delete-book-menu'); var newBookmarks = Bookmarks.cancel($(this).attr('data')); $(this).hide(); $(this).prev().css('opacity', '.4'); localStorage.setItem('ficstyle_bookmarks', JSON.stringify(newBookmarks)); }); // add estimated reading time var $words = $('dl.stats dd.words'); if ($words.length) { $words.each(function() { var numWords = $(this).text().replace(/,/g, ''); //debugging('numWorkWords', numWords); $(this).after('
    Time:
    ' + countTime(numWords) + '
    '); //debugging('countTime(numWords)', 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().split('.'); var minutes = Math.round(parseInt(timeReading[1]) / 100 * 60); timeReading = timeReading[0] + 'hr ' + minutes.toString() + 'min'; } return timeReading; } // CSS changes function addCSS(id, css) { //debugging('addCSS '+ id + '.length', $('style#' + id).length); if (!$('style#' + id).length) $('head').append(''); else $('style#' + id).html(css); //debugging('addCSS '+ id, css); } /** ONLY ON SEARCH PAGES **/ var Blacklist = { where: 'li.blurb.group' }; if ($(Blacklist.where).length) { //debugging('SEARCH PAGE'); var BL = []; Blacklist.what = [ '.tags .tag', '.required-tags span.text' ]; Blacklist.show = localStorage.getItem('ficstyle_blacklist_show') ? localStorage.getItem('ficstyle_blacklist_show') : true; Blacklist.get = function() { if (localStorage.getItem('ficstyle_blacklist')) { BL = JSON.parse(localStorage.getItem('ficstyle_blacklist')); } }; Blacklist.set = function(v) { var items = v ? v.replace(/\"/g, '\\\"').trim().split(',').join('","') : ''; items = items ? '["' + items + '"]' : '[]'; //debugging('Blacklist.set(' + w + ', ' + v + ')', items); localStorage.setItem('ficstyle_blacklist', items); this.get(); }; Blacklist.get(); var Blacklisting = { ifMatch: function(t) { for (var j = 0; j < BL.length; j++) { var b = BL[j].trim().toLowerCase(); if (b.length) { var r = b.replace(/[.+?^${}()|[\]\\]/g, '\\$&') .replace(/\*/g, '.*'); var reg = new RegExp('^' + r + '$'); //debugging('reg.test(' + t +') == ' + b, reg.test(t)); if (r.length && reg.test(t)) return true; } } return false; }, findMatch: function(w) { $(Blacklist.where + ' ' + w).each(function() { var tag = $(this).text().trim().toLowerCase(); if (tag && Blacklisting.ifMatch(tag)) { //debugging('Match Found', tag); if (Blacklist.show) { $(this).closest(Blacklist.where).attr('data-visibility', 'hide'); var reasons = $(this).closest(Blacklist.where).attr('data-reasons'); $(this).closest(Blacklist.where).attr('data-reasons', (!reasons) ? tag : reasons.trim() + ', ' + tag ); //debugging('reasons', $(this).closest('.work.blurb.group').attr('data-reasons')); } else { $(this).closest(Blacklist.where).attr('data-visibility', 'remove'); } } }); }, addReasons: function() { $(Blacklist.where + '[data-reasons]').each(function() { $(this).find('h4.heading').after('
    blacklisted ' + $(this).attr('data-reasons') + '
    '); }); }, clear: function() { $(Blacklist.where + '[data-visibility]').each(function() { $(this).removeAttr('data-visibility'); $(this).removeAttr('data-reasons'); if (Blacklist.show) $(this).find('.reasons').remove(); }); }, search: function() { this.clear(); if (!BL.length) return 'empty'; for (var i = 0; i < Blacklist.what.length; i++) { this.findMatch(Blacklist.what[i]); } if (Blacklist.show) this.addReasons(); }, updateTextareas: function() { var values = BL.toString(); //debugging('saveTextareas] [ficstyle_blacklist', values); $('#fs-blacklist').val(values); Blacklist.set(values); this.search(); }, saveTextareas: function() { //debugging('saveTextareas] [ficstyle_blacklist'], $('#fs-blacklist').val()); Blacklist.set($('#fs-blacklist').val()); this.search(); } }; addCSS('ficstyle-blacklist', '#menu-blacklist ul li { text-align: center!important; }' + '#fs-save-ta {color: #900!important; font-weight: bold; } ' + '#menu-blacklist textarea { font-size: .9em; line-height: 1.2em; min-height: 10em; margin: .5em!important; padding: .3em; box-shadow: 0 0 0 1px #888; width: calc(100% - 1em); border: 0; box-sizing: border-box; resize: vertical; } ' + '#menu-blacklist .fs-black-info { font-size: .9em; font-variant: small-caps; }' + '#menu-blacklist .fs-black-info span { padding: 0 2em; }' + Blacklist.where + '[data-visibility="remove"] { display: none; } ' + Blacklist.where + '[data-visibility="hide"] { opacity: .6; } ' + Blacklist.where + '[data-visibility="hide"] > *:not(.header), ' + Blacklist.where + '[data-visibility="hide"] .required-tags, ' + Blacklist.where + '[data-visibility="hide"] .fandoms.heading:not(.reasons) { display: none; }' + Blacklist.where + '[data-visibility="hide"] > .header { margin: 0!important; }' + Blacklist.where + '[data-visibility="hide"] .reasons > span { color: #fff; background-color: #900; padding: 0 .2em; } ' ); $('#header > ul').append(''); if (!Blacklist.show) $('#fs-blacklist-show').text('Show reasons for blacklisting'); Blacklisting.updateTextareas(); $('#fs-save-ta').on('click', function() { Blacklisting.saveTextareas(); $(this).text('SAVED'); setTimeout(function(){ $('#fs-save-ta').text('SAVE'); }, 1000); }); $('#fs-blacklist-show').on('click', function() { if (Blacklist.show) { Blacklist.show = false; $('#fs-blacklist-show').text('Show reasons for blacklisting'); } else { Blacklist.show = true; $('#fs-blacklist-show').text('Hide blacklisted works'); } localStorage.setItem('ficstyle_blacklist_show', Blacklist.show); Blacklisting.search(); }); } // end search page /** ONLY ON THE FIC'S PAGE **/ // include: (whatever)/works/(numbers) and (whatever)/works/(numbers)/chapters/(numbers) and exclude: navigate if (/\/works\/\d+(\/chapters\/\d+)?(?!.*navigate)/.test(window.location.pathname)) { //debugging('WORK PAGE'); var $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'] } }; addCSS('ficstyle-general', '#workskin { margin: 0; text-align: justify; max-width: none!important; } ' + '#main > div.wrapper, #main > div.work > div.wrapper { margin-bottom: 1em; } ' + '.actions { font-family: \'Lucida Grande\', \'Lucida Sans Unicode\', \'GNU Unifont\', Verdana, Helvetica, sans-serif; font-size: 14px; } ' + '.chapter .preface { margin-bottom: 0; } ' + '.chapter .preface[role="complementary"] { margin-top: 0; padding-top: 0; } ' + '#workskin .notes, #workskin .summary { font-family: inherit; font-size: 15px; } ' + '.preface.group { color: inherit; background-color: inherit; } ' + 'div.afterword { font-size: 14px } ' + '#workskin #chapters, #workskin #chapters .userstuff { width: 100%!important; box-sizing: border-box; } ' + '#workskin #chapters .userstuff p { font-family: inherit; margin: .6em auto; text-align: justify; line-height: 1.5em; } ' + '#workskin #chapters .userstuff { font-family: inherit; text-align: justify; line-height: 1.5em } ' + '#workskin #chapters .userstuff br { display: block; margin-top: .6em; content: " "; } ' + '.userstuff hr { width: 100%; height: 1px; border: 0; background-image: linear-gradient(to right, transparent, rgba(0, 0, 0, .5), transparent); margin: 1.5em 0; } ' + '#workskin #chapters a, #chapters a:link, #chapters a:visited { color: inherit; } ' + 'blockquote { font-family: inherit; } ' + '#workskin #chapters .userstuff blockquote { padding-top: 1px; padding-bottom: 1px; margin: 0 .5em; } ' + '.userstuff img { max-width: 100%; height: auto; display: block; margin: auto; } ' + '#options, .ficleft { position: fixed; bottom: 10px; margin: 0; padding: 0; font-family: Consolas, monospace; font-size: 16px; line-height: 18px; color: #000; text-shadow: 0 0 2px rgba(0, 0, 0, .4); z-index: 999; } ' + '#options { right: 10px; } ' + '.ficleft { display: none; left: 10px; } ' + '#options > div { display: none; margin: 5px 0 0 0; padding: 0 5px; cursor: pointer; } ' + '#options > div:last-child { display: block; padding: 2px 5px; color: #fff; background-color: rgba(0, 0, 0, .2); } ' + '.ficleft a, #options a { border: 0; color: #000; } ' + 'div.preface .notes, div.preface .summary, div.preface .series, div.preface .children { min-height: 0; } ' + '.notes-hidden { cursor: pointer; position: fixed; width: 50%; max-height: 50%; left: 50px; bottom: 50px; color: rgb(42, 42, 42); 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: 0; text-align: center; color: #666; } ' + '.chapterWords { font-size: .9em; color: inherit; font-family: verdana, sans-serif; font-variant: small-caps; text-align: center; margin: 2em 0 .6em; }' ); // CSS changes depending on the user var Variables = { init: function() { if (!localStorage.getItem('ficstyle')) { var all = { fontName: Options.fontName[0], fontSize: Options.fontSize, padding: Options.padding, colors: Object.keys(Options.colors)[0] }; localStorage.setItem('ficstyle', JSON.stringify(all)); } }, get: function() { //debugging('get', JSON.stringify(localStorage.getItem('ficstyle'))); return JSON.parse(localStorage.getItem('ficstyle')); }, set: function(a, b) { var all = this.get(); if (a && b) { switch (a) { case 'fontName': all.fontName = b; break; case 'fontSize': all.fontSize = b; break; case 'padding': all.padding = b; break; case 'colors': all.colors = b; break; } localStorage.setItem('ficstyle', JSON.stringify(all)); } //debugging('set', JSON.stringify(all)); addCSS('ficstyle-user-changes', '#workskin { font-family: ' + all.fontName + '; padding: 0 ' + all.padding + '%; font-size: ' + all.fontSize + '%; background-color: ' + Options.colors[all.colors][0] + '; color: ' + Options.colors[all.colors][1] + '; }' ); } }; Variables.init(); Variables.set(); // saved changes by user // remove all the non-breaking white spaces $('#chapters').html($('#chapters').html().replace(/ /g, ' ')); // # words and time for every chapter var numChapters = $('#chapters > .chapter').length; // if the fic has chapters //debugging('numChapters', numChapters); if (numChapters) { var chTexts = $('#chapters > .chapter > div.userstuff.module'); chTexts.each(function() { var text = $(this).text().replace(/(\s-\s)|(-)/g, '').replace(/[\."“”?!\)\(]/g, ' '); //debugging('wordsChapter', text); var words = text.match(/\S+\s/g); //debugging('wordsChapter', words.join(' | ')); var numWords = words.length - 2;// -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 $('body').append('
    ' + '
    «
    ' + '
    »
    ' + '
    -
    ' + '
    +
    ' + '
    ' + '
    ' + '
    ' + '
    r
    ' + '
    ' + '
    '); $('#show-hide').on('click', function() { $('#options > div:nth-last-child(n+2)').slideToggle('300'); }); // to remain more or less in the same position in the text when changes are happening var percent = 0; var checkPosition = function() { percent = $(document).scrollTop() / $(document).height(); }; var returnBack = function() { var r = percent * $(document).height(); $('html, body').scrollTop(r); }; var changeVar = function(v, d, t) { var cur = Variables.get()[v]; var opts, end; if (t === 'obj') opts = Object.keys(Options[v]); else opts = Options[v]; if (d === 1) end = opts.length; else end = d; for (var i = 0; i < opts.length; i++) { if (cur === opts[i]) { var j = i + d; if (j === end) { var u = end === d ? opts.length - 1 : 0; cur = opts[u]; } else { cur = opts[j]; } Variables.set(v, cur); break; } } }; // changes triggered by the user $('#reset-local-storage').on('click', function() { checkPosition(); localStorage.removeItem('ficstyle'); Variables.init(); Variables.set(); returnBack(); }); $('#workskin-colors').on('click', function() { changeVar('colors', 1, 'obj'); }); $('#font-name-minus').on('click', function() { checkPosition(); changeVar('fontName', -1, 'array'); returnBack(); }); $('#font-name-plus').on('click', function() { checkPosition(); changeVar('fontName', 1, 'array'); returnBack(); }); $('#font-size-minus').on('click', function() { checkPosition(); Variables.set('fontSize', Variables.get().fontSize - 2.5); returnBack(); }); $('#font-size-plus').on('click', function() { checkPosition(); Variables.set('fontSize', Variables.get().fontSize + 2.5); returnBack(); }); $('#padding-plus').on('click', function() { checkPosition(); var curPadding = Variables.get().padding + 1; if (curPadding > 40) { curPadding = 40; } Variables.set('padding', curPadding); returnBack(); }); $('#padding-minus').on('click', function() { checkPosition(); var curPadding = Variables.get().padding - 1; if (curPadding < 0) { curPadding = 0; } Variables.set('padding', curPadding); returnBack(); }); // FULL SCREEN MODE $workskin.prepend('
    ' + '' + '
    Full Screen
    ' + '
    '); $('body').append('
    ' + '^ + ' + '' + '
    '); // changes to create full screen mode var isFullScreen = false; var fullScreen = function() { //debugging('fullScreen'); $('#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').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); }); $('#full-screen a').prepend('Exit from '); $('.ficleft').show(); if (Bookmarks.checkIfExist()) { $('#delete-book').show(); $('#go-to-book').show(); } $(document).scrollTop(0); $workskin.append($('#feedback > ul.actions').css({ 'font-size': '60%', 'width': '100%', 'padding': ' 0 0 10px 0' })); $('#workskin > ul.actions > li:nth-child(1), #show_comments_link').remove(); isFullScreen = true; }; $('#workskin .preface .module').on('click', function() { // show/hide summary and notes $(this).children('.notes-hidden').fadeToggle(300); }); $('#full-screen').on('click', function() { // open/close full screen mode if (!isFullScreen) fullScreen(); else window.location.reload(); }); $('#arrow').on('click', function() { // go to top $('html, body').animate({ scrollTop: 0 }, 600); }); $('#bookmark').on('click', function() { // set new bookmark //debugging('setBookmark'); Bookmarks.getNew(); $('#go-to-book').show(); $('#delete-book').show(); $('#bookmark').css('color', '#900'); setTimeout(function() { $('#bookmark').css('color', 'inherit'); }, 1500); }); $('#go-to-book').on('click', function() { // go to the position of the bookmark //debugging('goToBook'); var book = Bookmarks.checkIfExist('book'); $('html, body').animate({ scrollTop: book }, 600); }); $('#delete-book').on('click', function() { // delete bookmark //debugging('deleteBookmark'); var newBookmarks = Bookmarks.cancel(); localStorage.setItem('ficstyle_bookmarks', JSON.stringify(newBookmarks)); $('#delete-book').hide(); $('#go-to-book').hide(); }); } // end fic page })(jQuery);