JavaScript Canvas Save Tutorial
last modified April 3, 2025
In this article, we explore the Canvas save and restore methods in JavaScript. These methods are essential for managing the drawing state when working with complex canvas graphics. Mastering state management is crucial for efficient canvas programming.
Basic Definition
Canvas save and restore refer to methods that manage the drawing state stack.
The save()
method pushes the current state onto the stack, while
restore()
pops the most recently saved state from the stack.
The drawing state includes transformations, styles, clipping paths, and more. These methods allow temporary changes without affecting subsequent drawings. They are particularly useful in complex drawing operations.
Basic Save and Restore
This example demonstrates the basic usage of save and restore methods.
<!DOCTYPE html> <html> <head> <title>Basic Canvas Save/Restore</title> </head> <body> <canvas id="myCanvas" width="300" height="200"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Original state ctx.fillStyle = 'blue'; ctx.fillRect(10, 10, 100, 50); // Save state ctx.save(); // Modify state ctx.fillStyle = 'red'; ctx.fillRect(130, 10, 100, 50); // Restore state ctx.restore(); // Uses original blue color ctx.fillRect(250, 10, 100, 50); </script> </body> </html>
This example shows how save and restore affect the drawing state. We first
draw a blue rectangle, then save the current state with save()
.
After changing the fill style to red and drawing another rectangle, we call
restore()
to return to the previous state. The third rectangle
uses the original blue color.
Nested Save and Restore
This example demonstrates nesting multiple save and restore operations.
<!DOCTYPE html> <html> <head> <title>Nested Save/Restore</title> </head> <body> <canvas id="myCanvas" width="400" height="300"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Initial state ctx.fillStyle = 'black'; ctx.fillRect(10, 10, 100, 50); // First save ctx.save(); ctx.fillStyle = 'blue'; ctx.fillRect(130, 10, 100, 50); // Second save ctx.save(); ctx.fillStyle = 'red'; ctx.fillRect(250, 10, 100, 50); // First restore (back to blue) ctx.restore(); ctx.fillRect(10, 80, 100, 50); // Second restore (back to black) ctx.restore(); ctx.fillRect(130, 80, 100, 50); </script> </body> </html>
This example shows how save and restore operations work in a stack (LIFO) order. We first save the black fill state, then change to blue and save again.
After changing to red and drawing, we restore twice. The first restore returns to blue, and the second returns to the original black state. This demonstrates the nested nature of state management.
Save/Restore with Transformations
This example shows how save and restore work with canvas transformations.
<!DOCTYPE html> <html> <head> <title>Save/Restore with Transformations</title> </head> <body> <canvas id="myCanvas" width="400" height="300"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Draw original rectangle ctx.fillStyle = 'blue'; ctx.fillRect(50, 50, 100, 50); // Save state ctx.save(); // Apply transformation ctx.translate(100, 100); ctx.rotate(Math.PI/4); ctx.fillStyle = 'red'; ctx.fillRect(50, 50, 100, 50); // Restore state ctx.restore(); // Draw with original transformation ctx.fillStyle = 'green'; ctx.fillRect(200, 50, 100, 50); </script> </body> </html>
This example demonstrates that save and restore affect not just styles but also transformations. We first draw a blue rectangle in the default state.
After saving, we apply translation and rotation transformations and draw a red rectangle. After restoring, the green rectangle is drawn without these transformations, showing the state was properly restored.
Save/Restore with Clipping
This example demonstrates using save and restore with clipping paths.
<!DOCTYPE html> <html> <head> <title>Save/Restore with Clipping</title> </head> <body> <canvas id="myCanvas" width="400" height="300"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Draw without clipping ctx.fillStyle = 'blue'; ctx.fillRect(50, 50, 300, 200); // Save state ctx.save(); // Create clipping path ctx.beginPath(); ctx.arc(200, 150, 100, 0, Math.PI * 2); ctx.clip(); // Draw with clipping ctx.fillStyle = 'red'; ctx.fillRect(50, 50, 300, 200); // Restore state (removes clipping) ctx.restore(); // Draw without clipping again ctx.fillStyle = 'green'; ctx.fillRect(50, 50, 100, 100); </script> </body> </html>
This example shows how clipping paths are affected by save and restore. We first draw a blue rectangle without any clipping. Then we save the state.
After creating a circular clipping path, we draw a red rectangle that only appears within the clip. After restoring, the green rectangle is drawn without clipping, demonstrating the state was properly restored.
Advanced State Management
This example shows complex state management with multiple properties.
<!DOCTYPE html> <html> <head> <title>Advanced State Management</title> </head> <body> <canvas id="myCanvas" width="500" height="400"></canvas> <script> const canvas = document.getElementById('myCanvas'); const ctx = canvas.getContext('2d'); // Set initial styles ctx.fillStyle = 'blue'; ctx.strokeStyle = 'black'; ctx.lineWidth = 2; ctx.font = '20px Arial'; // Draw with initial state ctx.fillRect(50, 50, 100, 50); ctx.strokeRect(50, 50, 100, 50); ctx.fillText('Initial', 50, 130); // Save state ctx.save(); // Modify multiple properties ctx.fillStyle = 'red'; ctx.strokeStyle = 'green'; ctx.lineWidth = 5; ctx.font = 'italic 16px Times'; // Draw with modified state ctx.fillRect(200, 50, 100, 50); ctx.strokeRect(200, 50, 100, 50); ctx.fillText('Modified', 200, 130); // Save nested state ctx.save(); // Modify again ctx.fillStyle = 'yellow'; ctx.strokeStyle = 'purple'; ctx.lineWidth = 1; // Draw with second modified state ctx.fillRect(350, 50, 100, 50); ctx.strokeRect(350, 50, 100, 50); ctx.fillText('Nested', 350, 130); // Restore to first modified state ctx.restore(); ctx.fillRect(200, 200, 100, 50); ctx.strokeRect(200, 200, 100, 50); // Restore to initial state ctx.restore(); ctx.fillRect(50, 200, 100, 50); ctx.strokeRect(50, 200, 100, 50); </script> </body> </html>
This comprehensive example demonstrates how save and restore affect multiple drawing state properties simultaneously. We modify fill, stroke, line width, and font properties at different levels.
The example shows nested state management, with two save operations and corresponding restores. Each restore returns to exactly the state that was saved, including all style properties and transformations.
Source
In this article, we have explored various techniques for managing canvas state using save and restore methods. These powerful tools are essential for complex canvas drawings and animations in web applications.
Author
List all JS Canvas tutorials.