Ratings on IMDb

Adds ratings from Rotten Tomatoes and Metacritic to IMDb

As of 2021-08-29. See the latest version.

// ==UserScript==
// @name            Ratings on IMDb
// @name:it         Valutazioni su IMDb
// @author          Davide <iFelix18@protonmail.com>
// @namespace       https://github.com/iFelix18
// @icon            https://www.google.com/s2/favicons?sz=64&domain=imdb.com
// @description     Adds ratings from Rotten Tomatoes and Metacritic to IMDb
// @description:it  Aggiunge valutazioni da Rotten Tomatoes e Metacritic a IMDb
// @copyright       2021, Davide (https://github.com/iFelix18)
// @license         MIT
// @version         1.0.5
//
// @homepageURL     https://github.com/iFelix18/Userscripts#readme
// @supportURL      https://github.com/iFelix18/Userscripts/issues
//
// @require         https://cdn.jsdelivr.net/gh/sizzlemctwizzle/GM_config@43fd0fe4de1166f343883511e53546e87840aeaf/gm_config.min.js
// @require         https://cdn.jsdelivr.net/gh/iFelix18/Userscripts@abce8796cedbe28ac8e072d9824c4b9342985098/lib/utils/utils.min.js
// @require         https://cdn.jsdelivr.net/gh/iFelix18/Userscripts@bced30119a3304aff1c4f71c77bd1781cefde396/lib/api/omdb.min.js
// @require         https://cdn.jsdelivr.net/npm/gm4-polyfill@1.0.1/gm4-polyfill.min.js#sha256-qmLl2Ly0/+2K+HHP76Ul+Wpy1Z41iKtzptPD1Nt8gSk=
// @require         https://cdn.jsdelivr.net/npm/jquery@3.6.0/dist/jquery.min.js#sha256-/xUj+3OJU5yExlq6GSYGSHk7tPXikynS7ogEvDej/m4=
// @require         https://cdn.jsdelivr.net/npm/handlebars@4.7.7/dist/handlebars.min.js#sha256-ZSnrWNaPzGe8v25yP0S6YaMaDLMTDHC+4mHTw0xydEk=
//
// @match           *://www.imdb.com/title/*
// @connect         omdbapi.com
//
// @grant           GM.deleteValue
// @grant           GM.getValue
// @grant           GM.info
// @grant           GM.listValues
// @grant           GM.registerMenuCommand
// @grant           GM.setValue
// @grant           GM.xmlHttpRequest
//
// @grant           GM_deleteValue
// @grant           GM_getValue
// @grant           GM_info
// @grant           GM_listValues
// @grant           GM_registerMenuCommand
// @grant           GM_setValue
// @grant           GM_xmlhttpRequest
//
// @run-at          document-start
// @inject-into     page
// ==/UserScript==

/* global $, GM_config, Handlebars, MonkeyUtils, OMDb */

