Untitled

 avatar
unknown
plain_text
2 years ago
23 kB
4
Indexable
// ==UserScript==
// @name         Instant Offer Sender
// @namespace    https://github.com/peleicht/backpack-offer-sender
// @homepage     https://github.com/peleicht
// @version      1.3.0
// @description  Adds a button on backpack.tf listings that instantly sends the offer.
// @author       Brom127
// @updateURL    https://github.com/peleicht/backpack-offer-sender/raw/main/offer_sender.user.js
// @downloadURL  https://github.com/peleicht/backpack-offer-sender/raw/main/offer_sender.user.js
// @include      /^https?:\/\/backpack\.tf\/(stats|classifieds|u).*/
// @include      /^https?:\/\/next\.backpack\.tf\/.*/
// @include      https://steamcommunity.com/tradeoffer/new*
// @icon         data:image/svg+xml,<svg xmlns=%22http://www.w3.org/2000/svg%22 viewBox=%220 0 100 100%22><text y=%22.9em%22 font-size=%2290%22>💠</text></svg>
// @run-at       document-start
// ==/UserScript==

const allow_change = true;
const btn_color = "#02d6d6";
const next_btn_color = "#00ffff";
const btn_text = "Send Tradeoffer automatically.";

let internal_request_sent = false;

main();

