// ==UserScript==
// @name AO3: Fic's Style, Blacklist, Bookmarks
// @namespace https://github.com/Schegge
// @version 3.5.3
// @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') !== 353) {
setStorage(`${NAMESPACE}_version`, 353);
return true;
}
return false;
},
// on search pages but not on personal user profile
black: function() {
let user = document.querySelector('#greeting .user a[href *= "/users/"]') || false;
user = user && window.location.pathname.includes(user.href.split('/users/')[1]);
return document.querySelector('li.blurb.group:not(.collection):not(.tagset)') && !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
if (Check.version()) {
document.body.insertAdjacentHTML('beforeend', `
You can now blacklist authors by putting @author_name in the text area of the blacklist (wildcards are allowed, so @*something* is ok and it'll only target the authors' names).
`;
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 **/
let Bookmarks = {};
if (Feature.book) {
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 = '