Bundle Helper

Add tools for many bundle sites.

当前为 2016-09-15 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name        Bundle Helper
// @namespace   iFantz7E.BundleHelper
// @version     0.07
// @description Add tools for many bundle sites.
// @match      	*://www.hrkgame.com/*
// @match      	*://www.bundlestars.com/*/bundle/*
// @match      	*://www.bundlestars.com/*/promotions/*
// @match      	*://www.reddit.com/r/*/comments/*
// @match      	*://groupees.com/*
// @match      	*://www.indiegala.com/*
// @match      	*://cubicbundle.com/*
// @match      	*://www.dailyindiegame.com/*
// @match      	*://dailyindiegame.com/*
// @match      	*://www.bundlekings.com/*
// @match      	*://www.orlygift.com/*
// @match      	*://otakumaker.com/*
// @match      	*://www.otakumaker.com/*
// @match      	*://www.superduperbundle.com/*
// @match      	*://www.humblebundle.com/*
// @match      	*://store.steampowered.com/widget/*
// @match      	*://store.steampowered.com/app/*
// @match      	*://steamcommunity.com/*/home/
// @match      	*://forums.steampowered.com/forums/showthread.php?*
// @match      	*://www.gamebundle.com/*
// @match      	*://steamcompanion.com/gifts/*
// @match      	*://whosgamingnow.net/discussion/*
// @run-at		document-start
// @grant       GM_addStyle
// @grant       GM_xmlhttpRequest
// @icon      	http://store.steampowered.com/favicon.ico
// @copyright	2016, 7-elephant
// ==/UserScript==

