Xbox Cloud Gaming Localization

Set Xbox Cloud Gaming' game language to your browser's preferred language.

بۇ قوليازمىنى قاچىلاش؟
ئاپتورنىڭ تەۋسىيەلىگەن قوليازمىسى

سىز بەلكىم Xbox Cloud Gaming Region Unlocker نى ياقتۇرۇشىڭىز مۇمكىن.

بۇ قوليازمىنى قاچىلاش
  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. })();