DigDig.IO Server Selector

Server selector for digdig.io

目前為 2021-07-06 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

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

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         DigDig.IO Server Selector
// @namespace    http://tampermonkey.net/
// @version      0.1
// @description  Server selector for digdig.io
// @author       Zertalious (Zert)
// @match        http://digdig.io/*
// @icon         
// @grant        unsafeWindow
// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// ==/UserScript==

GM_addStyle(`
.container {
    width: 100%;
    height: 100%;
    position: fixed;
    top: 0;
    left: 0;
    pointer-events: none;
    user-select: none;
}
 
.active {
    pointer-events: auto;
}
 
.btn,
.modalUI {
    background: #aeaeae;
    color: #fff;
    font-size: 14px;
    font-family: Ubuntu;
    border: 4px solid #8f8f8f;
    border-radius: 6px;
    pointer-events: auto;
    text-shadow: -1px -1px 0 #000, 1px -1px 0 #000, -1px 1px 0 #000, 1px 1px 0 #000;
}
 
.btn {
    position: absolute;
    right: 10px;
    bottom: 10px;
    padding: 5px 15px;
    cursor: pointer;
    outline: 0;
    transition: 0.2s;
}
 
.btn-active {
    transform: translate(0, 0);
}
 
.container:not(.active) .btn:hover {
    background: #c7c7c7;
}
 
.active .btn {
    background: #c7c7c7;
}
 
.modalUI {
    position: absolute;
    bottom: 55px;
    right: 6px;
    padding: 10px;
    width: 300px;
    max-height: 400px;
    transition: 0.2s;
    transform: translate(0, calc(100% + 55px));
    overflow: auto;
}
 
.active .modalUI {
    transform: translate(0, 0);
}
 
.modalUI-title {
    font-size: 1.6em;
    text-align: center;
    margin-top: 0;
}
 
.modalUI-close-btn {
    position: absolute;
    right: 5px;
    top: 7px;
    width: 24px;
    height: 24px;
    border-radius: 4px;
    background: #bb5555;
    border: 4px solid #974545;
    cursor: pointer;
    outline: 0;
}
 
.modalUI-close-btn:hover {
    background: #da6868;
}
 
.modalUI-close-btn:after,
.modalUI-close-btn:before {
    content: "";
    position: absolute;
    left: 51%;
    top: 52.5%;
    width: 3px;
    height: 100%;
    background: #cccccc;
    border-radius: 4px;
}
 
.modalUI-close-btn:after {
    transform: translate(-50%, -50%) rotate(45deg);
}
 
.modalUI-close-btn:before {
    transform: translate(-50%, -50%) rotate(-45deg);
}
 
.modalUI-element {
    background: rgba(0, 0, 0, 0.2);
    padding: 10px;
    border-radius: 5px;
    margin-bottom: 5px;
    cursor: pointer;
}
 
.modalUI-element:last-child {
    margin-bottom: 0;
}
 
.modalUI-element-active {
    box-shadow: inset 0 0 0 2px rgba(0, 0, 0, 0.2);
}
 
.modalUI-loader {
    display: flex;
    flex-direction: column;
    align-items: center;
    padding: 10px;
}
 
.spinner {
    width: 40px;
    height: 40px;
    border: 10px solid rgba(0, 0, 0, 0.5);
    border-top-color: rgba(0, 0, 0, 0.8);
    border-radius: 50%;
    animation: spin 0.5s infinite;
}
 
.spinner-text {
    margin-top: 5px;
    font-size: 1.25em;
}
 
.modalUI-info {
    position: absolute;
    right: 30px;
    top: 6px;
    font-size: 1.20em;
    padding: 4px 5px;
}
 
.modalUI-info:hover .modalUI-info-content {
    opacity: 1;
    pointer-events: inherit;
}
 
.modalUI-info-content {
    position: absolute;
    bottom: 170%;
    right: 0;
    background: rgba(0, 0, 0, 0.4);
    border-radius: 4px;
    opacity: 0;
    padding: 8px;
    font-size: 0.85rem;
    width: 450px;
    transition: 0.2s;
    pointer-events: none;
    display: flex;
    flex-direction: row;
}
 
.modalUI-info-title {
    margin: 0;
    margin-bottom: 5px;
    font-size: 1.40rem;
}
 
.version-label {
    font-size: 0.75em;
    text-align: right;
}
 
.divider {
    width: 3px;
    background: rgba(0, 0, 0, 0.3);
    border-radius: 5px;
    margin: 0 10px;
}
 
.modalUI-info-content > div:not(.divider) {
    flex: 1;
}
 
@keyframes spin {
    from {
        transform: rotate(0deg);
    }
    to {
        transform: rotate(360deg);
    }
}
`);

