SteamGifts Steam Ratings

Show Steam ratings on SteamGifts.

// ==UserScript==
// @name        SteamGifts Steam Ratings
// @namespace   SG Ratings
// @include     *://www.steamgifts.com/
// @include     *://www.steamgifts.com/giveaways/search?*
// @include     *://www.steamgifts.com/giveaway/*
// @include     *://www.steamgifts.com/group/*
// @include     *://www.steamgifts.com/account/settings/giveaways/filters*
// @version     1.7.17
// @require     https://code.jquery.com/jquery-3.2.1.min.js
// @grant       GM_xmlhttpRequest
// @grant       GM.xmlHttpRequest
// @grant       GM_addStyle
// @grant       GM.addStyle
// @grant       GM_registerMenuCommand
// @grant       GM.registerMenuCommand
// @grant       GM_getValue
// @grant       GM.getValue
// @grant       GM_setValue
// @grant       GM.setValue
// @grant       GM_listValues
// @grant       GM.listValues
// @grant       GM_deleteValue
// @grant       GM.deleteValue
// @connect     store.steampowered.com
// @connect     query.yahooapis.com
// @run-at      document-end
// @description:en Show Steam ratings on SteamGifts.
// @description Show Steam ratings on SteamGifts.
// ==/UserScript==

// GM4+ compatiblity code adapted from https://github.com/StigNygaard/GMCommonAPI.js STARTS HERE //
let info = (GM_info ? GM_info : GM.info);
if (info.scriptHandler === 'Greasemonkey' && parseInt(info.version,10)>=4){
    console.log("Using compatibility mode for GM4+");
    var GM4 = true;
} else {
    var GM4 = false;
}

function GM__registerMenuCommand(caption, commandFunc, options) {
    if (!GM4){
        GM_registerMenuCommand(caption, commandFunc, options);
    } else {
        if (typeof options === 'string') {
            options = {'accessKey': options};
        } else if (typeof options === 'undefined') {
            options = {};
        }
        if (!options['disabled']) {
            let prefix = '';
            if (options['type'] === 'radio') {
                prefix = options['checked'] ? '\u26AB ' : '\u26AA ';
            } else if (options['type'] === 'checkbox') {
                prefix = options['checked'] ? '\u2B1B ' : '\u2B1C ';
            }
        }
        let oMenu = document.createElement('menu');
        if (oMenu.type !== 'undefined') {
            if (!document.body) {
                return;
            }
            let topMenu = null;
            if (document.body.getAttribute('contextmenu')) {
                topMenu = document.querySelector('menu#' + document.body.getAttribute('contextmenu'));
            }
            if (!topMenu) {
                topMenu = document.createElement('menu');
                topMenu.setAttribute('type', 'context');
                topMenu.setAttribute('id', 'gm-registered-menu');
                document.body.appendChild(topMenu);
                document.body.setAttribute('contextmenu', topMenu.getAttribute('id'));
            }
            let menuItem = document.createElement('menuitem');
            menuItem.setAttribute('type', options['type'] ? options['type'] : 'command');
            menuItem.setAttribute('label', caption);
            if (options['id']) menuItem.setAttribute('id', options['id']);
            if (options['name']) menuItem.setAttribute('name', options['name']);
            if (options['checked']) menuItem.setAttribute('checked', 'checked');
            if (options['disabled']) menuItem.setAttribute('disabled', 'disabled');
            if (options['icon']) menuItem.setAttribute('icon', options['icon']);
            if (options['topLevel']) {
                topMenu.appendChild(menuItem);
            } else {
                let scriptMenu = topMenu.querySelector('menu[label="SteamGifts Steam Ratings"]');
                if (!scriptMenu) {
                    scriptMenu = document.createElement('menu');
                    scriptMenu.setAttribute('label', "SteamGifts Steam Ratings");
                    topMenu.appendChild(scriptMenu);
                }
                scriptMenu.appendChild(menuItem);
            }
            menuItem.addEventListener('click', commandFunc, false);
        }
    }
}

function GM__setValue(name, value) {
    if (!GM4){
        GM_setValue(name, value);
    } else {
        localStorage.setItem("SGSR" + '_' + name, value);
    }
}

function GM__getValue(name, defvalue) {
    if (!GM4){
        return GM_getValue(name, defvalue);
    } else {
        if (("SGSR"+'_'+name) in localStorage) {
            return localStorage.getItem("SGSR"+'_'+name);
        } else {
            return defvalue;
        }
    }
}

function GM__deleteValue(name) {
    if (!GM4){
        GM_deleteValue(name);
    } else {
        localStorage.removeItem("SGSR" + '_' + name);
    }
}

function GM__listValues() {
    if (!GM4){
        return GM_listValues();
    } else {
        let values = [];
        let prefix = "SGSR";
        let prelen = 4;
        for (let i = 0; i < localStorage.length; i++) {
            if (localStorage.key(i).substr(0, prelen) === prefix) {
                values.push(localStorage.key(i).substr(prelen+1));
            }
        }
        return values;
    }
}

function GM__addStyle(style) {
    if (!GM4){
        return GM_addStyle(style);
    } else {
        let head = document.getElementsByTagName('head')[0];
        if (head) {
            let styleElem = document.createElement('style');
            styleElem.setAttribute('type', 'text/css');
            styleElem.textContent = style;
            head.appendChild(styleElem);
            return styleElem;
        }
    }
}

function GM__xmlhttpRequest(details) {
    if (!GM4){
        return GM_xmlhttpRequest(details);
    } else {
        return GM.xmlHttpRequest(details);
    }
}
// GM4+ compatiblity code adapted from https://github.com/StigNygaard/GMCommonAPI.js ENDS HERE //

