sc-letterboxd-rating

Add rating and link to Letterboxd to SC torrent pages.

目前為 2025-10-01 提交的版本,檢視 最新版本

// ==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://secret-cinema.pw/forums.php?action=viewthread&threadid=902
//
// @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
              const originalIMDBElement = document.querySelector('.meta__imdb').remove();
              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
            const originalRottenElement = document.querySelector('.meta__rotten').remove();
            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();

})();