Tutorial: Drawable and Pannable-Zoomable Canvas in Vanilla JS

Abdullah Ahmad
10 min readDec 22, 2024

--

A sample drawing tool (don’t worry, we aren’t building this completely)

Introduction

A few weeks ago, I needed to make a canvas freehand drawing tool for a designers’ Fiverr app in SvelteKit. Since the requirements were a bit complex and subject to change, I hesitated to rely on an npm package, as it might not fully align with my client’s needs.

Instead of choosing the uncertain route, I chose the more challenging path and decided to build a quick MVP myself first, following Harrison Milbradt’s articles: An exploration into free drawing in HTML canvas and Panning and Zooming in HTML Canvas. I highly recommend reading them — they’re incredibly insightful.

This tutorial essentially combines the code from both articles and enhances it with additional functionality. The contents of this tutorial are as follows:

  1. The Goal: This section outlines what we aim to build, including a brief description of additional potential features, some of which are explored in Harrison’s articles.
  2. The Implementation: Here, we discuss the technical essence of our solution, providing a high-level understanding of the approach before delving into the specifics.
  3. The Detailed Guide: This part consists of six steps: creating two canvases, drawing a background image on the background canvas, implementing drawing and erasing, implementing panning, and implementing zooming.

The Goal:

The goal is to create a canvas with an image already drawn on it. On top of this image, the designer should be able to freely draw lines and erase any of their previous drawings using an erase tool.

Additionally, the designer should have the ability to zoom in and pan around the image, enabling them to focus on specific parts and interact with the draw tool.

This feature can be extended to include functionalities such as different pen colors, adjustable pen widths, a pen drawing cursor indicator, and the ability to draw shapes. However, these enhancements are straightforward to implement once the foundational functionality is in place.

The Implementation:

At its core, our solution involves two canvases working seamlessly together. The background canvas will display the image we want to draw on, ensuring that drawing and erasing lines do not affect the image’s pixels. The foreground canvas, referred to as the drawing layer canvas, will handle interactions like drawing and erasing lines.

Mouse event handlers (e.g., mouseup, mousedown, mousemove, and mousewheel) will update the canvas’s context based on user actions. Any panning or zooming on the drawing layer canvas must be precisely mirrored in the background canvas to avoid misalignment of drawn lines.

Additionally, the contexts of both canvases need to stay synchronized so that any drawn or erased lines scale and pan correctly, maintaining a visually consistent and accurate result. This synchronization is the essence of our solution.

The Detailed Guide:

Step 1: Create the 2 Canvases

We will proceed by creating two canvases, as described below:

  <canvas width="500" height="500" id="backgroundLayerCanvas"></canvas>
<canvas width="500" height="500" id="drawingCanvas"></canvas>

Let’s style them a bit as well to enhance their appearance, and to have one layer over the other:

#drawingCanvas {
position: absolute;
top: 0;
left: 0;
z-index: 1;
border: 2px solid black;
}

#backgroundLayerCanvas {
border: 2px solid black;
background-color: greenyellow;
}

The result:

Don’t mind the non-overlapping borders. It was a minor CSS issue in an app made for R&D, which I didn’t have the time to fix because it didn’t occur in the main app I worked on. I am pretty sure you can fix this yourself.

Step 2: Draw the Image on the Background Canvas

Let’s write some JavaScript to retrieve the elements in our script and draw a random duck image onto the background canvas:

const backgroundCanvas = document.getElementById('backgroundLayerCanvas')
const backgroundCtx = backgroundCanvas.getContext('2d')
const canvas = document.getElementById('drawingCanvas')
const ctx = canvas.getContext('2d')

imageObj = new Image();
imageObj.onload = function () {
backgroundCtx.drawImage(imageObj, 0, 0);
}
const duckImageSrc = ''
imageObj.src = duckImageSrc; // You can have this come from an uploaded image as well

The result:

Step 3: Implement Drawing and Erasing of Lines on Drawing Layer Canvas

First, let’s define a drawLine function, which will take the x- and y-coordinates of the starting and ending positions to draw a line:

const drawLine = (x, y, previousPosX, previousPosY) => {
ctx.beginPath(); // this will be drawing lines on context of drawing layer canvas only (ctx)
ctx.moveTo(previousPosX, previousPosY);
ctx.lineTo(x, y);
ctx.strokeStyle = penColor;
ctx.lineWidth = penWidth;
ctx.lineCap = 'round';
ctx.stroke();
}

Similarly, let’s define a function eraseLine for erasing a line, which will take the same arguments as the drawLine function:

const eraseLine = (x, y, previousPosX, previousPosY) => {
ctx.save()
ctx.beginPath()
ctx.moveTo(previousPosX, previousPosY)
ctx.lineTo(x, y)
ctx.globalCompositeOperation = 'destination-out'
ctx.lineWidth = penWidth
ctx.lineCap = 'round'
ctx.stroke()
ctx.restore()
}

