Add Ratings from Letterboxd to the torrent page.
// ==UserScript==
// @name UNIT3D Add Letterboxd rating
// @version 1.0
// @description Add Ratings from Letterboxd to the torrent page.
// @match *://*/torrents/*
// @match *://*/requests/*
// @match *://*/torrents/similar/*
// @namespace https://www.tampermonkey.net/
// @author boisterous-larva
// @license MIT License
// @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) {
handleLetterboxd(id);
}
}
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 articleElement = document.querySelector("ul.meta__ids");
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);
articleElement.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);
},
});
});
}
getIMDBID();
})();