async function main() {
	"use strict";

	if (location.hostname == "backpack.tf" && location.pathname.match(/\/(stats|classifieds|u)/)) {
		await awaitDocumentReady();

		//add new button with item and price info in url query
		const list_elements = document.getElementsByClassName("media-list");
		let order_elements = [];
		for (let elements of list_elements) {
			const buy_sell_listings = Array.from(elements.getElementsByTagName("li"));
			order_elements = order_elements.concat(buy_sell_listings);
		}

		for (let order of order_elements) {
			//get item info
			const header = document.querySelector("#" + order.id + " > div.listing-body > div.listing-header > div.listing-title > h5");
			const item_name = header.firstChild.textContent
				.trim()
				.replace("\n", " ")
				.replace(/ #\d+$/, ""); //\n and # dont work in urls

			const info = document.querySelector("#" + order.id + " > div.listing-item > div");
			const price = info.getAttribute("data-listing_price");

			//ignore specific buy orders
			let item_id_text = "";
			if (info.getAttribute("data-listing_intent") == "buy") {
				if (
					item_name.includes("Unusual") &&
					!item_name.includes("Haunted Metal Scrap") &&
					!item_name.includes("Horseless Headless Horsemann's Headtaker")
				) {
					continue; //ignore generic unusual buy orders
				}

				const attributes = ["data-spell_1", "data-part_name_1", "data-killstreaker", "data-sheen", "data-level", "data-paint_name"];
				let modified = false;
				for (let a of attributes) {
					if (info.hasAttribute(a)) {
						if (a == "data-paint_name" && item_name.includes(info.getAttribute("data-paint_name"))) continue; //dont ignore paint cans (they're always painted)
						modified = true;
						break;
					}
				}
				if (modified) continue; //ignore modified buy orders
			} else {
				item_id_text = "&tscript_id=" + info.getAttribute("data-id");
			}

			const btn_selector = "#" + order.id + " > div.listing-body > div.listing-header > div.listing-buttons > a.btn.btn-bottom.btn-xs.btn-";
			let send_offer_btn = document.querySelector(btn_selector + "success");
			if (!send_offer_btn) send_offer_btn = document.querySelector(btn_selector + "primary"); //button is blue (negotiable listing)
			if (!send_offer_btn || send_offer_btn.getAttribute("href").startsWith("steam://")) continue; //no tradeoffer button, stop

			//add new button
			const btn_clone = send_offer_btn.cloneNode(true);
			const url = encodeURI(btn_clone.getAttribute("href") + item_id_text + "&tscript_price=" + price + "&tscript_name=" + item_name);
			btn_clone.setAttribute("href", url);
			btn_clone.style.backgroundColor = btn_color;
			btn_clone.style.borderColor = btn_color;
			if (!btn_text) {
				btn_clone.removeAttribute("title");
				btn_clone.removeAttribute("data-tip");
			} else {
				btn_clone.setAttribute("title", btn_text);
			}

			document.querySelector("#" + order.id + " > div.listing-body > div.listing-header > div.listing-buttons").append(btn_clone);
		}
	} else if (location.hostname == "next.backpack.tf") {
		//next does not refresh page between pages, so script needs to run on any next page
		let listings_data = undefined;
		interceptSearchRequests();
		if (location.pathname.startsWith("/stats")) {
			await awaitDocumentReady();
			while (!__NUXT__?.fetch || !__NUXT__?.fetch["data-v-58d43071:0"]?.listings) {
				await waitFor(0.1); //wait for listings request ready
			}
			const listings = __NUXT__.fetch["data-v-58d43071:0"].listings;
			listings_data = listings.buy.items.concat(listings.sell.items);
			addSenderButtons();
		}

		/**
		 * Intercepts the classifieds search results and adds buttons once data is ready
		 */
		function interceptSearchRequests() {
			let old_open = XMLHttpRequest.prototype.open;
			XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
				if (url.match(/https:\/\/next\.backpack\.tf\/cors\/_classifieds\/(search|item)/)) {
					const this_ref = this;
					(async () => {
						while (true) {
							await waitFor(0.1);
							if (this_ref.readyState == 4) {
								const listings = JSON.parse(this_ref.responseText);
								listings_data = listings.buy.items.concat(listings.sell.items);
								await awaitDocumentReady();
								await waitFor(0.2);
								console.log("go!");
								addSenderButtons();
								break;
							}
						}
					})();
				}

				return old_open.apply(this, arguments);
			};
		}

		function addSenderButtons() {
			//add new button with item and price info in url query (for next.backpack.tf)
			const listings = Array.from(document.getElementsByClassName("listing"));

			for (let i = 0; i < listings.length; i++) {
				const listing = listings[i];
				const header = listing.children[0].children[1].children[0]; //everythings a div, nothing has an id why ;(

				//get info
				const item_name = header.children[0].innerText
					.trim()
					.replace("\n", " ")
					.replace(/ #\d+$/, ""); //\n and # dont work in urls

				const info = listing.children[0].children[0];
				const listing_id = info.getAttribute("href").replace("/classifieds/", "");
				const price = listings_data.find(l => l.id == listing_id).value.long;

				//ignore buy orders on specific items
				let item_id_text = "";
				if (header.getElementsByClassName("text-buy").length != 0) {
					if (
						item_name.includes("Unusual") &&
						!item_name.includes("Haunted Metal Scrap") &&
						!item_name.includes("Horseless Headless Horsemann's Headtaker")
					) {
						continue; //ignore generic unusual buy orders
					}

					const modified_traits = ["fa-wrench", "fa-fill-drip", "-spell", "fa-shoe-prints", "fa-flash-round-potion"];
					const special_traits = Array.from(info.children[0].children).map(e => e.getAttribute("class"));
					let modified = false;
					for (let trait of special_traits) {
						const found_trait = modified_traits.find(t => trait.includes(t));
						if (found_trait) {
							if (found_trait == "fa-fill-drip" && info.getAttribute("style").includes("Paint_Can")) continue; //dont ignore paint cans (they're always painted)
							modified = true;
							break;
						}
					}

					if (modified) continue; //ignore modified buy orders
				} else {
					const id = /\/classifieds\/440_(\d+)/.exec(info.getAttribute("href"));
					item_id_text = "&tscript_id=" + id[1];
				}

				const btn_box = header.getElementsByClassName("listing__details__actions")[0];
				const send_offer_btn = btn_box.getElementsByClassName("listing__details__actions__action")[0];
				const href = send_offer_btn.getAttribute("href");
				if (!href || href.startsWith("steam://") || href.startsWith("https://marketplace.tf")) continue;

				//add new button
				const btn_clone = send_offer_btn.cloneNode(true);
				const url = encodeURI(href + item_id_text + "&tscript_price=" + price + "&tscript_name=" + item_name);
				btn_clone.setAttribute("href", url);
				btn_clone.id = "instant-button-" + i;
				const icon = btn_clone.children[0];
				icon.style.color = next_btn_color;

				const existing_button = document.getElementById(btn_clone.id); //remove if another button exists already
				if (existing_button) existing_button.remove();

				btn_box.append(btn_clone);
			}
		}
	} else if (location.hostname == "steamcommunity.com" && location.pathname.startsWith("/tradeoffer/new")) {
		const params = new URLSearchParams(location.search);
		if (!params.has("tscript_price")) return;

		interceptInventoryRequest();
		await awaitDocumentReady();

		const items_to_give = [];
		const items_to_receive = [];

		const [our_inventory, their_inventory] = await getInventories();
		window.our_inv = our_inventory;
		window.their_inv = their_inventory;

		if (!params.has("tscript_id")) {
			//sell your item
			const needed_item_name = params.get("tscript_name").replace("u0023", "#");
			const needed_item = our_inventory.find(i => i.name == needed_item_name);
			if (!needed_item) return throwError("Could not find item in your inventory.");

			items_to_give.push(toTradeOfferItem(needed_item.id));

			//get partner currencies
			const currency_string = params.get("tscript_price");
			const currencies = toCurrencyTypes(currency_string);
			const [their_currency, change] = pickCurrency(their_inventory, ...currencies);
			if (change.find(c => c != 0)) {
				const [our_currency, change2] = pickCurrency(our_inventory, 0, ...currencies);
				if (change2.find(c => c != 0)) return throwError("Could not balance currencies");
				for (let c of our_currency) items_to_give.push(toTradeOfferItem(c.id));
			}

			for (let c of their_currency) items_to_receive.push(toTradeOfferItem(c.id));
		} else {
			//buy partners item
			const item_id = params.get("tscript_id");
			let needed_item = their_inventory.find(i => i.id == item_id);
			if (!needed_item) {
				const needed_item_name = params.get("tscript_name").replace("u0023", "#"); //get other instance of same item if item with exact id already sold
				needed_item = our_inventory.find(i => i.name == needed_item_name);
			}
			if (!needed_item) return throwError("Item has already been sold.");

			items_to_receive.push(toTradeOfferItem(needed_item.id));

			//get your currencies
			const currency_string = params.get("tscript_price");
			const currencies = toCurrencyTypes(currency_string);
			const [our_currency, change] = pickCurrency(our_inventory, ...currencies);
			if (change.find(c => c != 0)) {
				const [their_currency, change2] = pickCurrency(their_inventory, 0, ...change);
				if (change2.find(c => c != 0)) return throwError("Could not balance currencies");
				for (let c of their_currency) items_to_receive.push(toTradeOfferItem(c.id));
			}

			for (let c of our_currency) items_to_give.push(toTradeOfferItem(c.id));
		}

		const offer_id = await sendOffer(items_to_give, items_to_receive);
		if (offer_id) window.close(); //success
	}
}

function getInventories() {
	return new Promise(async res => {
		while (!UserYou.rgContexts["440"]) {
			await waitFor(0.1);
		}

		if (!internal_request_sent) UserYou.getInventory(440, 2);
		UserThem.LoadForeignAppContextData(g_ulTradePartnerSteamID, 440, 2);

		let done = false;
		setTimeout(() => {
			if (!done) throwError("Timeout waiting for inventory data.");
		}, 15000);

		const inventories = await Promise.all([getSingleInventory(UserYou), getSingleInventory(UserThem)]);
		done = true;

		res(inventories);
	});

	function getSingleInventory(User) {
		return new Promise(async res => {
			let inv = User.rgContexts["440"]["2"].inventory?.rgInventory;
			if (!inv || User.cLoadsInFlight != 0) {
				if (User.cLoadsInFlight == 0) User.loadInventory();
				inv = await waitForInventoryLoad();
			} else inv = Object.values(inv);

			res(parseInventory(inv));
		});

		function waitForInventoryLoad() {
			return new Promise(async res => {
				let done = false;

				//poll for inventory ready
				(async () => {
					let inv = User.rgContexts["440"]["2"].inventory?.rgInventory;
					while (!inv) {
						await waitFor(0.5);
						if (done) return;
						inv = User.rgContexts["440"]["2"].inventory?.rgInventory;
					}

					done = true;
					const parsed_inv = Object.values(inv);
					res(parsed_inv);
				})();

				//wait for intercepted request, fast but less reliable
				const on_load = User.OnLoadInventoryComplete;
				User.OnLoadInventoryComplete = function (data, appid, contextid) {
					if (appid == 440 && contextid == 2) {
						done = true;
						res(Object.values(data.responseJSON.rgInventory));
					}

					User.OnLoadInventoryComplete = on_load;
					return on_load.apply(this, arguments);
				};
				const on_fail = User.OnInventoryLoadFailed;
				User.OnInventoryLoadFailed = async function (data, appid, contextid) {
					if (appid == 440 && contextid == 2) {
						console.log("load failed, requesting manually");
						const inv = await getInventory(User.strSteamId);
						done = true;
						res(inv);
					}

					User.OnInventoryLoadFailed = on_fail;
					return on_fail.apply(this, arguments);
				};
			});
		}
	}

	function parseInventory(items) {
		return items.map(item => {
			return {
				id: item.id,
				name: nameFromItem(item),
			};
		});
	}
}

async function getInventory(steam_id) {
	let body;
	try {
		const response = await fetch("https://steamcommunity.com/inventory/" + steam_id + "/440/2?count=2000&l=english");
		if (!response.ok) throw response.status;
		body = await response.json();

		if (body.more_items) {
			const more_response = await fetch("https://steamcommunity.com/inventory/" + steam_id + "/440/2?count=1000&more_start=1000&l=english");
			if (!more_response.ok) throw more_response.status;
			const more_body = await more_response.json();

			body.assets = body.assets.concat(more_body.assets);
			body.descriptions = body.descriptions.concat(more_body.descriptions);
		}
	} catch (err) {
		return throwError("Could not obtain inventory data: " + err);
	}

	const quickDescriptionLookup = {};
	const inv = [];

	for (let i = 0; i < body.assets.length; i++) {
		const description = getDescription(body.descriptions, body.assets[i].classid, body.assets[i].instanceid);
		description.id = body.assets[i].assetid;
		description.name = nameFromItem(description);
		inv.push(JSON.parse(JSON.stringify(description)));
	}

	return inv;

	/**
	 * @credit node-steamcommunity by DoctorMcKay
	 */
	function getDescription(descriptions, classID, instanceID) {
		const key = classID + "_" + (instanceID || "0");

		if (quickDescriptionLookup[key]) {
			return quickDescriptionLookup[key];
		}

		for (let i = 0; i < descriptions.length; i++) {
			quickDescriptionLookup[descriptions[i].classid + "_" + (descriptions[i].instanceid || "0")] = descriptions[i];
		}

		return quickDescriptionLookup[key];
	}
}

async function sendOffer(items_to_give, items_to_receive) {
	const params = new URLSearchParams(location.search);

	const body = {
		sessionid: g_sessionID,
		serverid: 1,
		partner: g_ulTradePartnerSteamID,
		tradeoffermessage: "",
		json_tradeoffer: JSON.stringify({
			newversion: true,
			version: items_to_give.length + items_to_receive.length + 1,
			me: { assets: items_to_give, currency: [], ready: false },
			them: { assets: items_to_receive, currency: [], ready: false },
		}),
		captcha: "",
		trade_offer_create_params: JSON.stringify({
			trade_offer_access_token: params.get("token"),
		}),
	};
	const form = new FormData();
	for (let key in body) form.append(key, body[key]);

	try {
		const response_body = await (
			await fetch("https://steamcommunity.com/tradeoffer/new/send", {
				method: "POST",
				body: form,
			})
		).json();

		if (response_body.strError) return throwError(response_body.strError);

		return response_body.tradeofferid;
	} catch {}
}

function nameFromItem(item) {
	let name = item.market_hash_name;

	if (item.descriptions != undefined) {
		for (let i = 0; i < item.descriptions.length; i++) {
			const desc = item.descriptions[i];

			if (desc.value.includes("''")) continue;
			else if (desc.value == "( Not Usable in Crafting )") name = "Non-Craftable " + name;
			else if (desc.value.startsWith("★ Unusual Effect: ")) {
				for (let tag of item.tags) {
					if (tag.category == "Type" && tag.internal_name == "Supply Crate") continue; //crates have normal unusual tag
				}
				const effect = desc.value.substring("★ Unusual Effect: ".length);
				name = name.replace("Unusual", effect);
			}
		}
	}

	name = name.replace("\n", " ");
	name = name.replace("Series #", "#"); //case 'series' keyword not included in bp names
	name = name.replace(/ #\d+$/, ""); //remove case number

	return name;
}

function toTradeOfferItem(id) {
	return {
		appid: 440,
		contextid: "2",
		amount: 1,
		assetid: id,
	};
}
function toCurrencyTypes(currency_string) {
	const match = currency_string.match(/^(\d+ keys?,? ?)?(\d+(?:\.\d+)? ref)?$/);
	if (!match) return throwError("Could not parse currency " + currency_string);

	let keys = 0;
	let metal = 0;
	if (match[1]) {
		const key_length = match[1].indexOf(" ");
		keys = Number(match[1].slice(0, key_length));
	}
	if (match[2]) {
		const ref_length = match[2].indexOf(" ");
		metal = Number(match[2].slice(0, ref_length));
	}

	const ref = Math.floor(metal);
	const small_metal = Math.round((metal % 1) * 100);
	const rec = Math.floor(small_metal / 33);
	const scrap = (small_metal / 11) % 3;

	if (small_metal != 0 && String(small_metal)[0] != String(small_metal)[1]) return throwError("Invalid currency " + currency_string);

	return [keys, ref, rec, scrap];
}

function pickCurrency(inventory, keys, ref, rec, scrap) {
	const inv_keys = inventory.filter(item => item.name == "Mann Co. Supply Crate Key");
	const inv_ref = inventory.filter(item => item.name == "Refined Metal");
	const inv_rec = inventory.filter(item => item.name == "Reclaimed Metal");
	const inv_scrap = inventory.filter(item => item.name == "Scrap Metal");

	if (inv_keys.length < keys) return throwError("Insufficient Keys");
	if (allow_change && inv_ref.length + inv_rec.length / 3 + inv_scrap.length / 9 < ref + rec / 3 + scrap / 9) return throwError("Insufficient Metal");
	if (!allow_change && (inv_ref.length < ref || inv_rec.length < rec || inv_scrap.length < scrap)) return throwError("Insufficient Metal");

	let leftover_ref = inv_ref.length - ref;
	let leftover_rec = inv_rec.length - rec;
	let leftover_scrap = inv_scrap.length - scrap;
	let change = { ref: 0, rec: 0, scrap: 0 };

	//use rec if not enough scrap
	if (leftover_scrap < 0) {
		leftover_scrap = -leftover_scrap;
		rec += Math.ceil(leftover_scrap / 3);
		leftover_rec -= Math.ceil(leftover_scrap / 3);
		change.scrap += 3 - (leftover_scrap % 3);
		change.scrap = change.scrap % 3;
		scrap -= leftover_scrap;
		leftover_scrap = 0;
	}

	//use ref if not enough rec
	if (leftover_rec < 0) {
		leftover_rec = -leftover_rec;
		ref += Math.ceil(leftover_rec / 3);
		leftover_ref -= Math.ceil(leftover_rec / 3);
		change.rec += 3 - (leftover_rec % 3);
		change.rec = change.rec % 3;
		rec -= leftover_rec;
		leftover_rec = 0;
	}

	//use rec if not enough ref
	while (leftover_ref < 0) {
		if (leftover_rec >= -leftover_ref * 3) {
			ref -= -leftover_ref;
			rec += -leftover_ref * 3;
			leftover_rec -= -leftover_ref * 3;
			leftover_ref = 0;
		}
	}

	//calculate change needed from other inventory
	if (ref != 0 && change.ref != 0) {
		let reduce = Math.min(ref, change.ref);
		ref -= reduce;
		change.ref -= reduce;
	}
	if (rec != 0 && change.rec != 0) {
		let reduce = Math.min(rec, change.rec);
		rec -= reduce;
		change.rec -= reduce;
	}
	if (scrap != 0 && change.scrap != 0) {
		let reduce = Math.min(scrap, change.scrap);
		scrap -= reduce;
		change.scrap -= reduce;
	}

	//start taking items from random position; possible ranges are between 0 and length-amount
	const key_start = Math.floor(Math.random() * (inv_keys.length - keys + 1));
	const ref_start = Math.floor(Math.random() * (inv_ref.length - ref + 1));
	const rec_start = Math.floor(Math.random() * (inv_rec.length - rec + 1));
	const scrap_start = Math.floor(Math.random() * (inv_scrap.length - scrap + 1));

	//actually take the items
	const take_keys = inv_keys.slice(key_start, key_start + keys);
	const take_ref = inv_ref.slice(ref_start, ref_start + ref);
	const take_rec = inv_rec.slice(rec_start, rec_start + rec);
	const take_scrap = inv_scrap.slice(scrap_start, scrap_start + scrap);
	let items = take_keys;
	items = items.concat(take_ref);
	items = items.concat(take_rec);
	items = items.concat(take_scrap);

	//checks if anything went wrong. This should never happen but lets check anyways.
	if (
		keys < 0 ||
		ref < 0 ||
		rec < 0 ||
		scrap < 0 ||
		change.ref < 0 ||
		change.rec < 0 ||
		change.scrap < 0 ||
		key_start < 0 ||
		ref_start < 0 ||
		rec_start < 0 ||
		scrap_start < 0 ||
		keys == undefined ||
		ref == undefined ||
		rec == undefined ||
		scrap == undefined ||
		keys > inv_keys.length ||
		ref > inv_ref.length ||
		rec > inv_rec.length ||
		scrap > inv_scrap.length ||
		items.length < keys ||
		take_keys.length != keys ||
		take_ref.length != ref ||
		take_rec.length != rec ||
		take_scrap.length != scrap
	) {
		console.log("Something went wrong balancing currencies:");
		console.log(
			[
				inv_keys.length,
				inv_ref.length,
				inv_rec.length,
				inv_scrap.length,
				keys,
				ref,
				rec,
				scrap,
				key_start,
				ref_start,
				rec_start,
				scrap_start,
				take_keys,
				take_ref,
				take_rec,
				take_scrap,
				JSON.stringify(items, undefined, 4),
			].join("\n")
		);
		return throwError("Could not balance currencies");
	}

	return [items, [change.ref, change.rec, change.scrap]];
}

/**
 * Sets internal_request_sent to true once a request to the internal inventory api has been made.
 */
function interceptInventoryRequest() {
	let old_open = XMLHttpRequest.prototype.open;
	XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
		if (url.endsWith("/json/440/2/?trading=1")) {
			internal_request_sent = true;
			XMLHttpRequest.prototype.open = old_open;
		}

		return old_open.apply(this, arguments);
	};
}

function awaitDocumentReady() {
	return new Promise(async res => {
		if (document.readyState != "loading") res();
		else document.addEventListener("DOMContentLoaded", res);
	});
}

function waitFor(seconds) {
	return new Promise(res => setTimeout(res, seconds * 1000));
}

function throwError(err) {
	const params = new URLSearchParams(location.search);
	const buy_sell = params.has("for_item") ? "Buy" : "Sell";
	const item = params.get("tscript_name");
	const pre_string = "Unable to " + buy_sell + " " + item + ": ";

	window.alert(pre_string + err);
	//window.close();
	throw err;
}
Editor is loading...