Greasy Fork

Filmweb External Ratings

Add IMDb, Rotten Tomatoes, and Metacritic ratings to Filmweb

目前为 2025-03-29 提交的版本。查看 最新版本

// ==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
// ==/UserScript==

(function() {
    'use strict';

    const ICONS = {
        imdb: '',
        rotten: '',
        metacritic: ''
    };

    async function getIMDbRating(title, originalTitle, year) {
        try {
            console.log('Getting IMDb rating for:', {title, originalTitle, year});
            
            const imdbLinkElem = document.querySelector('a[href*="www.imdb.com/title/tt"]');
            if (imdbLinkElem) {
                const imdbId = imdbLinkElem.href.match(/tt\d+/)[0];
                console.log('Found IMDb ID from page link:', imdbId);
                
                return new Promise((resolve, reject) => {
                    GM_xmlhttpRequest({
                        method: 'GET',
                        url: `https://www.imdb.com/title/${imdbId}/`,
                        headers: {
                            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                        },
                        onload: function(response) {
                            console.log('IMDb page response:', response.status);
                            const ratingMatch = response.responseText.match(/ratingValue":{"@type":"AggregateRating"[^}]+"ratingValue":"([^"]+)"/);
                            if (ratingMatch) {
                                resolve({
                                    rating: ratingMatch[1] + '/10',
                                    url: `https://www.imdb.com/title/${imdbId}/`
                                });
                            } else {
                                const oldRatingMatch = response.responseText.match(/class="ipc-button__text"[^>]*>([0-9.]+)<\/span>/);
                                if (oldRatingMatch) {
                                    resolve({
                                        rating: oldRatingMatch[1] + '/10',
                                        url: `https://www.imdb.com/title/${imdbId}/`
                                    });
                                } else {
                                    resolve(null);
                                }
                            }
                        },
                        onerror: (error) => {
                            console.error('IMDb page fetch error:', error);
                            resolve(null);
                        }
                    });
                });
            }
            
            const searchTitle = originalTitle || title;
            const searchUrl = `https://www.imdb.com/find?q=${encodeURIComponent(searchTitle)}+${year}&s=tt`;
            console.log('Searching IMDb:', searchUrl);
            
            return new Promise((resolve, reject) => {
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: searchUrl,
                    headers: {
                        'User-Agent': 'Mozilla/5.0'
                    },
                    onload: function(response) {
                        const match = response.responseText.match(/href="\/title\/(tt\d+)/);
                        if (!match) {
                            console.log('No IMDb search results');
                            resolve(null);
                            return;
                        }

                        const imdbId = match[1];
                        console.log('Found IMDb ID from search:', imdbId);

                        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":{"@type":"AggregateRating"[^}]+"ratingValue":"([^"]+)"/);
                                if (ratingMatch) {
                                    resolve({
                                        rating: ratingMatch[1] + '/10',
                                        url: `https://www.imdb.com/title/${imdbId}/`
                                    });
                                } else {
                                    resolve(null);
                                }
                            }
                        });
                    }
                });
            });
        } catch (error) {
            console.error('Error in IMDb function:', error);
            return null;
        }
    }

    async function getRottenTomatoesRating(title, originalTitle, year) {
        try {
            const searchTitle = originalTitle || title;
            console.log('Getting RT rating for:', {title: searchTitle, year});
            
            const movieSlug = searchTitle.toLowerCase()
                .replace(/[^a-z0-9]+/g, '_')
                .replace(/^_+|_+$/g, '');
            
            const directUrl = `https://www.rottentomatoes.com/m/${movieSlug}_${year}`;
            console.log('Trying direct RT URL:', directUrl);

            return new Promise((resolve, reject) => {
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: directUrl,
                    headers: {
                        'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                    },
                    onload: function(response) {
                        if (response.status === 200) {
                            const scoreMatch = response.responseText.match(/"ratingValue":"(\d+)"/);
                            if (scoreMatch) {
                                resolve({
                                    rating: scoreMatch[1] + '%',
                                    url: directUrl
                                });
                                return;
                            }
                        }
                        
                        GM_xmlhttpRequest({
                            method: 'GET',
                            url: `https://www.rottentomatoes.com/search?search=${encodeURIComponent(searchTitle)}`,
                            headers: {
                                'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                            },
                            onload: function(searchResponse) {
                                try {
                                    const movieMatch = searchResponse.responseText.match(
                                        new RegExp(`href="(/m/[^"]+${year}[^"]*)"[^>]*>([^<]+)</a>`)
                                    );
                                    if (!movieMatch) {
                                        console.log('No RT search results');
                                        resolve(null);
                                        return;
                                    }

                                    const movieUrl = 'https://www.rottentomatoes.com' + movieMatch[1];
                                    console.log('Found RT movie:', movieMatch[2], movieUrl);

                                    GM_xmlhttpRequest({
                                        method: 'GET',
                                        url: movieUrl,
                                        headers: {
                                            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                                        },
                                        onload: function(movieResponse) {
                                            const scoreMatch = movieResponse.responseText.match(/"ratingValue":"(\d+)"/);
                                            if (scoreMatch) {
                                                resolve({
                                                    rating: scoreMatch[1] + '%',
                                                    url: movieUrl
                                                });
                                            } else {
                                                resolve(null);
                                            }
                                        }
                                    });
                                } catch (error) {
                                    console.error('Error processing RT response:', error);
                                    resolve(null);
                                }
                            }
                        });
                    }
                });
            });
        } catch (error) {
            console.error('Error in RT function:', error);
            return null;
        }
    }

    async function getMetacriticRating(title, originalTitle, year) {
        try {
            console.log('Getting Metacritic rating for:', {title, originalTitle, year});
            
            const searchTitle = (originalTitle || title).toLowerCase()
                .replace(/[^\w\s-]/g, '')
                .replace(/\s+/g, '-');
            
            const directUrl = `https://www.metacritic.com/movie/${searchTitle}`;
            console.log('Trying Metacritic URL:', directUrl);

            return new Promise((resolve, reject) => {
                GM_xmlhttpRequest({
                    method: 'GET',
                    url: directUrl,
                    headers: {
                        'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                    },
                    onload: function(response) {
                        if (response.status === 200) {
                            const scoreMatch = response.responseText.match(/metascore_w[^>]+>(\d+)</);
                            if (scoreMatch) {
                                resolve({
                                    rating: scoreMatch[1],
                                    url: directUrl
                                });
                                return;
                            }
                        }

                        const searchUrl = `https://www.metacritic.com/search/movie/${encodeURIComponent(searchTitle)}/results`;
                        console.log('Searching Metacritic:', searchUrl);

                        GM_xmlhttpRequest({
                            method: 'GET',
                            url: searchUrl,
                            headers: {
                                'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                            },
                            onload: function(searchResponse) {
                                try {
                                    const movieMatch = searchResponse.responseText.match(
                                        new RegExp(`href="([^"]+${year}[^"]*)"[^>]*>([^<]+)</a>`)
                                    );
                                    if (!movieMatch) {
                                        console.log('No Metacritic search results');
                                        resolve(null);
                                        return;
                                    }

                                    const movieUrl = 'https://www.metacritic.com' + movieMatch[1];
                                    console.log('Found Metacritic movie:', movieMatch[2], movieUrl);

                                    GM_xmlhttpRequest({
                                        method: 'GET',
                                        url: movieUrl,
                                        headers: {
                                            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
                                        },
                                        onload: function(movieResponse) {
                                            const scoreMatch = movieResponse.responseText.match(/metascore_w[^>]+>(\d+)</);
                                            if (scoreMatch) {
                                                resolve({
                                                    rating: scoreMatch[1],
                                                    url: movieUrl
                                                });
                                            } else {
                                                resolve(null);
                                            }
                                        }
                                    });
                                } catch (error) {
                                    console.error('Error processing Metacritic response:', error);
                                    resolve(null);
                                }
                            }
                        });
                    }
                });
            });
        } catch (error) {
            console.error('Error in Metacritic 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 originalTitle = document.querySelector('.filmCoverSection__originalTitle')?.firstChild?.textContent?.trim();
        const title = document.querySelector('.filmCoverSection__title')?.textContent?.trim();
        const year = document.querySelector('.filmCoverSection__year')?.textContent?.trim() || '';

        console.log('Found titles:', {title, originalTitle, 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),
            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, originalTitle, year),
                getRottenTomatoesRating(title, originalTitle, year),
                getMetacriticRating(title, originalTitle, 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);
        }
    }

    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', init);
    } else {
        init();
    }
})();