// ==UserScript== // @name Filmweb External Ratings // @namespace http://tampermonkey.net/ // @version 1.0 // @description Add IMDb, Rotten Tomatoes, and Metacritic ratings to Filmweb // @author mrkkr // @match http*://www.filmweb.pl/serial/* // @match http*://www.filmweb.pl/film/* // @match http*://www.filmweb.pl/tvshow/* // @match http*://www.imdb.com/* // @match http*://www.rottentomatoes.com/* // @match http*://www.metacritic.com/* // @grant GM_xmlhttpRequest // @grant GM.xmlhttpRequest // @connect * // @run-at document-end // @downloadURL none // ==/UserScript== (function() { 'use strict'; const ICONS = { imdb: '', rotten: '', metacritic: '' }; async function getExternalUrls(title, year) { console.log('Getting external URLs for:', {title, year}); const imdbLink = document.querySelector('a[href*="www.imdb.com/title/tt"]'); const imdbId = imdbLink ? imdbLink.href.match(/tt\d+/)[0] : null; const encodedTitle = encodeURIComponent(title); return { imdb: imdbId ? `https://www.imdb.com/title/${imdbId}/` : `https://www.imdb.com/find?q=${encodedTitle}+${year}`, rotten: `https://www.rottentomatoes.com/search?search=${encodedTitle}` }; } async function getIMDbRating(url) { try { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: 'GET', url: url, headers: { 'User-Agent': 'Mozilla/5.0' }, onload: function(response) { console.log('IMDb response received'); const ratingMatch = response.responseText.match(/class="sc-d541859f-1[^>]+>([0-9.]+)<\/span>/); if (ratingMatch) { resolve({ rating: ratingMatch[1] + '/10', url: url }); } else { resolve(null); } }, onerror: function(error) { console.error('IMDb fetch error:', error); resolve(null); } }); }); } catch (error) { console.error('Error in IMDb rating function:', error); return null; } } async function getRottenTomatoesRating(url) { try { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: 'GET', url: url, headers: { 'User-Agent': 'Mozilla/5.0' }, onload: function(response) { console.log('RT response received'); const criticMatch = response.responseText.match(/tomatometer">(\d+)%/); const audienceMatch = response.responseText.match(/audience-score">(\d+)%/); if (criticMatch || audienceMatch) { resolve({ rating: (criticMatch ? criticMatch[1] + '% ' : 'N/A ') + (audienceMatch ? '👥' + audienceMatch[1] + '%' : ''), url: url }); } else { resolve(null); } }, onerror: function(error) { console.error('RT fetch error:', error); resolve(null); } }); }); } catch (error) { console.error('Error in RT rating function:', error); return null; } } function createRatingElement(icon, rating, url) { const container = document.createElement('div'); container.style.cssText = ` display: inline-flex; align-items: center; margin: 0 10px; cursor: pointer; padding: 5px 10px; border-radius: 4px; background: rgba(0,0,0,0.05); transition: background 0.2s; `; container.onmouseover = () => container.style.background = 'rgba(0,0,0,0.1)'; container.onmouseout = () => container.style.background = 'rgba(0,0,0,0.05)'; const img = document.createElement('img'); img.src = icon; img.style.cssText = ` width: 16px; height: 16px; margin-right: 8px; `; const span = document.createElement('span'); span.textContent = rating || 'N/A'; span.style.cssText = ` font-weight: bold; color: #333; `; container.appendChild(img); container.appendChild(span); if (url) { container.addEventListener('click', () => window.open(url, '_blank')); } return container; } async function init() { console.log('Initializing script...'); if (!document.querySelector('.filmCoverSection__title')) { console.log('Title element not found, retrying...'); setTimeout(init, 100); return; } const titleElement = document.querySelector('.filmCoverSection__title'); const title = titleElement.textContent.trim(); const year = document.querySelector('.filmCoverSection__year')?.textContent.trim() || ''; console.log('Found title and year:', {title, year}); const ratingsContainer = document.createElement('div'); ratingsContainer.style.cssText = ` margin-top: 15px; display: flex; justify-content: left; flex-wrap: wrap; gap: 10px; `; const filmCoverSection = document.querySelector('.filmCoverSection__card'); if (!filmCoverSection) return; const placeholders = { imdb: createRatingElement(ICONS.imdb, 'Loading...', null), rotten: createRatingElement(ICONS.rotten, 'Loading...', null) }; Object.values(placeholders).forEach(el => ratingsContainer.appendChild(el)); filmCoverSection.appendChild(ratingsContainer); try { const urls = await getExternalUrls(title, year); console.log('External URLs:', urls); const [imdb, rotten] = await Promise.all([ getIMDbRating(urls.imdb), getRottenTomatoesRating(urls.rotten) ]); if (imdb) placeholders.imdb.replaceWith(createRatingElement(ICONS.imdb, imdb.rating, imdb.url)); if (rotten) placeholders.rotten.replaceWith(createRatingElement(ICONS.rotten, rotten.rating, rotten.url)); } catch (error) { console.error('Error fetching ratings:', error); } } if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', init); } else { init(); } })();