defaults = {
  expiretime:3*24*60*60,
  expiretimeenabled:true,
  GAratings:true,
  GAfeatures:2,
  GAtags:2,
  GASratings:true,
  GASfeatures:true,
  GAStags:true,
  GASmeta:false,
  timer:false,
  iconsid:2,
  cardsicon: false,
  hidereviews:0,
  hidereviewsonsg:0,
  hidemissingreviews: false,
  betaTitle:true,
  usesteamdb:false,
  forceobserver: false,
  delayscan:0.3*1000,
  dbversion:10
};

function savedata(a,b){
  GM__setValue(a, JSON.stringify(b));
}

function loaddata(a,b){
  var val = GM__getValue(a,null);
  return (val !== null ? JSON.parse(val) : (b !== undefined ? b : false));
}

function setdefault(restoredefault){
  $.each(defaults, function( name, value ) {
    if (restoredefault===true){ GM__deleteValue(name); }
    if (name!=="dbversion"){
      window[name] = loaddata(name,value);
    } else {
      window[name] = value;
    }
  });
}
setdefault();

if(timer){ console.time('SG Ratings Internal'); console.time('SG Ratings'); }
var run = 0;

function log(text) {
    if(window.console && console.log) {
        console.log("SG Ratings: " + text);
    }
}

function clearstorage(a){
  var toscan = [];
  e = GM__listValues();
  for (var i=0; i<e.length; i++){ if(!isNaN(e[i])){ GM__deleteValue(e[i]); } }
  log((typeof(a)!=='undefined'?a+' ':'') + "Cleaning database.");
}

function toggleconfig(name,e){
  e = e||!GM__getValue(name,defaults[name]);
  GM__setValue(name,e);
  alert(name + ': ' + e);
}

function betaTitleIn(el,features,tags,rdate){
  var tab = document.createElement("table");
  tab.style.position = "absolute";
  tab.style.display = "inline";
  tab.style.padding = "0 8px";
  tab.style.marginLeft = "4px"
  tab.style.border = "1px solid #d2d6e0";
  tab.style.backgroundImage = "linear-gradient(rgb(255, 255, 255) 0%, rgba(255, 255, 255, 0.98) 100%)";
  tab.style.zIndex = "1000";
  tab.style.lineHeight = "20px";
  if (el.offsetWidth < el.scrollWidth){ tab.style.left = el.getBoundingClientRect().left+105+4 + "px"; }
  if (!!features){
      var features = features.split('\n');
      var features2 = [];
      for (var bb=0; bb<features.length; bb++){
          var featuresid = features[bb].split('::')[0];
          var img = (featuresid!=='0'&& !isNaN(featuresid) && featuresid <= featuresicons.length ? '<img class="category ' + featuresicons[featuresid].replace(/_/g,'-').toLowerCase() + '"></img> ':'');
          features2.push(img + features[bb].split('::')[1]);
      }
      var features = features2.join("<br>");
  }
  tab.innerHTML = (!!rdate ? '<span>Release: ' + rdate + '</span>' : '') + (!!features ? '<td style="padding-right: 8px;">' + features + '</td><td style="padding-left: 8px;border-left: 1px solid;">' : "<td>") + tags.replace(/\b.+:/g,'').split('\n').join("<br>") + "</td>";
  el.appendChild(tab);
}

function betaTitleOut(el){
  el.removeChild(el.lastChild);
}

