ets-fe-ng-sdk 17.0.310 → 17.0.312
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/Shared/components/file-upload/file-upload.component.mjs +3 -3
- package/esm2022/lib/Shared/components/webcam-media/webcam-media.component.mjs +40 -39
- package/fesm2022/ets-fe-ng-sdk.mjs +39 -39
- package/fesm2022/ets-fe-ng-sdk.mjs.map +1 -1
- package/lib/Services/utility.service.d.ts +1 -1
- package/lib/Shared/components/btn/btn.component.d.ts +1 -1
- package/lib/Shared/components/info-dialog/info-dialog.component.d.ts +1 -1
- package/lib/Shared/components/table-https/table-https.component.d.ts +1 -1
- package/lib/Shared/components/webcam-media/webcam-media.component.d.ts +8 -4
- package/package.json +1 -1
|
@@ -72,11 +72,11 @@ export class FileUploadComponent {
|
|
|
72
72
|
inp.click();
|
|
73
73
|
}
|
|
74
74
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.1", ngImport: i0, type: FileUploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
75
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.1", type: FileUploadComponent, isStandalone: true, selector: "app-file-upload", inputs: { help: { classPropertyName: "help", publicName: "help", isSignal: false, isRequired: false, transformFunction: null }, label: { classPropertyName: "label", publicName: "label", isSignal: false, isRequired: false, transformFunction: null }, mediaWidth: { classPropertyName: "mediaWidth", publicName: "mediaWidth", isSignal: true, isRequired: false, transformFunction: null }, class: { classPropertyName: "class", publicName: "class", isSignal: true, isRequired: false, transformFunction: null }, useWebcam: { classPropertyName: "useWebcam", publicName: "useWebcam", isSignal: true, isRequired: false, transformFunction: null }, fileName: { classPropertyName: "fileName", publicName: "fileName", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: false, isRequired: false, transformFunction: null }, multiple: { classPropertyName: "multiple", publicName: "multiple", isSignal: false, isRequired: false, transformFunction: null }, mini: { classPropertyName: "mini", publicName: "mini", isSignal: true, isRequired: false, transformFunction: null }, useDocumentModal: { classPropertyName: "useDocumentModal", publicName: "useDocumentModal", isSignal: false, isRequired: false, transformFunction: null }, listFiles: { classPropertyName: "listFiles", publicName: "listFiles", isSignal: true, isRequired: false, transformFunction: null }, _accept: { classPropertyName: "_accept", publicName: "accept", isSignal: false, isRequired: false, transformFunction: null }, file: { classPropertyName: "file", publicName: "file", isSignal: false, isRequired: false, transformFunction: null }, files: { classPropertyName: "files", publicName: "files", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { mediaWidth: "mediaWidthChange", fileChange: "fileChange", filesChange: "filesChange" }, ngImport: i0, template: "@if (listFiles()) {\n <div [ngClass]=\"{ meta: mini() }\" class=\"{{ class() }} form-label mb-2 hide-scroll\">\n @for (item of files; track item) {\n <div class=\"row align-items-start mb-1\">\n <div class=\"col\">\n <div class=\"hide-scroll file-name\">\n {{ item?.name }}\n </div>\n </div>\n <div class=\"col-auto text-end\">\n <span class=\"text-danger pointer p-1 fa fa-close\" (click)=\"removeFile($index)\"></span>\n </div>\n </div>\n }\n </div>\n}\n<!-- <input type=\"file\" style=\"display: none;\" accept=\"{{accept}}\" (change)=\"onUpload($event)\" #uploadInput [multiple]=\"multiple\"> -->\n<app-btn\n icon=\"upload\"\n (mclick)=\"useWebcam() ? null : openDialog()\"\n [type]=\"file ? 'primary' : 'secondary'\"\n [disabled]=\"disabled\"\n [help]=\"help\"\n [text]=\"label\"\n [matMenuTriggerFor]=\"useWebcam() ? menu : null\" />\n\n<mat-menu #menu=\"matMenu\">\n <button mat-menu-item (click)=\"openDialog()\">Upload</button>\n <button mat-menu-item (click)=\"webcamModal.open()\">Use Camera</button>\n</mat-menu>\n\n<modal-comp #webcamModal width=\"1080px\" header=\"Webcam\">\n <ng-template modalBody>\n <lib-webcam-media [fileName]=\"fileName()\" [width]=\"mediaWidth()\" (closed)=\"webcamModal.close(); acceptFiles($event)\" />\n </ng-template>\n</modal-comp>\n", styles: [".meta,.file-name{height:23px}.meta{overflow-y:auto}.file-name{overflow:auto}\n"], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: BtnComponent, selector: "app-btn", inputs: ["formSchema", "debug", "centerBtn", "danger", "warning", "icon", "rightIcon", "leftIcon", "type", "group", "actionType", "animate", "excludeLogging", "loggingValue", "badge", "class", "customIcon", "disabled", "form", "forms", "help", "iconBtn", "loading", "mclass", "showHelpIcon", "rightCustomIcon", "leftCustomIcon", "text", "valid", "mini", "onFormInvalid"], outputs: ["mclick"] }, { kind: "component", type: i1.ModalComponent, selector: "modal-comp", inputs: ["header", "bodyTemplateRef", "footerTemplateRef", "showHeader", "loading", "isFullscreen", "showFooter", "width", "minWidth", "height", "maxHeight", "icon", "data", "disableClose", "hasBackdrop"], outputs: ["modalOpen", "modalClose"] }, { kind: "directive", type: i2.ModalBodyDirective, selector: "[modalBody]" }, { kind: "ngmodule", type: MatMenuModule }, { kind: "component", type: i3.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i3.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i3.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "component", type: WebcamMediaComponent, selector: "lib-webcam-media", inputs: ["isVideo", "useAudio", "fileName", "fileNameFactory", "fileType", "extraClass", "recordingNotificationInterval", "isBackground", "width"], outputs: ["closed", "recorderStopped", "recorderErrored", "recorderStarted", "recordingState", "recordingChanged", "widthChange"] }] }); }
|
|
75
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.1", type: FileUploadComponent, isStandalone: true, selector: "app-file-upload", inputs: { help: { classPropertyName: "help", publicName: "help", isSignal: false, isRequired: false, transformFunction: null }, label: { classPropertyName: "label", publicName: "label", isSignal: false, isRequired: false, transformFunction: null }, mediaWidth: { classPropertyName: "mediaWidth", publicName: "mediaWidth", isSignal: true, isRequired: false, transformFunction: null }, class: { classPropertyName: "class", publicName: "class", isSignal: true, isRequired: false, transformFunction: null }, useWebcam: { classPropertyName: "useWebcam", publicName: "useWebcam", isSignal: true, isRequired: false, transformFunction: null }, fileName: { classPropertyName: "fileName", publicName: "fileName", isSignal: true, isRequired: false, transformFunction: null }, disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: false, isRequired: false, transformFunction: null }, multiple: { classPropertyName: "multiple", publicName: "multiple", isSignal: false, isRequired: false, transformFunction: null }, mini: { classPropertyName: "mini", publicName: "mini", isSignal: true, isRequired: false, transformFunction: null }, useDocumentModal: { classPropertyName: "useDocumentModal", publicName: "useDocumentModal", isSignal: false, isRequired: false, transformFunction: null }, listFiles: { classPropertyName: "listFiles", publicName: "listFiles", isSignal: true, isRequired: false, transformFunction: null }, _accept: { classPropertyName: "_accept", publicName: "accept", isSignal: false, isRequired: false, transformFunction: null }, file: { classPropertyName: "file", publicName: "file", isSignal: false, isRequired: false, transformFunction: null }, files: { classPropertyName: "files", publicName: "files", isSignal: false, isRequired: false, transformFunction: null } }, outputs: { mediaWidth: "mediaWidthChange", fileChange: "fileChange", filesChange: "filesChange" }, ngImport: i0, template: "@if (listFiles()) {\n <div [ngClass]=\"{ meta: mini() }\" class=\"{{ class() }} form-label mb-2 hide-scroll\">\n @for (item of files; track item) {\n <div class=\"row align-items-start mb-1\">\n <div class=\"col\">\n <div class=\"hide-scroll file-name\">\n {{ item?.name }}\n </div>\n </div>\n <div class=\"col-auto text-end\">\n <span class=\"text-danger pointer p-1 fa fa-close\" (click)=\"removeFile($index)\"></span>\n </div>\n </div>\n }\n </div>\n}\n<!-- <input type=\"file\" style=\"display: none;\" accept=\"{{accept}}\" (change)=\"onUpload($event)\" #uploadInput [multiple]=\"multiple\"> -->\n<app-btn\n icon=\"upload\"\n (mclick)=\"useWebcam() ? null : openDialog()\"\n [type]=\"file ? 'primary' : 'secondary'\"\n [disabled]=\"disabled\"\n [help]=\"help\"\n [text]=\"label\"\n [matMenuTriggerFor]=\"useWebcam() ? menu : null\" />\n\n<mat-menu #menu=\"matMenu\">\n <button mat-menu-item (click)=\"openDialog()\">Upload</button>\n <button mat-menu-item (click)=\"webcamModal.open()\">Use Camera</button>\n</mat-menu>\n\n<modal-comp #webcamModal width=\"1080px\" [showFooter]=\"false\" header=\"Webcam\">\n <ng-template modalBody>\n <lib-webcam-media [fileName]=\"fileName()\" [width]=\"mediaWidth()\" (closed)=\"webcamModal.close(); $event ? acceptFiles($event) : null\" />\n </ng-template>\n</modal-comp>\n", styles: [".meta,.file-name{height:23px}.meta{overflow-y:auto}.file-name{overflow:auto}\n"], dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: BtnComponent, selector: "app-btn", inputs: ["formSchema", "debug", "centerBtn", "danger", "warning", "icon", "rightIcon", "leftIcon", "type", "group", "actionType", "animate", "excludeLogging", "loggingValue", "badge", "class", "customIcon", "disabled", "form", "forms", "help", "iconBtn", "loading", "mclass", "showHelpIcon", "rightCustomIcon", "leftCustomIcon", "text", "valid", "mini", "onFormInvalid"], outputs: ["mclick"] }, { kind: "component", type: i1.ModalComponent, selector: "modal-comp", inputs: ["header", "bodyTemplateRef", "footerTemplateRef", "showHeader", "loading", "isFullscreen", "showFooter", "width", "minWidth", "height", "maxHeight", "icon", "data", "disableClose", "hasBackdrop"], outputs: ["modalOpen", "modalClose"] }, { kind: "directive", type: i2.ModalBodyDirective, selector: "[modalBody]" }, { kind: "ngmodule", type: MatMenuModule }, { kind: "component", type: i3.MatMenu, selector: "mat-menu", inputs: ["backdropClass", "aria-label", "aria-labelledby", "aria-describedby", "xPosition", "yPosition", "overlapTrigger", "hasBackdrop", "class", "classList"], outputs: ["closed", "close"], exportAs: ["matMenu"] }, { kind: "component", type: i3.MatMenuItem, selector: "[mat-menu-item]", inputs: ["role", "disabled", "disableRipple"], exportAs: ["matMenuItem"] }, { kind: "directive", type: i3.MatMenuTrigger, selector: "[mat-menu-trigger-for], [matMenuTriggerFor]", inputs: ["mat-menu-trigger-for", "matMenuTriggerFor", "matMenuTriggerData", "matMenuTriggerRestoreFocus"], outputs: ["menuOpened", "onMenuOpen", "menuClosed", "onMenuClose"], exportAs: ["matMenuTrigger"] }, { kind: "component", type: WebcamMediaComponent, selector: "lib-webcam-media", inputs: ["isVideo", "useAudio", "fileName", "fileNameFactory", "fileType", "extraClass", "recordingNotificationInterval", "isBackground", "width"], outputs: ["closed", "recorderStopped", "recorderErrored", "recorderStarted", "recordingState", "recordingChanged", "widthChange"] }] }); }
|
|
76
76
|
}
|
|
77
77
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.1", ngImport: i0, type: FileUploadComponent, decorators: [{
|
|
78
78
|
type: Component,
|
|
79
|
-
args: [{ selector: 'app-file-upload', standalone: true, imports: [NgIf, NgClass, NgFor, BtnComponent, ModalComponents, MatMenuModule, WebcamMediaComponent], template: "@if (listFiles()) {\n <div [ngClass]=\"{ meta: mini() }\" class=\"{{ class() }} form-label mb-2 hide-scroll\">\n @for (item of files; track item) {\n <div class=\"row align-items-start mb-1\">\n <div class=\"col\">\n <div class=\"hide-scroll file-name\">\n {{ item?.name }}\n </div>\n </div>\n <div class=\"col-auto text-end\">\n <span class=\"text-danger pointer p-1 fa fa-close\" (click)=\"removeFile($index)\"></span>\n </div>\n </div>\n }\n </div>\n}\n<!-- <input type=\"file\" style=\"display: none;\" accept=\"{{accept}}\" (change)=\"onUpload($event)\" #uploadInput [multiple]=\"multiple\"> -->\n<app-btn\n icon=\"upload\"\n (mclick)=\"useWebcam() ? null : openDialog()\"\n [type]=\"file ? 'primary' : 'secondary'\"\n [disabled]=\"disabled\"\n [help]=\"help\"\n [text]=\"label\"\n [matMenuTriggerFor]=\"useWebcam() ? menu : null\" />\n\n<mat-menu #menu=\"matMenu\">\n <button mat-menu-item (click)=\"openDialog()\">Upload</button>\n <button mat-menu-item (click)=\"webcamModal.open()\">Use Camera</button>\n</mat-menu>\n\n<modal-comp #webcamModal width=\"1080px\" header=\"Webcam\">\n <ng-template modalBody>\n <lib-webcam-media [fileName]=\"fileName()\" [width]=\"mediaWidth()\" (closed)=\"webcamModal.close(); acceptFiles($event)\" />\n </ng-template>\n</modal-comp>\n", styles: [".meta,.file-name{height:23px}.meta{overflow-y:auto}.file-name{overflow:auto}\n"] }]
|
|
79
|
+
args: [{ selector: 'app-file-upload', standalone: true, imports: [NgIf, NgClass, NgFor, BtnComponent, ModalComponents, MatMenuModule, WebcamMediaComponent], template: "@if (listFiles()) {\n <div [ngClass]=\"{ meta: mini() }\" class=\"{{ class() }} form-label mb-2 hide-scroll\">\n @for (item of files; track item) {\n <div class=\"row align-items-start mb-1\">\n <div class=\"col\">\n <div class=\"hide-scroll file-name\">\n {{ item?.name }}\n </div>\n </div>\n <div class=\"col-auto text-end\">\n <span class=\"text-danger pointer p-1 fa fa-close\" (click)=\"removeFile($index)\"></span>\n </div>\n </div>\n }\n </div>\n}\n<!-- <input type=\"file\" style=\"display: none;\" accept=\"{{accept}}\" (change)=\"onUpload($event)\" #uploadInput [multiple]=\"multiple\"> -->\n<app-btn\n icon=\"upload\"\n (mclick)=\"useWebcam() ? null : openDialog()\"\n [type]=\"file ? 'primary' : 'secondary'\"\n [disabled]=\"disabled\"\n [help]=\"help\"\n [text]=\"label\"\n [matMenuTriggerFor]=\"useWebcam() ? menu : null\" />\n\n<mat-menu #menu=\"matMenu\">\n <button mat-menu-item (click)=\"openDialog()\">Upload</button>\n <button mat-menu-item (click)=\"webcamModal.open()\">Use Camera</button>\n</mat-menu>\n\n<modal-comp #webcamModal width=\"1080px\" [showFooter]=\"false\" header=\"Webcam\">\n <ng-template modalBody>\n <lib-webcam-media [fileName]=\"fileName()\" [width]=\"mediaWidth()\" (closed)=\"webcamModal.close(); $event ? acceptFiles($event) : null\" />\n </ng-template>\n</modal-comp>\n", styles: [".meta,.file-name{height:23px}.meta{overflow-y:auto}.file-name{overflow:auto}\n"] }]
|
|
80
80
|
}], ctorParameters: () => [], propDecorators: { help: [{
|
|
81
81
|
type: Input
|
|
82
82
|
}], label: [{
|
|
@@ -99,4 +99,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.1", ngImpor
|
|
|
99
99
|
}], filesChange: [{
|
|
100
100
|
type: Output
|
|
101
101
|
}] } });
|
|
102
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-upload.component.js","sourceRoot":"","sources":["../../../../../../../projects/ets-fe-ng-sdk/src/lib/Shared/components/file-upload/file-upload.component.ts","../../../../../../../projects/ets-fe-ng-sdk/src/lib/Shared/components/file-upload/file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output, inject, input, model } from '@angular/core';\nimport { NgIf, NgClass, NgFor } from '@angular/common';\nimport { BtnComponent } from '../btn/btn.component';\nimport { UtilityService } from '../../../Services/utility.service';\nimport ModalComponents from '../modal-components/modal.components';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { WebcamMediaComponent } from '../webcam-media/webcam-media.component';\n\n@Component({\n  selector: 'app-file-upload',\n  templateUrl: './file-upload.component.html',\n  styleUrls: ['./file-upload.component.scss'],\n  standalone: true,\n  imports: [NgIf, NgClass, NgFor, BtnComponent,ModalComponents,MatMenuModule,WebcamMediaComponent],\n})\nexport class FileUploadComponent implements OnInit {\n  @Input() help: string;\n  @Input() label: string;\n  mediaWidth = model<number>(1080); \n  class = input<string>();\n  useWebcam = input<boolean>(false);\n  /**\n   * Only useful if using the webcam option\n   */\n  fileName = input<string>();\n  @Input() disabled: boolean;\n  @Input() multiple: boolean;\n  mini = input(true);\n  @Input() useDocumentModal: boolean;\n  listFiles = input(true);\n  accept: string;\n  @Input('accept') set _accept(v: string) {\n    switch (v) {\n      case 'image':\n        this.accept = '.png,.jpg,.docx,.pdf';\n        break;\n\n      default:\n        this.accept = v;\n        break;\n    }\n  }\n  // @ViewChild('uploadInput') uploadInputRef: ElementRef<HTMLInputElement>;\n  @Input() file: File;\n  @Output() fileChange = new EventEmitter<File>();\n  @Input() files: File[];\n  @Output() filesChange = new EventEmitter<File[]>();\n\n  public uS = inject(UtilityService);\n\n  constructor() {}\n\n  ngOnInit(): void {}\n\n  get isSingle() {\n    return !this.multiple;\n  }\n  protected acceptFiles(...files: File[]) {\n    this.files = files;\n    this.file = files ? files[0] : null;\n    this.emitFiles();\n  }\n  protected emitFiles() {\n    if (this.multiple) this.filesChange.emit(this.files);\n    else this.fileChange.emit(this.file);\n  }\n\n  onUpload(event: any) {\n    const files = Array.from<File>(event.target.files);\n    this.acceptFiles(...files);\n  }\n\n  removeFile(index: number) {\n    this.files.splice(index, 1);\n    this.file = this.files ? this.files[0] : null;\n\n    this.fileChange.emit(this.file);\n    this.filesChange.emit(this.files);\n  }\n  openDialog() {\n    const inp = document.createElement('input');\n    inp.type = 'file';\n    inp.accept = this.accept;\n    inp.multiple = this.multiple;\n    inp.onchange = (e) => {\n      this.onUpload(e);\n      inp.remove();\n    };\n    // document.body.appendChild(inp);\n    inp.click();\n  }\n}\n","@if (listFiles()) {\n  <div [ngClass]=\"{ meta: mini() }\" class=\"{{ class() }} form-label mb-2 hide-scroll\">\n    @for (item of files; track item) {\n      <div class=\"row align-items-start mb-1\">\n        <div class=\"col\">\n          <div class=\"hide-scroll file-name\">\n            {{ item?.name }}\n          </div>\n        </div>\n        <div class=\"col-auto text-end\">\n          <span class=\"text-danger pointer p-1 fa fa-close\" (click)=\"removeFile($index)\"></span>\n        </div>\n      </div>\n    }\n  </div>\n}\n<!-- <input type=\"file\"  style=\"display: none;\" accept=\"{{accept}}\" (change)=\"onUpload($event)\" #uploadInput [multiple]=\"multiple\"> -->\n<app-btn\n  icon=\"upload\"\n  (mclick)=\"useWebcam() ? null : openDialog()\"\n  [type]=\"file ? 'primary' : 'secondary'\"\n  [disabled]=\"disabled\"\n  [help]=\"help\"\n  [text]=\"label\"\n  [matMenuTriggerFor]=\"useWebcam() ? menu : null\" />\n\n<mat-menu #menu=\"matMenu\">\n  <button mat-menu-item (click)=\"openDialog()\">Upload</button>\n  <button mat-menu-item (click)=\"webcamModal.open()\">Use Camera</button>\n</mat-menu>\n\n<modal-comp #webcamModal width=\"1080px\" [showFooter]=\"false\" header=\"Webcam\">\n  <ng-template modalBody>\n    <lib-webcam-media [fileName]=\"fileName()\" [width]=\"mediaWidth()\" (closed)=\"webcamModal.close(); $event ? acceptFiles($event) : null\" />\n  </ng-template>\n</modal-comp>\n"]}
|
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
2
|
import { Component, EventEmitter, Output, computed, input, model, signal, viewChild, } from '@angular/core';
|
|
3
3
|
import { BtnComponent } from '../btn/btn.component';
|
|
4
|
-
import {
|
|
4
|
+
import { map, Observable } from 'rxjs';
|
|
5
5
|
import { toSignal } from '@angular/core/rxjs-interop';
|
|
6
|
+
import { LoaderComponent } from '../loader/loader.component';
|
|
6
7
|
import * as i0 from "@angular/core";
|
|
7
8
|
import * as i1 from "@angular/common";
|
|
8
9
|
export class WebcamMediaComponent {
|
|
@@ -55,81 +56,81 @@ export class WebcamMediaComponent {
|
|
|
55
56
|
// will be set by the startup() function.
|
|
56
57
|
this.canvas = signal(undefined);
|
|
57
58
|
this.mediaFile = signal(undefined);
|
|
58
|
-
this.
|
|
59
|
-
this.videoSrc = signal(undefined);
|
|
59
|
+
this.mediaFileLink = computed(() => (this.mediaFile() ? URL.createObjectURL(this.mediaFile()) : undefined));
|
|
60
60
|
this.stream = signal(undefined);
|
|
61
61
|
this.isRecording = toSignal(this.recordingState.pipe(map((r) => r == 'recording' || r == 'paused')));
|
|
62
62
|
this.recordedBlobs = [];
|
|
63
|
-
|
|
63
|
+
/**In Milliseconds */
|
|
64
|
+
this.recordedTimeMS = signal(0);
|
|
65
|
+
this.recordedTime = computed(() => Math.round(this.recordedTimeMS() / 1000));
|
|
66
|
+
this.loading = signal(false);
|
|
64
67
|
this.startRecording = (stream = this.stream()) => {
|
|
65
68
|
if (!this.isVideo())
|
|
66
69
|
return;
|
|
67
70
|
this.recorder = new MediaRecorder(stream);
|
|
68
71
|
this.recordedBlobs = [];
|
|
69
|
-
this.
|
|
70
|
-
const
|
|
72
|
+
this.recordedTimeMS.set(0);
|
|
73
|
+
const startTime = Date.now();
|
|
71
74
|
this.recorder.ondataavailable = (event) => {
|
|
72
75
|
this.recordedBlobs.push(event.data);
|
|
76
|
+
this.recordedTimeMS.set(Date.now() - startTime);
|
|
73
77
|
this.mediaFile.set(new File(this.recordedBlobs, this.computedFileName(), { type: this.computedFileType() }));
|
|
74
|
-
|
|
78
|
+
this.recordingChanged.emit(this.mediaFile());
|
|
75
79
|
};
|
|
76
80
|
// this.recorder.ondataavailable = (event) => {
|
|
77
81
|
// this.recordedBlobs = [event.data];
|
|
78
82
|
// lengths.push(event.data.size);
|
|
79
83
|
// };
|
|
80
84
|
// this.recorder.ondataavailable = (event) => this.recordedBlobs.push(event.data);
|
|
81
|
-
this.recorder.start();
|
|
82
|
-
let notifSub;
|
|
83
|
-
const startTimer = () => {
|
|
84
|
-
let lastDateTime = Date.now();
|
|
85
|
-
notifSub = timer(1000, 1000)
|
|
86
|
-
.pipe(tap(() => this.recordedTime.update((t) => (t += 1))), tap(() => console.log(Date.now(), this.computedRecordingNotificationInterval(), this.recordingNotificationInterval())), filter(() => Date.now() - lastDateTime > this.computedRecordingNotificationInterval()), tap(() => this.recorder.requestData()))
|
|
87
|
-
.subscribe((r) => {
|
|
88
|
-
lastDateTime = Date.now();
|
|
89
|
-
this.recordingChanged.emit(this.mediaFile());
|
|
90
|
-
});
|
|
91
|
-
};
|
|
85
|
+
this.recorder.start(this.computedRecordingNotificationInterval());
|
|
92
86
|
this.recorder.onstart = (e) => {
|
|
93
87
|
this.recordingState.emit(this.recorder.state);
|
|
94
88
|
this.recorderStarted.emit(e);
|
|
95
|
-
startTimer();
|
|
96
89
|
};
|
|
97
90
|
this.recorder.onstop = (e) => {
|
|
98
91
|
this.recordingState.emit(this.recorder.state);
|
|
99
92
|
this.recorderStopped.emit(e);
|
|
100
|
-
console.log('new length', lengths);
|
|
101
|
-
notifSub?.unsubscribe();
|
|
102
93
|
};
|
|
103
94
|
this.recorder.onpause = (e) => {
|
|
104
95
|
this.recordingState.emit(this.recorder.state);
|
|
105
|
-
notifSub?.unsubscribe();
|
|
106
96
|
};
|
|
107
97
|
this.recorder.onresume = (e) => {
|
|
108
98
|
this.recordingState.emit(this.recorder.state);
|
|
109
|
-
startTimer();
|
|
110
99
|
};
|
|
111
100
|
this.recorder.onerror = (e) => {
|
|
112
101
|
this.recordingState.emit(this.recorder.state);
|
|
113
102
|
this.recorderErrored.emit(e);
|
|
114
|
-
notifSub?.unsubscribe();
|
|
115
103
|
};
|
|
116
104
|
};
|
|
105
|
+
this.extractImg = new Observable((sub) => this.canvas().toBlob((blob) => {
|
|
106
|
+
if (blob) {
|
|
107
|
+
this.mediaFile.set(new File([blob], this.computedFileName()));
|
|
108
|
+
sub.next(this.mediaFile());
|
|
109
|
+
sub.complete();
|
|
110
|
+
}
|
|
111
|
+
else
|
|
112
|
+
sub.error(`No ${this.subject()} file generated`);
|
|
113
|
+
}, this.computedFileType(), 1));
|
|
117
114
|
}
|
|
118
115
|
ngAfterViewInit() {
|
|
119
116
|
this.startUpWebcam();
|
|
120
117
|
}
|
|
121
118
|
ngOnDestroy() {
|
|
122
119
|
this.sub?.unsubscribe();
|
|
120
|
+
this.closeWebcam();
|
|
123
121
|
}
|
|
124
122
|
/**
|
|
125
123
|
* Start the webcam stream
|
|
126
124
|
*/
|
|
127
125
|
startUpWebcam() {
|
|
126
|
+
this.loading.set(true);
|
|
128
127
|
this.clear();
|
|
129
128
|
navigator.mediaDevices
|
|
130
129
|
.getUserMedia({ video: true, audio: this.useAudio() })
|
|
131
130
|
.then((stream) => {
|
|
132
131
|
this.streaming.set(true);
|
|
132
|
+
this.loading.set(false);
|
|
133
|
+
// stream.onaddtrack = () => this.loading.set(false);
|
|
133
134
|
this.canvas.set(document.createElement('canvas'));
|
|
134
135
|
this.canvas()?.setAttribute('width', this.width() + 'px');
|
|
135
136
|
this.canvas()?.setAttribute('height', this.height() + 'px');
|
|
@@ -147,15 +148,7 @@ export class WebcamMediaComponent {
|
|
|
147
148
|
*/
|
|
148
149
|
save() {
|
|
149
150
|
if (this.isPicture())
|
|
150
|
-
|
|
151
|
-
if (blob) {
|
|
152
|
-
this.mediaFile.set(new File([blob], this.computedFileName()));
|
|
153
|
-
sub.next(this.mediaFile());
|
|
154
|
-
sub.complete();
|
|
155
|
-
}
|
|
156
|
-
else
|
|
157
|
-
sub.error(`No ${this.subject()} file generated`);
|
|
158
|
-
}, this.computedFileType(), 1)).subscribe(() => {
|
|
151
|
+
this.extractImg.subscribe(() => {
|
|
159
152
|
this.closed.emit(this.mediaFile());
|
|
160
153
|
this.closeWebcam();
|
|
161
154
|
});
|
|
@@ -182,11 +175,18 @@ export class WebcamMediaComponent {
|
|
|
182
175
|
context.fillStyle = '#AAA';
|
|
183
176
|
context.fillRect(0, 0, this.canvas().width, this.canvas().height);
|
|
184
177
|
const data = this.canvas().toDataURL(this.computedFileType());
|
|
185
|
-
this.
|
|
178
|
+
this.mediaFile.set(undefined);
|
|
179
|
+
// this.photoSrc.set(data);
|
|
186
180
|
this.streaming.set(true);
|
|
187
181
|
this.preview.set(false);
|
|
188
182
|
}
|
|
189
183
|
}
|
|
184
|
+
retake() {
|
|
185
|
+
if (!this.stream())
|
|
186
|
+
this.startUpWebcam();
|
|
187
|
+
else
|
|
188
|
+
this.clear();
|
|
189
|
+
}
|
|
190
190
|
/**
|
|
191
191
|
* Shutdown the webcam stream
|
|
192
192
|
*/
|
|
@@ -223,8 +223,9 @@ export class WebcamMediaComponent {
|
|
|
223
223
|
return canvas;
|
|
224
224
|
});
|
|
225
225
|
context.drawImage(this.recordingVideoRefEl(), 0, 0, width, height);
|
|
226
|
-
|
|
227
|
-
this.
|
|
226
|
+
this.extractImg.subscribe();
|
|
227
|
+
// const data = this.canvas()!.toDataURL('image/png');
|
|
228
|
+
// this.photoSrc.set(data);
|
|
228
229
|
}
|
|
229
230
|
else
|
|
230
231
|
this.clear();
|
|
@@ -245,11 +246,11 @@ export class WebcamMediaComponent {
|
|
|
245
246
|
this.recorder.pause();
|
|
246
247
|
}
|
|
247
248
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.1", ngImport: i0, type: WebcamMediaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
248
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.1", type: WebcamMediaComponent, isStandalone: true, selector: "lib-webcam-media", inputs: { isVideo: { classPropertyName: "isVideo", publicName: "isVideo", isSignal: true, isRequired: false, transformFunction: null }, useAudio: { classPropertyName: "useAudio", publicName: "useAudio", isSignal: true, isRequired: false, transformFunction: null }, fileName: { classPropertyName: "fileName", publicName: "fileName", isSignal: true, isRequired: false, transformFunction: null }, fileNameFactory: { classPropertyName: "fileNameFactory", publicName: "fileNameFactory", isSignal: true, isRequired: false, transformFunction: null }, fileType: { classPropertyName: "fileType", publicName: "fileType", isSignal: true, isRequired: false, transformFunction: null }, extraClass: { classPropertyName: "extraClass", publicName: "extraClass", isSignal: true, isRequired: false, transformFunction: null }, recordingNotificationInterval: { classPropertyName: "recordingNotificationInterval", publicName: "recordingNotificationInterval", isSignal: true, isRequired: false, transformFunction: null }, isBackground: { classPropertyName: "isBackground", publicName: "isBackground", isSignal: true, isRequired: false, transformFunction: null }, width: { classPropertyName: "width", publicName: "width", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { closed: "closed", recorderStopped: "recorderStopped", recorderErrored: "recorderErrored", recorderStarted: "recorderStarted", recordingState: "recordingState", recordingChanged: "recordingChanged", width: "widthChange" }, viewQueries: [{ propertyName: "streamingVideoTemplate", first: true, predicate: ["streamVideoTemp"], descendants: true, isSignal: true }, { propertyName: "recordingVideoRef", first: true, predicate: ["recordingVideoTag"], descendants: true, isSignal: true }], ngImport: i0, template: "<div class=\"\" [hidden]=\"streaming() != true || isBackground()\">\n
|
|
249
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.1", type: WebcamMediaComponent, isStandalone: true, selector: "lib-webcam-media", inputs: { isVideo: { classPropertyName: "isVideo", publicName: "isVideo", isSignal: true, isRequired: false, transformFunction: null }, useAudio: { classPropertyName: "useAudio", publicName: "useAudio", isSignal: true, isRequired: false, transformFunction: null }, fileName: { classPropertyName: "fileName", publicName: "fileName", isSignal: true, isRequired: false, transformFunction: null }, fileNameFactory: { classPropertyName: "fileNameFactory", publicName: "fileNameFactory", isSignal: true, isRequired: false, transformFunction: null }, fileType: { classPropertyName: "fileType", publicName: "fileType", isSignal: true, isRequired: false, transformFunction: null }, extraClass: { classPropertyName: "extraClass", publicName: "extraClass", isSignal: true, isRequired: false, transformFunction: null }, recordingNotificationInterval: { classPropertyName: "recordingNotificationInterval", publicName: "recordingNotificationInterval", isSignal: true, isRequired: false, transformFunction: null }, isBackground: { classPropertyName: "isBackground", publicName: "isBackground", isSignal: true, isRequired: false, transformFunction: null }, width: { classPropertyName: "width", publicName: "width", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { closed: "closed", recorderStopped: "recorderStopped", recorderErrored: "recorderErrored", recorderStarted: "recorderStarted", recordingState: "recordingState", recordingChanged: "recordingChanged", width: "widthChange" }, viewQueries: [{ propertyName: "streamingVideoTemplate", first: true, predicate: ["streamVideoTemp"], descendants: true, isSignal: true }, { propertyName: "recordingVideoRef", first: true, predicate: ["recordingVideoTag"], descendants: true, isSignal: true }], ngImport: i0, template: "<loader [loading]=\"loading()\">\n <div class=\"\" [hidden]=\"streaming() != true || isBackground()\">\n <ng-container *ngTemplateOutlet=\"streamVideoTemp\" />\n\n @if (!isBackground()) {\n <div class=\"wm-buttons\">\n <div class=\"row row-cols-lg-auto g-3 justify-content-center\">\n @if (!isRecording()) {\n <div>\n <app-btn [text]=\"isPicture() ? 'Take Picture' : 'Record Video'\" (mclick)=\"isPicture() ? takePicture() : startRecording()\" />\n </div>\n } @else {\n <div>\n <app-btn\n text=\"{{ isPaused() ? 'Play' : 'Pause' }} Recording\"\n (mclick)=\"pauseRecording()\"\n type=\"outline\"\n [icon]=\"isPaused() ? 'play' : 'pause'\" />\n </div>\n <div>\n <app-btn [text]=\"'Stop Recording'\" (mclick)=\"stopRecording()\" type=\"danger\" icon=\"stop\" />\n </div>\n }\n </div>\n </div>\n }\n </div>\n\n <ng-template #streamVideoTemp>\n <div class=\"wm-recording-video-cont\">\n <video #recordingVideoTag [srcObject]=\"stream()\" autoplay class=\"wm-recording-video {{ extraClass() }}\"></video>\n @if (isRecording()) {\n <div class=\"wm-recordingDot\"></div>\n }\n </div>\n </ng-template>\n\n @if (computedPreview()) {\n <div class=\"wm-preview\">\n <div>\n @if (isPicture()) {\n <img class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" [src]=\"mediaFileLink()\" />\n } @else {\n <video class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" controls autoplay [src]=\"mediaFileLink()\"></video>\n }\n </div>\n <div class=\"mt-3 row row-cols-lg-auto justify-content-center g-3\">\n <div>\n <app-btn text=\"Retake {{ subject() }}\" type=\"outline\" (mclick)=\"retake()\" />\n </div>\n <div>\n <app-btn (mclick)=\"save()\" text=\"Done\" type=\"primary\" icon=\"save\" />\n </div>\n <div>\n <app-btn (mclick)=\"cancel()\" text=\"Cancel\" type=\"danger\" icon=\"cancel\" />\n </div>\n </div>\n </div>\n }\n</loader>\n", styles: [":host{display:block}:host .wm-recording-video{width:100%}:host .wm-preview image,:host .wm-preview video{object-fit:contain}:host .wm-recording-video-cont{position:relative}@keyframes wmAnimateDot{0%{background-color:transparent}50%{background-color:#ff000086}}:host .wm-recordingDot{animation-name:wmAnimateDot;animation-duration:1s;animation-iteration-count:infinite;border-radius:5px;--dimen: 20px;width:var(--dimen);height:var(--dimen);position:absolute;top:10px;right:10px}:host .wm-buttons{margin-top:var(--space-16)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: BtnComponent, selector: "app-btn", inputs: ["formSchema", "debug", "centerBtn", "danger", "warning", "icon", "rightIcon", "leftIcon", "type", "group", "actionType", "animate", "excludeLogging", "loggingValue", "badge", "class", "customIcon", "disabled", "form", "forms", "help", "iconBtn", "loading", "mclass", "showHelpIcon", "rightCustomIcon", "leftCustomIcon", "text", "valid", "mini", "onFormInvalid"], outputs: ["mclick"] }, { kind: "component", type: LoaderComponent, selector: "loader", inputs: ["class", "text", "loading", "height"] }] }); }
|
|
249
250
|
}
|
|
250
251
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.1", ngImport: i0, type: WebcamMediaComponent, decorators: [{
|
|
251
252
|
type: Component,
|
|
252
|
-
args: [{ selector: 'lib-webcam-media', standalone: true, imports: [CommonModule, BtnComponent], template: "<div class=\"\" [hidden]=\"streaming() != true || isBackground()\">\n
|
|
253
|
+
args: [{ selector: 'lib-webcam-media', standalone: true, imports: [CommonModule, BtnComponent, LoaderComponent], template: "<loader [loading]=\"loading()\">\n <div class=\"\" [hidden]=\"streaming() != true || isBackground()\">\n <ng-container *ngTemplateOutlet=\"streamVideoTemp\" />\n\n @if (!isBackground()) {\n <div class=\"wm-buttons\">\n <div class=\"row row-cols-lg-auto g-3 justify-content-center\">\n @if (!isRecording()) {\n <div>\n <app-btn [text]=\"isPicture() ? 'Take Picture' : 'Record Video'\" (mclick)=\"isPicture() ? takePicture() : startRecording()\" />\n </div>\n } @else {\n <div>\n <app-btn\n text=\"{{ isPaused() ? 'Play' : 'Pause' }} Recording\"\n (mclick)=\"pauseRecording()\"\n type=\"outline\"\n [icon]=\"isPaused() ? 'play' : 'pause'\" />\n </div>\n <div>\n <app-btn [text]=\"'Stop Recording'\" (mclick)=\"stopRecording()\" type=\"danger\" icon=\"stop\" />\n </div>\n }\n </div>\n </div>\n }\n </div>\n\n <ng-template #streamVideoTemp>\n <div class=\"wm-recording-video-cont\">\n <video #recordingVideoTag [srcObject]=\"stream()\" autoplay class=\"wm-recording-video {{ extraClass() }}\"></video>\n @if (isRecording()) {\n <div class=\"wm-recordingDot\"></div>\n }\n </div>\n </ng-template>\n\n @if (computedPreview()) {\n <div class=\"wm-preview\">\n <div>\n @if (isPicture()) {\n <img class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" [src]=\"mediaFileLink()\" />\n } @else {\n <video class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" controls autoplay [src]=\"mediaFileLink()\"></video>\n }\n </div>\n <div class=\"mt-3 row row-cols-lg-auto justify-content-center g-3\">\n <div>\n <app-btn text=\"Retake {{ subject() }}\" type=\"outline\" (mclick)=\"retake()\" />\n </div>\n <div>\n <app-btn (mclick)=\"save()\" text=\"Done\" type=\"primary\" icon=\"save\" />\n </div>\n <div>\n <app-btn (mclick)=\"cancel()\" text=\"Cancel\" type=\"danger\" icon=\"cancel\" />\n </div>\n </div>\n </div>\n }\n</loader>\n", styles: [":host{display:block}:host .wm-recording-video{width:100%}:host .wm-preview image,:host .wm-preview video{object-fit:contain}:host .wm-recording-video-cont{position:relative}@keyframes wmAnimateDot{0%{background-color:transparent}50%{background-color:#ff000086}}:host .wm-recordingDot{animation-name:wmAnimateDot;animation-duration:1s;animation-iteration-count:infinite;border-radius:5px;--dimen: 20px;width:var(--dimen);height:var(--dimen);position:absolute;top:10px;right:10px}:host .wm-buttons{margin-top:var(--space-16)}\n"] }]
|
|
253
254
|
}], propDecorators: { closed: [{
|
|
254
255
|
type: Output
|
|
255
256
|
}], recorderStopped: [{
|
|
@@ -263,4 +264,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.1", ngImpor
|
|
|
263
264
|
}], recordingChanged: [{
|
|
264
265
|
type: Output
|
|
265
266
|
}] } });
|
|
266
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webcam-media.component.js","sourceRoot":"","sources":["../../../../../../../projects/ets-fe-ng-sdk/src/lib/Shared/components/webcam-media/webcam-media.component.ts","../../../../../../../projects/ets-fe-ng-sdk/src/lib/Shared/components/webcam-media/webcam-media.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAEL,SAAS,EAET,YAAY,EACZ,MAAM,EAEN,QAAQ,EACR,KAAK,EACL,KAAK,EACL,MAAM,EACN,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AACpD,OAAO,EAA0B,MAAM,EAAuB,GAAG,EAAE,UAAU,EAAoB,GAAG,EAAE,KAAK,EAAE,MAAM,MAAM,CAAC;AAC1H,OAAO,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;;;AAStD,MAAM,OAAO,oBAAoB;IAPjC;QAQY,WAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;QAClC,oBAAe,GAAG,IAAI,YAAY,EAAS,CAAC;QAC5C,oBAAe,GAAG,IAAI,YAAY,EAAS,CAAC;QAC5C,oBAAe,GAAG,IAAI,YAAY,EAAS,CAAC;QAC5C,mBAAc,GAAG,IAAI,YAAY,EAAkB,CAAC;QACpD,qBAAgB,GAAG,IAAI,YAAY,EAAoB,CAAC;QAElE,YAAO,GAAG,KAAK,EAAW,CAAC;QAC3B,aAAQ,GAAG,KAAK,EAAW,CAAC;QAC5B,aAAQ,GAAG,KAAK,EAAU,CAAC;QAC3B,oBAAe,GAAG,KAAK,EAA0C,CAAC;QAClE,aAAQ,GAAG,KAAK,EAAU,CAAC;QAC3B,eAAU,GAAG,KAAK,EAAU,CAAC;QAC7B;;WAEG;QACH,kCAA6B,GAAG,KAAK,CAAS,CAAC,CAAC,CAAC;QACjD,iBAAY,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;QAErC,0CAAqC,GAAG,QAAQ,CAAS,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,6BAA6B,EAAE,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;QACnH,cAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAE5C,YAAO,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;QAEjE,aAAQ,GAAG,QAAQ,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,QAAQ,CAAC,CAAC,CAAC,CAAC;QAEzE,qBAAgB,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC/B,MAAM,EAAE,GAAG,IAAI,CAAC,QAAQ,EAAE,EACxB,GAAG,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;YACzB,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;QACH,0BAAqB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,gBAAgB,EAAE,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/E,qBAAgB,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC/B,MAAM,EAAE,GAAG,IAAI,CAAC,eAAe,EAAE,EAC/B,EAAE,GAAG,IAAI,CAAC,QAAQ,EAAE,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,EAClC,EAAE,GAAG,IAAI,CAAC,qBAAqB,EAAE,EACjC,OAAO,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;YAC3B,OAAO,EAAE,EAAE,QAAQ,CAAC,GAAG,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,OAAO,IAAI,EAAE,EAAE,CAAC;QAChE,CAAC,CAAC,CAAC;QAEH,2BAAsB,GAAG,SAAS,CAAmB,iBAAiB,CAAC,CAAC;QACxE,sBAAiB,GAAG,SAAS,CAA+B,mBAAmB,CAAC,CAAC;QACjF,wBAAmB,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,EAAE,EAAE,aAAa,CAAC,CAAC;QAE9E,cAAS,GAAG,MAAM,CAAsB,SAAS,CAAC,CAAC;QACnD,YAAO,GAAG,MAAM,CAAU,KAAK,CAAC,CAAC;QACjC,oBAAe,GAAG,QAAQ,CAAU,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAElF,8DAA8D;QAC9D,0DAA0D;QAC1D,4DAA4D;QAE5D,UAAK,GAAG,KAAK,CAAS,IAAI,CAAC,CAAC,CAAC,wCAAwC;QACrE,WAAM,GAAG,QAAQ,CAAS,GAAG,EAAE;YAC7B,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;YAC3B,OAAO,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QACzB,CAAC,CAAC,CAAC,CAAC,kDAAkD;QAEtD,mEAAmE;QACnE,yCAAyC;QAE/B,WAAM,GAAG,MAAM,CAAgC,SAAS,CAAC,CAAC;QACpE,cAAS,GAAG,MAAM,CAAmB,SAAS,CAAC,CAAC;QAChD,aAAQ,GAAG,MAAM,CAAqB,SAAS,CAAC,CAAC;QACjD,aAAQ,GAAG,MAAM,CAAqB,SAAS,CAAC,CAAC;QAEjD,WAAM,GAAG,MAAM,CAAc,SAAS,CAAC,CAAC;QAIxC,gBAAW,GAAG,QAAQ,CAAU,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,WAAW,IAAI,CAAC,IAAI,QAAQ,CAAC,CAAC,CAAC,CAAC;QACzG,kBAAa,GAAW,EAAE,CAAC;QAE3B,iBAAY,GAAG,MAAM,CAAS,CAAC,CAAC,CAAC;QAkIjC,mBAAc,GAAG,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,EAAE;YAC1C,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;gBAAE,OAAO;YAC5B,IAAI,CAAC,QAAQ,GAAG,IAAI,aAAa,CAAC,MAAM,CAAC,CAAC;YAE1C,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;YACxB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YACzB,MAAM,OAAO,GAAuB,EAAE,CAAC;YACvC,IAAI,CAAC,QAAQ,CAAC,eAAe,GAAG,CAAC,KAAK,EAAE,EAAE;gBACxC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;gBACpC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,gBAAgB,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,gBAAgB,EAAE,EAAE,CAAC,CAAC,CAAC;gBAC7G,OAAO,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;YACzD,CAAC,CAAC;YACF,+CAA+C;YAC/C,uCAAuC;YACvC,mCAAmC;YACnC,KAAK;YACL,kFAAkF;YAElF,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;YACtB,IAAI,QAAsB,CAAC;YAC3B,MAAM,UAAU,GAAG,GAAG,EAAE;gBACtB,IAAI,YAAY,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBAC9B,QAAQ,GAAG,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC;qBACzB,IAAI,CACH,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EACpD,GAAG,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,qCAAqC,EAAE,EAAE,IAAI,CAAC,6BAA6B,EAAE,CAAC,CAAC,EACtH,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,YAAY,GAAG,IAAI,CAAC,qCAAqC,EAAE,CAAC,EACtF,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC,CACvC;qBACA,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;oBACf,YAAY,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;oBAC1B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;gBAC/C,CAAC,CAAC,CAAC;YACP,CAAC,CAAC;YACF,IAAI,CAAC,QAAQ,CAAC,OAAO,GAAG,CAAC,CAAC,EAAE,EAAE;gBAC5B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAC9C,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBAC7B,UAAU,EAAE,CAAC;YACf,CAAC,CAAC;YACF,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,EAAE,EAAE;gBAC3B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAC9C,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBAC7B,OAAO,CAAC,GAAG,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;gBACnC,QAAQ,EAAE,WAAW,EAAE,CAAC;YAC1B,CAAC,CAAC;YACF,IAAI,CAAC,QAAQ,CAAC,OAAO,GAAG,CAAC,CAAC,EAAE,EAAE;gBAC5B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAC9C,QAAQ,EAAE,WAAW,EAAE,CAAC;YAC1B,CAAC,CAAC;YACF,IAAI,CAAC,QAAQ,CAAC,QAAQ,GAAG,CAAC,CAAC,EAAE,EAAE;gBAC7B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAC9C,UAAU,EAAE,CAAC;YACf,CAAC,CAAC;YACF,IAAI,CAAC,QAAQ,CAAC,OAAO,GAAG,CAAC,CAAC,EAAE,EAAE;gBAC5B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;gBAC9C,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;gBAC7B,QAAQ,EAAE,WAAW,EAAE,CAAC;YAC1B,CAAC,CAAC;QACJ,CAAC,CAAC;KAeH;IAzMC,eAAe;QACb,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED,WAAW;QACT,IAAI,CAAC,GAAG,EAAE,WAAW,EAAE,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,aAAa;QACX,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,SAAS,CAAC,YAAY;aACnB,YAAY,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;aACrD,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;YACf,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACzB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC;YAClD,IAAI,CAAC,MAAM,EAAE,EAAE,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC,KAAK,EAAE,GAAG,IAAI,CAAC,CAAC;YAC1D,IAAI,CAAC,MAAM,EAAE,EAAE,YAAY,CAAC,QAAQ,EAAE,IAAI,CAAC,MAAM,EAAE,GAAG,IAAI,CAAC,CAAC;YAC5D,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACxB,YAAY;QACd,CAAC,CAAC;aACD,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,oBAAoB,EAAE,GAAG,CAAC,CAAC;QAC3C,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;OAIG;IACH,IAAI;QACF,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,UAAU,CAAO,CAAC,GAAG,EAAE,EAAE,CAC3B,IAAI,CAAC,MAAM,EAAG,CAAC,MAAM,CACnB,CAAC,IAAI,EAAE,EAAE;gBACP,IAAI,IAAI,EAAE;oBACR,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC,CAAC;oBAC9D,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;oBAC3B,GAAG,CAAC,QAAQ,EAAE,CAAC;iBAChB;;oBAAM,GAAG,CAAC,KAAK,CAAC,MAAM,IAAI,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;YAC1D,CAAC,EACD,IAAI,CAAC,gBAAgB,EAAE,EACvB,CAAC,CACF,CACF,CAAC,SAAS,CAAC,GAAG,EAAE;gBACf,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;gBACnC,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,CAAC,CAAC,CAAC;aACA;YACH,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YACnC,IAAI,CAAC,WAAW,EAAE,CAAC;SACpB;IACH,CAAC;IAED;;;;OAIG;IACH,MAAM;QACJ,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAED;;OAEG;IACH,KAAK;QACH,IAAI,IAAI,CAAC,MAAM,EAAE,EAAE;YACjB,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,UAAU,CAAC,IAAI,CAAC,CAAC;YAChD,OAAQ,CAAC,SAAS,GAAG,MAAM,CAAC;YAC5B,OAAQ,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,MAAM,EAAG,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,EAAG,CAAC,MAAM,CAAC,CAAC;YAErE,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,EAAG,CAAC,SAAS,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC;YAC/D,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACxB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACzB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;SACzB;IACH,CAAC;IAED;;OAEG;IACH,WAAW;QACT,IAAI,CAAC,MAAM,EAAE;YACX,EAAE,SAAS,EAAE;aACZ,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QAC5B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC3B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC9B,uDAAuD;IACzD,CAAC;IAED;;;;;;;;OAQG;IACH,WAAW;QACT,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YAAE,OAAO;QAC9B,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,EAAG,CAAC,UAAU,CAAC,IAAI,CAAC,EAC7C,KAAK,GAAG,IAAI,CAAC,KAAK,EAAE,EACpB,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;QACzB,IAAI,KAAK,IAAI,MAAM,IAAI,OAAO,EAAE;YAC9B,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE;gBAC5B,IAAI,MAAM;oBAAE,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC;gBACjC,OAAO,MAAM,CAAC;YAChB,CAAC,CAAC,CAAC;YACH,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE;gBAC5B,IAAI,MAAM;oBAAE,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC;gBACnC,OAAO,MAAM,CAAC;YAChB,CAAC,CAAC,CAAC;YACH,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,mBAAmB,EAAG,EAAE,CAAC,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;YAEpE,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,EAAG,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;YACnD,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;SACzB;;YAAM,IAAI,CAAC,KAAK,EAAE,CAAC;QAEpB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IA8DD,aAAa;QACX,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,KAAK,WAAW;YAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QAE9D,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAC5B,CAAC;IAED,cAAc;QACZ,YAAY;QACZ,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,KAAK,QAAQ;YAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;;YACxD,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;IAC7B,CAAC;8GAnRU,oBAAoB;kGAApB,oBAAoB,wyDCzBjC,87DAsDA,skBDjCY,YAAY,sMAAE,YAAY;;2FAIzB,oBAAoB;kBAPhC,SAAS;+BACE,kBAAkB,cAChB,IAAI,WACP,CAAC,YAAY,EAAE,YAAY,CAAC;8BAK3B,MAAM;sBAAf,MAAM;gBACG,eAAe;sBAAxB,MAAM;gBACG,eAAe;sBAAxB,MAAM;gBACG,eAAe;sBAAxB,MAAM;gBACG,cAAc;sBAAvB,MAAM;gBACG,gBAAgB;sBAAzB,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Output,\n  TemplateRef,\n  computed,\n  input,\n  model,\n  signal,\n  viewChild,\n} from '@angular/core';\nimport { BtnComponent } from '../btn/btn.component';\nimport { debounce, debounceTime, filter, fromEvent, interval, map, Observable, of, Subscription, tap, timer } from 'rxjs';\nimport { toSignal } from '@angular/core/rxjs-interop';\n\n@Component({\n  selector: 'lib-webcam-media',\n  standalone: true,\n  imports: [CommonModule, BtnComponent],\n  templateUrl: './webcam-media.component.html',\n  styleUrl: './webcam-media.component.scss',\n})\nexport class WebcamMediaComponent {\n  @Output() closed = new EventEmitter<File>();\n  @Output() recorderStopped = new EventEmitter<Event>();\n  @Output() recorderErrored = new EventEmitter<Event>();\n  @Output() recorderStarted = new EventEmitter<Event>();\n  @Output() recordingState = new EventEmitter<RecordingState>();\n  @Output() recordingChanged = new EventEmitter<File | undefined>();\n\n  isVideo = input<boolean>();\n  useAudio = input<boolean>();\n  fileName = input<string>();\n  fileNameFactory = input<(comp: WebcamMediaComponent) => string>();\n  fileType = input<string>();\n  extraClass = input<string>();\n  /**\n   * Seconds\n   */\n  recordingNotificationInterval = input<number>(5);\n  isBackground = input<boolean>(false);\n\n  computedRecordingNotificationInterval = computed<number>(() => (this.recordingNotificationInterval() || 5) * 1000);\n  isPicture = computed(() => !this.isVideo());\n\n  subject = computed(() => (this.isPicture() ? 'Image' : 'Video'));\n\n  isPaused = toSignal(this.recordingState.pipe(map((r) => r == 'paused')));\n\n  computedFileType = computed(() => {\n    const ft = this.fileType(),\n      isp = this.isPicture();\n    return ft || (isp ? 'image/jpeg' : 'video/mp4');\n  });\n  computedFileExtension = computed(() => this.computedFileType()?.split('/')[1]);\n  computedFileName = computed(() => {\n    const ff = this.fileNameFactory(),\n      fn = this.fileName() || ff?.(this),\n      fx = this.computedFileExtension(),\n      subject = this.subject();\n    return fn?.includes('.' + fx) ? fn : `${fn || subject}.${fx}`;\n  });\n\n  streamingVideoTemplate = viewChild<TemplateRef<any>>('streamVideoTemp');\n  recordingVideoRef = viewChild<ElementRef<HTMLVideoElement>>('recordingVideoTag');\n  recordingVideoRefEl = computed(() => this.recordingVideoRef()?.nativeElement);\n\n  streaming = signal<boolean | undefined>(undefined);\n  preview = signal<boolean>(false);\n  computedPreview = computed<boolean>(() => !this.isBackground() && this.preview());\n\n  // The width and height of the captured photo. We will set the\n  // width to the value defined here, but the height will be\n  // calculated based on the aspect ratio of the input stream.\n\n  width = model<number>(1080); // We will scale the photo width to this\n  height = computed<number>(() => {\n    const width = this.width();\n    return width / (4 / 3);\n  }); // This will be computed based on the input stream\n\n  // The various HTML elements we need to configure or control. These\n  // will be set by the startup() function.\n\n  protected canvas = signal<HTMLCanvasElement | undefined>(undefined);\n  mediaFile = signal<File | undefined>(undefined);\n  photoSrc = signal<string | undefined>(undefined);\n  videoSrc = signal<string | undefined>(undefined);\n\n  stream = signal<MediaStream>(undefined);\n\n  sub: Subscription;\n  recorder: MediaRecorder;\n  isRecording = toSignal<boolean>(this.recordingState.pipe(map((r) => r == 'recording' || r == 'paused')));\n  recordedBlobs: Blob[] = [];\n\n  recordedTime = signal<number>(0);\n\n  ngAfterViewInit(): void {\n    this.startUpWebcam();\n  }\n\n  ngOnDestroy(): void {\n    this.sub?.unsubscribe();\n  }\n\n  /**\n   * Start the webcam stream\n   */\n  startUpWebcam() {\n    this.clear();\n    navigator.mediaDevices\n      .getUserMedia({ video: true, audio: this.useAudio() })\n      .then((stream) => {\n        this.streaming.set(true);\n        this.canvas.set(document.createElement('canvas'));\n        this.canvas()?.setAttribute('width', this.width() + 'px');\n        this.canvas()?.setAttribute('height', this.height() + 'px');\n        this.stream.set(stream);\n        // debugger;\n      })\n      .catch((err) => {\n        console.error(`An error occurred:`, err);\n      });\n  }\n\n  /**\n   * Indicate that the user is done with the webcam component.\n   *\n   * Any media obtained will be emitted\n   */\n  save() {\n    if (this.isPicture())\n      new Observable<File>((sub) =>\n        this.canvas()!.toBlob(\n          (blob) => {\n            if (blob) {\n              this.mediaFile.set(new File([blob], this.computedFileName()));\n              sub.next(this.mediaFile());\n              sub.complete();\n            } else sub.error(`No ${this.subject()} file generated`);\n          },\n          this.computedFileType(),\n          1,\n        ),\n      ).subscribe(() => {\n        this.closed.emit(this.mediaFile());\n        this.closeWebcam();\n      });\n    else { \n      this.closed.emit(this.mediaFile());\n      this.closeWebcam();\n    }\n  }\n\n  /**\n   * Indicate that the user is done with the webcam component.\n   *\n   * Any media obtained will be deleted\n   */\n  cancel() {\n    this.closed.emit(null);\n    this.closeWebcam();\n  }\n\n  /**\n   * Clear the media taken and close preview\n   */\n  clear() {\n    if (this.canvas()) {\n      const context = this.canvas()?.getContext('2d');\n      context!.fillStyle = '#AAA';\n      context!.fillRect(0, 0, this.canvas()!.width, this.canvas()!.height);\n\n      const data = this.canvas()!.toDataURL(this.computedFileType());\n      this.photoSrc.set(data);\n      this.streaming.set(true);\n      this.preview.set(false);\n    }\n  }\n\n  /**\n   * Shutdown the webcam stream\n   */\n  closeWebcam() {\n    this.stream()\n      ?.getTracks()\n      .forEach((x) => x.stop());\n    this.stream.set(undefined);\n    this.streaming.set(undefined);\n    // this.stream.set(this.stream()?.getTracks[0].stop());\n  }\n\n  /**\n   * Capture a photo by fetching the current contents of the video\n   * and drawing it into a canvas, then converting that to a PNG\n   * format data URL.\n   *\n   * By drawing it on an offscreen canvas and then\n   * drawing that to the screen, we can change its size and/or apply\n   * other changes before drawing it.\n   */\n  takePicture() {\n    if (!this.isPicture()) return;\n    const context = this.canvas()!.getContext('2d'),\n      width = this.width(),\n      height = this.height();\n    if (width && height && context) {\n      this.canvas.update((canvas) => {\n        if (canvas) canvas.width = width;\n        return canvas;\n      });\n      this.canvas.update((canvas) => {\n        if (canvas) canvas.height = height;\n        return canvas;\n      });\n      context.drawImage(this.recordingVideoRefEl()!, 0, 0, width, height);\n\n      const data = this.canvas()!.toDataURL('image/png');\n      this.photoSrc.set(data);\n    } else this.clear();\n\n    this.streaming.set(false);\n    this.preview.set(true);\n  }\n\n  startRecording = (stream = this.stream()) => {\n    if (!this.isVideo()) return;\n    this.recorder = new MediaRecorder(stream);\n\n    this.recordedBlobs = [];\n    this.recordedTime.set(0);\n    const lengths: [number, number][] = [];\n    this.recorder.ondataavailable = (event) => {\n      this.recordedBlobs.push(event.data);\n      this.mediaFile.set(new File(this.recordedBlobs, this.computedFileName(), { type: this.computedFileType() }));\n      lengths.push([this.mediaFile().size, event.data.size]);\n    };\n    // this.recorder.ondataavailable = (event) => {\n    //   this.recordedBlobs = [event.data];\n    //   lengths.push(event.data.size);\n    // };\n    // this.recorder.ondataavailable = (event) => this.recordedBlobs.push(event.data);\n\n    this.recorder.start();\n    let notifSub: Subscription;\n    const startTimer = () => {\n      let lastDateTime = Date.now();\n      notifSub = timer(1000, 1000)\n        .pipe(\n          tap(() => this.recordedTime.update((t) => (t += 1))),\n          tap(() => console.log(Date.now(), this.computedRecordingNotificationInterval(), this.recordingNotificationInterval())),\n          filter(() => Date.now() - lastDateTime > this.computedRecordingNotificationInterval()),\n          tap(() => this.recorder.requestData()),\n        )\n        .subscribe((r) => {\n          lastDateTime = Date.now(); \n          this.recordingChanged.emit(this.mediaFile());\n        });\n    };\n    this.recorder.onstart = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      this.recorderStarted.emit(e);\n      startTimer();\n    };\n    this.recorder.onstop = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      this.recorderStopped.emit(e);\n      console.log('new length', lengths);\n      notifSub?.unsubscribe();\n    };\n    this.recorder.onpause = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      notifSub?.unsubscribe();\n    };\n    this.recorder.onresume = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      startTimer();\n    };\n    this.recorder.onerror = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      this.recorderErrored.emit(e);\n      notifSub?.unsubscribe();\n    };\n  };\n\n  stopRecording() {\n    if (this.recorder.state === 'recording') this.recorder.stop();\n\n    this.preview.set(true);\n    this.streaming.set(false);\n  }\n\n  pauseRecording() {\n    // debugger;\n    if (this.recorder.state === 'paused') this.recorder.resume();\n    else this.recorder.pause();\n  }\n \n}\n","<div class=\"\" [hidden]=\"streaming() != true || isBackground()\">\n  <ng-container *ngTemplateOutlet=\"streamVideoTemp\" />\n\n  @if (!isBackground()) {\n    <div class=\"wm-buttons\">\n      <div class=\"  row row-cols-lg-auto g-3 justify-content-start\">\n        @if (!isRecording()) {\n          <div>\n            <app-btn [text]=\"isPicture() ? 'Take Picture' : 'Record Video'\" (mclick)=\"isPicture() ? takePicture() : startRecording()\" />\n          </div>\n        } @else {\n          <div>\n            <app-btn text=\"{{isPaused()?'Play': 'Pause'}} Recording\" (mclick)=\"pauseRecording()\" type=\"outline\" [icon]=\"isPaused()?'play':'pause'\" />\n          </div>\n          <div>\n            <app-btn [text]=\"'Stop Recording'\" (mclick)=\"stopRecording()\" type=\"danger\" icon=\"stop\" />\n          </div>\n        }\n      </div>\n    </div>\n  }\n</div>\n\n<ng-template #streamVideoTemp>\n  <div class=\"wm-recording-video-cont\">\n    <video #recordingVideoTag [srcObject]=\"stream()\" autoplay class=\"wm-recording-video {{ extraClass() }}\"></video>\n    @if (isRecording()) {\n      <div class=\"wm-recordingDot\"></div>\n    }\n  </div>\n</ng-template>\n\n@if (computedPreview()) {\n  <div class=\"wm-preview\">\n    <div>\n      @if (isPicture()) {\n        <img class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" [src]=\"photoSrc()\" />\n      } @else {\n        <video class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" [src]=\"mediaFile()\"></video>\n      }\n    </div>\n    <div class=\"mt-3 row row-cols-lg-auto g-3\">\n      <div>\n        <app-btn text=\"Retake {{ subject() }}\" type=\"outline\" (mclick)=\"clear()\" />\n      </div>\n      <div>\n        <app-btn (mclick)=\"save()\" text=\"Done\" type=\"primary\" icon=\"save\" />\n      </div>\n      <div>\n        <app-btn (mclick)=\"cancel()\" text=\"Cancel\" type=\"danger\" icon=\"cancel\" />\n      </div>\n    </div>\n  </div>\n}\n"]}
|
|
267
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webcam-media.component.js","sourceRoot":"","sources":["../../../../../../../projects/ets-fe-ng-sdk/src/lib/Shared/components/webcam-media/webcam-media.component.ts","../../../../../../../projects/ets-fe-ng-sdk/src/lib/Shared/components/webcam-media/webcam-media.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Output,\n  TemplateRef,\n  computed,\n  input,\n  model,\n  signal,\n  viewChild,\n} from '@angular/core';\nimport { BtnComponent } from '../btn/btn.component';\nimport { debounce, debounceTime, filter, fromEvent, interval, map, Observable, of, Subscription, tap, timer } from 'rxjs';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { LoaderComponent } from '../loader/loader.component';\n\n@Component({\n  selector: 'lib-webcam-media',\n  standalone: true,\n  imports: [CommonModule, BtnComponent, LoaderComponent],\n  templateUrl: './webcam-media.component.html',\n  styleUrl: './webcam-media.component.scss',\n})\nexport class WebcamMediaComponent {\n  @Output() closed = new EventEmitter<File>();\n  @Output() recorderStopped = new EventEmitter<Event>();\n  @Output() recorderErrored = new EventEmitter<Event>();\n  @Output() recorderStarted = new EventEmitter<Event>();\n  @Output() recordingState = new EventEmitter<RecordingState>();\n  @Output() recordingChanged = new EventEmitter<File | undefined>();\n\n  isVideo = input<boolean>();\n  useAudio = input<boolean>();\n  fileName = input<string>();\n  fileNameFactory = input<(comp: WebcamMediaComponent) => string>();\n  fileType = input<string>();\n  extraClass = input<string>();\n  /**\n   * Seconds\n   */\n  recordingNotificationInterval = input<number>(5);\n  isBackground = input<boolean>(false);\n\n  computedRecordingNotificationInterval = computed<number>(() => (this.recordingNotificationInterval() || 5) * 1000);\n  isPicture = computed(() => !this.isVideo());\n\n  subject = computed(() => (this.isPicture() ? 'Image' : 'Video'));\n\n  isPaused = toSignal(this.recordingState.pipe(map((r) => r == 'paused')));\n\n  computedFileType = computed(() => {\n    const ft = this.fileType(),\n      isp = this.isPicture();\n    return ft || (isp ? 'image/jpeg' : 'video/mp4');\n  });\n  computedFileExtension = computed(() => this.computedFileType()?.split('/')[1]);\n  computedFileName = computed(() => {\n    const ff = this.fileNameFactory(),\n      fn = this.fileName() || ff?.(this),\n      fx = this.computedFileExtension(),\n      subject = this.subject();\n    return fn?.includes('.' + fx) ? fn : `${fn || subject}.${fx}`;\n  });\n\n  streamingVideoTemplate = viewChild<TemplateRef<any>>('streamVideoTemp');\n  recordingVideoRef = viewChild<ElementRef<HTMLVideoElement>>('recordingVideoTag');\n  recordingVideoRefEl = computed(() => this.recordingVideoRef()?.nativeElement);\n\n  streaming = signal<boolean | undefined>(undefined);\n  preview = signal<boolean>(false);\n  computedPreview = computed<boolean>(() => !this.isBackground() && this.preview());\n\n  // The width and height of the captured photo. We will set the\n  // width to the value defined here, but the height will be\n  // calculated based on the aspect ratio of the input stream.\n\n  width = model<number>(1080); // We will scale the photo width to this\n  height = computed<number>(() => {\n    const width = this.width();\n    return width / (4 / 3);\n  }); // This will be computed based on the input stream\n\n  // The various HTML elements we need to configure or control. These\n  // will be set by the startup() function.\n\n  protected canvas = signal<HTMLCanvasElement | undefined>(undefined);\n  mediaFile = signal<File | undefined>(undefined);\n  mediaFileLink = computed<string | undefined>(() => (this.mediaFile() ? URL.createObjectURL(this.mediaFile()) : undefined));\n\n  stream = signal<MediaStream>(undefined);\n\n  sub: Subscription;\n  recorder: MediaRecorder;\n  isRecording = toSignal<boolean>(this.recordingState.pipe(map((r) => r == 'recording' || r == 'paused')));\n  recordedBlobs: Blob[] = [];\n\n  /**In Milliseconds */\n  recordedTimeMS = signal<number>(0);\n  recordedTime = computed<number>(() => Math.round(this.recordedTimeMS() / 1000));\n  loading = signal<boolean>(false);\n\n  ngAfterViewInit(): void {\n    this.startUpWebcam();\n  }\n\n  ngOnDestroy(): void {\n    this.sub?.unsubscribe();\n    this.closeWebcam();\n  }\n\n  /**\n   * Start the webcam stream\n   */\n  startUpWebcam() {\n    this.loading.set(true);\n    this.clear();\n    navigator.mediaDevices\n      .getUserMedia({ video: true, audio: this.useAudio() })\n      .then((stream) => {\n        this.streaming.set(true);\n        this.loading.set(false);\n        // stream.onaddtrack = () => this.loading.set(false);\n        this.canvas.set(document.createElement('canvas'));\n        this.canvas()?.setAttribute('width', this.width() + 'px');\n        this.canvas()?.setAttribute('height', this.height() + 'px');\n        this.stream.set(stream);\n        // debugger;\n      })\n      .catch((err) => {\n        console.error(`An error occurred:`, err);\n      });\n  }\n\n  /**\n   * Indicate that the user is done with the webcam component.\n   *\n   * Any media obtained will be emitted\n   */\n  save() {\n    if (this.isPicture())\n      this.extractImg.subscribe(() => {\n        this.closed.emit(this.mediaFile());\n        this.closeWebcam();\n      });\n    else {\n      this.closed.emit(this.mediaFile());\n      this.closeWebcam();\n    }\n  }\n\n  /**\n   * Indicate that the user is done with the webcam component.\n   *\n   * Any media obtained will be deleted\n   */\n  cancel() {\n    this.closed.emit(null);\n    this.closeWebcam();\n  }\n\n  /**\n   * Clear the media taken and close preview\n   */\n  clear() {\n    if (this.canvas()) {\n      const context = this.canvas()?.getContext('2d');\n      context!.fillStyle = '#AAA';\n      context!.fillRect(0, 0, this.canvas()!.width, this.canvas()!.height);\n\n      const data = this.canvas()!.toDataURL(this.computedFileType());\n      this.mediaFile.set(undefined);\n      // this.photoSrc.set(data);\n      this.streaming.set(true);\n      this.preview.set(false);\n    }\n  }\n\n  retake() {\n    if (!this.stream()) this.startUpWebcam();\n    else this.clear();\n  }\n\n  /**\n   * Shutdown the webcam stream\n   */\n  closeWebcam() {\n    this.stream()\n      ?.getTracks()\n      .forEach((x) => x.stop());\n    this.stream.set(undefined);\n    this.streaming.set(undefined);\n    // this.stream.set(this.stream()?.getTracks[0].stop());\n  }\n\n  /**\n   * Capture a photo by fetching the current contents of the video\n   * and drawing it into a canvas, then converting that to a PNG\n   * format data URL.\n   *\n   * By drawing it on an offscreen canvas and then\n   * drawing that to the screen, we can change its size and/or apply\n   * other changes before drawing it.\n   */\n  takePicture() {\n    if (!this.isPicture()) return;\n    const context = this.canvas()!.getContext('2d'),\n      width = this.width(),\n      height = this.height();\n    if (width && height && context) {\n      this.canvas.update((canvas) => {\n        if (canvas) canvas.width = width;\n        return canvas;\n      });\n      this.canvas.update((canvas) => {\n        if (canvas) canvas.height = height;\n        return canvas;\n      });\n      context.drawImage(this.recordingVideoRefEl()!, 0, 0, width, height);\n      this.extractImg.subscribe();\n      // const data = this.canvas()!.toDataURL('image/png');\n      // this.photoSrc.set(data);\n    } else this.clear();\n\n    this.streaming.set(false);\n    this.preview.set(true);\n  }\n\n  startRecording = (stream = this.stream()) => {\n    if (!this.isVideo()) return;\n    this.recorder = new MediaRecorder(stream);\n\n    this.recordedBlobs = [];\n    this.recordedTimeMS.set(0);\n    const startTime = Date.now();\n    this.recorder.ondataavailable = (event) => {\n      this.recordedBlobs.push(event.data);\n      this.recordedTimeMS.set(Date.now() - startTime);\n      this.mediaFile.set(new File(this.recordedBlobs, this.computedFileName(), { type: this.computedFileType() }));\n      this.recordingChanged.emit(this.mediaFile());\n    };\n    // this.recorder.ondataavailable = (event) => {\n    //   this.recordedBlobs = [event.data];\n    //   lengths.push(event.data.size);\n    // };\n    // this.recorder.ondataavailable = (event) => this.recordedBlobs.push(event.data);\n\n    this.recorder.start(this.computedRecordingNotificationInterval());\n\n    this.recorder.onstart = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      this.recorderStarted.emit(e);\n    };\n    this.recorder.onstop = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      this.recorderStopped.emit(e);\n    };\n    this.recorder.onpause = (e) => {\n      this.recordingState.emit(this.recorder.state);\n    };\n    this.recorder.onresume = (e) => {\n      this.recordingState.emit(this.recorder.state);\n    };\n    this.recorder.onerror = (e) => {\n      this.recordingState.emit(this.recorder.state);\n      this.recorderErrored.emit(e);\n    };\n  };\n\n  stopRecording() {\n    if (this.recorder.state === 'recording') this.recorder.stop();\n\n    this.preview.set(true);\n    this.streaming.set(false);\n  }\n\n  pauseRecording() {\n    // debugger;\n    if (this.recorder.state === 'paused') this.recorder.resume();\n    else this.recorder.pause();\n  }\n\n  extractImg = new Observable<File>((sub) =>\n    this.canvas()!.toBlob(\n      (blob) => {\n        if (blob) {\n          this.mediaFile.set(new File([blob], this.computedFileName()));\n          sub.next(this.mediaFile());\n          sub.complete();\n        } else sub.error(`No ${this.subject()} file generated`);\n      },\n      this.computedFileType(),\n      1,\n    ),\n  );\n}\n","<loader [loading]=\"loading()\">\n  <div class=\"\" [hidden]=\"streaming() != true || isBackground()\">\n    <ng-container *ngTemplateOutlet=\"streamVideoTemp\" />\n\n    @if (!isBackground()) {\n      <div class=\"wm-buttons\">\n        <div class=\"row row-cols-lg-auto g-3 justify-content-center\">\n          @if (!isRecording()) {\n            <div>\n              <app-btn [text]=\"isPicture() ? 'Take Picture' : 'Record Video'\" (mclick)=\"isPicture() ? takePicture() : startRecording()\" />\n            </div>\n          } @else {\n            <div>\n              <app-btn\n                text=\"{{ isPaused() ? 'Play' : 'Pause' }} Recording\"\n                (mclick)=\"pauseRecording()\"\n                type=\"outline\"\n                [icon]=\"isPaused() ? 'play' : 'pause'\" />\n            </div>\n            <div>\n              <app-btn [text]=\"'Stop Recording'\" (mclick)=\"stopRecording()\" type=\"danger\" icon=\"stop\" />\n            </div>\n          }\n        </div>\n      </div>\n    }\n  </div>\n\n  <ng-template #streamVideoTemp>\n    <div class=\"wm-recording-video-cont\">\n      <video #recordingVideoTag [srcObject]=\"stream()\" autoplay class=\"wm-recording-video {{ extraClass() }}\"></video>\n      @if (isRecording()) {\n        <div class=\"wm-recordingDot\"></div>\n      }\n    </div>\n  </ng-template>\n\n  @if (computedPreview()) {\n    <div class=\"wm-preview\">\n      <div>\n        @if (isPicture()) {\n          <img class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" [src]=\"mediaFileLink()\" />\n        } @else {\n          <video class=\"w-100\" [style.width.px]=\"width()\" [style.height.px]=\"height()\" controls autoplay [src]=\"mediaFileLink()\"></video>\n        }\n      </div>\n      <div class=\"mt-3 row row-cols-lg-auto justify-content-center g-3\">\n        <div>\n          <app-btn text=\"Retake {{ subject() }}\" type=\"outline\" (mclick)=\"retake()\" />\n        </div>\n        <div>\n          <app-btn (mclick)=\"save()\" text=\"Done\" type=\"primary\" icon=\"save\" />\n        </div>\n        <div>\n          <app-btn (mclick)=\"cancel()\" text=\"Cancel\" type=\"danger\" icon=\"cancel\" />\n        </div>\n      </div>\n    </div>\n  }\n</loader>\n"]}
|