Douban Rating for Olevod

Display Douban rating for the videos on Olevod. Click on the rating to jump to the Douban page.

// ==UserScript==
// @name         Douban Rating for Olevod
// @name:zh-CN   欧乐影院添加豆瓣评分
// @namespace    https://github.com/HerTio/tmscripts
// @homepageURL  https://greasyfork.org/zh-CN/scripts/448977
// @supportURL   https://github.com/HerTio/tmscripts/issues
// @version      0.1.0
// @description  Display Douban rating for the videos on Olevod. Click on the rating to jump to the Douban page.
// @description:zh-CN  显示欧乐页面上影视作品的豆瓣评分。点击评分跳转至豆瓣页面查看详情。
// @author       HerTio
// @match        *://*.olevod.com/index.php*
// @require      https://code.jquery.com/jquery-3.6.0.min.js
// @icon         
// @connect      douban.com
// @license      GPL-3.0
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// ==/UserScript==

const DOUBAN_SEARCH_URL = 'https://www.douban.com/search?cat=1002&q=';
const RANKLIST_REQUEST_INTERVAL = 1000;
const VODLIST_REQUEST_INTERVAL = 2000;
const TODAY = Date.now();
const LOG_LEVEL = 'error';

(function () {
    'use strict';

    let config = loadConfig();
    initConfigOptionMenuCommand(config);

    if (config.main) {
        changeMainRating();
    }

    if (config.ranklist) {
        changeRankItemRating();
    }

    if (config.vodlist) {
        changeVodListItemRating();
    }
})();

async function getDoubanRating(id, title, timeout) {
    const key = id + '_douban'
    const data = GM_getValue(key);

    if (data && TODAY < data.expired) {
        return data.rating;
    }

    const rating = await requestDoubanRating(title, timeout);

    storeRatings(key, rating);

    return rating;
};

function requestDoubanRating(title, timeout) {
    const url = DOUBAN_SEARCH_URL + title;
    return new Promise(function (resolve, reject) {
        log().debug("Request: " + url);

        GM_xmlhttpRequest({
            "method": "GET",
            "url": url,
            "onload": (r) => {
                const response = $($.parseHTML(r.response));
                if (r.status !== 200) {
                    reject(new Error(`Response: ${r.status}: ${r.statusText}`));
                } else {
                    log().debug(`Response: ${r.status}: ${r.statusText}`);
                    try {
                        let msg = getDoubanRatingMessage(response, url);
                        setTimeout(() => resolve(msg), timeout);
                    } catch (error) {
                        reject(error);
                    }
                }
            }
        });
    });
}

function getDoubanRatingMessage(data, search_url) {
    const s = data.find('.result-list .result:first-child');

    if (s.length === 0) {
        throw Error("Douban search result not found.");
    }

    const rating_nums = s.find('.rating_nums').text() || '暂无评分';
    const douban_link = s.find('.content .title a').attr('href') || '';
    const get_url = () => {
        try {
            return (new URL(douban_link)).searchParams.get('url');
        } catch (error) {
            log.error(error.message);
            return search_url;
        }
    };
    const url = get_url();
    const matches = douban_link.match(/subject%2F(\d+)%2F/);
    const sid = matches ? matches[1] : "N/A";

    const rating_message = {
        rating_nums,
        url,
        sid
    };
    return rating_message;
}

function isDetailPage() {
    return /.+\/detail\/id\/\d+.*/.test(location.href);
}

function isPlayPage() {
    return /.+\/play\/id\/\d+.*/.test(location.href);
}

function changeMainRating() {
    // get title
    const title_obj = $('h2.title');
    const title = getMainTitle(title_obj);
    const id = getOleId(location.href);

    // Set main item rating.
    getDoubanRating(id, title)
        .then(data => {
            const douban_link = doubanRatingTag().mainRatingLink(data);
            setMainRating(douban_link);
        })
        .catch(err => {
            log().error(err);
            const douban_link = doubanRatingTag().search(DOUBAN_SEARCH_URL + title);
            setMainRating(douban_link);
        });
}

function changeRankItemRating() {
    // Set top 1 item rating.
    const top1_title = $('h4.title').text().trim();
    const top_1_link = $('li.ranklist_item a').attr('href') || '';
    const top_1_id = getOleId(top_1_link);

    getDoubanRating(top_1_id, top1_title)
        .then(data => {
            const rating = doubanRatingTag().link(data);
            setTopOneRating(rating);
        })
        .catch(err => {
            log().error(err);
            const data = { "url": DOUBAN_SEARCH_URL + top1_title, "rating_nums": "🔍" };
            const link = doubanRatingTag().link(data);
            setTopOneRating(link);
        });

    // Set the rating of the rest of the rank items
    $('li.part_eone a').each((i, a) => {
        const a_obj = $(a);
        const title = getTitleFromRankItem(a_obj);
        const link = a_obj.attr('href') || '';
        const id = getOleId(link);

        getDoubanRating(id, title, i * RANKLIST_REQUEST_INTERVAL)
            .then(data => {
                const rating = doubanRatingTag().link(data);
                setRankItemRating(a_obj, rating);
            })
            .catch(err => {
                log().error(err);
                const data = { "url": DOUBAN_SEARCH_URL + title, "rating_nums": "🔍" };
                const link = doubanRatingTag().link(data);
                setRankItemRating(a_obj, link);
            });
    });
}

