Xbox Cloud Gaming Localization

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

Szkript telepítése?
A szerző által javasolt szkript

Ez is érdekelhet: Xbox Cloud Gaming Region Unlocker

Szkript telepítése
// ==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);
        }

    }
})();