// ==UserScript==
// @name sc-letterboxd-rating
// @namespace https://tampermonkey.net/
// @version 1.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 extraHeader = getElementByInnerText("h2", "extra information");
if (!extraHeader) return;
let ratingFloat = parseFloat(rating);
let ratingColor = "var(--meta-chip-name-fg)"; // 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
: ratingFloat < 3.5
? "rgba(212, 195, 36, 0.8)" // Yellow for ratings 2.5 and above
: ratingFloat < 4.5
? "rgba(0,224,84, 0.8)" // Green for ratings 3.5 and above
: "rgba(113, 251, 255, 0.8)"; // Light blue for ratings 4.5 and above
}
const logoLink = logo;
const img = document.createElement("img");
img.className = `${siteName.toLowerCase()}-chip__icon`;
img.src = logoLink;
const iconStyle = `
.${siteName.toLowerCase()}-chip__icon{
grid-area: image;
text-align: center;
line-height: 40px;
font-size: 14px;
color: var(--meta-chip-name-fg);
width: 35px;
height: 35px;
border-radius: 4%;
filter: drop-shadow(0 0 1rem ${ratingColor});
}`;
const linkbox = document.querySelector(".linkbox").first();
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";
ratingName.style = "font-size:14px; margin-bottom:0;";
ratingValue.className = "meta-chip__value";
ratingValue.style = `font-size:12px; 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);
linkbox.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();
})();