ZetCode

JavaScript Canvas isPointInPath Tutorial

last modified April 3, 2025

This tutorial explores the Canvas isPointInPath method in JavaScript. This method detects if a point is inside the current path, enabling hit detection. It's essential for interactive canvas applications like games and diagrams.

Basic Definition

isPointInPath checks if specified coordinates are inside the current path. It returns true if the point is inside, false otherwise. This is useful for detecting clicks or hovers on canvas elements.

The method has two forms: isPointInPath(x, y) and isPointInPath(path, x, y, fillRule). The second form works with Path2D objects and optional fill rules (nonzero or evenodd).

Basic isPointInPath Usage

This example shows how to detect clicks inside a rectangle path.

index.html
<!DOCTYPE html>
<html>
<head>
    <title>Basic isPointInPath</title>
</head>
<body>

<canvas id="myCanvas" width="300" height="200"></canvas>
<p id="output">Click on the rectangle</p>

<script>
    const canvas = document.getElementById('myCanvas');
    const ctx = canvas.getContext('2d');
    const output = document.getElementById('output');
    
    // Draw rectangle
    ctx.beginPath();
    ctx.rect(50, 50, 200, 100);
    ctx.fillStyle = 'lightblue';
    ctx.fill();
    ctx.stroke();
    
    canvas.addEventListener('click', (e) => {
        const rect = canvas.getBoundingClientRect();
        const x = e.clientX - rect.left;
        const y = e.clientY - rect.top;
        
        if (ctx.isPointInPath(x, y)) {
            output.textContent = 'Clicked inside the rectangle!';
        } else {
            output.textContent = 'Clicked outside the rectangle';
        }
    });
</script>

</body>
</html>

This code creates a blue rectangle on canvas. When clicked, it checks if the click coordinates are inside the rectangle path using isPointInPath.

The click coordinates are adjusted relative to the canvas position. The result is displayed in a paragraph element below the canvas.

Multiple Shapes Detection

This example demonstrates detecting clicks on multiple shapes.

index.html
<!DOCTYPE html>
<html>
<head>
    <title>Multiple Shapes Detection</title>
</head>
<body>

<canvas id="myCanvas" width="400" height="300"></canvas>
<p id="output">Click on a shape</p>

<script>
    const canvas = document.getElementById('myCanvas');
    const ctx = canvas.getContext('2d');
    const output = document.getElementById('output');
    
    // Draw shapes
    ctx.beginPath();
    ctx.rect(50, 50, 100, 100);
    ctx.fillStyle = 'lightgreen';
    ctx.fill();
    ctx.stroke();
    
    ctx.beginPath();
    ctx.arc(250, 100, 50, 0, Math.PI * 2);
    ctx.fillStyle = 'lightcoral';
    ctx.fill();
    ctx.stroke();
    
    ctx.beginPath();
    ctx.moveTo(300, 200);
    ctx.lineTo(350, 250);
    ctx.lineTo(250, 250);
    ctx.closePath();
    ctx.fillStyle = 'lightblue';
    ctx.fill();
    ctx.stroke();
    
    canvas.addEventListener('click', (e) => {
        const rect = canvas.getBoundingClientRect();
        const x = e.clientX - rect.left;
        const y = e.clientY - rect.top;
        
        // Check rectangle
        ctx.beginPath();
        ctx.rect(50, 50, 100, 100);
        if (ctx.isPointInPath(x, y)) {
            output.textContent = 'Clicked on the square';
            return;
        }
        
        // Check circle
        ctx.beginPath();
        ctx.arc(250, 100, 50, 0, Math.PI * 2);
        if (ctx.isPointInPath(x, y)) {
            output.textContent = 'Clicked on the circle';
            return;
        }
        
        // Check triangle
        ctx.beginPath();
        ctx.moveTo(300, 200);
        ctx.lineTo(350, 250);
        ctx.lineTo(250, 250);
        ctx.closePath();
        if (ctx.isPointInPath(x, y)) {
            output.textContent = 'Clicked on the triangle';
            return;
        }
        
        output.textContent = 'Clicked outside all shapes';
    });
