// ==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 // @connect imdb.com // @connect rottentomatoes.com // @connect metacritic.com // @downloadURL none // ==/UserScript== (function() { 'use strict'; const ICONS = { imdb: '', rotten: '', metacritic: '' }; async function getIMDbRating(title, year) { try { const imdbIdMatch = document.querySelector('a[href*="www.imdb.com/title/tt"]'); if (!imdbIdMatch) return null; const imdbId = imdbIdMatch.href.match(/tt\d+/)[0]; return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: 'GET', url: `https://www.imdb.com/title/${imdbId}/`, headers: { 'User-Agent': 'Mozilla/5.0' }, onload: function(response) { const ratingMatch = response.responseText.match(/"ratingValue":\s*"([^"]+)"/); if (!ratingMatch) resolve(null); resolve({ rating: ratingMatch[1], url: `https://www.imdb.com/title/${imdbId}/` }); }, onerror: function(error) { console.error('Error fetching IMDb rating:', error); resolve(null); } }); }); } catch (error) { console.error('Error fetching IMDb rating:', error); return null; } } async function getRottenTomatoesRating(title, year) { try { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: 'GET', url: `https://www.rottentomatoes.com/search?search=${encodeURIComponent(title)}`, headers: { 'User-Agent': 'Mozilla/5.0', 'Origin': 'https://www.rottentomatoes.com' }, onload: async function(response) { try { const rtUrlMatch = response.responseText.match(new RegExp(`href="([^"]+${year}[^"]+)"`)); if (!rtUrlMatch) { resolve(null); return; } const rtUrl = 'https://www.rottentomatoes.com' + rtUrlMatch[1]; GM_xmlhttpRequest({ method: 'GET', url: rtUrl, headers: { 'User-Agent': 'Mozilla/5.0', 'Origin': 'https://www.rottentomatoes.com' }, onload: function(movieResponse) { const ratingMatch = movieResponse.responseText.match(/tomatometer">(\d+)%/); if (!ratingMatch) { resolve(null); return; } resolve({ rating: ratingMatch[1] + '%', url: rtUrl }); }, onerror: function(error) { console.error('Error fetching RT movie page:', error); resolve(null); } }); } catch (error) { console.error('Error processing RT response:', error); resolve(null); } }, onerror: function(error) { console.error('Error fetching RT search:', error); resolve(null); } }); }); } catch (error) { console.error('Error in RT rating function:', error); return null; } } async function getMetacriticRating(title, year) { try { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: 'GET', url: `https://www.metacritic.com/search/movie/${encodeURIComponent(title)}/results`, headers: { 'User-Agent': 'Mozilla/5.0', 'Origin': 'https://www.metacritic.com' }, onload: async function(response) { try { const mcUrlMatch = response.responseText.match(new RegExp(`href="([^"]+${year}[^"]+)"`)); if (!mcUrlMatch) { resolve(null); return; } const mcUrl = 'https://www.metacritic.com' + mcUrlMatch[1]; GM_xmlhttpRequest({ method: 'GET', url: mcUrl, headers: { 'User-Agent': 'Mozilla/5.0', 'Origin': 'https://www.metacritic.com' }, onload: function(movieResponse) { const ratingMatch = movieResponse.responseText.match(/metascore_w[^>]+>(\d+) 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() { if (!document.querySelector('.filmCoverSection__title')) { setTimeout(init, 100); return; } const titleElement = document.querySelector('.filmCoverSection__title'); if (!titleElement) return; const title = titleElement.textContent.trim(); const year = document.querySelector('.filmCoverSection__year')?.textContent.trim() || ''; const ratingsContainer = document.createElement('div'); ratingsContainer.style.cssText = ` margin-top: 15px; display: flex; justify-content: center; 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), metacritic: createRatingElement(ICONS.metacritic, 'Loading...', null) }; Object.values(placeholders).forEach(el => ratingsContainer.appendChild(el)); filmCoverSection.appendChild(ratingsContainer); try { const [imdb, rotten, metacritic] = await Promise.all([ getIMDbRating(title, year), getRottenTomatoesRating(title, year), getMetacriticRating(title, year) ]); if (imdb) placeholders.imdb.replaceWith(createRatingElement(ICONS.imdb, imdb.rating, imdb.url)); if (rotten) placeholders.rotten.replaceWith(createRatingElement(ICONS.rotten, rotten.rating, rotten.url)); if (metacritic) placeholders.metacritic.replaceWith(createRatingElement(ICONS.metacritic, metacritic.rating, metacritic.url)); } catch (error) { console.error('Error fetching ratings:', error); } } init(); })();