<cancas id="canvas"></canvas>
<script>
const canvas = document.getElementById("canvas");
const ctx = canvas.getContext("2d");
</script>
Matt Toegel matthew.toegel@njit.edu
Learn about the <canvas>
element
Utilize 2d
context to draw text/shapes
Leverage vanilla JavaScript to draw, animate, and respond to user input
http://vanilla-js.com/ (it’s a joke)
Review practical examples
Apply concepts to simple 2D browser games
Finale: Chicken Vs. Zombie
At face value it’s just an empty element, like a <div>
or a <span>
You can include content inside of the tag that’ll render if the browser lacks Canvas
support
The magic of it comes from the Canvas API
Default size is 300x150 pixels
Supported in all major/modern browsers (basically > IE for the most part)
JavaScript interacts with the <canvas>
by getting a reference
document.getElementById("canvasId")
for example
From that reference we get the context
element.getContext('2d')
Canvas supports both 2d and 3d (webgl) but the latter is more complex with vanilla JavaScript
With the context reference the Canvas API will be used
<cancas id="canvas"></canvas>
<script>
const canvas = document.getElementById("canvas");
const ctx = canvas.getContext("2d");
</script>
Note: It’s best to check if canvas was found and if the browser supports getContext
rather than just assuming
2D context is very simple; Uses
Lines
Stroke/Fill
Rects
Arcs
Paths/Curves
Images
Coordinate system is (0,0)
top-left (inf,inf)
bottom right
-x
is left, +x
is right
-y
is up, +y
is down
More examples/reference: https://developer.mozilla.org/en-US/docs/Web/API/Canvas_API/Tutorial/Drawing_shapes
Use lineTo(x,y)
to draw a line from the current position to the destination x
,y
The start point depends on previously drawn paths or moveTo(x,y)
moveTo(x,y)
moves the "pen" to the x
,y
coordinate
These are usually used in conjunction with beginPath()
and stroke()
beginPath()
starts a new path (group of lines, shapes, curves)
Likewise, there’s a closePath()
, adds a straight line from the current point to the first path coordinate
stroke()
applies outline
fill()
fills a shape’s or path’s area
Stroke is used for individual line segmenets
Rectangles are drawn from top-left to bottom-right
The x
,y
refers to a corner rather than the center
fillRect(x, y, width, height)
draws a filled rectanlge starting at the top-left point extended the width and height (all in pixels)
strokeRect(x, y, width, height)
does the same except as an outline
clearRect(x, y, width, height)
clears an area making it transparent
Arcs (and circles) are centered at x
, y
arc(x, y, radius, startAngle, endAngle, counterclockwise)
draws an arc centered at x
,y
with a given radius; angles are in radians
radians = (Math.PI/180)*degrees
arcTo(x1, y1, x2, y2, radius)
draws an arc with control points
stroke()
and fill()
can be used for arcs
Text can be rendered to a Canvas
You’ll have to take care on how it is written/wrapped/etc
fillText(text, x, y, maxWidth [optional])
fills the text string at the coordinate (top-left of the bounding box); optionally uses maxWidth to handle scaling/kerning
This displays text as you’d normally expect to see it
strokeText(text, x, y, maxWidth [optional])
outlines the text string at the coordinate (top-left of the bounding box); optionally uses maxWidth to handle scaling/kerning
Can be styled
font
changes font family and some style (default is 10px sans-serif
)
textAlign
changes alignment (start
, end
, left
, right
, center
) (default is start
)
textBaseline
baseline alignment (top
, hanging
, middle
, alphabetic
, ideographic
, bottom
) (default is alphabetic
)
direction
directionality (ltr
, rtl
, inherit
) (default is inherit
)
There are a few ways to handle images, but I’ll keep this brief
An Image reference is required, for ease new Image()
and setting a source will be used
There are also many effects that can be applied, but will just cover the basic draw method
drawImage(image, x, y)
draws the image at the coordinate (top left)
drawImage(image, x, y, width, height)
draws the image with scaling
drawImage(image, sx, sy, sWidth, sHeight, dx, dy, dWidth, dHeight)
draws an image from source coord/area to destination coord/area (scales if size differs)
Maintains aspect ratio
Can be used with spritesheets for animation
Note: drawImage()
is quite slower than the other shape methods
What is an animation really?
Animations are basically separate frames that are drawn sequentionally; fast enough to trick our eyes/brain into seeing movement
Generally this is 24 FPS or higher for "smooth" animation
There are primarily two ways to do animation
setInterval()
to run a function at a set tick rate
requestAnimationFrame()
browser controlled animation tick rate (typically display refresh rate)
Uses a timestamp to calculate time delta between frames for frame rate independent animation
Essentially we want a loop to do the following
Update state
Clear viewport
draw current state
Uses frame time delta for decoupled animation
Let’s tie in user input along with our drawing and animation topics
Use left/right or a/d to move a paddle to catch falling objects
Use left/right or a/d to move player to avoid falling objects
Use space to shoot
This last example is a more complete game that leverages numerous spritesheets and mechanics
Premise:
You play as a chicken that lays explosive eggs with space bar
You’re objective is to defeat waves of zombies
Defeating zombies earns points to upgrade your skills
Play until you lose
Repo Link: https://github.com/MattToegel/html5-canvas/tree/main/cvz
To further extend our capabilities
Canvas API supports controllers and joysticks
WebSockets can provide multiplayer capabilities
Database services/platforms can be used to save/load game state, player state, etc
Use a PWA web app or similar you can deploy your Canvas program to multiple devices like consoles