</script>

</body>
</html>

This example draws three shapes: a square, circle, and triangle. On click, it checks each shape's path to determine which was clicked.

For each shape, we recreate its path before checking isPointInPath. The method returns immediately when a hit is detected to optimize performance.

Using Path2D Objects

This example shows how to use Path2D objects with isPointInPath.

index.html
<!DOCTYPE html>
<html>
<head>
    <title>Path2D with isPointInPath</title>
</head>
<body>

<canvas id="myCanvas" width="400" height="300"></canvas>
<p id="output">Click on a shape</p>

<script>
    const canvas = document.getElementById('myCanvas');
    const ctx = canvas.getContext('2d');
    const output = document.getElementById('output');
    
    // Create Path2D objects
    const starPath = new Path2D();
    starPath.moveTo(100, 25);
    starPath.lineTo(120, 75);
    starPath.lineTo(175, 75);
    starPath.lineTo(135, 100);
    starPath.lineTo(150, 150);
    starPath.lineTo(100, 125);
    starPath.lineTo(50, 150);
    starPath.lineTo(65, 100);
    starPath.lineTo(25, 75);
    starPath.lineTo(80, 75);
    starPath.closePath();
    
    const heartPath = new Path2D();
    heartPath.moveTo(250, 75);
    heartPath.bezierCurveTo(250, 37, 300, 25, 300, 75);
    heartPath.bezierCurveTo(300, 125, 250, 150, 250, 175);
    heartPath.bezierCurveTo(250, 150, 200, 125, 200, 75);
    heartPath.bezierCurveTo(200, 25, 250, 37, 250, 75);
    heartPath.closePath();
    
    // Draw paths
    ctx.fillStyle = 'pink';
    ctx.fill(starPath);
    ctx.stroke(starPath);
    
    ctx.fillStyle = 'red';
    ctx.fill(heartPath);
    ctx.stroke(heartPath);
    
    canvas.addEventListener('click', (e) => {
        const rect = canvas.getBoundingClientRect();
        const x = e.clientX - rect.left;
        const y = e.clientY - rect.top;
        
        if (ctx.isPointInPath(starPath, x, y)) {
            output.textContent = 'Clicked on the star';
        } else if (ctx.isPointInPath(heartPath, x, y)) {
            output.textContent = 'Clicked on the heart';
        } else {
            output.textContent = 'Clicked outside shapes';
        }
    });
</script>

</body>
</html>

This example uses Path2D objects to create complex shapes (star and heart). Path2D allows path reuse without recreating them for hit detection.

The isPointInPath method accepts Path2D as first argument. This makes the code cleaner and more efficient than recreating paths.

Fill Rule Demonstration

This example demonstrates the effect of different fill rules on point detection.

index.html
<!DOCTYPE html>
<html>
<head>
    <title>Fill Rule with isPointInPath</title>
</head>
<body>

<canvas id="myCanvas" width="400" height="300"></canvas>
<p id="output">Click inside the concentric circles</p>

<script>
    const canvas = document.getElementById('myCanvas');
    const ctx = canvas.getContext('2d');
    const output = document.getElementById('output');
    
    // Draw concentric circles
    ctx.beginPath();
    ctx.arc(150, 150, 100, 0, Math.PI * 2);
    ctx.arc(150, 150, 50, 0, Math.PI * 2);
    ctx.fillStyle = 'lightgray';
    ctx.fill();
    ctx.stroke();
    
    canvas.addEventListener('click', (e) => {
        const rect = canvas.getBoundingClientRect();
        const x = e.clientX - rect.left;
        const y = e.clientY - rect.top;
        
        // Recreate path for hit detection
        ctx.beginPath();
        ctx.arc(150, 150, 100, 0, Math.PI * 2);
        ctx.arc(150, 150, 50, 0, Math.PI * 2);
        
        // Check with different fill rules
        const nonzero = ctx.isPointInPath(x, y, 'nonzero');
        const evenodd = ctx.isPointInPath(x, y, 'evenodd');
        
        output.textContent = `Nonzero: ${nonzero}, Evenodd: ${evenodd}`;
    });
