Skip to content

Instantly share code, notes, and snippets.

@ldgrp
Last active August 2, 2025 11:48
Show Gist options
  • Save ldgrp/df0b90dd8f9a5d19bf199f2c9253e99a to your computer and use it in GitHub Desktop.
Save ldgrp/df0b90dd8f9a5d19bf199f2c9253e99a to your computer and use it in GitHub Desktop.
HTML Energy 2025
<!DOCTYPE html>
<html>
<head>
<title>sixteen times</title>
<meta name="author" content="leo@ldgrp.me">
<meta name="viewport" content="width=device-width, initial-scale=1" />
<style type="text/css">
.canvas {
border: 1px solid black;
width: 150px;
height: 150px;
background-color: white;
position: relative;
user-select: none;
}
#drawArea {
touch-action: none;
}
.cursor,
.ghost {
position: absolute;
width: 10px;
height: 10px;
background-color: red;
border-radius: 50%;
}
.ghost {
transition: opacity 1s ease-out;
}
span {
background-color: rgb(0, 255, 0);
}
body {
background-image: url()
}
@media (min-width: 900px) {
body {
display: grid;
grid-template-columns: 600px auto 1fr;
}
}
</style>
</head>
<body>
<div>
<table>
<tbody>
<tr>
<td class="canvas" id="canvas-16">
<div class="cursor" id="cursor-16"></div>
</td>
<td class="canvas" id="canvas-15">
<div class="cursor" id="cursor-15"></div>
</td>
<td class="canvas" id="canvas-14">
<div class="cursor" id="cursor-14"></div>
</td>
<td class="canvas" id="canvas-13">
<div class="cursor" id="cursor-13"></div>
</td>
</tr>
<tr>
<td class="canvas" id="canvas-12">
<div class="cursor" id="cursor-12"></div>
</td>
<td class="canvas" id="canvas-11">
<div class="cursor" id="cursor-11"></div>
</td>
<td class="canvas" id="canvas-10">
<div class="cursor" id="cursor-10"></div>
</td>
<td class="canvas" id="canvas-9">
<div class="cursor" id="cursor-9"></div>
</td>
</tr>
<tr>
<td class="canvas" id="canvas-8">
<div class="cursor" id="cursor-8"></div>
</td>
<td class="canvas" id="canvas-7">
<div class="cursor" id="cursor-7"></div>
</td>
<td class="canvas" id="canvas-6">
<div class="cursor" id="cursor-6"></div>
</td>
<td class="canvas" id="canvas-5">
<div class="cursor" id="cursor-5"></div>
</td>
</tr>
<tr>
<td class="canvas" id="canvas-4">
<div class="cursor" id="cursor-4"></div>
</td>
<td class="canvas" id="canvas-3">
<div class="cursor" id="cursor-3"></div>
</td>
<td class="canvas" id="canvas-2">
<div class="cursor" id="cursor-2"></div>
</td>
<td class="canvas" id="canvas-1">
<div class="cursor" id="cursor-1"></div>
</td>
</tr>
</tbody>
</table>
</div>
<div style="align-self: end; margin-bottom: 3px;">
<div>
<p>
<span>move inside the box</span><br />
<span>your pattern will be recorded</span>
</p>
<p>
<span>just the movement of your mouse</span><br />
<span>not even a name</span>
</p>
<p>
<span>sixteen times</span><br />
<span>your pattern will be played</span>
</p>
<p>
<span>then it fades</span><br />
<span>not erased</span><br />
<span>just replaced</span>
</p>
</div>
<div>
<progress id="timer" value="0" max="10000"></progress>
<button id="startDrawingButton" onclick="startDrawing()">Start drawing</button>
<input type="color" id="colorPicker" value="#ff0000" title="Choose cursor color">
<div class="canvas" id="drawArea">
<div class="cursor" id="cursor"></div>
</div>
</div>
</div>
<script>
const url = 'https://ldgrp--58faeb246f8611f0b42c0224a6c84d84.web.val.run/';
const timerMeter = document.getElementById("timer")
const startDrawingButton = document.getElementById("startDrawingButton")
const drawArea = document.getElementById("drawArea")
const cursor = document.getElementById("cursor")
const colorPicker = document.getElementById("colorPicker")
let startDrawingInterval;
let isActive = false;
const START_TIMER_VALUE = 10000;
const TIMER_INTERVAL = 10;
const CURSOR_OFFSET = 5;
let canvasTime = 0;
let temp = Array(START_TIMER_VALUE / TIMER_INTERVAL).fill(null)
let cursorPositions = [];
let isMouseDown = false;
function getTimeIndex(t) {
return Math.floor(t / TIMER_INTERVAL);
}
function setCursorPosition(element, x, y) {
element.style.left = (x - CURSOR_OFFSET) + 'px';
element.style.top = (y - CURSOR_OFFSET) + 'px';
}
function getCursorsPos() {
fetch(url)
.then(response => response.json())
.then(data => cursorPositions = data);
}
function putCursorPos() {
fetch(url, {
method: 'PUT',
body: JSON.stringify(temp),
});
}
drawArea.onmousedown = () => isMouseDown = true;
drawArea.onmouseup = () => isMouseDown = false;
// Color picker event handler
colorPicker.onchange = () => {
cursor.style.backgroundColor = colorPicker.value;
};
// Initialize cursor color
cursor.style.backgroundColor = colorPicker.value;
/**
* Mouse/touch events
*/
drawArea.ontouchstart = (event) => {
event.preventDefault();
isMouseDown = true;
};
drawArea.ontouchend = (event) => {
event.preventDefault();
isMouseDown = false;
};
drawArea.onmousemove = (event) => {
handleMove(event.clientX, event.clientY);
};
drawArea.ontouchmove = (event) => {
event.preventDefault();
if (event.touches.length > 0) {
handleMove(event.touches[0].clientX, event.touches[0].clientY);
}
};
function handleMove(clientX, clientY) {
if (!isActive) return;
const index = getTimeIndex(START_TIMER_VALUE - timerMeter.value);
const rect = drawArea.getBoundingClientRect();
const x = clientX - rect.left;
const y = clientY - rect.top;
temp[index] = { x, y, isMouseDown, color: colorPicker.value };
setCursorPosition(cursor, x, y);
if (isMouseDown) {
createGhost(drawArea, x, y, colorPicker.value);
}
}
function createGhost(parent, x, y, color) {
const ghost = document.createElement('div');
ghost.className = 'ghost';
ghost.style.backgroundColor = color;
setCursorPosition(ghost, x, y);
parent.appendChild(ghost);
setTimeout(() => {
ghost.style.opacity = '0';
setTimeout(() => ghost.parentNode?.removeChild(ghost), 1000);
}, 1000);
}
function tickDrawArea() {
timerMeter.value -= TIMER_INTERVAL;
if (timerMeter.value <= 0) {
isActive = false;
clearInterval(startDrawingInterval);
startDrawingButton.disabled = false;
if (temp.length) {
cursorPositions.unshift(temp);
putCursorPos();
}
}
}
function tickCanvas() {
if (cursorPositions.length === 0) return;
const index = getTimeIndex(canvasTime);
for (let i = 0; i < Math.min(16, cursorPositions.length); i++) {
const positions = cursorPositions[i];
const pos = positions[index];
if (pos) {
const cursor = document.getElementById(`cursor-${i + 1}`);
const canvas = document.getElementById(`canvas-${i + 1}`);
setCursorPosition(cursor, pos.x, pos.y);
if (pos.color) {
cursor.style.backgroundColor = pos.color;
}
if (pos.isMouseDown) {
createGhost(canvas, pos.x, pos.y, pos.color);
}
}
}
canvasTime = (canvasTime + TIMER_INTERVAL) % START_TIMER_VALUE;
}
function startDrawing() {
temp = [];
isActive = true;
timerMeter.value = START_TIMER_VALUE;
startDrawingButton.disabled = true;
startDrawingInterval = setInterval(tickDrawArea, TIMER_INTERVAL);
}
setInterval(tickCanvas, TIMER_INTERVAL);
getCursorsPos();
</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment