UNPKG

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
/** * @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"]}