@testgorilla/tgo-coding-test 1.0.0 → 2.0.1
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/fesm2022/testgorilla-tgo-coding-test.mjs +94 -94
- package/fesm2022/testgorilla-tgo-coding-test.mjs.map +1 -1
- package/lib/components/configurations/configurations.component.d.ts +1 -1
- package/package.json +15 -17
- package/esm2022/index.mjs +0 -12
- package/esm2022/lib/components/code-editor/code-editor.component.mjs +0 -335
- package/esm2022/lib/components/code-editor/code-editor.service.mjs +0 -72
- package/esm2022/lib/components/code-editor/helpers/c-helper.mjs +0 -40
- package/esm2022/lib/components/code-editor/helpers/code-editor-helper.base.mjs +0 -11
- package/esm2022/lib/components/code-editor/helpers/code-editor-helper.model.mjs +0 -2
- package/esm2022/lib/components/code-editor/helpers/cpp-helper.mjs +0 -41
- package/esm2022/lib/components/code-editor/helpers/csharp-helper.mjs +0 -42
- package/esm2022/lib/components/code-editor/helpers/go-helper.mjs +0 -42
- package/esm2022/lib/components/code-editor/helpers/index.mjs +0 -16
- package/esm2022/lib/components/code-editor/helpers/java-helper.mjs +0 -42
- package/esm2022/lib/components/code-editor/helpers/javascript-helper.mjs +0 -26
- package/esm2022/lib/components/code-editor/helpers/kotlin-helper.mjs +0 -42
- package/esm2022/lib/components/code-editor/helpers/php-helper.mjs +0 -26
- package/esm2022/lib/components/code-editor/helpers/python-helper.mjs +0 -26
- package/esm2022/lib/components/code-editor/helpers/r-helper.mjs +0 -26
- package/esm2022/lib/components/code-editor/helpers/ruby-helper.mjs +0 -26
- package/esm2022/lib/components/code-editor/helpers/scala-helper.mjs +0 -41
- package/esm2022/lib/components/code-editor/helpers/sql-helper.mjs +0 -34
- package/esm2022/lib/components/code-editor/helpers/swift-helper.mjs +0 -40
- package/esm2022/lib/components/code-editor/helpers/typescript-helper.mjs +0 -41
- package/esm2022/lib/components/code-editor/models/code-editor.model.mjs +0 -2
- package/esm2022/lib/components/code-editor/models/coding-snapshot.model.mjs +0 -2
- package/esm2022/lib/components/coding-question/coding-question.component.mjs +0 -126
- package/esm2022/lib/components/coding-section/coding-section.component.mjs +0 -188
- package/esm2022/lib/components/common/truncated-text/truncated-text.component.mjs +0 -38
- package/esm2022/lib/components/configurations/configurations.component.mjs +0 -97
- package/esm2022/lib/components/instructions/instructions.component.mjs +0 -139
- package/esm2022/lib/components/panel/panel.component.mjs +0 -34
- package/esm2022/lib/components/runnable-editor/runnable-editor.component.mjs +0 -169
- package/esm2022/lib/components/tests/test-cases/test-cases.component.mjs +0 -198
- package/esm2022/lib/components/tests/test-cases-content/test-cases-content.component.mjs +0 -96
- package/esm2022/lib/components/tests/test-cases-status/test-cases-status.component.mjs +0 -21
- package/esm2022/lib/components/tests/test-results.component.mjs +0 -127
- package/esm2022/lib/components/tgo-coding-test/tgo-coding-test.component.mjs +0 -280
- package/esm2022/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.mjs +0 -476
- package/esm2022/lib/config/index.mjs +0 -2
- package/esm2022/lib/config/tgo-coding-test.config.mjs +0 -2
- package/esm2022/lib/config/tgo-coding-test.provider.mjs +0 -34
- package/esm2022/lib/config/tgo-coding-test.token.mjs +0 -14
- package/esm2022/lib/models/auto-saved-data.mjs +0 -2
- package/esm2022/lib/models/code-event.mjs +0 -2
- package/esm2022/lib/models/coderunner-execution-results.mjs +0 -2
- package/esm2022/lib/models/configs.mjs +0 -2
- package/esm2022/lib/models/language-change-action.mjs +0 -2
- package/esm2022/lib/models/lat-languages.mjs +0 -3
- package/esm2022/lib/models/mixpanel-events.mjs +0 -2
- package/esm2022/lib/models/mode.mjs +0 -2
- package/esm2022/lib/models/paste-data.mjs +0 -2
- package/esm2022/lib/models/programming-language.mjs +0 -2
- package/esm2022/lib/models/test-cases.mjs +0 -7
- package/esm2022/lib/models/theme.mjs +0 -2
- package/esm2022/lib/models/translations.mjs +0 -2
- package/esm2022/lib/models/view-mode.mjs +0 -8
- package/esm2022/lib/pipes/memoize-func.pipe.mjs +0 -39
- package/esm2022/lib/services/candidate-coding-test-services/candidature-api.service.mjs +0 -19
- package/esm2022/lib/services/candidate-coding-test-services/coderunner-api.service.mjs +0 -58
- package/esm2022/lib/services/candidate-coding-test-services/coding-test-tour.service.mjs +0 -89
- package/esm2022/lib/services/candidate-coding-test-services/coding-test.service.mjs +0 -490
- package/esm2022/lib/services/candidate-coding-test-services/index.mjs +0 -5
- package/esm2022/lib/services/coding-test-config.service.mjs +0 -51
- package/esm2022/lib/services/configurations.service.mjs +0 -89
- package/esm2022/lib/services/lib-coding-test.service.mjs +0 -106
- package/esm2022/lib/services/storage.service.mjs +0 -624
- package/esm2022/lib/services/test-cases.service.mjs +0 -30
- package/esm2022/lib/services/theme.service.mjs +0 -36
- package/esm2022/lib/utils/additional-languages/erlang.mjs +0 -103
- package/esm2022/lib/utils/resize-element.mjs +0 -13
- package/esm2022/lib/utils/time-to-ms.util.mjs +0 -11
- package/esm2022/shared/index.mjs +0 -5
- package/esm2022/shared/lib/components/audio-animation/audio-animation.component.mjs +0 -114
- package/esm2022/shared/lib/components/audio-animation/index.mjs +0 -2
- package/esm2022/shared/lib/components/index.mjs +0 -3
- package/esm2022/shared/lib/components/vimeo-video/index.mjs +0 -2
- package/esm2022/shared/lib/components/vimeo-video/vimeo-video.component.mjs +0 -101
- package/esm2022/shared/lib/models/answer.mjs +0 -2
- package/esm2022/shared/lib/models/assessment.mjs +0 -2
- package/esm2022/shared/lib/models/environment.mjs +0 -2
- package/esm2022/shared/lib/models/index.mjs +0 -9
- package/esm2022/shared/lib/models/question-component.mjs +0 -2
- package/esm2022/shared/lib/models/question.mjs +0 -2
- package/esm2022/shared/lib/models/test.mjs +0 -2
- package/esm2022/shared/lib/models/translations.mjs +0 -2
- package/esm2022/shared/lib/models/window.mjs +0 -2
- package/esm2022/shared/lib/services/api/api.service.mjs +0 -97
- package/esm2022/shared/lib/services/api/mocked-api.service.mjs +0 -131
- package/esm2022/shared/lib/services/environment/environment.service.mjs +0 -13
- package/esm2022/shared/lib/services/index.mjs +0 -10
- package/esm2022/shared/lib/services/localization/languages.model.mjs +0 -19
- package/esm2022/shared/lib/services/localization/transloco-lazy-module-utils.mjs +0 -27
- package/esm2022/shared/lib/services/localization/transloco-testing.module.mjs +0 -11
- package/esm2022/shared/lib/services/media/media.service.mjs +0 -129
- package/esm2022/shared/lib/services/mixpanel/mixpanel.service.mjs +0 -30
- package/esm2022/shared/lib/services/theme/theme.service.mjs +0 -24
- package/esm2022/shared/test-mocks/assessment-test.mock.mjs +0 -112
- package/esm2022/shared/test-mocks/index.mjs +0 -3
- package/esm2022/shared/test-mocks/tgo-ui.mock.mjs +0 -39
- package/esm2022/testgorilla-tgo-coding-test.mjs +0 -5
|
@@ -1,198 +0,0 @@
|
|
|
1
|
-
import { __decorate, __metadata } from "tslib";
|
|
2
|
-
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, ViewChild, } from '@angular/core';
|
|
3
|
-
import { CommonModule } from '@angular/common';
|
|
4
|
-
import { MatTabGroup, MatTabsModule } from '@angular/material/tabs';
|
|
5
|
-
import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
|
|
6
|
-
import { filter } from 'rxjs';
|
|
7
|
-
import { IconComponentModule, ButtonComponentModule } from '@testgorilla/tgo-ui';
|
|
8
|
-
import { TestCaseStatus, } from '../../../models/test-cases';
|
|
9
|
-
import { LibCodingTestService } from '../../../services/lib-coding-test.service';
|
|
10
|
-
import { StorageCodingService } from '../../../services/storage.service';
|
|
11
|
-
import { TestCasesService } from '../../../services/test-cases.service';
|
|
12
|
-
import { TestCasesStatusComponent } from '../test-cases-status/test-cases-status.component';
|
|
13
|
-
import { TestCasesContentComponent } from '../test-cases-content/test-cases-content.component';
|
|
14
|
-
import * as i0 from "@angular/core";
|
|
15
|
-
import * as i1 from "../../../services/lib-coding-test.service";
|
|
16
|
-
import * as i2 from "../../../services/test-cases.service";
|
|
17
|
-
import * as i3 from "../../../services/storage.service";
|
|
18
|
-
import * as i4 from "@angular/common";
|
|
19
|
-
import * as i5 from "@angular/material/tabs";
|
|
20
|
-
import * as i6 from "@testgorilla/tgo-ui";
|
|
21
|
-
let TestCasesComponent = class TestCasesComponent {
|
|
22
|
-
libCodingTestService;
|
|
23
|
-
changeDetectorRef;
|
|
24
|
-
testCasesService;
|
|
25
|
-
StorageCodingService;
|
|
26
|
-
exampleTestCases;
|
|
27
|
-
apiTestCasesResult;
|
|
28
|
-
testCasesStatus;
|
|
29
|
-
testCases;
|
|
30
|
-
translations;
|
|
31
|
-
areTestsRun;
|
|
32
|
-
applicationTheme;
|
|
33
|
-
canAddCustomTestCases;
|
|
34
|
-
isSQL;
|
|
35
|
-
tabGroup;
|
|
36
|
-
testCasesResult;
|
|
37
|
-
activeTestCaseIndex;
|
|
38
|
-
maxTestCasesAllowed = 6; //Could be taken from backend
|
|
39
|
-
isAddTestCasesDisabled;
|
|
40
|
-
isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
|
|
41
|
-
isContentViewOpened;
|
|
42
|
-
constructor(libCodingTestService, changeDetectorRef, testCasesService, StorageCodingService) {
|
|
43
|
-
this.libCodingTestService = libCodingTestService;
|
|
44
|
-
this.changeDetectorRef = changeDetectorRef;
|
|
45
|
-
this.testCasesService = testCasesService;
|
|
46
|
-
this.StorageCodingService = StorageCodingService;
|
|
47
|
-
}
|
|
48
|
-
ngOnInit() {
|
|
49
|
-
this.testCasesService.activeTestCaseIndex$.pipe(untilDestroyed(this)).subscribe(index => {
|
|
50
|
-
this.activeTestCaseIndex = index;
|
|
51
|
-
});
|
|
52
|
-
this.setDefaultTestCases();
|
|
53
|
-
if (this.canAddCustomTestCases) {
|
|
54
|
-
this.updateAddCustomTestCaseDisabled();
|
|
55
|
-
this.listenTestCaseDeletion();
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
ngOnChanges({ apiTestCasesResult }) {
|
|
59
|
-
if (!apiTestCasesResult) {
|
|
60
|
-
return;
|
|
61
|
-
}
|
|
62
|
-
const apiTestCases = apiTestCasesResult.currentValue;
|
|
63
|
-
if (apiTestCases) {
|
|
64
|
-
this.testCasesResult = apiTestCases.length ? this.handleEmptyTestCases(apiTestCases) : [];
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
setDefaultTestCases() {
|
|
68
|
-
if (!this.testCases.length) {
|
|
69
|
-
this.testCases = [...(this.exampleTestCases || [])];
|
|
70
|
-
this.onTestCasesChanged();
|
|
71
|
-
this.setActiveTestCaseIndex(this.isMobile ? null : 0);
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
addTestCase() {
|
|
75
|
-
const testCaseLocalId = this.testCasesService.createTestCaseLocalId();
|
|
76
|
-
this.testCases.push({
|
|
77
|
-
localId: testCaseLocalId,
|
|
78
|
-
name: `Test ${testCaseLocalId}`,
|
|
79
|
-
input: '',
|
|
80
|
-
expectedOutput: '',
|
|
81
|
-
preloaded: false,
|
|
82
|
-
});
|
|
83
|
-
this.onTestCasesChanged();
|
|
84
|
-
this.testCasesService.setActiveTestCase(this.testCases.length - 1);
|
|
85
|
-
this.isContentViewOpened = this.isMobile;
|
|
86
|
-
this.updateAddCustomTestCaseDisabled();
|
|
87
|
-
if (!this.isMobile) {
|
|
88
|
-
this.setFocus();
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
removeTestCase(index) {
|
|
92
|
-
this.libCodingTestService.handleTestCaseDeletion(index);
|
|
93
|
-
}
|
|
94
|
-
modifyTestCase(index, modifiedValue) {
|
|
95
|
-
if (!this.testCases[index].preloaded) {
|
|
96
|
-
this.testCases[index].input = modifiedValue.input;
|
|
97
|
-
this.testCases[index].expectedOutput = modifiedValue.expectedOutput;
|
|
98
|
-
this.onTestCasesChanged();
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
setFocus() {
|
|
102
|
-
const tabHeaderId = `mat-tab-label-${this.tabGroup['_groupId']}-${this.activeTestCaseIndex}`;
|
|
103
|
-
const tabHeaderElement = document.getElementById(tabHeaderId);
|
|
104
|
-
if (tabHeaderElement) {
|
|
105
|
-
tabHeaderElement.focus();
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
onTestCasesChanged() {
|
|
109
|
-
this.libCodingTestService.updateTestCases([...this.testCases]);
|
|
110
|
-
this.StorageCodingService.updateTestCases([...this.testCases]);
|
|
111
|
-
}
|
|
112
|
-
setActiveTestCaseIndex(index) {
|
|
113
|
-
this.testCasesService.setActiveTestCase(index);
|
|
114
|
-
this.isContentViewOpened = index !== null ? this.isMobile : false;
|
|
115
|
-
}
|
|
116
|
-
closeTestCaseMobileView() {
|
|
117
|
-
this.isContentViewOpened = false;
|
|
118
|
-
this.testCasesService.setActiveTestCase(null);
|
|
119
|
-
}
|
|
120
|
-
listenTestCaseDeletion() {
|
|
121
|
-
this.libCodingTestService.testCaseDeletion$
|
|
122
|
-
.pipe(filter(({ action }) => action === "confirm" /* TestCaseDeletionAction.Confirm */), untilDestroyed(this))
|
|
123
|
-
.subscribe(({ index }) => {
|
|
124
|
-
this.testCases.splice(index, 1);
|
|
125
|
-
this.testCasesResult.splice(index, 1);
|
|
126
|
-
this.onTestCasesChanged();
|
|
127
|
-
if (this.isMobile) {
|
|
128
|
-
this.closeTestCaseMobileView();
|
|
129
|
-
}
|
|
130
|
-
else {
|
|
131
|
-
this.testCasesService.setActiveTestCase(index - 1);
|
|
132
|
-
}
|
|
133
|
-
this.updateAddCustomTestCaseDisabled();
|
|
134
|
-
this.changeDetectorRef.detectChanges();
|
|
135
|
-
});
|
|
136
|
-
}
|
|
137
|
-
handleEmptyTestCases(apiTestCasesResults) {
|
|
138
|
-
const testCasesResult = [...apiTestCasesResults];
|
|
139
|
-
this.testCases.forEach(({ input, expectedOutput }, index) => {
|
|
140
|
-
const isEmpty = this.isSQL ? !expectedOutput : !input || !expectedOutput;
|
|
141
|
-
if (isEmpty) {
|
|
142
|
-
testCasesResult.splice(index, 0, {
|
|
143
|
-
actualOutput: null,
|
|
144
|
-
logs: null,
|
|
145
|
-
status: TestCaseStatus.Empty,
|
|
146
|
-
});
|
|
147
|
-
}
|
|
148
|
-
});
|
|
149
|
-
return testCasesResult;
|
|
150
|
-
}
|
|
151
|
-
updateAddCustomTestCaseDisabled() {
|
|
152
|
-
const exampleTestCasesCount = this.exampleTestCases?.length || 0;
|
|
153
|
-
this.isAddTestCasesDisabled = this.testCases.length - exampleTestCasesCount < this.maxTestCasesAllowed;
|
|
154
|
-
}
|
|
155
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TestCasesComponent, deps: [{ token: i1.LibCodingTestService }, { token: i0.ChangeDetectorRef }, { token: i2.TestCasesService }, { token: i3.StorageCodingService }], target: i0.ɵɵFactoryTarget.Component });
|
|
156
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TestCasesComponent, isStandalone: true, selector: "tgo-test-cases", inputs: { exampleTestCases: "exampleTestCases", apiTestCasesResult: "apiTestCasesResult", testCasesStatus: "testCasesStatus", testCases: "testCases", translations: "translations", areTestsRun: "areTestsRun", applicationTheme: "applicationTheme", canAddCustomTestCases: "canAddCustomTestCases", isSQL: "isSQL" }, viewQueries: [{ propertyName: "tabGroup", first: true, predicate: ["tabGroup"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"test-case-container\">\n <div\n id=\"test-cases\"\n class=\"test-case-tabs-wrapper\"\n [ngClass]=\"{\n 'test-case-tabs-wrapper-tests-run': areTestsRun,\n 'test-case-tabs-wrapper-mobile': isMobile,\n 'test-case-tabs-wrapper-hidden': isContentViewOpened\n }\"\n >\n @if (!isMobile) {\n <mat-tab-group\n #tabGroup\n class=\"test-cases-tabs\"\n [selectedIndex]=\"activeTestCaseIndex\"\n (selectedIndexChange)=\"setActiveTestCaseIndex($event)\"\n >\n @for (testCase of testCases; track testCase; let index = $index) {\n <mat-tab label=\"{{ testCase.name }}\">\n <ng-template mat-tab-label>\n <div class=\"tab-label-wrapper\">\n <ng-container\n [ngTemplateOutlet]=\"testCaseHeading\"\n [ngTemplateOutletContext]=\"{ testCase: testCase, index: index }\"\n ></ng-container>\n </div>\n </ng-template>\n </mat-tab>\n }\n </mat-tab-group>\n } @else {\n <div class=\"test-cases-tabs test-cases-tabs-mobile\">\n @for (testCase of testCases; track testCase; let index = $index) {\n <div class=\"tab-label-wrapper\" (click)=\"setActiveTestCaseIndex(index)\">\n <div class=\"test-case-name-status-wrapper\">\n <ng-container\n [ngTemplateOutlet]=\"testCaseHeading\"\n [ngTemplateOutletContext]=\"{ testCase: testCase, index: index }\"\n ></ng-container>\n </div>\n @if (isMobile) {\n <ui-icon [name]=\"'Arrow-chevron-right-filled'\" [size]=\"'24'\" class=\"themed-button\"></ui-icon>\n }\n </div>\n }\n </div>\n } @if (canAddCustomTestCases) {\n <ui-button\n class=\"add-test-case themed-button\"\n [variant]=\"'tertiary'\"\n [iconName]=\"'Plus'\"\n [iconPosition]=\"'left'\"\n [disabled]=\"!isAddTestCasesDisabled\"\n [fullWidth]=\"true\"\n [tooltip]=\"!isAddTestCasesDisabled ? translations['TOOLTIPS']['CANT_ADD_TEST_CASE'] : null\"\n [label]=\"\n translations['TEST_RESULTS']['TEST_CASES']['ADD_TEST_CASE'] +\n ' (' +\n (testCases?.length - exampleTestCases?.length) +\n '/' +\n maxTestCasesAllowed +\n ')'\n \"\n (buttonClickEvent)=\"addTestCase()\"\n >\n </ui-button>\n }\n </div>\n\n @if (!isMobile) {\n <div class=\"test-case-content-wrapper\">\n <ng-container [ngTemplateOutlet]=\"testCasesContentTpl\"></ng-container>\n </div>\n } @if (isMobile && isContentViewOpened) {\n <div class=\"test-case-content-wrapper-mobile\">\n <div class=\"test-case-mobile-name-wrapper\">\n <ui-button\n [variant]=\"'icon-button'\"\n [iconName]=\"'Arrow-chevron-left-filled'\"\n [tooltip]=\"''\"\n [size]=\"'small'\"\n (buttonClickEvent)=\"closeTestCaseMobileView()\"\n class=\"themed-button\"\n >\n </ui-button>\n <ng-container\n [ngTemplateOutlet]=\"testCaseHeading\"\n [ngTemplateOutletContext]=\"{\n testCase: testCases[activeTestCaseIndex],\n index: activeTestCaseIndex,\n hideStatus: true\n }\"\n ></ng-container>\n </div>\n <ng-container [ngTemplateOutlet]=\"testCasesContentTpl\"></ng-container>\n </div>\n }\n</div>\n\n<ng-template #testCaseHeading let-testCase=\"testCase\" let-index=\"index\" let-hideStatus=\"hideStatus\">\n <div class=\"tab-label\">\n @if (testCase.preloaded) {\n <ui-icon [name]=\"'Lock'\" class=\"preloaded-icon\"></ui-icon>\n }\n <span class=\"tab-label-value\">{{ testCase.name }}</span>\n </div>\n\n @if (!hideStatus && testCasesResult?.length) {\n <tgo-test-cases-status\n [status]=\"testCasesResult?.[index]?.status\"\n [translations]=\"translations['TEST_RESULTS']['TEST_CASES']['STATUSES']\"\n ></tgo-test-cases-status>\n } @if (!testCase.preloaded) {\n <ui-button\n [variant]=\"'icon-button'\"\n [iconName]=\"'Delete'\"\n [tooltip]=\"translations['TOOLTIPS']['DELETE_TEST_CASE'] || ''\"\n (buttonClickEvent)=\"removeTestCase(index)\"\n class=\"delete-test-case themed-button\"\n >\n </ui-button>\n }\n</ng-template>\n\n<ng-template #testCasesContentTpl>\n <tgo-test-cases-content\n [testCase]=\"testCases[activeTestCaseIndex]\"\n [testCaseResult]=\"testCasesResult?.[activeTestCaseIndex]\"\n [areTestsRun]=\"areTestsRun\"\n [translations]=\"translations['TEST_RESULTS']['TEST_CASES']\"\n [isSQL]=\"isSQL\"\n [applicationTheme]=\"applicationTheme\"\n (valueChange)=\"modifyTestCase(activeTestCaseIndex, $event)\"\n ></tgo-test-cases-content>\n</ng-template>\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}.test-case-container{display:flex;height:100%}.test-case-container .test-case-tabs-wrapper{display:flex;flex-direction:column;position:relative;overflow-y:auto;border-right:1px solid var(--border-bg-color);height:100%;min-width:230px}.test-case-container .test-case-tabs-wrapper-mobile{flex:1}.test-case-container .test-case-tabs-wrapper-mobile.test-case-tabs-wrapper-hidden{display:none}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab .mdc-tab-indicator{width:3px}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab .mdc-tab-indicator .mdc-tab-indicator__content{border-left:3px solid transparent;height:100%}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab:hover .delete-test-case{display:block}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab.mdc-tab--active .tab-label-value{font-weight:600}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab.mdc-tab--active .mdc-tab-indicator__content{border-left-color:var(--company-color, #D410AA)}.test-case-container .test-case-tabs-wrapper .test-cases-tabs{flex-direction:row;flex-grow:1}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__content{justify-content:flex-start;padding:14px 16px;width:100%}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__content .mdc-tab__text-label{flex-grow:1}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__content .tab-label-wrapper{display:flex;align-items:center;justify-content:space-between;width:100%}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mat-mdc-tab-labels{flex-direction:column}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mat-mdc-tab-header{width:100%;border-bottom:none}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab{height:48px;padding:0;border-bottom:1px solid var(--border-bg-color);color:var(--secondary-text-color)}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__text-label{width:100%}.test-case-container .test-case-tabs-wrapper .test-cases-tabs.mat-mdc-tab-group.mat-primary ::ng-deep .mat-ink-bar{background-color:transparent}.test-case-container .test-case-tabs-wrapper .test-cases-tabs-mobile .test-case-name-status-wrapper{display:flex;justify-content:space-between;flex:1}.test-case-container .test-case-tabs-wrapper .test-cases-tabs-mobile .tab-label-wrapper{display:flex;padding:14px 16px;cursor:pointer;border-bottom:1px solid var(--border-bg-color)}.test-case-container .test-case-tabs-wrapper .add-test-case{position:sticky;bottom:0;left:0;right:0;border-top:1px solid var(--border-bg-color);border-radius:0;z-index:1}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep mat-icon{color:var(--company-color, #D410AA)}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button{justify-content:flex-start;background-color:var(--bg-mat-card)}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button:disabled{opacity:1}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button:disabled .mdc-button__label{opacity:.75}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button span{color:var(--company-theme-specific-color, #D410AA)!important}.test-case-container .delete-test-case{display:none;position:absolute;right:16px;cursor:pointer}.test-case-container .delete-test-case ::ng-deep .icon-button:not(:hover){background-color:var(--bg-mat-card)!important}.test-case-container .delete-test-case ::ng-deep svg{color:var(--icon-bg-color)}.test-case-container ::ng-deep .mat-tab-label.cdk-program-focused .delete-test-case svg rect{fill:#484c4d00}.test-case-container .test-case-content-wrapper{flex-grow:1;overflow-y:auto}.test-case-container .preloaded-icon{margin-right:4px}.test-case-container .preloaded-icon ::ng-deep mat-icon svg{color:var(--icon-secondary-bg-color)}.test-case-container .tab-label{display:flex;align-items:center;color:var(--main-text-color)}.test-case-container .chevron-icon{margin-left:8px}.test-case-container .test-case-content-wrapper-mobile{overflow:auto}.test-case-container .test-case-mobile-name-wrapper{display:flex;position:relative;padding:14px 16px 0}.test-case-container .test-case-mobile-name-wrapper .chevron-icon{margin:0 4px 0 0}.test-case-container .test-case-mobile-name-wrapper .delete-test-case{display:block}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: MatTabsModule }, { kind: "directive", type: i5.MatTabLabel, selector: "[mat-tab-label], [matTabLabel]" }, { kind: "component", type: i5.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i5.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor", "aria-label", "aria-labelledby"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: IconComponentModule }, { kind: "component", type: i6.IconComponent, selector: "ui-icon", inputs: ["size", "cssClass", "name", "color", "filled", "toggleIconStyle", "applicationTheme", "useFullIconName"] }, { kind: "ngmodule", type: ButtonComponentModule }, { kind: "component", type: i6.ButtonComponent, selector: "ui-button", inputs: ["size", "variant", "label", "iconPosition", "justIcon", "iconName", "disabled", "loading", "fullWidth", "url", "urlTarget", "value", "tooltip", "isPremium", "type", "companyColor", "buttonBadgeConfig", "applicationTheme", "disabledScaleOnClick", "ariaLabel", "ariaRequired", "ariaLabelledby", "ariaDescribedby", "preventDefault", "hasBackground", "tooltipPosition", "role", "iconFilled"], outputs: ["buttonClickEvent", "buttonHoverEvent"] }, { kind: "component", type: TestCasesStatusComponent, selector: "tgo-test-cases-status", inputs: ["status", "translations"] }, { kind: "component", type: TestCasesContentComponent, selector: "tgo-test-cases-content", inputs: ["testCase", "testCaseResult", "areTestsRun", "translations", "applicationTheme", "isSQL"], outputs: ["valueChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
157
|
-
};
|
|
158
|
-
TestCasesComponent = __decorate([
|
|
159
|
-
UntilDestroy(),
|
|
160
|
-
__metadata("design:paramtypes", [LibCodingTestService,
|
|
161
|
-
ChangeDetectorRef,
|
|
162
|
-
TestCasesService,
|
|
163
|
-
StorageCodingService])
|
|
164
|
-
], TestCasesComponent);
|
|
165
|
-
export { TestCasesComponent };
|
|
166
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TestCasesComponent, decorators: [{
|
|
167
|
-
type: Component,
|
|
168
|
-
args: [{ standalone: true, selector: 'tgo-test-cases', changeDetection: ChangeDetectionStrategy.OnPush, imports: [
|
|
169
|
-
CommonModule,
|
|
170
|
-
MatTabsModule,
|
|
171
|
-
IconComponentModule,
|
|
172
|
-
ButtonComponentModule,
|
|
173
|
-
TestCasesStatusComponent,
|
|
174
|
-
TestCasesContentComponent,
|
|
175
|
-
], template: "<div class=\"test-case-container\">\n <div\n id=\"test-cases\"\n class=\"test-case-tabs-wrapper\"\n [ngClass]=\"{\n 'test-case-tabs-wrapper-tests-run': areTestsRun,\n 'test-case-tabs-wrapper-mobile': isMobile,\n 'test-case-tabs-wrapper-hidden': isContentViewOpened\n }\"\n >\n @if (!isMobile) {\n <mat-tab-group\n #tabGroup\n class=\"test-cases-tabs\"\n [selectedIndex]=\"activeTestCaseIndex\"\n (selectedIndexChange)=\"setActiveTestCaseIndex($event)\"\n >\n @for (testCase of testCases; track testCase; let index = $index) {\n <mat-tab label=\"{{ testCase.name }}\">\n <ng-template mat-tab-label>\n <div class=\"tab-label-wrapper\">\n <ng-container\n [ngTemplateOutlet]=\"testCaseHeading\"\n [ngTemplateOutletContext]=\"{ testCase: testCase, index: index }\"\n ></ng-container>\n </div>\n </ng-template>\n </mat-tab>\n }\n </mat-tab-group>\n } @else {\n <div class=\"test-cases-tabs test-cases-tabs-mobile\">\n @for (testCase of testCases; track testCase; let index = $index) {\n <div class=\"tab-label-wrapper\" (click)=\"setActiveTestCaseIndex(index)\">\n <div class=\"test-case-name-status-wrapper\">\n <ng-container\n [ngTemplateOutlet]=\"testCaseHeading\"\n [ngTemplateOutletContext]=\"{ testCase: testCase, index: index }\"\n ></ng-container>\n </div>\n @if (isMobile) {\n <ui-icon [name]=\"'Arrow-chevron-right-filled'\" [size]=\"'24'\" class=\"themed-button\"></ui-icon>\n }\n </div>\n }\n </div>\n } @if (canAddCustomTestCases) {\n <ui-button\n class=\"add-test-case themed-button\"\n [variant]=\"'tertiary'\"\n [iconName]=\"'Plus'\"\n [iconPosition]=\"'left'\"\n [disabled]=\"!isAddTestCasesDisabled\"\n [fullWidth]=\"true\"\n [tooltip]=\"!isAddTestCasesDisabled ? translations['TOOLTIPS']['CANT_ADD_TEST_CASE'] : null\"\n [label]=\"\n translations['TEST_RESULTS']['TEST_CASES']['ADD_TEST_CASE'] +\n ' (' +\n (testCases?.length - exampleTestCases?.length) +\n '/' +\n maxTestCasesAllowed +\n ')'\n \"\n (buttonClickEvent)=\"addTestCase()\"\n >\n </ui-button>\n }\n </div>\n\n @if (!isMobile) {\n <div class=\"test-case-content-wrapper\">\n <ng-container [ngTemplateOutlet]=\"testCasesContentTpl\"></ng-container>\n </div>\n } @if (isMobile && isContentViewOpened) {\n <div class=\"test-case-content-wrapper-mobile\">\n <div class=\"test-case-mobile-name-wrapper\">\n <ui-button\n [variant]=\"'icon-button'\"\n [iconName]=\"'Arrow-chevron-left-filled'\"\n [tooltip]=\"''\"\n [size]=\"'small'\"\n (buttonClickEvent)=\"closeTestCaseMobileView()\"\n class=\"themed-button\"\n >\n </ui-button>\n <ng-container\n [ngTemplateOutlet]=\"testCaseHeading\"\n [ngTemplateOutletContext]=\"{\n testCase: testCases[activeTestCaseIndex],\n index: activeTestCaseIndex,\n hideStatus: true\n }\"\n ></ng-container>\n </div>\n <ng-container [ngTemplateOutlet]=\"testCasesContentTpl\"></ng-container>\n </div>\n }\n</div>\n\n<ng-template #testCaseHeading let-testCase=\"testCase\" let-index=\"index\" let-hideStatus=\"hideStatus\">\n <div class=\"tab-label\">\n @if (testCase.preloaded) {\n <ui-icon [name]=\"'Lock'\" class=\"preloaded-icon\"></ui-icon>\n }\n <span class=\"tab-label-value\">{{ testCase.name }}</span>\n </div>\n\n @if (!hideStatus && testCasesResult?.length) {\n <tgo-test-cases-status\n [status]=\"testCasesResult?.[index]?.status\"\n [translations]=\"translations['TEST_RESULTS']['TEST_CASES']['STATUSES']\"\n ></tgo-test-cases-status>\n } @if (!testCase.preloaded) {\n <ui-button\n [variant]=\"'icon-button'\"\n [iconName]=\"'Delete'\"\n [tooltip]=\"translations['TOOLTIPS']['DELETE_TEST_CASE'] || ''\"\n (buttonClickEvent)=\"removeTestCase(index)\"\n class=\"delete-test-case themed-button\"\n >\n </ui-button>\n }\n</ng-template>\n\n<ng-template #testCasesContentTpl>\n <tgo-test-cases-content\n [testCase]=\"testCases[activeTestCaseIndex]\"\n [testCaseResult]=\"testCasesResult?.[activeTestCaseIndex]\"\n [areTestsRun]=\"areTestsRun\"\n [translations]=\"translations['TEST_RESULTS']['TEST_CASES']\"\n [isSQL]=\"isSQL\"\n [applicationTheme]=\"applicationTheme\"\n (valueChange)=\"modifyTestCase(activeTestCaseIndex, $event)\"\n ></tgo-test-cases-content>\n</ng-template>\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}.test-case-container{display:flex;height:100%}.test-case-container .test-case-tabs-wrapper{display:flex;flex-direction:column;position:relative;overflow-y:auto;border-right:1px solid var(--border-bg-color);height:100%;min-width:230px}.test-case-container .test-case-tabs-wrapper-mobile{flex:1}.test-case-container .test-case-tabs-wrapper-mobile.test-case-tabs-wrapper-hidden{display:none}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab .mdc-tab-indicator{width:3px}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab .mdc-tab-indicator .mdc-tab-indicator__content{border-left:3px solid transparent;height:100%}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab:hover .delete-test-case{display:block}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab.mdc-tab--active .tab-label-value{font-weight:600}.test-case-container .test-case-tabs-wrapper:not(.test-case-container .test-case-tabs-wrapper-mobile) ::ng-deep .mat-mdc-tab.mdc-tab--active .mdc-tab-indicator__content{border-left-color:var(--company-color, #D410AA)}.test-case-container .test-case-tabs-wrapper .test-cases-tabs{flex-direction:row;flex-grow:1}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__content{justify-content:flex-start;padding:14px 16px;width:100%}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__content .mdc-tab__text-label{flex-grow:1}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__content .tab-label-wrapper{display:flex;align-items:center;justify-content:space-between;width:100%}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mat-mdc-tab-labels{flex-direction:column}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mat-mdc-tab-header{width:100%;border-bottom:none}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab{height:48px;padding:0;border-bottom:1px solid var(--border-bg-color);color:var(--secondary-text-color)}.test-case-container .test-case-tabs-wrapper .test-cases-tabs ::ng-deep .mdc-tab__text-label{width:100%}.test-case-container .test-case-tabs-wrapper .test-cases-tabs.mat-mdc-tab-group.mat-primary ::ng-deep .mat-ink-bar{background-color:transparent}.test-case-container .test-case-tabs-wrapper .test-cases-tabs-mobile .test-case-name-status-wrapper{display:flex;justify-content:space-between;flex:1}.test-case-container .test-case-tabs-wrapper .test-cases-tabs-mobile .tab-label-wrapper{display:flex;padding:14px 16px;cursor:pointer;border-bottom:1px solid var(--border-bg-color)}.test-case-container .test-case-tabs-wrapper .add-test-case{position:sticky;bottom:0;left:0;right:0;border-top:1px solid var(--border-bg-color);border-radius:0;z-index:1}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep mat-icon{color:var(--company-color, #D410AA)}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button{justify-content:flex-start;background-color:var(--bg-mat-card)}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button:disabled{opacity:1}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button:disabled .mdc-button__label{opacity:.75}.test-case-container .test-case-tabs-wrapper .add-test-case ::ng-deep .mdc-button span{color:var(--company-theme-specific-color, #D410AA)!important}.test-case-container .delete-test-case{display:none;position:absolute;right:16px;cursor:pointer}.test-case-container .delete-test-case ::ng-deep .icon-button:not(:hover){background-color:var(--bg-mat-card)!important}.test-case-container .delete-test-case ::ng-deep svg{color:var(--icon-bg-color)}.test-case-container ::ng-deep .mat-tab-label.cdk-program-focused .delete-test-case svg rect{fill:#484c4d00}.test-case-container .test-case-content-wrapper{flex-grow:1;overflow-y:auto}.test-case-container .preloaded-icon{margin-right:4px}.test-case-container .preloaded-icon ::ng-deep mat-icon svg{color:var(--icon-secondary-bg-color)}.test-case-container .tab-label{display:flex;align-items:center;color:var(--main-text-color)}.test-case-container .chevron-icon{margin-left:8px}.test-case-container .test-case-content-wrapper-mobile{overflow:auto}.test-case-container .test-case-mobile-name-wrapper{display:flex;position:relative;padding:14px 16px 0}.test-case-container .test-case-mobile-name-wrapper .chevron-icon{margin:0 4px 0 0}.test-case-container .test-case-mobile-name-wrapper .delete-test-case{display:block}\n"] }]
|
|
176
|
-
}], ctorParameters: () => [{ type: i1.LibCodingTestService }, { type: i0.ChangeDetectorRef }, { type: i2.TestCasesService }, { type: i3.StorageCodingService }], propDecorators: { exampleTestCases: [{
|
|
177
|
-
type: Input
|
|
178
|
-
}], apiTestCasesResult: [{
|
|
179
|
-
type: Input
|
|
180
|
-
}], testCasesStatus: [{
|
|
181
|
-
type: Input
|
|
182
|
-
}], testCases: [{
|
|
183
|
-
type: Input
|
|
184
|
-
}], translations: [{
|
|
185
|
-
type: Input
|
|
186
|
-
}], areTestsRun: [{
|
|
187
|
-
type: Input
|
|
188
|
-
}], applicationTheme: [{
|
|
189
|
-
type: Input
|
|
190
|
-
}], canAddCustomTestCases: [{
|
|
191
|
-
type: Input
|
|
192
|
-
}], isSQL: [{
|
|
193
|
-
type: Input
|
|
194
|
-
}], tabGroup: [{
|
|
195
|
-
type: ViewChild,
|
|
196
|
-
args: ['tabGroup']
|
|
197
|
-
}] } });
|
|
198
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-cases.component.js","sourceRoot":"","sources":["../../../../../../../../packages/tgo-coding-test/src/lib/components/tests/test-cases/test-cases.component.ts","../../../../../../../../packages/tgo-coding-test/src/lib/components/tests/test-cases/test-cases.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  Input,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { MatTabGroup, MatTabsModule } from '@angular/material/tabs';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { filter } from 'rxjs';\nimport { ApplicationTheme, IconComponentModule, ButtonComponentModule } from '@testgorilla/tgo-ui';\nimport {\n  AllowedModificationFields,\n  CustomTestCase,\n  ExampleTestCase,\n  TestCaseDeletionAction,\n  TestCaseResult,\n  TestCaseStatus,\n} from '../../../models/test-cases';\nimport { LibCodingTestService } from '../../../services/lib-coding-test.service';\nimport { StorageCodingService } from '../../../services/storage.service';\nimport { TestCasesService } from '../../../services/test-cases.service';\nimport { TestCasesStatusComponent } from '../test-cases-status/test-cases-status.component';\nimport { TestCasesContentComponent } from '../test-cases-content/test-cases-content.component';\n\n@UntilDestroy()\n@Component({\n  standalone: true,\n  selector: 'tgo-test-cases',\n  templateUrl: 'test-cases.component.html',\n  styleUrls: ['test-cases.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CommonModule,\n    MatTabsModule,\n    IconComponentModule,\n    ButtonComponentModule,\n    TestCasesStatusComponent,\n    TestCasesContentComponent,\n  ],\n})\nexport class TestCasesComponent implements OnInit, OnChanges {\n  @Input() exampleTestCases: ExampleTestCase[];\n  @Input() apiTestCasesResult: TestCaseResult[];\n  @Input() testCasesStatus: any;\n  @Input() testCases: (CustomTestCase | ExampleTestCase)[];\n  @Input() translations: Record<string, unknown>;\n  @Input() areTestsRun: boolean;\n  @Input() applicationTheme: ApplicationTheme;\n  @Input() canAddCustomTestCases: boolean;\n  @Input() isSQL: boolean;\n\n  @ViewChild('tabGroup') tabGroup: MatTabGroup;\n\n  testCasesResult: TestCaseResult[];\n\n  activeTestCaseIndex: number | null;\n  maxTestCasesAllowed = 6; //Could be taken from backend\n\n  isAddTestCasesDisabled: boolean;\n  isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);\n\n  isContentViewOpened: boolean;\n\n  constructor(\n    private libCodingTestService: LibCodingTestService,\n    private changeDetectorRef: ChangeDetectorRef,\n    private testCasesService: TestCasesService,\n    private StorageCodingService: StorageCodingService\n  ) {}\n\n  ngOnInit() {\n    this.testCasesService.activeTestCaseIndex$.pipe(untilDestroyed(this)).subscribe(index => {\n      this.activeTestCaseIndex = index;\n    });\n\n    this.setDefaultTestCases();\n    if (this.canAddCustomTestCases) {\n      this.updateAddCustomTestCaseDisabled();\n      this.listenTestCaseDeletion();\n    }\n  }\n\n  ngOnChanges({ apiTestCasesResult }: SimpleChanges): void {\n    if (!apiTestCasesResult) {\n      return;\n    }\n    const apiTestCases: TestCaseResult[] = apiTestCasesResult.currentValue;\n\n    if (apiTestCases) {\n      this.testCasesResult = apiTestCases.length ? this.handleEmptyTestCases(apiTestCases) : [];\n    }\n  }\n\n  private setDefaultTestCases() {\n    if (!this.testCases.length) {\n      this.testCases = [...(this.exampleTestCases || [])];\n      this.onTestCasesChanged();\n      this.setActiveTestCaseIndex(this.isMobile ? null : 0);\n    }\n  }\n\n  addTestCase(): void {\n    const testCaseLocalId = this.testCasesService.createTestCaseLocalId();\n\n    this.testCases.push({\n      localId: testCaseLocalId,\n      name: `Test ${testCaseLocalId}`,\n      input: '',\n      expectedOutput: '',\n      preloaded: false,\n    });\n    this.onTestCasesChanged();\n    this.testCasesService.setActiveTestCase(this.testCases.length - 1);\n    this.isContentViewOpened = this.isMobile;\n    this.updateAddCustomTestCaseDisabled();\n\n    if (!this.isMobile) {\n      this.setFocus();\n    }\n  }\n\n  removeTestCase(index: number): void {\n    this.libCodingTestService.handleTestCaseDeletion(index);\n  }\n\n  modifyTestCase(index: number, modifiedValue: AllowedModificationFields): void {\n    if (!this.testCases[index].preloaded) {\n      this.testCases[index].input = modifiedValue.input;\n      this.testCases[index].expectedOutput = modifiedValue.expectedOutput;\n      this.onTestCasesChanged();\n    }\n  }\n\n  private setFocus() {\n    const tabHeaderId = `mat-tab-label-${this.tabGroup['_groupId']}-${this.activeTestCaseIndex}`;\n    const tabHeaderElement: HTMLElement = document.getElementById(tabHeaderId);\n    if (tabHeaderElement) {\n      tabHeaderElement.focus();\n    }\n  }\n\n  private onTestCasesChanged() {\n    this.libCodingTestService.updateTestCases([...this.testCases]);\n    this.StorageCodingService.updateTestCases([...this.testCases]);\n  }\n\n  setActiveTestCaseIndex(index: number): void {\n    this.testCasesService.setActiveTestCase(index);\n    this.isContentViewOpened = index !== null ? this.isMobile : false;\n  }\n\n  closeTestCaseMobileView(): void {\n    this.isContentViewOpened = false;\n    this.testCasesService.setActiveTestCase(null);\n  }\n\n  private listenTestCaseDeletion(): void {\n    this.libCodingTestService.testCaseDeletion$\n      .pipe(\n        filter(({ action }) => action === TestCaseDeletionAction.Confirm),\n        untilDestroyed(this)\n      )\n      .subscribe(({ index }) => {\n        this.testCases.splice(index, 1);\n        this.testCasesResult.splice(index, 1);\n        this.onTestCasesChanged();\n\n        if (this.isMobile) {\n          this.closeTestCaseMobileView();\n        } else {\n          this.testCasesService.setActiveTestCase(index - 1);\n        }\n\n        this.updateAddCustomTestCaseDisabled();\n        this.changeDetectorRef.detectChanges();\n      });\n  }\n\n  private handleEmptyTestCases(apiTestCasesResults: TestCaseResult[]): TestCaseResult[] {\n    const testCasesResult = [...apiTestCasesResults];\n\n    this.testCases.forEach(({ input, expectedOutput }, index) => {\n      const isEmpty = this.isSQL ? !expectedOutput : !input || !expectedOutput;\n      if (isEmpty) {\n        testCasesResult.splice(index, 0, {\n          actualOutput: null,\n          logs: null,\n          status: TestCaseStatus.Empty,\n        });\n      }\n    });\n\n    return testCasesResult;\n  }\n\n  private updateAddCustomTestCaseDisabled(): void {\n    const exampleTestCasesCount = this.exampleTestCases?.length || 0;\n    this.isAddTestCasesDisabled = this.testCases.length - exampleTestCasesCount < this.maxTestCasesAllowed;\n  }\n}\n","<div class=\"test-case-container\">\n  <div\n    id=\"test-cases\"\n    class=\"test-case-tabs-wrapper\"\n    [ngClass]=\"{\n      'test-case-tabs-wrapper-tests-run': areTestsRun,\n      'test-case-tabs-wrapper-mobile': isMobile,\n      'test-case-tabs-wrapper-hidden': isContentViewOpened\n    }\"\n  >\n    @if (!isMobile) {\n    <mat-tab-group\n      #tabGroup\n      class=\"test-cases-tabs\"\n      [selectedIndex]=\"activeTestCaseIndex\"\n      (selectedIndexChange)=\"setActiveTestCaseIndex($event)\"\n    >\n      @for (testCase of testCases; track testCase; let index = $index) {\n      <mat-tab label=\"{{ testCase.name }}\">\n        <ng-template mat-tab-label>\n          <div class=\"tab-label-wrapper\">\n            <ng-container\n              [ngTemplateOutlet]=\"testCaseHeading\"\n              [ngTemplateOutletContext]=\"{ testCase: testCase, index: index }\"\n            ></ng-container>\n          </div>\n        </ng-template>\n      </mat-tab>\n      }\n    </mat-tab-group>\n    } @else {\n    <div class=\"test-cases-tabs test-cases-tabs-mobile\">\n      @for (testCase of testCases; track testCase; let index = $index) {\n      <div class=\"tab-label-wrapper\" (click)=\"setActiveTestCaseIndex(index)\">\n        <div class=\"test-case-name-status-wrapper\">\n          <ng-container\n            [ngTemplateOutlet]=\"testCaseHeading\"\n            [ngTemplateOutletContext]=\"{ testCase: testCase, index: index }\"\n          ></ng-container>\n        </div>\n        @if (isMobile) {\n        <ui-icon [name]=\"'Arrow-chevron-right-filled'\" [size]=\"'24'\" class=\"themed-button\"></ui-icon>\n        }\n      </div>\n      }\n    </div>\n    } @if (canAddCustomTestCases) {\n    <ui-button\n      class=\"add-test-case themed-button\"\n      [variant]=\"'tertiary'\"\n      [iconName]=\"'Plus'\"\n      [iconPosition]=\"'left'\"\n      [disabled]=\"!isAddTestCasesDisabled\"\n      [fullWidth]=\"true\"\n      [tooltip]=\"!isAddTestCasesDisabled ? translations['TOOLTIPS']['CANT_ADD_TEST_CASE'] : null\"\n      [label]=\"\n        translations['TEST_RESULTS']['TEST_CASES']['ADD_TEST_CASE'] +\n        ' (' +\n        (testCases?.length - exampleTestCases?.length) +\n        '/' +\n        maxTestCasesAllowed +\n        ')'\n      \"\n      (buttonClickEvent)=\"addTestCase()\"\n    >\n    </ui-button>\n    }\n  </div>\n\n  @if (!isMobile) {\n  <div class=\"test-case-content-wrapper\">\n    <ng-container [ngTemplateOutlet]=\"testCasesContentTpl\"></ng-container>\n  </div>\n  } @if (isMobile && isContentViewOpened) {\n  <div class=\"test-case-content-wrapper-mobile\">\n    <div class=\"test-case-mobile-name-wrapper\">\n      <ui-button\n        [variant]=\"'icon-button'\"\n        [iconName]=\"'Arrow-chevron-left-filled'\"\n        [tooltip]=\"''\"\n        [size]=\"'small'\"\n        (buttonClickEvent)=\"closeTestCaseMobileView()\"\n        class=\"themed-button\"\n      >\n      </ui-button>\n      <ng-container\n        [ngTemplateOutlet]=\"testCaseHeading\"\n        [ngTemplateOutletContext]=\"{\n          testCase: testCases[activeTestCaseIndex],\n          index: activeTestCaseIndex,\n          hideStatus: true\n        }\"\n      ></ng-container>\n    </div>\n    <ng-container [ngTemplateOutlet]=\"testCasesContentTpl\"></ng-container>\n  </div>\n  }\n</div>\n\n<ng-template #testCaseHeading let-testCase=\"testCase\" let-index=\"index\" let-hideStatus=\"hideStatus\">\n  <div class=\"tab-label\">\n    @if (testCase.preloaded) {\n    <ui-icon [name]=\"'Lock'\" class=\"preloaded-icon\"></ui-icon>\n    }\n    <span class=\"tab-label-value\">{{ testCase.name }}</span>\n  </div>\n\n  @if (!hideStatus && testCasesResult?.length) {\n  <tgo-test-cases-status\n    [status]=\"testCasesResult?.[index]?.status\"\n    [translations]=\"translations['TEST_RESULTS']['TEST_CASES']['STATUSES']\"\n  ></tgo-test-cases-status>\n  } @if (!testCase.preloaded) {\n  <ui-button\n    [variant]=\"'icon-button'\"\n    [iconName]=\"'Delete'\"\n    [tooltip]=\"translations['TOOLTIPS']['DELETE_TEST_CASE'] || ''\"\n    (buttonClickEvent)=\"removeTestCase(index)\"\n    class=\"delete-test-case themed-button\"\n  >\n  </ui-button>\n  }\n</ng-template>\n\n<ng-template #testCasesContentTpl>\n  <tgo-test-cases-content\n    [testCase]=\"testCases[activeTestCaseIndex]\"\n    [testCaseResult]=\"testCasesResult?.[activeTestCaseIndex]\"\n    [areTestsRun]=\"areTestsRun\"\n    [translations]=\"translations['TEST_RESULTS']['TEST_CASES']\"\n    [isSQL]=\"isSQL\"\n    [applicationTheme]=\"applicationTheme\"\n    (valueChange)=\"modifyTestCase(activeTestCaseIndex, $event)\"\n  ></tgo-test-cases-content>\n</ng-template>\n"]}
|
|
@@ -1,96 +0,0 @@
|
|
|
1
|
-
import { __decorate } from "tslib";
|
|
2
|
-
import { ChangeDetectionStrategy, Component, EventEmitter, inject, Input, Output, } from '@angular/core';
|
|
3
|
-
import { CommonModule } from '@angular/common';
|
|
4
|
-
import { FormBuilder, FormControl, ReactiveFormsModule } from '@angular/forms';
|
|
5
|
-
import { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';
|
|
6
|
-
import { FieldComponentModule } from '@testgorilla/tgo-ui';
|
|
7
|
-
import { TestCaseStatus, } from '../../../models/test-cases';
|
|
8
|
-
import { MemoizeFuncPipe } from '../../../pipes/memoize-func.pipe';
|
|
9
|
-
import { TruncatedTextComponent } from '../../common/truncated-text/truncated-text.component';
|
|
10
|
-
import { TestCasesStatusComponent } from '../test-cases-status/test-cases-status.component';
|
|
11
|
-
import * as i0 from "@angular/core";
|
|
12
|
-
import * as i1 from "@angular/forms";
|
|
13
|
-
import * as i2 from "@testgorilla/tgo-ui";
|
|
14
|
-
let TestCasesContentComponent = class TestCasesContentComponent {
|
|
15
|
-
fb = inject(FormBuilder);
|
|
16
|
-
testCase;
|
|
17
|
-
testCaseResult;
|
|
18
|
-
areTestsRun;
|
|
19
|
-
translations;
|
|
20
|
-
applicationTheme;
|
|
21
|
-
isSQL;
|
|
22
|
-
valueChange = new EventEmitter();
|
|
23
|
-
isCustomTestCase;
|
|
24
|
-
// todo: add validators when backend implements validation on their side
|
|
25
|
-
form = this.fb.group({
|
|
26
|
-
input: new FormControl(''),
|
|
27
|
-
expectedOutput: new FormControl(''),
|
|
28
|
-
});
|
|
29
|
-
ngOnInit() {
|
|
30
|
-
if (this.testCase?.input && this.testCase?.expectedOutput) {
|
|
31
|
-
this.form.patchValue({
|
|
32
|
-
input: this.testCase.input,
|
|
33
|
-
expectedOutput: this.testCase.expectedOutput,
|
|
34
|
-
});
|
|
35
|
-
}
|
|
36
|
-
if (this.testCase?.preloaded) {
|
|
37
|
-
this.form.disable();
|
|
38
|
-
}
|
|
39
|
-
this.listenFormChanges();
|
|
40
|
-
}
|
|
41
|
-
ngOnChanges(changes) {
|
|
42
|
-
if (changes.testCase?.currentValue) {
|
|
43
|
-
this.form.patchValue({
|
|
44
|
-
input: this.testCase.input,
|
|
45
|
-
expectedOutput: this.testCase.expectedOutput,
|
|
46
|
-
});
|
|
47
|
-
if (this.testCase.preloaded) {
|
|
48
|
-
this.form.disable();
|
|
49
|
-
}
|
|
50
|
-
else {
|
|
51
|
-
this.form.enable();
|
|
52
|
-
}
|
|
53
|
-
this.isCustomTestCase = !this.testCase.preloaded;
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
isEmptyTestCase(status) {
|
|
57
|
-
return status === TestCaseStatus.Empty;
|
|
58
|
-
}
|
|
59
|
-
listenFormChanges() {
|
|
60
|
-
this.form.valueChanges.pipe(untilDestroyed(this)).subscribe((value) => {
|
|
61
|
-
this.valueChange.emit(value);
|
|
62
|
-
});
|
|
63
|
-
}
|
|
64
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TestCasesContentComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
65
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TestCasesContentComponent, isStandalone: true, selector: "tgo-test-cases-content", inputs: { testCase: "testCase", testCaseResult: "testCaseResult", areTestsRun: "areTestsRun", translations: "translations", applicationTheme: "applicationTheme", isSQL: "isSQL" }, outputs: { valueChange: "valueChange" }, usesOnChanges: true, ngImport: i0, template: "@if (testCase) {\n<div class=\"test-cases-content\" [class.test-cases-content-tests-run]=\"areTestsRun\">\n @if (areTestsRun && isCustomTestCase && (testCaseResult?.status | memoizeFunc: isEmptyTestCase)) {\n <div class=\"custom-tests-hint-container\">\n <p class=\"custom-tests-info\">{{ translations['BOTH_FIELDS_MUST_BE_FILLED'] }}</p>\n </div>\n } @if (!isCustomTestCase && !areTestsRun) {\n <p class=\"custom-tests-info custom-tests-info-indented\">\n {{ translations['EXAMPLE_TEST_CASE_INFO'] }}\n </p>\n } @if (isCustomTestCase && (!testCaseResult || (testCaseResult?.status | memoizeFunc: isEmptyTestCase))) {\n <p class=\"custom-tests-info custom-tests-info-indented\">\n {{ translations['FILL_ALL_FIELDS'] }}\n </p>\n } @if (testCaseResult && !(testCaseResult?.status | memoizeFunc: isEmptyTestCase)) {\n <div class=\"custom-tests-logs\">\n <div class=\"custom-tests-log-row\">\n <h4 class=\"custom-tests-log-header\">{{ translations['LOG'] }}</h4>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text [text]=\"testCaseResult.logs\" class=\"custom-tests-info\"></tgo-truncated-text>\n </div>\n </div>\n @if (isSQL) {\n <div class=\"custom-tests-log-row custom-tests-log-row-cols\">\n <div class=\"custom-tests-log-col custom-tests-log-col-expected-query\">\n <div class=\"custom-tests-log-heading\">\n <h4 class=\"custom-tests-log-header\">{{ translations['EXPECTED_QUERY'] }}</h4>\n </div>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text\n [text]=\"testCase.expectedOutput\"\n [isWordWrap]=\"true\"\n class=\"custom-tests-info\"\n ></tgo-truncated-text>\n </div>\n </div>\n <div class=\"custom-tests-log-col\">\n <div class=\"custom-tests-log-heading\">\n <h4 class=\"custom-tests-log-header\">{{ translations['QUERY_RESULT'] }}</h4>\n <tgo-test-cases-status\n [status]=\"testCaseResult.status\"\n [translations]=\"translations['STATUSES']\"\n ></tgo-test-cases-status>\n </div>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text\n [text]=\"testCaseResult.actualOutput\"\n [isWordWrap]=\"true\"\n class=\"custom-tests-info\"\n ></tgo-truncated-text>\n </div>\n </div>\n </div>\n } @else {\n <div class=\"custom-tests-log-row\">\n <div class=\"custom-tests-log-heading\">\n <h4 class=\"custom-tests-log-header\">{{ translations['OUTPUT'] }}</h4>\n <tgo-test-cases-status\n [status]=\"testCaseResult.status\"\n [translations]=\"translations['STATUSES']\"\n ></tgo-test-cases-status>\n </div>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text [text]=\"testCaseResult.actualOutput\" class=\"custom-tests-info\"></tgo-truncated-text>\n </div>\n </div>\n }\n </div>\n } @if (!isSQL) {\n <form [formGroup]=\"form\">\n <div class=\"custom-tests-inputs-container\">\n <ui-field\n class=\"custom-tests-input\"\n [label]=\"translations['INPUT']\"\n [type]=\"'multi-line'\"\n [disabled]=\"testCase.preloaded\"\n [maxRows]=\"3\"\n [applicationTheme]=\"applicationTheme\"\n formControlName=\"input\"\n ></ui-field>\n <ui-field\n class=\"custom-tests-input\"\n [label]=\"translations['EXPECTED_OUTPUT']\"\n [type]=\"'multi-line'\"\n [maxRows]=\"3\"\n [disabled]=\"testCase.preloaded\"\n [applicationTheme]=\"applicationTheme\"\n formControlName=\"expectedOutput\"\n ></ui-field>\n </div>\n </form>\n }\n</div>\n}\n", styles: [".test-cases-content{padding:16px;font-size:14px;font-weight:400}.test-cases-content-tests-run{padding-bottom:50px}@media screen and (min-width: 1920px){.test-cases-content{max-width:800px}}.custom-tests-log-header,.custom-tests-info{margin:0;font-size:14px;font-weight:400}.custom-tests-log-heading{display:flex;justify-content:space-between;align-items:center;margin:0}.custom-tests-log-content{padding:4px 12px;margin:4px 0;background-color:var(--status-bg-color)}.custom-tests-log-row,.custom-tests-hint-container,.custom-tests-info-indented{margin-bottom:16px}.custom-tests-log-row-cols{display:flex;gap:16px}@media screen and (max-width: 599px){.custom-tests-log-row-cols{flex-direction:column}}@media screen and (min-width: 600px){.custom-tests-log-col{flex:1 1 50%;max-width:50%}}.custom-tests-log-col-expected-query{order:2}@media screen and (min-width: 600px){.custom-tests-log-col-expected-query{order:unset}}.custom-tests-hint-container{padding:12px 16px;background-color:#f0d6bb;border-radius:4px;color:#000}.custom-tests-inputs-container{display:flex;justify-content:space-between;flex-wrap:wrap}@media screen and (min-width: 600px){.custom-tests-inputs-container{gap:16px}.custom-tests-inputs-container .custom-tests-input{min-width:135px}}.custom-tests-input{flex:1;font-size:14px}::ng-deep .custom-tests-input.mat-focused .mat-form-field-outline-thick,::ng-deep .custom-tests-input.mat-focused .mat-form-field-label{color:var(--company-color, --petrol-text-color)!important}::ng-deep .custom-tests-input.mat-focused .mat-input-element{caret-color:var(--company-color, --petrol-text-color)!important}::ng-deep .custom-tests-input.mat-focused .mat-form-field-wrapper{padding-bottom:0}::ng-deep .custom-tests-input:hover .mat-form-field-outline-thick{color:var(--company-color, --petrol-text-color)!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: FieldComponentModule }, { kind: "component", type: i2.FieldComponent, selector: "ui-field", inputs: ["fullWidth", "fullHeight", "label", "labelHtml", "labelIcon", "fieldName", "placeholder", "id", "value", "badgeVariant", "errors", "disabled", "required", "readOnly", "hintMessage", "type", "updateOnBlur", "allowOnlyDigits", "isAutocompleteOff", "allowNegative", "showBottomContent", "applicationTheme", "ariaLabel", "loading", "isValid", "maxCharacters", "trimOnBlur", "trimOnSubmit", "maxRows", "hasTextAreaCounter", "hideBuiltInErrors", "hideLabelInErrors", "max", "min", "textareaHeight", "borderless", "autosizableTextarea", "isAIVariant", "ariaLabelledby", "ariaDescribedby", "hasError"], outputs: ["validateEvent", "fieldBlur"] }, { kind: "pipe", type: MemoizeFuncPipe, name: "memoizeFunc" }, { kind: "component", type: TruncatedTextComponent, selector: "tgo-truncated-text", inputs: ["text", "maxChars", "isWordWrap"] }, { kind: "component", type: TestCasesStatusComponent, selector: "tgo-test-cases-status", inputs: ["status", "translations"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
66
|
-
};
|
|
67
|
-
TestCasesContentComponent = __decorate([
|
|
68
|
-
UntilDestroy()
|
|
69
|
-
], TestCasesContentComponent);
|
|
70
|
-
export { TestCasesContentComponent };
|
|
71
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TestCasesContentComponent, decorators: [{
|
|
72
|
-
type: Component,
|
|
73
|
-
args: [{ standalone: true, selector: 'tgo-test-cases-content', changeDetection: ChangeDetectionStrategy.OnPush, imports: [
|
|
74
|
-
CommonModule,
|
|
75
|
-
ReactiveFormsModule,
|
|
76
|
-
FieldComponentModule,
|
|
77
|
-
MemoizeFuncPipe,
|
|
78
|
-
TruncatedTextComponent,
|
|
79
|
-
TestCasesStatusComponent,
|
|
80
|
-
], template: "@if (testCase) {\n<div class=\"test-cases-content\" [class.test-cases-content-tests-run]=\"areTestsRun\">\n @if (areTestsRun && isCustomTestCase && (testCaseResult?.status | memoizeFunc: isEmptyTestCase)) {\n <div class=\"custom-tests-hint-container\">\n <p class=\"custom-tests-info\">{{ translations['BOTH_FIELDS_MUST_BE_FILLED'] }}</p>\n </div>\n } @if (!isCustomTestCase && !areTestsRun) {\n <p class=\"custom-tests-info custom-tests-info-indented\">\n {{ translations['EXAMPLE_TEST_CASE_INFO'] }}\n </p>\n } @if (isCustomTestCase && (!testCaseResult || (testCaseResult?.status | memoizeFunc: isEmptyTestCase))) {\n <p class=\"custom-tests-info custom-tests-info-indented\">\n {{ translations['FILL_ALL_FIELDS'] }}\n </p>\n } @if (testCaseResult && !(testCaseResult?.status | memoizeFunc: isEmptyTestCase)) {\n <div class=\"custom-tests-logs\">\n <div class=\"custom-tests-log-row\">\n <h4 class=\"custom-tests-log-header\">{{ translations['LOG'] }}</h4>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text [text]=\"testCaseResult.logs\" class=\"custom-tests-info\"></tgo-truncated-text>\n </div>\n </div>\n @if (isSQL) {\n <div class=\"custom-tests-log-row custom-tests-log-row-cols\">\n <div class=\"custom-tests-log-col custom-tests-log-col-expected-query\">\n <div class=\"custom-tests-log-heading\">\n <h4 class=\"custom-tests-log-header\">{{ translations['EXPECTED_QUERY'] }}</h4>\n </div>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text\n [text]=\"testCase.expectedOutput\"\n [isWordWrap]=\"true\"\n class=\"custom-tests-info\"\n ></tgo-truncated-text>\n </div>\n </div>\n <div class=\"custom-tests-log-col\">\n <div class=\"custom-tests-log-heading\">\n <h4 class=\"custom-tests-log-header\">{{ translations['QUERY_RESULT'] }}</h4>\n <tgo-test-cases-status\n [status]=\"testCaseResult.status\"\n [translations]=\"translations['STATUSES']\"\n ></tgo-test-cases-status>\n </div>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text\n [text]=\"testCaseResult.actualOutput\"\n [isWordWrap]=\"true\"\n class=\"custom-tests-info\"\n ></tgo-truncated-text>\n </div>\n </div>\n </div>\n } @else {\n <div class=\"custom-tests-log-row\">\n <div class=\"custom-tests-log-heading\">\n <h4 class=\"custom-tests-log-header\">{{ translations['OUTPUT'] }}</h4>\n <tgo-test-cases-status\n [status]=\"testCaseResult.status\"\n [translations]=\"translations['STATUSES']\"\n ></tgo-test-cases-status>\n </div>\n <div class=\"custom-tests-log-content\">\n <tgo-truncated-text [text]=\"testCaseResult.actualOutput\" class=\"custom-tests-info\"></tgo-truncated-text>\n </div>\n </div>\n }\n </div>\n } @if (!isSQL) {\n <form [formGroup]=\"form\">\n <div class=\"custom-tests-inputs-container\">\n <ui-field\n class=\"custom-tests-input\"\n [label]=\"translations['INPUT']\"\n [type]=\"'multi-line'\"\n [disabled]=\"testCase.preloaded\"\n [maxRows]=\"3\"\n [applicationTheme]=\"applicationTheme\"\n formControlName=\"input\"\n ></ui-field>\n <ui-field\n class=\"custom-tests-input\"\n [label]=\"translations['EXPECTED_OUTPUT']\"\n [type]=\"'multi-line'\"\n [maxRows]=\"3\"\n [disabled]=\"testCase.preloaded\"\n [applicationTheme]=\"applicationTheme\"\n formControlName=\"expectedOutput\"\n ></ui-field>\n </div>\n </form>\n }\n</div>\n}\n", styles: [".test-cases-content{padding:16px;font-size:14px;font-weight:400}.test-cases-content-tests-run{padding-bottom:50px}@media screen and (min-width: 1920px){.test-cases-content{max-width:800px}}.custom-tests-log-header,.custom-tests-info{margin:0;font-size:14px;font-weight:400}.custom-tests-log-heading{display:flex;justify-content:space-between;align-items:center;margin:0}.custom-tests-log-content{padding:4px 12px;margin:4px 0;background-color:var(--status-bg-color)}.custom-tests-log-row,.custom-tests-hint-container,.custom-tests-info-indented{margin-bottom:16px}.custom-tests-log-row-cols{display:flex;gap:16px}@media screen and (max-width: 599px){.custom-tests-log-row-cols{flex-direction:column}}@media screen and (min-width: 600px){.custom-tests-log-col{flex:1 1 50%;max-width:50%}}.custom-tests-log-col-expected-query{order:2}@media screen and (min-width: 600px){.custom-tests-log-col-expected-query{order:unset}}.custom-tests-hint-container{padding:12px 16px;background-color:#f0d6bb;border-radius:4px;color:#000}.custom-tests-inputs-container{display:flex;justify-content:space-between;flex-wrap:wrap}@media screen and (min-width: 600px){.custom-tests-inputs-container{gap:16px}.custom-tests-inputs-container .custom-tests-input{min-width:135px}}.custom-tests-input{flex:1;font-size:14px}::ng-deep .custom-tests-input.mat-focused .mat-form-field-outline-thick,::ng-deep .custom-tests-input.mat-focused .mat-form-field-label{color:var(--company-color, --petrol-text-color)!important}::ng-deep .custom-tests-input.mat-focused .mat-input-element{caret-color:var(--company-color, --petrol-text-color)!important}::ng-deep .custom-tests-input.mat-focused .mat-form-field-wrapper{padding-bottom:0}::ng-deep .custom-tests-input:hover .mat-form-field-outline-thick{color:var(--company-color, --petrol-text-color)!important}\n"] }]
|
|
81
|
-
}], propDecorators: { testCase: [{
|
|
82
|
-
type: Input
|
|
83
|
-
}], testCaseResult: [{
|
|
84
|
-
type: Input
|
|
85
|
-
}], areTestsRun: [{
|
|
86
|
-
type: Input
|
|
87
|
-
}], translations: [{
|
|
88
|
-
type: Input
|
|
89
|
-
}], applicationTheme: [{
|
|
90
|
-
type: Input
|
|
91
|
-
}], isSQL: [{
|
|
92
|
-
type: Input
|
|
93
|
-
}], valueChange: [{
|
|
94
|
-
type: Output
|
|
95
|
-
}] } });
|
|
96
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-cases-content.component.js","sourceRoot":"","sources":["../../../../../../../../packages/tgo-coding-test/src/lib/components/tests/test-cases-content/test-cases-content.component.ts","../../../../../../../../packages/tgo-coding-test/src/lib/components/tests/test-cases-content/test-cases-content.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormBuilder, FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { UntilDestroy, untilDestroyed } from '@ngneat/until-destroy';\nimport { ApplicationTheme, FieldComponentModule } from '@testgorilla/tgo-ui';\nimport {\n  AllowedModificationFields,\n  CustomTestCase,\n  ExampleTestCase,\n  TestCaseResult,\n  TestCaseStatus,\n} from '../../../models/test-cases';\nimport { MemoizeFuncPipe } from '../../../pipes/memoize-func.pipe';\nimport { TruncatedTextComponent } from '../../common/truncated-text/truncated-text.component';\nimport { TestCasesStatusComponent } from '../test-cases-status/test-cases-status.component';\n\n@UntilDestroy()\n@Component({\n  standalone: true,\n  selector: 'tgo-test-cases-content',\n  templateUrl: './test-cases-content.component.html',\n  styleUrls: ['./test-cases-content.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    FieldComponentModule,\n    MemoizeFuncPipe,\n    TruncatedTextComponent,\n    TestCasesStatusComponent,\n  ],\n})\nexport class TestCasesContentComponent implements OnInit, OnChanges {\n  private readonly fb = inject(FormBuilder);\n  \n  @Input() testCase: ExampleTestCase | CustomTestCase;\n  @Input() testCaseResult: TestCaseResult;\n  @Input() areTestsRun: boolean;\n  @Input() translations: Record<string, unknown>;\n  @Input() applicationTheme: ApplicationTheme;\n  @Input() isSQL: boolean;\n\n  @Output() valueChange = new EventEmitter<AllowedModificationFields>();\n\n  isCustomTestCase: boolean;\n  // todo: add validators when backend implements validation on their side\n  form = this.fb.group({\n    input: new FormControl<string>(''),\n    expectedOutput: new FormControl<string>(''),\n  });\n\n  ngOnInit(): void {\n    if (this.testCase?.input && this.testCase?.expectedOutput) {\n      this.form.patchValue({\n        input: this.testCase.input,\n        expectedOutput: this.testCase.expectedOutput,\n      });\n    }\n\n    if (this.testCase?.preloaded) {\n      this.form.disable();\n    }\n\n    this.listenFormChanges();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.testCase?.currentValue) {\n      this.form.patchValue({\n        input: this.testCase.input,\n        expectedOutput: this.testCase.expectedOutput,\n      });\n      if (this.testCase.preloaded) {\n        this.form.disable();\n      } else {\n        this.form.enable();\n      }\n\n      this.isCustomTestCase = !this.testCase.preloaded;\n    }\n  }\n\n  isEmptyTestCase(status: TestCaseStatus): boolean {\n    return status === TestCaseStatus.Empty;\n  }\n\n  private listenFormChanges(): void {\n    this.form.valueChanges.pipe(untilDestroyed(this)).subscribe((value: AllowedModificationFields) => {\n      this.valueChange.emit(value);\n    });\n  }\n}\n","@if (testCase) {\n<div class=\"test-cases-content\" [class.test-cases-content-tests-run]=\"areTestsRun\">\n  @if (areTestsRun && isCustomTestCase && (testCaseResult?.status | memoizeFunc: isEmptyTestCase)) {\n  <div class=\"custom-tests-hint-container\">\n    <p class=\"custom-tests-info\">{{ translations['BOTH_FIELDS_MUST_BE_FILLED'] }}</p>\n  </div>\n  } @if (!isCustomTestCase && !areTestsRun) {\n  <p class=\"custom-tests-info custom-tests-info-indented\">\n    {{ translations['EXAMPLE_TEST_CASE_INFO'] }}\n  </p>\n  } @if (isCustomTestCase && (!testCaseResult || (testCaseResult?.status | memoizeFunc: isEmptyTestCase))) {\n  <p class=\"custom-tests-info custom-tests-info-indented\">\n    {{ translations['FILL_ALL_FIELDS'] }}\n  </p>\n  } @if (testCaseResult && !(testCaseResult?.status | memoizeFunc: isEmptyTestCase)) {\n  <div class=\"custom-tests-logs\">\n    <div class=\"custom-tests-log-row\">\n      <h4 class=\"custom-tests-log-header\">{{ translations['LOG'] }}</h4>\n      <div class=\"custom-tests-log-content\">\n        <tgo-truncated-text [text]=\"testCaseResult.logs\" class=\"custom-tests-info\"></tgo-truncated-text>\n      </div>\n    </div>\n    @if (isSQL) {\n    <div class=\"custom-tests-log-row custom-tests-log-row-cols\">\n      <div class=\"custom-tests-log-col custom-tests-log-col-expected-query\">\n        <div class=\"custom-tests-log-heading\">\n          <h4 class=\"custom-tests-log-header\">{{ translations['EXPECTED_QUERY'] }}</h4>\n        </div>\n        <div class=\"custom-tests-log-content\">\n          <tgo-truncated-text\n            [text]=\"testCase.expectedOutput\"\n            [isWordWrap]=\"true\"\n            class=\"custom-tests-info\"\n          ></tgo-truncated-text>\n        </div>\n      </div>\n      <div class=\"custom-tests-log-col\">\n        <div class=\"custom-tests-log-heading\">\n          <h4 class=\"custom-tests-log-header\">{{ translations['QUERY_RESULT'] }}</h4>\n          <tgo-test-cases-status\n            [status]=\"testCaseResult.status\"\n            [translations]=\"translations['STATUSES']\"\n          ></tgo-test-cases-status>\n        </div>\n        <div class=\"custom-tests-log-content\">\n          <tgo-truncated-text\n            [text]=\"testCaseResult.actualOutput\"\n            [isWordWrap]=\"true\"\n            class=\"custom-tests-info\"\n          ></tgo-truncated-text>\n        </div>\n      </div>\n    </div>\n    } @else {\n    <div class=\"custom-tests-log-row\">\n      <div class=\"custom-tests-log-heading\">\n        <h4 class=\"custom-tests-log-header\">{{ translations['OUTPUT'] }}</h4>\n        <tgo-test-cases-status\n          [status]=\"testCaseResult.status\"\n          [translations]=\"translations['STATUSES']\"\n        ></tgo-test-cases-status>\n      </div>\n      <div class=\"custom-tests-log-content\">\n        <tgo-truncated-text [text]=\"testCaseResult.actualOutput\" class=\"custom-tests-info\"></tgo-truncated-text>\n      </div>\n    </div>\n    }\n  </div>\n  } @if (!isSQL) {\n  <form [formGroup]=\"form\">\n    <div class=\"custom-tests-inputs-container\">\n      <ui-field\n        class=\"custom-tests-input\"\n        [label]=\"translations['INPUT']\"\n        [type]=\"'multi-line'\"\n        [disabled]=\"testCase.preloaded\"\n        [maxRows]=\"3\"\n        [applicationTheme]=\"applicationTheme\"\n        formControlName=\"input\"\n      ></ui-field>\n      <ui-field\n        class=\"custom-tests-input\"\n        [label]=\"translations['EXPECTED_OUTPUT']\"\n        [type]=\"'multi-line'\"\n        [maxRows]=\"3\"\n        [disabled]=\"testCase.preloaded\"\n        [applicationTheme]=\"applicationTheme\"\n        formControlName=\"expectedOutput\"\n      ></ui-field>\n    </div>\n  </form>\n  }\n</div>\n}\n"]}
|
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
2
|
-
import { IconLabelComponentModule } from '@testgorilla/tgo-ui';
|
|
3
|
-
import { TestCaseStatus } from '../../../models/test-cases';
|
|
4
|
-
import * as i0 from "@angular/core";
|
|
5
|
-
import * as i1 from "@testgorilla/tgo-ui";
|
|
6
|
-
export class TestCasesStatusComponent {
|
|
7
|
-
status;
|
|
8
|
-
translations;
|
|
9
|
-
TestCaseStatus = TestCaseStatus;
|
|
10
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TestCasesStatusComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
11
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: TestCasesStatusComponent, isStandalone: true, selector: "tgo-test-cases-status", inputs: { status: "status", translations: "translations" }, ngImport: i0, template: "@switch (status) { @case (TestCaseStatus.Passed) {\n<div class=\"test-case-status test-case-status-passed\">\n <ui-icon-label class=\"label-icon\" [iconName]=\"'Check'\" [iconSize]=\"'16'\" [text]=\"translations['PASSED']\">\n </ui-icon-label>\n</div>\n} @case (TestCaseStatus.Error) {\n<div class=\"test-case-status test-case-status-error\">\n <ui-icon-label class=\"label-icon\" [iconName]=\"'Close'\" [iconSize]=\"'16'\" [text]=\"translations['ERROR']\">\n </ui-icon-label>\n</div>\n} @case (TestCaseStatus.Empty) {\n<div class=\"test-case-status test-case-status-empty\">\n <ui-icon-label class=\"label-icon\" [iconName]=\"'Help'\" [iconSize]=\"'16'\" [text]=\"translations['EMPTY']\">\n </ui-icon-label>\n</div>\n} }\n", styles: [":host{display:flex}.test-case-status{display:flex;align-items:center}.test-case-status-passed{color:var(--petrol-text-color)}.test-case-status-passed ::ng-deep mat-icon svg{color:var(--petrol-text-color)}.test-case-status-error{color:#cb7b7a}.test-case-status-error ::ng-deep mat-icon svg{color:#cb7b7a}.test-case-status-empty{color:var(--status-empty-color)}.test-case-status-empty ::ng-deep mat-icon svg{color:var(--status-empty-color)}.test-case-status .label-icon ::ng-deep mat-icon{font-size:16px;width:16px;height:16px;min-width:16px;min-height:16px;line-height:16px}\n"], dependencies: [{ kind: "ngmodule", type: IconLabelComponentModule }, { kind: "component", type: i1.IconLabelComponent, selector: "ui-icon-label", inputs: ["iconSize", "iconName", "text", "iconColor", "iconIndent", "applicationTheme"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
12
|
-
}
|
|
13
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TestCasesStatusComponent, decorators: [{
|
|
14
|
-
type: Component,
|
|
15
|
-
args: [{ standalone: true, selector: 'tgo-test-cases-status', changeDetection: ChangeDetectionStrategy.OnPush, imports: [IconLabelComponentModule], template: "@switch (status) { @case (TestCaseStatus.Passed) {\n<div class=\"test-case-status test-case-status-passed\">\n <ui-icon-label class=\"label-icon\" [iconName]=\"'Check'\" [iconSize]=\"'16'\" [text]=\"translations['PASSED']\">\n </ui-icon-label>\n</div>\n} @case (TestCaseStatus.Error) {\n<div class=\"test-case-status test-case-status-error\">\n <ui-icon-label class=\"label-icon\" [iconName]=\"'Close'\" [iconSize]=\"'16'\" [text]=\"translations['ERROR']\">\n </ui-icon-label>\n</div>\n} @case (TestCaseStatus.Empty) {\n<div class=\"test-case-status test-case-status-empty\">\n <ui-icon-label class=\"label-icon\" [iconName]=\"'Help'\" [iconSize]=\"'16'\" [text]=\"translations['EMPTY']\">\n </ui-icon-label>\n</div>\n} }\n", styles: [":host{display:flex}.test-case-status{display:flex;align-items:center}.test-case-status-passed{color:var(--petrol-text-color)}.test-case-status-passed ::ng-deep mat-icon svg{color:var(--petrol-text-color)}.test-case-status-error{color:#cb7b7a}.test-case-status-error ::ng-deep mat-icon svg{color:#cb7b7a}.test-case-status-empty{color:var(--status-empty-color)}.test-case-status-empty ::ng-deep mat-icon svg{color:var(--status-empty-color)}.test-case-status .label-icon ::ng-deep mat-icon{font-size:16px;width:16px;height:16px;min-width:16px;min-height:16px;line-height:16px}\n"] }]
|
|
16
|
-
}], propDecorators: { status: [{
|
|
17
|
-
type: Input
|
|
18
|
-
}], translations: [{
|
|
19
|
-
type: Input
|
|
20
|
-
}] } });
|
|
21
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC1jYXNlcy1zdGF0dXMuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvdGdvLWNvZGluZy10ZXN0L3NyYy9saWIvY29tcG9uZW50cy90ZXN0cy90ZXN0LWNhc2VzLXN0YXR1cy90ZXN0LWNhc2VzLXN0YXR1cy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy90Z28tY29kaW5nLXRlc3Qvc3JjL2xpYi9jb21wb25lbnRzL3Rlc3RzL3Rlc3QtY2FzZXMtc3RhdHVzL3Rlc3QtY2FzZXMtc3RhdHVzLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzFFLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQy9ELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQzs7O0FBVTVELE1BQU0sT0FBTyx3QkFBd0I7SUFDMUIsTUFBTSxDQUFpQjtJQUN2QixZQUFZLENBQTBCO0lBRXRDLGNBQWMsR0FBRyxjQUFjLENBQUM7d0dBSjlCLHdCQUF3Qjs0RkFBeEIsd0JBQXdCLDZJQ1pyQywydEJBZ0JBLHduQkROWSx3QkFBd0I7OzRGQUV2Qix3QkFBd0I7a0JBUnBDLFNBQVM7aUNBQ0ksSUFBSSxZQUNOLHVCQUF1QixtQkFHaEIsdUJBQXVCLENBQUMsTUFBTSxXQUN0QyxDQUFDLHdCQUF3QixDQUFDOzhCQUcxQixNQUFNO3NCQUFkLEtBQUs7Z0JBQ0csWUFBWTtzQkFBcEIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBJY29uTGFiZWxDb21wb25lbnRNb2R1bGUgfSBmcm9tICdAdGVzdGdvcmlsbGEvdGdvLXVpJztcbmltcG9ydCB7IFRlc3RDYXNlU3RhdHVzIH0gZnJvbSAnLi4vLi4vLi4vbW9kZWxzL3Rlc3QtY2FzZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgc2VsZWN0b3I6ICd0Z28tdGVzdC1jYXNlcy1zdGF0dXMnLFxuICB0ZW1wbGF0ZVVybDogJy4vdGVzdC1jYXNlcy1zdGF0dXMuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi90ZXN0LWNhc2VzLXN0YXR1cy5jb21wb25lbnQuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgaW1wb3J0czogW0ljb25MYWJlbENvbXBvbmVudE1vZHVsZV0sXG59KVxuZXhwb3J0IGNsYXNzIFRlc3RDYXNlc1N0YXR1c0NvbXBvbmVudCB7XG4gIEBJbnB1dCgpIHN0YXR1czogVGVzdENhc2VTdGF0dXM7XG4gIEBJbnB1dCgpIHRyYW5zbGF0aW9uczogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG5cbiAgcmVhZG9ubHkgVGVzdENhc2VTdGF0dXMgPSBUZXN0Q2FzZVN0YXR1cztcbn1cbiIsIkBzd2l0Y2ggKHN0YXR1cykgeyBAY2FzZSAoVGVzdENhc2VTdGF0dXMuUGFzc2VkKSB7XG48ZGl2IGNsYXNzPVwidGVzdC1jYXNlLXN0YXR1cyB0ZXN0LWNhc2Utc3RhdHVzLXBhc3NlZFwiPlxuICA8dWktaWNvbi1sYWJlbCBjbGFzcz1cImxhYmVsLWljb25cIiBbaWNvbk5hbWVdPVwiJ0NoZWNrJ1wiIFtpY29uU2l6ZV09XCInMTYnXCIgW3RleHRdPVwidHJhbnNsYXRpb25zWydQQVNTRUQnXVwiPlxuICA8L3VpLWljb24tbGFiZWw+XG48L2Rpdj5cbn0gQGNhc2UgKFRlc3RDYXNlU3RhdHVzLkVycm9yKSB7XG48ZGl2IGNsYXNzPVwidGVzdC1jYXNlLXN0YXR1cyB0ZXN0LWNhc2Utc3RhdHVzLWVycm9yXCI+XG4gIDx1aS1pY29uLWxhYmVsIGNsYXNzPVwibGFiZWwtaWNvblwiIFtpY29uTmFtZV09XCInQ2xvc2UnXCIgW2ljb25TaXplXT1cIicxNidcIiBbdGV4dF09XCJ0cmFuc2xhdGlvbnNbJ0VSUk9SJ11cIj5cbiAgPC91aS1pY29uLWxhYmVsPlxuPC9kaXY+XG59IEBjYXNlIChUZXN0Q2FzZVN0YXR1cy5FbXB0eSkge1xuPGRpdiBjbGFzcz1cInRlc3QtY2FzZS1zdGF0dXMgdGVzdC1jYXNlLXN0YXR1cy1lbXB0eVwiPlxuICA8dWktaWNvbi1sYWJlbCBjbGFzcz1cImxhYmVsLWljb25cIiBbaWNvbk5hbWVdPVwiJ0hlbHAnXCIgW2ljb25TaXplXT1cIicxNidcIiBbdGV4dF09XCJ0cmFuc2xhdGlvbnNbJ0VNUFRZJ11cIj5cbiAgPC91aS1pY29uLWxhYmVsPlxuPC9kaXY+XG59IH1cbiJdfQ==
|