// ==UserScript== // @name Manga OnlineViewer // @author Tago // @namespace https://github.com/TagoDR // @description Shows all pages at once in online view for these sites: Asura Scans, Flame Scans, Batoto, ComiCastle, DisasterScans, Dynasty-Scans, Leitor, LHTranslation, MangaDex, MangaFox, MangaHere, MangaFreak, mangahosted, MangaHub, MangaKakalot, MangaNelo, MangaNato, MangaPark, Mangareader, MangaSee, Manga4life, MangaTown, NineManga, PandaManga, RawDevart, ReadComicsOnline, ReadManga Today, Funmanga, MangaDoom, MangaInn, SenManga(Raw), TenManga, TuMangaOnline, UnionMangas, Manga33, FoOlSlide, Kireicake, Yuri-ism, Sense-Scans, Madara WordPress Plugin, MangaHaus, Isekai Scan, Comic Kiba, Zinmanga, mangatx, Toonily, Mngazuki, ReaperScans, JaiminisBox // @version 2022.07.04 // @license MIT // @grant GM_getValue // @grant GM_setValue // @grant GM_listValues // @grant GM_deleteValue // @grant GM_xmlhttpRequest // @connect * // @require https://cdnjs.cloudflare.com/ajax/libs/color-scheme/1.0.1/color-scheme.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jquery.imagesloaded/5.0.0/imagesloaded.pkgd.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.10.0/jszip.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/nprogress/0.2.0/nprogress.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/limonte-sweetalert2/11.4.19/sweetalert2.min.js // @include /https?:\/\/(www.)?(asurascans|flamescans).(com|org)\/.+/ // @include /https?:\/\/(www.)?bato.to\/chapter.*/ // @include /https?:\/\/(www.)?comicastle.org\/read\/.+\/[0-9]+.*/ // @include /https?:\/\/(www.)?disasterscans.com\/manga\/.+\/chapter-.+/ // @include /https?:\/\/(www.)?dynasty-scans.com\/chapters\/.+/ // @include /https?:\/\/(www.)?leitor.net\/manga\/.+\/.+\/.+/ // @include /https?:\/\/(www.)?lhtranslation.net\/read.+/ // @include /https?:\/\/(www.)?mangadex.org\/chapter\/.+(\/.+)?/ // @include /https?:\/\/(www.)?(fanfox.net|mangahere.cc)\/manga\/.+\/.+\// // @include /https?:\/\/.{3,4}?(mangafreak).net\/Read.+/ // @include /https?:\/\/(www.)?mangahosted.com\/manga\/.+\/.+/ // @include /https?:\/\/(www.)?(mangahub).io\/chapter\/.+\/.+/ // @include /https?:\/\/(www.)?((manganelo|mangakakalot).com\/chapter\/.+\/.+|(manganato|readmanganato).com\/manga-\w\w\d+\/chapter-\d+)/ // @include /https?:\/\/(www.)?mangapark.(com|me|org|net)\/(manga|chapter|comic)\/.+\/.+/ // @include /https?:\/\/(www.)?mangareader.to\/read\/.+\/.+\/.+/ // @include /https?:\/\/(www.)?(mangasee123|manga4life).com\/read-online\/.+/ // @include /https?:\/\/(www.|m.)?mangatown.com\/manga\/.+\/.+/ // @include /https?:\/\/(www.)?ninemanga.com\/chapter\/.+\/.+\.html/ // @include /https?:\/\/(www.)?pandamanga.xyz\/.+\/.+\/.+/ // @include /https?:\/\/(www.)?rawdevart.com\/comic\/.+\/.+\// // @include /https?:\/\/(www.)?readcomicsonline.ru\/comic\/.*\/\d*/ // @include /https?:\/\/(www.)?(funmanga|mngdoom|readmng|mangainn).(com|net)\/.+\/\d+/ // @include /https?:\/\/raw.senmanga.com\/.+\/.+\/?/ // @include /https?:\/\/(www.)?(tenmanga|gardenmanage).com\/(chapter|statuses)\/.+/ // @include /https?:\/\/(www.)?(tmofans|lectortmo|followmanga).com\/.+\/.+\/(paginated|cascade)/ // @include /https?:\/\/(www.)?unionleitor.top\/leitor\/.+\/.+/ // @include /https?:\/\/(www.)?(manga33).com\/manga\/.+/ // @include /^(?!.*jaiminisbox).*\/read\/.+/ // @include /https?:\/\/.+\/(manga|series)\/.+\/.+/ // @exclude /https?:\/\/(www.)?tsumino.com\/.+/ // @exclude /https?:\/\/(www.)?pururin.io\/.+/ // @downloadURL none // ==/UserScript== (function () { 'use strict'; // == AsuraScans and FlameScans ==================================================================== var asurasflamecans = { name: ['Asura Scans', 'Flame Scans'], url: /https?:\/\/(www.)?(asurascans|flamescans).(com|org)\/.+/, homepage: ['https://www.asurascans.com/', 'https://flamescans.org/'], language: ['English'], category: 'manga', waitEle: '#chapter option:nth-child(2)', run() { const chapter = document.querySelector('#chapter option:checked'); const images = [...document.querySelectorAll('#readerarea p img')]; return { title: document.querySelector('.entry-title')?.textContent?.trim(), series: document.querySelector('.allc a')?.getAttribute('href'), pages: images.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listImages: images.map((img) => img.getAttribute('src')), }; }, }; // == Batoto ======================================================================================= var batoto = { name: 'Batoto', url: /https?:\/\/(www.)?bato.to\/chapter.*/, homepage: 'http://bato.to/', language: ['English'], category: 'manga', run() { const images = [...document.querySelectorAll('.page-img')]; return { title: document.querySelector('.nav-title a')?.textContent?.trim(), series: document.querySelector('.nav-title a')?.getAttribute('href'), pages: images.length, prev: document.querySelector('.nav-prev a')?.getAttribute('href'), next: document.querySelector('.nav-next a')?.getAttribute('href'), listImages: images.map((img) => img.getAttribute('src')), }; }, }; // == ComiCastle =================================================================================== var comicastle = { name: 'ComiCastle', url: /https?:\/\/(www.)?comicastle.org\/read\/.+\/[0-9]+.*/, homepage: 'http://www.comicastle.org/', language: ['English'], category: 'comic', waitEle: '.form-control option:nth-child(1)', run() { const images = [...document.querySelectorAll('.form-control')[1].querySelectorAll('option')]; const chapter = document.querySelectorAll('.form-control')[0].querySelector('option:checked'); return { title: chapter?.textContent?.trim(), series: document.querySelector('.navbar-header a')?.getAttribute('href'), pages: images.length, prev: chapter?.previousElementSibling?.getAttribute('value'), next: chapter?.nextElementSibling?.getAttribute('value'), listImages: images.map((img) => img.getAttribute('alt')), }; }, }; // == DisasterScans ================================================================================ var disasterscans = { name: 'DisasterScans', url: /https?:\/\/(www.)?disasterscans.com\/manga\/.+\/chapter-.+/, homepage: 'https://disasterscans.com/', language: ['English'], category: 'manga', waitEle: 'select.single-chapter-select option', waitVar: 'mangaNav', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const chapter = document.querySelector('select.single-chapter-select option:checked'); return { title: document.querySelector('#chapter-heading')?.textContent?.trim(), series: W.mangaNav.mangaUrl, pages: W.chapter_preloaded_images.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listImages: W.chapter_preloaded_images, }; }, }; // == DynastyScans ================================================================================= var dysnatyscans = { name: 'Dynasty-Scans', url: /https?:\/\/(www.)?dynasty-scans.com\/chapters\/.+/, homepage: 'https://dynasty-scans.com/', language: ['English'], category: 'manga', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; return { title: document.querySelector('#chapter-title')?.textContent?.trim(), series: document.querySelector('#chapter-title a')?.getAttribute('href'), pages: W.pages.length, prev: document.querySelector('#prev_link')?.getAttribute('href'), next: document.querySelector('#next_link')?.getAttribute('href'), listImages: W.pages.map((x) => x.image), }; }, }; // == FoOlSlide ==================================================================================== var foolslide = { name: ['FoOlSlide', 'Kireicake', 'Yuri-ism', 'Sense-Scans'], url: /^(?!.*jaiminisbox).*\/read\/.+/, homepage: [ '#', 'https://reader.kireicake.com', 'https://www.yuri-ism.net', 'https://sensescans.com/', ], language: ['English'], obs: 'Any Site that uses FoOLSlide', category: 'manga', waitEle: 'img.open', run() { const chapter = [...document.querySelectorAll('.topbar_left .dropdown_parent:last-of-type li')]; const origin = chapter.findIndex((item) => { const url = item.querySelector('a')?.getAttribute('href'); if (url) return window.location.href.startsWith(url); return false; }); const pages = [...document.querySelectorAll('.topbar_right .dropdown li')]; const images = [...document.querySelectorAll('.inner img:not(.open)')]; const num = images.length > 1 ? images.length : pages.length; return { title: chapter.at(origin)?.querySelector('a')?.textContent?.trim(), series: document.querySelector('div.tbtitle div.text a')?.getAttribute('href'), pages: num, prev: chapter .at(origin + 1) ?.querySelector('a') ?.getAttribute('href'), next: chapter .at(origin - 1) ?.querySelector('a') ?.getAttribute('href'), listPages: images.length > 1 ? null : Array(num) .fill(0) .map((_, i) => `${window.location.href.replace(/\/\d+$/, '')}/${i + 1}`), listImages: images.length > 1 ? images.map((img) => img.getAttribute('src')) : null, img: 'img.open', }; }, }; // == Leitor ======================================================================================= var leitor = { name: 'Leitor', url: /https?:\/\/(www.)?leitor.net\/manga\/.+\/.+\/.+/, homepage: 'https://leitor.net/', language: ['Portuguese'], category: 'manga', async run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const url = `https://leitor.net/leitor/pages/${W.READER_ID_RELEASE}.json?key=${W.READER_TOKEN}`; const api = await fetch(url).then((res) => res.json()); const chapter = document.querySelector('.chapter-list .selected'); return { title: document.querySelector('title')?.textContent?.trim(), series: document.querySelector('.series-cover a')?.getAttribute('href'), pages: api.images.length, prev: chapter?.nextElementSibling?.querySelector('a')?.getAttribute('href'), next: chapter?.previousElementSibling?.querySelector('a')?.getAttribute('href'), listImages: api.images.map((img) => img.avif || img.legacy), }; }, }; // == LHTranslation ================================================================================ var lhtranslation = { name: 'LHTranslation', url: /https?:\/\/(www.)?lhtranslation.net\/read.+/, homepage: 'https://lhtranslation.net/', language: ['English'], category: 'manga', run() { const chapter = document.querySelector('.form-control option:checked'); const images = [...document.querySelectorAll('img.chapter-img')]; return { title: document.querySelector('.chapter-img.tieude font')?.textContent?.trim(), series: document.querySelector('.navbar-brand.manga-name')?.getAttribute('href'), pages: images.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listImages: images.map((img) => img.getAttribute('src')), }; }, }; // == Madara WordPress Plugin ====================================================================== // https://themeforest.net/item/madara-wordpress-theme-for-manga/20849828 var madarawp = { name: [ 'Madara WordPress Plugin', 'MangaHaus', 'Isekai Scan', 'Comic Kiba', 'Zinmanga', 'mangatx', 'Toonily', 'Mngazuki', 'ReaperScans', 'JaiminisBox', ], url: /https?:\/\/.+\/(manga|series)\/.+\/.+/, homepage: [ '#', 'https://manhuaus.com', 'https://isekaiscan.com/', 'https://comickiba.com/', 'https://zinmanga.com/', 'https://mangatx.com/', 'https://toonily.net/', 'https://mangazuki.me/', 'https://reaperscans.com/', 'https://jaiminisbox.net', ], language: ['English'], obs: 'Any Site that uses Madara Wordpress Plugin', category: 'manga', run() { const images = [ ...document.querySelectorAll('.wp-manga-chapter-img, .blocks-gallery-item img'), ]; return { title: document.querySelector('#chapter-heading')?.textContent?.trim(), series: document.querySelector('.breadcrumb li:nth-child(2) a')?.getAttribute('href'), pages: images.length, prev: document.querySelector('.prev_page')?.getAttribute('href'), next: document.querySelector('.next_page')?.getAttribute('href'), listImages: images.map((img) => img.getAttribute('src') || img.getAttribute('data-src') || img.getAttribute('data-full-url')), }; }, }; // == MangaDex ===================================================================================== var mangadex = { name: 'MangaDex', url: /https?:\/\/(www.)?mangadex.org\/chapter\/.+(\/.+)?/, homepage: 'https://mangadex.org/', language: ['English'], category: 'manga', waitEle: "a[href^='/chapter/']", async run() { const chapterId = window.location.pathname.match(/\/chapter\/([^/]+)(\/\d+)?/)[1]; const home = `https://api.mangadex.org/at-home/server/${chapterId}`; const server = await fetch(home).then((res) => res.json()); const images = server.chapter.data; const chapters = document.querySelectorAll("a[href^='/chapter/']"); return { title: document.querySelector('title')?.text.replace(' - MangaDex', ''), series: document.querySelector("a.text-primary[href^='/title/']")?.getAttribute('href'), pages: images.length, prev: chapters[1].getAttribute('href'), next: chapters[0].getAttribute('href'), listImages: images.map((img) => `${server.baseUrl}/data/${server.chapter.hash}/${img}`), }; }, }; // == MangaFox ===================================================================================== var mangafox = { name: ['MangaFox', 'MangaHere'], url: /https?:\/\/(www.)?(fanfox.net|mangahere.cc)\/manga\/.+\/.+\//, homepage: ['https://fanfox.net/', 'https://www.mangahere.cc/'], language: ['English'], category: 'manga', waitVar: 'chapterid', async run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const key = document.querySelector('#dm5_key')?.getAttribute('value'); const options = { method: 'GET', headers: { 'Content-Type': 'text/plain', }, }; const src = Array(W.imagecount) .fill(0) .map(async (_, i) => { const url = `chapterfun.ashx?cid=${W.chapterid || W.chapter_id}&page=${i}&key=${key}`; const api = await fetch(url, options).then((res) => res.text()); // eslint-disable-next-line no-eval (0, eval)(api); // @ts-ignore return d; }); const images = await Promise.all(src); return { title: document.querySelector('.reader-header-title div')?.textContent?.trim(), series: document.querySelector('.reader-header-title a')?.getAttribute('href'), pages: W.imagecount, prev: W.prechapterurl, next: W.nextchapterurl, listImages: images.map((img, i) => img[i === 0 ? 0 : 1]), }; }, }; // == MangaFreak =================================================================================== var mangafreak = { name: 'MangaFreak', url: /https?:\/\/.{3,4}?(mangafreak).net\/Read.+/, homepage: 'https://mangafreak.net/', language: ['English'], category: 'manga', run() { const chapter = document.querySelector('.chapter_list select option:checked'); const images = [...document.querySelectorAll('.mySlides img')]; return { title: document.querySelector('title')?.textContent?.trim(), series: document.querySelector('.title a')?.getAttribute('href'), pages: images.length, prev: chapter?.previousElementSibling?.getAttribute('value'), next: chapter?.nextElementSibling?.getAttribute('value'), listImages: images.map((img) => img.getAttribute('src')), }; }, }; // == mangahosted =================================================================================== var mangahosted = { name: 'mangahosted', url: /https?:\/\/(www.)?mangahosted.com\/manga\/.+\/.+/, homepage: 'https://mangahosted.com/', language: ['Portuguese'], category: 'manga', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const images = [...document.querySelectorAll('picture img')]; return { title: $('.breadcrumb li:eq(3)').text().trim(), series: $('.breadcrumb li:eq(2) a').attr('href'), pages: images.length, prev: W.$read_prev, next: W.$read_next, listImages: images.map((img) => img.getAttribute('src')), }; }, }; // == MangaHub ===================================================================================== var mangahub = { name: 'MangaHub', url: /https?:\/\/(www.)?(mangahub).io\/chapter\/.+\/.+/, homepage: 'https://mangahub.io/', language: ['English'], category: 'manga', waitEle: '#select-chapter', async run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const slug = W.CURRENT_MANGA_SLUG || window.location.pathname.split('/')[2]; const number = window.location.pathname.split('/')[3].replace('chapter-', ''); const data = { query: `{chapter(x:m01,slug:"${slug}",number:${number}){pages}}` }; const options = { method: 'POST', body: JSON.stringify(data), headers: { 'Content-Type': 'application/json', }, }; const api = await fetch('https://api.mghubcdn.com/graphql', options).then((res) => res.json()); const images = Object.values(JSON.parse(api?.data.chapter.pages.toString())); return { title: document.querySelector('#mangareader h3')?.textContent?.trim(), series: document.querySelector('#mangareader a')?.getAttribute('href'), pages: images.length, prev: document.querySelector('.previous a')?.getAttribute('href'), next: document.querySelector('.next a')?.getAttribute('href'), listImages: images.map((i) => `https://img.mghubcdn.com/file/imghub/${i}`), }; }, }; // == MangaKakalot ================================================================================= var mangakakalot = { name: ['MangaKakalot', 'MangaNelo', 'MangaNato'], url: /https?:\/\/(www.)?((manganelo|mangakakalot).com\/chapter\/.+\/.+|(manganato|readmanganato).com\/manga-\w\w\d+\/chapter-\d+)/, homepage: [ 'https://mangakakalot.com/page', 'http://www.manganelo.com/', 'http://www.manganato.com/', ], language: ['English'], category: 'manga', run() { const images = [...document.querySelectorAll('#vungdoc img, .container-chapter-reader img')]; return { title: document .querySelector('.info-top-chapter h2, .imageOptions-chapter-info-top h1') ?.textContent?.trim(), series: document.querySelectorAll('span a[title]').item(1).getAttribute('href'), pages: images.length, prev: document.querySelector('.navi-change-chapter-btn-prev, .next')?.getAttribute('href'), next: document.querySelector('.navi-change-chapter-btn-next, .back')?.getAttribute('href'), listImages: images.map((img) => img.getAttribute('src')), }; }, }; var mangapark = { name: 'MangaPark', url: /https?:\/\/(www.)?mangapark.(com|me|org|net)\/(manga|chapter|comic)\/.+\/.+/, homepage: 'https://mangapark.net/', language: ['English'], category: 'manga', waitVar: 'CryptoJS', run() { const pass = JSON.parse(CryptoJS.AES.decrypt(amWord, amPass).toString(CryptoJS.enc.Utf8)); return { title: `${amSub_name} - ${mpEpi_name}`, series: currSubUrl, pages: imgPathLis.length, prev: prevEpiUrl, next: nextEpiUrl, listImages: imgPathLis.map((i, index) => `${imgCdnHost + i}?${pass[index]}`), }; }, }; // == Mangareader ================================================================================== var mangareader = { name: 'Mangareader', url: /https?:\/\/(www.)?mangareader.to\/read\/.+\/.+\/.+/, homepage: 'https://mangareader.to', language: ['English'], category: 'manga', obs: 'Some galleries will not be usable', waitEle: '.ds-image, .iv-card', run() { const chapter = document.querySelector('.chapter-item.active'); const images = [ ...document.querySelectorAll('.ds-image:not(.shuffled)[data-url], .iv-card:not(.shuffled)[data-url]'), ]; return { title: document.querySelector('.hr-manga h2')?.textContent?.trim(), series: document.querySelector('.hr-manga')?.getAttribute('href'), pages: images.length, prev: chapter?.nextElementSibling?.querySelector('a')?.getAttribute('href'), next: chapter?.previousElementSibling?.querySelector('a')?.getAttribute('href'), listImages: images.map((img) => img.getAttribute('data-url')), }; }, }; // == MangaSee ===================================================================================== var mangasee = { name: ['MangaSee', 'Manga4life'], url: /https?:\/\/(www.)?(mangasee123|manga4life).com\/read-online\/.+/, homepage: ['https://mangasee123.com/', 'https://manga4life.com/'], language: ['English'], category: 'manga', waitAttr: ['.img-fluid', 'src'], run() { const src = document.querySelector('.img-fluid')?.getAttribute('src') || ''; const script = [...document.querySelectorAll('body script:not([src])')].at(-1)?.textContent; const textCurChapter = script?.match(/CurChapter = ({.+});/) || []; const CurChapter = JSON.parse(textCurChapter[1]); const textCHAPTERS = script?.match(/CHAPTERS = (\[\{.+}]);/) || []; const CHAPTERS = JSON.parse(textCHAPTERS[1]); const CurChapterIndex = CHAPTERS.findIndex((chap) => chap.Chapter === CurChapter.Chapter); function ChapterURLEncode(reference) { let ChapterString = CHAPTERS[CurChapterIndex + reference]; if (ChapterString === undefined) { return '#'; } ChapterString = ChapterString.Chapter; let Index = ''; const IndexString = ChapterString.substring(0, 1); if (IndexString !== '1') { Index = `-index-${IndexString}`; } const Chapter = parseInt(ChapterString.slice(1, -1), 10); let Odd = ''; const OddString = ChapterString[ChapterString.length - 1]; if (OddString !== '0') { Odd = `.${OddString}`; } return window.location.href.replace(/-chapter-.+/, `-chapter-${Chapter}${Odd}${Index}.html`); } return { title: document .querySelector('title') ?.textContent?.replace(/ Page .+/, '') .trim(), series: document.querySelector('.MainContainer a')?.getAttribute('href'), pages: parseInt(CurChapter.Page, 10), prev: ChapterURLEncode(-1), next: ChapterURLEncode(+1), listImages: Array(parseInt(CurChapter.Page, 10)) .fill(0) .map((_, i) => src.replace(/-\d\d\d.png/, `-${String(`000${i + 1}`).slice(-3)}.png`)), }; }, }; // == MangaTown ==================================================================================== var mangatown = { name: 'MangaTown', url: /https?:\/\/(www.|m.)?mangatown.com\/manga\/.+\/.+/, homepage: 'https://www.mangatown.com/', language: ['English'], category: 'manga', waitVar: 'chapter_id', async run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const key = document.querySelector('#dm5_key')?.getAttribute('value'); const options = { method: 'GET', headers: { 'Content-Type': 'text/plain', }, }; const src = Array(W.total_pages) .fill(0) .map(async (_, i) => { const url = `chapterfun.ashx?cid=${W.chapter_id}&page=${i}&key=${key}`; const api = await fetch(url, options).then((res) => res.text()); // eslint-disable-next-line no-eval (0, eval)(api); // @ts-ignore return d; }); const images = await Promise.all(src); const chapter = document.querySelector('#top_chapter_list option:checked'); return { title: document.querySelector('.title h1')?.textContent, series: W.series_url, pages: images.length, prev: chapter?.previousElementSibling?.getAttribute('value'), next: chapter?.nextElementSibling?.getAttribute('value'), listImages: images.map((img, i) => img[i === 0 ? 0 : 1]), }; }, }; // == NineManga ==================================================================================== var ninemanga = { name: 'NineManga', url: /https?:\/\/(www.)?ninemanga.com\/chapter\/.+\/.+\.html/, homepage: 'https://ninemanga.com/', language: ['English'], category: 'manga', run() { const chapter = document.querySelector('#chapter option:checked'); const pages = [...document.querySelector('#page').querySelectorAll('option')]; return { title: document.querySelector('.tip a')?.textContent?.trim(), series: document.querySelector('.subgiude > li:nth-child(2) > a')?.getAttribute('href'), pages: pages.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listPages: pages.map((item) => $(item).val()), img: '.manga_pic', }; }, }; // == PandaManga ================================================================================== var pandamanga = { name: 'PandaManga', url: /https?:\/\/(www.)?pandamanga.xyz\/.+\/.+\/.+/, homepage: 'https://www.pandamanga.com/', language: ['English'], category: 'manga', run() { const chapter = document.querySelector('.select-chapter option:checked'); const data = JSON.parse(document.getElementById('__NEXT_DATA__').textContent); const images = data.props.pageProps.mangaview.source .split(',') .filter((url) => url.length > 0); return { title: data.props.pageProps.mangaview.nameSeoChapter, series: document.querySelector('.allc a')?.getAttribute('href'), pages: images.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listImages: images, }; }, }; // == RawDevart =================================================================================== var rawdevart = { name: 'RawDevart', url: /https?:\/\/(www.)?rawdevart.com\/comic\/.+\/.+\//, homepage: 'https://rawdevart.com', language: ['Japanese'], category: 'manga', waitVar: 'rconfig', waitEle: '#chapter-list select', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const chapter = document.querySelector('#chapter-list option:checked'); const images = [...document.querySelectorAll('#img-container img')]; return { title: W.rconfig.chapterTitle, series: W.rconfig.prefix, pages: images.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listImages: images.map((item) => $(item).attr('data-src') || $(item).attr('src')), }; }, }; // == ReadComicsOnline ============================================================================= var readcomicsonline = { name: 'ReadComicsOnline', url: /https?:\/\/(www.)?readcomicsonline.ru\/comic\/.*\/\d*/, homepage: 'https://readcomicsonline.ru/', language: ['English'], category: 'comic', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const images = [...document.querySelectorAll('#all img')]; return { title: W.title.replace(/ - Page \d+/, ''), series: document.querySelector('div.pager-cnt a')?.getAttribute('href'), pages: W.pages.length, prev: W.prev_chapter, next: W.next_chapter, listImages: images.map((img) => img.getAttribute('data-src')), }; }, }; // == ReadManga.Today ============================================================================== var readmangatoday = { name: ['ReadManga Today', 'Funmanga', 'MangaDoom', 'MangaInn'], url: /https?:\/\/(www.)?(funmanga|mngdoom|readmng|mangainn).(com|net)\/.+\/\d+/, homepage: [ 'https://www.readmng.com/', 'https://funmanga.com/', 'https://mngdoom.com/', 'https://www.mangainn.net/', ], language: ['English'], category: 'manga', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; return { title: W.chapter_page_title, series: W.manga_url, pages: W.images.length, prev: W.prev_chapter_url, next: W.next_chapter_url, listImages: W.images.map((item) => item.url), }; }, }; // == SenManga ===================================================================================== var senmanga = { name: 'SenManga(Raw)', url: /https?:\/\/raw.senmanga.com\/.+\/.+\/?/, homepage: 'https://raw.senmanga.com/', language: ['Original'], category: 'manga', run() { const url = `/${window.location.pathname.split('/')[1]}/${window.location.pathname.split('/')[2]}`; const num = parseInt(document.querySelector('.page-list select option:last-child')?.getAttribute('value') || '0', 10); const chapter = [...document.querySelectorAll('.dropdown-chapter li')]; const origin = chapter.findIndex((item) => item.querySelector('a')?.getAttribute('href') === window.location.href); return { title: $('.title').text().trim(), series: document.querySelector('.breadcrumb li:nth-child(2) a')?.getAttribute('href'), pages: num, prev: chapter .at(origin + 1) ?.querySelector('a') ?.getAttribute('href'), next: chapter .at(origin - 1) ?.querySelector('a') ?.getAttribute('href'), listPages: Array(num) .fill(0) .map((_, i) => `${url}/${i + 1}/`), img: '.picture', }; }, }; // == TenManga ===================================================================================== var tenmanga = { name: 'TenManga', url: /https?:\/\/(www.)?(tenmanga|gardenmanage).com\/(chapter|statuses)\/.+/, homepage: 'https://www.tenmanga.com/', language: ['English'], category: 'manga', waitVar: '_pageCtrl', run() { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const chapter = document.querySelector('.mangaread-pagenav select option:checked'); // eslint-disable-next-line no-underscore-dangle const images = W._pageCtrl.options.all_imgs_url; return { title: document.querySelector('.title h1')?.textContent?.trim(), series: document.querySelector('.title a:nth-child(2)')?.getAttribute('href'), pages: images.length, prev: chapter?.nextElementSibling?.getAttribute('value'), next: chapter?.previousElementSibling?.getAttribute('value'), listImages: images, }; }, }; // == TMOFans ================================================================================== var tmofans = { name: 'TuMangaOnline', url: /https?:\/\/(www.)?(tmofans|lectortmo|followmanga).com\/.+\/.+\/(paginated|cascade)/, homepage: 'https://lectortmo.com/', language: ['Spanish'], category: 'manga', run() { const images = [...document.querySelectorAll('.img-container img')]; const pages = [ ...document.querySelectorAll('div.container:nth-child(4) select#viewer-pages-select option'), ]; const num = images.length > 1 ? images.length : pages.length; return { title: document.querySelector('title')?.textContent?.trim(), series: document.querySelector('a[title="Volver"]')?.getAttribute('href'), pages: num, prev: document.querySelector('.chapter-prev a')?.getAttribute('href'), next: document.querySelector('.chapter-next a')?.getAttribute('href'), listPages: images.length > 1 ? null : Array(num) .fill(0) .map((_, i) => `${window.location.href.replace(/\/\d+$/, '')}/${i + 1}`), listImages: images.length > 1 ? images.map((item) => $(item).attr('data-src')) : null, img: '#viewer-container img, .viewer-page', }; }, }; // == UnionMangas ================================================================================= var unionmangas = { name: 'UnionMangas', url: /https?:\/\/(www.)?unionleitor.top\/leitor\/.+\/.+/, homepage: 'https://unionleitor.top/', language: ['Portuguese'], category: 'manga', run() { const chapter = document.querySelector('#capitulo_trocar option:checked'); const images = [...document.querySelectorAll('.img-manga')]; return { title: document.querySelector('.titulo-leitura')?.textContent?.trim(), series: document.querySelector('.breadcrumbs a:nth-child(3)')?.getAttribute('href'), pages: images.length, prev: chapter?.previousElementSibling?.getAttribute('value'), next: chapter?.nextElementSibling?.getAttribute('value'), listImages: images.map((img) => img.getAttribute('src')), }; }, }; // == WPManga ====================================================================================== var wpmanga = { name: ['Manga33'], url: /https?:\/\/(www.)?(manga33).com\/manga\/.+/, homepage: ['https://manga33.com/'], language: ['English'], category: 'manga', run() { const images = [...document.querySelectorAll('.chapter-content img')]; return { title: document.querySelector('title')?.textContent?.trim(), series: document.querySelector('.navbar-brand')?.getAttribute('href'), pages: images.length, prev: document.querySelector('a.prev')?.getAttribute('href'), next: document.querySelector('a.next')?.getAttribute('href'), listImages: images.map((img) => img.getAttribute('src')), before() { if (window.location.pathname.match(/all.html$/)) return; if (window.location.pathname.match(/\d+-\d+.html$/)) window.location.pathname = window.location.pathname.replace(/-\d+.html$/, '-all.html'); }, }; }, }; /* eslint-disable no-unused-vars,@typescript-eslint/no-unused-vars */ const sites = [ asurasflamecans, batoto, comicastle, disasterscans, dysnatyscans, leitor, lhtranslation, mangadex, mangafox, mangafreak, mangahosted, mangahub, mangakakalot, mangapark, mangareader, mangasee, mangatown, ninemanga, pandamanga, rawdevart, readcomicsonline, readmangatoday, senmanga, tenmanga, tmofans, unionmangas, wpmanga, foolslide, madarawp, // Must be at the end because is a generic check ]; /* eslint-disable camelcase */ function logScript(...text) { // eslint-disable-next-line no-console console.log('MangaOnlineViewer: ', ...text); return text; } // Replacement function for GM_info allowing for debugging in console const getInfoGM = typeof GM_info !== 'undefined' ? GM_info : { scriptHandler: 'Console', script: { name: 'Debug', version: 'Testing', }, }; // Replacement function for GM_getValue allowing for debugging in console function getValueGM(name, defaultValue = null) { if (typeof GM_getValue !== 'undefined') { // logScript('Getting: ', name, ' = ', defaultValue); return GM_getValue(name, defaultValue); } logScript('Fake Getting: ', name, ' = ', defaultValue); return defaultValue; } // Replacement function for GM_setValue allowing for debugging in console function setValueGM(name, value) { try { GM_setValue(name, value); // logScript('Setting: ', name, ' = ', value as any); return value.toString(); } catch (e) { logScript('Fake Setting: ', name, ' = ', value); return String(value); } } // See https://stackoverflow.com/a/2401861/331508 for optional browser sniffing code. function getBrowser() { const ua = navigator.userAgent; let tem; let M = ua.match(/(opera|chrome|safari|firefox|msie|trident(?=\/))\/?\s*(\d+)/i) || []; if (/trident/i.test(M[1])) { tem = /\brv[ :]+(\d+)/g.exec(ua) || []; return `IE ${tem[1] || ''}`; } if (M[1] === 'Chrome') { tem = ua.match(/\b(OPR|Edge)\/(\d+)/); if (tem !== null) { return tem.slice(1).join(' ').replace('OPR', 'Opera'); } } M = M[2] ? [M[1], M[2]] : [navigator.appName, navigator.appVersion, '-?']; tem = ua.match(/version\/(\d+)/i); if (tem !== null) { M.splice(1, 1, tem[1]); } return M.join(' '); } // See // https://stackoverflow.com/questions/27487828/how-to-detect-if-a-userscript-is-installed-from-the-chrome-store function getEngine() { return getInfoGM.scriptHandler || 'Greasemonkey'; } const isMobile = window.matchMedia('screen and (max-width: 1024px)').matches; /** * Checks if a JavaScript value is empty * @example * isEmpty(null) // true * isEmpty(undefined) // true * isEmpty([]) // true * isEmpty({}) // true * isEmpty("") // true * isEmpty(false) // false * isEmpty(0) // false * isEmpty([{},{"0":false},{"":0},{"0":0}]) // false * isEmpty(42) // false * isEmpty([{"":1},{"0":1}]) // false * @param {any} value - item to test * @returns {boolean} true if empty, otherwise false */ function isEmpty(value) { return (value === null || // check for null typeof value === 'undefined' || value === undefined || // check for undefined (typeof value === 'string' && value === '') || // check for empty string (Array.isArray(value) && value.length === 0) || // check for empty array (typeof value === 'object' && Object.keys(value).length === 0)); } /** * Checks if value is nothing. Deep-checks arrays and objects * @example * isNothing(null) // true * isNothing(undefined) // true * isNothing([]) // true * isNothing({}) // true * isNothing("") // true * isNothing(false) // true * isNothing(0) // true * isNothing([{},{"0":false},{"":0},{"0":0}]) // true * isNothing(42) // false * isNothing([{"":1},{"0":1}]) // false * @param {any} value - item to test * @returns {boolean} true if nothing, otherwise false */ function isNothing(value) { const isEmptyObject = (a) => { if (!Array.isArray(a)) { // it's an Object, not an Array const hasNonempty = Object.keys(a).some((element) => !isNothing(a[element])); return hasNonempty ? false : isEmptyObject(Object.keys(a)); } // check if array is really not empty as JS thinks at least one element should be non-empty return !a.some((element) => !isNothing(element)); }; return ( // eslint-disable-next-line eqeqeq value == false || value === 0 || isEmpty(value) || (typeof value === 'object' && isEmptyObject(value))); } // language=CSS var cssStyles = ` html { font-size: 100%; } body { margin: 0; font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; font-size: 14px; line-height: 20px; color: #333; background-color: #FFF; padding: 0; } a { color: #08C; text-decoration: none; } img { height: auto; max-width: 100%; vertical-align: middle; border: 0 none; } #nprogress .bar { background: #29d; position: fixed; z-index: 1031; top: 0; left: 0; width: 100%; height: 4px; } #MangaOnlineViewer { width: 100%; height: 100%; padding-bottom: 40px; min-height: 1080px; } #MangaOnlineViewer #Chapter { text-align: center; display: block; } #MangaOnlineViewer #Chapter.WebComic .PageFunctions { } #MangaOnlineViewer #Chapter.WebComic .PageContent { margin-bottom: 0; margin-top: -23px; } #MangaOnlineViewer #Chapter.FluidLTR .MangaPage { width: auto; } #MangaOnlineViewer #Chapter.FluidRTL .MangaPage { width: auto; } #MangaOnlineViewer #Chapter.FluidLTR { direction: ltr; } #MangaOnlineViewer #Chapter.FluidRTL { direction: rtl; } #MangaOnlineViewer #ViewerControls { padding: 8px; position: fixed; top: 0; left: 405px; width: auto; z-index: 1000; transition: transform 0.3s ease-in, background-color 0.3s linear; display: none; } #MangaOnlineViewer #ViewerControls.visible { display: block; } #MangaOnlineViewer #ViewerControls .ControlLabel { display: list-item; list-style: none; } #MangaOnlineViewer #ViewerShortcuts { padding: 8px; position: fixed; top: 65px; left: 0; display: none; } #MangaOnlineViewer #ViewerShortcuts.visible { display: block; } #MangaOnlineViewer select { height: 20px; padding: 0; margin-bottom: 5px; } #MangaOnlineViewer .ControlButton { cursor: pointer; border-radius: 5px; border-width: 1px; } #MangaOnlineViewer .ControlButton:hover { opacity: 0.8; } #MangaOnlineViewer #ImageOptions { left: 0; position: absolute; top: 0; width: 405px; z-index:1000; } #MangaOnlineViewer #ImageOptions .panel { padding: 5px; position: inherit; } #MangaOnlineViewer #ImageOptions:hover { position: fixed; } #MangaOnlineViewer #ImageOptions.settingsOpen { position: fixed; } #MangaOnlineViewer #ImageOptions #menu { position: fixed; height: 64px; width: 200px; top: 0; } #MangaOnlineViewer #ImageOptions #Zoom { position: absolute; left: 18px; bottom: -65px; } #MangaOnlineViewer .MangaPage { width: 100%; display: inline-block; text-align: center; transform: translate3d(0, 0, 0); backface-visibility: hidden; perspective: 1000px; } #MangaOnlineViewer .PageContent { margin: 0 0 15px; text-align: center; display: inline-block; } #MangaOnlineViewer .PageContent.hide{ display: none; } #MangaOnlineViewer .PageContent .PageImg[src=""], #MangaOnlineViewer .PageContent .PageImg:not([src]) { width: 500px; height: 750px; display: inline-block; } #MangaOnlineViewer .fitWidthIfOversize .PageContent .PageImg { max-width: 100%; } #MangaOnlineViewer #gotoPage { width: 35px; } #MangaOnlineViewer #ThemeSelector { width: 110px; } #MangaOnlineViewer #Header { display: flex; justify-content: space-around; align-items: center; flex-flow: row nowrap; transition: transform 0.3s ease-in, background-color 0.3s linear; position: sticky; top: 0; left: 0; right: 0; background-color: inherit; z-index: 900; } #MangaOnlineViewer #Header.scroll-hide { transform: translateY(-100%); } #MangaOnlineViewer #Header.scroll-show{ transform: translateY(-1%); } #MangaOnlineViewer #Header.mouseOverMenu{ position: static; transform: none; } #MangaOnlineViewer #Header.scroll-end, #MangaOnlineViewer #Header.visible{ transform: translateY(-1%); position: sticky; } #MangaOnlineViewer #MangaTitle { padding: 2px; margin: 0; font-size: 1.2rem; font-weight: normal; } #MangaOnlineViewer #GlobalControls { flex-basis: 30%; } #MangaOnlineViewer #ChapterNavigation { display: flex; flex-flow: column nowrap; justify-content: center; align-items: center; flex-basis: 30%; } #MangaOnlineViewer .ChapterControl { } #MangaOnlineViewer .ChapterControl .NavigationControlButton { display: inline-flex; width: 80px; height: 25px; margin: 3px; justify-content: center; align-items: center; } #MangaOnlineViewer .ChapterControl .NavigationControlButton[href='#'], #MangaOnlineViewer .ChapterControl .NavigationControlButton[href=''] { visibility: hidden } #MangaOnlineViewer .ViewerTitle { text-align: center; min-height: 60px; max-width: 500px; display: flex; justify-content: center; align-items: center; flex-direction: column; padding: 5px; } #MangaOnlineViewer #Counters { } #MangaOnlineViewer .PageFunctions { font-family: monospace; display: flex; justify-content: flex-end; align-items: center; margin: 0; padding: 0; gap: 3px; position: relative; } #MangaOnlineViewer .PageFunctions > .PageIndex { min-width: 20px; text-align: center; display: inline-block; padding: 2px 10px } #MangaOnlineViewer .PageFunctions > .ControlButton { height: 16px; width: 16px; padding: 0 5px; margin: 0; border-width: 0; } #MangaOnlineViewer .PageFunctions .ControlButton { opacity: 0.5; } #MangaOnlineViewer .PageFunctions:hover .ControlButton { opacity: 1; } #MangaOnlineViewer .PageFunctions .ControlButton:hover { opacity: 0.9; } #MangaOnlineViewer.hideControls .PageFunctions { visibility: hidden; } #MangaOnlineViewer #NavigationCounters { margin-top: 5px; width: 100%; } #MangaOnlineViewer #Navigation { bottom: -180px; height: 190px; overflow-x: hidden; overflow-y: hidden; padding-bottom: 20px; position: fixed; white-space: nowrap; width: 100%; text-align: center; transition: transform 0.3s ease-in, background-color 0.3s linear; } #MangaOnlineViewer #Navigation #Thumbnails { overflow-x: auto; overflow-y: hidden; } #MangaOnlineViewer #Navigation:hover { transform: translateY(-180px); } #MangaOnlineViewer #Navigation.disabled { display: none; } #MangaOnlineViewer #Navigation.visible { transform: translateY(-180px); } #MangaOnlineViewer #Navigation .Thumbnail { display: inline-block; height: 150px; margin: 0 5px; position: relative; } #MangaOnlineViewer #Navigation .Thumbnail .ThumbnailIndex { display: block; opacity: 0.8; position: relative; top: -30px; width: 100%; } #MangaOnlineViewer #Navigation .Thumbnail .ThumbnailImg { align-content: center; cursor: pointer; display: inline-block; margin-bottom: -10px; margin-top: 10px; max-height: 150px; min-height: 150px; min-width: 80px; max-width: 160px; } #MangaOnlineViewer #Navigation .nav { transform: rotate(-90deg); } #MangaOnlineViewer #ImageOptions .menuOuterArrow { width: 0; height: 0; border-top: 10px solid transparent; border-bottom: 10px solid transparent; border-left: 10px solid blue; display: inline-block; position: absolute; bottom: 0; } #MangaOnlineViewer #ImageOptions .menuInnerArrow { width: 0; height: 0; border-top: 5px solid transparent; border-bottom: 5px solid transparent; border-left: 5px solid white; left: -10px; position: absolute; top: -5px; display: inline-block; } #MangaOnlineViewer #blob{ display: none; } /* Mobile styles*/ @media (max-width: 768px) { #MangaOnlineViewer #Header { flex-direction: column; } #MangaOnlineViewer .ViewerTitle { order: 1; flex-basis: 100%; } #MangaOnlineViewer #GlobalControls { order: 2; } #MangaOnlineViewer #ChapterNavigation { order: 3; } #MangaOnlineViewer #Navigation { display: none; } #MangaOnlineViewer .PageFunctions { padding: 0; } #MangaOnlineViewer .PageFunctions a:not(.Bookmark) { display: none; } #MangaOnlineViewer .PageFunctions a.Bookmark { opacity: 1; } #MangaOnlineViewer .PageFunctions span { right: 0; position: inherit; text-align: center; } #MangaOnlineViewer .PageContent { margin: 0; width: 100%; } #MangaOnlineViewer .PageContent img { width: 100% !important; } #MangaOnlineViewer .fitWidthIfOversize .PageContent img { max-width: 100%; } #MangaOnlineViewer #ImageOptions img:not(#settings) { display: none; } #MangaOnlineViewer #ViewerShortcuts { display: none !important; } #MangaOnlineViewer #ViewerControls { padding: 8px; position: fixed; top: 0; left: 45px; width: auto; transition: transform 0.3s ease-in, background-color 0.3s linear; display: none; } #MangaOnlineViewer #ViewerControls.visible { display: block; } #MangaOnlineViewer #ViewerControls .DefaultZoom, #MangaOnlineViewer #ViewerControls .viewMode, #MangaOnlineViewer #ViewerControls .fitIfOversize, #MangaOnlineViewer #ViewerControls .showThumbnails, #MangaOnlineViewer #ViewerControls .lazyLoadImages, #MangaOnlineViewer #ViewerControls .downloadZip { display: none; } #MangaOnlineViewer #ImageOptions #menu { display: none; } #MangaOnlineViewer #ImageOptions #Zoom { display: none; } #MangaOnlineViewer .ViewerTitle { height: auto; } #MangaOnlineViewer .ChapterControl { margin: 10px; display: block; text-align: center; } #MangaOnlineViewer .ChapterControl .download { display: none; } #MangaOnlineViewer #Counters { position: inherit; text-align: center; margin: 10px; } #MangaOnlineViewer #Chapter { margin: 5px auto 0; } } `; const externalScripts = [ '', '', '', '', '', '', ]; externalScripts.map((script) => { const find = script.match(/src="(.+?)"/); return find ? find[1] : ''; }); const externalCSS = [ '', '', '', '', ]; externalCSS.map((script) => { const find = script.match(/href="(.+?)"/); return find ? find[1] : ''; }); // Configuration const settings$1 = { theme: getValueGM('Theme', 'Light'), customTheme: getValueGM('CustomTheme', '#3d0099'), customThemeBody: getValueGM('CustomThemeBody', '#000000'), customThemeText: getValueGM('CustomThemeText', '#ffffff'), customThemeLines: getValueGM('CustomThemeLines', '#666666'), customThemePanel: getValueGM('CustomThemePanel', '#333333'), customThemeButton: getValueGM('CustomThemeButton', '#282828'), fitWidthIfOversize: getValueGM('FitWidthIfOversize', true), showThumbnails: getValueGM('ShowThumbnails', true), downloadZip: getValueGM('DownloadZip', false), throttlePageLoad: getValueGM('Timer', 1000), zoom: getValueGM('Zoom', 100), zoomStep: getValueGM('ZoomStep', 25), loadMode: getValueGM('LoadMode', 'wait'), viewMode: getValueGM('ViewMode', ''), bookmarks: JSON.parse(getValueGM('Bookmarks', '[]')), lazyLoadImages: getValueGM('LazyLoadImages', false), lazyStart: getValueGM('LazyStart', 50), hidePageControls: getValueGM('HidePageControls', false), mouseOverMenu: getValueGM('MouseOverMenu', false), }; // Force Settings for mobile if (isMobile) { settings$1.lazyLoadImages = true; settings$1.lazyStart = parseInt(getValueGM('LazyStart', 5), 10); settings$1.fitWidthIfOversize = true; settings$1.showThumbnails = false; settings$1.viewMode = ''; } // Clear old Bookmarks const bookmarkTimeLimit = 1000 * 60 * 60 * 24 * 30 * 12; // year settings$1.bookmarks = settings$1.bookmarks.filter((el) => Date.now() - el.date < bookmarkTimeLimit); setValueGM('Bookmarks', JSON.stringify(settings$1.bookmarks)); // Icons in Base64 format // Source: http://www.iconarchive.com/show/farm-fresh-icons-by-fatcow.html // Source: http://www.iconarchive.com/show/oxygen-icons-by-oxygen-icons.org.7.html // Source: http://www.iconarchive.com/show/ivista-2-icons-by-gakuseisean.html const icon = { enlarge: '', reduce: '%3D%3D', restore: '%3D', fitWidth: '', fitHeight: '', reload: '%3D%3D', zoomIn: '%3D%3D', zoomOut: '', zoomRestore: '%3D', zoomWidth: '%3D', hide: '%3D', settings: '%3D', menu: '%3D', webComic: '%3D', bookmark: '', pictureRight: '', pictureDown: '', pictureLeft: '', controls: '', }; const scheme = new ColorScheme().scheme('mono').variation('default'); // Add custom Themes to the page function addTheme(theme) { return ``; } function addCustomTheme(color) { const bg = scheme.from_hex(color.replace('#', '')).colors(); return (addTheme(['Custom_Dark', '#000000', `#${bg[2]}`, `#${bg[3]}`, `#${bg[0]}`, `#${bg[1]}`]) + addTheme(['Custom_Light', '#eeeeec', `#${bg[3]}`, `#${bg[2]}`, `#${bg[0]}`, `#${bg[1]}`])); } function addFullCustomTheme(body, text, lines, panel, buttons) { return addTheme(['Full_Custom', body, text, lines, panel, buttons]); } function loadThemes() { const bg = scheme.from_hex(settings$1.customTheme.replace('#', '')).colors(); return [ // 1-body 2-text 3-lines 4-imageOptions 5-buttons ['Dark', '#000000', '#ffffff', '#666666', '#333333', '#282828'], ['Light', '#eeeeec', '#2e3436', '#888a85', '#babdb6', '#c8cec2'], ['Clear', '#ffffff', '#2e3436', '#888a85', '#eeeeec', '#d3d7cf'], ['Dark_Blue', '#000000', '#91a0b0', '#586980', '#3e4b5b', '#222c3b'], ['Tango_Blue', '#000000', '#82a0bf', '#3d669b', '#304c77', '#102747'], ['Lime', '#000000', '#8abd59', '#608d34', '#38531f', '#233413'], ['Plum', '#000000', '#ad7fa8', '#75507b', '#49324d', '#311b37'], ['Light_Plum', '#eeeeec', '#5c3566', '#9b71a2', '#ad7fa8', '#d2b8ce'], ['Earthy', '#000000', '#ffffff', '#693d3d', '#46211a', '#683327'], ['Cool_Blues', '#000000', '#c4dfe6', '#66a5ad', '#07575b', '#003b46'], ['Custom_Dark', '#000000', `#${bg[2]}`, `#${bg[3]}`, `#${bg[0]}`, `#${bg[1]}`], ['Custom_Light', '#eeeeec', `#${bg[3]}`, `#${bg[2]}`, `#${bg[0]}`, `#${bg[1]}`], [ 'Full_Custom', settings$1.customThemeBody, settings$1.customThemeText, settings$1.customThemeLines, settings$1.customThemePanel, settings$1.customThemeButton, ], ]; } const themes = loadThemes(); const themesSelector = themes.map((theme) => ``); const themesCSS = themes.map(addTheme).join(''); // Inject CSS for this script const readerCSS = ``; function head(manga) { return ` ${manga.title} ${externalScripts.join('\n')} ${externalCSS.join('\n')} ${readerCSS} ${themesCSS} `; } // Add Pages Place-holders const listPages = (times) => Array(times) .fill(null) .map((_, index) => `
${index + 1}
PageImg${index + 1}
`); const imageOptions = `
Enlarge Restore Reduce Fit Width Fit Height Web Comic Mode Left to Right Mode Vertical Mode Right to Left Mode Toggle Page Controls settings
Zoom: ${settings$1.zoom} %
`; const controls$1 = `
Theme:
-Base:
-Body:
-Text:
-Lines:
-Painels:
-Buttons:
Default Load Mode:
Pages/Second:
Default Zoom:
Zoom Change Step (between 5 and 50):
${settings$1.zoomStep}
Default View Mode:
Fit Width if Oversize:
Show Thumbnails:
Lazy Load Images:
Lazy Start From Page (between 5 and 100):
${settings$1.lazyStart}
Download Images as Zip Automatically:
Always Hide Page Controls:
Toggle Sticky Header / MouseOverMenu:
`; const keybindings = `
Numpad 5//: Open Settings
Numpad +/=: Global Zoom in pages (enlarge)
Numpad -/-: Global Zoom out pages (reduce)
Numpad //9: Global Restore pages to original
Numpad */0: Global Fit window width
V: Vertical Mode
C: WebComic Mode
N: Right to Left Mode
B: Left to Right Mode
/D/Numpad 6/. : Next Chapter
/A/Numpad 4/, : Previous Chapter
/W/Numpad 8: Scroll Up
/S/Numpad 2: Scroll Down
`; const listThumbnails = (times) => Array(times) .fill(null) .map((_, index) => `
ThumbnailImg${index + 1} ${index + 1}
`); const listOptions = (times) => Array(times) .fill(null) .map((_, index) => ``); const body = (manga, begin = 0) => `
${listPages(manga.pages).slice(begin).join('')}
`; const cache = { zip: new JSZip(), downloadFiles: 0, Data: {}, }; const getExtension = (mimeType) => ((/image\/(?jpe?g|png|webp)/.exec(mimeType) || {}).groups || {}).ext || '' || 'png'; const getFilename = (name, index, total, ext) => `${name}${(index + 1).toString().padStart(Math.floor(Math.log10(total)) + 1, '0')}.${ext.replace('jpeg', 'jpg')}`; // Generate Zip File for download function generateZip() { // Source: // http://stackoverflow.com/questions/8778863/downloading-an-image-using-xmlhttprequest-in-a-userscript/8781262#8781262 if (cache.downloadFiles === 0) { const filenameRegex = /^(?.*?)(?\d+)\.(?\w+)$/; const images = [...document.querySelectorAll('.PageImg')]; const filenames = (() => { const result = []; for (let i = 0; i < images.length; i += 1) { const $img = images[i]; const filename = $img.getAttribute('src')?.split(/[?#]/)[0].split('/').pop() ?? ''; const match = filenameRegex.exec(filename); if (!match || !match.groups) break; const fixedFilename = getFilename(match.groups.name, parseInt(match.groups.index, 10), images.length, match.groups?.ext); if (result.length > 0 && fixedFilename <= result[result.length - 1]) break; result.push(fixedFilename); } if (result.length < images.length) return []; return result; })(); images.forEach((img, index) => { const src = img.getAttribute('src') ?? ''; const base64 = /^data:(?image\/\w+);base64,+(?.+)/.exec(src); if (base64 && base64.groups) { const filename = getFilename('Page ', index, images.length, getExtension(base64.groups?.mimeType)); cache.zip.file(filename, base64.groups.data, { base64: true, createFolders: true, }); logScript(`${filename} Added to Zip from Base64 Image, From: ${src}`); cache.downloadFiles += 1; } else { try { GM_xmlhttpRequest({ method: 'GET', url: src, overrideMimeType: 'text/plain; charset=x-user-defined', responseType: 'blob', onload(request) { const filename = filenames[index] || getFilename('Page ', index, images.length, getExtension(request.response.type)); cache.zip.file(filename, request.response, { base64: true, createFolders: true, compression: 'DEFLATE', }); logScript(`${filename} Added to Zip as Base64 Image, From: ${src}, Data:`, request.response); cache.downloadFiles += 1; }, }); } catch (e) { logScript(e); } } }); } const total = document.querySelectorAll('.PageImg').length; if (cache.downloadFiles < total) { logScript(`Waiting for Files to Download ${cache.downloadFiles} of ${total}`); setTimeout(generateZip, 2000); } else { try { logScript('Generating Zip'); cache.zip .generateAsync({ type: 'blob', }) .then((content) => { logScript('Download Ready'); const zipName = `${document.querySelector('#MangaTitle')?.textContent?.trim()}.zip`; FileSaver.saveAs(content, zipName); }); } catch (e) { logScript(e); } } } function isImagesManga(manga) { return 'listImages' in manga && !isNothing(manga.listImages); } function isPagesManga(manga) { return 'listPages' in manga && !isNothing(manga.listPages); } function isBruteforceManga(manga) { return 'bruteForce' in manga && !isNothing(manga.bruteForce); } var Language; (function (Language) { Language["ENGLISH"] = "English"; Language["SPANISSH"] = "Spanish"; Language["PORTUGUESE"] = "Portuguese"; Language["JAPANESE"] = "Japanese"; })(Language || (Language = {})); var Category; (function (Category) { Category["MANGA"] = "manga"; Category["COMIC"] = "comic"; Category["HENTAI"] = "hentai"; })(Category || (Category = {})); function fetchText(url, format) { return new Promise((resolve) => { logScript('Fetching page: ', url); fetch(url) .then((response) => // When the page is loaded convert it to text response.text()) .then((html) => { // Initialize the DOM parser const parser = new DOMParser(); // Parse the text const doc = parser.parseFromString(html, format); // You can now even select part of that html as you would in the regular DOM // Example: // var docArticle = doc.querySelector('article').innerHTML; // console.log(doc); resolve(doc); }) .catch((err) => { logScript('Failed to fetch page: ', err); }); }); } function fetchHtml(url) { return fetchText(url, 'text/html'); } function getElementAttribute(url, selector, attribute) { return fetchHtml(url).then((doc) => doc.querySelector(selector)?.getAttribute(attribute)); } /** * Settings the lazy load will obey */ const settings = { threshold: 2000, throttle: 500, lazyAttribute: 'data-src', targetAttribute: 'src', }; /** * List of elements that will be lazy loaded */ let listElements = []; /** * Check if the image ins nearing the viewport, so it needs to load. * @param value */ function filterInView(value) { const { element } = value; if (!element.offsetParent) return false; const ele = element.offsetParent; const top = ele.offsetTop + element.height >= window.scrollY - settings.threshold; const bottom = ele.offsetTop <= window.scrollY + window.innerHeight + settings.threshold; return top && bottom; } /** * Execute the loading of the image * @param item */ function showElement(item) { const value = item.element.getAttribute(settings.lazyAttribute); if (value) item.element.setAttribute(settings.targetAttribute, value); item.callback(item.element); } /** * Lookup images that should be loaded, and update the current list */ function executeCheck() { const inView = listElements.filter(filterInView); listElements = listElements.filter((item) => !filterInView(item)); inView.forEach(showElement); } /** * Throttle controller */ let wait; /** * Function responsible for observing the screen move/change */ function observerEvent() { if (listElements.length === 0) { window.removeEventListener('scroll', observerEvent); window.removeEventListener('touchmove', observerEvent); window.removeEventListener('resize', observerEvent); // console.info('All items lazy loaded'); return; } if (wait) { return; } executeCheck(); wait = setTimeout(() => { wait = undefined; }, settings.throttle); } /** * Simple lazy loading for images. * Add an image element to a list, wait for it to be close to appearing on screen then load its 'src' from 'data-src' * then call a callback function. * @param element * @param callback */ function lazyLoad(element, callback) { if (listElements.length === 0) { // console.info('Initializing lazy load'); window.addEventListener('scroll', observerEvent, { passive: true, }); window.addEventListener('touchmove', observerEvent, { passive: true, }); window.addEventListener('resize', observerEvent, { passive: true, }); } listElements.push({ element, callback }); } // After pages load apply default Zoom function applyZoom(pages = '.PageContent img', zoom = settings$1.zoom) { const pg = [...document.querySelectorAll(pages)]; pg.forEach((value) => { const img = value; img.removeAttribute('width'); img.removeAttribute('height'); img.removeAttribute('style'); if (zoom === 1000) { img.style.width = `${window.innerWidth}px`; } else if (zoom === -1000) { const nav = document.querySelector('#Navigation')?.classList.contains('disabled'); const chap = document.querySelector('#Chapter')?.classList.contains('WebComic'); const nextHeight = window.innerHeight + (nav ? 0 : -34) + (chap ? 0 : -35); img.style.height = `${nextHeight}px`; } else { img.style.width = `${img.naturalWidth * (zoom / 100)}px`; } }); } // Force reload the image function reloadImage(img) { const src = img.getAttribute('src'); if (src) { img.removeAttribute('src'); setTimeout(() => { img.setAttribute('src', src); }, 500); } } function onImagesDone() { logScript('Images Loading Complete'); if (!settings$1.lazyLoadImages) { document.querySelector('.download')?.setAttribute('href', '#download'); logScript('Download Available'); if (settings$1.downloadZip) { document.querySelector('#blob')?.dispatchEvent(new Event('click')); } } } function updateProgress() { const total = document.querySelectorAll('.PageContent .PageImg').length; const loaded = document.querySelectorAll('.PageContent .PageImg.imgLoaded').length; const percentage = Math.floor((loaded / total) * 100); const title = document.querySelector('title'); if (title) { title.innerHTML = `(${percentage}%) ${document.querySelector('#MangaTitle')?.textContent}`; } document.querySelectorAll('#Counters i, #NavigationCounters i').forEach((ele) => { ele.textContent = loaded.toString(); }); NProgress.configure({ showSpinner: false, }).set(loaded / total); logScript(`Progress: ${percentage}%`); if (loaded === total) onImagesDone(); } // change class if the image is loaded or broken function onImagesProgress(instance, image) { if (image) { if (image.isLoaded) { image.img.classList.add('imgLoaded'); image.img.classList.remove('imgBroken'); image.img.getAttribute('id'); const thumbId = image.img.getAttribute('id').replace('PageImg', 'ThumbnailImg'); const thumb = document.getElementById(thumbId); if (thumb) { thumb.onload = () => applyZoom(`#${image.img.getAttribute('id')}`); thumb.setAttribute('src', image.img.getAttribute('src')); } } else { image.img.classList.add('imgBroken'); reloadImage(image.img); const imgLoad = imagesLoaded(image.img.parentElement); imgLoad.on('progress', onImagesProgress); } } updateProgress(); } // Corrects urls function normalizeUrl(url = '') { let uri = url.trim(); if (uri.startsWith('//')) { uri = `https:${uri}`; } return uri; } // Adds an image to the place-holder div function addImg(index, imageSrc) { const src = normalizeUrl(imageSrc); const img = document.querySelector(`#PageImg${index}`); if (img) { if (!settings$1.lazyLoadImages || index < settings$1.lazyStart) { img.setAttribute('src', src); img.setAttribute('src', src); const imgLoad = imagesLoaded(img.parentElement); imgLoad.on('progress', onImagesProgress); logScript('Loaded Image:', index, 'Source:', src); } else { img.setAttribute('data-src', src); lazyLoad(img, () => { const imgLoad = imagesLoaded(img.parentElement); imgLoad.on('progress', onImagesProgress); logScript('Lazy Image: ', index, ' Source: ', img.getAttribute('src')); }); } } } function findPage(manga, index, pageUrl, lazy) { return async () => { const src = await getElementAttribute(pageUrl, manga.img, manga.lazyAttr ?? 'src'); const img = document.querySelector(`#PageImg${index}`); if (src && img) { img.setAttribute('src', src); img.style.width = 'auto'; const imgLoad = imagesLoaded(img.parentElement); imgLoad.on('progress', onImagesProgress); logScript(`${lazy && 'Lazy '}Page: `, index, ' Source: ', img.getAttribute('src')); } }; } // Adds a page to the place-holder div async function addPage(manga, index, pageUrl) { const img = document.querySelector(`#PageImg${index}`); if (img) { if (!settings$1.lazyLoadImages || index < settings$1.lazyStart) { await findPage(manga, index, pageUrl, false)(); } else { img.setAttribute('data-src', ''); lazyLoad(img, findPage(manga, index, pageUrl, false)); } } } // daley the use of an url/src function delayAdd(src, wait = settings$1.throttlePageLoad) { return new Promise((resolve) => { setTimeout(() => { resolve(src); }, wait); }); } // use a list of pages to fill the viewer function loadMangaPages(begin, manga) { return manga.listPages?.map((url, index) => index >= begin ? delayAdd(url, (manga.timer || settings$1.throttlePageLoad) * (index - begin)).then((response) => addPage(manga, index + 1, response)) : null); } // use a list of images to fill the viewer function loadMangaImages(begin, manga) { return manga.listImages?.map((src, index) => index >= begin ? delayAdd(src, (manga.timer || settings$1.throttlePageLoad) * (index - begin)).then((response) => addImg(index + 1, response)) : null); } // Entry point for loading hte Manga pages function loadManga(manga, begin = 1) { settings$1.lazyLoadImages = manga.lazy || settings$1.lazyLoadImages; logScript('Loading Images'); logScript(`Intervals: ${manga.timer || settings$1.throttlePageLoad || 'Default(1000)'}`); logScript(`Lazy: ${settings$1.lazyLoadImages}`); if (settings$1.lazyLoadImages) { logScript('Download may not work with Lazy Loading Images'); } if (isImagesManga(manga)) { logScript('Method: Images:', manga.listImages); loadMangaImages(begin - 1, manga); } else if (isPagesManga(manga)) { logScript('Method: Pages:', manga.listPages); loadMangaPages(begin - 1, manga); } else if (isBruteforceManga(manga)) { logScript('Method: Brute Force'); manga.bruteForce({ begin, addImg, loadImages: (list) => loadMangaImages(begin - 1, { ...manga, listImages: list }), loadPages: (list, img, lazyAttr) => loadMangaPages(begin - 1, { ...manga, listPages: list, img, lazyAttr, }), wait: settings$1.throttlePageLoad, }); } } // Goto Page and Thumbnails function scrollToElement(ele) { window.scroll(0, ele?.offsetTop || 0); } // Clean key press configurations and set some when specified function setKeyDownEvents() { try { // $(document).off('keyup'); // $(document).off('keydown'); // $(document).off('keypress'); // $(document).off('onload'); // $(window).off('keyup'); // $(window).off('keydown'); // $(window).off('keypress'); // $(window).off('onload'); document.onkeydown = null; document.onkeypress = null; window.onkeydown = null; window.onkeypress = null; window.onload = null; document.body.onload = null; } catch (e) { logScript(`Keybinds error: ${e}`); } function processKey(e) { const a = e.code; console.log('Keyboard:', a, ' Event:', e); const usedKeys = [ 'KeyW', 'Numpad8', 'KeyS', 'Numpad2', 'ArrowRight', 'Period', 'KeyD', 'Numpad6', 'ArrowLeft', 'Comma', 'KeyA', 'Numpad4', 'Equal', 'NumpadAdd', 'KeyE', 'Minus', 'NumpadSubtract', 'KeyQ', 'Digit9', 'NumpadDivide', 'KeyR', 'Digit0', 'NumpadMultiply', 'KeyF', 'Slash', 'Numpad5', 'KeyX', 'KeyC', 'KeyV', 'KeyB', 'KeyN', ]; if (!e.ctrlKey && !e.altKey && !e.shiftKey && !e.metaKey && usedKeys.some((i) => i === a)) { e.preventDefault(); e.stopPropagation(); e.stopImmediatePropagation(); switch (a) { case 'ArrowUp': case 'KeyW': case 'Numpad8': if (settings$1.zoom === -1000) { const next = [...document.querySelectorAll('.MangaPage')].find((element) => element.offsetTop - window.scrollY > 10); scrollToElement(next?.previousElementSibling); } else { window.scrollBy({ top: -window.innerHeight / 2, behavior: 'smooth', }); } break; case 'ArrowDown': case 'KeyS': case 'Numpad2': if (settings$1.zoom === -1000) { const next = [...document.querySelectorAll('.MangaPage')].find((element) => element.offsetTop - window.scrollY > 10); scrollToElement(next); } else { window.scrollBy({ top: window.innerHeight / 2, behavior: 'smooth', }); } break; case 'ArrowRight': case 'Period': case 'KeyD': case 'Numpad6': document.querySelector('#next')?.dispatchEvent(new Event('click')); break; case 'ArrowLeft': case 'Comma': case 'KeyA': case 'Numpad4': document.querySelector('#prev')?.dispatchEvent(new Event('click')); break; case 'Equal': case 'NumpadAdd': case 'KeyE': document.querySelector('#enlarge')?.dispatchEvent(new Event('click')); break; case 'Minus': case 'NumpadSubtract': case 'KeyQ': document.querySelector('#reduce')?.dispatchEvent(new Event('click')); break; case 'Digit9': case 'NumpadDivide': case 'KeyR': document.querySelector('#restore')?.dispatchEvent(new Event('click')); break; case 'Digit0': case 'NumpadMultiply': case 'KeyF': document.querySelector('#fitWidth')?.dispatchEvent(new Event('click')); break; case 'Slash': case 'Numpad5': case 'KeyX': document.querySelector('#settings')?.dispatchEvent(new Event('click')); break; case 'KeyC': document.querySelector('#webComic')?.dispatchEvent(new Event('click')); break; case 'KeyV': document.querySelector('#verticalMode')?.dispatchEvent(new Event('click')); break; case 'KeyN': document.querySelector('#rtlMode')?.dispatchEvent(new Event('click')); break; case 'KeyB': document.querySelector('#ltrMode')?.dispatchEvent(new Event('click')); break; } return false; } return true; } document.addEventListener('keydown', processKey); } function updateZoomPercent(percent = settings$1.zoom) { const zoom = document.querySelector('#ZoomPercent'); if (zoom) { zoom.textContent = percent.toString(); } } // Controls for the extra features added to the sites function controls() { // Size Controls // Global Zoom In Button document.querySelector('#enlarge')?.addEventListener('click', () => { settings$1.zoom += settings$1.zoomStep; updateZoomPercent(); applyZoom(); }); // Global Zoom Out Button document.querySelector('#reduce')?.addEventListener('click', () => { settings$1.zoom -= settings$1.zoomStep; updateZoomPercent(); applyZoom(); }); // Global Zoomm Restore Button document.querySelector('#restore')?.addEventListener('click', () => { settings$1.zoom = 100; updateZoomPercent(); applyZoom(); }); // Global Fit Width Button document.querySelector('#fitWidth')?.addEventListener('click', () => { settings$1.zoom = 1000; updateZoomPercent(); applyZoom(); }); // Global Fit height Button document.querySelector('#fitHeight')?.addEventListener('click', () => { settings$1.zoom = -1000; updateZoomPercent(); applyZoom(); }); // Zoom Step Slider document.querySelector('#zoomStep')?.addEventListener('change', (event) => { const step = event.currentTarget.value; setValueGM('ZoomStep', parseInt(step, 10)); logScript(`ZoomStep: ${getValueGM('ZoomStep')}`); }); // WebComic View Mode Button document.querySelector('#webComic')?.addEventListener('click', () => { document.querySelector('#Chapter')?.classList.add('WebComic'); document.querySelector('#Chapter')?.classList.remove('FluidLTR'); document.querySelector('#Chapter')?.classList.remove('FluidRTL'); applyZoom(); }); // Fluid LTR View Mode Button document.querySelector('#ltrMode')?.addEventListener('click', () => { document.querySelector('#Chapter')?.classList.remove('WebComic'); document.querySelector('#Chapter')?.classList.add('FluidLTR'); document.querySelector('#Chapter')?.classList.remove('FluidRTL'); applyZoom(); }); // Fluid RTL View Mode Button document.querySelector('#rtlMode')?.addEventListener('click', () => { document.querySelector('#Chapter')?.classList.remove('WebComic'); document.querySelector('#Chapter')?.classList.remove('FluidLTR'); document.querySelector('#Chapter')?.classList.add('FluidRTL'); applyZoom(); }); // Vertical View Mode Button document.querySelector('#verticalMode')?.addEventListener('click', () => { document.querySelector('#Chapter')?.classList.remove('WebComic'); document.querySelector('#Chapter')?.classList.remove('FluidLTR'); document.querySelector('#Chapter')?.classList.remove('FluidRTL'); applyZoom(); }); // Image Fit width if Oversized Toggle document.querySelector('#fitIfOversize')?.addEventListener('change', (event) => { document.querySelector('#Chapter')?.classList.toggle('fitWidthIfOversize'); if (event.currentTarget.checked) { setValueGM('FitWidthIfOversize', true); } else { setValueGM('FitWidthIfOversize', false); } logScript(`fitIfOversize: ${getValueGM('FitWidthIfOversize')}`); }); // Default View mode Selector document.querySelector('#viewMode')?.addEventListener('change', (event) => { const mode = event.currentTarget.value; document.querySelector('#Chapter')?.classList.remove('WebComic'); document.querySelector('#Chapter')?.classList.remove('FluidLTR'); document.querySelector('#Chapter')?.classList.remove('FluidRTL'); document.querySelector('#Chapter')?.classList.add(mode); setValueGM('ViewMode', mode); logScript(`ViewMode: ${getValueGM('ViewMode')}`); applyZoom(); }); // Start/Load mode Selector document.querySelector('#loadMode')?.addEventListener('change', (event) => { const mode = event.currentTarget.value; setValueGM('LoadMode', mode); logScript(`MangaLoadMode: ${getValueGM('LoadMode')}`); }); // Show Thumbnail Toggle document.querySelector('#showThumbnails')?.addEventListener('change', (event) => { document.querySelector('#Navigation')?.classList.toggle('disabled'); if (event.currentTarget.checked) { setValueGM('ShowThumbnails', true); } else { setValueGM('ShowThumbnails', false); } logScript(`MangaShowThumbnails: ${getValueGM('ShowThumbnails')}`); applyZoom(); }); // Download auto start toggle document.querySelector('#downloadZip')?.addEventListener('change', (event) => { if (event.currentTarget.checked) { setValueGM('DownloadZip', true); Swal.fire({ title: 'Attention', text: 'Next time a chapter finish loading you will be prompted to save automatically', timer: 10000, icon: 'info', }); } else { setValueGM('DownloadZip', false); } logScript(`MangaDownloadZip: ${getValueGM('DownloadZip')}`); }); // Download starter document.querySelector('#blob')?.addEventListener('click', generateZip, { once: true }); document.querySelector('.download')?.addEventListener('click', () => { logScript('Downloading Chapter'); document.querySelector('#blob')?.dispatchEvent(new Event('click')); }); // Lazy load Toggle document.querySelector('#lazyLoadImages')?.addEventListener('change', (event) => { if (event.currentTarget.checked) { setValueGM('LazyLoadImages', true); Swal.fire({ title: 'Warning', html: `Lazy load is incompatible with zip download, you will not be able to download with this setting ON.
Suggestion: Disable Thumbnails to save Bandwidth/Memory.`, icon: 'warning', }); } else { setValueGM('LazyLoadImages', false); } logScript(`MangaLazyLoadImages: ${getValueGM('LazyLoadImages')}`); }); // Lazy load starting point Slider document.querySelector('#lazyStart')?.addEventListener('change', (event) => { const start = event.currentTarget.value; setValueGM('LazyStart', start); logScript(`lazyStart: ${getValueGM('LazyStart')}`); }); // Images load speed Selector document.querySelector('#PagesPerSecond')?.addEventListener('change', (event) => { setValueGM('Timer', parseInt(event.currentTarget.value, 10)); logScript(`MangaTimer: ${getValueGM('Timer')}`); }); // Global Default Zoom Selector document.querySelector('#DefaultZoom')?.addEventListener('change', (event) => { settings$1.zoom = parseInt(event.currentTarget.value, 10); updateZoomPercent(); setValueGM('Zoom', parseInt(settings$1.zoom.toString(), 10)); logScript(`MangaZoom: ${getValueGM('Zoom')}`); applyZoom(); }); // Show/hide Image Controls Button document.querySelector('#pageControls')?.addEventListener('click', () => { document.querySelector('#MangaOnlineViewer')?.classList.toggle('hideControls'); }); // Show/hide Image Controls Toggle document.querySelector('#hidePageControls')?.addEventListener('change', (event) => { document.querySelector('#MangaOnlineViewer')?.classList.toggle('hideControls'); if (event.currentTarget.checked) { setValueGM('HidePageControls', true); } else { setValueGM('HidePageControls', false); } logScript(`MangaHidePageControls: ${getValueGM('HidePageControls')}`); }); // Sticky Header or MouseOverMenu Toggle document.querySelector('#mouseOverMenu')?.addEventListener('change', (event) => { document.querySelector('#Header')?.classList.toggle('mouseOverMenu'); if (event.currentTarget.checked) { setValueGM('MouseOverMenu', true); } else { setValueGM('MouseOverMenu', false); } logScript(`MangaHidePageControls: ${getValueGM('MouseOverMenu')}`); }); // Theme Control Selector document.querySelector('#ThemeSelector')?.addEventListener('change', (event) => { const target = event.currentTarget.value; [...document.querySelectorAll('#MangaOnlineViewer , body')].forEach((elem) => { elem.className = ''; elem.classList.add(event.currentTarget.value); }); logScript('Theme', target); setValueGM('Theme', target); const ct = [...document.querySelectorAll('.CustomTheme')]; if (target === 'Custom_Dark' || target === 'Custom_Light') { ct.forEach((elem) => { elem.style.display = 'inherit'; }); } else { ct.forEach((elem) => { elem.style.display = 'none'; }); } const fc = [...document.querySelectorAll('.FullCustom')]; if (target === 'Full_Custom') { fc.forEach((elem) => { elem.style.display = 'inherit'; }); } else { fc.forEach((elem) => { elem.style.display = 'none'; }); } }); // Light/Dark Custom theme Color Input document.querySelector('#CustomThemeHue')?.addEventListener('change', (event) => { const target = event.currentTarget.value; logScript(`CustomTheme: ${target}`); document .querySelectorAll('style[title="Custom_Light"], style[title="Custom_Dark"]') .forEach((elem) => elem.remove()); document.head.append(addCustomTheme(target)); setValueGM('CustomTheme', target); logScript(`MangaCustomTheme: ${getValueGM('CustomTheme')}`); }); // Full Custom theme Color Input document.querySelectorAll('.FullCustom')?.forEach((input) => input.addEventListener('change', () => { logScript('FullCustomTheme: ', document.querySelector('#CustomThemeHueBody')?.value, document.querySelector('#CustomThemeHueText')?.value, document.querySelector('#CustomThemeHueLines')?.value, document.querySelector('#CustomThemeHuePanel')?.value, document.querySelector('#CustomThemeHueButton')?.value); document.querySelectorAll('style[title="Full_Custom"]').forEach((elem) => elem.remove()); document.head.append(addFullCustomTheme(document.querySelector('#CustomThemeHueBody').value, document.querySelector('#CustomThemeHueText').value, document.querySelector('#CustomThemeHueLines').value, document.querySelector('#CustomThemeHuePanel').value, document.querySelector('#CustomThemeHueButton').value)); setValueGM('CustomThemeBody', document.querySelector('#CustomThemeHueBody').value); setValueGM('CustomThemeText', document.querySelector('#CustomThemeHueText').value); setValueGM('CustomThemeLines', document.querySelector('#CustomThemeHueLines').value); setValueGM('CustomThemePanel', document.querySelector('#CustomThemeHuePanel').value); setValueGM('CustomThemeButton', document.querySelector('#CustomThemeHueButton').value); })); // Goto Navigation document.querySelector('#gotoPage')?.addEventListener('change', (event) => { applyZoom(); scrollToElement(document.querySelector(`#Page${event.currentTarget.textContent}`)); }); // Thumbnail Navigation document.querySelectorAll('.Thumbnail')?.forEach((elem) => elem.addEventListener('click', (event) => { applyZoom(); scrollToElement(document.querySelector(`#Page${event.currentTarget.querySelector('.ThumbnailIndex')?.textContent}`)); })); // Settings Control document.querySelector('#settings')?.addEventListener('click', () => { document.querySelector('#ViewerControls')?.classList.toggle('visible'); document.querySelector('#ViewerShortcuts')?.classList.toggle('visible'); document.querySelector('#ImageOptions')?.classList.toggle('settingsOpen'); document.querySelector('#Navigation')?.classList.toggle('visible'); document.querySelector('#Header')?.classList.toggle('visible'); }); // Individual Page functions // Bookmark Page to resume reading document.querySelectorAll('.Bookmark')?.forEach((elem) => elem.addEventListener('click', (event) => { const num = parseInt(event.currentTarget.parentElement?.querySelector('.PageIndex') ?.textContent || '0', 10); const mark = { url: window.location.href, page: num, date: Date.now(), }; const found = settings$1.bookmarks.filter((el) => el.url === mark.url).length > 0; settings$1.bookmarks = settings$1.bookmarks.filter((el) => el.url !== mark.url); if (found) { Swal.fire({ title: 'Bookmark Removed', timer: 10000, icon: 'error', }); } else { settings$1.bookmarks.push(mark); Swal.fire({ title: 'Saved Bookmark', html: `Next time you open this chapter it will resume from:

Page ${num}

(Only ONCE per Bookmark, will be removed after a year unused)`, icon: 'success', }); } setValueGM('Bookmarks', JSON.stringify(settings$1.bookmarks)); logScript(`MangaBookmarks: ${getValueGM('Bookmarks')}`); })); // Reload Page document.querySelectorAll('.Reload')?.forEach((elem) => elem.addEventListener('click', (event) => { const img = event.currentTarget.parentElement?.parentElement?.querySelector('.PageImg'); reloadImage(img); })); // ZoomIn document.querySelectorAll('.ZoomIn')?.forEach((elem) => elem.addEventListener('click', (event) => { const img = event.currentTarget.parentElement?.parentElement?.querySelector('.PageImg'); const ratio = (img.width / img.naturalWidth) * (100 + settings$1.zoomStep); applyZoom(`#${img.getAttribute('id')}`, ratio); })); // ZoomOut document.querySelectorAll('.ZoomOut')?.forEach((elem) => elem.addEventListener('click', (event) => { const img = event.currentTarget.parentElement?.parentElement?.querySelector('.PageImg'); const ratio = (img.width / img.naturalWidth) * (100 - settings$1.zoomStep); applyZoom(`#${img.getAttribute('id')}`, ratio); })); // ZoomRestore document.querySelectorAll('.ZoomRestore')?.forEach((elem) => elem.addEventListener('click', () => { document.querySelector('.PageContent .PageImg')?.removeAttribute('width'); })); // ZoomWidth document.querySelectorAll('.ZoomWidth')?.forEach((elem) => elem.addEventListener('click', (event) => { const img = event.currentTarget.parentElement?.parentElement?.querySelector('.PageImg'); applyZoom(`#${img.getAttribute('id')}`, 1000); })); // ZoomHeight document.querySelectorAll('.ZoomHeight')?.forEach((elem) => elem.addEventListener('click', (event) => { const img = event.currentTarget.parentElement?.parentElement?.querySelector('.PageImg'); applyZoom(`#${img.getAttribute('id')}`, -1000); })); // Hide document.querySelectorAll('.Hide')?.forEach((elem) => elem.addEventListener('click', (event) => { const img = event.currentTarget.parentElement?.parentElement?.querySelector('.PageContent'); img.classList.toggle('hide'); })); /** * Changes header class when scrolling up or down to show/hide it * @param showEnd [default 0]px from end of the screen to show header */ function useScrollDirection(showEnd = 0) { let prevOffset = 0; const header = document.querySelector('#Header'); const setScrollDirection = (classSuffix) => { header.classList.remove('scroll-end'); header.classList.remove('scroll-hide'); header.classList.remove('scroll-show'); if (classSuffix) header.classList.add(`scroll-${classSuffix}`); }; function toggleScrollDirection() { const { scrollY } = window; if (showEnd && scrollY + window.innerHeight + showEnd > document.body.offsetHeight) { setScrollDirection('end'); } else if (scrollY > prevOffset && scrollY > 50) { setScrollDirection('hide'); } else if (scrollY < prevOffset && scrollY > 50) { setScrollDirection('show'); } else { setScrollDirection(''); } prevOffset = scrollY; } window.addEventListener('scroll', toggleScrollDirection); } useScrollDirection(100); } function display(manga, begin) { window.stop(); if (manga.before !== undefined) { manga.before(); } document.head.innerHTML = head(manga); document.body.innerHTML = body(manga, begin); document.body.className = ''; document.body.removeAttribute('style'); // document.documentElement.innerHTML = reader(manga, begin); logScript('Rebuilding Site'); setTimeout(() => { try { controls(); setKeyDownEvents(); setTimeout(() => { window.scrollTo(0, 0); loadManga(manga, begin); }, 50); // Clear used Bookmarks if (!isNothing(settings$1.bookmarks.filter((el) => el.url === window.location.href))) { logScript(`Bookmark Removed ${window.location.href}`); settings$1.bookmarks = settings$1.bookmarks.filter((el) => el.url !== window.location.href); setValueGM('Bookmarks', JSON.stringify(settings$1.bookmarks)); } } catch (e) { logScript(e); } }, 50); if (manga.after !== undefined) { manga.after(); } } async function formatPage(manga, begin = 0) { display(manga, begin); } // language=CSS var sweetalertStyle = `.swal2-popup.swal2-toast{box-sizing:border-box;grid-column:1/4 !important;grid-row:1/4 !important;grid-template-columns:1fr 99fr 1fr;padding:1em;overflow-y:hidden;background:#fff;box-shadow:0 0 1px hsla(0deg, 0%, 0%, 0.075), 0 1px 2px hsla(0deg, 0%, 0%, 0.075), 1px 2px 4px hsla(0deg, 0%, 0%, 0.075), 1px 3px 8px hsla(0deg, 0%, 0%, 0.075), 2px 4px 16px hsla(0deg, 0%, 0%, 0.075);pointer-events:all}.swal2-popup.swal2-toast > *{grid-column:2}.swal2-popup.swal2-toast #swal2-title{margin:0.5em 1em;padding:0;font-size:1em;text-align:initial}.swal2-popup.swal2-toast .swal2-loading{justify-content:center}.swal2-popup.swal2-toast .swal2-input{height:2em;margin:0.5em;font-size:1em}.swal2-popup.swal2-toast .swal2-validation-message{font-size:1em}.swal2-popup.swal2-toast .swal2-footer{margin:0.5em 0 0;padding:0.5em 0 0;font-size:0.8em}.swal2-popup.swal2-toast .swal2-close{grid-column:3/3;grid-row:1/99;align-self:center;width:0.8em;height:0.8em;margin:0;font-size:2em}.swal2-popup.swal2-toast .swal2-html-container{margin:0.5em 1em;padding:0;font-size:1em;text-align:initial}.swal2-popup.swal2-toast .swal2-html-container:empty{padding:0}.swal2-popup.swal2-toast .swal2-loader{grid-column:1;grid-row:1/99;align-self:center;width:2em;height:2em;margin:0.25em}.swal2-popup.swal2-toast .swal2-icon{grid-column:1;grid-row:1/99;align-self:center;width:2em;min-width:2em;height:2em;margin:0 0.5em 0 0}.swal2-popup.swal2-toast .swal2-icon .swal2-icon-content{display:flex;align-items:center;font-size:1.8em;font-weight:700}.swal2-popup.swal2-toast .swal2-icon.swal2-success .swal2-success-ring{width:2em;height:2em}.swal2-popup.swal2-toast .swal2-icon.swal2-error [class^='swal2-x-mark-line']{top:0.875em;width:1.375em}.swal2-popup.swal2-toast .swal2-icon.swal2-error [class^='swal2-x-mark-line'][class$='left']{left:0.3125em}.swal2-popup.swal2-toast .swal2-icon.swal2-error [class^='swal2-x-mark-line'][class$='right']{right:0.3125em}.swal2-popup.swal2-toast .swal2-actions{justify-content:flex-start;height:auto;margin:0.5em 0 0;padding:0 0.5em}.swal2-popup.swal2-toast .swal2-styled{margin:0.25em 0.5em;padding:0.4em 0.6em;font-size:1em}.swal2-popup.swal2-toast .swal2-success{border-color:#a5dc86}.swal2-popup.swal2-toast .swal2-success [class^='swal2-success-circular-line']{position:absolute;width:1.6em;height:3em;transform:rotate(45deg);border-radius:50%}.swal2-popup.swal2-toast .swal2-success [class^='swal2-success-circular-line'][class$='left']{top:-0.8em;left:-0.5em;transform:rotate(-45deg);transform-origin:2em 2em;border-radius:4em 0 0 4em}.swal2-popup.swal2-toast .swal2-success [class^='swal2-success-circular-line'][class$='right']{top:-0.25em;left:0.9375em;transform-origin:0 1.5em;border-radius:0 4em 4em 0}.swal2-popup.swal2-toast .swal2-success .swal2-success-ring{width:2em;height:2em}.swal2-popup.swal2-toast .swal2-success .swal2-success-fix{top:0;left:0.4375em;width:0.4375em;height:2.6875em}.swal2-popup.swal2-toast .swal2-success [class^='swal2-success-line']{height:0.3125em}.swal2-popup.swal2-toast .swal2-success [class^='swal2-success-line'][class$='tip']{top:1.125em;left:0.1875em;width:0.75em}.swal2-popup.swal2-toast .swal2-success [class^='swal2-success-line'][class$='long']{top:0.9375em;right:0.1875em;width:1.375em}.swal2-popup.swal2-toast .swal2-success.swal2-icon-show .swal2-success-line-tip{-webkit-animation:swal2-toast-animate-success-line-tip 0.75s;animation:swal2-toast-animate-success-line-tip 0.75s}.swal2-popup.swal2-toast .swal2-success.swal2-icon-show .swal2-success-line-long{-webkit-animation:swal2-toast-animate-success-line-long 0.75s;animation:swal2-toast-animate-success-line-long 0.75s}.swal2-popup.swal2-toast.swal2-show{-webkit-animation:swal2-toast-show 0.5s;animation:swal2-toast-show 0.5s}.swal2-popup.swal2-toast.swal2-hide{-webkit-animation:swal2-toast-hide 0.1s forwards;animation:swal2-toast-hide 0.1s forwards}.swal2-container{display:grid;position:fixed;z-index:1060;top:0;right:0;bottom:0;left:0;box-sizing:border-box;grid-template-areas:'top-start top top-end' 'center-start center center-end' 'bottom-start bottom-center bottom-end';grid-template-rows:minmax(-webkit-min-content, auto) minmax(-webkit-min-content, auto) minmax( -webkit-min-content, auto );grid-template-rows:minmax(min-content, auto) minmax(min-content, auto) minmax( min-content, auto );height:100%;padding:0.625em;overflow-x:hidden;transition:background-color 0.1s;-webkit-overflow-scrolling:touch}.swal2-container.swal2-backdrop-show,.swal2-container.swal2-noanimation{background:rgba(0, 0, 0, 0.4)}.swal2-container.swal2-backdrop-hide{background:0 0 !important}.swal2-container.swal2-bottom-start,.swal2-container.swal2-center-start,.swal2-container.swal2-top-start{grid-template-columns:minmax(0, 1fr) auto auto}.swal2-container.swal2-bottom,.swal2-container.swal2-center,.swal2-container.swal2-top{grid-template-columns:auto minmax(0, 1fr) auto}.swal2-container.swal2-bottom-end,.swal2-container.swal2-center-end,.swal2-container.swal2-top-end{grid-template-columns:auto auto minmax(0, 1fr)}.swal2-container.swal2-top-start > .swal2-popup{align-self:start}.swal2-container.swal2-top > .swal2-popup{grid-column:2;align-self:start;justify-self:center}.swal2-container.swal2-top-end > .swal2-popup,.swal2-container.swal2-top-right > .swal2-popup{grid-column:3;align-self:start;justify-self:end}.swal2-container.swal2-center-left > .swal2-popup,.swal2-container.swal2-center-start > .swal2-popup{grid-row:2;align-self:center}.swal2-container.swal2-center > .swal2-popup{grid-column:2;grid-row:2;align-self:center;justify-self:center}.swal2-container.swal2-center-end > .swal2-popup,.swal2-container.swal2-center-right > .swal2-popup{grid-column:3;grid-row:2;align-self:center;justify-self:end}.swal2-container.swal2-bottom-left > .swal2-popup,.swal2-container.swal2-bottom-start > .swal2-popup{grid-column:1;grid-row:3;align-self:end}.swal2-container.swal2-bottom > .swal2-popup{grid-column:2;grid-row:3;justify-self:center;align-self:end}.swal2-container.swal2-bottom-end > .swal2-popup,.swal2-container.swal2-bottom-right > .swal2-popup{grid-column:3;grid-row:3;align-self:end;justify-self:end}.swal2-container.swal2-grow-fullscreen > .swal2-popup,.swal2-container.swal2-grow-row > .swal2-popup{grid-column:1/4;width:100%}.swal2-container.swal2-grow-column > .swal2-popup,.swal2-container.swal2-grow-fullscreen > .swal2-popup{grid-row:1/4;align-self:stretch}.swal2-container.swal2-no-transition{transition:none !important}.swal2-popup{display:none;position:relative;box-sizing:border-box;grid-template-columns:minmax(0, 100%);width:32em;max-width:100%;padding:0 0 1.25em;border:none;border-radius:5px;background:#fff;color:#545454;font-family:inherit;font-size:1rem}.swal2-popup:focus{outline:0}.swal2-popup.swal2-loading{overflow-y:hidden}#swal2-title{position:relative;max-width:100%;margin:0;padding:0.8em 1em 0;color:inherit;font-size:1.875em;font-weight:600;text-align:center;text-transform:none;word-wrap:break-word}.swal2-actions{display:flex;z-index:1;box-sizing:border-box;flex-wrap:wrap;align-items:center;justify-content:center;width:auto;margin:1.25em auto 0;padding:0}.swal2-actions:not(.swal2-loading) .swal2-styled[disabled]{opacity:0.4}.swal2-actions:not(.swal2-loading) .swal2-styled:hover{background-image:linear-gradient(rgba(0, 0, 0, 0.1), rgba(0, 0, 0, 0.1))}.swal2-actions:not(.swal2-loading) .swal2-styled:active{background-image:linear-gradient(rgba(0, 0, 0, 0.2), rgba(0, 0, 0, 0.2))}.swal2-loader{display:none;align-items:center;justify-content:center;width:2.2em;height:2.2em;margin:0 1.875em;-webkit-animation:swal2-rotate-loading 1.5s linear 0s infinite normal;animation:swal2-rotate-loading 1.5s linear 0s infinite normal;border-width:0.25em;border-style:solid;border-radius:100%;border-color:#2778c4 transparent #2778c4 transparent}.swal2-styled{margin:0.3125em;padding:0.625em 1.1em;transition:box-shadow 0.1s;box-shadow:0 0 0 3px transparent;font-weight:500}.swal2-styled:not([disabled]){cursor:pointer}.swal2-styled.swal2-confirm{border:0;border-radius:0.25em;background:initial;background-color:#7066e0;color:#fff;font-size:1em}.swal2-styled.swal2-confirm:focus{box-shadow:0 0 0 3px rgba(112, 102, 224, 0.5)}.swal2-styled.swal2-deny{border:0;border-radius:0.25em;background:initial;background-color:#dc3741;color:#fff;font-size:1em}.swal2-styled.swal2-deny:focus{box-shadow:0 0 0 3px rgba(220, 55, 65, 0.5)}.swal2-styled.swal2-cancel{border:0;border-radius:0.25em;background:initial;background-color:#6e7881;color:#fff;font-size:1em}.swal2-styled.swal2-cancel:focus{box-shadow:0 0 0 3px rgba(110, 120, 129, 0.5)}.swal2-styled.swal2-default-outline:focus{box-shadow:0 0 0 3px rgba(100, 150, 200, 0.5)}.swal2-styled:focus{outline:0}.swal2-styled::-moz-focus-inner{border:0}.swal2-footer{justify-content:center;margin:1em 0 0;padding:1em 1em 0;border-top:1px solid #eee;color:inherit;font-size:1em}.swal2-timer-progress-bar-container{position:absolute;right:0;bottom:0;left:0;grid-column:auto !important;overflow:hidden;border-bottom-right-radius:5px;border-bottom-left-radius:5px}.swal2-timer-progress-bar{width:100%;height:0.25em;background:rgba(0, 0, 0, 0.2)}.swal2-image{max-width:100%;margin:2em auto 1em}.swal2-close{z-index:2;align-items:center;justify-content:center;width:1.2em;height:1.2em;margin-top:0;margin-right:0;margin-bottom:-1.2em;padding:0;overflow:hidden;transition:color 0.1s, box-shadow 0.1s;border:none;border-radius:5px;background:0 0;color:#ccc;font-family:serif;font-family:monospace;font-size:2.5em;cursor:pointer;justify-self:end}.swal2-close:hover{transform:none;background:0 0;color:#f27474}.swal2-close:focus{outline:0;box-shadow:inset 0 0 0 3px rgba(100, 150, 200, 0.5)}.swal2-close::-moz-focus-inner{border:0}.swal2-html-container{z-index:1;justify-content:center;margin:1em 1.6em 0.3em;padding:0;overflow:auto;color:inherit;font-size:1.125em;font-weight:400;line-height:normal;text-align:center;word-wrap:break-word;word-break:break-word}.swal2-checkbox,.swal2-file,.swal2-input,.swal2-radio,.swal2-select,.swal2-textarea{margin:1em 2em 3px}.swal2-file,.swal2-input,.swal2-textarea{box-sizing:border-box;width:auto;transition:border-color 0.1s, box-shadow 0.1s;border:1px solid #d9d9d9;border-radius:0.1875em;background:0 0;box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.06), 0 0 0 3px transparent;color:inherit;font-size:1.125em}.swal2-file.swal2-inputerror,.swal2-input.swal2-inputerror,.swal2-textarea.swal2-inputerror{border-color:#f27474 !important;box-shadow:0 0 2px #f27474 !important}.swal2-file:focus,.swal2-input:focus,.swal2-textarea:focus{border:1px solid #b4dbed;outline:0;box-shadow:inset 0 1px 1px rgba(0, 0, 0, 0.06), 0 0 0 3px rgba(100, 150, 200, 0.5)}.swal2-file::-moz-placeholder,.swal2-input::-moz-placeholder,.swal2-textarea::-moz-placeholder{color:#ccc}.swal2-file:-ms-input-placeholder,.swal2-input:-ms-input-placeholder,.swal2-textarea:-ms-input-placeholder{color:#ccc}.swal2-file::placeholder,.swal2-input::placeholder,.swal2-textarea::placeholder{color:#ccc}.swal2-range{margin:1em 2em 3px;background:#fff}.swal2-range input{width:80%}.swal2-range output{width:20%;color:inherit;font-weight:600;text-align:center}.swal2-range input,.swal2-range output{height:2.625em;padding:0;font-size:1.125em;line-height:2.625em}.swal2-input{height:2.625em;padding:0 0.75em}.swal2-file{width:75%;margin-right:auto;margin-left:auto;background:0 0;font-size:1.125em}.swal2-textarea{height:6.75em;padding:0.75em}.swal2-select{min-width:50%;max-width:100%;padding:0.375em 0.625em;background:0 0;color:inherit;font-size:1.125em}.swal2-checkbox,.swal2-radio{align-items:center;justify-content:center;background:#fff;color:inherit}.swal2-checkbox label,.swal2-radio label{margin:0 0.6em;font-size:1.125em}.swal2-checkbox input,.swal2-radio input{flex-shrink:0;margin:0 0.4em}.swal2-input-label{display:flex;justify-content:center;margin:1em auto 0}.swal2-validation-message{align-items:center;justify-content:center;margin:1em 0 0;padding:0.625em;overflow:hidden;background:#f0f0f0;color:#666;font-size:1em;font-weight:300}.swal2-validation-message::before{content:'!';display:inline-block;width:1.5em;min-width:1.5em;height:1.5em;margin:0 0.625em;border-radius:50%;background-color:#f27474;color:#fff;font-weight:600;line-height:1.5em;text-align:center}.swal2-icon{position:relative;box-sizing:content-box;justify-content:center;width:5em;height:5em;margin:2.5em auto 0.6em;border:0.25em solid transparent;border-radius:50%;border-color:#000;font-family:inherit;line-height:5em;cursor:default;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.swal2-icon .swal2-icon-content{display:flex;align-items:center;font-size:3.75em}.swal2-icon.swal2-error{border-color:#f27474;color:#f27474}.swal2-icon.swal2-error .swal2-x-mark{position:relative;flex-grow:1}.swal2-icon.swal2-error [class^='swal2-x-mark-line']{display:block;position:absolute;top:2.3125em;width:2.9375em;height:0.3125em;border-radius:0.125em;background-color:#f27474}.swal2-icon.swal2-error [class^='swal2-x-mark-line'][class$='left']{left:1.0625em;transform:rotate(45deg)}.swal2-icon.swal2-error [class^='swal2-x-mark-line'][class$='right']{right:1em;transform:rotate(-45deg)}.swal2-icon.swal2-error.swal2-icon-show{-webkit-animation:swal2-animate-error-icon 0.5s;animation:swal2-animate-error-icon 0.5s}.swal2-icon.swal2-error.swal2-icon-show .swal2-x-mark{-webkit-animation:swal2-animate-error-x-mark 0.5s;animation:swal2-animate-error-x-mark 0.5s}.swal2-icon.swal2-warning{border-color:#facea8;color:#f8bb86}.swal2-icon.swal2-warning.swal2-icon-show{-webkit-animation:swal2-animate-error-icon 0.5s;animation:swal2-animate-error-icon 0.5s}.swal2-icon.swal2-warning.swal2-icon-show .swal2-icon-content{-webkit-animation:swal2-animate-i-mark 0.5s;animation:swal2-animate-i-mark 0.5s}.swal2-icon.swal2-info{border-color:#9de0f6;color:#3fc3ee}.swal2-icon.swal2-info.swal2-icon-show{-webkit-animation:swal2-animate-error-icon 0.5s;animation:swal2-animate-error-icon 0.5s}.swal2-icon.swal2-info.swal2-icon-show .swal2-icon-content{-webkit-animation:swal2-animate-i-mark 0.8s;animation:swal2-animate-i-mark 0.8s}.swal2-icon.swal2-question{border-color:#c9dae1;color:#87adbd}.swal2-icon.swal2-question.swal2-icon-show{-webkit-animation:swal2-animate-error-icon 0.5s;animation:swal2-animate-error-icon 0.5s}.swal2-icon.swal2-question.swal2-icon-show .swal2-icon-content{-webkit-animation:swal2-animate-question-mark 0.8s;animation:swal2-animate-question-mark 0.8s}.swal2-icon.swal2-success{border-color:#a5dc86;color:#a5dc86}.swal2-icon.swal2-success [class^='swal2-success-circular-line']{position:absolute;width:3.75em;height:7.5em;transform:rotate(45deg);border-radius:50%}.swal2-icon.swal2-success [class^='swal2-success-circular-line'][class$='left']{top:-0.4375em;left:-2.0635em;transform:rotate(-45deg);transform-origin:3.75em 3.75em;border-radius:7.5em 0 0 7.5em}.swal2-icon.swal2-success [class^='swal2-success-circular-line'][class$='right']{top:-0.6875em;left:1.875em;transform:rotate(-45deg);transform-origin:0 3.75em;border-radius:0 7.5em 7.5em 0}.swal2-icon.swal2-success .swal2-success-ring{position:absolute;z-index:2;top:-0.25em;left:-0.25em;box-sizing:content-box;width:100%;height:100%;border:0.25em solid rgba(165, 220, 134, 0.3);border-radius:50%}.swal2-icon.swal2-success .swal2-success-fix{position:absolute;z-index:1;top:0.5em;left:1.625em;width:0.4375em;height:5.625em;transform:rotate(-45deg)}.swal2-icon.swal2-success [class^='swal2-success-line']{display:block;position:absolute;z-index:2;height:0.3125em;border-radius:0.125em;background-color:#a5dc86}.swal2-icon.swal2-success [class^='swal2-success-line'][class$='tip']{top:2.875em;left:0.8125em;width:1.5625em;transform:rotate(45deg)}.swal2-icon.swal2-success [class^='swal2-success-line'][class$='long']{top:2.375em;right:0.5em;width:2.9375em;transform:rotate(-45deg)}.swal2-icon.swal2-success.swal2-icon-show .swal2-success-line-tip{-webkit-animation:swal2-animate-success-line-tip 0.75s;animation:swal2-animate-success-line-tip 0.75s}.swal2-icon.swal2-success.swal2-icon-show .swal2-success-line-long{-webkit-animation:swal2-animate-success-line-long 0.75s;animation:swal2-animate-success-line-long 0.75s}.swal2-icon.swal2-success.swal2-icon-show .swal2-success-circular-line-right{-webkit-animation:swal2-rotate-success-circular-line 4.25s ease-in;animation:swal2-rotate-success-circular-line 4.25s ease-in}.swal2-progress-steps{flex-wrap:wrap;align-items:center;max-width:100%;margin:1.25em auto;padding:0;background:0 0;font-weight:600}.swal2-progress-steps li{display:inline-block;position:relative}.swal2-progress-steps .swal2-progress-step{z-index:20;flex-shrink:0;width:2em;height:2em;border-radius:2em;background:#2778c4;color:#fff;line-height:2em;text-align:center}.swal2-progress-steps .swal2-progress-step.swal2-active-progress-step{background:#2778c4}.swal2-progress-steps .swal2-progress-step.swal2-active-progress-step ~ .swal2-progress-step{background:#add8e6;color:#fff}.swal2-progress-steps .swal2-progress-step.swal2-active-progress-step ~ .swal2-progress-step-line{background:#add8e6}.swal2-progress-steps .swal2-progress-step-line{z-index:10;flex-shrink:0;width:2.5em;height:0.4em;margin:0 -1px;background:#2778c4}[class^='swal2']{-webkit-tap-highlight-color:transparent}.swal2-show{-webkit-animation:swal2-show 0.3s;animation:swal2-show 0.3s}.swal2-hide{-webkit-animation:swal2-hide 0.15s forwards;animation:swal2-hide 0.15s forwards}.swal2-noanimation{transition:none}.swal2-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}.swal2-rtl .swal2-close{margin-right:initial;margin-left:0}.swal2-rtl .swal2-timer-progress-bar{right:0;left:auto}.swal2-no-war{display:flex;position:fixed;z-index:1061;top:0;left:0;align-items:center;justify-content:center;width:100%;height:3.375em;background:#20232a;color:#fff;text-align:center}.swal2-no-war a{color:#61dafb;text-decoration:none}.swal2-no-war a:hover{text-decoration:underline}@-webkit-keyframes swal2-toast-show{0%{transform:translateY(-0.625em) rotateZ(2deg)}33%{transform:translateY(0) rotateZ(-2deg)}66%{transform:translateY(0.3125em) rotateZ(2deg)}100%{transform:translateY(0) rotateZ(0)}}@keyframes swal2-toast-show{0%{transform:translateY(-0.625em) rotateZ(2deg)}33%{transform:translateY(0) rotateZ(-2deg)}66%{transform:translateY(0.3125em) rotateZ(2deg)}100%{transform:translateY(0) rotateZ(0)}}@-webkit-keyframes swal2-toast-hide{100%{transform:rotateZ(1deg);opacity:0}}@keyframes swal2-toast-hide{100%{transform:rotateZ(1deg);opacity:0}}@-webkit-keyframes swal2-toast-animate-success-line-tip{0%{top:0.5625em;left:0.0625em;width:0}54%{top:0.125em;left:0.125em;width:0}70%{top:0.625em;left:-0.25em;width:1.625em}84%{top:1.0625em;left:0.75em;width:0.5em}100%{top:1.125em;left:0.1875em;width:0.75em}}@keyframes swal2-toast-animate-success-line-tip{0%{top:0.5625em;left:0.0625em;width:0}54%{top:0.125em;left:0.125em;width:0}70%{top:0.625em;left:-0.25em;width:1.625em}84%{top:1.0625em;left:0.75em;width:0.5em}100%{top:1.125em;left:0.1875em;width:0.75em}}@-webkit-keyframes swal2-toast-animate-success-line-long{0%{top:1.625em;right:1.375em;width:0}65%{top:1.25em;right:0.9375em;width:0}84%{top:0.9375em;right:0;width:1.125em}100%{top:0.9375em;right:0.1875em;width:1.375em}}@keyframes swal2-toast-animate-success-line-long{0%{top:1.625em;right:1.375em;width:0}65%{top:1.25em;right:0.9375em;width:0}84%{top:0.9375em;right:0;width:1.125em}100%{top:0.9375em;right:0.1875em;width:1.375em}}@-webkit-keyframes swal2-show{0%{transform:scale(0.7)}45%{transform:scale(1.05)}80%{transform:scale(0.95)}100%{transform:scale(1)}}@keyframes swal2-show{0%{transform:scale(0.7)}45%{transform:scale(1.05)}80%{transform:scale(0.95)}100%{transform:scale(1)}}@-webkit-keyframes swal2-hide{0%{transform:scale(1);opacity:1}100%{transform:scale(0.5);opacity:0}}@keyframes swal2-hide{0%{transform:scale(1);opacity:1}100%{transform:scale(0.5);opacity:0}}@-webkit-keyframes swal2-animate-success-line-tip{0%{top:1.1875em;left:0.0625em;width:0}54%{top:1.0625em;left:0.125em;width:0}70%{top:2.1875em;left:-0.375em;width:3.125em}84%{top:3em;left:1.3125em;width:1.0625em}100%{top:2.8125em;left:0.8125em;width:1.5625em}}@keyframes swal2-animate-success-line-tip{0%{top:1.1875em;left:0.0625em;width:0}54%{top:1.0625em;left:0.125em;width:0}70%{top:2.1875em;left:-0.375em;width:3.125em}84%{top:3em;left:1.3125em;width:1.0625em}100%{top:2.8125em;left:0.8125em;width:1.5625em}}@-webkit-keyframes swal2-animate-success-line-long{0%{top:3.375em;right:2.875em;width:0}65%{top:3.375em;right:2.875em;width:0}84%{top:2.1875em;right:0;width:3.4375em}100%{top:2.375em;right:0.5em;width:2.9375em}}@keyframes swal2-animate-success-line-long{0%{top:3.375em;right:2.875em;width:0}65%{top:3.375em;right:2.875em;width:0}84%{top:2.1875em;right:0;width:3.4375em}100%{top:2.375em;right:0.5em;width:2.9375em}}@-webkit-keyframes swal2-rotate-success-circular-line{0%{transform:rotate(-45deg)}5%{transform:rotate(-45deg)}12%{transform:rotate(-405deg)}100%{transform:rotate(-405deg)}}@keyframes swal2-rotate-success-circular-line{0%{transform:rotate(-45deg)}5%{transform:rotate(-45deg)}12%{transform:rotate(-405deg)}100%{transform:rotate(-405deg)}}@-webkit-keyframes swal2-animate-error-x-mark{0%{margin-top:1.625em;transform:scale(0.4);opacity:0}50%{margin-top:1.625em;transform:scale(0.4);opacity:0}80%{margin-top:-0.375em;transform:scale(1.15)}100%{margin-top:0;transform:scale(1);opacity:1}}@keyframes swal2-animate-error-x-mark{0%{margin-top:1.625em;transform:scale(0.4);opacity:0}50%{margin-top:1.625em;transform:scale(0.4);opacity:0}80%{margin-top:-0.375em;transform:scale(1.15)}100%{margin-top:0;transform:scale(1);opacity:1}}@-webkit-keyframes swal2-animate-error-icon{0%{transform:rotateX(100deg);opacity:0}100%{transform:rotateX(0);opacity:1}}@keyframes swal2-animate-error-icon{0%{transform:rotateX(100deg);opacity:0}100%{transform:rotateX(0);opacity:1}}@-webkit-keyframes swal2-rotate-loading{0%{transform:rotate(0)}100%{transform:rotate(360deg)}}@keyframes swal2-rotate-loading{0%{transform:rotate(0)}100%{transform:rotate(360deg)}}@-webkit-keyframes swal2-animate-question-mark{0%{transform:rotateY(-360deg)}100%{transform:rotateY(0)}}@keyframes swal2-animate-question-mark{0%{transform:rotateY(-360deg)}100%{transform:rotateY(0)}}@-webkit-keyframes swal2-animate-i-mark{0%{transform:rotateZ(45deg);opacity:0}25%{transform:rotateZ(-25deg);opacity:0.4}50%{transform:rotateZ(15deg);opacity:0.8}75%{transform:rotateZ(-5deg);opacity:1}100%{transform:rotateX(0);opacity:1}}@keyframes swal2-animate-i-mark{0%{transform:rotateZ(45deg);opacity:0}25%{transform:rotateZ(-25deg);opacity:0.4}50%{transform:rotateZ(15deg);opacity:0.8}75%{transform:rotateZ(-5deg);opacity:1}100%{transform:rotateX(0);opacity:1}}body.swal2-shown:not(.swal2-no-backdrop):not(.swal2-toast-shown){overflow:hidden}body.swal2-height-auto{height:auto !important}body.swal2-no-backdrop .swal2-container{background-color:transparent !important;pointer-events:none}body.swal2-no-backdrop .swal2-container .swal2-popup{pointer-events:all}body.swal2-no-backdrop .swal2-container .swal2-modal{box-shadow:0 0 10px rgba(0, 0, 0, 0.4)}@media print{body.swal2-shown:not(.swal2-no-backdrop):not(.swal2-toast-shown){overflow-y:scroll !important}body.swal2-shown:not(.swal2-no-backdrop):not(.swal2-toast-shown) > [aria-hidden='true']{display:none}body.swal2-shown:not(.swal2-no-backdrop):not(.swal2-toast-shown) .swal2-container{position:static !important}}body.swal2-toast-shown .swal2-container{box-sizing:border-box;width:360px;max-width:100%;background-color:transparent;pointer-events:none}body.swal2-toast-shown .swal2-container.swal2-top{top:0;right:auto;bottom:auto;left:50%;transform:translateX(-50%)}body.swal2-toast-shown .swal2-container.swal2-top-end,body.swal2-toast-shown .swal2-container.swal2-top-right{top:0;right:0;bottom:auto;left:auto}body.swal2-toast-shown .swal2-container.swal2-top-left,body.swal2-toast-shown .swal2-container.swal2-top-start{top:0;right:auto;bottom:auto;left:0}body.swal2-toast-shown .swal2-container.swal2-center-left,body.swal2-toast-shown .swal2-container.swal2-center-start{top:50%;right:auto;bottom:auto;left:0;transform:translateY(-50%)}body.swal2-toast-shown .swal2-container.swal2-center{top:50%;right:auto;bottom:auto;left:50%;transform:translate(-50%, -50%)}body.swal2-toast-shown .swal2-container.swal2-center-end,body.swal2-toast-shown .swal2-container.swal2-center-right{top:50%;right:0;bottom:auto;left:auto;transform:translateY(-50%)}body.swal2-toast-shown .swal2-container.swal2-bottom-left,body.swal2-toast-shown .swal2-container.swal2-bottom-start{top:auto;right:auto;bottom:0;left:0}body.swal2-toast-shown .swal2-container.swal2-bottom{top:auto;right:auto;bottom:0;left:50%;transform:translateX(-50%)}body.swal2-toast-shown .swal2-container.swal2-bottom-end,body.swal2-toast-shown .swal2-container.swal2-bottom-right{top:auto;right:0;bottom:0;left:auto}`; async function lateStart(site, begin = 1) { const manga = await site.run(); logScript('LateStart'); const options = { title: 'Starting
MangaOnlineViewer', input: 'range', inputAttributes: { min: '1', max: manga.pages.toString(), step: '1', }, inputValue: begin || 1, text: 'Choose the Page to start from:', showCancelButton: true, cancelButtonColor: '#d33', reverseButtons: true, icon: 'question', }; Swal.fire(options).then((result) => { if (result.value) { logScript(`Choice: ${result.value}`); formatPage(manga, result.value - 1); } else { logScript(result.dismiss); } }); } function createLateStartButton(site, beginning) { const button = document.createElement('button'); button.innerText = 'Start MangaOnlineViewer'; button.id = 'StartMOV'; button.onclick = () => { lateStart(site, beginning); }; document.body.appendChild(button); const css = ` #StartMOV { font-size: 20px; font-weight: bold; color: #fff; cursor: pointer; margin: 20px; padding: 10px 20px; text-align: center; border: none; background-size: 300% 100%; border-radius: 50px; transition: all 0.4s ease-in-out; background-image: linear-gradient(to right, #667eea, #764ba2, #6b8dd6, #8e37d7); box-shadow: 0 4px 15px 0 rgba(116, 79, 168, 0.75); position: fixed; top: 10px; right: 10px; z-index: 10000; } #StartMOV:hover { background-position: 100% 0; transition: all 0.4s ease-in-out; } #StartMOV:focus { outline: none; } `; const style = document.createElement('style'); style.appendChild(document.createTextNode(css)); document.head.appendChild(style); logScript('Start Button added to page', button); } // Organize the site adding place-holders for the manga pages function preparePage(site, manga, begin = 0) { logScript(`Found Pages: ${manga.pages}`); if (manga.pages > 0) { let beginning = begin; if (beginning === 0) { beginning = settings$1?.bookmarks?.find((b) => b.url === window.location.href)?.page || 0; } const style = document.createElement('style'); style.appendChild(document.createTextNode(sweetalertStyle)); document.body.appendChild(style); // window.mov = (b: number) => lateStart(site, b || beginning); switch (site.start || settings$1?.loadMode) { case 'never': createLateStartButton(site, beginning); break; case 'always': formatPage(manga, 0); break; case 'wait': default: Swal.fire({ title: 'Starting
MangaOnlineViewer', html: `${beginning > 1 ? `Resuming reading from Page ${beginning}.
` : ''}Please wait, 3 seconds...`, showCancelButton: true, cancelButtonColor: '#d33', reverseButtons: true, timer: 3000, }).then((result) => { if (result.value || result.dismiss === Swal.DismissReason.timer) { formatPage(manga, beginning); } else { createLateStartButton(site, beginning); logScript(result.dismiss); } }); break; } } } // Wait for something on the site to be ready before executing the script async function waitExec(site, waitElapsed = 0) { if (waitElapsed >= (site.waitMax || 5000)) { preparePage(site, await site.run()); return; } if (site.waitAttr !== undefined) { const wait = document.querySelector(site.waitAttr[0])?.getAttribute(site.waitAttr[1]); if (isNothing(wait)) { logScript(`Waiting for Attribute ${site.waitAttr[1]} of ${site.waitAttr[0]} = ${wait}`); setTimeout(() => { waitExec(site, waitElapsed + (site.waitStep || 1000)); }, site.waitStep || 1000); return; } logScript(`Found Attribute ${site.waitAttr[1]} of ${site.waitAttr[0]} = ${wait}`); } if (site.waitEle !== undefined) { const wait = document.querySelector(site.waitEle); if (isNothing(wait?.tagName)) { logScript(`Waiting for Element ${site.waitEle} = `, wait); setTimeout(() => { waitExec(site, waitElapsed + (site.waitStep || 1000)); }, site.waitStep || 1000); return; } logScript(`Found Element ${site.waitEle} = `, wait); } if (site.waitVar !== undefined) { const W = typeof unsafeWindow !== 'undefined' ? unsafeWindow : window; const wait = W[site.waitVar]; if (isNothing(wait)) { logScript(`Waiting for Variable ${site.waitVar} = ${wait}`); setTimeout(() => { waitExec(site, waitElapsed + (site.waitStep || 1000)); }, site.waitStep || 1000); return; } logScript(`Found Variable ${site.waitVar} = ${wait}`); } preparePage(site, await site.run()); } // Script Entry point function start(sites) { logScript(`Starting ${getInfoGM.script.name} ${getInfoGM.script.version} on ${getBrowser()} with ${getEngine()}`); // window.InfoGM = getInfoGM; logScript(`${sites.length} Known Manga Sites, Looking for a match...`); const site = sites.find((s) => s.url.test(window.location.href)); if (site) { logScript(`Found site: ${site.name}`); waitExec(site); } else { logScript(`Sorry, didnt find any valid site`); } } start(sites); })();