(function ()
{	
	function attachOnLoad(callback)
	{
		window.addEventListener("load", function (e) 
		{
			callback();
		});
	}

	function attachOnReady(callback) 
	{
		document.addEventListener("DOMContentLoaded", function (e) 
		{
			callback();
		});
	}

	function insertBeforeElement(newNode, referenceNode) 
	{
		referenceNode.parentNode.insertBefore(newNode, referenceNode);
	}

	function insertAfterElement(newNode, referenceNode) 
	{
		referenceNode.parentNode.insertBefore(newNode, referenceNode.nextSibling);
	}

	function reload()
	{
		window.location = window.location.href;
	}

	function getQueryByName(name, url) 
	{
		if (url == null)
			url = location.search;
		name = name.replace(/[\[]/, "\\[").replace(/[\]]/, "\\]");
		var regex = new RegExp("[\\?&]" + name + "=([^&#]*)");
		var results = regex.exec(url);
		return results == null ? "" : decodeURIComponent(results[1].replace(/\+/g, " "));
	}

	function clickToSelect(ele)
	{
		if (ele != null)
		{
			var range = document.createRange();
			range.setStartBefore(ele.firstChild);
			range.setEndAfter(ele.lastChild);

			var sel = window.getSelection();
			sel.removeAllRanges();
			sel.addRange(range);
		}
	}
	
	function normalizeArray(arr)
	{
		arr = arr.filter(function(elem, index, self) 
		{
			return index == self.indexOf(elem);
		});
		return arr;
	}
	
	function randNum(min, max)
	{
		return Math.round(Math.random() * (max - min) + min);
	}
	
	function markOwned(query, markOwnedCallback)
	{		
		var rgxId = /[0-9]+/;
		var rgxApp = /^http[s]?:\/\/store\.steampowered\.com\/app\/[0-9]+/i;
		var rgxSub = /^http[s]?:\/\/store\.steampowered\.com\/sub\/[0-9]+/i;
		
		GM_xmlhttpRequest(
		{
			method: "GET",
			url: "https://store.steampowered.com/dynamicstore/userdata/?t=" + randNum(1000, 9999),
			onload: function(response) 
			{
				var dataRes = JSON.parse(response.responseText);
				
				var countOwned = [0, 0];
				var countAll = [0, 0];
				
				if (typeof dataRes["rgOwnedApps"] !== "undefined"
					&& typeof dataRes["rgOwnedPackages"] !== "undefined"
					&& typeof dataRes["rgIgnoredApps"] !== "undefined")
				{
					var eleApps = document.querySelectorAll(query);
					for (var i = 0; i < eleApps.length; i++)
					{
						var attrHref = eleApps[i].getAttribute("href");
						var id = rgxId.exec(attrHref);
						if (id != null)
						{
							var valId = parseInt(id);
								
							if (rgxApp.test(attrHref))
							{
								if (dataRes["rgOwnedApps"].indexOf(valId) > -1)
								{
									markOwnedCallback(eleApps[i]);
									countOwned[0]++;
								}
								else
								{
									console.log("App: not owned - http://store.steampowered.com/app/" + id + "/");
								}
								countAll[0]++;
							}
							else if (rgxSub.test(attrHref))
							{								
								if (dataRes["rgOwnedPackages"].indexOf(valId) > -1)
								{
									markOwnedCallback(eleApps[i]);
									countOwned[1]++;
								}
								else
								{
									console.log("Sub: not owned - http://store.steampowered.com/sub/" + id + "/");
								}
								countAll[1]++;
							}
						}
					}
					
				}
				
				console.log("App: " + countOwned[0] + "/" + countAll[0]);
				console.log("Sub: " + countOwned[1] + "/" + countAll[1]);
				
			} // End onload
		});
	}
	
	function markOwned_old(query, getLabelCallback)
	{
		var apps = [];
		
		var eleApps = document.querySelectorAll(query);
		
		for (var i = 0; i < eleApps.length; i++)
		{
			var app = /[0-9]+/.exec(eleApps[i].getAttribute("href"));
			if (app != null)
			{
				apps.push(app[0]);
			}
		}
		
		apps = apps.filter(function(elem, index, self) 
		{
			return index == self.indexOf(elem);
		});
		
		console.log("Apps: " + apps.length);
		var appAll = apps.join(",");
		
		GM_xmlhttpRequest(
		{
			method: "GET",
			headers: 
			{
				"Cache-Control": "max-age=0"
			},
			url: "https://store.steampowered.com/api/appuserdetails/?appids=" + appAll,
			onload: function(response) 
			{
				var dataRes = JSON.parse(response.responseText);
				
				var countOwned = 0;
				
				var eleApps = document.querySelectorAll(query);
				for (var i = 0; i < eleApps.length; i++)
				{
					var appUrl = eleApps[i].getAttribute("href");
					if (appUrl.indexOf("http://store.steampowered.com/app/") > -1)
					{
						var app = /[0-9]+/.exec(appUrl);
						if (app != null)
						{
							if (typeof dataRes[app] !== "undefined")
							{
								if (dataRes[app].success)
								{
									if (dataRes[app].data.is_owned)
									{
										var eleLabel = getLabelCallback(eleApps[i]);
										eleLabel.classList.add("bh_owned");
										countOwned++;
									}
									else
									{
										console.log("App: not owned - http://store.steampowered.com/app/" + app + "/");
									}
								}
								else
								{
									console.log("App: not success - https://steamdb.info/app/" + app + "/");
								}
							}
						}
					}
				}
				
				console.log("Apps: owned - " + countOwned);
				
			} // End onload
		});
	}

	function main()
	{
		// #8BC349
		// #6EA028
		// #2ECC71
		// #92B300
		
		GM_addStyle(
			"   .bh_button { "
			+ "	  border-radius: 2px; border: medium none; padding: 10px; display: inline-block; "
			+ "   cursor: pointer; background: #67C1F5 none repeat scroll 0% 0%; "
			+ "   position: fixed; right: 20px; bottom: 20px; z-index:3; } "
			+ " .bh_button a { "
			+ "   text-decoration: none !important; color: #FFF !important; "
			+ "   padding: 0px 2px; } "
			+ " .bh_button:hover a { "
			+ "   color: #0079BF !important; } "
			+ " .bh_button, .bh_button a { "
			+ "   font-family: Verdana; font-size: 12px; "
			+ "   line-height: 16px; } "
			+ " .bh_owned { background-color: #7CA156 !important; "
			+ "   transition: background 500ms ease 0s; } "
		);
		
		var url = document.documentURI;
		
		if (url.indexOf("hrkgame.com") > -1)
		{
			GM_addStyle(
				" .bh_owned { background-color: #97BA22 !important; } "
			);
			
			if (url.indexOf("/randomkeyshop/make-bundle") > -1)
			{
				// Add load button
				{
					var divButton = document.createElement("div");
					divButton.classList.add("bh_button");
					divButton.id = "bh_loadAll";
					divButton.setAttribute("style", "bottom: 80px;");
					divButton.setAttribute("onclick", " \
						this.firstElementChild.textContent = \"Loading...\"; \
						window.scrollTo(0,document.body.scrollHeight); \
						var countHidden = 5; \
						var idx = setInterval(function(ele) \
						{ \
							var eleLoad = document.querySelector(\"#loader-icon\"); \
							if (eleLoad) \
							{ \
								window.scrollTo(0,document.body.scrollHeight); \
								if (eleLoad.style.display == \"none\") \
								{ \
									countHidden--; \
								} \
								else \
								{ \
									countHidden = 5; \
								} \
							} \
							else \
							{ \
								countHidden--; \
							} \
							if (countHidden < 0) \
							{ \
								clearInterval(idx); \
								ele.style.display=\"none\"; \
								var eleRes = document.querySelector(\"#result\"); \
								if (eleRes) \
								{ \
									eleRes.scrollIntoView(true); \
									window.scrollBy(0, -80); \
								} \
							} \
						}, 500, this); \
						return false; \
					");
					divButton.innerHTML = "<a onclick='return false;'>Load All</a>";
					document.body.appendChild(divButton);
				}
				
				// Add mark button
				{
					var divButton = document.createElement("div");
					divButton.classList.add("bh_button");
					divButton.id = "bh_markOwned";
					
					var eleA = document.createElement("a");
					eleA.setAttribute("onclick", "return false;");
					eleA.textContent = "Mark Owned";
					
					divButton.appendChild(eleA);
					document.body.appendChild(divButton);				
					
					divButton.addEventListener("click", function()
					{						
						GM_xmlhttpRequest(
						{
							method: "GET",
							url: "https://store.steampowered.com/dynamicstore/userdata/?t=" + randNum(1000, 9999),
							onload: function(response) 
							{
								var dataResSteam = JSON.parse(response.responseText);
								
								if (typeof dataResSteam["rgOwnedApps"] == "undefined"
									|| dataResSteam["rgOwnedApps"].length == 0)
								{
									console.log("not logged in");
								}	
								else if (typeof dataResSteam["rgOwnedApps"] !== "undefined"
									&& typeof dataResSteam["rgOwnedPackages"] !== "undefined"
									&& typeof dataResSteam["rgIgnoredApps"] !== "undefined")
								{
									var parser = new DOMParser();
									var rgxId = /[0-9]+/;
									var rgxApp = /:\/\/store\.steampowered\.com\/app\/[0-9]+/i;
									var rgxSub = /:\/\/store\.steampowered\.com\/sub\/[0-9]+/i;
									
									var elesProduct = document.querySelectorAll("#result .content");
									for (var i = 0; i < elesProduct.length; i++)
									{
										var dataHref = elesProduct[i].firstElementChild.getAttribute("data-href");
										if (dataHref != null)
										{
											var fullHref = "https://www.hrkgame.com" + dataHref + "/";
											elesProduct[i].setAttribute("bh_href", fullHref);
											
											GM_xmlhttpRequest(
											{
												method: "GET",
												url: fullHref,
												onload: function(response) 
												{
													var isOwned = false;
													
													var dataRes = parser.parseFromString(response.responseText, "text/html");
													
													var eleA = dataRes.querySelector(".storeside a.item[href*='store.steampowered.com/']");
													if (eleA != null)
													{
														var attrHref = eleA.href;
														var id = rgxId.exec(attrHref);
														if (id != null)
														{
															var valId = parseInt(id);
																
															if (rgxApp.test(attrHref))
															{
																if (dataResSteam["rgOwnedApps"].indexOf(valId) > -1)
																{
																	isOwned = true;
																}
																else
																{
																	console.log("App: not owned - http://store.steampowered.com/app/" + id + "/");
																}
															}
															else if (rgxSub.test(attrHref))
															{								
																if (dataResSteam["rgOwnedPackages"].indexOf(valId) > -1)
																{
																	isOwned = true;
																}
																else
																{
																	console.log("Sub: not owned - http://store.steampowered.com/sub/" + id + "/");
																}
															}
															
															if (isOwned)
															{
																for (var i = 0; i < elesProduct.length; i++)
																{
																	if (elesProduct[i].getAttribute("bh_href") == response.finalUrl)
																	{
																		elesProduct[i].classList.add("bh_owned");
																		break;
																	}
																}
															}
														}
													}
													else
													{
														console.log("Info: not found - " + response.finalUrl);
													}
													
												} // End onload
											});
										}
									}
									
								}
								
							} // End onload
						});
					});
				}
			}
			else if (url.indexOf("/library") > -1)
			{				
				var clientScript = ' \
					confirm = function() \
					{ \
						return true; \
					}; \
				';

				var eleClientScript = document.createElement("script");
				eleClientScript.innerHTML = clientScript;
				document.head.appendChild(eleClientScript);
			}
		}
		else if (url.indexOf("bundlestars.com") > -1)
		{
			GM_addStyle(
				//" .bh_owned { background-color: #A7CC00 !important; } "
				" .bh_owned { background-color: #D0FE00 !important; } "
				+ " .bh_owned:hover { background-color: #BBE500 !important; } "
				+ " .bh_owned * { color: #444 !important; } "
				+ " .bh_owned .was, .bh_owned .was * { color: #777 !important; } "
				+ " #launcher { bottom: 100px !important; } "
			);
			
			if (url.indexOf("/bundle/") > -1)
			{
				// Add mark button
				{
					var divButton = document.createElement("div");
					divButton.classList.add("bh_button");
					divButton.id = "bh_markOwned";
					
					var eleA = document.createElement("a");
					eleA.setAttribute("onclick", "return false;");
					eleA.textContent = "Mark Owned";
					
					divButton.appendChild(eleA);
					document.body.appendChild(divButton);
					
					divButton.addEventListener("click", function()
					{
						markOwned(".bundle-accordian a.btn-bundle-more[href*='store.steampowered.com/']", function(ele)
						{
							ele.parentElement.parentElement
								.parentElement.parentElement.parentElement
								.parentElement.parentElement.parentElement
								.parentElement.firstElementChild
								.classList.add("bh_owned");
						});
					});
				}
			}
			else if (url.indexOf("/promotions/") > -1)
			{
				// Add mark button
				{
					var divButton = document.createElement("div");
					divButton.classList.add("bh_button");
					divButton.id = "bh_markOwned";
					
					var eleA = document.createElement("a");
					eleA.setAttribute("onclick", "return false;");
					eleA.textContent = "Mark Owned";
					
					divButton.appendChild(eleA);
					document.body.appendChild(divButton);
					
					divButton.param_promo = url.substr(url.indexOf("/promotions/") + 12)
						.replace(/\?.*/, "").replace(/#.*/, "");
					eleA.param_promo = divButton.param_promo;
					
					divButton.addEventListener("click", function(e)
					{						
						var promo = e.target.param_promo;
						
						GM_xmlhttpRequest(
						{
							method: "GET",
							url: "https://store.steampowered.com/dynamicstore/userdata/?t=" + randNum(1000, 9999),
							onload: function(response) 
							{
								var dataResSteam = JSON.parse(response.responseText);
								
								if (typeof dataResSteam["rgOwnedApps"] == "undefined"
									|| dataResSteam["rgOwnedApps"].length == 0)
								{
									console.log("not logged in");
								}	
								else if (typeof dataResSteam["rgOwnedApps"] !== "undefined"
									&& typeof dataResSteam["rgOwnedPackages"] !== "undefined"
									&& typeof dataResSteam["rgIgnoredApps"] !== "undefined")
								{										
									var elesProduct = document.querySelectorAll(".bs-card-body");
									for (var i = 0; i < elesProduct.length; i++)
									{
										var sref = elesProduct[i].firstElementChild.getAttribute("ui-sref");
										if (sref != null)
										{
											elesProduct[i].setAttribute("bh_slug"
												, sref.replace("base.game({ slug: '", "")
													.replace("base.dlc({ slug: '", "")
													.replace("' })", ""));
										}
									}
									
									GM_xmlhttpRequest(
									{
										method: "GET",
										url: "https://www.bundlestars.com/api/promotions/" + promo,
										onload: function(response) 
										{
											var dataRes = JSON.parse(response.responseText);
											
											var slugs = [];
											
											var i = dataRes.length - 1;
											//for (var i = 0; i < dataRes.length; i++)
											{
												
												for (var j = 0; j < dataRes[i].products.length; j++)
												{
													if (dataRes[i].products[j].drm.steam)
													{
														slugs.push(dataRes[i].products[j].slug);
													}
												}
											}
											
											slugs = normalizeArray(slugs);
											
											for (var i = 0; i < slugs.length; i++)
											{
												GM_xmlhttpRequest(
												{
													method: "GET",
													url: "https://www.bundlestars.com/api/products/" + slugs[i],
													onload: function(response) 
													{
														var isOwned = false;
														
														var dataRes = JSON.parse(response.responseText);
														
														if (!dataRes.steam.sub)
														{
															if (dataResSteam["rgOwnedApps"].indexOf(dataRes.steam.id) > -1)
															{
																isOwned = true;
															}
															else
															{
																console.log("App: not owned - http://store.steampowered.com/app/" + dataRes.steam.id + "/ - " + dataRes.slug);
															}
														}
														else
														{
															if (dataResSteam["rgOwnedPackages"].indexOf(dataRes.steam.id) > -1)
															{
																isOwned = true;
															}
															else
															{
																console.log("Sub: not owned - http://store.steampowered.com/sub/" + dataRes.steam.id + "/ - " + dataRes.slug);
															}
														}
														
														if (isOwned)
														{
															for (var i = 0; i < elesProduct.length; i++)
															{
																if (elesProduct[i].getAttribute("bh_slug") == dataRes.slug)
																{
																	elesProduct[i].classList.add("bh_owned");
																	break;
																}
															}
														}
														
													} // End onload
												});
											}
											
										} // End onload
									});
								
								}
								
							} // End onload
						});
					});
				}
			}
		}
		else if (url.indexOf("reddit.com") > -1)
		{
			GM_addStyle(
				" .bh_owned , .md .bh_owned code { background-color: #DFF0D8 !important; } "
			);
			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					markOwned("td > a[href*='store.steampowered.com/']", function(ele)
					{
						ele.parentElement.parentElement.classList.add("bh_owned");
					});					
				});
			}
		}
		else if (url.indexOf("groupees.com") > -1)
		{
			GM_addStyle(
				"   .bh_owned { background-color: #DFF0D8 !important; } "
				+ " #subscribe-form { display: none; } "
				+ " input[name='search'] { position: fixed; z-index: 1099; left: 18%; top: 16px; } "
				+ " button[role='show3dKeyModal'] { position: fixed; z-index: 1099; left: 72%; top: 16px; } "
				+ " .cancel-spin { position: fixed !important; z-index: 1099 !important; left: 39.5% !important; top: 22px !important; } "
				+ " .bh_owned_dark { background-color: rgba(140, 197, 63, 0.6) !important; } "	
			);
			/*
			var a = document.querySelector("input[name='search']");
			var b = document.querySelector("#subscribe-form");
			b.parentElement.insertBefore(a,b);
			b.parentElement.removeChild(b);
			*/
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);				
				
				divButton.addEventListener("click", function()
				{
					var apps = [];
					
					var eleApps = document.querySelectorAll(".bundle > .products .info .description a[href*='store.steampowered.com/app/']"
						+ ", .expanded .product-info a[href*='store.steampowered.com/app/']"
						+ ", .product-details.hidden .external-links a[href*='store.steampowered.com/app/']");
					console.log("Apps: " + eleApps.length);
					
					for (var i = 0; i < eleApps.length; i++)
					{
						var app = /[0-9]+/.exec(eleApps[i].getAttribute("href"));
						if (app != null)
						{
							apps.push(app[0]);
						}
					}
					
					apps = apps.filter(function(elem, index, self) 
					{
						return index == self.indexOf(elem);
					});
					
					var appAll = apps.join(",");
					
					GM_xmlhttpRequest(
					{
						method: "GET",
						headers: 
						{
							"Cache-Control": "max-age=0"
						},
						url: "https://store.steampowered.com/api/appuserdetails/?appids=" + appAll,
						onload: function(response) 
						{
							var dataRes = JSON.parse(response.responseText);
							
							var countOwned = 0;
							
							var elePurchases = null;
							var elePrds = document.querySelectorAll(".bundle > .products .product h3"
								+ ", .expanded .details, .product-info");
							var eleApps = document.querySelectorAll(".bundle > .products .info"
								+ ", .expanded .details, .product-details.hidden");
							for (var i = 0; i < eleApps.length; i++)
							{								
								var eleApp = eleApps[i].querySelector(".description a[href*='store.steampowered.com/app/']"
									+ ", .product-info a[href*='store.steampowered.com/app/']"
									+ ", .external-links a[href*='store.steampowered.com/app/']");
								if (eleApp != null)
								{
									var app = /[0-9]+/.exec(eleApp.getAttribute("href"));
									if (app != null)
									{
										if (typeof dataRes[app] !== "undefined")
										{
											if (dataRes[app].success)
											{
												if (dataRes[app].data.is_owned)
												{
													var eleLabel = elePrds[i];
													if (eleLabel.classList.contains("product-info"))
													{
														eleLabel.classList.add("bh_owned_dark");
														
														// Mark game in build bundles
														{
															var eleName = eleLabel.querySelector("h4");
															if (eleName)
															{
																var name = eleName.textContent.trim();
																
																if (elePurchases == null)
																{
																	elePurchases = document.querySelectorAll(".purchase-products > .bundle-product > .input > label");
																}
																
																if (elePurchases != null)
																{
																	for (var j = 0; j < elePurchases.length; j++)
																	{
																		if (elePurchases[j].textContent.trim() == name)
																		{
																			elePurchases[j].parentElement.parentElement.classList.add("bh_owned_dark");
																		}
																	}
																}
															}
														}
														
													}
													else
													{
														eleLabel.classList.add("bh_owned");
													}
													countOwned++;
												}
												else
												{
													console.log("App: not owned - " + app);
												}
											}
											else
											{
												console.log("App: not success - " + app);
											}
										}
									}
								}
							}
							
							console.log("Apps: owned - " + countOwned);
							
						} // End onload
					});
					
				});
			}
		}
		else if (url.indexOf("indiegala.com") > -1)
		{
			GM_addStyle(
				"   .bh_owned { background-color: rgba(125, 174, 45, 0.9) !important; } "			
				+ " .bundle-item-trading-cards-cont span { opacity: 0.7; } "
				+ " .span-title .title_game, .span-title .title_drm, .span-title .title_music { "
				+ "   line-height: 43px !important; margin: 10px 0px 10px 15px !important; "
				+ "   padding-left: 10px !important; border-radius: 3px !important; } "
			);
			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					var apps = [];
					
					var eleApps = document.querySelectorAll(".game-opened-switcher a[href*='store.steampowered.com/app/']"
						+ ", .in .in .in .game-steam-url[href*='store.steampowered.com/app/']"
						+ ", .game-opened-switcher a[href*='steamcommunity.com/app/']"
						+ ", .in .in .in .game-steam-url[href*='steamcommunity.com/app/']");
					console.log("Apps: " + eleApps.length);
					
					for (var i = 0; i < eleApps.length; i++)
					{
						var app = /[0-9]+/.exec(eleApps[i].getAttribute("href"));
						if (app != null)
						{
							apps.push(app[0]);
						}
					}
					
					apps = apps.filter(function(elem, index, self) 
					{
						return index == self.indexOf(elem);
					});
					
					var appAll = apps.join(",");
					
					GM_xmlhttpRequest(
					{
						method: "GET",
						headers: 
						{
							"Cache-Control": "max-age=0"
						},
						url: "https://store.steampowered.com/api/appuserdetails/?appids=" + appAll,
						onload: function(response) 
						{
							var dataRes = JSON.parse(response.responseText);
							
							var countOwned = 0;
							
							var elePrds = document.querySelectorAll(".bundle-item-link, .in .in .in .game-steam-url");
							var eleApps = document.querySelectorAll(
								".game-opened-switcher a[href*='store.steampowered.com/app/']:not(.game-steam-url)"
								+ ", .game-opened-switcher a[href*='steamcommunity.com/app/']:not(.game-steam-url)"
								+ ", .in .in .in .game-steam-url");
							for (var i = 0; i < eleApps.length; i++)
							{
								var appUrl = eleApps[i].getAttribute("href");
								if (appUrl.indexOf("/app/") > -1)
								{
									var app = /[0-9]+/.exec(appUrl);
									if (app != null)
									{
										if (typeof dataRes[app] !== "undefined")
										{
											if (dataRes[app].success)
											{
												if (dataRes[app].data.is_owned)
												{
													var eleLabel = elePrds[i].parentElement;
													eleLabel.classList.add("bh_owned");
													countOwned++;
												}
												else
												{
													console.log("App: not owned - " + app);
												}
											}
											else
											{
												console.log("App: not success - " + app);
											}
										}
									}
								}
							}
							
							console.log("Apps: owned - " + countOwned);
							
						} // End onload
					});
					
				});
			}
			
			// Change title
			
			var elesPrice = document.querySelectorAll(".bundle-claim-phrase");
			for (var i = elesPrice.length - 1; i > -1; i--)
			{
				var elePrice = elesPrice[i].querySelector("span");
				if (elePrice != null)
				{
					var price = elePrice.textContent.trim();
					if (price.indexOf("$") == 0)
					{
						document.title = price + " " + document.title;
						break;
					}
				}
			}
		}
		else if (url.indexOf("orlygift.com") > -1)
		{
			if (window.self != window.top)
				return;
			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);				
				
				divButton.addEventListener("click", function()
				{
					var apps = [];
					
					var eleApps = document.querySelectorAll("div[id^='game-detail-'] a.btn-primary[href^='http://store.steampowered.com/app/']");
					console.log("Apps: " + eleApps.length);
					
					for (var i = 0; i < eleApps.length; i++)
					{
						var app = /[0-9]+/.exec(eleApps[i].getAttribute("href"));
						if (app != null)
						{
							apps.push(app[0]);
						}
					}
					
					apps = apps.filter(function(elem, index, self) 
					{
						return index == self.indexOf(elem);
					});
					
					var appAll = apps.join(",");
					
					GM_xmlhttpRequest(
					{
						method: "GET",
						headers: 
						{
							"Cache-Control": "max-age=0"
						},
						url: "https://store.steampowered.com/api/appuserdetails/?appids=" + appAll,
						onload: function(response) 
						{
							var dataRes = JSON.parse(response.responseText);
							
							var countOwned = 0;
							
							var elePrds = document.querySelectorAll(".box-game");
							var eleApps = document.querySelectorAll("div[id^='game-detail-']");
							for (var i = 0; i < eleApps.length; i++)
							{
								var eleApp = eleApps[i].querySelector("a.btn-primary[href^='http://store.steampowered.com/app/']");
								if (eleApp != null)
								{
									var app = /[0-9]+/.exec(eleApp.getAttribute("href"));
									if (app != null)
									{
										if (typeof dataRes[app] !== "undefined")
										{
											if (dataRes[app].success)
											{
												if (dataRes[app].data.is_owned)
												{
													var eleLabel = elePrds[i];
													eleLabel.classList.add("bh_owned");
													countOwned++;
												}
												else
												{
													console.log("App: not owned - " + app);
												}
											}
											else
											{
												console.log("App: not success - " + app);
											}
										}
									}
								}
							}
							
							console.log("Apps: owned - " + countOwned);
							
						} // End onload
					});
					
				});
			}
		}
		else if (url.indexOf("cubicbundle.com") > -1)
		{
			GM_addStyle(
				"   .bh_owned { background-color: #91BA07 !important; } "
			);
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					markOwned(".price a[href*='store.steampowered.com/']", function(ele)
					{
						ele.parentElement.parentElement.parentElement.classList.add("bh_owned");
					});						
				});
			}
		}
		else if (url.indexOf("dailyindiegame.com") > -1)
		{			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);				
				
				divButton.addEventListener("click", function()
				{
					if (document.querySelectorAll(".DIG-content a[href*='store.steampowered.com/']").length > 0)
					{
						markOwned(".DIG-content a[href*='store.steampowered.com/']", function(ele)
						{
							ele.parentElement
								.parentElement.parentElement
								.parentElement.parentElement
								.classList.add("bh_owned");
						});
					}
					else
					{
						markOwned(".DIG2content a[href*='store.steampowered.com/']", function(ele)
						{
							ele.parentElement
								.parentElement
								.classList.add("bh_owned");
						});
					}
				});
			}
		}
		else if (url.indexOf("bundlekings.com") > -1)
		{			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					markOwned(".content-wrap a[href*='store.steampowered.com/']", function(ele)
					{
						ele.parentElement.parentElement.parentElement.classList.add("bh_owned");
					});					
				});
			}
		}
		else if (url.indexOf("otakumaker.com") > -1)
		{			
			GM_addStyle(
				"   .bh_owned { background-color: #91BA07 !important; } "
			);
			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					markOwned(".gantry-width-spacer a[href*='store.steampowered.com/']", function(ele)
					{
						ele.parentElement.parentElement.classList.add("bh_owned");
					});					
				});
			}
		}
		else if (url.indexOf("superduperbundle.com") > -1)
		{			
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					markOwned("#gameslist a[href*='store.steampowered.com/']", function(ele)
					{
						ele.parentElement.parentElement.classList.add("bh_owned");
					});					
				});
			}
		}
		else if (url.indexOf("gamebundle.com") > -1)
		{				
			GM_addStyle(
				"   .bh_owned { background-color: #A0CC41 !important; border-bottom: 45px solid rgba(233, 233, 233, 0.5);} "
			);
	
			// Add mark button
			{
				var divButton = document.createElement("div");
				divButton.classList.add("bh_button");
				divButton.id = "bh_markOwned";
				
				var eleA = document.createElement("a");
				eleA.setAttribute("onclick", "return false;");
				eleA.textContent = "Mark Owned";
				
				divButton.appendChild(eleA);
				document.body.appendChild(divButton);
				
				divButton.addEventListener("click", function()
				{
					markOwned(".activebundle_game_section_full a[href*='store.steampowered.com/']", function(ele)
					{
						ele.parentElement.classList.add("bh_owned");
					});					
				});
			}
		}
		else if (url.indexOf("humblebundle.com") > -1)
		{
			
			var elePrice = document.querySelector(".section-heading.bta");
			if (elePrice != null)
			{
				var price = elePrice.textContent.trim()
					.replace("Pay more than the average of ", "")
					.replace(" to also unlock!", "").trim();
				
				if (price.indexOf("$") == 0)
				{
					document.title = price + " " + document.title;
				}
			}
			
			var eleSold = document.querySelector(".heading-bundles-sold .mini-digit-holder");
			if (eleSold != null) 
			{
				var sold = eleSold.getAttribute("data-initial-value") || "";
				eleSold.parentElement.parentElement.setAttribute("title", sold);
			}
			
			if (url.indexOf("/downloads") > -1)
			{
			GM_addStyle(
				"   #steam-tab .redeem-instructions, #steam-tab .recommend-this-game { display: none; } "
			);
			
				setTimeout(function()
				{
					var elesKey = document.querySelectorAll(".sr-redeemed-bubble");
					for (var i = 0; i < elesKey.length; i++)
					{
						elesKey[i].addEventListener("click", function (e)
						{
							var ele = e.target;
							clickToSelect(ele);
						});
					}
				}, 3000);
			}
		}
		else if (url.indexOf("steamcompanion.com") > -1)
		{			
			GM_addStyle(
				"   .bh_owned.banner { margin-bottom: 5px !important; margin-top: 35px !important; "
				+ "   padding-bottom: 15px !important; padding-top: 15px !important; } "
				+ " .bh_owned.giveaway-links { opacity: 0.75; } "
			);
			
			markOwned("#hero a[href*='store.steampowered.com/']", function(ele)
			{
				ele.classList.add("bh_owned");
			});
			
			// Mark
			{
				var query = ".giveaway-links img[src^='https://steamcdn-a.akamaihd.net/steam/apps/']";
				var getLabelCallback = function(ele)
				{
					return ele.parentElement.parentElement.parentElement;
				};
				
				var apps = [];
				
				var eleApps = document.querySelectorAll(query);
				
				for (var i = 0; i < eleApps.length; i++)
				{
					var app = /[0-9]+/.exec(eleApps[i].getAttribute("src"));
					if (app != null)
					{
						apps.push(app[0]);
					}
				}
				
				apps = apps.filter(function(elem, index, self) 
				{
					return index == self.indexOf(elem);
				});
				
				console.log("Apps: " + apps.length);
				var appAll = apps.join(",");
				
				GM_xmlhttpRequest(
				{
					method: "GET",
					headers: 
					{
						"Cache-Control": "max-age=0"
					},
					url: "https://store.steampowered.com/api/appuserdetails/?appids=" + appAll,
					onload: function(response) 
					{
						var dataRes = JSON.parse(response.responseText);
						
						var countOwned = 0;
						
						var eleApps = document.querySelectorAll(query);
						for (var i = 0; i < eleApps.length; i++)
						{
							var appUrl = eleApps[i].getAttribute("src");
							if (appUrl.indexOf("https://steamcdn-a.akamaihd.net/steam/apps/") > -1)
							{
								var app = /[0-9]+/.exec(appUrl);
								if (app != null)
								{
									if (typeof dataRes[app] !== "undefined")
									{
										if (dataRes[app].success)
										{
											if (dataRes[app].data.is_owned)
											{
												var eleLabel = getLabelCallback(eleApps[i]);
												eleLabel.classList.add("bh_owned");
												countOwned++;
											}
											else
											{
												//console.log("App: not owned - http://store.steampowered.com/app/" + app + "/");
											}
										}
										else
										{
											//console.log("App: not success - https://steamdb.info/app/" + app + "/");
										}
									}
								}
							}
						}
						
						console.log("Apps: owned - " + countOwned);
						
					} // End onload
				});
			}
		}
		else if (url.indexOf("store.steampowered.com") > -1)
		{
			if (url.indexOf("/widget/") > -1) 
			{
				GM_addStyle(
					"   .bh_owned { background-color: transparent !important; } "
					+ " .bh_owned a { color: #71A034 !important; }"
				);
			
				markOwned(".main_text a[href*='store.steampowered.com/']", function(ele)
				{
					ele.parentElement.classList.add("bh_owned");
				});	
			}
			else if (url.indexOf("/app/") > -1) 
			{
				GM_addStyle(
					"   .bh_owned { "
					+ "   background-color: #6D8C1A !important; "
					+ "   padding: 0px 2px 0px 2px; "
					+ " } "
				);
				
				markOwned(".glance_details p > a[href*='store.steampowered.com/']"
					+ ", .game_area_dlc_bubble  a[href*='store.steampowered.com/']", 
					function(ele)
					{
						ele.classList.add("bh_owned");
					});	
			}
		}
		else if (url.indexOf("steamcommunity.com") > -1)
		{
			GM_addStyle(
				"   .bh_owned { background-color: #71A034 !important; "
				+ "   padding: 0px 2px 0px 2px;"
				+ " } "
			);
			
			markOwned(".blotter_gamepurchase_details a[href*='store.steampowered.com/']", function(ele)
			{
				ele.classList.add("bh_owned");
			});	
		}
		else if (url.indexOf("forums.steampowered.com") > -1)
		{
			GM_addStyle(
				"   .bh_owned { background-color: #71A034 !important; "
				+ "   padding: 0px 2px 0px 2px;"
				+ " } "
			);
			
			markOwned("div[id^='post_message'] a[href*='store.steampowered.com/']", function(ele)
			{
				ele.classList.add("bh_owned");
			});	
		}
		else if (url.indexOf("whosgamingnow.net") > -1)
		{
			GM_addStyle(
				"   .bh_owned { "
				+ "   padding: 0px 2px 0px 2px;"
				+ " } "
			);
			
			markOwned(".MessageList a[href*='store.steampowered.com/']", function(ele)
			{
				ele.classList.add("bh_owned");
			});	
		}
	}

	attachOnReady(main);
	
})();