// ==UserScript== // @name AO3: Fic's Style, Blacklist, Bookmarks // @namespace https://github.com/Schegge // @version 3.2.1 // @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 // @icon  // @downloadURL none // ==/UserScript== (function() { // CSS changes function addCSS(id, css) { if (!document.querySelector('style#' + id)) { let style = document.createElement('style'); style.id = id; style.textContent = css; document.getElementsByTagName('head')[0].appendChild(style); } else { document.querySelector('style#' + id).textContent = css; } } // estimate reading time: 200 wpm function countTime(num) { let time = (parseInt(num, 10) / 200 / 60).toFixed(2).toString().split('.'); return (time[0] !== '0' ? time[0] + 'hr ' : '') + Math.round(parseInt(time[1], 10) / 100 * 60) + 'min'; } function getScroll() { return Math.max(document.documentElement.scrollTop, window.scrollY); } function setScroll(s) { window.scroll(0, s ? s : 0); } function getDocHeight() { return Math.max(document.documentElement.scrollHeight, document.documentElement.offsetHeight, document.body.scrollHeight, document.body.offsetHeight); } // BOOKMARKS var BM = []; var Bookmarks = { get: function() { if (localStorage.getItem('ficstyle_bookmarks')) BM = JSON.parse(localStorage.getItem('ficstyle_bookmarks')); }, set: function() { localStorage.setItem('ficstyle_bookmarks', JSON.stringify(BM)); }, getUrl: window.location.pathname.split('/works/')[1], getTitle: function() { let title = document.querySelector('#workskin .preface.group h2.title.heading').textContent; // cut long titles title = title.trim().substring(0, 28); // if chapter by chapter, get the number of the chapter if (this.getUrl.indexOf('chapters') !== -1) { let chapter = document.querySelector('#chapters > .chapter > .chapter.preface.group > h3 > a').textContent; chapter = chapter.replace('Chapter ', 'ch'); title += ' (' + chapter + ')'; } return title; }, getNewBook: function() { let newbook = getScroll(); // if chapter by chapter view or work completed (number/number is the same), calculate as a percent if (window.location.pathname.indexOf('chapters') !== -1 || /(\d+)\/\1/.test(document.querySelector('dl.stats dd.chapters').textContent)) { newbook = (newbook / getDocHeight()).toFixed(4) + '%'; } return newbook; }, checkIfExist: function(a, b) { let url = b || this.getUrl; for (let i = 0; i < BM.length; i++) { // if the same fic if (BM[i][0].split('/chapters/')[0] === url.split('/chapters/')[0]) { if (a === 'cancel') { return i; // if the same chapter } else if (BM[i][0] === url) { if (a === 'book') { let book = BM[i][2]; if (book.toString().indexOf('%') !== -1) { book = parseFloat(book.replace('%', '')); book *= getDocHeight(); } return book; } return true; } } } return false; }, cancel: function(url) { let found = this.checkIfExist('cancel', url); if (found !== false) BM.splice(found, 1); }, getNew: function() { this.cancel(); BM.push([this.getUrl, this.getTitle(), this.getNewBook()]); this.set(); } }; Bookmarks.get(); // 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; } ' ); var bmMenu = document.createElement('li'); var bmMenuDrop = document.createElement('ul'); bmMenu.id = 'menu-bookmarks'; bmMenu.className = 'dropdown'; bmMenu.innerHTML = 'Bookmarks'; bmMenuDrop.className = 'menu dropdown-menu'; bmMenu.appendChild(bmMenuDrop); document.querySelector('#header > ul').appendChild(bmMenu); if (BM.length) { let clickDelete = function() { Bookmarks.cancel(this.getAttribute('data-url')); Bookmarks.set(); this.style.display = 'none'; this.previousSibling.style.opacity = '.4'; }; for (let z = 0; z < BM.length; z++) { let bmLi = document.createElement('li'); bmLi.innerHTML = '' + BM[z][1] + ''; let deleteBookMenu = document.createElement('a'); deleteBookMenu.className = 'delete-book-menu'; deleteBookMenu.title = 'delete bookmark'; deleteBookMenu.setAttribute('data-url', BM[z][0]); deleteBookMenu.textContent = 'x'; deleteBookMenu.addEventListener('click', clickDelete); bmLi.appendChild(deleteBookMenu); bmMenuDrop.appendChild(bmLi); } } else { bmMenuDrop.innerHTML = '