$.each([
        ["List current settings", function(){
            var set = [];
            $.each(defaults, function( name, value ) {
                if (name==='dbversion'){
                    set.push('\nDatabase version: ' + GM__getValue(name,value));
                } else {
                    set.push(name + ' = ' + GM__getValue(name,value) + ((GM__getValue(name,value)!=defaults[name])?" [default is " + defaults[name] + "]":""));
                }
            });
            set.push('Database entries: '+GM__listValues().length);
            alert(set.join('\n'));
        }],
        ["GA page: Show ratings",function(){ toggleconfig('GAratings'); }],
        ["GA page: Choose method to show features",function(){
            var temp = prompt("Default: " + defaults['GAfeatures'] + "\n\n0 = Disabled\n1 = Show on mouseover\n2 = Show on page", GM__getValue('GAfeatures',GAfeatures));
            if (!!temp.match(/0|1|2/)){ toggleconfig('GAfeatures',temp); } else { alert("invalid option"); }
        }],
        ["GA page: Choose method to show tags",function(){
            var temp = prompt("Default: " + defaults['GAtags'] + "\n\n0 = Disabled\n1 = Show on mouseover [default]\n2 = Show on page", GM__getValue('GAtags',GAtags));
            if (!!temp.match(/0|1|2/)){ toggleconfig('GAtags',temp); } else { alert("invalid option"); }
        }],
        ["GAS page: Show ratings",function(){ toggleconfig('GASratings'); }],
        ["GAS page: Show features",function(){ toggleconfig('GASfeatures'); }],
        ["GAS page: Show tags", function(){ toggleconfig('GAStags'); }],
        ["GAS page: Alternative mode to show features and tags [BETA]", function(){ toggleconfig('betaTitle'); }],
        ["GAS page: Show metascore", function(){ toggleconfig('GASmeta'); }],
        ["GAS page: Use SteamDB formula for ratings", function(){ toggleconfig('usesteamdb'); }],
        ["GAS page: Show Card/Achievements icons", function(){ toggleconfig('cardsicon'); }],
        ["Choose icons",function(){
            var temp = prompt("Default: " + defaults['iconsid'] + "\n\n-1 = auto\n0 = Thumbs up/down (unicode)\n1 = Alternative unicode icons\n2 = Thumbs up/down (images)", GM__getValue('iconsid',iconsid));
            if (!!temp.match(/-1|0|1|2/)){ toggleconfig('iconsid',temp); } else { alert("invalid option"); }
        }],
        ["Choose which games to hide (on browser)", function() {
    		let temp = prompt("Default: " + defaults['hidereviews'] + "\n\n0 = None\n1 = Very negative\n2 = Negative and Very negative\n3 = Mixed and below", GM__getValue('hidereviews', hidereviews));
    		if (!!temp.match(/0|1|2|3/)){ toggleconfig('hidereviews', temp); } else { alert("invalid option"); }
    	}],
        ["Choose which games to hide (on SG)", function() {
    		let temp = prompt("Unlike the previous option (\"Choose which games to hide (on browser)\") that will only hide games in the browser, this will add the game to your SG hidden games list (same as clicking on the eye icon on the giveaway).\nThe main benefit of this is that games that were hidden using this method will remain hidden if you use a browser that don't have the script installed.\nThe downside is that games that were hidden will not be unhiden automatically (unlike the previous option) if you change your mind later and reduce this option value.\nPS: This will only work if set to less than or equal to the previous option.\n\nDefault: " + defaults['hidereviewsonsg'] + "\n\n0 = None\n1 = Very negative\n2 = Negative and Very negative\n3 = Mixed and below", GM__getValue('hidereviewsonsg', hidereviewsonsg));
    		if (!!temp.match(/0|1|2|3/)) { toggleconfig('hidereviewsonsg', temp); } else { alert("invalid option"); }
    	}],
		["Hide on browser games that are missing reviews", function() {
			toggleconfig('hidemissingreviews');
		}],
        ["Set time between queries to Steam Store", function(){
            var temp = prompt("Time in milliseconds\n\nDefault: " + defaults['delayscan'] + "\nMinimum: 100 (0,1s)\nMaximum: 15000 (15s)\n\nAnything lower than 100 = off", GM__getValue('delayscan',delayscan));
            if (isNaN(temp)){ alert('Not a number'); return false; } else if (temp!==null){ GM__setValue('delayscan',Math.min(temp,15000)); }
        }],
        ["Force endless scroll observer to run", function() {
          toggleconfig('forceobserver');
        }],
        ["Clear storage", function(){ clearstorage(); alert('Storage cleaned'); }],
        ["Restore default settings", function(){ setdefault(true); alert('Settings restored to defaults'); }]
    ], function(a,b){ GM__registerMenuCommand(b[0],b[1]); }
);

if(loaddata('dbversion') !== false){ clearstorage("Old version."); restoredefault(); }

