// ==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.02
// @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/jquery/3.6.0/jquery.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.9.1/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.10/sweetalert2.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/color-scheme/1.0.1/color-scheme.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/jquery.imagesloaded/5.0.0/imagesloaded.pkgd.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\/.+/
// ==/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)));
}
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;
display: none;
z-index: 1000;
}
#MangaOnlineViewer #ViewerShortcuts {
padding: 8px;
position: fixed;
top: 65px;
left: 0;
}
#MangaOnlineViewer #ViewerControls .ControlLabel {
display: list-item;
list-style: none;
}
#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;
z
}
#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 .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;
position: relative;
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,
#MangaOnlineViewer #Header.visible{
transform: translateY(-1%);
}
#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.2;
}
#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;
}
#MangaOnlineViewer #ViewerControls span.DefaultZoom,
#MangaOnlineViewer #ViewerControls span.viewMode,
#MangaOnlineViewer #ViewerControls span.fitIfOversize,
#MangaOnlineViewer #ViewerControls span.showThumbnails,
#MangaOnlineViewer #ViewerControls span.lazyLoadImages,
#MangaOnlineViewer #ViewerControls span.downloadZip {
display: none;
}
#MangaOnlineViewer #ViewerControls {
padding: 8px;
position: fixed;
top: 0;
left: 45px;
width: auto;
}
#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 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;
margin-top: 0.5em;
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;
}
`;
const externalScripts = [
'<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.0/jquery.min.js" integrity="sha512-894YE6QWD5I59HgZOGReFYm4dnWc1Qt5NtvYSaNcOP+u1T9qYdvdihz0PPSiiqn/+/3e7Jo4EaG7TubfWGUrMQ==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>',
'<script src="https://cdnjs.cloudflare.com/ajax/libs/jszip/3.9.1/jszip.min.js" integrity="sha512-amNoSoOK3jIKx6qlDrv36P4M/h7vc6CHwiBU3XG9/1LW0ZSNe8E3iZL1tPG/VnfCrVrZc2Zv47FIJ7fyDX4DMA==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>',
'<script src="https://cdnjs.cloudflare.com/ajax/libs/nprogress/0.2.0/nprogress.min.js" integrity="sha256-XWzSUJ+FIQ38dqC06/48sNRwU1Qh3/afjmJ080SneA8=" crossorigin="anonymous"></script>',
'<script src="https://cdnjs.cloudflare.com/ajax/libs/limonte-sweetalert2/11.4.10/sweetalert2.min.js" integrity="sha512-OgIASmUioEN3o3gYIAxYUeW4G5FWdhORLq0p7UhTM6Xrm5XGY4IrSDM3uYTCNh4ik4V8BX0NaX9/Z/VMqigCzg==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>',
'<script src="https://cdnjs.cloudflare.com/ajax/libs/color-scheme/1.0.1/color-scheme.min.js" integrity="sha256-7IUC8vhyoPLh1tuQJnffPB5VO6HpR4VWK4Y1ciOOoBY=" crossorigin="anonymous"></script>',
'<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery.imagesloaded/5.0.0/imagesloaded.pkgd.min.js" integrity="sha512-kfs3Dt9u9YcOiIt4rNcPUzdyNNO9sVGQPiZsub7ywg6lRW5KuK1m145ImrFHe3LMWXHndoKo2YRXWy8rnOcSKg==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>',
];
externalScripts.map((script) => {
const find = script.match(/src="(.+?)"/);
return find ? find[1] : '';
});
const externalCSS = [
'<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/normalize/8.0.1/normalize.min.css" integrity="sha256-l85OmPOjvil/SOvVt3HnSSjzF1TUMyT9eV0c2BzEGzU=" crossorigin="anonymous" />',
'<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/nprogress/0.2.0/nprogress.min.css" integrity="sha256-pMhcV6/TBDtqH9E9PWKgS+P32PVguLG8IipkPyqMtfY=" crossorigin="anonymous" />',
'<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@gerhobbelt/[email protected]/keys.css" integrity="sha256-a/1ebfXeoX0xLUcQCJLQsm6APQNBwrm03/XFcvW7xAI=" crossorigin="anonymous">',
'<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/limonte-sweetalert2/11.4.10/sweetalert2.min.css" integrity="sha512-R9EM3xazxo9xyo8pz3IMTw7SIO1qKnG1Vs3yJnVApYhqDwemdSJJbcd5KROicK87kRiFksOhhtW/s2c4Mdjrwg==" crossorigin="anonymous" referrerpolicy="no-referrer" />',
];
externalCSS.map((script) => {
const find = script.match(/href="(.+?)"/);
return find ? find[1] : '';
});
// Configuration
const settings$1 = {
configVersion: 0,
throttlePageLoad: 0,
widthScale: 0,
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),
timer: 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),
};
// 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 `<style type='text/css' name='${theme[0]}'>
.${theme[0]} .ControlLabel, .${theme[0]} .ViewerTitle, .${theme[0]}, .PageFunctions a.visible, .${theme[0]} a, .${theme[0]} a:link, .${theme[0]} a:visited, .${theme[0]} a:active, .${theme[0]} a:focus{ text-decoration:none; color: ${theme[2]};}
.${theme[0]} {background-repeat: repeat;background-position: 0 0;background-image: none;background-color: ${theme[1]};background-attachment: scroll;}
.${theme[0]} #ImageOptions #menu .menuOuterArrow {border-left-width: 10px;border-left-style: solid;border-left-color: ${theme[4]};}
.${theme[0]} #ImageOptions #menu .menuInnerArrow {border-left-width: 5px;border-left-style: solid;border-left-color: ${theme[1]};}
.${theme[0]} .PageFunctions { border: 1px solid ${theme[3]}; border-bottom: medium none; border-left: medium none; border-right: medium none;}
/*.${theme[0]} #Chapter { border: 1px solid ${theme[3]}; border-top: medium none; border-left: medium none; border-right: medium none;}*/
.${theme[0]} .PageFunctions > span, .${theme[0]} .Thumbnail span {background: none repeat scroll 0 0 ${theme[4]};}
.${theme[0]} .panel {background: none repeat scroll 0 0 ${theme[4]}; border: thin solid ${theme[3]};}
.${theme[0]} .PageContent, .${theme[0]} .Thumbnail img { outline: 2px solid ${theme[3]}; background: none repeat scroll 0 0 ${theme[4]};}
.${theme[0]} .ChapterControl a { border: 1px solid ${theme[3]}; background-color: ${theme[5]};
</style>`;
}
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) => `<option value='${theme[0]}' ${settings$1.theme === theme[0] ? 'selected' : ''}>${theme[0].replace('_', ' ')}</option>`);
const themesCSS = themes.map(addTheme).join('');
// Inject CSS for this script
const readerCSS = `<style type='text/css'>
${cssStyles}
#MangaOnlineViewer .PageFunctions .Bookmark {background: url('${icon.bookmark}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .Reload {background: url('${icon.reload}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .Hide {background: url('${icon.hide}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .ZoomIn {background: url('${icon.zoomIn}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .ZoomOut {background: url('${icon.zoomOut}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .ZoomRestore {background: url('${icon.zoomRestore}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .ZoomWidth {background: url('${icon.zoomWidth}') no-repeat scroll center center transparent;}
#MangaOnlineViewer .PageFunctions .ZoomHeight {background: url('${icon.zoomWidth}') no-repeat scroll center center transparent;}
</style>`;
function head(manga) {
return `
<head>
<title>${manga.title}</title>
<meta charset='UTF-8'>
${externalScripts.join('\n')}
${externalCSS.join('\n')}
${readerCSS}
${themesCSS}
</head>
`;
}
// Add Pages Place-holders
const listPages = (times) => Array(times)
.fill(null)
.map((_, index) => `
<div id='Page${index + 1}' class='MangaPage'>
<div class='PageFunctions'>
<button class='Bookmark ControlButton' title='Bookmark'></button>
<button class='ZoomIn ControlButton' title='Zoom In'></button>
<button class='ZoomRestore ControlButton' title='Zoom Restore'></button>
<button class='ZoomOut ControlButton' title='Zoom Out'></button>
<button class='ZoomWidth ControlButton' title='Zoom to Width'></button>
<button class='ZoomHeight ControlButton' title='Zoom to Height'></button>
<button class='Hide ControlButton' title='Hide'></button>
<button class='Reload ControlButton' title='Reload'></button>
<span class='PageIndex'>${index + 1}</span>
</div>
<div class='PageContent'>
<img id='PageImg${index + 1}' alt='PageImg${index + 1}' class='PageImg' />
</div>
</div>`);
const imageOptions = `<div id='ImageOptions'>
<div id='menu'>
<span class='menuOuterArrow'><span class='menuInnerArrow'></span></span>
</div>
<div class='panel'>
<img id='enlarge' alt='Enlarge' title='Enlarge' src='${icon.enlarge}' class='ControlButton' />
<img id='restore' alt='Restore' title='Restore' src='${icon.restore}' class='ControlButton' />
<img id='reduce' alt='Reduce' title='Reduce' src='${icon.reduce}' class='ControlButton' />
<img id='fitWidth' alt='Fit Width' title='Fit Width' src='${icon.fitWidth}' class='ControlButton' />
<img id='fitHeight' alt='Fit Height' title='Fit Height' src='${icon.fitHeight}' class='ControlButton' />
<img id='webComic' alt='Web Comic Mode' title='Web Comic Mode' src='${icon.webComic}' class='ControlButton' />
<img id='ltrMode' alt='Left to Right Mode' title='Left to Right Mode' src='${icon.pictureLeft}' class='ControlButton'/>
<img id='verticalMode' alt='Vertical Mode' title='Vertical Mode' src='${icon.pictureDown}' class='ControlButton'/>
<img id='rtlMode' alt='Right to Left Mode' title='Right to Left Mode' src='${icon.pictureRight}' class='ControlButton'/>
<img id='pageControls' alt='Toggle Page Controls' title='Toggle Page Controls' src='${icon.controls}' class='ControlButton'/>
<img id='settings' alt='settings' title='settings' src='${icon.settings}' class='ControlButton' />
</div>
<div id='Zoom' class='ControlLabel'>Zoom: <b>${settings$1.zoom}</b> %</div>
</div>`;
const controls$1 = `<div id='ViewerControls' class='panel'>
<span class='ControlLabel ThemeSelector'>Theme:
<select id='ThemeSelector'>
${themesSelector}
</select>
<span class='CustomTheme' ${settings$1.theme !== 'Custom_Dark' && settings$1.theme !== 'Custom_Light'
? 'style="display: none;"'
: ''}><br/>-Base:<input id='CustomThemeHue' type='color' value='${settings$1.customTheme}' class='colorpicker CustomTheme'></span>
<span class='FullCustom' ${settings$1.theme !== 'Full_Custom' ? 'style="display: none;"' : ''}><br/>-Body:<input id='CustomThemeHueBody' type='color' value='${settings$1.customThemeBody}' class='colorpicker FullCustom'></span>
<span class='FullCustom' ${settings$1.theme !== 'Full_Custom' ? 'style="display: none;"' : ''}><br/>-Text:<input id='CustomThemeHueText' type='color' value=${settings$1.customThemeText}' class='colorpicker FullCustom'></span>
<span class='FullCustom' ${settings$1.theme !== 'Full_Custom' ? 'style="display: none;"' : ''}><br/>-Lines:<input id='CustomThemeHueLines' type='color' value='${settings$1.customThemeLines}' class='colorpicker FullCustom'></span>
<span class='FullCustom' ${settings$1.theme !== 'Full_Custom' ? 'style="display: none;"' : ''}><br/>-Painels:<input id='CustomThemeHuePanel' type='color' value='${settings$1.customThemePanel}' class='colorpicker FullCustom'></span>
<span class='FullCustom' ${settings$1.theme !== 'Full_Custom' ? 'style="display: none;"' : ''}><br/>-Buttons:<input id='CustomThemeHueButton' type='color' value='${settings$1.customThemeButton}' class='colorpicker FullCustom'></span>
</span>
<span class='ControlLabel loadMode'>Default Load Mode:
<select id='loadMode'>
<option value='wait' ${settings$1.loadMode === 'wait' ? 'selected' : ''}>Normal(Wait 3 sec)</option>
<option value='always' ${settings$1.loadMode === 'always' ? 'selected' : ''}>Always(Immediately)</option>
<option value='never' ${settings$1.loadMode === 'never' ? 'selected' : ''}>Never(Manually)</option>
</select>
</span>
<span class='ControlLabel PagesPerSecond'>Pages/Second:
<select id='PagesPerSecond'>
<option value='3000' ${settings$1.timer === 3000 ? 'selected' : ''}>0.3(Slow)</option>
<option value='2000' ${settings$1.timer === 2000 ? 'selected' : ''}>0.5</option>
<option value='1000' ${settings$1.timer === 1000 ? 'selected' : ''}>01(Normal)</option>
<option value='500' ${settings$1.timer === 500 ? 'selected' : ''}>02</option>
<option value='250' ${settings$1.timer === 250 ? 'selected' : ''}>04(Fast)</option>
<option value='125' ${settings$1.timer === 125 ? 'selected' : ''}>08</option>
<option value='100' ${settings$1.timer === 100 ? 'selected' : ''}>10(Extreme)</option>
</select>
</span>
<span class='ControlLabel DefaultZoom'>Default Zoom:
<select id='DefaultZoom'>
<option value='50' ${settings$1.zoom === 50 ? 'selected' : ''}>50%</option>
<option value='75' ${settings$1.zoom === 75 ? 'selected' : ''}>75%</option>
<option value='100' ${settings$1.zoom === 100 ? 'selected' : ''}>100%</option>
<option value='125' ${settings$1.zoom === 125 ? 'selected' : ''}>125%</option>
<option value='150' ${settings$1.zoom === 150 ? 'selected' : ''}>150%</option>
<option value='175' ${settings$1.zoom === 175 ? 'selected' : ''}>175%</option>
<option value='200' ${settings$1.zoom === 200 ? 'selected' : ''}>200%</option>
<option value='1000' ${settings$1.zoom === 1000 ? 'selected' : ''}>Fit Width</option>
<option value='-1000' ${settings$1.zoom === -1000 ? 'selected' : ''}>Fit Height</option>
</select>
</span>
<span class='ControlLabel zoomStep'>Zoom Change Step (between 5 and 50): <br/>
<input type='range' value='${settings$1.zoomStep}' name='zoomStep' id='zoomStep' min='5' max='50' step='5' oninput='zoomStepVal.value = this.value'>
<output id='zoomStepVal'>${settings$1.zoomStep}</output>
</span>
<span class='ControlLabel viewMode'>Default View Mode:
<select id='viewMode'>
<option value='' ${settings$1.viewMode === '' ? 'selected' : ''}>Vertical</option>
<option value='WebComic' ${settings$1.viewMode === 'WebComic' ? 'selected' : ''}>WebComic</option>
<option value='FluidLTR' ${settings$1.viewMode === 'FluidLTR' ? 'selected' : ''}>Left to Right</option>
<option value='FluidRTL' ${settings$1.viewMode === 'FluidRTL' ? 'selected' : ''}>Right to Left</option>
</select>
</span>
<span class='ControlLabel fitIfOversize'>Fit Width if Oversize:
<input type='checkbox' value='true' name='fitIfOversize' id='fitIfOversize' ${settings$1.fitWidthIfOversize ? 'checked' : ''}>
</span>
<span class='ControlLabel showThumbnails'>Show Thumbnails:
<input type='checkbox' value='true' name='showThumbnails' id='showThumbnails' ${settings$1.showThumbnails ? 'checked' : ''}>
</span>
<span class='ControlLabel lazyLoadImages'>Lazy Load Images:
<input type='checkbox' value='true' name='lazyLoadImages' id='lazyLoadImages' ${settings$1.lazyLoadImages ? 'checked' : ''}>
</span>
<span class='ControlLabel lazyStart'>Lazy Start From Page (between 5 and 100):<br/>
<input type='range' value='${settings$1.lazyStart}' name='lazyStart' id='lazyStart' min='5' max='100' step='5' oninput='lazyStartVal.value = this.value'>
<output id='lazyStartVal'>${settings$1.lazyStart}</output>
</span>
<span class='ControlLabel downloadZip'>Download Images as Zip Automatically:
<input type='checkbox' value='false' name='downloadZip' id='downloadZip' ${settings$1.downloadZip ? 'checked' : ''}>
</span>
<span class='ControlLabel hidePageControls'>Always Hide Page Controls:
<input type='checkbox' value='false' name='hidePageControls' id='hidePageControls' ${settings$1.hidePageControls ? 'checked' : ''}>
</span>
</div>`;
const keybindings = `<div id='ViewerShortcuts' class='panel' style='display: none;'>
<kbd class='dark'>Numpad 5</kbd>/<kbd class='dark'>/</kbd>: Open Settings<br/>
<kbd class='dark'>Numpad +</kbd>/<kbd class='dark'>=</kbd>: Global Zoom in pages (enlarge)<br/>
<kbd class='dark'>Numpad -</kbd>/<kbd class='dark'>-</kbd>: Global Zoom out pages (reduce)<br/>
<kbd class='dark'>Numpad /</kbd>/<kbd class='dark'>9</kbd>: Global Restore pages to original<br/>
<kbd class='dark'>Numpad *</kbd>/<kbd class='dark'>0</kbd>: Global Fit window width<br/>
<kbd class='dark'>V</kbd>: Vertical Mode<br/>
<kbd class='dark'>C</kbd>: WebComic Mode<br/>
<kbd class='dark'>N</kbd>: Right to Left Mode<br/>
<kbd class='dark'>B</kbd>: Left to Right Mode<br/>
<kbd class='dark'>→</kbd>/<kbd class='dark'>D</kbd>/<kbd class='dark'>Numpad 6</kbd>/<kbd class='dark'>.</kbd> : Next Chapter<br/>
<kbd class='dark'>←</kbd>/<kbd class='dark'>A</kbd>/<kbd class='dark'>Numpad 4</kbd>/<kbd class='dark'>,</kbd> : Previous Chapter<br/>
<kbd class='dark'>↑</kbd>/<kbd class='dark'>W</kbd>/<kbd class='dark'>Numpad 8</kbd>: Scroll Up<br/>
<kbd class='dark'>↓</kbd>/<kbd class='dark'>S</kbd>/<kbd class='dark'>Numpad 2</kbd>: Scroll Down<br/>
</div>`;
const listThumbnails = (times) => Array(times)
.fill(null)
.map((_, index) => `
<div id='Thumbnail${index + 1}' class='Thumbnail'>
<img id='ThumbnailImg${index + 1}' alt='ThumbnailImg${index + 1}' class='ThumbnailImg' src=''/>
<span class='ThumbnailIndex'>${index + 1}</span>
</div>`);
const listOptions = (times) => Array(times)
.fill(null)
.map((_, index) => `<option value='${index + 1}'>${index + 1}</option>`);
const body = (manga, begin = 0) => `
<div id='MangaOnlineViewer'
class='${settings$1.theme} ${isMobile ? 'mobile' : ''} ${settings$1.hidePageControls ? 'hideControls' : ''}'>
<header id="Header">
<aside id='GlobalControls'>
${imageOptions}
${controls$1}
${keybindings}
</aside>
<div class='ViewerTitle'>
<h1 id='MangaTitle'>${manga.title}</h1>
<a id='series' href='${manga.series}'>(Return to Chapter List)</a>
</div>
<nav id='ChapterNavigation'>
<div id='Counters' class='ControlLabel'>
<i>0</i> of <b>${manga.pages}</b> Pages Loaded
<span class='ControlLabel'>Go to Page:</span>
<select id='gotoPage'>
<option selected>#</option>
${listOptions(manga.pages).slice(begin).join('')}
</select>
</div>
<div id='ChapterControl' class='ChapterControl'>
<a href='#' class='download NavigationControlButton ControlButton'>
Download
</a>
<a class='prev NavigationControlButton ControlButton' id='prev' href='${manga.prev || ''}'>
Previous
</a>
<a class='next NavigationControlButton ControlButton' id='next' href='${manga.next || ''}'>
Next
</a>
</div>
</nav>
</header>
<main id='Chapter' class='${settings$1.fitWidthIfOversize === true ? 'fitWidthIfOversize' : ''} ${settings$1.viewMode}'>
${listPages(manga.pages).slice(begin).join('')}
</main>
<nav id='Navigation' class='panel ${settings$1.showThumbnails ? '' : 'disabled'}'>
<div id='NavigationCounters' class='ControlLabel'>
<img alt='Thumbnails' title='Thumbnails' src='${icon.menu}' class='nav' />
<i>0</i> of <b>${manga.pages}</b> Pages Loaded
</div>
<div id='Thumbnails'>
${listThumbnails(manga.pages).slice(begin).join('')}
</div>
</nav>
<a href='#' id='blob' style='display: none;'>Download</a>
</div>`;
var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};
var FileSaver_min = {exports: {}};
(function (module, exports) {
(function(a,b){b();})(commonjsGlobal,function(){function b(a,b){return "undefined"==typeof b?b={autoBom:!1}:"object"!=typeof b&&(console.warn("Deprecated: Expected third argument to be a object"),b={autoBom:!b}),b.autoBom&&/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob(["\uFEFF",a],{type:a.type}):a}function c(a,b,c){var d=new XMLHttpRequest;d.open("GET",a),d.responseType="blob",d.onload=function(){g(d.response,b,c);},d.onerror=function(){console.error("could not download file");},d.send();}function d(a){var b=new XMLHttpRequest;b.open("HEAD",a,!1);try{b.send();}catch(a){}return 200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent("click"));}catch(c){var b=document.createEvent("MouseEvents");b.initMouseEvent("click",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b);}}var f="object"==typeof window&&window.window===window?window:"object"==typeof self&&self.self===self?self:"object"==typeof commonjsGlobal&&commonjsGlobal.global===commonjsGlobal?commonjsGlobal:void 0,a=f.navigator&&/Macintosh/.test(navigator.userAgent)&&/AppleWebKit/.test(navigator.userAgent)&&!/Safari/.test(navigator.userAgent),g=f.saveAs||("object"!=typeof window||window!==f?function(){}:"download"in HTMLAnchorElement.prototype&&!a?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement("a");g=g||b.name||"download",j.download=g,j.rel="noopener","string"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target="_blank")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href);},4E4),setTimeout(function(){e(j);},0));}:"msSaveOrOpenBlob"in navigator?function(f,g,h){if(g=g||f.name||"download","string"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else {var i=document.createElement("a");i.href=f,i.target="_blank",setTimeout(function(){e(i);});}}:function(b,d,e,g){if(g=g||open("","_blank"),g&&(g.document.title=g.document.body.innerText="downloading..."),"string"==typeof b)return c(b,d,e);var h="application/octet-stream"===b.type,i=/constructor/i.test(f.HTMLElement)||f.safari,j=/CriOS\/[\d]+/.test(navigator.userAgent);if((j||h&&i||a)&&"undefined"!=typeof FileReader){var k=new FileReader;k.onloadend=function(){var a=k.result;a=j?a:a.replace(/^data:[^;]*;/,"data:attachment/file;"),g?g.location.href=a:location=a,g=null;},k.readAsDataURL(b);}else {var l=f.URL||f.webkitURL,m=l.createObjectURL(b);g?g.location=m:location.href=m,g=null,setTimeout(function(){l.revokeObjectURL(m);},4E4);}});f.saveAs=g.saveAs=g,(module.exports=g);});
} (FileSaver_min));
const cache = {
zip: new JSZip(),
downloadFiles: 0,
Data: {},
};
const getExtension = (mimeType) => ((/image\/(?<ext>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 = /^(?<name>.*?)(?<index>\d+)\.(?<ext>\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:(?<mimeType>image\/\w+);base64,+(?<data>.+)/.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_min.exports.saveAs(content, zipName);
});
}
catch (e) {
logScript(e);
}
}
}
// Todo: Test Download
var imagesloaded = {exports: {}};
var evEmitter = {exports: {}};
var hasRequiredEvEmitter;
function requireEvEmitter () {
if (hasRequiredEvEmitter) return evEmitter.exports;
hasRequiredEvEmitter = 1;
(function (module) {
( function( global, factory ) {
if ( module.exports ) {
module.exports = factory();
} else {
global.EvEmitter = factory();
}
}( typeof window != 'undefined' ? window : commonjsGlobal, function() {
function EvEmitter() {}
let proto = EvEmitter.prototype;
proto.on = function( eventName, listener ) {
if ( !eventName || !listener ) return this;
let events = this._events = this._events || {};
let listeners = events[ eventName ] = events[ eventName ] || [];
if ( !listeners.includes( listener ) ) {
listeners.push( listener );
}
return this;
};
proto.once = function( eventName, listener ) {
if ( !eventName || !listener ) return this;
this.on( eventName, listener );
let onceEvents = this._onceEvents = this._onceEvents || {};
let onceListeners = onceEvents[ eventName ] = onceEvents[ eventName ] || {};
onceListeners[ listener ] = true;
return this;
};
proto.off = function( eventName, listener ) {
let listeners = this._events && this._events[ eventName ];
if ( !listeners || !listeners.length ) return this;
let index = listeners.indexOf( listener );
if ( index != -1 ) {
listeners.splice( index, 1 );
}
return this;
};
proto.emitEvent = function( eventName, args ) {
let listeners = this._events && this._events[ eventName ];
if ( !listeners || !listeners.length ) return this;
listeners = listeners.slice( 0 );
args = args || [];
let onceListeners = this._onceEvents && this._onceEvents[ eventName ];
for ( let listener of listeners ) {
let isOnce = onceListeners && onceListeners[ listener ];
if ( isOnce ) {
this.off( eventName, listener );
delete onceListeners[ listener ];
}
listener.apply( this, args );
}
return this;
};
proto.allOff = function() {
delete this._events;
delete this._onceEvents;
return this;
};
return EvEmitter;
} ) );
} (evEmitter));
return evEmitter.exports;
}
(function (module) {
( function( window, factory ) {
if ( module.exports ) {
module.exports = factory( window, requireEvEmitter() );
} else {
window.imagesLoaded = factory( window, window.EvEmitter );
}
} )( typeof window !== 'undefined' ? window : commonjsGlobal,
function factory( window, EvEmitter ) {
let $ = window.jQuery;
let console = window.console;
function makeArray( obj ) {
if ( Array.isArray( obj ) ) return obj;
let isArrayLike = typeof obj == 'object' && typeof obj.length == 'number';
if ( isArrayLike ) return [ ...obj ];
return [ obj ];
}
function ImagesLoaded( elem, options, onAlways ) {
if ( !( this instanceof ImagesLoaded ) ) {
return new ImagesLoaded( elem, options, onAlways );
}
let queryElem = elem;
if ( typeof elem == 'string' ) {
queryElem = document.querySelectorAll( elem );
}
if ( !queryElem ) {
console.error(`Bad element for imagesLoaded ${queryElem || elem}`);
return;
}
this.elements = makeArray( queryElem );
this.options = {};
if ( typeof options == 'function' ) {
onAlways = options;
} else {
Object.assign( this.options, options );
}
if ( onAlways ) this.on( 'always', onAlways );
this.getImages();
if ( $ ) this.jqDeferred = new $.Deferred();
setTimeout( this.check.bind( this ) );
}
ImagesLoaded.prototype = Object.create( EvEmitter.prototype );
ImagesLoaded.prototype.getImages = function() {
this.images = [];
this.elements.forEach( this.addElementImages, this );
};
const elementNodeTypes = [ 1, 9, 11 ];
ImagesLoaded.prototype.addElementImages = function( elem ) {
if ( elem.nodeName === 'IMG' ) {
this.addImage( elem );
}
if ( this.options.background === true ) {
this.addElementBackgroundImages( elem );
}
let { nodeType } = elem;
if ( !nodeType || !elementNodeTypes.includes( nodeType ) ) return;
let childImgs = elem.querySelectorAll('img');
for ( let img of childImgs ) {
this.addImage( img );
}
if ( typeof this.options.background == 'string' ) {
let children = elem.querySelectorAll( this.options.background );
for ( let child of children ) {
this.addElementBackgroundImages( child );
}
}
};
const reURL = /url\((['"])?(.*?)\1\)/gi;
ImagesLoaded.prototype.addElementBackgroundImages = function( elem ) {
let style = getComputedStyle( elem );
if ( !style ) return;
let matches = reURL.exec( style.backgroundImage );
while ( matches !== null ) {
let url = matches && matches[2];
if ( url ) {
this.addBackground( url, elem );
}
matches = reURL.exec( style.backgroundImage );
}
};
ImagesLoaded.prototype.addImage = function( img ) {
let loadingImage = new LoadingImage( img );
this.images.push( loadingImage );
};
ImagesLoaded.prototype.addBackground = function( url, elem ) {
let background = new Background( url, elem );
this.images.push( background );
};
ImagesLoaded.prototype.check = function() {
this.progressedCount = 0;
this.hasAnyBroken = false;
if ( !this.images.length ) {
this.complete();
return;
}
let onProgress = ( image, elem, message ) => {
setTimeout( () => {
this.progress( image, elem, message );
} );
};
this.images.forEach( function( loadingImage ) {
loadingImage.once( 'progress', onProgress );
loadingImage.check();
} );
};
ImagesLoaded.prototype.progress = function( image, elem, message ) {
this.progressedCount++;
this.hasAnyBroken = this.hasAnyBroken || !image.isLoaded;
this.emitEvent( 'progress', [ this, image, elem ] );
if ( this.jqDeferred && this.jqDeferred.notify ) {
this.jqDeferred.notify( this, image );
}
if ( this.progressedCount === this.images.length ) {
this.complete();
}
if ( this.options.debug && console ) {
console.log( `progress: ${message}`, image, elem );
}
};
ImagesLoaded.prototype.complete = function() {
let eventName = this.hasAnyBroken ? 'fail' : 'done';
this.isComplete = true;
this.emitEvent( eventName, [ this ] );
this.emitEvent( 'always', [ this ] );
if ( this.jqDeferred ) {
let jqMethod = this.hasAnyBroken ? 'reject' : 'resolve';
this.jqDeferred[ jqMethod ]( this );
}
};
function LoadingImage( img ) {
this.img = img;
}
LoadingImage.prototype = Object.create( EvEmitter.prototype );
LoadingImage.prototype.check = function() {
let isComplete = this.getIsImageComplete();
if ( isComplete ) {
this.confirm( this.img.naturalWidth !== 0, 'naturalWidth' );
return;
}
this.proxyImage = new Image();
if ( this.img.crossOrigin ) {
this.proxyImage.crossOrigin = this.img.crossOrigin;
}
this.proxyImage.addEventListener( 'load', this );
this.proxyImage.addEventListener( 'error', this );
this.img.addEventListener( 'load', this );
this.img.addEventListener( 'error', this );
this.proxyImage.src = this.img.currentSrc || this.img.src;
};
LoadingImage.prototype.getIsImageComplete = function() {
return this.img.complete && this.img.naturalWidth;
};
LoadingImage.prototype.confirm = function( isLoaded, message ) {
this.isLoaded = isLoaded;
let { parentNode } = this.img;
let elem = parentNode.nodeName === 'PICTURE' ? parentNode : this.img;
this.emitEvent( 'progress', [ this, elem, message ] );
};
LoadingImage.prototype.handleEvent = function( event ) {
let method = 'on' + event.type;
if ( this[ method ] ) {
this[ method ]( event );
}
};
LoadingImage.prototype.onload = function() {
this.confirm( true, 'onload' );
this.unbindEvents();
};
LoadingImage.prototype.onerror = function() {
this.confirm( false, 'onerror' );
this.unbindEvents();
};
LoadingImage.prototype.unbindEvents = function() {
this.proxyImage.removeEventListener( 'load', this );
this.proxyImage.removeEventListener( 'error', this );
this.img.removeEventListener( 'load', this );
this.img.removeEventListener( 'error', this );
};
function Background( url, element ) {
this.url = url;
this.element = element;
this.img = new Image();
}
Background.prototype = Object.create( LoadingImage.prototype );
Background.prototype.check = function() {
this.img.addEventListener( 'load', this );
this.img.addEventListener( 'error', this );
this.img.src = this.url;
let isComplete = this.getIsImageComplete();
if ( isComplete ) {
this.confirm( this.img.naturalWidth !== 0, 'naturalWidth' );
this.unbindEvents();
}
};
Background.prototype.unbindEvents = function() {
this.img.removeEventListener( 'load', this );
this.img.removeEventListener( 'error', this );
};
Background.prototype.confirm = function( isLoaded, message ) {
this.isLoaded = isLoaded;
this.emitEvent( 'progress', [ this, this.element, message ] );
};
ImagesLoaded.makeJQueryPlugin = function( jQuery ) {
jQuery = jQuery || window.jQuery;
if ( !jQuery ) return;
$ = jQuery;
$.fn.imagesLoaded = function( options, onAlways ) {
let instance = new ImagesLoaded( this, options, onAlways );
return instance.jqDeferred.promise( $( this ) );
};
};
ImagesLoaded.makeJQueryPlugin();
return ImagesLoaded;
} );
} (imagesloaded));
var imagesLoaded = imagesloaded.exports;
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 img').length;
const loaded = document.querySelectorAll('.PageContent img.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.timer) {
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.timer) * (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.timer) * (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.timer || '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.timer,
});
}
}
// Goto Page and Thumbnails
function scrollToElement(ele) {
$(window)
.scrollTop(ele.offset()?.top || 0)
.scrollLeft(ele.offset()?.left || 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.originalEvent?.code;
if (!e.originalEvent?.ctrlKey &&
!e.originalEvent?.altKey &&
!e.originalEvent?.shiftKey &&
!e.originalEvent?.metaKey &&
$.inArray(a, [
'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',
]) !== -1) {
e.preventDefault();
e.stopPropagation();
e.stopImmediatePropagation();
switch (a) {
case 'ArrowUp':
case 'KeyW':
case 'Numpad8':
if (settings$1.zoom === -1000) {
const next = $('.MangaPage')
.get()
.map((item) => $(item).offset().top - $(window).scrollTop())
.findIndex((element) => element > 10);
scrollToElement($('.MangaPage').eq(next - 2));
}
else {
window.scrollBy({
top: -$(window).height() / 2,
behavior: 'smooth',
});
}
break;
case 'ArrowDown':
case 'KeyS':
case 'Numpad2':
if (settings$1.zoom === -1000) {
const next = $('.MangaPage')
.get()
.map((item) => $(item).offset().top - $(window).scrollTop())
.findIndex((element) => element > 10);
scrollToElement($('.MangaPage').eq(next));
}
else {
window.scrollBy({
top: $(window).height() / 2,
behavior: 'smooth',
});
}
break;
case 'ArrowRight':
case 'Period':
case 'KeyD':
case 'Numpad6':
$('.ChapterControl:first .next').trigger('click');
break;
case 'ArrowLeft':
case 'Comma':
case 'KeyA':
case 'Numpad4':
$('.ChapterControl:first .prev').trigger('click');
break;
case 'Equal':
case 'NumpadAdd':
case 'KeyE':
$('#enlarge').trigger('click');
break;
case 'Minus':
case 'NumpadSubtract':
case 'KeyQ':
$('#reduce').trigger('click');
break;
case 'Digit9':
case 'NumpadDivide':
case 'KeyR':
$('#restore').trigger('click');
break;
case 'Digit0':
case 'NumpadMultiply':
case 'KeyF':
$('#fitWidth').trigger('click');
break;
case 'Slash':
case 'Numpad5':
case 'KeyX':
$('#settings').trigger('click');
break;
case 'KeyC':
$('#webComic').trigger('click');
break;
case 'KeyV':
$('#verticalMode').trigger('click');
break;
case 'KeyN':
$('#rtlMode').trigger('click');
break;
case 'KeyB':
$('#ltrMode').trigger('click');
break;
}
return false;
}
return true;
}
$(document).on('keydown', processKey);
}
// Controls for the extra features added to the sites
function controls() {
// Size Controls
$('#enlarge').on('click', () => {
settings$1.zoom += settings$1.zoomStep;
$('#Zoom b').html(settings$1.zoom.toString());
applyZoom();
});
$('#reduce').on('click', () => {
settings$1.zoom -= settings$1.zoomStep;
$('#Zoom b').html(settings$1.zoom.toString());
applyZoom();
});
$('#restore').on('click', () => {
settings$1.zoom = 100;
$('#Zoom b').html(settings$1.zoom.toString());
applyZoom();
});
$('#fitWidth').on('click', () => {
settings$1.zoom = 1000;
$('#Zoom b').html(settings$1.zoom.toString());
applyZoom();
});
$('#fitHeight').on('click', () => {
settings$1.zoom = -1000;
$('#Zoom b').html(settings$1.zoom.toString());
applyZoom();
});
$('#zoomStep').on('change', (event) => {
const step = $(event.target).val();
setValueGM('ZoomStep', parseInt(step, 10));
logScript(`zoomStep: ${getValueGM('ZoomStep')}`);
});
// WebComic View Mode
$('#webComic').on('click', () => {
$('#Chapter').addClass('WebComic').removeClass('FluidLTR').removeClass('FluidRTL');
applyZoom();
});
// Fluid LTR View Mode
$('#ltrMode').on('click', () => {
$('#Chapter').removeClass('WebComic').addClass('FluidLTR').removeClass('FluidRTL');
applyZoom();
});
// Fluid RTL View Mode
$('#rtlMode').on('click', () => {
$('#Chapter').removeClass('WebComic').removeClass('FluidLTR').addClass('FluidRTL');
applyZoom();
});
// Vertical View Mode
$('#verticalMode').on('click', () => {
$('#Chapter').removeClass('WebComic').removeClass('FluidLTR').removeClass('FluidRTL');
applyZoom();
});
$('#fitIfOversize').on('change', (event) => {
$('#Chapter').toggleClass('fitWidthIfOversize');
if ($(event.target).is(':checked')) {
setValueGM('FitWidthIfOversize', true);
}
else {
setValueGM('FitWidthIfOversize', false);
}
logScript(`fitIfOversize: ${getValueGM('FitWidthIfOversize')}`);
});
$('#viewMode').on('change', (event) => {
const mode = $(event.target).val();
$('#Chapter')
.removeClass('WebComic')
.removeClass('FluidLTR')
.removeClass('FluidRTL')
.addClass(mode);
setValueGM('ViewMode', mode);
logScript(`ViewMode: ${getValueGM('ViewMode')}`);
applyZoom();
});
$('#loadMode').on('change', (event) => {
const mode = $(event.target).val();
setValueGM('LoadMode', mode);
logScript(`MangaLoadMode: ${getValueGM('LoadMode')}`);
});
$('#showThumbnails').on('change', (event) => {
$('#Navigation').toggleClass('disabled');
if ($(event.target).is(':checked')) {
setValueGM('ShowThumbnails', true);
}
else {
setValueGM('ShowThumbnails', false);
}
logScript(`MangaShowThumbnails: ${getValueGM('ShowThumbnails')}`);
applyZoom();
});
// Download
$('#downloadZip').on('change', (event) => {
if ($(event.target).is(':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')}`);
});
$('#blob').one('click', generateZip);
$('.download').on('click', () => {
logScript('Downloading Chapter');
$('#blob').trigger('click');
});
$('#lazyLoadImages').on('change', (event) => {
if ($(event.target).is(':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.<br/>
Suggestion: <span style="color:red;font-weight:bold">Disable Thumbnails</span> to save Bandwidth/Memory.`,
icon: 'warning',
});
}
else {
setValueGM('LazyLoadImages', false);
}
logScript(`MangaLazyLoadImages: ${getValueGM('LazyLoadImages')}`);
});
$('#lazyStart').on('change', (event) => {
const start = $(event.target).val();
setValueGM('LazyStart', start);
logScript(`lazyStart: ${getValueGM('LazyStart')}`);
});
$('#PagesPerSecond').on('change', (event) => {
setValueGM('Timer', parseInt($(event.target).val(), 10));
logScript(`MangaTimer: ${getValueGM('Timer')}`);
});
$('#DefaultZoom').on('change', (event) => {
settings$1.zoom = parseInt($(event.target).val(), 10);
$('#Zoom b').html(settings$1.zoom.toString);
setValueGM('Zoom', parseInt(settings$1.zoom.toString(), 10));
logScript(`MangaZoom: ${getValueGM('Zoom')}`);
applyZoom();
});
// Toggle Controls
$('#pageControls').on('click', () => {
$('#MangaOnlineViewer').toggleClass('hideControls');
});
$('#hidePageControls').on('change', (event) => {
$('#MangaOnlineViewer').toggleClass('hideControls');
if ($(event.target).is(':checked')) {
setValueGM('HidePageControls', true);
}
else {
setValueGM('HidePageControls', false);
}
logScript(`MangaHidePageControls: ${getValueGM('HidePageControls')}`);
});
// Theme Control
$('#ThemeSelector').on('change', (event) => {
const target = $(event.target);
$('#MangaOnlineViewer , body')
.removeClass()
.addClass(target.val());
logScript('Theme', target.val());
setValueGM('Theme', target.val());
if (target.val() === 'Custom_Dark' || target.val() === 'Custom_Light') {
$('.CustomTheme').show();
}
else {
$('.CustomTheme').hide();
}
if (target.val() === 'Full_Custom') {
$('.FullCustom').show();
}
else {
$('.FullCustom').hide();
}
});
// try {
// jscolor.presets.default = {
// position: 'right',
// format: 'hex',
// palette: [
// '#000000', '#7d7d7d', '#870014', '#ec1c23', '#ff7e26',
// '#fef100', '#22b14b', '#00a1e7', '#3f47cc', '#a349a4',
// '#ffffff', '#c3c3c3', '#b87957', '#feaec9', '#ffc80d',
// '#eee3af', '#b5e61d', '#99d9ea', '#7092be', '#c8bfe7',
// ],
// // paletteCols: 12,
// hideOnPaletteClick: true,
// closeButton: true,
// shadow: false,
// alphaChannel: false,
// paletteSetsAlpha: false,
// };
// jscolor.install();
// } catch (e) {
// logScript(e);
// }
// $('INPUT.colorpicker').minicolors();
$('#CustomThemeHue').on('change', (event) => {
const target = $(event.target).val();
logScript(`CustomTheme: ${target}`);
$('style[title="Custom_Light"], style[title="Custom_Dark"]').remove();
$('head').append(addCustomTheme(target));
setValueGM('CustomTheme', target);
logScript(`MangaCustomTheme: ${getValueGM('CustomTheme')}`);
});
$('.FullCustom').on('change', () => {
logScript('FullCustomTheme: ', $('#CustomThemeHueBody').val(), $('#CustomThemeHueText').val(), $('#CustomThemeHueLines').val(), $('#CustomThemeHuePanel').val(), $('#CustomThemeHueButton').val());
$('style[title="Full_Custom"]').remove();
$('head').append(addFullCustomTheme($('#CustomThemeHueBody').val(), $('#CustomThemeHueText').val(), $('#CustomThemeHueLines').val(), $('#CustomThemeHuePanel').val(), $('#CustomThemeHueButton').val()));
setValueGM('CustomThemeBody', $('#CustomThemeHueBody').val());
setValueGM('CustomThemeText', $('#CustomThemeHueText').val());
setValueGM('CustomThemeLines', $('#CustomThemeHueLines').val());
setValueGM('CustomThemePanel', $('#CustomThemeHuePanel').val());
setValueGM('CustomThemeButton', $('#CustomThemeHueButton').val());
});
$('#gotoPage').on('change', (event) => {
applyZoom();
scrollToElement($(`#Page${$(event.target).val()}`));
});
$('.Thumbnail').on('click', (event) => {
applyZoom();
scrollToElement($(`#Page${$(event.currentTarget).find('span').html()}`));
});
// Settings Control
$('#settings').on('click', () => {
$('#ViewerControls').slideToggle();
$('#ViewerShortcuts').slideToggle();
$('#ImageOptions').toggleClass('settingsOpen');
$('#Navigation').toggleClass('visible');
$('#Header').toggleClass('visible');
});
// Individual Page functions
// Bookmark Page to resume reading
$('.Bookmark').on('click', (event) => {
const num = parseInt($(event.target).parents('.MangaPage').find('.PageFunctions span').text(), 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:<h4>Page ${num}</h4>(Only <i>ONCE</i> per Bookmark, will be removed after a year unused)`,
icon: 'success',
});
}
setValueGM('Bookmarks', JSON.stringify(settings$1.bookmarks));
logScript(`MangaBookmarks: ${getValueGM('Bookmarks')}`);
});
// Reload Page
$('.Reload').on('click', (event) => {
reloadImage($(event.target).parents('.MangaPage').find('.PageContent img')[0]);
});
// ZoomIn
$('.ZoomIn').on('click', (event) => {
const img = $(event.target).parents('.MangaPage').find('.PageContent img');
const ratio = (img.width() / img.prop('naturalWidth')) * (100 + settings$1.zoomStep);
applyZoom(`#${$(event.target).attr('id')}`, ratio);
});
// ZoomOut
$('.ZoomOut').on('click', (event) => {
const img = $(event.target).parents('.MangaPage').find('.PageContent img');
const ratio = (img.width() / img.prop('naturalWidth')) * (100 - settings$1.zoomStep);
applyZoom(`#${$(event.target).attr('id')}`, ratio);
});
// ZoomRestore
$('.ZoomRestore').on('click', () => {
$('.PageContent img').removeAttr('width');
});
// ZoomWidth
$('.ZoomWidth').on('click', (event) => {
$(event.target).parents('.MangaPage').find('.PageContent img');
applyZoom(`#${$(event.target).attr('id')}`, 1000);
});
// ZoomHeight
$('.ZoomHeight').on('click', (event) => {
$(event.target).parents('.MangaPage').find('.PageContent img');
applyZoom(`#${$(event.target).attr('id')}`, -1000);
});
// Hide
$('.Hide').on('click', (event) => {
const img = $(event.target).parents('.MangaPage').find('.PageContent');
img.slideToggle('slow');
});
const useScrollDirection = (showEnd = 0) => {
let prevOffset = 0;
const header = document.querySelector('#Header');
const setScrollDirection = (show) => {
if (show == null) {
header.classList.remove('scroll-hide');
header.classList.remove('scroll-show');
}
else if (show) {
header.classList.add('scroll-show');
header.classList.remove('scroll-hide');
}
else {
header.classList.remove('scroll-show');
header.classList.add('scroll-hide');
}
};
const toggleScrollDirection = () => {
const { scrollY } = window;
if (showEnd && scrollY + window.innerHeight + showEnd > document.body.offsetHeight) {
setScrollDirection(true);
}
else if (scrollY > prevOffset && scrollY > 50) {
setScrollDirection(false);
}
else if (scrollY < prevOffset && scrollY > 50) {
setScrollDirection(true);
}
else {
setScrollDirection(null);
}
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);
}
async function lateStart(site, begin = 1) {
const manga = await site.run();
logScript('LateStart');
const options = {
title: 'Starting<br>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<br>MangaOnlineViewer',
html: `${beginning > 1 ? `Resuming reading from Page ${beginning}.<br/>` : ''}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);
})();