When we try to draw a line, we trigger the mousedown event once with our mouse, followed by several mousemove events as we drag the cursor around. We only need to track the mousemove events that occur between the mousedown and mouseup events. Once the mouseup event occurs, we will detach the mousemove event listener from the canvas.

Before defining those functions, we need some variables to keep track of the x- and y-coordinates that will update with every mouse event. Let’s initialize these variables, along with some others, as follows:

// NEW CODE
let drawModeType = 'freehand'
let previousXDrawing = 0, previousYDrawing = 0; // these are previous mouse event's X and Y coordinate positions for the canvas TODO: resolve confusion
let lines = [];
let isDrawingMode = true
const penWidth = 5
const penColor = 'red'

// OLD CODE
const backgroundCanvas = document.getElementById('backgroundLayerCanvas')
const backgroundCtx = backgroundCanvas.getContext('2d')
const canvas = document.getElementById('drawingCanvas')
const ctx = canvas.getContext('2d')

Now, let’s go ahead and define those functions:

const onMouseDown = (e) => {
// This is my attempt to make the drawLine logic work
if (isDrawingMode) {
var bounding = canvas.getBoundingClientRect();
var x = e.clientX - bounding.left;
var y = e.clientY - bounding.top;

const p = DOMPoint.fromPoint({ x, y }); // Create a DOMPoint for the pixel coordinates
const t = ctx.getTransform().inverse(); // Get the inverse of the transformation applied to the canvas context
const { x: adjustedX, y: adjustedY } = t.transformPoint(p); // Use it to calculate context coordinates for your pixel point

previousXDrawing = adjustedX
previousYDrawing = adjustedY
}

canvas.addEventListener("mousemove", onMouseMove);
}

const onMouseMove = (e) => {
if (isDrawingMode) {
var bounding = canvas.getBoundingClientRect();
var x = e.clientX - bounding.left;
var y = e.clientY - bounding.top;

const p = DOMPoint.fromPoint({ x, y }); // Create a DOMPoint for the pixel coordinates
const t = ctx.getTransform().inverse(); // Get the inverse of the transformation applied to the canvas context
const { x: adjustedX, y: adjustedY } = t.transformPoint(p); // Use it to calculate context coordinates for your pixel point

const xToDraw = adjustedX
const yToDraw = adjustedY

if (e.shiftKey) {
eraseLine(xToDraw, yToDraw, previousXDrawing, previousYDrawing)
lines.push({ x: xToDraw, y: yToDraw, previousX: previousXDrawing, previousY: previousYDrawing, isEraseLine: true }) // we somehow need to remember that this is a erased line. and handle it that way
} else {
if (drawModeType === 'freehand') {
drawLine(xToDraw, yToDraw, previousXDrawing, previousYDrawing)
lines.push({ x: xToDraw, y: yToDraw, previousX: previousXDrawing, previousY: previousYDrawing, isEraseLine: false })
}
}
previousXDrawing = adjustedX
previousYDrawing = adjustedY
}
}

const onMouseUp = (e) => {
canvas.removeEventListener("mousemove", onMouseMove);
}

Let’s also attach the event handlers to their respective event listeners:

// Adding mousedown event listener to drawing canvas
canvas.addEventListener("mousedown", onMouseDown)

// Adding mouseup event listener to drawing canvas
canvas.addEventListener("mouseup", onMouseUp)

The result:

Step 4: Implement Panning on both the Canvases

Before proceeding, we need to toggle between drawing and view modes. Both panning and drawing require a mousedown event followed by a series of mousemove events, which can lead to conflicts between the two. The simplest solution is to separate them into distinct modes:

<body>
<canvas width="500" height="500" id="backgroundLayerCanvas"></canvas>
<canvas width="500" height="500" id="drawingCanvas"></canvas>

<label for="">Drawing Mode</label>
<input type="checkbox" checked onchange="toggleDrawMode()">

<script src="canvas.js"></script>
</body>

<script>

function toggleDrawMode() {
isDrawingMode = !isDrawingMode
}

...
</script>

Let’s initialize additional variables to keep track of the x- and y-coordinates for panning separately:

let previousX = 0, previousY = 0; // these are previous mouse event's X and Y coordinate positions for the canvas

Let’s modify the mousedown event handler:

const onMouseDown = (e) => {
// NEW CODE: This is needed for ensuring smooth panning
previousX = e.clientX;
previousY = e.clientY;

// OLD CODE:
if (isDrawingMode) {
var bounding = canvas.getBoundingClientRect();
var x = e.clientX - bounding.left;
var y = e.clientY - bounding.top;
const p = DOMPoint.fromPoint({ x, y }); // Create a DOMPoint for the pixel coordinates
const t = ctx.getTransform().inverse(); // Get the inverse of the transformation applied to the canvas context
const { x: adjustedX, y: adjustedY } = t.transformPoint(p); // Use it to calculate context coordinates for your pixel point
previousXDrawing = adjustedX
previousYDrawing = adjustedY
}
}

