// ==UserScript== // @name Add Radarr link to each letterboxd film // @namespace http://tampermonkey.net/ // @version 1.31 // @description Adds an Radarr link to add movie for every film entry on letterboxd // @author CedricPelchat // @match https://letterboxd.com/* // @exclude https://letterboxd.com/film/* // @exclude https://letterboxd.com/*/film/* // @icon  // @license MIT // @grant none // @downloadURL none // ==/UserScript== if( document.readyState == "complete" || document.readyState == "loaded" || document.readyState == "interactive" ) { main(); } else { document.addEventListener('DOMContentLoaded', main); } function main() { const apiKey = ""; // YOUR RADARR API KEY const radarrUrl = ""; // YOUR RADARR API URL, BY DEFAULT "http://localhost:7878/api/v3/movie" function showBanner(message, isError) { const banner = document.createElement("div"); banner.textContent = message; banner.style.position = "fixed"; banner.style.bottom = "10px"; banner.style.right = "10px"; banner.style.padding = "10px"; banner.style.backgroundColor = "#000"; banner.style.color = "#fff"; banner.style.borderRadius = "5px"; banner.style.zIndex = "9999"; banner.classList.add('banner'); if (isError) { banner.style.backgroundColor = "red"; } document.body.appendChild(banner); setTimeout(() => { banner.remove(); }, 3000); } const addMovie = async (tmdbId, title) => { try { const response = await fetch(`${radarrUrl}?apikey=${apiKey}`, { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ "tmdbId": tmdbId, "title": title, "qualityProfileId": 1, "rootFolderPath": "", // YOUR ROOTFOLDERPATH "monitored": true, "addOptions": { "searchForMovie": true } }) }); if (response.ok) { showBanner(`${title} added to Radarr!`); } else { showBanner(`Error adding ${title} to Radarr: ${response.statusText}`, true); } } catch (error) { console.error(`Error adding ${title} to Radarr: ${error.message}`); } } // Function used to build icons function createIcon(cont, title, href, icon) { if (cont.querySelector('.radarr-icon')) { return; } var a = document.createElement('a'); a.href = href; a.title = title; a.setAttribute('target', '_blank'); var img = document.createElement('img'); img.src = icon; img.setAttribute('height', '27em'); img.setAttribute('width', '27em'); a.appendChild(img); var span = document.createElement('span'); span.classList.add('radarr-icon'); // Add class to identify icon span.appendChild(a); cont.appendChild(span); console.log(title, 'icon built successfully.'); } //Get TMDB ID from title with TMDB API function getTmdbId(filmTitle, filmYear) { const api_key = ''; // YOUR TMDB API KEY const url = `https://api.themoviedb.org/3/search/movie?api_key=${api_key}&query=${filmTitle}&year=${filmYear}`; return fetch(url) .then(response => response.json()) .then(data => { if (data.total_results > 0) { return data.results[0].id; } else { throw new Error(`No movie was found with the title "${filmTitle}" and release year "${filmYear}".`); } }) .catch(error => { throw error; }); } // Add Radarr link to each film on list page //Get all films, only gets the first row, list pages, and Popular reviews this week in Films page. document.querySelectorAll('li.poster-container, li.listitem, li.film-detail, .frame').forEach((film) => { let iconCreated = false; film.classList.add('film-list'); console.log(film); film.addEventListener('mouseover', () => { // Get film title const filmTitleElement = film.querySelector('.frame-title'); const filmTitle = filmTitleElement.textContent; // Get overlay actions element const overlayActions = film.querySelector('.overlay-actions'); // Check that overlayActions element exists if (!overlayActions) { console.log('No overlay actions element found for film:', film); } // Check that filmTitle is not empty if (!filmTitle) { console.log('Empty film title:', film); } const overlayActionsW70 = film.querySelector('.-w70'); if (overlayActions&& !iconCreated) { // Create new span element and add to overlay actions const span = document.createElement('span'); span.style.display = 'inline-block'; overlayActions.appendChild(span); // Create new icon and add to span element const img= '' const filmLink='http://localhost:7878/add/new?term='+filmTitle; const iconEl = document.createElement('span'); const radarr =createIcon(iconEl, 'Radarr',filmLink, img); iconCreated = true; // Add click event listener to add movie iconEl.addEventListener('click', () => { iconEl.style.opacity='0.6'; setTimeout(() => { iconEl.style.opacity='1'; }, 100); //Separate title and Year const titleWithYear = filmTitleElement.textContent; const regex = /\((\d{4})\)/; // matches four digits inside parentheses const match = titleWithYear.match(regex); let title; let year; if (match) { // extract the title and year from the titleWithYear string title = titleWithYear.slice(0, match.index).trim(); year = match[1]; } else { // no year was found in the titleWithYear string title = titleWithYear.trim(); } const api_key = ''; // YOUR TMDB API KEY const url = `https://api.themoviedb.org/3/search/movie?api_key=${api_key}&query=${title}&year=${year}`; fetch(url) .then(response => response.json()) .then(data => { if (data.total_results > 0) { const tmdb_id = data.results[0].id; console.log(`The TMDB ID grabbed from their API for "${title}" (${year}) is ${tmdb_id}.`); addMovie(tmdb_id, title); } else { console.log(`No movie was found with the title "${title}".`); } }) .catch(error => { console.log(error); }); }); if (overlayActionsW70) { iconEl.style.position = 'absolute'; iconEl.style.right ='9px'; iconEl.style.top = '-20px'; iconEl.style.height = '20px'; iconEl.style.width = '12px'; } span.appendChild(iconEl); } } ); } ); }