ThePlaceHolders/static/canvas.js

117 lines
3.1 KiB
JavaScript
Raw Normal View History

2024-10-09 23:18:42 -05:00
const canvas = document.getElementById("canvas");
const ctx = canvas.getContext("2d");
2024-11-05 11:51:17 -06:00
const CANVAS_SIZE = 500;
const DATA_SIZE = 200;
let scale = 1;
2024-11-06 02:02:36 -06:00
let offx = 0;
let offy = 0;
let clickx = 0;
let clicky = 0;
let mouseClicked = false;
2024-11-05 11:51:17 -06:00
// Array of [red, blue, green, transparency] * width * height
2024-11-06 02:02:36 -06:00
let image = new ImageData(DATA_SIZE, DATA_SIZE);
// Drawable image
let bitmap = null;
window.addEventListener("load", async (e) => {
2024-11-06 18:02:08 -06:00
for (let x = 0; x < DATA_SIZE * DATA_SIZE * 4; x++) {
image.data[x] = 255;
}
2024-11-06 02:02:36 -06:00
// Create drawable image, update it every half second
await redraw();
setInterval(redraw, 500);
setInterval(draw, 500);
});
async function redraw() {
bitmap = await createImageBitmap(image);
}
2024-10-29 21:32:06 -05:00
2024-11-05 11:51:17 -06:00
// Color is an array of four numbers 0-255 (RGB + transparency)
function setPixel(data, x, y, color) {
let start = (y * DATA_SIZE + x) * 4;
for (let i = 0; i < 4; i++) {
data[start + i] = color[i];
2024-10-16 22:30:14 -05:00
}
}
2024-10-16 18:41:26 -05:00
2024-11-06 02:02:36 -06:00
function draw() {
2024-11-05 11:51:17 -06:00
ctx.fillStyle = "rgb(0 0 0 255)";
ctx.clearRect(0, 0, canvas.width, canvas.height);
let drawScale = CANVAS_SIZE / DATA_SIZE * scale;
ctx.imageSmoothingEnabled = false;
ctx.scale(drawScale, drawScale);
2024-11-06 02:02:36 -06:00
ctx.drawImage(bitmap, offx, offy);
2024-11-05 11:51:17 -06:00
ctx.setTransform(1, 0, 0, 1, 0, 0);
2024-10-16 18:41:26 -05:00
}
2024-11-06 18:02:08 -06:00
function mousePosition(e) {
let x = e.clientX - canvas.getBoundingClientRect().left;
let y = e.clientY - canvas.getBoundingClientRect().top;
let cx = Math.round(x / (CANVAS_SIZE * scale) * DATA_SIZE - offx - 0.5);
let cy = Math.round(y / (CANVAS_SIZE * scale) * DATA_SIZE - offy - 0.5);
return {x: cx, y: cy};
}
2024-11-05 11:51:17 -06:00
canvas.addEventListener("wheel", async (e) => {
2024-11-06 18:02:08 -06:00
let mouse = mousePosition(e);
console.log(mouse);
2024-11-06 02:02:36 -06:00
let oldScale = scale;
2024-11-05 11:51:17 -06:00
if (e.deltaY < 0) {
scale += 0.1;
} else if (e.deltaY > 0) {
scale -= 0.1;
2024-10-30 16:30:24 -05:00
}
2024-11-06 18:02:08 -06:00
let diff = (CANVAS_SIZE / oldScale) - (CANVAS_SIZE / scale);
let ratiox = (mouse.x - (CANVAS_SIZE / 2)) / CANVAS_SIZE;
let ratioy = (mouse.y - (CANVAS_SIZE / 2)) / CANVAS_SIZE;
offx += diff * ratiox;
offy += diff * ratioy;
2024-11-06 02:02:36 -06:00
draw(image);
2024-10-30 16:30:24 -05:00
});
2024-11-05 11:51:17 -06:00
canvas.addEventListener("contextmenu", (e) => {
e.preventDefault();
})
2024-10-30 16:30:24 -05:00
2024-11-05 11:51:17 -06:00
canvas.addEventListener("mousedown", async (e) => {
// Left click
if (e.button == 0) {
2024-11-06 02:02:36 -06:00
mouseClicked = true;
setTimeout(() => {
mouseClicked = false;
}, 200);
}
});
canvas.addEventListener("mouseup", async (e) => {
if (e.button == 0 && mouseClicked) {
mouseClicked = false;
clickx = e.clientX;
clicky = e.clientY;
2024-11-06 18:02:08 -06:00
let mouse = mousePosition(e);
let cx = mouse.x;
let cy = mouse.y;
2024-11-05 11:51:17 -06:00
2024-11-06 02:02:36 -06:00
if (cx < 0 || cx > CANVAS_SIZE || cy < 0 || cy > CANVAS_SIZE) {
return;
}
setPixel(image.data, cx, cy, [255, 0, 0, 255]);
await redraw();
draw();
2024-10-30 16:30:24 -05:00
}
2024-10-23 18:09:48 -05:00
});
2024-11-06 02:02:36 -06:00
canvas.addEventListener("mousemove", async (e) => {
// `buttons` is a bitflag for some dumb reason
if (e.buttons & 1) {
offx += e.movementX * (1.0 / scale) * 0.45;
offy += e.movementY * (1.0 / scale) * 0.45;
draw(image);
}
});