(async function() {
	const html = `
    <button class="btn">Servers</button>
    <div class="modalUI">
      <h1 class="modalUI-title">Servers</h1>
      <button class="modalUI-close-btn"></button>
      <div class="modalUI-element modalUI-loader">
        <div class="spinner"></div>
        <div class="spinner-text">Fetching...</div>
      </div>
    </div>
    `;
    const container = document.createElement("div");
    container.classList.add('container');
    container.innerHTML = html;
    document.body.appendChild(container);
    const modalUIEl = document.querySelector(".modalUI");
    const btnEl = document.querySelector(".btn");
    const closeBtnEl = document.querySelector(".modalUI-close-btn");
    const loaderEl = document.querySelector(".modalUI-loader");
    btnEl.onclick = () => {
        container.classList.toggle("active");
    }
    closeBtnEl.onclick = () => {
        closeModal();
    }
    const closeModal = () => {
        container.classList.remove("active");
    }
    window.addEventListener('click', (evt) => {
        if (evt.target === container) {
            closeModal();
        }
    });
    function removeActiveClass() {
        let activeEl = document.querySelector(".modalUI-element-active");
        if (activeEl) {
            activeEl.classList.remove("modalUI-element-active");
        }
    }

	const sockets = [];
	let gameSocket = null;
	const KEY = 'digdig-selected-server';
	unsafeWindow.WebSocket = new Proxy(unsafeWindow.WebSocket, {
		construct(target, args) {
			let isGameSocket;
			if (servers) {
				for (let key in servers) {
					if (args[0].indexOf(servers[key].ipv4) > -1) { // is a game socket
						// disconnect socket and connect to selected server
						isGameSocket = true;
						console.log('game server found!');
						if (args[0].indexOf(GM_getValue(KEY)) == -1) {
							let old = args[0];
							args[0] = args[0].split('//')[0] + '//' + GM_getValue(KEY) + ':443';
							console.log('Was connecting to ' + old + ', overriding to ' + GM_getValue(KEY) + '...');
						} else {
							GM_setValue(KEY, args[0].split('//')[1].split(':')[0]);
							console.log('No saved server found! Saving current server...');
							const el = document.querySelector('[data-server="' + GM_getValue(KEY) + '"]');
							if (el) {
								el.classList.add('modalUI-element-active');
							}
						}
						break;
					}
				}
			}
			const socket = Reflect.construct(...arguments);
			if (isGameSocket) {
				gameSocket = socket;
			}
			sockets.push(socket);
			return socket;
		}
	});
	const response = await fetch('https://api.n.m28.io/endpoint/digdig/findEach/');
	const json = await response.json();
	const servers = json.servers;
	for (let i = 0; i < sockets.length; i++) {
		const socket = sockets[i];
		for (let key in servers) {
			if (socket.url.indexOf(servers[key].ipv4) > -1) {
				gameSocket = socket;
				reconnect();
			}
		}
	}
	for (let key in servers) {
		const server = servers[key];
		let el = document.createElement("div");
        el.classList.add("modalUI-element");
        el.innerHTML = key;
        el.setAttribute('data-server', server.ipv4);
        el.onclick = function() {
            removeActiveClass();
            el.classList.add("modalUI-element-active");
            GM_setValue(KEY, server.ipv4);
            reconnect();
        }
        if (server.ipv4 == GM_getValue(KEY)) {
        	el.classList.add('modalUI-element-active');
        }
        modalUIEl.appendChild(el);
	}
	loaderEl.style.display = 'none';
	function reconnect() {
		while (sockets.length > 0) {
			try {
				sockets.shift().close();
			} catch (e) {}
		}
	}
})();