Xbox Cloud Gaming Server Selection

Manually select servers for Xbox Cloud Gaming.

スクリプトをインストールするには、Tampermonkey, GreasemonkeyViolentmonkey のような拡張機能のインストールが必要です。

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

スクリプトをインストールするには、TampermonkeyViolentmonkey のような拡張機能のインストールが必要です。

スクリプトをインストールするには、TampermonkeyUserscripts のような拡張機能のインストールが必要です。

このスクリプトをインストールするには、Tampermonkeyなどの拡張機能をインストールする必要があります。

このスクリプトをインストールするには、ユーザースクリプト管理ツールの拡張機能をインストールする必要があります。

(ユーザースクリプト管理ツールは設定済みなのでインストール!)

このスタイルをインストールするには、Stylusなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus などの拡張機能をインストールする必要があります。

このスタイルをインストールするには、Stylus tなどの拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

このスタイルをインストールするには、ユーザースタイル管理用の拡張機能をインストールする必要があります。

(ユーザースタイル管理ツールは設定済みなのでインストール!)

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください
// ==UserScript==
// @name                 Xbox Cloud Gaming Server Selection
// @name:zh-CN           Xbox Cloud Gaming 云游戏服务器选择
// @name:zh-TW           Xbox Cloud Gaming 雲端游戲伺服器選取
// @namespace            http://tampermonkey.net/
// @version              0.2
// @description          Manually select servers for Xbox Cloud Gaming.
// @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
// @grant                GM_setValue
// @grant                GM_getValue
// @grant                GM_registerMenuCommand
// @grant                GM_unregisterMenuCommand
// ==/UserScript==

(function() {
    'use strict';

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

    // Your code here...

    let selectedServer = "Auto";
    let currentAutoServer = "Unknown";
    let serverList = [];
    let menuItemList = [];

    function checkSelected(name) {
        let selected = GM_getValue("XCLOUD_SERVER_SELECT");
        return name == selected;
    }

    function registerSelectableMenuItem(name) {
        return GM_registerMenuCommand((checkSelected(name) ? "✅" : "🔲") + " " + name + (name === "Auto" ? " (" + currentAutoServer + ")" : ""), function() {
            selectedServer = name;
            saveSelectedServer();
            updateSelectableMenuItem();
        });
    }

    function loadSelectedServer() {
        let selected = GM_getValue("XCLOUD_SERVER_SELECT");
        if (selected) {
            selectedServer = selected;
        } else {
            GM_setValue("XCLOUD_SERVER_SELECT", selectedServer);
        }
        let list = GM_getValue("XCLOUD_SERVER_LIST");
        if (list) {
            serverList = JSON.parse(list);
        } else {
            GM_setValue("XCLOUD_SERVER_LIST", JSON.stringify(serverList));
        }
    }

    function saveSelectedServer() {
        GM_setValue("XCLOUD_SERVER_SELECT", selectedServer);
        GM_setValue("XCLOUD_SERVER_LIST", JSON.stringify(serverList));
    }

    async function updateSelectableMenuItem() {
        for(let command of menuItemList) {
            await GM_unregisterMenuCommand(command);
        }
        menuItemList = [];
        menuItemList.push(await registerSelectableMenuItem("Auto"));
        for(let server of serverList) {
            menuItemList.push(await registerSelectableMenuItem(server));
        }
    }

    function buildInjectedHeader(headers) {
        if (!headers){
            headers = new Headers();
        }
        headers.set("X-Forwarded-For", forwardedIp);
        return headers;
    }

    function checkCorsAllowed(url) {
        return (url.startsWith("http://") || url.startsWith("https://") || url.startsWith("://")) === false;
    }

    function checkCorsWhitelistAllowed(url) {
        return (url.indexOf('://xgpuweb.gssv-play-prod.xboxlive.com') > -1 || url.indexOf('://www.xbox.com') > -1);
    }

    function checkCorsAllAllowed(url) {
        if (url.indexOf('://xgpuweb.gssv-play-prod.xboxlive.com') > -1) {
            removeHeaderNavBar();
        }
        return (checkCorsAllowed(url) || checkCorsWhitelistAllowed(url));
    }

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

        if (url.indexOf('/v2/login/user') > -1) {
            // Area Select
            return new Promise((resolve, reject) => {
                originFetchSS(...arg).then(res => {
                    res.json().then(json => {
                        let newServerList = [];
                        json["offeringSettings"]["regions"].forEach((region) => {
                            newServerList.push(region["name"]);
                            if (region["isDefault"] === true) {
                                currentAutoServer = region["name"];
                            }
                        });
                        // console.log(newServerList);
                        serverList = newServerList;
                        saveSelectedServer();
                        updateSelectableMenuItem();
                        if (selectedServer !== "Auto" && newServerList.includes(selectedServer)) {
                            json["offeringSettings"]["regions"].forEach((region) => {
                                if (region["name"] === selectedServer) {
                                    region["isDefault"] = true;
                                } else {
                                    region["isDefault"] = false;
                                }
                            });
                        }
                        let body = JSON.stringify(json);
                        // console.log(body);
                        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 originFetchSS(...arg);
        }
    }

    if (!!self.GM_registerMenuCommand && !!self.GM_unregisterMenuCommand && !!self.GM_setValue && !!self.GM_getValue) {
        loadSelectedServer();
        updateSelectableMenuItem();
    } else {
        console.warn("[Xbox Cloud Gaming Server Selection] your userscript extensions not support region select");
    }

    // setInterval(removeHeaderNavBar, 1000);
})();