kwikid-camera
Version:
KwikID's Camera Component
188 lines • 21.4 kB
JavaScript
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"]}