function changeVodListItemRating() {
    // Set the rating of the vod list items
    getVodlistItemsObj().each((i, item) => {
        const item_obj = $(item);
        const title = item_obj.find('.vodlist_title a').attr('title') || '';
        const id = item_obj.find('a').attr('dids') || 0;

        getDoubanRating(id, title, i * VODLIST_REQUEST_INTERVAL)
            .then(data => {
                const rating = doubanRatingTag().unchanged(data);
                setVodListItemRating(item_obj, '⭐️ ' + rating);
            })
            .catch(err => {
                log().error(err);
                setVodListItemRating(item_obj, "");
            });
    });
}

function getOleId(url) {
    const matches = url.match(/id\/(\d+)\D*/);
    const id = matches ? matches[1] : 0;
    return id;
}

function getVodlistItemsObj() {
    if (isDetailPage()) {
        return $('.vodlist_sh .vodlist_item');
    } else if (isPlayPage()) {
        return $('.vodlist_sm .vodlist_item');
    }
}

function getMainTitle(title_obj) {
    let title = title_obj.clone();
    title.children().remove();
    return title.text().trim();
}

function setMainRating(rating) {
    if (isDetailPage()) {
        setMainRatingOnDetailPage(rating);
    } else if (isPlayPage()) {
        setMainRatingOnPlayPage(rating);
    }
}

function setMainRatingOnDetailPage(rating_element) {
    let rating_obj = $('#rating');
    const rating_num_obj = rating_obj.find('.star_tips');

    // exchange rating
    const rating = rating_num_obj.text();
    rating_obj.children().remove();
    let small_rating = $('.content_detail .data>.text_muted:first-child');
    small_rating.text("欧乐评分:" + rating);

    rating_obj.append(rating_element);
}

function setMainRatingOnPlayPage(rating_element) {
    let play_text_obj = $('.play_text');
    const rating_obj = play_text_obj.find('.text_score');
    const ole_rating = rating_obj.text();
    const replaced_text = play_text_obj.html().replace('豆瓣评分:', '欧乐评分:' + ole_rating);
    play_text_obj.html(replaced_text);

    let new_rating_obj = play_text_obj.find('.text_score');
    new_rating_obj.text('');
    new_rating_obj.append(rating_element);
}

function setTopOneRating(rating) {
    let top1_obj = $('.ranklist_item span.text_muted');
    top1_obj.empty();
    top1_obj.append(rating);
}

function setRankItemRating(a_obj, rating) {
    let item_obj = a_obj.find('span.text_muted');
    item_obj.empty();
    item_obj.append(rating);
}

function getTitleFromRankItem(item) {
    let item_clone = item.clone();
    item_clone.children().remove();
    return item_clone.text().trim();
}

function setVodListItemRating(item_obj, rating_element) {
    let rating_obj = item_obj.find('.text_right.text_dy');
    rating_obj.empty();
    rating_obj.text(rating_element);
}

function ratingTag(site) {
    return {
        "unchanged": (data) => data.rating_nums,
        "prefix": (data) => site + data.rating_nums,
        "link": (data) => `<a href="${data.url}" target="_blank">${data.rating_nums}</a>`,
        "mainRatingLink": (data) => `<a href="${data.url}" target="_blank"><span class="star_tips">${site}:${data.rating_nums}</span></a>`,
        "search": (url) => `<a href="${url}" target="_blank">🔍 ${site}搜索` + (site === "豆瓣" ? "(由于豆瓣的请求限制,可能需要登陆)" : "") + "</a>"
    };
}

function doubanRatingTag() {
    return ratingTag("豆瓣");
}

function getExpiredDate() {
    let expired = new Date();
    expired.setDate(expired.getDate() + 1);
    return expired.valueOf();
}

function storeRatings(key, rating) {
    let expired = getExpiredDate();
    const data = {
        rating,
        expired
    };

    GM_setValue(key, data);
}

function onOrOff(bool_value) {
    return bool_value ? '开' : '关';
}

function loadConfig() {
    return GM_getValue('config', {
        "main": true,
        "ranklist": true,
        "vodlist": false,
    });
}

function configOptionClick(option_key) {
    return () => {
        let config = loadConfig();
        config[option_key] = !config[option_key];
        GM_setValue('config', config);
        location.reload();
    };
}

function initConfigOptionMenuCommand(config) {
    GM_registerMenuCommand('主评分:' + onOrOff(config.main), configOptionClick("main"), 'm');
    GM_registerMenuCommand('排行榜评分:' + onOrOff(config.ranklist), configOptionClick("ranklist"), 'r');
    GM_registerMenuCommand('推荐评分:' + onOrOff(config.vodlist), configOptionClick("vodlist"), 'v');
}

function log() {
    return {
        "debug": (msg) => {
            if (LOG_LEVEL === 'debug') {
                console.log(msg);
            }
        },
        "error": (error) => {
            if (LOG_LEVEL === 'error' || LOG_LEVEL == 'debug') {
                console.error(error.message);
            }
        },
        "none": (_) => { }
    };
}