// ==UserScript== // @name CS.RIN.RU Enhanced (External) // @name:fr CS.RIN.RU Amélioré (Externe) // @name:pt CS.RIN.RU Melhorado (Externo) // @name:tr Genişletilmiş CS.RIN.RU (Ek) // @namespace https://github.com/Altansar69/CS.RIN.RU-Enhanced-external // @version 1.1.7 // @description Everything that concerns CS.RIN.RU - Steam Underground Community but does not act on the site. // @description:fr Tout ce qui concerne CS.RIN.RU - Steam Underground Community mais qui n'agit pas sur le site. // @description:pt W.I.P. // @description:tr CS.RIN.RU sitesini ilgilendiren her şey - Steam Underground Topluluğu ancak sitede faaliyet göstermemektedir. // @author CS.RIN.RU community // @match *://store.steampowered.com/app/* // @match *://steamdb.info/app/* // @match *://www.pcgamingwiki.com/wiki/* // @icon https://i.ibb.co/p1k6cq6/image.png // @grant GM_xmlhttpRequest // @homepageURL https://github.com/Altansar69/CS.RIN.RU-Enhanced-external // @supportURL https://cs.rin.ru/forum/viewtopic.php?f=14&t=75717 // @updateURL https://raw.githubusercontent.com/Altansar69/CS.RIN.RU-Enhanced-external/master/CS-RIN-RU-ENHANCED-external.user.js // @downloadURL https://raw.githubusercontent.com/Altansar69/CS.RIN.RU-Enhanced-external/master/CS-RIN-RU-ENHANCED-external.user.js // ==/UserScript== let defaultTag; //defaultTag = "Cracked (by default)" function addRinLinkToSteam() { if (!document.location.origin.match("store.steampowered.com")) return; const page = "steam" const rinButton = addRinButton(page); const dlcPage = document.querySelector("div.game_area_bubble.game_area_dlc_bubble"); const pageUrl = dlcPage?.querySelector("div > p > a")?.href ?? document.location.pathname; const appName = document.querySelector("#appHubAppName").textContent; const regex = /\/app\/(\d+)\//; const appId = pageUrl.match(regex)[1]; const developer = encodeURIComponent(document.querySelector("#developers_list").firstElementChild.textContent); updatePage(appId, appName, developer, rinButton, page); } addRinLinkToSteam(); function addRinLinkToSteamDB() { if (!document.location.origin.match("steamdb.info")) return; const page = "steamdb" const rinButton = addRinButton(page); const appName = document.querySelector("h1").textContent; const firstEntry = document.querySelector('.span3'); const appId = firstEntry.nextElementSibling.textContent; const developer = encodeURIComponent(firstEntry.parentElement .nextElementSibling.nextElementSibling .firstElementChild.nextElementSibling.textContent.replace(/\n/g, "")) updatePage(appId, appName, developer, rinButton, page) } addRinLinkToSteamDB(); function addRinLinkToPCGW() { if (!document.location.origin.match("pcgamingwiki.com")) return; const page = "PCGW" const rinButton = addRinButton(page); const pageUrl = document.querySelector('.infobox-steamdb > a').getAttribute("href"); const appName = document.querySelector("h1").textContent; const regex = /\/app\/(\d+)\//; const appId = pageUrl.match(regex)[1]; const developer = encodeURIComponent(document.querySelector(".template-infobox-info").textContent); updatePage(appId, appName, developer, rinButton, page); } addRinLinkToPCGW(); function addRinButton(page) { const rinImage = ""; const rinButton = document.createElement("a"); rinButton.className = "btnv6_blue_hoverfade btn_medium"; rinButton.style.marginLeft = "0.280em"; const spanElement = document.createElement("span"); spanElement.dataset.tooltipText = "View on CS.RIN.RU"; const imgElement = document.createElement("img"); imgElement.className = "ico16"; imgElement.setAttribute("src", rinImage); spanElement.appendChild(imgElement); // Add text for RIN button on SteamDB if (page === "steamdb") { imgElement.style.height = "16px"; imgElement.style.width = "16px"; spanElement.append(" CS.RIN.RU"); } // Make sure the button has the same size as the other buttons if (page === "PCGW") { rinButton.className = "svg-icon template-infobox-icon"; } rinButton.append(spanElement); const siteSelectors = { "steam": () => document.querySelector('.apphub_OtherSiteInfo'), "steamdb": () => document.querySelectorAll('.app-links')[1], "PCGW": () => document.querySelectorAll('.template-infobox-icons')[0] }; const otherSiteInfo = (siteSelectors[page] || (() => null))(); otherSiteInfo.insertBefore(rinButton, otherSiteInfo.firstChild); return rinButton; } function updatePage(appId, appName, developer, rinButton, page) { getRinTopic(appId, appName, developer, function (url, tags) { // Adds the cs.rin topic "href" attribute to the button addRinUrl(rinButton, url); addRinTags(tags, page); }); } function getRinTopic(appId, appName, developer, callback) { const rinSearchUrl = `https://cs.rin.ru/forum/search.php?keywords=${appId}&fid%5B%5D=10&sr=topics&sf=firstpost`; console.log(rinSearchUrl); GM_xmlhttpRequest({ method: "GET", url: rinSearchUrl, onload: function (response) { const doc = new DOMParser().parseFromString(response.responseText, "text/html"); const topicSelectors = doc.querySelectorAll(".titles:not(:first-child), .topictitle"); if (topicSelectors.length > 1) { getRinTopicAdvanced(appId, appName, developer, callback); } else { processResponse(appName, response.responseText, callback, function () { getRinTopic(appId, "", developer, callback); // Retry getRinTopic if search fails }); } } }); } function getRinTopicAdvanced(appId, appName, developer, callback) { const rinSearchUrl = `https://cs.rin.ru/forum/search.php?keywords=${appId}+${developer}&fid%5B%5D=10&sr=topics&sf=firstpost`; console.log(rinSearchUrl); GM_xmlhttpRequest({ method: "GET", url: rinSearchUrl, onload: function (response) { processResponse(appName, response.responseText, callback, function () { getRinTopicAdvanced(appId, appName, developer, callback); // Retry getRinTopicAdvanced if search fails }); } }); } let retryScheduled = false; // Flag to track if a retry is scheduled function processResponse(appName, responseText, callback, retryFunction) { if (retryScheduled) return; // If a retry is scheduled, don't do anything const doc = new DOMParser().parseFromString(responseText, "text/html"); // Check if search was successful const checkElement = doc.querySelector("#wrapcentre > form > table.tablebg > tbody > tr:nth-child(1) > th:nth-child(1)"); if (!checkElement) { if (retryFunction) { retryScheduled = true; // Set the flag to true to block further retries setTimeout(() => { retryScheduled = false; // Reset the flag when the retry function is called retryFunction(); // Call the retryFunction, whichever function called this function }, 200); } return; } // Get all topics const topics = doc.querySelectorAll(".titles:not(:first-child), .topictitle"); let topicSelector = null; for (let potentialTopic of topics) { if (potentialTopic.textContent.includes(appName)) { topicSelector = potentialTopic; break; } } // Default to first topic if (!topicSelector) { topicSelector = doc.querySelector(".titles:not(:first-child), .topictitle"); } const rinURL = topicSelector ? topicSelector.getAttribute("href") : "posting.php?mode=post&f=10"; const redirectUrl = "https://cs.rin.ru/forum/" + rinURL.split("&hilit")[0]; const tags = topicSelector ? topicSelector.text.match(/(? document.getElementById("appHubAppName"), "steamdb": () => document.querySelector('[itemprop="name"]'), "PCGW": () => document.getElementsByClassName("article-title")[0] }; // const titleElem = (tagFunctions[page] || (() => null))(tags); const titleLocation = (titleLocations[page] || (() => null))(); const titleElem = appendRinTags(tags, titleLocation); // Add colours to the tags const bracketRegex = /[\[\]]/g; let newContent = titleElem.textContent; tags.forEach(tag => { const color = colorize(tag, titleElem); const tagSpan = `[${tag.replace(bracketRegex, "")}]`; newContent = newContent.replace(tag, tagSpan); }); titleElem.innerHTML = newContent; } function hexToRgb(hex) { return [parseInt(hex.substring(0, 2), 16), parseInt(hex.substring(2, 4), 16), parseInt(hex.substring(4, 6), 16)]; } function colorize(str, parentElem) { let lstr = str.toLowerCase(); let hash = 0; for (let i = 0; i < lstr.length; i++) { hash = lstr.charCodeAt(i) + ((hash << 5) - hash); } let color = Math.floor(Math.abs((Math.sin(hash) * 10000) % 1 * 16777216)).toString(16); let rgb = hexToRgb(color); while (!getComputedStyle(parentElem).getPropertyValue("background-color").match(/^rgb\((\d+),\s*(\d+),\s*(\d+)\)$/)) { parentElem = parentElem.parentElement; } let bgColour = getComputedStyle(parentElem).getPropertyValue("background-color"); let matches = bgColour.match(/^rgb\((\d+),\s*(\d+),\s*(\d+)\)$/); const bgRgb = [parseInt(matches[1]), parseInt(matches[2]), parseInt(matches[3])]; while (Math.abs(rgb[0] + rgb[1] + rgb[2] - (bgRgb[0] + bgRgb[1] + bgRgb[2])) < 300) { hash = (hash << 5) - hash; color = Math.floor(Math.abs((Math.sin(hash) * 10000) % 1 * 16777216)).toString(16); rgb = hexToRgb(color); } return '#' + color.padStart(6, '0'); }