@tilde-nlp/ngx-translate 7.0.60 → 7.0.61

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.
@@ -93,6 +93,7 @@ export class TldDictateComponent {
93
93
  this.isRecording = isRecording;
94
94
  },
95
95
  onError: (error) => {
96
+ this.hasAnyMicrophone = false;
96
97
  this.stopRecording();
97
98
  if (error.name === this.notAllowedErrorName) {
98
99
  this.emitError(AudioErrorCase.MIC_NOT_ALLOWED);
@@ -127,4 +128,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
127
128
  }], systemId: [{
128
129
  type: Input
129
130
  }] } });
130
- //# 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, EventEmitter, Input, OnInit, Output } from '@angular/core';\r\nimport { AsrClient, AsrClientConfiguration, RecognitionStatus, SAMPLE_RATE_PLACEHOLDER, SpeechRecognitionResult } from '@tilde-nlp/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  recordLength: Date = new Date(0, 0, 0, 0, 0, 0);\r\n  isProcessing: boolean;\r\n  hasAnyMicrophone: boolean;\r\n  // asrApiClient: ASRApiClient;\r\n  asrClient: AsrClient;\r\n  private readonly notAllowedErrorName = \"NotAllowedError\";\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\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  /**\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 _systemId: string;\r\n  @Input() set systemId(value: string) {\r\n    this._systemId = value;\r\n    this.stopRecording();\r\n  }\r\n\r\n  constructor() {\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.hasAnyAudioInput();\r\n  }\r\n\r\n  startRecording() {\r\n    this.createClient();\r\n    this.asrClient.beginVoiceRecognition();\r\n  }\r\n\r\n  stopRecording() {\r\n    this.asrClient?.endVoiceRecognition();\r\n    this.asrClient = null;\r\n  }\r\n\r\n  private hasAnyAudioInput() {\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;\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  private createClient() {\r\n    const config: AsrClientConfiguration = {\r\n      url: `${this.configuration.apiUrl}/${this._systemId}?contentType=audio/x-raw&sampleRate=${SAMPLE_RATE_PLACEHOLDER}&x-api-key=${this.configuration.apiKey}`,\r\n      replaceSampleRateInUrl: true,\r\n      onResult: (result) => { this.handleResult(result as SpeechRecognitionResult) },\r\n      onRecordingStartStop: (isRecording) => {\r\n        this.isRecording = isRecording;\r\n      },\r\n      onError: (error) => {\r\n        this.stopRecording();\r\n        if (error.name === this.notAllowedErrorName) {\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}\r\n","<ng-container *ngIf=\"!isRecording\">\r\n\t<button\r\n\t\t[matTooltip]=\"'AUDIO.START' | translate\"\r\n\t\t[disabled]=\"!hasAnyMicrophone\"\r\n\t\tmat-icon-button\r\n\t\tfxLayoutAlign=\"center center\"\r\n\t\t(click)=\"startRecording()\"\r\n\t\t[attr.aria-label]=\"(hasAnyMicrophone ? 'AUDIO.START' : 'AUDIO.MIC_NOT_FOUND') | translate\"\r\n\t>\r\n\t\t<span class=\"material-icons\" *ngIf=\"hasAnyMicrophone\" [matTooltip]=\"'AUDIO.START' | translate\"> mic </span>\r\n\t\t<span class=\"material-icons\" [matTooltip]=\"'AUDIO.MIC_NOT_FOUND' | translate\" *ngIf=\"!hasAnyMicrophone\"> mic_off </span>\r\n\t</button>\r\n</ng-container>\r\n<ng-container *ngIf=\"isRecording\">\r\n\t<div fxLayout=\"row\" fxLayoutAlign=\"start end\">\r\n\t\t<div fxLayout=\"column\">\r\n\t\t\t<button\r\n\t\t\t\tmat-icon-button\r\n\t\t\t\tfxLayoutAlign=\"center center\"\r\n\t\t\t\t[disabled]=\"!isRecording\"\r\n\t\t\t\t(click)=\"stopRecording()\"\r\n\t\t\t\t[matTooltip]=\"'AUDIO.STOP' | translate\"\r\n\t\t\t\t[attr.aria-label]=\"'AUDIO.STOP' | translate\"\r\n\t\t\t>\r\n\t\t\t\t<span class=\"material-icons\"> stop </span>\r\n\t\t\t</button>\r\n\t\t</div>\r\n\t</div>\r\n</ng-container>\r\n"]}
131
+ //# 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, EventEmitter, Input, OnInit, Output } from '@angular/core';\r\nimport { AsrClient, AsrClientConfiguration, RecognitionStatus, SAMPLE_RATE_PLACEHOLDER, SpeechRecognitionResult } from '@tilde-nlp/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  recordLength: Date = new Date(0, 0, 0, 0, 0, 0);\r\n  isProcessing: boolean;\r\n  hasAnyMicrophone: boolean;\r\n  // asrApiClient: ASRApiClient;\r\n  asrClient: AsrClient;\r\n  private readonly notAllowedErrorName = \"NotAllowedError\";\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\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  /**\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 _systemId: string;\r\n  @Input() set systemId(value: string) {\r\n    this._systemId = value;\r\n    this.stopRecording();\r\n  }\r\n\r\n  constructor() {\r\n  }\r\n\r\n  ngOnInit() {\r\n    this.hasAnyAudioInput();\r\n  }\r\n\r\n  startRecording() {\r\n    this.createClient();\r\n    this.asrClient.beginVoiceRecognition();\r\n  }\r\n\r\n  stopRecording() {\r\n    this.asrClient?.endVoiceRecognition();\r\n    this.asrClient = null;\r\n  }\r\n\r\n  private hasAnyAudioInput() {\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;\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  private createClient() {\r\n    const config: AsrClientConfiguration = {\r\n      url: `${this.configuration.apiUrl}/${this._systemId}?contentType=audio/x-raw&sampleRate=${SAMPLE_RATE_PLACEHOLDER}&x-api-key=${this.configuration.apiKey}`,\r\n      replaceSampleRateInUrl: true,\r\n      onResult: (result) => { this.handleResult(result as SpeechRecognitionResult) },\r\n      onRecordingStartStop: (isRecording) => {\r\n        this.isRecording = isRecording;\r\n      },\r\n      onError: (error) => {\r\n        this.hasAnyMicrophone = false;\r\n        this.stopRecording();\r\n        if (error.name === this.notAllowedErrorName) {\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}\r\n","<ng-container *ngIf=\"!isRecording\">\r\n\t<button\r\n\t\t[matTooltip]=\"'AUDIO.START' | translate\"\r\n\t\t[disabled]=\"!hasAnyMicrophone\"\r\n\t\tmat-icon-button\r\n\t\tfxLayoutAlign=\"center center\"\r\n\t\t(click)=\"startRecording()\"\r\n\t\t[attr.aria-label]=\"(hasAnyMicrophone ? 'AUDIO.START' : 'AUDIO.MIC_NOT_FOUND') | translate\"\r\n\t>\r\n\t\t<span class=\"material-icons\" *ngIf=\"hasAnyMicrophone\" [matTooltip]=\"'AUDIO.START' | translate\"> mic </span>\r\n\t\t<span class=\"material-icons\" [matTooltip]=\"'AUDIO.MIC_NOT_FOUND' | translate\" *ngIf=\"!hasAnyMicrophone\"> mic_off </span>\r\n\t</button>\r\n</ng-container>\r\n<ng-container *ngIf=\"isRecording\">\r\n\t<div fxLayout=\"row\" fxLayoutAlign=\"start end\">\r\n\t\t<div fxLayout=\"column\">\r\n\t\t\t<button\r\n\t\t\t\tmat-icon-button\r\n\t\t\t\tfxLayoutAlign=\"center center\"\r\n\t\t\t\t[disabled]=\"!isRecording\"\r\n\t\t\t\t(click)=\"stopRecording()\"\r\n\t\t\t\t[matTooltip]=\"'AUDIO.STOP' | translate\"\r\n\t\t\t\t[attr.aria-label]=\"'AUDIO.STOP' | translate\"\r\n\t\t\t>\r\n\t\t\t\t<span class=\"material-icons\"> stop </span>\r\n\t\t\t</button>\r\n\t\t</div>\r\n\t</div>\r\n</ng-container>\r\n"]}
@@ -6350,6 +6350,7 @@ class TldDictateComponent {
6350
6350
  this.isRecording = isRecording;
6351
6351
  },
6352
6352
  onError: (error) => {
6353
+ this.hasAnyMicrophone = false;
6353
6354
  this.stopRecording();
6354
6355
  if (error.name === this.notAllowedErrorName) {
6355
6356
  this.emitError(AudioErrorCase.MIC_NOT_ALLOWED);