Stream or download your favorite anime series effortlessly with AniLINK! Unlock the power to play any anime series directly in your preferred video player or download entire seasons in a single click using popular download managers like IDM. AniLINK generates direct download links for all episodes, conveniently sorted by quality. Elevate your anime-watching experience now!
// ==UserScript== // @name AniLINK - Episode Link Extractor // @namespace https://greasyfork.org/en/users/781076-jery-js // @version 6.21.2 // @description Stream or download your favorite anime series effortlessly with AniLINK! Unlock the power to play any anime series directly in your preferred video player or download entire seasons in a single click using popular download managers like IDM. AniLINK generates direct download links for all episodes, conveniently sorted by quality. Elevate your anime-watching experience now! // @icon https://www.google.com/s2/favicons?domain=animepahe.ru // @author Jery // @license MIT // @match https://anitaku.*/* // @match https://anitaku.io/* // @match https://gogoanime.*/* // @match https://gogoanime3.cc/* // @match https://gogoanime3.*/* // @match https://animepahe.*/play/* // @match https://animepahe.*/anime/* // @match https://animepahe.si/play/* // @match https://yugenanime.*/anime/*/*/watch/ // @match https://yugenanime.tv/anime/*/*/watch/ // @match https://yugenanime.sx/anime/*/*/watch/ // @match https://hianime.*/watch/* // @match https://hianime.to/watch/* // @match https://hianime.nz/watch/* // @match https://hianime.sz/watch/* // @match https://otaku-streamers.com/info/*/* // @match https://beta.otaku-streamers.com/watch/*/* // @match https://beta.otaku-streamers.com/title/*/* // @match https://animeheaven.me/anime.php?* // @match https://animez.org/*/* // @match https://animeyy.com/*/* // @match https://*.miruro.to/* // @match https://*.miruro.tv/* // @match https://*.miruro.online/* // @match https://anizone.to/anime/* // @match https://anixl.to/title/* // @match https://sudatchi.com/watch/*/* // @match https://hianime.*/watch/* // @match https://hianime.to/watch/* // @match https://hianime.nz/watch/* // @match https://hianimeZ.*/watch/* // @match https://aninow.tv/w/* // @match https://www.animegg.org/* // @match https://www.animeonsen.xyz/watch/* // @match https://kaido.to/watch/* // @match https://animetsu.cc/watch/* // @match https://animekai.to/watch/* // @match https://animekai.ac/watch/* // @match https://animekai.cc/watch/* // @match https://anikai.to/watch/* // @grant GM_registerMenuCommand // @grant GM_xmlhttpRequest // @grant GM.xmlHttpRequest // @require https://cdn.jsdelivr.net/npm/@trim21/[email protected] // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // ==/UserScript== // track last version as it might be needed by potential future updates if (GM_info.script.version > GM_getValue('script_version', '0')) { GM_setValue('script_version', GM_info.script.version); } /** * Represents an anime episode with metadata and streaming links. */ class Episode { /** * @param {string} number - The episode number. * @param {string} animeTitle - The title of the anime. * @param {Object.<string, {stream: string, type: '.m3u8'|'.mp4'|'.mpd'|'embed', tracks: Array<{file: string, kind: 'caption'|'audio', label: string}>}>, referer: string} links - An object containing streaming links and tracks for each source along with the referer (for use in CORS requests). * @param {string} thumbnail - The URL of the episode's thumbnail image. * @param {string} [epTitle] - The title of the episode (optional). */ constructor(number, animeTitle, links, thumbnail, epTitle) { this.number = String(number); // The episode number this.animeTitle = animeTitle; // The title of the anime. this.epTitle = epTitle; // The title of the episode (this can be the specific ep title or blank). this.links = this._processLinks(links); // An object containing streaming links and tracks for each source: {"source1":{stream:"url", type:"m3u8|mp4", tracks:[{file:"url", kind:"caption|audio", label:"name"}]}}} this.thumbnail = thumbnail; // The URL of the episode's thumbnail image (if unavailable, then just any image is fine. Thumbnail property isnt really used in the script yet). this.filename = `${this.animeTitle} - ${this.number.padStart(3, '0')}${this.epTitle ? ` - ${this.epTitle}` : ''}${Object.values(this.links)[0]?.type || ''}`; // The formatted name of the episode, combining anime name, number and title and extension. this.title = this.epTitle ?? this.animeTitle; } // Processes the links to ensure they are in right format and are absolute URLs. _processLinks(links) { for (const linkObj of Object.values(links)) { linkObj.stream &&= new URL(linkObj.stream, location.origin).href; // Ensure stream URLs are absolute linkObj.referer ??= location.href; // Set referer to current page if not present linkObj.type = (linkObj.type.startsWith('.') || (linkObj.type === 'embed')) ? linkObj.type : `.${linkObj.type}`; // Ensure type starts with a dot, but not for 'embed' linkObj.tracks?.forEach?.(track => track.kind = /^(caption|subtitle)s?/.test(track.kind) ? 'caption' : track.kind); // normalize all 'kind' values's subtitle(s) or caption(s) to 'caption' linkObj.tracks?.forEach?.(track => track.file &&= new URL(track.file, location.origin).href); // Ensure track file URLs are absolute } return links; } } /** * @typedef {Object} Websites[] * @property {string} name - The name of the website (required). * @property {string[]} url - An array of URL patterns that identify the website (required). * @property {string} thumbnail - A CSS selector to identify the episode thumbnail on the website (required). * @property {Function} addStartButton - A function to add the "Generate Download Links" button to the website (required). * @property {AsyncGeneratorFunction} extractEpisodes - An async generator function to extract episode information from the website (required). * @property {string} epLinks - A CSS selector to identify the episode links on the website (optional). * @property {string} epTitle - A CSS selector to identify the episode title on the website (optional). * @property {string} linkElems - A CSS selector to identify the download link elements on the website (optional). * @property {string} [animeTitle] - A CSS selector to identify the anime title on the website (optional). * @property {string} [epNum] - A CSS selector to identify the episode number on the website (optional). * @property {Function} [_getVideoLinks] - A function to extract video links from the website (optional). * @property {string} [styles] - Custom CSS styles to be applied to the website (optional). * * @description An array of website configurations for extracting episode links. * * @note To add a new website, follow these steps: * 1. Create a new object with the following properties: * - `name`: The name of the website. * - `url`: An array of URL patterns that identify the website. * - `thumbnail`: A CSS selector to identify the episode thumbnail on the website. * - `addStartButton`: A function to add the "Generate Download Links" button to the website. * - `extractEpisodes`: An async generator function to extract episode information from the website. * 2. Optionally, add the following properties if needed (they arent used by the script, but they will come in handy when the animesite changes its layout): * - `animeTitle`: A CSS selector to identify the anime title on the website. * - `epLinks`: A CSS selector to identify the episode links on the website. * - `epTitle`: A CSS selector to identify the episode title on the website. * - `linkElems`: A CSS selector to identify the download link elements on the website. * - `epNum`: A CSS selector to identify the episode number on the website. * - `_getVideoLinks`: A function to extract video links from the website. * - `styles`: Custom CSS styles to be applied to the website. * 3. Implement the `addStartButton` function to add the "Generate Download Links" button to the website. * - This function should create a element and append it to the appropriate location on the website. * - The button should have an ID of "AniLINK_startBtn". * 4. Implement the `extractEpisodes` function to extract episode information from the website. * - This function should be an async generator function that yields Episode objects (To ensure fast processing, using chunks is recommended). * - Use the `fetchPage` function to fetch the HTML content of each episode page. * - Parse the HTML content to extract the episode title, number, links, and thumbnail. * - Create an `Episode` object for each episode and yield it using the `yieldEpisodesFromPromises` function. * 5. Optionally, implement the `_getVideoLinks` function to extract video links from the website. * - This function should return a promise that resolves to an object containing video links. * - Use this function if the video links require additional processing or API calls. * - Tip: use GM_xmlhttpRequest to make cross-origin requests if needed (I've used proxy.sh so far which I plan to change in the future since GM_XHR seems more reliable). */ const Websites = [ { name: 'GoGoAnime', url: ['anitaku.to/', 'gogoanime3.co/', 'gogoanime3', 'anitaku.bz', 'gogoanime'], epLinks: '#episode_related > li > a', epTitle: '.title_name > h2', linkElems: '.cf-download > a', thumbnail: '.headnav_left > a > img', addStartButton: function () { const button = Object.assign(document.createElement('a'), { id: "AniLINK_startBtn", style: "cursor: pointer; background-color: #145132;", innerHTML: document.querySelector("div.user_auth a[href='/login.html']") ? `<b style="color:#FFC119;">AniLINK:</b> Please <a href="/login.html"><u>log in</u></a> to download` : '<i class="icongec-dowload"></i> Generate Download Links' }); const target = location.href.includes('/category/') ? '#episode_page' : '.cf-download'; document.querySelector(target)?.appendChild(button); return button; }, extractEpisodes: async function* (status) { const throttleLimit = 12; // Number of episodes to extract in parallel const allEpLinks = Array.from(document.querySelectorAll(this.epLinks)); const epLinks = await applyEpisodeRangeFilter(allEpLinks); for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { try { const page = await fetchPage(epLink.href); const [, epTitle, epNumber] = page.querySelector(this.epTitle).textContent.match(/(.+?) Episode (\d+(?:\.\d+)?)/); const thumbnail = page.querySelector(this.thumbnail).src; status.text = `Extracting ${epTitle} - ${epNumber.padStart(3, '0')}...`; const links = [...page.querySelectorAll(this.linkElems)].reduce((obj, elem) => ({ ...obj, [elem.textContent.trim()]: { stream: elem.href, type: 'mp4' } }), {}); status.text = `Extracted ${epTitle} - ${epNumber.padStart(3, '0')}`; return new Episode(epNumber, epTitle, links, thumbnail); // Return Episode object } catch (e) { showToast(e); return null; } }); // Handle errors and return null yield* yieldEpisodesFromPromises(episodePromises); // Use helper function } } }, { name: "Anitaku", url: ['anitaku.io'], extractEpisodes: async function* (status) { const epLinks = document.querySelectorAll('.episodelist li > a'); for (let i = 0, l = [...await applyEpisodeRangeFilter(epLinks)]; i < l.length; i += 12) yield* yieldEpisodesFromPromises(l.slice(i, i + 12).map(async a => { const pg = await fetchPage(a.href); const epNum = a.href.match(/-episode-(\d+)-/)[1]; status.text = `Extracting Episodes ${(epNum-Math.min(1, epNum)+1)} - ${epNum}...`; const links = {}; for (const [sel, name, attr, ref] of [['.fa-cloud-download-alt', u => 'GoFile', 'href', 0], ['iframe', u => u.includes('megaplay') ? 'MegaPlay' : 'VKSpeed', 'src', 1]]) { try { const el = pg.querySelector(sel); if (!el) continue; const url = attr === 'href' ? el.closest('a')[attr] : el[attr]; const src = await Extractors.use(url, ref ? location.href : undefined); links[typeof name === 'function' ? name(url) : name] = { stream: src.file, tracks: src.tracks || [], type: src.type || 'm3u8', ...(ref && { referer: location.href }) }; } catch (e) { showToast(`${typeof name === 'function' ? 'iframe' : name} error ep ${epNum}: ${e}`); } } return new Episode(epNum, pg.querySelector('.det > h2 > a').textContent.trim(), links, pg.querySelector('img').src); })); } }, { name: 'YugenAnime', url: ['yugenanime.tv', 'yugenanime.sx'], epLinks: '.ep-card > a.ep-thumbnail', animeTitle: '.ani-info-ep .link h1', epTitle: 'div.col.col-w-65 > div.box > h1', thumbnail: 'a.ep-thumbnail img', addStartButton: function () { return document.querySelector(".content .navigation").appendChild(Object.assign(document.createElement('a'), { id: "AniLINK_startBtn", className: "link p-15", textContent: "Generate Download Links" })); }, extractEpisodes: async function* (status) { status.text = 'Getting list of episodes...'; const allEpLinks = Array.from(document.querySelectorAll(this.epLinks)); const epLinks = await applyEpisodeRangeFilter(allEpLinks); const throttleLimit = 6; // Number of episodes to extract in parallel for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async (epLink, index) => { try { status.text = `Loading ${epLink.pathname}`; const page = await fetchPage(epLink.href); const animeTitle = page.querySelector(this.animeTitle).textContent; const epNumber = epLink.href.match(/(\d+)\/?$/)[1]; const epTitle = page.querySelector(this.epTitle).textContent.match(/^${epNumber} : (.+)$/) || animeTitle; const thumbnail = document.querySelectorAll(this.thumbnail)[index].src; status.text = `Extracting ${`${epNumber.padStart(3, '0')} - ${animeTitle}` + (epTitle != animeTitle ? `- ${epTitle}` : '')}...`; const rawLinks = await this._getVideoLinks(page, status, epTitle); const links = Object.entries(rawLinks).reduce((acc, [quality, url]) => ({ ...acc, [quality]: { stream: url, type: 'm3u8' } }), {}); return new Episode(epNumber, epTitle, links, thumbnail); } catch (e) { showToast(e); return null; } }); yield* yieldEpisodesFromPromises(episodePromises); } }, _getVideoLinks: async function (page, status, episodeTitle) { const embedLinkId = page.body.innerHTML.match(new RegExp(`src="//${page.domain}/e/(.*?)/"`))[1]; const embedApiResponse = await fetch(`https://${page.domain}/api/embed/`, { method: 'POST', headers: { "X-Requested-With": "XMLHttpRequest" }, body: new URLSearchParams({ id: embedLinkId, ac: "0" }) }); const json = await embedApiResponse.json(); const m3u8GeneralLink = json.hls[0]; status.text = `Parsing ${episodeTitle}...`; // Fetch the m3u8 file content const m3u8Response = await fetch(m3u8GeneralLink); const m3u8Text = await m3u8Response.text(); // Parse the m3u8 file to extract different qualities const qualityMatches = m3u8Text.matchAll(/#EXT-X-STREAM-INF:.*RESOLUTION=\d+x\d+.*NAME="(\d+p)"\n(.*\.m3u8)/g); const links = {}; for (const match of qualityMatches) { const [_, quality, m3u8File] = match; links[quality] = `${m3u8GeneralLink.slice(0, m3u8GeneralLink.lastIndexOf('/') + 1)}${m3u8File}`; } return links; } }, { name: 'AnimePahe', url: ['animepahe.si', 'animepahe'], epLinks: (location.pathname.startsWith('/anime/')) ? 'a.play' : '.dropup.episode-menu a.dropdown-item', epTitle: '.theatre-info > h1', linkElems: '#resolutionMenu > button', thumbnail: '.theatre-info > a > img', _chunkSize: 36, // Setting high throttle limit actually improves performance addStartButton: function () { GM_addStyle(`.theatre-settings .col-sm-3 { max-width: 20%; }`); (document.location.pathname.startsWith('/anime/')) ? document.querySelector(".col-6.bar").innerHTML += `<div class="btn-group btn-group-toggle"><label id="AniLINK_startBtn" class="btn btn-dark btn-sm">Generate Download Links</label></div>` : document.querySelector("div.theatre-settings > div.row").innerHTML += `<div class="col-12 col-sm-3"><div class="dropup"><a class="btn btn-secondary btn-block" id="AniLINK_startBtn">Generate Download Links</a></div></div>`; return document.getElementById("AniLINK_startBtn"); }, extractEpisodes: async function* (status) { const allEpLinks = Array.from(document.querySelectorAll(this.epLinks)); const epLinks = await applyEpisodeRangeFilter(allEpLinks); // Resolve the ep numbering offset (sometimes, a 2nd cour can have ep.num=13 while its s2e1) const firstEp = () => document.querySelector(this.epLinks).textContent.match(/.*\s(\d+)/)[1]; let firstEpNum = firstEp(); if (document.querySelector('.btn.active')?.innerText == 'desc') { document.querySelector('.episode-bar .btn').click(); await new Promise(r => { const c = () => firstEp() !== firstEpNum ? r() : setTimeout(c, 500); c(); }); firstEpNum = firstEp(); } for (let i = 0; i < epLinks.length; i += this._chunkSize) yield* yieldEpisodesFromPromises(epLinks.slice(i, i + this._chunkSize).map(async epLink => { const page = await fetchPage(epLink.href); const [, animeTitle, epNum] = page.querySelector(this.epTitle).outerText.split(/Watch (.+) - (\d+(?:\.\d+)?) Online$/); const epNumber = (epNum - firstEpNum + 1).toString(); const thumbnail = page.querySelector(this.thumbnail).src; status.text = `Extracting episodes ${epNumber-Math.min(epNumber, this._chunkSize)+1} - ${epNumber}...`; const links = Object.fromEntries(await Promise.all([...page.querySelectorAll(this.linkElems)].map(async elm => [elm.textContent, { stream: await Extractors.use(elm.getAttribute('data-src')), type: 'm3u8' }]))); return new Episode(epNumber, animeTitle, links, thumbnail); })); }, styles: `div#AniLINK_LinksContainer { font-size: 10px; } #Quality > b > div > ul {font-size: 16px;}` }, { name: 'Beta-Otaku-Streamers', url: ['beta.otaku-streamers.com'], epLinks: (document.location.pathname.startsWith('/title/')) ? '.item-title a' : '.video-container .clearfix > a', epTitle: '.title > a', epNum: '.watch_curep', thumbnail: 'video', addStartButton: function () { (document.location.pathname.startsWith('/title/') ? document.querySelector(".album-top-box") : document.querySelector('.video-container .title-box')) .innerHTML += `<a id="AniLINK_startBtn" class="btn btn-outline rounded-btn">Generate Download Links</a>`; return document.getElementById("AniLINK_startBtn"); }, extractEpisodes: async function* (status) { const allEpLinks = Array.from(document.querySelectorAll(this.epLinks)); const epLinks = await applyEpisodeRangeFilter(allEpLinks); const throttleLimit = 12; for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { try { const page = await fetchPage(epLink.href); const epTitle = page.querySelector(this.epTitle).textContent.trim(); const epNumber = page.querySelector(this.epNum).textContent.replace("Episode ", ''); const thumbnail = page.querySelector(this.thumbnail).poster; status.text = `Extracting ${epTitle} - ${epNumber}...`; const links = { 'Video Links': { stream: page.querySelector('video > source').src, type: 'mp4' } }; return new Episode(epNumber, epTitle, links, thumbnail); } catch (e) { showToast(e); return null; } }); yield* yieldEpisodesFromPromises(episodePromises); } } }, { name: 'Otaku-Streamers', url: ['otaku-streamers.com'], epLinks: 'table > tbody > tr > td:nth-child(2) > a', epTitle: '#strw_player > table > tbody > tr:nth-child(1) > td > span:nth-child(1) > a', epNum: '#video_episode', thumbnail: 'otaku-streamers.com/images/os.jpg', addStartButton: function () { const button = document.createElement('a'); button.id = "AniLINK_startBtn"; button.style.cssText = `cursor: pointer; background-color: #145132; float: right;`; button.innerHTML = 'Generate Download Links'; document.querySelector('table > tbody > tr:nth-child(2) > td > div > table > tbody > tr > td > h2').appendChild(button); return button; }, extractEpisodes: async function* (status) { const allEpLinks = Array.from(document.querySelectorAll(this.epLinks)); const epLinks = await applyEpisodeRangeFilter(allEpLinks); const throttleLimit = 12; // Number of episodes to extract in parallel for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { try { const page = await fetchPage(epLink.href); const epTitle = page.querySelector(this.epTitle).textContent; const epNumber = page.querySelector(this.epNum).textContent.replace("Episode ", '') status.text = `Extracting ${epTitle} - ${epNumber}...`; const links = { 'mp4': { stream: page.querySelector('video > source').src, type: 'mp4' } }; return new Episode(epNumber, epTitle, links, this.thumbnail); // Return Episode object } catch (e) { showToast(e); return null; } }); // Handle errors and return null yield* yieldEpisodesFromPromises(episodePromises); // Use helper function } } }, { name: 'AnimeHeaven', url: ['animeheaven.me'], epLinks: 'a.ac3', epTitle: 'a.c2.ac2', epNumber: '.boxitem.bc2.c1.mar0', thumbnail: 'img.posterimg', addStartButton: function () { const button = document.createElement('a'); button.id = "AniLINK_startBtn"; button.style.cssText = `cursor: pointer; border: 2px solid red; padding: 4px;`; button.innerHTML = 'Generate Download Links'; document.querySelector("div.linetitle2.c2").parentNode.insertBefore(button, document.querySelector("div.linetitle2.c2")); return button; }, extractEpisodes: async function* (status) { const allEpLinks = Array.from(document.querySelectorAll(this.epLinks)); const epLinks = await applyEpisodeRangeFilter(allEpLinks); const throttleLimit = 12; // Number of episodes to extract in parallel for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { try { const page = await fetchPage(epLink.href); const epTitle = page.querySelector(this.epTitle).textContent; const epNumber = page.querySelector(this.epNumber).textContent.replace("Episode ", ''); const thumbnail = document.querySelector(this.thumbnail).src; status.text = `Extracting ${epTitle} - ${epNumber}...`; const links = [...page.querySelectorAll('#vid > source')].reduce((acc, source) => ({ ...acc, [source.src.match(/\/\/(\w+)\./)[1]]: { stream: source.src, type: 'mp4' } }), {}); return new Episode(epNumber, epTitle, links, thumbnail); // Return Episode object } catch (e) { showToast(e); return null; } }); // Handle errors and return null yield* yieldEpisodesFromPromises(episodePromises); // Use helper function } } }, { name: 'AnimeZ', url: ['animez.org', 'animeyy.com'], epLinks: 'li.wp-manga-chapter a', epTitle: '#title-detail-manga', epNum: '.wp-manga-chapter.active', thumbnail: '.Image > figure > img', addStartButton: function () { (document.querySelector(".MovieTabNav.ControlPlayer") || document.querySelector(".mb-3:has(#keyword_chapter)")) .innerHTML += `<div class="Lnk AAIco-link" id="AniLINK_startBtn">Extract Episode Links</div>`; return document.getElementById("AniLINK_startBtn"); }, extractEpisodes: async function* (status) { /// work in progress- stopped when animes.org started redirecting to some random manhwa site status.text = 'Fetching Episodes List...'; const mangaId = (window.location.pathname.match(/-(\d+)(?:\/|$)/) || [])[1] || document.querySelector('[data-manga-id]')?.getAttribute('data-manga-id'); if (!mangaId) return showToast('Could not determine manga_id for episode list.'); const nav = [...document.querySelectorAll('#nav_list_chapter_id_detail li > :not(a.next)')]; const maxPage = Math.max(1, ...Array.from(nav).map(a => +(a.getAttribute('onclick')?.match(/load_list_chapter\((\d+)\)/)?.[1] || 0)).filter(Boolean)); // Parse all episode links from all pages in parallel status.text = `Loading all ${maxPage} episode pages...`; let allEpLinks = []; try { await Promise.all(Array.from({ length: maxPage }, (_, i) => fetch(`/?act=ajax&code=load_list_chapter&manga_id=${mangaId}&page_num=${i + 1}&chap_id=0&keyword=`).then(r => r.text()).then(t => { let html = JSON.parse(t).list_chap; const doc = document.implementation.createHTMLDocument('eps'); doc.body.innerHTML = html; allEpLinks.push(...doc.querySelectorAll(this.epLinks)); }))); } catch (e) { showToast('Failed to load Episodes List: ' + e); return null; } // Remove duplicates allEpLinks = allEpLinks.filter((el, idx, self) => self.findIndex(e => e.href === el.href && e.textContent.trim() === el.textContent.trim()) === idx); const epLinks = await applyEpisodeRangeFilter(allEpLinks); const throttleLimit = 12; for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { try { const page = await fetchPage(epLink.href); const epTitle = page.querySelector(this.epTitle).textContent; const isDub = page.querySelector(this.epNum).textContent.includes('-Dub'); const epNumber = page.querySelector(this.epNum).textContent.replace(/-Dub/, '').trim(); const thumbnail = document.querySelector(this.thumbnail).src; status.text = `Extracting ${epTitle} - ${epNumber}...`; const links = { [isDub ? "Dub" : "Sub"]: { stream: page.querySelector('iframe').src.replace('/embed/', '/anime/'), type: 'm3u8' } }; return new Episode(epNumber, epTitle, links, thumbnail); // Return Episode object } catch (e) { showToast(e); return null; } }); yield* yieldEpisodesFromPromises(episodePromises); } } }, { name: 'Miruro', url: ['miruro.to', 'miruro.tv', 'miruro.online'], animeTitle: '.anime-title > a', thumbnail: 'a[href^="/info?id="] > img', baseApiUrl: `${location.origin}/api`, addStartButton: function (id) { let last_known = { location: location.href, source: null }; const intervalId = setInterval(() => { const currSource = [...document.querySelectorAll('select')].slice(1).map(e => e.value).toString(); if (last_known.location !== location.href || last_known.source !== currSource) { last_known = { location: location.href, source: currSource }; document.getElementById('AniLINK_Overlay')?.remove(); } // Append the extract button const target = document.querySelector('.App + div > div > div + div > div > div > div > div + div > div + div'); if (target && !document.getElementById(id)) { // clearInterval(intervalId); const btn = document.createElement('button'); btn.id = id; btn.style.cssText = `${target.lastChild.style.cssText} display: flex; justify-content: center; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; flex: auto;`; btn.className = target.lastChild.className; btn.innerHTML = ` <svg xmlns="http://www.w3.org/2000/svg" height="1em" viewBox="3 3 18 18"><path fill="currentColor" d="M5 21q-.825 0-1.413-.588T3 19V5q0-.825.588-1.413T5 3h14q.825 0 1.413.588T21 5v14q0 .825-.588 1.413T19 21H5Zm0-2h14V5H5v14Zm3-4.5h2.5v-6H8v6Zm5.25 0h2.5v-6h-2.5v6Zm5.25 0h2.5v-6h-2.5v6Z"/></svg> <div style="display: flex; justify-content: center; align-items: center;">Extract Episode Links</div> `; btn.addEventListener('click', extractEpisodes); target.appendChild(btn); } }, 500); }, extractEpisodes: async function* (status) { status.text = 'Fetching episode list...'; const animeTitle = (document.querySelector('p.title-romaji') || document.querySelector(this.animeTitle)).textContent; const malId = document.querySelector(`a[href*="/myanimelist.net/anime/"]`)?.href.split('/').pop(); if (!malId) return showToast('MAL ID not found.'); const res = await this._secureFetch(`${this.baseApiUrl}/episodes`, { query: { malId } }); const eps = Object.entries(res.providers).reduce((a, [provider, { episodes }]) => ( Object.entries(episodes).forEach(([type, list]) => list.forEach(ep => (a[ep.number] ??= []).push({ ...ep, provider, type }))), a ), {}); showToast('Found Providers: ' + Object.entries(Object.values(eps).flat().reduce((m, ep) => ((m[this._getLocalSourceName(ep.provider)] ??= new Set()).add(ep.type), m), {})).map(([p, t]) => `${p.toLowerCase()} (${[...t].join(', ')})`).join(', ')); for (const epNum of await applyEpisodeRangeFilter(Object.keys(eps).sort((a, b) => a - b))) { const baseEp = eps[epNum][0]; status.text = `Fetching Ep ${epNum}...`; const links = {}; await Promise.all(eps[epNum].map(async ({ id, provider, type }) => { if ([...document.querySelectorAll('select')][2].value.includes(provider.toLowerCase()) && [...document.querySelectorAll('select')][1].value.includes(type)) { const source = this._getLocalSourceName(provider, type); try { const sresJson = await this._secureFetch(`${this.baseApiUrl}/sources`, { query: { episodeId: id, provider, category: type } }); const referer = provider == 'KICKASSANIME' ? 'https://kaa.to/' : provider == 'ZORO' ? 'https://megacloud.blog/' : location.href; links[this._getLocalSourceName(source)] = { stream: sresJson.streams[0].url, type: "m3u8", tracks: sresJson.tracks || [], referer }; } catch (e) { showToast(`Failed to fetch ep-${epNum} from ${source}: ${e}`); } } })); yield new Episode(epNum, animeTitle, links, baseEp.image, baseEp.title); } }, _secureFetch: async (url, options = {}) => { const payload = { path: url.split('/api/').pop(), method: 'GET', query: options.query || {}, body: null, version: '0.1.0'}; const encode = o => btoa(encodeURIComponent(JSON.stringify(o)).replace(/%([0-9A-F]{2})/g, (_, h) => String.fromCharCode(parseInt(h, 16)))).replace(/\+/g, '-').replace(/\//g, '_').replace(/=/g, ''); const decode = async s => JSON.parse(new TextDecoder().decode(await new Response(new Blob([Uint8Array.from(atob(s.replace(/-/g, '+').replace(/_/g, '/')), c => c.charCodeAt(0))]).stream().pipeThrough(new DecompressionStream('gzip'))).arrayBuffer())); const res = await fetch(`${location.origin}/api/secure/pipe?e=${encode(payload)}`, {headers: { 'x-protocol-version': payload.version }}); if (res.headers.get('x-obfuscated') === '1') return await decode(await res.text()); return await res.json(); }, _getLocalSourceName: function (source, type) { source = source.toLowerCase(); const sourceNames = { 'animepahe': 'kiwi', 'animekai': 'arc', 'animez': 'jet', 'zoro': 'zoro', 'kickassanime': 'kaa' }; return (sourceNames[source] || source) + (type !== undefined ? `-${type.toLowerCase()}` : ''); }, }, { name: 'AniZone', url: ['anizone.to/'], animeTitle: 'nav > span', epTitle: 'div.space-y-2 > div.text-center', epNumber: 'a[x-ref="activeEps"] > div > div', thumbnail: 'media-poster', epLinks: () => [...new Set(Array.from(document.querySelectorAll('a[wire\\:key][href^="https://anizone.to/anime/"]')).map(a => a.href))], addStartButton: function () { const target = document.querySelector('button > span.truncate')?.parentElement || document.querySelector('.grow + div select'); const button = Object.assign(document.createElement('button'), { id: "AniLINK_startBtn", className: target.className, style: "display: flex; justify-content: center; align-items: center; width: 100%;", innerHTML: `<svg xmlns="http://www.w3.org/2000/svg" style="margin-right: 4px;" height="1em" viewBox="3 3 18 18"><path fill="currentColor" d="M5 21q-.825 0-1.413-.588T3 19V5q0-.825.588-1.413T5 3h14q.825 0 1.413.588T21 5v14q0 .825-.588 1.413T19 21H5Zm0-2h14V5H5v14Zm3-4.5h2.5v-6H8v6Zm5.25 0h2.5v-6h-2.5v6Zm5.25 0h2.5v-6h-2.5v6Z"/></svg><span class="truncate">Extract Episode Links</span>` }); target.parentElement.appendChild(button); return button; }, extractEpisodes: async function* (status) { const epLinks = await applyEpisodeRangeFilter(this.epLinks()); const throttleLimit = 12; // Limit concurrent requests for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { try { const page = await fetchPage(epLink); const animeTitle = page.querySelector(this.animeTitle)?.textContent.trim(); const epNum = page.querySelector(this.epNumber)?.textContent.trim(); const epTitle = page.querySelector(this.epTitle)?.textContent.trim(); const thumbnail = page.querySelectorAll('media-poster')[0].outerHTML.match(/src="([^"]*)"/)[1]; // using outerHTML as workaround for a weird bug status.text = `Extracting ${epNum} - ${epTitle}...`; const links = { [page.querySelector('button > span.truncate').textContent]: { stream: page.querySelector("media-player").getAttribute("src"), type: "m3u8", tracks: [...page.querySelectorAll("media-provider>track")].map(t => ({ file: t.src, kind: t.kind, label: t.label })) } }; return new Episode(epNum, animeTitle, links, thumbnail, epTitle); } catch (e) { showToast(e); return null; } }); yield* yieldEpisodesFromPromises(episodePromises); } } }, { name: 'AniXL', url: ['anixl.to/'], animeTitle: () => document.querySelector('a.link[href^="/title/"]').textContent, epLinks: () => [...document.querySelectorAll('div[q\\:key^="F0_"] a')].map(e=>e.href), addStartButton: () => document.querySelector('div.join')?.prepend( Object.assign(document.createElement('button'), {id: "AniLINK_startBtn", className: "btn btn-xs", textContent: "Generate Download Links"}) ), extractEpisodes: async function* (status) { const epLinks = await applyEpisodeRangeFilter(this.epLinks()); const throttleLimit = 12; // Limit concurrent requests for (let i = 0; i < epLinks.length; i += throttleLimit) { const chunk = epLinks.slice(i, i + throttleLimit); const episodePromises = chunk.map(async epLink => { return await fetchPage(epLink).then(page => { const [, epNum, epTitle] = page.querySelector('a[q\\:id="1s"]').textContent.match(/Ep (\d+) : (.*)/d) status.text = `Extracting ${epNum} - ${epTitle}...`; const links = page.querySelector('script[type="qwik/json"]').textContent.match(/"[ds]ub","https:\/\/[^"]+\/media\/[^"]+\/[^"]+\.m3u8"/g)?.map(s => s.split(',').map(JSON.parse)).reduce((acc, [type, url]) => ({ ...acc, [type]: { stream: url, type: 'm3u8', tracks: [] } }), {}); return new Episode(epNum, this.animeTitle(), links, null, epTitle); }).catch(e => { showToast(e); return null; }); }); yield* yieldEpisodesFromPromises(episodePromises); } } }, { name: 'Sudatchi', url: ['sudatchi.com/'], epLinks: () => [...document.querySelectorAll('.text-sm.rounded-lg')].map(e => `${location.href}/../${e.textContent}`), extractEpisodes: async function* (status) { for (let i = 0, l = await applyEpisodeRangeFilter(this.epLinks()); i < l.length; i += 6) yield* yieldEpisodesFromPromises(l.slice(i, i + 6).map(async link => await fetchPage(link).then(p => { status.text = `Extracting ${link.split('/').pop().padStart(3, '0')}...`; const tracks = JSON.parse([...p.scripts].flatMap(s => s.textContent.match(/\[{.*"}/)).filter(Boolean)[0].replaceAll('\\', '') + ']').map(i => ({ file: i.file.replace('/ipfs/', 'https://sudatchi.com/api/proxy/'), label: i.label, kind: i.kind })); const links = { 'Sudatchi': { stream: p.querySelector('meta[property="og:video"]').content.replace(/http.*:8888/, location.origin), type: 'm3u8', tracks } }; return new Episode(link.split('/').pop(), p.querySelector('p').textContent, links, p.querySelector('video').poster); }).catch(e => { showToast(e); return null; }) )); } }, { name: 'HiAnime', url: ['hianime.to/', 'hianimez.is/', 'hianimez.to/', 'hianime.nz/', 'hianime.bz/', 'hianime.pe/', 'hianime.cx/', 'hianime.gs/'], _chunkSize: 1, // Number of episodes to extract in parallel extractEpisodes: async function* (status) { for (let i = 0, epList = await applyEpisodeRangeFilter($('.ss-list > a').get()); i < epList.length; i += this._chunkSize) { yield* yieldEpisodesFromPromises(epList.slice(i, i + this._chunkSize).map(async e => { const [epId, epNum, epTitle] = [$(e).data('id'), $(e).data('number'), $(e).find('.ep-name').text()]; let thumbnail = ''; status.text = `Extracting Episode ${epNum-Math.min(this._chunkSize, epNum)+1}...`; const servers = await $((await $.get(`/ajax/v2/episode/servers?episodeId=${epId}`, r => $(r).responseJSON)).html).find('.server-item').map((_, i) => [[$(i).text().trim(), { id: $(i).data('id'), type: $(i).data('type') }]]).get(); // Prefer HD-2 if available. (HD-1 and HD-3 might have CORS issues) const filteredServers = servers.filter(([s]) => !['HD-1', 'HD-3'].includes(s)); const links = await (filteredServers.length ? filteredServers : servers).reduce(async (linkAcc, [server, { id, type }]) => {try { const data = await fetch(`/ajax/v2/episode/sources?id=${id}`).then(r => r.json()); const src = await Extractors.use(data.link, location.href); return {...await linkAcc, [`${server}-${type}`]: { stream: src.file, type: 'm3u8', tracks: src.tracks, referer: src.referer || location.href }}; } catch (e) { showToast(`Failed to fetch Ep ${epNum} from ${server}-${type}: (${e.status}): ${e.message || e}`); return linkAcc; }}, Promise.resolve({})); return new Episode(epNum, ($('.film-name > a').first().text()), links, thumbnail, epTitle); }))} } }, { name: 'AniNow', url: ['aninow.tv/'], _chunkSize: 6, // Number of episodes to extract in parallel _decryptUrl: async (encryptedUrl) => (new TextDecoder()).decode(await crypto.subtle.decrypt({ name: 'AES-CBC', iv: (encryptedBytes => encryptedBytes.slice(0, 16))(Uint8Array.from(atob(encryptedUrl), c => c.charCodeAt(0))) }, await crypto.subtle.importKey('raw', (new TextEncoder()).encode('superaninowq8hgl1'.padEnd(32, '\0').slice(0, 32)), { name: 'AES-CBC' }, false, ['decrypt']), (encryptedBytes => encryptedBytes.slice(16))(Uint8Array.from(atob(encryptedUrl), c => c.charCodeAt(0))))), extractEpisodes: async function* (status) { for (let i = 0, l = await applyEpisodeRangeFilter([...document.querySelectorAll('a[data-episode]')]); i < l.length; i+=this._chunkSize) yield* yieldEpisodesFromPromises(l.slice(i, i + this._chunkSize).map(async a => { const epNum = a.innerText; status.text = `Extracting Episodes ${(epNum-Math.min(this._chunkSize, epNum)+1)} - ${epNum}...`; const data = await fetchPage(a.href).then(p => JSON.parse(p.querySelector("#media-sources-data").dataset.mediaSources)).then(d => d.filter(l => !!l.url)); const links = Object.fromEntries(await Promise.all(data.map(async m => [ `${m.providerdisplayname}-${m.language}-${m.quality}`, { stream: !m.url.startsWith('videos/') ? m.url : await this._decryptUrl((await fetch('https://aninow.tv/api/presigned/media/' + m.url).then(r => r.json())).url), type: m.url.endsWith('mp4') ? 'mp4' : 'm3u8', tracks: m.subtitles.map(s => ({ file: s.filename.startsWith('subtitles/') ? 'https://aninow.tv/api/subtitles/C:/Users/GraceAshby/OneDrive/aninow-copy/subzzzzzz/' + s.filename : s.filename, label: s.displayname, kind: 'caption' })) } ]))); return new Episode(epNum.padStart(3, '0'), document.querySelector('h1').innerText, links, document.querySelector('a>img').src); })); }, }, { name: "Animegg", url: ['animegg.org/'], extractEpisodes: async function* (status) { const epLinks = $((!!$('.anm_det_pop').length) ? document : $(await fetchPage($('.nap > a[href^="/series/"]').get(0).href))).find('.newmanga > li > div').get().reverse(); for (let i = 0, l = await applyEpisodeRangeFilter(epLinks); i < l.length; i += 1) yield* yieldEpisodesFromPromises(l.slice(i, i + 1).map(async div => { const pg = $(await fetchPage($(div).find('.anm_det_pop').get(0).href)); const epNum = pg.find('.info > a').text().split(' ').pop(); status.text = `Extracting Episodes ${(epNum-Math.min(1, epNum)+1)} - ${epNum}...`; const links = Object.fromEntries(await Promise.all(pg.find('#videos a').get().map(async a => [a.dataset.version, { stream: (await fetch((await fetchPage('/embed/' + a.dataset.id)).querySelector('[property="og:video"]')?.content, { method: 'HEAD' }).catch(e => showToast(`Error fetching ep ${epNum} - ${a.dataset.version}: ${e}`)))?.url, type: 'mp4', referer: location.origin }])).then(r => r.filter(([_, v]) => v.stream))); return new Episode(epNum, pg.find('.titleep a').text().trim(), links, $('a > img').get(0).src, $(div).find('.anititle').text()); })); }, }, { name: "AnimeOnsen", url: ['animeonsen.xyz/'], extractEpisodes: async function* (status) { for (let i = 0, epLinks = await applyEpisodeRangeFilter([..._$('.ao-player-metadata-episode').options].map(o=>o.value.split('-')[1])); i < epLinks.length; i += 12) { yield* yieldEpisodesFromPromises(epLinks.slice(i, i + 12).map(async epNum => { status.text = `Extracting Episodes ${(epNum-Math.min(12, epNum)+1)} - ${epNum}...`; const token = atob(decodeURIComponent(document.cookie.match(new RegExp('(^|;\\s*)' + 'ao.session' + '=([^;]*)'))[2])).split("").map(c => String.fromCharCode(c.charCodeAt(0) + 1)).join(""); const data = await fetch(`https://api.animeonsen.xyz/v4/content/${document.querySelector('[name="ao-content-id"]').content}/video/${epNum}`, { headers: { 'Authorization': `Bearer ${token}` } }).then(r => r.json()); const links = { "AnimeOnsen": { stream: data.uri.stream, type: ".mpd", tracks: Object.entries(data.uri.subtitles).map(([label, file]) => ({ file, label, kind: 'caption' })), referer: location.origin } }; return new Episode(epNum.toString().padStart(3, '0'), data.metadata.content_title, links, _$('[property="og:image"]').content, data.metadata.episode[1].contentTitle_episode_en); })); } } }, { name: "Kaido", url: ["kaido.to"], extractEpisodes: async function* (status) { for (let i = 0, epLinks = await applyEpisodeRangeFilter([..._$$('a.ep-item')]); i < epLinks.length; i += 12) yield* yieldEpisodesFromPromises(epLinks.slice(i, i + 12).map(async epLink => { const epNum = epLink.dataset.number; status.text = `Extracting Episodes ${(epNum-Math.min(12, epNum)+1)} - ${epNum}...`; return await fetch(`/ajax/episode/servers?episodeId=${epLink.dataset.id}`).then(async r => (await r.json()).html).then(t => (new DOMParser()).parseFromString(t, 'text/html')) .then(h => [...h.querySelectorAll('[data-server-id]')].map(e => ({id: e.dataset.id, type: e.dataset.type, name: e.textContent.trim()}))) .then(async servers => { const links = Object.fromEntries(await Promise.all(servers.map(async s => fetch(`/ajax/episode/sources?id=${s.id}`).then(r => r.json()) .then(d => GM_fetch(d.link.replace('/e-1/', '/e-1/getSources?id=').replace('?z=', '')).then(r => r.json()) .then(src => src.encrypted ? undefined : [`${s.name}-${s.type}`, { stream: src.sources[0].file, tracks: src.tracks, type: 'm3u8', referer: src.server == 4 ? 'https://megacloud.blog/' : undefined }]))))); return new Episode(epNum, _$('h2.film-name > a').textContent, links, _$('.film-poster > img').src, epLink.querySelector('.ep-name').textContent) }); })); } }, { name: "Gojo", url: ["animetsu.cc"], addStartButton: function (id) { // Use same logic as Miruro, but target gojo layout let last_known_location = location.href; setInterval(() => { if (last_known_location !== location.href) { last_known_location = location.href; document.getElementById('AniLINK_Overlay')?.remove() }; // Prepend the extract button const target = document.querySelector('.Video .items-center.gap-2 + div'); if (target && !document.getElementById(id)) { const btn = Object.assign(document.createElement('button'), {id, className: (target.lastChild?.className || '') + " font-light w-fit !shrink-0 text-[.6rem] sm:text-xs justify-center items-center whitespace-nowrap overflow-hidden text-ellipsis flex", innerHTML: `<svg xmlns="http://www.w3.org/2000/svg" height="1em" viewBox="3 3 18 18" style="margin-right:6px;"><path fill="currentColor" d="M5 21q-.825 0-1.413-.588T3 19V5q0-.825.588-1.413T5 3h14q.825 0 1.413.588T21 5v14q0 .825-.588 1.413T19 21H5Zm0-2h14V5H5v14Zm3-4.5h2.5v-6H8v6Zm5.25 0h2.5v-6h-2.5v6Zm5.25 0h2.5v-6h-2.5v6Z"/></svg>Extract Episode Links`}); btn.addEventListener('click', extractEpisodes); target.prepend(btn); } }, 500); }, extractEpisodes: async function* (status) { const id = location.pathname.split('/').pop(); for (let i = 0, epElms = await applyEpisodeRangeFilter([..._$$('.Episode button:not(:nth-child(1))')]); i < epElms.length; i += 3) yield* yieldEpisodesFromPromises(epElms.slice(i, i + 3).map(async epElm => { const epNum = epElm.querySelector('.font-medium').textContent.split(' ').pop(); status.text = `Extracting Episodes ${(epNum-Math.min(3, epNum)+1)} - ${epNum}...`; const servers = await fetch(`https://backend.animetsu.cc/api/anime/servers?id=${id}&num=${epNum}`).then(r => r.json()); const links = Object.fromEntries((await Promise.allSettled(servers.flatMap(srv => { if (!_$('button[disabled]').textContent.includes(srv.id)) return []; // process only selected server return ['sub', ...(srv.hasDub ? ['dub'] : [])].map(async subType => fetch(`https://backend.animetsu.cc/api/anime/tiddies?server=${srv.id}&id=${id}&num=${epNum}&subType=${subType}`).then(r => r.json()) .then(data => data.sources.map(src => [`${srv.id}-${subType}-${src.quality}`, { stream: src.url, type: 'm3u8', tracks: data.subtitles?.map(s => ({ file: s.url, label: s.lang, kind: 'caption' })) || [] }])) .catch(e => { showToast(`Failed to fetch Ep ${epNum} from ${srv.id}-${subType}: ${e.message || e}`); return []; }) ); }))).flatMap(r => r.status === 'fulfilled' ? r.value : [])); return new Episode(epNum, _$('.cover + div span').textContent, links, epElm.querySelector('img')?.src || '', epElm.querySelector('.text-sm').textContent); })); } }, { name: 'AnimeKai', url: ['animekai.to/', 'animekai.ac/', 'animekai.cc/', 'anikai.to/'], _chunkSize: 12, addStartButton: function (id) { setInterval(() => { if ($('#' + id).get(0)) return; const button = Object.assign(document.createElement('button'), { id, className: "btn btn-primary", textContent: "Extract Episode Links" }); const target = document.querySelector('.episode-section'); if (target) target.appendChild(button); else document.querySelector('.eplist-nav')?.appendChild(button); button.addEventListener('click', extractEpisodes); }, 500); }, extractEpisodes: async function* (status) { status.text = 'Fetching episode list...'; // const epItems = await applyEpisodeRangeFilter($('a[num]').get().map(e=> ({id: e.getAttribute('token'), num: e.getAttribute('num'), type: e.getAttribute('langs'), name: e.querySelector('span').textContent}))); const epElms = await applyEpisodeRangeFilter($('a[num]').get()); for (let i = 0; i < epElms.length; i += this._chunkSize) yield* yieldEpisodesFromPromises(epElms.slice(i, i + this._chunkSize).map(async ep => { const epNum = ep.getAttribute('num'); status.text = `Extracting Episodes ${(epNum-Math.min(this._chunkSize, epNum)+1)} - ${epNum}...`; const servers = await fetch(`/ajax/links/list?token=${ep.getAttribute('token')}&_=${await this._decode(ep.getAttribute('token'))}`).then(r => r.json().then(d => d.result)).then(t => (new DOMParser()).parseFromString(t, 'text/html')) .then(doc => $(doc).find('.server').map((i, e) => ({ lid: e.dataset.lid, name: `${this._typeSuffix(e.closest('div').dataset.id)} - ${e.textContent}` })).get()) .catch(e => showToast(`Failed to fetch servers for Ep ${epNum}`)); const links = {}; await Promise.all(servers.map(async s => { links[s.name] = await fetch(`/ajax/links/view?id=${s.lid}&_=${await this._decode(s.lid)}`).then(r => r.json().then(d => d.result)) .then(val => this._decode(val, 'd').then(JSON.parse)).then(async d => await Extractors.use(d.url)) .catch(e => showToast(`Failed to fetch Ep ${epNum} from ${s.name}: ${e.message || e}`)) })); return new Episode(epNum, $('h1').text(), links, $('.poster-wrap-bg').attr('style').match(/https.*\.[a-z]+/g)[0], ep.querySelector('span').textContent); })) }, _decode: async (s, t = 'e') => await GM_fetch(`https://c-kai-8090.amarullz.com/?f=${t}&d=${s}`).then(r => r.text()), _typeSuffix: type => ({ sub: "Hard Sub", softsub: "Soft Sub", dub: "Dub & S-Sub" }[type] || type) } ]; const USER_AGENT_HEADER = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.0.0 Safari/537.36 Edg/141.0.0.0"; const Extractors = { use: function (url, ...args) { const extractor = this[(new URL(url)).host]; if (!extractor) throw new Error(`No extractor found for ${url}`); return extractor(url, ...args); }, 'kwik.cx': async function (kwikUrl, referer = location.href) { const response = await fetch(kwikUrl, { headers: { referer } }); const data = await response.text(); return eval(/(eval)(\(f.*?)(\n<\/script>)/s.exec(data)[2].replace("eval", "")).match(/https.*?m3u8/)[0]; }, 'megaplay.buzz': async function (embed, referer) { referer = referer || 'https://megaplay.buzz/'; const id = await fetch(embed, { headers: { Referer: referer } }).then(r=>r.text()).then(t => t.match(/<title>File ([0-9]+)/)[1]); const src = await GM_fetch('https://megaplay.buzz/stream/getSources?id=' + id, { headers: { 'X-Requested-With': 'XMLHttpRequest' } }).then(e => e.json()) return { file: src.sources?.file, type: 'm3u8', tracks: src.tracks || []} }, 'megacloud.blog': async function (embed, referer) { // adapted from https://github.com/yuzono/aniyomi-extensions/blob/master/lib/megacloud-extractor/src/main/java/eu/kanade/tachiyomi/lib/megacloudextractor/MegaCloudExtractor.kt const res = await GM_fetch(embed, { headers: { referer, 'User-Agent': USER_AGENT_HEADER } }); const retryAfter = res.headers.get('Retry-After'); // Rate limit Policy: 10 requests per minute if (retryAfter) { const hhmmss = new Date(new Date().getTime() + parseInt(retryAfter) * 1000).toLocaleTimeString([], { hour: '2-digit', minute: '2-digit', second: '2-digit', hour12: true }); showToast(`Rate limited by megacloud.blog, retrying in ${retryAfter} seconds (at ${hhmmss})...`, parseInt(retryAfter) * 1000); return await new Promise(res => setTimeout(res, 500 + parseInt(retryAfter) * 1000)).then(() => Extractors['megacloud.blog'](embed, referer)); // recursive retry } const html = await res.text(); const match1 = html.match(/\b[a-zA-Z0-9]{48}\b/), match2 = html.match(/\b([a-zA-Z0-9]{16})\b.*?\b([a-zA-Z0-9]{16})\b.*?\b([a-zA-Z0-9]{16})\b/); const nonce = match1?.[0] || (match2 ? match2[1] + match2[2] + match2[3] : null); if (!nonce) throw new Error('Failed to extract nonce from response'); const sId = embed.split('/e-1/')[1]?.split('?')[0]; const origin = (new URL(embed)).origin; const url = `${origin}/embed-2/v3/e-1/getSources?id=${sId}&_k=${nonce}`; const data = await GM_fetch(url, { headers: { 'Accept': '*/*', 'X-Requested-With': 'XMLHttpRequest', 'Referer': origin+'/' } }).then(r => r.json()); if (!data.encrypted || data.sources[0].file.includes('.m3u8')) return { file: data.sources[0].file, type: data.sources[0].type, tracks: data.tracks || [], referer: origin+'/' }; const secret = await fetch('https://raw.githubusercontent.com/yogesh-hacker/MegacloudKeys/refs/heads/main/keys.json').then(r => r.json()).then(j => j['mega']); const decryptUrl = `https://megacloud-api-nine.vercel.app/?encrypted_data=${encodeURIComponent(data.sources[0].file)}&nonce=${encodeURIComponent(nonce)}&secret=${encodeURIComponent(secret)}`; const decrypted = await GM_fetch(decryptUrl).then(r => r.text()); const m3u8 = decrypted.match(/"file":"(.*?)"/)?.[1]; if (!m3u8) throw new Error('Video URL not found in decrypted response'); return { file: m3u8, type: 'hls', tracks: data.tracks || [] }; }, 'gofile.io': async function (url) { const id = url.split('/').pop(); const stored = JSON.parse(localStorage.gofile_token || '{}'); let token = stored.token; if (!token || Date.now() - stored.timestamp > 604800000) { if (token !== 'fetching') { localStorage.gofile_token = JSON.stringify({ token: 'fetching', timestamp: Date.now() }); token = (await GM_fetch('https://api.gofile.io/accounts', { method: 'POST', body: '{}' }).then(r => r.json())).data.token; localStorage.gofile_token = JSON.stringify({ token, timestamp: Date.now() }); } else { while ((token = JSON.parse(localStorage.gofile_token || '{}').token) === 'fetching') await new Promise(r => setTimeout(r, 500)); } } const data = await GM_fetch(`https://api.gofile.io/contents/${id}?wt=4fd6sg89d7s6`, { headers: { Authorization: `Bearer ${token}` } }).then(r => r.json()); if (data.status !== 'ok') throw new Error(data.status); const file = Object.values(data.data.children || {}).find(f => f.name?.endsWith('.m3u8') || f.mimetype?.startsWith('video/')); if (!file) throw new Error('No video file found'); return { file: file.link, type: file.name?.endsWith('.m3u8') ? 'm3u8' : 'mp4', tracks: [] }; }, 'vkspeed.com': async function(url) { const html = await GM_fetch(url).then(r => r.text()); const [, e, r, c, d] = html.match(/eval\(function\(p,a,c,k,e,d\)\{while\(c--\)if\(k\[c\]\)p=p\.replace\(new RegExp\('\\\\b'\+c\.toString\(a\)\+'\\\\b','g'\),k\[c\]\);return p\}\('(.+?)',(\d+),(\d+),'(.+?)'\.split\('\|'\)\)\)/) || []; if (!e) throw new Error('No packed script found'); let decoded = e; const dict = d.split('|'); for (let i = +c - 1; i >= 0; i--) if (dict[i]) decoded = decoded.replace(new RegExp('\\b' + i.toString(+r) + '\\b', 'g'), dict[i]); const sources = eval(decoded.match(/sources:\[.*?\]/)[0]); const source = sources.reduce((best, curr) => (s => parseInt(s.label) || 0)(curr) > (s => parseInt(s.label) || 0)(best) ? curr : best, sources[0]); return { file: source.file, type: source.file.includes('.m3u8') ? 'm3u8' : 'mp4', tracks: [] }; }, 'megaup.live': async function(url, referer='https://megaup.live/') { // workaround: use GM_xmlhttpRequest to avoid passing cookies (coudnt do that with GM_fetch) const encToken = await new Promise((r, j) => GM_xmlhttpRequest({ method: 'GET', url: url.replace('/e/', '/media/'), headers: { 'User-Agent': USER_AGENT_HEADER }, anonymous: true, onload: res => { try { r(JSON.parse(res.responseText).result); } catch (e) { j(e); } }, onerror: j })); const src = (await GM_fetch('https://enc-dec.app/api/dec-mega', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({text: encToken, agent: USER_AGENT_HEADER}) }).then(r => r.json())).result; return { stream: src.sources[0].file, type: 'm3u8', tracks: src.tracks?.map(t => ({ file: t.file, label: t.label, kind: t.kind, default: !!t.default })), referer: 'https://megaup.cc/' }; } } /** * Fetches the HTML content of a given URL and parses it into a DOM object. * * @param {string} url - The URL of the page to fetch. * @returns {Promise<Document>} A promise that resolves to a DOM Document object. * @throws {Error} If the fetch operation fails. */ async function fetchPage(url, options = {}) { const response = await fetch(url, options); if (response.ok) { const page = (new DOMParser()).parseFromString(await response.text(), 'text/html'); return page; } else { showToast(`Failed to fetch HTML for ${url} : ${response.status}`); throw new Error(`Failed to fetch HTML for ${url} : ${response.status}`); } } /** * Fetches a URL with retry logic for handling rate limits or temporary errors. * * @returns {Promise<Response>} A promise that resolves to the response object. */ async function fetchWithRetry(url, options = {}, retries = 3, sleep = 1000) { const response = await fetch(url, options); if (!response.ok) { if (response.status === 503 && retries > 0) { // 503 is a common status when rate limited console.log(`Retrying ${url}, ${retries} retries remaining`); await new Promise(resolve => setTimeout(resolve, sleep)); // Wait 1 second before retrying return fetchWithRetry(url, options, retries - 1, sleep); // Pass options and sleep to the next call } throw new Error(`${response.status} - ${response.statusText}`); } return response; } /** * Asynchronously processes an array of episode promises and yields each resolved episode. * * @param {Array<Promise>} episodePromises - An array of promises, each resolving to an episode. * @returns {AsyncGenerator} An async generator yielding each resolved episode. */ async function* yieldEpisodesFromPromises(episodePromises) { for (const episodePromise of episodePromises) { const episode = await episodePromise; if (episode) { yield episode; } } } /** * encodes a string to base64url format thats safe for URLs */ const safeBtoa = str => btoa(unescape(encodeURIComponent(str))).replace(/\+/g, '-').replace(/\//g, '_').replace(/=+$/, ''); /** * Analyzes the given media url to return duration, size, and resolution of the media. * @param {string} mediaUrl - The URL of the media to analyze. * @return {Promise<{duration: string, size: string, resolution: string}>} A promise that resolves to an object * containing duration (in hh:mm:ss), size of the media (in MB), and resolution (e.g., 1920x1080). * @TODO: Not Yet Implemented */ async function analyzeMedia(mediaUrl) { if (_analyzedMediaCache.has(mediaUrl)) return _analyzedMediaCache.get(mediaUrl); let metadata = { duration: 'N/A', resolution: 'N/A', size: 'N/A' }; try { if (mediaUrl.endsWith('.mp4')) { const r = await GM_fetch(mediaUrl, { method: 'HEAD' }); if (r.ok) { const sz = parseFloat(r.headers.get('Content-Length')) || 0; metadata.size = `${(sz / 1048576).toFixed(2)} MB`; } } else if (mediaUrl.endsWith('.m3u8')) { const r = await GM_fetch(mediaUrl); if (r.ok) { const t = await r.text(); const res = t.match(/RESOLUTION=(\d+x\d+)/i); if (res) metadata.resolution = res[1]; let d = 0; for (const m of t.matchAll(/#EXTINF:([\d.]+)/g)) d += parseFloat(m[1]); if (d > 0) { const h = Math.floor(d / 3600), m = Math.floor((d % 3600) / 60), s = Math.floor(d % 60); metadata.duration = [h, m, s].map(v => String(v).padStart(2, '0')).join(':'); } } } if (metadata.duration === 'N/A' || metadata.resolution === 'N/A') { await new Promise(res => { const v = document.createElement('video'); v.src = mediaUrl; v.preload = 'metadata'; v.muted = true; v.onloadedmetadata = () => { if (v.duration && metadata.duration === 'N/A') { const h = Math.floor(v.duration / 3600), m = Math.floor((v.duration % 3600) / 60), s = Math.floor(v.duration % 60); metadata.duration = [h, m, s].map(x => String(x).padStart(2, '0')).join(':'); } if (v.videoWidth && v.videoHeight && metadata.resolution === 'N/A') metadata.resolution = `${v.videoWidth}x${v.videoHeight}`; res(); }; v.onerror = () => res(); setTimeout(res, 2000); }); } } catch (e) { } _analyzedMediaCache.set(mediaUrl, metadata); return metadata; } const _analyzedMediaCache = new Map(); // Cache to store analyzed media results for the above function // initialize console.log('Initializing AniLINK...'); const site = Websites.find(site => site.url.some(url => window.location.href.includes(url))); // register menu command to start script GM_registerMenuCommand('Extract Episodes', extractEpisodes); // attach start button to page try { const startBtnId = "AniLINK_startBtn"; (site.addStartButton(startBtnId) || document.getElementById(startBtnId))?.addEventListener('click', extractEpisodes); } catch (e) { console.warn('Could not add start button to site. This might be due to the function not being implemented for this site.'); } // append site specific css styles document.body.style.cssText += (site.styles || ''); /*************************************************************** * This function creates an overlay on the page and displays a list of episodes extracted from a website * The function is triggered by a user command registered with `GM_registerMenuCommand`. * The episode list is generated by calling the `extractEpisodes` method of a website object that matches the current URL. ***************************************************************/ async function extractEpisodes() { // Restore last overlay if it exists if (document.getElementById("AniLINK_Overlay")) { document.getElementById("AniLINK_Overlay").style.display = "flex"; return; } // Flag to control extraction process let status = { isExtracting: true, text: 'Initializing...', stopped: false, error: null }; // --- Materialize CSS Initialization --- GM_addStyle(` @import url('https://fonts.googleapis.com/icon?family=Material+Icons'); #AniLINK_Overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background-color: rgba(0, 0, 0, 0.8); z-index: 1000; display: flex; align-items: center; justify-content: center; } #AniLINK_LinksContainer { width: 80%; max-height: 85%; background-color: #222; color: #eee; padding: 20px; border-radius: 8px; overflow-y: auto; display: flex; flex-direction: column;} .anlink-status-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 10px; } /* Header for status bar and stop button */ .anlink-status-bar { color: #eee; flex-grow: 1; margin-right: 10px; display: block; } /* Status bar takes space */ .anlink-status-icon { background: transparent; border: none; color: #eee; cursor: pointer; padding-right: 10px; } /* status icon style */ .anlink-status-icon i { font-size: 24px; transition: transform 0.3s ease-in-out; } /* Icon size and transition */ .anlink-status-icon i::before { content: 'check_circle'; } /* Show check icon when not extracting */ .anlink-status-icon i.extracting::before { content: 'auto_mode'; animation: spinning 2s linear infinite; } /* Spinner animation class */ .anlink-status-icon i.retry::before { content: 'refresh'; } /* Retry icon */ .anlink-status-icon i.error::before { content: 'error'; } /* Error icon */ .anlink-status-icon:hover i.extracting::before { content: 'stop_circle'; animation: stop; } /* Show stop icon on hover when extracting */ .anlink-header-buttons { display: flex; gap: 10px; } .anlink-header-buttons button { background-color: #26a69a; color: white; border: none; padding: 8px 15px; border-radius: 4px; cursor: pointer; } .anlink-header-buttons button:hover { background-color: #2bbbad; } .anlink-quality-section { margin-top: 20px; margin-bottom: 10px; border-bottom: 1px solid #444; padding-bottom: 5px; } .anlink-quality-header { display: flex; justify-content: space-between; align-items: center; } .anlink-quality-header > span { color: #26a69a; font-size: 1.5em; display: flex; align-items: center; flex-grow: 1; } /* Flex and align items for icon and text */ .anlink-quality-count { cursor: pointer; margin-right: 8px; opacity: 0.7; transition: opacity 0.2s; } .anlink-quality-count:hover { opacity: 1; } .anlink-quality-name { cursor: pointer; flex-grow: 1; } .anlink-quality-header i { margin-right: 8px; transition: transform 0.3s ease-in-out; } .anlink-quality-header i.rotate { transform: rotate(90deg); } /* Rotate class */ .anlink-episode-list { list-style: none; padding-left: 0; margin-top: 0; overflow: hidden; transition: max-height 0.5s ease-in-out; } /* Transition for max-height */ .anlink-episode-item { margin-bottom: 5px; padding: 8px; border-bottom: 1px solid #333; display: flex; flex-direction: column; } .anlink-episode-item:last-child { border-bottom: none; } .anlink-episode-main { display: flex; align-items: center; } .anlink-episode-main > label { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } /* Single line & Ellipsis for long links */ .anlink-episode-main > label > span { user-select: none; cursor: pointer; color: #26a69a; } /* Disable selecting the 'Ep: 1' prefix */ .anlink-episode-main > label > span > img { vertical-align: middle; display: inline; } /* Ensure the mpv icon is in the same line */ .anlink-episode-checkbox { appearance: none; width: 20px; height: 20px; margin-right: 10px; margin-bottom: -5px; border: 1px solid #26a69a; border-radius: 4px; outline: none; cursor: pointer; transition: background-color 0.3s, border-color 0.3s; } .anlink-episode-checkbox:checked { background-color: #26a69a; border-color: #26a69a; } .anlink-episode-checkbox:checked::after { content: '✔'; display: block; color: white; font-size: 14px; text-align: center; line-height: 20px; animation: checkTilt 0.3s; } .anlink-episode-link { color: #ffca28; text-decoration: none; display: inline; } .anlink-episode-link:hover { color: #fff; } .anlink-subs-toggle { font-size: 0.85em; color: #888; cursor: pointer; margin-left: 10px; user-select: none; transition: color 0.2s; white-space: nowrap; } .anlink-subs-toggle:hover { color: #26a69a; } .anlink-subs-list { margin-left: 30px; margin-top: 5px; font-size: 0.9em; color: #bbb; max-height: 0; overflow: hidden; transition: max-height 0.3s ease-in-out; } .anlink-subs-list.expanded { max-height: 300px; } .anlink-sub-item { padding: 2px 0; width: max-content; user-select: none; } .anlink-sub-item a { color: #64b5f6; text-overflow: ellipsis; overflow: hidden; display: inline; user-select: text; } .anlink-sub-item a:hover { color: #90caf9; text-decoration: underline; } @keyframes spinning { from { transform: rotate(0deg); } to { transform: rotate(360deg); } } /* Spinning animation */ @keyframes checkTilt { from { transform: rotate(-20deg); } to { transform: rotate(0deg); } } /* Checkmark tilt animation */ `); // Create an overlay to cover the page const overlayDiv = document.createElement("div"); overlayDiv.id = "AniLINK_Overlay"; document.body.appendChild(overlayDiv); overlayDiv.onclick = e => !linksContainer.contains(e.target) && (document.querySelector('.anlink-status-bar')?.textContent.startsWith("Cancelled") ? overlayDiv.remove() : overlayDiv.style.display = "none"); // Create a container for links const linksContainer = document.createElement('div'); linksContainer.id = "AniLINK_LinksContainer"; overlayDiv.appendChild(linksContainer); // Status bar header - container for status bar and status icon const statusBarHeader = document.createElement('div'); statusBarHeader.className = 'anlink-status-header'; linksContainer.appendChild(statusBarHeader); // Create dynamic status icon const statusIconElement = document.createElement('a'); statusIconElement.className = 'anlink-status-icon'; statusIconElement.innerHTML = '<i class="material-icons extracting"></i>'; statusIconElement.title = 'Stop Extracting'; statusBarHeader.appendChild(statusIconElement); statusIconElement.addEventListener('click', () => { if (status.stopped) return; // TODO: add retry functionality with continuing with past links status = { isExtracting: false, text: "Extraction Stopped by User.", stopped: true }; }); // Create a status bar const statusBar = document.createElement('span'); statusBar.className = "anlink-status-bar"; statusBar.textContent = status.text; statusBarHeader.appendChild(statusBar); // Create header buttons (Export & Play) const headerButtons = document.createElement('div'); headerButtons.className = 'anlink-header-buttons'; headerButtons.innerHTML = ` <button type="button" class="anlink-export-all">Export</button> <button type="button" class="anlink-play-all">Play with MPV</button> `; statusBarHeader.appendChild(headerButtons); // start interval to update status text const statusInterval = setInterval(() => { if (JSON.stringify(status) !== JSON.stringify(_lastStatus)) { _lastStatus = { ...status }; statusBar.textContent = status.text; if (status.isExtracting) { statusIconElement.querySelector('i').classList.add('extracting'); // Start spinner animation } else { statusIconElement.title = 'Restart Extraction.'; statusIconElement.querySelector('i').classList.remove('extracting'); // Stop spinner animation if (status.stopped) { statusIconElement.querySelector('i').classList.add('retry'); // Show retry icon } if (status.error) { statusIconElement.querySelector('i').classList.add('error'); // Show error icon statusBar.textContent += ` : ${status.error}`; // Update status bar with error statusBar.style.color = 'red'; // Change status bar color to red } else statusBar.style.color = ''; // Reset status bar color clearInterval(statusInterval); // Stop updating statusBar } } }, 100); _lastStatus = { ...status }; // Store a shallow copy of the last status for reference // Create a container for qualities and episodes const qualitiesContainer = document.createElement('div'); qualitiesContainer.id = "AniLINK_QualitiesContainer"; linksContainer.appendChild(qualitiesContainer); // Update counts on checkbox change (event delegation) qualitiesContainer.addEventListener('change', e => { if (e.target.classList.contains('anlink-episode-checkbox')) { const section = e.target.closest('.anlink-quality-section'); const total = section.querySelectorAll('.anlink-episode-checkbox').length; const checked = section.querySelectorAll('.anlink-episode-checkbox:checked').length; section.querySelector('.anlink-quality-count').textContent = checked ? `(${checked}/${total})` : `(${total})`; } }); // --- Process Episodes using Generator --- window._anilink_episodes = []; try { const episodeGenerator = site.extractEpisodes(status); const qualityLinkLists = {}; const startTime = Date.now(); for await (const episode of episodeGenerator) { if (!status.isExtracting) { // Check if extraction is stopped statusIconElement.querySelector('i').classList.remove('extracting'); // Stop spinner animation return; // Exit if extraction is stopped } if (!episode) continue; window._anilink_episodes.push(episode); // Get all links into format - {[qual1]:[ep1,2,3,4], [qual2]:[ep1,2,3,4], ...} for (const quality in episode.links) { qualityLinkLists[quality] = qualityLinkLists[quality] || []; qualityLinkLists[quality].push(episode); } // Update UI in real-time - RENDER UI HERE BASED ON qualityLinkLists renderQualityLinkLists(qualityLinkLists, qualitiesContainer); } const duration = ((Date.now() - startTime) / 1000).toFixed(2); statusIconElement.querySelector('i').classList.remove('extracting'); if (qualityLinkLists && Object.keys(qualityLinkLists).length > 0) { status = { isExtracting: false, text: `Extraction Complete in ${duration} seconds` }; } else { status = { isExtracting: false, text: "No episodes found." }; } } catch (error) { console.error('Error during episode extraction:', error); status = { isExtracting: false, text: `Extraction Failed after ${duration} seconds.`, error: error.message || error.toString() }; } // Renders quality link lists inside a given container element function renderQualityLinkLists(sortedLinks, container) { // Track expanded state for each quality section const expandedState = {}; container.querySelectorAll('.anlink-quality-section').forEach(section => { const quality = section.dataset.quality; const episodeList = section.querySelector('.anlink-episode-list'); expandedState[quality] = episodeList && episodeList.style.maxHeight !== '0px'; }); for (const quality in sortedLinks) { let qualitySection = container.querySelector(`.anlink-quality-section[data-quality="${quality}"]`); let episodeListElem; const episodes = sortedLinks[quality].sort((a, b) => a.number - b.number); if (!qualitySection) { // Create new section if it doesn't exist qualitySection = document.createElement('div'); qualitySection.className = 'anlink-quality-section'; qualitySection.dataset.quality = quality; const headerDiv = document.createElement('div'); headerDiv.className = 'anlink-quality-header'; headerDiv.title = 'Shift+Click to select/deselect all episodes in this quality'; const qualitySpan = document.createElement('span'); const count = document.createElement('i'); count.className = 'anlink-quality-count'; count.textContent = `(${sortedLinks[quality].length})`; count.title = 'Click to select/deselect all'; count.dataset.total = sortedLinks[quality].length; count.addEventListener('click', e => { e.stopPropagation(); toggleSelectAll(qualitySection); }); const icon = document.createElement('i'); icon.className = 'material-icons'; icon.textContent = 'chevron_right'; const name = document.createElement('span'); name.className = 'anlink-quality-name'; name.textContent = quality; name.addEventListener('click', toggleQualitySection); qualitySpan.appendChild(count); qualitySpan.appendChild(icon); qualitySpan.appendChild(name); headerDiv.appendChild(qualitySpan); qualitySection.appendChild(headerDiv); // --- Add Empty episodes list elm to the quality section --- episodeListElem = document.createElement('ul'); episodeListElem.className = 'anlink-episode-list'; episodeListElem.style.maxHeight = '0px'; qualitySection.appendChild(episodeListElem); container.appendChild(qualitySection); // Shift+Click to select all episodes in this quality headerDiv.addEventListener('mousedown', e => e.shiftKey && _$$('.anlink-episode-checkbox').forEach(cb => cb.checked = !cb.checked)); } else { // Update header count const countElem = qualitySection.querySelector('.anlink-quality-count'); if (countElem) { const checked = qualitySection.querySelectorAll('.anlink-episode-checkbox:checked').length; countElem.textContent = checked ? `(${checked}/${sortedLinks[quality].length})` : `(${sortedLinks[quality].length})`; countElem.dataset.total = sortedLinks[quality].length; } episodeListElem = qualitySection.querySelector('.anlink-episode-list'); } // Update episode list items episodeListElem.innerHTML = ''; episodes.forEach(ep => { const listItem = document.createElement('li'); listItem.className = 'anlink-episode-item'; const hasSubs = ep.links[quality].tracks?.some(t => /^(caption|subtitle)s?/.test(t.kind)); listItem.innerHTML = ` <div class="anlink-episode-main"> <label> <input type="checkbox" class="anlink-episode-checkbox" /> <span class="mpv-epnum" title="Play in MPV">Ep ${ep.number.replace(/^0+/, '')}: </span> <a href="${ep.links[quality].stream}" class="anlink-episode-link" download="${encodeURI(ep.filename)}" data-epnum="${ep.number}" data-ep=${encodeURI(JSON.stringify({ ...ep, links: undefined }))} >${ep.links[quality].stream}</a> </label> ${hasSubs ? '<span class="anlink-subs-toggle" title="Shift+Click to toggle all episodes\' subtitles">🄰 Subs ▼</span>' : ''} </div> ${hasSubs ? '<div class="anlink-subs-list"></div>' : ''} `; const episodeLinkElement = listItem.querySelector('.anlink-episode-link'); const epnumSpan = listItem.querySelector('.mpv-epnum'); const link = episodeLinkElement.href; const name = decodeURIComponent(episodeLinkElement.download); // On hover, show MPV icon & file name listItem.addEventListener('mouseenter', () => { window.getSelection().isCollapsed && (episodeLinkElement.textContent = name); epnumSpan.innerHTML = `<img width="20" height="20" fill="#26a69a" src="https://a.fsdn.com/allura/p/mpv-player-windows/icon?1517058933"> ${ep.number.replace(/^0+/, '')}: `; }); listItem.addEventListener('mouseleave', () => { episodeLinkElement.textContent = decodeURIComponent(link); epnumSpan.textContent = `Ep ${ep.number.replace(/^0+/, '')}: `; }); epnumSpan.addEventListener('click', e => { e.preventDefault(); location.replace('mpv-handler://play/' + safeBtoa(link) + `/?v_title=${safeBtoa(name)}&cookies=${location.hostname}.txt` + (ep.links[quality].tracks?.some(t => t.kind === 'caption') ? `&subfile=${safeBtoa(ep.links[quality].tracks.filter(t => /^caption/.test(t.kind)).map(t => t.file).join(';'))}` : '')); showToast('Sent to MPV. If nothing happened, install v0.4.0+ of <a href="https://github.com/akiirui/mpv-handler" target="_blank" style="color:#1976d2;">mpv-handler</a>.'); }); episodeLinkElement.addEventListener('click', () => { fetch(episodeLinkElement.href) .then(r => r.blob()) .then(b => Object.assign(document.createElement('a'), { href: URL.createObjectURL(b), download: decodeURIComponent(episodeLinkElement.download) }).click()); // workaround to force download with correct filename (some browsers ignore download attr for cross-origin links) }); // Subtitle toggle functionality const subsToggle = listItem.querySelector('.anlink-subs-toggle'); const subsList = listItem.querySelector('.anlink-subs-list'); if (subsToggle && subsList) { subsToggle.addEventListener('mousedown', e => { // shift+click to toggle all episode subtitles if (e.shiftKey) { return document.querySelectorAll('.anlink-subs-list').forEach(sl => sl.previousElementSibling.querySelector('.anlink-subs-toggle').dispatchEvent(new MouseEvent('mousedown', { bubbles: false }))); } const isExpanded = subsList.classList.toggle('expanded'); subsToggle.textContent = isExpanded ? '🄰 Subs ▲' : '🄰 Subs ▼'; if (isExpanded && !subsList.hasChildNodes()) { ep.links[quality].tracks.filter(t => /^caption/.test(t.kind)).forEach(track => { const subItem = document.createElement('div'); subItem.className = 'anlink-sub-item'; subItem.innerHTML = `└─ ${track.label || 'Subtitle'}: <a href="${track.file}" target="_blank">${track.file}</a>`; subsList.appendChild(subItem); }); } const epList = subsList.closest('.anlink-episode-list'); epList.style.maxHeight = +epList.style.maxHeight.replace('px','') + subsList.scrollHeight + 'px'; // Adjust max-height to fit new content }); } episodeListElem.appendChild(listItem); // Fix checkbox state double toggling due to label click (listItem.querySelector('.anlink-episode-checkbox')).onclick = e => e.stopPropagation(); }); // Restore expand state only if section was previously expanded if (expandedState[quality]) { const icon = qualitySection.querySelector('.material-icons'); episodeListElem.style.maxHeight = `${episodeListElem.scrollHeight}px`; icon.classList.add('rotate'); } } } function toggleQualitySection(event) { const qualityName = event.currentTarget; const qualitySection = qualityName.closest('.anlink-quality-section'); const episodeList = qualitySection.querySelector('.anlink-episode-list'); const icon = qualitySection.querySelector('.material-icons'); const isCollapsed = episodeList.style.maxHeight === '0px'; if (isCollapsed) { episodeList.style.maxHeight = `${episodeList.scrollHeight}px`; // Expand to content height with animation icon.classList.add('rotate'); // Rotate icon on expand } else { episodeList.style.maxHeight = '0px'; // Collapse icon.classList.remove('rotate'); // Reset icon rotation } } function toggleSelectAll(qualitySection) { const checkboxes = Array.from(qualitySection.querySelectorAll('.anlink-episode-checkbox')); const allChecked = checkboxes.every(cb => cb.checked); checkboxes.forEach(cb => cb.checked = !allChecked); // also select all the text if (!allChecked) { const range = document.createRange(); range.selectNodeContents(qualitySection.querySelector('ul')); const sel = window.getSelection(); sel.removeAllRanges(); sel.addRange(range); } } // Attach header button handlers (function attachHeaderButtons() { const exportBtn = linksContainer.querySelector('.anlink-export-all'); const playBtn = linksContainer.querySelector('.anlink-play-all'); exportBtn.addEventListener('click', () => onExportAll(exportBtn)); playBtn.addEventListener('click', () => onPlayAll(playBtn)); })(); // Helper to get all selected episodes across all qualities function getAllSelectedEpisodes() { const selected = {}; document.querySelectorAll('.anlink-quality-section').forEach(section => { const quality = section.dataset.quality; const items = Array.from(section.querySelectorAll('.anlink-episode-item input:checked')) .map(cb => cb.closest('.anlink-episode-item')); if (items.length) selected[quality] = items; }); return selected; } // Helper to prepare m3u8 playlist string function preparePlaylist(episodes, quality) { let content = '#EXTM3U\n'; const referer = Object.values(episodes[0]?.links)[0]?.referer; if (referer) content += `#EXTVLCOPT:http-referrer=${referer}\n`; episodes.forEach(ep => { const link = ep.links[quality]; if (!link) return; if (link.tracks?.length) { link.tracks.forEach(t => { const type = t.kind?.startsWith('audio') ? 'AUDIO' : /^(caption|subtitle)s?/.test(t.kind) ? 'SUBTITLES' : null; if (type) content += `#EXT-X-MEDIA:TYPE=${type},GROUP-ID="${type.toLowerCase()}${ep.number}",NAME="${t.label || type}",DEFAULT=${t.default ? 'YES' : 'NO'},URI="${t.file}"\n`; }); } // content += `#EXT-X-STREAM-INF:BANDWIDTH=0,RESOLUTION=0x0,CODECS="mp4a.40.2,avc1.42E01E"${link.tracks?.length ? `,AUDIO="audio${ep.number}",SUBTITLES="subtitles${ep.number}"` : ''}\n`; // commented out cuz ffmpeg (used by mpv) doesnt have https:// on its whitelist for EXT-X-MEDIA lines content += `#EXTINF:-1,${ep.filename.replaceAll('/', '|')}\n${link.stream}\n`; }); return content; } async function onExportAll(btn) { const selected = getAllSelectedEpisodes(); if (!Object.keys(selected).length) return showToast('No episodes selected'); let allContent = '#EXTM3U\n'; const qualities = Object.keys(selected).join(', '); for (const [quality, items] of Object.entries(selected)) { const epNums = items.map(i => i.querySelector('[data-epnum]').dataset.epnum); const episodes = (window._anilink_episodes || []).filter(ep => ep.links[quality] && epNums.includes(ep.number)); const referer = episodes[0]?.links[quality]?.referer; if (referer && !allContent.includes(referer)) allContent += `#EXTVLCOPT:http-referrer=${referer}\n`; episodes.forEach(ep => { const link = ep.links[quality]; if (link?.tracks?.length) link.tracks.forEach(t => { const type = t.kind?.startsWith('audio') ? 'AUDIO' : /^(caption|subtitle)s?/.test(t.kind) ? 'SUBTITLES' : null; if (type) allContent += `#EXT-X-MEDIA:TYPE=${type},GROUP-ID="${type.toLowerCase()}${ep.number}",NAME="${t.label || type}",DEFAULT=${t.default ? 'YES' : 'NO'},URI="${t.file}"\n`; }); allContent += `#EXTINF:-1,${ep.filename.replaceAll('/', '|')}${GM_getValue('include_source_in_filename', true) ? ` [${quality}]` : ''}\n${link.stream}\n`; }); } const fileName = (window._anilink_episodes?.[0]?.animeTitle || 'Anime') + (Object.keys(selected).length > 1 ? ` [${qualities}]` : `${GM_getValue('include_source_in_filename', true) ? ` [${qualities}]` : ''}`) + '.m3u8'; Object.assign(document.createElement('a'), { href: URL.createObjectURL(new Blob([allContent], { type: 'application/vnd.apple.mpegurl' })), download: fileName }).click(); btn.textContent = 'Exported'; setTimeout(() => btn.textContent = 'Export', 1000); } async function onPlayAll(btn) { const selected = getAllSelectedEpisodes(); if (!Object.keys(selected).length) return showToast('No episodes selected'); btn.textContent = 'Processing...'; let allContent = '#EXTM3U\n'; for (const [quality, items] of Object.entries(selected)) { const epNums = items.map(i => i.querySelector('[data-epnum]').dataset.epnum); const episodes = (window._anilink_episodes || []).filter(ep => ep.links[quality] && epNums.includes(ep.number)); const referer = episodes[0]?.links[quality]?.referer; if (referer && !allContent.includes(referer)) allContent += `#EXTVLCOPT:http-referrer=${referer}\n`; episodes.forEach(ep => { const link = ep.links[quality]; if (link?.tracks?.length) link.tracks.forEach(t => { const type = t.kind?.startsWith('audio') ? 'AUDIO' : /^(caption|subtitle)s?/.test(t.kind) ? 'SUBTITLES' : null; if (type) allContent += `#EXT-X-MEDIA:TYPE=${type},GROUP-ID="${type.toLowerCase()}${ep.number}",NAME="${t.label || type}",DEFAULT=${t.default ? 'YES' : 'NO'},URI="${t.file}"\n`; }); allContent += `#EXTINF:-1,${ep.filename.replaceAll('/', '|')}${GM_getValue('include_source_in_filename', true) ? ` [${quality}]` : ''}\n${link.stream}\n`; }); } // Use mpv-handler:// protocol to pass the paste.rs link to mpv (requires mpv-handler installed) const url = await GM_fetch('https://paste.rs/', { method: 'POST', body: allContent }).then(r => r.text()).then(t => t + '.m3u8'); console.log(`Playlist URL:`, url); location.replace('mpv-handler://play/' + safeBtoa(url) + '/?v_title=' + safeBtoa((window._anilink_episodes?.[0]?.animeTitle || 'Anime'))); btn.textContent = 'Sent to MPV'; setTimeout(() => { btn.textContent = 'Play with MPV'; showToast('If nothing happened, install v0.4.0+ of <a href="https://github.com/akiirui/mpv-handler" target="_blank" style="color:#1976d2;">mpv-handler</a>.'); }, 1000); } } /*************************************************************** * Modern Episode Range Selector with Keyboard Navigation ***************************************************************/ async function showEpisodeRangeSelector(total) { return new Promise(resolve => { const modal = Object.assign(document.createElement('div'), { innerHTML: ` <div class="anlink-modal-backdrop"> <div class="anlink-modal"> <div class="anlink-modal-header"> <div class="anlink-modal-icon">📺</div> <h2>Episode Range</h2> <div class="anlink-episode-count">${total} episodes found</div> <small style="display:block;color:#ccc;font-size:11px;margin-top:2px;"> Note: Range is by episode count, not episode number<br>(e.g., 1-6 means the first 6 episodes listed). </small> </div> <div class="anlink-modal-body"> <div class="anlink-range-inputs"> <div class="anlink-input-group"> <label>From</label> <input type="number" id="start" min="1" max="${total}" value="1" tabindex="1"> </div> <div class="anlink-range-divider">—</div> <div class="anlink-input-group"> <label>To</label> <input type="number" id="end" min="1" max="${total}" value="${Math.min(24, total)}" tabindex="2"> </div> </div> <div class="anlink-quick-select"> <button class="anlink-quick-btn" data-range="1,24" tabindex="3">First 24</button> <button class="anlink-quick-btn" data-range="${Math.max(1, total - 23)},${total}" tabindex="4">Last 24</button> <button class="anlink-quick-btn" data-range="1,${total}" tabindex="5">All ${total}</button> </div> <div class="anlink-help-text"> Use <kbd>Tab</kbd> to navigate • <kbd>↑↓</kbd> to adjust values • <kbd>Enter</kbd> to extract • <kbd>Esc</kbd> to cancel </div> </div> <div class="anlink-modal-footer"> <button class="anlink-btn anlink-btn-cancel" data-key="Escape" tabindex="6"><kbd>Esc</kbd> Cancel</button> <button class="anlink-btn anlink-btn-primary" data-key="Enter" tabindex="7"><kbd>Enter</kbd> Extract</button> </div> </div> </div> `, style: 'position:fixed;top:0;left:0;width:100%;height:100%;z-index:1001;' }); // Enhanced styling with keyboard indicators GM_addStyle(` .anlink-modal-backdrop { display: flex; align-items: center; justify-content: center; width: 100%; height: 100%; background: rgba(0,0,0,0.8); backdrop-filter: blur(4px); } .anlink-modal { background: linear-gradient(135deg, #1a1a1a 0%, #2d2d2d 100%); border-radius: 16px; box-shadow: 0 20px 40px rgba(0,0,0,0.4); width: 420px; max-width: 90vw; color: #fff; overflow: hidden; } .anlink-modal-header { text-align: center; padding: 24px 24px 16px; background: linear-gradient(135deg, #26a69a 0%, #20847a 100%); } .anlink-modal-icon { font-size: 48px; margin-bottom: 8px; } .anlink-modal h2 { margin: 0 0 8px; font-size: 24px; font-weight: 600; } .anlink-episode-count { opacity: 0.9; font-size: 14px; } .anlink-modal-body { padding: 24px; } .anlink-range-inputs { display: flex; align-items: center; gap: 16px; margin-bottom: 20px; } .anlink-input-group { flex: 1; } .anlink-input-group label { display: block; margin-bottom: 8px; font-size: 14px; color: #26a69a; font-weight: 500; } .anlink-input-group input { width: 100%; padding: 12px; border: 2px solid #444; border-radius: 8px; background: #1a1a1a; color: #fff; font-size: 16px; text-align: center; transition: all 0.2s; } .anlink-input-group input:focus { outline: none; border-color: #26a69a; box-shadow: 0 0 0 3px rgba(38,166,154,0.1); } .anlink-range-divider { color: #26a69a; font-weight: bold; font-size: 18px; margin-top: 24px; } .anlink-quick-select { display: flex; gap: 8px; margin-bottom: 16px; } .anlink-quick-btn { flex: 1; padding: 8px 12px; border: 1px solid #444; border-radius: 6px; background: transparent; color: #ccc; cursor: pointer; font-size: 12px; transition: all 0.2s; position: relative; } .anlink-quick-btn:hover, .anlink-quick-btn:focus { border-color: #26a69a; color: #26a69a; background: rgba(38,166,154,0.1); outline: none; } .anlink-help-text { font-size: 11px; color: #888; text-align: center; margin-top: 12px; } .anlink-modal-footer { display: flex; gap: 12px; padding: 0 24px 24px; } .anlink-btn { flex: 1; padding: 12px 24px; border: none; border-radius: 8px; font-size: 14px; font-weight: 500; cursor: pointer; transition: all 0.2s; position: relative; } .anlink-btn:focus { outline: 2px solid #26a69a; outline-offset: 2px; } .anlink-btn-cancel { background: #444; color: #ccc; } .anlink-btn-cancel:hover, .anlink-btn-cancel:focus { background: #555; } .anlink-btn-primary { background: linear-gradient(135deg, #26a69a 0%, #20847a 100%); color: #fff; } .anlink-btn-primary:hover, .anlink-btn-primary:focus { transform: translateY(-1px); box-shadow: 0 4px 12px rgba(38,166,154,0.3); } kbd { background: rgba(255,255,255,0.1); border: 1px solid rgba(255,255,255,0.2); border-radius: 3px; padding: 1px 4px; font-size: 10px; margin-right: 4px; } `); document.body.appendChild(modal); const [startInput, endInput] = modal.querySelectorAll('input'); const buttons = modal.querySelectorAll('button'); const primaryBtn = modal.querySelector('.anlink-btn-primary'); const cancelBtn = modal.querySelector('.anlink-btn-cancel'); const validate = () => { const s = Math.max(1, Math.min(total, +startInput.value)); const e = Math.max(s, Math.min(total, +endInput.value)); startInput.value = s; endInput.value = e; }; const cleanup = () => modal.remove(); const accept = () => { validate(); cleanup(); resolve({ start: +startInput.value, end: +endInput.value }); }; const cancel = () => { cleanup(); resolve(null); }; // Keyboard navigation with arrow keys for number inputs modal.addEventListener('keydown', e => { switch (e.key) { case 'Escape': e.preventDefault(); cancel(); break; case 'Enter': e.preventDefault(); accept(); break; case 'f': case 'F': if (!e.target.matches('input') && !e.ctrlKey && !e.altKey) { e.preventDefault(); startInput.focus(); startInput.select(); } break; } }); // Input validation and arrow key navigation for number inputs [startInput, endInput].forEach(input => { input.addEventListener('input', validate); input.addEventListener('keydown', e => { if (e.key === 'ArrowUp') { e.preventDefault(); input.value = Math.min(total, (+input.value || 0) + 1); validate(); } else if (e.key === 'ArrowDown') { e.preventDefault(); input.value = Math.max(1, (+input.value || 2) - 1); validate(); } else if (e.key === 'Tab' && !e.shiftKey && input === endInput) { e.preventDefault(); modal.querySelector('.anlink-quick-btn').focus(); } }); }); // Quick select buttons modal.querySelectorAll('.anlink-quick-btn').forEach((btn, index) => { btn.addEventListener('click', () => { const [s, e] = btn.dataset.range.split(',').map(Number); startInput.value = s; endInput.value = e; validate(); // Focus extract button after quick select setTimeout(() => primaryBtn.focus(), 100); }); // Arrow key navigation between quick select buttons btn.addEventListener('keydown', e => { if (e.key === 'ArrowLeft' && index > 0) { e.preventDefault(); modal.querySelectorAll('.anlink-quick-btn')[index - 1].focus(); } else if (e.key === 'ArrowRight' && index < 2) { e.preventDefault(); modal.querySelectorAll('.anlink-quick-btn')[index + 1].focus(); } else if (e.key === 'Tab' && !e.shiftKey && index === 2) { e.preventDefault(); cancelBtn.focus(); } }); }); // Button handlers with enhanced keyboard navigation cancelBtn.addEventListener('click', cancel); cancelBtn.addEventListener('keydown', e => { if (e.key === 'ArrowRight') { e.preventDefault(); primaryBtn.focus(); } }); primaryBtn.addEventListener('click', accept); primaryBtn.addEventListener('keydown', e => { if (e.key === 'ArrowLeft') { e.preventDefault(); cancelBtn.focus(); } }); // Focus management - start with first input and select all text setTimeout(() => { startInput.focus(); startInput.select(); }, 100); }); } /*************************************************************** * Apply episode range filtering with modern UI ***************************************************************/ async function applyEpisodeRangeFilter(allEpLinks) { const status = document.querySelector('.anlink-status-bar'); const epRangeThreshold = GM_getValue('ep_range_threshold', 12); if (allEpLinks.length <= epRangeThreshold) return allEpLinks; status.text = `Found ${allEpLinks.length} episodes. Waiting for selection...`; const selection = await showEpisodeRangeSelector(allEpLinks.length); if (!selection) { status.text = 'Cancelled by user.'; return null; } const filtered = allEpLinks.slice(selection.start - 1, selection.end); status.text = `Extracting episodes ${selection.start}-${selection.end} of ${allEpLinks.length}...`; return filtered; } /*************************************************************** * Display a simple toast message on the top right of the screen ***************************************************************/ let toasts = []; function showToast(message, duration = 5000) { const maxToastHeight = window.innerHeight * 0.5; const toastHeight = 70; const maxToasts = Math.floor(maxToastHeight / toastHeight); console.log(message); // Inject toast styles if not already present if (!document.getElementById('anlink-toast-styles')) { GM_addStyle(` @keyframes anlink-toast-slide-in { from { transform: translateX(400px); opacity: 0; } to { transform: translateX(0); opacity: 1; } } @keyframes anlink-toast-slide-out { from { transform: translateX(0); opacity: 1; } to { transform: translateX(400px); opacity: 0; } } .anlink-toast { position: fixed; right: 20px; min-width: 300px; max-width: 400px; background: linear-gradient(135deg, #ffffff 0%, #f8f9fa 100%); border: 1px solid rgba(0, 0, 0, 0.08); border-radius: 12px; padding: 16px 20px; box-shadow: 0 8px 32px rgba(0, 0, 0, 0.12), 0 2px 8px rgba(0, 0, 0, 0.08); z-index: 10000; display: flex; align-items: flex-start; gap: 12px; animation: anlink-toast-slide-in 0.3s cubic-bezier(0.16, 1, 0.3, 1); backdrop-filter: blur(10px); transition: top 0.4s cubic-bezier(0.16, 1, 0.3, 1); } .anlink-toast.slide-out { animation: anlink-toast-slide-out 0.3s cubic-bezier(0.7, 0, 0.84, 0) forwards; } .anlink-toast-icon { flex-shrink: 0; width: 24px; height: 24px; display: flex; align-items: center; justify-content: center; background: linear-gradient(135deg, #26a69a 0%, #20847a 100%); border-radius: 50%; color: white; font-size: 14px; font-weight: bold; } .anlink-toast-content { flex: 1; color: #1a1a1a; font-size: 14px; line-height: 1.5; font-weight: 500; } .anlink-toast-content a { color: #26a69a; text-decoration: none; font-weight: 600; border-bottom: 1px solid transparent; transition: border-color 0.2s; } .anlink-toast-content a:hover { border-bottom-color: #26a69a; } .anlink-toast-close { flex-shrink: 0; width: 20px; height: 20px; display: flex; align-items: center; justify-content: center; background: rgba(0, 0, 0, 0.05); border: none; border-radius: 50%; color: #666; cursor: pointer; font-size: 16px; line-height: 1; transition: all 0.2s; padding: 0; } .anlink-toast-close:hover { background: rgba(0, 0, 0, 0.1); color: #1a1a1a; transform: scale(1.1); } /* Dark mode support */ @media (prefers-color-scheme: dark) { .anlink-toast { background: linear-gradient(135deg, #2d2d2d 0%, #1a1a1a 100%); border-color: rgba(255, 255, 255, 0.1); } .anlink-toast-content { color: #e0e0e0; } .anlink-toast-close { background: rgba(255, 255, 255, 0.1); color: #ccc; } .anlink-toast-close:hover { background: rgba(255, 255, 255, 0.2); color: #fff; } } `); const styleTag = document.createElement('style'); styleTag.id = 'anlink-toast-styles'; document.head.appendChild(styleTag); } // Create the new toast element const toast = document.createElement("div"); toast.className = "anlink-toast"; toast.style.top = `${20 + toasts.length * toastHeight}px`; // Infer toast type and icon from message content const lowerMsg = message.toString().toLowerCase(); const iconMap = { error: ['❌', '#ef5350'], success: ['✅', '#66bb6a'], warning: ['⚠️', '#ffa726'], loading: ['⏳', '#42a5f5'], help: ['💡', '#ab47bc'], info: ['ℹ️', null] }; const typeChecks = [ [['error', 'failed', 'couldn\'t', 'could not'], 'error'], [['success', 'complete', 'copied', 'exported', 'sent to'], 'success'], [['warning', 'no episodes', 'not found', 'rate limited'], 'warning'], [['loading', 'fetching', 'extracting', 'processing'], 'loading'], [['install', 'mpv', 'handler'], 'help'] ]; const toastType = typeChecks.find(([keywords]) => keywords.some(k => lowerMsg.includes(k)))?.[1] || 'info'; const [icon, borderColor] = iconMap[toastType]; if (borderColor) toast.style.borderLeft = `4px solid ${borderColor}`; toast.innerHTML = ` <div class="anlink-toast-icon">${icon}</div> <div class="anlink-toast-content">${message}</div> <button class="anlink-toast-close" aria-label="Close">×</button> `; document.body.appendChild(toast); // Close button handler const closeBtn = toast.querySelector('.anlink-toast-close'); const removeToast = () => { toast.classList.add('slide-out'); setTimeout(() => { if (document.body.contains(toast)) document.body.removeChild(toast); toasts = toasts.filter(t => t !== toast); // Reposition remaining toasts toasts.forEach((t, index) => { t.style.top = `${20 + index * toastHeight}px`; }); }, 300); }; closeBtn.addEventListener('click', removeToast); // Add the new toast to the list toasts.push(toast); // Auto-remove after delay (or dont remove if duration is 0) if (duration > 0) { setTimeout(() => removeToast(), duration); } // Limit the number of toasts to maxToasts if (toasts.length > maxToasts) { const oldestToast = toasts.shift(); oldestToast.classList.add('slide-out'); setTimeout(() => { if (document.body.contains(oldestToast)) { document.body.removeChild(oldestToast); } }, 300); // Reposition remaining toasts toasts.forEach((t, index) => { t.style.top = `${20 + index * toastHeight}px`; }); } } // On overlay open, show a help link for mpv-handler if not detected function showMPVHandlerHelp() { showToast('To play directly in MPV, install <a href="https://github.com/akiirui/mpv-handler" target="_blank" style="color:#1976d2;">mpv-handler</a> and reload this page.'); } // Simple query selector shortcuts const _$ = s => document.querySelector(s); const _$$ = s => document.querySelectorAll(s);