ngx-face-api-js
Version:
Angular directives for face detection and face recognition in the browser. It is a wrapper for face-api.js, so it is not dependent on the browser implementation.
191 lines • 15.8 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
import * as tslib_1 from "tslib";
import { Component, ViewChild, ElementRef, Renderer2, HostListener, } from '@angular/core';
import { Subscription, Subject, combineLatest } from 'rxjs';
import * as faceapi from 'face-api.js';
import { map, startWith } from 'rxjs/operators';
import { DetectTask } from '../../classes/DetectTask';
import { FaceDetectorService } from '../../services/face-detector.service';
export class DetectionResultComponent {
/**
* @param {?} task
* @param {?} el
* @param {?} renderer
* @param {?} faceDetector
*/
constructor(task, el, renderer, faceDetector) {
this.task = task;
this.el = el;
this.renderer = renderer;
this.faceDetector = faceDetector;
this.subscription = new Subscription();
this.resize$ = new Subject();
}
/**
* @private
* @return {?}
*/
get canvas() {
return this.canvasEl.nativeElement;
}
/**
* @return {?}
*/
onResize() {
this.resize$.next('onResize');
}
/**
* @private
* @param {?} result
* @return {?}
*/
convertResultToArray(result) {
if (Array.isArray(result)) {
return result;
}
return [result];
}
/**
* @return {?}
*/
ngOnInit() {
this.subscription.add(combineLatest(this.faceDetector.detect(this.task), this.resize$.pipe(startWith('init')))
.pipe(map((/**
* @param {?} __0
* @return {?}
*/
([result]) => this.convertResultToArray(result))))
.subscribe((/**
* @param {?} result
* @return {?}
*/
result => this.draw(result))));
}
/**
* @return {?}
*/
ngOnDestroy() {
this.subscription.unsubscribe();
}
/**
* @private
* @param {?} results
* @return {?}
*/
draw(results) {
return tslib_1.__awaiter(this, void 0, void 0, function* () {
/** @type {?} */
const target = yield this.task.target;
let { width, height } = target;
if (target instanceof HTMLVideoElement) {
height = target.videoHeight;
width = target.videoWidth;
}
/** @type {?} */
const detectionsForSize = faceapi.resizeResults(results.map((/**
* @param {?} result
* @return {?}
*/
result => result instanceof faceapi.FaceDetection ? result : result.detection)), { width, height });
this.canvas.width = width;
this.canvas.height = height;
this.renderer.setStyle(this.canvas, 'width', `${width}px`);
this.renderer.setStyle(this.canvas, 'height', `${height}px`);
if (this.task.tokens.length >= 1) {
faceapi.draw.drawDetections(this.canvas, detectionsForSize);
/** @type {?} */
const resizeResults = faceapi.resizeResults(results, { width, height });
if (this.task.tokens.includes('expressions')) {
faceapi.draw.drawFaceExpressions(this.canvas, resizeResults.map((/**
* @param {?} __0
* @return {?}
*/
({ detection, expressions }) => ({
position: detection.box,
expressions,
}))));
}
if (this.task.tokens.includes('landmarks')) {
faceapi.draw.drawFaceLandmarks(this.canvas, resizeResults.map((/**
* @param {?} __0
* @return {?}
*/
({ landmarks }) => landmarks)));
}
if (this.task.tokens.includes('ageAndGender')) {
resizeResults.forEach((/**
* @param {?} result
* @return {?}
*/
result => {
const { age, gender, genderProbability } = result;
/** @type {?} */
const text = new faceapi.draw.DrawTextField([
`${faceapi.round(age, 0)} years`,
`${gender} (${faceapi.round(genderProbability)})`,
], result.detection.box.bottomLeft);
text.draw(this.canvas);
}));
}
}
else {
faceapi.draw.drawDetections(this.canvas, detectionsForSize);
}
});
}
}
DetectionResultComponent.decorators = [
{ type: Component, args: [{
template: "<canvas #canvas></canvas>\n",
styles: ["canvas{width:100%;height:100%}"]
}] }
];
/** @nocollapse */
DetectionResultComponent.ctorParameters = () => [
{ type: DetectTask },
{ type: ElementRef },
{ type: Renderer2 },
{ type: FaceDetectorService }
];
DetectionResultComponent.propDecorators = {
canvasEl: [{ type: ViewChild, args: ['canvas',] }],
onResize: [{ type: HostListener, args: ['window:resize',] }]
};
if (false) {
/** @type {?} */
DetectionResultComponent.prototype.subscription;
/**
* @type {?}
* @private
*/
DetectionResultComponent.prototype.canvasEl;
/**
* @type {?}
* @private
*/
DetectionResultComponent.prototype.resize$;
/**
* @type {?}
* @private
*/
DetectionResultComponent.prototype.task;
/**
* @type {?}
* @private
*/
DetectionResultComponent.prototype.el;
/**
* @type {?}
* @private
*/
DetectionResultComponent.prototype.renderer;
/**
* @type {?}
* @private
*/
DetectionResultComponent.prototype.faceDetector;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"detection-result.component.js","sourceRoot":"ng://ngx-face-api-js/","sources":["lib/components/detection-result/detection-result.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  ViewChild,\n  ElementRef,\n  Renderer2,\n  OnDestroy,\n  HostListener,\n} from '@angular/core';\nimport { Subscription, Subject, combineLatest } from 'rxjs';\n\nimport * as faceapi from 'face-api.js';\nimport { map, startWith } from 'rxjs/operators';\nimport { DetectTask } from '../../classes/DetectTask';\nimport { FaceDetectorService } from '../../services/face-detector.service';\n\n@Component({\n  templateUrl: './detection-result.component.html',\n  styleUrls: ['./detection-result.component.scss'],\n})\nexport class DetectionResultComponent implements OnInit, OnDestroy {\n  subscription = new Subscription();\n\n  @ViewChild('canvas')\n  private canvasEl: ElementRef<HTMLCanvasElement>;\n\n  private get canvas(): HTMLCanvasElement {\n    return this.canvasEl.nativeElement;\n  }\n\n  private resize$ = new Subject();\n\n  @HostListener('window:resize')\n  public onResize() {\n    this.resize$.next('onResize');\n  }\n\n  constructor(\n    private task: DetectTask,\n    private el: ElementRef,\n    private renderer: Renderer2,\n    private faceDetector: FaceDetectorService,\n  ) {}\n\n  private convertResultToArray(result: any): any[] {\n    if (Array.isArray(result)) {\n      return result;\n    }\n    return [result];\n  }\n\n  ngOnInit() {\n    this.subscription.add(\n      combineLatest(\n        this.faceDetector.detect(this.task),\n        this.resize$.pipe(startWith('init')),\n      )\n        .pipe(map(([result]) => this.convertResultToArray(result)))\n        .subscribe(result => this.draw(result)),\n    );\n  }\n\n  ngOnDestroy() {\n    this.subscription.unsubscribe();\n  }\n  private async draw(results: any[]) {\n    const target = await this.task.target;\n    let { width, height } = target;\n    if (target instanceof HTMLVideoElement) {\n      height = target.videoHeight;\n      width = target.videoWidth;\n    }\n\n    const detectionsForSize = faceapi.resizeResults(\n      results.map(result =>\n        result instanceof faceapi.FaceDetection ? result : result.detection,\n      ),\n      { width, height },\n    );\n\n    this.canvas.width = width;\n    this.canvas.height = height;\n    this.renderer.setStyle(this.canvas, 'width', `${width}px`);\n    this.renderer.setStyle(this.canvas, 'height', `${height}px`);\n    if (this.task.tokens.length >= 1) {\n      faceapi.draw.drawDetections(this.canvas, detectionsForSize);\n\n      const resizeResults = faceapi.resizeResults(results, { width, height });\n      if (this.task.tokens.includes('expressions')) {\n        faceapi.draw.drawFaceExpressions(\n          this.canvas,\n          resizeResults.map(({ detection, expressions }) => ({\n            position: detection.box,\n            expressions,\n          })),\n        );\n      }\n\n      if (this.task.tokens.includes('landmarks')) {\n        faceapi.draw.drawFaceLandmarks(\n          this.canvas,\n          resizeResults.map(({ landmarks }) => landmarks),\n        );\n      }\n\n      if (this.task.tokens.includes('ageAndGender')) {\n        resizeResults.forEach(result => {\n          const { age, gender, genderProbability } = result;\n          const text = new faceapi.draw.DrawTextField(\n            [\n              `${faceapi.round(age, 0)} years`,\n              `${gender} (${faceapi.round(genderProbability)})`,\n            ],\n            result.detection.box.bottomLeft,\n          );\n          text.draw(this.canvas);\n        });\n      }\n    } else {\n      faceapi.draw.drawDetections(this.canvas, detectionsForSize);\n    }\n  }\n}\n"]}