// ==UserScript==
// @name Gota.io Features by Donut
// @namespace https://www.youtube.com/channel/UCIpCflcKEN9YgaO9qDahpRg
// @version 1.5.3
// @description Hold down S to freeze. Press K to show/hide skins, N to show/hide names, M to show/hide mass, F to show/hide food, H to show/hide chat. For Team Scrimmage players: auto respawn in Team Scrimmage, "Team 2v2" is selected by default, press Enter to queue a match or leave a match that has ended, L to leave a match at any time. You can change the keys if you wish. You can also set custom keys for triple split, show/hide minimap, score panel, party panel, leaderboard, and auto decline party invites.
// @match http://gota.io/web/*
// @author Donut
// @grant none
// @run-at document-end
// @downloadURL none
// ==/UserScript==
(function(){
function getVersionParts(v, validate = false) {
var parts = v.split('.');
if (validate) for (var i = 0; i < parts.length; i++) if (!/^\d+$/.test(parts[i])) return false;
return parts;
}
function compareVersions(v1, v2) {
var v1parts = getVersionParts(v1), v2parts = getVersionParts(v2);
while (v1parts.length < v2parts.length) v1parts.push('0');
while (v2parts.length < v1parts.length) v2parts.push('0');
v1parts = v1parts.map(Number);
v2parts = v2parts.map(Number);
for (var i = 0; i < v1parts.length; ++i) {
if (v1parts[i] > v2parts[i]) return 1;
else if (v1parts[i] < v2parts[i]) return -1;
}
return 0;
}
var importantUpdates = ['1.5.3'];
importantUpdates.sort(compareVersions);
var version = '1.5.3', storageVersion = localStorage['donut-version'], notify = false;
if (!storageVersion || !getVersionParts(storageVersion, true) || compareVersions(version, storageVersion) == 1 && (importantUpdates.lastIndexOf(version) >= 0 || compareVersions(importantUpdates[importantUpdates.length - 1], storageVersion) == 1)) notify = true;
else localStorage['donut-version'] = version;
var styles = {
'.donut-features-table': {
'margin': 'auto',
'border-collapse': 'collapse'
},
'.donut-features-table td:nth-child(1)': {
'text-align': 'left'
},
'.donut-features-table th, .donut-features-table td': {
'padding': '2px 8px'
},
'.donut-features-table input[type="text"]': {
'width': '100px',
'text-align': 'center'
},
'.donut-features-div': {
'padding-top': '4px'
},
'.donut-checkbox': {
'position': 'relative',
'top': '1px',
'margin': '0 5px'
},
'.donut-channel-link': {
'box-sizing': 'border-box',
'display': 'block',
'margin': '3px auto 0',
'padding': '4px 8px 4px 5.5px',
'width': '85px',
'height': '24px',
'color': '#fefefe',
'font-family': 'Arial, Helvetica, sans-serif',
'font-size': '12px',
'line-height': 'normal',
'text-align': 'center',
'text-decoration': 'none',
'background-color': '#e62117',
'border': 'solid 1px transparent',
'border-radius': '2px',
'white-space': 'nowrap',
'vertical-align': 'middle',
'box-shadow': '0 1px 0 rgba(0,0,0,0.05)'
},
'.donut-channel-link::before': {
'content': '""',
'position': 'relative',
'top': '-1px',
'display': 'inline-block',
'margin-right': '6px',
'width': '16px',
'height': '12px',
'background': 'no-repeat url(//s.ytimg.com/yts/imgbin/www-hitchhiker-vfl-Nn88d.png) -721px -88px',
'background-size': 'auto',
'vertical-align': 'middle'
},
'.donut-channel-link>span': {
'display': 'inline-block',
'-moz-box-sizing': 'border-box',
'box-sizing': 'border-box'
},
'.donut-channel-link:hover': {
'background-color': '#cc181e'
},
'.donut-channel-link:active': {
'background-color': '#b31217'
}
};
if (notify) {
styles['.donut-features-btn::before'] = {
'content': '"!"',
'position': 'absolute',
'top': '-6px',
'right': '-6px'
};
styles['.donut-features-btn::before, .donut-updated'] = {
'width': '12px',
'height': '12px',
'color': 'white',
'font-size': '11px',
'font-weight': 'bold',
'text-align': 'center',
'line-height': '12px',
'background-color': 'red',
'border-radius': '50%',
'overflow': 'hidden',
'-webkit-user-select': 'none',
'-moz-user-select': 'none',
'-ms-user-select': 'none',
'user-select': 'none',
'display': 'inline-flex',
'align-items': 'center',
'justify-content': 'center'
};
styles['.donut-updated'] = {
'position': 'relative',
'top': '-1px',
'vertical-align': 'middle'
};
}
var style = document.createElement('style');
document.head.appendChild(style);
var stylesheet = style.sheet;
for (var selector in styles)
for (var property in styles[selector])
stylesheet.insertRule(selector + '{' + property + ':' + styles[selector][property] + ';' + '}', stylesheet.cssRules.length);
var bottom = document.getElementsByClassName('main-bottom')[0], stats = document.getElementsByClassName('main-bottom-stats')[0];
bottom.style.position = 'relative';
bottom.style.left = '-7px';
stats.style.width = '100%';
stats.style.height = '140px';
var btnTemplate = document.createElement('button');
btnTemplate.className = 'gota-btn bottom-btn';
btnTemplate.style.marginTop = '12px';
btnTemplate.style.color = 'white';
btnTemplate.style.backgroundColor = 'rgba(23,22,23,.9)';
var btn = btnTemplate.cloneNode();
btn.className += ' donut-features-btn';
btn.style.position = 'relative';
btn.style.left = '2px';
btn.style.padding = '4px 3px 3px';
btn.style.width = '100%';
btn.style.fontSize = '16px';
btn.innerText = 'Features by Donut';
btn.addEventListener('click', function() {
blackout.style.opacity = '0';
blackout.style.display = 'block';
resize();
blackout.style.opacity = '1';
if (notify) {
localStorage['donut-version'] = version;
stylesheet.insertRule('.donut-features-btn::before{content:none;}', stylesheet.cssRules.length);
}
});
document.getElementsByClassName('main-bottom-right')[0].appendChild(btn);
var blackout = document.createElement('div');
blackout.style.position = 'fixed';
blackout.style.top = '0';
blackout.style.right = '0';
blackout.style.bottom = '0';
blackout.style.left = '0';
blackout.style.display = 'none';
blackout.style.background = 'rgba(0,0,0,.5)';
blackout.style.overflow = 'auto';
blackout.style.zIndex = '100';
document.body.appendChild(blackout);
var win = document.createElement('div');
win.style.position = 'absolute';
win.style.top = '50%';
win.style.left = '50%';
win.style.margin = '0';
win.style.padding = '15px';
win.style.color = 'white';
win.style.fontFamily = 'Arial, Helvetica, sans-serif';
win.style.fontSize = '16px';
win.style.lineHeight = '22px';
win.style.textAlign = 'center';
win.style.backgroundColor = 'rgb(14,14,14)';
win.style.border = 'solid 2px rgba(255,255,255,.2)';
win.style.borderRadius = '5px';
win.style.transform = 'translate(-50%, -50%)';
win.innerHTML = `
Feature
Default
Custom key
Freeze
S
Show/hide skins
K
Show/hide names
N
Show/hide mass
M
Show/hide food
F
Show/hide chat
H
Triple split
Show/hide minimap
Show/hide score panel
Show/hide party panel
Show/hide leaderboard
Toggle Decline party invites
Toggle Auto respawn
Team Scrimmage
Queue
Enter
Leave a match
L
You should only use the Leave a match hotkey when playing with random teammates.
If you want to disable a hotkey, type Delete. You can support the developer by checking out his YouTube channel:
YouTube`;
if (notify && version == '1.5.3') win.innerHTML = `
If you were one of the first to install the extension, you may have missed some of the new features. Since now, to let you know about all the new features you'll be shown the ! icon whenever the script is updated.
` + win.innerHTML;
blackout.appendChild(win);
function resize() {
if (blackout.style.display == 'block') {
if (window.innerHeight < win.offsetHeight + 20) {
win.style.top = '0';
win.style.margin = '10px';
win.style.transform = 'translate(-50%, 0%)';
} else {
win.style.top = '50%';
win.style.margin = '0';
win.style.transform = 'translate(-50%, -50%)';
}
}
}
window.addEventListener('resize', resize);
document.addEventListener('click', function(e) {
if (blackout.style.display == 'block' && e.target != win && !win.contains(e.target)) blackout.style.display = 'none';
}, true);
var done = btnTemplate.cloneNode();
done.style.display = 'block';
done.style.margin = '12px auto 0';
done.innerText = 'Done';
done.addEventListener('click', function() {
blackout.style.display = 'none';
});
win.appendChild(done);
var keyCodes = {
8 : 'Backspace',
9 : 'Tab',
13 : 'Enter',
33 : 'Page Up',
34 : 'Page Down',
35 : 'End',
36 : 'Home',
37 : 'Left Arrow',
38 : 'Up Arrow',
39 : 'Right Arrow',
40 : 'Down Arrow',
48 : '0',
49 : '1',
50 : '2',
51 : '3',
52 : '4',
53 : '5',
54 : '6',
55 : '7',
56 : '8',
57 : '9',
65 : 'A',
66 : 'B',
67 : 'C',
68 : 'D',
69 : 'E',
70 : 'F',
71 : 'G',
72 : 'H',
73 : 'I',
74 : 'J',
75 : 'K',
76 : 'L',
77 : 'M',
78 : 'N',
79 : 'O',
80 : 'P',
81 : 'Q',
82 : 'R',
83 : 'S',
84 : 'T',
85 : 'U',
86 : 'V',
87 : 'W',
88 : 'X',
89 : 'Y',
90 : 'Z',
96 : 'Numpad 0',
97 : 'Numpad 1',
98 : 'Numpad 2',
99 : 'Numpad 3',
100 : 'Numpad 4',
101 : 'Numpad 5',
102 : 'Numpad 6',
103 : 'Numpad 7',
104 : 'Numpad 8',
105 : 'Numpad 9'
};
var defaultKeys = {
freeze: 83,
skins: 75,
names: 78,
mass: 77,
food: 70,
chat: 72,
tripleSplit: null,
minimap: null,
scorePanel: null,
partyPanel: null,
leaderboard: null,
autoDecline: null,
autoRespawn: null,
queue: 13,
leave: 76
};
var inputs = document.querySelectorAll('.donut-features-table input[type="text"][data-donut-feature]'), errorKeys = 'Gota.io Features by Donut: An error occurred. We had to reset your custom keys.';
function fillInputs(keys) {
for (var i = 0; i < inputs.length; i++) {
var feature = inputs[i].dataset.donutFeature, key = keys[feature];
if (typeof key == 'undefined' && feature in defaultKeys) {
key = keys[feature] = defaultKeys[feature];
localStorage['donut-keys'] = JSON.stringify(keys);
}
if (key === null) continue;
var value = keyCodes[key];
if (value) inputs[i].value = value;
else throw errorKeys;
}
}
function tryLocalStorage(index, defaultObj, func, error) {
var obj; index = 'donut-' + index;
try {
if (!localStorage[index]) throw null;
obj = JSON.parse(localStorage[index]);
func(obj);
} catch (e) {
obj = JSON.parse(JSON.stringify(defaultObj));
if (e) console.error(error);
localStorage[index] = JSON.stringify(obj);
func(obj);
}
return obj;
}
var keys = tryLocalStorage('keys', defaultKeys, fillInputs, errorKeys);
for (var j = 0; j < inputs.length; j++) {
inputs[j].addEventListener('keydown', function(e) {
e.preventDefault();
var feature = this.dataset.donutFeature, key = e.which || e.keyCode || 0;
if (key == 46) {
this.value = '';
keys[feature] = null;
localStorage['donut-keys'] = JSON.stringify(keys);
}
else if (key != keys[feature] && key in keyCodes) {
this.value = keyCodes[key];
for (var k in keys) {
if (keys[k] == key) {
keys[k] = null;
for (var l = 0; l < inputs.length; l++) if (inputs[l].dataset.donutFeature == k) {
inputs[l].value = '';
break;
}
}
}
keys[feature] = key;
localStorage['donut-keys'] = JSON.stringify(keys);
}
});
}
function triggerCheckbox(id) {
$('#' + id).prop('checked', !$('#' + id).prop('checked')).trigger('change');
}
function triggerDonutCheckbox(id) {
var checkbox = document.getElementById(id);
checkbox.checked = !checkbox.checked;
checkbox.dispatchEvent(new Event('change'));
}
function fadeOutMain() {
if (document.getElementById('main').style.display == 'block') window.onkeydown({keyCode: 27, which: 27});
}
var alertWin = win.cloneNode();
alertWin.style.top = '10px';
alertWin.style.display = 'none';
alertWin.style.lineHeight = '19px';
alertWin.style.backgroundColor = 'rgba(23,23,23,.9)';
alertWin.style.boxShadow = '0 2px 3px rgba(0,0,0,.25)';
alertWin.style.transform = 'translate(-50%, 0%)';
alertWin.style.zIndex = '3';
document.body.appendChild(alertWin);
var alertFadeOutTimer = 0;
function Alert(html, timeout) {
alertWin.innerHTML = html;
$(alertWin).fadeIn(500);
if (alertFadeOutTimer) clearTimeout(alertFadeOutTimer);
alertFadeOutTimer = setTimeout(function() {
$(alertWin).fadeOut(500);
alertFadeOutTimer = 0;
}, timeout);
}
var frozen = false, x = 0, y = 0, originalMousemove = window.onmousemove;
window.onmousemove = function(e) {
x = e.clientX; y = e.clientY;
if (!frozen) originalMousemove(e);
};
window.addEventListener('keydown', function(e) {
if (!e.altKey && !e.ctrlKey && !e.metaKey) {
var key = e.which || e.keyCode || 0;
if (key == 13 && document.activeElement === document.getElementById('name-box')) team2v2();
else if (key == 27) {
if (!firstEscape) firstEscape = true;
if (alertWin.offsetHeight) $(alertWin).fadeOut(500);
}
else if (!e.shiftKey && document.activeElement.tagName != 'INPUT' && document.activeElement.tagName != 'TEXTAREA') switch (key) {
case keys.freeze:
X = window.innerWidth/2;
Y = window.innerHeight/2;
originalMousemove({clientX: X, clientY: Y});
frozen = true;
break;
case keys.skins:
triggerCheckbox('cHideSkins');
break;
case keys.names:
triggerCheckbox('cHideNames');
break;
case keys.mass:
triggerCheckbox('cShowMass');
break;
case keys.food:
triggerCheckbox('cHideFood');
break;
case keys.chat:
triggerCheckbox('cHideChat');
break;
case keys.tripleSplit:
for (var i = 0; i < 3; i++) $(window).trigger($.Event('keydown', {keyCode: 32, which: 32}));
break;
case keys.minimap:
triggerCheckbox('cHideMinimap');
break;
case keys.scorePanel:
triggerDonutCheckbox('donut-hideScorePanel');
break;
case keys.partyPanel:
triggerDonutCheckbox('donut-hidePartyPanel');
break;
case keys.leaderboard:
triggerDonutCheckbox('donut-hideLeaderboard');
break;
case keys.autoDecline:
triggerCheckbox('cAutoDecline');
Alert('Auto Decline Party Invites: ' + (document.getElementById('cAutoDecline').checked ? 'On' : 'Off') + '', 2500);
break;
case keys.autoRespawn:
var id = done2v2 ? 'donut-scrimmageAutoRespawn' : 'cAutoRespawn';
if (done2v2) triggerDonutCheckbox(id);
else triggerCheckbox(id);
Alert('Auto Respawn' + (done2v2 ? ' in Team Scrimmage' : '') + ': ' + (document.getElementById(id).checked ? 'On' : 'Off') + '', 2500);
break;
case keys.queue:
if (document.getElementById('scrimmage-btn-leave').style.display == 'block') $('#btn-leave-match').trigger('click');
else if (document.getElementById('main-scrimmage').style.display == 'block') $('#btn-queue').trigger('click');
fadeOutMain();
break;
case keys.leave:
if (scrimmage) {
var cp = document.getElementById('chat-panel'), hidden = cp.style.display == 'none';
if (hidden) {
cp.style.opacity = '0';
cp.style.display = 'block';
}
$('#chat-input').val('/leave').focus().trigger($.Event('keypress', {keyCode: 13, which: 13})).blur();
scrimmage = false;
if (hidden) {
cp.style.display = 'none';
cp.style.opacity = '1';
}
fadeOutMain();
}
}
}
});
window.addEventListener('keyup', function(e) {
var key = e.which || e.keyCode || 0;
if (key == keys.freeze && !e.altKey && !e.ctrlKey && !e.metaKey && !e.shiftKey && document.activeElement.tagName != 'INPUT' && document.activeElement.tagName != 'TEXTAREA') {
frozen = false;
originalMousemove({clientX: x, clientY: y});
}
});
var defaultSwitches = {
hideScorePanel: false,
hidePartyPanel: false,
hideLeaderboard: false,
scrimmageAutoRespawn: true,
startWithScrimmage: false
};
var checkboxes = document.getElementsByClassName('donut-checkbox'), errorSwitches = 'Gota.io Features by Donut: An error occurred. We had to reset your settings.';
function fillCheckboxes(switches) {
for (var i = 0; i < checkboxes.length; i++) {
var feature = checkboxes[i].dataset.donutFeature, Switch = switches[feature];
if (typeof Switch == 'undefined' && feature in defaultSwitches) {
Switch = switches[feature] = defaultSwitches[feature];
localStorage['donut-switches'] = JSON.stringify(switches);
}
if (typeof Switch == 'boolean') checkboxes[i].checked = Switch;
else throw errorSwitches;
}
}
var switches = tryLocalStorage('switches', defaultSwitches, fillCheckboxes, errorSwitches);
var anchorObd = document.querySelector('.options-container>p:nth-of-type(2)'), obd = anchorObd.cloneNode(true);
obd.firstChild.innerText = 'Options by Donut';
anchorObd.parentNode.insertBefore(obd, anchorObd);
function duplicatedCheckbox(el, feature, text) {
var checkbox = document.createElement('input');
checkbox.type = 'checkbox';
checkbox.className = 'checkbox-options';
checkbox.dataset.donutFeature = feature;
el.parentNode.insertBefore(checkbox, el);
var span = document.createElement('span');
span.innerText = text;
el.parentNode.insertBefore(span, el);
var br = document.createElement('br');
el.parentNode.insertBefore(br, el);
return checkbox;
}
for (var feature in defaultSwitches) {
var bro = document.querySelector('.donut-checkbox[data-donut-feature="' + feature + '"]');
duplicatedCheckbox(anchorObd, feature, feature == 'startWithScrimmage' ? 'Join Scrimmage as I open the game' : bro.parentNode.innerText).checked = switches[feature];
}
checkboxes = document.querySelectorAll('input[type="checkbox"][data-donut-feature]');
for (var k = 0; k < checkboxes.length; k++) {
checkboxes[k].addEventListener('change', function(e) {
var feature = this.dataset.donutFeature;
switches[feature] = this.checked;
localStorage['donut-switches'] = JSON.stringify(switches);
document.querySelector('input[type="checkbox"][data-donut-feature="' + feature + '"]' + (this.classList.contains('donut-checkbox') ? ':not(.donut-checkbox)' : '.donut-checkbox')).checked = this.checked;
switch (feature) {
case 'hideScorePanel':
if (this.checked) hideScorePanel();
else {
document.getElementById('party-panel').style.top = document.getElementById('score-panel').offsetHeight + 20 + 'px';
document.getElementById('score-panel').style.opacity = '1';
}
break;
case 'hidePartyPanel':
visibility('party-panel', !this.checked);
break;
case 'hideLeaderboard':
visibility('leaderboard-panel', !this.checked);
break;
case 'scrimmageAutoRespawn':
if (this.checked && scrimmage) scrimRespawn();
}
});
}
function hideScorePanel() {
document.getElementById('score-panel').style.opacity = '0';
document.getElementById('party-panel').style.top = '10px';
}
if (switches.hideScorePanel) hideScorePanel();
document.getElementById('cHideId').addEventListener('change', function() {
if (document.getElementById('score-panel').style.opacity == '1') document.getElementById('party-panel').style.top = document.getElementById('score-panel').offsetHeight + 20 + 'px';
});
function visibility(id, visible) {
document.getElementById(id).style.visibility = visible ? 'visible' : 'hidden';
}
if (switches.hideLeaderboard) visibility('leaderboard-panel', false);
if (switches.hidePartyPanel) visibility('party-panel', false);
var scrimmage = false, respawnCheckInterval = 0, respawnTimer = 0;
document.getElementById('btn-queue').addEventListener('click', function() {
scrimmage = true;
if (switches.scrimmageAutoRespawn) scrimRespawn();
});
function scrimmageFalse() {
scrimmage = false;
if (respawnTimer) {
clearTimeout(respawnTimer);
respawnTimer = respawnCheckInterval = 0;
}
}
document.getElementById('btn-leave-match').addEventListener('click', scrimmageFalse);
function scrimRespawn() {
var cells = document.getElementById('playerCells'), cellsNum = parseInt(cells.innerText, 10);
if (respawnCheckInterval) clearInterval(respawnCheckInterval);
respawnCheckInterval = setInterval(function() {
if (switches.scrimmageAutoRespawn && scrimmage) {
var temp = parseInt(cells.innerText, 10);
if (temp != cellsNum) {
cellsNum = temp;
if (cellsNum === 0) {
if (document.getElementById('main-scrimmage').style.display == 'none') respawnTimer = setTimeout(function() {
if (switches.scrimmageAutoRespawn && scrimmage) {
$('#btn-play').trigger('click');
respawnTimer = 0;
scrimRespawn();
}
}, 10000);
clearInterval(respawnCheckInterval);
respawnCheckInterval = 0;
}
}
} else {
clearInterval(respawnCheckInterval);
respawnCheckInterval = 0;
}
}, 500);
}
var firstEscape = false;
function startScrimmage() {
var region = document.getElementsByClassName('server-active')[0].getAttribute('region');
if (region != 'ap') {
var id = region == 'eu' ? 's_BETA' : 's_Jet';
var interval = setInterval(function() {
if (document.getElementById(id) && document.getElementsByClassName('server-selected').length) {
$('#' + id).trigger('click');
$('#btn-play').trigger('click');
setTimeout(function() {
if (!firstEscape && !done2v2) Alert('It looks like the server is not responding. You may want to press Esc to go back to the game menu.', 20000);
}, 5000);
clearInterval(interval);
}
}, 100);
}
}
if (switches.startWithScrimmage) startScrimmage();
var lastScrimmageServer = null;
function isTeamScrimmage() {
var servers = ['BETA', 'Jet'];
for (var i = 0; i < servers.length; i++) if (document.getElementById('s_' + servers[i]).classList.contains('server-selected')) return servers[i];
return false;
}
var done2v2 = false, checkigTean2v2 = false;
function team2v2() {
if (!checkigTean2v2) {
var scrimmageServer = isTeamScrimmage();
if (scrimmageServer) {
if (!done2v2 || scrimmageServer != lastScrimmageServer) {
lastScrimmageServer = scrimmageServer;
checkigTean2v2 = true;
var interval = setInterval(function() {
if (document.getElementById('main-scrimmage').style.display == 'block' && document.getElementById('scrimmage-mode-select').options.selectedIndex === 0) {
document.getElementById('scrimmage-mode-select').options.selectedIndex = 1;
$('#scrimmage-mode-select').trigger('change');
clearTimeout(timeout);
clearInterval(interval);
done2v2 = true;
checkigTean2v2 = false;
if (switches.startWithScrimmage && alertWin.offsetHeight) alertWin.style.display = 'none';
}
}, 100), timeout = setTimeout(function() {
clearInterval(interval);
checkigTean2v2 = false;
}, 25000);
}
} else {
scrimmageFalse();
done2v2 = false;
}
}
}
document.getElementById('btn-play').addEventListener('click', team2v2);
document.getElementById('btn-spec').addEventListener('click', team2v2);
})();