// ==UserScript==
// @name PhantomBot-ChatGuessr
// @namespace http://tampermonkey.net/
// @version 0.1
// @description try to take over the world!
// @author Sora
// @match https://*.geoguessr.com/*
// @icon https://www.google.com/s2/favicons?domain=geoguessr.com
// @require https://code.jquery.com/jquery-3.6.0.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/datatables/1.10.21/js/jquery.dataTables.min.js
// @grant none
// ==/UserScript==
if(localStorage.getItem('lauth')==null || localStorage.getItem('lauth')=="") localStorage.setItem('lauth',prompt("Enter Current Auth token",""))
var socket = null;
var isInGame = false;
var table;
var guesses = {};
window.MAP = null;
class GameHelper {
static isGameURL(url){ return url.includes("/game/"); }
static getGameId(url){ return url.substring(url.lastIndexOf("/") + 1); }
static sortByDistance(guesses){ return guesses.sort((a, b) => a.distance - b.distance); }
static sortByScore(guesses){ return guesses.sort((a, b) => b.score - a.score); }
}
const toMeter = (distance) => (distance >= 1 ? parseFloat(distance.toFixed(1)) + "km" : parseInt(distance * 1000) + "m");
const renderGuesses = (guesses) => {
let sguesses = GameHelper.sortByScore(guesses);
const rows = sguesses.map((guess, i) => {
return {
Position: i+1,
Player: `${guess.user}`,
Streak: "-",
Distance: guess.distance,
Score: guess.score,
};
});
table.clear().draw();
table.rows.add(rows).draw();
};
let markers = [];
let polylines = [];
function populateMap(location, scores) {
scores = GameHelper.sortByScore(scores);
const infowindow = new google.maps.InfoWindow();
const icon = {
path: `M13.04,41.77c-0.11-1.29-0.35-3.2-0.99-5.42c-0.91-3.17-4.74-9.54-5.49-10.79c-3.64-6.1-5.46-9.21-5.45-12.07
c0.03-4.57,2.77-7.72,3.21-8.22c0.52-0.58,4.12-4.47,9.8-4.17c4.73,0.24,7.67,3.23,8.45,4.07c0.47,0.51,3.22,3.61,3.31,8.11
c0.06,3.01-1.89,6.26-5.78,12.77c-0.18,0.3-4.15,6.95-5.1,10.26c-0.64,2.24-0.89,4.17-1,5.48C13.68,41.78,13.36,41.78,13.04,41.77z
`,
fillColor: "#de3e3e",
fillOpacity: 0.7,
scale: 1.2,
strokeColor: "#000000",
strokeWeight: 1,
anchor: new google.maps.Point(14, 43),
labelOrigin: new google.maps.Point(13.5, 15),
};
const locationMarker = new google.maps.Marker({
position: location,
url: `http://maps.google.com/maps?q=&layer=c&cbll=${location.lat},${location.lng}`,
icon: icon,
map: MAP,
});
google.maps.event.addListener(locationMarker, "click", () => {
window.open(locationMarker.url, "_blank");
});
markers.push(locationMarker);
icon.scale = 1;
scores.forEach((score, index) => {
const color = index == 0 ? "#E3BB39" : index == 1 ? "#C9C9C9" : index == 2 ? "#A3682E" : score.color;
icon.fillColor = color;
let pos = {lat:parseFloat(score.location.split(',')[0]),lng:parseFloat(score.location.split(',')[1])}
const guessMarker = new google.maps.Marker({
position: pos,
icon: icon,
map: MAP,
label: { color: "#000", fontWeight: "bold", fontSize: "16px", text: `${score.user.substr(0,2)}` },
});
google.maps.event.addListener(guessMarker, "mouseover", () => {
infowindow.setContent(`
${score.user}
${score.distance >= 1 ? parseFloat(score.distance.toFixed(1)) + "km" : parseInt(score.distance * 1000) + "m"}
${score.score}
`);
infowindow.open(MAP, guessMarker);
});
google.maps.event.addListener(guessMarker, "mouseout", () => {
infowindow.close();
});
markers.push(guessMarker);
polylines.push(
new google.maps.Polyline({
strokeColor: color,
strokeWeight: 4,
strokeOpacity: 0.6,
geodesic: true,
map: MAP,
path: [pos, location],
})
);
});
}
function clearMarkers() {
while (markers[0]) {
markers.pop().setMap(null);
}
while (polylines[0]) {
polylines.pop().setMap(null);
}
}
function hijackMap() {
const MAPS_API_URL = "https://maps.googleapis.com/maps/api/js?";
const GOOGLE_MAPS_PROMISE = new Promise((resolve, reject) => {
let scriptObserver = new MutationObserver((mutations) => {
for (const mutation of mutations) {
for (const node of mutation.addedNodes) {
if (node.tagName === "SCRIPT" && node.src.startsWith(MAPS_API_URL)) {
node.onload = () => {
scriptObserver.disconnect();
scriptObserver = undefined;
resolve();
};
}
}
}
});
let bodyDone = false;
let headDone = false;
new MutationObserver((_, observer) => {
if (!bodyDone && document.body) {
if (scriptObserver) {
scriptObserver.observe(document.body, {
childList: true,
});
bodyDone = true;
}
}
if (!headDone && document.head) {
if (scriptObserver) {
scriptObserver.observe(document.head, {
childList: true,
});
headDone = true;
}
}
if (headDone && bodyDone) {
observer.disconnect();
}
}).observe(document.documentElement, {
childList: true,
subtree: true,
});
});
function runAsClient(f) {
const script = document.createElement("script");
script.type = "text/javascript";
script.text = `(${f.toString()})()`;
document.body.appendChild(script);
}
GOOGLE_MAPS_PROMISE.then(() => {
runAsClient(() => {
const google = window.google;
const isGamePage = () => location.pathname.startsWith("/results/") || location.pathname.startsWith("/game/");
const onMapUpdate = (map) => {
try {
if (!isGamePage()) return;
MAP = map;
} catch (error) {
console.error("GeoguessrHijackMap Error:", error);
}
};
const oldMap = google.maps.Map;
google.maps.Map = Object.assign(
function (...args) {
const res = oldMap.apply(this, args);
this.addListener("idle", () => {
if (MAP != null) return;
onMapUpdate(this);
});
return res;
},
{
prototype: Object.create(oldMap.prototype),
}
);
});
});
}
const markerRemover = document.createElement("style");
markerRemover.innerHTML = ".map-pin{display:none}";
let gameState='none'
const handleGameState = async()=>{
let prevGameState = gameState;
let nextRoundBtn = document.querySelector('[data-qa="close-round-result"]');
let addGuessBtn = document.querySelector('[data-qa="perform-guess"]');
let playAgainBtn = document.querySelector('[data-qa="play-same-map"]');
if(gameState=='none'){
if(nextRoundBtn) gameState='result';
if(addGuessBtn) gameState='guess';
if(playAgainBtn) gameState='end';
}
if(gameState=='guess' && addGuessBtn){
markerRemover.remove();
addGuessBtn.addEventListener("click", async() => {
await new Promise(r => setTimeout(r, 500))
gameState='result';
socket.sendCommand('cg_refresh_gs_g', 'cga refresh', ()=>{});
handleGameState();
});
}else if(gameState=='result' && nextRoundBtn){
document.body.appendChild(markerRemover);
nextRoundBtn.addEventListener("click", async() => {
await new Promise(r => setTimeout(r, 500))
gameState='guess';
socket.sendCommand('cg_refresh_gs_r', 'cga refresh', ()=>{});
handleGameState();
});
}else if(gameState=='end' && playAgainBtn){
document.body.appendChild(markerRemover);
clearMarkers();
if(guesses){
renderGuesses(guesses.final);
for(let i = 0; i<= guesses.round; ++i){
populateMap(guesses.rounds[i],guesses[i])
}
}
playAgainBtn.addEventListener("click", async() => {
await new Promise(r => setTimeout(r, 500))
gameState='none';
socket.sendCommand('cg_refresh_gs_r', 'cga end', ()=>{ });
});
}else{
await new Promise(r => setTimeout(r, 500))
gameState='none'
handleGameState();
}
console.log(gameState)
}
const head = document.getElementsByTagName("head")[0];
const styles = document.createElement("style");
styles.innerHTML = `
.container_content__H3tXS > div:nth-child(1),.container_content__H3tXS > div:nth-child(2),.container_content__H3tXS > div:nth-child(3){display:none}
[data-qa=result-view-top]{max-height:100vh}
[data-qa=result-view-bottom]>div>div:last-child{display:none}
[data-qa=score]{display:none}
[data-qa=result-view-bottom]{position:fixed;width:100%;left:0;bottom:0}
[data-qa=result-view-bottom]>div{background:linear-gradient(transparent,#000),linear-gradient(90deg,#28374c36,#221d6ca1)}
.classic_section__19Ttr{display:none}button:focus{outline:0}.hud-button-group:last-child{display:none}.result-map__line{display:none}
.gm-ui-hover-effect{display:none!important}.gm-style-iw-c{top:-4px!important;padding:10px 0 0 17px!important}
.gm-style-iw-t::after{top:-4px!important}
.gm-iw__content{color:#000;text-align:center;font-weight:700; width:100%;}
#showScoreboard:hover{cursor:pointer;background:rgba(0,0,0,.7)}#settingsIcon span,#showScoreboard span{margin:-1px 0 0 1px}#scoreboardContainer{display:none;width:420px;position:absolute;top:0;left:0;right:0;bottom:0;overflow:hidden;pointer-events:none}#scoreboard{font-family:Montserrat,sans-serif;position:relative;min-width:380px;min-height:180px;max-width:1800px;max-height:1000px;padding:5px;background-color:rgba(0,0,0,.4);box-shadow:2px 2px 7px -2px #000;color:#fff;text-align:center;border-radius:10px;pointer-events:auto;user-select:none;overflow:hidden;z-index:999999}.dataTables_scrollHeadInner{width:100%!important}#scoreboardHeader{display:grid;grid-template-columns:90px auto 80px;justify-items:center;height:35px;font-size:18px;align-items:center}#scoreboardTitle{margin-top:-8px}.dataTables_scrollBody::-webkit-scrollbar{display:none}.dataTables_scrollBody{padding-top:2px;-ms-overflow-style:none;scrollbar-width:none}table{font-size:15px;line-height:.8;font-weight:700;width:100%!important;margin:0 auto;clear:both;border-collapse:collapse;table-layout:fixed;word-wrap:break-word;overflow:hidden}thead{font-size:14px;background-color:rgba(0,0,0,.5)}tbody td,thead th{padding:8px 0;line-height:1em;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}tbody .odd{background-color:rgba(0,0,0,.1)}tbody .even{background-color:rgba(0,0,0,.2)}tbody>tr:hover{-webkit-transition:.2s;transition:.2s;background-color:rgba(0,0,0,.4)}tbody tr:hover>.sorting_1,tbody tr>.sorting_1{background-color:rgba(0,0,0,.1)}th.sorting,th.sorting_asc,th.sorting_desc{cursor:pointer}th.sorting:hover,th.sorting_asc:hover,th.sorting_desc:hover{-webkit-transition:.1s;transition:.2s;color:#d6d6d6}.dataTables_empty{display:none}.icon{font-size:20px;line-height:0}.username{text-shadow:.05em 0 .05em #fff}.expand{animation:expand .3s ease-in-out}@keyframes expand{from{transform:scale(0);opacity:0}}.ui-draggable .ui-dialog-titlebar{cursor:move}.ui-draggable-handle{-ms-touch-action:none;touch-action:none}.ui-resizable{position:relative}.ui-resizable-handle{position:absolute;font-size:.1px;touch-action:none}.ui-resizable-autohide .ui-resizable-handle,.ui-resizable-disabled .ui-resizable-handle{display:none}.ui-resizable-n{cursor:n-resize;height:12px;width:100%;top:-5px;left:0}.ui-resizable-s{cursor:s-resize;height:12px;width:100%;bottom:0;left:0}.ui-resizable-e{cursor:e-resize;width:12px;height:100%;right:-3px;top:0}.ui-resizable-w{cursor:w-resize;width:12px;height:100%;left:0;top:0}.ui-resizable-se{cursor:se-resize;width:12px;height:12px;right:0;bottom:0}.ui-resizable-sw{cursor:sw-resize;width:12px;height:12px;left:0;bottom:0}.ui-resizable-nw{cursor:nw-resize;width:12px;height:12px;left:0;top:0}.ui-resizable-ne{cursor:ne-resize;width:12px;height:12px;right:-3px;top:-3px}.dt-buttons{position:absolute;top:0;margin-top:16px;width:100%;text-align:left}.dt-button{vertical-align:middle;font-size:14px;cursor:pointer;border-radius:5px;border:1px solid #000;background-size:200% auto;background-image:linear-gradient(to right,#2e2e2e 0,#454545 51%,#2e2e2e 100%);transition:.3s;-webkit-transition:.3s}.dt-button:hover{background-position:right center;box-shadow:2px 2px 5px -2px #000}.dt-button-collection{margin-top:-29px;position:absolute;padding-left:45px}.dt-button-collection div{background-color:#333;border-radius:8px;padding:4px 6px 4px 6px}.buttons-columnVisibility{color:#fff;padding:3px 7px}.buttons-columnVisibility.active{background-image:linear-gradient(to right,#1cd997 0,#33b09b 51%,#1cd997 100%)}.colvis-btn{width:40px;height:25px}.scrollBtn{text-align:center;width:38px;height:23px;margin-right:4px;float:left}.scrollBtn label span{margin-top:2px;height:100%;display:block;cursor:pointer}.scrollBtn label input{position:absolute;top:-50px}.scrollBtn input:checked+span{color:#1cd997}#switchContainer{margin-top:-9px;margin-left:auto;z-index:99;position:relative;display:inline-block;width:37px;height:21px;-webkit-transition:.1s;transition:.1s}#switchContainer:hover{box-shadow:2px 2px 7px -2px #000}#switchContainer input{display:none}.slider{position:absolute;cursor:pointer;top:0;left:0;right:0;bottom:0;border-radius:4px;background-color:#e04352;-webkit-transition:.1s;transition:.1s}.slider:before{position:absolute;content:"";height:16px;width:16px;left:2px;bottom:3px;border-radius:3px;background-color:#fff;-webkit-transition:.2s;transition:.2s}input:checked+.slider{background-color:#1cd997;box-shadow:2px 2px 7px -2px #000}input:checked+.slider:before{-webkit-transform:translateX(17px)}#scrollSpeedSlider{display:none;height:3px;padding:0;-webkit-appearance:none;width:calc(100% - 12px);background:#fff;outline:0;opacity:.2;-webkit-transition:.3s;transition:opacity .3s;direction:rtl}#scrollSpeedSlider:hover{opacity:1}#scrollSpeedSlider::-webkit-slider-thumb{-webkit-appearance:none;appearance:none;width:30px;height:7px;background:#63db85;cursor:pointer}#scrollSpeedSlider::-moz-range-thumb{width:30px;height:7px;background:#63db85;cursor:pointer}.btn{margin:30px;padding:5px;background-color:#22da8d;color:#fff;font-weight:700}.btn:hover{background-color:#18cc81}.btn:active{background-color:#15b472}`
head.appendChild(styles);
hijackMap();
const scoreboardContainer = document.createElement("div");
scoreboardContainer.setAttribute("id", "scoreboardContainer");
scoreboardContainer.innerHTML = `
# |
Player |
Streak |
Distance |
Score |
`;
document.body.appendChild(scoreboardContainer);
const script1 = document.createElement('script')
script1.src = 'https://twitchbot.helcel.net/common/reconnecting-websocket/reconnectingWS.min.js';
const script2= document.createElement('script')
script2.src = 'https://twitchbot.helcel.net/common/js/wsConfig.js';
const script3 = document.createElement('script')
script3.src ='https://twitchbot.helcel.net/custom/js/socketWrapper.js'
document.body.appendChild(script1)
document.body.appendChild(script2)
document.body.appendChild(script3)
script3.addEventListener('load', async () => {
while(!window.socket){await new Promise(r => setTimeout(r, 500))}
socket = window.socket;
if(socket){
socket.addFamilyHandler("chatguessr", (e)=>{
try {
let rawMessage = e.data, message = JSON.parse(rawMessage);
if(!message.hasOwnProperty('eventFamily') || message.eventFamily != 'chatguessr' ||
!message.hasOwnProperty('eventType') || !message.hasOwnProperty('data')) return;
if(message.eventType == 'guesses') {
const gd = JSON.parse(message.data)
guesses = gd;
if(gd.round!=undefined && gd[gd.round] && gd.rounds){
clearMarkers();
renderGuesses(gd[gd.round]);
if(gameState=="result"){
populateMap(gd.rounds[gd.round],gd[gd.round])
}
}else if(gd.round!= undefined && gd[gd.round - 1]){
clearMarkers();
renderGuesses([]);
}
}
} catch (ex) {console.log(ex)}
});
while(socket.getReadyState() === 0){await new Promise(r => setTimeout(r, 500))}
socket.sendCommand('cga_gg', 'cga gg', ()=>{});
var ccc = '';
setInterval(async()=>{
if(ccc != $('#__next > div').attr('class')){
if(GameHelper.isGameURL(window.location.href)){
$('#scoreboardContainer').show();
if(isInGame){
socket.sendCommand('cg_refresh', 'cga refresh', ()=>{});
}else {
isInGame = true;
socket.sendCommand('cga_start', 'cga start '+GameHelper.getGameId(window.location.href), ()=>{});
handleGameState();
}
isInGame = true;
}else if(isInGame){
isInGame = false;
socket.sendCommand('cga_end', 'cga end', ()=>{});
$('#scoreboardContainer').hide();
}else{
$('#scoreboardContainer').hide();
}
}
ccc = $('#__next > div').attr('class')
}, 1000);
}
socket.getDBValue('cga_init_btn', 'cgstatus', 'isOpen', (v)=>$('#switchBtn').prop("checked","true"==v.cgstatus))
$('#switchBtn').change(()=>{
if($('#switchBtn').prop('checked')) socket.sendCommand('cga_open_btn', 'cga open', ()=>{});
else socket.sendCommand('cga_close_btn', 'cga close', ()=>{});
});
table = $("#datatable").DataTable({
info: false,
searching: false,
autoWidth: true,
paging: false,
scrollY: 166,
scrollResize: true,
scrollCollapse: true,
language: { zeroRecords: " " },
dom: "Bfrtip",
buttons: [],
columns: [
{ data: "Position" },
{ data: "Player" },
{ data: "Streak" },
{
data: "Distance",
render: (data, type) => ((type === "display" || type === "filter")? toMeter(data) : data)
},
{ data: "Score" },
],
columnDefs: [
{ targets: 0, width: "35px"},
{ targets: 1, width: "auto"},
{ targets: 2, width: "55px" },
{ targets: 3, width: "100px" },
{ targets: 4, width: "75px", type: "natural" },
],
});
});