// ==UserScript== // @name Plex Letterboxd link and rating // @namespace http://tampermonkey.net/ // @description Add Letterboxd link and rating to its corresponding Plex film's page // @author CarnivalHipster // @match https://app.plex.tv/* // @icon https://www.google.com/s2/favicons?sz=64&domain=letterboxd.com // @license MIT // @grant GM_xmlhttpRequest // @connect letterboxd.com // @version 2.7.1 // @downloadURL none // ==/UserScript== //Edge cases: //Vietnam: A Television History is a tv show logged as movie in Tmdb so doesn't get an icon //Films that have both same year and name and one of them has no directors like Cargo 2006 and Cargo 2006 by Clive Gordon //Directors that are very unknown don't get the icon, don't know why //Also Clive Gordon's director's page makes the script stop after getting the title for unknown reasons //The Shining has a bug on letterboxd where the-shining-1980 links to the-shining-1997 //I did a fall back to remove the year from the url for a last try for stupid cases like The Shining, which may cause problems elsewhere, but probably not. //Letterboxd api will make all this obsolete so its not really worth the time. (function() { 'use strict'; //const letterboxdImg = ''; const letterboxdImg = 'https://www.google.com/s2/favicons?sz=64&domain=letterboxd.com'; const globalParser = new DOMParser(); var lastTitle = undefined; var lastYear = undefined; var lastDirector = undefined; function extractTitleAndYear() { const titleElement = document.querySelector('h1[data-testid="metadata-title"]'); const yearElement = document.querySelector('span[data-testid="metadata-line1"]'); if (titleElement) { const title = titleElement.textContent.trim() || titleElement.innerText.trim(); if (title !== lastTitle) { lastTitle = title; console.log('The title is:', lastTitle); } } else { lastTitle = ''; // Reset if no title is found } if (yearElement) { const text = yearElement.textContent.trim() || yearElement.innerText.trim(); const match = text.match(/\b\d{4}\b/); if (match && match[0] !== lastYear) { lastYear = match[0]; console.log('The year is:', lastYear); } } else { lastYear = ''; // Reset if no year is found } } function extractDirectorFromPage() { const directedByText = 'Directed by'; const spans = Array.from(document.querySelectorAll('span')); const directorSpan = spans.find(span => span.textContent.includes(directedByText)); if (directorSpan) { const directorLink = directorSpan.parentElement.querySelector('a'); if (directorLink) { const directorName = directorLink.textContent.trim(); if (directorName && directorName !== lastDirector) { lastDirector = directorName; console.log('Director in Plex: ', lastDirector); } } } else { if (lastDirector !== undefined) { lastDirector = undefined; console.log('The director has been reset.'); } } } function checkLink(url) { return new Promise((resolve, reject) => { GM.xmlHttpRequest({ method: 'HEAD', url: url, onload: function(response) { if (response.status >= 200 && response.status < 300) { resolve({url: url, status: response.status, accessible: true}); } else { resolve({url: url, status: response.status, accessible: false}); } }, onerror: function() { reject(new Error(url + ' could not be reached or is blocked by CORS policy.')); } }); }); } function updateOrCreateLetterboxdIcon(link, rating) { let metadataElement = document.querySelector('div[data-testid="metadata-ratings"]'); if (!metadataElement) { metadataElement = document.querySelector('div[data-testid="metadata-children"]'); } const existingContainer = document.querySelector('.letterboxd-container'); if (existingContainer) { existingContainer.querySelector('a').href = link; const ratingElement = existingContainer.querySelector('.letterboxd-rating'); if (ratingElement) { ratingElement.textContent = rating; } } else if (metadataElement) { const container = document.createElement('div'); container.classList.add('letterboxd-container'); container.style.cssText = 'display: flex; align-items: center; gap: 8px;'; const icon = document.createElement('img'); icon.src = letterboxdImg; icon.alt = 'Letterboxd Icon'; icon.style.cssText = 'width: 24px; height: 24px; cursor: pointer;'; const ratingText = document.createElement('span'); ratingText.classList.add('letterboxd-rating'); ratingText.textContent = rating; ratingText.style.cssText = 'font-size: 14px;'; // Style as needed const linkElement = document.createElement('a'); linkElement.href = link; linkElement.appendChild(icon); container.appendChild(linkElement); container.appendChild(ratingText); metadataElement.insertAdjacentElement('afterend', container); } } function buildDefaultLetterboxdUrl(title, year) { const normalizedTitle = title.normalize("NFD").replace(/[\u0300-\u036f]/g, ""); const titleSlug = normalizedTitle.trim().toLowerCase() .replace(/&/g, 'and') .replace(/-/g, ' ') .replace(/[^\w\s]/g, '') .replace(/\s+/g, '-'); const letterboxdBaseUrl = 'https://letterboxd.com/film/'; return `${letterboxdBaseUrl}${titleSlug}-${year}/`; } function removeYearFromUrl(url) { const yearPattern = /-\d{4}(?=\/$)/; return url.replace(yearPattern, ''); } function replaceFilmWithDirector(url) { return url.replace('film','director'); } function buildLetterboxdUrl(title, year) { let defaultUrl = buildDefaultLetterboxdUrl(title, year); return checkLink(defaultUrl).then(result => { if (result.accessible) { console.log(result.url, 'is accessible, status:', result.status); return result.url; } else { console.log(result.url, 'is not accessible, status:', result.status); let yearRemovedUrl = removeYearFromUrl(result.url); console.log('Trying URL without year:', yearRemovedUrl); return checkLink(yearRemovedUrl).then(yearRemovedResult => { if (yearRemovedResult.accessible) { console.log(yearRemovedUrl, 'is accessible, status:', yearRemovedResult.status); return yearRemovedUrl; } else { console.log(yearRemovedUrl, 'is not accessible, status:', yearRemovedResult.status); let directorUrl = replaceFilmWithDirector(yearRemovedUrl); console.log('Trying director URL:', directorUrl); return checkLink(directorUrl).then(result =>{ if (result.accessible){ return directorUrl; }else{ console.log(result.url, 'is not accessible, status:', result.status); } }); } }); } }).catch(error => { console.error('Error after checking both film and year:', error.message); let newUrl = removeYearFromUrl(defaultUrl); return newUrl; }); } function fetchLetterboxdPage(url) { return new Promise((resolve, reject) => { GM.xmlHttpRequest({ method: 'GET', url: url, onload: function(response) { if (response.status >= 200 && response.status < 300) { resolve(response.responseText); } else { reject(new Error('Failed to load Letterboxd page')); } }, onerror: function() { reject(new Error('Network error while fetching Letterboxd page')); } }); }); } function roundToOneDecimal(numberString) { const number = parseFloat(numberString); return isNaN(number) ? null : (Math.round(number * 10) / 10).toFixed(1); } function extractRating(doc) { const ratingElement = doc.querySelector('meta[name="twitter:data2"]'); if (ratingElement && ratingElement.content) { const match = ratingElement.getAttribute('content').match(/\b\d+\.\d{1,2}\b/); if (match) { return roundToOneDecimal(match[0]); } } else { console.log('Rating element not found.'); return null; } } function extractYearFromMeta(doc) { const metaTag = doc.querySelector('meta[property="og:title"]'); if (metaTag) { const content = metaTag.getAttribute('content'); const yearMatch = content.match(/\b\d{4}\b/); if (yearMatch) { return yearMatch[0]; } else { console.log('Year not found in the html'); return null; } } else { console.log('Meta tag not found in the html'); return null; } } function extractDirectorFromMeta(doc) { const directorMetaTag = doc.querySelector('meta[name="twitter:data1"]'); return directorMetaTag ? directorMetaTag.content : null; } function subtractYearFromUrl(url, lastYear) { const yearPattern = /-(\d{4})\/$/; const match = url.match(yearPattern); if (match) { const year = parseInt(match[1], 10) - 1; return url.replace(yearPattern, `-${year}/`); } else { const previousYear = parseInt(lastYear, 10) - 1; return url.replace(/\/$/, `-${previousYear}/`); } } async function processLetterboxdUrl(initialUrl) { let url = initialUrl; let shouldContinue = true; const hasOneSeason = document.querySelector('[title*="Season 1"]'); while (shouldContinue) { try { const html = await fetchLetterboxdPage(url); const doc = globalParser.parseFromString(html, "text/html"); if (hasOneSeason && !doc.querySelector('a[href*="themoviedb.org/tv/"]')) { console.log(`Plex got a tv show but Letterboxd is on a movie or director page.`); console.log(`Icon creation aborted.`); break; } if (url.startsWith('https://letterboxd.com/director')) { updateOrCreateLetterboxdIcon(url, 'Letterboxd'); break; } else { const yearInHtml = extractYearFromMeta(doc); console.log('The year in the html is : ' + yearInHtml); if (yearInHtml != lastYear) { const directorInHtml = extractDirectorFromMeta(doc); console.log('Director in Html: ' + directorInHtml); if (!directorInHtml.includes(lastDirector)) { if (lastYear != NaN) { break; } let subtractedYearUrl = subtractYearFromUrl(url, lastYear); console.log('Trying substracted year url: ' + subtractedYearUrl); let result = await checkLink(subtractedYearUrl); if (result.accessible) { console.log(subtractedYearUrl, 'Url with substracted year is accessible, status:', result.status); const newHtml = await fetchLetterboxdPage(subtractedYearUrl); const newDoc = globalParser.parseFromString(newHtml, "text/html"); const newDirectorInHtml = extractDirectorFromMeta(newDoc); if (newDirectorInHtml.includes(lastDirector)) { url = subtractedYearUrl; continue; } else { console.log(`Director in Plex [${lastDirector}] doesn't match director in html [${newDirectorInHtml}]`); } } else { console.log('Url with substracted year is inaccessible'); } // Fallback to URL without year let urlWithoutYear = removeYearFromUrl(url); result = await checkLink(urlWithoutYear); if (result.accessible) { console.log(result.url, 'is accessible, status:', result.status); console.log('Going back to url without year as fallback'); url = urlWithoutYear; // Update URL to the one without year continue; // Check again with the updated URL } else { console.log(result.url, 'is not accessible, status:', result.status); console.log(`Icon creation aborted`); break; } } else { console.log('Movie has different year but same director and name, probably due to differing metadata. Icon created'); const rating = extractRating(doc); updateOrCreateLetterboxdIcon(url, rating); break; } } else { const rating = extractRating(doc); updateOrCreateLetterboxdIcon(url, rating); break; } } } catch (error) { console.error('Error fetching or parsing Letterboxd page:', error); break; } } } if(document.readyState === 'complete' || document.readyState === 'loaded' || document.readyState === 'interactive') { main(); } else { document.addEventListener('DOMContentLoaded', main); } function main() { var lastProcessedTitle = undefined; var lastProcessedYear = undefined; var lastProcessedDirector = undefined; function observerCallback(mutationsList, observer) { const isAlbumPage = document.querySelector('[class^="AlbumDisc"]'); const isFullSeries = document.querySelector('[title*="Season 4"], [title*="Season 5"], [title*="Season 6"]'); if (isAlbumPage || isFullSeries) { return; } extractTitleAndYear(); extractDirectorFromPage(); if (lastTitle !== lastProcessedTitle || lastYear !== lastProcessedYear || lastDirector !== lastProcessedDirector ) { lastProcessedTitle = lastTitle; lastProcessedYear = lastYear; lastProcessedDirector = lastDirector; if (lastTitle && lastYear ) { buildLetterboxdUrl(lastTitle, lastYear).then(url => { if (!url){ return; } processLetterboxdUrl(url, lastYear, lastDirector); }).catch(error => { console.error('Error building Letterboxd URL:', error); }); } } } const observer = new MutationObserver(observerCallback); observer.observe(document.body, { childList: true, characterData: true, subtree: true }); } })();