// ==UserScript== // @name DoubanRatingForMovie // @name:zh-CN 在线电影添加豆瓣评分 // @namespace https://github.com/ciphersaw/DoubanRatingForMovie // @version 1.2.2 // @description Display Douban rating for online movies such as Tencent Video, iQIYI, Youku and so on. // @description:zh-CN 在腾讯视频、爱奇艺、优酷等主流电影网站上显示豆瓣评分。 // @author CipherSaw // @match *://*.olehdtv.com/index.php* // @match *://*.olevod.com/details* // @match *://*.olevod.com/player/vod/* // @match *://*.olevod.tv/details* // @match *://*.olevod.tv/player/vod/* // @match *://v.qq.com/x/cover/* // @match *://www.iqiyi.com/v_* // @match *://v.youku.com/v_show/* // @require https://code.jquery.com/jquery-3.6.0.min.js // @icon data:image/gif;base64,R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw== // @connect douban.com // @license GPL-3.0 // @grant GM_xmlhttpRequest // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @grant GM_listValues // @supportURL https://github.com/ciphersaw/DoubanRatingForMovie/issues // @downloadURL none // ==/UserScript== 'use strict'; const LOG_LEVELS = { NONE: 0, ERROR: 1, INFO: 2, DEBUG: 3 }; class Logger { constructor(initialLevel = 'INFO') { this.currentLogLevel = LOG_LEVELS[initialLevel] || LOG_LEVELS.INFO; } error(...args) { if (this.currentLogLevel >= LOG_LEVELS.ERROR) { console.error(...args); } } info(...args) { if (this.currentLogLevel >= LOG_LEVELS.INFO) { console.info(...args); } } debug(...args) { if (this.currentLogLevel >= LOG_LEVELS.DEBUG) { console.debug(...args); } } } const logger = new Logger('INFO'); const TERM_OF_VALID_CACHE = 1; const PERIOD_OF_CLEARING_CACHE = 1; const DOUBAN_RATING_API = 'https://www.douban.com/search?cat=1002&q='; (function () { clearExpiredCache(); const host = location.hostname; if (host === 'www.olehdtv.com') { OLEHDTV_setRating(); } else if (host === 'www.olevod.com' || host === 'www.olevod.tv') { // Both main site and test site. OLEVOD_setRating(); } else if (host === 'v.qq.com') { VQQ_setRating(); } else if (host === 'www.iqiyi.com') { IQIYI_setRating(); } else if (host === 'v.youku.com') { YOUKU_setRating(); } })(); // ==OLEHDTV== function OLEHDTV_setRating() { const id = OLEHDTV_getID(); const title = OLEHDTV_getTitle(); const director = OLEHDTV_getDirector(); const year = OLEHDTV_getYear(); getDoubanRating(`olehdtv_${id}`, title, director, year) .then(data => { OLEHDTV_setMainRating(data.ratingNums, data.url); }) .catch(err => { OLEHDTV_setMainRating("N/A", DOUBAN_RATING_API + encodeSpaces(title)); }); } function OLEHDTV_getID() { const id = /id\/(\d+)/.exec(location.href); return id ? id[1] : 0; } function OLEHDTV_getTitle() { // Remove the annotated suffix of title. const suffixRegex = /【.*】$/; let clone = $('h2.title').clone(); clone.children().remove(); return clone.text().trim().replace(suffixRegex, ''); } function OLEHDTV_getDirector() { let selector = ''; if (OLEHDTV_isDetailPage()) { selector = '.content_min li.data:last'; } else if (OLEHDTV_isPlayPage()) { selector = '.play_content p:first-child'; } const directorText = $(selector).text().trim(); const directors = /^导演:(.+)$/.exec(directorText); if (directors) { const array = directors[1].split(/\s+/); return array[0]; } else { return ''; } } function OLEHDTV_getYear() { let selector = ''; if (OLEHDTV_isDetailPage()) { selector = 'ul li.data:first-child'; } else if (OLEHDTV_isPlayPage()) { selector = '.play_text a'; } const yearText = $(selector).text().trim(); const year = /\d{4}/.exec(yearText); return year ? year[0] : ''; } function OLEHDTV_setMainRating(ratingNums, url) { const doubanLink = `豆瓣评分:${ratingNums}`; if (OLEHDTV_isDetailPage()) { let ratingObj = $('.content_detail .data>.text_muted:first-child'); ratingObj.empty(); ratingObj.append(doubanLink); } else if (OLEHDTV_isPlayPage()) { let ratingObj = $('.play_text .nstem'); const replacedHTML = ratingObj.html().replace('豆瓣评分:', ''); ratingObj.html(replacedHTML); ratingObj.append(doubanLink); } } function OLEHDTV_isDetailPage() { return /.+\/vod\/detail\/id\/\d+.*/.test(location.href); } function OLEHDTV_isPlayPage() { return /.+\/vod\/play\/id\/\d+.*/.test(location.href); } // ==OLEVOD== async function OLEVOD_setRating() { const id = OLEVOD_getID(); let title = ''; try { title = await OLEVOD_waitForTitle(1000, 10); } catch (error) { logger.error(`OLEVOD_waitForTitle: id=${id} error=${error}`); return; } // Note that director and year must be selected after OLEVOD_waitForTitle, // which means elements have been already loaded and do not need to wait again. const director = OLEVOD_getDirector(); const year = OLEVOD_getYear(); getDoubanRating(`olevod_${id}`, title, director, year) .then(data => { OLEVOD_setMainRating(data.ratingNums, data.url); }) .catch(err => { OLEVOD_setMainRating("N/A", DOUBAN_RATING_API + encodeSpaces(title)); }); } function OLEVOD_getID() { const id = /\d{1}-\d{4,5}/.exec(location.href); return id ? id[0] : 0; } function OLEVOD_waitForTitle(delay, iterations) { let selector = ''; if (OLEVOD_isDetailPage()) { selector = ".pc-container .info .title"; } else if (OLEVOD_isPlayPage()) { selector = ".el-tabs__content .tab-label"; } return new Promise((resolve, reject) => { let count = 0; const intervalID = setInterval(() => { count++; if (count === iterations) { const error = new Error(`ResolveError: title is not found and iterations have reached the maximum`); clearInterval(intervalID); reject(error); } const obj = $(selector); if (obj.length > 0) { const title = OLEVOD_resolveTitle(obj); if (title !== "") { clearInterval(intervalID); resolve(title); } } }, delay); }); } function OLEVOD_resolveTitle(obj) { // Remove the annotated suffix of title. const suffixRegex = /【.*】$/; if (OLEVOD_isDetailPage()) { return obj.text().trim().replace(suffixRegex, ''); } else if (OLEVOD_isPlayPage()) { const clone = obj.clone(); clone.children().remove(); return clone.text().trim().replace(suffixRegex, ''); } } function OLEVOD_getDirector() { if (OLEVOD_isDetailPage()) { const directorText = $('.pc-container .info p:nth-of-type(2)').text().trim(); const directors = /^导演:(.+)$/.exec(directorText); if (directors) { const array = directors[1].split(/\s*\/\s*/); return array[0]; } else { return ''; } } else if (OLEVOD_isPlayPage()) { // Director is not found in the movie play page. return ''; } } function OLEVOD_getYear() { let selector = ''; if (OLEVOD_isDetailPage()) { selector = '.pc-container .info .label'; } else if (OLEVOD_isPlayPage()) { selector = '.el-tabs__content .tab-label p.wes'; } const yearText = $(selector).text().trim(); const year = /\d{4}/.exec(yearText); return year ? year[0] : ''; } function OLEVOD_setMainRating(ratingNums, url) { if (OLEVOD_isDetailPage()) { let ratingObj = $('.pc-container .info .label:first-child'); ratingObj.before(`豆瓣评分:${ratingNums}`); // Set MutationObserver for the title element of current page. const titleObj = $('.pc-container .info .title'); const originalText = titleObj.text().trim(); if (titleObj.length > 0) { const observer = new MutationObserver(observerCallback); observer.observe(titleObj[0], { subtree: true, characterData: true }); // Stop watching for mutations before page is unloaded. window.onbeforeunload = () => { if (observer) { observer.disconnect(); } }; function observerCallback(mutations, observer) { mutations.forEach(function (mutation) { // Check if the character data is changed. if (mutation.type === 'characterData') { const changedText = mutation.target.data.trim(); // If the movie page is reloaded by AJAX, // remove the Douban rating of current page and reset for the new page. if (originalText !== changedText) { let ratingObj = $('.pc-container .info .label:first-child'); if (/豆瓣/.test(ratingObj.text().trim())) { ratingObj.remove(); } observer.disconnect(); OLEVOD_setRating(); } } }); } } } else if (OLEVOD_isPlayPage()) { let ratingObj = $('#pane-first .tab-label .wes'); const clone = ratingObj.clone(); clone.children().remove(); const originalText = clone.text().trim(); const array = originalText.split(/ +/); if (array.length === 2) { const revisedHTML = `${array[0]} 豆瓣${ratingNums}/${array[1]}`; ratingObj.html(revisedHTML); } } } function OLEVOD_isDetailPage() { return /.+\/details-\d{1}-\d{4,5}\.html/.test(location.href); } function OLEVOD_isPlayPage() { return /.+\/player\/vod\/\d{1}-\d{4,5}-\d{1}\.html/.test(location.href); } // ==VQQ== function VQQ_setRating() { const id = VQQ_getID(); const title = VQQ_getTitle(); // It is hard to get director in VQQ, so set them to null temporarily. const director = ''; const year = VQQ_getYear(); getDoubanRating(`vqq_${id}`, title, director, year) .then(data => { VQQ_setMainRating(data.ratingNums, data.url); }) .catch(err => { VQQ_setMainRating("N/A", DOUBAN_RATING_API + encodeSpaces(title)); }); } function VQQ_getID() { const id = /x\/cover\/(\S+)\//.exec(location.href); return id ? id[1] : 0; } function VQQ_getTitle() { // Remove the annotated suffix of title. const suffixRegex = /\[.*\]$/; const title = $('h1.playlist-intro__title'); return title.text().trim().replace(suffixRegex, ''); } function VQQ_getYear() { const yearText = $('span.playlist-intro-info__item').text(); const year = /\S*· (\d{4}) ·\S*/.exec(yearText); return year ? year[1] : ''; } function VQQ_setMainRating(ratingNums, url) { let ratingObj = $('h1.playlist-intro__title'); ratingObj.after(`豆瓣${ratingNums}`); } // ==IQIYI== async function IQIYI_setRating() { const id = IQIYI_getID(); let title = ''; try { title = await IQIYI_waitForTitle(1000, 10); } catch (error) { logger.error(`IQIYI_waitForTitle: id=${id} error=${error}`); return; } // It is hard to get director and year in IQIYI, so set them to null temporarily. const director = ''; const year = ''; getDoubanRating(`iqiyi_${id}`, title, director, year) .then(data => { IQIYI_setMainRating(data.ratingNums, data.url); }) .catch(err => { IQIYI_setMainRating("N/A", DOUBAN_RATING_API + encodeSpaces(title)); }); } function IQIYI_getID() { const id = /v_(\S+).html/.exec(location.href); return id ? id[1] : 0; } function IQIYI_waitForTitle(delay, iterations) { const selector = '.meta_titleNotCloud__O2Ffr'; return new Promise((resolve, reject) => { let count = 0; const intervalID = setInterval(() => { count++; if (count === iterations) { const error = new Error(`ResolveError: title is not found and iterations have reached the maximum`); clearInterval(intervalID); reject(error); } const obj = $(selector); if (obj.length > 0) { const title = obj.text().trim(); if (title !== "") { clearInterval(intervalID); resolve(title); } } }, delay); }); } function IQIYI_setMainRating(ratingNums, url) { let count = 0; const intervalID = setInterval(() => { const obj = $('#doubanRating'); if (obj.length === 0) { count = 0; // Set the align-items to center, for the parent div element with flex layout. let flexObj = $('.meta_titleContent__cUi2t'); flexObj.css("align-items", "center"); // Insert rating div element after title div element. let ratingObj = $('.meta_titleNotCloud__O2Ffr'); ratingObj.after(`
`); } else { count++; } // If rating div element is not overwritten and removed in 10s, then clear interval. if (count === 10) { clearInterval(intervalID); } }, 1000); } // ==YOUKU== async function YOUKU_setRating() { const id = YOUKU_getID(); const title = YOUKU_getTitle(); let director = ''; try { director = await YOUKU_waitForDirector(1000, 10); } catch (error) { logger.error(`YOUKU_waitForDirector: id=${id} error=${error}`); return; } const year = YOUKU_getYear(); getDoubanRating(`youku_${id}`, title, director, year) .then(data => { YOUKU_setMainRating(data.ratingNums, data.url); }) .catch(err => { YOUKU_setMainRating("N/A", DOUBAN_RATING_API + encodeSpaces(title)); }); } function YOUKU_getID() { const id = /id_(\S+).html/.exec(location.href); return id ? id[1] : 0; } function YOUKU_getTitle() { const title = $('h3.new-title-name'); return title.text().trim(); } function YOUKU_waitForDirector(delay, iterations) { const selector = '.starBox .star .starName:first'; return new Promise((resolve, reject) => { let count = 0; const intervalID = setInterval(() => { count++; if (count === iterations) { const error = new Error(`ResolveError: title is not found and iterations have reached the maximum`); clearInterval(intervalID); reject(error); } const obj = $(selector); if (obj.length > 0) { const director = obj.text().trim(); if (director !== "") { clearInterval(intervalID); resolve(director); } } }, delay); }); } function YOUKU_getYear() { const yearText = $('.new-title-name-left span:last-child').text(); const year = /\S*·(\d{4})·\S*/.exec(yearText); return year ? year[1] : ''; } function YOUKU_setMainRating(ratingNums, url) { let ratingObj = $('.new-title-name-left span:last-child'); const originalText = ratingObj.text().trim(); const revisedHTML = `豆瓣${ratingNums}·${originalText}`; const revisedAttr = `豆瓣${ratingNums}·${originalText}`; ratingObj.html(revisedHTML); ratingObj.attr('title', revisedAttr); } // ==COMMON== function clearExpiredCache() { const t = GM_getValue('clear_time'); if (!t || !isValidTime(new Date(t), PERIOD_OF_CLEARING_CACHE)) { logger.info(`clearExpiredCache: clear_time=${t}`); const idList = GM_listValues(); idList.forEach(function (id) { // Delete the expired IDs periodically. const data = GM_getValue(id); if (data.uptime && !isValidTime(new Date(data.uptime), TERM_OF_VALID_CACHE)) { GM_deleteValue(id); } }); GM_setValue('clear_time', new Date().toISOString()); } } async function getDoubanRating(key, title, director, year) { const data = GM_getValue(key); if (data && isValidTime(new Date(data.uptime), TERM_OF_VALID_CACHE)) { logger.info(`getDoubanRating: title=${title} rating=${data.ratingData.ratingNums} uptime=${data.uptime}`); return data.ratingData; } const url = DOUBAN_RATING_API + encodeSpaces(title); logger.info(`getDoubanRating: title=${title} searchURL=${url}`); const ratingData = await new Promise((resolve, reject) => { GM_xmlhttpRequest({ "method": "GET", "url": url, "onload": (r) => { const response = $($.parseHTML(r.response)); if (r.status !== 200) { const error = new Error(`StatusError: response status is ${r.status} and message is ${r.statusText}`); reject(error); } else { try { let data = resolveDoubanRatingResult(url, director, year, response); logger.info(`getDoubanRating: title=${title} rating=${data.ratingNums}`); resolve(data); } catch (error) { logger.error(`getDoubanRating: title=${title} error=${error}`); reject(error); } } } }); }); cacheDoubanRatingData(key, ratingData); return ratingData; } function isValidTime(uptime, term) { const oneDayMillis = 24 * 60 * 60 * 1000; const nowDate = new Date(); const diffMillis = nowDate.getTime() - uptime.getTime(); return diffMillis < oneDayMillis * term; } function cacheDoubanRatingData(key, ratingData) { const uptime = new Date().toISOString(); const data = { ratingData, uptime }; GM_setValue(key, data); } function resolveDoubanRatingResult(searchURL, director, year, data) { const s = getDoubanRatingItem(director, year, data); if (s === null) { throw Error("ResolveError: search result is not found"); } const ratingNums = s.find('.rating_nums').text() || '暂无评分'; const doubanLink = s.find('.content .title a').attr('href') || ''; let url = resolveDoubanURL(doubanLink); if (url === "") { url = searchURL; } const ratingData = { ratingNums, url } return ratingData; } function getDoubanRatingItem(director, year, data) { let item = null; if (director === '' && year === '') { item = data.find('.result-list .result:first-child'); } else { const list = data.find('.result-list').children(); list.each(function () { const info = $(this).find('.subject-cast').text(); const array = info.split(/\s*\/\s*/); // e.g. ['原名:毕业那年', '姚宇', '顾莉雅', '2012'] if (array.length > 0 && array[0].includes('原名')) { array.shift(); } let releaseYear = null; if (/^\d{4}$/.test(array[array.length - 1])) { releaseYear = array.pop(); } if (director !== '' && array.indexOf(director) === -1) { return true; } if (year !== '' && releaseYear !== year) { return true; } item = $(this); return false; }); if (item === null) { item = data.find('.result-list .result:first-child'); } } return item; } function resolveDoubanURL(doubanLink) { try { return (new URL(doubanLink)).searchParams.get('url'); } catch (error) { logger.error(`resolveDoubanURL: error=${error.message}`); return ""; } } function encodeSpaces(text) { return text.replace(/ /g, '%20'); }