alpe / SteamGifts Steam Ratings

// ==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/*
// @version     1.6.9.2
// @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
// @license MIT
// @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,
  hidereviews:0,
  betaTitle:false,
  usesteamdb:false,
  delayscan:0.3*1000,
  dbversion:8
};

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(a,b,c){
  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 (a.offsetWidth < a.scrollWidth){ tab.style.left = a.getBoundingClientRect().left+105+4 + "px"; }
  if (!!b){
      var b = b.split('\n');
      var b2 = [];
      for (var bb=0; bb<b.length; bb++){
          var featuresid = b[bb].split('::')[0];
          var img = (featuresid!=='0'&&!isNaN(featuresid)?'<img class="category ' + featuresicons[featuresid].replace('_','-').toLowerCase() + '"></img> ':'');
          b2.push(img + b[bb].split('::')[1]);
      }
      var b = b2.join("<br>");
  }
  tab.innerHTML = (!!b ? '<td style="padding-right: 8px;">' + b + '</td><td style="padding-left: 8px;border-left: 1px solid;">' : "<td>") + c.replace(/\b.+:/g,'').split('\n').join("<br>") + "</td>";
  a.appendChild(tab);
}

function betaTitleOut(a){
  a.removeChild(a.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'); }],
        ["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",function(){
            var 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"); }
        }],
        ["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)); }
        }],
        ["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(); }

if (window.location.pathname.match(/^\/giveaways/) || window.location.pathname === "/" || window.location.pathname.length === 0 || window.location.pathname.match(/^\/group\//)){ var isgaspage = true; } else { var isgaspage = false; }
if (isgaspage){ var isgapage = false; var issearchpage = !!document.URL.match(/type=(wishlist|recommended)|q=.+/); } else { var isgapage = !!window.location.pathname.match(/^\/giveaway\//); var issearchpage = false; }
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"];
    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%; }\
    .sidebar--wide{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{max-width:105px;text-overflow:ellipsis;overflow: hidden; }"); }
  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(',','')));
    }
  }
  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-text]', 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-text");
      if (ratingtest.length === 2){ rating2 = ratingtest[0].getAttribute("data-tooltip-text"); }
    } 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 = $('a.name, div.DRM_notice div:first, div.DRM_notice:not(:has(*))', req).map(function() { return ((this.href!==undefined && this.href.indexOf("vrsupport")===-1)?this.href.match(/=([0-9]+)\&/)[1]:'0') + '::' + this.textContent.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');
    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});
    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--wide')[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>' : "") +
                '<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 (!!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)?'<img class="category ' + featuresicons[featuresid].replace('_','-').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 (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"><img class="meta"></img><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 (!!saved.features || !!saved.tags){
            if (betaTitle){
                change.getElementsByClassName("review")[0].onmouseenter = function(){ betaTitleIn(this,saved.features,saved.tags) };
                change.getElementsByClassName("review")[0].onmouseleave = function(){ betaTitleOut(this) };
            } else if (GASfeatures){
                change.parentElement.title = saved.features.replace(/\b[0-9]+::/g,'') + (GAStags?(!!saved.features ? '\n\n':'')+saved.tags.replace(/\b.+:/g,''):'');
            }
        }
        if(hidereviews>0 && !issearchpage && change.className === "giveaway__row-inner-wrap" && image !== "" && image.match(/[0-9]/)[0] <= hidereviews){
          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"><img class="meta"></img><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 });
    }
  }, 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);
    }
  }
}

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