@tilde-nlp/ngx-translate 7.0.35 → 7.0.37
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/components/shared/text-to-speech/models/text-to-speech-settings.model.mjs +1 -1
- package/esm2022/lib/components/shared/text-to-speech/text-to-speech.component.mjs +14 -13
- package/esm2022/lib/modules/tld-audio/models/dictate-configurationi.model.mjs +1 -1
- package/esm2022/lib/modules/tld-audio/tld-dictate/tld-dictate.component.mjs +49 -45
- package/esm2022/lib/modules/tld-common/models/configs/tld-audio-config.model.mjs +1 -1
- package/esm2022/lib/modules/tld-common/models/configs/tld-speech-config.model.mjs +2 -0
- package/esm2022/lib/modules/tld-common/models/configs/tld-translate-config.model.mjs +1 -1
- package/esm2022/lib/modules/tld-common/models/index.mjs +2 -2
- package/esm2022/lib/modules/tld-common/services/tld-translate-config.service.mjs +22 -13
- package/esm2022/lib/modules/tld-common/services/tld-translate-system-management.service.mjs +28 -15
- package/esm2022/lib/modules/tld-text/components/tld-translate-text/tld-translate-text.component.mjs +13 -12
- package/esm2022/lib/modules/tld-text/services/tld-translate-text.service.mjs +18 -17
- package/esm2022/lib/modules/tld-tooltip/translation-system-picker/dropdown/translation-system-picker-dropdown.component.mjs +34 -6
- package/fesm2022/tilde-nlp-ngx-translate.mjs +168 -112
- package/fesm2022/tilde-nlp-ngx-translate.mjs.map +1 -1
- package/lib/components/shared/text-to-speech/models/text-to-speech-settings.model.d.ts +4 -13
- package/lib/components/shared/text-to-speech/text-to-speech.component.d.ts +1 -0
- package/lib/modules/tld-audio/models/dictate-configurationi.model.d.ts +2 -3
- package/lib/modules/tld-audio/tld-dictate/tld-dictate.component.d.ts +7 -5
- package/lib/modules/tld-common/models/configs/tld-audio-config.model.d.ts +0 -9
- package/lib/modules/tld-common/models/configs/tld-speech-config.model.d.ts +24 -0
- package/lib/modules/tld-common/models/configs/tld-translate-config.model.d.ts +2 -2
- package/lib/modules/tld-common/models/index.d.ts +1 -1
- package/lib/modules/tld-common/services/tld-translate-config.service.d.ts +3 -3
- package/lib/modules/tld-common/services/tld-translate-system-management.service.d.ts +9 -6
- package/lib/modules/tld-text/components/tld-translate-text/tld-translate-text.component.d.ts +1 -1
- package/lib/modules/tld-tooltip/translation-system-picker/dropdown/translation-system-picker-dropdown.component.d.ts +2 -0
- package/package.json +1 -1
- package/src/assets/webcomponent/tilde-translate-box.js +1 -1
- package/esm2022/lib/modules/tld-common/models/configs/tld-text-to-speech-config.model.mjs +0 -2
- package/lib/modules/tld-common/models/configs/tld-text-to-speech-config.model.d.ts +0 -11
package/esm2022/lib/components/shared/text-to-speech/models/text-to-speech-settings.model.mjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dC10by1zcGVlY2gtc2V0dGluZ3MubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90bGQtdHJhbnNsYXRlL3NyYy9saWIvY29tcG9uZW50cy9zaGFyZWQvdGV4dC10by1zcGVlY2gvbW9kZWxzL3RleHQtdG8tc3BlZWNoLXNldHRpbmdzLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIFRleHRUb1NwZWVjaFNldHRpbmdzIHtcclxuICAvKipcclxuICAgKiBUZXh0IHRvIGJlIHNwb2tlbiBvdXRcclxuICAgKi9cclxuICB0ZXh0OiBzdHJpbmc7XHJcbiAgLyoqXHJcbiAgICogVFRTIHN5dGVtIGlkXHJcbiAgICovXHJcbiAgc3lzdGVtSWQ6IHN0cmluZztcclxuICAgIC8qKlxyXG4gICAqIE1heGltdW0gYWxsb3dlZCBjaGFyYWN0ZXJzXHJcbiAgICovXHJcbiAgICBtYXhDaGFyc0luVGV4dD86IG51bWJlcjtcclxufVxyXG4iXX0=
|
|
@@ -1,9 +1,11 @@
|
|
|
1
|
-
import { Component, Input } from '@angular/core';
|
|
1
|
+
import { Component, Input, inject } from '@angular/core';
|
|
2
|
+
import { TextToSpeechService } from '@tilde-nlp/ngx-services';
|
|
2
3
|
import * as i0 from "@angular/core";
|
|
3
4
|
import * as i1 from "@angular/material/button";
|
|
4
5
|
import * as i2 from "@angular/material/tooltip";
|
|
5
6
|
import * as i3 from "@ngx-translate/core";
|
|
6
7
|
export class TextToSpeechComponent {
|
|
8
|
+
#ttsService = inject(TextToSpeechService);
|
|
7
9
|
get isPaused() {
|
|
8
10
|
return this.audio?.paused;
|
|
9
11
|
}
|
|
@@ -29,19 +31,18 @@ export class TextToSpeechComponent {
|
|
|
29
31
|
}
|
|
30
32
|
play() {
|
|
31
33
|
if (!this.audio.src) {
|
|
32
|
-
this.audio = new Audio();
|
|
33
34
|
const trimmedText = (this.settings?.maxCharsInText && this.settings.maxCharsInText < this.settings.text.length) ? this.settings.text.slice(0, this.settings.maxCharsInText) : this.settings.text;
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
|
|
35
|
+
this.#ttsService.generateSpeech({ text: trimmedText, system: this.settings.systemId }).subscribe((blob) => {
|
|
36
|
+
this.audio = new Audio();
|
|
37
|
+
const url = URL.createObjectURL(blob);
|
|
38
|
+
this.audio.src = url;
|
|
39
|
+
this.audio.load();
|
|
40
|
+
this.audio.play();
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
else {
|
|
44
|
+
this.audio.play();
|
|
43
45
|
}
|
|
44
|
-
this.audio.play();
|
|
45
46
|
}
|
|
46
47
|
ngOnDestroy() {
|
|
47
48
|
this.disposeAudio();
|
|
@@ -61,4 +62,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
61
62
|
}], ctorParameters: () => [], propDecorators: { settings: [{
|
|
62
63
|
type: Input
|
|
63
64
|
}] } });
|
|
64
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
65
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dC10by1zcGVlY2guY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdGxkLXRyYW5zbGF0ZS9zcmMvbGliL2NvbXBvbmVudHMvc2hhcmVkL3RleHQtdG8tc3BlZWNoL3RleHQtdG8tc3BlZWNoLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RsZC10cmFuc2xhdGUvc3JjL2xpYi9jb21wb25lbnRzL3NoYXJlZC90ZXh0LXRvLXNwZWVjaC90ZXh0LXRvLXNwZWVjaC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBYSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFcEUsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0seUJBQXlCLENBQUM7Ozs7O0FBTzlELE1BQU0sT0FBTyxxQkFBcUI7SUFDdkIsV0FBVyxHQUFHLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBQ25ELElBQUksUUFBUTtRQUNWLE9BQU8sSUFBSSxDQUFDLEtBQUssRUFBRSxNQUFNLENBQUM7SUFDNUIsQ0FBQztJQUNPLEtBQUssQ0FBbUI7SUFFeEIsU0FBUyxDQUF1QjtJQUN4QyxJQUFJLFFBQVEsS0FBSyxPQUFPLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDO0lBQ3pDLElBQWEsUUFBUSxDQUFDLEtBQUs7UUFDekIsSUFBSSxDQUFDLFNBQVMsR0FBRyxLQUFLLENBQUM7UUFDdkIsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO1FBQ3BCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxLQUFLLEVBQUUsQ0FBQztJQUMzQixDQUFDO0lBRUQsZ0JBQWdCLENBQUM7SUFFakIsT0FBTztRQUNMLElBQUksSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1lBQ2xCLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNkLENBQUM7YUFDSSxDQUFDO1lBQ0osSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQ2YsQ0FBQztJQUNILENBQUM7SUFFTyxLQUFLO1FBQ1gsSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUNyQixDQUFDO0lBRU8sSUFBSTtRQUNWLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsRUFBRSxDQUFDO1lBQ3BCLE1BQU0sV0FBVyxHQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxjQUFjLElBQUksSUFBSSxDQUFDLFFBQVEsQ0FBQyxjQUFjLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLEVBQUUsSUFBSSxDQUFDLFFBQVEsQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUM7WUFDak0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxjQUFjLENBQUMsRUFBRSxJQUFJLEVBQUUsV0FBVyxFQUFFLE1BQU0sRUFBRSxJQUFJLENBQUMsUUFBUSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsSUFBSSxFQUFFLEVBQUU7Z0JBQ3hHLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxLQUFLLEVBQUUsQ0FBQztnQkFDekIsTUFBTSxHQUFHLEdBQUcsR0FBRyxDQUFDLGVBQWUsQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDdEMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDO2dCQUNyQixJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDO2dCQUNsQixJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQ3BCLENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQzthQUNJLENBQUM7WUFDSixJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ3BCLENBQUM7SUFDSCxDQUFDO0lBRUQsV0FBVztRQUNULElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBRU8sWUFBWTtRQUNsQixJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNmLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxHQUFHLElBQUksQ0FBQztZQUN0QixJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQztRQUNwQixDQUFDO0lBQ0gsQ0FBQzt3R0F2RFUscUJBQXFCOzRGQUFyQixxQkFBcUIsNEZDVGxDLDJQQU1BOzs0RkRHYSxxQkFBcUI7a0JBTGpDLFNBQVM7K0JBQ0Usb0JBQW9CO3dEQWFqQixRQUFRO3NCQUFwQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT25EZXN0cm95LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgVGV4dFRvU3BlZWNoU2V0dGluZ3MgfSBmcm9tICcuL21vZGVscy90ZXh0LXRvLXNwZWVjaC1zZXR0aW5ncy5tb2RlbCc7XHJcbmltcG9ydCB7IFRleHRUb1NwZWVjaFNlcnZpY2UgfSBmcm9tICdAdGlsZGUtbmxwL25neC1zZXJ2aWNlcyc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ3RsZC10ZXh0LXRvLXNwZWVjaCcsXHJcbiAgdGVtcGxhdGVVcmw6ICcuL3RleHQtdG8tc3BlZWNoLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybHM6IFsnLi90ZXh0LXRvLXNwZWVjaC5jb21wb25lbnQuc2NzcyddXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBUZXh0VG9TcGVlY2hDb21wb25lbnQgaW1wbGVtZW50cyBPbkRlc3Ryb3kge1xyXG4gIHJlYWRvbmx5ICN0dHNTZXJ2aWNlID0gaW5qZWN0KFRleHRUb1NwZWVjaFNlcnZpY2UpO1xyXG4gIGdldCBpc1BhdXNlZCgpIHtcclxuICAgIHJldHVybiB0aGlzLmF1ZGlvPy5wYXVzZWQ7XHJcbiAgfVxyXG4gIHByaXZhdGUgYXVkaW86IEhUTUxBdWRpb0VsZW1lbnQ7XHJcblxyXG4gIHByaXZhdGUgX3NldHRpbmdzOiBUZXh0VG9TcGVlY2hTZXR0aW5ncztcclxuICBnZXQgc2V0dGluZ3MoKSB7IHJldHVybiB0aGlzLl9zZXR0aW5nczsgfVxyXG4gIEBJbnB1dCgpIHNldCBzZXR0aW5ncyh2YWx1ZSkge1xyXG4gICAgdGhpcy5fc2V0dGluZ3MgPSB2YWx1ZTtcclxuICAgIHRoaXMuZGlzcG9zZUF1ZGlvKCk7XHJcbiAgICB0aGlzLmF1ZGlvID0gbmV3IEF1ZGlvKCk7XHJcbiAgfVxyXG5cclxuICBjb25zdHJ1Y3RvcigpIHsgfVxyXG5cclxuICBjbGlja2VkKCkge1xyXG4gICAgaWYgKHRoaXMuaXNQYXVzZWQpIHtcclxuICAgICAgdGhpcy5wbGF5KCk7XHJcbiAgICB9XHJcbiAgICBlbHNlIHtcclxuICAgICAgdGhpcy5wYXVzZSgpO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBwYXVzZSgpIHtcclxuICAgIHRoaXMuYXVkaW8ucGF1c2UoKTtcclxuICB9XHJcblxyXG4gIHByaXZhdGUgcGxheSgpIHtcclxuICAgIGlmICghdGhpcy5hdWRpby5zcmMpIHtcclxuICAgICAgY29uc3QgdHJpbW1lZFRleHQgPSAodGhpcy5zZXR0aW5ncz8ubWF4Q2hhcnNJblRleHQgJiYgdGhpcy5zZXR0aW5ncy5tYXhDaGFyc0luVGV4dCA8IHRoaXMuc2V0dGluZ3MudGV4dC5sZW5ndGgpID8gdGhpcy5zZXR0aW5ncy50ZXh0LnNsaWNlKDAsIHRoaXMuc2V0dGluZ3MubWF4Q2hhcnNJblRleHQpIDogdGhpcy5zZXR0aW5ncy50ZXh0O1xyXG4gICAgICB0aGlzLiN0dHNTZXJ2aWNlLmdlbmVyYXRlU3BlZWNoKHsgdGV4dDogdHJpbW1lZFRleHQsIHN5c3RlbTogdGhpcy5zZXR0aW5ncy5zeXN0ZW1JZCB9KS5zdWJzY3JpYmUoKGJsb2IpID0+IHtcclxuICAgICAgICB0aGlzLmF1ZGlvID0gbmV3IEF1ZGlvKCk7XHJcbiAgICAgICAgY29uc3QgdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChibG9iKTtcclxuICAgICAgICB0aGlzLmF1ZGlvLnNyYyA9IHVybDtcclxuICAgICAgICB0aGlzLmF1ZGlvLmxvYWQoKTtcclxuICAgICAgICB0aGlzLmF1ZGlvLnBsYXkoKTtcclxuICAgICAgfSk7XHJcbiAgICB9XHJcbiAgICBlbHNlIHtcclxuICAgICAgdGhpcy5hdWRpby5wbGF5KCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcclxuICAgIHRoaXMuZGlzcG9zZUF1ZGlvKCk7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGRpc3Bvc2VBdWRpbygpIHtcclxuICAgIGlmICh0aGlzLmF1ZGlvKSB7XHJcbiAgICAgIHRoaXMuYXVkaW8uc3JjID0gbnVsbDtcclxuICAgICAgdGhpcy5hdWRpbyA9IG51bGw7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxufVxyXG4iLCI8YnV0dG9uIG1hdC1pY29uLWJ1dHRvbiAoY2xpY2spPVwiY2xpY2tlZCgpXCJcclxuICBbbWF0VG9vbHRpcF09XCIoaXNQYXVzZWQ/ICdURVhUX1RPX1NQRUVDSC5QTEFZJzonVEVYVF9UT19TUEVFQ0guUEFVU0UnKXx0cmFuc2xhdGVcIj5cclxuICA8c3BhbiBjbGFzcz1cIm1hdGVyaWFsLWljb25zXCI+XHJcbiAgICB7e2lzUGF1c2VkPyBcInZvbHVtZV91cFwiOiBcInBhdXNlXCIgfX1cclxuICA8L3NwYW4+XHJcbjwvYnV0dG9uPlxyXG4iXX0=
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGljdGF0ZS1jb25maWd1cmF0aW9uaS5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RsZC10cmFuc2xhdGUvc3JjL2xpYi9tb2R1bGVzL3RsZC1hdWRpby9tb2RlbHMvZGljdGF0ZS1jb25maWd1cmF0aW9uaS5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBJRGljdGF0ZUNvbmZpZ3VyYXRpb24ge1xyXG4gICAgbWF4TGVuZ3RoOiBudW1iZXIsXHJcbiAgICBhcGlVcmw6IHN0cmluZztcclxuICAgIGFwaUtleTogc3RyaW5nO1xyXG59XHJcbiJdfQ==
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
|
|
2
|
-
import { AsrClient, RecognitionStatus } from '@tilde-nlp/asr-client';
|
|
2
|
+
import { AsrClient, RecognitionStatus, SAMPLE_RATE_PLACEHOLDER } from '@tilde-nlp/asr-api-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";
|
|
@@ -18,6 +18,7 @@ export class TldDictateComponent {
|
|
|
18
18
|
hasAnyMicrophone;
|
|
19
19
|
// asrApiClient: ASRApiClient;
|
|
20
20
|
asrClient;
|
|
21
|
+
notAllowedErrorName = "NotAllowedError";
|
|
21
22
|
get maxLength() { return this.configuration?.maxLength; }
|
|
22
23
|
;
|
|
23
24
|
_isRecording;
|
|
@@ -52,50 +53,18 @@ export class TldDictateComponent {
|
|
|
52
53
|
isDictating = new EventEmitter();
|
|
53
54
|
onError = new EventEmitter();
|
|
54
55
|
configuration;
|
|
55
|
-
|
|
56
|
-
set
|
|
57
|
-
this.
|
|
58
|
-
this.
|
|
56
|
+
_systemId;
|
|
57
|
+
set systemId(value) {
|
|
58
|
+
this._systemId = value;
|
|
59
|
+
this.stopRecording();
|
|
59
60
|
}
|
|
60
61
|
constructor() {
|
|
61
62
|
}
|
|
62
63
|
ngOnInit() {
|
|
63
|
-
|
|
64
|
-
appId: this.configuration.appId,
|
|
65
|
-
appSecret: this.configuration.appSecret,
|
|
66
|
-
language: this._language,
|
|
67
|
-
apiUrl: this.configuration.apiUrl,
|
|
68
|
-
visualizerConfig: {
|
|
69
|
-
visualizerId: this.visualizataionId
|
|
70
|
-
},
|
|
71
|
-
onResult: (result) => { this.handleResult(result); },
|
|
72
|
-
onRecordingStartStop: (isRecording) => {
|
|
73
|
-
this.isRecording = isRecording;
|
|
74
|
-
setTimeout(() => {
|
|
75
|
-
this.setCanvasWidth();
|
|
76
|
-
if (this.asrClient.audioVisualizer?.strokeStyle) {
|
|
77
|
-
this.asrClient.audioVisualizer.strokeStyle = this.audioVisualizationColor;
|
|
78
|
-
}
|
|
79
|
-
this.asrClient.audioVisualizer?.visualizeAudio();
|
|
80
|
-
}, 1);
|
|
81
|
-
},
|
|
82
|
-
onError: (error) => {
|
|
83
|
-
if (error.name === "NotAllowedError") {
|
|
84
|
-
this.emitError(AudioErrorCase.MIC_NOT_ALLOWED);
|
|
85
|
-
return;
|
|
86
|
-
}
|
|
87
|
-
throw error;
|
|
88
|
-
},
|
|
89
|
-
onMessage: (message) => {
|
|
90
|
-
if (message === RecognitionStatus.NoSpeech) {
|
|
91
|
-
this.emitError(AudioErrorCase.NO_SPEECH);
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
};
|
|
95
|
-
this.asrClient = new AsrClient(config);
|
|
96
|
-
this.findMic();
|
|
64
|
+
this.hasAnyAudioInput();
|
|
97
65
|
}
|
|
98
66
|
startRecording() {
|
|
67
|
+
this.createClient();
|
|
99
68
|
this.asrClient.beginVoiceRecognition();
|
|
100
69
|
this.interval = setInterval(() => {
|
|
101
70
|
const recordTimeInSeconds = this.asrClient?.audioContext?.currentTime ?? 0;
|
|
@@ -108,12 +77,13 @@ export class TldDictateComponent {
|
|
|
108
77
|
}, 1000);
|
|
109
78
|
}
|
|
110
79
|
stopRecording() {
|
|
111
|
-
this.asrClient
|
|
80
|
+
this.asrClient?.endVoiceRecognition();
|
|
81
|
+
this.asrClient = null;
|
|
112
82
|
}
|
|
113
83
|
setCanvasWidth() {
|
|
114
84
|
this.canvasWidth = this.canvasWrapper?.nativeElement.offsetWidth;
|
|
115
85
|
}
|
|
116
|
-
|
|
86
|
+
hasAnyAudioInput() {
|
|
117
87
|
navigator.mediaDevices.enumerateDevices().then((devices) => {
|
|
118
88
|
this.hasAnyMicrophone = false;
|
|
119
89
|
for (const device of devices) {
|
|
@@ -138,12 +108,46 @@ export class TldDictateComponent {
|
|
|
138
108
|
emitError(error) {
|
|
139
109
|
this.onError.emit(error);
|
|
140
110
|
}
|
|
111
|
+
createClient() {
|
|
112
|
+
const config = {
|
|
113
|
+
url: `${this.configuration.apiUrl}/${this._systemId}?contentType=audio/x-raw&sampleRate=${SAMPLE_RATE_PLACEHOLDER}&x-api-key=${this.configuration.apiKey}`,
|
|
114
|
+
replaceSampleRateInUrl: true,
|
|
115
|
+
visualizerConfig: {
|
|
116
|
+
visualizerId: this.visualizataionId
|
|
117
|
+
},
|
|
118
|
+
onResult: (result) => { this.handleResult(result); },
|
|
119
|
+
onRecordingStartStop: (isRecording) => {
|
|
120
|
+
this.isRecording = isRecording;
|
|
121
|
+
setTimeout(() => {
|
|
122
|
+
this.setCanvasWidth();
|
|
123
|
+
if (this.asrClient.audioVisualizer?.strokeStyle) {
|
|
124
|
+
this.asrClient.audioVisualizer.strokeStyle = this.audioVisualizationColor;
|
|
125
|
+
}
|
|
126
|
+
this.asrClient.audioVisualizer?.visualizeAudio();
|
|
127
|
+
}, 1);
|
|
128
|
+
},
|
|
129
|
+
onError: (error) => {
|
|
130
|
+
this.stopRecording();
|
|
131
|
+
if (error.name === this.notAllowedErrorName) {
|
|
132
|
+
this.emitError(AudioErrorCase.MIC_NOT_ALLOWED);
|
|
133
|
+
return;
|
|
134
|
+
}
|
|
135
|
+
throw error;
|
|
136
|
+
},
|
|
137
|
+
onMessage: (message) => {
|
|
138
|
+
if (message === RecognitionStatus.NoSpeech) {
|
|
139
|
+
this.emitError(AudioErrorCase.NO_SPEECH);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
};
|
|
143
|
+
this.asrClient = new AsrClient(config);
|
|
144
|
+
}
|
|
141
145
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TldDictateComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
142
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TldDictateComponent, selector: "tld-dictate", inputs: { configuration: "configuration",
|
|
146
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TldDictateComponent, selector: "tld-dictate", inputs: { configuration: "configuration", systemId: "systemId" }, outputs: { result: "result", partialResult: "partialResult", isDictating: "isDictating", onError: "onError" }, host: { listeners: { "window:resize": "onResize()" } }, viewQueries: [{ propertyName: "canvasWrapper", first: true, predicate: ["canvasWrapper"], descendants: true }], ngImport: i0, template: "<ng-container *ngIf=\"!isRecording\">\r\n <button [disabled]=\"!hasAnyMicrophone\" 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-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>\r\n", styles: ["button{border:1px solid rgba(0,0,0,.12);min-width:40px}#dictate-visualization{margin-left:1em}.tld-audio-timer{font-size:10px}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "directive", type: i3.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { kind: "directive", type: i3.DefaultLayoutAlignDirective, selector: " [fxLayoutAlign], [fxLayoutAlign.xs], [fxLayoutAlign.sm], [fxLayoutAlign.md], [fxLayoutAlign.lg], [fxLayoutAlign.xl], [fxLayoutAlign.lt-sm], [fxLayoutAlign.lt-md], [fxLayoutAlign.lt-lg], [fxLayoutAlign.lt-xl], [fxLayoutAlign.gt-xs], [fxLayoutAlign.gt-sm], [fxLayoutAlign.gt-md], [fxLayoutAlign.gt-lg]", inputs: ["fxLayoutAlign", "fxLayoutAlign.xs", "fxLayoutAlign.sm", "fxLayoutAlign.md", "fxLayoutAlign.lg", "fxLayoutAlign.xl", "fxLayoutAlign.lt-sm", "fxLayoutAlign.lt-md", "fxLayoutAlign.lt-lg", "fxLayoutAlign.lt-xl", "fxLayoutAlign.gt-xs", "fxLayoutAlign.gt-sm", "fxLayoutAlign.gt-md", "fxLayoutAlign.gt-lg"] }, { kind: "directive", type: i3.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { kind: "component", type: i4.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "pipe", type: i1.DatePipe, name: "date" }, { kind: "pipe", type: i5.TranslatePipe, name: "translate" }] });
|
|
143
147
|
}
|
|
144
148
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TldDictateComponent, decorators: [{
|
|
145
149
|
type: Component,
|
|
146
|
-
args: [{ selector: 'tld-dictate', template: "<ng-container *ngIf=\"!isRecording\">\r\n <button [disabled]=\"!hasAnyMicrophone\" 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-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
|
|
150
|
+
args: [{ selector: 'tld-dictate', template: "<ng-container *ngIf=\"!isRecording\">\r\n <button [disabled]=\"!hasAnyMicrophone\" 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-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>\r\n", styles: ["button{border:1px solid rgba(0,0,0,.12);min-width:40px}#dictate-visualization{margin-left:1em}.tld-audio-timer{font-size:10px}\n"] }]
|
|
147
151
|
}], ctorParameters: () => [], propDecorators: { onResize: [{
|
|
148
152
|
type: HostListener,
|
|
149
153
|
args: ['window:resize', []]
|
|
@@ -160,7 +164,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
160
164
|
type: Output
|
|
161
165
|
}], configuration: [{
|
|
162
166
|
type: Input
|
|
163
|
-
}],
|
|
167
|
+
}], systemId: [{
|
|
164
168
|
type: Input
|
|
165
169
|
}] } });
|
|
166
|
-
//# 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  private readonly audioVisualizationColor = '#0A70C2';\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\r\n          if (this.asrClient.audioVisualizer?.strokeStyle) {\r\n            this.asrClient.audioVisualizer.strokeStyle = this.audioVisualizationColor;\r\n          }\r\n\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-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-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>"]}
|
|
170
|
+
//# 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, 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\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  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    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  private readonly audioVisualizationColor = '#0A70C2';\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 _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    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    this.asrClient = null;\r\n  }\r\n\r\n  private setCanvasWidth() {\r\n    this.canvasWidth = this.canvasWrapper?.nativeElement.offsetWidth;\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 + (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  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      visualizerConfig: {\r\n        visualizerId: this.visualizataionId\r\n      },\r\n      onResult: (result) => { this.handleResult(result as SpeechRecognitionResult) },\r\n      onRecordingStartStop: (isRecording) => {\r\n        this.isRecording = isRecording;\r\n        setTimeout(() => {\r\n          this.setCanvasWidth();\r\n\r\n          if (this.asrClient.audioVisualizer?.strokeStyle) {\r\n            this.asrClient.audioVisualizer.strokeStyle = this.audioVisualizationColor;\r\n          }\r\n\r\n          this.asrClient.audioVisualizer?.visualizeAudio();\r\n        }, 1);\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    <button [disabled]=\"!hasAnyMicrophone\" 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-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>\r\n"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGxkLWF1ZGlvLWNvbmZpZy5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RsZC10cmFuc2xhdGUvc3JjL2xpYi9tb2R1bGVzL3RsZC1jb21tb24vbW9kZWxzL2NvbmZpZ3MvdGxkLWF1ZGlvLWNvbmZpZy5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBJVGxkQXVkaW9Db25maWcge1xyXG4gICAgLyoqIFVybCBmb3IgYXBpIHRoYXQgcmV0dXJucyB0ZXh0IGZyb20gYXVkaW8gaW5wdXQuIFRoaXMgd2FzIHVzZWQgZm9yIEVFTVQgYXBpLiAqL1xyXG4gICAgYXVkaW9BcGlVcmw/OiBzdHJpbmc7XHJcbiAgICAvKiogTWF4aW11bSBzaXplIGZvciBhdWRpbyBmaWxlICovXHJcbiAgICBhdWRpb01heFNpemVCeXRlcz86IG51bWJlcjtcclxuICAgIC8qKiBBcnJheSBvZiBsYW5ndWFnZXMgZm9yIHdoaWNoIGF1ZGlvIGlucHV0IGlzIHN1cHBvcnRlZC4gKi9cclxuICAgIHNvdXJjZUxhbmd1YWdlcz86IHN0cmluZ1tdO1xyXG4gICAgLyoqIEhvdyBvZnRlbiBzaG91bGQgZmlsZSBzaXplIGJlIGNoZWNrZWQuIFZhbHVlIG11c3QgYmUgaW4gbWlsbGlzZWNvbmRzLiAqL1xyXG4gICAgZGF0YVVwZGF0ZVRpbWVGcmFtZT86IG51bWJlclxyXG4gICAgZGlzYWJsZUF1dG9tYXRpY1RyYW5zbGF0aW9uPzogYm9vbGVhbjtcclxuICAgIC8qKiBTaG93IG1lc3NhZ2UgdG8gY2hlY2sgcHVuY3R1YXRpb24gYWZ0ZXIgc3BlZWNoIGlucHV0IGlzIHN0b3BwZWQuICovXHJcbiAgICBwdW5jdHVhdGlvbk1lc3NhZ2U/OiBib29sZWFuO1xyXG59XHJcbiJdfQ==
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGxkLXNwZWVjaC1jb25maWcubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90bGQtdHJhbnNsYXRlL3NyYy9saWIvbW9kdWxlcy90bGQtY29tbW9uL21vZGVscy9jb25maWdzL3RsZC1zcGVlY2gtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIElUbGRTcGVlY2hDb25maWcge1xyXG4gIHRleHRUb1NwZWVjaD86IHtcclxuICAgIGVuYWJsZWQ/OiBib29sZWFuO1xyXG4gICAgbWF4Q2hhcnNJblRleHQ/OiBudW1iZXI7XHJcbiAgICB1cmw6IHN0cmluZztcclxuICB9LFxyXG4gIG9ubGluZUFzcj86IHtcclxuICAgIGVuYWJsZWQ/OiBib29sZWFuO1xyXG4gICAgdXJsOiBzdHJpbmc7XHJcbiAgICAvKiogTWF4aW11bSBhbGxvd2VkIHRpbWUgZm9yIHJlY29yZGluZywgaW4gc2Vjb25kcy4gKi9cclxuICAgIG1heGltdW1SZWNvcmRUaW1lPzogbnVtYmVyO1xyXG4gICAgLyoqIEFmdGVyIHJlY29yZGluZywgc2hvdyBtZXNzYWdlIHRvIGNoZWNrIHB1bmN0dWF0aW9uICovXHJcbiAgICBwdW5jdHVhdGlvbk1lc3NhZ2U/OiBib29sZWFuO1xyXG4gIH1cclxuICAvKipcclxuICAgKiBVUkwgZm9yIHRoZSByZXNvdXJjZSBzZXJ2aWNlLiBUaGlzIGlzIHVzZWQgdG8gZ2V0IHRoZSBsaXN0IG9mIGF2YWlsYWJsZSByZXNvdXJjZXMgZm9yIFRUUyBhbmQgQVNSLlxyXG4gICAqIFRoZSBVUkwgc2hvdWxkIHBvaW50IHRvIHRoZSBzcGVlY2ggcmVzb3VyY2Ugc2VydmljZSBlbmRwb2ludC5cclxuICAgKi9cclxuICByZXNvdXJjZVNlcnZpY2VVcmw/OiBzdHJpbmc7XHJcbiAgLyoqXHJcbiAgICogQXBpIGtleSBnZW5lcmF0ZWQgZm9yIHNwZWVjaCBzZXJ2aWNlIGNhbGxzLiBOb3RlLCB0aGF0IHRoaXMgaXMgZGlmZmVyZW50IHRoYW4gZGVmYXVsdCByZXF1ZXN0cyBhcGkga2V5LlxyXG4gICAqL1xyXG4gIGFwaUtleTogc3RyaW5nO1xyXG59XHJcbiJdfQ==
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGxkLXRyYW5zbGF0ZS1jb25maWcubW9kZWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90bGQtdHJhbnNsYXRlL3NyYy9saWIvbW9kdWxlcy90bGQtY29tbW9uL21vZGVscy9jb25maWdzL3RsZC10cmFuc2xhdGUtY29uZmlnLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJVGxkQXVkaW9Db25maWcgfSBmcm9tIFwiLi90bGQtYXVkaW8tY29uZmlnLm1vZGVsXCI7XHJcbmltcG9ydCB7IElUbGRUZXJtQ29uZmlnIH0gZnJvbSBcIi4vdGxkLXRlcm0tY29uZmlnLm1vZGVsXCI7XHJcbmltcG9ydCB7IElUbGRUcmFuc2xhdGVGaWxlQ29uZmlnIH0gZnJvbSBcIi4vdGxkLXRyYW5zbGF0ZS1maWxlLWNvbmZpZy5tb2RlbFwiO1xyXG5pbXBvcnQgeyBJVGxkVHJhbnNsYXRlQ29yZUNvbmZpZyB9IGZyb20gXCIuL3RsZC10cmFuc2xhdGUtY29yZS1jb25maWcubW9kZWxcIjtcclxuaW1wb3J0IHsgSVRsZFRyYW5zbGF0ZVRleHRDb25maWcgfSBmcm9tIFwiLi90bGQtdHJhbnNsYXRlLXRleHQtY29uZmlnLm1vZGVsXCI7XHJcbmltcG9ydCB7IElUbGRUcmFuc2xhdGVXZWJzaXRlQ29uZmlnIH0gZnJvbSBcIi4vdGxkLXRyYW5zbGF0ZS13ZWItY29uZmlnLm1vZGVsXCI7XHJcbmltcG9ydCB7IElUbGRRdW90YUNvbmZpZyB9IGZyb20gXCIuL3RsZC1xdW90YS5tb2RlbFwiO1xyXG5pbXBvcnQgeyBJVGxkU3Vic2NyaXB0aW9uQ29uZmlnIH0gZnJvbSBcIi4vdGxkLXN1YnNjcmlwdGlvbi1jb25maWcubW9kZWxcIjtcclxuaW1wb3J0IHsgSVRsZExMTUNvbmZpZyB9IGZyb20gXCIuL3RsZC1sbG0tY29uZmlnLm1vZGVsXCI7XHJcbmltcG9ydCB7IElUbGRTcGVlY2hDb25maWcgfSBmcm9tIFwiLi90bGQtc3BlZWNoLWNvbmZpZy5tb2RlbFwiO1xyXG5cclxuZXhwb3J0IGludGVyZmFjZSBJVGxkVHJhbnNsYXRlQ29uZmlne1xyXG4gICAgYXVkaW8/OiBJVGxkQXVkaW9Db25maWc7XHJcbiAgICBjb3JlPzogSVRsZFRyYW5zbGF0ZUNvcmVDb25maWc7XHJcbiAgICB0ZXJtPzogSVRsZFRlcm1Db25maWc7XHJcbiAgICBmaWxlPzogSVRsZFRyYW5zbGF0ZUZpbGVDb25maWc7XHJcbiAgICB0ZXh0PzogSVRsZFRyYW5zbGF0ZVRleHRDb25maWc7XHJcbiAgICBsbG0/OiBJVGxkTExNQ29uZmlnO1xyXG4gICAgcXVvdGE/OiBJVGxkUXVvdGFDb25maWc7XHJcbiAgICBzdWJzY3JpcHRpb24/OiBJVGxkU3Vic2NyaXB0aW9uQ29uZmlnO1xyXG4gICAgd2ViVHJhbnNsYXRlPzogSVRsZFRyYW5zbGF0ZVdlYnNpdGVDb25maWc7XHJcbiAgICBzcGVlY2g/OiBJVGxkU3BlZWNoQ29uZmlnO1xyXG59XHJcbiJdfQ==
|
|
@@ -7,7 +7,6 @@ export * from './configs/tld-translate-web-config.model';
|
|
|
7
7
|
export * from './configs/tld-term-config.model';
|
|
8
8
|
export * from './configs/tld-translate-config.model';
|
|
9
9
|
export * from './configs/tld-audio-config.model';
|
|
10
|
-
export * from './configs/tld-text-to-speech-config.model';
|
|
11
10
|
export * from './auth-options.model';
|
|
12
11
|
export * from './lang-list-item.model';
|
|
13
12
|
export * from './selected-language.model';
|
|
@@ -15,4 +14,5 @@ export * from './core';
|
|
|
15
14
|
export * from './services';
|
|
16
15
|
export * from './errors/error-codes.type';
|
|
17
16
|
export * from './errors/error.model';
|
|
18
|
-
|
|
17
|
+
export * from './configs/tld-speech-config.model';
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90bGQtdHJhbnNsYXRlL3NyYy9saWIvbW9kdWxlcy90bGQtY29tbW9uL21vZGVscy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYywyQ0FBMkMsQ0FBQztBQUMxRCxjQUFjLDJDQUEyQyxDQUFDO0FBQzFELGNBQWMsMkNBQTJDLENBQUM7QUFDMUQsY0FBYywwQ0FBMEMsQ0FBQztBQUN6RCxjQUFjLGlDQUFpQyxDQUFDO0FBQ2hELGNBQWMsc0NBQXNDLENBQUM7QUFDckQsY0FBYyxrQ0FBa0MsQ0FBQztBQUNqRCxjQUFjLHNCQUFzQixDQUFDO0FBQ3JDLGNBQWMsd0JBQXdCLENBQUM7QUFDdkMsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLFFBQVEsQ0FBQztBQUN2QixjQUFjLFlBQVksQ0FBQztBQUMzQixjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsc0JBQXNCLENBQUM7QUFDckMsY0FBYyxtQ0FBbUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vbWVzc2FnZS5tb2RlbCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGFuZ3VhZ2Utd2l0aC10cmFuc2xhdGlvbi5tb2RlbCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vY29uZmlncy90bGQtdHJhbnNsYXRlLWNvcmUtY29uZmlnLm1vZGVsJztcclxuZXhwb3J0ICogZnJvbSAnLi9jb25maWdzL3RsZC10cmFuc2xhdGUtdGV4dC1jb25maWcubW9kZWwnO1xyXG5leHBvcnQgKiBmcm9tICcuL2NvbmZpZ3MvdGxkLXRyYW5zbGF0ZS1maWxlLWNvbmZpZy5tb2RlbCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vY29uZmlncy90bGQtdHJhbnNsYXRlLXdlYi1jb25maWcubW9kZWwnO1xyXG5leHBvcnQgKiBmcm9tICcuL2NvbmZpZ3MvdGxkLXRlcm0tY29uZmlnLm1vZGVsJztcclxuZXhwb3J0ICogZnJvbSAnLi9jb25maWdzL3RsZC10cmFuc2xhdGUtY29uZmlnLm1vZGVsJztcclxuZXhwb3J0ICogZnJvbSAnLi9jb25maWdzL3RsZC1hdWRpby1jb25maWcubW9kZWwnO1xyXG5leHBvcnQgKiBmcm9tICcuL2F1dGgtb3B0aW9ucy5tb2RlbCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGFuZy1saXN0LWl0ZW0ubW9kZWwnO1xyXG5leHBvcnQgKiBmcm9tICcuL3NlbGVjdGVkLWxhbmd1YWdlLm1vZGVsJztcclxuZXhwb3J0ICogZnJvbSAnLi9jb3JlJztcclxuZXhwb3J0ICogZnJvbSAnLi9zZXJ2aWNlcyc7XHJcbmV4cG9ydCAqIGZyb20gJy4vZXJyb3JzL2Vycm9yLWNvZGVzLnR5cGUnO1xyXG5leHBvcnQgKiBmcm9tICcuL2Vycm9ycy9lcnJvci5tb2RlbCc7XHJcbmV4cG9ydCAqIGZyb20gJy4vY29uZmlncy90bGQtc3BlZWNoLWNvbmZpZy5tb2RlbCc7XHJcbiJdfQ==
|