// ==UserScript==
// @name Cookie Clicker Mod Menu
// @namespace https://github.com/qba210/cookie-clicker-mod-menu
// @version 1.1
// @description Mod menu for Cookie Clicker
// @author qba210
// @license MIT
// @match *://orteil.dashnet.org/cookieclicker/*
// @icon 
// @grant none
// ==/UserScript==
(async function() {
'use strict';
class Logger {
/**
*
* @param {object} msg
*/
static LogInfo(msg) {
console.log("\n[Mod Menu] ", msg);
}
/**
*
* @param {object} msg
*/
static LogDebug(msg) {
Logger.LogInfo(msg);
}
}
Logger.LogInfo("Waiting for Game...")
let gamePromise = new Promise((res) => {var int = setInterval(() => {if (Game.Achievements) {clearInterval(int); res(Game);}}, 10)});
await gamePromise;
Logger.LogInfo("Game loaded!")
let cheatMenu = document.createElement("div");
const translations = [
{
lang: "pl",
langName: "Polski",
hacks: {
"silent-mode": {
name: "Tryb cichy",
desc: "Unika wykrycia przez grę hackowania (np. osiągnięcie 'Oszukane ciastka smakują najgorzej')"
},
"cookie-spam": {
name: "Spamowanie ciastka",
desc: "Po przyciśnięciu ciastka samoczynnie zaczyna na nie klikać (kończy po odciśnięciu)"
},
"autoclicker": {
name: "Autokliker",
desc: "Auttomatycznie klika w ciastko"
},
"dev-tools": {
name: "Menu dewelop.",
desc: "Otwiera menu deweloperskie"
},
"set-cookies": {
name: "Ustaw ilość ciastek",
desc: "Pozwala na zmianę ilości ciastek"
},
"delete-save": {
name: "Usuń zapis"
},
"earn-achievement": {
name: "Odblokuj osiągnięcie",
desc: "Wybierz osiągnięcie z listy i kliknij przycisk aby je zdobyć!"
},
"revoke-achievement": {
name: "Usuń osiągnięcie",
desc: "Wybierz osiągnięcie z listy i kliknij przycisk aby je usunąć!"
},
"finish-game": {
name: "Ukończ grę",
desc: "Kończy grę i odblokowywuje wszystko"
}
},
strings: {
"dev-tools-confirm": "Jesteś pewien? Tryb cichy nie uchroni Cię od hacków które włączycz w menu deweloperskim.",
"changes-as-you-type": "Wartość zmienia się jak piszesz",
"confirm-save-delete": "Jesteś pewien?",
"set-object-amount": "Ustaw ilość %s"
}
},
{
lang: "en",
langName: "English",
hacks: {
"silent-mode": {
name: "Silent mode",
desc: "Avoids detecting hacks by game (ex.: achievement 'Cheated cookies tastes awful')"
},
"cookie-spam": {
name: "Cookie spam",
desc: "After holding the cookie, starts to click it automatically (ends after releasing)"
},
"autoclicker": {
name: "Autoclicker",
desc: "Automatically clicks cookie"
},
"dev-tools": {
name: "Dev menu",
desc: "Opens developer menu"
},
"set-cookies": {
name: "Set cookie count",
desc: "Allows you to change cookie count"
},
"delete-save": {
name: "Delete save"
},
"earn-achievement": {
name: "Earn achievement",
desc: "Select achievement from list, then click button to get it!"
},
"revoke-achievement": {
name: "Remove achievement",
desc: "Select achievement from list, then click button to remove it!"
},
"finish-game": {
name: "Finish game",
desc: "Finishes game and unlocks everything"
}
},
strings: {
"dev-tools-confirm": "Are you sure? Silent mode cannot prevent detecting cheats you activate in developer menu.",
"changes-as-you-type": "Changes as you type",
"confirm-save-delete": "Are you sure?",
"set-object-amount": "Set %s amount"
}
}
]
let lang = translations.find((lng) => lng.lang === (localStorage.getItem("cheats_lang") ?? "en"));
document.body.append(cheatMenu);
cheatMenu.outerHTML = `
<div id="hack-menu" style="left: 50%; top: 35px;">
<div id="hack-popup">Hack</div>
<br>
<div id="hacks" style="display: none;">
<div class="hack-list" id="main-hacks">
<select id="hack-lang-select"></select>
<div class="hack hack-bool" id="hack-silent-mode"></div>
<div class="hack hack-bool" id="hack-cookie-spam"></div>
<div class="hack hack-bool" id="hack-autoclicker"></div>
<div class="hack hack-btn" id="hack-dev-tools"></div>
<div class="hack hack-btn" id="hack-set-cookies"></div>
<div class="hack hack-btn" id="hack-delete-save"></div>
<div class="hack hack-btn" id="hack-finish-game"></div>
<div class="hack hack-select" id="hack-earn-achievement"></div>
<div class="hack hack-select" id="hack-revoke-achievement"></div>
</div>
<div class="hack-list" id="objects-hacks">
</div>
</div>
</div>
<div id="hack-tooltip" style="opacity: 0;left:0;top:0"></div>
<div id="hack-alert-input" style="opacity: 0;display: none;">
<div id="hack-alert-input-popup">
<h1 id="hack-alert-input-popup-title"></h1>
<div id="hack-alert-input-popup-desc"></div><br>
<input type="number" id="hack-alert-input-popup-input"/><br>
<input type="button" id="hack-alert-input-popup-ok" value="OK"/>
</div>
</div>
`;
let styles = document.createElement("style");
styles.innerText = `
#hack-menu, #hack-tooltip {
z-index: 9000000000;
position: absolute;
box-sizing: border-box;
}
#hack-menu, #hack-menu *:not(#hacks){
box-sizing: border-box;
}
#hack-menu {
display: flex;
flex-direction: column;
align-items: center;
width: clamp(150px, 225px, 25vw);
}
#hacks {
backdrop-filter: blur(5px);
display: flex;
flex-direction: row;
gap: 10px;
padding: 10px;
background-color: rgba(0, 0, 0, .5);
border-radius: 13px;
width: 200%;
box-sizing: content-box;
}
.hack-list {
display: flex;
flex-direction: column;
gap: 10px;
padding: 10px;
width: 100%;
box-sizing: content-box;
overflow-y: auto;
overflow-x: hidden;
max-height: 70vh;
}
.hack-list::-webkit-scrollbar-thumb {
border: 4px solid rgb(10, 87, 242);
background-color: rgba(10, 87, 242, .2);
border-radius: 5px;
backdrop-filter: blur(3px);
box-shadow: none;
transition: background-color .3s linear, border .3s linear
}
.hack-list::-webkit-scrollbar-thumb:hover {
border: 4px solid #00a2ff;
background-color: rgba(0, 162, 255, .4);
}
.hack-list::-webkit-scrollbar-track {
border: 4px solid black;
background-color: transparent;
border-radius: 5px;
}
#hack-popup, .hack {
padding: 10px;
font-size: 20px;
background-color: #00a2ff;
border: 7px solid black;
text-align: center;
width: 100%;
}
.hack {
cursor: pointer;
}
.hack:active {
filter: brightness(0.85);
}
.hack-bool {
transition: background-color .3s ease-in;
background-color: red;
}
.hack-bool[on] {
background-color: green;
}
#hack-tooltip {
position: absolute;
backdrop-filter: blur(7px);
filter: blur(0px);
transition: all .3s ease-out, left 0s ease, top 0s ease;
background-color: rgba(0, 0, 0, .5);
color: white;
transform: translate(15px, 9.5px);
width: 200px;
padding: 10px;
border-radius: 13px;
pointer-events: none;
}
#hack-lang-select {
z-index: 90000000001;
background-color: black;
color: white;
border-color: white;
width: 100%
}
#hack-lang-select > option {
color: white;
}
#hack-popup {
transition: opacity .35s ease-out;
}
#hack-popup:hover, #hack-popup[open] {
opacity: 1;
}
#hack-popup:not(:hover):not([open]) {
opacity: .8;
}
#hack-alert-input {
z-index: 9500000000;
background-color: rgba(0, 0, 0, .5);
transition: opacity .3s ease-out;
position: absolute;
left: 0;
top: 0;
right: 0;
bottom: 0;
backdrop-filter: blur(5px);
}
#hack-alert-input-popup {
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
width: 60%;
height: 50%;
background-color: black;
color: white;
display: flex;
flex-direction: column;
justify-content: center;
font-family: verdana;
text-align: center;
padding: 20px;
border: 3px solid white;
border-radius: 10px;
}
#hack-alert-input-popup-title {
font-size: 2.5em;
}
#hack-alert-input-popup-desc {
font-size: 1.5em;
}
#hack-alert-input-popup-ok {
background-color: #00a2ff;
border: 4px solid white;
text-align: center;
width: 100%;
height: 50px;
font-weight: bold;
cursor: pointer;
}
#hack-alert-input-popup-input {
background-color: black;
color: white;
border-color: white;
}
.select-in-hack {
width: 100%;
border-color: black;
color: white;
background-color: black;
}
`;
document.head.append(styles);
if (!localStorage.getItem("cheats_lang")) {
localStorage.setItem("cheats_lang", "en");
}
let $hackmenu = document.getElementById("hack-menu");
let $hacks = document.getElementById("hacks");
let $popup = document.getElementById("hack-popup");
let $tooltip = document.getElementById("hack-tooltip");
/**@type {HTMLSelectElement} */
let $langselect = document.getElementById("hack-lang-select");
let $hack_earnachievement = document.getElementById("hack-earn-achievement")
let $hack_revokeachievement = document.getElementById("hack-revoke-achievement")
let $inputalert = {
alert: document.getElementById("hack-alert-input"),
popup: document.getElementById("hack-alert-input-popup"),
title: document.getElementById("hack-alert-input-popup-title"),
desc: document.getElementById("hack-alert-input-popup-desc"),
input: document.getElementById("hack-alert-input-popup-input"),
ok: document.getElementById("hack-alert-input-popup-ok")
}
let $objecthacks = document.getElementById("objects-hacks")
let $cookie = document.getElementById("bigCookie");
dragElement($hackmenu, $popup);
//boolean hack manager
$hacks.querySelectorAll(".hack-bool").forEach(node => {
node.addEventListener("click", (e) => {
if (node.hasAttribute("on")) {
node.removeAttribute("on");
} else {
node.setAttribute("on", "");
}
})
})
//populate object hacks
gamePromise.then(() => {
Object.entries(Game.Objects).forEach(entry => {
let [key, value] = entry;
let hack = document.createElement("div");
hack.className = "hack hack-btn";
hack.id = `hack-object-amount-${value.bsingle}`;
hack.setAttribute("lang-string", "set-object-amount");
hack.setAttribute("supply-val", value.dname);
hack.setAttribute("key", key);
hack.onclick = (e) => {
showInputAlert(hack.innerText, lang.strings["changes-as-you-type"], "number", value.amount, (val) => {
value.getFree(+val - value.amount)
value.free = 0;
})
}
$objecthacks.append(hack);
})
// update language after
reloadLangs();
})
reloadLangs();
//class loop
$hackmenu.querySelectorAll(".hack").forEach(_node => {
/**@type {HTMLElement} */
let node = _node;
//tooltip
node.addEventListener("mousemove", function (e) {
if (e.target === this) {
showTooltip(node.id, e.pageX, e.pageY);
} else if (e.target instanceof HTMLSelectElement) {
if (e.target.value && e.target.value !== "") {
showTooltip(Array.from(this.querySelectorAll("option")).find(opt => opt.value === e.target.value).innerHTML, e.pageX, e.pageY, false);
}
} else {
hideTooltip();
}
})
node.addEventListener("mouseleave", (e) => {
hideTooltip();
})
})
//menu hide/show
$popup.addEventListener("click", (e) => {
if ($hacks.style.display === "none") {
$hacks.style.display = "";
$popup.setAttribute("open", "");
} else {
$hacks.style.display = "none";
$popup.removeAttribute("open");
}
})
//add event listener to language select
$langselect.addEventListener("change", (e) => {
setLang($langselect.value);
})
//select language
translations.forEach((translation) => {
let option = document.createElement("option");
option.value = translation.lang;
option.innerText = translation.langName;
$langselect.append(option);
Logger.LogInfo(`Loaded language ${translation.lang}: ${translation.langName}`);
})
//set current language
$langselect.value = lang.lang;
// Debug menu hack
document.getElementById("hack-dev-tools").addEventListener("click", (e) => {
if (isSilentMode()) {
if (confirm(lang.strings["dev-tools-confirm"])) {
Game.OpenSesame();
}
} else {
Game.OpenSesame();
}
})
// Cookie spam hack
let cookieSpamWorker;
$cookie.addEventListener("mousedown", (e) =>
{
if (document.getElementById("hack-cookie-spam").hasAttribute("on"))
cookieSpamWorker = setInterval(() => $cookie.dispatchEvent(new Event("click")));
})
$cookie.addEventListener("mouseup", (e) => {
clearInterval(cookieSpamWorker);
})
// Autoclicker hack
let autoclickerWorker;
document.getElementById("hack-autoclicker").addEventListener("click", (e) =>
{
let cookrect = $cookie.getBoundingClientRect();
if (document.getElementById("hack-autoclicker").hasAttribute("on"))
autoclickerWorker = setInterval(() => $cookie.dispatchEvent(new Event("click")), 0);
else
clearInterval(autoclickerWorker);
})
// Set cookies hack
document.getElementById("hack-set-cookies").addEventListener("click", (e) => {
//show input to user
showInputAlert(lang.hacks["set-cookies"].name, lang.strings["changes-as-you-type"], "number", Game.cookies.toString(), (val) => {
//earn mode decided by silent mode
if (isSilentMode()) {
Game.Earn(-Game.cookies + +val);
} else {
Game.cookies = +val;
}
})
});
// Delete save hack
document.getElementById("hack-delete-save").addEventListener("click", (e) => {
//ask user to confirm
if (confirm(lang.strings["confirm-save-delete"])) {
//delete save from localstorage
localStorage.removeItem("CookieClickerGame");
//after this reload to confirm
window.location.reload();
}
});
//earn achievement hack
$hack_earnachievement.addEventListener("click", function (e) {
//if clicked on select dont gain achevement
if (e.target !== this) return;
//gain selected achievement
Game.Win($hack_earnachievement.querySelector("select").value)
})
//revoke achievement hack (simillar to earn)
$hack_revokeachievement.addEventListener("click", function (e) {
if (e.target !== this) return;
Game.RemoveAchiev($hack_revokeachievement.querySelector("select").value)
})
gamePromise.then(() => {
document.getElementById("hack-finish-game").addEventListener("click", (e) => {
Game.RuinTheFun(true);
})
})
/**
* @param {string} lang
*/
function setLang(lang) {
localStorage.setItem("cheats_lang", lang);
reloadLangs();
}
function reloadLangs() {
lang = translations.find((lng) => lng.lang === (localStorage.getItem("cheats_lang") ?? "en"))
//for every hack
$hackmenu.querySelectorAll(".hack").forEach(node => {
//add value
/**@type {string} */
let transval = (lang.hacks[node.id.replace("hack-", "")] ?? {name: node.id} ).name;
if (node.hasAttribute("lang-string")) {
transval = lang.strings[node.getAttribute("lang-string")];
}
if (node.hasAttribute("supply-val")) {
transval = transval.replace("%s", node.getAttribute("supply-val"));
}
node.innerText = (transval ?? node.id ) ?? node.id
})
//support for select class
$hackmenu.querySelectorAll(".hack-select").forEach(node => {
let select = document.createElement("select");
select.className = "select-in-hack";
node.append(select);
})
// Earn achevement hack
//when Game is avaible do
gamePromise.then(() => {
// foreach achevement ingame
Object.entries(Game.Achievements).forEach((entry) => {
const [key, value] = entry;
//create option with achievement
let option = document.createElement("option");
option.value = key;
option.innerHTML = value.dname;
//add it to hack
$hack_earnachievement.querySelector("select").append(option);
$hack_revokeachievement.querySelector("select").append(option.cloneNode(true));
});
})
//set current language
$langselect.value = lang.lang;
}
function isSilentMode() {
return document.getElementById("hack-silent-mode").hasAttribute("on");
}
/**
*
* @param {string} id
* @param {number} x
* @param {number} y
*/
function showTooltip(id, x, y, translatable = true) {
//show only if id have description
if (translatable) {
if (!lang.hacks[id.replace("hack-", "")]) return;
if (!lang.hacks[id.replace("hack-", "")].desc) return;
}
$tooltip.innerText = translatable ? lang.hacks[id.replace("hack-", "")].desc ?? "" : id;
$tooltip.style.left = x.toString() + "px";
$tooltip.style.top = y.toString() + "px";
$tooltip.style.opacity = 1;
}
function hideTooltip() {
//$tooltip.innerText = ""; //its needed to be comented because transistion
$tooltip.style.opacity = 0;
}
/**
* Shows input alert.
*
* @param {string} title
* @param {string} desc
* @param {string} type
* @param {string} value
* @param {(value: string) => void} setter
*/
function showInputAlert(title, desc, type, value, setter) {
$inputalert.ok.onclick = () => {
setTimeout(() => $inputalert.alert.style.display = "none", 301);
setter($inputalert.input.value);
$inputalert.alert.style.opacity = "0";
}
$inputalert.title.innerText = title;
$inputalert.desc.innerText = desc;
$inputalert.input.type = type;
$inputalert.input.value = value.toString();
$inputalert.input.onchange = (e) => setter($inputalert.input.value);
$inputalert.alert.style.display = "";
$inputalert.alert.style.opacity = "1";
}
/**
*
* @param {HTMLElement} elmnt element to drag
* @param {HTMLElement?} dragger element dragging elmnt
*/
function dragElement(elmnt, dragger) {
var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
if (document.getElementById(elmnt.id + "header") || dragger) {
// if present, the header is where you move the DIV from:
(document.getElementById(elmnt.id + "header") ?? dragger).onmousedown = dragMouseDown;
} else {
// otherwise, move the DIV from anywhere inside the DIV:
elmnt.onmousedown = dragMouseDown;
}
function dragMouseDown(e) {
e = e || window.event;
e.preventDefault();
// get the mouse cursor position at startup:
pos3 = e.clientX;
pos4 = e.clientY;
document.onmouseup = closeDragElement;
// call a function whenever the cursor moves:
document.onmousemove = elementDrag;
if (document.getElementById(elmnt.id + "header") || dragger) {
(document.getElementById(elmnt.id + "header") ?? dragger).setAttribute("dragging", "");
}else {
elmnt.setAttribute("dragging", "");
}
}
function elementDrag(e) {
e = e || window.event;
e.preventDefault();
// calculate the new cursor position:
pos1 = pos3 - e.clientX;
pos2 = pos4 - e.clientY;
pos3 = e.clientX;
pos4 = e.clientY;
// set the element's new position:
elmnt.style.top = (elmnt.offsetTop - pos2) + "px";
elmnt.style.left = (elmnt.offsetLeft - pos1) + "px";
}
function closeDragElement() {
// stop moving when mouse button is released:
document.onmouseup = null;
document.onmousemove = null;
if (document.getElementById(elmnt.id + "header") || dragger) {
(document.getElementById(elmnt.id + "header") ?? dragger).removeAttribute("dragging");
}else {
elmnt.removeAttribute("dragging");
}
}
}
})();