// ==UserScript==
// @name AO3: Fic's Style and Bookmarks
// @namespace https://github.com/Schegge
// @version 2.3.2
// @description Change font, size, width, background.. + number of words for every chapter + estimated reading time + fullscreen 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($) {
/*function debugging(varName, variable) {
var message = 'FS\t[' + varName + ']';
if (variable !== undefined) {
message += '\t(' + typeof variable + ') ' + variable;
}
console.log(message);
}*/
// BOOKMARKS
var Bookmarks = {
getBooks: function() {
var bookmarks = localStorage.getItem('ficstyle_bookmarks');
bookmarks = bookmarks.trim();
if (!bookmarks) {
bookmarks = '[]';
localStorage.setItem('ficstyle_bookmarks', bookmarks);
} else if (bookmarks.charAt(0) === '@') {
bookmarks = bookmarks.substring(1);
bookmarks = bookmarks.replace(/@/g, '"],["');
bookmarks = bookmarks.replace(/#/g, '","');
bookmarks = '[["' + bookmarks;
bookmarks += '"]]';
//debugging('bookmarks', bookmarks);
localStorage.setItem('ficstyle_bookmarks', bookmarks);
}
//debugging('getBooks', JSON.parse(bookmarks));
return JSON.parse(bookmarks);
},
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].toString();
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)', 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] + ' ');
}
} 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();
numWords = numWords.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);
}
// BELOW 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)) return;
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 } ' +
'#chapters .userstuff p { font-family: inherit; margin: .6em auto; text-align: justify; line-height: 1.5em } ' +
'#chapters .userstuff { font-family: inherit; text-align: justify; line-height: 1.5em } ' +
'#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; } ' +
'#chapters a, #chapters a:link, #chapters a:visited { color: inherit; } ' +
'blockquote { font-family: inherit; } ' +
'#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: localStorage.getItem('ficstyle_fontName') ? localStorage.getItem('ficstyle_fontName') : Options.fontName[0],
fontSize: localStorage.getItem('ficstyle_fontSize') ? parseFloat(localStorage.getItem('ficstyle_fontSize')) : Options.fontSize,
padding: localStorage.getItem('ficstyle_padding') ? parseInt(localStorage.getItem('ficstyle_padding')) : Options.padding,
colors: localStorage.getItem('ficstyle_colors') ? localStorage.getItem('ficstyle_colors') : Object.keys(Options.colors)[0]
};
localStorage.removeItem('ficstyle_fontName');
localStorage.removeItem('ficstyle_fontSize');
localStorage.removeItem('ficstyle_padding');
localStorage.removeItem('ficstyle_colors');
localStorage.setItem('ficstyle', JSON.stringify(all));
}
},
get: function() {
var all = localStorage.getItem('ficstyle');
//debugging('get', JSON.stringify(all));
return JSON.parse(all);
},
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 { padding: 0 ' + all.padding + '%; font-family: ' + all.fontName + '; 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();
text = text.replace(/(\s-\s)|(-)/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
$('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;
function checkPosition() {
percent = $(document).scrollTop() / $(document).height();
}
function returnBack() {
var r = percent * $(document).height();
$('html, body').scrollTop(r);
}
// 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() {
var curColors = Variables.get().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('foundOld', Object.keys(Options.colors)[i]);
var j = i + 1;
if (j === Object.keys(Options.colors).length) {
curColor = Object.keys(Options.colors)[0];
} else {
curColor = Object.keys(Options.colors)[j];
}
//debugging('foundNew', curColor);
Variables.set('colors', curColor);
break;
}
}
});
$('#font-name-minus').on('click', function() {
checkPosition();
var curFont = Variables.get().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;
curFont = Options.fontName[u];
} else {
curFont = Options.fontName[j];
}
Variables.set('fontName', curFont);
break;
}
}
returnBack();
});
$('#font-name-plus').on('click', function() {
checkPosition();
var curFont = Variables.get().fontName;
for(var i = 0; i < Options.fontName.length; i++) {
if (curFont === Options.fontName[i]) {
var j = i + 1;
if (j === Options.fontName.length) {
curFont = Options.fontName[0];
} else {
curFont = Options.fontName[j];
}
Variables.set('fontName', curFont);
break;
}
}
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('');
$('body').append('');
// changes to create full screen mode
var isFullScreen = false;
function fullScreen() {
//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': '80%', '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();
});
})(jQuery);