(() => {
  'use strict'

  //* GM_config
  GM_config.init({
    id: 'ratings-config',
    title: `${GM.info.script.name} v${GM.info.script.version} Settings`,
    fields: {
      OMDbApiKey: {
        label: 'OMDb API Key',
        section: ['You can request a free OMDb API Key at:', 'https://www.omdbapi.com/apikey.aspx'],
        type: 'text',
        title: 'Your OMDb API Key',
        size: 70,
        default: ''
      },
      magic: {
        label: 'Remove',
        section: ['Remove old data from the cache'],
        type: 'button',
        click: async () => {
          const values = await GM.listValues()

          values.forEach(async (value) => {
            const cache = await GM.getValue(value) // get cache
            if ((Date.now() - cache.time) > cachePeriod) { GM.deleteValue(value) } // delete old cache
          })

          GM_config.close()
        }
      },
      logging: {
        label: 'Logging',
        section: ['Develop'],
        labelPos: 'above',
        type: 'checkbox',
        default: false
      },
      debugging: {
        label: 'Debugging',
        labelPos: 'above',
        type: 'checkbox',
        default: false
      }
    },
    /* cspell: disable-next-line */
    css: '#ratings-config{background-color:#343434;color:#fff}#ratings-config *{font-family:varela round,helvetica neue,Helvetica,Arial,sans-serif}#ratings-config .section_header{background-color:#282828;border:1px solid #282828;border-bottom:none;color:#fff;font-size:10pt}#ratings-config .section_desc{background-color:#282828;border:1px solid #282828;border-top:none;color:#fff;font-size:10pt}#ratings-config #ratings-config_field_magic{margin:0 auto;display:block}#ratings-config .reset{color:#fff}',
    events: {
      init: () => {
        if (!GM_config.isOpen && GM_config.get('OMDbApiKey') === '') {
          window.onload = () => GM_config.open()
        }
      },
      save: () => {
        if (!GM_config.isOpen && GM_config.get('OMDbApiKey') === '') {
          window.alert(`${GM.info.script.name}: check your settings and save`)
        } else {
          window.alert(`${GM.info.script.name}: settings saved`)
          GM_config.close()
          window.location.reload(false)
        }
      }
    }
  })
  GM.registerMenuCommand('Configure', () => GM_config.open())

  //* MonkeyUtils
  const MU = new MonkeyUtils({
    name: GM.info.script.name,
    version: GM.info.script.version,
    author: GM.info.script.author,
    color: '#ff0000',
    logging: GM_config.get('logging')
  })
  MU.init('ratings-config')

  //* OMDb API
  const omdb = new OMDb({
    apikey: GM_config.get('OMDbApiKey'),
    debug: GM_config.get('debugging')
  })

  //* Handlebars
  Handlebars.registerHelper('ifEqual', function (a, b, options) {
    if (a === b) return options.fn(this)
    return options.inverse(this)
  })

  //* Constants
  const cachePeriod = 3600000 // 1 hours
  const logos = {
    metacritic: '',
    rotten: '',
    fresh: ''
  }

  //* Functions
  /**
   * Returns IMDb ID
   * @returns {string}
   */
  const getID = () => {
    return $('meta[property="imdb:pageConst"]').first().attr('content')
  }

  /**
   * Returns ratings from OMDb
   * @param {*} id IMDb ID
   * @returns {Promise}
   */
  const getRatings = async (id) => {
    const cache = await GM.getValue(id) // get cache

    return new Promise((resolve, reject) => {
      if (cache !== undefined && ((Date.now() - cache.time) < cachePeriod)) { // cache valid
        MU.log('data from cache')
        resolve(elaborateResponse(cache.response))
      } else { // cache not valid
        omdb.get({
          id: id
        }).then((response) => {
          GM.setValue(id, { response, time: Date.now() }) // set cache
          MU.log('data from OMDb')
          resolve(elaborateResponse(response))
        })
      }
    }).catch((error) => MU.error(error))
  }

  /**
   * Returns elaborated response
   * @param {Object} response
   * @returns {Object}
   */
  const elaborateResponse = (response) => {
    return ([
      {
        logo: (response.Ratings[1] !== undefined) ? ((parseFloat(response.Ratings[1].Value) < 60) ? logos.rotten : logos.fresh) : logos.fresh,
        rating: (response.Ratings[1] !== undefined) ? ((response.Ratings[1].Source === 'Rotten Tomatoes') ? response.Ratings[1].Value.replace(/%/g, '') : 'N/A') : 'N/A',
        symbol: '%',
        title: 'TOMATOMETER',
        url: response.tomatoURL,
        votes: (response.Ratings[1] !== undefined) ? ((parseFloat(response.Ratings[1].Value) < 60) ? 'Rotten' : 'Fresh') : 'N/A'
      },
      {
        logo: logos.metacritic,
        rating: response.Metascore,
        symbol: '',
        title: 'METASCORE',
        url: 'criticreviews',
        votes: (response.Metascore < 40) ? '#ff0000' : (response.Metascore >= 40 && response.Metascore <= 60) ? '#ffcc33' : '#66cc33'
      }
    ])
  }

  /**
   * Add template
   */
  const addTemplate = () => {
    /* cspell: disable-next-line */
    const template = '<div class="external-ratings idYUsR"style=margin-right:.5rem></div><script id=external-ratings-template type=text/x-handlebars-template>{{#each ratings}}<div class="jQXoLQ rating-bar__base-button"> <div class="bufoWn">{{this.title}}</div><a class="ipc-button ipc-text-button ipc-button--core-baseAlt ipc-button--on-textPrimary jjcqHZ" role="button"{{#ifEqual this.url "N/A"}}{{else}}href="{{this.url}}{{/ifEqual}}"> <div class="ipc-button__text"> <div class="jodtvN"> <div class="dwhzFZ"> <img src="{{this.logo}}" alt="logo" width="24"> </div><div class="hmJkIS"> <div class="bmbYRW"> <span class="iTLWoV">{{this.rating}}</span>{{#ifEqual this.rating "N/A"}}<span></span>{{else}}<span>{{this.symbol}}</span>{{/ifEqual}}</div><div class="fKXaGo"></div>{{#ifEqual this.rating "N/A"}}<div class="jkCVKJ"></div>{{else}}{{#ifEqual this.title "TOMATOMETER"}}<div class="jkCVKJ">{{this.votes}}</div>{{else}}<div class="jkCVKJ" style="background-color:{{this.votes}};height: 10px;width: 100%;margin-top: 3px;margin-bottom: 3px;"></div>{{/ifEqual}}{{/ifEqual}}</div></div></div></a> </div>{{/each}}</script>'
    const target = '.hglRHk div[class^="RatingBar__ButtonContainer"] div[class^="RatingBarButtonBase__ContentWrap"]:nth-child(1)'

    $(template).insertAfter(target)
  }

  //* Script
  $(document).ready(() => {
    const id = getID()

    if (!id) return

    MU.log(`ID is '${id}'`)

    addTemplate()
    getRatings(id).then((response) => {
      MU.log(response)

      const template = Handlebars.compile($('#external-ratings-template').html())
      const context = { ratings: response }
      const compile = template(context)
      $('.external-ratings').html(compile)
    }).catch((error) => MU.error(error))
  })
})()