您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Greasy Fork is available in English.
Adds direct links to MyAnimeList and AniList on HiAnime watch pages and its grid poster
当前为
// ==UserScript== // @name HiAnime Links+ // @namespace http://greasyfork.icu/users/1470715 // @author cattishly6060 // @author forked_bytes // @description Adds direct links to MyAnimeList and AniList on HiAnime watch pages and its grid poster // @icon https://icons.duckduckgo.com/ip3/hianime.to.ico // @match https://hianime.to/* // @grant GM_openInTab // @version 1.0 // @license 0BSD // ==/UserScript== (function () { 'use strict'; // Add images const malBase64Img = ''; const anilstBase64Img = ''; // Add minimal spinner CSS const style = document.createElement('style'); style.textContent = ` @keyframes spin { to { transform: rotate(360deg); } } .mal-mini-spinner { display: inline-block; width: 12px; height: 12px; border: 2px solid rgba(0,0,0,0.2); border-top-color: #3498db; border-radius: 50%; animation: spin 0.8s linear infinite; margin-left: 5px; vertical-align: middle; } .mal-link-loading .tick { opacity: 0.3; } `; document.head.appendChild(style); /** @typedef {{mal_id: ?number, anilist_id: ?number}} Link */ /** @type {Map<string, ?Link>} */ const cachedLinkMap = new Map(); /** @type {Object<string, string>} */ const linkType = { MAL: 'mal', ANILIST: 'anilist', }; function addHiAnimeBtnAndSubTitle() { Array.from(document.querySelectorAll("div.flw-item"))?.filter(e => e.querySelector('div.film-detail') && e.querySelector('div.film-poster') || []).forEach(e => { if (e.querySelector('.custom-jp-title')) return; const title = e.querySelector('.film-name > a')?.textContent || ""; const titleJp = (e.querySelector('.film-name > a')?.getAttribute('data-jname') || title || "").replace('[Uncensored]', '').trim(); const query = encodeURIComponent(titleJp.slice(0, 100)); const uriMAL = `https://myanimelist.net/search/all?q=${query}&cat=all#anime`; const uriAnilist = `https://anilist.co/search/anime?search=${query}`; const linkEndpoint = e.querySelector('a.film-poster-ahref')?.href; const topPad = e.querySelector('.tick-rate') ? 35 : 10; const imgSize = 30; const aHiAnime = createLink(uriMAL, titleJp, linkEndpoint, linkType.MAL); aHiAnime.innerHTML = `<img class="tick" width="${imgSize}" height="${imgSize}" style="position: absolute; left: 10px; top: ${topPad}px;" src="${malBase64Img}">`; e.querySelector('div.film-poster').appendChild(aHiAnime); const aAnilist = createLink(uriAnilist, titleJp, linkEndpoint, linkType.ANILIST); aAnilist.innerHTML = `<img class="tick" width="${imgSize}" height="${imgSize}" style="position: absolute; left: 10px; top: ${topPad + imgSize + 3}px;" src="${anilstBase64Img}">`; e.querySelector('div.film-poster').appendChild(aAnilist); const jpTitleElement = `<h3 class="film-name custom-jp-title" style="font-size: 12px; color: gray;">${titleJp}</h3>`; e.querySelector('.film-detail')?.children?.[0]?.insertAdjacentHTML('afterend', jpTitleElement); [aHiAnime, aAnilist].forEach(e => e.addEventListener('click', async function (e) { e.preventDefault(); // Prevent default behavior // Get configuration from data attributes const endpoint = this.dataset.endpoint; const fallbackUrl = this.dataset.fallbackUrl; const type = this.dataset.type; // Get cached finalUrl const cachedFinalUrl = this.dataset.finalUrl; if (cachedFinalUrl) { GM_openInTab(cachedFinalUrl, {active: true}); return; } // Get cached link /** @type {?Link} */ const cachedLink = cachedLinkMap.get(endpoint); if (cachedLink) { let finalUrl; if (type === linkType.MAL && cachedLink.mal_id) { finalUrl = `https://myanimelist.net/anime/${cachedLink.mal_id}`; } else if (type === linkType.ANILIST && cachedLink.anilist_id) { finalUrl = `https://anilist.co/anime/${cachedLink.anilist_id}`; } else { finalUrl = fallbackUrl; } GM_openInTab(finalUrl || fallbackUrl, {active: true}); return; } // Show loading state // Add small spinner next to existing icon const spinner = document.createElement('span'); spinner.className = 'mal-mini-spinner'; // Add loading class to parent this.classList.add('mal-link-loading'); // Position spinner absolutely near the icon spinner.style.cssText = ` position: absolute; left: 30px; // top: ${topPad + 9}px; top: ${type === linkType.MAL ? topPad + 9 : topPad + imgSize + 3 + 9}px; `; this.appendChild(spinner); this.style.pointerEvents = 'none'; try { // Fetch required data const res = await fetch(endpoint); if (!res?.ok) { GM_openInTab(fallbackUrl, {active: true}); return; } const data = await res.text(); const malId = data.match(/"mal_id":"(\d+)",/)?.[1]; const anilistId = data.match(/"anilist_id":"(\d+)",/)?.[1]; if (malId || anilistId) { cachedLinkMap.set(endpoint, { mal_id: malId, anilist_id: anilistId }); } if ((!malId && type === linkType.MAL) || (!anilistId && type === linkType.ANILIST) || (!anilistId && !malId) || !type) { GM_openInTab(fallbackUrl, {active: true}); return; } // Open the link const finalUrl = type === linkType.MAL ? `https://myanimelist.net/anime/${malId}` : `https://anilist.co/anime/${anilistId}`; this.dataset.finalUrl = finalUrl; GM_openInTab(finalUrl, {active: true}); } catch (error) { console.error('Failed to process link:', error); } finally { // Clean up (loading) spinner.remove(); this.classList.remove('mal-link-loading'); this.style.pointerEvents = ''; } })); }); } window.addEventListener('load', function () { addHiAnimeBtnAndSubTitle(); }); addHiAnimeBtnAndSubTitle(); const syncData = JSON.parse(document.getElementById('syncData')?.textContent || null); if (!syncData) return; const title = document.getElementsByClassName('film-name')[0]; if (!title) return; if (syncData.mal_id) { const a = createLink(`https://myanimelist.net/anime/${syncData.mal_id}`, 'MyAnimeList'); a.innerHTML = ` <img width="25" height="25" src="${malBase64Img}">`; title.appendChild(a); } if (syncData.anilist_id) { const a = createLink(`https://anilist.co/anime/${syncData.anilist_id}`, 'AniList'); a.innerHTML = ` <img width="25" height="25" src="${anilstBase64Img}">`; title.appendChild(a); } function createLink(href, title, endpoint, type) { const a = document.createElement('a'); a.target = '_blank'; a.rel = 'noreferrer,noopener'; a.href = href; a.title = title; if (endpoint) { a.href = "#"; a.dataset.endpoint = endpoint; a.dataset.fallbackUrl = href; a.dataset.type = type; } return a; } })();