ngx-webcam
Version:
A simple Angular webcam component. Pure & minimal, no Flash-fallback. <a href="https://basst314.github.io/ngx-webcam/?" target="_blank">See the Demo!</a>
516 lines (507 loc) • 27.5 kB
JavaScript
import * as i0 from '@angular/core';
import { EventEmitter, Component, Input, Output, ViewChild, NgModule } from '@angular/core';
import * as i1 from '@angular/common';
import { CommonModule } from '@angular/common';
/**
* Container class for a captured webcam image
* @author basst314, davidshen84
*/
class WebcamImage {
constructor(imageAsDataUrl, mimeType, imageData) {
this._mimeType = null;
this._imageAsBase64 = null;
this._imageAsDataUrl = null;
this._imageData = null;
this._mimeType = mimeType;
this._imageAsDataUrl = imageAsDataUrl;
this._imageData = imageData;
}
/**
* Extracts the Base64 data out of the given dataUrl.
* @param dataUrl the given dataUrl
* @param mimeType the mimeType of the data
*/
static getDataFromDataUrl(dataUrl, mimeType) {
return dataUrl.replace(`data:${mimeType};base64,`, '');
}
/**
* Get the base64 encoded image data
* @returns base64 data of the image
*/
get imageAsBase64() {
return this._imageAsBase64 ? this._imageAsBase64
: this._imageAsBase64 = WebcamImage.getDataFromDataUrl(this._imageAsDataUrl, this._mimeType);
}
/**
* Get the encoded image as dataUrl
* @returns the dataUrl of the image
*/
get imageAsDataUrl() {
return this._imageAsDataUrl;
}
/**
* Get the ImageData object associated with the canvas' 2d context.
* @returns the ImageData of the canvas's 2d context.
*/
get imageData() {
return this._imageData;
}
}
class WebcamUtil {
/**
* Lists available videoInput devices
* @returns a list of media device info.
*/
static getAvailableVideoInputs() {
if (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {
return Promise.reject('enumerateDevices() not supported.');
}
return new Promise((resolve, reject) => {
navigator.mediaDevices.enumerateDevices()
.then((devices) => {
resolve(devices.filter((device) => device.kind === 'videoinput'));
})
.catch(err => {
reject(err.message || err);
});
});
}
}
class WebcamComponent {
constructor() {
/** Defines the max width of the webcam area in px */
this.width = 640;
/** Defines the max height of the webcam area in px */
this.height = 480;
/** Defines base constraints to apply when requesting video track from UserMedia */
this.videoOptions = WebcamComponent.DEFAULT_VIDEO_OPTIONS;
/** Flag to enable/disable camera switch. If enabled, a switch icon will be displayed if multiple cameras were found */
this.allowCameraSwitch = true;
/** Flag to control whether an ImageData object is stored into the WebcamImage object. */
this.captureImageData = false;
/** The image type to use when capturing snapshots */
this.imageType = WebcamComponent.DEFAULT_IMAGE_TYPE;
/** The image quality to use when capturing snapshots (number between 0 and 1) */
this.imageQuality = WebcamComponent.DEFAULT_IMAGE_QUALITY;
/** EventEmitter which fires when an image has been captured */
this.imageCapture = new EventEmitter();
/** Emits a mediaError if webcam cannot be initialized (e.g. missing user permissions) */
this.initError = new EventEmitter();
/** Emits when the webcam video was clicked */
this.imageClick = new EventEmitter();
/** Emits the active deviceId after the active video device was switched */
this.cameraSwitched = new EventEmitter();
/** available video devices */
this.availableVideoInputs = [];
/** Indicates whether the video device is ready to be switched */
this.videoInitialized = false;
/** Index of active video in availableVideoInputs */
this.activeVideoInputIndex = -1;
/** MediaStream object in use for streaming UserMedia data */
this.mediaStream = null;
/** width and height of the active video stream */
this.activeVideoSettings = null;
}
/**
* If the given Observable emits, an image will be captured and emitted through 'imageCapture' EventEmitter
*/
set trigger(trigger) {
if (this.triggerSubscription) {
this.triggerSubscription.unsubscribe();
}
// Subscribe to events from this Observable to take snapshots
this.triggerSubscription = trigger.subscribe(() => {
this.takeSnapshot();
});
}
/**
* If the given Observable emits, the active webcam will be switched to the one indicated by the emitted value.
* @param switchCamera Indicates which webcam to switch to
* true: cycle forwards through available webcams
* false: cycle backwards through available webcams
* string: activate the webcam with the given id
*/
set switchCamera(switchCamera) {
if (this.switchCameraSubscription) {
this.switchCameraSubscription.unsubscribe();
}
// Subscribe to events from this Observable to switch video device
this.switchCameraSubscription = switchCamera.subscribe((value) => {
if (typeof value === 'string') {
// deviceId was specified
this.switchToVideoInput(value);
}
else {
// direction was specified
this.rotateVideoInput(value !== false);
}
});
}
/**
* Get MediaTrackConstraints to request streaming the given device
* @param deviceId
* @param baseMediaTrackConstraints base constraints to merge deviceId-constraint into
* @returns
*/
static getMediaConstraintsForDevice(deviceId, baseMediaTrackConstraints) {
const result = baseMediaTrackConstraints ? baseMediaTrackConstraints : this.DEFAULT_VIDEO_OPTIONS;
if (deviceId) {
result.deviceId = { exact: deviceId };
}
return result;
}
/**
* Tries to harvest the deviceId from the given mediaStreamTrack object.
* Browsers populate this object differently; this method tries some different approaches
* to read the id.
* @param mediaStreamTrack
* @returns deviceId if found in the mediaStreamTrack
*/
static getDeviceIdFromMediaStreamTrack(mediaStreamTrack) {
if (mediaStreamTrack.getSettings && mediaStreamTrack.getSettings() && mediaStreamTrack.getSettings().deviceId) {
return mediaStreamTrack.getSettings().deviceId;
}
else if (mediaStreamTrack.getConstraints && mediaStreamTrack.getConstraints() && mediaStreamTrack.getConstraints().deviceId) {
const deviceIdObj = mediaStreamTrack.getConstraints().deviceId;
return WebcamComponent.getValueFromConstrainDOMString(deviceIdObj);
}
}
/**
* Tries to harvest the facingMode from the given mediaStreamTrack object.
* Browsers populate this object differently; this method tries some different approaches
* to read the value.
* @param mediaStreamTrack
* @returns facingMode if found in the mediaStreamTrack
*/
static getFacingModeFromMediaStreamTrack(mediaStreamTrack) {
if (mediaStreamTrack) {
if (mediaStreamTrack.getSettings && mediaStreamTrack.getSettings() && mediaStreamTrack.getSettings().facingMode) {
return mediaStreamTrack.getSettings().facingMode;
}
else if (mediaStreamTrack.getConstraints && mediaStreamTrack.getConstraints() && mediaStreamTrack.getConstraints().facingMode) {
const facingModeConstraint = mediaStreamTrack.getConstraints().facingMode;
return WebcamComponent.getValueFromConstrainDOMString(facingModeConstraint);
}
}
}
/**
* Determines whether the given mediaStreamTrack claims itself as user facing
* @param mediaStreamTrack
*/
static isUserFacing(mediaStreamTrack) {
const facingMode = WebcamComponent.getFacingModeFromMediaStreamTrack(mediaStreamTrack);
return facingMode ? 'user' === facingMode.toLowerCase() : false;
}
/**
* Extracts the value from the given ConstrainDOMString
* @param constrainDOMString
*/
static getValueFromConstrainDOMString(constrainDOMString) {
if (constrainDOMString) {
if (constrainDOMString instanceof String) {
return String(constrainDOMString);
}
else if (Array.isArray(constrainDOMString) && Array(constrainDOMString).length > 0) {
return String(constrainDOMString[0]);
}
else if (typeof constrainDOMString === 'object') {
if (constrainDOMString['exact']) {
return String(constrainDOMString['exact']);
}
else if (constrainDOMString['ideal']) {
return String(constrainDOMString['ideal']);
}
}
}
return null;
}
ngAfterViewInit() {
this.detectAvailableDevices()
.then(() => {
// start video
this.switchToVideoInput(null);
})
.catch((err) => {
this.initError.next({ message: err });
// fallback: still try to load webcam, even if device enumeration failed
this.switchToVideoInput(null);
});
}
ngOnDestroy() {
this.stopMediaTracks();
this.unsubscribeFromSubscriptions();
}
/**
* Takes a snapshot of the current webcam's view and emits the image as an event
*/
takeSnapshot() {
// set canvas size to actual video size
const _video = this.nativeVideoElement;
const dimensions = { width: this.width, height: this.height };
if (_video.videoWidth) {
dimensions.width = _video.videoWidth;
dimensions.height = _video.videoHeight;
}
const _canvas = this.canvas.nativeElement;
_canvas.width = dimensions.width;
_canvas.height = dimensions.height;
// paint snapshot image to canvas
const context2d = _canvas.getContext('2d');
context2d.drawImage(_video, 0, 0);
// read canvas content as image
const mimeType = this.imageType ? this.imageType : WebcamComponent.DEFAULT_IMAGE_TYPE;
const quality = this.imageQuality ? this.imageQuality : WebcamComponent.DEFAULT_IMAGE_QUALITY;
const dataUrl = _canvas.toDataURL(mimeType, quality);
// get the ImageData object from the canvas' context.
let imageData = null;
if (this.captureImageData) {
imageData = context2d.getImageData(0, 0, _canvas.width, _canvas.height);
}
this.imageCapture.next(new WebcamImage(dataUrl, mimeType, imageData));
}
/**
* Switches to the next/previous video device
* @param forward
*/
rotateVideoInput(forward) {
if (this.availableVideoInputs && this.availableVideoInputs.length > 1) {
const increment = forward ? 1 : (this.availableVideoInputs.length - 1);
const nextInputIndex = (this.activeVideoInputIndex + increment) % this.availableVideoInputs.length;
this.switchToVideoInput(this.availableVideoInputs[nextInputIndex].deviceId);
}
}
/**
* Switches the camera-view to the specified video device
*/
switchToVideoInput(deviceId) {
this.videoInitialized = false;
this.stopMediaTracks();
this.initWebcam(deviceId, this.videoOptions);
}
/**
* Event-handler for video resize event.
* Triggers Angular change detection so that new video dimensions get applied
*/
videoResize() {
// here to trigger Angular change detection
}
get videoWidth() {
const videoRatio = this.getVideoAspectRatio();
return Math.min(this.width, this.height * videoRatio);
}
get videoHeight() {
const videoRatio = this.getVideoAspectRatio();
return Math.min(this.height, this.width / videoRatio);
}
get videoStyleClasses() {
let classes = '';
if (this.isMirrorImage()) {
classes += 'mirrored ';
}
return classes.trim();
}
get nativeVideoElement() {
return this.video.nativeElement;
}
/**
* Returns the video aspect ratio of the active video stream
*/
getVideoAspectRatio() {
// calculate ratio from video element dimensions if present
const videoElement = this.nativeVideoElement;
if (videoElement.videoWidth && videoElement.videoWidth > 0 &&
videoElement.videoHeight && videoElement.videoHeight > 0) {
return videoElement.videoWidth / videoElement.videoHeight;
}
// nothing present - calculate ratio based on width/height params
return this.width / this.height;
}
/**
* Init webcam live view
*/
initWebcam(deviceId, userVideoTrackConstraints) {
const _video = this.nativeVideoElement;
if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
// merge deviceId -> userVideoTrackConstraints
const videoTrackConstraints = WebcamComponent.getMediaConstraintsForDevice(deviceId, userVideoTrackConstraints);
navigator.mediaDevices.getUserMedia({ video: videoTrackConstraints })
.then((stream) => {
this.mediaStream = stream;
_video.srcObject = stream;
_video.play();
this.activeVideoSettings = stream.getVideoTracks()[0].getSettings();
const activeDeviceId = WebcamComponent.getDeviceIdFromMediaStreamTrack(stream.getVideoTracks()[0]);
this.cameraSwitched.next(activeDeviceId);
// Initial detect may run before user gave permissions, returning no deviceIds. This prevents later camera switches. (#47)
// Run detect once again within getUserMedia callback, to make sure this time we have permissions and get deviceIds.
this.detectAvailableDevices()
.then(() => {
this.activeVideoInputIndex = activeDeviceId ? this.availableVideoInputs
.findIndex((mediaDeviceInfo) => mediaDeviceInfo.deviceId === activeDeviceId) : -1;
this.videoInitialized = true;
})
.catch(() => {
this.activeVideoInputIndex = -1;
this.videoInitialized = true;
});
})
.catch((err) => {
this.initError.next({ message: err.message, mediaStreamError: err });
});
}
else {
this.initError.next({ message: 'Cannot read UserMedia from MediaDevices.' });
}
}
getActiveVideoTrack() {
return this.mediaStream ? this.mediaStream.getVideoTracks()[0] : null;
}
isMirrorImage() {
if (!this.getActiveVideoTrack()) {
return false;
}
// check for explicit mirror override parameter
{
let mirror = 'auto';
if (this.mirrorImage) {
if (typeof this.mirrorImage === 'string') {
mirror = String(this.mirrorImage).toLowerCase();
}
else {
// WebcamMirrorProperties
if (this.mirrorImage.x) {
mirror = this.mirrorImage.x.toLowerCase();
}
}
}
switch (mirror) {
case 'always':
return true;
case 'never':
return false;
}
}
// default: enable mirroring if webcam is user facing
return WebcamComponent.isUserFacing(this.getActiveVideoTrack());
}
/**
* Stops all active media tracks.
* This prevents the webcam from being indicated as active,
* even if it is no longer used by this component.
*/
stopMediaTracks() {
if (this.mediaStream && this.mediaStream.getTracks) {
// pause video to prevent mobile browser freezes
this.nativeVideoElement.pause();
// getTracks() returns all media tracks (video+audio)
this.mediaStream.getTracks()
.forEach((track) => track.stop());
}
}
/**
* Unsubscribe from all open subscriptions
*/
unsubscribeFromSubscriptions() {
if (this.triggerSubscription) {
this.triggerSubscription.unsubscribe();
}
if (this.switchCameraSubscription) {
this.switchCameraSubscription.unsubscribe();
}
}
/**
* Reads available input devices
*/
detectAvailableDevices() {
return new Promise((resolve, reject) => {
WebcamUtil.getAvailableVideoInputs()
.then((devices) => {
this.availableVideoInputs = devices;
resolve(devices);
})
.catch(err => {
this.availableVideoInputs = [];
reject(err);
});
});
}
}
WebcamComponent.DEFAULT_VIDEO_OPTIONS = { facingMode: 'environment' };
WebcamComponent.DEFAULT_IMAGE_TYPE = 'image/jpeg';
WebcamComponent.DEFAULT_IMAGE_QUALITY = 0.92;
WebcamComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: WebcamComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
WebcamComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.0.0", type: WebcamComponent, selector: "webcam", inputs: { width: "width", height: "height", videoOptions: "videoOptions", allowCameraSwitch: "allowCameraSwitch", mirrorImage: "mirrorImage", captureImageData: "captureImageData", imageType: "imageType", imageQuality: "imageQuality", trigger: "trigger", switchCamera: "switchCamera" }, outputs: { imageCapture: "imageCapture", initError: "initError", imageClick: "imageClick", cameraSwitched: "cameraSwitched" }, viewQueries: [{ propertyName: "video", first: true, predicate: ["video"], descendants: true, static: true }, { propertyName: "canvas", first: true, predicate: ["canvas"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"webcam-wrapper\" (click)=\"imageClick.next();\">\r\n <video #video [width]=\"videoWidth\" [height]=\"videoHeight\" [class]=\"videoStyleClasses\" autoplay muted playsinline (resize)=\"videoResize()\"></video>\r\n <div class=\"camera-switch\" *ngIf=\"allowCameraSwitch && availableVideoInputs.length > 1 && videoInitialized\" (click)=\"rotateVideoInput(true)\"></div>\r\n <canvas #canvas [width]=\"width\" [height]=\"height\"></canvas>\r\n</div>\r\n", styles: [".webcam-wrapper{display:inline-block;position:relative;line-height:0}.webcam-wrapper video.mirrored{transform:scaleX(-1)}.webcam-wrapper canvas{display:none}.webcam-wrapper .camera-switch{background-color:#0000001a;background-image:url();background-repeat:no-repeat;border-radius:5px;position:absolute;right:13px;top:10px;height:48px;width:48px;background-size:80%;cursor:pointer;background-position:center;transition:background-color .2s ease}.webcam-wrapper .camera-switch:hover{background-color:#0000002e}\n"], directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: WebcamComponent, decorators: [{
type: Component,
args: [{ selector: 'webcam', template: "<div class=\"webcam-wrapper\" (click)=\"imageClick.next();\">\r\n <video #video [width]=\"videoWidth\" [height]=\"videoHeight\" [class]=\"videoStyleClasses\" autoplay muted playsinline (resize)=\"videoResize()\"></video>\r\n <div class=\"camera-switch\" *ngIf=\"allowCameraSwitch && availableVideoInputs.length > 1 && videoInitialized\" (click)=\"rotateVideoInput(true)\"></div>\r\n <canvas #canvas [width]=\"width\" [height]=\"height\"></canvas>\r\n</div>\r\n", styles: [".webcam-wrapper{display:inline-block;position:relative;line-height:0}.webcam-wrapper video.mirrored{transform:scaleX(-1)}.webcam-wrapper canvas{display:none}.webcam-wrapper .camera-switch{background-color:#0000001a;background-image:url();background-repeat:no-repeat;border-radius:5px;position:absolute;right:13px;top:10px;height:48px;width:48px;background-size:80%;cursor:pointer;background-position:center;transition:background-color .2s ease}.webcam-wrapper .camera-switch:hover{background-color:#0000002e}\n"] }]
}], propDecorators: { width: [{
type: Input
}], height: [{
type: Input
}], videoOptions: [{
type: Input
}], allowCameraSwitch: [{
type: Input
}], mirrorImage: [{
type: Input
}], captureImageData: [{
type: Input
}], imageType: [{
type: Input
}], imageQuality: [{
type: Input
}], imageCapture: [{
type: Output
}], initError: [{
type: Output
}], imageClick: [{
type: Output
}], cameraSwitched: [{
type: Output
}], video: [{
type: ViewChild,
args: ['video', { static: true }]
}], canvas: [{
type: ViewChild,
args: ['canvas', { static: true }]
}], trigger: [{
type: Input
}], switchCamera: [{
type: Input
}] } });
const COMPONENTS = [
WebcamComponent
];
class WebcamModule {
}
WebcamModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: WebcamModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
WebcamModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: WebcamModule, declarations: [WebcamComponent], imports: [CommonModule], exports: [WebcamComponent] });
WebcamModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: WebcamModule, imports: [[
CommonModule
]] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: WebcamModule, decorators: [{
type: NgModule,
args: [{
imports: [
CommonModule
],
declarations: [
COMPONENTS
],
exports: [
COMPONENTS
]
}]
}] });
class WebcamInitError {
constructor() {
this.message = null;
this.mediaStreamError = null;
}
}
class WebcamMirrorProperties {
}
/**
* Generated bundle index. Do not edit.
*/
export { WebcamComponent, WebcamImage, WebcamInitError, WebcamMirrorProperties, WebcamModule, WebcamUtil };