Xbox Cloud Gaming Localization

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

Install this script?
Author's suggested script

You may also like Xbox Cloud Gaming Region Unlocker.

Install this script
// ==UserScript==
// @name                 Xbox Cloud Gaming Localization
// @name:zh-CN           Xbox Cloud Gaming 云游戏语言本地化
// @name:zh-TW           Xbox Cloud Gaming 雲端游戲語言本地化
// @namespace            http://tampermonkey.net/
// @version              1.3
// @description          Set Xbox Cloud Gaming' game language to your browser's preferred language.
// @description:zh-CN    将 Xbox Cloud Gaming 的游戏设置为浏览器首选语言
// @description:zh-TW    將 Xbox Cloud Gaming 的遊戲設定為瀏覽器首選語言
// @author               TGSAN
// @match                https://www.xbox.com/*/play*
// @icon                 
// @inject-into          page
// @run-at               document-start
// @grant                unsafeWindow
// ==/UserScript==

(function() {
    'use strict';

    let windowCtx = self.window;
    if (self.unsafeWindow) {
        console.log("[Xbox Cloud Gaming Localization] use unsafeWindow mode");
        windowCtx = self.unsafeWindow;
    } else {
        console.log("[Xbox Cloud Gaming Localization] use window mode (your userscript extensions not support unsafeWindow)");
    }

    // Your code here...
    let allFullLanguages = [];
    let allShortLanguages = [];
    let browserFirstLanguage = "";

    navigator.languages.forEach(language => {
        const reg = /^[a-z]{2}-[A-Z]{2}$/;
        const isFullLanguage = reg.test(language);
        if (isFullLanguage) {
            allFullLanguages.push(language);
        } else {
            allShortLanguages.push(language);
        }
    });

    if (allFullLanguages.length > 0) {
        browserFirstLanguage = allFullLanguages[0];
    }

    const originFetch = windowCtx.fetch;
    windowCtx.fetch = (...arg) => {
        // console.log('fetch arg', ...arg);

        let arg0 = arg[0];
        let url = "";
        let isRequest = false;
        switch (typeof arg0) {
            case "object":
                url = arg0.url;
                isRequest = true;
                break;
            case "string":
                url = arg0;
                break;
            default:
                break;
        }

        if (url.indexOf('/v5/sessions/cloud/play') > -1) {
            // Start Configuration
            return new Promise(async (resolve, reject) => {
                // eg: /en-US/play/launch/forza-horizon-4-standard-edition/9PNJXVCVWD4K
                const regex = /\/([a-zA-Z0-9]+)\/?/gm;
                let matches;
                let latestMatch;
                while ((matches = regex.exec(document.location.pathname)) !== null) {
                    if (matches.index === regex.lastIndex) {
                        regex.lastIndex++;
                    }
                    matches.forEach((match, groupIndex) => {
                        // console.log(`Found match, group ${groupIndex}: ${match}`);
                        latestMatch = match;
                    });
                }
                let selectedLanguage = browserFirstLanguage;
                if (latestMatch) {
                    let pid = latestMatch;
                    try {
                        let res = await fetch(
                            "https://catalog.gamepass.com/products?market=US&language=en-US&hydration=PCInline", {
                            "headers": {
                                "content-type": "application/json;charset=UTF-8",
                            },
                            "body": "{\"Products\":[\"" + pid + "\"]}",
                            "method": "POST",
                            "mode": "cors",
                            "credentials": "omit"
                        });
                        let jsonObj = await res.json();
                        let languageSupport = jsonObj["Products"][pid]["LanguageSupport"]
                        if (languageSupport) {
                            let supportedlanguages = Object.keys(languageSupport);
                            if (supportedlanguages.length > 0) {
                                let matched = false;
                                console.log("[Xbox Cloud Gaming Localization] Browser first language: " + browserFirstLanguage);
                                for (let fullLanguage of allFullLanguages) {
                                    if (matched === false) {
                                        if (supportedlanguages.indexOf(fullLanguage) > -1) {
                                            matched = true;
                                            selectedLanguage = fullLanguage;
                                            console.log("[Xbox Cloud Gaming Localization] Game support: " + fullLanguage + " (Browser language: " + browserFirstLanguage + ")");
                                            break;
                                        } else {
                                            console.log("[Xbox Cloud Gaming Localization] Game not support: " + fullLanguage);
                                        }
                                    }
                                }
                                if (matched === false) {
                                    console.log("[Xbox Cloud Gaming Localization] Start fuzzy matching");
                                    for (let shortLanguage of allShortLanguages) {
                                        supportedlanguages.forEach(language => {
                                            if (matched === false) {
                                                if (language.startsWith(shortLanguage)) {
                                                    if (matched === false) {
                                                        matched = true;
                                                        selectedLanguage = language;
                                                        console.log("[Xbox Cloud Gaming Localization] Game support: " + fullLanguage + " (Browser language: " + browserFirstLanguage + ")");
                                                    }
                                                }
                                            }
                                        });
                                    }
                                }
                            } else {
                                console.warn("[Xbox Cloud Gaming Localization] Game no supported languages list.");
                            }
                        }
                    } catch (err) {
                        console.warn("[Xbox Cloud Gaming Localization] fallback to first browser language")
                    }
                }
                if (isRequest && arg0.method == "POST") {
                    arg0.json().then(json => {
                        if (selectedLanguage != "") {
                            console.log("Selected: " + selectedLanguage);
                            json["settings"]["locale"] = selectedLanguage;
                        } else {
                            console.log("Use default language");
                        }
                        let body = JSON.stringify(json);
                        arg[0] = new Request(url, {
                            method: arg0.method,
                            headers: arg0.headers,
                            body: body,
                            mode: arg0.mode,
                            credentials: arg0.credentials,
                            cache: arg0.cache,
                            redirect: arg0.redirect,
                            referrer: arg0.referrer,
                            integrity: arg0.integrity
                        });
                        originFetch(...arg).then(res => {
                            resolve(res);
                        }).catch(err => {
                            reject(err);
                        });
                    });
                } else {
                    console.error("[Xbox Cloud Gaming Localization] [ERROR] Not a request.");
                    return originFetch(...arg);
                }
            });
        } else if (url.indexOf('/v2/login/user') > -1) {
            // Area Select
            return new Promise((resolve, reject) => {
                originFetch(...arg).then(res => {
                    res.json().then(json => {
                        // console.error(json);
                        json["offeringSettings"]["allowRegionSelection"] = true;
                        let body = JSON.stringify(json);
                        let newRes = new Response(body, {
                            status: res.status,
                            statusText: res.statusText,
                            headers: res.headers
                        })
                        resolve(newRes);
                    }).catch(err => {
                        reject(err);
                    });
                }).catch(err => {
                    reject(err);
                });
            });
        } else {
            return originFetch(...arg);
        }

    }
})();