npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

camera-capture

v0.0.11

Published

Super portable, fast camera capture library for node.js (server). TypeScript/JavaScript easy to use APIs. Uses puppeteer headless browser to capture webcam video (audio/desktop, recording, etc) and stream back to node.js frame by frame in plain image data

Downloads

64

Readme

camera-capture

Portable Camera, audio, desktop capture Node.js library.

Contents

What / Why ?

After searching for an easy to use portable library to access the webcam directly from node.js I didn't found a library that works in windows, macOs and linux, without native dependencies that users need ot manually install (or even so, they won't work).

This library solves the problem with an easy approach. Use headless browser to capture the video, draw in canvas and pass the image data the Node.js context as fast as possible (age.exposeFunction()) and with minimal processing. It uses HTMLCanvasElement getImageData when returning raw image data or HTMLCanvasElement.toBlob() when retuning encoded images such as png, jpg. In both cases using ArrayBuffer

Install

npm install camera-capture puppeteer

(puppeteer is a peer dependency you must install it by yourself)

JavaScript API

Managed frame read

import {VideoCapture} from 'camera-capture'
const c = new VideoCapture()
c.addFrameListener(frame => {  
  // frame by default is unencoded raw Image Data `{width: 480, height: 320, data: UIntArray}``
  // which is often what image processing / surfaces interfaces expect for fast processing. 
  // Use `mime` option to receive it in other formats (see examples below)
  surface.putImageData(0,0,frame.width, frame.height, frame.data)
})
// pause / resume frame emission (without tunning off the camera)
setTimeout(()=>c.pause(), 1000)
setTimeout(()=>c.resume(), 2000)
// shutdown everything, including, camera, browser, server:
setTimeout(()=>c.stop(), 3000)
console.log('Capturing camera');
await c.start() // promise will be resolved only when `stop`
console.log('Stopping camera capture');

Manual frame read

Instead of using start() and being notified on each frame, just call initialize() and read frames programmatically:

import {VideoCapture} from 'camera-capture'
const c = new VideoCapture({
  mime: 'image/png'
})
await c.initialize()
let f = await c.readFrame()               // PNG as configured
writeFileSync('tmp.png', f.data)
f = await c.readFrame('image/webp')       // take another shot this time as webp image
writeFileSync('tmp.webp', f.data)
f = await c.readFrame('image/jpeg') // jpeg
writeFileSync('tmp.jpg', f.data)
f = await c.readFrame('rgba')       // raw image data (as default)
writeFileSync('tmp-8bit-200x200.rgba', f.data)

Recording camera video

The following uses DOM MediaRecorder API to record video. Notice that it all happens in the browser, on memory, so the result is a excellent quality video but it could consume lots of memory on long recordings. If that's an issue perhaps it's better to store frame by frame to hard drive and then use a video assembler like ffmpeg / imagemagick. (in the roadmap):

import {VideoCapture} from 'camera-capture'
const c = new VideoCapture({ port: 8082 })
await c.initialize()
await c.startRecording()
await sleep(500)
const data = await c.stopRecording()
writeFileSync('tmp6.webm', data)

Command line

TODO - TBD

Summary

I didn't found any library that provides an interface to capture webcam video so I show the video and filter frame by frame in my Node.s desktop app (not based on electron - no canvas / HTML5 available - rendering on cairo/opengl surface that complies with

  • Don't require users to install native complex dependencies (like opencv or native applications installed)
  • Don't include any binary code that needs to be compiled.
  • works on windows, macOs, and linux
  • provides a stream-like API for video frames
  • fast so it can be used for a "real-time" video filter demo
  • usable without electron/canvas/html5 - imagine I'm rendering in a native surface like cairo, gtk, etc
  • portable - no surprises - working in latest node.js versions
  • Optionally the frames can be encoded as in jpg/png or even a video created .
  • Also provides simple filtering API.

Design summary

  • Use puppeteer (which is google chrome headless browser) to capture camera video. Expose frames as fast as possible.
  • not focused on encoding more than the ones supported by the browser
  • API based on raw image data - users responsible of compose an output video with ffmpeg, imagemagick, opencv, etc. Format encoding is not the objective of this project

Status

Observed behavior:

About, 30 frames per second (size 600x400, format: raw image data)

  • JavaScript API (managing the capturing loop)
  • javaScript API (manual capture)
  • image encoded as jpeg, png, webp
  • camera video recording using DOM MediaRecorder

Reference API

TODO / Road map

  • [ ] investigate why/how to pass the buffer / array buffer view directly without transforming it to number[] / and array buffer views
    • se TextEncoder/TextDecoder to serialize the data as a single char-per-byte string (using windows-1252 encoding) and deserialize it in Node on the other side which is fast (since passing strings is much faster).
  • [ ] test if toDataUrl is faster than toBlob
  • [ ] probably for frames a generator / or observable is more appropriate than even listeners.
  • [ ] perhaps is faster to do the capture loop all together inside the DOM, instead calling evaluate() on each iteration?
  • [ ] CLI
  • [ ] pause/resume / start/stop should work for recording too.
  • [ ] performance tests (fps raw image data and encoded images)
  • [ ] do we really need to serialize constrains ?
  • [ ] video recording formats other than webm?
  • [ ] video recording constraints - size -
  • [ ] audio recording only API
  • [ ] record desktop ? possible ?
  • [ ] desktop screenshot only API
  • [ ] browser screenshot only API
  • [ ] webcam screenshot only API
  • [ ] geo location (get the coords) ? (need https?)
  • [ ] change video size dynamically ?
  • [x] check c.addFrameListener() with encoded images
  • [x] real world example: native app
  • [x] encode in browser supported formats (png, jpg)
  • [x] c.readFrame() users read manually instead listener - loop controlled by users.
  • [x] listener API managed loop
  • [x] API docs
  • [x] add api docs descriptions to class, options and
  • [x] record capture using dom api (output is mp4/avi video)

low priority

  • [ ] research how fast/slow is painting canvas pixel by pixel from image data than showImage in node-gui
  • [ ] TODO: support fps control like in opencv
  • [ ]