// ==UserScript== // @id History Bastard // @name History Bastard // @version 1.92 // @namespace localhost // @author https://steamcommunity.com/id/_naknak_/ // @description Bastardize steam inventory history page // @description count repeated items // @description sort by item type // @description note properties // @description replace custom item names with canonical names // @description load 2 pages if on page 1 // @description highlight and summarize on-hold items // @description use full page width and fluid layout // @description add bp.tf links // @description v1.1, v1.2: work with https // @description v1.3: mark page 2, only load pg2 if page 1 lacks 72 hours of history; accumulate metal counts; SHOW_HELD_FIRST option // @description v1.4: include levels; given items are inline-block (to allow wrapping); sort by name if quality is equal; text-only trade log popups // @description v1.5: dedup everything; fix metal rounding so .99=1.00; text log aesthetics, show only held trades, fix 2nd page url when current page ends in # // @description v1.6: filter by keyword // @description v1.7: mention hours remaining on each held trade; include outgoing items in summary; bugfix warning on <72h of history available; readability tweaks // @description v1.8: bugfix trade hold ending next year // @description v1.9: page 2 properties work // @description v1.91: bugfix filter box // @description v1.92: more readable hold-timer countdowns; continuous update of timers // @include /^https?:\/\/steamcommunity\.com\/.*\/inventoryhistory.*/ // @run-at window-load // @require https://ajax.googleapis.com/ajax/libs/jquery/2.1.3/jquery.min.js // @grant GM_addStyle // ==/UserScript== LOAD_SECOND_PAGE=true; // or false; only loads if less than ESCROW_DURATION hours of history on page 1 SHOW_HELD_FIRST=false; // if true, put held trades first, then completed trades ESCROW_DURATION=72; // hours GM_addStyle(` .daybreak { border-top: 2px solid #555 } span.nx { color: cyan; vertical-align: top; padding-right: 2px; font-weight: bold; font-size: 16px; } span.nx::after { content: "ˣ"; } span.desc { vertical-align: super; font-size: 10px; } span.desc * { display: inline; } .held .tradehistory_content, #heldrow { background-color: #191500; } #heldrow { overflow:hidden; padding-bottom:6px; } #heldbanner { font-size: 18px; color: #999; margin-left: 20px; line-height: 22px; padding-bottom: 3px; } #heldbanner .warning { color:#f44; font-size:14px; } .held .tradehistory_event_description { color: #666; } .held .tradehistory_event_description > br, .held .tradehistory_event_description > .heldnotice, .profile_small_header_bg, .header_installsteam_btn_content, .header_installsteam_btn_leftcap, #account_pulldown, .tradehistory_items_received .tradehistory_items_plusminus { display: none; } .tradehistory_date.unique:not(.showall) { color: #457; font-size: 125%; text-align: left; } .tradehistory_date.dupe:not(.showall) { display:none; } .tradehistory_date.showall { display:block; } .maincontent, #mainContents { width: 100%!important; max-width: 100%!important; margin: 0!important; } .tradehistoryrow.hidden { display:none; } .tradehistoryrow { padding: 3px; margin: 0; } .tradehistory_event_description { margin-bottom: 2px; } .tradehistory_items_received { visibility: hidden; line-height: normal; margin-bottom: 1px; } .tradehistory_items_received > * { visibility: visible; line-height: normal; } .tradehistory_items_received * { vertical-align: top; display: inline-block; } .tradehistory_items_received img { margin-left: 0; vertical-align: top; width: 40px; height: 40px; } .tradehistory_items_received .history_item { min-height: 44px; max-height: 120px; overflow: hidden; } .tradehistory_items_given { border-top:1px solid rgba(66,66,66,0.1); } .history_item { counter-increment: n; margin-left: 4px; } .history_item::after { color: #ccc; border-radius: 4px; font-size: 10px; padding: 1px 2px 1px 2px; margin-left: 2px; vertical-align: top; } .history_item_name .desc span { color: #767676; } .history_item:not(:nth-child(-n+4)):last-child::after { background: #141; content: counter(n); } .tradehistory_items_received .history_item .history_item_name { display: inline-block; /* max-width: 132px; */ max-width: 150px; word-wrap: break-word; white-space: normal; line-height: normal; } .tradehistory_items_given .history_item { margin-left: 30px; display:inline-block; } .pagebreak { background-color: black; color: #999; font-size: 14px; font-weight: bold; line-height: 32px; border-top: 2px solid white; border-bottom: 2px solid white; padding-left: 20px; } .textbut:hover, .heldbut:hover { color:#ccc } .textbut, .heldbut { float:right; color: #444; margin-left:20px; line-height:normal; padding-left:2px; padding-right:2px; font-family: monospace; text-decoration:underline; } .textbut.closed::before { font-size: 11px; content:"text"; } .textbut.open::before { font-size: 11px; content:"hide"; } .heldbut.hidden::before { font-size: 11px; content:"show all trades"; } .heldbut.shown::before { font-size: 11px; content:"show only held trades"; } #heldrow.empty { display:none } .textpop { background-color: rgba(220,220,220,0.2); float:right; width:auto; z-index:99; position:relative; margin:6px 10px 4px 10px; } .textpop_content { padding:6px 20px 8px 6px; font-size:11px; width:auto; color: #111; } .textpop.open { display: inline-block; } .textpop.closed { display: none; } .textpop_close::before { content:"x"; } .textpop_close { position:absolute; top:-5px; right:-5px; cursor:pointer; color: #fff; border: 2px solid #AEAEAE; border-radius: 30px; background: #333; font-size: 31px; font-weight: bold; display: inline-block; line-height: 0px; padding: 8px 5px 10px 5px; } .filterrow { background-color:black; display:block; } .tradehistoryrow.filtered { display:none; } #filterbox { margin:-8px 6px 8px 10px; border-radius:3px; background-color:#333; border-color:#555; color:white; padding:3px 2px 3px 8px; } .filtercount { color:gold; font-style:italic; } ::-webkit-input-placeholder { color:#666; font-style:italic; } .inventory_history_pagingrow { margin-bottom:0; border-bottom:none; } .heldendtime.changed { transition: none; background-color:gold; } .heldendtime { transition: background-color 2s ease-in; background-color:inherit; } `); PROPS = propmap( "", g_rgHistoryInventory, $("script") ); String.prototype.isMatch = function(s){ return this.match(s)!==null ;}; if( window.location.href.isMatch(/inventoryhistory\/?(\?p=1)?#*$/) && LOAD_SECOND_PAGE && ESCROW_DURATION > age( $(".tradehistory_date").last().html(), $(".tradehistory_timestamp").last().html() ) ) { console.log("loading page 2"); $("
").load(window.location.href.replace(/\?.*/,'').replace(/#.*/,'') + '?p=2', function() { var scripts=$(this).find("script"); scripts.each(function() { var json=$(this).text().match(/\svar\s+g_rgHistoryInventory\s+=\s+(\{.*\});\s*\n/); if (json!==null) { var props=JSON.parse(json[1]); // console.log(prop); $.extend( PROPS, propmap( "page2_", props, scripts ) ); } }); $(this).find(".history_item").each( function() { this.id="page2_" + this.id; }); $(".tradehistoryrow").last().after( $("
").addClass("pagebreak").html("page 2") ); $(".pagebreak").last().after( $(this).find(".tradehistoryrow") ); console.log( "loaded" ); go(); $(".inventory_history_pagingrow").contents().filter(function() { return this.nodeType == 3; }).each(function(){ this.textContent = this.textContent.replace(/(1 - )\d+( of \d+ History Items)/,'$1' + $(".tradehistoryrow").length + '$2'); }); }); } else { go(); } function go() { describe(); findheld(); dedup(); bptflinks(); textevents(); dates(); filterrow(); } function propmap( prefix, props, maps ) { var rePropMap=new RegExp(/HistoryPageCreateItemHover\( '(trade\d+_\w+item\d+)', '?(\d+)'?, '?(\d+)'?, '?(\w+)'?, '?(\d+)'? \);/g); var propmap={}; var hit; $(maps).each( function() { /* jshint -W084 */ while(hit = rePropMap.exec( $(this).text() )) { /* jshint +W084 */ // console.log([ hit[2], hit[3], hit[4] ]); propmap[ prefix + hit[1] ] = props[ hit[2] ][ hit[3] ][ hit[4] ]; } }); // console.log(propmap); return propmap; } function age(date, time) { if (!date.isMatch(/\b20\d\d\b/)) date+=" "+ new Date().getFullYear(); var hours = (Date.now() - Date.parse(date)) / 3600 / 1000; //console.log("hours=" + hours); var hmampm = time.match(/(\d+):(\d\d)\s*(p?)/i); if (hmampm === null) return hours; hours -= parseInt(hmampm[2],10)/60; var h = parseInt(hmampm[1],10); if (h==12 && !hmampm[3]) h=0; else h += (h < 12 && hmampm[3]) ? 12 : 0; hours-=h; // console.log("date=" + date + "; time=" + time + "; hours=" + hours); return hours; } function dates() { var last=""; $(".tradehistory_date").each(function() { var txt=$(this).text(); if (last===txt) $(this).addClass("dupe"); else { $(this).parent().addClass("daybreak"); $(this).addClass("unique"); } last=txt; }); } function s(n,u) { return n + " " + u + (n===1 ? "":"s") + " "; } function hours2text(hours) { var sign=(hours<0 ? " from now" : " ago"); hours=Math.abs(hours); var days=parseInt((hours+.5)/24); hours-=days*24; var minutes=0; var seconds=0; if (hours<1.6 && days===0) { minutes=hours*60; hours=0; if(minutes<1.6) { seconds=Math.round(minutes*60 /10)*10; minutes=0; if(seconds<30) seconds=""; } else minutes=Math.round(minutes); } hours=Math.round(hours); return ( (days ? s(days,"day") + "and " : "") + (hours+days? s(hours,"hour") : (minutes?s(minutes,"minute") : seconds+" seconds ") ) + sign); } function endtimes() { var n=0; var c=0; var waith=0.5; $(".heldendtime").each(function() { n++; var d=$(this).data("date"); var t=$(this).data("time"); var h=age(d,t); if (waith > Math.abs(h)/60) waith=Math.abs(h)/60; var txt="— " + (h<0 ? "on hold until ":"completed ") + d + t + " (" + hours2text(h) + ")"; if ($(this).text() !== txt) { if ($(this).text().isMatch(/^—/)) $(this).addClass("changed"); $(this).text(txt); c++; } }); if (c) setTimeout(function() { $(".heldendtime.changed").removeClass("changed") }, 100); if (waith<2.5/3600) waith=2.5/3600; // console.log("waitsec=" + waith*3600); if (n) setTimeout(endtimes, Math.round(waith * 3600 * 1000)); return n; } function findheld() { $(".tradehistoryrow > .tradehistory_content > .tradehistory_event_description > span").filter(function() { return $(this).text() === "Trade on Hold"; }).addClass("heldnotice").parent().children('span:not([class])').each(function() { var c= $(this).text().match(/.*the trade hold period is over \(around (\d+ \w+[^@]*) @( \d+:\d+[ap]m)\)/); if (c !== null) { $(this).addClass("heldendtime"); $(this).data("date", c[1]); $(this).data("time", c[2]); } }).parent().parent().parent().addClass("held"); endtimes(); if(SHOW_HELD_FIRST) { var i=0; $("#mainContents").children().each( function() { $(this).data("held", 0+$(this).hasClass("held")); $(this).data("index", i++); }); $("#mainContents").html( $("#mainContents").children().sort( function(a,b) { return ( ( $(b).data("held")-$(a).data("held") ) || ( $(a).data("index")-$(b).data("index") ) ) } ) ) } var held=document.createElement("div"); held.id="heldrow"; held.appendChild( document.createElement("div") ); held.firstChild.id="heldbanner"; held.appendChild( document.createElement("div") ); held.lastChild.className="tradehistory_items tradehistory_items_received"; held.lastChild.id="heldrec"; held.appendChild( document.createElement("div") ); held.lastChild.className="tradehistory_items tradehistory_items_given"; held.lastChild.id="heldgiv"; var nheld=$(".held").length; var nheld_s = nheld==1 ? "" : "s"; var heldrec=$(".held .tradehistory_items_received .history_item"); var nheldrec=heldrec.length; var nheldrec_s = nheldrec==1 ? "" : "s"; var heldgiv=$(".held .tradehistory_items_given .history_item"); var nheldgiv=heldgiv.length; var nheldgiv_s = nheldgiv==1 ? "" : "s"; var hours=parseInt(age( $(".tradehistory_date").last().html(), $(".tradehistory_timestamp").last().html() )); if(nheld) { held.firstChild.innerText="" + nheld + " trade" + nheld_s + (nheldrec ? " and " + nheldrec + " incoming item" + nheldrec_s : "") + " held" + (nheldgiv ? "; " + nheldgiv + " outgoing held item"+nheldgiv_s : "") if (hours < ESCROW_DURATION) { held.firstChild.appendChild(document.createElement("br")); held.firstChild.appendChild(document.createElement("span")); held.firstChild.lastChild.className="warning"; held.firstChild.lastChild.innerText="More held trades may be on the next page: only " + hours + " hours of history was found here "; } } else held.className="empty"; var heldbut=document.createElement("a"); heldbut.href="#"; heldbut.className="heldbut shown"; heldbut.onclick=function(e) { e.preventDefault; var rows=document.querySelectorAll(".tradehistoryrow:not(.held)"); for (var i=0; i " + bt); var ret=0; [ /^Mann Co. Supply Crate Key/, /^Refined Metal/, /^Reclaimed Metal/, /^Scrap Metal/, /^Unusual /, /^Strange /, /^Genuine /, /^Vintage /, /^Haunted / ].forEach(function(thing){ ret=ret||(bt.isMatch(thing) - at.isMatch(thing)); }); return (ret || (bt>at ? -1:1) ); }) ); var seen = {}; var count = {}; var metal=0; $(this).find(".history_item").each( function() { var txt=$(this).text(); if ( [ "Refined Metal", "Reclaimed Metal", "Scrap Metal" ].indexOf(txt)>-1 ) { if(metal==0) { $(this).find(".history_item_name").text( "Metal" ); seen["metal"]=nx(this); } else { $(this).text(''); $(this).css('margin-left','0'); } if (txt==="Refined Metal") metal+=9; else if (txt==="Reclaimed Metal") metal+=3; else if (txt==="Scrap Metal") metal+=1; $( seen["metal"] ).text( Math.round(metal/9*10000).toString().replace(/(\d\d)\d\d$/,'.$1') ) } else if (seen[txt]) { $(this).text(''); $(this).css('margin-left','0'); // $(this).next().remove(); count[txt]++; if(count[txt]==2) seen[txt]=nx(seen[txt]); $(seen[txt]).text( count[txt] ); } else { // only uniques, CSGO whites, steam inv whites // var color=$(this).find(".history_item_name").first().css("color"); // if ( [ "rgb(125, 109, 0)", "rgb(210, 210, 210)", "rgb(180, 180, 180)"].indexOf(color)>-1 ) seen[txt]=this; seen[txt]=this; count[txt]=1; } }); }); } function nx(e) { var span=document.createElement("span"); span.className="nx"; $(e).prepend(span); return span; } function bptflinks() { $(".tradehistory_event_description a").each( function() { var a=$(this)[0]; var match= a.href.match(/^.*steamcommunity.com(\/profiles\/\d+|\/id\/[-\w]+)$/); if (match.length) { var bp=document.createElement("a"); bp.href='https://backpack.tf' + match[1]; var img=document.createElement('img'); img.src=''; img.height=img.width=12; bp.appendChild(img); a.parentNode.insertBefore(bp, a.nextSibling.nextSibling); } }); } function describe() { String.prototype.isMatch = function(s){ return this.match(s)!==null ;}; // add desriptions (paint/sheen/ks/1st part/effect) var reWhitelist=new RegExp(/^(?:Limited )?(L)e(v)e(l) (\d+) \S|^(?:Sheen|Paint Color|★ Unusual Effect|Killstreaker|Halloween): ([^(]*[^(\s]).*|^\((.*): [,\d]+\)$|^\s*(Gift) from: .*|^(?:Exterior: |.*Grade.*\()(Battle Scarred|Factory New|Minimal Wear|Field-Tested|Well-Worn)\)?$|^(?!This is a limited)(?!StatTrak™ Confirmed Kills)(.*): \d+$/); var strWhitelist=' $1$2$3$4$5$6$7$8$9'; var reLevel=new RegExp(/^(?:Limited )?Level (\d+) (?!Tool$|Special Taunt$|Craft Item$|Ticket$)\S.*$/); var rePunc=new RegExp(/\s*\((Battle Scarred|Factory New|Minimal Wear|Field-Tested|Well-Worn)\)$/g); var items=document.getElementsByClassName("history_item"); for(var i=0; i
"; for (var i=0; i"; text+=name.firstChild.textContent; var desc=name.querySelector(":scope .desc"); if (desc && desc.textContent.length) text+=": " + desc.textContent; text+="
" } } return text; } function selectElementContents(el) { var range = document.createRange(); range.selectNodeContents(el); var sel = window.getSelection(); sel.removeAllRanges(); sel.addRange(range); } var row=this.parentNode.parentNode.parentNode; var textpop=row.querySelector(":scope .textpop"); if(row.querySelector(":scope .textpop.open")) { row.querySelector(":scope .textpop.open").className="textpop closed"; this.className="textbut closed"; } else if (row.querySelector(":scope .textpop.closed")) { row.querySelector(":scope .textpop.closed").className="textpop open"; this.className="textbut open"; } else { textpop=document.createElement("div"); textpop.className="textpop closed"; var popcontent=document.createElement("div"); popcontent.className="textpop_content"; textpop.appendChild(popcontent); var closebut=document.createElement("a"); closebut.title="close text log"; closebut.href="#"; closebut.className="textpop_close"; closebut.onclick=function(e) { e.preventDefault; this.parentNode.className="textpop closed"; return false; } textpop.appendChild(closebut); var text=""; // console.log(row); var datetime=row.querySelector(":scope .tradehistory_date").textContent + " " + row.querySelector(":scope .tradehistory_timestamp").textContent; var prof=row.querySelector(":scope .tradehistory_event_description a").href; var who ="" + row.querySelector(":scope .tradehistory_event_description a").textContent.replace(/^\s+|\s+$/g, '') + ""; var desc=row.querySelector(":scope .tradehistory_event_description").firstChild.textContent.replace(/^\s*|\s*$/g, ' '); e.preventDefault; popcontent.innerHTML=datetime + desc + who + "
" + prof + sel2txt(row, "given") + sel2txt(row, "received"); textpop.className="textpop open"; this.className="textbut open"; textpop.onfocus = function(e) { var el = this; requestAnimationFrame(function() { selectElementContents(el); }); }; this.parentNode.appendChild(textpop); } textpop.focus(); return false; } $(this).find(".tradehistory_event_description").append(textbut); }); }