@testgorilla/tgo-ai-interview-test 0.0.1 → 1.0.0

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.
Files changed (102) hide show
  1. package/README.md +32 -0
  2. package/esm2022/index.mjs +4 -0
  3. package/esm2022/lib/components/ai-interview-test/ai-interview-test.component.mjs +180 -0
  4. package/esm2022/lib/components/index.mjs +4 -0
  5. package/esm2022/lib/components/interview-stream/interview-stream.component.mjs +240 -0
  6. package/esm2022/lib/components/interview-video/interview-video.component.mjs +61 -0
  7. package/esm2022/lib/models/index.mjs +6 -0
  8. package/esm2022/lib/models/question-component.mjs +2 -0
  9. package/esm2022/lib/models/translations.mjs +3 -0
  10. package/esm2022/shared/index.mjs +5 -0
  11. package/esm2022/shared/lib/components/audio-animation/audio-animation.component.mjs +114 -0
  12. package/esm2022/shared/lib/components/audio-animation/index.mjs +2 -0
  13. package/esm2022/shared/lib/components/index.mjs +3 -0
  14. package/esm2022/shared/lib/components/vimeo-video/index.mjs +2 -0
  15. package/esm2022/shared/lib/components/vimeo-video/vimeo-video.component.mjs +101 -0
  16. package/esm2022/shared/lib/models/answer.mjs +2 -0
  17. package/esm2022/shared/lib/models/assessment.mjs +2 -0
  18. package/esm2022/shared/lib/models/environment.mjs +2 -0
  19. package/esm2022/shared/lib/models/index.mjs +9 -0
  20. package/esm2022/shared/lib/models/question-component.mjs +2 -0
  21. package/esm2022/shared/lib/models/question.mjs +2 -0
  22. package/esm2022/shared/lib/models/test.mjs +2 -0
  23. package/esm2022/shared/lib/models/translations.mjs +2 -0
  24. package/esm2022/shared/lib/models/window.mjs +2 -0
  25. package/esm2022/shared/lib/services/api/api.service.mjs +97 -0
  26. package/esm2022/shared/lib/services/api/mocked-api.service.mjs +131 -0
  27. package/esm2022/shared/lib/services/environment/environment.service.mjs +13 -0
  28. package/esm2022/shared/lib/services/index.mjs +10 -0
  29. package/esm2022/shared/lib/services/localization/languages.model.mjs +19 -0
  30. package/esm2022/shared/lib/services/localization/transloco-lazy-module-utils.mjs +27 -0
  31. package/esm2022/shared/lib/services/localization/transloco-testing.module.mjs +11 -0
  32. package/esm2022/shared/lib/services/media/media.service.mjs +129 -0
  33. package/esm2022/shared/lib/services/mixpanel/mixpanel.service.mjs +30 -0
  34. package/esm2022/shared/lib/services/theme/theme.service.mjs +24 -0
  35. package/esm2022/shared/test-mocks/assessment-test.mock.mjs +112 -0
  36. package/esm2022/shared/test-mocks/index.mjs +3 -0
  37. package/esm2022/shared/test-mocks/tgo-ui.mock.mjs +39 -0
  38. package/esm2022/testgorilla-tgo-ai-interview-test.mjs +5 -0
  39. package/fesm2022/testgorilla-tgo-ai-interview-test.mjs +484 -0
  40. package/fesm2022/testgorilla-tgo-ai-interview-test.mjs.map +1 -0
  41. package/lib/components/ai-interview-test/ai-interview-test.component.d.ts +44 -0
  42. package/lib/components/interview-stream/interview-stream.component.d.ts +46 -0
  43. package/lib/components/interview-video/interview-video.component.d.ts +16 -0
  44. package/lib/models/index.d.ts +3 -0
  45. package/lib/models/question-component.d.ts +5 -0
  46. package/lib/models/translations.d.ts +1 -0
  47. package/package.json +18 -9
  48. package/shared/index.d.ts +4 -0
  49. package/shared/lib/components/audio-animation/audio-animation.component.d.ts +27 -0
  50. package/shared/lib/components/audio-animation/index.d.ts +1 -0
  51. package/shared/lib/components/index.d.ts +2 -0
  52. package/shared/lib/components/vimeo-video/index.d.ts +1 -0
  53. package/shared/lib/components/vimeo-video/vimeo-video.component.d.ts +24 -0
  54. package/shared/lib/models/answer.d.ts +17 -0
  55. package/shared/lib/models/assessment.d.ts +80 -0
  56. package/shared/lib/models/environment.d.ts +1 -0
  57. package/shared/lib/models/index.d.ts +8 -0
  58. package/shared/lib/models/question-component.d.ts +54 -0
  59. package/shared/lib/models/question.d.ts +102 -0
  60. package/shared/lib/models/test.d.ts +81 -0
  61. package/shared/lib/models/translations.d.ts +1 -0
  62. package/shared/lib/models/window.d.ts +6 -0
  63. package/shared/lib/services/api/api.service.d.ts +25 -0
  64. package/shared/lib/services/api/mocked-api.service.d.ts +35 -0
  65. package/shared/lib/services/environment/environment.service.d.ts +6 -0
  66. package/shared/lib/services/index.d.ts +9 -0
  67. package/shared/lib/services/localization/languages.model.d.ts +15 -0
  68. package/shared/lib/services/localization/transloco-lazy-module-utils.d.ts +11 -0
  69. package/shared/lib/services/localization/transloco-testing.module.d.ts +2 -0
  70. package/shared/lib/services/media/media.service.d.ts +29 -0
  71. package/shared/lib/services/mixpanel/mixpanel.service.d.ts +10 -0
  72. package/shared/lib/services/theme/theme.service.d.ts +8 -0
  73. package/shared/test-mocks/assessment-test.mock.d.ts +21 -0
  74. package/shared/test-mocks/index.d.ts +2 -0
  75. package/shared/test-mocks/tgo-ui.mock.d.ts +21 -0
  76. package/.eslintrc.json +0 -46
  77. package/jest.config.ts +0 -29
  78. package/ng-package.json +0 -16
  79. package/project.json +0 -37
  80. package/src/lib/components/ai-interview-test/ai-interview-test.component.html +0 -42
  81. package/src/lib/components/ai-interview-test/ai-interview-test.component.scss +0 -167
  82. package/src/lib/components/ai-interview-test/ai-interview-test.component.spec.ts +0 -211
  83. package/src/lib/components/ai-interview-test/ai-interview-test.component.ts +0 -193
  84. package/src/lib/components/interview-stream/interview-stream.component.html +0 -9
  85. package/src/lib/components/interview-stream/interview-stream.component.scss +0 -5
  86. package/src/lib/components/interview-stream/interview-stream.component.spec.ts +0 -285
  87. package/src/lib/components/interview-stream/interview-stream.component.ts +0 -321
  88. package/src/lib/components/interview-video/interview-video.component.html +0 -8
  89. package/src/lib/components/interview-video/interview-video.component.scss +0 -7
  90. package/src/lib/components/interview-video/interview-video.component.spec.ts +0 -140
  91. package/src/lib/components/interview-video/interview-video.component.ts +0 -68
  92. package/src/lib/models/index.ts +0 -13
  93. package/src/lib/models/question-component.ts +0 -13
  94. package/src/lib/models/translations.ts +0 -3
  95. package/src/test-setup.ts +0 -28
  96. package/tsconfig.json +0 -17
  97. package/tsconfig.lib.json +0 -15
  98. package/tsconfig.lib.prod.json +0 -10
  99. package/tsconfig.spec.json +0 -13
  100. /package/{src/assets → assets}/i18n/en.json +0 -0
  101. /package/{src/index.ts → index.d.ts} +0 -0
  102. /package/{src/lib/components/index.ts → lib/components/index.d.ts} +0 -0
package/README.md CHANGED
@@ -32,6 +32,38 @@ import { AiInterviewTestComponent } from '@testgorilla/tgo-ai-interview-test';
32
32
  export class MyComponent {}
