// ==UserScript==
// @name AO3: Fic's Style, Blacklist, Bookmarks
// @namespace https://github.com/Schegge
// @version 3.5.2
// @description Change font, size, width, background... of a work + blacklist: hide works that contain certains tags, have too many fandoms/relations/chapters/words and other options + fullscreen reading mode + bookmarks: save the position you stopped reading a fic + number of words for each chapter and estimated reading time
// @author Schegge
// @include http*://archiveofourown.org/*
// @grant none
// @icon 
// @downloadURL none
// ==/UserScript==
(function() {
const NAMESPACE = 'ficstyle';
/** FEATURES **/
const Feature = {
style: true,
book: true,
black: true,
wpm: 250
}
// Object.assign changes only the same keys
Object.assign(Feature, getStorage(`${NAMESPACE}_feature`, '{}'));
// check which page
const Check = {
// script version
version: function() {
if (getStorage(`${NAMESPACE}_version`, '1') !== 352) {
setStorage(`${NAMESPACE}_version`, 352);
return true;
}
return false;
},
// on search pages but not on personal user profile
black: function() {
let user = document.querySelector('#greeting .user a[href *= "/users/"]');
user = user && window.location.pathname.includes(user.href.split('/users/')[1]);
return document.querySelector('li.blurb.group:not(.collection)') && !user;
},
// include /works/(numbers) and /works/(numbers)/chapters/(numbers) and exclude /works/(whatever)navigate
work: function() {
return /\/works\/\d+(\/chapters\/\d+)?(?!.*navigate)/.test(window.location.pathname);
},
// Full Screen
fullScreen: false
};
// new version notification + cleaning old stuff
if (Check.version()) {
localStorage.removeItem(`${NAMESPACE}_blacklist_langs`);
if (localStorage.getItem(NAMESPACE)) {
localStorage.setItem(`${NAMESPACE}_styling`, localStorage.getItem(NAMESPACE));
localStorage.removeItem(NAMESPACE);
}
if (localStorage.getItem(`${NAMESPACE}_wpm`)) {
Feature.wpm = localStorage.getItem(`${NAMESPACE}_wpm`);
localStorage.removeItem(`${NAMESPACE}_wpm`);
}
document.body.insertAdjacentHTML('beforeend', `
Every feature is now optional, you can disable them in the menu "Features". Disabling them doesn't erase the values saved previously. To disable the estimated reading time set the words per minute to 0.
I've slightly changed how to save of the preferences, none of them should be lost apart from your preferred languages in the blacklist.
`;
document.querySelector('#header > ul').appendChild(featureMenu);
document.getElementById(`${NAMESPACE}-feature-save`).addEventListener('click', function() {
Feature.style = document.getElementById(`${NAMESPACE}-feature-style`).checked;
Feature.book = document.getElementById(`${NAMESPACE}-feature-book`).checked;
Feature.black = document.getElementById(`${NAMESPACE}-feature-black`).checked;
let wpm = document.getElementById(`${NAMESPACE}-feature-wpm`).value;
Feature.wpm = wpm ? Math.min(Math.max(parseInt(wpm, 10), 0), 1000) : 0;
setStorage(`${NAMESPACE}_feature`, Feature);
this.textContent = 'SAVING...';
window.location.replace(window.location.href);
});
// add estimated reading time for every fic found
if (Feature.wpm) {
document.querySelectorAll('dl.stats dd.words').forEach(w => {
let numWords = w.textContent.replace(/,/g, '');
w.insertAdjacentHTML('afterend', `
Time:
${countTime(numWords)}
`);
});
}
/** BOOKMARKS **/
const Bookmarks = {
list: [],
get: function() {
this.list = getStorage(`${NAMESPACE}_bookmarks`, '[]');
},
set: function() {
setStorage(`${NAMESPACE}_bookmarks`, this.list);
},
fromBook: window.location.search === '?bookmark',
getUrl: window.location.pathname.split('/works/')[1],
getTitle: function() {
let title = document.querySelector('#workskin .preface.group h2.title.heading').textContent.trim().substring(0, 28);
// get the number of the chapter if chapter by chapter
if (this.getUrl.includes('/chapters/')) title += ` (${document.querySelector('#chapters > .chapter > .chapter.preface.group > h3 > a').textContent.replace('Chapter ', 'ch')})`;
return title;
},
getPosition: function() {
let position = getScroll();
// calculate % if chapter by chapter view or work completed (number/number is the same)
if (window.location.pathname.includes('/chapters/') || /(\d+)\/\1/.test(document.querySelector('dl.stats dd.chapters').textContent)) {
position = (position / getDocHeight()).toFixed(4) + '%';
}
return position;
},
checkIfExist: function(what, link) {
let url = link || this.getUrl;
for (let i = 0, len = this.list.length; i < len; i++) {
// check if the same fic already exists
if (this.list[i][0].split('/chapters/')[0] === url.split('/chapters/')[0]) {
// i need the index to delete the old bookmark (for change or cancel)
if (what === 'cancel') {
return i;
// check if the same chapter
} else if (this.list[i][0] === url) {
// retrieve the bookmark position
if (what === 'book') {
let book = this.list[i][2];
// if the bookmark is in %
if (book.toString().includes('%')) {
book = parseFloat(book.replace('%', ''));
book *= getDocHeight();
}
return book;
}
// just check if a bookmark exist
return true;
}
}
}
return false;
},
cancel: function(url) {
let found = this.checkIfExist('cancel', url);
if (found !== false) this.list.splice(found, 1);
},
getNew: function() {
this.cancel();
this.list.push([this.getUrl, this.getTitle(), this.getPosition()]);
this.set();
},
html: function() {
let bookMenu = document.createElement('li');
bookMenu.id = `${NAMESPACE}-book`;
bookMenu.className = 'dropdown';
bookMenu.innerHTML = 'Bookmarks';
let bookMenuDrop = document.createElement('ul');
bookMenuDrop.className = 'menu dropdown-menu';
bookMenu.appendChild(bookMenuDrop);
document.querySelector('#header > ul').appendChild(bookMenu);
if (this.list.length) {
let self = this;
let clickDelete = function() {
self.cancel(this.getAttribute('data-url'));
self.set();
this.style.display = 'none';
this.previousSibling.style.opacity = '.4';
};
this.list.forEach(item => {
let bookMenuLi = document.createElement('li');
bookMenuLi.innerHTML = `${item[1]}`;
let bookMenuDelete = document.createElement('a');
bookMenuDelete.className = `${NAMESPACE}-book-delete`;
bookMenuDelete.title = 'delete bookmark';
bookMenuDelete.setAttribute('data-url', item[0]);
bookMenuDelete.textContent = 'x';
bookMenuDelete.addEventListener('click', clickDelete);
bookMenuLi.appendChild(bookMenuDelete);
bookMenuDrop.appendChild(bookMenuLi);
});
} else {
bookMenuDrop.innerHTML = '