To maintain consistency, we also need the viewportTransform, which will track all the transformations and actions performed on the canvases:

const viewportTransform = {
x: 0,
y: 0,
scale: 1
}

Finally, let’s modify the mousemove event handler to account for panning:

const onMouseMove = (e) => {
// OLD CODE ...
if (isDrawingMode) {
...
} else { // NEW CODE ...
updatePanning(e)
render()
}
}

Some explanation is needed for these critical methods: updatePanning and render:

  • updatePanning: This function updates the viewportTransform object, which tracks the transformations applied to the canvas. It ensures that when panning occurs, the positions of the already drawn pixels are correctly adjusted. This way, when we draw new lines, they will appear exactly where the cursor is positioned, taking into account any prior panning.
  • render: The render function first resets both canvases by clearing any previously drawn content. It then sets the transform matrix of the drawing layer canvas to reflect any actions performed during panning (and zooming, if applicable). Next, it redraws the background image on the background canvas and re-renders the previously drawn lines on the drawing layer canvas. Remember the lines.push call from earlier? This is exactly why we stored the lines—to redraw them accurately during each render!
const updatePanning = (e) => {
const localX = e.clientX;
const localY = e.clientY;

viewportTransform.x += localX - previousX;
viewportTransform.y += localY - previousY;

previousX = localX;
previousY = localY;
}

const render = () => {
// Code for drawing layer canvas
ctx.setTransform(1, 0, 0, 1, 0, 0);
ctx.clearRect(0, 0, canvas.width, canvas.height);
ctx.setTransform(viewportTransform.scale, 0, 0, viewportTransform.scale, viewportTransform.x, viewportTransform.y);

// Code for background layer canvas
backgroundCtx.setTransform(1, 0, 0, 1, 0, 0);
backgroundCtx.clearRect(0, 0, canvas.width, canvas.height);
backgroundCtx.setTransform(viewportTransform.scale, 0, 0, viewportTransform.scale, viewportTransform.x, viewportTransform.y);
backgroundCtx.drawImage(imageObj, 0, 0);

// Ok, so we DO need to redraw the lines. That means we DO need to KEEP TRACK of them!
// This drawing just needs absolute figures, the same as they were when the line was being made! The viewportTransform will automatically adjust shit here and there.
lines.forEach((line, idx) => {
if (line.isEraseLine) {
eraseLine(line.x, line.y, line.previousX, line.previousY)
} else {
drawLine(line.x, line.y, line.previousX, line.previousY)
}
});

}

Here’s a TypeScript interface for the Lineobject:

interface Line {
x: number; // The line's end point's x coordinate
y: number; // The line's end point's y coordinate
previousX: number; // The line's start point's x coordinate
previousY: number; // The line's start point's y coordinate
isEraseLine: boolean; // we will call either the drawLine or eraseLine method based on this
}

The end result:

Step 5: Implement Zooming on both the Canvases

Finally, we need to implement zooming as well. Let’s define a function to update the zooming properties of the canvas context:

const updateZooming = (e) => {
const oldX = viewportTransform.x;
const oldY = viewportTransform.y;

const localX = e.clientX;
const localY = e.clientY;

const previousScale = viewportTransform.scale;

const newScale = viewportTransform.scale += e.deltaY * -0.01;

const newX = localX - (localX - oldX) * (newScale / previousScale);
const newY = localY - (localY - oldY) * (newScale / previousScale);

viewportTransform.x = newX;
viewportTransform.y = newY;
viewportTransform.scale = newScale;
}

We also need to define an event listener that uses the updateZooming method:

const onMouseWheel = (e) => {
updateZooming(e)
render()
}

We also want to prevent zooming out until the zoom level reaches zero, as this could introduce bugs (such as drawings not appearing at all). To handle this, we introduce an isZoomAllowed method:

// If we are trying to zoom out to a level lesser than zoom level 1, then we will not do anything
const isZoomAllowed = (viewportTransform, deltaY) => {
return viewportTransform.scale + deltaY * -0.01 >= 1
}

const onMouseWheel = (e) => {
if (isZoomAllowed(viewportTransform, e.deltaY)) {
updateZooming(e) // just like updatePanning changes the transform matrix, so does updateZooming
render() // after making necessary adjustments to the transform matrix, we need to redraw everything, hence render is called again
}
}

// Adding mousewheel event listener to drawing canvas
canvas.addEventListener("wheel", onMouseWheel);

The result:

Ending

You can find all the code for this hosted here. I am planning to create an npm package (Svelte-specific) for this functionality soon, so expect its release in a few weeks.

For a deeper understanding of how the drawLine and eraseLine methods work with the canvas API, I highly recommend checking out Harrison's articles. They were instrumental in enhancing my understanding of the topic:

  1. Panning and Zooming in HTML Canvas
  2. An exploration into free drawing in HTML canvas

Happy coding!

--

--

No responses yet