Skip to content

Latest commit

 

History

History

canvas

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 

Screen Buffers, Canvas, and RGB

React and Animated Canvases

How do you animate a React component?

We know the component has a lifecycle and a render() method and it gets created when it is rendered or the state changes.

But what if we want to update the component several times per second?

requestAnimationFrame()

There's a way in Javascript to get "frame-by-frame" callbacks. That is, once per "frame", however long that is, you'll get a callback. It'll probably be ~30 times per second. This makes it very suitable for doing animations, as the name suggests.

For every one time you call requestAnimationFrame(), your given callback will be called. It's really common for the callback to call requestAnimationFrame() to get another frame as soon as possible.

let prevTimestamp = null;

function onAnimFrame(timestamp) {

    // Request another animation frame for the future
    requestAnimationFrame(onAnimFrame);

    // If we haven't yet stored the previous time, fake it
    if (prevTimestamp === null) {
        prevTimestamp = timestamp - 30; // milliseconds
    }

    // Compute how long it took between frames
    const elapsed = timestamp - prevTimestamp

    // Remember this for next frame
    prevTimestamp = timestamp;

    console.log(`Current time: ${timestamp} ms, frame time: ${elapsed} ms`);

    // TODO: Do animation stuff to the canvas
}

// Request the first animation frame to kick things off
requestAnimationFrame(onAnimFrame);

We'll be using this to change the contents in the canvas within the React component.

Canvas in the React Component

Having a canvas in a react component is pretty easy:

class MyComponent extends Component {

    // ...

    /**
     * Render the canvas
     */
    render() {
        return <canvas ref="canvas" width={this.props.width} height={this.props.height} />
    }

    // ...
}

What's that weird ref="canvas" bit?

Turns out we can use that later in JS code to get a reference to the canvas that was rendered. This will be very useful when we actually want to draw on it.

Inside the requestAnimationFrame() handler, you can refer to the canvas with:

const canvas = this.refs.canvas; // refers to the ref attribute in render()

const context = canvas.getContext('2d'); // etc.

Canvas in React with requestAnimationFrame()

How do we merge these two things together to get animation?

We need to call requestAnimationFrame() one time to kick off the process, and then we need to call it from our callback if we're still showing the component.

Fortunately, in the React component lifecycle, there are a couple handlers we can make use of to control the process.

We'll request an initial animation frame in the componentDidMount() handler, and we'll stop animating in the componentWillUnmount() handler.

All componentDidMount() needs to do is call requestAnimationFrame(). But since the decision to keep animating or not is made in the callback, componentWillUnmount() just needs to set a flag to stop animating.

Exercises

Implement a React App with Animated Canvas

class MyComponent extends Component {

    /**
     * Constructor
     */
    constructor(props) {
        super(props);

        this.continueAnimation = true;
    }

    /**
     * After the component has mounted
     */
    componentDidMount() {
        // Request initial animation frame
        requestAnimationFrame((timestamp) => { this.onAnimFrame(timestamp); });
    }

    /**
     * When the component is about to unmount
     */
    componentWillUnmount() {
        // Stop animating
        this.continueAnimation = false;
    }

    /**
     * Called every frame of animation
     */
    onAnimFrame(timestamp) {
        // If desired, request another anim frame for later
        if (this.continueAnimation) {
            requestAnimationFrame((timestamp) => { this.onAnimFrame(timestamp); });
        }

        // TODO animate stuff
    }

    /**
     * Render the canvas
     */
    render() {
        return <canvas ref="canvas" width={this.props.width} height={this.props.height} />
    }
}

Animate a Pixel Across the Screen

Using the above code, move a pixel repeatedly across the canvas.

Using math, make it take exactly 1 second for the pixel to move across the canvas.