UNPKG

kwikid-camera

Version:
188 lines 21.4 kB
import { __awaiter, __decorate } from "tslib"; import { Injectable } from "@angular/core"; import { isNotEmptyValue, logMethod } from "kwikid-toolkit"; import * as RecordRTC from "recordrtc"; import { Subject } from "rxjs"; import { IRecordingState } from "./kwikid-camera-record.definitions"; import { getBase64FromVideoFile } from "./kwikid-camera-record.helper"; import * as i0 from "@angular/core"; export class KwikIDCameraRecordService { constructor() { this._mediaStream = new Subject(); this._blob = new Subject(); this.state = IRecordingState.NONE; this._state = new Subject(); this.isFrontCamera = false; this.videoDeviceId = null; this.startTime = 0; this.pauseTime = 0; this.resumeTime = 0; this.totalPausedTime = 0; this.totalRecordingTime = 0; this.minRecordingTimeInSeconds = null; this.maxRecordingTimeInSeconds = null; this._timerUpdate = new Subject(); } // getObservable getMediaStream() { return this._mediaStream.asObservable(); } getBlob() { return this._blob.asObservable(); } getTimerUpdate() { return this._timerUpdate.asObservable(); } getRecordingState() { return this._state.asObservable(); } setRecordingState(state) { this.state = state; this._state.next(state); } resetTimer() { clearInterval(this.timerInterval); this.startTime = 0; this.pauseTime = 0; this.resumeTime = 0; this.totalPausedTime = 0; this.totalRecordingTime = 0; this._timerUpdate.next(0); } updateTimer() { this.timerInterval = setInterval(() => { if (this.state === IRecordingState.RECORDING) { this.totalRecordingTime = Math.round((Date.now() - this.startTime - this.totalPausedTime) / 1000); if (isNotEmptyValue(this.maxRecordingTimeInSeconds) && this.totalRecordingTime > this.maxRecordingTimeInSeconds) { this.stopRecording(); } else { this._timerUpdate.next(this.totalRecordingTime); } } if (this.state === IRecordingState.NONE || this.state === IRecordingState.RECORDED) { this._timerUpdate.next(0); } }, 1000); } handleRecording() { return __awaiter(this, void 0, void 0, function* () { this.startTime = Date.now(); if (isNotEmptyValue(this.videoDeviceId)) { this.mediaStream = yield navigator.mediaDevices.getUserMedia({ audio: true, video: { deviceId: { exact: this.videoDeviceId } } }); } else { this.mediaStream = yield navigator.mediaDevices.getUserMedia({ audio: true, video: { facingMode: this.isFrontCamera ? "user" : "environment" } }); } this._mediaStream.next(this.mediaStream); this.recorder = new RecordRTC(this.mediaStream, { type: "video", disableLogs: true, mimeType: "video/webm;codecs=vp9", frameInterval: 60 }); this.recorder.startRecording(); this.setRecordingState(IRecordingState.RECORDING); }); } startRecording({ minRecordingTimeInSeconds = null, maxRecordingTimeInSeconds = null, isFrontCamera = false, videoDeviceId = null }) { this.isFrontCamera = isFrontCamera; this.videoDeviceId = videoDeviceId; this.minRecordingTimeInSeconds = minRecordingTimeInSeconds; this.maxRecordingTimeInSeconds = maxRecordingTimeInSeconds; this.handleRecording(); this.updateTimer(); } pauseRecording() { if (!this.recorder) { return; } this.pauseTime = Date.now(); this.setRecordingState(IRecordingState.PAUSED); this.recorder.pauseRecording(); } resumeRecording() { if (!this.recorder) { this.handleRecording(); } else { this.resumeTime = Date.now(); this.totalPausedTime += this.resumeTime - this.pauseTime; this.recorder.resumeRecording(); this.setRecordingState(IRecordingState.RECORDING); } } stopRecording() { if (!this.recorder) { return; } if (this.minRecordingTimeInSeconds > this.totalRecordingTime) { return; } this.recorder.stopRecording(() => __awaiter(this, void 0, void 0, function* () { this.blob = this.recorder.getBlob(); const videoBase64 = yield getBase64FromVideoFile(this.blob); this._blob.next(videoBase64); this.mediaStream.stop(); this.recorder.destroy(); this.mediaStream = null; this.recorder = null; this.setRecordingState(IRecordingState.RECORDED); this.resetTimer(); })); } downloadRecording() { RecordRTC.invokeSaveAsDialog(this.blob, `${Date.now()}.webm`); } clearRecording() { this.blob = null; this.recorder = null; this.mediaStream = null; this.setRecordingState(IRecordingState.NONE); this.resetTimer(); } } /** @nocollapse */ KwikIDCameraRecordService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: KwikIDCameraRecordService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); /** @nocollapse */ KwikIDCameraRecordService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: KwikIDCameraRecordService, providedIn: "root" }); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "handleRecording", null); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "startRecording", null); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "pauseRecording", null); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "resumeRecording", null); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "stopRecording", null); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "downloadRecording", null); __decorate([ logMethod ], KwikIDCameraRecordService.prototype, "clearRecording", null); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: KwikIDCameraRecordService, decorators: [{ type: Injectable, args: [{ providedIn: "root" }] }], propDecorators: { handleRecording: [], startRecording: [], pauseRecording: [], resumeRecording: [], stopRecording: [], downloadRecording: [], clearRecording: [] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kwikid-camera-record.service.js","sourceRoot":"","sources":["../../../../../../projects/kwikid-camera/src/lib/components/kwikid-camera-record/kwikid-camera-record.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from \"@angular/core\";\nimport { isNotEmptyValue, logMethod } from \"kwikid-toolkit\";\nimport * as RecordRTC from \"recordrtc\";\nimport { Subject } from \"rxjs\";\nimport {\n  IRecordingState,\n  TRecordingState\n} from \"./kwikid-camera-record.definitions\";\nimport { getBase64FromVideoFile } from \"./kwikid-camera-record.helper\";\n\n@Injectable({\n  providedIn: \"root\"\n})\nexport class KwikIDCameraRecordService {\n  private mediaStream: any;\n\n  private recorder: any;\n\n  private blob: any;\n\n  private _mediaStream = new Subject<any>();\n\n  private _blob = new Subject<any>();\n\n  private state: TRecordingState = IRecordingState.NONE;\n\n  private _state = new Subject<TRecordingState>();\n\n  private isFrontCamera = false;\n\n  private videoDeviceId: string = null;\n\n  private timerInterval: any;\n\n  private startTime = 0;\n\n  private pauseTime = 0;\n\n  private resumeTime = 0;\n\n  private totalPausedTime = 0;\n\n  private totalRecordingTime = 0;\n\n  private minRecordingTimeInSeconds: number = null;\n\n  private maxRecordingTimeInSeconds: number = null;\n\n  private _timerUpdate = new Subject<number>();\n\n  // getObservable\n  getMediaStream() {\n    return this._mediaStream.asObservable();\n  }\n\n  getBlob() {\n    return this._blob.asObservable();\n  }\n\n  getTimerUpdate() {\n    return this._timerUpdate.asObservable();\n  }\n\n  getRecordingState() {\n    return this._state.asObservable();\n  }\n\n  setRecordingState(state: TRecordingState) {\n    this.state = state;\n    this._state.next(state);\n  }\n\n  private resetTimer() {\n    clearInterval(this.timerInterval);\n    this.startTime = 0;\n    this.pauseTime = 0;\n    this.resumeTime = 0;\n    this.totalPausedTime = 0;\n    this.totalRecordingTime = 0;\n    this._timerUpdate.next(0);\n  }\n\n  private updateTimer() {\n    this.timerInterval = setInterval(() => {\n      if (this.state === IRecordingState.RECORDING) {\n        this.totalRecordingTime = Math.round(\n          (Date.now() - this.startTime - this.totalPausedTime) / 1000\n        );\n\n        if (\n          isNotEmptyValue(this.maxRecordingTimeInSeconds) &&\n          this.totalRecordingTime > this.maxRecordingTimeInSeconds\n        ) {\n          this.stopRecording();\n        } else {\n          this._timerUpdate.next(this.totalRecordingTime);\n        }\n      }\n      if (\n        this.state === IRecordingState.NONE ||\n        this.state === IRecordingState.RECORDED\n      ) {\n        this._timerUpdate.next(0);\n      }\n    }, 1000);\n  }\n\n  @logMethod\n  async handleRecording() {\n    this.startTime = Date.now();\n\n    if (isNotEmptyValue(this.videoDeviceId)) {\n      this.mediaStream = await navigator.mediaDevices.getUserMedia({\n        audio: true,\n        video: {\n          deviceId: {\n            exact: this.videoDeviceId\n          }\n        }\n      });\n    } else {\n      this.mediaStream = await navigator.mediaDevices.getUserMedia({\n        audio: true,\n        video: {\n          facingMode: this.isFrontCamera ? \"user\" : \"environment\"\n        }\n      });\n    }\n    this._mediaStream.next(this.mediaStream);\n    this.recorder = new RecordRTC(this.mediaStream, {\n      type: \"video\",\n      disableLogs: true,\n      mimeType: \"video/webm;codecs=vp9\",\n      frameInterval: 60\n    });\n    this.recorder.startRecording();\n    this.setRecordingState(IRecordingState.RECORDING);\n  }\n\n  @logMethod\n  startRecording({\n    minRecordingTimeInSeconds = null,\n    maxRecordingTimeInSeconds = null,\n    isFrontCamera = false,\n    videoDeviceId = null\n  }) {\n    this.isFrontCamera = isFrontCamera;\n    this.videoDeviceId = videoDeviceId;\n    this.minRecordingTimeInSeconds = minRecordingTimeInSeconds;\n    this.maxRecordingTimeInSeconds = maxRecordingTimeInSeconds;\n\n    this.handleRecording();\n    this.updateTimer();\n  }\n\n  @logMethod\n  pauseRecording() {\n    if (!this.recorder) {\n      return;\n    }\n    this.pauseTime = Date.now();\n\n    this.setRecordingState(IRecordingState.PAUSED);\n    this.recorder.pauseRecording();\n  }\n\n  @logMethod\n  resumeRecording() {\n    if (!this.recorder) {\n      this.handleRecording();\n    } else {\n      this.resumeTime = Date.now();\n      this.totalPausedTime += this.resumeTime - this.pauseTime;\n\n      this.recorder.resumeRecording();\n      this.setRecordingState(IRecordingState.RECORDING);\n    }\n  }\n\n  @logMethod\n  stopRecording() {\n    if (!this.recorder) {\n      return;\n    }\n\n    if (this.minRecordingTimeInSeconds > this.totalRecordingTime) {\n      return;\n    }\n\n    this.recorder.stopRecording(async () => {\n      this.blob = this.recorder.getBlob();\n\n      const videoBase64 = await getBase64FromVideoFile(this.blob);\n\n      this._blob.next(videoBase64);\n\n      this.mediaStream.stop();\n      this.recorder.destroy();\n      this.mediaStream = null;\n      this.recorder = null;\n\n      this.setRecordingState(IRecordingState.RECORDED);\n      this.resetTimer();\n    });\n  }\n\n  @logMethod\n  downloadRecording() {\n    RecordRTC.invokeSaveAsDialog(this.blob, `${Date.now()}.webm`);\n  }\n\n  @logMethod\n  clearRecording() {\n    this.blob = null;\n    this.recorder = null;\n    this.mediaStream = null;\n\n    this.setRecordingState(IRecordingState.NONE);\n    this.resetTimer();\n  }\n}\n"]}