Xbox Cloud Gaming Localization

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

Tendrás que instalar una extensión para tu navegador como Tampermonkey, Greasemonkey o Violentmonkey si quieres utilizar este script.

You will need to install an extension such as Tampermonkey to install this script.

Tendrás que instalar una extensión como Tampermonkey o Violentmonkey para instalar este script.

Necesitarás instalar una extensión como Tampermonkey o Userscripts para instalar este script.

Tendrás que instalar una extensión como Tampermonkey antes de poder instalar este script.

Necesitarás instalar una extensión para administrar scripts de usuario si quieres instalar este script.

(Ya tengo un administrador de scripts de usuario, déjame instalarlo)

Tendrás que instalar una extensión como Stylus antes de poder instalar este script.

Tendrás que instalar una extensión como Stylus antes de poder instalar este script.

Tendrás que instalar una extensión como Stylus antes de poder instalar este script.

Para poder instalar esto tendrás que instalar primero una extensión de estilos de usuario.

Para poder instalar esto tendrás que instalar primero una extensión de estilos de usuario.

Para poder instalar esto tendrás que instalar primero una extensión de estilos de usuario.

(Ya tengo un administrador de estilos de usuario, déjame instalarlo)

// ==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);
        }

    }
})();