您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Add rating and link to Letterboxd to SC torrent pages.
// ==UserScript== // @name sc-letterboxd-rating // @namespace https://tampermonkey.net/ // @version 2.1 // @author boisterous-larva // @description Add rating and link to Letterboxd to SC torrent pages. // @homepage https://github.com/boisterous-larva/sc-letterboxd-rating/blob/master/sc-letterboxd-rating.user.js // @icon https://letterboxd.com/favicon.ico // @match https://*.secret-cinema.pw/torrents.php?id=* // @grant GM.xmlHttpRequest // ==/UserScript== (function () { "use strict"; function addStyle(css) { const style = document.createElement("style"); style.textContent = css; document.head.appendChild(style); } function getIMDBID() { let a = document.querySelector('[href*="://www.imdb.com/title/tt"]'); if (!a) return; let id = a.href.match(/tt\d+/)[0]; if (id) { handleIMDB(id) handleLetterboxd(id); } } // function getRottenID() { // let rottenURL = document.querySelector('.meta__rotten a').href; // if (rottenURL) { // handleRotten(rottenURL); // } else return; // } function getElementByInnerText(tag, text) { return Array.from(document.querySelectorAll(tag)).find( (el) => el.innerText.trim().toLowerCase() === text ); } function buildElement(siteName, url, logo, rating, count) { if (!rating) return; const linkbox = document.querySelector(".linkbox"); // Selects the first linkbox on the page. Consider improving this by making it more specific. if (!linkbox) return; let ratingFloat = parseFloat(rating); let ratingColor = "white"; // Default. if (ratingFloat){ if (siteName === "IMDb") ratingFloat = ratingFloat / 2; // IMDb ratings are out of 10, adjust to match other ratings if (siteName === "RT") ratingFloat = (ratingFloat / 100) * 5 // Rotten scores are out of 100, adjust to match other ratings ratingColor = ratingFloat < 2.5 ? "rgba(212, 36, 36, 0.8)" // Red for ratings below 2.5 (5.0 IMDb and 50% RT) : ratingFloat < 3.5 ? "rgba(212, 195, 36, 0.8)" // Yellow for ratings 2.5 and above (5.0 IMDb and 50% RT) : ratingFloat < 4.5 ? "rgba(0,224,84, 0.8)" // Green for ratings 3.5 and above (7.0 IMDb and 70& RT) : "rgba(113, 251, 255, 0.8)"; // Light blue for ratings 4.5 and above (9.0 IMDb and 90% RT) } const logoLink = logo; const img = document.createElement("img"); img.className = `${siteName.toLowerCase()}-chip__icon`; img.src = logoLink; const iconStyle = ` .meta-chip { display: grid; grid-template-areas: "image name" "image value"; grid-template-columns: 32px auto; grid-template-rows: auto auto; border-radius: 26px; padding: 6px 12px 6px 12px; background: black; } .${siteName.toLowerCase()}-chip__icon{ grid-area: image; text-align: center; line-height: 40px; font-size: 12px; width: 25px; height: 25px; border-radius: 4%; filter: drop-shadow(0 0 0.3rem ${ratingColor}); } .meta-chip__name { color: white; grid-area: name; display: inline; margin: 0; font-weight: 400; font-size: 12px; line-height: 1; align-self: end; } .meta-chip__value { color:white; grid-area: value; display: inline; margin: 0; padding: 0; line-height: 1; align-self: start; height: 100%; font-weight: 400; font-size: 10px; }`; const container = document.createElement("div"); const ratingName = document.createElement("h2"); const ratingValue = document.createElement("h3"); const meta_id_tag = document.createElement("a"); meta_id_tag.className = "meta-chip"; meta_id_tag.style = "column-gap:4px; row-gap:0; padding-right:18px;"; ratingName.className = "meta-chip__name"; ratingValue.className = "meta-chip__value"; ratingValue.style = `color:${ratingColor};`; meta_id_tag.href = url; meta_id_tag.target = "_blank"; meta_id_tag.append(img); ratingName.innerText = siteName; ratingValue.innerText = `${rating} / ${count} Votes`; meta_id_tag.append(ratingName); meta_id_tag.append(ratingValue); container.className = "ratings-container"; container.style = `display:flex; margin-top:4px; gap:6px;`; const ratingsContainer = document.querySelector('.ratings-container'); if (!ratingsContainer) { container.append(meta_id_tag); linkbox.append(container); } else ratingsContainer.append(meta_id_tag); addStyle(iconStyle); console.log(`Added ${siteName} rating: ${rating} / ${count} Votes`); } function handleLetterboxd(id) { const letterboxdURL = "https://letterboxd.com/imdb/"; const siteName = "Letterboxd"; const logoURL = ""; const url = `${letterboxdURL}${id}`; return new Promise((resolve, reject) => { GM.xmlHttpRequest({ method: "GET", url: url, onload: function (response) { if (response.status === 200) { const responseText = response.responseText; // Get the relevant info from the response const scriptMatch = responseText.match( /<script type="application\/ld\+json">\n\/\* <!\[CDATA\[ \*\/\n([\s\S]*?)\/\* ]]> \*\/\n<\/script>/ ); if (scriptMatch && scriptMatch[1]) { const jsonData = JSON.parse(scriptMatch[1]); const aggregateRating = jsonData.aggregateRating; if (aggregateRating) { console.log("Letterboxd data found."); const ratingValue = aggregateRating.ratingValue; const ratingCount = aggregateRating.ratingCount; buildElement(siteName, response.finalUrl, logoURL, ratingValue, ratingCount); } } else { console.error("Letterboxd data not found."); return; } } else { console.error( "Failed to fetch the webpage. Status:", response.status ); reject(`Failed to fetch the webpage. Status: ${response.status}`); } }, onerror: function (error) { console.error("Error fetching the webpage:", error); reject(error); }, }); }); } function handleIMDB(id) { const siteName = "IMDb"; const logoURL = ""; const imdbURL = `https://www.imdb.com/title/${id}`; return new Promise((resolve, reject) => { // Step 1: Use GM.xmlHttpRequest to fetch the IMDb page GM.xmlHttpRequest({ method: 'GET', url: imdbURL, onload: function(response) { try { // Step 2: Parse the HTML content const parser = new DOMParser(); const doc = parser.parseFromString(response.responseText, 'text/html'); if (doc){ console.log("IMDB data found."); } else { console.error('IMDB data not found.'); return; } // Step 3: Extract the rating const ratingBarParent = doc.querySelector('[data-testid="hero-rating-bar__aggregate-rating__score"]'); if (!ratingBarParent) { throw new Error('IMDb rating element not found'); } const ratingElement = ratingBarParent.querySelector('span'); const rating = ratingElement.textContent.trim(); // e.g., "8.7" // Step 4: Extract the votes const parent = ratingBarParent.parentElement; const votesElement = parent.lastChild; const votes = votesElement.textContent.trim(); // e.g., "13K" // Step 5: Resolve with the results resolve(buildElement(siteName, imdbURL, logoURL, rating, votes)); // Assemble data and build IMDB element. } catch (error) { console.error('Error:', error.message); reject(error); } }, onerror: function(error) { console.error('Request failed:', error); reject(new Error('Failed to fetch IMDb page')); } }); }); } function handleRotten(rottenURL) { const siteName = "RT"; const logoURL = ""; const url = rottenURL; return new Promise((resolve, reject) => { // Step 1: Use GM.xmlHttpRequest to fetch the IMDb page GM.xmlHttpRequest({ method: 'GET', url: url, onload: function(response) { try { // Step 2: Parse the HTML content const parser = new DOMParser(); const doc = parser.parseFromString(response.responseText, 'text/html'); if (doc){ console.log("RT data found."); } else { console.error('RT data not found.'); return; } // Step 3: Extract the ratings let criticsScore = doc.querySelector('[slot="criticsScore"]').textContent.trim(); if (!criticsScore) { criticsScore = '-'; console.error('Rotten critics score not found'); } let audienceScore = doc.querySelector('[slot="audienceScore"]').textContent.trim(); if (!audienceScore) { audienceScore = '-'; console.error('Rotten audience score not found'); } // Step 4: Extract number of reviews let reviews = doc.querySelector('[slot="criticsReviews"]').textContent.trim().match(/\d+/)[0]; if (!reviews) { reviews = '-'; console.error('Rotten number of reviews not found'); } // Step 5: Resolve with the results resolve(buildElement(siteName, url, logoURL, criticsScore, `${audienceScore} / ${reviews}`)); // Assemble data and build element. } catch (error) { console.error('Error:', error.message); reject(error); } }, onerror: function(error) { console.error('Request failed:', error); reject(new Error('Failed to fetch Rotten page')); } }); }); } getIMDBID(); // getRottenID(); })();