// ==UserScript==
// @name Manga OnlineViewer
// @author Tago
// @supportURL https://github.com/TagoDR/MangaOnlineViewer/issues
// @namespace https://github.com/TagoDR
// @description Shows all pages at once in online view for these sites
// @version 2024.01.11
// @license MIT
// @icon https://cdn-icons-png.flaticon.com/32/2281/2281832.png
// @run-at document-end
// @grant unsafeWindow
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_listValues
// @grant GM_deleteValue
// @grant GM_xmlhttpRequest
// @noframes on
// @connect *
// @require https://cdnjs.cloudflare.com/ajax/libs/tinycolor/1.6.0/tinycolor.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/jquery.imagesloaded/5.0.0/imagesloaded.pkgd.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.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.8/sweetalert2.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.21/lodash.min.js
// @require https://cdn.jsdelivr.net/npm/hotkeys-js@3.13.2/dist/hotkeys.min.js
// @require https://cdn.jsdelivr.net/npm/range-slider-input@2.4.4/dist/rangeslider.nostyle.umd.min.js
// @include /https?:\/\/alandal.com\/chapter\/.+\/\d+/
// @include /https?:\/\/(www\.)?bato.to\/(chapter|title).*/
// @include /https?:\/\/(www\.)?(bilibilicomics).com\/.+\/.+/
// @include /https?:\/\/comic\.nizamkomputer.com\/read\/.+\/\d+.*/
// @include /https?:\/\/(www\.)?dynasty-scans.com\/chapters\/.+/
// @include /https?:\/\/(www\.)?(asura.*|flamecomics|rizzcomic|void-scans|luminousscans|shimascans|nightscans|manhwafreak|manhwa-freak|ozulscansen|azuremanga|cypherscans).(com|org|gg|xyz|to|net)\/.+/
// @include /https?:\/\/(comics\.)?inkr.com\/title\/.+\/chapter\/.+/
// @include /https?:\/\/(www\.)?inmanga.com\/ver\/manga\/.+\/.+/
// @include /https?:\/\/(www\.)?klmanga.com\/.+chapter.+/
// @include /https?:\/\/(www\.)?leitor.net\/manga\/.+\/.+\/.+/
// @include /https?:\/\/(www\.)?lhtranslation.net\/read.+/
// @include /(file:\/\/\/.+(index)?.html)/
// @include /https?:\/\/(www\.)?lynxscans.com\/comics?\/.+/
// @include /https?:\/\/(www\.)?mangabuddy.com\/.+\/chapter.+/
// @include /https?:\/\/(www\.)?mangadex.org\/chapter\/.+(\/.+)?/
// @include /https?:\/\/(www\.)?(fanfox.net|mangahere.cc)\/manga\/.+\/.+\//
// @include /https?:\/\/.{3,4}?(mangafreak).net\/Read.+/
// @include /https?:\/\/(www\.)?mangago.me\/.*\/.*\/.*/
// @include /https?:\/\/(www\.)?mangahosted.com\/manga\/.+\/.+/
// @include /https?:\/\/(www\.)?(mangahub).io\/chapter\/.+\/.+/
// @include /https?:\/\/(www\.)?mangas.in\/manga\/.+\/.+\/\d+/
// @include /https?:\/\/(www\.)?(read|chap)?(manganelo|mangakakalot|manganato).(com|to).*\/chapter.+/
// @include /https?:\/\/(www\.)?mangapark.(com|me|org|net)\/title\/.+\/.+/
// @include /https?:\/\/(www\.)?mangareader.to\/read\/.+\/.+\/.+/
// @include /https?:\/\/(www\.)?(mangasee123|manga4life).com\/read-online\/.+/
// @include /https?:\/\/(www\.)?mangatigre.net\/.+\/.+\/.+/
// @include /https?:\/\/.*mangatoon.mobi\/.+\/watch\/.+/
// @include /https?:\/\/(www\.|m\.)?mangatown.com\/manga\/.+\/.+/
// @include /https?:\/\/(www\.)?manhuascan.io\/.+chapter.+/
// @include /https?:\/\/(www\.)?mgeko.com?\/reader\/.*/
// @include /https?:\/\/(www\.)?(naniscans).com\/chapters\/.+\/read/
// @include /https?:\/\/(www\.)?ninemanga.com\/chapter\/.+\/.+\.html/
// @include /https?:\/\/(www\.)?olympusscans.com\/capitulo\/.+\/.+/
// @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?:\/\/(www\.)?reaperscans\.com\/comics\/.+\/chapters\/.+/
// @include /https?:\/\/raw\.senmanga.com\/.+\/.+\/?/
// @include /https?:\/\/(www\.)?tapas.io\/episode\/.+/
// @include /https?:\/\/(www\.)?(tenmanga|gardenmanage).com\/(chapter|statuses)\/.+/
// @include /https?:\/\/(www\.)?(.+).com\/(viewer|news)\/.+\/(paginated|cascade)/
// @include /https?:\/\/(www\.)?tumanhwas.com\/news\/.+/
// @include /https?:\/\/(www\.)?unionleitor.top\/leitor\/.+\/.+/
// @include /https?:\/\/(www\.)?webnovel.com\/comic\/.+/
// @include /https?:\/\/(www\.)?webtoons.com\/.+viewer.+/
// @include /https?:\/\/(www\.)?(manga33).com\/manga\/.+/
// @include /https?:\/\/(www\.)?(yugenmangas).(com|net|lat)\/series\/.+/
// @include /https?:\/\/(www\.)?zscans.com\/comics\/.+/
// @include /^(?!.*jaiminisbox).*\/read\/.+/
// @include /https?:\/\/.+\/(manga|series|manhua|comic|ch|novel)\/.+\/.+/
// @exclude /https?:\/\/(www\.)?tsumino.com\/.+/
// @exclude /https?:\/\/(www\.)?pururin.io\/.+/
// @downloadURL none
// ==/UserScript==
(function () {
"use strict";
const alandal = {
name: "Alandal",
url: /https?:\/\/alandal.com\/chapter\/.+\/\d+/,
homepage: "https://alandal.com/",
language: ["English"],
category: "manga",
run() {
const images = [...document.querySelectorAll('img[alt^="Page"]')];
const chapter = document
?.querySelector('[aria-label="chapter list"]')
?.parentElement?.parentElement?.parentElement?.parentElement?.querySelectorAll(
"a",
);
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector('a[href^="/series/"]')
?.getAttribute("href"),
pages: images.length,
prev: chapter?.item(0)?.getAttribute("href"),
next: chapter?.item(1)?.getAttribute("href"),
listImages: images.map((img) => img.getAttribute("src")),
};
},
};
const batoto = {
name: "Batoto",
url: /https?:\/\/(www\.)?bato.to\/(chapter|title).*/,
homepage: "https://bato.to/",
language: ["English"],
category: "manga",
waitEle: 'div[name="image-item"] img, .page-img',
run() {
if (window.location.pathname.startsWith("/title")) {
if (window.location.search !== "?load=2") {
window.location.search = "?load=2";
}
const images2 = [
...document.querySelectorAll('div[name="image-item"] img'),
];
return {
title: document.querySelector("h6")?.textContent?.trim(),
series: document.querySelector("h3 a")?.getAttribute("href"),
pages: images2.length,
prev: [...document.querySelectorAll("span")]
.find((item) => item.textContent?.endsWith("Prev Chapter"))
?.parentElement?.getAttribute("href"),
next: [...document.querySelectorAll("span")]
.find((item) => item.textContent?.startsWith("Next Chapter"))
?.parentElement?.getAttribute("href"),
listImages: images2.map((img) => img.getAttribute("src")),
};
}
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")),
};
},
};
const bilibilicomics = {
name: "BilibiliComics",
url: /https?:\/\/(www\.)?(bilibilicomics).com\/.+\/.+/,
homepage: "https://www.bilibilicomics.com/",
language: ["English"],
category: "manga",
waitEle: ".read-nav",
async run() {
const api = await fetch(
"https://www.bilibilicomics.com/twirp/comic.v1.Comic/GetImageIndex?device=pc&platform=web&lang=en&sys_lang=en",
{
method: "post",
headers: {
Accept: "application/json",
"Content-Type": "application/json",
},
body: JSON.stringify({
ep_id: window.location.href.split("/").pop(),
credential: "",
}),
},
)
.then(async (res) => res.json())
.then(({ data }) =>
data.images.map((image) => `${image.path}@2000w.webp`),
)
.then(JSON.stringify)
.then(async (urls) =>
fetch(
"https://www.bilibilicomics.com/twirp/comic.v1.Comic/ImageToken?device=pc&platform=web&lang=en&sys_lang=en",
{
method: "post",
headers: {
Accept: "application/json",
"Content-Type": "application/json",
},
body: JSON.stringify({ urls }),
},
),
)
.then(async (res) => res.json())
.then((tokens) => tokens.data.map((i) => `${i.url}?token=${i.token}`));
return {
title: document.querySelector(".read-nav")?.textContent?.trim(),
series: document.querySelector(".manga-title")?.getAttribute("href"),
pages: api.length,
prev: document
.querySelector(".navigate a button[title=Previous]")
?.parentElement?.getAttribute("href"),
next: document
.querySelector(".navigate a button[title=Next]")
?.parentElement?.getAttribute("href"),
listImages: api,
};
},
};
const comicastle = {
name: "ComiCastle",
url: /https?:\/\/comic\.nizamkomputer.com\/read\/.+\/\d+.*/,
homepage: "https://comic.nizamkomputer.com/",
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")),
};
},
};
const dysnatyscans = {
name: "Dynasty-Scans",
url: /https?:\/\/(www\.)?dynasty-scans.com\/chapters\/.+/,
homepage: "https://dynasty-scans.com/",
language: ["English"],
category: "manga",
run() {
return {
title: document.querySelector("#chapter-title")?.textContent?.trim(),
series: document
.querySelector("#chapter-title a")
?.getAttribute("href"),
pages: unsafeWindow.pages.length,
prev: document.querySelector("#prev_link")?.getAttribute("href"),
next: document.querySelector("#next_link")?.getAttribute("href"),
listImages: unsafeWindow.pages.map((x) => x.image),
};
},
};
const foolslide = {
name: ["FoOlSlide", "Kireicake"],
url: /^(?!.*jaiminisbox).*\/read\/.+/,
homepage: [
"https://github.com/saintly2k/FoOlSlideX",
"https://reader.kireicake.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() ??
document.querySelector("title")?.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",
};
},
};
const inkr = {
name: "INKR",
url: /https?:\/\/(comics\.)?inkr.com\/title\/.+\/chapter\/.+/,
homepage: "https://inkr.com/",
language: ["English"],
category: "manga",
waitFunc: () =>
document.querySelector(
'[data-container="file-horizontal-scroll-view"] img',
)?.naturalWidth !== void 0 &&
document.querySelectorAll(
'[data-container="file-horizontal-scroll-view"] img',
).length > 2,
run() {
const images = [
...document.querySelectorAll(
'[data-container="file-horizontal-scroll-view"] img',
),
];
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector('[aria-label="Previous Chapter"] + div a')
?.getAttribute("href"),
pages: images.length,
prev: document
.querySelector('a[aria-label="Previous Chapter"]')
?.getAttribute("href"),
next: document
.querySelector('a[aria-label="Next Chapter"]')
?.getAttribute("href"),
listImages: images.map((img) =>
img.getAttribute("src")?.replace("/t.", "/p."),
),
};
},
};
const inmanga = {
name: "InManga",
url: /https?:\/\/(www\.)?inmanga.com\/ver\/manga\/.+\/.+/,
homepage: "https://inmanga.com//",
language: ["Spanish"],
category: "manga",
waitVar: "pageController",
run() {
const images = [...document.querySelectorAll("#PageList option")];
const chapter = document.querySelector("#ChapList option:checked");
const src = unsafeWindow.pageController._containers.pageUrl;
return {
title: document.querySelector("title")?.textContent?.trim(),
series: `../${unsafeWindow.pageController._containers.mangaIdentification}`,
pages: images.length,
prev: chapter?.previousElementSibling?.getAttribute("value"),
next: chapter?.nextElementSibling?.getAttribute("value"),
listImages: images.map((img, index) =>
src
.replace("identification", img.getAttribute("value"))
.replace("pageNumber", index),
),
};
},
};
const klmanga = {
name: "KLManga",
url: /https?:\/\/(www\.)?klmanga.com\/.+chapter.+/,
homepage: "https://klmanga.com/",
language: ["Raw"],
category: "manga",
run() {
const images = [...document.querySelectorAll(".chapter-content img")];
const chapter = document
.querySelectorAll(".form-control")[0]
.querySelector("option:checked");
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document.querySelector(".navbar-brand")?.getAttribute("href"),
pages: images.length,
prev: chapter?.nextElementSibling?.getAttribute("value"),
next: chapter?.previousElementSibling?.getAttribute("value"),
listImages: images.map((img) => img.getAttribute("src")),
};
},
};
const leitor = {
name: "Leitor",
url: /https?:\/\/(www\.)?leitor.net\/manga\/.+\/.+\/.+/,
homepage: "https://leitor.net/",
language: ["Portuguese"],
category: "manga",
async run() {
const url = `https://leitor.net/leitor/pages/${unsafeWindow.READER_ID_RELEASE}.json?key=${unsafeWindow.READER_TOKEN}`;
const api = await fetch(url).then(async (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),
};
},
};
const 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")),
};
},
};
const concatenateTemplateLiteralTag = (raw, ...keys) =>
keys.length === 0 ? raw[0] : String.raw({ raw }, ...keys);
const html = concatenateTemplateLiteralTag;
const css = concatenateTemplateLiteralTag;
const colors = {
dark: {
name: "dark",
50: "#C1C2C5",
100: "#A6A7AB",
200: "#909296",
300: "#5c5f66",
400: "#373A40",
500: "#2C2E33",
600: "#25262b",
700: "#1A1B1E",
800: "#141517",
900: "#101113",
},
gray: {
name: "gray",
50: "#f8f9fa",
100: "#f1f3f5",
200: "#e9ecef",
300: "#dee2e6",
400: "#ced4da",
500: "#adb5bd",
600: "#868e96",
700: "#495057",
800: "#343a40",
900: "#212529",
},
red: {
name: "red",
50: "#fff5f5",
100: "#ffe3e3",
200: "#ffc9c9",
300: "#ffa8a8",
400: "#ff8787",
500: "#ff6b6b",
600: "#fa5252",
700: "#f03e3e",
800: "#e03131",
900: "#c92a2a",
},
pink: {
name: "pink",
50: "#fff0f6",
100: "#ffdeeb",
200: "#fcc2d7",
300: "#faa2c1",
400: "#f783ac",
500: "#f06595",
600: "#e64980",
700: "#d6336c",
800: "#c2255c",
900: "#a61e4d",
},
grape: {
name: "grape",
50: "#f8f0fc",
100: "#f3d9fa",
200: "#eebefa",
300: "#e599f7",
400: "#da77f2",
500: "#cc5de8",
600: "#be4bdb",
700: "#ae3ec9",
800: "#9c36b5",
900: "#862e9c",
},
violet: {
name: "violet",
50: "#f3f0ff",
100: "#e5dbff",
200: "#d0bfff",
300: "#b197fc",
400: "#9775fa",
500: "#845ef7",
600: "#7950f2",
700: "#7048e8",
800: "#6741d9",
900: "#5f3dc4",
},
indigo: {
name: "purple",
50: "#edf2ff",
100: "#dbe4ff",
200: "#bac8ff",
300: "#91a7ff",
400: "#748ffc",
500: "#5c7cfa",
600: "#4c6ef5",
700: "#4263eb",
800: "#3b5bdb",
900: "#364fc7",
},
blue: {
name: "blue",
50: "#e7f5ff",
100: "#d0ebff",
200: "#a5d8ff",
300: "#74c0fc",
400: "#4dabf7",
500: "#339af0",
600: "#228be6",
700: "#1c7ed6",
800: "#1971c2",
900: "#1864ab",
},
cyan: {
name: "cyan",
50: "#e3fafc",
100: "#c5f6fa",
200: "#99e9f2",
300: "#66d9e8",
400: "#3bc9db",
500: "#22b8cf",
600: "#15aabf",
700: "#1098ad",
800: "#0c8599",
900: "#0b7285",
},
teal: {
name: "teal",
50: "#e6fcf5",
100: "#c3fae8",
200: "#96f2d7",
300: "#63e6be",
400: "#38d9a9",
500: "#20c997",
600: "#12b886",
700: "#0ca678",
800: "#099268",
900: "#087f5b",
},
green: {
name: "green",
50: "#ebfbee",
100: "#d3f9d8",
200: "#b2f2bb",
300: "#8ce99a",
400: "#69db7c",
500: "#51cf66",
600: "#40c057",
700: "#37b24d",
800: "#2f9e44",
900: "#2b8a3e",
},
lime: {
name: "lime",
50: "#f4fce3",
100: "#e9fac8",
200: "#d8f5a2",
300: "#c0eb75",
400: "#a9e34b",
500: "#94d82d",
600: "#82c91e",
700: "#74b816",
800: "#66a80f",
900: "#5c940d",
},
yellow: {
name: "yellow",
50: "#fff9db",
100: "#fff3bf",
200: "#ffec99",
300: "#ffe066",
400: "#ffd43b",
500: "#fcc419",
600: "#fab005",
700: "#f59f00",
800: "#f08c00",
900: "#e67700",
},
orange: {
name: "orange",
50: "#fff4e6",
100: "#ffe8cc",
200: "#ffd8a8",
300: "#ffc078",
400: "#ffa94d",
500: "#ff922b",
600: "#fd7e14",
700: "#f76707",
800: "#e8590c",
900: "#d9480f",
},
darkblue: {
name: "darkblue",
50: "#E8F4F9",
100: "#D9DEE9",
200: "#B7C2DA",
300: "#6482C0",
400: "#4267B2",
500: "#385898",
600: "#314E89",
700: "#29487D",
800: "#223B67",
900: "#1E355B",
},
};
const darkest = 10;
const lightest = 95;
function setLightness(hsl, lightness) {
hsl.l = lightness;
return tinycolor(hsl).toHexString();
}
function getTextColor(hex) {
const color = tinycolor(hex);
const hsl = color.toHsl();
return setLightness(hsl, color.isDark() ? lightest : darkest);
}
function isDark(color) {
return tinycolor(color).getBrightness() <= 120;
}
function isBackgroundColorDark(element) {
return isDark(window.getComputedStyle(element).backgroundColor);
}
function svgToUrl(str) {
const cleaned = str.replace(/[\t\n\r]/gim, "").replace(/\s\s+/g, " ");
const encoded = encodeURIComponent(cleaned)
.replace(/\(/g, "%28")
.replace(/\)/g, "%29");
return `data:image/svg+xml;charset=UTF-8,${encoded}`;
}
const rulerMarkerLength = (len) => {
if (len % 100 === 0) {
return 15;
}
if (len % 50 === 0) {
return 10;
}
if (len % 25 === 0) {
return 5;
}
return 2.5;
};
function rectangleRuler(width, height, bgColor, textColor) {
let markers = "";
for (let x = 0; x <= width; x += 5) {
const tick = html` `;
markers += tick;
if (x !== 0 && x % 50 === 0) {
const label = html`
${x}
`;
markers += label;
}
}
for (let y = 0; y <= height; y += 5) {
const tick = html` `;
markers += tick;
if (y !== 0 && y % 50 === 0) {
const label = html`
${y}
`;
markers += label;
}
}
return html`
${width}x${height}
${markers}
`;
}
function placeholder(
width,
height,
bgColor = "#0F1C3F",
textColor = "#ECEAD9",
) {
const str = rectangleRuler(width, height, bgColor, textColor);
return svgToUrl(str);
}
const backgrounds = Object.values(colors).map((i) => i["900"]);
const widths = [400, 600, 900, 1200, 1400, 1600, 1970];
const heights = [600, 800, 1e3, 1200, 1400, 2e3, 2600];
function randomPlaceholder() {
const randomWidth = Math.floor(Math.random() * widths.length);
const randomHeight = Math.floor(Math.random() * heights.length);
const randomColor = Math.floor(Math.random() * backgrounds.length);
return placeholder(
widths[randomWidth],
heights[randomHeight],
backgrounds[randomColor],
);
}
const localhost = {
name: "Local Files",
url: /(file:\/\/\/.+(index)?.html)/,
homepage: "/index.html?raw=1",
language: ["Raw"],
category: "manga",
run() {
const num = parseInt(
/\d+/.exec(window.location.search)?.toString() ?? "5",
10,
);
const comments = document.createElement("div");
comments.innerHTML = Array(100).fill("Testing Comment ").join("");
return {
title: "Placeholder Manga Loaded",
series: "?reload",
pages: num,
begin: 1,
prev: "?pages=50",
next: "?pages=1",
listImages: [
placeholder(1970, 1400, "#2D1657"),
placeholder(985, 1400, "#152C55"),
placeholder(985, 1400, "#7A1420"),
placeholder(985, 1400, "#0F5B30"),
placeholder(1970, 1400, "#806D15"),
...Array(num).fill(0).map(randomPlaceholder),
],
comments,
};
},
};
const lynxscans = {
name: "LynxScans",
url: /https?:\/\/(www\.)?lynxscans.com\/comics?\/.+/,
homepage: "https://lynxscans.com/",
language: ["English"],
category: "manga",
waitAttr: ["#app", "data-page"],
run() {
const data = JSON.parse(
document.querySelector("#app").getAttribute("data-page"),
);
return {
title: document.querySelector("title")?.textContent?.trim(),
series: data.props.home,
pages: data.props.pages.length,
prev: data.props.previousChapter,
next: data.props.nextChapter,
listImages: data.props.pages.map((img) => img.url),
};
},
};
function findImages() {
return [
...document.querySelectorAll(
".wp-manga-chapter-img, .blocks-gallery-item img, .reading-content img",
),
].map(
(img) =>
img?.getAttribute("src") ??
img?.getAttribute("data-src") ??
img?.getAttribute("data-full-url"),
);
}
const madarawp = {
name: [
"Madara WordPress Plugin",
"MangaHaus",
"Isekai Scan",
"Comic Kiba",
"Zinmanga",
"mangatx",
"Toonily",
"Mngazuki",
"JaiminisBox",
"DisasterScans",
"ManhuaPlus",
"TopManhua",
"NovelMic",
"Reset-Scans",
"LeviatanScans",
"Dragon Tea",
"SetsuScans",
],
url: /https?:\/\/.+\/(manga|series|manhua|comic|ch|novel)\/.+\/.+/,
homepage: [
"https://mangabooth.com/",
"https://manhuaus.com",
"https://isekaiscan.com/",
"https://comickiba.com/",
"https://zinmanga.com/",
"https://mangatx.com/",
"https://toonily.net/",
"https://mangazuki.me/",
"https://jaiminisbox.net",
"https://disasterscans.com/",
"https://manhuaplus.com/",
"https://www.topmanhua.com/",
"https://novelmic.com/",
"https://reset-scans.com/",
"https://leviatanscans.com/",
"https://dragontea.ink/",
"https://setsuscans.com/",
],
language: ["English"],
obs: "Any Site that uses Madara Wordpress Plugin",
category: "manga",
waitFunc: () =>
findImages().every(
(s) =>
s &&
/^([\t\n])*(https?:\/\/)?.+\.(jpg|jpeg|png|gif|bmp|webp).*$/.test(s),
),
run() {
const images = findImages();
return {
title: document.querySelector("#chapter-heading")?.textContent?.trim(),
series: (
document.querySelector(".breadcrumb li:nth-child(3) a") ??
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,
};
},
};
const mangabuddy = {
name: "MangaBuddy",
url: /https?:\/\/(www\.)?mangabuddy.com\/.+\/chapter.+/,
homepage: "https://mangabuddy.com/",
language: ["English"],
category: "manga",
waitVar: "chapImages",
run() {
const images = unsafeWindow.chapImages.split(",");
return {
title: document.querySelector(".chapter-info")?.textContent?.trim(),
series: document
.querySelector("#breadcrumbs-container div:nth-child(2) a")
?.getAttribute("href"),
pages: images.length,
prev: document.querySelector("a.prev")?.getAttribute("href"),
next: document.querySelector("a.next")?.getAttribute("href"),
listImages: images,
};
},
};
const mangadex = {
name: "MangaDex",
url: /https?:\/\/(www\.)?mangadex.org\/chapter\/.+(\/.+)?/,
homepage: "https://mangadex.org/",
language: ["English"],
category: "manga",
waitEle: ".md--reader-menu a[href^='/chapter/']",
async run() {
const chapterId = /\/chapter\/([^/]+)(\/\d+)?/
.exec(window.location.pathname)
?.at(1);
const home = `https://api.mangadex.org/at-home/server/${chapterId}`;
const server = await fetch(home).then(async (res) => res.json());
const images = server.chapter.data;
const chapters = [
...document.querySelectorAll(".md--reader-menu a[href^='/chapter/']"),
].map((a) => a.getAttribute("href"));
return {
title: document.querySelector("title")?.text.replace(" - MangaDex", ""),
series: document
.querySelector("a.text-primary[href^='/title/']")
?.getAttribute("href"),
pages: images.length,
prev: chapters[0] !== window.location.pathname ? chapters[0] : "#",
next: chapters[1] !== window.location.pathname ? chapters[1] : "#",
listImages: images.map(
(img) => `${server.baseUrl}/data/${server.chapter.hash}/${img}`,
),
};
},
};
const 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 key = document.querySelector("#dm5_key")?.getAttribute("value");
const options = {
method: "GET",
headers: {
"Content-Type": "text/plain",
},
};
const src = Array(unsafeWindow.imagecount)
.fill(0)
.map(async (_, i) => {
const url = `chapterfun.ashx?cid=${
unsafeWindow.chapterid ?? unsafeWindow.chapter_id
}&page=${i}&key=${key}`;
const api = await fetch(url, options).then(async (res) => res.text());
(0, eval)(api);
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: unsafeWindow.imagecount,
prev: unsafeWindow.prechapterurl,
next: unsafeWindow.nextchapterurl,
listImages: images.map((img, i) => img[i === 0 ? 0 : 1]),
};
},
};
const 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")),
};
},
};
const mangago = {
name: "Mangago",
url: /https?:\/\/(www\.)?mangago.me\/.*\/.*\/.*/,
homepage: "https://www.mangago.me/",
language: ["English"],
category: "manga",
waitVar: "imgsrcs",
run() {
const key = CryptoJS.enc.Hex.parse("e11adc3949ba59abbe56e057f20f883e");
const iv = CryptoJS.enc.Hex.parse("1234567890abcdef1234567890abcdef");
const opinion = { iv, padding: CryptoJS.pad.ZeroPadding };
const images = CryptoJS.AES.decrypt(unsafeWindow.imgsrcs, key, opinion)
.toString(CryptoJS.enc.Utf8)
.split(",");
return {
title: `${unsafeWindow.manga_name} ${unsafeWindow.chapter_name}`,
series: unsafeWindow.mid,
pages: unsafeWindow.total_pages,
prev: document
.querySelector(".recom p:nth-child(5) a")
?.getAttribute("href"),
next: unsafeWindow.next_c_url,
listImages: images,
};
},
};
const mangahosted = {
name: "MangaHosted",
url: /https?:\/\/(www\.)?mangahosted.com\/manga\/.+\/.+/,
homepage: "https://mangahosted.com/",
language: ["Portuguese"],
category: "manga",
run() {
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: unsafeWindow.$read_prev,
next: unsafeWindow.$read_next,
listImages: images.map((img) => img.getAttribute("src")),
};
},
};
const mangahub = {
name: "MangaHub",
url: /https?:\/\/(www\.)?(mangahub).io\/chapter\/.+\/.+/,
homepage: "https://mangahub.io/",
language: ["English"],
category: "manga",
waitEle: "#select-chapter",
async run() {
function getCookie(name) {
const re = new RegExp(`${name}=([^;]+)`);
const value = re.exec(document.cookie);
return value != null ? decodeURIComponent(value[1]) : null;
}
const slug =
unsafeWindow.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",
"x-mhub-access": getCookie("mhub_access") ?? "",
},
};
const api = await fetch("https://api.mghubcdn.com/graphql", options).then(
async (res) => res.json(),
);
const images = JSON.parse(api?.data.chapter.pages.toString());
return {
title: document.querySelector("#mangareader h3")?.textContent?.trim(),
series: document.querySelector("#mangareader a")?.getAttribute("href"),
pages: images.i.length,
prev: document.querySelector(".previous a")?.getAttribute("href"),
next: document.querySelector(".next a")?.getAttribute("href"),
listImages: images.i.map(
(i) => `https://img.mghubcdn.com/file/imghub/${images.p + i}`,
),
};
},
};
const mangasin = {
name: "MangasIn",
url: /https?:\/\/(www\.)?mangas.in\/manga\/.+\/.+\/\d+/,
homepage: "https://mangas.in/",
language: ["Spanish"],
category: "manga",
run() {
const images = [...document.querySelectorAll("#all img")];
const chapter = document.querySelector("#chapter-list li.active");
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector("#navbar-collapse-1 ul:nth-child(2) a")
?.getAttribute("href"),
pages: images.length,
prev: chapter?.nextElementSibling?.firstElementChild?.getAttribute(
"href",
),
next: chapter?.previousElementSibling?.firstElementChild?.getAttribute(
"href",
),
listImages: images.map((img) => img.getAttribute("data-src")),
};
},
};
const mangakakalot = {
name: ["MangaKakalot", "MangaNelo", "MangaNato"],
url: /https?:\/\/(www\.)?(read|chap)?(manganelo|mangakakalot|manganato).(com|to).*\/chapter.+/,
homepage: [
"https://mangakakalot.com/",
"https://www.manganelo.com/",
"https://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, .panel-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")),
};
},
};
const mangapark = {
name: "MangaPark",
url: /https?:\/\/(www\.)?mangapark.(com|me|org|net)\/title\/.+\/.+/,
homepage: "https://mangapark.net/",
language: ["English"],
category: "manga",
waitEle: "main div div a.btn-primary",
run() {
const json = JSON.parse(
document.querySelector("#__NEXT_DATA__")?.innerHTML ?? "",
);
const data =
json.props.pageProps.dehydratedState.queries[0].state.data.data
.imageSet;
const images = data.httpLis.map(
(img, index) => `${img}?${data.wordLis[index]}`,
);
return {
title: [
...document.querySelectorAll(
".comic-detail h3 a, .comic-detail h6 span",
),
]
.map((e) => e.textContent?.trim())
.join(" "),
series: document.querySelector(".comic-detail a")?.getAttribute("href"),
pages: images.length,
prev: document
.querySelectorAll("main div div a.btn-primary")
?.item(0)
?.getAttribute("href"),
next: document
.querySelectorAll("main div div a.btn-primary")
?.item(1)
?.getAttribute("href"),
listImages: images,
};
},
};
const 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",
async run() {
const chapter = document.querySelector(".chapter-item.active");
const images = [
...document.querySelectorAll(".ds-image[data-url], .iv-card[data-url]"),
];
const src = images.map(async (img) => {
const url = img.getAttribute("data-url");
if (url && img.classList.contains("shuffled")) {
return (await imgReverser(url)).toDataURL();
}
return url;
});
return {
title: document.querySelector(".hr-manga h2")?.textContent?.trim(),
series: document.querySelector(".hr-manga")?.getAttribute("href"),
pages: src.length,
prev: chapter?.nextElementSibling
?.querySelector("a")
?.getAttribute("href"),
next: chapter?.previousElementSibling
?.querySelector("a")
?.getAttribute("href"),
listImages: await Promise.all(src),
};
},
};
const 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 === void 0) {
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(i + 1)
.padStart(3, "0")
.slice(-3)}.png`,
),
),
};
},
};
const mangastreamwp = {
name: [
"MangaStream WordPress Plugin",
"Asura Scans",
"Flame Comics",
"Rizzcomic",
"Voids-Scans",
"Luminous Scans",
"Shimada Scans",
"Night Scans",
"Manhwa-Freak",
"OzulScansEn",
"AzureManga",
"CypherScans",
],
url: /https?:\/\/(www\.)?(asura.*|flamecomics|rizzcomic|void-scans|luminousscans|shimascans|nightscans|manhwafreak|manhwa-freak|ozulscansen|azuremanga|cypherscans).(com|org|gg|xyz|to|net)\/.+/,
homepage: [
"https://themesia.com/mangastream-wordpress-theme/",
"https://asuratoon.com/",
"https://flamecomics.com/",
"https://rizzcomic.com/",
"https://void-scans.com/",
"https://luminousscans.com/",
"https://shimadascans.com/",
"https://nightscans.net/",
"https://manhwa-freak.com/",
"https://ozulscansen.com/",
"https://azuremanga.com/",
"https://cypherscans.xyz/",
],
language: ["English"],
category: "manga",
// waitTime: 2000,
waitEle: "#chapter option:nth-child(2)",
run() {
const chapter = document.querySelector("#chapter option:checked");
const images = [
...document.querySelectorAll("#readerarea img:not(.asurascans)"),
];
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("data-src") ??
img.getAttribute("data-lazy-src") ??
img.getAttribute("src"),
),
};
},
};
const mangatigre = {
name: "MangaTigre",
url: /https?:\/\/(www\.)?mangatigre.net\/.+\/.+\/.+/,
homepage: "https://www.mangatigre.net/",
language: ["Spanish"],
category: "manga",
run() {
const images = [...document.querySelectorAll(".chapter-content img")];
const chapter = document.querySelector(".form-control option:checked");
return {
title: document.querySelector(".page-title")?.textContent?.trim(),
series: document
.querySelector(".breadcrumb li:nth-child(3) a")
?.getAttribute("href"),
pages: images.length,
prev: chapter?.nextElementSibling?.getAttribute("value"),
next: chapter?.previousElementSibling?.getAttribute("value"),
listImages: images.map(
(img) => img.getAttribute("data-src") ?? img.getAttribute("src"),
),
};
},
};
const mangatoon = {
name: "MangaToons",
url: /https?:\/\/.*mangatoon.mobi\/.+\/watch\/.+/,
homepage: "https://mangatoon.mobi/",
language: ["English"],
category: "manga",
waitEle: ".pictures img:not(.cover)",
run() {
const images = [
...document.querySelectorAll(".pictures img:not(.cover)"),
];
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document.querySelector(".top-left a")?.getAttribute("href"),
pages: images.length,
prev: document.querySelector(".page-icons-prev")?.getAttribute("href"),
next: document.querySelector(".page-icons-next")?.getAttribute("href"),
listImages: images.map((img) => img.getAttribute("data-src")),
};
},
};
const 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 key = document.querySelector("#dm5_key")?.getAttribute("value");
const options = {
method: "GET",
headers: {
"Content-Type": "text/plain",
},
};
const src = Array(unsafeWindow.total_pages)
.fill(0)
.map(async (_, i) => {
const url = `chapterfun.ashx?cid=${unsafeWindow.chapter_id}&page=${i}&key=${key}`;
const api = await fetch(url, options).then(async (res) => res.text());
(0, eval)(api);
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: unsafeWindow.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]),
};
},
};
const manhuascan = {
name: "ManhuaScan",
url: /https?:\/\/(www\.)?manhuascan.io\/.+chapter.+/,
homepage: "https://manhuascan.io/",
language: ["English"],
category: "manga",
waitVar: "imgsrcs",
run() {
const key = CryptoJS.enc.Hex.parse("e11adc3949ba59abbe56e057f20f131e");
const iv = CryptoJS.enc.Hex.parse("1234567890abcdef1234567890abcdef");
const opinion = { iv, padding: CryptoJS.pad.ZeroPadding };
const images = CryptoJS.AES.decrypt(unsafeWindow.imgsrcs, key, opinion)
.toString(CryptoJS.enc.Utf8)
.split(",");
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector(".breadcrumb li:nth-child(3) a")
?.getAttribute("href"),
pages: images.length,
prev: document
.querySelector(".chapter-select a:first-of-type")
?.getAttribute("href"),
next: document
.querySelector(".chapter-select a:last-of-type")
?.getAttribute("href"),
listImages: images,
};
},
};
const mgeko = {
name: ["MangaGeko"],
url: /https?:\/\/(www\.)?mgeko.com?\/reader\/.*/,
homepage: ["https://www.mgeko.com/"],
language: ["English"],
category: "manga",
run() {
const images = [...document.querySelectorAll("#chapter-reader img")];
return {
title: document.querySelector(".titles")?.textContent?.trim(),
series: document.querySelector(".titles a")?.getAttribute("href"),
pages: images.length,
prev: document.querySelector(".chnav.prev")?.getAttribute("href"),
next: document.querySelector(".chnav.next")?.getAttribute("href"),
listImages: images.map((img) => img.getAttribute("src")),
};
},
};
const naniscans = {
name: "NaniScans",
url: /https?:\/\/(www\.)?(naniscans).com\/chapters\/.+\/read/,
homepage: "https://naniscans.com/",
language: ["English"],
category: "manga",
waitVar: "chapterListRoute",
async run() {
const api = await fetch(
window.location.href.replace("read", "json"),
).then(async (res) => res.json());
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector('a[href^="/titles/"]')
?.getAttribute("href"),
pages: api.pages.length,
prev: unsafeWindow.previousChapterRoute,
next: unsafeWindow.nextChapterRoute,
listImages: api.pages.map((i) => i.address),
};
},
};
const 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",
};
},
};
const olympusscans = {
name: "OlympusScans",
url: /https?:\/\/(www\.)?olympusscans.com\/capitulo\/.+\/.+/,
homepage: "https://olympusscans.com/",
language: ["Spanish"],
category: "manga",
waitVar: "__NUXT__",
run() {
const images =
unsafeWindow.__NUXT__.data[window.location.pathname].chapter.pages;
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector("h1")
?.parentElement?.getAttribute("href"),
pages: images.length,
prev: document
.querySelector(".i-heroicons-chevron-left-20-solid")
?.parentElement?.getAttribute("href"),
next: document
.querySelector(".i-heroicons-chevron-right-20-solid")
?.parentElement?.getAttribute("href"),
listImages: images,
};
},
};
const 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,
};
},
};
const rawdevart = {
name: "RawDevart",
url: /https?:\/\/(www\.)?rawdevart.com\/comic\/.+\/.+\//,
homepage: "https://rawdevart.com",
language: ["Raw"],
category: "manga",
waitVar: "rconfig",
waitEle: "#chapter-list select",
run() {
const chapter = document.querySelector("#chapter-list option:checked");
const images = [...document.querySelectorAll("#img-container img")];
return {
title: unsafeWindow.rconfig.chapterTitle,
series: unsafeWindow.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"),
),
};
},
};
const readcomicsonline = {
name: "ReadComicsOnline",
url: /https?:\/\/(www\.)?readcomicsonline.ru\/comic\/.*\/\d*/,
homepage: "https://readcomicsonline.ru/",
language: ["English"],
category: "comic",
run() {
const images = [...document.querySelectorAll("#all img")];
return {
title: unsafeWindow.title.replace(/ - Page \d+/, ""),
series: document.querySelector("div.pager-cnt a")?.getAttribute("href"),
pages: unsafeWindow.pages.length,
prev: unsafeWindow.prev_chapter,
next: unsafeWindow.next_chapter,
listImages: images.map((img) => img.getAttribute("data-src")),
};
},
};
const 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() {
return {
title: unsafeWindow.chapter_page_title,
series: unsafeWindow.manga_url,
pages: unsafeWindow.images.length,
prev: unsafeWindow.prev_chapter_url,
next: unsafeWindow.next_chapter_url,
listImages: unsafeWindow.images.map((item) => item.url),
};
},
};
const reaperscans = {
name: "ReaperScans",
url: /https?:\/\/(www\.)?reaperscans\.com\/comics\/.+\/chapters\/.+/,
homepage: "https://reaperscans.com/",
language: ["English"],
category: "manga",
waitEle: "main img",
run() {
const images = [...document.querySelectorAll("main img")];
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector(".fa-list")
?.parentElement?.getAttribute("href"),
pages: images.length,
prev: document
.querySelector(".fa-arrow-left-long")
?.parentElement?.getAttribute("href"),
next: document
.querySelector(".fa-arrow-right-long")
?.parentElement?.getAttribute("href"),
listImages: images.map(
(img) => img.getAttribute("data-src") ?? img.getAttribute("src"),
),
};
},
};
const senmanga = {
name: "SenManga(Raw)",
url: /https?:\/\/raw\.senmanga.com\/.+\/.+\/?/,
homepage: "https://raw.senmanga.com/",
language: ["Raw"],
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",
};
},
};
const tapas = {
name: "KLManga",
url: /https?:\/\/(www\.)?tapas.io\/episode\/.+/,
homepage: "https://tapas.io/",
language: ["English"],
category: "manga",
run() {
const images = [
...document.querySelectorAll(".viewer__body img.content__img"),
];
const chapter = document.querySelector(
".js-episodes .body__item--selected",
);
return {
title: document
.querySelector(".viewer__header .title")
?.textContent?.trim(),
series: document.querySelector(".vw-nav__left a")?.getAttribute("href"),
pages: images.length,
prev: chapter?.previousElementSibling?.getAttribute("data-href"),
next: chapter?.nextElementSibling?.getAttribute("data-href"),
listImages: images.map(
(img) => img.getAttribute("data-src") ?? img.getAttribute("src"),
),
};
},
};
const tenmanga = {
name: "TenManga",
url: /https?:\/\/(www\.)?(tenmanga|gardenmanage).com\/(chapter|statuses)\/.+/,
homepage: "https://www.tenmanga.com/",
language: ["English"],
category: "manga",
waitVar: "_pageCtrl",
run() {
const chapter = document.querySelector(
".mangaread-pagenav select option:checked",
);
const images = unsafeWindow._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,
};
},
};
const tmofans = {
name: "TuMangaOnline",
url: /https?:\/\/(www\.)?(.+).com\/(viewer|news)\/.+\/(paginated|cascade)/,
homepage: "https://lectortmo.com/",
language: ["Spanish"],
category: "manga",
run() {
const images = [
...document.querySelectorAll(
".img-container img, .viewer-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"),
...(images.length > 1
? {
listImages: images.map((item) => $(item).attr("data-src")),
}
: {
listPages: Array(pages.length)
.fill(0)
.map(
(_, i) =>
`${window.location.href.replace(/\/\d+$/, "")}/${i + 1}`,
),
img: "#viewer-container img, .viewer-page",
}),
};
},
};
const tumanhwas = {
name: "TuManhwas",
url: /https?:\/\/(www\.)?tumanhwas.com\/news\/.+/,
homepage: "https://tumanhwas.com/",
language: ["Spanish"],
category: "manga",
run() {
const images = [...document.querySelectorAll("#chapter_imgs img")];
return {
title: document.querySelector(".entry-title")?.textContent?.trim(),
series: document
.querySelector(".nextprev a:nth-child(2)")
?.getAttribute("href"),
pages: images.length,
prev: document
.querySelector(".nextprev a:nth-child(1)")
?.getAttribute("href"),
next: document
.querySelector(".nextprev a:nth-child(3)")
?.getAttribute("href"),
listImages: images.map((item) => $(item).attr("src")),
};
},
};
const 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")),
};
},
};
const webnovel = {
name: "WebNovel",
url: /https?:\/\/(www\.)?webnovel.com\/comic\/.+/,
homepage: "https://www.webnovel.com/",
language: ["English"],
category: "manga",
waitVar: "g_data",
run() {
const images = unsafeWindow.g_data.chapter.chapterInfo.chapterPage.map(
(img) => img.url,
);
return {
title: document.querySelector("title")?.textContent?.trim(),
series: "./",
pages: images.length,
prev: `${unsafeWindow.g_data.chapter.chapterInfo.preChapterName}_${unsafeWindow.g_data.chapter.chapterInfo.preChapterId}`,
next: `${unsafeWindow.g_data.chapter.chapterInfo.nextChapterName}_${unsafeWindow.g_data.chapter.chapterInfo.nextChapterId}`,
listImages: images,
};
},
};
const webtoons = {
name: "WebToons",
url: /https?:\/\/(www\.)?webtoons.com\/.+viewer.+/,
homepage: "https://www.webtoons.com/",
language: ["English"],
category: "manga",
run() {
const images = [...document.querySelectorAll("#_imageList img")];
return {
title: document.querySelector(".subj_info")?.textContent?.trim(),
series: document.querySelector(".subj_info a")?.getAttribute("href"),
pages: images.length,
prev: document.querySelector("._prevEpisode")?.getAttribute("href"),
next: document.querySelector("._nextEpisode")?.getAttribute("href"),
listImages: images.map(
(img) =>
img.getAttribute("data-url") ??
img.getAttribute("data-src") ??
img.getAttribute("src"),
),
};
},
};
const 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 (/all.html$/.exec(window.location.pathname)) {
return;
}
if (/\d+-\d+.html$/.exec(window.location.pathname)) {
window.location.pathname = window.location.pathname.replace(
/-\d+.html$/,
"-all.html",
);
}
},
};
},
};
const yugenmangas = {
name: "YugenMangas",
url: /https?:\/\/(www\.)?(yugenmangas).(com|net|lat)\/series\/.+/,
homepage: "https://yugenmangas.lat/",
language: ["Spanish"],
category: "manga",
async run() {
const images = [...document.querySelectorAll("p.flex > img")];
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector("div.justify-between:nth-child(2) > a:nth-child(2)")
?.getAttribute("href"),
pages: images.length,
prev: document
.querySelector("div.justify-between:nth-child(2) > a:nth-child(1)")
?.getAttribute("href"),
next: document
.querySelector("div.justify-between:nth-child(2) > a:nth-child(3)")
?.getAttribute("href"),
listImages: images.map((img) =>
img.classList.contains("lazy")
? img.getAttribute("data-src")
: img.getAttribute("src"),
),
};
},
};
const zeroscans = {
name: "ZeroScans",
url: /https?:\/\/(www\.)?zscans.com\/comics\/.+/,
homepage: "https://zscans.com/",
language: ["English"],
category: "manga",
waitVar: "__ZEROSCANS__",
run() {
const images =
unsafeWindow.__ZEROSCANS__.data.at(0).current_chapter.high_quality;
const chapters = document.querySelectorAll(".v-btn--router");
return {
title: document.querySelector("title")?.textContent?.trim(),
series: document
.querySelector(".v-breadcrumbs li:nth-child(2) + a")
?.getAttribute("href"),
pages: images.length,
prev: chapters[0]?.getAttribute("href"),
next: chapters[1]?.getAttribute("href"),
listImages: images,
};
},
};
const sites = [
alandal,
// Asurascans,
batoto,
bilibilicomics,
comicastle,
dysnatyscans,
mangastreamwp,
inkr,
inmanga,
klmanga,
leitor,
// Leviatanscans,
lhtranslation,
localhost,
lynxscans,
mangabuddy,
mangadex,
mangafox,
mangafreak,
mangago,
mangahosted,
mangahub,
mangasin,
mangakakalot,
mangapark,
mangareader,
mangasee,
mangatigre,
mangatoon,
mangatown,
manhuascan,
mgeko,
naniscans,
ninemanga,
olympusscans,
pandamanga,
rawdevart,
readcomicsonline,
readmangatoday,
reaperscans,
// Resetscans, deprecated
senmanga,
tapas,
tenmanga,
tmofans,
tumanhwas,
unionmangas,
webnovel,
webtoons,
wpmanga,
yugenmangas,
zeroscans,
foolslide,
// Must be at the end because is a generic check
madarawp,
// Must be at the end because is a generic check
];
const rangeSliderStyles =
".range-slider{touch-action:none;-webkit-tap-highlight-color:transparent;-webkit-user-select:none;user-select:none;cursor:pointer;display:block;position:relative;width:100%;height:8px;background:#ddd;border-radius:4px}.range-slider[data-vertical]{height:100%;width:8px}.range-slider[data-disabled]{opacity:.5;cursor:not-allowed}.range-slider .range-slider__thumb{position:absolute;z-index:3;top:50%;width:24px;height:24px;transform:translate(-50%,-50%);border-radius:50%;background:#2196f3}.range-slider .range-slider__thumb:focus-visible{outline:0;box-shadow:0 0 0 6px rgba(33,150,243,.5)}.range-slider[data-vertical] .range-slider__thumb{left:50%}.range-slider .range-slider__thumb[data-disabled]{z-index:2}.range-slider .range-slider__range{position:absolute;z-index:1;transform:translate(0,-50%);top:50%;width:100%;height:100%;background:#51adf6}.range-slider[data-vertical] .range-slider__range{left:50%;transform:translate(-50%,0)}.range-slider input[type=range]{-webkit-appearance:none;pointer-events:none;position:absolute;z-index:2;top:0;left:0;width:0;height:0;background-color:transparent}.range-slider input[type=range]::-webkit-slider-thumb{-webkit-appearance:none;appearance:none}.range-slider input[type=range]::-moz-range-thumb{width:0;height:0;border:0}.range-slider input[type=range]:focus{outline:0}";
function logScript(...text) {
console.log("MangaOnlineViewer: ", ...text);
return text;
}
function getListGM() {
return typeof GM_listValues !== "undefined" ? GM_listValues() : [];
}
function removeValueGM(name) {
if (typeof GM_deleteValue !== "undefined") {
GM_deleteValue(name);
} else {
logScript("Removing: ", name);
}
}
const getInfoGM =
typeof GM_info !== "undefined"
? GM_info
: {
scriptHandler: "Console",
script: {
name: "Debug",
version: "Testing",
},
};
function getValueGM(name, defaultValue = null) {
if (typeof GM_getValue !== "undefined") {
return GM_getValue(name, defaultValue);
}
logScript("Fake Getting: ", name, " = ", defaultValue);
return defaultValue;
}
function getJsonGM(name, defaultValue = null) {
const result = getValueGM(name, defaultValue);
if (typeof result === "string") {
return JSON.parse(result);
}
return result;
}
function getSettings(defaultSettings) {
return getJsonGM("settings", defaultSettings);
}
function setValueGM(name, value) {
try {
GM_setValue(name, value);
return value.toString();
} catch (e) {
logScript("Fake Setting: ", name, " = ", value);
return String(value);
}
}
function setSettings(value) {
return setValueGM("settings", value);
}
function getBrowser() {
let tem;
const M =
/(opera|chrome|safari|firefox|msie|trident(?=\/))\/?\s*(\d+)/i.exec(
navigator.userAgent,
) ?? [];
if (/trident/i.test(M[1])) {
tem = /\brv[ :]+(\d+)/g.exec(navigator.userAgent) ?? [];
return `IE ${tem[1] ?? ""}`;
}
if (M[1] === "Chrome") {
tem = /\b(OPR|Edge)\/(\d+)/.exec(navigator.userAgent);
if (tem !== null) {
return tem.slice(1).join(" ").replace("OPR", "Opera");
}
}
const tempM = [M[1], M[2]];
tem = /version\/(\d+)/i.exec(navigator.userAgent);
if (tem !== null) {
tempM.splice(1, 1, tem[1]);
}
return tempM.join(" ");
}
function getEngine() {
return getInfoGM.scriptHandler ?? "Greasemonkey";
}
const isMobile = () =>
window.matchMedia("screen and (max-width: 768px)").matches;
const diffObj = (changed, original) => {
const changes = (object, base) =>
_.transform(
object,
(result, value, key) => {
if (!_.isEqual(value, base[key])) {
if (_.isArray(value)) {
result[key] = _.difference(value, base[key]);
} else if (_.isObject(value) && _.isObject(base[key])) {
result[key] = changes(value, base[key]);
} else {
result[key] = value;
}
}
},
/* Omit accumulator */
);
return changes(changed, original);
};
const en_US = {
ID: "en_US",
NAME: "English (US)",
STARTING: "Starting Manga OnlineViewer",
RESUME: "Resuming reading from Page ",
WAITING: "Please wait, 3 seconds...",
CHOOSE_BEGINNING: "Choose the Page to start from:",
BUTTON_START: "Start Manga OnlineViewer",
SETTINGS: "Settings",
LANGUAGE: "Language",
COLOR_SCHEME: "Color Scheme",
THEME: "Theme",
THEME_HUE: "Theme Primary Color Hue",
THEME_SHADE: "Theme Primary Color Shade",
DEFAULT_LOAD_MODE: "Default Load Mode",
LOAD_MODE_NORMAL: "Normal(Wait 3 sec)",
LOAD_MODE_ALWAYS: "Always(Immediately)",
LOAD_MODE_NEVER: "Never(Manually)",
LOAD_SPEED: "Load Speed Pages/Second",
DEFAULT_ZOOM: "Default Zoom (between 5 and 200)",
DEFAULT_ZOOM_MODE: "Default Zoom Mode",
MINIMUM_ZOOM:
"Minimum Zoom relative to the width of screen (between 30 and 100)",
ZOOM_STEP: "Zoom Change Step (between 5 and 50)",
DEFAULT_VIEW_MODE: "Default View Mode",
VIEW_MODE_VERTICAL: "Vertical",
VIEW_MODE_LEFT: "Left to Right",
VIEW_MODE_RIGHT: "Right to Left",
VIEW_MODE_WEBCOMIC: "WebComic",
FIT_WIDTH_OVERSIZED: "Fit Width if Oversized",
SHOW_THUMBNAILS: "Show Thumbnails",
HIDE_CONTROLS: "Always Hide Page Controls",
HEADER_TYPE: "Change Header Type",
HEADER_HOVER: "Hover",
HEADER_SCROLL: "Scroll",
HEADER_CLICK: "Click",
HEADER_FIXED: "Fixed",
HEADER_SIMPLE: "Simple",
BUTTON_DOWNLOAD: "Download",
DOWNLOAD_ZIP: "Download Zip file",
DOWNLOAD_IMAGES: "Download Images as Zip Automatically",
BUTTON_NEXT: "Next",
NEXT_CHAPTER: "Next Chapter",
BUTTON_PREVIOUS: "Previous",
PREVIOUS_CHAPTER: "Previous Chapter",
BOOKMARKS: "Bookmarks",
BOOKMARK: "Bookmark",
BOOKMARK_REMOVED: "Bookmark Removed",
BOOKMARK_SAVED: "Bookmark Saved",
BOOKMARK_MESSAGE:
"Next time you open this chapter it will resume from:
Page ##num## (Only ONCE per Bookmark)",
KEYBINDINGS: "Keybindings",
EDIT_KEYBINDS: "Edit KeyBindings",
SAVE_KEYBINDS: "Save KeyBindings",
BUTTON_EDIT: "Edit",
BUTTON_SAVE: "Save",
KEYBIND_RULES: `
Supported Keys
Allowed modifiers: shift, option, alt, ctrl, control, command.
Special keys: backspace, tab, clear, enter, return, esc, escape, space, up, down, left, right, home, end, pageup, pagedown, del, delete, f1 - f19, num_0 - num_9, num_multiply, num_add, num_enter, num_subtract, num_decimal, num_divide.
Examples: a , ctrl+a , shift+a , num_2 , 2
`,
ATTENTION: "Attention",
WARNING: "Warning",
BUTTON_RESET_SETTINGS: "Reset Settings",
SETTINGS_RESET: "Settings have been reset, reload the page to take effect",
LANGUAGE_CHANGED:
"Language has been changed, reload the page to take effect",
AUTO_DOWNLOAD:
"Next time a chapter finish loading you will be prompted to save automatically",
LAZY_LOAD:
"Lazy load is incompatible with zip download, you will not be able to download with this setting ON. Suggestion: Disable Thumbnails to save Bandwidth/Memory.",
LAZY_LOAD_IMAGES_ENABLE: "Enable Lazy Load Images",
LAZY_LOAD_IMAGES: "Lazy Start From Page (between 5 and 100)",
RETURN_CHAPTER_LIST: "Return to Chapter List",
PAGES_LOADED: "Pages Loaded",
GO_TO_PAGE: "Go to Page",
ENLARGE: "Enlarge",
RESTORE: "Restore",
REDUCE: "Restore",
FIT_WIDTH: "Fit Width",
FIT_HEIGHT: "Fit Height",
PERCENT: "Percent",
TOGGLE_CONTROLS: "Toggle page controls",
ZOOM_IN: "Zoom In",
ZOOM_OUT: "Zoom Out",
ZOOM_RESET: "Zoom Reset",
ZOOM_WIDTH: "Zoom to Width",
ZOOM_HEIGHT: "Zoom to Height",
HIDE: "Hide",
RELOAD: "Reload",
SLOWLY: "Slowly",
NORMAL: "Normal",
FAST: "Fast",
EXTREME: "Extreme",
ALL_PAGES: "All Pages",
SPEED_WARNING: "Loading Speed too High",
SPEED_WARNING_MESSAGE:
"This speed is not recommended. It may hurt some servers or get your IP marked as DDoS attacker. Please use with caution!",
SCROLL_UP: "Scroll Up",
SCROLL_DOWN: "Scroll Down",
CLOSE: "Close",
LIST_EMPTY: "List Empty",
DISPLAY_COMMENTS: "Display Comments",
COMMENTS: "Comments Section",
SCROLL_START: "Toggle Auto Scroll",
AUTO_SCROLL_HEIGHT: "Auto Scroll Speed in Pixels",
};
const pt_BR = {
ID: "pt_BR",
NAME: "Portugues (Brasil)",
STARTING: "Iniciando Manga OnlineViewer",
RESUME: "Continuando leitura na Pagina ",
WAITING: "Por Favor espere, 3 segundos...",
CHOOSE_BEGINNING: "Escolha a pagina de onde começar:",
BUTTON_START: "Iniciar Manga OnlineViewer",
SETTINGS: "Configurações",
LANGUAGE: "Idioma",
COLOR_SCHEME: "Esquema de Color",
THEME: "Tema",
THEME_HUE: "Coloração primaria",
THEME_SHADE: "Saturação de Cor",
DEFAULT_LOAD_MODE: "Forma de Carregamento Padrão",
LOAD_MODE_NORMAL: "Normal(Esperando 3 sec)",
LOAD_MODE_ALWAYS: "Sempre(Imediatamente)",
LOAD_MODE_NEVER: "Nunca(Manualmente)",
LOAD_SPEED: "Velocidade de Carregamento Paginas/Segundo",
DEFAULT_ZOOM: "Zoom padrão (entre 5 e 200)",
DEFAULT_ZOOM_MODE: "Modo de Zoom padrão",
MINIMUM_ZOOM: "Zoom minimo, relativo ao tamanho da tela (entre 30 e 100)",
ZOOM_STEP: "Precisão da Mudança do Zoom (entre 5 e 50)",
DEFAULT_VIEW_MODE: "Modo de Visualização Padrão",
VIEW_MODE_VERTICAL: "Vertical",
VIEW_MODE_LEFT: "Esquerda para Direita",
VIEW_MODE_RIGHT: "Direita para Esquerda",
VIEW_MODE_WEBCOMIC: "WebComic",
FIT_WIDTH_OVERSIZED: "Encher a tela se grande demais",
SHOW_THUMBNAILS: "Mostra Miniaturas",
HIDE_CONTROLS: "Sempre esconder controles das paginas",
HEADER_TYPE: "Mudar Tipo de Cabeçalho",
HEADER_HOVER: "Passar por perto",
HEADER_SCROLL: "Rolagem do Mouse",
HEADER_CLICK: "Click",
HEADER_FIXED: "Fixo",
HEADER_SIMPLE: "Simples",
BUTTON_DOWNLOAD: "Download",
DOWNLOAD_ZIP: "Baixar arquivo Zip",
DOWNLOAD_IMAGES: "Download das Imagens como Zip Automaticamente",
BUTTON_NEXT: "Proximo",
NEXT_CHAPTER: "Proximo Capitulo",
BUTTON_PREVIOUS: "Anterior",
PREVIOUS_CHAPTER: "Capitulo Anterior",
BOOKMARKS: "Marca paginas",
BOOKMARK: "Marcar pagina",
BOOKMARK_REMOVED: "Marca pagina Removido",
BOOKMARK_SAVED: "Marca pagina Salvo",
BOOKMARK_MESSAGE:
"Proxima vez que abrir este capitulo continuará a partir da Pagina ##num## (Apenas UMA VEZ por marca pagina)",
KEYBINDINGS: "Atalhos",
EDIT_KEYBINDS: "Editar Atalhos",
SAVE_KEYBINDS: "Salvar Atalhos",
BUTTON_EDIT: "Editar",
BUTTON_SAVE: "Salvar",
KEYBIND_RULES: `
Teclas Suportadas
Modificadores permitidos: shift, option, alt, ctrl, control, command.
Teclas Especiais: backspace, tab, clear, enter, return, esc, escape, space, up, down, left, right, home, end, pageup, pagedown, del, delete, f1 - f19, num_0 - num_9, num_multiply, num_add, num_enter, num_subtract, num_decimal, num_divide.
Exemplos: a , ctrl+a , shift+a , num_2 , 2
`,
ATTENTION: "Atenção",
WARNING: "Alerta",
BUTTON_RESET_SETTINGS: "Limpar Configurações(Reset Settings)",
SETTINGS_RESET:
"Configurações foram limpas, recarregue o site para efetivar a alteração",
LANGUAGE_CHANGED:
"Idioma foi alterado, recarregue o site para efetivar a alteração",
AUTO_DOWNLOAD:
"Proxima vez que abrir um capitulo download iniciara automaticamente",
LAZY_LOAD:
"Carregamento preguiçoso não é compativel com download de zip, não conseguira com essa configuração ativa. Sugestão: Desative Miniaturas para economizar memoria e cota de internet.",
LAZY_LOAD_IMAGES_ENABLE: "Ativar Carregamento de imagens preguiçoso",
LAZY_LOAD_IMAGES:
"Carregamento de paginas preguiçoso começa a partir de (entre 5 e 100)",
RETURN_CHAPTER_LIST: "Voltar a lista de Capitulos",
PAGES_LOADED: "Paginas Carregadas",
GO_TO_PAGE: "Pular para",
ENLARGE: "Aumentar",
RESTORE: "Restaurar",
REDUCE: "Diminuir",
FIT_WIDTH: "Preencher Largura",
FIT_HEIGHT: "Preencher Altura ",
PERCENT: "Percentual",
TOGGLE_CONTROLS: "Mostar controles de pagina",
ZOOM_IN: "Mais Zoom",
ZOOM_OUT: "Menos Zoom",
ZOOM_RESET: "Resetar Zoom",
ZOOM_WIDTH: "Zoom para Largura",
ZOOM_HEIGHT: "Zoom para Altura",
HIDE: "Esconder",
RELOAD: "Recarregar",
SLOWLY: "Devagar",
NORMAL: "Normal",
FAST: "Rapido",
EXTREME: "Extremo",
ALL_PAGES: "Todas as Paginas",
SPEED_WARNING: "Velocidade de Carregamento muito alta",
SPEED_WARNING_MESSAGE:
"Essa velocidade não é recomendada. Ela pode derrubar um servidor or marcar voce como um ataque hacker de DDoS. Use com cuidado!",
SCROLL_UP: "Subir Pagina",
SCROLL_DOWN: "Descer Pagina",
CLOSE: "Fechar",
LIST_EMPTY: "Lista Vazia",
DISPLAY_COMMENTS: "Mostar Comentarios",
COMMENTS: "Seção de comentários",
SCROLL_START: "Ativar Rolagem Automatica",
AUTO_SCROLL_HEIGHT: "Velocidade da Rolagem Automatica em Pixels",
};
const zh_CN = {
ID: "zh_CN",
NAME: "中文 (简体)",
STARTING: "正在启动 Manga OnlineViewer",
RESUME: "从页面继续阅读 ",
WAITING: "请等待3秒钟...",
CHOOSE_BEGINNING: "选择要开始的页数:",
BUTTON_START: "启动Manga OnlineViewer",
SETTINGS: "设置",
LANGUAGE: "语言",
COLOR_SCHEME: "配色方案",
THEME: "主题",
THEME_HUE: "主题色调",
THEME_SHADE: "主题阴影",
DEFAULT_LOAD_MODE: "默认加载模式",
LOAD_MODE_NORMAL: "等待模式(等待3秒自动加载 )",
LOAD_MODE_ALWAYS: "自动模式(无需等待)",
LOAD_MODE_NEVER: "手动模式(点击启动)",
LOAD_SPEED: "加载速度页数/秒",
DEFAULT_ZOOM: "默认缩放 (最小 5 最大 200)",
DEFAULT_ZOOM_MODE: "默认缩放模式",
MINIMUM_ZOOM: "相对于屏幕宽度的最小缩放 (最小 30 最大 100)",
ZOOM_STEP: "缩放级别 (最小 5 最大 50)",
DEFAULT_VIEW_MODE: "默认视图模式",
VIEW_MODE_VERTICAL: "垂直有缝",
VIEW_MODE_LEFT: "从左到右",
VIEW_MODE_RIGHT: "从右到左",
VIEW_MODE_WEBCOMIC: "垂直无缝",
FIT_WIDTH_OVERSIZED: "如果尺寸过大、则适合宽度",
SHOW_THUMBNAILS: "显示缩略图",
HIDE_CONTROLS: "始终隐藏页面控件",
HEADER_TYPE: "更改标题显示方式",
HEADER_HOVER: "悬停",
HEADER_SCROLL: "滚动",
HEADER_CLICK: "点击",
HEADER_FIXED: "固定",
HEADER_SIMPLE: "简单",
BUTTON_DOWNLOAD: "下载",
DOWNLOAD_ZIP: "下载压缩文件",
DOWNLOAD_IMAGES: "自动将图片下载成ZIP",
BUTTON_NEXT: "下一页",
NEXT_CHAPTER: "下一章",
BUTTON_PREVIOUS: "上一页",
PREVIOUS_CHAPTER: "上一章",
BOOKMARKS: "书签",
BOOKMARK: "Bookmark",
BOOKMARK_REMOVED: "删除书签",
BOOKMARK_SAVED: "保存书签",
BOOKMARK_MESSAGE:
"下次打开本章时,将从:页码 ##num## (仅一次 每个书签)",
KEYBINDINGS: "快捷键",
EDIT_KEYBINDS: "编辑键绑定",
SAVE_KEYBINDS: "保存键绑定",
BUTTON_EDIT: "编辑",
BUTTON_SAVE: "救",
KEYBIND_RULES: `
支持的密钥
允许的修饰符: shift, option, alt, ctrl, control, command.
特殊键: backspace, tab, clear, enter, return, esc, escape, space, up, down, left, right, home, end, pageup, pagedown, del, delete, f1 - f19, num_0 - num_9, num_multiply, num_add, num_enter, num_subtract, num_decimal, num_divide.
例子: a , ctrl+a , shift+a , num_2 , 2
`,
ATTENTION: "注意",
WARNING: "警告",
BUTTON_RESET_SETTINGS: "重置设置(Reset Settings)",
SETTINGS_RESET: "设置已重置、重新加载页面才能生效",
LANGUAGE_CHANGED: "语言已更改、重新加载页面才能生效",
AUTO_DOWNLOAD: "下次章节加载完成时、系统将提示您自动保存",
LAZY_LOAD:
"延迟加载与zip下载不兼容、您将无法使用此设置下载. 建议: 禁用缩略图 以节省流量和内存.",
LAZY_LOAD_IMAGES_ENABLE: "启用延迟加载图像",
LAZY_LOAD_IMAGES: "惰性加载从页面 (最小 5 最大 100)",
RETURN_CHAPTER_LIST: "返回章节列表",
PAGES_LOADED: "已加载的页数",
GO_TO_PAGE: "转到页数",
ENLARGE: "放大",
RESTORE: "还原",
REDUCE: "缩小",
FIT_WIDTH: "适合宽度",
FIT_HEIGHT: "适合高度",
PERCENT: "百分之",
TOGGLE_CONTROLS: "显示隐藏页面控件",
ZOOM_IN: "放大",
ZOOM_OUT: "缩小",
ZOOM_RESET: "还原",
ZOOM_WIDTH: "适合宽度",
ZOOM_HEIGHT: "适合高度",
HIDE: "显示隐藏页面控件",
RELOAD: "重新加载",
SLOWLY: "慢速",
NORMAL: "正常",
FAST: "快速",
EXTREME: "极端",
ALL_PAGES: "所有页面",
SPEED_WARNING: "加载速度过高",
SPEED_WARNING_MESSAGE:
"不建议使用此速度. 它可能会伤害某些服务器或将您的 IP 标记为 DDoS 攻击者. 请谨慎使用!",
SCROLL_UP: "向上滚动",
SCROLL_DOWN: "向下滚动",
CLOSE: "关闭",
LIST_EMPTY: "没有收藏书签",
DISPLAY_COMMENTS: "显示注释",
COMMENTS: "评论部分",
SCROLL_START: "切换自动滚动",
AUTO_SCROLL_HEIGHT: "自动滚动速度(以像素为单位)",
};
const es_ES = {
ID: "es_ES",
NAME: "Español (ES)",
STARTING: "Iniciando Manga OnlineViewer",
RESUME: "Continuando lectura desde la Página ",
WAITING: "Por favor espere, 3 segundos...",
CHOOSE_BEGINNING: "Elija la página en la que comenzar:",
BUTTON_START: "Iniciar Manga OnlineViewer",
SETTINGS: "Ajustes",
LANGUAGE: "Idioma",
COLOR_SCHEME: "Esquema de color",
THEME: "Tema",
THEME_HUE: "Matiz del color primario",
THEME_SHADE: "Saturación del color primario",
DEFAULT_LOAD_MODE: "Modo de carga por defecto",
LOAD_MODE_NORMAL: "Normal (Espera 3s)",
LOAD_MODE_ALWAYS: "Siempre (Inmediatamente)",
LOAD_MODE_NEVER: "Nunca (Manualmente)",
LOAD_SPEED: "Velocidad carga página/segundo",
DEFAULT_ZOOM: "Zoom por defecto (entre 5 y 200)",
DEFAULT_ZOOM_MODE: "Modo de zoom por defecto",
MINIMUM_ZOOM: "Zoom mínimo relativo al ancho de la pantalla",
ZOOM_STEP: "Paso entre cambios de zoom (entre 5 y 50)",
DEFAULT_VIEW_MODE: "Modo de visualización por defecto",
VIEW_MODE_VERTICAL: "Vertical",
VIEW_MODE_LEFT: "Izquierda a derecha",
VIEW_MODE_RIGHT: "Derecha a izquierda",
VIEW_MODE_WEBCOMIC: "WebComic",
FIT_WIDTH_OVERSIZED: "Ajustar ancho si es demasiado grande",
SHOW_THUMBNAILS: "Mostrar miniaturas",
HIDE_CONTROLS: "Ocultar siempre la barra de controles",
HEADER_TYPE: "Cambiar tipo de cabecera",
HEADER_HOVER: "Pasar por encima",
HEADER_SCROLL: "Desplazamiento",
HEADER_CLICK: "Hacer click",
HEADER_FIXED: "Fijo",
HEADER_SIMPLE: "Sencillo",
BUTTON_DOWNLOAD: "Descargar",
DOWNLOAD_ZIP: "Descargar fichero Zip",
DOWNLOAD_IMAGES: "Autodescargar imágenes como Zip",
BUTTON_NEXT: "Siguiente",
NEXT_CHAPTER: "Siguiente capítulo",
BUTTON_PREVIOUS: "Anterior",
PREVIOUS_CHAPTER: "Capítulo anterior",
BOOKMARKS: "Marcadores",
BOOKMARK: "Marcador",
BOOKMARK_REMOVED: "Marcador eliminado",
BOOKMARK_SAVED: "Marcador guardado",
BOOKMARK_MESSAGE:
"La próxima vez que abra este capítulo, continuará desde la página ##num## (Sólo UNA VEZ por Marcador)",
KEYBINDINGS: "Atajos de teclado",
EDIT_KEYBINDS: "Editar atajos",
SAVE_KEYBINDS: "Guardar atajos",
BUTTON_EDIT: "Editar",
BUTTON_SAVE: "Guardar",
KEYBIND_RULES: `
Teclas soportadas
Modificadores permitidos: shift, option, alt, ctrl, control, command.
Teclas especiales: backspace, tab, clear, enter, return, esc, escape, space, up, down, left, right, home, end, pageup, pagedown, del, delete, f1 - f19, num_0 - num_9, num_multiply, num_add, num_enter, num_subtract, num_decimal, num_divide.
Ejemplos: a , ctrl+a , shift+a , num_2 , 2
`,
ATTENTION: "Atención",
WARNING: "Alerta",
BUTTON_RESET_SETTINGS: "Reiniciar ajustes(Reset Settings)",
SETTINGS_RESET:
"Se han restablecido los ajustes, vuelve a cargar la página para que surta efecto",
LANGUAGE_CHANGED:
"Se ha cambiado el idioma, vuelve a cargar la página para que surta efecto",
AUTO_DOWNLOAD:
"La próxima vez que termine de cargarse un capítulo, se le pedirá que guarde automáticamente",
LAZY_LOAD:
"La carga diferida es incompatible con la descarga zip, no podrá descargar con este ajuste activado. Sugerencia: Desactivar miniaturas para ahorrar Ancho de banda/Memoria.",
LAZY_LOAD_IMAGES_ENABLE: "Habilitar carga de imágenes diferida",
LAZY_LOAD_IMAGES:
"Empezar carga diferida a partir de la página (entre 5 y 100)",
RETURN_CHAPTER_LIST: "Regresar a la lista de capítulos",
PAGES_LOADED: "Páginas cargadas",
GO_TO_PAGE: "Ir a página",
ENLARGE: "Agrandar",
RESTORE: "Restaurar",
REDUCE: "Reducir",
FIT_WIDTH: "Ajustar al ancho",
FIT_HEIGHT: "Ajustar al alto",
PERCENT: "Porcentual",
TOGGLE_CONTROLS: "Alternar controles de página",
ZOOM_IN: "Acercar",
ZOOM_OUT: "Alejar",
ZOOM_RESET: "Restablecer zoom",
ZOOM_WIDTH: "Zoom al ancho",
ZOOM_HEIGHT: "Zoom al alto",
HIDE: "Ocultar",
RELOAD: "Recargar",
SLOWLY: "Lento",
NORMAL: "Normal",
FAST: "Rápido",
EXTREME: "Extremo",
ALL_PAGES: "Todas las páginas",
SPEED_WARNING: "Velocidad de carga muy alta",
SPEED_WARNING_MESSAGE:
"No se recomienda esta velocidad. Puede dañar algunos servidores o marcar su IP como atacante DDoS. ¡Utilícelo con precaución!",
SCROLL_UP: "Desplazar arriba",
SCROLL_DOWN: "Desplazar abajo",
CLOSE: "Cerrar",
LIST_EMPTY: "Lista vacía",
DISPLAY_COMMENTS: "Mostrar comentarios",
COMMENTS: "Sección de comentarios",
SCROLL_START: "Alternar desplazamiento automático",
AUTO_SCROLL_HEIGHT: "Velocidad de desplazamiento automático en píxeles",
};
const locales = [en_US, es_ES, pt_BR, zh_CN];
function isEmpty(value) {
return (
value === null || // Check for null
typeof value === "undefined" ||
value === void 0 || // 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)
);
}
function isNothing(value) {
const isEmptyObject = (a) => {
if (!Array.isArray(a)) {
const hasNonempty = Object.keys(a).some(
(element) => !isNothing(a[element]),
);
return hasNonempty ? false : isEmptyObject(Object.keys(a));
}
return !a.some(
(element) => !isNothing(element),
//
);
};
return (
// eslint-disable-next-line eqeqeq
value == false ||
value === 0 ||
isEmpty(value) ||
(typeof value === "object" && isEmptyObject(value))
);
}
const defaultSettings = {
locale: "en_US",
theme: "darkblue",
customTheme: "#263e3a",
themeShade: 600,
colorScheme: "dark",
fitWidthIfOversize: true,
showThumbnails: true,
downloadZip: false,
throttlePageLoad: 1e3,
zoomMode: "percent",
defaultZoom: 100,
zoomStep: 25,
minZoom: 30,
loadMode: "wait",
viewMode: "WebComic",
bookmarks: [],
lazyLoadImages: false,
lazyStart: 50,
hidePageControls: false,
header: "hover",
maxReload: 5,
scrollHeight: 20,
keybinds: {
SCROLL_UP: ["up", "W", "num_8"],
SCROLL_DOWN: ["down", "S", "num_2"],
NEXT_CHAPTER: ["right", "/", "D", "num_6"],
PREVIOUS_CHAPTER: ["left", ";", "A", "num_4"],
ENLARGE: ["-", "num_add", "E"],
REDUCE: ["=", "num_subtract", "Q"],
RESTORE: ["9", "num_divide", "R"],
FIT_WIDTH: ["0", "num_multiply", "F"],
FIT_HEIGHT: ["H"],
SETTINGS: ["num_divide", "num_5", "X"],
VIEW_MODE_WEBCOMIC: ["C"],
VIEW_MODE_VERTICAL: ["V"],
VIEW_MODE_LEFT: ["N"],
VIEW_MODE_RIGHT: ["B"],
SCROLL_START: ["space"],
},
};
let settings$2 = _.defaultsDeep(
getSettings(defaultSettings),
defaultSettings,
);
if (isMobile()) {
settings$2.lazyLoadImages = true;
settings$2.fitWidthIfOversize = true;
settings$2.showThumbnails = false;
settings$2.viewMode = "WebComic";
settings$2.header = "click";
}
function getUserSettings() {
return settings$2;
}
function getLocaleString(name) {
const locale = locales.find((l) => l.ID === settings$2.locale);
if (locale?.[name]) {
return locale[name];
}
if (locales?.at(1)?.[name]) {
return locales[1][name];
}
return "##MISSING_STRING##";
}
function updateSettings(newValue) {
logScript(JSON.stringify(newValue));
settings$2 = { ...settings$2, ...newValue };
setSettings(diffObj(settings$2, defaultSettings));
}
function resetSettings() {
getListGM().forEach((setting) => {
removeValueGM(setting);
});
updateSettings(defaultSettings);
}
function isBookmarked(url = window.location.href) {
return settings$2.bookmarks.find((el) => el.url === url)?.page;
}
const bookmarkTimeLimit = 1e3 * 60 * 60 * 24 * 30 * 12;
const refreshedBookmark = settings$2.bookmarks.filter(
(el) => Date.now() - new Date(el.date).valueOf() < bookmarkTimeLimit,
);
if (settings$2.bookmarks.length !== refreshedBookmark.length) {
updateSettings({ bookmarks: refreshedBookmark });
}
function clearBookmark(url = window.location.href) {
if (!isNothing(isBookmarked())) {
logScript(`Bookmark Removed ${window.location.href}`);
updateSettings({
bookmarks: settings$2.bookmarks.filter((el) => el.url !== url),
});
}
}
const IconArrowAutofitDown =
'\n \n \n \n \n \n\n\n';
const IconArrowAutofitHeight =
'\n \n \n \n \n \n \n \n\n\n';
const IconArrowAutofitLeft =
'\n \n \n \n \n \n\n\n';
const IconArrowAutofitRight =
'\n \n \n \n \n \n\n\n';
const IconArrowAutofitWidth =
'\n \n \n \n \n \n \n \n\n\n';
const IconArrowBigLeft =
'\n \n \n \n\n\n';
const IconArrowBigRight =
'\n \n \n \n\n\n';
const IconBookmark =
'\n \n \n \n\n\n';
const IconBookmarkOff =
'\n \n \n \n \n\n\n';
const IconBookmarks =
'\n \n \n \n \n\n\n';
const IconCategory =
'\n \n \n \n \n \n \n\n\n';
const IconCheck =
'\n \n \n \n\n\n';
const IconDeviceFloppy =
'\n \n \n \n \n \n\n\n';
const IconExternalLink =
'\n \n \n \n \n \n\n\n';
const IconEye =
'\n \n \n \n \n\n\n';
const IconEyeOff =
'\n \n \n \n \n \n\n\n';
const IconFileDownload =
'\n \n \n \n \n \n \n\n\n';
const IconKeyboard =
'\n \n \n \n \n \n \n \n \n \n \n\n\n';
const IconListNumbers =
'\n \n \n \n \n \n \n \n\n\n';
const IconLoader2 =
'\n \n \n \n\n\n';
const IconMenu2 =
'\n\n\n';
const IconMessage =
'\n \n \n \n \n \n\n\n';
const IconMoon =
'\n \n \n \n\n\n';
const IconPalette =
'\n \n \n \n \n \n \n\n\n';
const IconPencil =
'\n \n \n \n \n\n\n';
const IconPhoto =
'\n \n \n \n \n \n \n\n\n';
const IconPhotoOff =
'\n \n \n \n \n \n \n \n\n\n';
const IconRefresh =
'\n \n \n \n \n\n\n';
const IconSettings =
'\n \n \n \n \n\n\n';
const IconSpacingVertical =
'\n \n \n \n \n \n\n\n';
const IconSun =
'\n \n \n \n \n\n\n';
const IconTrash =
'\n \n \n \n \n \n \n \n\n\n';
const IconX =
'\n \n \n \n \n\n\n';
const IconZoomCancel =
'\n \n \n \n \n \n \n\n\n';
const IconZoomIn =
'\n \n \n \n \n \n \n\n\n';
const IconZoomInArea =
'\n \n \n \n \n \n \n \n \n \n \n \n\n\n';
const IconZoomOut =
'\n \n \n \n \n \n\n\n';
const IconZoomOutArea =
'\n \n \n \n \n \n \n \n \n \n \n\n\n';
const IconZoomPan =
'\n \n \n \n \n \n \n \n \n\n\n';
const IconPlayerPlay =
'\n \n \n \n\n\n';
const IconPlayerPause =
'\n \n \n \n \n\n\n';
const styles =
":root {\n --theme-body-background: #25262b;\n --theme-body-text-color: #c1c2c5;\n --theme-text-color: #c1c2c5;\n --theme-primary-color: #1a1b1e;\n --theme-primary-text-color: #c1c2c5;\n --theme-background-color: #25262b;\n --theme-hightlight-color: #2c2e33;\n --theme-border-color: #373a40;\n}\n\n#MangaOnlineViewer {\n height: 100%;\n width: 100%;\n text-decoration: none;\n color: var(--theme-body-text-color);\n background-color: var(--theme-body-background);\n}\n\n#MangaOnlineViewer #Chapter {\n display: grid;\n grid-template-columns: repeat(1, 1fr);\n min-width: 225px;\n}\n\n#MangaOnlineViewer #Chapter.Vertical,\n#MangaOnlineViewer #Chapter.WebComic {\n padding-bottom: 40px;\n}\n\n#MangaOnlineViewer #Chapter.Vertical .PageContent {\n margin-bottom: 15px;\n}\n\n#MangaOnlineViewer .closeButton {\n width: fit-content;\n height: fit-content;\n position: absolute;\n right: 10px;\n top: 10px;\n}\n\n#MangaOnlineViewer .overlay {\n position: fixed;\n display: none;\n width: 100%;\n height: 100%;\n top: 0;\n left: 0;\n right: 0;\n bottom: 0;\n background-color: rgba(0, 0, 0, 0.5);\n z-index: 950;\n cursor: pointer;\n}\n\n#MangaOnlineViewer .overlay.visible {\n display: block;\n}\n\n#MangaOnlineViewer select {\n height: 20px;\n padding: 0;\n margin-bottom: 5px;\n}\n\n#MangaOnlineViewer .ControlButton {\n cursor: pointer;\n border-radius: 5px;\n border-width: 1px;\n padding: 2px;\n min-height: 32px;\n color: var(--theme-primary-text-color);\n background-color: var(--theme-primary-color);\n border-color: var(--theme-border-color);\n}\n\n#MangaOnlineViewer .ControlButton:active,\n#MangaOnlineViewer .ControlButton:hover {\n opacity: 0.8;\n}\n\n#MangaOnlineViewer .panel {\n padding: 5px;\n position: inherit;\n border-radius: 5px;\n background-color: var(--theme-background-color);\n}\n\n#MangaOnlineViewer .fitWidthIfOversize .PageContent .PageImg {\n max-width: 100%;\n}\n\n#MangaOnlineViewer .ControlButton.hidden,\n#MangaOnlineViewer.light #ColorScheme > .icon-tabler-sun,\n#MangaOnlineViewer:not(.light) #ColorScheme > .icon-tabler-moon,\n#MangaOnlineViewer .ChapterControl #download.loading > .icon-tabler-file-download,\n#MangaOnlineViewer .ChapterControl #download:not(.loading) > .icon-tabler-loader-2,\n#MangaOnlineViewer .MangaPage.hide .ControlButton.Hide > .icon-tabler-eye-off,\n#MangaOnlineViewer .MangaPage:not(.hide) .ControlButton.Hide > .icon-tabler-eye,\n#MangaOnlineViewer.bookmarked .ControlButton.Bookmark > .icon-tabler-bookmark,\n#MangaOnlineViewer:not(.bookmarked) .ControlButton.Bookmark > .icon-tabler-bookmark-off,\n#MangaOnlineViewer #AutoScroll.running > .icon-tabler-player-play,\n#MangaOnlineViewer #AutoScroll:not(.running) > .icon-tabler-player-pause {\n display: none;\n}\n\n#MangaOnlineViewer.hideControls .PageFunctions {\n visibility: hidden;\n}\n";
const icons =
".icon-tabler {\n height: 1rem;\n width: 1rem;\n vertical-align: sub;\n}\n\n.icon-tabler-file-download > :nth-child(n + 4) {\n /* 4, 5 */\n color: gold;\n}\n.icon-tabler-arrow-autofit-width > :nth-child(n + 3) {\n /* 3,4,5,6 */\n color: yellow;\n}\n.icon-tabler-arrow-autofit-height > :nth-child(n + 3) {\n /* 3,4,5,6 */\n color: yellow;\n}\n.icon-tabler-zoom-in-area > :nth-child(2),\n.icon-tabler-zoom-in-area > :nth-child(3) {\n color: lime;\n}\n.icon-tabler-zoom-out-area > :nth-child(2) {\n color: red;\n}\n.icon-tabler-zoom-pan > :nth-child(n + 4) {\n color: #9966ff;\n}\n.icon-tabler-arrow-autofit-down > :nth-child(n + 3) {\n color: #28ffbf;\n}\n.icon-tabler-arrow-autofit-left > :nth-child(n + 3) {\n color: #28ffbf;\n}\n.icon-tabler-arrow-autofit-right > :nth-child(n + 3) {\n color: #28ffbf;\n}\n.icon-tabler-spacing-vertical > :nth-child(4) {\n color: fuchsia;\n}\n.icon-tabler-list-numbers > :nth-child(n + 5) {\n color: #e48900;\n}\n.icon-tabler-bookmarks > :nth-child(n + 2) {\n color: orange;\n}\n.icon-tabler-bookmark > * {\n color: orange;\n}\n.icon-tabler-bookmark-off > * {\n color: orange;\n}\n.icon-tabler-bookmark-off > :nth-child(3) {\n color: red;\n}\n.icon-tabler-eye-off > :nth-child(4) {\n color: red;\n}\n.icon-tabler-zoom-cancel > :nth-child(3),\n.icon-tabler-zoom-cancel > :nth-child(4) {\n color: #9966ff;\n}\n.icon-tabler-zoom-in > :nth-child(3),\n.icon-tabler-zoom-in > :nth-child(4) {\n color: lime;\n}\n.icon-tabler-zoom-out > :nth-child(3) {\n color: red;\n}\n.icon-tabler-refresh > :nth-child(n + 2) {\n color: cyan;\n}\n.icon-tabler-photo > * {\n color: silver;\n}\n.icon-tabler-photo-off > * {\n color: silver;\n}\n.icon-tabler-photo-off > :nth-child(5) {\n color: orange;\n}\n.icon-tabler-message > :nth-child(2),\n.icon-tabler-message > :nth-child(3) {\n color: greenyellow;\n}\n";
const normalize$1 =
"/* Simple Normalizer */\nhtml {\n font-size: 100%;\n}\n\nbody {\n margin: 0;\n font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif;\n font-size: 14px;\n line-height: 20px;\n color: var(--theme-body-text-color);\n background-color: var(--theme-body-background);\n padding: 0;\n}\n\na,\na:link,\na:visited,\na:active,\na:focus {\n color: var(--theme-body-text-color);\n text-decoration: none;\n}\n\nimg {\n height: auto;\n vertical-align: middle;\n border: 0 none;\n}\n";
const media =
"@media (max-width: 992px) {\n #MangaOnlineViewer #Header {\n display: flex;\n flex-direction: row;\n flex-wrap: wrap;\n }\n\n #MangaOnlineViewer #Header.click {\n padding-left: initial;\n }\n\n #MangaOnlineViewer #Header.click.headroom-end {\n transform: translateY(0%);\n position: sticky;\n top: 0;\n }\n\n #MangaOnlineViewer .PageContent .PageImg {\n max-width: 100%;\n }\n\n #MangaOnlineViewer .ViewerTitle {\n order: 1;\n min-height: auto;\n padding: 0;\n margin: 0;\n flex-grow: 1;\n flex-shrink: 1;\n flex-basis: 100%;\n }\n\n #MangaOnlineViewer #GlobalFunctions {\n width: auto;\n order: 2;\n padding: 5px;\n }\n\n #MangaOnlineViewer #ChapterNavigation {\n order: 3;\n }\n\n #MangaOnlineViewer #GlobalFunctions #ZoomSlider,\n #MangaOnlineViewer #GlobalFunctions .ControlButton:not(.tablets, .phones) {\n display: none;\n }\n}\n\n/* Small devices (portrait phones) */\n@media (max-width: 600px) {\n #MangaOnlineViewer #Header {\n flex-direction: row;\n flex-wrap: wrap;\n justify-content: center;\n align-items: center;\n }\n\n #MangaOnlineViewer #Header.click {\n position: sticky;\n }\n\n #MangaOnlineViewer #MangaTitle {\n word-wrap: anywhere;\n }\n\n #MangaOnlineViewer .ViewerTitle {\n order: 1;\n margin-top: 0;\n height: auto;\n padding: 0;\n }\n\n #MangaOnlineViewer #GlobalFunctions {\n order: 2;\n padding: 0;\n width: auto;\n flex-basis: 35px;\n }\n\n #MangaOnlineViewer #ChapterNavigation {\n order: 3;\n width: min-content;\n min-width: 205px;\n }\n\n #MangaOnlineViewer .ChapterControl {\n flex-direction: row;\n flex-wrap: wrap;\n }\n\n #MangaOnlineViewer .ChapterControl .NavigationControlButton {\n flex-grow: 1;\n }\n\n #MangaOnlineViewer .PageFunctions {\n padding: 0;\n }\n\n #MangaOnlineViewer .PageFunctions .ControlButton.Bookmark {\n opacity: 1;\n }\n\n #MangaOnlineViewer .PageContent {\n margin: 0;\n width: 100%;\n }\n\n #MangaOnlineViewer .PageContent .PageImg {\n max-width: 100%;\n }\n\n #MangaOnlineViewer #Navigation,\n #MangaOnlineViewer #GlobalFunctions #ZoomSlider,\n #MangaOnlineViewer #GlobalFunctions .ControlButton:not(.phones),\n #MangaOnlineViewer .PageFunctions .ControlButton:not(.Bookmark),\n #MangaOnlineViewer #SettingsPanel .DefaultZoomMode,\n #MangaOnlineViewer #SettingsPanel .DefaultZoom,\n #MangaOnlineViewer #SettingsPanel .viewMode,\n #MangaOnlineViewer #SettingsPanel .fitIfOversize,\n #MangaOnlineViewer #SettingsPanel .showThumbnails,\n #MangaOnlineViewer #SettingsPanel .lazyLoadImages,\n #MangaOnlineViewer #SettingsPanel .downloadZip,\n #MangaOnlineViewer #SettingsPanel .minZoom,\n #MangaOnlineViewer #SettingsPanel .zoomStep,\n #MangaOnlineViewer #SettingsPanel .headerType,\n #MangaOnlineViewer #SettingsPanel .autoScroll,\n #MangaOnlineViewer #KeybindingsPanel,\n #MangaOnlineViewer .ChapterControl .download,\n #MangaOnlineViewer #Counters {\n display: none;\n }\n}\n";
const animation =
"@-webkit-keyframes spin {\n to {\n transform: rotate(360deg);\n }\n}\n\n@keyframes spin {\n to {\n transform: rotate(360deg);\n }\n}\n\n@-webkit-keyframes spin-reverse {\n 0% {\n transform: rotate(360deg);\n }\n\n to {\n transform: rotate(0);\n }\n}\n\n@keyframes spin-reverse {\n 0% {\n transform: rotate(360deg);\n }\n\n to {\n transform: rotate(0);\n }\n}\n\n.icon-tabler-loader-2,\n.animate-spin {\n -webkit-animation: spin 1s linear infinite;\n animation: spin 1s linear infinite;\n}\n\n.animate-spin-reverse {\n -webkit-animation: spin-reverse 1s linear infinite;\n animation: spin-reverse 1s linear infinite;\n}\n";
const header =
"#MangaOnlineViewer #gotoPage {\n min-width: 35px;\n}\n\n#MangaOnlineViewer #Header {\n display: flex;\n justify-content: space-between;\n align-items: center;\n flex-flow: row nowrap;\n transition: transform 0.3s ease-in;\n position: sticky;\n top: 0;\n left: 0;\n right: 0;\n background-color: inherit;\n z-index: 900;\n}\n\n#MangaOnlineViewer #Header.click.headroom-end,\n#MangaOnlineViewer #Header.click {\n position: fixed;\n top: -100%;\n padding-left: 40px;\n}\n\n#MangaOnlineViewer #Header.hover,\n#MangaOnlineViewer #Header.simple {\n position: static;\n}\n\n/*#MangaOnlineViewer #Header.hover.headroom-hide,*/\n#MangaOnlineViewer #Header.scroll.headroom-hide {\n transform: translateY(-100%);\n}\n\n#MangaOnlineViewer #Header.headroom-end,\n#MangaOnlineViewer #Header.scroll.headroom-show,\n#MangaOnlineViewer #Header.hover.visible,\n#MangaOnlineViewer #Header.click.visible,\n#MangaOnlineViewer #Header.fixed {\n transform: translateY(0%);\n position: sticky;\n top: 0;\n}\n\n#MangaOnlineViewer #menu {\n position: fixed;\n z-index: 1;\n color: var(--theme-body-text-color);\n height: 40px;\n width: 40px;\n}\n\n#MangaOnlineViewer #menu .icon-tabler {\n position: relative;\n top: 4px;\n left: 4px;\n height: 32px;\n width: 32px;\n stroke-width: 1.25;\n}\n\n#MangaOnlineViewer #menu:not(.click, .hover),\n#MangaOnlineViewer #menu.hide {\n display: none;\n}\n\n#MangaOnlineViewer #menu.click {\n z-index: 901;\n}\n\n#MangaOnlineViewer #MangaTitle {\n padding: 2px;\n margin: 0;\n font-size: 1.2rem;\n font-weight: 400;\n}\n\n#MangaOnlineViewer #GlobalFunctions {\n display: flex;\n gap: 3px;\n padding: 3px 3px 3px 0;\n flex-wrap: wrap;\n width: 300px;\n z-index: 100;\n}\n\n#MangaOnlineViewer #GlobalFunctions span,\n#MangaOnlineViewer .ChapterControl span {\n display: flex;\n flex-wrap: nowrap;\n justify-content: space-evenly;\n}\n\n#MangaOnlineViewer .ChapterControl span {\n flex-grow: 1;\n}\n\n#MangaOnlineViewer .ChapterControl span > * {\n flex-basis: 50%;\n}\n\n#MangaOnlineViewer #ScrollControl .icon-tabler,\n#MangaOnlineViewer #GlobalFunctions .icon-tabler {\n width: 25px;\n height: 25px;\n}\n\n#MangaOnlineViewer #GlobalFunctions #ZoomSlider {\n display: flex;\n align-items: center;\n}\n\n#MangaOnlineViewer #GlobalFunctions #Zoom {\n margin: 2px 5px;\n width: 160px;\n}\n\n#MangaOnlineViewer #GlobalFunctions #ZoomVal {\n width: 40px;\n display: inline-block;\n color: var(--theme-primary-text-color);\n line-height: 20px;\n text-align: center;\n border-radius: 3px;\n background: var(--theme-primary-color);\n padding: 2px 5px;\n}\n\n#MangaOnlineViewer #ChapterNavigation {\n display: flex;\n flex-flow: column nowrap;\n justify-content: center;\n align-items: end;\n padding: 5px;\n max-width: 350px;\n}\n\n#MangaOnlineViewer #Counters {\n padding-right: 5px;\n}\n\n#MangaOnlineViewer #ChapterControl {\n display: flex;\n}\n\n#MangaOnlineViewer #ChapterControl .NavigationControlButton {\n display: inline-flex;\n margin: 2px;\n justify-content: center;\n align-items: center;\n padding: 3px;\n gap: 0.5em;\n}\n\n#MangaOnlineViewer #ChapterControl .NavigationControlButton .icon-tabler {\n flex-shrink: 0;\n align-self: center;\n width: 1rem;\n height: 1rem;\n}\n\n#MangaOnlineViewer #ChapterControl .NavigationControlButton[href='#'],\n#MangaOnlineViewer #ChapterControl .NavigationControlButton[href=''],\n#MangaOnlineViewer #ChapterControl .NavigationControlButton[href='undefined'] {\n visibility: hidden;\n}\n\n#MangaOnlineViewer #ChapterControl #download.loading {\n cursor: not-allowed;\n pointer-events: none;\n opacity: 0.6;\n}\n\n#MangaOnlineViewer #ChapterControl .NavigationControlButton.disabled {\n pointer-events: none;\n filter: grayscale(0.9);\n}\n\n#MangaOnlineViewer .ViewerTitle {\n text-align: center;\n min-height: 60px;\n display: flex;\n justify-content: center;\n align-items: center;\n flex-direction: column;\n padding: 5px;\n flex-basis: 60%;\n}\n\n#MangaOnlineViewer #Header .ViewerTitle #series[href='#'],\n#MangaOnlineViewer #Header .ViewerTitle #series[href=''],\n#MangaOnlineViewer #Header .ViewerTitle #series[href='undefined'] {\n /*visibility: hidden;*/\n}\n";
const keybindings$1 =
"#MangaOnlineViewer #KeybindingsPanel {\n padding: 10px;\n position: fixed;\n top: 0;\n right: 0;\n bottom: 0;\n transition: transform 0.3s ease-in-out;\n transform: translateX(100%);\n line-height: 1.5em;\n z-index: 1000;\n overflow-y: auto;\n width: 360px;\n max-width: 100vw;\n}\n\n#MangaOnlineViewer #KeybindingsPanel.visible {\n transform: translateX(0);\n display: block;\n}\n\n#MangaOnlineViewer #KeybindingsPanel #KeybindingsList {\n display: grid;\n grid-template-columns: 1fr 2fr;\n gap: 5px;\n}\n\n#MangaOnlineViewer #KeybindingsPanel .ControlButton {\n margin-left: 3px;\n justify-content: center;\n align-items: center;\n padding: 5px 10px;\n gap: 0.5em;\n}\n\n#MangaOnlineViewer #KeybindingsPanel label {\n display: ruby;\n}\n#MangaOnlineViewer #KeybindingsPanel input {\n display: inline-block;\n width: 100%;\n}\n\n#MangaOnlineViewer #KeybindingsPanel #HotKeysRules {\n grid-column: span 2;\n}\n";
const page =
"#MangaOnlineViewer .MangaPage {\n width: 100%;\n display: inline-block;\n text-align: center;\n line-height: 0;\n min-height: 22px;\n min-width: 100%;\n}\n\n#MangaOnlineViewer .PageContent {\n text-align: center;\n display: inline-block;\n overflow-x: auto;\n max-width: 100%;\n transition: all 0.3s ease-in-out;\n height: 100%;\n overflow-y: hidden;\n}\n\n#MangaOnlineViewer .MangaPage.hide .PageContent {\n height: 0;\n}\n\n#MangaOnlineViewer .PageContent .PageImg[src=''],\n#MangaOnlineViewer .PageContent .PageImg:not([src]) {\n width: 40vw;\n height: 80vh;\n display: inline-block;\n background-position: center;\n background-repeat: no-repeat;\n background-size: 20%;\n background-color: var(--theme-hightlight-color);\n}\n\n#MangaOnlineViewer .PageContent .PageImg.imgBroken {\n width: 40vw;\n height: 80vh;\n display: inline-block;\n background-position: center;\n background-repeat: no-repeat;\n background-size: 20%;\n background-color: var(--theme-hightlight-color);\n}\n\n#MangaOnlineViewer .PageFunctions {\n font-family: monospace;\n display: flex;\n justify-content: flex-end;\n align-items: center;\n margin: 0;\n padding: 0;\n gap: 3px;\n position: absolute;\n right: 0;\n}\n\n#MangaOnlineViewer .PageFunctions > .PageIndex {\n background-color: var(--theme-primary-color);\n color: var(--theme-primary-text-color);\n min-width: 20px;\n text-align: center;\n display: inline-block;\n padding: 3px 5px;\n line-height: 1rem;\n border-radius: 5px;\n}\n\n#MangaOnlineViewer .PageFunctions .ControlButton {\n padding: 3px;\n display: flex;\n justify-content: center;\n align-items: center;\n margin: 0;\n border-width: 0;\n min-height: auto;\n opacity: 0.5;\n}\n\n#MangaOnlineViewer .PageFunctions:hover .ControlButton {\n opacity: 1;\n}\n\n#MangaOnlineViewer .PageFunctions .ControlButton:hover {\n opacity: 0.9;\n}\n";
const fluid =
"#MangaOnlineViewer #Chapter.FluidLTR,\n#MangaOnlineViewer #Chapter.FluidRTL {\n display: flex;\n overflow-x: auto;\n min-width: auto;\n /*height: 100%;*/\n\n .ZoomWidth {\n display: none;\n }\n\n .PageImg {\n min-width: unset;\n }\n\n .MangaPage {\n width: initial;\n min-width: fit-content;\n position: relative;\n max-height: 100%;\n }\n\n .MangaPage.DoublePage {\n grid-column: span 2;\n }\n}\n\n#MangaOnlineViewer #Chapter.FluidLTR {\n flex-direction: row;\n\n .MangaPage .PageFunctions {\n right: auto;\n left: 0;\n direction: rtl;\n }\n}\n\n#MangaOnlineViewer #Chapter.FluidRTL {\n flex-direction: row-reverse;\n}\n";
const settings$1 =
"#MangaOnlineViewer #SettingsPanel {\n color: var(--theme-text-color);\n padding: 10px;\n position: fixed;\n top: 0;\n left: 0;\n bottom: 0;\n z-index: 1000;\n transition:\n transform 0.3s ease-in,\n background-color 0.3s linear;\n transform: translateX(-100%);\n display: flex;\n flex-flow: column;\n gap: 5px;\n overflow-y: auto;\n max-width: 100vw;\n width: 305px;\n}\n\n#MangaOnlineViewer #SettingsPanel.visible {\n transform: translateX(0);\n}\n\n#MangaOnlineViewer #SettingsPanel .ControlLabel {\n display: flex;\n flex-flow: row wrap;\n justify-content: space-between;\n align-items: center;\n}\n\n#MangaOnlineViewer #SettingsPanel .ControlLabelItem {\n display: flex;\n align-items: center;\n justify-content: space-between;\n}\n\n#MangaOnlineViewer #SettingsPanel .ControlLabelItem:not(.show) {\n display: none;\n}\n\n#MangaOnlineViewer #SettingsPanel input[type='range'] {\n width: 100%;\n}\n\n#MangaOnlineViewer #SettingsPanel .RangeValue {\n display: inline-block;\n color: var(--theme-primary-text-color);\n line-height: 20px;\n text-align: center;\n border-radius: 3px;\n background: var(--theme-primary-color);\n padding: 2px 5px;\n margin-left: 8px;\n}\n\n#MangaOnlineViewer #SettingsPanel datalist {\n display: flex;\n flex-direction: column;\n justify-content: space-between;\n align-items: center;\n writing-mode: vertical-lr;\n width: 100%;\n}\n\n#MangaOnlineViewer #SettingsPanel datalist option {\n padding: 0;\n}\n#MangaOnlineViewer #ThemeSection {\n border: 1px solid var(--theme-body-text-color);\n border-radius: 10px;\n padding: 10px;\n}\n\n#MangaOnlineViewer .ThemeRadio {\n border: 1px solid var(--theme-text-color);\n color: var(--theme-primary-text-color);\n background-color: var(--theme-primary-color);\n height: 20px;\n width: 20px;\n border-radius: 50%;\n padding: 1px;\n margin: 2px 5px;\n position: relative;\n}\n\n#MangaOnlineViewer .ThemeRadio svg {\n position: absolute;\n top: 15%;\n right: 15%;\n}\n\n#MangaOnlineViewer .ThemeRadio.selected .icon-tabler-check {\n display: inline;\n}\n\n#MangaOnlineViewer .ThemeRadio:not(.selected) .icon-tabler-check {\n display: none;\n}\n\n#MangaOnlineViewer #ThemeSelector {\n width: 110px;\n}\n";
const thumbnails =
"#MangaOnlineViewer .Thumbnail .ThumbnailImg[src=''],\n#MangaOnlineViewer .Thumbnail .ThumbnailImg:not([src]) {\n width: 100px;\n height: 150px;\n display: inline-block;\n background-position: center;\n background-repeat: no-repeat;\n background-size: 20%;\n}\n\n#MangaOnlineViewer #NavigationCounters {\n margin: 5px;\n width: 100%;\n line-height: 1rem;\n}\n\n#MangaOnlineViewer #Navigation {\n color: var(--theme-text-color);\n background-color: var(--theme-hightlight-color);\n bottom: -180px;\n height: 185px;\n overflow-x: hidden;\n overflow-y: hidden;\n padding-bottom: 20px;\n position: fixed;\n white-space: nowrap;\n width: 100%;\n text-align: center;\n transition:\n transform 0.3s ease-in,\n background-color 0.3s linear;\n border-bottom-left-radius: 0;\n border-bottom-right-radius: 0;\n line-height: 0;\n}\n\n#MangaOnlineViewer #Navigation #Thumbnails {\n overflow-x: auto;\n overflow-y: hidden;\n margin-right: 10px;\n}\n\n#MangaOnlineViewer #Navigation:hover {\n transform: translateY(-180px);\n}\n\n#MangaOnlineViewer #Navigation.disabled {\n display: none;\n}\n\n#MangaOnlineViewer #Navigation.visible {\n transform: translateY(-180px);\n}\n\n#MangaOnlineViewer #Navigation .Thumbnail {\n display: inline-block;\n height: 150px;\n margin: 0 5px;\n border: 1px solid var(--theme-primary-color);\n}\n\n#MangaOnlineViewer #Navigation .Thumbnail .ThumbnailIndex {\n color: var(--theme-text-color);\n background-color: var(--theme-hightlight-color);\n display: block;\n opacity: 0.8;\n position: relative;\n bottom: 25%;\n width: 100%;\n line-height: 1rem;\n}\n\n#MangaOnlineViewer #Navigation .Thumbnail .ThumbnailImg {\n cursor: pointer;\n display: inline-block;\n max-height: 150px;\n min-height: 150px;\n min-width: 80px;\n max-width: 160px;\n}\n";
const bookmarks$1 =
"#MangaOnlineViewer #BookmarksPanel {\n position: fixed;\n top: 10%;\n width: 50%;\n left: 25%;\n right: 25%;\n text-align: center;\n max-height: 70%;\n transition: transform 0.3s ease-in-out;\n transform: scaleY(0);\n z-index: 1000;\n}\n\n#MangaOnlineViewer #BookmarksPanel.visible {\n transform: scaleY(1);\n display: block;\n}\n\n#MangaOnlineViewer #BookmarksList {\n padding: 0 15px;\n overflow: auto;\n max-height: 60vh;\n}\n\n#MangaOnlineViewer #BookmarksList .BookmarkItem {\n display: flex;\n flex-flow: row;\n justify-content: space-between;\n align-items: center;\n padding: 2px;\n}\n\n#MangaOnlineViewer #BookmarksList .bookmarkData {\n flex-basis: 15%;\n}\n\n#MangaOnlineViewer #BookmarksList .bookmarkURl {\n text-overflow: ellipsis;\n overflow: hidden;\n white-space: nowrap;\n flex-basis: 55%;\n}\n";
const comments =
"#MangaOnlineViewer #CommentsPanel {\n position: static;\n width: 90%;\n height: 0;\n top: 5%;\n left: 5%;\n text-align: center;\n transition: transform 0.3s ease-in-out;\n transform: scaleY(0);\n z-index: 1000;\n overflow-y: initial;\n background: var(--theme-body-background);\n opacity: 0;\n}\n\n#MangaOnlineViewer #CommentsPanel.visible {\n position: fixed;\n height: 90%;\n transform: scaleY(1);\n display: flex;\n justify-content: center;\n align-items: center;\n flex-direction: column;\n opacity: 1;\n}\n\n#MangaOnlineViewer #CommentsArea {\n overflow-y: auto;\n overscroll-behavior: contain;\n height: 100%;\n width: 100%;\n}\n";
const cssStyles = css`
:root,
.dark,
.dark .default,
[data-theme="dark"] {
--theme-body-background: ${colors.dark["600"]};
--theme-body-text-color: ${colors.dark["50"]};
--theme-text-color: ${colors.dark["50"]};
--theme-primary-color: ${colors.dark["700"]};
--theme-primary-text-color: ${colors.dark["50"]};
--theme-background-color: ${colors.dark["600"]};
--theme-hightlight-color: ${colors.dark["500"]};
--theme-border-color: ${colors.dark["400"]};
}
.light,
.light .default,
[data-theme="light"] {
--theme-body-background: ${colors.gray["50"]};
--theme-body-text-color: ${colors.gray["900"]};
--theme-text-color: ${colors.gray["900"]};
--theme-primary-color: ${colors.gray["300"]};
--theme-primary-text-color: ${colors.gray["900"]};
--theme-background-color: ${colors.gray["50"]};
--theme-hightlight-color: ${colors.gray["500"]};
--theme-border-color: ${colors.gray["100"]};
}
#MangaOnlineViewer .PageContent .PageImg[src=""],
#MangaOnlineViewer .PageContent .PageImg:not([src]) {
background-image: url("${svgToUrl(IconPhoto)}");
}
#MangaOnlineViewer .PageContent .PageImg.imgBroken {
background-image: url("${svgToUrl(IconPhotoOff)}");
}
#MangaOnlineViewer .Thumbnail .ThumbnailImg[src=""],
#MangaOnlineViewer .Thumbnail .ThumbnailImg:not([src]) {
background-image: url("${svgToUrl(IconPhoto)}");
}
#MangaOnlineViewer .ThemeRadio.custom {
/*background-image: url("${svgToUrl(IconPalette)}");*/
}
${normalize$1}
${styles}
${header}
${icons}
${keybindings$1}
${page}
${fluid}
${settings$1}
${thumbnails}
${bookmarks$1}
${comments}
${media}
${animation}
`;
function createStyleElement(id, content) {
const style = document.createElement("style");
style.id = id;
style.appendChild(document.createTextNode(content));
return style;
}
function appendStyleSheet(id, content) {
if (!document.querySelector(`#${id}`)) {
const head = document.head ?? document.querySelector("head");
head.appendChild(createStyleElement(id, content));
}
}
function removeStyleSheet(id) {
document.querySelectorAll(`style[id="${id}"]`).forEach((elem) => {
elem.remove();
});
}
function replaceStyleSheet(id, content) {
removeStyleSheet(id);
appendStyleSheet(id, content);
}
function wrapStyle(id, css) {
return html`
`;
}
function generateThemeCSS(name, primary, text) {
return css`
.${name}, [data-theme="${name}"] {
--theme-primary-color: ${primary};
--theme-primary-text-color: ${text};
}
`;
}
function getNormalThemeCSS(theme) {
return generateThemeCSS(
theme.name,
theme[getUserSettings().themeShade],
getUserSettings().themeShade < 500 ? theme["900"] : theme["50"],
);
}
function getCustomThemeCSS(hex) {
return generateThemeCSS("custom", hex, getTextColor(hex));
}
function addTheme(theme) {
return wrapStyle(theme.name, getNormalThemeCSS(theme));
}
function addCustomTheme(hex) {
replaceStyleSheet("custom", getCustomThemeCSS(hex));
}
const themes = () => Object.values(colors);
function refreshThemes() {
themes().forEach((theme) => {
replaceStyleSheet(theme.name, getNormalThemeCSS(theme));
});
replaceStyleSheet(
"custom",
getCustomThemeCSS(getUserSettings().customTheme),
);
}
const themesCSS =
themes().map(addTheme).join("") +
wrapStyle("custom", getCustomThemeCSS(getUserSettings().customTheme));
const sweetalert =
'.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 rgba(0,0,0,.075),0 1px 2px rgba(0,0,0,.075),1px 2px 4px rgba(0,0,0,.075),1px 3px 8px rgba(0,0,0,.075),2px 4px 16px rgba(0,0,0,.075);pointer-events:all}.swal2-popup.swal2-toast>*{grid-column:2}.swal2-popup.swal2-toast .swal2-title{margin:.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:.5em;font-size:1em}.swal2-popup.swal2-toast .swal2-validation-message{font-size:1em}.swal2-popup.swal2-toast .swal2-footer{margin:.5em 0 0;padding:.5em 0 0;font-size:.8em}.swal2-popup.swal2-toast .swal2-close{grid-column:3/3;grid-row:1/99;align-self:center;width:.8em;height:.8em;margin:0;font-size:2em}.swal2-popup.swal2-toast .swal2-html-container{margin:.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:.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 .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:.875em;width:1.375em}.swal2-popup.swal2-toast .swal2-icon.swal2-error [class^=swal2-x-mark-line][class$=left]{left:.3125em}.swal2-popup.swal2-toast .swal2-icon.swal2-error [class^=swal2-x-mark-line][class$=right]{right:.3125em}.swal2-popup.swal2-toast .swal2-actions{justify-content:flex-start;height:auto;margin:0;margin-top:.5em;padding:0 .5em}.swal2-popup.swal2-toast .swal2-styled{margin:.25em .5em;padding:.4em .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:-.8em;left:-.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:-.25em;left:.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:.4375em;width:.4375em;height:2.6875em}.swal2-popup.swal2-toast .swal2-success [class^=swal2-success-line]{height:.3125em}.swal2-popup.swal2-toast .swal2-success [class^=swal2-success-line][class$=tip]{top:1.125em;left:.1875em;width:.75em}.swal2-popup.swal2-toast .swal2-success [class^=swal2-success-line][class$=long]{top:.9375em;right:.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 .75s;animation:swal2-toast-animate-success-line-tip .75s}.swal2-popup.swal2-toast .swal2-success.swal2-icon-show .swal2-success-line-long{-webkit-animation:swal2-toast-animate-success-line-long .75s;animation:swal2-toast-animate-success-line-long .75s}.swal2-popup.swal2-toast.swal2-show{-webkit-animation:swal2-toast-show .5s;animation:swal2-toast-show .5s}.swal2-popup.swal2-toast.swal2-hide{-webkit-animation:swal2-toast-hide .1s forwards;animation:swal2-toast-hide .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:.625em;overflow-x:hidden;transition:background-color .1s;-webkit-overflow-scrolling:touch}.swal2-container.swal2-backdrop-show,.swal2-container.swal2-noanimation{background:rgba(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:.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:.4}.swal2-actions:not(.swal2-loading) .swal2-styled:hover{background-image:linear-gradient(rgba(0,0,0,.1),rgba(0,0,0,.1))}.swal2-actions:not(.swal2-loading) .swal2-styled:active{background-image:linear-gradient(rgba(0,0,0,.2),rgba(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:.25em;border-style:solid;border-radius:100%;border-color:#2778c4 transparent #2778c4 transparent}.swal2-styled{margin:.3125em;padding:.625em 1.1em;transition:box-shadow .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:.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,.5)}.swal2-styled.swal2-deny{border:0;border-radius:.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,.5)}.swal2-styled.swal2-cancel{border:0;border-radius:.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,.5)}.swal2-styled.swal2-default-outline:focus{box-shadow:0 0 0 3px rgba(100,150,200,.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:.25em;background:rgba(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 .1s,box-shadow .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,.5)}.swal2-close::-moz-focus-inner{border:0}.swal2-html-container{z-index:1;justify-content:center;margin:1em 1.6em .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 .1s,box-shadow .1s;border:1px solid #d9d9d9;border-radius:.1875em;background:inherit;box-shadow:inset 0 1px 1px rgba(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,.06),0 0 0 3px rgba(100,150,200,.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 .75em}.swal2-file{width:75%;margin-right:auto;margin-left:auto;background:inherit;font-size:1.125em}.swal2-textarea{height:6.75em;padding:.75em}.swal2-select{min-width:50%;max-width:100%;padding:.375em .625em;background:inherit;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 .6em;font-size:1.125em}.swal2-checkbox input,.swal2-radio input{flex-shrink:0;margin: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:.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 .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 .6em;border:.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:.3125em;border-radius:.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 .5s;animation:swal2-animate-error-icon .5s}.swal2-icon.swal2-error.swal2-icon-show .swal2-x-mark{-webkit-animation:swal2-animate-error-x-mark .5s;animation:swal2-animate-error-x-mark .5s}.swal2-icon.swal2-warning{border-color:#facea8;color:#f8bb86}.swal2-icon.swal2-warning.swal2-icon-show{-webkit-animation:swal2-animate-error-icon .5s;animation:swal2-animate-error-icon .5s}.swal2-icon.swal2-warning.swal2-icon-show .swal2-icon-content{-webkit-animation:swal2-animate-i-mark .5s;animation:swal2-animate-i-mark .5s}.swal2-icon.swal2-info{border-color:#9de0f6;color:#3fc3ee}.swal2-icon.swal2-info.swal2-icon-show{-webkit-animation:swal2-animate-error-icon .5s;animation:swal2-animate-error-icon .5s}.swal2-icon.swal2-info.swal2-icon-show .swal2-icon-content{-webkit-animation:swal2-animate-i-mark .8s;animation:swal2-animate-i-mark .8s}.swal2-icon.swal2-question{border-color:#c9dae1;color:#87adbd}.swal2-icon.swal2-question.swal2-icon-show{-webkit-animation:swal2-animate-error-icon .5s;animation:swal2-animate-error-icon .5s}.swal2-icon.swal2-question.swal2-icon-show .swal2-icon-content{-webkit-animation:swal2-animate-question-mark .8s;animation:swal2-animate-question-mark .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:-.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:-.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:-.25em;left:-.25em;box-sizing:content-box;width:100%;height:100%;border:.25em solid rgba(165,220,134,.3);border-radius:50%}.swal2-icon.swal2-success .swal2-success-fix{position:absolute;z-index:1;top:.5em;left:1.625em;width:.4375em;height:5.625em;transform:rotate(-45deg)}.swal2-icon.swal2-success [class^=swal2-success-line]{display:block;position:absolute;z-index:2;height:.3125em;border-radius:.125em;background-color:#a5dc86}.swal2-icon.swal2-success [class^=swal2-success-line][class$=tip]{top:2.875em;left:.8125em;width:1.5625em;transform:rotate(45deg)}.swal2-icon.swal2-success [class^=swal2-success-line][class$=long]{top:2.375em;right:.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 .75s;animation:swal2-animate-success-line-tip .75s}.swal2-icon.swal2-success.swal2-icon-show .swal2-success-line-long{-webkit-animation:swal2-animate-success-line-long .75s;animation:swal2-animate-success-line-long .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:inherit;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:.4em;margin:0 -1px;background:#2778c4}[class^=swal2]{-webkit-tap-highlight-color:transparent}.swal2-show{-webkit-animation:swal2-show .3s;animation:swal2-show .3s}.swal2-hide{-webkit-animation:swal2-hide .15s forwards;animation:swal2-hide .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}@-webkit-keyframes swal2-toast-show{0%{transform:translateY(-.625em) rotateZ(2deg)}33%{transform:translateY(0) rotateZ(-2deg)}66%{transform:translateY(.3125em) rotateZ(2deg)}100%{transform:translateY(0) rotateZ(0)}}@keyframes swal2-toast-show{0%{transform:translateY(-.625em) rotateZ(2deg)}33%{transform:translateY(0) rotateZ(-2deg)}66%{transform:translateY(.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:.5625em;left:.0625em;width:0}54%{top:.125em;left:.125em;width:0}70%{top:.625em;left:-.25em;width:1.625em}84%{top:1.0625em;left:.75em;width:.5em}100%{top:1.125em;left:.1875em;width:.75em}}@keyframes swal2-toast-animate-success-line-tip{0%{top:.5625em;left:.0625em;width:0}54%{top:.125em;left:.125em;width:0}70%{top:.625em;left:-.25em;width:1.625em}84%{top:1.0625em;left:.75em;width:.5em}100%{top:1.125em;left:.1875em;width:.75em}}@-webkit-keyframes swal2-toast-animate-success-line-long{0%{top:1.625em;right:1.375em;width:0}65%{top:1.25em;right:.9375em;width:0}84%{top:.9375em;right:0;width:1.125em}100%{top:.9375em;right:.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:.9375em;width:0}84%{top:.9375em;right:0;width:1.125em}100%{top:.9375em;right:.1875em;width:1.375em}}@-webkit-keyframes swal2-show{0%{transform:scale(.7)}45%{transform:scale(1.05)}80%{transform:scale(.95)}100%{transform:scale(1)}}@keyframes swal2-show{0%{transform:scale(.7)}45%{transform:scale(1.05)}80%{transform:scale(.95)}100%{transform:scale(1)}}@-webkit-keyframes swal2-hide{0%{transform:scale(1);opacity:1}100%{transform:scale(.5);opacity:0}}@keyframes swal2-hide{0%{transform:scale(1);opacity:1}100%{transform:scale(.5);opacity:0}}@-webkit-keyframes swal2-animate-success-line-tip{0%{top:1.1875em;left:.0625em;width:0}54%{top:1.0625em;left:.125em;width:0}70%{top:2.1875em;left:-.375em;width:3.125em}84%{top:3em;left:1.3125em;width:1.0625em}100%{top:2.8125em;left:.8125em;width:1.5625em}}@keyframes swal2-animate-success-line-tip{0%{top:1.1875em;left:.0625em;width:0}54%{top:1.0625em;left:.125em;width:0}70%{top:2.1875em;left:-.375em;width:3.125em}84%{top:3em;left:1.3125em;width:1.0625em}100%{top:2.8125em;left:.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:.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:.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(.4);opacity:0}50%{margin-top:1.625em;transform:scale(.4);opacity:0}80%{margin-top:-.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(.4);opacity:0}50%{margin-top:1.625em;transform:scale(.4);opacity:0}80%{margin-top:-.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:.4}50%{transform:rotateZ(15deg);opacity:.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:.4}50%{transform:rotateZ(15deg);opacity:.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,.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 normalize =
'/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */\n\n/* Document\n ========================================================================== */\n\n/**\n * 1. Correct the line height in all browsers.\n * 2. Prevent adjustments of font size after orientation changes in iOS.\n */\n\nhtml {\n line-height: 1.15; /* 1 */\n -webkit-text-size-adjust: 100%; /* 2 */\n}\n\n/* Sections\n ========================================================================== */\n\n/**\n * Remove the margin in all browsers.\n */\n\nbody {\n margin: 0;\n}\n\n/**\n * Render the `main` element consistently in IE.\n */\n\nmain {\n display: block;\n}\n\n/**\n * Correct the font size and margin on `h1` elements within `section` and\n * `article` contexts in Chrome, Firefox, and Safari.\n */\n\nh1 {\n font-size: 2em;\n margin: 0.67em 0;\n}\n\n/* Grouping content\n ========================================================================== */\n\n/**\n * 1. Add the correct box sizing in Firefox.\n * 2. Show the overflow in Edge and IE.\n */\n\nhr {\n box-sizing: content-box; /* 1 */\n height: 0; /* 1 */\n overflow: visible; /* 2 */\n}\n\n/**\n * 1. Correct the inheritance and scaling of font size in all browsers.\n * 2. Correct the odd `em` font sizing in all browsers.\n */\n\npre {\n font-family: monospace, monospace; /* 1 */\n font-size: 1em; /* 2 */\n}\n\n/* Text-level semantics\n ========================================================================== */\n\n/**\n * Remove the gray background on active links in IE 10.\n */\n\na {\n background-color: transparent;\n}\n\n/**\n * 1. Remove the bottom border in Chrome 57-\n * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.\n */\n\nabbr[title] {\n border-bottom: none; /* 1 */\n text-decoration: underline; /* 2 */\n text-decoration: underline dotted; /* 2 */\n}\n\n/**\n * Add the correct font weight in Chrome, Edge, and Safari.\n */\n\nb,\nstrong {\n font-weight: bolder;\n}\n\n/**\n * 1. Correct the inheritance and scaling of font size in all browsers.\n * 2. Correct the odd `em` font sizing in all browsers.\n */\n\ncode,\nkbd,\nsamp {\n font-family: monospace, monospace; /* 1 */\n font-size: 1em; /* 2 */\n}\n\n/**\n * Add the correct font size in all browsers.\n */\n\nsmall {\n font-size: 80%;\n}\n\n/**\n * Prevent `sub` and `sup` elements from affecting the line height in\n * all browsers.\n */\n\nsub,\nsup {\n font-size: 75%;\n line-height: 0;\n position: relative;\n vertical-align: baseline;\n}\n\nsub {\n bottom: -0.25em;\n}\n\nsup {\n top: -0.5em;\n}\n\n/* Embedded content\n ========================================================================== */\n\n/**\n * Remove the border on images inside links in IE 10.\n */\n\nimg {\n border-style: none;\n}\n\n/* Forms\n ========================================================================== */\n\n/**\n * 1. Change the font styles in all browsers.\n * 2. Remove the margin in Firefox and Safari.\n */\n\nbutton,\ninput,\noptgroup,\nselect,\ntextarea {\n font-family: inherit; /* 1 */\n font-size: 100%; /* 1 */\n line-height: 1.15; /* 1 */\n margin: 0; /* 2 */\n}\n\n/**\n * Show the overflow in IE.\n * 1. Show the overflow in Edge.\n */\n\nbutton,\ninput { /* 1 */\n overflow: visible;\n}\n\n/**\n * Remove the inheritance of text transform in Edge, Firefox, and IE.\n * 1. Remove the inheritance of text transform in Firefox.\n */\n\nbutton,\nselect { /* 1 */\n text-transform: none;\n}\n\n/**\n * Correct the inability to style clickable types in iOS and Safari.\n */\n\nbutton,\n[type="button"],\n[type="reset"],\n[type="submit"] {\n -webkit-appearance: button;\n}\n\n/**\n * Remove the inner border and padding in Firefox.\n */\n\nbutton::-moz-focus-inner,\n[type="button"]::-moz-focus-inner,\n[type="reset"]::-moz-focus-inner,\n[type="submit"]::-moz-focus-inner {\n border-style: none;\n padding: 0;\n}\n\n/**\n * Restore the focus styles unset by the previous rule.\n */\n\nbutton:-moz-focusring,\n[type="button"]:-moz-focusring,\n[type="reset"]:-moz-focusring,\n[type="submit"]:-moz-focusring {\n outline: 1px dotted ButtonText;\n}\n\n/**\n * Correct the padding in Firefox.\n */\n\nfieldset {\n padding: 0.35em 0.75em 0.625em;\n}\n\n/**\n * 1. Correct the text wrapping in Edge and IE.\n * 2. Correct the color inheritance from `fieldset` elements in IE.\n * 3. Remove the padding so developers are not caught out when they zero out\n * `fieldset` elements in all browsers.\n */\n\nlegend {\n box-sizing: border-box; /* 1 */\n color: inherit; /* 2 */\n display: table; /* 1 */\n max-width: 100%; /* 1 */\n padding: 0; /* 3 */\n white-space: normal; /* 1 */\n}\n\n/**\n * Add the correct vertical alignment in Chrome, Firefox, and Opera.\n */\n\nprogress {\n vertical-align: baseline;\n}\n\n/**\n * Remove the default vertical scrollbar in IE 10+.\n */\n\ntextarea {\n overflow: auto;\n}\n\n/**\n * 1. Add the correct box sizing in IE 10.\n * 2. Remove the padding in IE 10.\n */\n\n[type="checkbox"],\n[type="radio"] {\n box-sizing: border-box; /* 1 */\n padding: 0; /* 2 */\n}\n\n/**\n * Correct the cursor style of increment and decrement buttons in Chrome.\n */\n\n[type="number"]::-webkit-inner-spin-button,\n[type="number"]::-webkit-outer-spin-button {\n height: auto;\n}\n\n/**\n * 1. Correct the odd appearance in Chrome and Safari.\n * 2. Correct the outline style in Safari.\n */\n\n[type="search"] {\n -webkit-appearance: textfield; /* 1 */\n outline-offset: -2px; /* 2 */\n}\n\n/**\n * Remove the inner padding in Chrome and Safari on macOS.\n */\n\n[type="search"]::-webkit-search-decoration {\n -webkit-appearance: none;\n}\n\n/**\n * 1. Correct the inability to style clickable types in iOS and Safari.\n * 2. Change font properties to `inherit` in Safari.\n */\n\n::-webkit-file-upload-button {\n -webkit-appearance: button; /* 1 */\n font: inherit; /* 2 */\n}\n\n/* Interactive\n ========================================================================== */\n\n/*\n * Add the correct display in Edge, IE 10+, and Firefox.\n */\n\ndetails {\n display: block;\n}\n\n/*\n * Add the correct display in all browsers.\n */\n\nsummary {\n display: list-item;\n}\n\n/* Misc\n ========================================================================== */\n\n/**\n * Add the correct display in IE 10+.\n */\n\ntemplate {\n display: none;\n}\n\n/**\n * Add the correct display in IE 10.\n */\n\n[hidden] {\n display: none;\n}\n';
const nprogress =
"/* Make clicks pass-through */\n#nprogress {\n pointer-events: none;\n}\n\n#nprogress .bar {\n background: #29d;\n\n position: fixed;\n z-index: 1031;\n top: 0;\n left: 0;\n\n width: 100%;\n height: 2px;\n}\n\n/* Fancy blur effect */\n#nprogress .peg {\n display: block;\n position: absolute;\n right: 0px;\n width: 100px;\n height: 100%;\n box-shadow: 0 0 10px #29d, 0 0 5px #29d;\n opacity: 1.0;\n\n -webkit-transform: rotate(3deg) translate(0px, -4px);\n -ms-transform: rotate(3deg) translate(0px, -4px);\n transform: rotate(3deg) translate(0px, -4px);\n}\n\n/* Remove these to get rid of the spinner */\n#nprogress .spinner {\n display: block;\n position: fixed;\n z-index: 1031;\n top: 15px;\n right: 15px;\n}\n\n#nprogress .spinner-icon {\n width: 18px;\n height: 18px;\n box-sizing: border-box;\n\n border: solid 2px transparent;\n border-top-color: #29d;\n border-left-color: #29d;\n border-radius: 50%;\n\n -webkit-animation: nprogress-spinner 400ms linear infinite;\n animation: nprogress-spinner 400ms linear infinite;\n}\n\n.nprogress-custom-parent {\n overflow: hidden;\n position: relative;\n}\n\n.nprogress-custom-parent #nprogress .spinner,\n.nprogress-custom-parent #nprogress .bar {\n position: absolute;\n}\n\n@-webkit-keyframes nprogress-spinner {\n 0% { -webkit-transform: rotate(0deg); }\n 100% { -webkit-transform: rotate(360deg); }\n}\n@keyframes nprogress-spinner {\n 0% { transform: rotate(0deg); }\n 100% { transform: rotate(360deg); }\n}\n\n";
const keyscss =
'/**\r\n * KEYS.css\r\n *\r\n * A simple stylesheet for rendering beautiful keyboard-style elements.\r\n *\r\n * Author: Michael Hüneburg\r\n * Website: http://michaelhue.com/keyscss\r\n * License: MIT License (see LICENSE.txt)\r\n */\r\n\r\nkbd,\r\n.key {\r\n display: inline;\r\n display: inline-block;\r\n white-space: nowrap;\r\n min-width: 1em;\r\n padding: .3em .4em .2em .3em;\r\n font-style: normal;\r\n font-family: "Lucida Grande", Lucida, Arial, sans-serif;\r\n text-align: center;\r\n text-decoration: none;\r\n border-radius: .3em;\r\n border: none;\r\n background-color: #505050;\r\n background-color: gradient(linear, left top, left bottom, from(#3c3c3c), to(#505050));\r\n color: #fafafa;\r\n text-shadow: -1px -1px 0 #464646;\r\n -webkit-box-shadow: inset 0 0 1px #969696, inset 0 -0.05em 0.4em #505050, 0 0.1em 0 #1e1e1e, 0 0.1em 0.1em rgba(0, 0, 0, 0.3);\r\n box-shadow: inset 0 0 1px #969696, inset 0 -0.05em 0.4em #505050, 0 0.1em 0 #1e1e1e, 0 0.1em 0.1em rgba(0, 0, 0, 0.3);\r\n font-size: .85em;\r\n line-height: 1;\r\n cursor: default;\r\n -webkit-user-select: none;\r\n -moz-user-select: none;\r\n -ms-user-select: none;\r\n user-select: none;\r\n}\r\nkbd[title],\r\n.key[title] {\r\n cursor: help;\r\n}\r\nkbd.dark,\r\n.dark-keys kbd,\r\n.key.dark,\r\n.dark-keys .key {\r\n display: inline;\r\n display: inline-block;\r\n white-space: nowrap;\r\n min-width: 1em;\r\n padding: .3em .4em .2em .3em;\r\n font-style: normal;\r\n font-family: "Lucida Grande", Lucida, Arial, sans-serif;\r\n text-align: center;\r\n text-decoration: none;\r\n border-radius: .3em;\r\n border: none;\r\n background-color: #505050;\r\n background-color: gradient(linear, left top, left bottom, from(#3c3c3c), to(#505050));\r\n color: #fafafa;\r\n text-shadow: -1px -1px 0 #464646;\r\n -webkit-box-shadow: inset 0 0 1px #969696, inset 0 -0.05em 0.4em #505050, 0 0.1em 0 #1e1e1e, 0 0.1em 0.1em rgba(0, 0, 0, 0.3);\r\n box-shadow: inset 0 0 1px #969696, inset 0 -0.05em 0.4em #505050, 0 0.1em 0 #1e1e1e, 0 0.1em 0.1em rgba(0, 0, 0, 0.3);\r\n}\r\nkbd.light,\r\n.light-keys kbd,\r\n.key.light,\r\n.light-keys .key {\r\n display: inline;\r\n display: inline-block;\r\n white-space: nowrap;\r\n min-width: 1em;\r\n padding: .3em .4em .2em .3em;\r\n font-style: normal;\r\n font-family: "Lucida Grande", Lucida, Arial, sans-serif;\r\n text-align: center;\r\n text-decoration: none;\r\n border-radius: .3em;\r\n border: none;\r\n background-color: #fafafa;\r\n background-color: gradient(linear, left top, left bottom, from(#d2d2d2), to(#ffffff));\r\n color: #323232;\r\n text-shadow: 0 0 2px #ffffff;\r\n -webkit-box-shadow: inset 0 0 1px #ffffff, inset 0 0 0.4em #c8c8c8, 0 0.1em 0 #828282, 0 0.11em 0 rgba(0, 0, 0, 0.4), 0 0.1em 0.11em rgba(0, 0, 0, 0.9);\r\n box-shadow: inset 0 0 1px #ffffff, inset 0 0 0.4em #c8c8c8, 0 0.1em 0 #828282, 0 0.11em 0 rgba(0, 0, 0, 0.4), 0 0.1em 0.11em rgba(0, 0, 0, 0.9);\r\n}\r\nkbd.so,\r\n.so-keys kbd,\r\n.key.so,\r\n.so-keys .key {\r\n display: inline;\r\n display: inline-block;\r\n white-space: nowrap;\r\n min-width: 1em;\r\n padding: .3em .4em .2em .3em;\r\n font-style: normal;\r\n font-family: "Lucida Grande", Lucida, Arial, sans-serif;\r\n text-align: center;\r\n text-decoration: none;\r\n border-radius: .3em;\r\n border: none;\r\n margin: 0 .1em;\r\n padding: .1em .6em;\r\n font-family: Arial, "Helvetica Neue", Helvetica, sans-serif;\r\n line-height: 1.4;\r\n color: #242729;\r\n text-shadow: 0 1px 0 #FFF;\r\n background-color: #e1e3e5;\r\n border: 1px solid #adb3b9;\r\n border-radius: 0.27272727em;\r\n -webkit-box-shadow: 0 1px 0 rgba(12, 13, 14, 0.2), 0 0 0 2px #FFF inset;\r\n box-shadow: 0 1px 0 rgba(12, 13, 14, 0.2), 0 0 0 2px #FFF inset;\r\n}\r\nkbd.github,\r\n.github-keys kbd,\r\n.key.github,\r\n.github-keys .key {\r\n display: inline;\r\n display: inline-block;\r\n white-space: nowrap;\r\n min-width: 1em;\r\n padding: .3em .4em .2em .3em;\r\n font-style: normal;\r\n font-family: "Lucida Grande", Lucida, Arial, sans-serif;\r\n text-align: center;\r\n text-decoration: none;\r\n border-radius: .3em;\r\n border: none;\r\n padding: 0.27272727em 0.45454545em;\r\n font-size: 68.75%;\r\n line-height: 0.90909091;\r\n color: #444d56;\r\n vertical-align: middle;\r\n background-color: #fafbfc;\r\n border: solid 1px #c6cbd1;\r\n border-bottom-color: #959da5;\r\n border-radius: 0.27272727em;\r\n -webkit-box-shadow: inset 0 -1px 0 #959da5;\r\n box-shadow: inset 0 -1px 0 #959da5;\r\n font-family: "SFMono-Regular", Consolas, "Liberation Mono", Menlo, Courier, monospace;\r\n -webkit-box-sizing: border-box;\r\n box-sizing: border-box;\r\n text-shadow: none;\r\n}\r\n\r\n/*# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImtleXMuY3NzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztFQUVFLGdCQUFnQjtFQUNoQixzQkFBc0I7RUFDdEIsb0JBQW9CO0VBQ3BCLGVBQWU7RUFDZiw2QkFBNkI7RUFDN0IsbUJBQW1CO0VBQ25CLHdEQUF3RDtFQUN4RCxtQkFBbUI7RUFDbkIsc0JBQXNCO0VBQ3RCLG9CQUFvQjtFQUNwQixhQUFhO0VBQ2IsMEJBQTBCO0VBQzFCLHNGQUFzRjtFQUN0RixlQUFlO0VBQ2YsaUNBQWlDO0VBQ2pDLDhIQUFzSDtVQUF0SCxzSEFBc0g7RUFDdEgsaUJBQWlCO0VBQ2pCLGVBQWU7RUFDZixnQkFBZ0I7RUFDaEIsMEJBQWtCO0tBQWxCLHVCQUFrQjtNQUFsQixzQkFBa0I7VUFBbEIsa0JBQWtCO0NBQ25CO0FBQ0Q7O0VBRUUsYUFBYTtDQUNkO0FBQ0Q7Ozs7RUFJRSxnQkFBZ0I7RUFDaEIsc0JBQXNCO0VBQ3RCLG9CQUFvQjtFQUNwQixlQUFlO0VBQ2YsNkJBQTZCO0VBQzdCLG1CQUFtQjtFQUNuQix3REFBd0Q7RUFDeEQsbUJBQW1CO0VBQ25CLHNCQUFzQjtFQUN0QixvQkFBb0I7RUFDcEIsYUFBYTtFQUNiLDBCQUEwQjtFQUMxQixzRkFBc0Y7RUFDdEYsZUFBZTtFQUNmLGlDQUFpQztFQUNqQyw4SEFBc0g7VUFBdEgsc0hBQXNIO0NBQ3ZIO0FBQ0Q7Ozs7RUFJRSxnQkFBZ0I7RUFDaEIsc0JBQXNCO0VBQ3RCLG9CQUFvQjtFQUNwQixlQUFlO0VBQ2YsNkJBQTZCO0VBQzdCLG1CQUFtQjtFQUNuQix3REFBd0Q7RUFDeEQsbUJBQW1CO0VBQ25CLHNCQUFzQjtFQUN0QixvQkFBb0I7RUFDcEIsYUFBYTtFQUNiLDBCQUEwQjtFQUMxQixzRkFBc0Y7RUFDdEYsZUFBZTtFQUNmLDZCQUE2QjtFQUM3Qix3SkFBZ0o7VUFBaEosZ0pBQWdKO0NBQ2pKO0FBQ0Q7Ozs7RUFJRSxnQkFBZ0I7RUFDaEIsc0JBQXNCO0VBQ3RCLG9CQUFvQjtFQUNwQixlQUFlO0VBQ2YsNkJBQTZCO0VBQzdCLG1CQUFtQjtFQUNuQix3REFBd0Q7RUFDeEQsbUJBQW1CO0VBQ25CLHNCQUFzQjtFQUN0QixvQkFBb0I7RUFDcEIsYUFBYTtFQUNiLGVBQWU7RUFDZixtQkFBbUI7RUFDbkIsNERBQTREO0VBQzVELGlCQUFpQjtFQUNqQixlQUFlO0VBQ2YsMEJBQTBCO0VBQzFCLDBCQUEwQjtFQUMxQiwwQkFBMEI7RUFDMUIsNEJBQTRCO0VBQzVCLHdFQUFnRTtVQUFoRSxnRUFBZ0U7Q0FDakU7QUFDRDs7OztFQUlFLGdCQUFnQjtFQUNoQixzQkFBc0I7RUFDdEIsb0JBQW9CO0VBQ3BCLGVBQWU7RUFDZiw2QkFBNkI7RUFDN0IsbUJBQW1CO0VBQ25CLHdEQUF3RDtFQUN4RCxtQkFBbUI7RUFDbkIsc0JBQXNCO0VBQ3RCLG9CQUFvQjtFQUNwQixhQUFhO0VBQ2IsbUNBQW1DO0VBQ25DLGtCQUFrQjtFQUNsQix3QkFBd0I7RUFDeEIsZUFBZTtFQUNmLHVCQUF1QjtFQUN2QiwwQkFBMEI7RUFDMUIsMEJBQTBCO0VBQzFCLDZCQUE2QjtFQUM3Qiw0QkFBNEI7RUFDNUIsMkNBQW1DO1VBQW5DLG1DQUFtQztFQUNuQyxzRkFBc0Y7RUFDdEYsK0JBQXVCO1VBQXZCLHVCQUF1QjtFQUN2QixrQkFBa0I7Q0FDbkIiLCJmaWxlIjoidG1wMi5jc3MiLCJzb3VyY2VzQ29udGVudCI6WyJrYmQsXG4ua2V5IHtcbiAgZGlzcGxheTogaW5saW5lO1xuICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gIG1pbi13aWR0aDogMWVtO1xuICBwYWRkaW5nOiAuM2VtIC40ZW0gLjJlbSAuM2VtO1xuICBmb250LXN0eWxlOiBub3JtYWw7XG4gIGZvbnQtZmFtaWx5OiBcIkx1Y2lkYSBHcmFuZGVcIiwgTHVjaWRhLCBBcmlhbCwgc2Fucy1zZXJpZjtcbiAgdGV4dC1hbGlnbjogY2VudGVyO1xuICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gIGJvcmRlci1yYWRpdXM6IC4zZW07XG4gIGJvcmRlcjogbm9uZTtcbiAgYmFja2dyb3VuZC1jb2xvcjogIzUwNTA1MDtcbiAgYmFja2dyb3VuZC1jb2xvcjogZ3JhZGllbnQobGluZWFyLCBsZWZ0IHRvcCwgbGVmdCBib3R0b20sIGZyb20oIzNjM2MzYyksIHRvKCM1MDUwNTApKTtcbiAgY29sb3I6ICNmYWZhZmE7XG4gIHRleHQtc2hhZG93OiAtMXB4IC0xcHggMCAjNDY0NjQ2O1xuICBib3gtc2hhZG93OiBpbnNldCAwIDAgMXB4ICM5Njk2OTYsIGluc2V0IDAgLTAuMDVlbSAwLjRlbSAjNTA1MDUwLCAwIDAuMWVtIDAgIzFlMWUxZSwgMCAwLjFlbSAwLjFlbSByZ2JhKDAsIDAsIDAsIDAuMyk7XG4gIGZvbnQtc2l6ZTogLjg1ZW07XG4gIGxpbmUtaGVpZ2h0OiAxO1xuICBjdXJzb3I6IGRlZmF1bHQ7XG4gIHVzZXItc2VsZWN0OiBub25lO1xufVxua2JkW3RpdGxlXSxcbi5rZXlbdGl0bGVdIHtcbiAgY3Vyc29yOiBoZWxwO1xufVxua2JkLmRhcmssXG4uZGFyay1rZXlzIGtiZCxcbi5rZXkuZGFyayxcbi5kYXJrLWtleXMgLmtleSB7XG4gIGRpc3BsYXk6IGlubGluZTtcbiAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICB3aGl0ZS1zcGFjZTogbm93cmFwO1xuICBtaW4td2lkdGg6IDFlbTtcbiAgcGFkZGluZzogLjNlbSAuNGVtIC4yZW0gLjNlbTtcbiAgZm9udC1zdHlsZTogbm9ybWFsO1xuICBmb250LWZhbWlseTogXCJMdWNpZGEgR3JhbmRlXCIsIEx1Y2lkYSwgQXJpYWwsIHNhbnMtc2VyaWY7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICBib3JkZXItcmFkaXVzOiAuM2VtO1xuICBib3JkZXI6IG5vbmU7XG4gIGJhY2tncm91bmQtY29sb3I6ICM1MDUwNTA7XG4gIGJhY2tncm91bmQtY29sb3I6IGdyYWRpZW50KGxpbmVhciwgbGVmdCB0b3AsIGxlZnQgYm90dG9tLCBmcm9tKCMzYzNjM2MpLCB0bygjNTA1MDUwKSk7XG4gIGNvbG9yOiAjZmFmYWZhO1xuICB0ZXh0LXNoYWRvdzogLTFweCAtMXB4IDAgIzQ2NDY0NjtcbiAgYm94LXNoYWRvdzogaW5zZXQgMCAwIDFweCAjOTY5Njk2LCBpbnNldCAwIC0wLjA1ZW0gMC40ZW0gIzUwNTA1MCwgMCAwLjFlbSAwICMxZTFlMWUsIDAgMC4xZW0gMC4xZW0gcmdiYSgwLCAwLCAwLCAwLjMpO1xufVxua2JkLmxpZ2h0LFxuLmxpZ2h0LWtleXMga2JkLFxuLmtleS5saWdodCxcbi5saWdodC1rZXlzIC5rZXkge1xuICBkaXNwbGF5OiBpbmxpbmU7XG4gIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgbWluLXdpZHRoOiAxZW07XG4gIHBhZGRpbmc6IC4zZW0gLjRlbSAuMmVtIC4zZW07XG4gIGZvbnQtc3R5bGU6IG5vcm1hbDtcbiAgZm9udC1mYW1pbHk6IFwiTHVjaWRhIEdyYW5kZVwiLCBMdWNpZGEsIEFyaWFsLCBzYW5zLXNlcmlmO1xuICB0ZXh0LWFsaWduOiBjZW50ZXI7XG4gIHRleHQtZGVjb3JhdGlvbjogbm9uZTtcbiAgYm9yZGVyLXJhZGl1czogLjNlbTtcbiAgYm9yZGVyOiBub25lO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAjZmFmYWZhO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiBncmFkaWVudChsaW5lYXIsIGxlZnQgdG9wLCBsZWZ0IGJvdHRvbSwgZnJvbSgjZDJkMmQyKSwgdG8oI2ZmZmZmZikpO1xuICBjb2xvcjogIzMyMzIzMjtcbiAgdGV4dC1zaGFkb3c6IDAgMCAycHggI2ZmZmZmZjtcbiAgYm94LXNoYWRvdzogaW5zZXQgMCAwIDFweCAjZmZmZmZmLCBpbnNldCAwIDAgMC40ZW0gI2M4YzhjOCwgMCAwLjFlbSAwICM4MjgyODIsIDAgMC4xMWVtIDAgcmdiYSgwLCAwLCAwLCAwLjQpLCAwIDAuMWVtIDAuMTFlbSByZ2JhKDAsIDAsIDAsIDAuOSk7XG59XG5rYmQuc28sXG4uc28ta2V5cyBrYmQsXG4ua2V5LnNvLFxuLnNvLWtleXMgLmtleSB7XG4gIGRpc3BsYXk6IGlubGluZTtcbiAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICB3aGl0ZS1zcGFjZTogbm93cmFwO1xuICBtaW4td2lkdGg6IDFlbTtcbiAgcGFkZGluZzogLjNlbSAuNGVtIC4yZW0gLjNlbTtcbiAgZm9udC1zdHlsZTogbm9ybWFsO1xuICBmb250LWZhbWlseTogXCJMdWNpZGEgR3JhbmRlXCIsIEx1Y2lkYSwgQXJpYWwsIHNhbnMtc2VyaWY7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICBib3JkZXItcmFkaXVzOiAuM2VtO1xuICBib3JkZXI6IG5vbmU7XG4gIG1hcmdpbjogMCAuMWVtO1xuICBwYWRkaW5nOiAuMWVtIC42ZW07XG4gIGZvbnQtZmFtaWx5OiBBcmlhbCwgXCJIZWx2ZXRpY2EgTmV1ZVwiLCBIZWx2ZXRpY2EsIHNhbnMtc2VyaWY7XG4gIGxpbmUtaGVpZ2h0OiAxLjQ7XG4gIGNvbG9yOiAjMjQyNzI5O1xuICB0ZXh0LXNoYWRvdzogMCAxcHggMCAjRkZGO1xuICBiYWNrZ3JvdW5kLWNvbG9yOiAjZTFlM2U1O1xuICBib3JkZXI6IDFweCBzb2xpZCAjYWRiM2I5O1xuICBib3JkZXItcmFkaXVzOiAwLjI3MjcyNzI3ZW07XG4gIGJveC1zaGFkb3c6IDAgMXB4IDAgcmdiYSgxMiwgMTMsIDE0LCAwLjIpLCAwIDAgMCAycHggI0ZGRiBpbnNldDtcbn1cbmtiZC5naXRodWIsXG4uZ2l0aHViLWtleXMga2JkLFxuLmtleS5naXRodWIsXG4uZ2l0aHViLWtleXMgLmtleSB7XG4gIGRpc3BsYXk6IGlubGluZTtcbiAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICB3aGl0ZS1zcGFjZTogbm93cmFwO1xuICBtaW4td2lkdGg6IDFlbTtcbiAgcGFkZGluZzogLjNlbSAuNGVtIC4yZW0gLjNlbTtcbiAgZm9udC1zdHlsZTogbm9ybWFsO1xuICBmb250LWZhbWlseTogXCJMdWNpZGEgR3JhbmRlXCIsIEx1Y2lkYSwgQXJpYWwsIHNhbnMtc2VyaWY7XG4gIHRleHQtYWxpZ246IGNlbnRlcjtcbiAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICBib3JkZXItcmFkaXVzOiAuM2VtO1xuICBib3JkZXI6IG5vbmU7XG4gIHBhZGRpbmc6IDAuMjcyNzI3MjdlbSAwLjQ1NDU0NTQ1ZW07XG4gIGZvbnQtc2l6ZTogNjguNzUlO1xuICBsaW5lLWhlaWdodDogMC45MDkwOTA5MTtcbiAgY29sb3I6ICM0NDRkNTY7XG4gIHZlcnRpY2FsLWFsaWduOiBtaWRkbGU7XG4gIGJhY2tncm91bmQtY29sb3I6ICNmYWZiZmM7XG4gIGJvcmRlcjogc29saWQgMXB4ICNjNmNiZDE7XG4gIGJvcmRlci1ib3R0b20tY29sb3I6ICM5NTlkYTU7XG4gIGJvcmRlci1yYWRpdXM6IDAuMjcyNzI3MjdlbTtcbiAgYm94LXNoYWRvdzogaW5zZXQgMCAtMXB4IDAgIzk1OWRhNTtcbiAgZm9udC1mYW1pbHk6IFwiU0ZNb25vLVJlZ3VsYXJcIiwgQ29uc29sYXMsIFwiTGliZXJhdGlvbiBNb25vXCIsIE1lbmxvLCBDb3VyaWVyLCBtb25vc3BhY2U7XG4gIGJveC1zaXppbmc6IGJvcmRlci1ib3g7XG4gIHRleHQtc2hhZG93OiBub25lO1xufVxuIl19 */';
const fix =
"#nprogress .bar {\n background: #29d;\n position: fixed;\n z-index: 1031;\n top: 0;\n left: 0;\n width: 100%;\n height: 4px;\n}\n\n#pagesSlider {\n margin: 10px 0;\n}\n\n#pageInputs {\n display: flex;\n gap: 5px;\n align-items: center;\n justify-content: center;\n}\n\n#swal2-html-container .pageInput {\n border: 1px darkblue dashed;\n border-radius: 5px;\n text-align: center;\n background-color: aliceblue;\n color: black;\n max-width: 40%;\n}\n\n#swal2-title {\n color: navy;\n}\n\nbutton.swal2-styled {\n position: inherit;\n transform: inherit;\n}\n";
const sweetalertStyle = [normalize, sweetalert, fix, nprogress, keyscss].join(
"\n",
);
function head(manga) {
return html`
${manga.title}
${wrapStyle("externals", sweetalertStyle)}
${wrapStyle("reader", cssStyles)} ${themesCSS}
${wrapStyle(
"MinZoom",
`#MangaOnlineViewer .PageContent .PageImg {min-width: ${
getUserSettings().minZoom
}vw;}`,
)}
`;
}
const localeSelector = () =>
locales
.map(
(locale) => html`
${locale.NAME}
`,
)
.join("");
const themesSelector = () =>
[...Object.keys(colors).map((color) => colors[color].name)]
.map(
(theme2) => html`
${IconCheck}
`,
)
.join("");
const language = html`
${getLocaleString("LANGUAGE")}
${localeSelector()}
`;
const theme = html`
`;
const loadMode = html`
${getLocaleString("DEFAULT_LOAD_MODE")}
${getLocaleString("LOAD_MODE_NORMAL")}
${getLocaleString("LOAD_MODE_ALWAYS")}
${getLocaleString("LOAD_MODE_NEVER")}
`;
const loadSpeed = html`
${getLocaleString("LOAD_SPEED")}
0.3(${getLocaleString("SLOWLY")})
0.5
01(${getLocaleString("NORMAL")})
02
04(${getLocaleString("FAST")})
08
10(${getLocaleString("EXTREME")})
${getLocaleString("ALL_PAGES")}
`;
const defaultZoomMode = html`
${getLocaleString("DEFAULT_ZOOM_MODE")}
${getLocaleString("PERCENT")}
${getLocaleString("FIT_WIDTH")}
${getLocaleString("FIT_HEIGHT")}
`;
const defaultZoom = html`
${getLocaleString("DEFAULT_ZOOM")}
${getUserSettings().defaultZoom}%
5
25
50
75
100
125
150
175
200
`;
const minZoom = html`
${getLocaleString("MINIMUM_ZOOM")}
${getUserSettings().minZoom}%
`;
const zoomStep = html`
${getLocaleString("ZOOM_STEP")}
${getUserSettings().zoomStep}%
`;
const viewMode$1 = html`
${getLocaleString("DEFAULT_VIEW_MODE")}
${getLocaleString("VIEW_MODE_VERTICAL")}
${getLocaleString("VIEW_MODE_WEBCOMIC")}
${getLocaleString("VIEW_MODE_LEFT")}
${getLocaleString("VIEW_MODE_RIGHT")}
`;
const lazyLoad$1 = html`
${getLocaleString("LAZY_LOAD_IMAGES")}
${getUserSettings().lazyStart}
`;
const headerType = html`
`;
const checkboxOptions = html`
${getLocaleString("FIT_WIDTH_OVERSIZED")}
${getLocaleString("SHOW_THUMBNAILS")}
${getLocaleString("LAZY_LOAD_IMAGES_ENABLE")}
${lazyLoad$1}
${getLocaleString("DOWNLOAD_IMAGES")}
${getLocaleString("HIDE_CONTROLS")}
`;
const autoScroll = html`
${getLocaleString("AUTO_SCROLL_HEIGHT")}
${getUserSettings().scrollHeight} px
`;
const SettingsPanel = () => html`
${getLocaleString("SETTINGS")}
${IconX}
${getLocaleString("BUTTON_RESET_SETTINGS")}
${language} ${theme} ${loadMode} ${loadSpeed} ${defaultZoomMode}
${defaultZoom} ${minZoom} ${zoomStep} ${viewMode$1} ${checkboxOptions}
${headerType} ${autoScroll}
`;
const keybindList = () =>
Object.keys(getUserSettings().keybinds).map((kb) => {
const keys = getUserSettings().keybinds[kb]?.length
? getUserSettings()
.keybinds[kb]?.map((key) => html`${key} `)
.join(" / ")
: "";
return html`${getLocaleString(kb)}: ${keys} `;
});
const keybindEditor = () =>
Object.keys(getUserSettings().keybinds)
.map(
// Language=html
(kb) =>
html`${getLocaleString(kb)}:
`,
)
.concat(
html`${getLocaleString("KEYBIND_RULES")}
`,
);
const KeybindingsPanel = () => html`
${getLocaleString("KEYBINDINGS")}
${IconX}
${IconPencil} ${getLocaleString("BUTTON_EDIT")}
${IconDeviceFloppy} ${getLocaleString("BUTTON_SAVE")}
${keybindList().join("\n")}
`;
function indexList(repeat, begin = 1) {
return Array(repeat)
.fill(0)
.map((_, i) => i + 1)
.filter((i) => i >= begin);
}
const ThumbnailsPanel = (manga) => html`
${IconCategory}
0 /
${manga.begin > 1 ? manga.pages - (manga.begin - 1) : manga.pages}
${getLocaleString("PAGES_LOADED")}
${indexList(manga.pages, manga.begin)
.map(
(index) => html`
${index}
`,
)
.join("")}
`;
const listBookmarks = () => {
if (isEmpty(getUserSettings().bookmarks)) {
return [getLocaleString("LIST_EMPTY")];
}
return getUserSettings().bookmarks.map(
(mark, index) => html`
`,
);
};
const BookmarkPanel = () => html`
${IconX}
${getLocaleString("BOOKMARKS")}
`;
function reloadBookmarks() {
const list = document.getElementById("BookmarksList");
if (list) {
list.innerHTML = listBookmarks().join("");
}
}
const listOptions = (times, begin) =>
indexList(times, begin).map(
(index) => html` ${index} `,
);
const Header = (manga) => html`
`;
const listPages = (times, begin) =>
indexList(times, begin).map(
// Language=html
(index) => html`
${IconBookmark} ${IconBookmarkOff}
${IconZoomIn}
${IconZoomCancel}
${IconZoomOut}
${IconArrowAutofitWidth}
${IconArrowAutofitHeight}
${IconEye} ${IconEyeOff}
${IconRefresh}
${index}
`,
);
const Reader = (manga) => html`
${listPages(manga.pages, manga.begin).join("")}
`;
const commentsPanel = (manga) => html`
`;
function scrollToElement(ele) {
const chapter = document.querySelector("#Chapter");
if (
chapter?.classList.contains("FluidLTR") ||
chapter?.classList.contains("FluidRTL")
) {
chapter?.scroll(ele?.offsetLeft ?? 0, ele?.offsetTop ?? 0);
} else {
window?.scroll(ele?.offsetLeft ?? 0, ele?.offsetTop ?? 0);
}
}
function addEvent(ev, fn) {
return (elem) => elem.addEventListener(ev, fn);
}
function transformScrollToHorizontal(event) {
if (!event.deltaY) {
return;
}
event.currentTarget.scrollLeft += event.deltaY + event.deltaX;
event.preventDefault();
}
function transformScrollToHorizontalReverse(event) {
if (!event.deltaY) {
return;
}
event.currentTarget.scrollLeft -= event.deltaY + event.deltaX;
event.preventDefault();
}
function buttonBookmarksOpen() {
reloadBookmarks();
document.querySelector("#BookmarksPanel")?.classList.add("visible");
document.querySelector("#Overlay")?.classList.add("visible");
}
function buttonBookmarksClose() {
document.querySelector("#BookmarksPanel")?.classList.remove("visible");
document.querySelector("#Overlay")?.classList.remove("visible");
}
function buttonEraseBookmarks(event) {
const target = event.currentTarget.value;
const marks = getUserSettings().bookmarks.filter((el) => el.url !== target);
if (target === window.location.href) {
document
.querySelector("#MangaOnlineViewer")
?.classList.toggle("bookmarked");
}
logScript(`Bookmark Removed ${target}`);
Swal.fire({
title: getLocaleString("BOOKMARK_REMOVED"),
timer: 1e4,
icon: "error",
});
updateSettings({ bookmarks: marks });
reloadBookmarks();
document
.querySelectorAll(".BookmarkItem .erase")
?.forEach(addEvent("click", buttonEraseBookmarks));
}
function buttonBookmark(event) {
document
.querySelector("#MangaOnlineViewer")
?.classList.toggle("bookmarked");
const num = parseInt(
event.currentTarget.parentElement?.querySelector(".PageIndex")
?.textContent ?? "0",
10,
);
const mark = {
url: window.location.href,
page: num,
date: /* @__PURE__ */ new Date().toISOString().slice(0, 10),
};
if (isBookmarked(mark.url)) {
updateSettings({
bookmarks: getUserSettings().bookmarks.filter(
(el) => el.url !== mark.url,
),
});
Swal.fire({
title: getLocaleString("BOOKMARK_REMOVED"),
timer: 1e4,
icon: "error",
});
} else {
updateSettings({ bookmarks: [...getUserSettings().bookmarks, mark] });
Swal.fire({
title: getLocaleString("BOOKMARK_SAVED"),
html: getLocaleString("BOOKMARK_SAVED").replace(
"##NUM##",
num.toString(),
),
icon: "success",
});
}
reloadBookmarks();
document
.querySelectorAll(".BookmarkItem .erase")
?.forEach(addEvent("click", buttonEraseBookmarks));
}
function bookmarks() {
document
.querySelector("#bookmarks")
?.addEventListener("click", buttonBookmarksOpen);
document
.querySelectorAll(".closeButton")
?.forEach(addEvent("click", buttonBookmarksClose));
document
.querySelector("#Overlay")
?.addEventListener("click", buttonBookmarksClose);
document
.querySelectorAll(".BookmarkItem .erase")
?.forEach(addEvent("click", buttonEraseBookmarks));
document
.querySelectorAll(".Bookmark")
?.forEach(addEvent("click", buttonBookmark));
}
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);
var FileSaver_minExports = FileSaver_min.exports;
let zip;
const base64Regex = /^data:(?image\/\w+);base64,+(?.+)/;
const getExtension = (mimeType) =>
/image\/(?jpe?g|png|webp)/.exec(mimeType)?.groups?.ext ?? "png";
const getFilename = (name, index, total, ext) =>
`${name}${(index + 1)
.toString()
.padStart(Math.floor(Math.log10(total)) + 1, "0")}.${ext.replace(
"jpeg",
"jpg",
)}`;
async function getImage(src) {
return new Promise((resolve) => {
logScript(`Getting Image data: ${src}`);
GM_xmlhttpRequest({
method: "GET",
url: src,
headers: { referer: src, origin: src },
responseType: "blob",
onload(response) {
resolve(response);
},
});
});
}
async function getImageData(img, index, array) {
const src = img.getAttribute("src") ?? img.getAttribute("data-src");
if (src == null) {
return Promise.reject(new Error("Image source not specified"));
}
const base64 = base64Regex.exec(src);
if (base64?.groups) {
return Promise.resolve({
name: getFilename(
"Page-",
index,
array.length,
getExtension(base64.groups?.mimeType),
),
data: base64.groups.data,
});
}
return new Promise((resolve) => {
getImage(src)
.then((res) => {
resolve({
name: getFilename(
"Page-",
index,
array.length,
getExtension(res.response.type),
),
data: res.response,
});
})
.catch(logScript);
});
}
function addZip(img) {
logScript(`${img.name} Added to Zip from Base64 Image`);
zip.file(img.name, img.data, {
base64: true,
createFolders: true,
compression: "DEFLATE",
});
}
async function generateZip() {
zip = new JSZip();
const images = [...document.querySelectorAll(".PageImg")];
const data = await Promise.all(images.map(getImageData));
data.forEach(addZip);
logScript("Generating Zip");
zip
.generateAsync(
{
type: "blob",
},
// LogScript, progress
)
.then((content) => {
logScript("Download Ready");
const zipName = `${document
.querySelector("#MangaTitle")
?.textContent?.trim()}.zip`;
FileSaver_minExports.saveAs(content, zipName, { autoBom: false });
document.getElementById("download")?.classList.remove("loading");
})
.catch(logScript);
}
function buttonStartDownload(event) {
const button = event.currentTarget;
if (button.classList.contains("loading")) {
return;
}
logScript("Downloading Chapter");
button.classList.add("loading");
generateZip().catch((err) => logScript("Error downloading chapter", err));
}
function buttonGlobalHideImageControls() {
document
.querySelector("#MangaOnlineViewer")
?.classList.toggle("hideControls");
}
function buttonRedirectURL(event) {
const element = event.target;
const url = element.getAttribute("value") ?? element.getAttribute("href");
if (url) {
window.location.href = url;
} else {
window.history.back();
}
}
function buttonCommentsOpen() {
document.querySelector("#CommentsPanel")?.classList.add("visible");
document.querySelector("#Overlay")?.classList.add("visible");
}
function buttonCommentsClose() {
document.querySelector("#CommentsPanel")?.classList.remove("visible");
document.querySelector("#Overlay")?.classList.remove("visible");
}
function globals() {
document
.querySelector("#download")
?.addEventListener("click", buttonStartDownload);
document
.querySelector("#pageControls")
?.addEventListener("click", buttonGlobalHideImageControls);
document
.querySelector("#next")
?.addEventListener("click", buttonRedirectURL);
document
.querySelector("#prev")
?.addEventListener("click", buttonRedirectURL);
document
.querySelector("#series")
?.addEventListener("click", buttonRedirectURL);
document
.querySelector("#CommentsButton")
?.addEventListener("click", buttonCommentsOpen);
document
.querySelectorAll(".closeButton")
?.forEach(addEvent("click", buttonCommentsClose));
document
.querySelector("#Overlay")
?.addEventListener("click", buttonCommentsClose);
}
function headroom(showEnd = 0) {
let prevOffset = 0;
const setScrollDirection = (classSuffix) => {
const header = document.querySelector("#Header");
header.classList.remove("headroom-end");
header.classList.remove("headroom-hide");
header.classList.remove("headroom-show");
if (classSuffix) {
header.classList.add(`headroom-${classSuffix}`);
}
};
function toggleScrollDirection() {
const { scrollY } = window;
if (
showEnd &&
getUserSettings().zoomMode !== "height" &&
scrollY + window.innerHeight + showEnd > document.body.scrollHeight
) {
setScrollDirection("end");
} else if (scrollY > prevOffset && scrollY > 50) {
setScrollDirection("hide");
} else if (scrollY < prevOffset && scrollY > 50) {
setScrollDirection("show");
} else {
setScrollDirection("");
}
prevOffset = scrollY;
}
window.addEventListener("scroll", _.debounce(toggleScrollDirection, 50));
}
const doClick = (selector) =>
document.querySelector(selector)?.dispatchEvent(new Event("click"));
function doScrolling(sign) {
const chapter = document.querySelector("#Chapter");
if (
chapter?.classList.contains("FluidLTR") ||
chapter?.classList.contains("FluidRTL")
) {
chapter.scrollBy({
left:
(window.innerWidth / 2) *
sign *
(chapter?.classList.contains("FluidRTL") ? -1 : 1),
behavior: "smooth",
});
} else if (getUserSettings().zoomMode === "height") {
const pages = [...document.querySelectorAll(".MangaPage")];
const distance = pages.map((element) =>
Math.abs(element.offsetTop - window.scrollY),
);
const currentPage = distance.findIndex((d) => d <= 5);
const target = currentPage + sign;
const header = document.querySelector("#Header");
if (target < 0) {
scrollToElement(header);
} else if (target >= pages.length) {
header.classList.add("headroom-end");
} else {
logScript(
`Current array page ${currentPage},`,
`Scrolling to page ${target}`,
);
scrollToElement(pages.at(target));
}
} else {
window.scrollBy({
top: (sign * window.innerHeight) / 2,
behavior: "smooth",
});
}
}
const actions = {
SCROLL_UP() {
doScrolling(-1);
},
SCROLL_DOWN() {
doScrolling(1);
},
NEXT_CHAPTER() {
doClick("#next");
},
PREVIOUS_CHAPTER() {
doClick("#prev");
},
ENLARGE() {
doClick("#enlarge");
},
REDUCE() {
doClick("#reduce");
},
RESTORE() {
doClick("#restore");
},
FIT_WIDTH() {
doClick("#fitWidth");
},
FIT_HEIGHT() {
doClick("#fitHeight");
},
SETTINGS() {
doClick("#settings");
},
VIEW_MODE_WEBCOMIC() {
doClick("#webComic");
},
VIEW_MODE_VERTICAL() {
doClick("#verticalMode");
},
VIEW_MODE_LEFT() {
doClick("#rtlMode");
},
VIEW_MODE_RIGHT() {
doClick("#ltrMode");
},
SCROLL_START() {
doClick("#AutoScroll");
},
};
function keybindings() {
document.onkeydown = null;
document.onkeyup = null;
document.onkeypress = null;
window.onkeydown = null;
window.onkeyup = null;
window.onkeypress = null;
window.onload = null;
document.body.onload = null;
hotkeys.unbind();
Object.keys(getUserSettings().keybinds).forEach((key) => {
hotkeys(getUserSettings().keybinds[key]?.join(",") ?? "", (event) => {
event.preventDefault();
event.stopImmediatePropagation();
event.stopPropagation();
actions[key]();
});
});
}
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);
}
async function fetchText(url, format) {
return new Promise((resolve) => {
logScript("Fetching page: ", url);
fetch(url)
.then(async (response) =>
// When the page is loaded convert it to text
response.text(),
)
.then((html) => {
const parser = new DOMParser();
const doc = parser.parseFromString(html, format);
resolve(doc);
})
.catch((err) => {
logScript("Failed to fetch page: ", err);
});
});
}
async function fetchHtml(url) {
return fetchText(url, "text/html");
}
async function getElementAttribute(url, selector, attribute) {
return fetchHtml(url).then((doc) =>
doc.querySelector(selector)?.getAttribute(attribute),
);
}
const settings = {
threshold: 2e3,
throttle: 500,
lazyAttribute: "data-src",
targetAttribute: "src",
};
let listElements = [];
let setup = false;
function filterInView(value) {
const { element } = value;
const rect = element.getBoundingClientRect();
const target =
(window.innerHeight || document.documentElement.clientHeight) +
settings.threshold;
return rect.top <= target || rect.bottom <= target;
}
function showElement(item) {
const value = item.element.getAttribute(settings.lazyAttribute);
if (value) {
item.element.setAttribute(settings.targetAttribute, value);
}
item.callback(item.element)?.catch(logScript);
}
function executeCheck() {
const inView = listElements.filter(filterInView);
listElements = listElements.filter((item) => !filterInView(item));
inView.forEach(showElement);
}
const observerEvent = _.throttle(executeCheck, settings.throttle);
function lazyLoad(element, callback) {
if (!setup) {
window.addEventListener("scroll", observerEvent, {
passive: true,
});
window.addEventListener("touchmove", observerEvent, {
passive: true,
});
window.addEventListener("resize", observerEvent, {
passive: true,
});
setup = true;
}
listElements.push({ element, callback });
observerEvent();
}
function applyZoom(
zoom = getUserSettings().zoomMode,
pages = ".PageContent img",
) {
const pg = [...document.querySelectorAll(pages)];
pg.forEach((img) => {
img.removeAttribute("width");
img.removeAttribute("height");
img.removeAttribute("style");
if (zoom === "width") {
img.style.width = `${window.innerWidth}px`;
} else if (zoom === "height") {
const chapter = document.querySelector("#Chapter");
const fluid =
chapter?.classList.contains("FluidLTR") ||
chapter?.classList.contains("FluidRTL");
const nextHeight =
window.innerHeight +
(getUserSettings().showThumbnails || fluid ? -30 : 0);
img.style.height = `${nextHeight}px`;
img.style.minWidth = "unset";
} else if (zoom === "percent") {
img.style.width = `${
img.naturalWidth * (getUserSettings().defaultZoom / 100)
}px`;
} else if (zoom >= 0 && zoom !== 100) {
img.style.width = `${img.naturalWidth * (zoom / 100)}px`;
}
});
}
function invalidateImageCache(src, repeat) {
const url = src.replace(/[?&]cache=\d+$/, "");
const symbol = !url.includes("?") ? "?" : "&";
return `${url + symbol}cache=${repeat}`;
}
function getRepeatValue(src) {
let repeat = 1;
const cache = src?.match(/cache=(\d+)$/);
if (cache?.at(1)) {
repeat = parseInt(cache[1], 10) + 1;
}
return repeat;
}
function reloadImage(img) {
const src = img.getAttribute("src");
if (!src) {
return;
}
img.removeAttribute("src");
img.setAttribute("src", invalidateImageCache(src, getRepeatValue(src)));
}
function onImagesDone() {
logScript("Images Loading Complete");
if (getUserSettings().downloadZip) {
document.getElementById("download")?.dispatchEvent(new Event("click"));
}
document.getElementById("download")?.classList.remove("disabled");
}
function updateProgress() {
const total = document.querySelectorAll(".PageContent .PageImg").length;
const loaded = document.querySelectorAll(
".PageContent .PageImg.imgLoaded",
).length;
const percentage = Math.floor((loaded / total) * 100);
const title = document.querySelector("title");
if (title) {
title.innerHTML = html`(${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();
}
}
function onImagesSuccess(instance) {
instance.images.forEach((image) => {
image.img.classList.add("imgLoaded");
image.img.classList.remove("imgBroken");
const thumbId = image.img.id.replace("PageImg", "ThumbnailImg");
const thumb = document.getElementById(thumbId);
if (thumb) {
thumb.setAttribute("src", image.img.getAttribute("src"));
}
applyZoom(getUserSettings().zoomMode, `#${image.img.id}`);
updateProgress();
});
}
function onImagesFail(instance) {
instance.images.forEach((image) => {
image.img.classList.add("imgBroken");
const src = image.img.getAttribute("src");
if (src && getRepeatValue(src) <= getUserSettings().maxReload) {
setTimeout(() => {
reloadImage(image.img);
const imgLoad = imagesLoaded(image.img.parentElement);
imgLoad.on("done", onImagesSuccess);
imgLoad.on("fail", onImagesFail);
}, 2e3);
}
});
}
function normalizeUrl(url = "") {
let uri = url.trim();
if (uri.startsWith("//")) {
uri = `https:${uri}`;
}
return uri;
}
function addImg(manga, index, imageSrc, position) {
const relativePosition = position - manga.begin;
const src = normalizeUrl(imageSrc);
const img = document.querySelector(`#PageImg${index}`);
if (img) {
if (
!getUserSettings().lazyLoadImages ||
relativePosition <= getUserSettings().lazyStart
) {
setTimeout(
() => {
const imgLoad = imagesLoaded(img.parentElement);
imgLoad.on("done", onImagesSuccess);
imgLoad.on("fail", onImagesFail);
img.setAttribute("src", src);
logScript("Loaded Image:", index, "Source:", src);
},
(manga.timer ?? getUserSettings().throttlePageLoad) *
relativePosition,
);
} else {
img.setAttribute("data-src", src);
lazyLoad(img, () => {
const imgLoad = imagesLoaded(img.parentElement);
imgLoad.on("done", onImagesSuccess);
imgLoad.on("fail", onImagesFail);
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.style.width = "auto";
const imgLoad = imagesLoaded(img.parentElement);
imgLoad.on("done", onImagesSuccess);
imgLoad.on("fail", onImagesFail);
img.setAttribute("src", src);
logScript(
`${lazy && "Lazy "}Page: `,
index,
" Source: ",
img.getAttribute("src"),
);
}
};
}
async function addPage(manga, index, pageUrl, position) {
const relativePosition = position - manga.begin;
const img = document.querySelector(`#PageImg${index}`);
if (img) {
if (
!getUserSettings().lazyLoadImages ||
relativePosition <= getUserSettings().lazyStart
) {
setTimeout(
() => {
findPage(manga, index, pageUrl, false)().catch(logScript);
},
(manga.timer ?? getUserSettings().throttlePageLoad) *
relativePosition,
);
} else {
img.setAttribute(
"data-src",
"",
);
lazyLoad(img, findPage(manga, index, pageUrl, true));
}
}
}
function loadMangaPages(begin, manga) {
indexList(manga.pages, begin).forEach((index, position) => {
addPage(manga, index, manga.listPages[index - 1], position).catch(
logScript,
);
});
}
function loadMangaImages(begin, manga) {
indexList(manga.pages, begin).forEach((index, position) => {
addImg(manga, index, manga.listImages[index - 1], position);
});
}
function loadManga(manga, begin = 1) {
getUserSettings().lazyLoadImages =
manga.lazy ?? getUserSettings().lazyLoadImages;
logScript("Loading Images");
logScript(
`Intervals: ${
manga.timer ?? getUserSettings().throttlePageLoad ?? "Default(1000)"
}`,
);
logScript(
`Lazy: ${getUserSettings().lazyLoadImages}, Starting from: ${
getUserSettings().lazyStart
}`,
);
if (isImagesManga(manga)) {
logScript("Method: Images:", manga.listImages);
loadMangaImages(begin, manga);
} else if (isPagesManga(manga)) {
logScript("Method: Pages:", manga.listPages);
loadMangaPages(begin, manga);
} else if (isBruteforceManga(manga)) {
logScript("Method: Brute Force");
manga.bruteForce({
begin,
addImg,
loadImages(list) {
loadMangaImages(begin, { ...manga, listImages: list });
},
loadPages(list, img, lazyAttr) {
loadMangaPages(begin, {
...manga,
listPages: list,
img,
lazyAttr,
});
},
wait: getUserSettings().throttlePageLoad,
});
} else {
logScript("No Loading Method Found");
}
}
function buttonReloadPage(event) {
const img =
event.currentTarget.parentElement?.parentElement?.querySelector(
".PageImg",
);
reloadImage(img);
}
function buttonHidePage(event) {
const img = event.currentTarget.parentElement?.parentElement;
img.classList.toggle("hide");
}
function individual() {
document
.querySelectorAll(".Reload")
?.forEach(addEvent("click", buttonReloadPage));
document
.querySelectorAll(".Hide")
?.forEach(addEvent("click", buttonHidePage));
}
function selectGoToPage(event) {
const target = event.currentTarget.value;
applyZoom();
scrollToElement(document.querySelector(`#Page${target}`));
}
function clickThumbnail(event) {
applyZoom();
scrollToElement(
document.querySelector(
`#Page${
event.currentTarget.querySelector(".ThumbnailIndex")?.textContent
}`,
),
);
}
function navigation() {
document
.querySelector("#gotoPage")
?.addEventListener("change", selectGoToPage);
document
.querySelectorAll(".Thumbnail")
?.forEach(addEvent("click", clickThumbnail));
document
.querySelector("#Thumbnails")
?.addEventListener("wheel", transformScrollToHorizontal);
}
function buttonResetSettings() {
resetSettings();
const elem = document.getElementById("MangaOnlineViewer");
elem?.removeAttribute("locale");
elem?.dispatchEvent(new Event("locale"));
}
function changeLocale(event) {
const locale = event.currentTarget.value;
updateSettings({ locale });
const elem = document.getElementById("MangaOnlineViewer");
elem?.setAttribute("locale", locale);
elem?.dispatchEvent(new Event("locale"));
}
function changeLoadMode(event) {
const mode = event.currentTarget.value;
updateSettings({ loadMode: mode });
}
function checkFitWidthOversize(event) {
document.querySelector("#Chapter")?.classList.toggle("fitWidthIfOversize");
updateSettings({ fitWidthIfOversize: event.currentTarget.checked });
}
function checkShowThumbnails(event) {
document.querySelector("#Navigation")?.classList.toggle("disabled");
updateSettings({ showThumbnails: event.currentTarget.checked });
applyZoom();
}
function changeAutoDownload(event) {
updateSettings({ downloadZip: event.currentTarget.checked });
if (event.currentTarget.checked) {
Swal.fire({
title: getLocaleString("ATTENTION"),
text: getLocaleString("AUTO_DOWNLOAD"),
timer: 1e4,
icon: "info",
});
}
}
function checkLazyLoad(event) {
updateSettings({ lazyLoadImages: event.currentTarget.checked });
const start = document.querySelector(".lazyStart");
if (getUserSettings().lazyLoadImages) {
start?.classList.add("show");
} else {
start?.classList.remove("show");
}
if (event.currentTarget.checked) {
Swal.fire({
title: getLocaleString("WARNING"),
html: getLocaleString("LAZY_LOAD"),
icon: "warning",
});
}
}
function changeLazyStart(event) {
const start = event.currentTarget.value;
updateSettings({ lazyStart: parseInt(start, 10) });
}
function changePagesPerSecond(event) {
const timer = parseInt(event.currentTarget.value, 10);
updateSettings({ throttlePageLoad: timer });
if (timer < 100) {
Swal.fire({
title: getLocaleString("SPEED_WARNING"),
html: getLocaleString("SPEED_WARNING_MESSAGE"),
icon: "warning",
});
}
}
function changeZoomStep(event) {
const step = event.currentTarget.value;
updateSettings({ zoomStep: parseInt(step, 10) });
}
function changeMinZoom(event) {
const min = event.currentTarget.value;
replaceStyleSheet(
"MinZoom",
`#MangaOnlineViewer .PageContent .PageImg {min-width: ${min}vw;}`,
);
updateSettings({ minZoom: parseInt(min, 10) });
}
function checkHideImageControls(event) {
document
.querySelector("#MangaOnlineViewer")
?.classList.toggle("hideControls");
updateSettings({ hidePageControls: event.currentTarget.checked });
}
function changeHeaderType(event) {
const headerType = event.currentTarget.value;
document.querySelector("#Header").className = headerType;
document.querySelector("#menu").className = headerType;
updateSettings({ header: headerType });
}
function changeScrollHeight(event) {
const { value } = event.currentTarget;
updateSettings({ scrollHeight: parseInt(value, 10) });
}
function options() {
document
.querySelector("#ResetSettings")
?.addEventListener("click", buttonResetSettings);
document.querySelector("#locale")?.addEventListener("change", changeLocale);
document
.querySelector("#fitIfOversize")
?.addEventListener("change", checkFitWidthOversize);
document
.querySelector("#loadMode")
?.addEventListener("change", changeLoadMode);
document
.querySelector("#showThumbnails")
?.addEventListener("change", checkShowThumbnails);
document
.querySelector("#downloadZip")
?.addEventListener("change", changeAutoDownload);
document
.querySelector("#lazyLoadImages")
?.addEventListener("change", checkLazyLoad);
document
.querySelector("#lazyStart")
?.addEventListener("change", changeLazyStart);
document
.querySelector("#PagesPerSecond")
?.addEventListener("change", changePagesPerSecond);
document
.querySelector("#zoomStep")
?.addEventListener("change", changeZoomStep);
document
.querySelector("#minZoom")
?.addEventListener("input", changeMinZoom);
document
.querySelector("#hidePageControls")
?.addEventListener("change", checkHideImageControls);
document
.querySelector("#headerType")
?.addEventListener("change", changeHeaderType);
document
.querySelector("#scrollHeight")
?.addEventListener("change", changeScrollHeight);
}
function buttonHeaderClick() {
const header = document.querySelector("#Header");
if (header?.classList.contains("click")) {
header?.classList.toggle("visible");
}
}
function isMouseInsideRegion(event, headerWidth, headerHeight) {
return (
event.clientX >= 0 &&
event.clientX <= headerWidth &&
event.clientY >= 0 &&
event.clientY <= headerHeight
);
}
function headerHover(event) {
const header = document.querySelector("#Header");
if (header?.classList.contains("hover")) {
if (isMouseInsideRegion(event, header.clientWidth, header.clientHeight)) {
document.querySelector("#menu")?.classList.add("hide");
header?.classList.add("visible");
} else {
document.querySelector("#menu")?.classList.remove("hide");
header?.classList.remove("visible");
}
}
}
function buttonSettingsOpen() {
document.querySelector("#SettingsPanel")?.classList.add("visible");
document.querySelector("#Navigation")?.classList.add("visible");
document.querySelector("#Header")?.classList.add("visible");
document.querySelector("#Overlay")?.classList.add("visible");
}
function buttonSettingsClose() {
document.querySelector("#SettingsPanel")?.classList.remove("visible");
document.querySelector("#Navigation")?.classList.remove("visible");
document.querySelector("#Header")?.classList.remove("visible");
document.querySelector("#Overlay")?.classList.remove("visible");
}
function buttonKeybindingsOpen() {
document.querySelector("#KeybindingsList").innerHTML =
keybindList().join("\n");
document.querySelector("#SaveKeybindings")?.classList.add("hidden");
document.querySelector("#EditKeybindings")?.classList.remove("hidden");
document.querySelector("#KeybindingsPanel")?.classList.add("visible");
document.querySelector("#Overlay")?.classList.add("visible");
}
function buttonKeybindingsClose() {
document.querySelector("#SaveKeybindings")?.classList.add("hidden");
document.querySelector("#EditKeybindings")?.classList.remove("hidden");
document.querySelector("#KeybindingsPanel")?.classList.remove("visible");
document.querySelector("#Overlay")?.classList.remove("visible");
}
function saveKeybindings() {
const newkeybinds = getUserSettings().keybinds;
Object.keys(getUserSettings().keybinds).forEach((kb) => {
const keys = document
.querySelector(`#${kb}`)
?.value.split(",")
?.map((value) => value.trim());
newkeybinds[kb] = isNothing(keys) ? void 0 : keys;
});
updateSettings({ keybinds: newkeybinds });
document.querySelector("#KeybindingsList").innerHTML =
keybindList().join("\n");
document.querySelector("#SaveKeybindings")?.classList.add("hidden");
document.querySelector("#EditKeybindings")?.classList.remove("hidden");
keybindings();
}
function editKeybindings() {
document.querySelector("#KeybindingsList").innerHTML =
keybindEditor().join("\n");
document.querySelector("#SaveKeybindings")?.classList.remove("hidden");
document.querySelector("#EditKeybindings")?.classList.add("hidden");
}
function panels() {
document
.querySelector("#menu")
?.addEventListener("click", buttonHeaderClick);
document.addEventListener("mousemove", _.throttle(headerHover, 300));
document
.querySelector("#settings")
?.addEventListener("click", buttonSettingsOpen);
document
.querySelectorAll(".closeButton")
?.forEach(addEvent("click", buttonSettingsClose));
document
.querySelector("#Overlay")
?.addEventListener("click", buttonSettingsClose);
document
.querySelector("#keybindings")
?.addEventListener("click", buttonKeybindingsOpen);
document
.querySelectorAll(".closeButton")
?.forEach(addEvent("click", buttonKeybindingsClose));
document
.querySelector("#Overlay")
?.addEventListener("click", buttonKeybindingsClose);
document
.querySelector("#EditKeybindings")
?.addEventListener("click", editKeybindings);
document
.querySelector("#SaveKeybindings")
?.addEventListener("click", saveKeybindings);
}
function buttonZoomIn(event) {
const img =
event.currentTarget.parentElement?.parentElement?.querySelector(
".PageImg",
);
const ratio =
(img.width / img.naturalWidth) * (100 + getUserSettings().zoomStep);
applyZoom(ratio, `#${img.getAttribute("id")}`);
}
function buttonZoomOut(event) {
const img =
event.currentTarget.parentElement?.parentElement?.querySelector(
".PageImg",
);
const ratio =
(img.width / img.naturalWidth) * (100 - getUserSettings().zoomStep);
applyZoom(ratio, `#${img.getAttribute("id")}`);
}
function buttonRestoreZoom() {
document.querySelector(".PageContent .PageImg")?.removeAttribute("width");
}
function buttonZoomWidth(event) {
const page = event.currentTarget.parentElement?.parentElement;
const img = page?.querySelector(".PageImg");
applyZoom("width", `#${img.getAttribute("id")}`);
page?.classList.toggle("DoublePage");
}
function buttonZoomHeight(event) {
const img =
event.currentTarget.parentElement?.parentElement?.querySelector(
".PageImg",
);
applyZoom("height", `#${img.getAttribute("id")}`);
}
function size() {
document
.querySelectorAll(".ZoomIn")
?.forEach(addEvent("click", buttonZoomIn));
document
.querySelectorAll(".ZoomOut")
?.forEach(addEvent("click", buttonZoomOut));
document
.querySelectorAll(".ZoomRestore")
?.forEach(addEvent("click", buttonRestoreZoom));
document
.querySelectorAll(".ZoomWidth")
?.forEach(addEvent("click", buttonZoomWidth));
document
.querySelectorAll(".ZoomHeight")
?.forEach(addEvent("click", buttonZoomHeight));
}
function changeColorScheme() {
const isDark = getUserSettings().colorScheme === "dark";
updateSettings({ colorScheme: isDark ? "light" : "dark" });
const elem = document.getElementById("MangaOnlineViewer");
elem?.classList.remove(isDark ? "dark" : "light");
elem?.classList.add(getUserSettings().colorScheme);
}
function buttonSelectTheme(event) {
const target = event.currentTarget;
[...document.querySelectorAll(".ThemeRadio")].forEach((theme) => {
theme.classList.remove("selected");
});
target.classList.add("selected");
document
.getElementById("MangaOnlineViewer")
?.setAttribute("data-theme", target.title);
updateSettings({ theme: target.title });
const hue = document.querySelector("#Hue");
const shade = document.querySelector("#Shade");
if (target.title.startsWith("custom")) {
hue?.classList.add("show");
shade?.classList.remove("show");
} else {
hue?.classList.remove("show");
shade?.classList.add("show");
}
}
function changeCustomTheme(event) {
const target = event.currentTarget.value;
updateSettings({ customTheme: target });
addCustomTheme(target);
}
function changeThemeShade(event) {
const target = parseInt(event.currentTarget.value, 10);
updateSettings({ themeShade: target });
refreshThemes();
}
function theming() {
document
.querySelector("#ColorScheme")
?.addEventListener("click", changeColorScheme);
document
.querySelectorAll(".ThemeRadio")
.forEach(addEvent("click", buttonSelectTheme));
document
.querySelector("#CustomThemeHue")
?.addEventListener("change", changeCustomTheme);
document
.querySelector("#ThemeShade")
?.addEventListener("input", changeThemeShade);
}
function changeGlobalZoom(value) {
return () => {
if (typeof value !== "number") {
getUserSettings().zoomMode = value;
} else {
getUserSettings().zoomMode = "percent";
}
const globalZoomVal = document.querySelector("#ZoomVal");
if (Number.isInteger(value)) {
globalZoomVal.textContent = `${value}%`;
document.querySelector("#Zoom").value = value.toString();
} else {
globalZoomVal.textContent = value;
}
applyZoom(value);
};
}
function changeZoomByStep(sign = 1) {
return () => {
const globalZoom = document.querySelector("#Zoom");
const ratio =
parseInt(globalZoom.value, 10) + sign * getUserSettings().zoomStep;
globalZoom.value = ratio.toString();
globalZoom?.dispatchEvent(new Event("input", { bubbles: true }));
};
}
function changeDefaultZoomMode(event) {
const target = event.currentTarget.value;
updateSettings({ zoomMode: target });
changeGlobalZoom(target)();
const percent = document.querySelector(".DefaultZoom");
if (getUserSettings().zoomMode === "percent") {
percent?.classList.add("show");
} else {
percent?.classList.remove("show");
}
}
function changeDefaultZoom(event) {
const target = parseInt(event.currentTarget.value, 10);
updateSettings({ defaultZoom: target });
changeGlobalZoom(target)();
}
function changeZoom(event) {
const target = parseInt(event.currentTarget.value, 10);
changeGlobalZoom(target)();
document.querySelector("#ZoomVal").textContent = `${target}%`;
}
function zoom() {
document
.querySelector("#DefaultZoomMode")
?.addEventListener("change", changeDefaultZoomMode);
document
.querySelector("#DefaultZoom")
?.addEventListener("input", changeDefaultZoom);
document.querySelector("#Zoom")?.addEventListener("input", changeZoom);
document
.querySelector("#enlarge")
?.addEventListener("click", changeZoomByStep());
document
.querySelector("#reduce")
?.addEventListener("click", changeZoomByStep(-1));
document
.querySelector("#restore")
?.addEventListener("click", changeGlobalZoom(100));
document
.querySelector("#fitWidth")
?.addEventListener("click", changeGlobalZoom("width"));
document
.querySelector("#fitHeight")
?.addEventListener("click", changeGlobalZoom("height"));
}
function setupFluid(mode) {
const chapter = document.querySelector("#Chapter");
document.querySelector("#Header").className = "click";
document.querySelector("#menu").className = "click";
changeGlobalZoom("height")();
scrollToElement(chapter);
chapter?.addEventListener(
"wheel",
mode === "FluidLTR"
? transformScrollToHorizontal
: transformScrollToHorizontalReverse,
);
}
function updateViewMode(mode) {
return () => {
const chapter = document.querySelector("#Chapter");
const header = document.querySelector("#Header");
["Vertical", "WebComic", "FluidLTR", "FluidRTL"].forEach((c) =>
chapter?.classList.remove(c),
);
chapter?.classList.add(mode);
header.className = "";
chapter?.removeEventListener("wheel", transformScrollToHorizontal);
chapter?.removeEventListener("wheel", transformScrollToHorizontalReverse);
if (mode === "FluidLTR" || mode === "FluidRTL") {
setupFluid(mode);
} else {
document.querySelector("#Header").className = getUserSettings().header;
document.querySelector("#menu").className = getUserSettings().header;
changeGlobalZoom(100)();
}
};
}
function changeViewMode(event) {
const mode = event.currentTarget.value;
updateViewMode(mode)();
updateSettings({ viewMode: mode });
}
function viewMode() {
document
.querySelector("#viewMode")
?.addEventListener("change", changeViewMode);
document
.querySelector("#webComic")
?.addEventListener("click", updateViewMode("WebComic"));
document
.querySelector("#ltrMode")
?.addEventListener("click", updateViewMode("FluidLTR"));
document
.querySelector("#rtlMode")
?.addEventListener("click", updateViewMode("FluidRTL"));
document
.querySelector("#verticalMode")
?.addEventListener("click", updateViewMode("Vertical"));
if (
getUserSettings().viewMode === "FluidLTR" ||
getUserSettings().viewMode === "FluidRTL"
) {
setupFluid(getUserSettings().viewMode);
}
}
let scrollInterval;
function scroll() {
const chapter = document.querySelector("#Chapter");
if (
chapter?.classList.contains("FluidLTR") ||
chapter?.classList.contains("FluidRTL")
) {
chapter?.scrollBy({
top: 0,
left:
getUserSettings().scrollHeight *
(chapter?.classList.contains("FluidRTL") ? -1 : 1),
behavior: "smooth",
});
} else {
window.scrollBy({
top: getUserSettings().scrollHeight,
left: 0,
behavior: "smooth",
});
}
if (document.querySelector("#Header")?.classList.contains("headroom-end")) {
clearInterval(scrollInterval);
scrollInterval = void 0;
document.querySelector("#ScrollControl")?.classList.remove("running");
logScript("Finished auto scroll");
}
}
function toggleAutoScroll() {
const control = document.querySelector("#AutoScroll");
if (scrollInterval) {
clearInterval(scrollInterval);
scrollInterval = void 0;
control?.classList.remove("running");
logScript("Stopped auto scroll");
} else {
scroll();
scrollInterval = setInterval(scroll, 1e3 / 60);
control?.classList.add("running");
logScript("Start auto scroll");
}
}
let resume = false;
const debounceAutoScroll = _.debounce(() => {
toggleAutoScroll();
resume = false;
}, 500);
function manualScroll() {
if (!resume && scrollInterval) {
toggleAutoScroll();
resume = true;
}
if (resume && !scrollInterval) {
debounceAutoScroll();
}
}
function autoscroll() {
window.addEventListener("wheel", _.throttle(manualScroll, 500));
document
.querySelector("#AutoScroll")
?.addEventListener("click", toggleAutoScroll);
}
let setupEvents = false;
function events() {
if (!setupEvents) {
headroom(100);
keybindings();
individual();
size();
setupEvents = true;
}
bookmarks();
globals();
navigation();
options();
panels();
theming();
viewMode();
zoom();
autoscroll();
}
let loadedManga;
function hydrateApp() {
const elements = {
"#Header": Header(loadedManga),
"#CommentsPanel": commentsPanel(loadedManga),
"#SettingsPanel": SettingsPanel(),
"#KeybindingsPanel": KeybindingsPanel(),
"#Bookmarks": BookmarkPanel(),
};
if (
document.querySelector("#ScrollControl")?.classList.contains("running")
) {
toggleAutoScroll();
}
Object.entries(elements).forEach(([id, component]) => {
const tag = document.querySelector(id);
if (tag) {
tag.outerHTML = component;
}
});
document.querySelector("#Overlay")?.dispatchEvent(new Event("click"));
events();
}
const app = (manga) => {
loadedManga = manga;
return html`
${Header(manga)} ${Reader(manga)} ${ThumbnailsPanel(manga)}
${commentsPanel(manga)} ${KeybindingsPanel()} ${BookmarkPanel()}
${SettingsPanel()}
`;
};
function display(manga) {
document.head.innerHTML = head(manga);
document.body.innerHTML = app(manga);
events();
loadManga(manga);
document
.querySelector("#MangaOnlineViewer")
?.addEventListener("locale", hydrateApp);
if (manga.comments)
document.querySelector("#CommentsArea")?.append(manga.comments);
}
const cleanUpElement = (...ele) =>
ele.forEach((element) => {
element.getAttributeNames().forEach((attr) => {
element.removeAttribute(attr);
});
});
async function viewer(manga) {
if (manga.before !== void 0) {
await manga.before(manga.begin);
}
manga.comments = document.querySelector("#disqus_thread, #fb-comments");
logScript("Rebuilding Site");
setTimeout(() => {
try {
cleanUpElement(document.documentElement, document.head, document.body);
window.scrollTo(0, 0);
display(manga);
} catch (e) {
logScript(e);
} finally {
clearBookmark();
}
}, 50);
}
const startButton =
"#StartMOV {\n font-size: 2em;\n color: #fff;\n cursor: pointer;\n margin: 10px auto;\n padding: 10px 20px;\n text-align: center;\n border: none;\n background-size: 300% 100%;\n border-radius: 50px;\n transition: all 0.4s ease-in-out;\n background-image: linear-gradient(to right, #667eea, #764ba2, #6b8dd6, #8e37d7);\n box-shadow: 0 4px 15px 0 rgba(116, 79, 168, 0.75);\n position: fixed;\n right: 0;\n left: 0;\n bottom: 0;\n z-index: 105000;\n height: 3em;\n min-height: 50px;\n width: 80%;\n}\n\n#StartMOV:hover {\n background-position: 100% 0;\n transition: all 0.4s ease-in-out;\n}\n\n#StartMOV:focus {\n outline: none;\n}\n";
function waitForElm(selector, target = document.body) {
return new Promise((resolve) => {
if (document.querySelector(selector)) {
resolve(document.querySelector(selector));
return;
}
const observer = new MutationObserver(() => {
if (document.querySelector(selector)) {
resolve(document.querySelector(selector));
observer.disconnect();
}
});
observer.observe(target, {
childList: true,
subtree: true,
attributes: true,
});
});
}
function waitForAtb(selector, attribute, target = document.body) {
return new Promise((resolve) => {
if (target.querySelector(selector)?.getAttribute(attribute)) {
resolve(target.querySelector(selector)?.getAttribute(attribute) ?? "");
return;
}
const observer = new MutationObserver(() => {
if (target.querySelector(selector)?.getAttribute(attribute)) {
resolve(
target.querySelector(selector)?.getAttribute(attribute) ?? "",
);
observer.disconnect();
}
});
observer.observe(target, {
childList: true,
subtree: true,
attributes: true,
attributeFilter: [attribute],
});
});
}
function waitForVar(name, target = document.body) {
return new Promise((resolve) => {
if (!isNothing(unsafeWindow[name])) {
resolve(unsafeWindow[name]);
return;
}
const observer = new MutationObserver(() => {
if (!isNothing(unsafeWindow[name])) {
resolve(unsafeWindow[name]);
observer.disconnect();
}
});
observer.observe(target, {
childList: true,
subtree: true,
attributes: true,
});
});
}
async function testAttribute(site) {
if (site.waitAttr !== void 0) {
logScript(
`Waiting for Attribute ${site.waitAttr[1]} of ${site.waitAttr[0]}`,
);
const wait = await waitForAtb(site.waitAttr[0], site.waitAttr[1]);
logScript(
`Found Attribute ${site.waitAttr[1]} of ${site.waitAttr[0]} = ${wait}`,
);
}
}
async function testElement(site) {
if (site.waitEle !== void 0) {
logScript(`Waiting for Element ${site.waitEle}`);
const wait = await waitForElm(site.waitEle);
logScript(`Found Element ${site.waitEle} = `, wait);
}
}
async function testVariable(site) {
if (site.waitVar !== void 0) {
logScript(`Waiting for Variable ${site.waitVar}`);
const wait = await waitForVar(site.waitVar);
logScript(`Found Variable ${site.waitVar} = ${wait}`);
}
}
async function until(predFn) {
const poll = (done) => {
const result = predFn();
if (result) {
done(result);
} else {
setTimeout(() => {
poll(done);
}, 500);
}
};
return new Promise(poll);
}
async function testFunc(site) {
if (site.waitFunc !== void 0) {
logScript(`Waiting to pass Function check ${site.waitFunc}`);
const wait = await until(site.waitFunc);
logScript(`Found Function check ${site.waitFunc} = ${wait}`);
}
}
async function testTime(site) {
if (site.waitTime !== void 0) {
logScript(`Waiting to for ${site.waitTime} milliseconds`);
await new Promise((resolve) => {
setTimeout(resolve, site.waitTime);
});
logScript("Continuing");
}
}
const fileTypes = [
"image/apng",
"image/bmp",
"image/gif",
"image/jpeg",
"image/pjpeg",
"image/png",
"image/svg+xml",
"image/tiff",
"image/webp",
"image/x-icon",
];
const fileImageExt = /.(png|jpg|jpeg|gif|bmp|webp)$/i;
const orderFiles = (a, b) =>
a.localeCompare(b, navigator.languages[0] || navigator.language, {
numeric: true,
ignorePunctuation: true,
});
function validFileType(file) {
return fileTypes.includes(file.type);
}
const getImageBlob = (content) => {
const buffer = new Uint8Array(content);
const blob = new Blob([buffer.buffer]);
return URL.createObjectURL(blob);
};
async function loadZipFile(filePath) {
const zip = await JSZip.loadAsync(filePath);
const files = zip
.filter((_, file) => !file.dir && fileImageExt.test(file.name))
.sort((a, b) => orderFiles(a.name, b.name));
logScript("Files in zip:", zip.files);
return Promise.all(
files.map((file) => file.async("arraybuffer").then(getImageBlob)),
);
}
function displayUploadedFiles(title, listImages) {
viewer({
title,
series: "?reload",
pages: listImages.length,
begin: 1,
prev: "#",
next: "#",
listImages,
}).then(() => logScript("Page loaded"));
}
async function loadMangaFromZip(zipFile) {
const listImages = await loadZipFile(zipFile);
displayUploadedFiles(
typeof zipFile === "string" ? zipFile : zipFile.name,
listImages,
);
}
function openFileImages(evt) {
const input = evt.target;
const files = Array.from(input.files)
.filter(validFileType)
.sort((a, b) =>
orderFiles(
a.webkitRelativePath || a.name,
b.webkitRelativePath || b.name,
),
);
logScript(
"Local Files: ",
files,
files.map((f) => f.webkitRelativePath || f.name),
);
if (input.files?.[0]) {
displayUploadedFiles(
input.files[0].webkitRelativePath.split("/")[0] || "Local Images",
files.map(URL.createObjectURL),
);
}
}
function allowUpload() {
if (localhost.url.test(window.location.href)) {
if (document.querySelector("#MangaOnlineViewer, #LocalTest")) {
document
.querySelector("#LocalTest")
?.setAttribute("style", "display:none");
document.querySelector("#file")?.addEventListener("change", (evt) => {
const input = evt.target;
if (input.files?.[0]) loadMangaFromZip(input.files[0]);
});
document
.querySelector("#folder")
?.addEventListener("change", openFileImages);
document
.querySelector("#images")
?.addEventListener("change", openFileImages);
logScript(`Waiting for zip/images upload`);
}
return true;
}
return false;
}
function Unlock() {
try {
const originalAttachShadow = Element.prototype.attachShadow;
Element.prototype.attachShadow = function customAttachShadow() {
return originalAttachShadow.apply(this, [{ mode: "open" }]);
};
} catch (e) {
logScript("Failed to unlock Closed Shadow DOM", e);
}
}
function validateMin(valBegin, endPage, rs) {
let val = valBegin;
if (Number.isNaN(val) || val < rs.min()) {
val = rs.min();
} else if (val > rs.max()) {
val = rs.max();
} else if (val > endPage) {
val = endPage;
}
return val;
}
function validateMax(valEnd, beginPage, rs) {
let val = valEnd;
if (Number.isNaN(val) || val > rs.max()) {
val = rs.max();
} else if (val < rs.min()) {
val = rs.min();
} else if (val < beginPage) {
val = beginPage;
}
return val;
}
async function lateStart(site, begin = 1) {
const manga = await site.run();
logScript("LateStart");
let beginPage = begin;
let endPage = manga.pages;
const options = {
title: getLocaleString("STARTING"),
// Language=html
html: html`
${getLocaleString("CHOOSE_BEGINNING")}
`,
showCancelButton: true,
cancelButtonColor: "#d33",
reverseButtons: true,
icon: "question",
didOpen() {
const pageBeginInput = document.querySelector("#pageBegin");
const pageEndInput = document.querySelector("#pageEnd");
const rangeSliderElement = rangeSlider(
document.getElementById("pagesSlider"),
{
min: 1,
max: manga.pages,
value: [beginPage, endPage],
onInput(value, userInteraction) {
if (userInteraction) {
[beginPage, endPage] = value;
if (pageBeginInput) {
pageBeginInput.value = beginPage.toString();
}
if (pageEndInput) {
pageEndInput.value = endPage.toString();
}
}
},
},
);
function changedInput() {
if (pageBeginInput.value === "" || pageEndInput.value === "") {
return;
}
const valBegin = validateMin(
parseInt(pageBeginInput.value, 10),
endPage,
rangeSliderElement,
);
const valEnd = validateMax(
parseInt(pageEndInput.value, 10),
beginPage,
rangeSliderElement,
);
pageBeginInput.value = valBegin.toString();
pageEndInput.value = valEnd.toString();
beginPage = valBegin;
endPage = valEnd;
rangeSliderElement.value([valBegin, valEnd]);
}
const observerEvent = _.debounce(changedInput, 600);
["change", "mouseup", "keyup", "touchend"].forEach((event) => {
pageBeginInput?.addEventListener(event, observerEvent);
pageEndInput?.addEventListener(event, observerEvent);
});
},
};
Swal.fire(options).then((result) => {
if (result.value) {
logScript(`Choice: ${beginPage} - ${endPage}`);
manga.begin = beginPage;
manga.pages = endPage;
viewer(manga).then(() => logScript("Page loaded"));
} else {
logScript(result.dismiss);
}
});
}
function createLateStartButton(site, beginning) {
const button = document.createElement("button");
button.innerText = getLocaleString("BUTTON_START");
button.id = "StartMOV";
button.onclick = () => {
lateStart(site, beginning).catch(logScript);
};
document.body.appendChild(button);
const style = document.createElement("style");
style.appendChild(document.createTextNode(startButton + rangeSliderStyles));
document.head.appendChild(style);
logScript("Start Button added to page", button);
}
function showWaitPopup(site, manga) {
Swal.fire({
title: getLocaleString("STARTING"),
html: html`${manga.begin > 1
? `${getLocaleString("RESUME")}${manga.begin}. `
: ""}${getLocaleString("WAITING")}`,
showCancelButton: true,
cancelButtonColor: "#d33",
reverseButtons: true,
timer: 3e3,
}).then((result) => {
if (result.value || result.dismiss === Swal.DismissReason.timer) {
viewer(manga).then(() => logScript("Page loaded"));
} else {
createLateStartButton(site, manga.begin);
logScript(result.dismiss);
}
});
}
async function preparePage(site) {
const manga = await site.run();
logScript(`Found Pages: ${manga.pages}`);
if (manga.pages <= 0) {
return;
}
if (!manga.title) {
manga.title = document.querySelector("title")?.textContent?.trim();
}
manga.begin = isBookmarked() ?? manga.begin ?? 1;
const style = document.createElement("style");
style.appendChild(document.createTextNode(sweetalertStyle));
document.body.appendChild(style);
unsafeWindow.MOV = (startPage, endPage) => {
if (startPage !== void 0) {
manga.begin = startPage;
}
if (endPage !== void 0) {
manga.pages = endPage;
}
viewer(manga).then(() => logScript("Page loaded"));
};
switch (site.start ?? getUserSettings()?.loadMode) {
case "never":
createLateStartButton(site, manga.begin);
break;
case "always":
viewer(manga).then(() => logScript("Page loaded"));
break;
case "wait":
default:
showWaitPopup(site, manga);
break;
}
}
async function start(sites) {
Unlock();
logScript(
`Starting ${getInfoGM.script.name} ${
getInfoGM.script.version
} on ${getBrowser()} with ${getEngine()}`,
);
if (allowUpload()) return;
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}`);
await Promise.all([
testTime(site),
testElement(site),
testAttribute(site),
testVariable(site),
testFunc(site),
]).then(async () => preparePage(site).catch(logScript));
} else {
logScript("Sorry, didnt find any valid site");
}
}
start(sites).catch(logScript);
})();