Greasy Fork

Filmweb External Ratings

Add IMDb, Rotten Tomatoes ratings to Filmweb

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

// ==UserScript==
// @name         Filmweb External Ratings
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  Add IMDb, Rotten Tomatoes ratings to Filmweb
// @author       mrkkr
// @match        http*://www.filmweb.pl/serial/*
// @match        http*://www.filmweb.pl/film/*
// @match        http*://www.filmweb.pl/tvshow/*
// @grant        GM_xmlhttpRequest
// @grant        GM.xmlhttpRequest
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_addStyle
// @connect      *
// @run-at       document-end
// ==/UserScript==

(function() {
    'use strict';

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

    const OMDB_API_KEY = '9dd3bf83';

    async function getMovieInfo() {
        const originalTitleElem = document.querySelector('.filmCoverSection__originalTitle');
        const localTitleElem = document.querySelector('.filmCoverSection__title');
        const yearElem = document.querySelector('.filmCoverSection__year');
        
        const originalTitle = originalTitleElem ? originalTitleElem.firstChild.textContent.trim() : null;
        const localTitle = localTitleElem ? localTitleElem.textContent.trim() : null;
        const year = yearElem ? yearElem.textContent.trim() : '';
        
        console.log('Found movie info:', {originalTitle, localTitle, year});
        
        return {
            title: originalTitle || localTitle,
            year: year
        };
    }

    async function getOMDBInfo(title, year) {
        const url = `https://www.omdbapi.com/?apikey=${OMDB_API_KEY}&t=${encodeURIComponent(title)}&y=${year}`;
        console.log('Searching OMDB:', url);

        return new Promise((resolve) => {
            GM_xmlhttpRequest({
                method: 'GET',
                url: url,
                headers: {'User-Agent': 'Mozilla/5.0'},
                onload: function(response) {
                    try {
                        const data = JSON.parse(response.responseText);
                        if (data.Response === 'True') {
                            console.log('OMDB data:', data);
                            resolve({
                                imdbID: data.imdbID,
                                imdbRating: data.imdbRating,
                                tomatoURL: data.tomatoURL,
                                tomatoRating: data.tomatoRating,
                                tomatoUserRating: data.tomatoUserRating
                            });
                        } else {
                            resolve(null);
                        }
                    } catch (e) {
                        console.error('Error parsing OMDB response:', e);
                        resolve(null);
                    }
                },
                onerror: () => resolve(null)
            });
        });
    }

    async function getIMDbRating(imdbId) {
        if (!imdbId) return null;

        const url = `https://www.imdb.com/title/${imdbId}/`;
        console.log('Fetching IMDb rating:', url);

        return new Promise((resolve) => {
            GM_xmlhttpRequest({
                method: 'GET',
                url: url,
                headers: {'User-Agent': 'Mozilla/5.0'},
                onload: function(response) {
                    const ratingMatch = response.responseText.match(/class="sc-bde20123-1[^>]+>([0-9.]+)<\/span>/);
                    if (ratingMatch) {
                        resolve({
                            rating: ratingMatch[1] + '/10',
                            url: url
                        });
                    } else {
                        resolve(null);
                    }
                },
                onerror: () => resolve(null)
            });
        });
    }

    async function getRTRating(rtUrl) {
        if (!rtUrl) return null;

        console.log('Fetching RT rating:', rtUrl);

        return new Promise((resolve) => {
            GM_xmlhttpRequest({
                method: 'GET',
                url: rtUrl,
                headers: {'User-Agent': 'Mozilla/5.0'},
                onload: function(response) {
                    const criticMatch = response.responseText.match(/score-board__tomatometer">(\d+)%/);
                    const audienceMatch = response.responseText.match(/score-board__audiencescore">(\d+)%/);
                    
                    if (criticMatch || audienceMatch) {
                        resolve({
                            rating: `${criticMatch ? criticMatch[1] + '%' : 'N/A'} 👥${audienceMatch ? audienceMatch[1] + '%' : 'N/A'}`,
                            url: rtUrl
                        });
                    } else {
                        resolve(null);
                    }
                },
                onerror: () => resolve(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...');
        
        while (!document.querySelector('.filmCoverSection__title')) {
            await new Promise(r => setTimeout(r, 100));
        }

        const {title, year} = await getMovieInfo();
        if (!title) return;

        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 {
            // Get IDs and basic info from OMDB
            const omdbInfo = await getOMDBInfo(title, year);
            if (!omdbInfo) {
                console.error('Movie not found on OMDB');
                return;
            }

            // Get current ratings from respective sites
            const imdb = await getIMDbRating(omdbInfo.imdbID);
            const rotten = await getRTRating(omdbInfo.tomatoURL);

            // Update UI
            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();
    }
})();