// ==UserScript== // @name scrap.tf - Helper // @version 0.072 // @namespace http://nethack.no-ip.biz/dev/hsp-scraptf-helper/ // @description little helper that enhances desired hats on scrap.tf - hat bots' inventories (and probably hides ones that you don't) // @require http://www.javascripttoolbox.com/libsource.php/datadumper/source/datadumper.js // @include http://scrap.tf/hats?bot=* // @include http://scrap.tf/hats/* // @include http://scrap.tf/weapons?bot=* // @include http://scrap.tf/weapons/* // @include http://backpack.tf/profiles/76561198040005664 // @include http://nethack.no-ip.biz/stats/*tf*.html // @exclude http://www.tf2wh.com/backpack.php // @grant GM_getValue // @grant GM_setValue // @run-at document-end // ==/UserScript== var debug = (0==1); var optionsreset = (0==1); var statsreset = (0==1); var optionslenvars = 0; var statslenvars = 0; var pageneedsreloading = 1; var myprefsform = -1; var imon = ""; // TODO create a unique id and a timestamp when it could be stored globally (ie. when it was allowed to run). // also create a timeout for those timestamp (say 15 seconds) at which point the unique id saved becomes invalid (ie. reset to "") // every script (mymain or ignoreitem) is now having to wait for up to timestamp-timeout time, checking back every .5-2 seconds (random!) if it can access now. // IF it can access (ie. no unique id is set), it grabs the window of opportunity and writes its own unique id and timestamp of globalstore-entry. // // this is to circumvent problems with one session reading data from about:config, another (like ignoreitem) writing back in between and then getting its writebacks overwritten by the original data of the first session. // of course a database could circumvent this problem.. or a high numreloadafter* timeout. // // TODO if there is a new hat (ie. not-unwanted), only pick one per name, if more than one hat with the new name are available. // // TODO extend saveprefs dialog to update configdata on-the-fly from the globaldatastorage // TODO also in the same venue, save configdata on load time and compare with configdata on savetime and warn if some option (manually changed) has changed since loading the page. GM_setValue ("numconcurrentsessions", GM_getValue("numconcurrentsessions", 0) +1); var s = ''; if (optionsreset) { s = s + 'options, '; } if (statsreset) { s = s + 'stats, '; } if (s != '') { if (!debug) alert ('Doing ' +s.substring(0,s.length-2)+ ' reset..'); } var strvars = "The following variables have been loaded via GM_getValue:\n\n"; /// /// conversion tables /// var convertqualitytoname = { 1: 'Genuine ', 3: 'Vintage ', 5: 'Unusual ', 6: 'The ', 11: 'Strange ', 13: 'Haunted ' }; var converteffecttoname = { 36: 'Steaming ' }; // the following is used during bptf import to undo the name fuckingupper done by the steam api (or bptf) var convertnames = { 'Aladdin s Private Reserve': 'Aladdin\'s Private Reserve', 'Ali Baba s Wee Booties': 'Ali Baba\'s Wee Booties', 'Apparition s Aspect': 'Apparition\'s Aspect', 'Backbiter s Billycock': 'Backbiter\'s Billycock', 'Baseball Bill s Sports Shine': 'Baseball Bill\'s Sports Shine', 'Berliner s Bucket Helm': 'Berliner\'s Bucket Helm', 'Bloke s Bucket Hat': 'Bloke\'s Bucket Hat', 'Brock s Locks': 'Brock\'s Locks', 'Buccaneer s Bicorne': 'Buccaneer\'s Bicorne', 'Builder s Blueprints': 'Builder\'s Blueprints', 'Bushman s Boonie': 'Bushman\'s Boonie', 'Cadaver s Cranium': 'Cadaver\'s Cranium', 'Capo s Capper': 'Capo\'s Capper', 'Captain s Cocktails': 'Captain\'s Cocktails', 'Carouser s Capotain': 'Carouser\'s Capotain', 'Chieftain s Challenge': 'Chieftain\'s Challenge', 'Conjurer s Cowl': 'Conjurer\'s Cowl', 'Conniver s Kunai': 'Conniver\'s Kunai', 'Connoisseur s Cap': 'Connoisseur\'s Cap', 'Copper s Hard Top': 'Copper\'s Hard Top', 'Coupe D isaster': 'Coupe D\'isaster', 'Croft s Crest': 'Croft\'s Crest', 'Crone s Dome': 'Crone\'s Dome', 'Crusader s Crossbow': 'Crusader\'s Crossbow', 'Darwin s Danger Shield': 'Darwin\'s Danger Shield', 'Doc s Holiday': 'Doc\'s Holiday', 'Doctor s Sack': 'Doctor\'s Sack', 'Dual Core Devil Doll': 'Dual-Core Devil Doll', 'Dr s Dapper Topper': 'Dr\'s Dapper Topper', 'Ellis Cap': 'Ellis\' Cap', 'Engineer s Cap': 'Engineer\'s Cap', 'Fan O War': 'Fan-O-War', 'Fed Fightin Fedora': 'Fed-Fightin\' Fedora', 'Foster s Facade': 'Foster\'s Facade', 'Frenchman s Beret': 'Frenchman\'s Beret', 'FR 0': 'FR-0', 'Gentleman s Gatsby': 'Gentleman\'s Gatsby', 'Gentleman s Ushanka': 'Gentleman\'s Ushanka', 'Griffin s Gog': 'Griffin\'s Gog', 'Handyman s Handle': 'Handyman\'s Handle', 'Heavy s Hockey Hair': 'Heavy\'s Hockey Hair', 'Hetman s Headpiece': 'Hetman\'s Headpiece', 'Honcho s Headgear': 'Honcho\'s Headgear', 'Hottie s Hoodie': 'Hottie\'s Hoodie', 'Hustler s Hallmark': 'Hustler\'s Hallmark', 'Jumper s Jeepcap': 'Jumper\'s Jeepcap', 'K 9 Mane': 'K-9 Mane', 'Letch s LED': 'Letch\'s LED', 'L Inspecteur': 'L\'Inspecteur', 'Liquidator s Lid': 'Liquidator\'s Lid', 'Lord Cockswain s Novelty Mutton Chops and Pipe': 'Lord Cockswain\'s Novelty Mutton Chops and Pipe', 'Lord Cockswain s Pith Helmet': 'Lord Cockswain\'s Pith Helmet', 'Magistrate s Mullet': 'Magistrate\'s Mullet', 'Master s Yellow Belt': 'Master\'s Yellow Belt', 'Medic Mech bag': 'Medic Mech-bag', 'Medic s Mountain Cap': 'Medic\'s Mountain Cap', 'Officer s Ushanka': 'Officer\'s Ushanka', 'Ol Geezer': 'Ol\' Geezer', 'Otolaryngologist s Mirror': 'Otolaryngologist\'s Mirror', 'Plug In Prospector': 'Plug-In Prospector', 'Plumber s Pipe': 'Plumber\'s Pipe', 'Prancer s Pride': 'Prancer\'s Pride', 'Professional s Panama': 'Professional\'s Panama', 'Pugilist s Protector': 'Pugilist\'s Protector', 'Pyro s Boron Beanie': 'Pyro\'s Boron Beanie', 'Respectless Robo Glove': 'Respectless Robo-Glove', 'Ritzy Rick s Hair Fixative': 'Ritzy Rick\'s Hair Fixative', 'Rump o Lantern': 'Rump-o\'-Lantern', 'Safe n Sound': 'Safe\'n\'Sound', 'Scotsman s Stove Pipe': 'Scotsman\'s Stove Pipe', 'Sergeant s Drill Hat': 'Sergeant\'s Drill Hat', 'Shooter s Sola Topi': 'Shooter\'s Sola Topi', 'Shooter s Tin Topi': 'Shooter\'s Tin Topi', 'Sign of the Wolf s School': 'Sign of the Wolf\'s School', 'Soldier s Slope Scopers': 'Soldier\'s Slope Scopers', 'Soldier s Sparkplug': 'Soldier\'s Sparkplug', 'Sultan s Ceremonial': 'Sultan\'s Ceremonial', 'Surgeon s Side Satchel': 'Surgeon\'s Side Satchel', 'Surgeon s Stethoscope': 'Surgeon\'s Stethoscope', 'Tam O Shanter': 'Tam O\' Shanter', 'Texas Slim s Dome Shine': 'Texas Slim\'s Dome Shine', 'Texas Tin Gallon': 'Texas Tin-Gallon', 'Tin 1000': 'Tin-1000', 'Tippler s Tricorne': 'Tippler\'s Tricorne', 'Toss Proof Towel': 'Toss-Proof Towel', 'Tough Guy s Toque': 'Tough Guy\'s Toque', 'U clank a': 'U-clank-a' }; var botid2addfriend = { 8: { 'name': 'Soulbot', 'url': 'http://steamcommunity.com/actions/AddFriend/76561198075701728' }, 9: { 'name': 'Penultiscrap', 'url': 'http://steamcommunity.com/actions/AddFriend/76561198078813403' }, 11: { 'name': 'Scrapties are cool', 'url': 'http://steamcommunity.com/actions/AddFriend/76561198080375927' }, 19: { 'name': 'Scrapunny', 'url': 'http://steamcommunity.com/actions/AddFriend/76561198090766710' }, 20: { 'name': 'Scrappiversity Alum', 'url': 'http://steamcommunity.com/actions/AddFriend/76561198090904769' }, 21: { 'name': 'Scraplebeian', 'url': 'http://steamcommunity.com/actions/AddFriend/76561198090946118' } }; /// /// fixed options data /// var arrwanteditems = { // 'Amputator': 1, 'AWPer Hand': 1, 'Bat': 1, 'Bat Outta Hell': 0, 'Big Kill': 1, 'Black Rose': 1, 'Bonesaw': 1, 'Bottle': 1, 'Conscientious Objector': 1, 'Festive Ambassador': 1, 'Festive Axtinguisher': 1, 'Festive Bat': 1, 'Festive Buff Banner': 1, 'Festive Flame Thrower': 1, 'Festive Frontier Justice': 1, 'Festive Grenade Launcher': 1, 'Festive Holy Mackerel': 1, 'Festive Huntsman': 1, 'Festive Knife': 1, 'Festive Medi Gun': 1, 'Festive Minigun': 1, 'Festive Rocket Launcher': 1, 'Festive Sandvich': 1, 'Festive Scattergun': 1, 'Festive Sniper Rifle': 1, 'Festive Stickybomb Launcher': 1, 'Festive Ubersaw': 1, 'Festive Wrench': 1, 'Fire Axe': 1, 'Fishcake': 1, 'Fists': 1, 'Flame Thrower': 1, 'Freedom Staff': 1, 'Frying Pan': 1, 'Grenade Launcher': 1, 'Ham Shank': 1, 'Iron Curtain': 1, 'Knife': 1, 'Kukri': 1, 'Loose Cannon': 0, 'Lugermorph': 1, 'Maul': 1, 'Medi Gun': 1, 'Minigun': 1, 'Pistol': 1, 'Quäckenbirdt': 1, 'Rescue Ranger': 1, 'Revolver': 1, 'Rocket Jumper': 1, 'Rocket Launcher': 1, 'Scattergun': 1, 'Sharp Dresser': 1, 'Shotgun': 1, 'SMG': 1, 'Sniper Rifle': 1, 'Stickybomb Launcher': 0, 'Sticky Jumper': 0, 'Syringe Gun': 1, 'Three-Rune Blade': 1, 'Unarmed Combat': 1, 'Vaccinator': 1, 'Wanga Prick': 1, 'Wrench': 1 }; // the new options data storage that includes all of the latter // keep defaults right below, and only load GM_getValue data IF there is no optionsreset // and save optionsArray at the end of this script (TODO). var default_options = { 'arrunwantedcosts': { 11: 1, 12: 1 }, // 11 => 1.22 ref, 12 => 1.33 ref 'arrwanteditems': arrwanteditems, 'arrwantedlevels': { 100: 1 }, // 'arrwantedlevels': { 22: 1, 73: 1, 100: 1 }, 'numminpricebptf': 1.65, 'nummaxcraftno': 250, 'strmysteamid': '76561198040005664', 'numreloadafterrand': 50, // default: add 20% variance to numreloadafter(sctf|bptf) settings 'numreloadafterdecreasewithoutload': 25,// default: subtract 25% of reloadafter* from reloadafter*current if ncs == 1 'numreloadafterincreasewithload': 75, // default: add 100% of reloadafter* to reloadafter*current if ncs > 1 'numreloadaftersctf': 15, // default: every 20 seconds 'numreloadafterbptf': 3600, // default: every 60 minutes 'numreloadafterbptfbreakcount': 0, // not set by options dialog, but used by the next two '*break*' keywords 'numreloadaftersctfbreakcount': 0, // not set by options dialog, but used by the next two '*break*' keywords 'numreloadaftersctfcurrent': 20, // not set by options dialog, used by pagereloading way below 'numreloadafterbptfcurrent': 20, // not set by options dialog, used by pagereloading way below 'numreloadafterbreak': 5, // default: take a break every 5 cycles 'numreloadafterbreaktime': 5, // default: take a break that is 5 times the numreloadafter timeout 'numsimulateclickonwantedsnum': 4, // default: auto-click up to four items // 'numsimulateclickonwantedsnum': 1, // default: auto-click up to four items 'bolshowdivinfo': 1, 'bolshowtextlog': 1, 'boluseaudiocue': 1 }; var stats = { 'arrseenhats': { }, 'arrignorednames': { }, // item ids are added manually by rightclicking on a visible item in the results list 'arrignoredids': { }, // item ids are added manually by rightclicking on a visible item in the results list 'arrunwantednames' : { } }; var options = { }; /// /// persistent data handling (input) /// // do options for sctf as well as bptf related things if (optionsreset) { GM_setValue ("numconcurrentsessions", 1); GM_setValue ("optionsArray", JSON.stringify(default_options)); } if (statsreset) // save "empty" arrays during a statsreset; usually will only save those if something changes. { GM_setValue ("statsarrignorednamesArray", JSON.stringify(stats['arrignorednames'])); GM_setValue ("statsarrignoredidsArray", JSON.stringify(stats['arrignoredids'])); GM_setValue ("statsarrseenhatsArray", JSON.stringify(stats['arrseenhats'])); GM_setValue ("statsarrunwantednamesArray", JSON.stringify(stats['arrunwantednames'])); } // and load data options = load_var ("options", default_options); stats['arrignorednames'] = load_var ('statsarrignorednames', stats['arrignorednames']); stats['arrignoredids'] = load_var ('statsarrignoredids', stats['arrignoredids']); //stats['arrseenhats'] = load_var ('statsarrseenhats', stats['arrseenhats']); only loaded way below, when actually about to be filled with new values. stats['arrunwantednames'] = load_var ('statsarrunwantednames', stats['arrunwantednames']); /// functions /// /// simulator // to use it: // simulate(document.getElementById("btn"), "click"); // simulate(document.getElementById("btn"), "click", { pointerX: 123, pointerY: 321 }) // idea from: http://perfectionkills.com/cross-browser-mouse-events-simulation/ function simulate(element, eventName) { var options = extend(defaultOptions, arguments[2] || {}); var oEvent, eventType = null; for (var name in eventMatchers) { if (eventMatchers[name].test(eventName)) { eventType = name; break; } } if (!eventType) throw new SyntaxError('Only HTMLEvents and MouseEvents interfaces are supported'); if (document.createEvent) { oEvent = document.createEvent(eventType); if (eventType == 'HTMLEvents') { oEvent.initEvent(eventName, options.bubbles, options.cancelable); } else { oEvent.initMouseEvent(eventName, options.bubbles, options.cancelable, document.defaultView, options.button, options.pointerX, options.pointerY, options.pointerX, options.pointerY, options.ctrlKey, options.altKey, options.shiftKey, options.metaKey, options.button, element); } element.dispatchEvent(oEvent); } else { options.clientX = options.pointerX; options.clientY = options.pointerY; var evt = document.createEventObject(); oEvent = extend(evt, options); element.fireEvent('on' + eventName, oEvent); } return element; } function extend(destination, source) { for (var property in source) destination[property] = source[property]; return destination; } var eventMatchers = { 'HTMLEvents': /^(?:load|unload|abort|error|select|change|submit|reset|focus|blur|resize|scroll)$/, 'MouseEvents': /^(?:click|dblclick|mouse(?:down|up|over|move|out))$/ } var defaultOptions = { pointerX: 0, pointerY: 0, button: 0, ctrlKey: false, altKey: false, shiftKey: false, metaKey: false, bubbles: true, cancelable: true } //// end simulator function load_var(name, dfltarr) { var outarr; if ((name) && (name != "") && (dfltarr)) { var tmpobj = GM_getValue (name +"Array", JSON.stringify(dfltarr)); if ((tmpobj) && (tmpobj != "")) { outarr = JSON.parse (tmpobj); } } return (outarr); } // end func load_var function save_var(name, arr) { if ((name) && (name != "") && (arr)) { if (debug) { alert ("saving "+name+" with " +JSON.stringify(arr).length+ " length."); } GM_setValue (name+"Array", JSON.stringify(arr)); } } // end func save_sar function loadScript(url, callback) { // adding the script tag to the head as suggested before var head = document.getElementsByTagName('head')[0]; var script = document.createElement('script'); script.type = 'text/javascript'; script.src = url; // then bind the event to the callback function // there are several events for cross browser compatibility script.onreadystatechange = callback; script.onload = callback; // fire the loading head.appendChild(script); } Number.prototype.padZero= function(len){ var s= String(this), c= '0'; len= len || 2; while(s.length < len) s= c + s; return s; } String.prototype.trim= function(){ var s= String(this); while (s.substring(s.length-1,s.length) == "\n") { s = s.substring(0,s.length-1); } return s; } function gettype(myvar) { var tmp = myvar.constructor.toString(); tmp = tmp.substring(0,tmp.indexOf("()")); tmp = tmp.substring(tmp.indexOf(" ")+1, tmp.length); return (tmp); } function genuid ( s = '' ) { var i=16; while (i--) { s = s + new String(Math.floor(Math.random()*10)) + ((i%4==0) ? "-":""); } return (s.substring(0,s.length-1)); } function addarrunwantednames (arrun, newentry, pricebptf) { if (arrun) { if ( (newentry) && (newentry != "") ) { // var debugsearchie = "Whoopee"; // if (newentry.indexOf(debugsearchie) >= 0) alert ("newentry: "+newentry+" (pricebptf: " +pricebptf+")\ninarrun: price "+arrun[newentry]+".\n\nisinarr? " +((newentry in arrun))+ "\n arrun > newpricebptf? " +(arrun[newentry] > pricebptf)+ "\n."); if (!(newentry in arrun)) { // if (newentry.indexOf(debugsearchie) >= 0) alert ("++ found new unwanted name '" +newentry+ "'"); arrun[newentry] = pricebptf; } else if (arrun[newentry] > pricebptf) { // if (newentry.indexOf(debugsearchie) >= 0) alert ("-- found cheaper price for unwanted name '" +newentry+ "'"); arrun[newentry] = pricebptf; } else { // if (newentry.indexOf(debugsearchie) >= 0) alert ("-- already have unwanted name '" +newentry+ "'"); } } } return (arrun); } // end function addarrunwantednames function iswantedhat (hatdata) { // var s = ""; for (var i in hatdata) { if (i && hatdata[i]) { s = s + i +": "+hatdata[i]+ "\n"; } } alert (s); // var s = ""; for (var i in options['arrwantedlevels']) { if (i && options['arrwantedlevels'][i]) { s = s + i +": "+options['arrwantedlevels'][i]+ "\n"; } } alert (s); var reason = ""; var rc = (0==1); if ((hatdata) && (hatdata['el'])) { var namewithoutthe = hatdata['name']; if (namewithoutthe.indexOf("The ") >= 0) namewithoutthe = hatdata['name'].substring(4); // alert ( ('name' in hatdata) +" => "+ (hatdata['name'] in stats['arrunwantednames']) +" ? price: " +stats['arrunwantednames'][hatdata['name']]+ " > options['numminpricebptf'] " +options['numminpricebptf'] ); reason = "no-reason"; if ((hatdata['craftno'] > 0) && (parseInt(hatdata['craftno']) < parseInt(options['nummaxcraftno']))) // if craftno below ie. 500, show { reason = "craftno (= "+hatdata['craftno']+") below "+options['nummaxcraftno']+" found!"; rc = (0==0); } else if (('arrignoredids' in stats) && (hatdata['id'] in stats['arrignoredids'])) { reason = "ignored id found!"; rc = (0==1); } else if (('arrignorednames' in stats) && (namewithoutthe in stats['arrignorednames'])) { reason = "ignored name found!"; rc = (0==1); } else if (! (hatdata['cost-bptf'] in options['arrunwantedcosts'])) // if cost is not in array, show { reason = "wanted cost-bptf found!"; rc = (0==0); } else if (! (hatdata['cost-hat'] in options['arrunwantedcosts'])) // if cost is not in array, show { reason = "wanted cost-hat found!"; rc = (0==0); } else if ((hatdata['name'] in stats['arrunwantednames']) && (stats['arrunwantednames'][hatdata['name']] >= options['numminpricebptf']))// if price is above options['numminpricebptf'], show (ie. +ref per exchange) { reason = "wanted price-bptf (=" +stats['arrunwantednames'][hatdata['name']]+ " ref) found!"; rc = (0==0); } else if ((hatdata['level'] > 0) && (hatdata['level'] in options['arrwantedlevels'])) // if lvl is in array, show { reason = "wanted level found!"; rc = (0==0); } else if ('name' in hatdata) // if name is in array, dont show if it's already in inv, { // or if there is a higher quality version of it in inv, in this order: 5 unu,11 stra,1 genui,13 haun,3 vinta,6 unique if (stats['arrunwantednames'][convertqualitytoname[5]+namewithoutthe]) // Unusual { reason = "unwanted name '" +convertqualitytoname[5]+namewithoutthe+ "' found!"; rc = (0==1); } else if (stats['arrunwantednames'][convertqualitytoname[11]+namewithoutthe]) // Strange { reason = "unwanted name '" +convertqualitytoname[11]+namewithoutthe+ "' found!"; rc = (0==1); } else if (stats['arrunwantednames'][convertqualitytoname[1]+namewithoutthe]) // Genuine { reason = "unwanted name '" +convertqualitytoname[1]+namewithoutthe+ "' found!"; rc = (0==1); } else if (stats['arrunwantednames'][convertqualitytoname[13]+namewithoutthe]) // Haunted { reason = "unwanted name '" +convertqualitytoname[13]+namewithoutthe+ "' found!"; rc = (0==1); } else if (stats['arrunwantednames'][convertqualitytoname[3]+namewithoutthe]) // Vintage { reason = "unwanted name '" +convertqualitytoname[3]+namewithoutthe+ "' found!"; rc = (0==1); } else if (stats['arrunwantednames'][hatdata['name']]) // The { reason = "unwanted name '" +hatdata['name']+ "' found!"; rc = (0==1); } else { reason = "not-unwanted name '" +hatdata['name']+ "' found!"; rc = (0==0); } } else { reason = "unknown name '" +hatdata['name']+ "' found!"; rc = (0==0); } } // if nothing matched so far, show return ( [ rc, reason ] ); } // end function iswantedhat function iswantedweapon (weapdata) { // var s = ""; for (var i in weapdata) { if (i && weapdata[i]) { s = s + i +": "+weapdata[i]+ "\n"; } } alert (s); // var s = ""; for (var i in options['arrwantedlevels']) { if (i && options['arrwantedlevels'][i]) { s = s + i +": "+options['arrwantedlevels'][i]+ "\n"; } } alert (s); var reason = ""; var rc = (0==1); if ((weapdata) && (weapdata['el'])) { var namewithoutthe = weapdata['name']; if (namewithoutthe.indexOf("The ") >= 0) namewithoutthe = weapdata['name'].substring(4); // alert ( ('name' in weapdata) +" => "+ (weapdata['name'] in stats['arrunwantednames']) +" ? price: " +stats['arrunwantednames'][weapdata['name']]+ " > options['numminpricebptf'] " +options['numminpricebptf'] ); reason = "no-reason"; if ((weapdata['craftno'] > 0) && (parseInt(weapdata['craftno']) < parseInt(options['nummaxcraftno']))) // if craftno below ie. 500, show { reason = "craftno (= "+weapdata['craftno']+") below "+options['nummaxcraftno']+" found!"; rc = (0==0); } else if (('arrignoredids' in stats) && (weapdata['id'] in stats['arrignoredids'])) { reason = "ignored id found!"; rc = (0==1); } else if (('arrignorednames' in stats) && (namewithoutthe in stats['arrignorednames'])) { reason = "ignored name found!"; rc = (0==1); } else if (weapdata['name'] in arrwanteditems) // if is in arrwanteditems list, show { reason = "wanted item '" +weapdata['name']+ "' found!"; rc = (0==0); } else if (! (weapdata['cost-bptf'] in options['arrunwantedcosts'])) // if cost is not in array, show { reason = "wanted cost-bptf found!"; rc = (0==0); } else if ('name' in weapdata) // if name is in array, dont show if it's already in inv, { // or if there is a higher quality version of it in inv, in this order: 5 unu,11 stra,1 genui,13 haun,3 vinta,6 unique if (convertqualitytoname[5]+namewithoutthe in arrwanteditems) // Unusual { reason = "wanted item '" +convertqualitytoname[5]+namewithoutthe+ "' found!"; rc = (0==0); } else if (convertqualitytoname[11]+namewithoutthe in arrwanteditems) // Strange { reason = "wanted item '" +convertqualitytoname[11]+namewithoutthe+ "' found!"; rc = (0==0); } else if (convertqualitytoname[1]+namewithoutthe in arrwanteditems) // Genuine { reason = "wanted item '" +convertqualitytoname[1]+namewithoutthe+ "' found!"; rc = (0==0); } else if (convertqualitytoname[13]+namewithoutthe in arrwanteditems) // Haunted { reason = "wanted item '" +convertqualitytoname[13]+namewithoutthe+ "' found!"; rc = (0==0); } else if (convertqualitytoname[3]+namewithoutthe in arrwanteditems) // Vintage { reason = "wanted item '" +convertqualitytoname[3]+namewithoutthe+ "' found!"; rc = (0==0); } else if ((weapdata['name'] in arrwanteditems) || (convertqualitytoname[6]+namewithoutthe in arrwanteditems)) // The { reason = "wanted item '" +convertqualitytoname[6]+namewithoutthe+ "' found!"; rc = (0==0); } else { } } else { reason = "unknown name '" +weapdata['name']+ "' found!"; rc = (0==1); } } // if nothing matched so far, show return ( [ rc, reason ] ); } // end func iswantedweapon function _readattr (curel, wantedkey, oldvalue) { try { if ((curel) && (curel.attributes) && (wantedkey in curel.attributes) && (curel.attributes[wantedkey].value)) { oldvalue = ((oldvalue == undefined) || (oldvalue == "")) ? curel.attributes[wantedkey].value : oldvalue; } } catch(e) { console.log ("0 _readattr("+curel+","+wantedkey+","+oldvalue+"): exception '"+e+"' caught!"); } //alert ("- _readattr("+curel+","+wantedkey+","+oldvalue+")"); return (oldvalue); } function _walkelement (curel, alldata, tmpdata, whichtag) { var id, slot, classes, content, hatname, quality, effect, costbptf, costhat, pricebptf, costwhbp, cntmine; var lvl, craftno; /* * example of an li/a combo:
  • Strange Shovel

    Level 1 Shovel
    Kills: 0
    Origin: Found in Crate
    Suggested value: 0.11 ref
    TF2WH: 330 credits (0.11 ref)
    soldier melee
  • */ /* example of an sctf-h entry with colors (mann co orange
    */ // get hat data // try { if (imon == "whbp") { var localth = curel.getElementsByTagName("th")[0].getElementsByTagName("font")[0]; var localtds = curel.getElementsByTagName("td"); if ((localth) && (localtds) && (localtds[0]) && (gettype(localtds[0]) == "XrayWrapper function HTMLTableCellElement")) { var s = ""; hatname = localth.innerHTML.trim(); for (var i=0; i < localtds.length; i++) { s = s + "i:"+i+" '" +localtds[i].innerHTML.trim() + "' (" +gettype(localtds[i])+ ")\n"; } if (localtds[0].innerHTML.trim() != "") { costwhbp = localtds[0].innerHTML.trim().replace(/[^0-9]/g, ''); } if (localtds[1].innerHTML.trim() != "") { cntmine = localtds[1].innerHTML.trim().replace(/[^0-9]/g, ''); } if (debug) { alert ("found item:\nhatname '"+hatname+"' => costwhbp " +costwhbp+ ", cntmine "+cntmine); } id = genuid("whbp-"); while (alldata[id]) { id = genuid("whbp-"); } } } else { id = _readattr (curel, 'data-id', id); // div id = _readattr (curel, 'data-itemid', id); // li id = _readattr (curel, 'data-currentid', id); // a slot = _readattr (curel, 'data-slot', slot); // div classes = _readattr (curel, 'data-classes', classes); // div content = _readattr (curel, 'data-content', content); // div quality = _readattr (curel, 'data-quality', quality); // a (1 = genuine, 3 = vintage, 5 = unusual (data-effect for effect), 6 = unique, 11 = strange, 13 = haunted, ..) effect = _readattr (curel, 'data-effect', effect); // a (36 = steaming, ..) hatname = _readattr (curel, 'title', hatname); // div if ((hatname == undefined) || (hatname == "")) hatname = _readattr (curel, 'data-original-title', hatname); // div if ((hatname == undefined) || (hatname == "")) // a { hatname = _readattr (curel, 'data-name', hatname); if (hatname == undefined) hatname = ""; hatname = hatname.replace(/-/g, ' '); } hatname = hatname.replace(/'/ig, '\''); // sctf - replacement for ' lvl = _readattr (curel, 'data-level', lvl); // a pricebptf = _readattr (curel, 'data-price', pricebptf); // li costbptf = _readattr (curel, 'data-bptf-cost', costbptf); // div costhat = _readattr (curel, 'data-hat-cost', costhat); // div // craftno on bptf is innerHTML - text in a span class='value' ... we don't need it with the bptf data and so I won't implement it. // same for the "Crafted by username" data, which on bptf is in a span class='desc_positive'. Not interested in this, so I won't implement it. // } catch(e) { alert ("_walkelement: exception '"+e+"'"); } } // in case of a non-div, content is undefined.. to avoid a hickup below, we set it to "" if (content == undefined) content = ""; // process hat data (the default values for lvl and craftno are for easier sorting and deciding whether to highlight this one lateron) if ( ((lvl == undefined) || (lvl == "")) && (content.replace(/^Level: ([0-9]+).*$/gi, '$1') != content)) lvl = content.replace(/^.*Level: ([0-9]+).*$/gi, '$1'); if ((lvl == undefined) || (lvl == "")) lvl = -1; if ( ((craftno == undefined) || (craftno == "")) || (content.replace(/^.*Craft number: ([0-9]+).*$/gi, '$1') == content)) craftno = content.replace(/^.*Craft number: ([0-9]+).*$/gi, '$1'); if ((craftno == undefined) || (craftno == "")) craftno = 99999; // check hat data for validity if (whichtag == "div") { if ((id == undefined) || (hatname == undefined) || (lvl == undefined)) { /* no change */ } else { tmpdata = (id in alldata) ? alldata[id] : {}; if ((tmpdata['id'] == undefined) || (tmpdata['id'] == "")) tmpdata['id'] = id; if ((tmpdata['slot'] == undefined) || (tmpdata['slot'] == "")) tmpdata['slot'] = slot; if ((tmpdata['classes'] == undefined) || (tmpdata['classes'] == "")) tmpdata['classes'] = classes; if ((tmpdata['content'] == undefined) || (tmpdata['content'] == "")) tmpdata['content'] = content; if ((tmpdata['name'] == undefined) || (tmpdata['name'] == "")) tmpdata['name'] = hatname; if ((tmpdata['cost-bptf'] == undefined) || (tmpdata['cost-bptf'] == "")) tmpdata['cost-bptf'] = costbptf; if ((tmpdata['cost-hat'] == undefined) || (tmpdata['cost-hat'] == "")) tmpdata['cost-hat'] = costhat; if ((tmpdata['level'] == undefined) || (tmpdata['level'] == "")) tmpdata['level'] = lvl; if ((tmpdata['craftno'] == undefined) || (tmpdata['craftno'] == "")) tmpdata['craftno'] = craftno; if ((tmpdata['el'] == undefined) || (tmpdata['el'] == "")) tmpdata['el'] = curel; /* if (hatname.indexOf("Saint") >= 0) { alert ("hn5: "+hatname+ ", id: "+id+", slot: "+slot+", classes: "+classes+ ", costbptf: "+costbptf+", costhat: "+costhat+"\nhn5: "+tmpdata['name']+ ", id: "+tmpdata['id']+", slot: "+tmpdata['slot']+", classes: "+tmpdata['classes']+ ", costbptf: "+tmpdata['cost-bptf']+", costhat: "+tmpdata['cost-hat']+"."); } */ } } // end if tag is "div", ie. sctfh else if (whichtag == "li") // get itemid, name, price-bptf from li, and lvl from a (how to interject this, i have no idea) { if ((id == undefined) || (pricebptf == undefined)) { /* no change */ } else { tmpdata = (id in alldata) ? alldata[id] : {}; if ((tmpdata['id'] == undefined) || (tmpdata['id'] == "")) tmpdata['id'] = id; if ((tmpdata['price-bptf'] == undefined) || (tmpdata['price-bptf'] == "")) tmpdata['price-bptf'] = pricebptf; if ((tmpdata['el'] == undefined) || (tmpdata['el'] == "")) tmpdata['el'] = curel; } } // end if tag is "li", ie. bptf 1 else if (whichtag == "a") { if ((id == undefined) || (hatname == undefined) || (lvl <= 0)) { /* no change */ } else { tmpdata = (id in alldata) ? alldata[id] : {}; if ((tmpdata['name'] == undefined) || (tmpdata['name'] == "")) tmpdata['name'] = hatname; if ((tmpdata['level'] == undefined) || (tmpdata['level'] == "")) tmpdata['level'] = lvl; if ((tmpdata['quality'] == undefined) || (tmpdata['quality'] == "")) tmpdata['quality'] = quality; if ((tmpdata['effect'] == undefined) || (tmpdata['effect'] == "")) tmpdata['effect'] = effect; // alert ("_walkelem: name "+tmpdata['name']+", level "+tmpdata['level']+", quality "+tmpdata['quality']+", effect "+tmpdata['effect']+"."); } } // end if tag is "a", ie. bptf 2 else if (whichtag == "tr") { if ((hatname == undefined) || (costwhbp == undefined)) { /* no change */ } else { tmpdata = (id in alldata) ? alldata[id] : {}; if ((tmpdata['id'] == undefined) || (tmpdata['id'] == "")) tmpdata['id'] = id; if ((tmpdata['name'] == undefined) || (tmpdata['name'] == "")) tmpdata['name'] = pricebptf; if ((tmpdata['cost-whbp'] == undefined) || (tmpdata['cost-whbp'] == "")) tmpdata['cost-whbp'] = pricebptf; if ((tmpdata['el'] == undefined) || (tmpdata['el'] == "")) tmpdata['el'] = curel; } } else { if (debug) { alert ("_walkelement got unknown whichtag '" +whichtag+ "' !!"); } } return (tmpdata); } // end function _walkelement function walkelements (arr, infodata, whichtag) { var str = "walkelements(whichtag:" +whichtag+ "):\n"; if (arr) { var tmpdata = {}; for (index = 0; index < arr.length; ++index) { tmpdata = _walkelement (arr[index], infodata, tmpdata, whichtag); if ((tmpdata) && ('id' in tmpdata) && (tmpdata['id'] != undefined) && (tmpdata['id'] != "") && (tmpdata['id'] > 0)) { infodata[tmpdata['id']] = tmpdata; str = str + "-- got tmpdata with id " +tmpdata['id']+ " (lvl: " +tmpdata['level']+ ")\n"; } else { str = str + "00 got tmpdata with invalid id.. Skipping.\n"; } } } if (debug) alert (str); return (infodata); } // end function walkelements function playsound (el) { var sndwintada = "http://nethack.no-ip.biz/dev/hsp-scraptf-helper/tada.wav"; var sndoldtelebell = "http://nethack.no-ip.biz/dev/hsp-scraptf-helper/OldTeleBellLoud.mp3"; var sndgoodgrief = "http://nethack.no-ip.biz/dev/hsp-scraptf-helper/GoodGrief.wav"; var sndspherical = "http://nethack.no-ip.biz/dev/hsp-scraptf-helper/Wayfinder_-_Spherical.mp3"; if (('boluseaudiocue' in options) && (options['boluseaudiocue'] > 0)) { if (el) { var embedEl=document.createElement('audio'); embedEl.setAttribute('src', sndgoodgrief); embedEl.setAttribute('preload', 'auto'); embedEl.setAttribute('autobuffer', '1'); embedEl.setAttribute('controls', '1'); embedEl.setAttribute('volume', 100); el.appendChild(embedEl); // to get the speakers abooming embedEl.play(); // to get the screen aflashing document.title = "*** >>> " +document.title+ " <<< ***"; window.focus(); } } } // end function playsound function adddivinfo (basediv, el, cnt, s) { if (('bolshowdivinfo' in options) && (options['bolshowdivinfo'] > 0)) { if ((el) && (el['el']) && (s != "")) { // add the item name and info var headerEl=document.createElement("span"); var txtEl=document.createTextNode(s); headerEl.appendChild(txtEl); headerEl.style.textAlign = "center"; headerEl.style.fontSize = "1"; headerEl.style.lineHeight = "25%"; el['el'].appendChild(headerEl); // add an eventhandler on el-div: right-mb ignore-id, middle-mb ignore-name // el['el'].addEventListener('contextmenu', function () { ignoreitem (el, 'id') }, false); el['el'].addEventListener('mousedown', function (e) { if (e.which == 3) { ignoreitem (el, 'id'); } else if (e.which == 2) { ignoreitem (el, 'name'); } e.preventDefault(); }, false); } } } // end function adddivinfo function ignoreitem (tmpdata, ignoretype) { // if ((tmpdata) && (tmpdata['el']) && (tmpdata['name']) && (tmpdata['id']) && (ignoretype)) if ((tmpdata) && ('el' in tmpdata) && ('name' in tmpdata) && ('id' in tmpdata) && (ignoretype)) { if (ignoretype == "id") { stats['arrignoredids'] = load_var ('statsarrignoredids', stats['arrignoredids']); if ('arrignoredids' in stats) { if ( (!(tmpdata['id'] in stats['arrignoredids'])) || (stats['arrignoredids'][tmpdata['id']] < 1) ) { stats['arrignoredids'][tmpdata['id']] = 1; if (debug) { alert ("ignoreitem added to arrignoredids: id '"+tmpdata['id']+"' (name: '"+tmpdata['name']+"')"); } tmpdata['el'].style.backgroundColor = "#FF0000"; } else { stats['arrignoredids'][tmpdata['id']] = 0; if (debug) { alert ("ignoreitem removed from arrignoredids: id '"+tmpdata['id']+"' (name: '"+tmpdata['name']+"')"); } tmpdata['el'].style.backgroundColor = ""; } } // end if have arrignoredids array else { stats['arrignoredids'] = { }; // set up default } save_var ("statsarrignoredids", stats['arrignoredids']); } // end if ignoretype == id else if (ignoretype == "name") { stats['arrignorednames'] = load_var ('statsarrignorednames', stats['arrignorednames']); if ('arrignorednames' in stats) { if ( (!(tmpdata['name'] in stats['arrignorednames'])) || (stats['arrignorednames'][tmpdata['name']] < 1) ) { stats['arrignorednames'][tmpdata['name']] = 1; if (debug) { alert ("ignoreitem added to arrignorednames: id '"+tmpdata['id']+"' (name: '"+tmpdata['name']+"')"); } tmpdata['el'].style.backgroundColor = "#FF0000"; } else { stats['arrignorednames'][tmpdata['name']] = 0; if (debug) { alert ("ignoreitem removed from arrignorednames: id '"+tmpdata['id']+"' (name: '"+tmpdata['name']+"')"); } tmpdata['el'].style.backgroundColor = ""; } } // end if have arrignorednames array else { stats['arrignorednames'] = { }; // set up default } save_var ("statsarrignorednames", stats['arrignorednames']); } // end if ignoretype == name else { if (debug) { alert ("ignoreitem has gotten weird ignoretype: '" +ignoretype+ "'"); } } } else { if (debug) { try { alert ("ignoreitem got called with weird info: '"+tmpdata['name']+"', '"+tmpdata['id']+"'"); } catch(e) { alert ("ignoreitem got called with weird info: "+e); }; } } } // end function ignoreitem function addtextlog (el, s) { if (('bolshowtextlog' in options) && (options['bolshowtextlog'] > 0)) { if ((el) && (s != "")) { var preEl=document.createElement("pre"); var txtEl=document.createTextNode(s); preEl.appendChild(txtEl); preEl.style.textAlign = "left"; el.appendChild(preEl); } } } // end func addtextlog function saveprefschanges() { var optionschanged = 0; var s = ""; for (opt in options) { var inp = document.getElementById("opt"+opt); if ((inp) && (inp.name) && (inp.value)) { var name = inp.name.substring(3); var val = inp.value; if ((name) && (options[name]) && (options[name] != val)) { if (name == "num") { val = new Number(val); } else if (name == "str") { val = new String(val); } else if (name == "arr") { val = new Array(val); } else if (name == "obj") { val = new Object(val); } // if (debug) { alert ("found option with differing value with name '"+inp.name.substring(3)+"' (opt-val: " +options[opt]+ "; inp-val: "+inp.value+ ")"); } options[opt] = val; s = s + opt +", "; optionschanged++; } } } if (optionschanged > 0) { save_var ("options", options); alert (optionschanged+ " options (" +s.substring(0,s.length-2)+ ") had changed. Saved options!"); } } // end func saveprefschanges function resetprefs() { var optionschanged = 0; var s = ''; for (opt in default_options) { var inp = document.getElementById("opt"+opt); if ((inp) && (inp.name) && (inp.value)) { var ty = inp.name.substring(3); var name = inp.name.substring(3,inp.name.length); if ((name) && (default_options[name]) && (new String(default_options[name]) != new String(inp.value))) { inp.value = new String(default_options[name]); s = s + opt +', '; optionschanged = optionschanged+1; } } } if (optionschanged > 0) { alert (optionschanged+ " options (" +s.substring(0,s.length-2)+ ") have been reset. Press 'Save Prefs' to save."); } } // end func resetprefs function createinputandtext(label, idtouse, valuetouse, fieldsize = 3, createbutton = 0) { var newtrel = ""; if ((label) && (label != "")) { newtrel = document.createElement ("tr"); newtd1el = document.createElement ("td"); newtd1el.style.textAlign = "left"; newtrel.appendChild(newtd1el); newtd2el = document.createElement ("td"); newtd2el.style.textAlign = "left"; newtrel.appendChild(newtd2el); newtd3el = document.createElement ("td"); newtd3el.style.textAlign = "center"; newtrel.appendChild(newtd3el); var myhdel = ""; if ((label.indexOf("current") < 0) && (gettype(valuetouse) != "Array") && (gettype(valuetouse) != "Object")) { myhdel = document.createElement ("h4"); } else { myhdel = document.createElement ("h5"); } newtd1el.appendChild(myhdel); lblEl = document.createTextNode (label); myhdel.appendChild (lblEl); if (createbutton > 0) { var btnel = document.createElement("button"); btnel.id = "hspsaveprefsbtn" +idtouse; btnel.type = "button"; btnel.name = "hspsaveprefsbtn"; btnel.value = "Save Prefs"; btnel.style.height = "25px"; btnel.style.maxHeight = "25px"; btnel.style.width = "70pt"; btnel.style.maxWidth = "70pt"; btnel.addEventListener ("click", function() { saveprefschanges(); }, false); newtd2el.appendChild (btnel); btn2el = document.createElement("button"); btn2el.id = "hspresetprefsbtn" +idtouse; btn2el.type = "reset"; btn2el.name = "hspresetprefsbtn"; btn2el.value = "Reset!"; btn2el.style.height = "25px"; btn2el.style.maxHeight = "25px"; btn2el.style.width = "10pt"; btn2el.style.maxWidth = "10pt"; btn2el.addEventListener ("click", function() { resetprefs(); }, false); newtd2el.appendChild (btn2el); } else if ((idtouse) && (idtouse != "") && (valuetouse) && (gettype(valuetouse) != "Array") && (gettype(valuetouse) != "Object") && (fieldsize) && (fieldsize > 0)) { var inpEl = document.createElement("input"); inpEl.type = "text"; inpEl.id = idtouse; inpEl.name = idtouse; inpEl.value = valuetouse; inpEl.className = "weapon-selector span2"; inpEl.style.border = "1px solid black"; if (fieldsize > 0) { inpEl.size = fieldsize; inpEl.maxLength = fieldsize; inpEl.style.width = new String(fieldsize*5.75) +"pt"; inpEl.style.maxWidth = new String(fieldsize*5.75) +"pt"; } newtd2el.appendChild(inpEl); } } return (newtrel); } // end func createinputandtext function addprefsdialogtousername() { var hdEl = document.getElementsByClassName("dropdown"); hdEl = hdEl[hdEl.length-1]; if (hdEl) { myprefsform = document.createElement ("form"); myprefsform.id = "formhspprefs"; myprefsform.name = "formhspprefs"; baseEl = document.createElement ("table"); myprefsform.appendChild (baseEl); baseEl.className = "alert-container alert alert-success"; baseEl.style.position = "fixed"; baseEl.style.fontVariant = "normal"; baseEl.style.width = "470px"; baseEl.style.maxWidth = "470px"; hdEl.appendChild(baseEl); var newpair = createinputandtext ("Scrap.tf Helper - Prefs", "", 0, 0, 1); baseEl.appendChild(newpair); delete (newpair); for (opt in options) // type is "Array", "Object", "String", "Number", "Function" { // *count options are internal counters. if (opt.indexOf("count") >= 0) { continue; } newpair = ""; if (gettype(options[opt]) == "String") { newpair = createinputandtext (opt.substring(3) +": ", "opt"+opt, new String(options[opt]), new String(options[opt]).length+1); baseEl.appendChild(newpair); delete (newpair); } else if (gettype(options[opt]) == "Number") { newpair = createinputandtext (opt.substring(3) +": ", "opt"+opt, new Number(options[opt]), new String(options[opt]).length+1); baseEl.appendChild(newpair); delete (newpair); } else if (gettype(options[opt]) == "Array") { newpair = createinputandtext (opt.substring(3) +": ", "opt"+opt, options[opt], new Number(options[opt]).length+1); baseEl.appendChild(newpair); delete (newpair); } else if (gettype(options[opt]) == "Object") { newpair = createinputandtext (opt.substring(3) +": ", "opt"+opt, options[opt], new Number(options[opt]).length+1); baseEl.appendChild(newpair); delete (newpair); } } } } // end func addprefsdialogtousername function addtexttoheader(what = '', url = '') { if (!what) { what = ''; } var hdEl = document.getElementsByTagName("h2")[0]; if (hdEl) { if ((url) && (url != "")) { var imgel = document.getElementById ("foot_logo"); // var imgel = document.createElement("img"); if ((imgel) && (imgel.src)) { imgel.src = url; // imgel.width = 1; // imgel.height = 1; what = what + " -- imgurl added: \n"+url; } // hdEl.appendChild(imgel); } if ((what) && (what != '')) { var txtEl = document.createTextNode(what); hdEl.appendChild(txtEl); } } } // end func addsimultradetoheader function isinqueue() { // var nbqEl = document.getElementsByClassName("navbar-queue")[0]; if ((nbqEl) && (nbqEl.style) && (nbqEl.style.display) && (nbqEl.style.display == "none")) { // alert ("NOTinqueue: nbqel is "+nbqEl+" and dispstyle is: '"+nbqEl.style.display+ "'"); return (0==1); } else { // alert (">>>inqueue: nbqel is "+nbqEl+" and dispstyle is: '"+nbqEl.style.display+ "'"); return (0==0); } } function simulatefinishtradeclick(s='') { //
    // //
    if (! isinqueue()) { // the next thing is finish-hatbank for sctfh, finish-reverse for sctfw var finishtradebtnel = (imon == "sctfh") ? document.getElementById("finish-hatbank") : document.getElementById("finish-reverse"); // alert ("imon: "+imon+", findfinishtradebutton: btnel is " +finishtradebtnel); simulate (finishtradebtnel, "click"); addtexttoheader ("Simulated finishtrade click for the following items:\n" +s); // add friend automatically via steam url var botid = document.location.href.substring(document.location.href.indexOf("bot=")+4, document.location.href.length); if ((botid in botid2addfriend) && (botid2addfriend[botid]) && ('url' in botid2addfriend[botid]) && (botid2addfriend[botid]['url'] != "")) { addtexttoheader ('', botid2addfriend[botid]['url']); } } } // end function findfinishtradebutton function adjustreloadaftercurrent (what = "", ncs = 0, options = { }) { if ( (what != "") && ((what == "sctf") || (what == "bptf")) ) { var nracurbefore = options['numreloadafter'+what+'current']; if (ncs > 1) // add 75% of reloadaftersctf (only 1 out of 3 under-loads gets written back) { ncs = ncs-1; if (ncs > 0) { options['numreloadafter'+what+'current'] = new Number(nracurbefore) + (new Number(options['numreloadafterincreasewithload']) * new Number(ncs) * new Number(options['numreloadafter'+what])) / 100; } } else if (ncs > 0) // subtract 25% of reloadaftersctf (ie. no load, decrease slowly!) { options['numreloadafter'+what+'current'] = new Number(nracurbefore) - (new Number(options['numreloadafterdecreasewithoutload']) * new Number(options['numreloadafter'+what])) / 100; if (options['numreloadafter'+what+'current'] < options['numreloadafter'+what]) { options['numreloadafter'+what+'current'] = options['numreloadafter'+what]; } } if (nracurbefore != options['numreloadafter'+what+'current']) { addtexttoheader('>> adjust (ncs:' +ncs+ '): numreloadafter'+what+'current went from ' +nracurbefore+ ' to ' +options['numreloadafter'+what+'current']+ '.'); save_var ("options", options); } } return (options); } // end func adjustreloadaftercurrent // main if (debug) alert("begin"); //var afterloading = function() { try { mymain(); } catch(e) { alert ("Exception thrown in mymain: "+e); } }; loadScript('http://www.javascripttoolbox.com/libsource.php/datadumper/source/datadumper.js', afterloading); var afterloading = function() { mymain(); }; loadScript('http://www.javascripttoolbox.com/libsource.php/datadumper/source/datadumper.js', afterloading); // ################ // ################ // ################ function mymain() { var mysteam64id = options['strmysteamid']; var divs, index, div; var infodata = { }; // main infocollection imon = ((document.location.href.indexOf("backpack.tf/profiles/"+mysteam64id) >= 0) || (document.location.href.indexOf("nethack.no-ip.biz/stats/bptf.html") >= 0)) ? "bptf" : ""; if (imon == "") imon = ((document.location.href.indexOf("scrap.tf/hats") >= 0) || (document.location.href.indexOf("nethack.no-ip.biz/stats/sctfh.html") >= 0)) ? "sctfh" : ""; if (imon == "") imon = ((document.location.href.indexOf("scrap.tf/weapons") >= 0) || (document.location.href.indexOf("nethack.no-ip.biz/stats/sctfw.html") >= 0)) ? "sctfw" : ""; if (imon == "") imon = (document.location.href.indexOf("tf2wh.com/backpack.php") >= 0) ? "whbp" : ""; if (imon.indexOf("sctf") >= 0) { addprefsdialogtousername (); } var basediv = ""; basediv = (imon.indexOf("bptf") >= 0) ? document.getElementById("backpackhtml") : basediv; basediv = (imon.indexOf("sctf") >= 0) ? document.getElementsByClassName("items-wrapper")[0] : basediv; basediv = (imon.indexOf("whbp") >= 0) ? document.getElementsByClassName("data")[0] : basediv; // if website is down, skip the rest. if ((basediv != undefined) && (basediv != "")) { //alert ("imon is " +imon+", basediv is "+basediv); arrel1 = basediv.getElementsByTagName("div"); // only sctfh arrel2 = basediv.getElementsByTagName("li"); // only bptf 1 arrel3 = basediv.getElementsByTagName("a"); // only bptf 2 arrel4 = basediv.getElementsByClassName("overstocked"); // only tf2wh-bp arrel5 = basediv.getElementsByClassName("normal"); // only tf2wh-bp } // use 10M as warn limit for variable space for starters.. if ((debug) || (optionslenvars > 10247680) || (statslenvars > 10247680)) alert (strvars); /// if ((basediv != undefined) && (imon == "sctfh")) { if (debug) alert ("mid10"); infodata = walkelements (basediv.getElementsByTagName("div"), infodata, "div"); if (debug) alert ("mid11"); var str = "The following hats have been offered:\n"; if (infodata) { var cntvisible = 0; var listofids = []; var listvisibleids = []; for (var k in infodata) { listofids.push(k); } listofids.sort (function(a,b) { var name1 = infodata[a].name.toLowerCase() +parseInt(infodata[a].level).padZero(4); var name2 = infodata[b].name.toLowerCase() +parseInt(infodata[b].level).padZero(4); if (name1 < name2) return -1; if (name1 > name2) return 1; return 0; }); var simulcnt = options['numsimulateclickonwantedsnum']; var simulitems = ''; for (var idnum in listofids) { if ((idnum < 0) || (!(idnum in listofids))) { continue; } var k = listofids[idnum]; var tmpdata = infodata[k]; // now do something with the data gotten var t = iswantedhat (tmpdata); if (t) { if (t[0]) { listvisibleids[listvisibleids.length] = tmpdata; str = str +"\n++ " +idnum+ ". id '" +k+ "', name '" +tmpdata['name']+ "' (lvl " +tmpdata['level']+ ((tmpdata['craftno'] < 99999) ? ", CRAFTNO "+tmpdata['craftno'] : "") +"; reason: "+t[1]+")"; if (('el' in tmpdata) && (tmpdata['el'])) { adddivinfo (basediv, tmpdata, listvisibleids.length, tmpdata['name'].substring(0,20)+ "\nLvl=" +tmpdata['level']+ "\n" +((tmpdata['craftno'] < 99999) ? ", CRAFTNO="+tmpdata['craftno'] : "")); if (--simulcnt >= 0) { simulate (tmpdata['el'], "click"); simulitems = simulitems +"- " +k+ ": " +tmpdata['name']+ " (lvl " +tmpdata['level']+ ((tmpdata['craftno'] < 99999) ? ", CRAFTNO "+tmpdata['craftno'] : "") +"; reason: "+t[1]+")"; } } } // end if is wanted else { str = str +"\n-- " +idnum+ ". id '" +k+ "', name '" +tmpdata['name']+ "' (lvl " +tmpdata['level']+ ((tmpdata['craftno'] < 99999) ? ", CRAFTNO "+tmpdata['craftno'] : "") +"; reason: "+t[1]+")"; if (('el' in tmpdata) && (tmpdata['el']) && (tmpdata['el'].style)) { tmpdata['el'].style.display = "none"; // tmpdata['el'].style.visibility = "hidden"; } } // end if is unwanted } // end if got valid data from iswantedhat(..) // } catch (e) { alert ("sctfh-infodata-iswantedhat-check: Exception caught: "+e); } } // end for each item in sctfh-inventory var ncs = GM_getValue("numconcurrentsessions", 0); if (ncs > 2) { GM_setValue("numconcurrentsessions", 0); } // XXX in case we got a hickup, reset ncs addtexttoheader (" - ncs: " +new Number(ncs)+ "; Last load: " +new Date().toISOString()); options = adjustreloadaftercurrent ("sctf", ncs, options); if (simulcnt != options['numsimulateclickonwantedsnum']) { playsound(basediv); simulatefinishtradeclick(simulitems); pageneedsreloading = 0; } // end if we clicked something, click "finish trade" button as well addtextlog (basediv, str); // save items seen on sctfh for later usage stats['arrseenhats'] = load_var ('statsarrseenhats', stats['arrseenhats']); for (var id in infodata) { if ( (id) && (id != "") && (!(id in stats['arrseenhats'])) ) { stats['arrseenhats'][id] = infodata[id]; } } save_var ("statsarrseenhats", stats['arrseenhats']); } // end if infodata defined if (debug) alert(str); } // end if is sctfh else if ((basediv != undefined) && (imon == "sctfw")) { if (debug) alert ("mid20"); infodata = walkelements (basediv.getElementsByTagName("div"), infodata, "div"); if (debug) alert ("mid21"); var str = "The following weapons have been offered:\n"; if (infodata) { var cntvisible = 0; var listofids = []; var listvisibleids = []; for (var k in infodata) { listofids.push(k); } listofids.sort (function(a,b) { var name1 = infodata[a].name.toLowerCase() +parseInt(infodata[a].level).padZero(4); var name2 = infodata[b].name.toLowerCase() +parseInt(infodata[b].level).padZero(4); if (name1 < name2) return -1; if (name1 > name2) return 1; return 0; }); var simulcnt = options['numsimulateclickonwantedsnum']; var simulitems = ''; for (var idnum in listofids) { if ((idnum < 0) || (!(idnum in listofids))) { continue; } var k = listofids[idnum]; var tmpdata = infodata[k]; // now do something with the data gotten var t = iswantedweapon (tmpdata); if (t) { if (t[0]) { listvisibleids[listvisibleids.length] = tmpdata; str = str +"\n++ " +idnum+ ". id '" +k+ "', name '" +tmpdata['name']+ "' (lvl " +tmpdata['level']+ ((tmpdata['craftno'] < 99999) ? ", CRAFTNO "+tmpdata['craftno'] : "") +"; reason: "+t[1]+")"; if (('el' in tmpdata) && (tmpdata['el'])) { adddivinfo (basediv, tmpdata, listvisibleids.length, tmpdata['name'].substring(0,20)+ "\nLvl=" +tmpdata['level']+ "\n" +((tmpdata['craftno'] < 99999) ? ", CRAFTNO="+tmpdata['craftno'] : "")); if (--simulcnt >= 0) { simulate (tmpdata['el'], "click"); simulitems = simulitems +"- " +k+ ": " +tmpdata['name']+ " (lvl " +tmpdata['level']+ ((tmpdata['craftno'] < 99999) ? ", CRAFTNO "+tmpdata['craftno'] : "") +"; reason: "+t[1]+")"; } } } // end if is wanted else { str = str +"\n-- " +idnum+ ". id '" +k+ "', name '" +tmpdata['name']+ "' (lvl " +tmpdata['level']+ ((tmpdata['craftno'] < 99999) ? ", CRAFTNO "+tmpdata['craftno'] : "") +"; reason: "+t[1]+")"; if (('el' in tmpdata) && (tmpdata['el']) && (tmpdata['el'].style)) { tmpdata['el'].style.display = "none"; // tmpdata['el'].style.visibility = "hidden"; } } // end if is unwanted } // end if got valid data from iswantedweapon(..) // } catch (e) { alert ("sctfw-infodata-iswantedweapon-check: Exception caught: "+e); } } // end for each item in sctfw-inventory addtexttoheader (" - Last load: " +new Date().toISOString()); if (simulcnt != options['numsimulateclickonwantedsnum']) { playsound(basediv); simulatefinishtradeclick(simulitems); pageneedsreloading = 0; } // end if we clicked something, click "finish trade" button as well // in case there is no wanted weapon, provide a textlist of all the weapons on the page addtextlog (basediv, str); } // end if infodata defined if (debug) alert(str); } // end if is sctfw else if ((basediv != undefined) && (imon == "bptf")) { if (debug) alert ("mid30"); infodata = walkelements (basediv.getElementsByTagName("li"), infodata, "li"); if (debug) alert ("mid31"); infodata = walkelements (basediv.getElementsByTagName("a"), infodata, "a"); var str = "There are new items in your bptf inventory:\n\n"; var cntnewitems = 0; stats['arrunwantednames'] = load_var ('statsarrunwantednames', stats['arrunwantednames']); for (var k in infodata) { var tmpdata = infodata[k]; if ( (tmpdata) && ('name' in tmpdata) && (tmpdata['name'] != "") ) { cntnewitems++; if (tmpdata['name'].indexOf("The ") >= 0) { namewithoutthe = tmpdata['name'].substring(4); } else { namewithoutthe = tmpdata['name']; } if ((tmpdata['quality'] > 0) && (tmpdata['quality'] != 6)) { if (convertqualitytoname[tmpdata['quality']] != "") // all known qualities besides unique (ie. "The blabal") { if (debug) str = str + "orig "+tmpdata['name']+" => name+quality '" +convertqualitytoname[tmpdata['quality']] +namewithoutthe+ "' (price-bptf: " +tmpdata['price-bptf']+ ")\n"; stats['arrunwantednames'] = addarrunwantednames (stats['arrunwantednames'], convertqualitytoname[tmpdata['quality']] +namewithoutthe, tmpdata['price-bptf']); } } else { if (namewithoutthe in convertnames) namewithoutthe = convertnames[namewithoutthe]; // reinstate missing ' and - in the names supplied by bptf namewiththe = "The " +namewithoutthe; if (debug) str = str + "orig "+tmpdata['name']+" => namewiththe '" +namewiththe+"', namewithoutthe '" +namewithoutthe+ "' (price-bptf: " +tmpdata['price-bptf']+ ")\n"; stats['arrunwantednames'] = addarrunwantednames (stats['arrunwantednames'], namewiththe, tmpdata['price-bptf']); stats['arrunwantednames'] = addarrunwantednames (stats['arrunwantednames'], namewithoutthe, tmpdata['price-bptf']); } } } // end for each item in bptf-inventory save_var ("statsarrunwantednames", stats['arrunwantednames']); if (debug) { if (cntnewitems > 0) { var preEl=document.createElement("pre"); var txtEl=document.createTextNode(str); preEl.appendChild(txtEl); basediv.appendChild(preEl); preEl.style.textAlign = "left"; } } /* for (var k in stats['arrunwantednames']) { if (stats['arrunwantednames'][k].indexOf("Cyborg Stunt Helmet") >= 0) str = str + "- " +k+ " => " +stats['arrunwantednames'][k]+ "\n"; if (stats['arrunwantednames'][k].indexOf("Timeless Topper") >= 0) str = str + "- " +k+ " => " +stats['arrunwantednames'][k]+ "\n"; if (stats['arrunwantednames'][k].indexOf("Panama") >= 0) str = str + "- " +k+ " => " +stats['arrunwantednames'][k]+ "\n"; } */ if (debug) alert(str); } // end if is bptf else if ((basediv != undefined) && (imon == "whbp")) { if (debug) alert ("mid40"); infodata = walkelements (basediv.getElementsByTagName("tr"), infodata, "tr"); if (debug) alert ("mid41"); var str = "There are new items in your whbp backpack:\n\n"; var cntnewitems = 0; stats['arrunwantednames'] = load_var ('statsarrunwantednames', stats['arrunwantednames']); for (var k in infodata) { var tmpdata = infodata[k]; if ( (tmpdata) && ('name' in tmpdata) && (tmpdata['name'] != "") ) { cntnewitems++; if (tmpdata['name'].indexOf("The ") >= 0) { namewithoutthe = tmpdata['name'].substring(4); } else { namewithoutthe = tmpdata['name']; } if (namewithoutthe in convertnames) namewithoutthe = convertnames[namewithoutthe]; // reinstate missing ' and - in the names supplied by bptf namewiththe = "The " +namewithoutthe; if (debug) str = str + "orig "+tmpdata['name']+" => namewiththe '" +namewiththe+"', namewithoutthe '" +namewithoutthe+ "' (price-bptf: " +tmpdata['price-bptf']+ ")\n"; stats['arrunwantednames'] = addarrunwantednames (stats['arrunwantednames'], namewiththe, tmpdata['cost-whbp']); stats['arrunwantednames'] = addarrunwantednames (stats['arrunwantednames'], namewithoutthe, tmpdata['cost-whbp']); } } // end for each item in bptf-inventory save_var ("statsarrunwantednames", stats['arrunwantednames']); if (debug) { if (cntnewitems > 0) { var preEl=document.createElement("pre"); var txtEl=document.createTextNode(str); preEl.appendChild(txtEl); basediv.appendChild(preEl); preEl.style.textAlign = "left"; } } /* for (var k in stats['arrunwantednames']) { if (stats['arrunwantednames'][k].indexOf("Cyborg Stunt Helmet") >= 0) str = str + "- " +k+ " => " +stats['arrunwantednames'][k]+ "\n"; if (stats['arrunwantednames'][k].indexOf("Timeless Topper") >= 0) str = str + "- " +k+ " => " +stats['arrunwantednames'][k]+ "\n"; if (stats['arrunwantednames'][k].indexOf("Panama") >= 0) str = str + "- " +k+ " => " +stats['arrunwantednames'][k]+ "\n"; } */ if (debug) alert(str); } // end if is whbp else if (basediv != undefined) { alert ("got unknown imon '"+imon+"'.. No data has been imported."); } // end if is unknown imon /* TODO edit these by option dialog * TODO trigger optionsreset by option dialog and contextmenu * TODO make convertnames editable by option dialog * TODO turn off display:none behaviour temporarily (or explicitly off, only to turn on on a by case basis) * TODO add a way to manually add entries "stats['arrunwantednames']" that is not overwritten by bptf updates * TODO add a filter for coloured hats along the lines of options['numminpricebptf'] * TODO make option-configurable what urls to serve (ie. turn off hats, but leave on weapons filtering) GM_setValue ("options['arrunwantedcosts']Array", JSON.stringify(options['arrunwantedcosts'])); GM_setValue ("stats['arrunwantednames']Array", JSON.stringify(stats['arrunwantednames'])); GM_setValue ("options['arrwantedlevels']Array", JSON.stringify(options['arrwantedlevels'])); GM_setValue ("options['numminpricebptf']Scalar", options['numminpricebptf']); GM_setValue ("options['nummaxcraftno']Scalar", options['nummaxcraftno']); */ /// /// the page reloader /// /// so many lines for so little functionality *sigh* /// // TODO if the bot is offline, set pageneedsreloading to 0, no matter what it was before // then do a window.setTimeout ("reload window", numreloadaftersctf*10); // TODO if a new option to-be-created is set to 1, then the moment we are in the queue anywhere, every other window stops reloading altogether (to save cpu time and improve reception ;)) // at the same time, wait for us to stop being in the queue (checking GM_variables only, with a longer timeout maybe?) and then, once we step out of queue, reload the full window and return to scanning for wanted items. if (pageneedsreloading) { if (imon.indexOf("sctf") >= 0) { if (('numreloadaftersctfcurrent' in options) && ('numreloadafterrand' in options)) { if (options['numreloadaftersctfcurrent'] > 0) { var thisroundstimeout = 9999999; // this should be never used, so just make it a really long wait. if ((options['numreloadafterbreak'] > 0) && (options['numreloadaftersctfbreakcount'] > options['numreloadafterbreak'])) { thisroundstimeout = new Number(options['numreloadafterbreaktime'])*new Number(options['numreloadaftersctfcurrent']); options['numreloadaftersctfbreakcount'] = 0; if (debug) { alert ("Taking a reload break for " +thisroundstimeout+ " seconds.."); } else { window.setTimeout ("document.location.reload()", thisroundstimeout*1000); } } // end if take a reload break else { thisroundstimeout = new Number(options['numreloadaftersctfcurrent']) + new Number( (options['numreloadaftersctfcurrent'] * Math.random() * options['numreloadafterrand']) / 100 ); if (debug) { alert ("Reloading page in " +thisroundstimeout+ " seconds (numreloadaftersctfcurrent:" +new String(options['numreloadaftersctfcurrent'])+", numreloadafterrand:" +new String(options['numreloadafterrand'])+ ").."); } else { window.setTimeout ("document.location.reload()", thisroundstimeout*1000); } } // end if actually do the reload } else { if (debug) { alert ("Not reloading page. sctfnumreloadafter (value: "+options['sctfnumreloadafter']+ " s) has been deactivated."); } } } else { if (!('numreloadaftersctf' in options)) { options['numreloadaftersctf'] = default_options['numreloadaftersctf']; } if (!('numreloadaftersctfcurrent' in options)) { options['numreloadaftersctfcurrent'] = default_options['numreloadaftersctf']; } if (!('numreloadafterrand' in options)) { options['numreloadafterrand'] = default_options['numreloadafterrand']; } } } else if (imon.indexOf("bptf") >= 0) { if (('numreloadafterbptf' in options) && ('numreloadafterrand' in options)) { if (options['numreloadafterbptf'] > 0) { var thisroundstimeout = 9999999; // this should be never used, so just make it a really long wait. if ((options['numreloadafterbreak'] > 0) && (options['numreloadafterbptfbreakcount'] > options['numreloadafterbreak'])) { thisroundstimeout = new Number(options['numreloadafterbreaktime'])*new Number(options['numreloadafterbptf']); options['numreloadafterbptfbreakcount'] = 0; if (debug) { alert ("Taking a reload break for " +thisroundstimeout+ " seconds.."); } else { window.setTimeout ("document.location.reload()", thisroundstimeout*1000); } } // end if take a reload break else { thisroundstimeout = new Number(options['numreloadafterbptf']) + new Number( (options['numreloadafterbptf'] * Math.random() * options['numreloadafterrand']) / 100 ); if (debug) { alert ("Reloading page in " +thisroundstimeout+ " seconds.."); } else { window.setTimeout ("document.location.reload()", thisroundstimeout*1000); } } // end if actually do the reload } else { if (debug) { alert ("Not reloading page. bptfnumreloadafter (value: "+options['bptfnumreloadafter']+ " s) has been deactivated."); } } } else { if (!('numreloadafterbptf' in options)) { options['numreloadafterbptf'] = default_options['numreloadafterbptf']; } if (!('numreloadafterrand' in options)) { options['numreloadafterrand'] = default_options['numreloadafterrand']; } } } else { if (debug) { alert ("Not reloading page. Unknown imon '"+imon+"'."); } } } else { if (debug) { alert ("Not reloading page. Doesn't need reloading."); } } /// /// persistent data handling (output) /// // now save the options if something changed (very crude check, I know).. save_var ("options", options); if (debug) alert("end"); GM_setValue ("numconcurrentsessions", GM_getValue("numconcurrentsessions", 0) -1); } // end function mymain // ################ // ################ // ################