// ==UserScript== // @name SteamClientInfoBoxManage // @namespace http://tampermonkey.net/ // @version 1.2 // @description 隐藏"Steam个人游戏列表"页的客户端和下载信息,附加游戏列表排序功能 // @author 澪羽 // @match https://steamcommunity.com/profiles/*/games/* // @match https://steamcommunity.com/id/*/games/* // @icon  // @grant GM_setValue // @grant GM_getValue // @downloadURL none // ==/UserScript== const iGameContainer = "games_list_row_container" // all games container div const iGameList = "games_list_rows" // all games div const cActiveBackground = "gameListItemActive" // 活动状态背景 const iClientInfo = "clientConnBlock" // 客户端计算机相关信息 const cDiskStatus = "clientConnItemBlock" // 磁盘信息和下载图标 const cAppInstalled = "gameListItemInstalled" // 客户端已安装 var gDefaultSortApps = getApps() var gSortApps = new Array() var gDiskStatus = document.querySelectorAll('.' + cDiskStatus) var gmSettings = { "SortTypeIndex": 1, "DisplayClientInfo": true, "DisplayActiveBackground": true, "DisplayDiskStatus": true, "DisplayDownloadIcon": true } function getApps() { let aNodeList = document.getElementById(iGameList).childNodes let ret = Array.prototype.slice.call(aNodeList, 0) return ret.slice(1, ret.length) } function getAppName(DOM) { return DOM.childNodes[2].childNodes[1].childNodes[1].childNodes[1].innerText } function getTimeRecord(DOM) { return DOM.childNodes[2].childNodes[1].childNodes[1].childNodes[3].innerText } function getPlaytime(s) { var ret = 0.00 let aString = s.split(" ") var sNumStr for (let i = 0; i < aString.length; i++) { if (!!(aString[i].match(/\d/g))) { sNumStr = aString[i] } } if (sNumStr.includes(",")) { let aLeft = sNumStr.split(",") let sLeft = aLeft.join('') ret = parseFloat(sLeft) } else { ret = parseFloat(sNumStr) } return ret } function rerenderAppList() { for (let i = 0; i < gSortApps.length; i++) { document.getElementById(iGameList).appendChild(gSortApps[i].DOM) } return } function sortAndRerender(sFieldName, sSortDirection) { switch (sSortDirection) { case "DESC": gSortApps.sort(compareDESC(sFieldName)) break case "ASC": gSortApps.sort(compareASC(sFieldName)) break } rerenderAppList() return } function defaultSortAndRerender() { for (let i = 0; i < gDefaultSortApps.length; i++) { document.getElementById(iGameList).appendChild(gDefaultSortApps[i]) } return } function getFieldName(s) { return (s.split("_"))[0] } function getSortDirection(s) { return (s.split("_"))[1] } function initSortTypeSelect(dParent) { var dSortTypeArea = document.createElement('div') let lSortType = document.createElement('label') lSortType.innerText = "排序方式 " let sSortType = document.createElement('select') const sDefaultSort = "Default_Sort" const sPlaytimeDESC = "Playtime_DESC" const sPlaytimeASC = "Playtime_ASC" const sNameDESC = "Name_DESC" const sNameASC = "Name_ASC" sSortType.options.add(new Option("Default Sort(默认排序)", sDefaultSort)) sSortType.options.add(new Option("Playtime DESC(时间降序)", sPlaytimeDESC)) sSortType.options.add(new Option("Playtime ASC(时间升序)", sPlaytimeASC)) sSortType.options.add(new Option("Name DESC(名称降序)", sNameDESC)) sSortType.options.add(new Option("Name ASC(名称升序)", sNameASC)) sSortType.setAttribute('id', "SortType") sSortType.onchange = function () { let self = document.getElementById("SortType") let sFieldName = getFieldName(self.value) let sSortDirection = getSortDirection(self.value) switch (self.value) { case sDefaultSort: defaultSortAndRerender() break case sPlaytimeDESC: case sPlaytimeASC: case sNameDESC: case sNameASC: sortAndRerender(sFieldName, sSortDirection) break default: console.log("unknown SortType:" + self.value) return } GM_setValue("SortTypeIndex", self.selectedIndex) } sSortType.selectedIndex = gmSettings.SortTypeIndex dSortTypeArea.appendChild(lSortType) dSortTypeArea.appendChild(sSortType) dParent.appendChild(dSortTypeArea) return } function displayClientInfo(bHide) { GM_setValue("DisplayClientInfo", bHide) if (bHide) { document.getElementById(iClientInfo).style.display = 'none' } else { document.getElementById(iClientInfo).style.display = 'block' } return } function displayActiveBackground(bHide) { GM_setValue("DisplayActiveBackground", bHide) if (bHide) { for (let i = 0; i < gSortApps.length; i++) { if (gSortApps[i].ActiveBackgroundFlag) { let sAppClass = gSortApps[i].DOM.getAttribute('class') sAppClass = sAppClass.replace(cActiveBackground, "") gSortApps[i].DOM.setAttribute('class', sAppClass) } } } else { for (let i = 0; i < gSortApps.length; i++) { if (gSortApps[i].ActiveBackgroundFlag) { let sAppClass = gSortApps[i].DOM.getAttribute('class') sAppClass = sAppClass.concat(" " + cActiveBackground) gSortApps[i].DOM.setAttribute('class', sAppClass) } } } return } function displayDiskStatus(bHide) { GM_setValue("DisplayDiskStatus", bHide) if (bHide) { for (let i = 0; i < gDiskStatus.length; i++) { gDiskStatus[i].childNodes[1].style.display = 'none' } } else { for (let i = 0; i < gDiskStatus.length; i++) { gDiskStatus[i].childNodes[1].style.display = 'block' } } return } function displayDownloadIcon(bHide) { GM_setValue("DisplayDownloadIcon", bHide) if (bHide) { for (let i = 0; i < gDiskStatus.length; i++) { let aDownloadInfo = gDiskStatus[i].childNodes if ((aDownloadInfo.length) >= 5) { aDownloadInfo[3].style.display = 'none' } } } else { for (let i = 0; i < gDiskStatus.length; i++) { let aDownloadInfo = gDiskStatus[i].childNodes if ((aDownloadInfo.length) >= 5) { aDownloadInfo[3].style.display = 'block' } } } return } function initSettingCheckbox(dParent) { var dCheckboxArea = document.createElement('div') let dClientInfo = document.createElement('div') let cClientInfo = document.createElement('input') cClientInfo.setAttribute('type', 'checkbox') cClientInfo.setAttribute('id', "ClientInfo") cClientInfo.onclick = function () { let self = document.getElementById("ClientInfo") displayClientInfo(self.checked) } cClientInfo.checked = gmSettings.DisplayClientInfo dClientInfo.appendChild(cClientInfo) let tClientInfo = document.createTextNode("Hide Client Info(隐藏客户端信息框)") dClientInfo.appendChild(tClientInfo) let dActiveBackground = document.createElement('div') let cActiveBackground = document.createElement('input') cActiveBackground.setAttribute('type', 'checkbox') cActiveBackground.setAttribute('id', "ActiveBackground") cActiveBackground.onclick = function () { let self = document.getElementById("ActiveBackground") displayActiveBackground(self.checked) } cActiveBackground.checked = gmSettings.DisplayActiveBackground dActiveBackground.appendChild(cActiveBackground) let tActiveBackground = document.createTextNode("Hide Active Background(隐藏活动中游戏的背景颜色条)") dActiveBackground.appendChild(tActiveBackground) let dDiskStatus = document.createElement('div') let cDiskStatus = document.createElement('input') cDiskStatus.setAttribute('type', 'checkbox') cDiskStatus.setAttribute('id', "DiskStatus") cDiskStatus.onclick = function () { let self = document.getElementById("DiskStatus") displayDiskStatus(self.checked) } cDiskStatus.checked = gmSettings.DisplayDiskStatus dDiskStatus.appendChild(cDiskStatus) let tDiskStatus = document.createTextNode("Hide Disk Status(隐藏磁盘占用信息)") dDiskStatus.appendChild(tDiskStatus) let dDownloadIcon = document.createElement('div') let cDownloadIcon = document.createElement('input') cDownloadIcon.setAttribute('type', 'checkbox') cDownloadIcon.setAttribute('id', "DownloadIcon") cDownloadIcon.onclick = function () { let self = document.getElementById("DownloadIcon") displayDownloadIcon(self.checked) } cDownloadIcon.checked = gmSettings.DisplayDownloadIcon dDownloadIcon.appendChild(cDownloadIcon) let tDownloadIcon = document.createTextNode("Hide Download Icon(隐藏下载图标)") dDownloadIcon.appendChild(tDownloadIcon) dCheckboxArea.appendChild(dClientInfo) dCheckboxArea.appendChild(dActiveBackground) dCheckboxArea.appendChild(dDiskStatus) dCheckboxArea.appendChild(dDownloadIcon) dParent.appendChild(dCheckboxArea) return } function initSettingArea() { // root dom var dRootDiv = document.createElement('div') // todo:展开/收起功能设置区域+(文字描述or分隔线) var dHeader = document.createElement('div') initSortTypeSelect(dHeader) // 选项 var dSettingArea = document.createElement('div') initSettingCheckbox(dSettingArea) dRootDiv.appendChild(dHeader) dRootDiv.appendChild(dSettingArea) let dGameContainer = document.getElementById(iGameContainer) let dParent = dGameContainer.parentNode dParent.insertBefore(dRootDiv, dGameContainer) return } function loadStorageConfig() { if (!(typeof (GM_getValue("SortTypeIndex")) == 'undefined')) { gmSettings.SortTypeIndex = GM_getValue("SortTypeIndex") } if (!(typeof (GM_getValue("DisplayClientInfo")) == 'undefined')) { gmSettings.DisplayClientInfo = GM_getValue("DisplayClientInfo") } displayClientInfo(gmSettings.DisplayClientInfo) if (!(typeof (GM_getValue("DisplayActiveBackground")) == 'undefined')) { gmSettings.DisplayActiveBackground = GM_getValue("DisplayActiveBackground") } if (!(typeof (GM_getValue("DisplayDiskStatus")) == 'undefined')) { gmSettings.DisplayDiskStatus = GM_getValue("DisplayDiskStatus") } displayDiskStatus(gmSettings.DisplayDiskStatus) if (!(typeof (GM_getValue("DisplayDownloadIcon")) == 'undefined')) { gmSettings.DisplayDownloadIcon = GM_getValue("DisplayDownloadIcon") } displayDownloadIcon(gmSettings.DisplayDownloadIcon) return } (function () { 'use strict'; // Your code here... loadStorageConfig() initSettingArea() for (let i = 0; i < gDefaultSortApps.length; i++) { let dNode = { "ActiveBackgroundFlag": false, "Playtime": 0.00 } // ActiveBackground初始化 let sAppClass = gDefaultSortApps[i].getAttribute('class') if (sAppClass.indexOf(cActiveBackground) != -1) { dNode.ActiveBackgroundFlag = true if (gmSettings.DisplayActiveBackground) { sAppClass = sAppClass.replace(cActiveBackground, "") gDefaultSortApps[i].setAttribute('class', sAppClass) } } dNode.DOM = gDefaultSortApps[i] dNode.Name = getAppName(gDefaultSortApps[i]) let sTimeRecord = getTimeRecord(gDefaultSortApps[i]) if (!(sTimeRecord == "")) { dNode.Playtime = getPlaytime(sTimeRecord) } gSortApps.push(dNode) } if (gmSettings.SortTypeIndex > 0) { let sSortType = document.getElementById("SortType") let sFieldName = getFieldName(sSortType.options[gmSettings.SortTypeIndex].value) let sSortDirection = getSortDirection(sSortType.options[gmSettings.SortTypeIndex].value) sortAndRerender(sFieldName, sSortDirection) } return })(); function compareDESC(sFieldName) { return function (a, b) { let v1 = (a[sFieldName]) let v2 = (b[sFieldName]) if (v1 < v2) { return 1 } if (v1 > v2) { return -1 } return 0 } } function compareASC(sFieldName) { return function (a, b) { let v1 = a[sFieldName] let v2 = b[sFieldName] if (v1 < v2) { return -1 } if (v1 > v2) { return 1 } return 0 } }