var isgaspage, isgapage, issearchpage, ishiddenpage;
if (window.location.pathname.match(/^\/giveaways/) || window.location.pathname === "/" || window.location.pathname.length === 0 || window.location.pathname.match(/^\/group\//)) {
	isgaspage = true;
} else {
	isgaspage = false;
}
if (isgaspage) {
	isgapage = false;
	issearchpage = !!document.URL.match(/type=(wishlist|recommended)|q=.+/);
} else {
	isgapage = !!window.location.pathname.match(/^\/giveaway\//);
	issearchpage = false;
}
ishiddenpage = !!document.URL.match(/\/account\/settings\/giveaways\/filters/);

var icons = [["\uD83D\uDC4D","\uD83D\uDC4D","\u25FC","\uD83D\uDC4E","\uD83D\uDC4E"],["\u25B2","\u25B2","\u25AC ","\u25BC","\u25BC"],['','','','','']];
var toscan = [];

if (iconsid !== 2 && (icons[iconsid] === undefined || icons[iconsid].length !== 5)){
  var canvas = document.createElement("canvas");
  var context = canvas.getContext("2d");
  var text = "abcdefghijklmnopqrstuvwxyz0123456789";
  context.font = "72px monospace";
  var baselineSize = context.measureText(text).width;
  context.font = "72px 'Segoe UI Symbol', monospace";
  var newSize = context.measureText(text).width;
  delete canvas;
  if (newSize == baselineSize) {
    iconsid = 2;
    log('Font "Segoe UI Symbol" not found. Using image as fallback.');
  } else {
    iconsid = 0;
  }
}

{
  var styles = [];
  if (iconsid===2){
    styles.push(".rating{background-image: url(); width:13px;height:13px;display:inline-block;border-width: 0px 4px 0px 0px;border-style:solid;border-color:transparent;vertical-align:text-bottom;background-repeat:no-repeat}\
    .rating1 { background-position: 0 0%; }\
    .rating2 { background-position: 0 25%; }\
    .rating3 { background-position: 0 50%; }\
    .rating4 { background-position: 0 75%; }\
    .rating5 { background-position: 0 100%; }");
  } else {
    styles.push('.rating{font-size:16px;}\
    .rating5::before{color:#0000ff;content:"' + icons[iconsid][0] + '"}\
    .rating4::before{color:#33b4ff;content:"' + icons[iconsid][1] + '"}\
    .rating3::before{color:#ffa300;content:"' + icons[iconsid][2] + '"}\
    .rating2::before{color:#ff6666;content:"' + icons[iconsid][3] + '"}\
    .rating1::before{color:#ff0000;content:"' + icons[iconsid][4] + '"}');
  }
  if (betaTitle || (isgapage && GAfeatures===2)){
    var featuresicons = ['', 'ico_multiPlayer', 'ico_singlePlayer', '', '', '', 'ico_mod_hl2', 'ico_mod_hl', 'ico_vac', 'ico_coop', '', '', '', 'ico_cc', 'ico_commentary', 'ico_stats', 'ico_sdk', 'ico_editor', 'ico_partial_controller', 'ico_sdk', 'ico_multiPlayer', 'ico_dlc', 'ico_achievements', 'ico_cloud', 'ico_coop', 'ico_leaderboards', '', 'ico_multiPlayer', 'ico_controller', 'ico_cards', 'ico_workshop', 'VRIcon', 'ico_turn_notifications', '', '', 'ico_cart', 'ico_multiPlayer', 'ico_multiPlayer', 'ico_coop', 'ico_coop', 'ico_collectibles', 'ico_remote_play', 'ico_remote_play', 'ico_remote_play', 'ico_remote_play'];
    styles.push(".category{background-image: url();width:21px;height:13px;display:inline-block;background-color:rgb(0,0,0);background-color:rgba(0,0,0,0.35);vertical-align:text-bottom;background-repeat:no-repeat; }\
    .vricon { background-position: 0 0%; }\
    .ico-achievements { background-position: 0 4.761905%; }\
    .ico-cards { background-position: 0 9.52381%; }\
    .ico-cart { background-position: 0 14.285714%; }\
    .ico-cc { background-position: 0 19.047619%; }\
    .ico-cloud { background-position: 0 23.809524%; }\
    .ico-commentary { background-position: 0 28.571429%; }\
    .ico-controller { background-position: 0 33.333333%; }\
    .ico-coop { background-position: 0 38.095238%; }\
    .ico-dlc { background-position: 0 42.857143%; }\
    .ico-editor { background-position: 0 47.619048%; }\
    .ico-leaderboards { background-position: 0 52.380952%; }\
    .ico-mod-hl { background-position: 0 57.142857%; }\
    .ico-mod-hl2 { background-position: 0 61.904762%; }\
    .ico-multiplayer { background-position: 0 66.666667%; }\
    .ico-partial-controller { background-position: 0 71.428571%; }\
    .ico-sdk { background-position: 0 76.190476%; }\
    .ico-singleplayer { background-position: 0 80.952381%; }\
    .ico-stats { background-position: 0 85.714286%; }\
    .ico-turn-notifications { background-position: 0 90.47619%; }\
    .ico-vac { background-position: 0 95.238095%; }\
    .ico-workshop { background-position: 0 100%; }\
    .ico-turn-notifications { background-image: url() !important; }\
    .ico-remote-play { background-image: url() !important; }\
    .ico-collectibles { background-image: url() !important; }\
    .sidebar{max-width:336px; }\
    .steamdb{width:13px;height:13px;display:inline-block;border-width: 0px 4px 0px 0px;border-style:solid;border-color:transparent;background:url('');background-size:100% 100%;background-repeat:no-repeat;vertical-align:middle;}");
  }
  if (GASmeta || isgapage){ styles.push(".meta{width:13px;height:13px;display:inline-block;border-width: 0px 4px 0px 0px;border-style:solid;border-color:transparent;background:url('');background-size:100% 100%;background-repeat:no-repeat;vertical-align:middle;}"); }
  if (isgaspage){ styles.push(".review > a{max-width:105px;text-overflow:ellipsis;overflow: hidden;display:inline-block;} .review{max-height:26px;} .review > *{vertical-align:middle;} .review > img{margin-left:4px;}"); }
  var style = document.createElement('style');
  style.type = 'text/css';
  style.innerHTML = styles.join('');
  document.getElementsByTagName('head')[0].appendChild(style);
}

function cleanrating(rating, mode = false){
  if (rating.indexOf('%') !== -1){
    rating = rating.match(/%?([0-9,]+)(\s{0,1}%)?/g);
    ratinginvord = (rating[0].indexOf('%')!==-1) ? false : true;
    if (!mode){
        rating = rating[ratinginvord?1:0] + ' (' + rating[ratinginvord?0:1] + ')';
    } else {
        rating = Array((rating[ratinginvord?1:0].replace('%','')/100), Number(rating[ratinginvord?0:1].replace(/,/g,'')));
    }
  }
  return rating;
}

function geticon(rating){
  if (rating.indexOf('%') === -1){ return ""; }
  e = rating.match(/[0-9,]+/g);
  if (e.length !== 2){ return "";
  } else {
    imagetest = e[0];
    imagetest2 = e[1].replace(/,/g,'');
    if (imagetest>84 && imagetest2>=100){ var image = 5;
    } else if (imagetest>69){ var image = 4;
    } else if (imagetest>39){ var image = 3;
    } else if (imagetest>24 || imagetest2<=100){ var image = 2;
    } else { var image = 1; }
  }
  return '<span class="rating rating' + image + '" ></span>';
}

function processpage(req, hlink, change, gaid, steamstore){
  steamstore = !!steamstore ? " [Steam Store]" : "";
  ratingtest = $('div.glance_ctn_responsive_left:first div[data-tooltip-html]', req);
  ratingtest2 = $('.game_area_comingsoon:first div:first h1:first', req);
  ratingtest3 = $('.noReviewsYetTitle:first', req);
  if (req.title.indexOf('Site Error') === -1 && (!!steamstore || (!!ratingtest.length || !!ratingtest2.length || !!ratingtest3.length))){
    code = hlink.match(/[0-9]{2,6}/)[0];
    rating2 = "";
    if (!!ratingtest.length){
      rating = ratingtest[ratingtest.length === 2 ? 1 : 0].getAttribute("data-tooltip-html");
      if (ratingtest.length === 2){ rating2 = ratingtest[0].getAttribute("data-tooltip-html"); }
    } else {
      rating = ratingtest2.text()||ratingtest3.text()||"No ratings available";
    }
    meta = $('#game_area_metascore div', req);
    meta = (meta.length === 5 ? [meta[0].textContent.replace(/\s/g,""), $('#game_area_metalink a:first', meta)[0].href.split('/pc/')[1]] : "");
    saved = loaddata(code,false);
    features = $('#category_block > .game_area_features_list_ctn > a.game_area_details_specs_ctn, #category_block > div.DRM_notice div:first, #category_block > div.DRM_notice:not(:has(*)), #category_block > div.DRM_notice:has(br)', req).map(function() { return ((this.href && this.href.indexOf("vrsupport")===-1)?this.href.match(/=([0-9]+)\&/)[1]:'0') + '::' + (this.innerHTML.indexOf("<br>") === -1 ? this.textContent : this.innerHTML.split("<br>")[0]).replace(/^\s{2,}/,'').replace(/\s{2,}/g,' '); }).get().join('\n');
    genre = $('.blockbg:first a[href*="//store.steampowered.com/genre/"]', req);
    tags = (genre.length > 0 ? genre[0].href.match(/genre\/(.+)\//)[1] + ':[' + genre[0].innerText + ']\n' : '') + $('a.app_tag', req).slice(0,5).map(function() { return (this.href!==undefined?this.href.match(/\/tags\/([a-z\-]+)/)[1]:'0') + ':' + this.textContent.replace(/\s{2,}/g,''); }).get().join('\n');
    date = $('div.release_date > div.date', req);
    date = (date.length > 0 ? date[0].innerText : '');
    savedata(code,{version: dbversion, rate: rating, rate2: rating2, meta: meta, time: (rating.indexOf("%")!==-1 ? (Number(cleanrating(rating).match(/[0-9,]+/g)[1].replace(/,/g,''))>=1000 ? parseInt(Date.now()/1000)+parseInt(expiretime*Math.min((Number(cleanrating(rating).match(/[0-9,]+/g)[1].replace(/,/g,'')/1000)-1),1)) : parseInt(Date.now()/1000)) : parseInt(Date.now()/1000)-expiretime+86400), features: features, tags: tags, rdate: date});
    if (!saved){
      dup = "";
    } else {
      dup = " (dup)";
    }
    if (!!ratingtest.length){
      log("Loaded ratings for " + hlink + dup + steamstore);
    } else if (!!ratingtest2.length){
      log("No ratings available YET for " + hlink + dup + steamstore);
    } else {
      log("No ratings available for " + hlink + steamstore);
    }
    getrating(hlink, change, gaid);
  } else if(!steamstore){
    steamstorerequest(hlink, change, gaid);
  }
}

function steamstorerequest(hlink, change, gaid){
  GM__xmlhttpRequest({
    method:  'GET',
    url:     hlink,
    headers: { 'Cookie': 'birthtime=0; mature_content=1' },
    onload:  function(req) {
      var req = new DOMParser().parseFromString(req.response, 'text/html');
      processpage(req, hlink, change, gaid, true);
    },
  });
}

function getreviewlink(id,saved){
  var dlc = saved.features.indexOf('21:');
  if(dlc !== -1){
    return 'http://store.steampowered.com/app/' + id + '/#responsive_apppage_reviewblock_ctn';
  } else {
    return 'https://steamcommunity.com/app/' + id + '/reviews/?browsefilter=toprated';
  }
}

function steamdb(rating){
    var steamdbr = cleanrating(rating, true);
    return ((steamdbr[0] - ( steamdbr[0] - 0.5 ) * Math.pow( 2, -Math.log10( steamdbr[1] + 1 ) ))*100).toFixed(2);
}

function getrating(hlink, change, gaid, toscanarray, delayed){
  var delayed = delayed||false;
  var toscanarray = toscanarray||[];
  var code = hlink.match(/[0-9]{2,6}/)[0];
  var saved = loaddata(code,false);
  if (delayscan>=100 && !delayed && toscanarray.indexOf(code)>0 && (!saved || (expiretimeenabled && parseInt(Date.now()/1000)-expiretime >= saved.time))){
      setTimeout(function(){ getrating(hlink, change, gaid, toscanarray, true); }, delayscan*toscanarray.indexOf(code));
      return false;
  }
  if(!!saved){
    if (expiretimeenabled && parseInt(Date.now()/1000)-expiretime >= saved.time){
      log("Cache expired [ " + hlink + " ]");
      GM__deleteValue(code);
      getrating(hlink, change, gaid);
    } else if (saved.version !== dbversion){
      log("Different dbversion on cache [ " + hlink + " ]");
      GM__deleteValue(code);
      getrating(hlink, change, gaid);
    } else {
      rating = saved.rate;
      rating2 = saved.rate2;
      meta = saved.meta;
      if (isgapage){
        sidebarwide = $(document.getElementsByClassName('sidebar')[0]);
        if (GAratings){
            sidebarwide.append(
                '<h3 class="sidebar__heading">Steam Reviews</h3><ul class="sidebar__navigation"><li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link" href=' + getreviewlink(code,saved) + ' rel="nofollow" target="_blank">' + geticon(cleanrating(rating)) + '<span style="text-overflow: ellipsis; overflow: hidden; white-space: nowrap;">' + rating + '</span></a></li>' +
                ( rating2 !== "" ? '<li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link" href=' + getreviewlink(code,saved) + ' rel="nofollow" target="_blank">' + geticon(cleanrating(rating2)) + '<span style="text-overflow: ellipsis; overflow: hidden; white-space: nowrap;">' + rating2 + '</span></a></li>' : '') +
                ( rating.indexOf('%') !== -1 ? '<li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link" href="https://steamdb.info/app/' + code + '/" rel="nofollow" target="_blank"><span class="steamdb"></span><div class="sidebar__navigation__item__name">SteamDB Formula</div><div class="sidebar__navigation__item__underline"></div><div class="sidebar__navigation__item__count">' + steamdb(rating) + '%</div></a></li>' : '') +
                ( meta !== "" ? '<li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link" href="http://www.metacritic.com/game/pc/' + meta[1] + '" rel="nofollow" target="_blank"><span class="meta"></span><div class="sidebar__navigation__item__name">metacritic</div><div class="sidebar__navigation__item__underline"></div><div class="sidebar__navigation__item__count">' + meta[0] + '/100</div></a></li>' : '') + '</ul>'
            );
        }
        if(!!saved.rdate){
            sidebarwide.append('<h3 class="sidebar__heading">Steam Release Date</h3><ul class="sidebar__navigation"></ul>');
            $('ul:last', sidebarwide).append('<li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link"><div class="sidebar__navigation__item__name">'+ saved.rdate + '</div></a></li>');
        }
        if (!!GAfeatures && !!saved.features){
          if(GAfeatures === 2){
            sidebarwide.append('<h3 class="sidebar__heading">Steam Features</h3><ul class="sidebar__navigation"></ul>');
            var a = saved.features.split('\n');
            for (var i=0; i<a.length; i++){
              featuresid = a[i].split('::')[0];
              var img = (featuresid!=='0'&&!isNaN(featuresid)&&featuresid<=featuresicons.length?'<img class="category ' + featuresicons[featuresid].replace(/_/g,'-').toLowerCase() + '"></img> ':'');
              var link = (featuresid!=='0'&&!isNaN(featuresid)?' href="http://store.steampowered.com/search/?category2=' + featuresid + '&category1=' + featuresid +'&category3=' +  featuresid +'"':'');
              $('ul:last', sidebarwide).append('<li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link"' + link + '><div class="sidebar__navigation__item__name">' + img + a[i].split('::')[1] + '</div></a></li>');
            }
          } else if(GAfeatures === 1){
            change.parentElement.title = saved.features.replace(/\b[0-9]+:/g,'');
          }
        }
        if(!!GAtags && !!saved.tags){
          if(GAtags === 2){
            sidebarwide.append('<h3 class="sidebar__heading">Steam Genre/Tags</h3><ul class="sidebar__navigation"></ul>');
            var a = saved.tags.split('\n');
            for (var i=0; i<a.length; i++){
              $('ul:last', sidebarwide).append('<li class="sidebar__navigation__item"><a class="sidebar__navigation__item__link" href="' + (a[i].indexOf('[')===-1?'http://store.steampowered.com/tags/' + a[i].split(':')[0] + '/' + a[i].split(':')[1]:'http://store.steampowered.com/genre/'+a[i].split(':')[0]) + '/" rel="nofollow" target="_blank"><div class="sidebar__navigation__item__name">'+ a[i].split(':')[1] + '</div></a></li>');
            }
          } else if (GAtags === 1){
            change.parentElement.title += (!!change.parentElement.title?'\n\n':'') + saved.tags.replace(/\b.+:/g,'');
          }
        }
        var title = change.parentElement.title;
        if(!!title){
          $(document.getElementsByClassName('sidebar__heading')).filter(':contains("Steam")').attr('title', title).next('ul').attr('title', title);
        }
      } else if (ishiddenpage){
		if (GASmeta && meta !== ""){
			change.getElementsByClassName('table__column--width-fill')[0].outerHTML += '<div><a href="http://www.metacritic.com/game/pc/' + meta[1] + '" rel="nofollow" target="_blank"><span class="meta"></span><span>' + meta[0] + '/100</div></a></div>';
		}
		rating = cleanrating(rating);
		if (usesteamdb && rating.indexOf('%') !== -1) {
			rating = Math.round(steamdb(rating)) + "%" + rating.split('%')[1];
		}
		const image = geticon(rating);
		if (GASratings) {
			if (rating.indexOf('%') !== -1) {
				change.getElementsByClassName('table__column--width-fill')[0].outerHTML += '<div class="review table__column--width-small text-center"><a href=' + getreviewlink(code, saved) + ' rel="nofollow" target="_blank">' + image + rating + '</a></div>';
			}
		}
		if (change.getElementsByClassName("review").length){
			if (cardsicon) {
				if (saved.features.indexOf('22::') !== -1) {
					change.getElementsByClassName("review")[0].appendChild($(document.createElement("img")).attr("src", '')[0]);
				}
				if (saved.features.indexOf('29::') !== -1) {
					change.getElementsByClassName("review")[0].appendChild($(document.createElement("img")).attr("src", '')[0]);
				}
			}
			if (!!saved.features || !!saved.tags || !!saved.rdate) {
				if (betaTitle) {
					change.getElementsByClassName("review")[0].onmouseenter = function() {
						betaTitleIn(this, saved.features, saved.tags, saved.rdate)
					};
					change.getElementsByClassName("review")[0].onmouseleave = function() {
						betaTitleOut(this)
					};
				} else if (GASfeatures) {
					change.parentElement.title = [(!!saved.rdate ? "Release: " + saved.rdate : ''), saved.features.replace(/\b[0-9]+::/g, ''), (GAStags ? saved.tags.replace(/\b.+:/g, '') : '')].filter(String).join("\n\n");
				}
			}
		}
	  } else {
        if (gaid===0 && run === 0 && !!document.getElementById('ui-id-1')){ run++; getfeaturedga(); }
        if (GASmeta && meta !== "" && (!sgplusgrid || change.parentElement.parentElement.className.indexOf('pinned-giveaways__inner-wrap') !== -1)){
          change.getElementsByClassName('giveaway__columns')[0].children[0].outerHTML += '<div><a href="http://www.metacritic.com/game/pc/' + meta[1] + '" rel="nofollow" target="_blank"><span class="meta"></span><span>' + meta[0] + '/100</div></a></div>';
        }
        rating = cleanrating(rating);
        if(usesteamdb && rating.indexOf('%') !== -1){
            rating = Math.round(steamdb(rating)) + "%" + rating.split('%')[1];
        }
        image = geticon(rating);
        if (GASratings){
          if (!sgplusgrid || change.parentElement.parentElement.className.indexOf('pinned-giveaways__inner-wrap') !== -1){
            change.getElementsByClassName('giveaway__columns')[0].children[0].outerHTML += '<div class="review"><a href=' + getreviewlink(code,saved) + ' rel="nofollow" target="_blank">' + image + rating + '</a></div>';
          } else if (rating.indexOf('%') !== -1){
            if(change.className === "giveaway__row-inner-wrap"){
              change.getElementsByClassName('fa fa-clock-o')[0].outerHTML = image + rating.split(' ')[0] + ' ' + change.getElementsByClassName('fa fa-clock-o')[0].outerHTML;
            } else {
              change.parentElement.getElementsByClassName('fa fa-clock-o')[0].outerHTML = image + rating.split(' ')[0] + ' ' + change.parentElement.getElementsByClassName('fa fa-clock-o')[0].outerHTML;
            }
          }
        }

        if (cardsicon){
            if (saved.features.indexOf('22::') !== -1){ change.getElementsByClassName("review")[0].appendChild($(document.createElement("img")).attr("src", '')[0]); }
            if (saved.features.indexOf('29::') !== -1){ change.getElementsByClassName("review")[0].appendChild($(document.createElement("img")).attr("src", '')[0]); }
        }

        if (!!saved.features || !!saved.tags || !!saved.rdate){
            if (betaTitle){
                change.getElementsByClassName("review")[0].onmouseenter = function(){ betaTitleIn(this,saved.features,saved.tags,saved.rdate) };
                change.getElementsByClassName("review")[0].onmouseleave = function(){ betaTitleOut(this) };
            } else if (GASfeatures){
                change.parentElement.title = [(!!saved.rdate ? "Release: " + saved.rdate : ''), saved.features.replace(/\b[0-9]+::/g,''), (GAStags ? saved.tags.replace(/\b.+:/g,''):'')].filter(String).join("\n\n");
            }
        }
        if((hidemissingreviews && image === "") || (hidereviews>0 && !issearchpage && change.className === "giveaway__row-inner-wrap" && image !== "" && image.match(/[0-9]/)[0] <= hidereviews)){
            if (hidereviewsonsg > 0 && image.match(/[0-9]/)[0] <= hidereviewsonsg) {
                let xsrf_el = document.querySelector("[name=xsrf_token]");
                if (xsrf_el !== null && xsrf_el.getAttribute("value") !== null){
                    let xhr = new XMLHttpRequest();
                    xhr.open("POST", '/ajax.php', true);
                    xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
                    xhr.onreadystatechange = function() {
                        if (this.readyState === XMLHttpRequest.DONE && this.status === 200) {
                            log("Hidden Game on SG [" + $('.giveaway__heading__name:first', change).text() + " ( " + $('.giveaway__heading__name:first', change)[0].href + " )]");
                        }
                    }
                    xhr.send("xsrf_token=" + xsrf_el.getAttribute("value") + "&game_id=" + change.parentElement.getAttribute("data-game-id") + "&do=hide_giveaways_by_game_id");
                }
            }
            change.parentElement.style.display = "none";
        }
      }
    }
  } else {
      steamstorerequest(hlink, change, gaid);
  }
}

function getfeaturedga(){
  if (GASratings || GASfeatures){
    var code = false;
    var test1 = document.getElementsByClassName('featured__inner-wrap')[0].getElementsByTagName('img');
    if (!!test1.length && test1[0].src.indexOf('/apps/') !== -1){
      var code = test1[0].src.match(/[0-9]{2,6}/)[0];
    } else {
      var test2 = $(document.getElementsByClassName('giveaway__heading__name')).filter('[href$="'+document.getElementsByClassName('featured__inner-wrap')[0].getElementsByTagName('a')[0].href.replace(/.*\/giveaway\/.{5}/,'')+'"]:first');
      if (!!test2.length){
        var test2b = $(test2[0].parentElement.getElementsByClassName('giveaway__icon')).filter('[href*="/app/"]:first');
        if (!!test2b.length){
            var code = test2b[0].href.match(/[0-9]{2,6}/)[0];
        }
      }
    }
    if (!!code){
      var saved = loaddata(code,false);
      if (!!saved){
        if (GASmeta){
          var meta = saved.meta;
          if(meta !== ""){
            document.getElementsByClassName('featured__column')[0].outerHTML += '<div class="featured__column text-left"><a href="http://www.metacritic.com/game/pc/' + meta[1] + '" rel="nofollow" target="_blank"><span class="meta"></span><span>' + meta[0] + '/100</div></a></div>';
          }
        }
        if (GASratings){
          var rating = cleanrating(saved.rate);
          var image = geticon(rating);
          document.getElementsByClassName('featured__column')[0].outerHTML += '<div class="featured__column text-left"><a href=' + getreviewlink(code,saved) + ' rel="nofollow" target="_blank">' + image + rating + '</a></div>';
        }
        if (GASfeatures && (saved.features || !!saved.tags)){
          document.getElementsByClassName('featured__inner-wrap')[0].title = saved.features.replace(/\b[0-9]+::/g,'') + (GAStags?(!!saved.features ? '\n\n':'')+saved.tags.replace(/\b.+:/g,''):'');
        }
      }
    }
  }
}

function scangas(element){
  a = $('a.giveaway__icon[href*="/app/"]', 'div.pinned-giveaways__inner-wrap').length;
  if (timer){ console.time('SG Ratings pageLoad'); }
  var list = GM__listValues();
  var total = [];
  if (delayscan>=100){
      for (var i=0; i<element.length; i++){
          var code = element[i].href.match(/[0-9]{2,6}/)[0];
          var scan = (list.indexOf(code) !== -1);
          var expired = (!loaddata(code,false)||(expiretimeenabled && !!loaddata(code,false) && parseInt(Date.now()/1000)-expiretime >= loaddata(code).time));
          if (expired && total.indexOf(code)===-1){ total.push(code); }
      }
  }
  for (var i=0; i<element.length; i++){
    var code = element[i].href.match(/[0-9]{2,6}/)[0];
    var scan = (list.indexOf(code) !== -1);
    var expired = (expiretimeenabled && !!loaddata(code,false) && parseInt(Date.now()/1000)-expiretime >= loaddata(code).time);
    if (scan && toscan.indexOf(code)===-1){
      if (expired){ toscan.push(code); }
      getrating(element[i].href, element[i].parentElement.parentElement.parentElement,i-a,total);
    } else if(toscan.indexOf(code)===-1 && !expired){
      toscan.push(code);
      getrating(element[i].href, element[i].parentElement.parentElement.parentElement,i-a,total);
    } else {
      (function myLoop (i,element,code) {
        setTimeout(function () {
          if (!!GM__getValue(code,false)){
            log('Avoided (Dup) for ' + element.href);
            getrating(element.href, element.parentElement.parentElement.parentElement,10,total);
            i=1;
          }
          if (--i) myLoop(i,element,code);
        }, 2000);
      })(60,element[i],code);
    }
  }
}

if (isgaspage && (GASratings || GASfeatures || GAStags)){
  setTimeout(function(){
    sgplus = !!$('a.nav__row.SGPP__settings:first').length;
    sgplusgrid = (sgplus ? !!$('div.SGPP__gridView:first').length : false);
    if (sgplusgrid){ log('SG++ using Grid view'); }
    sgplus = (sgplus ? (JSON.parse(localStorage.SGPP_Settings||false).EndlessScrollGiveaways||{enabled:false}).enabled : false);
    extendedsg = !!$(document.getElementsByClassName('page-loading')).filter('[src*="Extended_Steamgifts"]').length;
    esgst = document.querySelector('div[class*="esgst-es-page"]');
    scangas($(document.getElementsByClassName('giveaway__icon')).filter('a[href*="/app/"]'));
    if(sgplus || extendedsg){
      log((sgplus ? 'SG++' : extendedsg ? 'Extended SteamGifts' : 'ESGST') + ' using endless scroll');
      log('Starting observer (Endless scroll compatibility)');
      var observer = new MutationObserver(function(mutations) {
        mutations.forEach(function(mutation) {
          var element = ( !sgplus ? $(mutation.addedNodes).filter('div:not([class])') : $(mutation.addedNodes).filter('div.table__heading'));
          if (element.length === 1){
            var element = $('a.giveaway__icon[href*="/app/"]', ( !sgplus ? element : element.parent().filter('div:not([class])')));
            if (element.length !== 0){ scangas(element); }
          }
        });
      });
      observer.observe(document.getElementsByClassName('widget-container')[0].children[1], { childList: true, subtree: sgplus });
    } else if (!!esgst){
      log('ESGST using endless scroll');
      log('Starting observer (Endless scroll compatibility)');
      var observer = new MutationObserver(function(mutations) {
        mutations.forEach(function(mutation) {
          var elements = $(mutation.addedNodes);
          for (var i=0; i<elements.length; i++){
            var element = $('a.giveaway__icon[href*="/app/"]', elements[i]);
            if (element.length===1){
              scangas(element);
            }
          }
        });
      });
      observer.observe(esgst.parentNode, { childList: true, subtree: false });
    } else if (forceobserver){
	  log('Starting observer (Endless scroll compatibility)');
	  var observer = new MutationObserver(function(mutations) {
		mutations.forEach(function(mutation) {
		  scangas($('a.giveaway__icon[href*="/app/"]', mutation.addedNodes));
		});
	  });
	  observer.observe(document.getElementsByClassName('widget-container')[0].children[1], {
		childList: true,
		subtree: true
	  });
	}
  }, 20);
} else if (isgapage && (GAratings || (GAfeatures===1||GAfeatures===2) || (GAtags===1||GAtags===2))) {
  var link = document.getElementsByClassName('global__image-outer-wrap--game-large');
  if (link.length !== 0){
    if (link[0].href.indexOf('store.steampowered.com/app/') !== -1){
      getrating(link[0].href, link[0].parentElement);
    }
  }
} else if (ishiddenpage){
    scangas($('a.table__column__secondary-link[href*="/app/"]'));
}

setTimeout(function(){ if(timer){ console.timeEnd('SG Ratings'); } }, 0);
if (timer) console.timeEnd('SG Ratings Internal');