Preload, navigation and other enhancements for acomics.ru comics viewer
// ==UserScript== // @name Enhanced viewer for acomics.ru // @name:ru Улучшенный просмотрщик для acomics.ru // @description:ru Предзагрузка, удобная навигация и прочие улучшения для сайта acomics.ru // @description Preload, navigation and other enhancements for acomics.ru comics viewer // @author Sanya_Zol (https://github.com/SanyaZol) // @license Unlicense // @icon  // @homepageURL https://greasyfork.org/ru/scripts/10521 // @supportURL https://greasyfork.org/ru/scripts/10521 // @namespace Sanya_Zol // @version 0.3.0 // @match https://acomics.ru/* // @run-at document-start // @grant none // @require https://code.jquery.com/jquery-3.7.1.min.js // ==/UserScript== // @ts-check // Greasy Fork: https://greasyfork.org/ru/scripts/10521 // Github: https://github.com/SanyaZol/acomics-enhanced-viewer-userscript /** @typedef {{Page:number,LastPage:number,Image:string|null,ContentSerialNomargin:string,ContentMargin:string,Title:string}} parsedPage */ (function () { class AcomicsViewer { settings = { // Блокировать случайное закрытие страницы blockUnload: false, // таймаут запроса в милисекундах (секундах*1000) AjaxRequestTimeoutMs: 10000, // чтобы деактивировать или снова активировать быстрый скролл, нажми Shift+[˄] или Shift+[˅] scrollFasterDefault: true, // сейчас при скролле страницы стрелками по вертикали [˄] и [˅] будет быстрый скролл на 300 пикселей scrollFasterPx: 300, // а при зажатом Ctrl - на 100 scrollFasterCtrlPx: 100, // время в милисекундах, за которое скроллить при использовании быстрого скролла scrollFasterSpeed: 75, // время в милисекундах, за которое скроллить наверх при смене страниц scrollSpeed: 75, // выпилить комментарии - они будут вылазить при нажатии [˄]+[˃] // (стрелки вправо при зажатой стрелке вверх) и выезжать при нажатии [˂]+[˃] contentMarginMover: false, // выводить отладочную информацию в левом верхнем углу страницы visualLog: false, }; constants = { eventNamespace: 'acomics-enhanced-viewer-userscript', // префикс localStorage localStorageKeyPrefix: '**zol**page**', logPrefix: '[acViewer] ', }; stage = 'noInit'; /** @type {JQuery<HTMLElement> | null} */ logOverlay = null; log(text) { let s = `[${this.stage}] ${text}`; if (this.settings.visualLog) { if (!this.logOverlay) { this.logOverlay = $('<div/>'); this.logOverlay.css({ position: 'fixed', // absolute zIndex: 1e6, left: '1px', top: '1px', padding: '1px', fontFamily: 'Verdana', fontSize: '7px', color: '#fff', textShadow: '#000 0px 0px 1px',//, #000 2px 2px 2px background: 'rgba(255,255,255,0.3)', borderRadius: '2px', border: '1px rgba(255,255,255,0.5) solid', overflowY: 'hidden', maxHeight: '60px' }).appendTo('body'); } $('<div/>').text(s).hide().prependTo(this.logOverlay).slideDown(200).delay(5000).slideUp(200, function () { $(this).remove(); }); this.logOverlay.fadeIn().finish().delay(5000).slideUp(200); } console.log(`${this.constants.logPrefix}${s}`); } preInitFailed = false; preInitGuard = false; preInitSucceeded = false; /** @type {string|null} */ comicsName = null; /** @type {number|null} */ comicsPage = null; /** * @param {number} id * @returns {string} */ makeUrl(id) { if (!this.comicsName) throw new Error("makeUrl() requires this.comicsName to be set"); if (id === null) throw new Error("makeUrl() requires id to be set"); return '/~' + this.comicsName + '/' + id; } /** @type {string|null} */ localStorageKey = null; /** @type {string|null} */ localStorageKeyPage = null; /** @returns {void} */ store() { if (!this.localStorageKey || this.comicsPage === null) throw new Error("store() requires this.localStorageKey to be set"); if (this.comicsPage !== this.wantPage) throw new Error("store() requires this.comicsPage === this.wantPage"); window.localStorage.setItem(this.localStorageKey, `${this.comicsPage}`); } /** @type {number|null} */ wantPage = null; static _rx1 = /(?:(?:Мой )?(?:патреон|яндекс|yandex|paypal|wm(?:z|r|e|u)|кошелек|счет)[\s.-]*(?:кошелек|money|деньги)?:?|(?:https?:\/\/)?(?:www\.)?patreon.com\/[^\s]*|41\d{13}\b|(?:wm)?(?:z|r|e|u)\d{12}\b)/gi; static _rx2 = /(?:(?:\s*(?:<p>\s*<\/p>|<br ?\/? ?>)\s*)+|^(?:\s*<br ?\/? ?>)+|(?:<br ?\/? ?>\s*)+$)/gi; static _dangerous_tags = (['script', 'style', 'link', 'embed', 'object'].join(', ')); static removeInlineHandlers(element) { // http://stackoverflow.com/a/3593250 http://stackoverflow.com/q/3593242 http://stackoverflow.com/questions/3593242/how-to-remove-all-attributes-from-body-with-js-or-jquery for (var i = element.attributes.length; i-- > 0;) { var attr = element.attributes[i]; if (attr.nodeName.toLowerCase().indexOf('on') == 0) { element.removeAttributeNode(attr); } } } static removeScripts(el) { var f; while (f = el.querySelector(AcomicsViewer._dangerous_tags)) { // we querying selector every time because tags can contain each other f.parentNode.removeChild(f); } var f = el.querySelectorAll('*'); for (var i = f.length; i-- > 0;) { AcomicsViewer.removeInlineHandlers(f[i]); } } /** * * @param {string} s * @returns {parsedPage|null} */ parsePageString(s) { // TODO: validate comics page is really from the comics that is in URL var o = {}; var parser = new DOMParser(); var d = parser.parseFromString(s, "text/html"); { let x1 = d.querySelector('.common-content .button-goto span'); if (!x1) { return null; } let x = $.trim(x1.innerHTML).match(/^(\d+)\/(\d+)$/); if (!x) { return null; } o.Page = parseInt(x[1]); o.LastPage = parseInt(x[2]); } { let x = d.querySelector('.common-content .issue-description-buttons'); if (x) { x.parentNode?.removeChild(x); }; } { let x = d.querySelector('.common-content .reader-comment-form'); if (x) { x.parentNode?.removeChild(x); }; } { let x = d.querySelector('.common-content .reader-issue img.issue')?.getAttribute('src'); o.Image = x || null; } { let x = d.querySelector('.common-content .reader-issue'); if (!x) return null; AcomicsViewer.removeScripts(x); o.ContentSerialNomargin = x.innerHTML; } { let x = d.querySelector('.common-content .view-container .view-article'); if (!x) return null; AcomicsViewer.removeScripts(x); let y = x.querySelector('.issue-description-text'); if (y) y.innerHTML = y.innerHTML.replace(AcomicsViewer._rx1, '').replace(AcomicsViewer._rx2, ''); o.ContentMargin = x.innerHTML; } { let x = d.querySelector('title'); o.Title = x ? ($.trim(x.innerText)) : '?'; } return o; } /** @type {number|null} */ comicsLastPage = null; /** @type {number|null} */ init_comicsPage = null; /** @type {number|null} */ init_wantPage = null; /** @type {parsedPage|null} */ tmp_curr = null; /** @returns {boolean} */ preInit() { if (this.preInitSucceeded) return true; if (this.preInitFailed || this.preInitGuard) { return false; } this.preInitFailed = true; this.preInitGuard = true; this.stage = 'preInit.1'; { const m = /^\/~([^\/]+)\/(\d*)(?:\D|$)/.exec(location.pathname.toString()); if (!m) { return false; } if (!window.$) { this.log('no window.$'); return false; } this.comicsName = m[1]; const page = parseInt(m[2], 10); if (m[2] == '' || isNaN(page) || page < 1) { return false; } this.comicsPage = page; } this.localStorageKey = this.constants.localStorageKeyPrefix + this.comicsName; this.localStorageKeyPage = this.localStorageKey + '*'; { /** @var {number|null} */ let stored = null; const storedString = window.localStorage.getItem(this.localStorageKey); if (storedString) { stored = parseInt(storedString, 10) || null; } if (stored !== null && stored != this.comicsPage) { this.wantPage = stored; } else { this.wantPage = this.comicsPage; } } this.stage = 'preInit.2'; let curr = this.parsePageString($('html')[0].outerHTML); if (!curr) { return false; } this.comicsLastPage = curr.LastPage; this.comicsPageTitle = curr.Title; this.init_comicsPage = this.comicsPage; this.init_wantPage = this.wantPage; this.stage = 'preInit.3'; if (this.comicsPage != curr.Page) { if ( confirm( 'Что-то пошло не так.' + '\n' + '\n[1] Сохраненная страница:\t' + this.comicsPage + '\n[2] Страница комикса (html):\t' + curr.Page + '\n[3] Максимальная страница (html):\t' + this.comicsLastPage + '\n' + '\nЗаменить сохраненную страницу ' + this.comicsPage + ' на ' + curr.Page + '?' ) ) { this.wantPage = this.comicsPage = curr.Page; this.store(); location.href = this.makeUrl(this.comicsPage); } return false; } this.tmp_curr = curr; // this.putCached( this.comicsPage, curr ); // this.putCached( this.tmp_curr.Page, this.tmp_curr ); delete this.tmp_curr; this.stage = 'preInit.4'; this.preInitSucceeded = true; return true; } initSucceeded = false; initFailed = false; initGuard = false; noPushState = false; firstPushState = false; changeHandlers() { this.store(); this.ShouldIDoSomething(); if (this.noPushState) { this.noPushState = false; return; } let stateObj = { page: this.comicsPage }; let title = `#${this.comicsPage} | ${this.comicsPageTitle}`; let url = this.makeUrl(this.comicsPage); if (this.firstPushState) { this.firstPushState = false; history.replaceState(stateObj, title, url); } else { history.pushState(stateObj, title, url); } } $htmlbody = null; doScroll(relative, scrollAmount) { this.$htmlbody || (this.$htmlbody = $('html,body')); if (relative) { this.$htmlbody.finish().animate({ scrollTop: Math.max(0, $(window).scrollTop() + scrollAmount) }, this.settings.scrollFasterSpeed); } else { this.$htmlbody.finish().animate({ scrollTop: scrollAmount }, this.settings.scrollSpeed); } } SessCacheQuery(page) { if (!this.localStorageKeyPage) throw new Error("No localStorageKeyPage"); var r = window.sessionStorage.getItem(this.localStorageKeyPage + page); if (!r) return false; try { r = JSON.parse(r); } catch (e) { return false; } return r; }; SessCachePut(page, data) { if (!this.localStorageKeyPage) throw new Error("No localStorageKeyPage"); var d = JSON.stringify(data); window.sessionStorage.setItem(this.localStorageKeyPage + page, d); }; /** @type {number|null} */ handleSwapTimeout = null; handleSwap(page, o) { this.log('page.swap: swapping page to #' + page + ( o.Preloaded ? '' : ' which doesn\'t have preloaded image' )); this.$contentMargin.html(o.ContentMargin); this.$contentSer.find('img.issue').attr('src', o.Image); if (this.handleSwapTimeout !== null) clearTimeout(this.handleSwapTimeout); this.handleSwapTimeout = setTimeout(() => { this.$contentSer.html(o.ContentSerialNomargin); this.handleSwapTimeout = null; }, 50); this.$contentCounter.html(`${o.Page}/${o.LastPage}`); if (o.Page < 2) this.$contentBtnPrev.addClass('button-inactive'); else this.$contentBtnPrev.removeClass('button-inactive'); if (o.Page >= o.LastPage) this.$contentBtnNext.addClass('button-inactive'); else this.$contentBtnNext.removeClass('button-inactive'); // title this.comicsPageTitle = o.Title; this.comicsPage = page; document.title = '#' + this.comicsPage + ' | ' + this.comicsPageTitle; // $(window).scrollTop(0); this.doScroll(false, 0); this.changeHandlers(); }; /** @type {Object} */ plData = Object.create(null); putCached(page, data) { this.plData[page] = data; this.SessCachePut(page, data); this.putCachedPreloadImage(page); }; putCachedPreloadImage(page) { this.plData[page].Preloaded = false; this.plData[page].DomImage = new Image(); this.plData[page].DomImage.onerror = (err) => { console.error('Cannot load image #' + page + ' (window._err_img): ' + this.plData[page].Image); console.error({ "this.plData[page].Image": this.plData[page].Image, "err (event)": err }); // TODO: debugger; window._err_img = this.plData[page].Image; if (typeof (this.plData[page]) != 'undefined') { delete this.plData[page]; } }; var log = false; this.plData[page].DomImage.onload = () => { delete this.plData[page].DomImage.onload; if (typeof (this.plData[page]) == 'object') { if (log) { this.log('page.cache: IMG preloaded #' + page); } this.plData[page].Preloaded = true; } else { this.log('page.cache: IMG preloaded #' + page + ' but OBJ DOES NOT EXISTS!!'); } }; this.plData[page].DomImage.src = this.plData[page].Image; setTimeout(() => { log = true; }, 30); }; /** @type {JQuery<HTMLElement>|null} */ $contentMargin = null; /** @type {JQuery<HTMLElement>|null} */ $content = null; /** @type {JQuery<HTMLElement>|null} */ $contentSer = null; /** @type {JQuery<HTMLImageElement>|null} */ $contentImage = null; /** @type {JQuery<HTMLElement>|null} */ $contentCounter = null; /** @type {JQuery<HTMLElement>|null} */ $contentBtnPrev = $('.common-content .button-previous'); /** @type {JQuery<HTMLElement>|null} */ $contentBtnNext = $('.common-content .button-next'); gcAllowed = false; gcAllowedSet() { this.gcAllowed = true; } gc() { if (!this.gcAllowed) { return; } var r = this.populateList(5); r.push(this.wantPage); for (var page_str in this.plData) { page = parseInt(page_str, 10); if ($.inArray(page, r) == -1) { this.log('this.gc: removing old page #' + page); delete this.plData[page]; window.sessionStorage.removeItem(this.localStorageKeyPage + page); } } var prefixl = this.localStorageKeyPage.length; for (var i = window.sessionStorage.length; i-- > 0;) { var k = window.sessionStorage.key(i); if (typeof (k) == 'string' && k.indexOf(this.localStorageKeyPage) == 0) { var page = parseInt(k.substr(prefixl), 10); if ($.inArray(page, r) == -1) { this.log('this.gc: removing sessionStorage page #' + page); window.sessionStorage.removeItem(this.localStorageKeyPage + page); } } } // key() this.gcAllowed = false; setTimeout(() => this.gcAllowedSet(), 10000); }; swapPage() { var page = this.wantPage; if (typeof (this.plData[page]) == 'undefined') { this.log('page.swap: NO CACHE FOR #' + page); return false; } if (typeof (this.plData[page]) == 'boolean') { this.log('page.swap: CACHE STILL WAITING #' + page); return true; } if (typeof (this.plData[page]) == 'object') { var o = this.plData[page]; this.handleSwap(page, o); } return true; }; ShouldIDoSomething() { if (this.wantPage < 1) { this.log('page.swap: wrong wantPage = ' + this.wantPage); this.wantPage = 1; } else if (this.wantPage > this.comicsLastPage) { this.log('page.swap: wrong wantPage = ' + this.wantPage); this.wantPage = this.comicsLastPage; } if (this.wantPage != this.comicsPage) { if (!this.swapPage()) { this.log('page.cache: requesting non-preloaded page #' + this.wantPage); this.ajaxPreload(this.wantPage); return true; } } var r = this.populateList(0); var rl = r.length; for (var i = 0; i < rl; i++) { var page = r[i]; if (typeof (this.plData[page]) == 'undefined') { var cached = this.ajaxPreload(page); if (!cached) { this.log('page.cache: preloading page #' + page); return true; } } } this.gc(); return false; }; ticker() { var to = this.ShouldIDoSomething() ? 300 : 1000; this.tickerTO = setTimeout(() => this.ticker(), to); }; populateList(Add) { var a = []; for (var i = 1; i <= (4 + Add); i++) { var np = this.wantPage + i; if (np <= this.comicsLastPage) { a.push(np); } } for (var i = 1; i <= (2 + Add); i++) { var np = this.wantPage - i; if (np >= 1) { a.push(np); } } return a; }; ajaxPreload(page) { // return cached var sc = this.SessCacheQuery(page); if (sc) { this.plData[page] = sc; // this.SessCachePut( page, data ); this.putCachedPreloadImage(page); return true; } this.plData[page] = true; $.ajax({ type: 'GET', dataType: 'text', cache: true, url: this.makeUrl(page), timeout: this.settings.AjaxRequestTimeoutMs, error: (...args) => { console.error({ _: `ajaxPreload(${page}) failed!`, args }); if (typeof (this.plData[page]) != 'undefined') { delete this.plData[page]; } }, success: (d) => { var parsed = this.parsePageString(d); if (!parsed) { console.error('Cannot parse data (see window._err_data)'); window._err_data = d; if (typeof (this.plData[page]) != 'undefined') { delete this.plData[page]; } } else { this.log('page.ajax: preloaded #' + page); this.putCached(page, parsed); this.ShouldIDoSomething(); } } }); return false; }; handlePrev() { // this.wantPage = this.comicsPage; this.wantPage--; this.ShouldIDoSomething(); }; handleNext() { this.wantPage++; this.ShouldIDoSomething(); }; uparrow_pressed = false; last_scroll = 0; documentKeydown(e) { var prevent = true; // if( e.which==13 && e.shiftKey ) { // Shift+Enter // } else if ((e.which == 38 || e.which == 40) && e.shiftKey) { // ^/v this.scroll_faster = !this.scroll_faster; } if (e.which == 37) { // < if (this.settings.contentMarginMover) { this.$contentMargin.hide(); } if (this.uparrow_pressed) { if (this.settings.contentMarginMover) { $(window).scrollTop(this.last_scroll); } this.$content.addClass('notitle'); } else { this.handlePrev(); } } else if (e.which == 39) { // > if (this.uparrow_pressed) { if (this.settings.contentMarginMover) { this.$contentMargin.show(); } this.$content.removeClass('notitle'); // $(window).scrollTop(0); this.doScroll(false, 0); } else { if (this.settings.contentMarginMover) { this.$contentMargin.hide(); } this.handleNext(); } } else if (e.which == 38) { // ^ this.uparrow_pressed = true; this.last_scroll = $(window).scrollTop(); if (this.settings.scrollFasterPx && this.scroll_faster) { // $(window).scrollTop( Math.max( 0, $(window).scrollTop()-this.settings.scrollFasterPx ) ); this.doScroll(true, -(e.ctrlKey ? this.settings.scrollFasterCtrlPx : this.settings.scrollFasterPx)); } else { prevent = false; } } else if (e.which == 40) { // v if (this.settings.scrollFasterPx && this.scroll_faster) { // $(window).scrollTop( $(window).scrollTop()+this.settings.scrollFasterPx ); this.doScroll(true, (e.ctrlKey ? this.settings.scrollFasterCtrlPx : this.settings.scrollFasterPx)); } else { prevent = false; } } else { prevent = false; } if (prevent) { e.preventDefault(); return false; } } documentKeyup(e) { if (e.which == 38) { // ^ this.uparrow_pressed = false; } } scroll_faster = false; /** @returns {boolean} */ init() { if (!this.preInit()) { return false; } if (this.initSucceeded) return true; if (this.initFailed || this.initGuard) { return false; } this.initFailed = true; this.initGuard = true; this.stage = 'init.1'; this.noPushState = false; // this.firstPushState = true; $(window) .off(`popstate.${this.constants.eventNamespace}`) .on(`popstate.${this.constants.eventNamespace}`, (e) => { var d = e?.originalEvent?.state || false; if (d && d.page) { this.wantPage = d.page; this.noPushState = true; this.ShouldIDoSomething(); } }); // TODO: add button and click event for this thing and script settings window.eval('window.z_SetPage=function(){ $(window).trigger(\'' + this.constants.localStorageKeyPrefix + ':setpage' + '\') };'); this.storePage = () => { // z_SetPage var p = parseInt(prompt('Set page to:', this.comicsPage), 10); if (isNaN(p) || p < 1 || p > this.comicsLastPage) { alert('Wrong page!\nMust be: 1 < page < ' + this.comicsLastPage); return; } if (!confirm('Warning! Really update page from ' + this.comicsPage + ' to ' + p + ' ?')) { return; } this.wantPage = p; this.ShouldIDoSomething(); }; $(window).on(this.constants.localStorageKeyPrefix + ':setpage', this.storePage); // this.wantPage = this.comicsPage; setTimeout(() => { if (this.wantPage == this.comicsPage) { this.swapPage(); } else { this.ShouldIDoSomething(); } }, 1); this.plData = Object.create(null); // this.wantPage = this.comicsPage; this.$content = $('.common-content'); this.$contentMargin = $('.common-content .view-container .view-article'); this.$contentSer = $('.common-content .reader-issue'); // this.$contentImage = $('.common-content .reader-issue img.issue'); this.$contentCounter = $('.common-content .button-goto span, .common-content .reader-issue-title > .number-without-name'); this.$contentBtnPrev = $('.common-content .button-previous'); this.$contentBtnNext = $('.common-content .button-next'); // custom stylesheets $('<style/>').attr('type', 'text/css').html('' + ' .common-header-background, .common-header {display:none !important;}' + ' .common-content > .serial-header, .common-content > .serial-reader-menu {display:none;}' + ' .common-content .reader-issue-title > a, .common-content #title > a {display:none;}' + ' .common-content .reader-issue-title, .common-content #title {padding:0 !important;min-height:0 !important;}' + ' .common-content.notitle .reader-issue-title, .common-content.notitle #title {display:none;}' // + ' .common-content img.issue {max-width: none !important; width: auto; }' + ' .page-margins {max-width: none !important;}' + ' .page-margins {background-color: transparent !important;}' + ' div.page-background {padding-top: 0 !important;}' + ' .common-footer {display:none;}' + ' body {grid-template-rows: 1fr !important;}' + ' .view-container {max-width:var(--max-width);margin-right:auto;margin-left:auto;background-color:#fff;box-shadow:0 3px 6px 1px rgba(0,0,0,0.05);}' + ' .common-content .reader-navigator .button-content>* {display:none;}' + ' .common-content .reader-navigator .button-first>* {display:none;}' + ' .common-content .reader-navigator .button-last>* {display:none;}' + ' .common-content .reader-navigator .button-random>* {display:none;}' ).appendTo('body'); this.$content.addClass('notitle'); if (this.settings.contentMarginMover) { this.$contentMargin .detach() .css({ position: 'absolute', left: '5px', top: '64px', border: '1px #000 solid', background: '#fff' }) .appendTo('body') .hide(); } this.gcAllowedSet(); this.tickerTO = setTimeout(() => this.ticker(), 10); this.uparrow_pressed = false; this.last_scroll = 0; this.scroll_faster = this.settings.scrollFasterDefault; $(document) .off(`keydown.${this.constants.eventNamespace}`) .on(`keydown.${this.constants.eventNamespace}`, (e) => this.documentKeydown(e)) .off(`keyup.${this.constants.eventNamespace}`) .on(`keyup.${this.constants.eventNamespace}`, (e) => this.documentKeyup(e)) ; this.$contentSer .off(`click.${this.constants.eventNamespace}`, 'a.reader-issue-next') .on(`click.${this.constants.eventNamespace}`, 'a.reader-issue-next', (e) => { e.preventDefault(); this.handleNext(); }); this.$contentSer .off(`click.${this.constants.eventNamespace}`, 'a.reader-issue-previous') .on(`click.${this.constants.eventNamespace}`, 'a.reader-issue-previous', (e) => { e.preventDefault(); this.handlePrev(); }); this.$contentBtnPrev.find('a') .attr('href', '#js-prev') .off(`click.${this.constants.eventNamespace}`) .on(`click.${this.constants.eventNamespace}`, (e) => { e.preventDefault(); this.handlePrev(); return false; }); this.$contentBtnNext.find('a') .attr('href', '#js-next') .off(`click.${this.constants.eventNamespace}`) .on(`click.${this.constants.eventNamespace}`, (e) => { e.preventDefault(); this.handleNext(); return false; }); $(window) .off(`beforeunload.${this.constants.eventNamespace}`) .on(`beforeunload.${this.constants.eventNamespace}`, (event) => { if (this.settings.blockUnload) { event.preventDefault(); return 'You shall not pass.'; } }); this.putCached(this.tmp_curr.Page, this.tmp_curr); this.tmp_curr = null; enableIntercept = true; this.stage = 'init.2'; this.log('init(finish)'); }; createRunInterface() { var current = this.init_comicsPage == this.init_wantPage; var label = 'Запуск читалки'; if (!current) { label = 'Продолжить чтение (стр. ' + this.init_wantPage + ')'; } var d = $('<div/>'); $('<button/>').html('×').css({ color: '#F00', fontWeight: 'bold', fontSize: '16px', position: 'absolute', left: '-10px', top: '-10px', width: '20px', height: '20px', padding: '1px' }).click((e) => { e.preventDefault(); d.remove(); }).appendTo(d); $('<button/>').html(label).css({ fontSize: '14px' }).click((e) => { e.preventDefault(); d.remove(); this.init(); // w.history.pushState({a:location.href}, "* Супер-читалка", location.href); }).appendTo(d); if (!current) { d.append('<div>Произойдет переход на стр. ' + this.init_wantPage + '</div>'); $('<button/>').html( 'Читать со страницы ' + this.init_comicsPage + '' ).css({ fontSize: '12px' }).click((e) => { e.preventDefault(); if (confirm('' + 'Вы собираетесь удалить закладку со страницы ' + this.init_wantPage + '\nи продолжить чтение со страницы ' + this.init_comicsPage + '\n\nВсё верно?' )) { this.wantPage = this.comicsPage = this.init_comicsPage; // this.store(); // location.href = this.makeUrl(this.comicsPage); d.remove(); this.init(); } }).appendTo(d); d.append('<div style="font-size:10px;">Закладка со стр. ' + this.init_wantPage + ' будет удалена.</div>'); } d.css({ borderRadius: '4px', position: 'fixed', // absolute zIndex: 1e6, right: '3px', top: '3px', padding: '4px', background: 'rgba(0,0,0,0.6)', color: '#fff', border: '2px rgba(0,0,0,0.9) solid' }).appendTo('body'); setTimeout(() => { var inner = $('header.common > div.inner'); if (!inner.length) return; if (d.offset().left < (inner.offset().left + inner.width() + 55)) { d.css({ top: Math.round(inner.height() + 3) + 'px' }); } }, 1); } } var enableIntercept = false; // disable built-in navigation with left/right arrow keys. let _addEventListener = document.addEventListener; document.addEventListener = function (en, fn, ...rest) { if (en == 'keydown') { let fnStr = fn?.toString(); if (fnStr.indexOf(`navElement.querySelector('a').getAttribute('href')`) !== -1) { let origFn = fn; fn = function (e) { if (enableIntercept) return; return origFn(e); }; } } return _addEventListener.call(this, en, fn, ...rest); }; class AcomicsViewerList2 { static makeUrlReference(id, comicsName) { return '/~' + comicsName + '/' + id; }; initGuard = false; init() { if (this.initGuard) return; this.initGuard = true; if (!/\/-[A-Za-z0-9_-]+\/list2/.test(location.pathname)) { return; } // $('div.contentSerialMargin > div.agrHolder >table.agr >tbody>tr:first-child>td.agrBody>div.numbers') var div = $('div.agrHolder'); if (!div.length) { return; } div.find('>table.agr').css({ marginBottom: '4px' }).each(function () { // var a = $(this).find('>tbody>tr:first-child>td.agrBody>h3>a'); // if(!a.length){return;} // a = a[0].pathname; // if( a.substr(0,2)!='/~' ){return;} // a=a.substr(2); var b = $(this).find('>tbody>tr:first-child>td.agrBody>div.numbers a').eq(0); if (!b.length) { return; } b = b[0].pathname; var m = /^\/~([^\/]+)\/(\d*)(?:\D|$)/.exec(b); if (!m) { return; } // parse page info var comicsName = m[1]; var comicsPage = parseInt(m[2], 10); if (m[2] == '' || isNaN(comicsPage) || comicsPage < 1) { return; } var k = this.constants.localStorageKeyPrefix + comicsName; // stored shit var stored = window.localStorage.getItem(k); if (stored) { stored = parseInt(stored, 10) || false; } else { stored = false; } if (!stored) { return; } if (stored == comicsPage) { $(this).css({ backgroundColor: '#ccc', outline: '1px #999 solid' }); } else if (stored < comicsPage) { $(this).css({ backgroundColor: '#cff', outline: '1px #9ff solid' }); var link = $('<a/>').addClass('agr-today').attr( 'href', AcomicsViewerList2.makeUrlReference(stored, comicsName) ).text( 'Продолжить чтение со страницы ' + stored + ' (из ' + comicsPage + ')' ).css({ fontSize: '18px' }); $(this).find('>tbody>tr:first-child>td.agrBody>div.numbers').prepend('<hr/>').prepend(link); } // this.comicsName = m[1]; // var comicsPage = parseInt(m[2],10); }); } } $(function () { (function () { // $('.common .inner td.nainmenu > nav > a[href$="/list2"]').css({backgroundColor:'#f00'}); var a = $('.common .inner td.logo > a'); var img = a.find('> img'); if (!img.length) { return; } var div = $('<div style="display:inline-block;overflow:hidden;width:55px;height:54px;height:100%;vertical-align:middle;padding-top:7px;" />'); img.detach(); div.append(img).appendTo(a); $('.common .inner td.logo').css('width', 'auto'); var x = $('.common .inner td.nainmenu > nav > a[href$="/list2"]'); x.css({ fontWeight: 'bold' }); var y = $('<div/>').html('обновления').css({ display: 'inline-block', background: '#99f', borderRadius: '4px', border: '1px #66f solid', position: 'absolute', marginTop: '-12px', marginLeft: '-48px', width: '64px', height: '14px', fontSize: '10px', fontFamily: 'sans-serif', textTransform: 'none', textAlign: 'center', fontWeight: 'normal' }); y.insertBefore(x.find('>span')); var live = $('.common .inner td.nainmenu > nav > a[href$="/live"]').contents().filter(function () { return this.nodeType === 3; }).eq(0); if (live.length) { live[0].textContent = 'Live'; } var Top = $('.common .inner td.nainmenu > nav > a[href$="//top.a-comics.ru/"]').contents().filter(function () { return this.nodeType === 3; }).eq(0); if (Top.length) { Top[0].textContent = 'ТОП'; } // window.lol_y = y; // window.lol_x = x; })(); let acomicsViewer = new AcomicsViewer(); if (!acomicsViewer.preInit()) { let list2 = new AcomicsViewerList2(); list2.init(); return; } acomicsViewer.createRunInterface(); }); })();