@tilde-nlp/ngx-translate 1.0.2 → 1.0.3

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.
@@ -1,5 +1,5 @@
1
1
  import { Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
2
- import { AsrClient, RecognitionStatus } from 'asr-api-client';
2
+ import { AsrClient, RecognitionStatus } from '@tilde-nlp/asr-client';
3
3
  import { AudioErrorCase } from '../models/audio-error-case.model';
4
4
  import * as i0 from "@angular/core";
5
5
  import * as i1 from "@angular/common";
@@ -148,4 +148,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
148
148
  }], language: [{
149
149
  type: Input
150
150
  }] } });
151
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tld-dictate.component.js","sourceRoot":"","sources":["../../../../../../../projects/tld-translate/src/lib/modules/tld-audio/tld-dictate/tld-dictate.component.ts","../../../../../../../projects/tld-translate/src/lib/modules/tld-audio/tld-dictate/tld-dictate.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, ViewChild } from '@angular/core';\r\nimport { AsrClient, AsrClientConfiguration, RecognitionStatus, SpeechRecognitionResult } from 'asr-api-client';\r\nimport { AudioErrorCase } from '../models/audio-error-case.model';\r\nimport { IDictateConfiguration } from '../models/dictate-configurationi.model';\r\n\r\n@Component({\r\n  selector: 'tld-dictate',\r\n  templateUrl: './tld-dictate.component.html',\r\n  styleUrls: ['./tld-dictate.component.scss']\r\n})\r\nexport class TldDictateComponent implements OnInit {\r\n\r\n  @HostListener('window:resize', [])\r\n  onResize() {\r\n    this.setCanvasWidth();\r\n  }\r\n\r\n  @ViewChild(\"canvasWrapper\") canvasWrapper: ElementRef<any>;\r\n  recordLength: Date = new Date(0, 0, 0, 0, 0, 0);\r\n  canvasWidth: number;\r\n  isProcessing: boolean;\r\n  hasAnyMicrophone: boolean;\r\n  // asrApiClient: ASRApiClient;\r\n  asrClient: AsrClient;\r\n\r\n  get maxLength() { return this.configuration?.maxLength };\r\n\r\n  private _isRecording: boolean;\r\n  get isRecording() { return this._isRecording; }\r\n  set isRecording(value: boolean) {\r\n    const shouldClearSource = !this.isRecording && value;\r\n    // if record ended, dictate text emitted one last time, so it translates automatically.\r\n    this._isRecording = value;\r\n    this.isDictating.emit(this._isRecording);\r\n    if (shouldClearSource) {\r\n      this.dictatedText = \"\";\r\n    }\r\n\r\n    if (!this.isRecording && this.interval) {\r\n      this.recordLength = new Date(0, 0, 0, 0, 0, 0);\r\n      clearInterval(this.interval);\r\n    }\r\n  }\r\n\r\n  private _dictatedText: string;\r\n  set dictatedText(value: string) {\r\n    this._dictatedText = value;\r\n    this.result.emit(this._dictatedText);\r\n  }\r\n  get dictatedText() { return this._dictatedText; }\r\n\r\n  readonly visualizataionId = \"dictate-visualization\";\r\n  private interval: any;\r\n\r\n  /**\r\n   * Last recognized text\r\n   */\r\n  @Output() result: EventEmitter<string> = new EventEmitter<string>();\r\n  @Output() partialResult: EventEmitter<string> = new EventEmitter<string>();\r\n  @Output() isDictating: EventEmitter<boolean> = new EventEmitter<boolean>();\r\n  @Output() onError: EventEmitter<AudioErrorCase> = new EventEmitter<AudioErrorCase>();\r\n  @Input() configuration: IDictateConfiguration;\r\n\r\n  private _language: string;\r\n  @Input() set language(value: string) {\r\n    this._language = value;\r\n    this.asrClient?.changeLanguage(value);\r\n  }\r\n\r\n  constructor() {\r\n  }\r\n\r\n  ngOnInit() {\r\n    const config: AsrClientConfiguration = {\r\n      appId: this.configuration.appId,\r\n      appSecret: this.configuration.appSecret,\r\n      language: this._language,\r\n      apiUrl: this.configuration.apiUrl,\r\n      visualizerConfig: {\r\n        visualizerId: this.visualizataionId\r\n      },\r\n      onResult: (result) => { this.handleResult(result) },\r\n      onRecordingStartStop: (isRecording) => {\r\n        this.isRecording = isRecording;\r\n        setTimeout(() => {\r\n          this.setCanvasWidth();\r\n          this.asrClient.audioVisualizer?.visualizeAudio();\r\n        }, 1);\r\n      },\r\n      onError: (error) => {\r\n        if (error.name === \"NotAllowedError\") {\r\n          this.emitError(AudioErrorCase.MIC_NOT_ALLOWED);\r\n          return;\r\n        }\r\n        throw error;\r\n      },\r\n      onMessage: (message) => {\r\n        if (message === RecognitionStatus.NoSpeech) {\r\n          this.emitError(AudioErrorCase.NO_SPEECH);\r\n        }\r\n      }\r\n    }\r\n    this.asrClient = new AsrClient(config);\r\n\r\n    this.findMic();\r\n  }\r\n\r\n  startRecording() {\r\n    this.asrClient.beginVoiceRecognition();\r\n    this.interval = setInterval(() => {\r\n      const recordTimeInSeconds = this.asrClient?.audioContext?.currentTime ?? 0;\r\n      if (this.maxLength && recordTimeInSeconds > this.maxLength) {\r\n        this.stopRecording();\r\n        this.emitError(AudioErrorCase.MAX_TIME_EXCEEDED);\r\n        return;\r\n      }\r\n      this.recordLength = new Date(0, 0, 0, 0, 0, recordTimeInSeconds);\r\n    }, 1000);\r\n  }\r\n\r\n  stopRecording() {\r\n    this.asrClient.endVoiceRecognition();\r\n  }\r\n\r\n  private setCanvasWidth() {\r\n    this.canvasWidth = this.canvasWrapper?.nativeElement.offsetWidth;\r\n  }\r\n\r\n  private findMic() {\r\n    navigator.mediaDevices.enumerateDevices().then((devices: MediaDeviceInfo[]) => {\r\n      this.hasAnyMicrophone = false;\r\n      for (const device of devices) {\r\n        if (device.kind === \"audioinput\") {\r\n          this.hasAnyMicrophone = true;\r\n          return;\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private handleResult(result: SpeechRecognitionResult) {\r\n    // not necessary to display <unk> tags\r\n    const resultText = result.hypotheses[0].transcript?.replace(\"<unk>\", \"\");\r\n    if (result.final) {\r\n      let text = (this.dictatedText.length > 0 ? \" \" : \"\") + resultText + (resultText.endsWith(\".\") ? \"\" : \".\");\r\n      this.dictatedText += text;\r\n    }\r\n    else {\r\n      this.partialResult.emit(resultText);\r\n    }\r\n  }\r\n\r\n  private emitError(error: AudioErrorCase) {\r\n    this.onError.emit(error);\r\n  }\r\n}\r\n","<ng-container *ngIf=\"!isRecording\">\r\n    <button [disabled]=\"!hasAnyMicrophone\" mat-button mat-icon-button fxLayoutAlign=\"center center\"\r\n        (click)=\"startRecording()\" color=\"accent\"\r\n        [attr.aria-label]=\"(hasAnyMicrophone?'AUDIO.START':'AUDIO.MIC_NOT_FOUND') | translate\">\r\n        <span class=\"material-icons\" *ngIf=\"hasAnyMicrophone\" color=\"accent\" [matTooltip]=\"'AUDIO.START' | translate\">\r\n            mic\r\n        </span>\r\n        <span class=\"material-icons\" [matTooltip]=\"'AUDIO.MIC_NOT_FOUND' | translate\" *ngIf=\"!hasAnyMicrophone\">\r\n            mic_off\r\n        </span>\r\n    </button>\r\n</ng-container>\r\n<ng-container *ngIf=\"isRecording\">\r\n    <div fxLayout=\"row\" fxLayoutAlign=\"start end\">\r\n        <div fxLayout=column>\r\n            <button mat-button mat-icon-button fxLayoutAlign=\"center center\" [disabled]=\"!isRecording\"\r\n                (click)=\"stopRecording()\" color=\"accent\" [matTooltip]=\"'AUDIO.STOP' | translate\"\r\n                [attr.aria-label]=\"'AUDIO.STOP' | translate\">\r\n                <span class=\"material-icons\">\r\n                    stop\r\n                </span>\r\n            </button>\r\n            <div class=\"tld-audio-timer\" fxLayoutAlign=\"center center\">\r\n                {{recordLength | date: 'mm:ss'}}\r\n            </div>\r\n        </div>\r\n        <div fxFlex #canvasWrapper>\r\n            <canvas [attr.width]=\"canvasWidth\" height=\"60\" [attr.id]=\"visualizataionId\"></canvas>\r\n        </div>\r\n    </div>\r\n</ng-container>"]}
151
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tld-dictate.component.js","sourceRoot":"","sources":["../../../../../../../projects/tld-translate/src/lib/modules/tld-audio/tld-dictate/tld-dictate.component.ts","../../../../../../../projects/tld-translate/src/lib/modules/tld-audio/tld-dictate/tld-dictate.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, ViewChild } from '@angular/core';\r\nimport { AsrClient, AsrClientConfiguration, RecognitionStatus, SpeechRecognitionResult } from '@tilde-nlp/asr-client';\r\nimport { AudioErrorCase } from '../models/audio-error-case.model';\r\nimport { IDictateConfiguration } from '../models/dictate-configurationi.model';\r\n\r\n@Component({\r\n  selector: 'tld-dictate',\r\n  templateUrl: './tld-dictate.component.html',\r\n  styleUrls: ['./tld-dictate.component.scss']\r\n})\r\nexport class TldDictateComponent implements OnInit {\r\n\r\n  @HostListener('window:resize', [])\r\n  onResize() {\r\n    this.setCanvasWidth();\r\n  }\r\n\r\n  @ViewChild(\"canvasWrapper\") canvasWrapper: ElementRef<any>;\r\n  recordLength: Date = new Date(0, 0, 0, 0, 0, 0);\r\n  canvasWidth: number;\r\n  isProcessing: boolean;\r\n  hasAnyMicrophone: boolean;\r\n  // asrApiClient: ASRApiClient;\r\n  asrClient: AsrClient;\r\n\r\n  get maxLength() { return this.configuration?.maxLength };\r\n\r\n  private _isRecording: boolean;\r\n  get isRecording() { return this._isRecording; }\r\n  set isRecording(value: boolean) {\r\n    const shouldClearSource = !this.isRecording && value;\r\n    // if record ended, dictate text emitted one last time, so it translates automatically.\r\n    this._isRecording = value;\r\n    this.isDictating.emit(this._isRecording);\r\n    if (shouldClearSource) {\r\n      this.dictatedText = \"\";\r\n    }\r\n\r\n    if (!this.isRecording && this.interval) {\r\n      this.recordLength = new Date(0, 0, 0, 0, 0, 0);\r\n      clearInterval(this.interval);\r\n    }\r\n  }\r\n\r\n  private _dictatedText: string;\r\n  set dictatedText(value: string) {\r\n    this._dictatedText = value;\r\n    this.result.emit(this._dictatedText);\r\n  }\r\n  get dictatedText() { return this._dictatedText; }\r\n\r\n  readonly visualizataionId = \"dictate-visualization\";\r\n  private interval: any;\r\n\r\n  /**\r\n   * Last recognized text\r\n   */\r\n  @Output() result: EventEmitter<string> = new EventEmitter<string>();\r\n  @Output() partialResult: EventEmitter<string> = new EventEmitter<string>();\r\n  @Output() isDictating: EventEmitter<boolean> = new EventEmitter<boolean>();\r\n  @Output() onError: EventEmitter<AudioErrorCase> = new EventEmitter<AudioErrorCase>();\r\n  @Input() configuration: IDictateConfiguration;\r\n\r\n  private _language: string;\r\n  @Input() set language(value: string) {\r\n    this._language = value;\r\n    this.asrClient?.changeLanguage(value);\r\n  }\r\n\r\n  constructor() {\r\n  }\r\n\r\n  ngOnInit() {\r\n    const config: AsrClientConfiguration = {\r\n      appId: this.configuration.appId,\r\n      appSecret: this.configuration.appSecret,\r\n      language: this._language,\r\n      apiUrl: this.configuration.apiUrl,\r\n      visualizerConfig: {\r\n        visualizerId: this.visualizataionId\r\n      },\r\n      onResult: (result) => { this.handleResult(result) },\r\n      onRecordingStartStop: (isRecording) => {\r\n        this.isRecording = isRecording;\r\n        setTimeout(() => {\r\n          this.setCanvasWidth();\r\n          this.asrClient.audioVisualizer?.visualizeAudio();\r\n        }, 1);\r\n      },\r\n      onError: (error) => {\r\n        if (error.name === \"NotAllowedError\") {\r\n          this.emitError(AudioErrorCase.MIC_NOT_ALLOWED);\r\n          return;\r\n        }\r\n        throw error;\r\n      },\r\n      onMessage: (message) => {\r\n        if (message === RecognitionStatus.NoSpeech) {\r\n          this.emitError(AudioErrorCase.NO_SPEECH);\r\n        }\r\n      }\r\n    }\r\n    this.asrClient = new AsrClient(config);\r\n\r\n    this.findMic();\r\n  }\r\n\r\n  startRecording() {\r\n    this.asrClient.beginVoiceRecognition();\r\n    this.interval = setInterval(() => {\r\n      const recordTimeInSeconds = this.asrClient?.audioContext?.currentTime ?? 0;\r\n      if (this.maxLength && recordTimeInSeconds > this.maxLength) {\r\n        this.stopRecording();\r\n        this.emitError(AudioErrorCase.MAX_TIME_EXCEEDED);\r\n        return;\r\n      }\r\n      this.recordLength = new Date(0, 0, 0, 0, 0, recordTimeInSeconds);\r\n    }, 1000);\r\n  }\r\n\r\n  stopRecording() {\r\n    this.asrClient.endVoiceRecognition();\r\n  }\r\n\r\n  private setCanvasWidth() {\r\n    this.canvasWidth = this.canvasWrapper?.nativeElement.offsetWidth;\r\n  }\r\n\r\n  private findMic() {\r\n    navigator.mediaDevices.enumerateDevices().then((devices: MediaDeviceInfo[]) => {\r\n      this.hasAnyMicrophone = false;\r\n      for (const device of devices) {\r\n        if (device.kind === \"audioinput\") {\r\n          this.hasAnyMicrophone = true;\r\n          return;\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  private handleResult(result: SpeechRecognitionResult) {\r\n    // not necessary to display <unk> tags\r\n    const resultText = result.hypotheses[0].transcript?.replace(\"<unk>\", \"\");\r\n    if (result.final) {\r\n      let text = (this.dictatedText.length > 0 ? \" \" : \"\") + resultText + (resultText.endsWith(\".\") ? \"\" : \".\");\r\n      this.dictatedText += text;\r\n    }\r\n    else {\r\n      this.partialResult.emit(resultText);\r\n    }\r\n  }\r\n\r\n  private emitError(error: AudioErrorCase) {\r\n    this.onError.emit(error);\r\n  }\r\n}\r\n","<ng-container *ngIf=\"!isRecording\">\r\n    <button [disabled]=\"!hasAnyMicrophone\" mat-button mat-icon-button fxLayoutAlign=\"center center\"\r\n        (click)=\"startRecording()\" color=\"accent\"\r\n        [attr.aria-label]=\"(hasAnyMicrophone?'AUDIO.START':'AUDIO.MIC_NOT_FOUND') | translate\">\r\n        <span class=\"material-icons\" *ngIf=\"hasAnyMicrophone\" color=\"accent\" [matTooltip]=\"'AUDIO.START' | translate\">\r\n            mic\r\n        </span>\r\n        <span class=\"material-icons\" [matTooltip]=\"'AUDIO.MIC_NOT_FOUND' | translate\" *ngIf=\"!hasAnyMicrophone\">\r\n            mic_off\r\n        </span>\r\n    </button>\r\n</ng-container>\r\n<ng-container *ngIf=\"isRecording\">\r\n    <div fxLayout=\"row\" fxLayoutAlign=\"start end\">\r\n        <div fxLayout=column>\r\n            <button mat-button mat-icon-button fxLayoutAlign=\"center center\" [disabled]=\"!isRecording\"\r\n                (click)=\"stopRecording()\" color=\"accent\" [matTooltip]=\"'AUDIO.STOP' | translate\"\r\n                [attr.aria-label]=\"'AUDIO.STOP' | translate\">\r\n                <span class=\"material-icons\">\r\n                    stop\r\n                </span>\r\n            </button>\r\n            <div class=\"tld-audio-timer\" fxLayoutAlign=\"center center\">\r\n                {{recordLength | date: 'mm:ss'}}\r\n            </div>\r\n        </div>\r\n        <div fxFlex #canvasWrapper>\r\n            <canvas [attr.width]=\"canvasWidth\" height=\"60\" [attr.id]=\"visualizataionId\"></canvas>\r\n        </div>\r\n    </div>\r\n</ng-container>"]}
@@ -55,7 +55,7 @@ import * as i2$3 from '@angular/cdk/clipboard';
55
55
  import { ClipboardModule } from '@angular/cdk/clipboard';
56
56
  import * as i8 from '@angular/cdk/layout';
57
57
  import { Breakpoints, LayoutModule } from '@angular/cdk/layout';
58
- import { RecognitionStatus, AsrClient } from 'asr-api-client';
58
+ import { RecognitionStatus, AsrClient } from '@tilde-nlp/asr-client';
59
59
  import * as i1$4 from '@angular/cdk/overlay';
60
60
  import * as i5$2 from '@angular/cdk/text-field';
61
61
  import { BrowserAnimationsModule } from '@angular/platform-browser/animations';