Enhanced viewer for acomics.ru

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('&times;').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();
  });

})();