Xbox Cloud Gaming Localization (compatible with Safari by using Userscripts)

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

// ==UserScript==
// @name                 Xbox Cloud Gaming Localization (compatible with Safari by using Userscripts)
// @name:zh-CN           Xbox Cloud Gaming 云游戏语言本地化
// @name:zh-TW           Xbox Cloud Gaming 雲端游戲語言本地化
// @namespace            http://tampermonkey.net/
// @version              0.2
// @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                 
// @run-at               document-start
// @inject-into          page
// ==/UserScript==

(function() {
    'use strict';

    // Your code here...
    let allLanguages = [];
    let selectedLanguage = "";

    allLanguages.push(navigator.language);
    navigator.languages.forEach(language => {
        allLanguages.push(language);
    });
    for (let language of allLanguages) {
        const reg = /^[a-z]{2}-[A-Z]{2}$/;
        const isFullLanguage = reg.test(language);
        if (isFullLanguage) {
            selectedLanguage = language;
            break;
        }
    }
    
    console.log('[localization]selectedLanguage', selectedLanguage);

    const originFetch = fetch;
    window.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((resolve, reject) => {
                console.log('[localization]Started for play')
                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("[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);
        }

    }
})();