NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Desura Collection Key Scraper // @author wisnoskij // @version 2.8.1 // @namespace http://indie-elitist.blogspot.ca/ // @description Gets, stores, and displays the keys of your games. With the appropriate notifications for changes. // @match http://www.desura.com/collection* // @include http://www.desura.com/collection* // @match http://www.desura.com/games/* // @include http://www.desura.com/games/* // @match https://www.desura.com/collection* // @include https://www.desura.com/collection* // @match https://www.desura.com/games/* // @include https://www.desura.com/games/* // @grant GM_getValue // @grant GM_setValue // @grant GM_deleteValue // @grant GM_listValues // @grant GM_addStyle // @grant GM_setClipboard // @require https://gist.github.com/raw/2625891/waitForKeyElements.js // @require http://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js // @require http://ajax.googleapis.com/ajax/libs/jqueryui/1.11.1/jquery-ui.min.js // ==/UserScript== /** Change Log ** ** Fixed download issue by using more robouse encoding scheme (encodeURIComponent()). **/ //Make the icon list that will go below every game box var icons = $('<div name="dcks_icons"><a title="Steam Keys" class="dcks_cat dcks_keyslist dcks_steamicon dcks_hide"></a> <a title="Desura Keys" class="dcks_cat dcks_keyslist dcks_desuraicon dcks_hide"></a> <a title="GOG Keys" class="dcks_cat dcks_keyslist dcks_gogicon dcks_hide"></a> <a title="In-Game Keys" class="dcks_cat dcks_keyslist dcks_ingameicon dcks_hide"></a> <a title="Misc Keys" class="dcks_cat dcks_keyslist dcks_miscicon dcks_hide"></a> <a title="New Keys" class="dcks_keyslist dcks_newicon dcks_remove"></a><a title="Not Scanned" class="dcks_keyslist dcks_notscanedicon"></a><a title="Scrape Keys" class="dcks_keysearch"></a></div>'); //Create icons list icons.find('.dcks_keysearch').bind('click', function(ev){ ev.preventDefault(); getKeys($(this).closest('.box')); }); //Make it search that particular game's Keys page var catLabels = ['_STEAM', '_DESURA', '_GOG', '_INGAME', '_MISC']; //GM_Value category names var catNames = ['Steam', 'Desura', 'GOG', 'In-Game', 'Misc']; //Nice readable names var stats = $('<span id="dcks_stats">(<span id="dcks_gamecount"></span><span id="dcks_newexist" class="dcks_remove">:<span id="dcks_newcount"></span></span>)<span title="Scrape Keys" class="dcks_keysearch"></span><span id="dcks_proc" class="dcks_remove">(<span id="dcks_procdata"></span>)</span><span id="dcks_collmenu" style="font-size: small;cursor: pointer;">▼</span></span>'); stats.find('.dcks_keysearch').bind('click', function(ev){ ev.preventDefault(); getKeys($('.box:not(.dcks_remove)')); }); //Make it search all games var mainMenu = $('<ul id="dcksmenu" style="z-index: 55;position: relative;top: 32px;left: 9px;width: 133px;display: none;">\ <li id="dcksmenu_filter">Filter\ <ul><li id="dcksmenuclearfilter">Clear</li>\ <hr>\ <li id="dcksmenuscanstar">Unscanned & Starred</li>\ <li id="dcksmenuunclaim">Unclaimed Keys</li>\ <li id="dcksmenusteam">Steam Keys</li>\ <li id="dcksmenudesura">Desura Keys</li>\ <li id="dcksmenugog">GOG Keys</li>\ <li id="dcksmenuingame">In-Game Keys</li>\ <li id="dcksmenumisc">Misc Keys</li>\ </ul></li>\ <li id="dcksmenusearch">Search\ <ul><li id="dcksmenusearchall">All</li>\ <li id="dcksmenusearchunscanned">Unscanned</li>\ </ul></li>\ <hr>\ <li id="dcksclearallstar" class="dcks_ar">Clear All<span class="dcks_newicon dcks_keyslist"></span></li>\ <li id="dcksclearalldata" class="dcks_ar">Clear Storage<span class="dcks_notscanedicon dcks_keyslist"></span></li>\ <li id="dcksex">Import/Export\ <ul><li id="dcksexdelete">Delete All</li>\ <li id="dcksexexport">Export</li>\ <li id="dckseximport">Import</li>\ <li id="dcksexexportdl">Export Download List</li>\ </ul></li>\ <hr>\ <li id="dcksinfomenu">Info\ <ul><li><b><a target="_blank" href="https://openuserjs.org/scripts/wisnoskij/Desura_Collection_Key_Scraper">Desura Collection Key Scraper</a></b></li>\ <li>Version: ' + GM_info.script.version + '</li>\ <li><i>By: Wisnoskij</i></li>\ </ul></li>\ <li id="dcksinfomenu">Controls\ <ul><li>Your Collection(#1[:#2])<span title="Scrape Keys" class="dcks_keysearch"></span><span style="font-size: small;cursor: pointer;">▼</span>\ <ul><li><b>#1</b>: The number of games in your collection.</li>\ <li><b>#2</b>: The number of new/unscanned games in your collection.</li>\ <li><span title="Scrape Keys" class="dcks_keysearch"></span> Scan entire selection</li>\ <li><span style="font-size: small;cursor: pointer;">▼</span>: The Main Menu for this script (The one you are in right now).</li></ul></li>\ <li><div name="dcks_icons"><a title="Steam Keys" class="dcks_cat dcks_keyslist dcks_steamicon"></a>(Steam) <a title="Desura Keys" class="dcks_cat dcks_keyslist dcks_desuraicon"></a>(Desura) <a title="GOG Keys" class="dcks_cat dcks_keyslist dcks_gogicon"></a>(GOG) <a title="In-Game Keys" class="dcks_cat dcks_keyslist dcks_ingameicon"></a>(In-Game) <a title="Misc Keys" class="dcks_cat dcks_keyslist dcks_miscicon"></a>(Misc)<br><a title="New Keys" class="dcks_keyslist dcks_newicon"></a>(New) <a title="Not Scanned" class="dcks_keyslist dcks_notscanedicon"></a>(Unscanned) <span title="Scrape Keys" class="dcks_keysearch"></span>(Scan)</div>\ <ul><li>Greyed-out game icons mean that the key has been claimed to this account.<br>Non greyed-out icons means the the key still needs to be requested.</li>\ <li>A <b>Star</b> icon means that that game has new or changed keys.\ <ul><li>The <b>Star</b> sticks around until you dismiss it enmass through the script menu,<br>or individually my mouseing over a game\'s icons.</li>\ <li>A greyed-out <b>Star</b> means that all the keys for that game have been claimed.</li></ul></li>\ <li>An <b>Unscanned</b> icon means that the script has no data for this game.</li>\ <li>Click the <b>Scan</b> icon to scan just that single game.</li>\ <li>Click any of the game icons to open a menu to:\ <ul><li>Toggle the game\'s Star on/off.</li>\ <li>Clear the game\'s saved key data.</li>\ <li>Look at the game\'s saved keys.\ <ul><li>Clicking on any of these keys will transfer them to your clipboard, if supported.<br>Else a prompt will open up with the key.</li></ul></li></ul></li></ul></li>\ </ul></li>\ </ul>'); mainMenu.find('#dcksmenuclearfilter').bind('click', function(ev){ ev.preventDefault(); $('.box.dcks_remove').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenuscanstar').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('a.dcks_newicon:not(.dcks_remove),a.dcks_notscanedicon:not(.dcks_remove)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenuunclaim').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('.dcks_cat:not(.dcks_hide,.dcks_greyout)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenusteam').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('.dcks_steamicon:not(.dcks_hide)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenudesura').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('.dcks_desuraicon:not(.dcks_hide)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenugog').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('.dcks_gogicon:not(.dcks_hide)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenuingame').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('.dcks_ingameicon:not(.dcks_hide)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenumisc').bind('click', function(ev){ ev.preventDefault(); $('.box').addClass('dcks_remove'); $('.dcks_miscicon:not(.dcks_hide)').closest('.box').removeClass('dcks_remove'); setStats(); }); mainMenu.find('#dcksmenusearchunscanned').bind('click', function(ev){ ev.preventDefault(); getKeys($('a.dcks_notscanedicon:not(.dcks_remove)').closest('.box:not(.dcks_remove)')); }); mainMenu.find('#dcksmenusearchall').bind('click', function(ev){ ev.preventDefault(); getKeys($('.box:not(.dcks_remove)')); }); mainMenu.find('#dcksclearallstar').bind('click', function(ev){ ev.preventDefault(); if(confirm('Are you sure you want to Clear All Stars from every game in your current selection?')){ $('.box:not(.dcks_remove)').find('a.dcks_newicon:not(.dcks_remove)').closest('[name=dcks_icons]').trigger('mouseenter'); } }); mainMenu.find('#dcksclearalldata').bind('click', function(ev){ ev.preventDefault(); if(confirm('Are you sure you want to Clear All Key Data from every game in your current selection?')){ clearStorage($('.box:not(.dcks_remove)')); } }); mainMenu.find('#dcksexdelete').bind('click', function(ev){ ev.preventDefault(); $('.box').removeClass('dcks_remove'); if(confirm('Are you sure you want to Delete everything?')){ deleteStorage(); location.reload(); } setStats(); }); mainMenu.find('#dcksexexport').bind('click', function(ev){ ev.preventDefault(); $('.box').removeClass('dcks_remove'); setStats(); var list = GM_listValues(); var datastr = ''; var sep = askSep(); var tmpval; for(var i=0; i<list.length; i++){ tmpval = GM_getValue(list[i],''); if(typeof tmpval === 'string'){ tmpval = tmpval.replace(new RegExp('":"', 'g'), sep[1]).replace(new RegExp('","', 'g'), sep[2]); } datastr += list[i] + sep[0] + tmpval + sep[3]; } download('DCKS_data.csv', datastr); }); mainMenu.find('#dckseximport').bind('click', function(ev){ ev.preventDefault(); $('.box').removeClass('dcks_remove'); if(confirm('Are you sure you want to Import some data? This will DELETE all of your current data.')){ deleteStorage(); var input = prompt('Enter Data'); var sep = askSep(); var nvPair = input.split(sep[3]); if(nvPair[nvPair.length-1] === ''){ nvPair.splice(nvPair.length-1, 1); } var tmpval; for(var i=0; i<nvPair.length; i++){ tmpval = nvPair[i].split(sep[0]); GM_setValue(tmpval[0], tmpval[1].replace(new RegExp(sep[1], 'g'), '":"').replace(new RegExp(sep[2], 'g'), '","')); } } location.reload(); }); mainMenu.find('#dcksexexportdl').bind('click', function(ev){ ev.preventDefault(); $('.box').removeClass('dcks_remove'); setStats(); var list = $('a.more'); var datastr = ''; var tmpval; var requests = []; proc = 0; for(var i=0; i<list.length; i++){ proc++; setProc(stats); $.get(list.eq(i).prop('href'), function(data){ var title = $(data).find('.title>a').prop('href').split('/').pop(); var downloads = $(data).find('a.clear[href*="gamedownloads"]'); for(var i=0; i<downloads.length; i++){ datastr += title + '\t' + downloads.eq(i).find('.icons').justtext() + '\t' + downloads.eq(i).find('.desc').justtext() + '\t' + downloads.eq(i).find('.buy').justtext() + '\t' + downloads.eq(i).prop('href') + '\r\n'; } }, 'html') .always(function(){ proc--; setProc(stats); if(proc === 0){ download('DCKS_downlod_links.csv', datastr); } }).fail(function(data, textStatus, jqXHR){ alert(textStatus); }); } }); mainMenu.menu(); jQuery.fn.justtext = function(){ return $(this).clone().children().remove().end().text(); }; //Run for every BOX element that gets loaded by ajax waitForKeyElements(".box", main); waitForKeyElements(".boxrowheading", collection); function download(name, data){ data = data.replace(/#/g, ""); var a = document.createElement('a'); a.setAttribute('download', name); a.setAttribute('title', name); a.setAttribute('name', name); a.href = 'data:application/csv;charset=utf-8;filename='+name+','+encodeURI(data); a.innerHTML = name; a.style.display = 'none'; document.body.appendChild(a); a.click(); } function deleteStorage(){ var list = GM_listValues(); for(var i=0; i<list.length; i++){ GM_deleteValue(list[i]); } } function askSep(){ var sep = ['"="', '":"', '","', '\\r\\n']; var ask = ['Enter Name/Value Pair Separator:', 'Enter Title:Key Separator:', 'Enter Keys Separtor:', 'Enter Name/Value Pair Ending:']; sep[0] = prompt(ask[0], sep[0]); sep[3] = prompt(ask[3], sep[3]).replace(new RegExp('\\\\n', 'g'), '\n').replace(new RegExp('\\\\r', 'g'), '\r') sep[1] = prompt(ask[1], sep[1]); sep[2] = prompt(ask[2], sep[2]); return(sep); } $("head").append('<link href="http://ajax.googleapis.com/ajax/libs/jqueryui/1.11.1/themes/ui-lightness/jquery-ui.min.css" rel="stylesheet" type="text/css">'); //JQuery UI Theme $('html').bind('click', function(ev){ var clickNode = $(ev.target); if(clickNode.closest('.ui-menu').length === 0){ //Do not do anything, if you clicked a menu $('.ui-menu').hide(); if(clickNode.closest('.dcks_keyslist').length){ clickNode.closest('.box').find('>.ui-menu').show(); //Show the menu if you clicked its icon list }else if(clickNode.closest('#dcks_collmenu').length){ $('#dcksmenu').show(); //Show the menu if you clicked its icon list } } }); //Close all menus on click, open one if its icons were clicked. If a menu was clicked, do not close it. window.onhashchange = function(){ setStats(); proc = 0; }; //The main function that gets called one for every loaded game box function main(box){ var gameId = getGameId(box); box.attr('id', gameId); box.append(setIcons(icons.clone(true), gameId)); //Clone generic icons, set them for this specific game, then append to game box box.append(setMenu(gameId)); box.find('>#' + gameId + '_dcksmenu').menu(); //Make menu setStats(); //Redundant except for Alpha/favourites } function collection(collNode){ setStats(); collNode.append(stats); collNode.after(mainMenu); } function setStats(){ stats.find('#dcks_gamecount').html($('.box:not(.dcks_remove)').length); setNew(); } function setNew(){ var count = $('.box:not(.dcks_remove)').find('a.dcks_newicon:not(.dcks_remove),a.dcks_notscanedicon:not(.dcks_remove)').length; if(count > 0){ stats.find('#dcks_newexist').removeClass('dcks_remove'); }else{ stats.find('#dcks_newexist').addClass('dcks_remove'); } stats.find('#dcks_newcount').html(count); } function setProc(statsNode){ if(proc > 0){ statsNode.find('#dcks_proc').removeClass('dcks_remove'); }else{ statsNode.find('#dcks_proc').addClass('dcks_remove'); } statsNode.find('#dcks_procdata').html(proc); return(statsNode); } //Set the icons based on stored data function setIcons(gameIcons, gameId){ var status = GM_getValue(gameId, null); if(status !== null){ //Never scanned, don't need to change anything so skip gameIcons.find('.dcks_notscanedicon').addClass('dcks_remove'); //remove the Not Scanned icon switch(status){ //Set the new star case(0): //Nothing new this scan, or nothing found. Do nothing break; case(1): //Newly scanned content(greyed out star) gameIcons.find('.dcks_newicon').removeClass('dcks_remove').addClass('dcks_greyout'); gameIcons.bind('mouseenter', function(){GM_setValue(gameId, 0); gameIcons.unbind('mouseenter').find('.dcks_newicon').addClass('dcks_remove'); setNew(); }); break; case(2): //New content(never before retrieved keys) gameIcons.find('.dcks_newicon').removeClass('dcks_remove'); gameIcons.bind('mouseenter', function(){GM_setValue(gameId, 0); gameIcons.unbind('mouseenter').find('.dcks_newicon').addClass('dcks_remove'); setNew(); }); break; } var data, node; for(var index=0; index<catLabels.length; index++){ //Set the category icons data = GM_getValue(gameId + catLabels[index], null); if(data !== null){ //if no data skip node = gameIcons.find('a').eq(index); node.removeClass('dcks_hide'); if(data.indexOf('":"Not Yet Reclaimed') === -1){ //If there exists non reclaimed keys. node.addClass('dcks_greyout'); } } } } return(gameIcons); } var clearStorageNode = $('<li class="dcks_ar">Clear<span class="dcks_notscanedicon dcks_keyslist"></li>').bind('click', function(ev){ ev.preventDefault(); clearStorage($(this).closest('.box')); }); var toggleStarNode = $('<li class="dcks_ar">Toggle<span class="dcks_newicon dcks_keyslist"></span></li>').bind('click', function(ev){ ev.preventDefault(); toggleStar($(this).closest('.box')); }); var menuTempDL = $('<ul name="dcks_menu"><hr></ul>').prepend(toggleStarNode, clearStorageNode).hide(); var menuTempDT = $('<li><ul></ul></li>\n').bind('click', function(ev){ ev.preventDefault(); var keyNode = $(this).find('li.ui-menu-item:eq(0)'); if(typeof GM_setClipboard === "function"){ GM_setClipboard(keyNode.find('.dcks_key').text(), 'text'); }else{ prompt(keyNode.find('.dcks_title').text(), keyNode.find('.dcks_key').text()); } }); var menuTempDD = $('<li></li>\n'); //Set the menu based on the stored data function setMenu(gameId){ var menu = menuTempDL.clone(true).attr('id', gameId + '_dcksmenu'); var tmpNode, data; for(var i=0; i<catNames.length; i++){ //It better be the same length data = GM_getValue(gameId + catLabels[i], null); if(data !== null){ tmpNode = menuTempDT.clone(true).prepend(catNames[i]); data = data.split('","'); for(var j=0; j<data.length; j++){ data[j] = data[j].split('":"'); tmpNode.find('ul').append(menuTempDD.clone().html('<span class="dcks_title">' + data[j][0] + '</span> - <span class="dcks_key">' + data[j][1] + '</span>')); } menu.append(tmpNode); } } return(menu); } function toggleStar(box){ var gameId, curS; for(var i=0; i<box.length; i++){ gameId = box.eq(i).attr('id'); curS = GM_getValue(gameId, null); if(curS === 0){ GM_setValue(gameId, 2); }else if(curS === null){ return(false); }else{ GM_setValue(gameId, 0); } propigate(gameId); } } function clearStorage(box){ var gameId; for(var i=0; i<box.length; i++){ gameId = box.eq(i).attr('id'); GM_deleteValue(gameId); for(var j=0; j<catLabels.length; j++){ GM_deleteValue(gameId + catLabels[j]); } propigate(gameId); } } //Get game id, what it is called in the website url. Example: "ice-cream-surfer" function getGameId(gameNode){ return(gameNode.find('.btnsfour>a:eq(1)').attr('href').split('/')[2]); //Select the keys anchor (Could use "nth-of-type" for multiple game selection). Extract the gameid. } var proc = 0; //Send a get request to the game key URL function getKeys(element){ element.addClass('dcks_processing'); var url; for(var i=0; i<element.length; i++){ processing(element.eq(i)); url = '/games/' + getGameId(element.eq(i)) + '/keys/'; $.get(url, process_html, 'html') .fail( function(){ proc--; element.removeClass('dcks_processing'); alert("Getting (" + url + ") failed."); setProc(stats); }); } } function processing(element){ proc++; setProc(stats); } //Process the key page data function process_html(data){ var content = $(data).find('.content'); var gameId = content.find('>p:first-of-type>a:first-of-type').attr('href').split('/')[2]; //Use the necessarily unique game URL as a unique identifier content = content.find('>span, >a'); var categoriedata = [[], [], [], [], []]; var newkey = false, changekey = false; content.each(function(index, element){ //List of already reclaimed keys, and the list of not yet reclaimed keys var keycontent = $(element); var type = keycontent.find('.action').text(); var title = keycontent.find('.heading>span').text(); var value = keycontent.find('input[name=key]').val(); if($.trim(keycontent.find('.price').text().toUpperCase()) == 'GET'){ //If the key is not available value = 'Not Yet Reclaimed'; newkey = true; } switch(type){ //Sort all the keys into the categories case('Steam Key'): categoriedata[0].push(title + '":"' + value); break; case('Desura Key'): categoriedata[1].push(title + '":"' + value); break; case('GOG Key'): categoriedata[2].push(title + '":"' + value); break; case('In-Game Key'): categoriedata[3].push(title + '":"' + value); break; default: categoriedata[4].push(type + ' - ' + title + '":"' + value); break; } }); categoriedata[0] = categoriedata[0].join('","'); //Convert the arrays into strings categoriedata[1] = categoriedata[1].join('","'); categoriedata[2] = categoriedata[2].join('","'); categoriedata[3] = categoriedata[3].join('","'); categoriedata[4] = categoriedata[4].join('","'); for(var index=0;index<catLabels.length;index++){ //Check current data against old stored data, replace and note if different if(GM_getValue(gameId + catLabels[index], '') !== categoriedata[index]){ GM_setValue(gameId + catLabels[index], categoriedata[index]); changekey = true; } } if(changekey){ //Calculate the new game id status changekey = 1; if(newkey){ changekey = 2; } }else{ changekey = 0; } var oldstatus = GM_getValue(gameId, null); if(oldstatus !== changekey){ //Set the game status (star) if((oldstatus === null) || (changekey > 0)){ //Don't remove the star just because scanned again. Wait for mouse-over. GM_setValue(gameId, changekey); } } propigate(gameId); proc--; setProc(stats); } function propigate(gameId){ var propig = $('#' + gameId).removeClass('dcks_processing'); propig.find('[name="dcks_icons"],[name="dcks_menu"]').remove(); main(propig); setNew(); } //Additional CSS, appended with acronym dcks if creating completely new class GM_addStyle( '#boxesholder{\ overflow: visible;\ top: -10px;\ }\ #footer{\ top: -10px;\ position: relative;\ }\ .box>ul{\ width: 76px;\ position: absolute;\ z-index: 9;\ }\ ul>li>ul.ui-menu{\ white-space: nowrap;\ z-index: 9;\ }\ .box{\ padding: 1px 10px !important;\ }\ .dcks_processing{\ opacity: 0.4;\ filter: alpha(opacity=40);\ }\ #dcks_proc{\ font-size: xx-small;\ vertical-align: super;\ }\ .ui-menu-item>.dcks_keyslist{\ float: left;\ margin-top: 2px;\ }\ .dcks_ar{\ text-align: right;\ }\ .dcks_keysearch{\ background-image: url();\ background-repeat: no-repeat;\ cursor: pointer;\ }\ a.dcks_keysearch{\ float: right;\ background-size: 12px Auto;\ width: 12px;\ height: 12px;\ }\ span.dcks_keysearch{\ width: 13px;\ height: 13px;\ background-size: 14px Auto;\ display: inline-block;\ }\ .dcks_greyout{\ opacity: 0.6;\ filter: alpha(opacity=60);\ }\ .dcks_hide{\ visibility:hidden;\ }\ .dcks_remove{\ display:none !important;\ }\ #dcks_searchall{\ width: 12px;\ height: 12px;\ background-size: 12px Auto;\ position: relative;\ top: 1px;\ }\ .dcks_keyslist{\ background-repeat: no-repeat;\ background-size: Auto 13px;\ display: inline-block;\ width: 13px;\ height: 13px;\ cursor: pointer;\ }\ .dcks_steamicon{\ background-image: url();\ width: 14px !important;\ }\ .dcks_desuraicon{\ background-image: url();\ width: 17px !important;\ }\ .dcks_gogicon{\ background-image: url();\ }\ .dcks_ingameicon{\ background-image: url();\ }\ .dcks_miscicon{\ background-image: url();\ }\ .dcks_newicon{\ background-image: url();\ width: 14px !important;\ }\ .dcks_notscanedicon{\ background-image: url();\ }\ .dcks_keydetails table{\ width: 898px;\ border: 1px dotted #ccc;\ }\ .dcks_keydetails th{\ border-left: 1px dotted #ccc;\ border-right: 1px dotted #ccc;\ border-bottom: 1px dotted #000;\ font-size: 15px;\ line-height: 20px;\ font-weight: bold;\ color: #1480cd;\ padding-left: 6px;\ width: 179.6px;\ }\ .dcks_keydetails td{\ border-left: 1px dotted #ccc;\ border-right: 1px dotted #ccc;\ text-align: center;\ width: 179.6px;\ }\ .dcks_closekeydetail{\ float: right;\ padding-right: 2px;\ font-weight: bolder;\ font-size: large;\ cursor: pointer;\ }');