Access the Camera in a PWA built with React

Chris Achard
InstructorChris Achard
Share this video with your friends

Social Share Links

Send Tweet

It's possible to access some, but not all, of the native device features from a PWA. One that we can access, is the camera and camera roll. We'll use two different methods to allow users to upload images to our application.

First, we'll add a file input that accepts images. We can specify a capture param, which will tell PWAs on mobile devices to prompt the user to take a picture with their camera.

Next, we'll access the camera feed directly by rendering a video tag, and accessing navigator.mediaDevices.getUserMedia directly - which will allow us to show a live video feed from the camera. Then, we can use a canvas tag to grab a frame from that video, and render the image.

Instructor: [00:00] In app.js, on the profile page, we can add a file and put component that accepts images, which is the easiest way to add an image upload to a PWA. Let's make the changeImage function. For now, we'll just have a bit of state called image, which we'll start as null.

[00:25] We'll make an object URL from the inputs file that was uploaded. That will allow us to display it directly as a user's profile image. When we build and start that in a browser, we can go to the profile page and upload a new image to be displayed as the profile image.

[00:50] When the PWA is installed on a mobile device, that will also allow the user to upload from the camera roll. If we want it to default to the camera instead of to photos, we can add a capture prop to the file input. Since this is for a user profile image, we can even specify that we want to default to the front-facing camera and specify user as the capture prop.

[01:12] That's the easiest way to get an image upload into our app. To interact more directly with the camera, first, we have to check that the browser supports the camera, by checking that media devices is in navigator. Then we can change the file upload to just a button, and make a new onClick function.

[01:36] Let's make a new onClick function called startChangeImage. In that function, we'll set some state to turn on the camera, and when that state gets set, we can render a new video component. That's where we'll show the live camera feed.

[01:54] When that video mounts, we need to save the ref. We'll also immediately ask for access to the user's camera. On navigator.media-devices, call that getUserMedia function, and tell it that we want video. Then we can set the source of a new video tag to the stream that's coming from the user's media device.

[02:19] This will show a popup which will ask for permission to access the camera. Make sure you only do this step when you're really ready to show the video feed, or else the user will just close your app.

[02:32] Now, we can display a button that will actually take an image. To get the image out of the video stream, we'll also need a blank canvas element. Let's put an invisible one here as well, and save the ref.

[02:49] Now, in the takeImageOnClick function, first, we'll set the height and the width of the canvas to the current video size. Then we can draw the image into the canvas directly from the video. Next, we'll tell the video to stop recording, and that will shut the user's camera off.

[03:13] Finally, we can set the image in our state to the canvas-context-image as a data URL, and turn off the video in our app by turning off the enableCamera bit and state. Now, let's build that and serve it to test it out.

[03:32] In a browser, we can click to toggle the camera, and we see a video feed. Then, when we click to take a picture, the picture is rendered to the canvas, and transferred to our state, and shown on the screen.

[03:46] The same process works for mobile devices as well, but it will only work over HTTPS. If we just serve this on our local computer over a local network, then it will never ask for camera access. Keep that in mind during development.