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.

228 lines 18.4 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'; var DetectionResultComponent = /** @class */ (function () { function DetectionResultComponent(task, el, renderer, faceDetector) { this.task = task; this.el = el; this.renderer = renderer; this.faceDetector = faceDetector; this.subscription = new Subscription(); this.resize$ = new Subject(); } Object.defineProperty(DetectionResultComponent.prototype, "canvas", { get: /** * @private * @return {?} */ function () { return this.canvasEl.nativeElement; }, enumerable: true, configurable: true }); /** * @return {?} */ DetectionResultComponent.prototype.onResize = /** * @return {?} */ function () { this.resize$.next('onResize'); }; /** * @private * @param {?} result * @return {?} */ DetectionResultComponent.prototype.convertResultToArray = /** * @private * @param {?} result * @return {?} */ function (result) { if (Array.isArray(result)) { return result; } return [result]; }; /** * @return {?} */ DetectionResultComponent.prototype.ngOnInit = /** * @return {?} */ function () { var _this = this; this.subscription.add(combineLatest(this.faceDetector.detect(this.task), this.resize$.pipe(startWith('init'))) .pipe(map((/** * @param {?} __0 * @return {?} */ function (_a) { var _b = tslib_1.__read(_a, 1), result = _b[0]; return _this.convertResultToArray(result); }))) .subscribe((/** * @param {?} result * @return {?} */ function (result) { return _this.draw(result); }))); }; /** * @return {?} */ DetectionResultComponent.prototype.ngOnDestroy = /** * @return {?} */ function () { this.subscription.unsubscribe(); }; /** * @private * @param {?} results * @return {?} */ DetectionResultComponent.prototype.draw = /** * @private * @param {?} results * @return {?} */ function (results) { return tslib_1.__awaiter(this, void 0, void 0, function () { var target, width, height, detectionsForSize, resizeResults; var _this = this; return tslib_1.__generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, this.task.target]; case 1: target = _a.sent(); width = target.width, height = target.height; if (target instanceof HTMLVideoElement) { height = target.videoHeight; width = target.videoWidth; } detectionsForSize = faceapi.resizeResults(results.map((/** * @param {?} result * @return {?} */ function (result) { return result instanceof faceapi.FaceDetection ? result : result.detection; })), { width: width, height: 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); resizeResults = faceapi.resizeResults(results, { width: width, height: height }); if (this.task.tokens.includes('expressions')) { faceapi.draw.drawFaceExpressions(this.canvas, resizeResults.map((/** * @param {?} __0 * @return {?} */ function (_a) { var detection = _a.detection, expressions = _a.expressions; return ({ position: detection.box, expressions: expressions, }); }))); } if (this.task.tokens.includes('landmarks')) { faceapi.draw.drawFaceLandmarks(this.canvas, resizeResults.map((/** * @param {?} __0 * @return {?} */ function (_a) { var landmarks = _a.landmarks; return landmarks; }))); } if (this.task.tokens.includes('ageAndGender')) { resizeResults.forEach((/** * @param {?} result * @return {?} */ function (result) { var age = result.age, gender = result.gender, genderProbability = result.genderProbability; /** @type {?} */ var 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); } return [2 /*return*/]; } }); }); }; DetectionResultComponent.decorators = [ { type: Component, args: [{ template: "<canvas #canvas></canvas>\n", styles: ["canvas{width:100%;height:100%}"] }] } ]; /** @nocollapse */ DetectionResultComponent.ctorParameters = function () { return [ { type: DetectTask }, { type: ElementRef }, { type: Renderer2 }, { type: FaceDetectorService } ]; }; DetectionResultComponent.propDecorators = { canvasEl: [{ type: ViewChild, args: ['canvas',] }], onResize: [{ type: HostListener, args: ['window:resize',] }] }; return DetectionResultComponent; }()); export { DetectionResultComponent }; 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":";;;;;AAAA,OAAO,EACL,SAAS,EAET,SAAS,EACT,UAAU,EACV,SAAS,EAET,YAAY,GACb,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,aAAa,EAAE,MAAM,MAAM,CAAC;AAE5D,OAAO,KAAK,OAAO,MAAM,aAAa,CAAC;AACvC,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,UAAU,EAAE,MAAM,0BAA0B,CAAC;AACtD,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAE3E;IAqBE,kCACU,IAAgB,EAChB,EAAc,EACd,QAAmB,EACnB,YAAiC;QAHjC,SAAI,GAAJ,IAAI,CAAY;QAChB,OAAE,GAAF,EAAE,CAAY;QACd,aAAQ,GAAR,QAAQ,CAAW;QACnB,iBAAY,GAAZ,YAAY,CAAqB;QApB3C,iBAAY,GAAG,IAAI,YAAY,EAAE,CAAC;QAS1B,YAAO,GAAG,IAAI,OAAO,EAAE,CAAC;IAY7B,CAAC;IAhBJ,sBAAY,4CAAM;;;;;QAAlB;YACE,OAAO,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC;QACrC,CAAC;;;OAAA;;;;IAKM,2CAAQ;;;IADf;QAEE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAChC,CAAC;;;;;;IASO,uDAAoB;;;;;IAA5B,UAA6B,MAAW;QACtC,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACzB,OAAO,MAAM,CAAC;SACf;QACD,OAAO,CAAC,MAAM,CAAC,CAAC;IAClB,CAAC;;;;IAED,2CAAQ;;;IAAR;QAAA,iBASC;QARC,IAAI,CAAC,YAAY,CAAC,GAAG,CACnB,aAAa,CACX,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EACnC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CACrC;aACE,IAAI,CAAC,GAAG;;;;QAAC,UAAC,EAAQ;gBAAR,0BAAQ,EAAP,cAAM;YAAM,OAAA,KAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC;QAAjC,CAAiC,EAAC,CAAC;aAC1D,SAAS;;;;QAAC,UAAA,MAAM,IAAI,OAAA,KAAI,CAAC,IAAI,CAAC,MAAM,CAAC,EAAjB,CAAiB,EAAC,CAC1C,CAAC;IACJ,CAAC;;;;IAED,8CAAW;;;IAAX;QACE,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;IAClC,CAAC;;;;;;IACa,uCAAI;;;;;IAAlB,UAAmB,OAAc;;;;;;4BAChB,qBAAM,IAAI,CAAC,IAAI,CAAC,MAAM,EAAA;;wBAA/B,MAAM,GAAG,SAAsB;wBAC/B,KAAK,GAAa,MAAM,MAAnB,EAAE,MAAM,GAAK,MAAM,OAAX;wBACnB,IAAI,MAAM,YAAY,gBAAgB,EAAE;4BACtC,MAAM,GAAG,MAAM,CAAC,WAAW,CAAC;4BAC5B,KAAK,GAAG,MAAM,CAAC,UAAU,CAAC;yBAC3B;wBAEK,iBAAiB,GAAG,OAAO,CAAC,aAAa,CAC7C,OAAO,CAAC,GAAG;;;;wBAAC,UAAA,MAAM;4BAChB,OAAA,MAAM,YAAY,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS;wBAAnE,CAAmE,EACpE,EACD,EAAE,KAAK,OAAA,EAAE,MAAM,QAAA,EAAE,CAClB;wBAED,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC;wBAC1B,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC;wBAC5B,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,OAAO,EAAK,KAAK,OAAI,CAAC,CAAC;wBAC3D,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,EAAK,MAAM,OAAI,CAAC,CAAC;wBAC7D,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,CAAC,EAAE;4BAChC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,EAAE,iBAAiB,CAAC,CAAC;4BAEtD,aAAa,GAAG,OAAO,CAAC,aAAa,CAAC,OAAO,EAAE,EAAE,KAAK,OAAA,EAAE,MAAM,QAAA,EAAE,CAAC;4BACvE,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAC,EAAE;gCAC5C,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAC9B,IAAI,CAAC,MAAM,EACX,aAAa,CAAC,GAAG;;;;gCAAC,UAAC,EAA0B;wCAAxB,wBAAS,EAAE,4BAAW;oCAAO,OAAA,CAAC;wCACjD,QAAQ,EAAE,SAAS,CAAC,GAAG;wCACvB,WAAW,aAAA;qCACZ,CAAC;gCAHgD,CAGhD,EAAC,CACJ,CAAC;6BACH;4BAED,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE;gCAC1C,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAC5B,IAAI,CAAC,MAAM,EACX,aAAa,CAAC,GAAG;;;;gCAAC,UAAC,EAAa;wCAAX,wBAAS;oCAAO,OAAA,SAAS;gCAAT,CAAS,EAAC,CAChD,CAAC;6BACH;4BAED,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,cAAc,CAAC,EAAE;gCAC7C,aAAa,CAAC,OAAO;;;;gCAAC,UAAA,MAAM;oCAClB,IAAA,gBAAG,EAAE,sBAAM,EAAE,4CAAiB;;wCAChC,IAAI,GAAG,IAAI,OAAO,CAAC,IAAI,CAAC,aAAa,CACzC;wCACK,OAAO,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,WAAQ;wCAC7B,MAAM,UAAK,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,MAAG;qCAClD,EACD,MAAM,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,CAChC;oCACD,IAAI,CAAC,IAAI,CAAC,KAAI,CAAC,MAAM,CAAC,CAAC;gCACzB,CAAC,EAAC,CAAC;6BACJ;yBACF;6BAAM;4BACL,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,EAAE,iBAAiB,CAAC,CAAC;yBAC7D;;;;;KACF;;gBAzGF,SAAS,SAAC;oBACT,uCAAgD;;iBAEjD;;;;gBANQ,UAAU;gBATjB,UAAU;gBACV,SAAS;gBASF,mBAAmB;;;2BASzB,SAAS,SAAC,QAAQ;2BASlB,YAAY,SAAC,eAAe;;IA0F/B,+BAAC;CAAA,AA1GD,IA0GC;SAtGY,wBAAwB;;;IACnC,gDAAkC;;;;;IAElC,4CACgD;;;;;IAMhD,2CAAgC;;;;;IAQ9B,wCAAwB;;;;;IACxB,sCAAsB;;;;;IACtB,4CAA2B;;;;;IAC3B,gDAAyC","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"]}