33
33
  ```
34
34
 
35
+ ## Assets (translations)
36
+
37
+ This package ships its translations under `assets/i18n`. Consumer apps must copy these assets into their build output so Transloco can load them at runtime.
38
+
39
+ - **Angular CLI / Nx**: add an `assets` entry pointing at the package:
40
+
41
+ ```json
42
+ {
43
+ "assets": [
44
+ "src/favicon.ico",
45
+ "src/assets",
46
+ {
47
+ "glob": "**/*",
48
+ "input": "node_modules/@testgorilla/tgo-ai-interview-test/assets",
49
+ "output": "assets/tgo-ai-interview-test"
50
+ }
51
+ ]
52
+ }
53
+ ```
54
+
55
+ If you develop inside this repo (consuming the workspace sources), also include the local path:
56
+
57
+ ```json
58
+ {
59
+ "glob": "**/*",
60
+ "input": "packages/tgo-ai-interview-test/src/assets",
61
+ "output": "assets/tgo-ai-interview-test"
62
+ }
63
+ ```
64
+
65
+ After adding the asset entries, rebuild/re-serve your app so `/assets/tgo-ai-interview-test/i18n/en.json` is available.
66
+
35
67
  ## API
36
68
 
37
69
  ### Inputs
@@ -0,0 +1,4 @@
1
+ export * from './lib/components';
2
+ export * from './lib/components/ai-interview-test/ai-interview-test.component';
3
+ export * from './lib/models';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy90Z28tYWktaW50ZXJ2aWV3LXRlc3Qvc3JjL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxnRUFBZ0UsQ0FBQztBQUMvRSxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vbGliL2NvbXBvbmVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY29tcG9uZW50cy9haS1pbnRlcnZpZXctdGVzdC9haS1pbnRlcnZpZXctdGVzdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbW9kZWxzJztcbiJdfQ==
@@ -0,0 +1,180 @@
1
+ import { InterviewStreamComponent } from '../interview-stream/interview-stream.component';
2
+ import { firstValueFrom, Observable, takeUntil } from 'rxjs';
3
+ import { transition, style, animate, trigger } from '@angular/animations';
4
+ import { ElementRef, signal, ViewChild, Component, EventEmitter, Input, Output, inject, Inject, ChangeDetectorRef, } from '@angular/core';
5
+ import { Subject } from 'rxjs';
6
+ import { ButtonComponentModule, DialogService, IconComponentModule, } from '@testgorilla/tgo-ui';
7
+ import { CommonModule } from '@angular/common';
8
+ import { TRANSLOCO_SCOPE, TranslocoModule, TranslocoService, } from '@ngneat/transloco';
9
+ import { AudioAnimationComponent, ROOT_TRANSLATIONS_SCOPE, MediaService, ThemeService, TranslocoLazyModuleUtils, getAvailableLangs, } from '../../../shared/index.mjs';
10
+ import * as i0 from "@angular/core";
11
+ import * as i1 from "@testgorilla/tgo-ui";
12
+ import * as i2 from "@angular/common";
13
+ export class AiInterviewTestComponent {
14
+ translationScope;
15
+ videoElement;
16
+ audioElement;
17
+ question;
18
+ test;
19
+ isFirstQuestion = false;
20
+ selectedMediaDevices;
21
+ conversationUrl;
22
+ mediaAccessChanged;
23
+ submissionStateChanged = new EventEmitter();
24
+ loadingStateChanged = new EventEmitter();
25
+ requestMediaAccess = new EventEmitter();
26
+ isInterviewInProgress = signal(false);
27
+ candidateVideoStreamReady = signal(false);
28
+ translations = {};
29
+ hasMediaPermissions = signal(false);
30
+ unsubscribe$ = new Subject();
31
+ mediaService = inject(MediaService);
32
+ translocoService = inject(TranslocoService);
33
+ cdr = inject(ChangeDetectorRef);
34
+ themeService = inject(ThemeService);
35
+ companyColor = this.themeService.getCompanyColor();
36
+ constructor(translationScope) {
37
+ this.translationScope = translationScope;
38
+ }
39
+ ngOnInit() {
40
+ this.initMediaAccessSubscription();
41
+ this.loadingStateChanged.emit(true);
42
+ this.mediaService.setSelectedMediaDevices(this.selectedMediaDevices);
43
+ void this.checkMediaPermissions();
44
+ void this.setTranslations();
45
+ void this.initVideoStream();
46
+ }
47
+ ngOnDestroy() {
48
+ this.unsubscribe$.next();
49
+ this.unsubscribe$.complete();
50
+ }
51
+ onVideoLoad() {
52
+ this.candidateVideoStreamReady.set(true);
53
+ }
54
+ interviewStarted() {
55
+ this.isInterviewInProgress.set(true);
56
+ this.loadingStateChanged.emit(false);
57
+ }
58
+ interviewEnded() {
59
+ this.isInterviewInProgress.set(false);
60
+ this.submissionStateChanged.emit({
61
+ text: '',
62
+ });
63
+ }
64
+ async checkMediaPermissions() {
65
+ if (!(await this.mediaService.checkPermission({ audio: true, video: false }))) {
66
+ this.hasMediaPermissions.set(false);
67
+ this.requestMediaAccess.emit();
68
+ return;
69
+ }
70
+ else {
71
+ this.hasMediaPermissions.set(true);
72
+ }
73
+ }
74
+ async initVideoStream() {
75
+ try {
76
+ const stream = await this.mediaService.getMediaStream({
77
+ video: true,
78
+ audio: false,
79
+ });
80
+ if (this.videoElement) {
81
+ this.videoElement.nativeElement.srcObject = stream;
82
+ await this.videoElement?.nativeElement.play();
83
+ }
84
+ }
85
+ catch (error) {
86
+ console.error('Error initializing video stream:', error);
87
+ this.candidateVideoStreamReady.set(false);
88
+ }
89
+ }
90
+ async setTranslations() {
91
+ this.translations = await firstValueFrom(this.translocoService.selectTranslateObject(`TEST`, {}, this.translationScope));
92
+ this.cdr.markForCheck();
93
+ }
94
+ initMediaAccessSubscription() {
95
+ this.mediaAccessChanged
96
+ ?.pipe(takeUntil(this.unsubscribe$))
97
+ .subscribe((selectedMediaDevices) => {
98
+ this.mediaService.setSelectedMediaDevices(selectedMediaDevices);
99
+ this.checkMediaPermissions();
100
+ void this.initVideoStream();
101
+ });
102
+ }
103
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AiInterviewTestComponent, deps: [{ token: TRANSLOCO_SCOPE }], target: i0.ɵɵFactoryTarget.Component });
104
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: AiInterviewTestComponent, isStandalone: true, selector: "tgo-ai-interview-test", inputs: { question: "question", test: "test", isFirstQuestion: "isFirstQuestion", selectedMediaDevices: "selectedMediaDevices", conversationUrl: "conversationUrl", mediaAccessChanged: "mediaAccessChanged" }, outputs: { submissionStateChanged: "submissionStateChanged", loadingStateChanged: "loadingStateChanged", requestMediaAccess: "requestMediaAccess" }, providers: [
105
+ TranslocoLazyModuleUtils.getScopeProvider('tgo-ai-interview-test', getAvailableLangs(), ROOT_TRANSLATIONS_SCOPE, (lang) => {
106
+ // Fetch from app assets; demo app copies the library assets to
107
+ // /assets/tgo-ai-interview-test via project.json.
108
+ const url = new URL(`assets/tgo-ai-interview-test/i18n/${lang}.json`, document.baseURI).href;
109
+ return fetch(url).then((res) => res.json());
110
+ }),
111
+ DialogService,
112
+ ThemeService,
113
+ ], viewQueries: [{ propertyName: "videoElement", first: true, predicate: ["video"], descendants: true }, { propertyName: "audioElement", first: true, predicate: ["audio"], descendants: true }], ngImport: i0, template: "<div class=\"ai-interview-test\">\n <div class=\"test-container\">\n <div\n class=\"media-container\"\n [class.is-video-visible]=\"isInterviewInProgress()\"\n >\n <div class=\"candidate-no-camera\" *ngIf=\"!candidateVideoStreamReady()\">\n <h3>&nbsp;</h3>\n <ui-icon name=\"User-profile-in-line\" color=\"white\" size=\"24\"></ui-icon>\n <h3 class=\"bold\">{{ translations['YOU'] }}</h3>\n </div>\n <div class=\"candidate-camera\" [hidden]=\"!candidateVideoStreamReady()\">\n <video\n height\n #video\n id=\"video\"\n playsinline\n (loadedmetadata)=\"onVideoLoad()\"\n ></video>\n <h3 class=\"bold\" *ngIf=\"candidateVideoStreamReady()\">\n {{ translations['YOU'] }}\n </h3>\n </div>\n <tgo-audio-animation\n *ngIf=\"isInterviewInProgress()\"\n [fakeData]=\"true\"\n ></tgo-audio-animation>\n <div class=\"interview-stream-container\">\n <tgo-interview-stream\n *ngIf=\"conversationUrl && hasMediaPermissions()\"\n [selectedMediaDevices]=\"selectedMediaDevices\"\n [conversationUrl]=\"conversationUrl\"\n (streamStart)=\"interviewStarted()\"\n (streamEnd)=\"interviewEnded()\"\n (checkMediaPermissions)=\"checkMediaPermissions()\"\n [translations]=\"translations\"\n ></tgo-interview-stream>\n </div>\n </div>\n </div>\n</div>\n\n", styles: [".bg-teal-60b{background:#1c443c}.bg-teal-30b{background:#31766a}.bg-teal-default{background:#46a997}.bg-teal-30w{background:#7ec3b6}.bg-teal-60w{background:#b5ddd5}.bg-teal-secondary{background:#cbd6cb}.bg-teal-90w{background:#ecf6f5}.bg-petrol-60b{background:#102930}.bg-petrol-30b{background:#1b4754}.bg-petrol-default{background:#276678}.bg-petrol-30w{background:#6894a0}.bg-petrol-60w{background:#a9c2c9}.bg-petrol-secondary{background:#c8d7de}.bg-petrol-90w{background:#e9f0f1}.bg-error-60b{background:#513131}.bg-error-30b{background:#8e5655}.bg-error-60w{background:#e3c3c6}.bg-error-secondary{background:#f0dad9}.bg-error-default{background:#cb7b7a}.bg-warning-secondary{background:#f0d6bb}.bg-warning-default{background:#cca45f}.bg-black{background:#000}.bg-dark{background:#888}.bg-medium{background:#e0e0e0}.bg-grey{background:#ededed}.bg-light{background:#f6f6f6}.bg-white{background:#fff}.bg-box-shadow{background:#00000014}.bg-navigation-subtitle{background:#528593}.bgc-teal-60b{background-color:#1c443c}.bgc-teal-30b{background-color:#31766a}.bgc-teal-default{background-color:#46a997}.bgc-teal-30w{background-color:#7ec3b6}.bgc-teal-60w{background-color:#b5ddd5}.bgc-teal-secondary{background-color:#cbd6cb}.bgc-teal-90w{background-color:#ecf6f5}.bgc-petrol-60b{background-color:#102930}.bgc-petrol-30b{background-color:#1b4754}.bgc-petrol-default{background-color:#276678}.bgc-petrol-30w{background-color:#6894a0}.bgc-petrol-60w{background-color:#a9c2c9}.bgc-petrol-secondary{background-color:#c8d7de}.bgc-petrol-90w{background-color:#e9f0f1}.bgc-error-60b{background-color:#513131}.bgc-error-30b{background-color:#8e5655}.bgc-error-60w{background-color:#e3c3c6}.bgc-error-secondary{background-color:#f0dad9}.bgc-error-default{background-color:#cb7b7a}.bgc-warning-secondary{background-color:#f0d6bb}.bgc-warning-default{background-color:#cca45f}.bgc-black{background-color:#000}.bgc-dark{background-color:#888}.bgc-medium{background-color:#e0e0e0}.bgc-grey{background-color:#ededed}.bgc-light{background-color:#f6f6f6}.bgc-white{background-color:#fff}.bgc-box-shadow{background-color:#00000014}.bgc-navigation-subtitle{background-color:#528593}.ai-interview-test{background-color:#242424;padding:24px;display:flex;justify-content:center;min-height:calc(100vh - 80px)}.ai-interview-test h3{color:#fff}.ai-interview-test .test-container{min-width:360px;width:100%;min-height:300px;height:min(725px,100%,max(100vw,100vh,200px));max-width:calc(177.7777777778vh - 80px - 48px);display:flex;flex-direction:column;justify-content:space-between;position:relative;background-color:#242424}.ai-interview-test .media-container{position:relative;color:#fff;flex-grow:1}.ai-interview-test .media-container .interview-stream-container{border-radius:10px;aspect-ratio:16/9;border:4px solid #0165FC;margin:0 auto}.ai-interview-test .media-container .candidate-no-camera{position:absolute;width:200px;aspect-ratio:16/9;z-index:3;top:32px;left:32px;border-radius:10px;background-color:#1a47aa;display:flex;justify-content:space-between;flex-direction:column;padding:16px}.ai-interview-test .media-container .candidate-no-camera ui-icon{margin:auto}.ai-interview-test .media-container .candidate-camera{position:absolute;width:200px;top:32px;left:32px;z-index:3}.ai-interview-test .media-container .candidate-camera video{width:200px;border-radius:10px}.ai-interview-test .media-container .candidate-camera h3{position:absolute;bottom:16px;left:16px}.ai-interview-test .media-container tgo-audio-animation{position:absolute;top:32px;right:32px;z-index:1}.ai-interview-test .media-container tgo-vimeo-video{display:none;position:absolute;height:100%;width:100%}.ai-interview-test .media-container.is-video-visible tgo-vimeo-video{display:block}.ai-interview-test .media-container.is-playing{border:4px solid #0165FC;border-bottom-width:3px}.ai-interview-test .media-container.is-answering .candidate-camera tgo-audio-animation,.ai-interview-test .media-container.is-answering .candidate-no-camera tgo-audio-animation{position:absolute;top:16px;right:16px}.ai-interview-test .media-container.is-answering .candidate-camera video,.ai-interview-test .media-container.is-answering .candidate-no-camera{border:3px solid #0165FC;border-radius:10px}.ai-interview-test .media-container.is-answering tgo-vimeo-video{display:block}.ai-interview-test .media-container .start,.ai-interview-test .media-container .audio-info,.ai-interview-test .media-container .overlay,.ai-interview-test .media-container .answer{display:flex;justify-content:center;align-items:center;flex-direction:column;gap:40px;height:100%}.ai-interview-test .media-container .preview{height:100%;display:flex;align-items:center;flex-direction:column;justify-content:flex-end;padding:24px}.ai-interview-test .media-container .preview p{color:#d3d3d3;margin:8px 0}.ai-interview-test .media-container .preview audio{margin:16px 0}.ai-interview-test .media-container .preview.hidden{display:none}@media screen and (max-width: 600px){.ai-interview-test{padding:0 24px;margin-top:16px}}\n"], dependencies: [{ kind: "ngmodule", type: TranslocoModule }, { kind: "ngmodule", type: ButtonComponentModule }, { kind: "ngmodule", type: IconComponentModule }, { kind: "component", type: i1.IconComponent, selector: "ui-icon", inputs: ["size", "cssClass", "name", "color", "filled", "toggleIconStyle", "applicationTheme", "useFullIconName"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: AudioAnimationComponent, selector: "tgo-audio-animation", inputs: ["volume", "fakeData"] }, { kind: "component", type: InterviewStreamComponent, selector: "tgo-interview-stream", inputs: ["conversationUrl", "selectedMediaDevices", "translations"], outputs: ["streamStart", "streamEnd", "checkMediaPermissions"] }], animations: [
114
+ trigger('fadeInFadeOut', [
115
+ transition(':enter', [
116
+ style({ opacity: 0 }),
117
+ animate('600ms', style({ opacity: 1 })),
118
+ ]),
119
+ transition(':leave', [animate('600ms', style({ opacity: 0 }))]),
120
+ ]),
121
+ ] });
122
+ }
123
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AiInterviewTestComponent, decorators: [{
124
+ type: Component,
125
+ args: [{ selector: 'tgo-ai-interview-test', animations: [
126
+ trigger('fadeInFadeOut', [
127
+ transition(':enter', [
128
+ style({ opacity: 0 }),
129
+ animate('600ms', style({ opacity: 1 })),
130
+ ]),
131
+ transition(':leave', [animate('600ms', style({ opacity: 0 }))]),
132
+ ]),
133
+ ], standalone: true, imports: [
134
+ TranslocoModule,
135
+ ButtonComponentModule,
136
+ IconComponentModule,
137
+ CommonModule,
138
+ AudioAnimationComponent,
139
+ InterviewStreamComponent,
140
+ ], providers: [
141
+ TranslocoLazyModuleUtils.getScopeProvider('tgo-ai-interview-test', getAvailableLangs(), ROOT_TRANSLATIONS_SCOPE, (lang) => {
142
+ // Fetch from app assets; demo app copies the library assets to
143
+ // /assets/tgo-ai-interview-test via project.json.
144
+ const url = new URL(`assets/tgo-ai-interview-test/i18n/${lang}.json`, document.baseURI).href;
145
+ return fetch(url).then((res) => res.json());
146
+ }),
147
+ DialogService,
148
+ ThemeService,
149
+ ], template: "<div class=\"ai-interview-test\">\n <div class=\"test-container\">\n <div\n class=\"media-container\"\n [class.is-video-visible]=\"isInterviewInProgress()\"\n >\n <div class=\"candidate-no-camera\" *ngIf=\"!candidateVideoStreamReady()\">\n <h3>&nbsp;</h3>\n <ui-icon name=\"User-profile-in-line\" color=\"white\" size=\"24\"></ui-icon>\n <h3 class=\"bold\">{{ translations['YOU'] }}</h3>\n </div>\n <div class=\"candidate-camera\" [hidden]=\"!candidateVideoStreamReady()\">\n <video\n height\n #video\n id=\"video\"\n playsinline\n (loadedmetadata)=\"onVideoLoad()\"\n ></video>\n <h3 class=\"bold\" *ngIf=\"candidateVideoStreamReady()\">\n {{ translations['YOU'] }}\n </h3>\n </div>\n <tgo-audio-animation\n *ngIf=\"isInterviewInProgress()\"\n [fakeData]=\"true\"\n ></tgo-audio-animation>\n <div class=\"interview-stream-container\">\n <tgo-interview-stream\n *ngIf=\"conversationUrl && hasMediaPermissions()\"\n [selectedMediaDevices]=\"selectedMediaDevices\"\n [conversationUrl]=\"conversationUrl\"\n (streamStart)=\"interviewStarted()\"\n (streamEnd)=\"interviewEnded()\"\n (checkMediaPermissions)=\"checkMediaPermissions()\"\n [translations]=\"translations\"\n ></tgo-interview-stream>\n </div>\n </div>\n </div>\n</div>\n\n", styles: [".bg-teal-60b{background:#1c443c}.bg-teal-30b{background:#31766a}.bg-teal-default{background:#46a997}.bg-teal-30w{background:#7ec3b6}.bg-teal-60w{background:#b5ddd5}.bg-teal-secondary{background:#cbd6cb}.bg-teal-90w{background:#ecf6f5}.bg-petrol-60b{background:#102930}.bg-petrol-30b{background:#1b4754}.bg-petrol-default{background:#276678}.bg-petrol-30w{background:#6894a0}.bg-petrol-60w{background:#a9c2c9}.bg-petrol-secondary{background:#c8d7de}.bg-petrol-90w{background:#e9f0f1}.bg-error-60b{background:#513131}.bg-error-30b{background:#8e5655}.bg-error-60w{background:#e3c3c6}.bg-error-secondary{background:#f0dad9}.bg-error-default{background:#cb7b7a}.bg-warning-secondary{background:#f0d6bb}.bg-warning-default{background:#cca45f}.bg-black{background:#000}.bg-dark{background:#888}.bg-medium{background:#e0e0e0}.bg-grey{background:#ededed}.bg-light{background:#f6f6f6}.bg-white{background:#fff}.bg-box-shadow{background:#00000014}.bg-navigation-subtitle{background:#528593}.bgc-teal-60b{background-color:#1c443c}.bgc-teal-30b{background-color:#31766a}.bgc-teal-default{background-color:#46a997}.bgc-teal-30w{background-color:#7ec3b6}.bgc-teal-60w{background-color:#b5ddd5}.bgc-teal-secondary{background-color:#cbd6cb}.bgc-teal-90w{background-color:#ecf6f5}.bgc-petrol-60b{background-color:#102930}.bgc-petrol-30b{background-color:#1b4754}.bgc-petrol-default{background-color:#276678}.bgc-petrol-30w{background-color:#6894a0}.bgc-petrol-60w{background-color:#a9c2c9}.bgc-petrol-secondary{background-color:#c8d7de}.bgc-petrol-90w{background-color:#e9f0f1}.bgc-error-60b{background-color:#513131}.bgc-error-30b{background-color:#8e5655}.bgc-error-60w{background-color:#e3c3c6}.bgc-error-secondary{background-color:#f0dad9}.bgc-error-default{background-color:#cb7b7a}.bgc-warning-secondary{background-color:#f0d6bb}.bgc-warning-default{background-color:#cca45f}.bgc-black{background-color:#000}.bgc-dark{background-color:#888}.bgc-medium{background-color:#e0e0e0}.bgc-grey{background-color:#ededed}.bgc-light{background-color:#f6f6f6}.bgc-white{background-color:#fff}.bgc-box-shadow{background-color:#00000014}.bgc-navigation-subtitle{background-color:#528593}.ai-interview-test{background-color:#242424;padding:24px;display:flex;justify-content:center;min-height:calc(100vh - 80px)}.ai-interview-test h3{color:#fff}.ai-interview-test .test-container{min-width:360px;width:100%;min-height:300px;height:min(725px,100%,max(100vw,100vh,200px));max-width:calc(177.7777777778vh - 80px - 48px);display:flex;flex-direction:column;justify-content:space-between;position:relative;background-color:#242424}.ai-interview-test .media-container{position:relative;color:#fff;flex-grow:1}.ai-interview-test .media-container .interview-stream-container{border-radius:10px;aspect-ratio:16/9;border:4px solid #0165FC;margin:0 auto}.ai-interview-test .media-container .candidate-no-camera{position:absolute;width:200px;aspect-ratio:16/9;z-index:3;top:32px;left:32px;border-radius:10px;background-color:#1a47aa;display:flex;justify-content:space-between;flex-direction:column;padding:16px}.ai-interview-test .media-container .candidate-no-camera ui-icon{margin:auto}.ai-interview-test .media-container .candidate-camera{position:absolute;width:200px;top:32px;left:32px;z-index:3}.ai-interview-test .media-container .candidate-camera video{width:200px;border-radius:10px}.ai-interview-test .media-container .candidate-camera h3{position:absolute;bottom:16px;left:16px}.ai-interview-test .media-container tgo-audio-animation{position:absolute;top:32px;right:32px;z-index:1}.ai-interview-test .media-container tgo-vimeo-video{display:none;position:absolute;height:100%;width:100%}.ai-interview-test .media-container.is-video-visible tgo-vimeo-video{display:block}.ai-interview-test .media-container.is-playing{border:4px solid #0165FC;border-bottom-width:3px}.ai-interview-test .media-container.is-answering .candidate-camera tgo-audio-animation,.ai-interview-test .media-container.is-answering .candidate-no-camera tgo-audio-animation{position:absolute;top:16px;right:16px}.ai-interview-test .media-container.is-answering .candidate-camera video,.ai-interview-test .media-container.is-answering .candidate-no-camera{border:3px solid #0165FC;border-radius:10px}.ai-interview-test .media-container.is-answering tgo-vimeo-video{display:block}.ai-interview-test .media-container .start,.ai-interview-test .media-container .audio-info,.ai-interview-test .media-container .overlay,.ai-interview-test .media-container .answer{display:flex;justify-content:center;align-items:center;flex-direction:column;gap:40px;height:100%}.ai-interview-test .media-container .preview{height:100%;display:flex;align-items:center;flex-direction:column;justify-content:flex-end;padding:24px}.ai-interview-test .media-container .preview p{color:#d3d3d3;margin:8px 0}.ai-interview-test .media-container .preview audio{margin:16px 0}.ai-interview-test .media-container .preview.hidden{display:none}@media screen and (max-width: 600px){.ai-interview-test{padding:0 24px;margin-top:16px}}\n"] }]
150
+ }], ctorParameters: () => [{ type: undefined, decorators: [{
151
+ type: Inject,
152
+ args: [TRANSLOCO_SCOPE]
153
+ }] }], propDecorators: { videoElement: [{
154
+ type: ViewChild,
155
+ args: ['video']
156
+ }], audioElement: [{
157
+ type: ViewChild,
158
+ args: ['audio']
159
+ }], question: [{
160
+ type: Input,
161
+ args: [{ required: true }]
162
+ }], test: [{
163
+ type: Input,
164
+ args: [{ required: true }]
165
+ }], isFirstQuestion: [{
166
+ type: Input
167
+ }], selectedMediaDevices: [{
168
+ type: Input
169
+ }], conversationUrl: [{
170
+ type: Input
171
+ }], mediaAccessChanged: [{
172
+ type: Input
173
+ }], submissionStateChanged: [{
174
+ type: Output
175
+ }], loadingStateChanged: [{
176
+ type: Output
177
+ }], requestMediaAccess: [{
178
+ type: Output
179
+ }] } });
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ai-interview-test.component.js","sourceRoot":"","sources":["../../../../../../../packages/tgo-ai-interview-test/src/lib/components/ai-interview-test/ai-interview-test.component.ts","../../../../../../../packages/tgo-ai-interview-test/src/lib/components/ai-interview-test/ai-interview-test.component.html"],"names":[],"mappings":"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","sourcesContent":["import { InterviewStreamComponent } from '../interview-stream/interview-stream.component';\nimport { firstValueFrom, Observable, takeUntil } from 'rxjs';\nimport { transition, style, animate, trigger } from '@angular/animations';\nimport {\n  ElementRef,\n  OnDestroy,\n  signal,\n  ViewChild,\n  Component,\n  EventEmitter,\n  Input,\n  Output,\n  inject,\n  OnInit,\n  Inject,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { Subject } from 'rxjs';\nimport {\n  ButtonComponentModule,\n  DialogService,\n  IconComponentModule,\n} from '@testgorilla/tgo-ui';\nimport { CommonModule } from '@angular/common';\nimport {\n  TRANSLOCO_SCOPE,\n  TranslocoModule,\n  TranslocoScope,\n  TranslocoService,\n} from '@ngneat/transloco';\nimport {\n  AudioAnimationComponent,\n  Question,\n  TestResultRead,\n  SelectedMediaDevices,\n  ISubmissionState,\n  ROOT_TRANSLATIONS_SCOPE,\n  MediaService,\n  ThemeService,\n  TranslocoLazyModuleUtils,\n  getAvailableLangs,\n} from '@testgorilla/tgo-test-shared';\nimport { IQuestionDataContract } from '../../models';\n\n@Component({\n  selector: 'tgo-ai-interview-test',\n  templateUrl: './ai-interview-test.component.html',\n  styleUrl: './ai-interview-test.component.scss',\n  animations: [\n    trigger('fadeInFadeOut', [\n      transition(':enter', [\n        style({ opacity: 0 }),\n        animate('600ms', style({ opacity: 1 })),\n      ]),\n      transition(':leave', [animate('600ms', style({ opacity: 0 }))]),\n    ]),\n  ],\n  standalone: true,\n  imports: [\n    TranslocoModule,\n    ButtonComponentModule,\n    IconComponentModule,\n    CommonModule,\n    AudioAnimationComponent,\n    InterviewStreamComponent,\n  ],\n  providers: [\n    TranslocoLazyModuleUtils.getScopeProvider(\n      'tgo-ai-interview-test',\n      getAvailableLangs(),\n      ROOT_TRANSLATIONS_SCOPE,\n      (lang: string) => {\n        // Fetch from app assets; demo app copies the library assets to\n        // /assets/tgo-ai-interview-test via project.json.\n        const url = new URL(\n          `assets/tgo-ai-interview-test/i18n/${lang}.json`,\n          document.baseURI\n        ).href;\n        return fetch(url).then((res) => res.json());\n      }\n    ),\n    DialogService,\n    ThemeService,\n  ],\n})\nexport class AiInterviewTestComponent\n  implements OnInit, OnDestroy, IQuestionDataContract\n{\n  @ViewChild('video') videoElement?: ElementRef<HTMLVideoElement>;\n  @ViewChild('audio') audioElement?: ElementRef<HTMLAudioElement>;\n  @Input({ required: true }) question!: Question;\n  @Input({ required: true }) test!: TestResultRead;\n  @Input() isFirstQuestion?: boolean = false;\n  @Input() selectedMediaDevices?: SelectedMediaDevices;\n  @Input() conversationUrl?: string;\n  @Input() mediaAccessChanged?: Observable<SelectedMediaDevices> | undefined;\n\n  @Output() submissionStateChanged =\n    new EventEmitter<ISubmissionState | null>();\n  @Output() loadingStateChanged: EventEmitter<boolean> =\n    new EventEmitter<boolean>();\n  @Output() requestMediaAccess: EventEmitter<void> = new EventEmitter<void>();\n\n  isInterviewInProgress = signal(false);\n  candidateVideoStreamReady = signal(false);\n  translations: { [key: string]: string } = {};\n  hasMediaPermissions = signal(false);\n\n  private unsubscribe$ = new Subject<void>();\n  private mediaService = inject(MediaService);\n  private translocoService = inject(TranslocoService);\n  private cdr = inject(ChangeDetectorRef);\n  private themeService = inject(ThemeService);\n\n  companyColor = this.themeService.getCompanyColor();\n\n  constructor(\n    @Inject(TRANSLOCO_SCOPE) private translationScope: TranslocoScope\n  ) {}\n\n  ngOnInit(): void {\n    this.initMediaAccessSubscription();\n    this.loadingStateChanged.emit(true);\n    this.mediaService.setSelectedMediaDevices(this.selectedMediaDevices);\n    void this.checkMediaPermissions();\n\n    void this.setTranslations();\n    void this.initVideoStream();\n  }\n\n  ngOnDestroy() {\n    this.unsubscribe$.next();\n    this.unsubscribe$.complete();\n  }\n\n  onVideoLoad() {\n    this.candidateVideoStreamReady.set(true);\n  }\n\n  interviewStarted() {\n    this.isInterviewInProgress.set(true);\n    this.loadingStateChanged.emit(false);\n  }\n\n  interviewEnded() {\n    this.isInterviewInProgress.set(false);\n    this.submissionStateChanged.emit({\n      text: '',\n    });\n  }\n\n  async checkMediaPermissions() {\n    if (\n      !(await this.mediaService.checkPermission({ audio: true, video: false }))\n    ) {\n      this.hasMediaPermissions.set(false);\n      this.requestMediaAccess.emit();\n      return;\n    } else {\n      this.hasMediaPermissions.set(true);\n    }\n  }\n\n  private async initVideoStream() {\n    try {\n      const stream = await this.mediaService.getMediaStream({\n        video: true,\n        audio: false,\n      });\n      if (this.videoElement) {\n        this.videoElement.nativeElement.srcObject = stream;\n        await this.videoElement?.nativeElement.play();\n      }\n    } catch (error) {\n      console.error('Error initializing video stream:', error);\n      this.candidateVideoStreamReady.set(false);\n    }\n  }\n\n  private async setTranslations() {\n    this.translations = await firstValueFrom(\n      this.translocoService.selectTranslateObject(\n        `TEST`,\n        {},\n        this.translationScope as string\n      )\n    );\n    this.cdr.markForCheck();\n  }\n\n  private initMediaAccessSubscription() {\n    this.mediaAccessChanged\n      ?.pipe(takeUntil(this.unsubscribe$))\n      .subscribe((selectedMediaDevices) => {\n        this.mediaService.setSelectedMediaDevices(selectedMediaDevices);\n        this.checkMediaPermissions();\n        void this.initVideoStream();\n      });\n  }\n}\n\n","<div class=\"ai-interview-test\">\n  <div class=\"test-container\">\n    <div\n      class=\"media-container\"\n      [class.is-video-visible]=\"isInterviewInProgress()\"\n    >\n      <div class=\"candidate-no-camera\" *ngIf=\"!candidateVideoStreamReady()\">\n        <h3>&nbsp;</h3>\n        <ui-icon name=\"User-profile-in-line\" color=\"white\" size=\"24\"></ui-icon>\n        <h3 class=\"bold\">{{ translations['YOU'] }}</h3>\n      </div>\n      <div class=\"candidate-camera\" [hidden]=\"!candidateVideoStreamReady()\">\n        <video\n          height\n          #video\n          id=\"video\"\n          playsinline\n          (loadedmetadata)=\"onVideoLoad()\"\n        ></video>\n        <h3 class=\"bold\" *ngIf=\"candidateVideoStreamReady()\">\n          {{ translations['YOU'] }}\n        </h3>\n      </div>\n      <tgo-audio-animation\n        *ngIf=\"isInterviewInProgress()\"\n        [fakeData]=\"true\"\n      ></tgo-audio-animation>\n      <div class=\"interview-stream-container\">\n        <tgo-interview-stream\n          *ngIf=\"conversationUrl && hasMediaPermissions()\"\n          [selectedMediaDevices]=\"selectedMediaDevices\"\n          [conversationUrl]=\"conversationUrl\"\n          (streamStart)=\"interviewStarted()\"\n          (streamEnd)=\"interviewEnded()\"\n          (checkMediaPermissions)=\"checkMediaPermissions()\"\n          [translations]=\"translations\"\n        ></tgo-interview-stream>\n      </div>\n    </div>\n  </div>\n</div>\n\n"]}
@@ -0,0 +1,4 @@
1
+ export * from './ai-interview-test/ai-interview-test.component';
2
+ export * from './interview-stream/interview-stream.component';
3
+ export * from './interview-video/interview-video.component';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy90Z28tYWktaW50ZXJ2aWV3LXRlc3Qvc3JjL2xpYi9jb21wb25lbnRzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsaURBQWlELENBQUM7QUFDaEUsY0FBYywrQ0FBK0MsQ0FBQztBQUM5RCxjQUFjLDZDQUE2QyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9haS1pbnRlcnZpZXctdGVzdC9haS1pbnRlcnZpZXctdGVzdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcnZpZXctc3RyZWFtL2ludGVydmlldy1zdHJlYW0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vaW50ZXJ2aWV3LXZpZGVvL2ludGVydmlldy12aWRlby5jb21wb25lbnQnO1xuIl19
@@ -0,0 +1,240 @@
1
+ import { ChangeDetectorRef, Component, EventEmitter, inject, Input, Output, signal, } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import DailyIframe from '@daily-co/daily-js';
4
+ import { InterviewVideoComponent } from '../interview-video/interview-video.component';
5
+ import * as i0 from "@angular/core";
6
+ const PLAYABLE_STATE = 'playable';
7
+ const LOADING_STATE = 'loading';
8
+ export class InterviewStreamComponent {
9
+ conversationUrl;
10
+ selectedMediaDevices;
11
+ translations = {};
12
+ streamStart = new EventEmitter();
13
+ streamEnd = new EventEmitter();
14
+ checkMediaPermissions = new EventEmitter();
15
+ callObject;
16
+ avatarParticipant;
17
+ candidateJoined = signal(false);
18
+ cdr = inject(ChangeDetectorRef);
19
+ ngOnInit() {
20
+ void this.setupCall();
21
+ }
22
+ ngOnDestroy() {
23
+ if (!this.callObject)
24
+ return;
25
+ this.leaveCall();
26
+ this.callObject
27
+ .off('joined-meeting', this.candidateJoinMeeting)
28
+ .off('participant-joined', this.participantJoined)
29
+ .off('app-message', this.handleNewMessage)
30
+ .off('track-started', this.handleTrackStartedStopped)
31
+ .off('track-stopped', this.handleTrackStartedStopped)
32
+ .off('participant-left', this.handleParticipantLeft)
33
+ .off('left-meeting', this.handleLeftMeeting)
34
+ .off('error', this.handleError);
35
+ }
36
+ async setupCall() {
37
+ this.callObject = DailyIframe.getCallInstance();
38
+ if (!this.callObject) {
39
+ this.callObject = DailyIframe.createCallObject();
40
+ }
41
+ if (this.selectedMediaDevices) {
42
+ await this.callObject.setInputDevicesAsync({
43
+ videoDeviceId: this.selectedMediaDevices.videoDeviceId,
44
+ audioDeviceId: this.selectedMediaDevices.audioDeviceId,
45
+ });
46
+ }
47
+ this.callObject.startRecording({
48
+ type: 'cloud',
49
+ });
50
+ this.callObject
51
+ .on('joined-meeting', this.candidateJoinMeeting)
52
+ .on('participant-joined', this.participantJoined)
53
+ .on('app-message', this.handleNewMessage)
54
+ .on('track-started', this.handleTrackStartedStopped)
55
+ .on('track-stopped', this.handleTrackStartedStopped)
56
+ .on('participant-left', this.handleParticipantLeft)
57
+ .on('left-meeting', this.handleLeftMeeting)
58
+ .on('error', this.handleError);
59
+ await this.callObject.join({
60
+ userName: 'Candidate',
61
+ url: this.conversationUrl,
62
+ });
63
+ }
64
+ formatParticipantObj(participant) {
65
+ const { video, audio } = participant.tracks;
66
+ const videoTrack = video?.persistentTrack;
67
+ const audioTrack = audio?.persistentTrack;
68
+ return {
69
+ videoTrack: videoTrack,
70
+ audioTrack: audioTrack,
71
+ videoReady: !!(videoTrack &&
72
+ (video.state === PLAYABLE_STATE || video.state === LOADING_STATE)),
73
+ audioReady: !!(audioTrack &&
74
+ (audio.state === PLAYABLE_STATE || audio.state === LOADING_STATE)),
75
+ userName: participant.user_name,
76
+ local: participant.local,
77
+ sessionId: participant.session_id,
78
+ };
79
+ }
80
+ updateTrack(participant, newTrackType) {
81
+ if (!this.avatarParticipant ||
82
+ this.avatarParticipant.sessionId !== participant.session_id) {
83
+ return;
84
+ }
85
+ const existingParticipant = this.avatarParticipant;
86
+ const currentParticipantCopy = this.formatParticipantObj(participant);
87
+ if (newTrackType === 'video') {
88
+ if (existingParticipant.videoReady !== currentParticipantCopy.videoReady) {
89
+ existingParticipant.videoReady = currentParticipantCopy.videoReady;
90
+ }
91
+ if (currentParticipantCopy.videoReady &&
92
+ existingParticipant.videoTrack?.id !==
93
+ currentParticipantCopy.videoTrack?.id) {
94
+ existingParticipant.videoTrack = currentParticipantCopy.videoTrack;
95
+ }
96
+ return;
97
+ }
98
+ if (newTrackType === 'audio') {
99
+ if (existingParticipant.audioReady !== currentParticipantCopy.audioReady) {
100
+ existingParticipant.audioReady = currentParticipantCopy.audioReady;
101
+ }
102
+ if (currentParticipantCopy.audioReady &&
103
+ existingParticipant.audioTrack?.id !==
104
+ currentParticipantCopy.audioTrack?.id) {
105
+ existingParticipant.audioTrack = currentParticipantCopy.audioTrack;
106
+ }
107
+ }
108
+ }
109
+ candidateJoinMeeting = (event) => {
110
+ if (!event || !this.callObject)
111
+ return;
112
+ this.candidateJoined.set(true);
113
+ this.streamStart.emit();
114
+ };
115
+ participantJoined = (event) => {
116
+ if (!event)
117
+ return;
118
+ this.avatarParticipant = this.formatParticipantObj(event.participant);
119
+ };
120
+ handleTrackStartedStopped = (event) => {
121
+ if (!event || !event.participant || !this.candidateJoined())
122
+ return;
123
+ if (event.action === 'track-stopped') {
124
+ this.checkMediaPermissions.emit();
125
+ }
126
+ this.updateTrack(event.participant, event.type);
127
+ this.cdr.detectChanges();
128
+ };
129
+ handleParticipantLeft = (event) => {
130
+ if (!event)
131
+ return;
132
+ this.leaveCall();
133
+ };
134
+ handleError = (event) => {
135
+ if (!event)
136
+ return;
137
+ console.error('Interview stream error', event);
138
+ this.leaveCall();
139
+ };
140
+ handleLeftMeeting = (event) => {
141
+ this.callObject?.stopRecording();
142
+ if (!event || !this.callObject)
143
+ return;
144
+ this.candidateJoined.set(false);
145
+ this.callObject.destroy();
146
+ this.streamEnd.emit();
147
+ };
148
+ leaveCall() {
149
+ if (!this.callObject)
150
+ return;
151
+ this.callObject.leave();
152
+ }
153
+ handleNewMessage = (event) => {
154
+ if (!event)
155
+ return;
156
+ if (event.data.event_type === 'conversation.tool_call') {
157
+ this.handleToolCall(event.data.properties);
158
+ }
159
+ };
160
+ getConversationId() {
161
+ return this.conversationUrl?.replace('https://tavus.daily.co/', '');
162
+ }
163
+ /*
164
+ This is a test implementation of tool calling.
165
+ These events will only be triggered if configured with a Tavus persona. The message content will be further refined by the IP Team.
166
+ https://docs.tavus.io/sections/event-schemas/conversation-toolcall
167
+ */
168
+ handleToolCall = (properties) => {
169
+ switch (properties.name) {
170
+ case 'next_question':
171
+ try {
172
+ const args = JSON.parse(properties.arguments);
173
+ if (args?.questionsLeft > 0) {
174
+ this.sendMessage(this.getToolCallTranslation('NEXT_QUESTION'));
175
+ window.setTimeout(() => {
176
+ this.sendMessage('Read next question', 'respond');
177
+ }, 1000);
178
+ }
179
+ else {
180
+ throw new Error('No more questions left');
181
+ }
182
+ }
183
+ catch (err) {
184
+ console.error('Failed to parse arguments for next_question tool call', err);
185
+ this.sendMessage(this.getToolCallTranslation('ALL_FOR_TODAY'));
186
+ window.setTimeout(() => {
187
+ this.leaveCall();
188
+ }, 5000);
189
+ }
190
+ break;
191
+ case 'end_conversation':
192
+ window.setTimeout(() => {
193
+ this.leaveCall();
194
+ }, 5000);
195
+ break;
196
+ default:
197
+ console.warn('Unknown tool call code:', properties);
198
+ break;
199
+ }
200
+ };
201
+ /*
202
+ Echo message is a message that avatar reads and candidate can hear
203
+ Respond message is a message that avatar reads and candidate cannot hear
204
+ */
205
+ sendMessage(text, type = 'echo') {
206
+ if (!this.callObject)
207
+ return;
208
+ this.callObject.sendAppMessage({
209
+ message_type: 'conversation',
210
+ event_type: 'conversation.' + type,
211
+ conversation_id: this.getConversationId(),
212
+ properties: {
213
+ text,
214
+ },
215
+ });
216
+ }
217
+ getToolCallTranslation(key) {
218
+ const toolCallTranslations = this.translations['TOOL_CALL'];
219
+ return toolCallTranslations[key];
220
+ }
221
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: InterviewStreamComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
222
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: InterviewStreamComponent, isStandalone: true, selector: "tgo-interview-stream", inputs: { conversationUrl: "conversationUrl", selectedMediaDevices: "selectedMediaDevices", translations: "translations" }, outputs: { streamStart: "streamStart", streamEnd: "streamEnd", checkMediaPermissions: "checkMediaPermissions" }, ngImport: i0, template: "<div class=\"interview-stream\">\n @if (avatarParticipant) {\n <tgo-interview-video\n [videoTrack]=\"avatarParticipant.videoTrack\"\n [audioTrack]=\"avatarParticipant.audioTrack\"\n ></tgo-interview-video>\n }\n</div>\n\n", styles: [".interview-stream{width:100%;height:100%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: InterviewVideoComponent, selector: "tgo-interview-video", inputs: ["videoTrack", "audioTrack"] }] });
223
+ }
224
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: InterviewStreamComponent, decorators: [{
225
+ type: Component,
226
+ args: [{ selector: 'tgo-interview-stream', standalone: true, imports: [CommonModule, InterviewVideoComponent], template: "<div class=\"interview-stream\">\n @if (avatarParticipant) {\n <tgo-interview-video\n [videoTrack]=\"avatarParticipant.videoTrack\"\n [audioTrack]=\"avatarParticipant.audioTrack\"\n ></tgo-interview-video>\n }\n</div>\n\n", styles: [".interview-stream{width:100%;height:100%}\n"] }]
227
+ }], propDecorators: { conversationUrl: [{
228
+ type: Input
229
+ }], selectedMediaDevices: [{
230
+ type: Input
231
+ }], translations: [{
232
+ type: Input
233
+ }], streamStart: [{
234
+ type: Output
235
+ }], streamEnd: [{
236
+ type: Output
237
+ }], checkMediaPermissions: [{
238
+ type: Output
239
+ }] } });
240
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interview-stream.component.js","sourceRoot":"","sources":["../../../../../../../packages/tgo-ai-interview-test/src/lib/components/interview-stream/interview-stream.component.ts","../../../../../../../packages/tgo-ai-interview-test/src/lib/components/interview-stream/interview-stream.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  signal,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport DailyIframe, {\n  DailyCall,\n  DailyEventObjectParticipant,\n  DailyParticipant,\n  DailyEventObjectFatalError,\n  DailyEventObjectParticipants,\n  DailyEventObjectNoPayload,\n  DailyEventObjectParticipantLeft,\n  DailyEventObjectTrack,\n  DailyEventObjectAppMessage,\n} from '@daily-co/daily-js';\nimport { SelectedMediaDevices } from '@testgorilla/tgo-test-shared';\nimport { InterviewVideoComponent } from '../interview-video/interview-video.component';\n\nexport type Participant = {\n  videoTrack?: MediaStreamTrack | undefined;\n  audioTrack?: MediaStreamTrack | undefined;\n  videoReady: boolean;\n  audioReady: boolean;\n  userName: string;\n  local: boolean;\n  sessionId: string;\n};\n\nconst PLAYABLE_STATE = 'playable';\nconst LOADING_STATE = 'loading';\n\ninterface EventData {\n  event_type: string;\n  properties: {\n    name: string;\n    arguments: string;\n  };\n}\n\n@Component({\n  selector: 'tgo-interview-stream',\n  templateUrl: './interview-stream.component.html',\n  styleUrls: ['./interview-stream.component.scss'],\n  standalone: true,\n  imports: [CommonModule, InterviewVideoComponent],\n})\nexport class InterviewStreamComponent implements OnInit, OnDestroy {\n  @Input() conversationUrl: string | undefined;\n  @Input() selectedMediaDevices: SelectedMediaDevices | undefined;\n  @Input() translations: { [key: string]: string } = {};\n  @Output() streamStart: EventEmitter<null> = new EventEmitter();\n  @Output() streamEnd: EventEmitter<null> = new EventEmitter();\n  @Output() checkMediaPermissions: EventEmitter<null> = new EventEmitter();\n\n  callObject: DailyCall | undefined;\n  avatarParticipant: Participant | undefined;\n  candidateJoined = signal(false);\n  private cdr = inject(ChangeDetectorRef);\n\n  ngOnInit(): void {\n    void this.setupCall();\n  }\n\n  ngOnDestroy(): void {\n    if (!this.callObject) return;\n    this.leaveCall();\n    this.callObject\n      .off('joined-meeting', this.candidateJoinMeeting)\n      .off('participant-joined', this.participantJoined)\n      .off('app-message', this.handleNewMessage)\n      .off('track-started', this.handleTrackStartedStopped)\n      .off('track-stopped', this.handleTrackStartedStopped)\n      .off('participant-left', this.handleParticipantLeft)\n      .off('left-meeting', this.handleLeftMeeting)\n      .off('error', this.handleError);\n  }\n\n  private async setupCall() {\n    this.callObject = DailyIframe.getCallInstance();\n    if (!this.callObject) {\n      this.callObject = DailyIframe.createCallObject();\n    }\n    if (this.selectedMediaDevices) {\n      await this.callObject.setInputDevicesAsync({\n        videoDeviceId: this.selectedMediaDevices.videoDeviceId,\n        audioDeviceId: this.selectedMediaDevices.audioDeviceId,\n      });\n    }\n    this.callObject.startRecording({\n      type: 'cloud',\n    });\n\n    this.callObject\n      .on('joined-meeting', this.candidateJoinMeeting)\n      .on('participant-joined', this.participantJoined)\n      .on('app-message', this.handleNewMessage)\n      .on('track-started', this.handleTrackStartedStopped)\n      .on('track-stopped', this.handleTrackStartedStopped)\n      .on('participant-left', this.handleParticipantLeft)\n      .on('left-meeting', this.handleLeftMeeting)\n      .on('error', this.handleError);\n\n    await this.callObject.join({\n      userName: 'Candidate',\n      url: this.conversationUrl,\n    });\n  }\n\n  formatParticipantObj(participant: DailyParticipant): Participant {\n    const { video, audio } = participant.tracks;\n\n    const videoTrack = video?.persistentTrack;\n    const audioTrack = audio?.persistentTrack;\n    return {\n      videoTrack: videoTrack,\n      audioTrack: audioTrack,\n      videoReady: !!(\n        videoTrack &&\n        (video.state === PLAYABLE_STATE || video.state === LOADING_STATE)\n      ),\n      audioReady: !!(\n        audioTrack &&\n        (audio.state === PLAYABLE_STATE || audio.state === LOADING_STATE)\n      ),\n      userName: participant.user_name,\n      local: participant.local,\n      sessionId: participant.session_id,\n    };\n  }\n\n  updateTrack(participant: DailyParticipant, newTrackType: string): void {\n    if (\n      !this.avatarParticipant ||\n      this.avatarParticipant.sessionId !== participant.session_id\n    ) {\n      return;\n    }\n    const existingParticipant = this.avatarParticipant as Participant;\n    const currentParticipantCopy = this.formatParticipantObj(participant);\n\n    if (newTrackType === 'video') {\n      if (\n        existingParticipant.videoReady !== currentParticipantCopy.videoReady\n      ) {\n        existingParticipant.videoReady = currentParticipantCopy.videoReady;\n      }\n\n      if (\n        currentParticipantCopy.videoReady &&\n        existingParticipant.videoTrack?.id !==\n          currentParticipantCopy.videoTrack?.id\n      ) {\n        existingParticipant.videoTrack = currentParticipantCopy.videoTrack;\n      }\n      return;\n    }\n\n    if (newTrackType === 'audio') {\n      if (\n        existingParticipant.audioReady !== currentParticipantCopy.audioReady\n      ) {\n        existingParticipant.audioReady = currentParticipantCopy.audioReady;\n      }\n\n      if (\n        currentParticipantCopy.audioReady &&\n        existingParticipant.audioTrack?.id !==\n          currentParticipantCopy.audioTrack?.id\n      ) {\n        existingParticipant.audioTrack = currentParticipantCopy.audioTrack;\n      }\n    }\n  }\n\n  private candidateJoinMeeting = (\n    event: DailyEventObjectParticipants | undefined\n  ): void => {\n    if (!event || !this.callObject) return;\n    this.candidateJoined.set(true);\n    this.streamStart.emit();\n  };\n\n  private participantJoined = (\n    event: DailyEventObjectParticipant | undefined\n  ) => {\n    if (!event) return;\n    this.avatarParticipant = this.formatParticipantObj(event.participant);\n  };\n\n  private handleTrackStartedStopped = (\n    event: DailyEventObjectTrack | undefined\n  ): void => {\n    if (!event || !event.participant || !this.candidateJoined()) return;\n    if (event.action === 'track-stopped') {\n      this.checkMediaPermissions.emit();\n    }\n    this.updateTrack(event.participant, event.type);\n    this.cdr.detectChanges();\n  };\n\n  private handleParticipantLeft = (\n    event: DailyEventObjectParticipantLeft | undefined\n  ): void => {\n    if (!event) return;\n    this.leaveCall();\n  };\n\n  private handleError = (\n    event: DailyEventObjectFatalError | undefined\n  ): void => {\n    if (!event) return;\n    console.error('Interview stream error', event);\n    this.leaveCall();\n  };\n\n  private handleLeftMeeting = (\n    event: DailyEventObjectNoPayload | undefined\n  ): void => {\n    this.callObject?.stopRecording();\n    if (!event || !this.callObject) return;\n    this.candidateJoined.set(false);\n    this.callObject.destroy();\n    this.streamEnd.emit();\n  };\n\n  private leaveCall(): void {\n    if (!this.callObject) return;\n    this.callObject.leave();\n  }\n\n  private handleNewMessage = (\n    event: DailyEventObjectAppMessage<EventData> | undefined\n  ): void => {\n    if (!event) return;\n    if (event.data.event_type === 'conversation.tool_call') {\n      this.handleToolCall(event.data.properties);\n    }\n  };\n\n  private getConversationId(): string | undefined {\n    return this.conversationUrl?.replace('https://tavus.daily.co/', '');\n  }\n\n  /*\n    This is a test implementation of tool calling.\n    These events will only be triggered if configured with a Tavus persona. The message content will be further refined by the IP Team.\n    https://docs.tavus.io/sections/event-schemas/conversation-toolcall\n  */\n  private handleToolCall = (properties: {\n    name: string;\n    arguments: string;\n  }): void => {\n    switch (properties.name) {\n      case 'next_question':\n        try {\n          const args = JSON.parse(properties.arguments);\n          if (args?.questionsLeft > 0) {\n            this.sendMessage(this.getToolCallTranslation('NEXT_QUESTION'));\n            window.setTimeout(() => {\n              this.sendMessage('Read next question', 'respond');\n            }, 1000);\n          } else {\n            throw new Error('No more questions left');\n          }\n        } catch (err) {\n          console.error(\n            'Failed to parse arguments for next_question tool call',\n            err\n          );\n\n          this.sendMessage(this.getToolCallTranslation('ALL_FOR_TODAY'));\n          window.setTimeout(() => {\n            this.leaveCall();\n          }, 5000);\n        }\n        break;\n      case 'end_conversation':\n        window.setTimeout(() => {\n          this.leaveCall();\n        }, 5000);\n        break;\n      default:\n        console.warn('Unknown tool call code:', properties);\n        break;\n    }\n  };\n\n  /*\n    Echo message is a message that avatar reads and candidate can hear\n    Respond message is a message that avatar reads and candidate cannot hear\n  */\n\n  private sendMessage(text: string, type: 'echo' | 'respond' = 'echo'): void {\n    if (!this.callObject) return;\n\n    this.callObject.sendAppMessage({\n      message_type: 'conversation',\n      event_type: 'conversation.' + type,\n      conversation_id: this.getConversationId(),\n      properties: {\n        text,\n      },\n    });\n  }\n\n  private getToolCallTranslation(key: string) {\n    const toolCallTranslations = this.translations['TOOL_CALL'] as unknown as {\n      [key: string]: string;\n    };\n    return toolCallTranslations[key];\n  }\n}\n\n","<div class=\"interview-stream\">\n  @if (avatarParticipant) {\n  <tgo-interview-video\n    [videoTrack]=\"avatarParticipant.videoTrack\"\n    [audioTrack]=\"avatarParticipant.audioTrack\"\n  ></tgo-interview-video>\n  }\n</div>\n\n"]}