Xbox Cloud Gaming 云游戏语言本地化

将 Xbox Cloud Gaming 的游戏设置为浏览器首选语言

  1. // ==UserScript==
  2. // @name Xbox Cloud Gaming Localization
  3. // @name:zh-CN Xbox Cloud Gaming 云游戏语言本地化
  4. // @name:zh-TW Xbox Cloud Gaming 雲端游戲語言本地化
  5. // @namespace http://tampermonkey.net/
  6. // @version 1.3
  7. // @description Set Xbox Cloud Gaming' game language to your browser's preferred language.
  8. // @description:zh-CN 将 Xbox Cloud Gaming 的游戏设置为浏览器首选语言
  9. // @description:zh-TW 將 Xbox Cloud Gaming 的遊戲設定為瀏覽器首選語言
  10. // @author TGSAN
  11. // @match https://www.xbox.com/*/play*
  12. // @icon 
  13. // @inject-into page
  14. // @run-at document-start
  15. // @grant unsafeWindow
  16. // ==/UserScript==
  17.  
  18. (function() {
  19. 'use strict';
  20.  
  21. let windowCtx = self.window;
  22. if (self.unsafeWindow) {
  23. console.log("[Xbox Cloud Gaming Localization] use unsafeWindow mode");
  24. windowCtx = self.unsafeWindow;
  25. } else {
  26. console.log("[Xbox Cloud Gaming Localization] use window mode (your userscript extensions not support unsafeWindow)");
  27. }
  28.  
  29. // Your code here...
  30. let allFullLanguages = [];
  31. let allShortLanguages = [];
  32. let browserFirstLanguage = "";
  33.  
  34. navigator.languages.forEach(language => {
  35. const reg = /^[a-z]{2}-[A-Z]{2}$/;
  36. const isFullLanguage = reg.test(language);
  37. if (isFullLanguage) {
  38. allFullLanguages.push(language);
  39. } else {
  40. allShortLanguages.push(language);
  41. }
  42. });
  43.  
  44. if (allFullLanguages.length > 0) {
  45. browserFirstLanguage = allFullLanguages[0];
  46. }
  47.  
  48. const originFetch = windowCtx.fetch;
  49. windowCtx.fetch = (...arg) => {
  50. // console.log('fetch arg', ...arg);
  51.  
  52. let arg0 = arg[0];
  53. let url = "";
  54. let isRequest = false;
  55. switch (typeof arg0) {
  56. case "object":
  57. url = arg0.url;
  58. isRequest = true;
  59. break;
  60. case "string":
  61. url = arg0;
  62. break;
  63. default:
  64. break;
  65. }
  66.  
  67. if (url.indexOf('/v5/sessions/cloud/play') > -1) {
  68. // Start Configuration
  69. return new Promise(async (resolve, reject) => {
  70. // eg: /en-US/play/launch/forza-horizon-4-standard-edition/9PNJXVCVWD4K
  71. const regex = /\/([a-zA-Z0-9]+)\/?/gm;
  72. let matches;
  73. let latestMatch;
  74. while ((matches = regex.exec(document.location.pathname)) !== null) {
  75. if (matches.index === regex.lastIndex) {
  76. regex.lastIndex++;
  77. }
  78. matches.forEach((match, groupIndex) => {
  79. // console.log(`Found match, group ${groupIndex}: ${match}`);
  80. latestMatch = match;
  81. });
  82. }
  83. let selectedLanguage = browserFirstLanguage;
  84. if (latestMatch) {
  85. let pid = latestMatch;
  86. try {
  87. let res = await fetch(
  88. "https://catalog.gamepass.com/products?market=US&language=en-US&hydration=PCInline", {
  89. "headers": {
  90. "content-type": "application/json;charset=UTF-8",
  91. },
  92. "body": "{\"Products\":[\"" + pid + "\"]}",
  93. "method": "POST",
  94. "mode": "cors",
  95. "credentials": "omit"
  96. });
  97. let jsonObj = await res.json();
  98. let languageSupport = jsonObj["Products"][pid]["LanguageSupport"]
  99. if (languageSupport) {
  100. let supportedlanguages = Object.keys(languageSupport);
  101. if (supportedlanguages.length > 0) {
  102. let matched = false;
  103. console.log("[Xbox Cloud Gaming Localization] Browser first language: " + browserFirstLanguage);
  104. for (let fullLanguage of allFullLanguages) {
  105. if (matched === false) {
  106. if (supportedlanguages.indexOf(fullLanguage) > -1) {
  107. matched = true;
  108. selectedLanguage = fullLanguage;
  109. console.log("[Xbox Cloud Gaming Localization] Game support: " + fullLanguage + " (Browser language: " + browserFirstLanguage + ")");
  110. break;
  111. } else {
  112. console.log("[Xbox Cloud Gaming Localization] Game not support: " + fullLanguage);
  113. }
  114. }
  115. }
  116. if (matched === false) {
  117. console.log("[Xbox Cloud Gaming Localization] Start fuzzy matching");
  118. for (let shortLanguage of allShortLanguages) {
  119. supportedlanguages.forEach(language => {
  120. if (matched === false) {
  121. if (language.startsWith(shortLanguage)) {
  122. if (matched === false) {
  123. matched = true;
  124. selectedLanguage = language;
  125. console.log("[Xbox Cloud Gaming Localization] Game support: " + fullLanguage + " (Browser language: " + browserFirstLanguage + ")");
  126. }
  127. }
  128. }
  129. });
  130. }
  131. }
  132. } else {
  133. console.warn("[Xbox Cloud Gaming Localization] Game no supported languages list.");
  134. }
  135. }
  136. } catch (err) {
  137. console.warn("[Xbox Cloud Gaming Localization] fallback to first browser language")
  138. }
  139. }
  140. if (isRequest && arg0.method == "POST") {
  141. arg0.json().then(json => {
  142. if (selectedLanguage != "") {
  143. console.log("Selected: " + selectedLanguage);
  144. json["settings"]["locale"] = selectedLanguage;
  145. } else {
  146. console.log("Use default language");
  147. }
  148. let body = JSON.stringify(json);
  149. arg[0] = new Request(url, {
  150. method: arg0.method,
  151. headers: arg0.headers,
  152. body: body,
  153. mode: arg0.mode,
  154. credentials: arg0.credentials,
  155. cache: arg0.cache,
  156. redirect: arg0.redirect,
  157. referrer: arg0.referrer,
  158. integrity: arg0.integrity
  159. });
  160. originFetch(...arg).then(res => {
  161. resolve(res);
  162. }).catch(err => {
  163. reject(err);
  164. });
  165. });
  166. } else {
  167. console.error("[Xbox Cloud Gaming Localization] [ERROR] Not a request.");
  168. return originFetch(...arg);
  169. }
  170. });
  171. } else if (url.indexOf('/v2/login/user') > -1) {
  172. // Area Select
  173. return new Promise((resolve, reject) => {
  174. originFetch(...arg).then(res => {
  175. res.json().then(json => {
  176. // console.error(json);
  177. json["offeringSettings"]["allowRegionSelection"] = true;
  178. let body = JSON.stringify(json);
  179. let newRes = new Response(body, {
  180. status: res.status,
  181. statusText: res.statusText,
  182. headers: res.headers
  183. })
  184. resolve(newRes);
  185. }).catch(err => {
  186. reject(err);
  187. });
  188. }).catch(err => {
  189. reject(err);
  190. });
  191. });
  192. } else {
  193. return originFetch(...arg);
  194. }
  195.  
  196. }
  197. })();

QingJ © 2025

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