UNPKG

biometry-angular-components

Version:

Angular UI component library for capturing biometric data

275 lines 47 kB
import { Component, ChangeDetectionStrategy, EventEmitter, Input, Output, ViewChild, } from '@angular/core'; import { CommonModule } from '@angular/common'; import * as i0 from "@angular/core"; import * as i1 from "../../services/permissions.service"; import * as i2 from "../../services/recorder.service"; import * as i3 from "@angular/common"; function numbersToPhrase(digits) { const words = ['zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine']; return digits.map(d => words[d]).join(' '); } function pickSupportedMimeType() { const candidates = [ 'video/webm;codecs=vp9,opus', 'video/webm;codecs=vp8,opus', 'video/webm', 'video/mp4;codecs=h264,aac', 'video/mp4', ]; for (const type of candidates) { // @ts-ignore - isTypeSupported exists in browsers that implement MediaRecorder if (typeof MediaRecorder !== 'undefined' && MediaRecorder.isTypeSupported?.(type)) { return type; } } return 'video/webm'; } export class FaceRecorderComponent { permissions; recorder; cdr; videoRef; countdownSeconds = 3; recordingSeconds = 10; /** Prefer vertical 9:16; request 720x1280 from camera */ videoWidth = 720; videoHeight = 1280; capture = new EventEmitter(); state = 'preparation'; digits = []; maskedDigits = []; countdownLeft = 0; recordLeft = 0; countdownTimer; recordTimer; stream; videoBlob = null; videoUrl = null; isConfirming = false; permissionsGranted = null; mimeType = pickSupportedMimeType(); constructor(permissions, recorder, cdr) { this.permissions = permissions; this.recorder = recorder; this.cdr = cdr; } ngOnInit() { this.generateDigits(); this.requestPermissions(); } ngAfterViewInit() { if (this.stream) { this.attachStream(); } } ngOnDestroy() { this.clearAllTimers(); this.cleanupPlayback(); this.stopTracks(); this.recorder.cancel(); } /** === Flow control === */ async requestPermissions() { this.permissionsGranted = null; this.cdr.markForCheck(); const { stream, granted } = await this.permissions.requestCamera({ width: this.videoWidth, height: this.videoHeight, audio: true, }); this.permissionsGranted = granted; if (granted && stream) { this.stream = stream; this.attachStream(); } else { this.stream = undefined; } this.cdr.markForCheck(); } attachStream() { const video = this.videoRef.nativeElement; video.srcObject = this.stream ?? null; video.src = ''; video.controls = false; video.muted = true; video.play().catch(() => { }); } startFlow() { if (!this.stream || this.state !== 'preparation') return; this.startCountdown(); } startCountdown() { this.state = 'countdown'; this.countdownLeft = this.countdownSeconds; this.cdr.markForCheck(); this.clearCountdown(); this.countdownTimer = setInterval(() => { this.countdownLeft -= 1; this.cdr.markForCheck(); if (this.countdownLeft <= 0) { this.clearCountdown(); this.startRecording(); } }, 1000); } startRecording() { if (!this.stream) return; // Keep preview running; MediaRecorder records from the same stream this.state = 'recording'; this.recordLeft = this.recordingSeconds; try { this.recorder.start(this.stream, this.mimeType); } catch (e) { console.error('Failed to start recorder', e); this.cancelRecording(); return; } this.clearRecordTimer(); this.recordTimer = setInterval(async () => { this.recordLeft -= 1; this.cdr.markForCheck(); if (this.recordLeft <= 0) { await this.finishRecording(); } }, 1000); this.cdr.markForCheck(); } async finishRecording() { if (this.state !== 'recording') return; this.state = 'processing'; this.clearRecordTimer(); this.cdr.markForCheck(); try { this.videoBlob = await this.recorder.stop(); } catch { this.videoBlob = null; } if (this.videoBlob && this.videoBlob.size) { if (this.videoUrl) URL.revokeObjectURL(this.videoUrl); this.videoUrl = URL.createObjectURL(this.videoBlob); // Switch the element from live preview to the recorded video const video = this.videoRef.nativeElement; video.srcObject = null; video.src = this.videoUrl; video.controls = true; video.muted = false; await video.play().catch(() => { }); this.state = 'result'; } else { // No data recorded: return to preparation with live preview this.state = 'preparation'; this.attachStream(); } this.cdr.markForCheck(); } cancelRecording() { if (this.state !== 'countdown' && this.state !== 'recording') return; this.clearCountdown(); this.clearRecordTimer(); this.recorder.cancel(); if (this.videoUrl) URL.revokeObjectURL(this.videoUrl); this.videoUrl = null; this.videoBlob = null; // Return to preparation and keep preview alive this.state = 'preparation'; this.attachStream(); this.cdr.markForCheck(); } handleDecline() { if (this.videoUrl) URL.revokeObjectURL(this.videoUrl); this.videoUrl = null; this.videoBlob = null; this.state = 'preparation'; this.generateDigits(); this.requestPermissions(); this.cdr.markForCheck(); } handleConfirm() { if (!this.videoBlob) return; const file = new File([this.videoBlob], 'face-video.webm', { type: this.videoBlob.type, }); this.capture.emit({ file, phrase: numbersToPhrase(this.digits) }); if (this.videoUrl) URL.revokeObjectURL(this.videoUrl); this.videoUrl = null; this.videoBlob = null; this.isConfirming = false; this.state = 'preparation'; this.requestPermissions(); this.cdr.markForCheck(); } /** === Helpers === */ generateDigits() { this.digits = Array.from({ length: 10 }, () => Math.floor(Math.random() * 10)); this.maskedDigits = Array(this.digits.length).fill('*'); } clearCountdown() { if (this.countdownTimer) { clearInterval(this.countdownTimer); this.countdownTimer = null; } } clearRecordTimer() { if (this.recordTimer) { clearInterval(this.recordTimer); this.recordTimer = null; } } clearAllTimers() { this.clearCountdown(); this.clearRecordTimer(); } cleanupPlayback() { const v = this.videoRef?.nativeElement; if (v) { v.pause(); v.srcObject = null; v.src = ''; v.removeAttribute('src'); } if (this.videoUrl) URL.revokeObjectURL(this.videoUrl); this.videoUrl = null; this.videoBlob = null; } stopTracks() { this.stream?.getTracks().forEach(t => t.stop()); this.stream = undefined; } get displayDigits() { return (this.state === 'recording' ? this.digits : this.maskedDigits).map(d => d.toString()); } static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FaceRecorderComponent, deps: [{ token: i1.PermissionsService }, { token: i2.RecorderService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: FaceRecorderComponent, isStandalone: true, selector: "bio-face-recorder", inputs: { countdownSeconds: "countdownSeconds", recordingSeconds: "recordingSeconds", videoWidth: "videoWidth", videoHeight: "videoHeight" }, outputs: { capture: "capture" }, viewQueries: [{ propertyName: "videoRef", first: true, predicate: ["videoEl"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"recorder-wrapper\" [class.result]=\"state === 'result'\">\n <!-- Video frame (9:16) -->\n <div class=\"frame\">\n <video #videoEl class=\"preview\" autoplay playsinline muted preload=\"metadata\"></video>\n\n <!-- Oval dashed overlay -->\n <div class=\"oval-area\" *ngIf=\"state !== 'result' && state !== 'processing'\">\n <span class=\"guidance-text\">Place your face in the oval</span>\n <div class=\"oval-dashed\"></div>\n </div>\n\n <!-- HUD: digits to read -->\n <!-- <div class=\"digits\" *ngIf=\"state !== 'processing'\">\n <div class=\"digits-label\">Read these digits aloud</div>\n <div class=\"digits-row\">\n <div class=\"digit-box\" *ngFor=\"let d of displayDigits\">\n {{ d }}\n </div>\n </div>\n </div> -->\n\n <!-- HUD: countdown -->\n <div class=\"overlay center\" *ngIf=\"state === 'countdown'\">\n <div class=\"countdown\">{{ countdownLeft }}</div>\n <button class=\"btn ghost\" type=\"button\" (click)=\"cancelRecording()\">Cancel</button>\n </div>\n\n <!-- HUD: recording timer + cancel -->\n <div class=\"overlay top\" *ngIf=\"state === 'recording'\">\n <div class=\"timer\">\n <span class=\"dot\"></span>\n <span>{{ recordLeft }}s</span>\n </div>\n <button class=\"btn danger\" type=\"button\" (click)=\"cancelRecording()\">Cancel</button>\n </div>\n\n <!-- HUD: processing -->\n <div class=\"overlay center\" *ngIf=\"state === 'processing'\">\n <div class=\"spinner\"></div>\n <div class=\"processing-label\">Processing\u2026</div>\n </div>\n </div>\n\n <div class=\"digits\" *ngIf=\"state !== 'processing'\">\n <div class=\"digits-label\">Read these digits aloud</div>\n <div class=\"digits-row\">\n <div class=\"digit-box\" *ngFor=\"let d of displayDigits\">\n {{ d }}\n </div>\n </div>\n </div>\n\n <!-- Controls / CTA -->\n <div class=\"controls\">\n <!-- Preparation -->\n <ng-container *ngIf=\"state === 'preparation'\">\n <div class=\"hint\" *ngIf=\"permissionsGranted === false\">\n Camera access denied. Please allow camera & audio and try again.\n </div>\n <button class=\"btn primary\" type=\"button\" (click)=\"startFlow()\" [disabled]=\"!permissionsGranted\">\n Start recording\n </button>\n <button class=\"btn\" type=\"button\" (click)=\"requestPermissions()\">Retry camera</button>\n </ng-container>\n\n <!-- Result -->\n <ng-container *ngIf=\"state === 'result'\">\n <div class=\"result-actions\">\n <button class=\"btn success\" type=\"button\" [disabled]=\"isConfirming\" (click)=\"handleConfirm()\">\n {{ isConfirming ? 'Confirming\u2026' : 'Use this video' }}\n </button>\n <button class=\"btn\" type=\"button\" (click)=\"handleDecline()\">Retake</button>\n </div>\n </ng-container>\n </div>\n</div>", styles: [".recorder-wrapper{display:grid;gap:12px;justify-items:center;width:100%}.frame{position:relative;width:100%;max-width:320px;aspect-ratio:9/16;background:#000;border-radius:20px;overflow:hidden;box-shadow:0 6px 20px #00000040}.preview{position:absolute;inset:0;width:100%;height:100%;object-fit:cover}.oval-area{position:absolute;inset:0;display:grid;place-items:center;pointer-events:none}.guidance-text{position:absolute;top:13%;left:50%;transform:translate(-50%);text-wrap:nowrap;color:#fff;background:#00000059;padding:6px 10px;text-align:center;border-radius:30px;font-size:12px;letter-spacing:.2px;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px)}.oval-dashed{width:70%;height:58%;border:3px dashed rgba(255,255,255,.8);border-radius:50%;box-shadow:0 0 0 9999px #00000026 inset}.digits{width:min(100%,420px);display:grid;gap:8px;justify-items:center;margin-top:4px}.digits-label{font-size:13px;opacity:.8;text-align:center}.digits-row{display:flex;justify-content:center;gap:8px;flex-wrap:nowrap;overflow-x:auto}.digit-box{width:28px;height:36px;background:#f5f5f5;border:2px solid #333;border-radius:6px;font-size:18px;font-weight:600;color:#333;display:flex;align-items:center;justify-content:center;box-shadow:0 1px 3px #00000014;transition:background .3s;flex-shrink:0}.overlay{position:absolute;inset-inline:0;display:grid;justify-items:center;gap:10px}.overlay.center{inset-block:0;place-content:center}.overlay.top{top:10px}.countdown{font-size:72px;font-weight:800;color:#fff;text-shadow:0 2px 8px rgba(0,0,0,.35)}.timer{display:inline-flex;align-items:center;gap:8px;padding:6px 10px;border-radius:999px;background:#00000059;color:#fff;font-weight:600}.timer .dot{width:8px;height:8px;border-radius:50%;background:currentColor;animation:pulse 1s infinite}.processing-label{color:#fff;font-weight:600}.spinner{width:28px;height:28px;border-radius:50%;border:3px solid rgba(255,255,255,.4);border-top-color:#fff;animation:spin .9s linear infinite}.controls{width:min(100%,420px);display:grid;gap:10px}.result-actions{display:grid;grid-auto-flow:column;gap:10px}.hint{font-size:13px;color:#a00;text-align:center}.btn{appearance:none;border:none;padding:10px 14px;border-radius:12px;font-weight:600;cursor:pointer;background:#eee}.btn.primary{background:#2b6ef2;color:#fff}.btn.success{background:#16a34a;color:#fff}.btn.danger{background:#ef4444;color:#fff}.btn.ghost{background:#00000059;color:#fff}.btn:disabled{opacity:.6;cursor:not-allowed}@keyframes spin{to{transform:rotate(360deg)}}@keyframes pulse{0%,to{opacity:.5;transform:scale(1)}50%{opacity:1;transform:scale(1.2)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FaceRecorderComponent, decorators: [{ type: Component, args: [{ selector: 'bio-face-recorder', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"recorder-wrapper\" [class.result]=\"state === 'result'\">\n <!-- Video frame (9:16) -->\n <div class=\"frame\">\n <video #videoEl class=\"preview\" autoplay playsinline muted preload=\"metadata\"></video>\n\n <!-- Oval dashed overlay -->\n <div class=\"oval-area\" *ngIf=\"state !== 'result' && state !== 'processing'\">\n <span class=\"guidance-text\">Place your face in the oval</span>\n <div class=\"oval-dashed\"></div>\n </div>\n\n <!-- HUD: digits to read -->\n <!-- <div class=\"digits\" *ngIf=\"state !== 'processing'\">\n <div class=\"digits-label\">Read these digits aloud</div>\n <div class=\"digits-row\">\n <div class=\"digit-box\" *ngFor=\"let d of displayDigits\">\n {{ d }}\n </div>\n </div>\n </div> -->\n\n <!-- HUD: countdown -->\n <div class=\"overlay center\" *ngIf=\"state === 'countdown'\">\n <div class=\"countdown\">{{ countdownLeft }}</div>\n <button class=\"btn ghost\" type=\"button\" (click)=\"cancelRecording()\">Cancel</button>\n </div>\n\n <!-- HUD: recording timer + cancel -->\n <div class=\"overlay top\" *ngIf=\"state === 'recording'\">\n <div class=\"timer\">\n <span class=\"dot\"></span>\n <span>{{ recordLeft }}s</span>\n </div>\n <button class=\"btn danger\" type=\"button\" (click)=\"cancelRecording()\">Cancel</button>\n </div>\n\n <!-- HUD: processing -->\n <div class=\"overlay center\" *ngIf=\"state === 'processing'\">\n <div class=\"spinner\"></div>\n <div class=\"processing-label\">Processing\u2026</div>\n </div>\n </div>\n\n <div class=\"digits\" *ngIf=\"state !== 'processing'\">\n <div class=\"digits-label\">Read these digits aloud</div>\n <div class=\"digits-row\">\n <div class=\"digit-box\" *ngFor=\"let d of displayDigits\">\n {{ d }}\n </div>\n </div>\n </div>\n\n <!-- Controls / CTA -->\n <div class=\"controls\">\n <!-- Preparation -->\n <ng-container *ngIf=\"state === 'preparation'\">\n <div class=\"hint\" *ngIf=\"permissionsGranted === false\">\n Camera access denied. Please allow camera & audio and try again.\n </div>\n <button class=\"btn primary\" type=\"button\" (click)=\"startFlow()\" [disabled]=\"!permissionsGranted\">\n Start recording\n </button>\n <button class=\"btn\" type=\"button\" (click)=\"requestPermissions()\">Retry camera</button>\n </ng-container>\n\n <!-- Result -->\n <ng-container *ngIf=\"state === 'result'\">\n <div class=\"result-actions\">\n <button class=\"btn success\" type=\"button\" [disabled]=\"isConfirming\" (click)=\"handleConfirm()\">\n {{ isConfirming ? 'Confirming\u2026' : 'Use this video' }}\n </button>\n <button class=\"btn\" type=\"button\" (click)=\"handleDecline()\">Retake</button>\n </div>\n </ng-container>\n </div>\n</div>", styles: [".recorder-wrapper{display:grid;gap:12px;justify-items:center;width:100%}.frame{position:relative;width:100%;max-width:320px;aspect-ratio:9/16;background:#000;border-radius:20px;overflow:hidden;box-shadow:0 6px 20px #00000040}.preview{position:absolute;inset:0;width:100%;height:100%;object-fit:cover}.oval-area{position:absolute;inset:0;display:grid;place-items:center;pointer-events:none}.guidance-text{position:absolute;top:13%;left:50%;transform:translate(-50%);text-wrap:nowrap;color:#fff;background:#00000059;padding:6px 10px;text-align:center;border-radius:30px;font-size:12px;letter-spacing:.2px;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px)}.oval-dashed{width:70%;height:58%;border:3px dashed rgba(255,255,255,.8);border-radius:50%;box-shadow:0 0 0 9999px #00000026 inset}.digits{width:min(100%,420px);display:grid;gap:8px;justify-items:center;margin-top:4px}.digits-label{font-size:13px;opacity:.8;text-align:center}.digits-row{display:flex;justify-content:center;gap:8px;flex-wrap:nowrap;overflow-x:auto}.digit-box{width:28px;height:36px;background:#f5f5f5;border:2px solid #333;border-radius:6px;font-size:18px;font-weight:600;color:#333;display:flex;align-items:center;justify-content:center;box-shadow:0 1px 3px #00000014;transition:background .3s;flex-shrink:0}.overlay{position:absolute;inset-inline:0;display:grid;justify-items:center;gap:10px}.overlay.center{inset-block:0;place-content:center}.overlay.top{top:10px}.countdown{font-size:72px;font-weight:800;color:#fff;text-shadow:0 2px 8px rgba(0,0,0,.35)}.timer{display:inline-flex;align-items:center;gap:8px;padding:6px 10px;border-radius:999px;background:#00000059;color:#fff;font-weight:600}.timer .dot{width:8px;height:8px;border-radius:50%;background:currentColor;animation:pulse 1s infinite}.processing-label{color:#fff;font-weight:600}.spinner{width:28px;height:28px;border-radius:50%;border:3px solid rgba(255,255,255,.4);border-top-color:#fff;animation:spin .9s linear infinite}.controls{width:min(100%,420px);display:grid;gap:10px}.result-actions{display:grid;grid-auto-flow:column;gap:10px}.hint{font-size:13px;color:#a00;text-align:center}.btn{appearance:none;border:none;padding:10px 14px;border-radius:12px;font-weight:600;cursor:pointer;background:#eee}.btn.primary{background:#2b6ef2;color:#fff}.btn.success{background:#16a34a;color:#fff}.btn.danger{background:#ef4444;color:#fff}.btn.ghost{background:#00000059;color:#fff}.btn:disabled{opacity:.6;cursor:not-allowed}@keyframes spin{to{transform:rotate(360deg)}}@keyframes pulse{0%,to{opacity:.5;transform:scale(1)}50%{opacity:1;transform:scale(1.2)}}\n"] }] }], ctorParameters: () => [{ type: i1.PermissionsService }, { type: i2.RecorderService }, { type: i0.ChangeDetectorRef }], propDecorators: { videoRef: [{ type: ViewChild, args: ['videoEl', { static: true }] }], countdownSeconds: [{ type: Input }], recordingSeconds: [{ type: Input }], videoWidth: [{ type: Input }], videoHeight: [{ type: Input }], capture: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"face-recorder.component.js","sourceRoot":"","sources":["../../../../../src/lib/components/face-recorder/face-recorder.component.ts","../../../../../src/lib/components/face-recorder/face-recorder.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild,\n  AfterViewInit,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { PermissionsService } from '../../services/permissions.service';\nimport { RecorderService } from '../../services/recorder.service';\n\ntype FaceState = 'preparation' | 'countdown' | 'recording' | 'processing' | 'result';\n\nfunction numbersToPhrase(digits: number[]): string {\n  const words = ['zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight', 'nine'];\n  return digits.map(d => words[d]).join(' ');\n}\n\nfunction pickSupportedMimeType(): string {\n  const candidates = [\n    'video/webm;codecs=vp9,opus',\n    'video/webm;codecs=vp8,opus',\n    'video/webm',\n    'video/mp4;codecs=h264,aac',\n    'video/mp4',\n  ];\n  for (const type of candidates) {\n    // @ts-ignore - isTypeSupported exists in browsers that implement MediaRecorder\n    if (typeof MediaRecorder !== 'undefined' && MediaRecorder.isTypeSupported?.(type)) {\n      return type;\n    }\n  }\n\n  return 'video/webm';\n}\n\n\n@Component({\n  selector: 'bio-face-recorder',\n  standalone: true,\n  imports: [CommonModule],\n  templateUrl: './face-recorder.component.html',\n  styleUrls: ['./face-recorder.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FaceRecorderComponent implements OnInit, AfterViewInit, OnDestroy {\n  @ViewChild('videoEl', { static: true }) videoRef!: ElementRef<HTMLVideoElement>;\n\n  @Input() countdownSeconds = 3;\n  @Input() recordingSeconds = 10;\n\n  /** Prefer vertical 9:16; request 720x1280 from camera */\n  @Input() videoWidth = 720;\n  @Input() videoHeight = 1280;\n\n  @Output() capture = new EventEmitter<{ file: File; phrase: string }>();\n\n  state: FaceState = 'preparation';\n  digits: number[] = [];\n  maskedDigits: string[] = [];\n\n  countdownLeft = 0;\n  recordLeft = 0;\n\n  private countdownTimer: any;\n  private recordTimer: any;\n\n  private stream?: MediaStream;\n  videoBlob: Blob | null = null;\n  videoUrl: string | null = null;\n\n  isConfirming = false;\n  permissionsGranted: boolean | null = null;\n\n  private mimeType = pickSupportedMimeType();\n\n  constructor(\n    private permissions: PermissionsService,\n    private recorder: RecorderService,\n    private cdr: ChangeDetectorRef\n  ) { }\n\n  ngOnInit(): void {\n    this.generateDigits();\n    this.requestPermissions();\n  }\n\n  ngAfterViewInit(): void {\n    if (this.stream) {\n      this.attachStream();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.clearAllTimers();\n    this.cleanupPlayback();\n    this.stopTracks();\n    this.recorder.cancel();\n  }\n\n  /** === Flow control === */\n\n  async requestPermissions() {\n    this.permissionsGranted = null;\n    this.cdr.markForCheck();\n\n    const { stream, granted } = await this.permissions.requestCamera({\n      width: this.videoWidth,\n      height: this.videoHeight,\n      audio: true,\n    });\n\n    this.permissionsGranted = granted;\n\n    if (granted && stream) {\n      this.stream = stream;\n      this.attachStream();\n    } else {\n      this.stream = undefined;\n    }\n\n    this.cdr.markForCheck();\n  }\n\n  private attachStream() {\n    const video = this.videoRef.nativeElement;\n    video.srcObject = this.stream ?? null;\n    video.src = '';\n    video.controls = false;\n    video.muted = true;\n    video.play().catch(() => {/* ignore autoplay race */ });\n  }\n\n  startFlow() {\n    if (!this.stream || this.state !== 'preparation') return;\n    this.startCountdown();\n  }\n\n  private startCountdown() {\n    this.state = 'countdown';\n    this.countdownLeft = this.countdownSeconds;\n    this.cdr.markForCheck();\n\n    this.clearCountdown();\n    this.countdownTimer = setInterval(() => {\n      this.countdownLeft -= 1;\n      this.cdr.markForCheck();\n      if (this.countdownLeft <= 0) {\n        this.clearCountdown();\n        this.startRecording();\n      }\n    }, 1000);\n  }\n\n  private startRecording() {\n    if (!this.stream) return;\n\n    // Keep preview running; MediaRecorder records from the same stream\n    this.state = 'recording';\n    this.recordLeft = this.recordingSeconds;\n\n    try {\n      this.recorder.start(this.stream, this.mimeType);\n    } catch (e) {\n      console.error('Failed to start recorder', e);\n      this.cancelRecording();\n      return;\n    }\n\n    this.clearRecordTimer();\n    this.recordTimer = setInterval(async () => {\n      this.recordLeft -= 1;\n      this.cdr.markForCheck();\n      if (this.recordLeft <= 0) {\n        await this.finishRecording();\n      }\n    }, 1000);\n    this.cdr.markForCheck();\n  }\n\n  async finishRecording() {\n    if (this.state !== 'recording') return;\n\n    this.state = 'processing';\n    this.clearRecordTimer();\n    this.cdr.markForCheck();\n\n    try {\n      this.videoBlob = await this.recorder.stop();\n    } catch {\n      this.videoBlob = null;\n    }\n\n    if (this.videoBlob && this.videoBlob.size) {\n      if (this.videoUrl) URL.revokeObjectURL(this.videoUrl);\n      this.videoUrl = URL.createObjectURL(this.videoBlob);\n      // Switch the element from live preview to the recorded video\n      const video = this.videoRef.nativeElement;\n      video.srcObject = null;\n      video.src = this.videoUrl;\n      video.controls = true;\n      video.muted = false;\n      await video.play().catch(() => { });\n      this.state = 'result';\n    } else {\n      // No data recorded: return to preparation with live preview\n      this.state = 'preparation';\n      this.attachStream();\n    }\n\n    this.cdr.markForCheck();\n  }\n\n  cancelRecording() {\n    if (this.state !== 'countdown' && this.state !== 'recording') return;\n\n    this.clearCountdown();\n    this.clearRecordTimer();\n    this.recorder.cancel();\n\n    if (this.videoUrl) URL.revokeObjectURL(this.videoUrl);\n    this.videoUrl = null;\n    this.videoBlob = null;\n\n    // Return to preparation and keep preview alive\n    this.state = 'preparation';\n    this.attachStream();\n    this.cdr.markForCheck();\n  }\n\n  handleDecline() {\n    if (this.videoUrl) URL.revokeObjectURL(this.videoUrl);\n    this.videoUrl = null;\n    this.videoBlob = null;\n\n    this.state = 'preparation';\n    this.generateDigits();\n    this.requestPermissions();\n    this.cdr.markForCheck();\n  }\n\n  handleConfirm() {\n    if (!this.videoBlob) return;\n\n    const file = new File([this.videoBlob], 'face-video.webm', {\n      type: this.videoBlob.type,\n    });\n\n    this.capture.emit({ file, phrase: numbersToPhrase(this.digits) });\n\n    if (this.videoUrl) URL.revokeObjectURL(this.videoUrl);\n    this.videoUrl = null;\n    this.videoBlob = null;\n\n    this.isConfirming = false;\n    this.state = 'preparation';\n    this.requestPermissions();\n    this.cdr.markForCheck();\n  }\n\n  /** === Helpers === */\n\n  private generateDigits() {\n    this.digits = Array.from({ length: 10 }, () => Math.floor(Math.random() * 10));\n    this.maskedDigits = Array(this.digits.length).fill('*');\n  }\n\n  private clearCountdown() {\n    if (this.countdownTimer) {\n      clearInterval(this.countdownTimer);\n      this.countdownTimer = null;\n    }\n  }\n\n  private clearRecordTimer() {\n    if (this.recordTimer) {\n      clearInterval(this.recordTimer);\n      this.recordTimer = null;\n    }\n  }\n\n  private clearAllTimers() {\n    this.clearCountdown();\n    this.clearRecordTimer();\n  }\n\n  private cleanupPlayback() {\n    const v = this.videoRef?.nativeElement;\n    if (v) {\n      v.pause();\n      v.srcObject = null;\n      v.src = '';\n      v.removeAttribute('src');\n    }\n    if (this.videoUrl) URL.revokeObjectURL(this.videoUrl);\n    this.videoUrl = null;\n    this.videoBlob = null;\n  }\n\n  private stopTracks() {\n    this.stream?.getTracks().forEach(t => t.stop());\n    this.stream = undefined;\n  }\n\n  get displayDigits(): string[] {\n    return (this.state === 'recording' ? this.digits : this.maskedDigits).map(d => d.toString());\n  }\n}\n","<div class=\"recorder-wrapper\" [class.result]=\"state === 'result'\">\n  <!-- Video frame (9:16) -->\n  <div class=\"frame\">\n    <video #videoEl class=\"preview\" autoplay playsinline muted preload=\"metadata\"></video>\n\n    <!-- Oval dashed overlay -->\n    <div class=\"oval-area\" *ngIf=\"state !== 'result' && state !== 'processing'\">\n      <span class=\"guidance-text\">Place your face in the oval</span>\n      <div class=\"oval-dashed\"></div>\n    </div>\n\n    <!-- HUD: digits to read -->\n    <!-- <div class=\"digits\" *ngIf=\"state !== 'processing'\">\n      <div class=\"digits-label\">Read these digits aloud</div>\n      <div class=\"digits-row\">\n        <div class=\"digit-box\" *ngFor=\"let d of displayDigits\">\n          {{ d }}\n        </div>\n      </div>\n    </div> -->\n\n    <!-- HUD: countdown -->\n    <div class=\"overlay center\" *ngIf=\"state === 'countdown'\">\n      <div class=\"countdown\">{{ countdownLeft }}</div>\n      <button class=\"btn ghost\" type=\"button\" (click)=\"cancelRecording()\">Cancel</button>\n    </div>\n\n    <!-- HUD: recording timer + cancel -->\n    <div class=\"overlay top\" *ngIf=\"state === 'recording'\">\n      <div class=\"timer\">\n        <span class=\"dot\"></span>\n        <span>{{ recordLeft }}s</span>\n      </div>\n      <button class=\"btn danger\" type=\"button\" (click)=\"cancelRecording()\">Cancel</button>\n    </div>\n\n    <!-- HUD: processing -->\n    <div class=\"overlay center\" *ngIf=\"state === 'processing'\">\n      <div class=\"spinner\"></div>\n      <div class=\"processing-label\">Processing…</div>\n    </div>\n  </div>\n\n  <div class=\"digits\" *ngIf=\"state !== 'processing'\">\n    <div class=\"digits-label\">Read these digits aloud</div>\n    <div class=\"digits-row\">\n      <div class=\"digit-box\" *ngFor=\"let d of displayDigits\">\n        {{ d }}\n      </div>\n    </div>\n  </div>\n\n  <!-- Controls / CTA -->\n  <div class=\"controls\">\n    <!-- Preparation -->\n    <ng-container *ngIf=\"state === 'preparation'\">\n      <div class=\"hint\" *ngIf=\"permissionsGranted === false\">\n        Camera access denied. Please allow camera & audio and try again.\n      </div>\n      <button class=\"btn primary\" type=\"button\" (click)=\"startFlow()\" [disabled]=\"!permissionsGranted\">\n        Start recording\n      </button>\n      <button class=\"btn\" type=\"button\" (click)=\"requestPermissions()\">Retry camera</button>\n    </ng-container>\n\n    <!-- Result -->\n    <ng-container *ngIf=\"state === 'result'\">\n      <div class=\"result-actions\">\n        <button class=\"btn success\" type=\"button\" [disabled]=\"isConfirming\" (click)=\"handleConfirm()\">\n          {{ isConfirming ? 'Confirming…' : 'Use this video' }}\n        </button>\n        <button class=\"btn\" type=\"button\" (click)=\"handleDecline()\">Retake</button>\n      </div>\n    </ng-container>\n  </div>\n</div>"]}