Skip to content

Instantly share code, notes, and snippets.

@notnotrishi
Last active November 14, 2025 07:03
Show Gist options
  • Select an option

  • Save notnotrishi/47dbe8088a614effec71e6a22a7af0ab to your computer and use it in GitHub Desktop.

Select an option

Save notnotrishi/47dbe8088a614effec71e6a22a7af0ab to your computer and use it in GitHub Desktop.
(() => {
// Detect environment
const detectEnvironment = () => {
const ua = navigator.userAgent;
const isVSCode = ua.includes('Visual Studio Code') ||
typeof acquireVsCodeApi !== 'undefined' ||
window.vscodeApi !== undefined;
const isElectron = ua.includes('Electron') || typeof require !== 'undefined';
const isChrome = /Chrome/.test(ua) && !/Edge/.test(ua);
const isFirefox = /Firefox/.test(ua);
const isSafari = /Safari/.test(ua) && !/Chrome/.test(ua);
return { isVSCode, isElectron, isChrome, isFirefox, isSafari, ua };
};
const ENV = detectEnvironment();
const SHOW_PULSING_ICON = ENV.isVSCode; // Only show in VS Code webviews
const isDark = typeof THEME !== "undefined" ? THEME === "dark" : true;
const COLORS = {
text: isDark ? "#0aae5cff" : "#006644",
boxBg: isDark ? "rgba(17,17,17,0.82)" : "rgba(255,255,255,0.82)",
boxBorder: isDark ? "#0aae5cff" : "#006644",
launcherBg: isDark ? "#111" : "#ddd",
launcherText: isDark ? "#0aae5cff" : "#006644"
};
// ๐ŸŽฎ launcher button (bottom-left)
const launcher = document.createElement("div");
launcher.textContent = "๐ŸŽฎ";
Object.assign(launcher.style, {
position: "fixed",
bottom: "10px",
left: "10px",
background: COLORS.launcherBg,
color: COLORS.launcherText,
padding: "6px 10px",
borderRadius: "8px",
cursor: "pointer",
zIndex: 999999,
fontFamily: "Consolas, Courier New, monospace",
fontSize: "14px",
boxShadow: "0 2px 6px rgba(0,0,0,0.3)",
userSelect: "none",
transition: "transform 0.15s ease, opacity 0.2s ease"
});
document.body.appendChild(launcher);
// ๐Ÿงฉ centered pulsing icon overlay
const icon = document.createElement("img");
icon.src = ""
// Ensure this path points to a valid icon image
Object.assign(icon.style, {
position: "fixed",
top: "50%",
left: "50%",
width: "120px",
height: "120px",
transform: "translate(-50%, -50%)",
opacity: "0.9",
zIndex: 999997,
pointerEvents: "none",
transition: "opacity 0.6s ease"
});
document.body.appendChild(icon);
// Pulse animation
const style = document.createElement("style");
style.textContent = `
@keyframes pulseIcon {
0%, 100% { transform: translate(-50%, -50%) scale(1); opacity: 0.85; }
50% { transform: translate(-50%, -50%) scale(1.1); opacity: 1; }
}
`;
document.head.appendChild(style);
const startPulse = () => {
icon.style.animation = "pulseIcon 2.5s ease-in-out infinite";
};
const stopPulse = () => {
icon.style.animation = "none";
};
let canvas, ctx, w, h, score, over, paddle, brick, animationId;
let gameActive = false;
let keys = {};
let keydownHandler, keyupHandler;
let particles = [];
let closeBtn, countdownTimer, countdown = 3;
let level = 1;
let starting = false;
// --- Single-tab (cross-tab) singleton support ---
const SINGLETON = { enable: true, heartbeatMs: 2000, staleMs: 6000, verifyMs: 120, pingMs: 500 };
const SINGLETON_KEY = "bricks_game_singleton";
const tabId = `${Date.now()}-${Math.random().toString(36).slice(2)}`;
let heartbeatTimer = null;
let otherOwnerAlerted = false;
let lockHeld = false;
let lockCurrentId = null;
const pendingPings = new Map();
const delay = ms => new Promise(r => setTimeout(r, ms));
const bc = 'BroadcastChannel' in window ? new BroadcastChannel('vscode-bricks-singleton') : null;
const now = () => Date.now();
const readLock = () => {
try {
const raw = localStorage.getItem(SINGLETON_KEY);
return raw ? JSON.parse(raw) : null;
} catch {
return null;
}
};
const writeLock = (data) => {
try {
localStorage.setItem(SINGLETON_KEY, JSON.stringify(data));
lockCurrentId = data?.id ?? null;
} catch {}
};
const clearLock = () => {
try {
const cur = readLock();
if (cur && cur.id === tabId) {
localStorage.removeItem(SINGLETON_KEY);
lockCurrentId = null;
}
} catch {}
};
const isStale = (lock) => !lock || (now() - lock.ts) > SINGLETON.staleMs;
const ownerResponds = (lock) => {
if (!bc || !lock?.id) return Promise.resolve(true); // treat as alive if we can't ping
const correlationId = `${tabId}-${Math.random().toString(36).slice(2)}`;
return new Promise((resolve) => {
const timeout = setTimeout(() => {
pendingPings.delete(correlationId);
resolve(false);
}, SINGLETON.pingMs);
pendingPings.set(correlationId, () => {
clearTimeout(timeout);
resolve(true);
});
try {
bc.postMessage({ type: 'ping', from: tabId, lockId: lock.id, correlationId });
} catch {
clearTimeout(timeout);
pendingPings.delete(correlationId);
resolve(true);
}
});
};
async function claimSingleton() {
if (!SINGLETON.enable) return true;
const current = readLock();
if (current && current.id !== tabId && !isStale(current)) {
if (await ownerResponds(current)) {
flashLauncherLocked();
notifyExisting();
requestShow();
return false;
}
// Someone else actively owns it
flashLauncherLocked();
notifyExisting();
requestShow();
return false;
}
// Optimistic claim -> verify shortly to avoid races
writeLock({ id: tabId, ts: now() });
await delay(SINGLETON.verifyMs);
const after = readLock();
if (!after || after.id !== tabId) {
flashLauncherLocked();
notifyExisting();
requestShow();
return false;
}
startHeartbeat();
lockHeld = true;
return true;
}
function requestShow() {
try { bc?.postMessage({ type: 'show' }); } catch {}
}
function startHeartbeat() {
stopHeartbeat();
lockHeld = true;
heartbeatTimer = setInterval(() => {
const cur = readLock();
if (!cur || cur.id === tabId) {
writeLock({ id: tabId, ts: now() });
}
}, SINGLETON.heartbeatMs);
}
function stopHeartbeat() {
if (heartbeatTimer) {
clearInterval(heartbeatTimer);
heartbeatTimer = null;
}
lockHeld = false;
}
function releaseSingleton() {
stopHeartbeat();
clearLock();
}
function flashLauncherLocked() {
launcher.textContent = "๐Ÿ”’";
launcher.style.transform = "scale(1.2)";
setTimeout(() => (launcher.style.transform = "scale(1)"), 250);
}
function notifyExisting() {
if (otherOwnerAlerted) return;
otherOwnerAlerted = true;
// Gentle notice in place of alert
icon.style.opacity = "1";
icon.title = "Game already running in another tab.";
}
window.addEventListener("storage", (e) => {
if (e.key !== SINGLETON_KEY) return;
const lock = readLock();
if (lock && lock.id !== tabId && gameActive) {
// We lost ownership (another tab took over after we became stale) -> stop locally
stopGame({ showIdle: SHOW_PULSING_ICON });
notifyExisting();
}
});
window.addEventListener("pagehide", releaseSingleton);
window.addEventListener("beforeunload", releaseSingleton);
// --- end singleton support ---
if (bc) {
bc.addEventListener("message", (e) => {
const data = e?.data;
if (!data) return;
if (data.type === "show" && gameActive) {
launcher.style.transform = "scale(1.25)";
setTimeout(() => (launcher.style.transform = "scale(1)"), 220);
} else if (data.type === "ping") {
if (lockHeld && lockCurrentId && data.lockId === lockCurrentId) {
try { bc.postMessage({ type: "pong", to: data.from, correlationId: data.correlationId }); } catch {}
}
} else if (data.type === "pong" && data.to === tabId) {
const resolver = pendingPings.get(data.correlationId);
if (resolver) {
resolver();
pendingPings.delete(data.correlationId);
}
}
});
}
const AudioCtx = window.AudioContext || window.webkitAudioContext;
const audioCtx = new AudioCtx();
const playTone = (freq, duration = 0.1, type = "sine") => {
const osc = audioCtx.createOscillator();
const gain = audioCtx.createGain();
osc.type = type;
osc.frequency.value = freq;
osc.connect(gain);
gain.connect(audioCtx.destination);
osc.start();
gain.gain.setValueAtTime(0.1, audioCtx.currentTime);
gain.gain.exponentialRampToValueAtTime(0.001, audioCtx.currentTime + duration);
osc.stop(audioCtx.currentTime + duration);
};
const playCatchSound = () => {
playTone(600, 0.07, "square");
setTimeout(() => playTone(800, 0.07, "square"), 50);
};
const playGameOverSound = () => {
playTone(300, 0.15, "triangle");
setTimeout(() => playTone(200, 0.15, "triangle"), 100);
};
const playStartChime = () => {
playTone(400, 0.1, "sine");
setTimeout(() => playTone(600, 0.1, "sine"), 100);
setTimeout(() => playTone(800, 0.15, "sine"), 200);
};
const isSpaceKey = e => e.code === "Space" || e.key === " " || e.key === "Spacebar";
const createCloseButton = () => {
if (closeBtn) {
closeBtn.style.display = "block";
closeBtn.style.setProperty("z-index", "2147483647", "important");
try { document.documentElement.appendChild(closeBtn); } catch {}
return closeBtn;
}
closeBtn = document.createElement("div");
closeBtn.textContent = "โœ–๏ธ";
Object.assign(closeBtn.style, {
position: "fixed",
top: "12px",
right: "12px",
fontSize: "20px",
background: "transparent", // Remove background
color: "#fff",
padding: "6px 8px",
borderRadius: "6px",
cursor: "pointer",
zIndex: 2147483647,
userSelect: "none",
fontFamily: "Consolas, Courier New, monospace",
transition: "opacity 0.2s ease",
pointerEvents: "auto"
});
closeBtn.onmouseover = () => (closeBtn.style.opacity = "0.8");
closeBtn.onmouseout = () => (closeBtn.style.opacity = "1");
try { document.documentElement.appendChild(closeBtn); } catch { document.body.appendChild(closeBtn); }
return closeBtn;
};
const removeCloseButton = () => closeBtn?.remove();
const drawTextBox = (textLines) => {
const boxWidth = 370;
const boxHeight = 120;
const boxX = (w - boxWidth) / 2;
const boxY = (h - boxHeight) / 2 - 20;
ctx.fillStyle = COLORS.boxBg;
ctx.fillRect(boxX, boxY, boxWidth, boxHeight);
ctx.strokeStyle = COLORS.boxBorder;
ctx.lineWidth = 2;
ctx.strokeRect(boxX, boxY, boxWidth, boxHeight);
ctx.fillStyle = COLORS.text;
ctx.textAlign = "center";
ctx.textBaseline = "middle";
textLines.forEach((line, i) => {
ctx.font = i === 0 ? "bold 36px Consolas, Courier New, monospace" : "18px Consolas, Courier New, monospace";
ctx.fillText(line.toUpperCase(), w / 2, boxY + 35 + i * 30);
});
};
const showCountdown = () => {
ctx.clearRect(0, 0, w, h);
drawTextBox([String(countdown), "PRESS โœ–๏ธ OR ESC MID-GAME TO EXIT"]);
};
const setupGame = () => {
w = canvas.width = innerWidth;
h = canvas.height = innerHeight;
paddle = { x: w / 2 - 60, y: h - 70, width: 120, height: 24, speed: 15 };
brick = { x: Math.random() * (w - 20), y: 0, size: 20, speed: 4 };
};
const showIcon = () => {
const btn = createCloseButton();
btn.style.display = "none"; // Hide the close button when showing pulsing icon
icon.style.opacity = "0.9";
icon.style.display = "block";
startPulse();
};
const hideIcon = () => {
icon.style.opacity = "0";
icon.style.display = "none";
stopPulse();
};
const startGame = async () => {
if (starting || gameActive) return;
if (SINGLETON.enable) {
const ok = await claimSingleton(); // verified claim
if (!ok) return;
}
starting = true;
launcher.style.transform = "scale(1.1)";
launcher.style.opacity = "0.7";
setTimeout(() => {
launcher.style.transform = "scale(1)";
launcher.style.opacity = "1";
}, 150);
score = 0;
level = 1;
launcher.textContent = "๐ŸŽฎ";
icon.style.display = "none";
gameActive = true;
over = false;
canvas = document.createElement("canvas");
Object.assign(canvas.style, {
position: "fixed",
top: 0,
left: 0,
width: "100vw",
height: "100vh",
background: "rgba(0,0,0,0.01)",
zIndex: 999998
});
document.body.appendChild(canvas);
ctx = canvas.getContext("2d");
addEventListener("resize", setupGame);
createCloseButton();
particles = [];
setupGame();
countdown = 3;
showCountdown();
countdownTimer = setInterval(() => {
countdown--;
if (countdown > 0) {
showCountdown();
} else {
clearInterval(countdownTimer);
playStartChime();
startGameplay();
starting = false;
}
}, 1000);
};
const startGameplay = () => {
keys = {};
keydownHandler = e => {
keys[e.key] = true;
if (isSpaceKey(e)) e.preventDefault();
if (e.key === "Escape") {
score = 0;
launcher.textContent = "๐ŸŽฎ";
stopGame({ showIdle: SHOW_PULSING_ICON });
}
if (over && isSpaceKey(e)) {
stopGame({ showIdle: false });
startGame();
}
};
keyupHandler = e => {
delete keys[e.key];
};
window.addEventListener("keydown", keydownHandler);
window.addEventListener("keyup", keyupHandler);
// when creating the close button
closeBtn.onclick = () => stopGame({ showIdle: true });
const makeParticles = (x, y, color = "#a84637") => {
for (let i = 0; i < 15; i++) {
particles.push({
x,
y,
vx: (Math.random() - 0.5) * 5,
vy: -Math.random() * 4,
life: 30,
color
});
}
};
const loop = () => {
ctx.clearRect(0, 0, w, h);
if (keys["ArrowLeft"]) paddle.x -= paddle.speed;
if (keys["ArrowRight"]) paddle.x += paddle.speed;
paddle.x = Math.max(0, Math.min(w - paddle.width, paddle.x));
if (!over) {
brick.y += brick.speed;
if (
brick.y + brick.size > paddle.y &&
brick.x + brick.size > paddle.x &&
brick.x < paddle.x + paddle.width
) {
score++;
playCatchSound();
launcher.textContent = `๐Ÿ† ${score}`;
makeParticles(brick.x + brick.size / 2, paddle.y, "#a84637");
brick.y = 0;
brick.x = Math.random() * (w - brick.size);
if (score % 10 === 0) {
brick.speed += 0.7;
level++;
}
}
if (brick.y > h) {
over = true;
playGameOverSound();
launcher.textContent = "๐ŸŽฎ";
}
ctx.fillStyle = "#a84637";
ctx.fillRect(brick.x, brick.y, brick.size, brick.size);
ctx.fillStyle = "#00FFFF";
ctx.fillRect(paddle.x, paddle.y, paddle.width, paddle.height);
ctx.font = "bold 13px Consolas, Courier New, monospace";
ctx.fillStyle = "#000000";
ctx.textAlign = "center";
ctx.textBaseline = "middle";
ctx.fillText(`LVL ${level}`.toUpperCase(), paddle.x + paddle.width / 2, paddle.y + paddle.height / 2);
particles = particles.filter(p => p.life > 0);
for (const p of particles) {
p.x += p.vx;
p.y += p.vy;
p.vy += 0.2;
p.life--;
ctx.fillStyle = `rgba(168,70,55,${p.life / 30})`;
ctx.fillRect(p.x, p.y, 3, 3);
}
animationId = requestAnimationFrame(loop);
} else {
drawTextBox([`GAME OVER`, `๐Ÿ† SCORE: ${score}`, `PRESS SPACE OR ๐ŸŽฎ TO RESTART`]);
}
};
loop();
};
const stopGame = ({ showIdle = true } = {}) => {
releaseSingleton();
gameActive = false;
starting = false;
cancelAnimationFrame(animationId);
clearInterval(countdownTimer);
if (keydownHandler) {
window.removeEventListener("keydown", keydownHandler);
keydownHandler = null;
}
if (keyupHandler) {
window.removeEventListener("keyup", keyupHandler);
keyupHandler = null;
}
keys = {};
canvas?.remove();
if (!showIdle) {
closeBtn?.style.setProperty("display", "none");
} else {
if (SHOW_PULSING_ICON) {
showIcon(); // only show the center icon if environment supports it
} else {
closeBtn?.style.setProperty("display", "none");
icon.style.display = "none";
}
}
};
// At the end, conditionally show icon
launcher.onclick = () => {
if (gameActive) {
if (over) {
stopGame({ showIdle: false });
startGame();
} else {
stopGame({ showIdle: SHOW_PULSING_ICON });
}
} else {
startGame();
}
};
// Allow spacebar to launch game from idle pulsating view
window.addEventListener("keydown", (e) => {
if (isSpaceKey(e) && !gameActive && !starting) {
e.preventDefault();
startGame();
}
});
// Show pulsing icon on idle start (only if enabled for this environment)
if (SHOW_PULSING_ICON) {
showIcon();
} else {
icon.style.display = "none";
}
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment