// ==UserScript== // @name Manga OnlineViewer // @author Tago // @namespace https://github.com/TagoDR // @description Shows all pages at once in online view for these sites: Batoto, ComiCastle, ReadComicsOnline, Dynasty-Scans, EatManga, Easy Going Scans, FoOlSlide, KissManga, MangaDoom, MangaFox, MangaGo, MangaHere, MangaInn, MangaLyght, MangaPark, MangaReader,MangaPanda, MangaStream, MangaTown, NineManga, ReadManga Today, SenManga(Raw), TenManga, TheSpectrum, MangaDeep, Funmanga, UnionMangas, MangaHost, Hoc Vien Truyen Tranh, JaiminisBox, MangaDex, HatigarmScans, MangaRock, MangaNelo, LHTranslation, JapScan.To // @version 13.76.0 // @license MIT // @date 2019-07-07 // @grant GM_getValue // @grant GM_setValue // @grant GM_listValues // @grant GM_deleteValue // @grant GM_xmlhttpRequest // @connect * // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jquery-scrollTo/2.1.2/jquery.scrollTo.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.1.5/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/6.11.5/sweetalert2.all.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jscolor/2.0.4/jscolor.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/color-scheme/1.0.0/color-scheme.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/ramda/0.24.1/ramda.min.js // @include /https?:\/\/(www.)?bato.to\/chapter.*/ // @include /https?:\/\/(www.)?comicastle.org\/comic\/.+\/[0-9]+.*/ // @include /https?:\/\/(www.)?readcomicsonline.ru\/comic\/.*\/[0-9]*/ // @include /https?:\/\/(www.)?dynasty-scans.com\/chapters\/.+/ // @include /https?:\/\/(www.)?eatmanga.me\/Manga-Scan\/.+\/.+\// // @include /https?:\/\/read.egscans.com\/.+/ // @include /^(?!.*jaiminisbox).*\/read\/.+/ // @include /https?:\/\/(www.)?kissmanga.com\/Manga\/.+\/.+?id=[0-9]+/ // @include /https?:\/\/(www.)?mngdoom.com\/.+\/[0-9]+/ // @include /https?:\/\/(www.)?fanfox.net\/manga\/.+\/.+\// // @include /https?:\/\/(www.)?mangago.me\/read-manga\/.+\/.+/ // @include /https?:\/\/(www.)?mangahere.cc\/manga\/.+\/.+/ // @include /https?:\/\/(www.)?mangainn.net\/.+\/[0-9]+(\/[0-9]*)?/ // @include /https?:\/\/manga.lyght.net\/series\/.+\.html/ // @include /https?:\/\/(www.)?mangapark.(com|me|org|net)\/(manga|chapter)\/.+\/.+/ // @include /https?:\/\/(www.)?(mangareader|mangapanda)(.net|.com)\/.+\/.+/ // @include /https?:\/\/(www.)?(mangastream|readms)(.net|.com)\/r.*\/.+/ // @include /https?:\/\/(www.)?mangatown.com\/manga\/.+\/.+/ // @include /https?:\/\/(www.)?ninemanga.com\/chapter\/.+\/.+\.html/ // @include /https?:\/\/(www.)?readmng.com\/.+\/[0-9.]+(\/[0-9]*)?/ // @include /https?:\/\/raw.senmanga.com\/.+\/.+\/?/ // @include /https?:\/\/(www.)?tenmanga.com\/chapter\/.+/ // @include /https?:\/\/view.thespectrum.net\/.+/ // @include /https?:\/\/(www.)?(mangadeep).com\/.+\/[0-9]+/ // @include /https?:\/\/(www.)?funmanga.com\/.+\/[0-9]+/ // @include /https?:\/\/(www.)?unionmangas.net\/leitor\/.+\/.+/ // @include /https?:\/\/(www.)?mangahost.net\/manga\/.+\/.+/ // @include /https?:\/\/(www.)?hocvientruyentranh.com\/chapter\/.+\/.+/ // @include /https?:\/\/(www.)?jaiminisbox.com\/reader\/read\/.+/ // @include /https?:\/\/(www.)?mangadex.org\/chapter\/.+(\/.+)?/ // @include /https?:\/\/(www.)?hatigarmscans.net\/manga\/.+\/.+(\/[0-9]*)?/ // @include /https?:\/\/(www.)?mangarock.com\/manga\/.+\/chapter\/.+/ // @include /https?:\/\/(www.)?manganelo.com\/chapter\/.+\/.+/ // @include /https?:\/\/(www.)?lhtranslation.net\/read.+/ // @include /https?:\/\/(www.)?japscan.to\/lecture-en-ligne\/.+\/.+/ // @exclude /https?:\/\/(www.)?tsumino.com\/.+/ // @exclude /https?:\/\/(www.)?pururin.io\/.+/ // @exclude /https?:\/\/hentai.cafe\/.+/ // @downloadURL none // ==/UserScript== (function() { 'use strict'; var W = (typeof unsafeWindow === undefined) ? window : unsafeWindow; function logScript(...text) { console.log('MangaOnlineViewer:', ...text); return text; } const logScriptC = R.curry((x, y) => logScript(x, y)[1]); const removeValueGM = GM_deleteValue || (name => logScript('Removing: ', name)); const getInfoGM = GM_info || { scriptHandler: 'Console', script: { name: 'Debug', version: 'Testing' } }; const getValueGM = GM_getValue || ((name, defaultValue = null) => logScript('Getting: ', name, '=', defaultValue)[3]); const setValueGM = GM_setValue || ((name, value) => logScript('Getting: ', name, '=', value)); function getBrowser() { const ua = navigator.userAgent; let tem; let M = ua.match(/(opera|chrome|safari|firefox|msie|trident(?=\/))\/?\s*(\d+)/i) || []; if (/trident/i.test(M[1])) { tem = /\brv[ :]+(\d+)/g.exec(ua) || []; return 'IE ' + String(tem[1] || ''); } if (M[1] === 'Chrome') { tem = ua.match(/\b(OPR|Edge)\/(\d+)/); if (tem !== null) { return tem.slice(1).join(' ').replace('OPR', 'Opera'); } } M = M[2] ? [M[1], M[2]] : [navigator.appName, navigator.appVersion, '-?']; tem = ua.match(/version\/(\d+)/i); if (tem !== null) { M.splice(1, 1, tem[1]); } return M.join(' '); } function getEngine() { return String(getInfoGM.scriptHandler || 'Greasemonkey') + ' ' + String(getInfoGM.version); } if (typeof getValueGM('MangaFitWidthIfOversized') === 'string') { setValueGM('MangaFitWidthIfOversized', true); setValueGM('MangaShowThumbnails', true); setValueGM('MangaDownloadZip', false); setValueGM('MangaAlwaysLoad', false); } if (typeof getValueGM('MangaZoom') === 'string') { setValueGM('MangaTimer', 1000); setValueGM('MangaZoom', 100); } removeValueGM('MangaAlwaysWebComic'); removeValueGM('MangaTheme:'); const settings = { Theme: getValueGM('MangaTheme', 'Light'), CustomTheme: getValueGM('MangaCustomTheme', '3d0099'), FitWidthIfOversized: getValueGM('MangaFitWidthIfOversized', true), ShowThumbnails: getValueGM('MangaShowThumbnails', true), DownloadZip: getValueGM('MangaDownloadZip', false), Timer: getValueGM('MangaTimer', 1000), Zoom: getValueGM('MangaZoom', 100), alwaysLoad: getValueGM('MangaAlwaysLoad', false), viewMode: getValueGM('MangaViewMode', ''), bookmarks: JSON.parse(getValueGM('MangaBookmarks', '[]')) }; const bookmarkTimeLimit = 1000 * 60 * 60 * 24 * 30 * 12; settings.bookmarks = settings.bookmarks.filter(el => Date.now() - el.date < bookmarkTimeLimit); setValueGM('MangaBookmarks', JSON.stringify(settings.bookmarks)); const icon = { enlage: '', reduce: '%3D%3D', restore: '%3D', fitwidth: '', reload: '%3D%3D', zoomin: '%3D%3D', zoomout: '', zoomrestore: '%3D', zoomwidth: '%3D', hide: '%3D', settings: '%3D', menu: '%3D', webcomic: '%3D', bookmark: '', pictureright: '', picturedown: '', pictureleft: '' }; const scheme = new ColorScheme().scheme('mono').variation('default'); function addTheme(theme) { return ''; } function addCustomTheme(color) { const bg = scheme.from_hex(color).colors(); return addTheme(['Custom_Dark', '#000000', '#' + String(bg[2]), '#' + String(bg[3]), '#' + String(bg[0]), '#' + String(bg[1])]) + addTheme(['Custom_Light', '#eeeeec', '#' + String(bg[3]), '#' + String(bg[2]), '#' + String(bg[0]), '#' + String(bg[1])]); } function loadThemes() { const bg = scheme.from_hex(settings.CustomTheme).colors(); return [ ['Dark', '#000000', '#ffffff', '#666666', '#333333', '#282828'], ['Light', '#eeeeec', '#2e3436', '#888a85', '#babdb6', '#c8cec2'], ['Clear', '#ffffff', '#2e3436', '#888a85', '#eeeeec', '#d3d7cf'], ['Dark_Blue', '#000000', '#91a0b0', '#586980', '#3e4b5b', '#222c3b'], ['Tango_Blue', '#000000', '#82a0bf', '#3d669b', '#304c77', '#102747'], ['Lime', '#000000', '#8abd59', '#608d34', '#38531f', '#233413'], ['Plum', '#000000', '#ad7fa8', '#75507b', '#49324d', '#311b37'], ['Light_Plum', '#eeeeec', '#5c3566', '#9b71a2', '#ad7fa8', '#d2b8ce'], ['Earthy', '#000000', '#ffffff', '#693d3d', '#46211a', '#683327'], ['Cool_Blues', '#000000', '#c4dfe6', '#66a5ad', '#07575b', '#003b46'], ['Custom_Dark', '#000000', '#' + String(bg[2]), '#' + String(bg[3]), '#' + String(bg[0]), '#' + String(bg[1])], ['Custom_Light', '#eeeeec', '#' + String(bg[3]), '#' + String(bg[2]), '#' + String(bg[0]), '#' + String(bg[1])] ]; } const themes = loadThemes(); const themesSelector = R.map(theme => '', themes); const themesCSS = R.map(theme => addTheme(theme), themes).join(''); const painel = '\n
\n
\n \n
\n
\n \n \n \n \n \n \n \n \n \n
\n
Zoom: ' + String(settings.Zoom) + ' %
\n
'; const shortcuts = '\n
\n + or = : Global Zoom in pages (enlarge)
\n - : Global Zoom out pages (reduce)
\n * or 8 : Global Restore pages to original
\n 5 : Global Fit window width
\n Arrow Right or . : Next Chapter
\n Arrow Left or , : Previous Chapter
\n
'; const controls = '\n
\n Theme:\n \n \n \n Pages/Second:\n \n \n Default Zoom:\n \n \n Fit Width if Oversized:\n \n \n Show Thumbnails:\n \n \n Default View Mode:\n \n \n Download Images as Zip Automatically:\n \n \n Always Load Script:\n \n \n
'; const chapterControl = R.curry((id, target, manga) => '\n
\n Bottom\n Download\n Previous\n Next\n
'); const chapterControlTop = chapterControl('ChapterControlTop', 'ChapterControlBottom'); const chapterControlBottom = chapterControl('ChapterControlBottom', 'MangaOnlineViewer'); const title = manga => '