</script>

</body>
</html>

This example shows how fill rules affect point detection in complex paths. We draw two concentric circles and check point inclusion with both rules.

The 'nonzero' rule (default) considers the center as inside, while 'evenodd' considers it outside. This demonstrates how fill rules change hit detection behavior for self-intersecting paths.

Interactive Drawing with Hit Detection

This example creates an interactive drawing app with shape selection.

index.html
<!DOCTYPE html>
<html>
<head>
    <title>Interactive Drawing with Hit Detection</title>
</head>
<body>

<canvas id="myCanvas" width="500" height="400"></canvas>
<div>
    <button id="addRect">Add Rectangle</button>
    <button id="addCircle">Add Circle</button>
    <p id="output">Click on shapes to select them</p>
</div>

<script>
    const canvas = document.getElementById('myCanvas');
    const ctx = canvas.getContext('2d');
    const output = document.getElementById('output');
    const addRect = document.getElementById('addRect');
    const addCircle = document.getElementById('addCircle');
    
    let shapes = [];
    let selectedShape = null;
    
    class Shape {
        constructor(path, type, color) {
            this.path = path;
            this.type = type;
            this.color = color;
            this.selected = false;
        }
    }
    
    function drawShapes() {
        ctx.clearRect(0, 0, canvas.width, canvas.height);
        
        shapes.forEach(shape => {
            ctx.fillStyle = shape.selected ? 'yellow' : shape.color;
            ctx.fill(shape.path);
            ctx.stroke(shape.path);
            
            // Draw selection indicator
            if (shape.selected) {
                ctx.strokeStyle = 'red';
                ctx.lineWidth = 3;
                ctx.stroke(shape.path);
                ctx.strokeStyle = 'black';
                ctx.lineWidth = 1;
            }
        });
    }
    
    // Add rectangle button
    addRect.addEventListener('click', () => {
        const x = Math.random() * 350 + 50;
        const y = Math.random() * 250 + 50;
        const width = Math.random() * 100 + 50;
        const height = Math.random() * 100 + 50;
        
        const path = new Path2D();
        path.rect(x, y, width, height);
        
        const colors = ['lightblue', 'lightgreen', 'pink', 'lavender'];
        const color = colors[Math.floor(Math.random() * colors.length)];
        
        shapes.push(new Shape(path, 'rectangle', color));
        drawShapes();
    });
    
    // Add circle button
    addCircle.addEventListener('click', () => {
        const x = Math.random() * 350 + 50;
        const y = Math.random() * 250 + 50;
        const radius = Math.random() * 50 + 25;
        
        const path = new Path2D();
        path.arc(x, y, radius, 0, Math.PI * 2);
        
        const colors = ['lightcoral', 'lightseagreen', 'plum', 'wheat'];
        const color = colors[Math.floor(Math.random() * colors.length)];
        
        shapes.push(new Shape(path, 'circle', color));
        drawShapes();
    });
    
    // Canvas click handler
    canvas.addEventListener('click', (e) => {
        const rect = canvas.getBoundingClientRect();
        const x = e.clientX - rect.left;
        const y = e.clientY - rect.top;
        
        // Deselect all first
        shapes.forEach(shape => shape.selected = false);
        selectedShape = null;
        
        // Check shapes in reverse order (top to bottom)
        for (let i = shapes.length - 1; i >= 0; i--) {
            if (ctx.isPointInPath(shapes[i].path, x, y)) {
                shapes[i].selected = true;
                selectedShape