JavaScript Canvas lineCap Tutorial
last modified April 3, 2025
In this article, we explore the Canvas lineCap property in JavaScript. This property controls how the ends of lines are rendered in HTML canvas drawings. Understanding lineCap is essential for creating polished graphics.
Basic Definition
The lineCap property determines the shape used to draw the endpoints of lines. It affects lines drawn with stroke(), strokeRect(), and other stroke methods. There are three possible values: butt, round, and square.
The default value is butt, which creates flat edges perpendicular to the line. Round adds semicircular ends, while square adds square ends extending beyond the line's length.
Basic lineCap Demonstration
This example shows all three lineCap values applied to simple lines.
<!DOCTYPE html> <html> <head> <title>Basic lineCap</title> </head> <body> <canvas id="myCanvas" width="400" height="200"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Guide line to show actual length ctx.strokeStyle = 'lightgray'; ctx.lineWidth = 1; ctx.beginPath(); ctx.moveTo(50, 50); ctx.lineTo(350, 50); ctx.stroke(); // Butt lineCap (default) ctx.strokeStyle = 'red'; ctx.lineWidth = 15; ctx.lineCap = 'butt'; ctx.beginPath(); ctx.moveTo(50, 50); ctx.lineTo(150, 50); ctx.stroke(); // Round lineCap ctx.strokeStyle = 'blue'; ctx.lineCap = 'round'; ctx.beginPath(); ctx.moveTo(200, 50); ctx.lineTo(300, 50); ctx.stroke(); // Square lineCap ctx.strokeStyle = 'green'; ctx.lineCap = 'square'; ctx.beginPath(); ctx.moveTo(350, 50); ctx.lineTo(450, 50); ctx.stroke(); </script> </body> </html>
This example draws three horizontal lines with different lineCap values. A light gray guide line shows the actual length of each line segment.
The red line uses butt (default), blue uses round, and green uses square. Notice how round and square extend beyond the line's endpoints.
lineCap with Dashed Lines
This example demonstrates how lineCap affects dashed lines.
<!DOCTYPE html> <html> <head> <title>lineCap with Dashes</title> </head> <body> <canvas id="myCanvas" width="400" height="200"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Set dash pattern ctx.setLineDash([20, 10]); // Butt lineCap ctx.strokeStyle = 'red'; ctx.lineWidth = 10; ctx.lineCap = 'butt'; ctx.beginPath(); ctx.moveTo(50, 50); ctx.lineTo(350, 50); ctx.stroke(); // Round lineCap ctx.strokeStyle = 'blue'; ctx.lineCap = 'round'; ctx.beginPath(); ctx.moveTo(50, 100); ctx.lineTo(350, 100); ctx.stroke(); // Square lineCap ctx.strokeStyle = 'green'; ctx.lineCap = 'square'; ctx.beginPath(); ctx.moveTo(50, 150); ctx.lineTo(350, 150); ctx.stroke(); </script> </body> </html>
Here we apply the same dash pattern ([20,10]) to three lines with different lineCap values. The pattern means 20px dashes with 10px gaps.
The red line (butt) has sharp ends on each dash. The blue line (round) has rounded ends, and green (square) has squared-off ends extending the dash.
lineCap with Arcs
This example shows how lineCap affects open arcs and circles.
<!DOCTYPE html> <html> <head> <title>lineCap with Arcs</title> </head> <body> <canvas id="myCanvas" width="400" height="300"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Butt lineCap arc ctx.strokeStyle = 'red'; ctx.lineWidth = 15; ctx.lineCap = 'butt'; ctx.beginPath(); ctx.arc(100, 100, 50, 0, Math.PI * 1.5, false); ctx.stroke(); // Round lineCap arc ctx.strokeStyle = 'blue'; ctx.lineCap = 'round'; ctx.beginPath(); ctx.arc(200, 100, 50, 0, Math.PI * 1.5, false); ctx.stroke(); // Square lineCap arc ctx.strokeStyle = 'green'; ctx.lineCap = 'square'; ctx.beginPath(); ctx.arc(300, 100, 50, 0, Math.PI * 1.5, false); ctx.stroke(); </script> </body> </html>
This example draws three 270-degree arcs with different lineCap values. Each arc starts at the right (0 radians) and goes counter-clockwise.
The red arc (butt) has flat ends at both endpoints. The blue arc (round) has rounded ends, while green (square) has squared-off ends extending beyond the arc's endpoints.
lineCap with Complex Paths
This example demonstrates lineCap behavior with more complex paths.
<!DOCTYPE html> <html> <head> <title>lineCap with Complex Paths</title> </head> <body> <canvas id="myCanvas" width="400" height="300"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Butt lineCap ctx.strokeStyle = 'red'; ctx.lineWidth = 10; ctx.lineCap = 'butt'; ctx.beginPath(); ctx.moveTo(50, 50); ctx.lineTo(150, 150); ctx.lineTo(250, 50); ctx.stroke(); // Round lineCap ctx.strokeStyle = 'blue'; ctx.lineCap = 'round'; ctx.beginPath(); ctx.moveTo(50, 150); ctx.lineTo(150, 250); ctx.lineTo(250, 150); ctx.stroke(); // Square lineCap ctx.strokeStyle = 'green'; ctx.lineCap = 'square'; ctx.beginPath(); ctx.moveTo(50, 250); ctx.lineTo(150, 350); ctx.lineTo(250, 250); ctx.stroke(); </script> </body> </html>
This example draws three zigzag paths with different lineCap values. Each path consists of two connected line segments forming a "V" shape.
The red path (butt) shows flat ends at the start and end points. The blue path (round) has rounded ends, while green (square) has squared-off ends extending beyond the path's endpoints.
Practical Application: Custom Arrowheads
This example shows how to use lineCap to create custom arrowheads.
<!DOCTYPE html> <html> <head> <title>Custom Arrowheads</title> </head> <body> <canvas id="myCanvas" width="400" height="200"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Draw arrow with round lineCap function drawArrow(x, y, length, color) { ctx.strokeStyle = color; ctx.lineWidth = 8; ctx.lineCap = 'round'; // Shaft ctx.beginPath(); ctx.moveTo(x, y); ctx.lineTo(x + length - 20, y); ctx.stroke(); // Arrowhead ctx.beginPath(); ctx.moveTo(x + length - 30, y - 15); ctx.lineTo(x + length, y); ctx.lineTo(x + length - 30, y + 15); ctx.stroke(); } drawArrow(50, 50, 300, 'blue'); drawArrow(50, 100, 250, 'red'); drawArrow(50, 150, 200, 'green'); </script> </body> </html>
This example creates a reusable arrow-drawing function that uses round lineCap for both the shaft and arrowhead. The round ends create a polished look.
The function draws the arrow shaft and then the arrowhead as two separate paths. The round lineCap gives the arrowhead a smooth appearance at the tip.
Source
MDN Canvas lineCap Documentation
In this article, we have explored the lineCap property and its effects on different types of canvas drawings. Mastering lineCap helps create more polished and professional-looking graphics in web applications.
Author
List all JS Canvas tutorials.