' + String(manga.title) + '
(Return to Chapter List)
'; const listPages = R.times(index => '\n
\n
\n \n \n \n \n \n \n \n ' + String(index + 1) + '\n
\n
\n \'PageImg'\n
\n
'); const listOptions = R.times(index => ''); const listThumbnails = R.times(index => '
\'ThumbnailImg'' + String(index + 1) + '
'); const body = (manga, begin = 0) => '\n
\n ' + String(title(manga)) + '\n ' + String(chapterControlTop(manga)) + '\n
\n ' + String(listPages(manga.quant).slice(begin).join('')) + ' \n
\n ' + String(title(manga)) + '\n ' + String(chapterControlBottom(manga)) + '\n ' + painel + ' \n ' + controls + '\n ' + shortcuts + ' \n
\n 0 of ' + String(manga.quant) + ' Pages Loaded \n Go to Page:\n \n
\n
\n
\n 0 of ' + String(manga.quant) + ' Pages Loaded\n
\n
\n ' + String(listThumbnails(manga.quant).slice(begin).join('')) + '\n
\n
\n Download\n
'; const readerCSS = '\n'; const externalScripts = ['', '', '', '', '', '', '', '']; const externalCSS = ['', '', '']; function reader(manga, begin = 0) { return '\n\n ' + String(manga.title) + '\n \n ' + String(externalScripts.join('\n')) + '\n ' + String(externalCSS.join('\n')) + '\n ' + readerCSS + '\n ' + String(themesCSS) + '\n\n\n ' + String(body(manga, begin > 0 ? begin - 1 : 0)) + '\n'; } const isEmpty = R.either(R.either(R.isNil, R.isEmpty), R.either(x => R.length(x) === 0, x => x === 0)); const mapIndexed = R.addIndex(R.map); function normalizeUrl(url) { let uri = url.trim(); if (uri.startsWith('//')) { uri = 'https:' + String(uri); } return uri; } function addImg(index, src) { const url = normalizeUrl(src); logScript('Image:', index, 'Source:', url); $('#PageImg' + String(index)).attr('src', url).parent().slideToggle(); $('#ThumbnailImg' + String(index)).attr('src', url); return index; } function addImgAlt(index, altsrc) { const url = normalizeUrl(altsrc); logScript('Image:', index, 'Alternative Source:', url); if (altsrc !== '') { $('#PageImg' + String(index)).attr('altsrc', url); $('#ThumbnailImg' + String(index)).attr('onerror', 'this.src=\'' + String(url) + '\';this.onerror=null;'); } return index; } function getPage(url, wait = settings.Timer) { return new Promise(resolve => { setTimeout(() => { logScript('Getting page: ' + String(url)); $.ajax({ type: 'GET', url, dataType: 'html', async: true, success: html => resolve(html), retryCount: 0, retryLimit: 10, retryTimeout: 10000, timeout: 1000, created: Date.now(), error() { this.retryCount += 1; if (this.retryCount <= this.retryLimit && Date.now() - this.created < this.retryTimeout) { logScript('Retrying Getting page: ' + String(url)); $.ajax(this); } else { logScript('Failed Getting page: ' + String(url)); } } }); }, wait); }); } const loadMangaPages = (begin, manga) => mapIndexed((url, index) => index >= begin ? getPage(url, (manga.timer || settings.Timer) * (index - begin)).then(response => addImg(index + 1, $(response).find(manga.img).attr('src'))) : null, manga.listPages); function getImages(src, wait = settings.Timer) { return new Promise(resolve => { setTimeout(() => { resolve(src); }, wait); }); } const loadMangaImages = (begin, manga) => mapIndexed((src, index) => index >= begin ? getImages(src, (manga.timer || settings.Timer) * (index - begin)).then(response => addImg(index + 1, response)) : null, manga.listImages); const loadMangaImagesAlt = (begin, manga) => mapIndexed((src, index) => index >= begin ? addImgAlt(index + 1, src) : null, manga.listImagesAlt); function loadManga(manga, begin = 1) { logScript('Loading Images'); logScript('Intervals: ' + String(manga.timer || settings.Timer || 'Default(1000)')); if (manga.listPages !== undefined) { logScript('Method: Pages:', manga.listPages); loadMangaPages(begin - 1, manga); } else if (manga.listImages !== undefined) { logScript('Method: Images:', manga.listImages); loadMangaImages(begin - 1, manga); if (manga.listImagesAlt !== undefined) { loadMangaImagesAlt(begin - 1, manga); } } else { logScript('Method: Brute Force'); manga.bruteForce({ begin, addImg, loadMangaImages: R.curry(loadMangaImages)(begin - 1), loadMangaPages: R.curry(loadMangaPages)(begin - 1), getPage, getImages, wait: settings.timer }); } } function reloadImage(img) { const src = img.attr('src'); const altsrc = img.attr('altsrc'); if (src !== undefined) { if (altsrc !== undefined) { img.removeAttr('src'); img.removeAttr('altsrc'); setTimeout(() => { img.attr('src', altsrc); img.attr('altsrc', src); }, 500); } else { img.removeAttr('src'); setTimeout(() => { img.attr('src', src); }, 500); } } } function applyZoom(page, newZoom) { const zoom = newZoom || settings.Zoom; const pages = page || '.PageContent img'; $(pages).each((index, value) => $(value).width(zoom === 1000 ? $('html').width() : $(value).prop('naturalWidth') * (zoom / 100))); } function checkImagesLoaded(manga) { const images = $('.PageContent img').get(); const total = images.length; const missing = images.filter(item => $(item).prop('naturalWidth') === 0); const loaded = images.filter(item => $(item).prop('naturalWidth') !== 0); loaded.filter(item => $(item).attr('width') === undefined).forEach(item => applyZoom($(item))); missing.forEach(item => reloadImage($(item))); NProgress.configure({ showSpinner: false }).set(loaded.length / total); $('#Counters i, #NavigationCounters i').html(loaded.length); logScript('Progress: ' + String(Math.floor(loaded.length / total * 100)) + '%'); $('title').html('(' + String(Math.floor(loaded.length / total * 100)) + '%) ' + String(manga.title)); if (loaded.length < total) { setTimeout(() => checkImagesLoaded(manga), 5000); } else { logScript('Images Loading Complete'); settings.bookmarks = settings.bookmarks.filter(el => el.url !== location.href); setValueGM('MangaBookmarks', JSON.stringify(settings.bookmarks)); $('.download').attr('href', '#download'); logScript('Download Avaliable'); if (settings.DownloadZip) { $('#blob').click(); } } } const cache = { zip: new JSZip(), downloadFiles: 0, Data: {} }; function generateZip() { if (cache.downloadFiles === 0) { $('.MangaPage img').get().forEach((value, index) => { const img = $(value); const src = img.attr('src'); const ext = src.substring(0, 20).match(/jpg|png|webp/ig) || ['png']; const filename = 'Page ' + String(String('000' + String(index + 1)).slice(-3)) + '.' + String(ext[0]); if (src.indexOf('base64') > -1) { let base64 = src.replace('data:image/png;base64,', ''); const i = base64.indexOf(','); if (i !== -1) { base64 = base64.substring(i + 1, base64.length); } cache.zip.file(filename, base64, { base64: true, createFolders: true }); logScript(filename + ' Added to Zip from Base64 Image, From: ' + String(src)); cache.downloadFiles += 1; } else { try { GM_xmlhttpRequest({ method: 'GET', url: src, overrideMimeType: 'text/plain; charset=x-user-defined', responseType: 'blob', onload(request) { cache.zip.file(filename, request.response, { base64: true, createFolders: true, compression: 'DEFLATE' }); logScript(filename + ' Added to Zip as Base64 Image, From: ' + String(src) + ', Data:', request.response); cache.downloadFiles += 1; } }); } catch (e) { logScript(e); } } }); } const total = parseInt($('#Counters').find('b').text(), 10); if (cache.downloadFiles < total) { logScript('Waiting for Files to Download ' + String(cache.downloadFiles) + ' of ' + String(total)); setTimeout(generateZip, 2000); } else { const blobLink = document.getElementById('blob'); try { blobLink.download = String($('title').text().trim()) + '.zip'; cache.zip.generateAsync({ type: 'blob' }).then(content => { blobLink.href = W.URL.createObjectURL(content); logScript('Download Ready'); $('#blob')[0].click(); }); } catch (e) { logScript(e); blobLink.innerHTML += ' (not supported on this browser)'; } } } function setKeyDownEvents() { try { $(document).unbind('keyup keydown keypress onload'); $(W).unbind('keyup keydown keypress onload'); document.onkeydown = null; document.onkeypress = null; W.onkeydown = null; W.onkeypress = null; W.onload = null; document.body.onload = null; } catch (e) { logScript('Keybinds error: ' + String(e)); } function processKey(e) { const a = e.keyCode || e.which; if ($.inArray(a, [39, 46, 190, 37, 44, 188, 43, 107, 61, 45, 109, 42, 106, 56, 104, 53, 101]) !== -1) { e.preventDefault(); e.stopPropagation(); e.stopImmediatePropagation(); switch (a) { case 39: case 46: case 190: $('.ChapterControl:first .next')[0].click(); break; case 37: case 44: case 188: $('.ChapterControl:first .prev')[0].click(); break; case 43: case 107: case 61: $('#enlarge').click(); break; case 45: case 109: $('#reduce').click(); break; case 42: case 106: case 56: case 104: $('#restore').click(); break; case 53: case 101: $('#fitwidth').click(); break; default: break; } return false; } return true; } if (navigator.userAgent.match(/mozilla/i)) { $(document).keypress(processKey); } else { $(document).keydown(processKey); } } function controls$1() { $('#enlarge').click(() => { settings.Zoom += 25; $('#Zoom b').html(settings.Zoom); applyZoom(); }); $('#reduce').click(() => { settings.Zoom -= 25; $('#Zoom b').html(settings.Zoom); applyZoom(); }); $('#restore').click(() => { settings.Zoom = 100; $('#Zoom b').html(settings.Zoom); $('.PageContent img').removeAttr('width'); $('.PageContent img').removeAttr('style'); }); $('#fitwidth').click(() => { settings.Zoom = 1000; $('#Zoom b').html(settings.Zoom); applyZoom(); }); $('#webcomic').click(() => { $('#Chapter').addClass('WebComic').removeClass('FluidLTR').removeClass('FluidRTL'); }); $('#ltrmode').click(() => { $('#Chapter').removeClass('WebComic').addClass('FluidLTR').removeClass('FluidRTL'); }); $('#rtlmode').click(() => { $('#Chapter').removeClass('WebComic').removeClass('FluidLTR').addClass('FluidRTL'); }); $('#verticalmode').click(() => { $('#Chapter').removeClass('WebComic').removeClass('FluidLTR').removeClass('FluidRTL'); }); $('#fitIfOversized').change(event => { $('#Chapter').toggleClass('fitWidthIfOversized'); if ($(event.target).is(':checked')) { setValueGM('MangaFitWidthIfOversized', true); } else { setValueGM('MangaFitWidthIfOversized', false); } logScript('fitIfOversized: ' + String(getValueGM('MangaFitWidthIfOversized'))); }); $('#viewMode').change(event => { const mode = $(event.target).val(); $('#Chapter').removeClass('WebComic').removeClass('FluidLTR').removeClass('FluidRTL').addClass(mode); setValueGM('MangaViewMode', mode); logScript('ViewMode: ' + String(getValueGM('MangaViewMode'))); }); $('#alwaysLoad').change(event => { if ($(event.target).is(':checked')) { setValueGM('MangaAlwaysLoad', true); } else { setValueGM('MangaAlwaysLoad', false); } logScript('MangaAlwaysLoad: ' + String(getValueGM('MangaAlwaysLoad'))); }); $('#showThumbnails').change(event => { $('#Navigation').toggleClass('disabled'); if ($(event.target).is(':checked')) { setValueGM('MangaShowThumbnails', true); } else { setValueGM('MangaShowThumbnails', false); } logScript('MangaShowThumbnails: ' + String(getValueGM('MangaShowThumbnails'))); }); $('#downloadZip').change(event => { if ($(event.target).is(':checked')) { setValueGM('MangaDownloadZip', true); swal({ title: 'Attention', text: 'Next time a chapter finish loading you will be promted to save automatically', timer: 10000, type: 'info', confirmButtonText: 'OK' }); } else { setValueGM('MangaDownloadZip', false); } logScript('MangaDownloadZip: ' + String(getValueGM('MangaDownloadZip'))); }); $('#blob').one('click', generateZip); $('.download').click(() => { logScript('Downloading Chapter'); $('#blob')[0].click(); }); $('#PagesPerSecond').change(event => { setValueGM('MangaTimer', parseInt($(event.target).val(), 10)); logScript('MangaTimer: ' + String(getValueGM('MangaTimer'))); }); $('#DefaultZoom').change(event => { settings.Zoom = parseInt($(event.target).val(), 10); $('#Zoom b').html(settings.Zoom); setValueGM('MangaZoom', settings.Zoom); logScript('MangaZoom: ' + String(getValueGM('MangaZoom'))); applyZoom(); }); $('#ThemeSelector').change(event => { const target = $(event.target); $('#MangaOnlineViewer , body').removeClass().addClass(target.val()); logScript('MangaTheme', target.val()); setValueGM('MangaTheme', target.val()); logScript('MangaTheme: ' + String(getValueGM('MangaTheme'))); if (target.val() === 'Custom_Dark' || target.val() === 'Custom_Light') { $('#CustomThemeHue').show(); } else { $('#CustomThemeHue').hide(); } }); jscolor(document.getElementById('CustomThemeHue')); $('#CustomThemeHue').change(event => { const target = $(event.target).val(); logScript('CustomTheme: #' + String(target)); $('style[title="Custom_Light"], style[title="Custom_Dark"]').remove(); $('head').append(addCustomTheme(target)); setValueGM('MangaCustomTheme', target); logScript('MangaCustomTheme: ' + String(getValueGM('MangaCustomTheme'))); }); function scrollToElement(ele) { $(W).scrollTop(ele.offset().top).scrollLeft(ele.offset().left); } $('#gotoPage').bind('change', event => { scrollToElement($('#Page' + String($(event.target).val()))); }); $('.Thumbnail').bind('click', event => { scrollToElement($('#Page' + String($(event.currentTarget).find('span').html()))); }); $('#settings').click(() => { $('#ViewerControls').slideToggle(); $('#ViewerShortcuts').slideToggle(); $('#ImageOptions').toggleClass('settingsOpen'); $('#Navigation').toggleClass('visible'); }); $('.Bookmark').click(event => { const num = parseInt($(event.target).parents('.MangaPage').find('.PageFunctions span').text(), 10); const mark = { url: location.href, page: num, date: Date.now() }; const found = settings.bookmarks.filter(el => el.url === mark.url).length > 0; settings.bookmarks = settings.bookmarks.filter(el => el.url !== mark.url); if (found) { swal({ title: 'Bookmark Removed', timer: 10000, type: 'error', confirmButtonText: 'OK' }); } else { settings.bookmarks.push(mark); swal({ title: 'Saved Bookmark', html: 'Next time you open this chapter it will resume from:

Page ' + String(num) + '

(Only ONCE per Bookmark, will be removed after a year unused)', type: 'success', confirmButtonText: 'OK' }); } setValueGM('MangaBookmarks', JSON.stringify(settings.bookmarks)); logScript('MangaBookmarks: ' + String(getValueGM('MangaBookmarks'))); }); $('.Reload').click(event => { reloadImage($(event.target).parents('.MangaPage').find('.PageContent img')); }); $('.ZoomIn').click(event => { const img = $(event.target).parents('.MangaPage').find('.PageContent img'); const ratio = img.width() / img.prop('naturalWidth') * 1.25 * 100; applyZoom(img, ratio); }); $('.ZoomOut').click(event => { const img = $(event.target).parents('.MangaPage').find('.PageContent img'); const ratio = img.width() / img.prop('naturalWidth') * 0.75 * 100; applyZoom(img, ratio); }); $('.ZoomRestore').click(() => { $('.PageContent img').removeAttr('width'); }); $('.ZoomWidth').click(event => { const img = $(event.target).parents('.MangaPage').find('.PageContent img'); applyZoom(img, 1000); }); $('.Hide').click(event => { const img = $(event.target).parents('.MangaPage').find('.PageContent'); img.slideToggle('slow'); }); } function formatPage(manga, begin = 0) { logScript('Found ' + String(manga.quant) + ' pages'); if (manga.quant > 0) { settings.starting = begin || settings.bookmarks .filter(x => x.url === location.href).map(x => x.page)[0] || 0; let cancel = false; if (!settings.alwaysLoad) { $('head').append(''); swal({ title: 'Starting MangaOnlineViewer', text: (settings.starting > 1 ? 'Resuming reading from Page ' + String(settings.starting) + '.\n' : '') + 'Please wait, 3 seconds...', showCancelButton: false, confirmButtonText: 'No, cancel!', confirmButtonColor: '#DD6B55', closeOnConfirm: true }).then(isConfirm => { cancel = isConfirm; W.mov = starting => formatPage(manga, starting); }); } setTimeout(() => { W.stop(); if (cancel) { logScript('Aborted'); return; } if (manga.before !== undefined) { manga.before(); } document.documentElement.innerHTML = reader(manga, settings.starting); setTimeout(() => { try { controls$1(manga); setKeyDownEvents(manga); checkImagesLoaded(manga); logScript('Site rebuild done'); setTimeout(() => { $('body').scrollTo(0); loadManga(manga, settings.starting); }, 50); } catch (e) { logScript(e); } }, 50); if (manga.after !== undefined) { manga.after(); } }, settings.alwaysLoad ? 50 : 3000); } } function start(sites) { logScript('Starting ' + String(getInfoGM.script.name) + ' ' + String(getInfoGM.script.version) + ' on ' + String(getBrowser()) + ' with ' + String(getEngine())); W.InfoGM = getInfoGM; logScript(String(sites.length) + ' Known Manga Sites'); let waitElapsed = 0; function waitExec(site) { let wait = ''; if (site.waitMax !== undefined) { if (waitElapsed >= site.waitMax) { formatPage(site.run()); return; } } if (site.waitAttr !== undefined) { wait = $(site.waitAttr[0]).attr(site.waitAttr[1]); logScript('Wating for ' + String(site.waitAttr[1]) + ' of ' + String(site.waitAttr[0]) + ' = ' + String(wait)); if (wait === undefined || isEmpty(wait)) { setTimeout(() => { waitExec(site); }, site.waitStep || 1000); waitElapsed += site.waitStep || 1000; return; } } if (site.waitEle !== undefined) { wait = $(site.waitEle).get(); logScript('Wating for ' + String(site.waitEle) + ' = ' + ('' + String(wait))); if (wait === undefined || isEmpty(wait)) { setTimeout(() => { waitExec(site); }, site.waitStep || 1000); waitElapsed += site.waitStep || 1000; return; } } if (site.waitVar !== undefined) { wait = W[site.waitVar]; logScript('Wating for ' + String(site.waitVar) + ' = ' + String(wait)); if (isEmpty(wait)) { setTimeout(() => { waitExec(site); }, site.waitStep || 1000); waitElapsed += site.waitStep || 1000; return; } } formatPage(site.run()); } logScript('Looking for a match...'); const test = R.compose(R.map(waitExec), R.map(logScriptC('Site Found:')), R.filter(x => R.test(x.url, location.href))); test(sites); } var batoto = { name: 'Batoto', url: /https?:\/\/(www.)?bato.to\/chapter.*/, homepage: 'http://bato.to/', language: ['English'], category: 'manga', run() { const num = $('#viewer .item').length; return { title: $('.nav-title a').text(), series: $('.nav-title a').attr('href'), quant: num, prev: $('.nav-prev a').attr('href'), next: $('.nav-next a').attr('href'), listImages: $('.page-img').get().map(i => $(i).attr('src')) }; } }; var comicastle = { name: 'ComiCastle', url: /https?:\/\/(www.)?comicastle.org\/comic\/.+\/[0-9]+.*/, homepage: 'http://www.comicastle.org/', language: ['English'], category: 'comic', run() { const url = $('.form-control:last option').get(); const chapter = $('.form-control:first option'); return { title: chapter.find(':selected').text(), series: $('.navbar-header a').attr('href'), quant: url.length, prev: chapter.find(':selected').prev().val(), next: chapter.find(':selected').next().val(), listPages: url.map(item => $(item).val()), img: '.chapter-img' }; } }; var readcomicsonline = { name: 'ReadComicsOnline', url: /https?:\/\/(www.)?readcomicsonline.ru\/comic\/.*\/[0-9]*/, homepage: 'http://readcomicsonline.ru/', language: ['English'], category: 'comic', run() { return { title: W.title.replace(/ - Page [0-9]+/, ''), series: $('div.pager-cnt a:first').attr('href'), quant: W.pages.length, prev: W.prev_chapter, next: W.next_chapter, listImages: $('#all img').get().map(i => $(i).attr('data-src')) }; } }; var dysnatyscans = { name: 'Dynasty-Scans', url: /https?:\/\/(www.)?dynasty-scans.com\/chapters\/.+/, homepage: 'https://dynasty-scans.com/', language: ['English'], category: 'manga', run() { return { title: $('#chapter-title').text(), series: '#', quant: W.pages.length, prev: $('#prev_link').attr('href'), next: $('#next_link').attr('href'), listImages: W.pages.map(x => x.image) }; } }; var eatmanga = { name: 'EatManga', url: /https?:\/\/(www.)?eatmanga.me\/Manga-Scan\/.+\/.+\//, homepage: 'http://eatmanga.me/', language: ['English'], category: 'manga', run() { const chapter = $('#top_chapter_list option:selected'); return { title: $('#main_content h1').text().split(',')[0].trim(), series: $('ul#crumbs li a:eq(2)').attr('href'), quant: $('select#pages option:last').html(), prev: chapter.next().val(), next: chapter.prev().val(), listPages: $('select#pages option').get().map(item => $(item).val()), img: '#eatmanga_image , #eatmanga_image_big' }; } }; var egscans = { name: 'Easy Going Scans', url: /https?:\/\/read.egscans.com\/.+/, homepage: 'http://read.egscans.com/', language: ['English'], category: 'manga', waitVar: 'img_url', run() { const src = W.img_url.slice(1); return { title: $('select[name="manga"] option:selected').text().trim(), series: '#', quant: src.length, prev: '../' + String(W.prev_chap), next: '../' + String(W.next_chap), listImages: src.map(encodeURI).map(x => '../' + String(x)) }; } }; var foolslide = { name: 'FoOlSlide', url: /^(?!.*jaiminisbox).*\/read\/.+/, homepage: '', language: ['English'], obs: 'Any Scanlator site that uses FoOLSlide', category: 'manga', run() { const temp = String(location.href.substr(0, location.href.lastIndexOf('/'))) + '/'; const url = temp.match(/page\/$/) ? temp : temp + 'page/'; const num = $('.topbar_right .dropdown li').length; const chapter = $('.topbar_left .dropdown_parent:last ul li a'); return { title: $('title').text().trim(), series: $('div.tbtitle div.text a:first').attr('href'), quant: num, prev: chapter.eq(chapter.index(chapter.filter('[href*=\'' + String(location.pathname.replace(/page.+/, '')) + '\']')) + 1).attr('href'), next: chapter.eq(chapter.index(chapter.filter('[href*=\'' + String(location.pathname.replace(/page.+/, '')) + '\']')) - 1).attr('href'), listPages: [...Array(num).keys()].map(i => url + (i + 1)), img: 'img.open' }; } }; var kissmanga = { name: 'KissManga', url: /https?:\/\/(www.)?kissmanga.com\/Manga\/.+\/.+?id=[0-9]+/, homepage: 'http://kissmanga.com/', language: ['English'], category: 'manga', run() { const chapter = $('.selectChapter option:selected'); const url = location.href.replace(/[^/]+$/, ''); return { title: $('title').text().replace('Read manga', '').replace('online in high quality', '').trim(), series: $('#navsubbar a').attr('href'), quant: W.lstImages.length, prev: url + chapter.prev().val(), next: url + chapter.next().val(), listImages: W.lstImages }; } }; var mangadoom = { name: 'MangaDoom', url: /https?:\/\/(www.)?mngdoom.com\/.+\/[0-9]+/, homepage: 'https://mngdoom.com/', language: ['English'], category: 'manga', run() { const url = $('.selectPage:first option').get(); const chapter = $('.chapterSelect:first option:selected'); return { title: $('.widget-heading > div > div:first').text().trim(), series: $('.widget-heading a').attr('href'), quant: url.length, prev: chapter.next().val(), next: chapter.prev().val(), listPages: url.map(item => $(item).val()), img: 'img.img-responsive' }; } }; var mangafox = { name: 'MangaFox', url: /https?:\/\/(www.)?fanfox.net\/manga\/.+\/.+\//, homepage: 'http://fanfox.net/', language: ['English'], category: 'manga', run() { function decode(data, page) { const toBeEval = data.match(/'[^']*'/g)[5].replace(/'/g, ''); const keyWords = data.match(/'[^']*'/g)[6].replace(/'/g, '').split('|'); function charFromPosition(i) { return (i < 31 ? '' : charFromPosition(parseInt(i / 31, 10))) + (i % 31 > 35 ? String.fromCharCode(i % 31 + 29) : (i % 31).toString(36)); } const replacingValues = {}; keyWords.forEach((ele, i) => { replacingValues[charFromPosition(i)] = ele || charFromPosition(i); }); const res = toBeEval.replace(new RegExp(/\b\w+\b/, 'g'), y => replacingValues[y]); return res.match(/pix=\"([^;]+)\";/)[1] + res.match(/pvalue=\[\"([^,]+)\",\"([^,\]]+)\"/)[page === 0 ? 1 : 2]; } const src = [...Array(W.imagecount).keys()].map(i => { let img = ''; $.ajax({ url: 'chapterfun.ashx', async: false, data: { cid: W.chapterid, page: i, key: $('#dm5_key').val() } }).done(data => { img = decode(data, i); }); return img; }); return { title: $('.reader-header-title div:first').text().trim(), series: $('.reader-header-title a').attr('href'), quant: W.imagecount, prev: W.prechapterurl, next: W.nextchapterurl, listImages: src }; } }; var mangago = { name: 'MangaGo', url: /https?:\/\/(www.)?mangago.me\/read-manga\/.+\/.+/, homepage: 'http://www.mangago.me/', language: ['English'], category: 'manga', run() { const origin = $('#series'); return { title: origin.text(), series: origin.attr('href'), quant: $('.page a:first').text().replace(/page 1 of /, ''), prev: $('.readtips p:eq(4) a:first').attr('href'), next: $('.readtips p:eq(3) a:first').attr('href'), listPages: $('.page a').get().map(item => $(item).attr('href')), img: '#page1' }; } }; var mangahere = { name: 'MangaHere', url: /https?:\/\/(www.)?mangahere.cc\/manga\/.+\/.+/, homepage: 'http://www.mangahere.cc/', language: ['English'], category: 'manga', run() { const num = $('.right select:first option').length - 1; const chapter = $('.reader_tip a'); return { title: $('.title h1').text(), series: $('div.title h2 a').attr('href'), quant: num, prev: chapter.eq(-1).attr('href'), next: chapter.eq(-2).attr('href'), listPages: [''].concat([...Array(num - 1).keys()].map(i => String(i + 2) + '.html')), img: 'img#image' }; } }; var mangainn = { name: 'MangaInn', url: /https?:\/\/(www.)?mangainn.net\/.+\/[0-9]+(\/[0-9]*)?/, homepage: 'http://www.mangainn.net/', language: ['English'], category: 'manga', run() { return { title: W.chapter_page_title.trim(), series: W.manga_url, quant: W.images.length, prev: W.prev_chapter_url, next: W.next_chapter_url, listImages: W.images.map(i => i.url) }; } }; var mangalyght = { name: 'MangaLyght', url: /https?:\/\/manga.lyght.net\/series\/.+\.html/, homepage: 'http://manga.lyght.net/', language: ['English'], category: 'manga', run() { const chapter = $('.selectchapter option:selected'); const url = String($('form[name=\'pageSelector1\']').attr('action')) + '?ch=' + String(chapter.val().replace(' ', '+')) + '&page='; const num = $('.selectpage option').length; const origin = $('div.entry h1 a'); return { title: origin.text().trim(), series: origin.attr('href'), quant: num, prev: (String(location.pathname) + '?ch=' + String(chapter.prev().val())).replace(' ', '+'), next: (String(location.pathname) + '?ch=' + String(chapter.next().val())).replace(' ', '+'), listPages: [...Array(num).keys()].map(i => url + (i + 1)), img: '#mainimage' }; } }; var mangapark = { name: 'MangaPark', url: /https?:\/\/(www.)?mangapark.(com|me|org|net)\/(manga|chapter)\/.+\/.+/, homepage: 'http://mangapark.net/', language: ['English'], category: 'manga', run() { const img = $('.img-link img').get(); return { title: $('.loc a:first, h4 a').text().trim(), series: $('.loc a:first, h4 a').attr('href'), quant: W.pages || img.length, prev: W._prev_link || $('span:contains(◀ Prev Chapter):first').parent('a').attr('href'), next: W._next_link || $('span:contains(Next Chapter ▶):first').parent('a').attr('href'), listImages: W.images || img.map(i => { if ($(i).hasClass('lazy')) { return $(i).attr('data-src'); } return $(i).attr('src'); }), before() { if (location.href.search(/\/1$/) !== -1) { location.href = location.href.replace('/1', ''); } } }; } }; var mangareader = { name: ['MangaReader', 'MangaPanda'], url: /https?:\/\/(www.)?(mangareader|mangapanda)(.net|.com)\/.+\/.+/, homepage: ['http://www.mangareader.net/', 'http://www.mangapanda.com/'], language: ['English'], category: 'manga', run() { const url = location.href + (location.href.lastIndexOf('/') !== location.href.length - 1 ? '/' : ''); const num = parseInt($('select#pageMenu option:last').html(), 10); const chapter = $('#mangainfo_bas a'); return { title: $('#mangainfo h1').text(), series: $('#mangainfo a').attr('href'), quant: num, prev: chapter.last().attr('href'), next: chapter.first().attr('href'), listPages: [...Array(num).keys()].map(i => url + (i + 1), num), img: 'img#img', before() { if (location.pathname.match(/\/.+\/.+\/chapter-[0-9]+.*/)) { const path = location.pathname.split('/'); location.pathname = '/' + String(path[2]) + '/' + String(path[3].match(/[0-9]+/)); } else if (location.search) { location.href = location.pathname; } } }; } }; var mangastream = { name: 'MangaStream', url: /https?:\/\/(www.)?(mangastream|readms)(.net|.com)\/r.*\/.+/, homepage: 'http://mangastream.com/', language: ['English'], category: 'manga', run() { const url = location.href.substring(0, location.href.lastIndexOf('/') + 1); const num = parseInt($('div.controls div.btn-group ul.dropdown-menu li:last').text().match(/[0-9]+/), 10); const chapter = $('.controls .dropdown-menu:first a'); return { title: $('.btn:eq(0)').text().trim(), series: $('div.controls div.btn-group ul.dropdown-menu:first li a:last').attr('href'), quant: num, prev: chapter.eq(chapter.index(chapter.filter('[href*=\'' + String(location.pathname) + '\']')) + 1).attr('href'), next: chapter.eq(chapter.index(chapter.filter('[href*=\'' + String(location.pathname) + '\']')) - 1).attr('href'), listPages: [...Array(num).keys()].map(i => url + (i + 1)), img: 'img#manga-page' }; } }; var mangatown = { name: 'MangaTown', url: /https?:\/\/(www.)?mangatown.com\/manga\/.+\/.+/, homepage: 'http://www.mangatown.com/', language: ['English'], category: 'manga', waitEle: '#top_chapter_list option', waitMax: 5000, run() { const num = $('.page_select select:first option').get().slice(0, -1); const chapter = $('#top_chapter_list option').eq(W.current_chapter_index); return { title: $('.title h1').text(), series: $('.title h2 a').attr('href'), quant: num.length, prev: chapter.prev().val(), next: chapter.next().val(), listPages: num.map(item => $(item).val()), img: '#image' }; } }; var ninemanga = { name: 'NineManga', url: /https?:\/\/(www.)?ninemanga.com\/chapter\/.+\/.+\.html/, homepage: 'http://ninemanga.com/', language: ['English'], category: 'manga', run() { return { title: $('.tip a:first').text(), series: $('.subgiude a:eq(1)').attr('href'), quant: $('#page:first option').length, prev: $('.chnav a:first').attr('href'), next: $('.chnav a:eq(1)').attr('href'), listPages: $('#page:first option').get().map(item => $(item).val()), img: '.manga_pic' }; } }; var readmangatoday = { name: 'ReadManga Today', url: /https?:\/\/(www.)?readmng.com\/.+\/[0-9.]+(\/[0-9]*)?/, homepage: 'http://www.readmng.com/', language: ['English'], category: 'manga', run() { return { title: W.chapter_page_title.trim(), series: W.manga_url, quant: W.images.length, prev: W.prev_chapter_url, next: W.next_chapter_url, listImages: W.images.map(i => i.url) }; } }; var senmanga = { name: 'SenManga(Raw)', url: /https?:\/\/raw.senmanga.com\/.+\/.+\/?/, homepage: 'http://raw.senmanga.com/', language: ['English'], category: 'manga', run() { const url = '/' + String(location.pathname.split('/')[1]) + '/' + String(location.pathname.split('/')[2]); const num = parseInt($('select[name=\'page\'] option:last').val(), 10); const chapter = $('select[name="chapter"] option:selected'); const origin = $('.title a'); return { title: $('.title').text().trim(), series: origin.attr('href'), quant: num, prev: origin.attr('href') + chapter.next().val(), next: origin.attr('href') + chapter.prev().val(), listPages: [...Array(num).keys()].map(i => url + '/' + String(i + 1) + '/'), img: '#picture', before() { $('body').contents().filter(() => this.nodeType === 3).remove(); } }; } }; var tenmanga = { name: 'TenManga', url: /https?:\/\/(www.)?tenmanga.com\/chapter\/.+/, homepage: 'http://www.tenmanga.com/', language: ['English'], category: 'manga', run() { const url = $('.sl-page:first option').get(); const chapter = $('.sl-chap:first option:selected'); return { title: $('.read-page a:eq(2)').text().replace('»', '').trim(), series: $('.read-page a:eq(1)').attr('href'), quant: url.length, prev: chapter.next().val(), next: chapter.prev().val(), listPages: url.map(item => $(item).val()), img: '.manga_pic' }; } }; var thespectrum = { name: 'TheSpectrum', url: /https?:\/\/view.thespectrum.net\/.+/, homepage: 'http://www.thespectrum.net/', language: ['English'], category: 'manga', run() { const url = String(location.pathname) + '?' + String($('form').serialize().substring(0, $('form').serialize().lastIndexOf('='))); const num = $('.selectpage option').length; const chapter = $('.selectchapter option:selected'); return { title: $('.viewerLabel:eq(1)').text(), series: '#', quant: num, prev: String(location.pathname) + '?ch=' + String(chapter.prev().val()), next: String(location.pathname) + '?ch=' + String(chapter.next().val()), listPages: [...Array(num).keys()].map(i => url + '=' + String(i + 1)), img: '#imgContainer img' }; } }; var wpmanga = { name: ['MangaDeep'], url: /https?:\/\/(www.)?(mangadeep).com\/.+\/[0-9]+/, homepage: ['http://mangadeep.com/'], language: ['English'], category: 'manga', run() { const url = '/' + String(location.pathname.split('/')[1]) + '/' + String(location.pathname.split('/')[2]); const num = parseInt($('select.cbo_wpm_pag:first option:last').html(), 10); const chapter = $('.cbo_wpm_chp option:selected'); const key = $('.cbo_wpm_chp').attr('onchange').replace(/location.href='/, ''); return { title: $('.wpm_pag h1').text().trim(), series: $('h1.ttl a').attr('href'), quant: num, prev: key.replace(/'.+/, chapter.next().val()), next: key.replace(/'.+/, chapter.prev().val()), listPages: [...Array(num).keys()].map(i => url + '/' + String(i + 1) + '/'), img: 'img.manga-page , .prw > a img, .prw a img' }; } }; var funmanga = { name: 'Funmanga', url: /https?:\/\/(www.)?funmanga.com\/.+\/[0-9]+/, homepage: 'http://funmanga.com/', language: ['English'], category: 'manga', run() { const chapter = $('.extra-buttons select:first option:selected'); const url = $('.widget-heading select option').get().slice(1); return { title: $('title').text().trim(), series: $('h5.widget-heading a:first').attr('href'), quant: url.length, prev: chapter.next('option').val(), next: chapter.prev('option').val(), listPages: url.map(item => $(item).val()), img: '.img-responsive' }; } }; var unionmangas = { name: 'UnionMangas', url: /https?:\/\/(www.)?unionmangas.net\/leitor\/.+\/.+/, homepage: 'http://unionmangas.net/', language: ['Portuguese'], category: 'manga', run() { const origin = $('#topo h1 a'); const chapter = $('#cap_manga1 option:selected'); const src = $('.item img.real').get(); return { title: origin.text(), series: origin.attr('href'), quant: $('.selectPage:first option').length, prev: chapter.prev().val(), next: chapter.next().val(), listImages: [$(src[0]).attr('src')].concat(src.splice(1).map(item => $(item).attr('data-lazy'))) }; } }; var mangahost = { name: 'MangaHost', url: /https?:\/\/(www.)?mangahost.net\/manga\/.+\/.+/, homepage: 'https://mangahost.net/', language: ['Portuguese'], category: 'manga', run() { const url = location.href + (location.href.lastIndexOf('/') !== location.href.length - 1 ? '/' : ''); const chapter = $('.viewerChapter:first option:selected'); const num = parseInt($('.viewerPage:first option:last').html(), 10); const manga = { title: $('.breadcrumb li:eq(3)').text().trim(), series: $('.breadcrumb li:eq(2) a').attr('href'), quant: num, prev: chapter.next().val(), next: chapter.prev().val(), img: '.image-content img' }; if ($('.read-slideshow img').get().length === 0) { manga.listPages = [...Array(num).keys()].map(i => url + (i + 1)); } else { manga.listImages = $('.read-slideshow img').get().map(item => $(item).attr('src')); } return manga; } }; var hocvien = { name: 'Hoc Vien Truyen Tranh', url: /https?:\/\/(www.)?hocvientruyentranh.com\/chapter\/.+\/.+/, homepage: 'http://hocvientruyentranh.com/', language: ['Vietnamese'], category: 'manga', run() { const src = $('.manga-container img').get(); return { title: $('.chapters-dropdown option:selected').text().trim(), series: $('.theNavi a').attr('href'), quant: src.length, prev: $('.top-nav a:first').attr('href'), next: $('.top-nav a:last').attr('href'), listImages: src.map(item => $(item).attr('src')) }; } }; var jaiminisbox = { name: 'JaiminisBox', url: /https?:\/\/(www.)?jaiminisbox.com\/reader\/read\/.+/, homepage: 'https://jaiminisbox.com/', language: ['English'], category: 'manga', run() { const chapter = $('.topbar_left .dropdown_parent:last ul li a'); return { title: $('title').text().trim(), series: $('div.tbtitle div.text a:first').attr('href'), quant: W.pages.length, prev: chapter.eq(chapter.index(chapter.filter('[href*=\'' + String(location.pathname.replace(/page.+/, '')) + '\']')) + 1).attr('href'), next: chapter.eq(chapter.index(chapter.filter('[href*=\'' + String(location.pathname.replace(/page.+/, '')) + '\']')) - 1).attr('href'), listImages: W.pages.map(i => i.url) }; } }; var mangadex = { name: 'MangaDex', url: /https?:\/\/(www.)?mangadex.org\/chapter\/.+(\/.+)?/, homepage: 'https://mangadex.org/', language: ['English'], category: 'manga', waitEle: '.total-pages', waitAttr: ['.reader-image-wrapper img', 'src'], run() { let api = null; const url = 'https://mangadex.org/api/chapter/' + String(location.pathname.match(/[0-9]+/)[0]); $.ajax({ type: 'GET', url, async: false, success(res) { api = res; } }); return { title: $('title').text().replace(' - MangaDex', ''), series: $('.manga-link').attr('href'), quant: api.page_array.length, prev: $('.chapter-link-left').attr('href'), next: $('.chapter-link-right').attr('href'), listImages: api.page_array.map(img => String(api.server + api.hash) + '/' + String(img)) }; } }; var hatigarmscans = { name: 'HatigarmScans', url: /https?:\/\/(www.)?hatigarmscans.net\/manga\/.+\/.+(\/[0-9]*)?/, homepage: 'https://www.hatigarmscans.net//', language: ['English'], category: 'manga', run() { const src = $('.scan-page').attr('src'); const url = src.substring(0, src.lastIndexOf('/') + 1); return { title: W.title.replace(/ - Page .+/, '').trim(), series: W.base_url.substring(0, W.base_url.lastIndexOf('/') + 1), quant: W.pages.length, prev: W.next_chapter, next: W.prev_chapter, listImages: W.pages.map(i => url + i.page_image) }; } }; var mangarock = { name: 'MangaRock', url: /https?:\/\/(www.)?mangarock.com\/manga\/.+\/chapter\/.+/, homepage: 'https://mangarock.com/', language: ['English'], category: 'manga', waitAttr: ['a[title]', 'href'], run() { let api = null; const url = 'https://api.mangarockhd.com/query/web401/pages?oid=' + String(location.pathname.match(/mrs-chapter-[0-9]+/)[0]); $.ajax({ type: 'GET', url, async: false, success(res) { api = res; } }); function decode(t) { const s = ['length']; const e = new Uint8Array(t[s[0]] + 15); const n = t[s[0]] + 7; e[0] = 82; e[1] = 73; e[2] = 70; e[3] = 70; e[7] = n >> 24 & 255; e[6] = n >> 16 & 255; e[5] = n >> 8 & 255; e[4] = 255 & n; e[8] = 87; e[9] = 69; e[10] = 66; e[11] = 80; e[12] = 86; e[13] = 80; e[14] = 56; for (let r = 0; r < t[s[0]]; r += 1) e[r + 15] = 101 ^ t[r]; return e; } function process(mri) { const image = decode(new Uint8Array(mri)); const e = []; for (let n = 0; n < image.length; n += 32768) { e.push(String.fromCharCode.apply(null, image.subarray(n, n + 32768))); } return 'data:image/webp;base64,' + String(btoa(e.join(''))); } function getMRI(index, src, e) { setTimeout(() => { GM_xmlhttpRequest({ method: 'GET', url: src, overrideMimeType: 'text/plain; charset=x-user-defined', responseType: 'arraybuffer', onload(request) { e.addImg(index, process(request.response)); } }); }, e.wait * (index - e.begin)); } return { title: $('title').text().trim(), series: $('a[title]').attr('href'), quant: api.data.length, prev: $('select:first option:selected').prev().val(), next: $('select:first option:selected').next().val(), bruteForce(e) { for (let i = 0; i < api.data.length; i += 1) { if (i >= e.begin - 1) { getMRI(i + 1, api.data[i], e); } } } }; } }; var manganelo = { name: 'MangaNelo', url: /https?:\/\/(www.)?manganelo.com\/chapter\/.+\/.+/, homepage: 'http://www.manganelo.com/', language: ['English'], category: 'manga', run() { const images = $('img.img_content').get(); return { title: $('.info-top-chapter h2').text().trim(), series: $('.rdfa-breadcrumb a span[itemprop="title"]').eq(1).parent().attr('href'), quant: images.length, prev: $('.btn-navigation-chap a:eq(0)').attr('href'), next: $('.btn-navigation-chap a:eq(1)').attr('href'), listImages: images.map(i => $(i).attr('data')) }; } }; var lhtranslation = { name: 'LHTranslation', url: /https?:\/\/(www.)?lhtranslation.net\/read.+/, homepage: 'http://lhtranslation.net/', language: ['English'], category: 'manga', run() { return { title: $('.chapter-img.tieude font').text(), series: $('.navbar-brand.manga-name').attr('href'), quant: $('img.chapter-img').length, prev: $('.form-control option:selected').next().val(), next: $('.form-control option:selected').prev().val(), listImages: $('img.chapter-img').get().map(item => $(item).attr('src')) }; } }; var japscan = { name: 'JapScan.To', url: /https?:\/\/(www.)?japscan.to\/lecture-en-ligne\/.+\/.+/, homepage: 'https://www.japscan.to/', language: ['French'], category: 'manga', waitAttr: ['#image img', 'src'], run() { const src = $('#image img').attr('src').replace(/\/[0-9]+\.[a-z]+$/, '/'); return { title: $('.container h1').text(), series: $('.breadcrumb a:last').attr('href'), quant: $('#pages option').get().length, prev: $('.card-body span + a:first').attr('href'), next: $('.card-body span + a:last').attr('href'), listImages: $('#pages option').get().map(item => src + $(item).attr('data-img')) }; } }; var sites = [batoto, comicastle, readcomicsonline, dysnatyscans, eatmanga, egscans, foolslide, kissmanga, mangadoom, mangafox, mangago, mangahere, mangainn, mangalyght, mangapark, mangareader, mangastream, mangatown, ninemanga, readmangatoday, senmanga, tenmanga, thespectrum, wpmanga, funmanga, unionmangas, mangahost, hocvien, jaiminisbox, mangadex, hatigarmscans, mangarock, manganelo, lhtranslation, japscan]; start(sites); }());