Skip to content

Instantly share code, notes, and snippets.

@Karmalakas
Last active February 11, 2025 18:21
Show Gist options
  • Save Karmalakas/1b63e946d3b3a3f393bb2d5d5cf19b1e to your computer and use it in GitHub Desktop.
Save Karmalakas/1b63e946d3b3a3f393bb2d5d5cf19b1e to your computer and use it in GitHub Desktop.
GuruShots mini game info
// ==UserScript==
// @name GuruShots mini-game info
// @description Show some info for each photo in a mini game popup
// @namespace http://karmalakas.lt/
// @version 1.0.2
// @author Karmalakas
// @updateURL https://github.com/Karmalakas/gurushots-user-scripts/raw/refs/heads/main/script/mini-game-info/script.user.js
// @downloadURL https://github.com/Karmalakas/gurushots-user-scripts/raw/refs/heads/main/script/mini-game-info/script.user.js
// @supportURL https://github.com/Karmalakas/gurushots-user-scripts/issues
// @match https://gurushots.com/*
// @run-at document-idle
// @grant GM_addStyle
// ==/UserScript==
(function() {
'use strict'
GM_addStyle('' +
'.GS__mini_game_image_info {' +
' z-index: 9999;' +
' position: absolute;' +
' padding: 8px;' +
' margin: 8px;' +
' font-weight: 600;' +
' border-radius: 22px 8px 8px;' +
' color: whitesmoke;' +
' background-color: rgba(0, 0, 0, 0.5);' +
'}'
);
const mutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver;
const observerOptions = {'childList': true, 'subtree': true, 'attributeFilter': ['class']};
let gameComponentElement;
async function request(url, data)
{
const resp = await (await fetch(url, {
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
'x-api-version': 13,
'x-env': 'WEB',
'x-requested-with': 'XMLHttpRequest',
'x-token': ((name) => {
var value = "; " + document.cookie;
var parts = value.split("; " + name + "=");
if (parts.length == 2) return parts.pop().split(";").shift();
})('gs_t')
},
method: 'POST',
body: new URLSearchParams(Object.entries(data)).toString()
})).json();
if (resp.success !== true) {
return null;
}
return resp.data;
}
async function processMiniGame(response, post)
{
gameComponentElement = document.querySelector('modal-mini-game');
const images = {};
for (const battle of response.images) {
if (
battle.is_success !== null
|| (
typeof images[battle.first_image.id] !== 'undefined'
&& typeof images[battle.second_image.id] !== 'undefined'
)
) {
continue;
}
images[battle.first_image.id] = await request('https://api.gurushots.com/rest/get_image_data', {id: battle.first_image.id});
images[battle.second_image.id] = await request('https://api.gurushots.com/rest/get_image_data', {id: battle.second_image.id});
}
initializeMiniGame(images);
}
function initializeMiniGame(imagesData)
{
fillImagesData(imagesData);
mutationObserver && new MutationObserver(function(mut, observer) {
for (const mutation of mut) {
if (Array.from(mutation.target.classList).includes('mini-game-voting-wrapper')) {
observer.disconnect();
fillImagesData(imagesData);
observer.observe(gameComponentElement, observerOptions);
return;
}
}
}).observe(gameComponentElement, observerOptions);
}
function fillImagesData(imagesData)
{
fillImageData(gameComponentElement.querySelector('app-mini-game-voting .image-first'), imagesData);
fillImageData(gameComponentElement.querySelector('app-mini-game-voting .image-second'), imagesData);
}
function fillImageData(imageHolder, imagesData)
{
const imageSrc = imageHolder.querySelector('app-ng-image.is-image-visible img').src;
const regex = /_(.*)\.jpg$/g;
const imageId = Array.from(imageSrc.matchAll(regex))[0][1];
const data = imagesData[imageId];
const achievementsCount = getAchievementsCount(data.achievements);
const el = document.createElement('div');
el.className = 'GS__mini_game_image_info';
el.innerHTML = `
<div>πŸ‘¨${data.member.member_status_name} | πŸ†${achievementsCount}</div>
<div>βœ”${data.votes} βˆ• πŸ‘${data.views} (%${Math.round((data.votes/data.views + Number.EPSILON) * 100) / 100})</div>
`;
const overlay = imageHolder.querySelector('.vote-result-overlay');
overlay.querySelector('.GS__mini_game_image_info')?.remove();
overlay.prepend(el);
}
function getAchievementsCount(achievements)
{
let total = 0;
for (const achivementType of achievements) {
if (
achivementType.unique_key.startsWith('CHALLENGE_LEVEL_ACHIEVEMENT.NO_PARAM')
|| achivementType.unique_key.startsWith('TOP_PHOTOGRAPHER_ACHIEVEMENT.PERCENT')
|| achivementType.unique_key.startsWith('TOP_PHOTO_ACHIEVEMENT.PERCENT')
) {
continue;
}
total += achivementType.count;
}
return total;
}
(function() {
const origSend = XMLHttpRequest.prototype.send;
XMLHttpRequest.prototype.send = function(postBody) {
this.addEventListener('load', function() {
if (this.responseURL === "https://api.gurushots.com/rest/get_challenge_turbo") {
processMiniGame(JSON.parse(this.response), new URLSearchParams(postBody));
}
});
origSend.apply(this, arguments);
};
})();
})();
@Karmalakas
Copy link
Author

Karmalakas commented Jul 22, 2024

Caution

This gist was moved to a separate repo and will no longer be updated
Please visit the new update URL to re-install to the latest version or wait for updates to appear on TamperMonkey


This script shows how many points each team gets in the battle, based on current vote count.

See images bellow.


This only works with TamperMonkey (or similar) extension. Supported browsers are Chrome, Firefox, Edge (maybe others too).

TamperMonkey for:

Once you have TamperMonkey installed, just go to the @downloadURL you see in a script.
If later TM fails to update script, just go to @updateURL you see in a script. This should trigger the update in a new window with TM open.

Any questions or suggestions are welcome.


If you like this script, please consider keeping me scripting Keep me scripting πŸ˜„

@Karmalakas
Copy link
Author

Karmalakas commented Jul 22, 2024

Info of the image includes:

  • Author level
  • All time achievements (excl. level and percentage; e. g. All Star, Elite, Top Photo 20%, etc.)
  • All time votes
  • All time views
  • All time ratio

You have to wait for a few seconds to load images info when popup is opened

Screenshot 2024-07-23 003348

!!! This does not mean, that better photo actually will win the battle for the specific challenge !!!

@mateussfernandess
Copy link

This helps a lot! @Karmalakas, Thank you for putting the time into this.

I would love if it was possible to have this ratio and rank in all photos of the site, not just in the game, it would be hard to create something like this out of this script?

@Karmalakas
Copy link
Author

Karmalakas commented Feb 11, 2025

I suppose it's possible, but I don't see any poin in that. Here it helps to decide which photo to choose, but elsewhere..? πŸ€”
I don't think I will be looking into this myself, but feel free to submit any changes on the new main repo (where I moved all of my GS scripts) πŸ˜‰

@Karmalakas
Copy link
Author

Caution

This gist was moved to a separate repo and will no longer be updated
Please visit the new update URL to re-install to the latest version or wait for updates to appear on TamperMonkey

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment