SteamBadgeShowNext

在徽章页面查看下一级徽章图标

  1. // ==UserScript==
  2. // @name SteamBadgeShowNext
  3. // @namespace SteamBadgeShowNext@Byzod.user.js
  4. // @description 在徽章页面查看下一级徽章图标
  5. // @include /^https?:\/\/steamcommunity\.com\/(profiles|id)\/[^\/]+\/badges/
  6. // @version 2017-6-19
  7. // @grant GM_xmlhttpRequest
  8. // jshint esversion:6
  9. // ==/UserScript==
  10.  
  11.  
  12. // Steam CDN 主机
  13. const STEAM_CDN_HOST = "cdn.steamstatic.com.8686c.com";
  14. // LAZY_LOAD_DISTANCE 像素内的徽章图标才会加载
  15. const LAZY_LOAD_DISTANCE = 500;
  16. // Steam loading 图标
  17. const STEAM_LOADING_INDICATOR_URL = "http://steamcommunity-a.akamaihd.net/public/images/login/throbber.gif";
  18. // SCE 徽章页面前缀 ( + appId )
  19. const SCE_BADGES_URL_PREFIX = "http://www.steamcardexchange.net/index.php?gamepage-appid-";
  20. // 升到顶级的提示图标,PNG最佳
  21. // const HIGHEST_LEVEL_INDICATOR_URL = "https://steamcommunity-a.akamaihd.net/public/images/badges/generic/ValveEmployee_80.png";
  22. const HIGHEST_LEVEL_INDICATOR_URL = "";
  23.  
  24. // “差不多这个意思”版延迟加载类
  25. // targets: 需要lazy load的目标们
  26. // callback: 目标可见后的回调处理
  27. // 有一个参数target,为状态变为可见的目标
  28. function LazyLoader(targets, callback){
  29. // 监视对象Set
  30. this.Targets = targets ? new Set(targets) : new Set();
  31. // 回调函数
  32. this.Callback = (callback && typeof(callback)==="function") ? callback : null;
  33. // Lazy load距离 (像素)
  34. this.Distance = 0;
  35. // 初始化
  36. this.Init = function(win){
  37. win = win ? win : window;
  38. win.addEventListener("scroll", ScrollEventHandler, false);
  39. // 先调用一次,处理目前可视区域
  40. ScrollEventHandler(null);
  41. }
  42. let ScrollEventHandler = e => {
  43. // console.log('[LazyLoader] 处理Scroll event, 例遍 %o 个目标...', this.Targets.size) // DEBUG
  44. this.Targets.forEach(target => {
  45. // target 可见
  46. if(this.Callback
  47. && typeof(this.Callback)==="function"
  48. && LazyLoader.IsElementVisible(target, this.Distance)){
  49. // console.log('[LazyLoader] 正在处理 ' + target.innerText) // DEBUG
  50. // 回调
  51. this.Callback(target);
  52. // 从Set里移除已回调的target
  53. this.Targets.delete(target);
  54. }
  55. });
  56. }
  57. }
  58.  
  59. // Static function
  60. // 对象是否可见?(在视野distance像素内)
  61. LazyLoader.IsElementVisible = function IsElementVisible(elem, distance){
  62. if(!distance){
  63. distance = 0;
  64. }
  65. if(elem){
  66. let viewport = {
  67. width : window.innerWidth,
  68. height : window.innerHeight
  69. };
  70. // console.log("[IsElementVisible]: viewport %o" + viewport); // DEBUG
  71. let rect = elem.getBoundingClientRect();
  72. // console.log("[IsElementVisible]: rect %o" + rect); // DEBUG
  73. return (
  74. rect.left < viewport.width + distance
  75. && rect.right > 0 - distance
  76. && rect.top < viewport.height + distance
  77. && rect.bottom > 0 - distance
  78. );
  79. } else {
  80. throw("[IsElementVisible]: Element not exist");
  81. }
  82. }
  83.  
  84. // 插入样式
  85. let badgeImageCSS = document.createElement("style");
  86. badgeImageCSS.innerHTML = `
  87. .badge_next {
  88. position: absolute;
  89. overflow: hidden;
  90. width: 100px;
  91. top: -5px;
  92. left: -120px;
  93. bottom: -5px;
  94. padding: 5px;
  95. background: linear-gradient( to bottom, #232424 5%, #141414 95%);
  96. }
  97. .badge_next_loading {
  98. position: absolute;
  99. top: 50%;
  100. left: 50%;
  101. transform: translate(-50%, -50%);
  102. }
  103. .showcase-element {
  104. position: absolute;
  105. left: 0;
  106. right: 0;
  107. }
  108. .element-image {
  109. width: 80px;
  110. height: 80px;
  111. margin: 0px auto 0px auto;
  112. display: block;
  113. float: none;
  114. }
  115. .element-text {
  116. text-align: center;
  117. display: block;
  118. padding-top: 2px;
  119. }
  120. .element-experience {
  121. text-align: center;
  122. display: block;
  123. padding-top: 2px;
  124. color: #8F98A0;
  125. }
  126. .badge_highest_level{
  127. padding-top: 20px;
  128. box-shadow:0 1px 4px rgba(255, 255, 255, 0.3), 0 0 20px 5px rgba(255, 255, 255, 0.1);
  129. }
  130. `
  131. document.head.appendChild(badgeImageCSS);
  132.  
  133.  
  134. // 插入下一级徽章图
  135. let badgeRows = document.querySelectorAll(".badge_row");
  136. // console.log("[Steam+]: badgeRows: " + badgeRows.length); // DEBUG
  137. // 延迟附加徽章图
  138. let lazyLoader = new LazyLoader(
  139. badgeRows,
  140. row => {
  141. let appIdMatch = row.querySelector(".badge_row_overlay").href.match(/gamecards\/(\d+)(\/\?border=1)?/);
  142. let appId = appIdMatch ? appIdMatch[1] : 0;
  143. let foil = appIdMatch && appIdMatch.length > 2 && appIdMatch[2] ? true : false;
  144. let currentLevelInfo = row.querySelector(".badge_info_title + div");
  145. let currentLevel = currentLevelInfo ? parseInt(currentLevelInfo.innerText.match(/\d+/)[0]) : 0;
  146. // console.log(`[Steam+]: 处理 ${appId} (lv.${currentLevel}${foil?" foil":""})`); // DEBUG
  147. // 获取徽章数据
  148. if(appId !== 0){
  149. let badgeNextShowcase = document.createElement("a");
  150. badgeNextShowcase.className = "badge_next";
  151. badgeNextShowcase.href = SCE_BADGES_URL_PREFIX + appId;
  152. badgeNextShowcase.target = "_blank";
  153. badgeNextShowcase.innerHTML = '<img class="badge_next_loading" src="' + STEAM_LOADING_INDICATOR_URL + '" alt="Loading" />';
  154. GM_xmlhttpRequest({
  155. method: "GET",
  156. url: SCE_BADGES_URL_PREFIX + appId,
  157. onload: function (response) {
  158. let badgeData = PraseBadgeData(response.responseText);
  159. if(badgeData){
  160. let showcase = GetNextLevelBadgeShowcase(
  161. foil ? badgeData.FoilBadgeShowcases : badgeData.BadgeShowcases,
  162. currentLevel
  163. );
  164. // 停止转圈圈
  165. badgeNextShowcase.innerHTML = '';
  166. if(showcase){
  167. // 一切正常就显示徽章
  168. // console.log("[Steam+]: showcase (foil: %o): %o: ", foil, showcase); // DEBUG
  169. badgeNextShowcase.appendChild(showcase);
  170. } else {
  171. // 找不到showcase,已经最高级了
  172. // 随便恭喜一下
  173. badgeNextShowcase.classList.add("badge_highest_level");
  174. badgeNextShowcase.innerHTML = '<div class="showcase-element badge_info"><img class="element-image badge_info_image" src="' + HIGHEST_LEVEL_INDICATOR_URL + '" alt="Top Level"><span class="element-text badge_info_description badge_info_title">已升至顶级</span></div>';
  175. }
  176. }
  177. }
  178. });
  179. // 添加徽章展柜
  180. row.appendChild(
  181. badgeNextShowcase
  182. );
  183. }
  184. }
  185. );
  186. // 设置lazy loader参数
  187. // 可视范围外额外lazy load的范围 (像素)
  188. lazyLoader.Distance = LAZY_LOAD_DISTANCE;
  189. lazyLoader.Init(window);
  190.  
  191. // 处理SCE数据
  192. function PraseBadgeData(data){
  193. // SCE 文档碎片
  194. let sceFrag;
  195. // Badge 数据
  196. let badgeData = null;
  197. // 徽章选择器
  198. let badgeSelector = ".badge>.showcase-element";
  199. // 将 SCE 页面中的链接替换成支持 https 的域名
  200. data = data.replace(/https?:\/\/(community\.edgecast\.steamstatic\.com|steamcommunity-a\.akamaihd\.net|cdn\.steamcommunity\.com)\//g, "//steamcommunity-a.akamaihd.net/");
  201. data = data.replace(/https?:\/\/(cdn\.edgecast\.steamstatic\.com|steamcdn-a\.akamaihd\.net|cdn\.akamai\.steamstatic\.com)\//g, "//steamcdn-a.akamaihd.net/");
  202. // 先去除又臭又长的下拉菜单选项……
  203. data = data.replace(/<select[^]+<\/select>/,"");
  204. // 替换为Steam样式的class
  205. data = data.replace(/class="showcase-element"/g, 'class="showcase-element badge_info"');
  206. data = data.replace(/class="element-image"/g, 'class="element-image badge_info_image"');
  207. data = data.replace(/class="element-text"/g, 'class="element-text badge_info_description badge_info_title"');
  208. data = data.replace(/class="element-experience"/g, 'class="element-experience badge_info_description"');
  209. // 转为DOM
  210. sceFrag = document.createRange().createContextualFragment(data);
  211. // 普通徽章
  212. let badgeRows = ClosetParentNode(
  213. sceFrag.querySelector(".showcase-element-container.badge"), // 第一个.badge 父节点即为普通徽章box
  214. ".content-box"
  215. );
  216. // 闪亮徽章
  217. let foilBadgeRows = badgeRows ? badgeRows.nextSibling : null; // 好兄弟排排坐
  218. if(badgeRows && foilBadgeRows){
  219. badgeData = {
  220. BadgeShowcases : badgeRows.querySelectorAll(badgeSelector),
  221. FoilBadgeShowcases : foilBadgeRows.querySelectorAll(badgeSelector)
  222. };
  223. } else {
  224. badgeData = null;
  225. }
  226. // console.log("[Steam+]: badgeData: %o: ", badgeData); // DEBUG
  227. return badgeData;
  228. }
  229.  
  230. // 简单实现JQuery的closet
  231. function ClosetParentNode(elem, selector){
  232. let parent = null;
  233. while (elem) {
  234. parent = elem.parentElement;
  235. if (parent && parent.matches(selector)) {
  236. return parent;
  237. }
  238. elem = parent;
  239. }
  240. return null;
  241. }
  242.  
  243. // 获取下一等级徽章showcase
  244. function GetNextLevelBadgeShowcase(badgeShowcases, currentLevel){
  245. let showcase = null;
  246. const LEVEL_SELECTOR = ".element-experience";
  247. // {level} or {level low} - {level high} or {level low} - ???
  248. const LEVEL_REGEX = /Level (\d+)(?: - (\d+|\?\?\?))?/;
  249. for(let badge of badgeShowcases){
  250. let levelElem = badge.querySelector(LEVEL_SELECTOR);
  251. // 没有level也不要慌,showcase有很多空的,略过就是
  252. if(levelElem){
  253. let levelMatch = levelElem.innerText.match(LEVEL_REGEX);
  254. // 只有low的时候取low,有low和high时取high
  255. let level = levelMatch
  256. ? (levelMatch.length > 2 && levelMatch[2]
  257. ? levelMatch[2]
  258. : levelMatch[1]
  259. )
  260. : 0;
  261. // 转为Int
  262. level = isNaN(level)
  263. ? (level === "???" ? Infinity : 0) // 特别的,high=???代表该徽章可无限升级。给夏促大佬递女装
  264. : parseInt(level);
  265. if(level > currentLevel){
  266. // 找到下一级了,走起
  267. showcase = badge;
  268. break;
  269. }
  270. }
  271. }
  272. return showcase;
  273. }

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址