@touchpoll/tp-survey 0.0.18 → 0.0.20
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/core/tp.survey.answer/tp.survey.answer.mjs +1 -4
- package/esm2022/lib/core/tp.survey.core/tp.survey.core.service.mjs +45 -9
- package/esm2022/lib/core/tp.survey.interview/tp.survey.interview.service.mjs +1 -1
- package/fesm2022/touchpoll-tp-survey.mjs +44 -11
- package/fesm2022/touchpoll-tp-survey.mjs.map +1 -1
- package/lib/core/tp.survey.answer/tp.survey.answer.d.ts +1 -2
- package/package.json +1 -1
|
@@ -23,8 +23,5 @@ export class Answer {
|
|
|
23
23
|
this.AnswerAdditional[Number(key)] = null;
|
|
24
24
|
}
|
|
25
25
|
}
|
|
26
|
-
addAnswer(answerValue, answerAdditional, asComplete) {
|
|
27
|
-
this.AnswerValue = answerValue;
|
|
28
|
-
}
|
|
29
26
|
}
|
|
30
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHAuc3VydmV5LmFuc3dlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RwLnN1cnZleS9zcmMvbGliL2NvcmUvdHAuc3VydmV5LmFuc3dlci90cC5zdXJ2ZXkuYW5zd2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQztBQUVwRCxNQUFNLE9BQU8sTUFBTTtJQVdqQixZQUFZLElBQVksRUFBRSxVQUFrQixDQUFDLENBQUM7UUFSckMscUJBQWdCLEdBQXFDLEVBQUUsQ0FBQztRQUN4RCw0QkFBdUIsR0FBa0IsRUFBRSxDQUFDO1FBQzVDLHVCQUFrQixHQUFrQixFQUFFLENBQUM7UUFDdkMsOEJBQXlCLEdBQTRCLEVBQUUsQ0FBQztRQUNqRSxjQUFTLEdBQUcsQ0FBQyxDQUFDO1FBQ2QsZUFBVSxHQUFHLENBQUMsQ0FBQztRQUNmLGdCQUFXLEdBQVEsYUFBYSxDQUFDO1FBQ2pDLHFCQUFnQixHQUFZLEtBQUssQ0FBQztRQUVoQyxJQUFJLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQztRQUN6QixJQUFJLENBQUMsZUFBZSxHQUFHLE9BQU8sQ0FBQztJQUNqQyxDQUFDO0lBQ0QsS0FBSztRQUNILElBQUksQ0FBQyxTQUFTLEdBQUcsQ0FBQyxDQUFDO1FBQ25CLElBQUksQ0FBQyxVQUFVLEdBQUcsQ0FBQyxDQUFDO1FBQ3BCLElBQUksQ0FBQyxXQUFXLEdBQUcsYUFBYSxDQUFDO1FBQ2pDLElBQUksQ0FBQyx1QkFBdUIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQ3hDLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQ25DLElBQUksQ0FBQyx5QkFBeUIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQzFDLEtBQUssTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsRUFBRSxDQUFDO1lBQ3JELElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUM7UUFDNUMsQ0FBQztJQUNILENBQUM7Q0FLRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SVF1ZXN0aW9uQW5zd2VyfSBmcm9tICcuLi8uLi90cC5zdXJ2ZXkuaW50ZXJmYWNlJztcclxuaW1wb3J0IHtBZGRpdGlvbmFsQW5zd2VyfSBmcm9tICcuLi8uLi90cC5zdXJ2ZXkudHlwZSc7XHJcbmltcG9ydCB7TUlTU0lOR19WQUxVRX0gZnJvbSAnLi4vLi4vdHAuc3VydmV5LmNvbnN0JztcclxuXHJcbmV4cG9ydCBjbGFzcyBBbnN3ZXIgaW1wbGVtZW50cyBJUXVlc3Rpb25BbnN3ZXIge1xyXG4gIHJlYWRvbmx5IFF1ZXN0aW9uR1VJRDogbnVtYmVyO1xyXG4gIHJlYWRvbmx5IFF1ZXN0aW9uVmVyc2lvbjogbnVtYmVyO1xyXG4gIHJlYWRvbmx5IEFuc3dlckFkZGl0aW9uYWw6IFJlY29yZDxzdHJpbmcsIEFkZGl0aW9uYWxBbnN3ZXI+ID0ge307XHJcbiAgcmVhZG9ubHkgVmlld2VkQWx0ZXJuYXRpdmVWYWx1ZXM6IEFycmF5PG51bWJlcj4gPSBbXTtcclxuICByZWFkb25seSBWaWV3ZWRTZWN0aW9uR3VpZHM6IEFycmF5PG51bWJlcj4gPSBbXTtcclxuICByZWFkb25seSBBbHRlcm5hdGl2ZVNlcXVlbmNlQ2xpY2tzOiBBcnJheTxbbnVtYmVyLCBudW1iZXJdPiA9IFtdO1xyXG4gIFN0YXJ0ZWRBdCA9IDA7XHJcbiAgRmluaXNoZWRBdCA9IDA7XHJcbiAgQW5zd2VyVmFsdWU6IGFueSA9IE1JU1NJTkdfVkFMVUU7XHJcbiAgYW5zd2VySXNDb21wbGV0ZTogYm9vbGVhbiA9IGZhbHNlO1xyXG4gIGNvbnN0cnVjdG9yKGd1aWQ6IG51bWJlciwgdmVyc2lvbjogbnVtYmVyID0gLTEpIHtcclxuICAgIHRoaXMuUXVlc3Rpb25HVUlEID0gZ3VpZDtcclxuICAgIHRoaXMuUXVlc3Rpb25WZXJzaW9uID0gdmVyc2lvbjtcclxuICB9XHJcbiAgY2xlYXIoKTogdm9pZCB7XHJcbiAgICB0aGlzLlN0YXJ0ZWRBdCA9IDA7XHJcbiAgICB0aGlzLkZpbmlzaGVkQXQgPSAwO1xyXG4gICAgdGhpcy5BbnN3ZXJWYWx1ZSA9IE1JU1NJTkdfVkFMVUU7XHJcbiAgICB0aGlzLlZpZXdlZEFsdGVybmF0aXZlVmFsdWVzLmxlbmd0aCA9IDA7XHJcbiAgICB0aGlzLlZpZXdlZFNlY3Rpb25HdWlkcy5sZW5ndGggPSAwO1xyXG4gICAgdGhpcy5BbHRlcm5hdGl2ZVNlcXVlbmNlQ2xpY2tzLmxlbmd0aCA9IDA7XHJcbiAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyh0aGlzLkFuc3dlckFkZGl0aW9uYWwpKSB7XHJcbiAgICAgIHRoaXMuQW5zd2VyQWRkaXRpb25hbFtOdW1iZXIoa2V5KV0gPSBudWxsO1xyXG4gICAgfVxyXG4gIH1cclxuICAvLyBhZGRBbnN3ZXIoYW5zd2VyVmFsdWU6IHVua25vd24sIGFuc3dlckFkZGl0aW9uYWw6IExvZ2ljT2JqZWN0LCBhc0NvbXBsZXRlOiBib29sZWFuKTogdm9pZCB7XHJcbiAgLy8gICB0aGlzLkFuc3dlclZhbHVlID0gYW5zd2VyVmFsdWU7XHJcbiAgLy8gfVxyXG5cclxufVxyXG4iXX0=
|
|
@@ -15,6 +15,7 @@ export class TpSurveyCoreService {
|
|
|
15
15
|
#surveyQuotas = new SurveyQuotas();
|
|
16
16
|
#questionary;
|
|
17
17
|
#displayedQuestions = []; // нужно для навигации по кнопке back
|
|
18
|
+
#questionLogicVarContainer = {}; //Нужно знать какая логическая переменная относится к какому вопросу. НапримерЮ когда нужно ее отчистить при переходе назад.
|
|
18
19
|
#destroyRef = inject(DestroyRef);
|
|
19
20
|
constructor(platformId, _logic, _interview) {
|
|
20
21
|
this.platformId = platformId;
|
|
@@ -86,6 +87,7 @@ export class TpSurveyCoreService {
|
|
|
86
87
|
if (!!previousQuestionNum) {
|
|
87
88
|
const prevQuestion = this.#questionary?.Questions[previousQuestionNum];
|
|
88
89
|
if (!!prevQuestion) {
|
|
90
|
+
this.#clearAnswerForCurrentQuestion(); // отчистим ответ на вопрос, если он уже есть.
|
|
89
91
|
this.#changeCurrentQuestion(prevQuestion);
|
|
90
92
|
}
|
|
91
93
|
}
|
|
@@ -94,22 +96,29 @@ export class TpSurveyCoreService {
|
|
|
94
96
|
const answer = this._interview.findAnswerByGuid(guid);
|
|
95
97
|
return answer?.answerIsComplete ?? false;
|
|
96
98
|
}
|
|
99
|
+
/*
|
|
100
|
+
* функция по номеру вопроса, ищет ответ на этот guid, и возвращает ответ на этот вопрос.
|
|
101
|
+
* если это ss ms, вместо номеров альтернатив подставляет текст.
|
|
102
|
+
* работает только для eqOL, eqSS, eqMS
|
|
103
|
+
* */
|
|
97
104
|
guidToTextAnswer(guid, languageId) {
|
|
98
105
|
const answer = this.findAnswerByGuid(guid);
|
|
99
106
|
const question = this.#findQuestionByGuid(guid);
|
|
107
|
+
//если ответа нет, ревнём пустую строку
|
|
100
108
|
if (!question || !answer || answer.AnswerValue === -97) {
|
|
101
109
|
return '';
|
|
102
110
|
}
|
|
103
111
|
const questionType = question.QuestionType;
|
|
104
|
-
console.log(0);
|
|
105
112
|
switch (questionType) {
|
|
106
113
|
case QuestionsType.eqOL: return answer.AnswerValue;
|
|
107
114
|
case QuestionsType.eqSS: {
|
|
115
|
+
//находим альтернативу и подставляем заголовок альтернативы с указанным языком, если заголовка нет, вернём alias
|
|
108
116
|
const alternative = question.Alternativelist.find(alt => alt.Value === answer.AnswerValue);
|
|
109
117
|
const altText = alternative?.MultiLangCaptions.find(obj => obj.lang === languageId);
|
|
110
118
|
return (altText?.CaptionHtml || alternative?.alias) ?? '';
|
|
111
119
|
}
|
|
112
120
|
case QuestionsType.eqMS: {
|
|
121
|
+
//всё то же самое, что и ss, только тут массив ответов,
|
|
113
122
|
const answerArray = answer.AnswerValue;
|
|
114
123
|
const alternativeArray = question.Alternativelist.filter(alt => answerArray.indexOf(alt.Value) > 0);
|
|
115
124
|
const altTextArray = alternativeArray.map(alt => (alt.MultiLangCaptions.find(obj => obj.lang === languageId)?.CaptionHtml || alt.alias) ?? '');
|
|
@@ -225,8 +234,8 @@ export class TpSurveyCoreService {
|
|
|
225
234
|
this.#eventBus.next({ key: SurveyEvents.seQuestionAnswersChange, data: { question: this.#currentQuestion.value, answerIsComplete: answer.answerIsComplete } });
|
|
226
235
|
//может возникнуть ситуация, когда нет ни одной альтернативы или секции для отображения.в этом случае вопрос показывать не нужно, как ответ записываем 0 и переходим на следующий
|
|
227
236
|
const question = this.#findQuestionByGuid(eventData.guid);
|
|
228
|
-
//ситуация для ss/ms
|
|
229
|
-
if (!!question && answer.answerIsComplete && (question.QuestionType === 'SS' || question.QuestionType === 'MS') && (eventData.viewedAlternativeValues.length === 0)) {
|
|
237
|
+
//ситуация для ss/ms вопросов и для неявных вопросов
|
|
238
|
+
if (!!question && answer.answerIsComplete && (question.QuestionType === 'SS' || question.QuestionType === 'MS') && ((eventData.viewedAlternativeValues.length === 0) || question.Implicit)) {
|
|
230
239
|
this.goToNextQuestion();
|
|
231
240
|
}
|
|
232
241
|
}
|
|
@@ -327,19 +336,31 @@ export class TpSurveyCoreService {
|
|
|
327
336
|
if (!answer) {
|
|
328
337
|
throw new Error('answer not found');
|
|
329
338
|
}
|
|
339
|
+
const logicVarSet = new Set();
|
|
340
|
+
this.#questionLogicVarContainer[question.QuestionGuid.toString()] = logicVarSet;
|
|
341
|
+
logicVarSet.add(question.SysName); //привязываем переменную к вопросу
|
|
330
342
|
this._logic.setLogicValue(question.SysName, MISSING_VALUE); // переменная самого вопроса
|
|
331
343
|
//добавим доп ответы и логику для них
|
|
332
344
|
question.AdditionalAnswerlist.forEach(aa => {
|
|
345
|
+
const aaSysName = `${question.SysName}${aa.SysName}`;
|
|
333
346
|
answer.AnswerAdditional[aa.Value] = null;
|
|
334
|
-
this._logic.setLogicValue(
|
|
347
|
+
this._logic.setLogicValue(aaSysName, MISSING_VALUE); // логика для переменных доп ответа
|
|
348
|
+
logicVarSet.add(aaSysName); //привязываем переменную к вопросу
|
|
335
349
|
});
|
|
336
350
|
// если в тексте postScript есть запись типа setVar('varName', value); нужно инициализировать переменную varName значением -97.
|
|
337
351
|
if (!!question.PostScript) {
|
|
338
|
-
(question.PostScript.match(/(\bsetvar\b)\(['|"](.*?)['|"]/gim) ?? []).map(scriptVar => scriptVar.match(/['|"](.+)['|"]/i)[1]).forEach(varName =>
|
|
352
|
+
(question.PostScript.match(/(\bsetvar\b)\(['|"](.*?)['|"]/gim) ?? []).map(scriptVar => scriptVar.match(/['|"](.+)['|"]/i)[1]).forEach(varName => {
|
|
353
|
+
logicVarSet.add(varName); //привязываем переменную к вопросу
|
|
354
|
+
this._logic.setLogicValue(varName, null);
|
|
355
|
+
});
|
|
339
356
|
}
|
|
340
357
|
// для Вопросов SS/MS нужно прописать логику альтернатив sysname вопроса + sysname альтернативы
|
|
341
358
|
if (question.Sectionlist.length <= 0) { // для Вопросов SS/MS
|
|
342
|
-
question.Alternativelist.forEach(alt =>
|
|
359
|
+
question.Alternativelist.forEach(alt => {
|
|
360
|
+
const aSysName = `${question.SysName}${alt.SysName}`;
|
|
361
|
+
this._logic.setLogicValue(aSysName, MISSING_VALUE);
|
|
362
|
+
logicVarSet.add(aSysName); //привязываем переменную к вопросу
|
|
363
|
+
});
|
|
343
364
|
}
|
|
344
365
|
else {
|
|
345
366
|
//для вопросов у которых есть секции проинициализируем каждую секцию
|
|
@@ -352,8 +373,13 @@ export class TpSurveyCoreService {
|
|
|
352
373
|
}
|
|
353
374
|
// для Вопросов SL нужно прописать логику альтернатив и секций sysname секции + sysname альтернативы
|
|
354
375
|
this._logic.setLogicValue(section.SysName, MISSING_VALUE); // переменная самой секции
|
|
376
|
+
logicVarSet.add(section.SysName); //привязываем переменную к вопросу
|
|
355
377
|
//логика альтернатив SL вопроса
|
|
356
|
-
question.Alternativelist.forEach(alt =>
|
|
378
|
+
question.Alternativelist.forEach(alt => {
|
|
379
|
+
const aSysName = `${section.SysName}${alt.SysName}`;
|
|
380
|
+
this._logic.setLogicValue(aSysName, MISSING_VALUE);
|
|
381
|
+
logicVarSet.add(aSysName); //привязываем переменную к вопросу
|
|
382
|
+
});
|
|
357
383
|
});
|
|
358
384
|
}
|
|
359
385
|
});
|
|
@@ -397,13 +423,23 @@ export class TpSurveyCoreService {
|
|
|
397
423
|
answer: questionAnswer,
|
|
398
424
|
additionalAnswers: {},
|
|
399
425
|
logicVars: {},
|
|
400
|
-
viewedAlternativeValues: [],
|
|
426
|
+
viewedAlternativeValues: Array.isArray(questionAnswer) ? questionAnswer : [questionAnswer],
|
|
401
427
|
viewedSectionGuids: [],
|
|
402
428
|
alternativeSequenceClicks: []
|
|
403
429
|
};
|
|
404
430
|
this.updateAnswer(answerChange);
|
|
405
431
|
}
|
|
406
432
|
}
|
|
433
|
+
#clearAnswerForCurrentQuestion() {
|
|
434
|
+
const currentQuestionGuid = this.#currentQuestion.value?.QuestionGuid ?? -1;
|
|
435
|
+
//удаляем сам ответ на вопрос
|
|
436
|
+
this._interview.findAnswerByGuid(currentQuestionGuid)?.clear();
|
|
437
|
+
//удаляем переменные которые относятся к этому вопросу
|
|
438
|
+
const logicSet = this.#questionLogicVarContainer[currentQuestionGuid];
|
|
439
|
+
if (!!logicSet) {
|
|
440
|
+
logicSet.forEach(sysName => this._logic.setLogicValue(sysName, MISSING_VALUE));
|
|
441
|
+
}
|
|
442
|
+
}
|
|
407
443
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: TpSurveyCoreService, deps: [{ token: PLATFORM_ID }, { token: i1.TpSurveyLogicService }, { token: i2.TpSurveyInterviewService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
408
444
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: TpSurveyCoreService, providedIn: 'root' }); }
|
|
409
445
|
}
|
|
@@ -416,4 +452,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImpor
|
|
|
416
452
|
type: Inject,
|
|
417
453
|
args: [PLATFORM_ID]
|
|
418
454
|
}] }, { type: i1.TpSurveyLogicService }, { type: i2.TpSurveyInterviewService }] });
|
|
419
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tp.survey.core.service.js","sourceRoot":"","sources":["../../../../../../projects/tp.survey/src/lib/core/tp.survey.core/tp.survey.core.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {DestroyRef, inject, Inject, Injectable, PLATFORM_ID} from '@angular/core';\r\nimport {BehaviorSubject, filter, Observable, Subject} from 'rxjs';\r\nimport {SurveyQuotas} from '../tp.servey.quota/tp.survey.quota';\r\nimport {TpSurveyLogicService} from '../tp.survey.logic/tp.survey.logic.service';\r\nimport {isPlatformBrowser} from '@angular/common';\r\nimport {map} from 'rxjs/operators';\r\nimport {AnswerChange, IdbQuestion, IdbQuestionary, IInterviewData, IInterviewLogicVar, IQuestionAnswer} from '../../tp.survey.interface';\r\nimport {TpSurveyInterviewService} from '../tp.survey.interview/tp.survey.interview.service';\r\nimport {QuestionsType, ResultativeStatus, SurveyEvents, ValidStatus} from '../../tp.survey.enum';\r\nimport {Answer} from '../tp.survey.answer/tp.survey.answer';\r\nimport {AnswerSlType, LogicObject} from '../../tp.survey.type';\r\nimport {MISSING_VALUE} from '../../tp.survey.const';\r\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\r\n\r\ninterface BroadcastEvent {\r\n  key: string;\r\n  data?: any;\r\n}\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class TpSurveyCoreService {\r\n  readonly #eventBus = new Subject<BroadcastEvent>();\r\n  readonly #currentQuestion = new BehaviorSubject<IdbQuestion | null>(null);\r\n  #surveyQuotas = new SurveyQuotas();\r\n  #questionary?: IdbQuestionary;\r\n  #displayedQuestions: Array<number> = []; // нужно для навигации по кнопке back\r\n  readonly #destroyRef = inject(DestroyRef);\r\n  constructor(@Inject(PLATFORM_ID) private readonly platformId: object, private readonly _logic: TpSurveyLogicService, private readonly _interview: TpSurveyInterviewService) {\r\n    // запускам проверку квот, после скрытия каждого(почти) вопроса (seQuestionHide)\r\n    this.#eventBus.asObservable().pipe(\r\n     filter(ev => ev.key === SurveyEvents.seQuestionHide ),\r\n     takeUntilDestroyed(this.#destroyRef),\r\n    ).subscribe({\r\n      next: () => this.#surveyQuotas.findFilledQuotas(this.#currentQuestion.value?.QuestionNum ?? -1, Object.assign({}, this._logic.getLogicVars()), this._interview.getAnswerList())\r\n    });\r\n  }\r\n\r\n  setupSurvey(questionaryRawData: IdbQuestionary, logicVars?: Array<IInterviewLogicVar>, answerList?: Array<IQuestionAnswer>, questionGuid?: number) {\r\n    //если анкеты нет, делать нечего.\r\n    if (!questionaryRawData) {\r\n      this.#currentQuestion.next(null);\r\n      return;\r\n    }\r\n\r\n    this.#questionary = questionaryRawData;\r\n\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      //узнаем есть ли вопрос который указали для перехода при заполнении квоты\r\n      const destinationQuestion = this.#findQuestionByGuid(questionaryRawData.BaseParams.QuotaLimitDestinationQuestionGuid);\r\n\r\n      //инициализируем квоты\r\n      this.#surveyQuotas.setQuotas(questionaryRawData.Quotas, questionaryRawData.Quotas2d, destinationQuestion?.QuestionGuid ?? -1, destinationQuestion?.QuestionNum ?? -1);\r\n    }\r\n\r\n    //создадим ответы и логические переменные\r\n    this.#initiateQuestions(this.#questionary.Questions)\r\n\r\n    // Может быть 2 варианта запуска интервью, 1-новое интервью, 2-перезапуск старого интервью. В первом случае начало это lang или первый вопрос. Во втором, с последнего вопроса на который ответили.\r\n    let startQuestionNum: number | undefined = undefined;\r\n    //если загружаем ранее не законченное интервью, продолжаем с места последнего вопроса (случай номер 2)\r\n    if (!!logicVars && !!answerList && !!questionGuid) {\r\n      //загружаем старые логические переменные и ответы\r\n      this.#fillOldInterviewData(logicVars, answerList);\r\n\r\n      //ищем номер вопроса, который был последний перед сохранением.\r\n      const questionLastSave = this.#findQuestionByGuid(questionGuid);\r\n\r\n      //Вопрос могли удалить. Тогда начнем сначала.\r\n      if (!!questionLastSave) {\r\n        startQuestionNum = questionLastSave.QuestionNum;\r\n      }\r\n    }\r\n    if (startQuestionNum === undefined) {\r\n      //Это новое интервью, без ранее сохранённых данных(случай 1). Запускаем опрос, первый экран это Lang вопрос, но если выбора языка нет, пропускаем и переходим на 1-й вопрос\r\n      startQuestionNum = this.#activeLangCount > 1 ? 0 : this.#nextQuestionNum();\r\n    }\r\n\r\n    //пошлём событие о начале опроса\r\n    this.#eventBus.next({key: SurveyEvents.seSurveyStart, data: {questionary: this.#questionary}});\r\n\r\n    //показываем вопрос\r\n    this.#changeCurrentQuestion(this.#questionary!.Questions[startQuestionNum]);\r\n  }\r\n\r\n  on(...key: Array<string>): Observable<any> {\r\n    return this.#eventBus.asObservable().pipe(\r\n     filter(event => key.includes(event.key)),\r\n     map(event => event.data)\r\n    );\r\n  }\r\n\r\n  goToNextQuestion(): void {\r\n    const currentQuestion = this.#currentQuestion.value;\r\n    if (!currentQuestion) {\r\n      return;\r\n    }\r\n    //если ответ на текущий вопрос не закончен, выходим\r\n    if (!this._interview.questionIsComplete(currentQuestion.QuestionGuid)) {\r\n      return ;\r\n    }\r\n\r\n    //если указан PostScript, выполним его\r\n    if (!!currentQuestion.PostScript) {\r\n      this._logic.executePostScript(currentQuestion.PostScript);\r\n    }\r\n    this.#changeCurrentQuestion(this.#questionary!.Questions[this.#nextQuestionNum()]);\r\n  }\r\n\r\n  goToBackQuestion(): void {\r\n    this.#displayedQuestions.pop(); // удалим информацию о том что показывали этот вопрос\r\n    const previousQuestionNum = this.#displayedQuestions.pop(); // узнаем номер вопроса на который нам надо перейти, и удалим информацию что-бы не дублировать\r\n    if (!!previousQuestionNum) {\r\n      const prevQuestion = this.#questionary?.Questions[previousQuestionNum];\r\n      if (!!prevQuestion) {\r\n        this.#changeCurrentQuestion(prevQuestion);\r\n      }\r\n    }\r\n  }\r\n\r\n  questionIsComplete(guid: number): boolean {\r\n    const answer = this._interview.findAnswerByGuid(guid);\r\n    return answer?.answerIsComplete ?? false\r\n  }\r\n  \r\n  guidToTextAnswer(guid: number, languageId: number): string {\r\n    const answer = this.findAnswerByGuid(guid);\r\n    const question = this.#findQuestionByGuid(guid);\r\n    if (!question || !answer || answer.AnswerValue === -97) {\r\n      return ''\r\n    }\r\n    const questionType = question.QuestionType;\r\n    console.log(0);\r\n    switch (questionType) {\r\n      \r\n      case QuestionsType.eqOL: return answer.AnswerValue;\r\n      case QuestionsType.eqSS: {\r\n        const alternative = question.Alternativelist.find(alt => alt.Value === answer.AnswerValue);\r\n        const altText = alternative?.MultiLangCaptions.find(obj => obj.lang === languageId);\r\n        return (altText?.CaptionHtml || alternative?.alias) ?? '';\r\n      }\r\n      case QuestionsType.eqMS: {\r\n        const answerArray = answer.AnswerValue as Array<number>;\r\n        const alternativeArray = question.Alternativelist.filter(alt => answerArray.indexOf(alt.Value) > 0);\r\n        const altTextArray = alternativeArray.map(alt => (alt.MultiLangCaptions.find(obj => obj.lang === languageId)?.CaptionHtml || alt.alias) ?? '');\r\n        return altTextArray.join(', ');\r\n      }\r\n      default: return '';\r\n    }\r\n  }\r\n\r\n  #fillOldInterviewData(logicVars: Array<IInterviewLogicVar>, answerList: Array<IQuestionAnswer>): void{\r\n    //пропишем старые логические переменные\r\n    if (Array.isArray(logicVars)) {\r\n      logicVars.forEach(logicVarObj => {\r\n        this._logic.setLogicValue(logicVarObj.nm, logicVarObj.vl);\r\n      })\r\n    }\r\n\r\n    //пропишем старые ответы\r\n    this._interview.overrideAnswers(answerList);\r\n\r\n    //нужно вернуть время начала интервью, отдельно оно не сохраняется, поэтому возьмём минимальное время начала из ответов\r\n    //фильтруем все вопросы у которых оно указано(>0) и трансформируем в число.\r\n    const startedAtArray =  answerList.filter(answer => answer.StartedAt > 0).map(answer => answer.StartedAt);\r\n\r\n    //берём минимальное\r\n    this._interview.startedAt = Math.min(...startedAtArray);\r\n  }\r\n\r\n  #changeCurrentQuestion(newQuestion: IdbQuestion): void {\r\n    const currentQuestion = this.#currentQuestion.value;\r\n    const currentTime = new Date().getTime();\r\n\r\n    if (!!currentQuestion) {\r\n      /*\r\n      Сохраним время начала интервью. Фиксировать будем не с самого начала, а с ответа на первый вопрос.\r\n      Потому что практика такая, открыть анкету на планшете и ходить искать респондентов и из-за этого длительность интервью может быть нерелевантно.\r\n      * */\r\n      if (this._interview.startedAt === 0 && currentQuestion.QuestionNum >= 0) {\r\n        this._interview.startedAt = currentTime;\r\n      }\r\n\r\n      //нужно сохранить FinishedAt в ответах для старого вопроса\r\n      const answer = this._interview.findAnswerByGuid(currentQuestion.QuestionGuid);\r\n      if (!!answer) {\r\n        answer.FinishedAt = currentTime;\r\n      }\r\n      this.#eventBus.next({key: SurveyEvents.seQuestionHide, data: currentQuestion});\r\n    }\r\n    this.#displayedQuestions.push(newQuestion.QuestionNum);\r\n    this.#currentQuestion.next(newQuestion);\r\n\r\n    //нужно сохранить StartedAt в ответах для нового вопроса\r\n    const answerNew = this._interview.findAnswerByGuid(newQuestion.QuestionGuid);\r\n    if (!!answerNew) {\r\n      answerNew.StartedAt = currentTime;\r\n    }\r\n    this.#eventBus.next({key: SurveyEvents.seQuestionShow, data: newQuestion});\r\n\r\n    //если новый экран это end тогда завершаем интервью\r\n    if (newQuestion.QuestionType === 'END') {\r\n      this.#finishInterview();\r\n    }\r\n    \r\n    //если это неявный вопрос и это ss или ms вопрос тогда посчитаем значения у этого вопроса\r\n    if (newQuestion.Implicit) {\r\n      this.#calculateImplicitQuestion(newQuestion);\r\n    }\r\n  }\r\n\r\n  currentQuestion$(): Observable<IdbQuestion | null> {\r\n    return this.#currentQuestion.asObservable();\r\n  }\r\n\r\n  findAnswerByGuid(guid: number): Answer | undefined {\r\n    return this._interview.findAnswerByGuid(guid);\r\n  }\r\n\r\n  setLogicValue(varName: string, value: any): void {\r\n    this._logic.setLogicValue(varName, value);\r\n  }\r\n\r\n  getLogicVars(): LogicObject {\r\n    return this._logic.getLogicVars();\r\n  }\r\n\r\n  getAnswerList(): Array<Answer> {\r\n    return this._interview.getAnswerList();\r\n  }\r\n\r\n  updateAnswer(eventData: AnswerChange): void {\r\n    const answer = this._interview.findAnswerByGuid(eventData.guid);\r\n    if (!answer) {\r\n      return;\r\n    }\r\n\r\n    //вопросы которые в своем составе имеет секции, хранят ответы чуть по другому, точнее ответ на каждую секцию это отдельный объект в  _interview.answerList\r\n    //поэтому ответ для таких типов вопросов это [key: string]: Array<number> | number, где key(guid секции) это ответ на секцию, если key нет, секция не была показана\r\n    if (this.#currentQuestion.value?.QuestionType === QuestionsType.eqSL) {\r\n      const sectionList = this.#currentQuestion.value?.Sectionlist;\r\n\r\n      //пройдёмся по каждой секции и если для неё есть ответ, запишем, в противном случае вместо ответа укажем -97(MISSING_VALUE)\r\n      if (!!sectionList) {\r\n        sectionList.forEach(section => {\r\n          const sectionInterviewAnswer =  this._interview.findAnswerByGuid(section.SectionGuid);\r\n          if (!sectionInterviewAnswer){\r\n            return\r\n          }\r\n          const sectionQuestionAnswer = eventData.answer ?  (eventData.answer as AnswerSlType)[section.SectionGuid.toString()] : undefined;\r\n          sectionInterviewAnswer.AnswerValue = sectionQuestionAnswer === undefined ? MISSING_VALUE : sectionQuestionAnswer;\r\n        });\r\n      }\r\n      //ответ самого исходного вопроса должен быть 1\r\n      answer.AnswerValue = 1;\r\n    } else {\r\n      //ответ на вопрос у которого нет секций\r\n      answer.AnswerValue = eventData.answer;\r\n    }\r\n\r\n    //на вопрос ответили тогда когда есть все ответы и выполняется логика кнопки далее(если её указали)\r\n    answer.answerIsComplete = eventData.isComplete && this.#nextButtonExpressionIsTrue();\r\n\r\n    //ViewedAlternativeValues и ViewedSectionGuids. Т.к. они readonly сначала удалим, потом добавим\r\n    answer.ViewedAlternativeValues.length = 0;\r\n    eventData.viewedAlternativeValues.forEach(oa => answer.ViewedAlternativeValues.push(oa));\r\n\r\n    answer.ViewedSectionGuids.length = 0;\r\n    eventData.viewedSectionGuids.forEach(oa => answer.ViewedSectionGuids.push(oa));\r\n\r\n    answer.AlternativeSequenceClicks.length = 0;\r\n    eventData.alternativeSequenceClicks.forEach(oa => answer.AlternativeSequenceClicks.push(oa));\r\n\r\n\r\n    Object.entries(answer.AnswerAdditional).forEach(([key]) => delete answer.AnswerAdditional[Number(key)]);\r\n    Object.entries(eventData.additionalAnswers).forEach(([key, value]) => answer.AnswerAdditional[Number(key)] = value);\r\n\r\n    this.updateLogic(eventData.logicVars);\r\n\r\n    this.#eventBus.next({key: SurveyEvents.seQuestionAnswersChange, data: {question: this.#currentQuestion.value, answerIsComplete: answer.answerIsComplete}});\r\n\r\n    //может возникнуть ситуация, когда нет ни одной альтернативы или секции для отображения.в этом случае вопрос показывать не нужно, как ответ записываем 0 и переходим на следующий\r\n    const question = this.#findQuestionByGuid(eventData.guid);\r\n    //ситуация для ss/ms вопросов,\r\n    if (!!question && answer.answerIsComplete && (question.QuestionType === 'SS' || question.QuestionType === 'MS') && (eventData.viewedAlternativeValues.length === 0)) {\r\n      this.goToNextQuestion()\r\n    }\r\n  }\r\n\r\n  updateLogic(logicVars: Record<string, any>): void{\r\n    Object.entries(logicVars).forEach(([varName, value]) => {\r\n      this.setLogicValue(varName, value)\r\n    })\r\n  }\r\n\r\n  //возвращает количество активных языков\r\n  get #activeLangCount(): number {\r\n    //если анкета по какой-то причине ну указана, вернём 0\r\n    return this.#questionary?.BaseParams.LanguageList.filter( langObj => langObj.isActive).length ?? 0;\r\n  }\r\n\r\n  //возвращает количество вопросов\r\n  get #questionsCount(): number {\r\n    return this.#questionary?.Questions.length ?? 0;\r\n  }\r\n\r\n  #finishInterview(): void {\r\n    const currentTime = new Date();\r\n    let resultDirty;\r\n    try {\r\n      resultDirty = this._logic.executeLogic(this.#questionary?.BaseParams.ResultativeScript ?? '');\r\n    }catch (e) {\r\n      resultDirty = false;\r\n    }\r\n\r\n    const interview: IInterviewData = {\r\n      Language: 0,    //todo lang\r\n      SpotId: 0,\r\n      DeviceId: 0,\r\n      DeviceType: 0,\r\n      Resultative: ResultativeStatus.rsUnset,\r\n      Valid: ValidStatus.vsUnset,\r\n      Wave: this.#questionary?.Info.current_wave ?? 0,\r\n      QuestionaryName: this.#questionary?.Info.name ?? '',\r\n      QuestionaryGuid: this.#questionary?.Info.questionary_guid ?? 0,\r\n      StartedAt: this._interview.startedAt,\r\n      FinishedAt: currentTime.getTime(),\r\n      TineZoneBias: currentTime.getTimezoneOffset(),\r\n      ResultatDirty: resultDirty,\r\n      Answers: {\r\n        AnsversList: Array.from(this._interview.getAnswerList())\r\n      },\r\n      additional_info: {\r\n        QuestionaryUpdateTimestamp: new Date(this.#questionary?.Info.last_update_timestamp ?? 0).getTime() ,\r\n      },\r\n      ResultQuotaGuids: [],\r\n      ResultQuota2dGuids: [],\r\n      LogicVars: [],\r\n    };\r\n\r\n    const logicVars = this._logic.getLogicVars();\r\n    Object.entries(logicVars).forEach(([key, value]) => interview.LogicVars.push({nm: key, vl: value}));\r\n\r\n    this.#eventBus.next({key: SurveyEvents.seSurveyEnd, data: interview});\r\n  }\r\n\r\n  #nextQuestionNum(): number {\r\n    const currentQuestion = this.#currentQuestion.value;\r\n    // проверим квоты\r\n    const currentGuid = currentQuestion?.QuestionGuid ?? 0;\r\n    if (this.#surveyQuotas.quotaIsCompleted && !this.#surveyQuotas.guidIsDestinationQuestion(currentGuid)) {\r\n      // у нас есть заполненная квота, найдем вопрос который нужно показать при заполнении квоты\r\n      const destinationQ = this.#findQuestionByGuid(this.#surveyQuotas.destinationQuestionGuid);\r\n      //если вопрос есть(его могли удалить из анкеты), показывает его\r\n      if (!!destinationQ) {\r\n        return destinationQ.QuestionNum;\r\n      }\r\n    }\r\n\r\n    // если нет заполненной квоты, переходим на следующий вопрос, ищем следующий вопрос который мы можем отобразить (до логика может запретить показ вопроса)\r\n    let nextQuestionNum = (currentQuestion?.QuestionNum ?? 0) + 1;\r\n\r\n    // но сначала проверим после логику\r\n    if (!!currentQuestion) {\r\n      const destQ = currentQuestion.logiclist.find(logObj => (logObj.DestinationQuestionGuid > 0) && this._logic.executeLogic(logObj.LogicText) );\r\n      if (!!destQ) {\r\n        const destQNum = this.#findQuestionByGuid(destQ.DestinationQuestionGuid)?.QuestionNum;\r\n        if (Number.isSafeInteger(destQNum)) { // вопрос мог быть удалён, а логику оставить\r\n          nextQuestionNum = destQNum!;\r\n        }\r\n      }\r\n    }\r\n\r\n    // общее количество вопросов, цикл будет считать до этого количества\r\n    const questionCount = this.#questionsCount;\r\n    do {\r\n      //потенциально следующий вопрос\r\n      const q = this.#questionary!.Questions[nextQuestionNum];\r\n\r\n      //если это последний вопрос или если выполняется BeforeExpression и этот вопрос не DestinationQuota, можем его показать\r\n      if (q.QuestionType === QuestionsType.eqEND || (this._logic.executeLogic(q.BeforeExpression) && !this.#surveyQuotas.guidIsDestinationQuestion(q.QuestionGuid)) ) {\r\n        return nextQuestionNum;\r\n      }\r\n      //вопрос нельзя показывать, увеличиваем счётчик текущего вопроса на 1\r\n      nextQuestionNum++;\r\n    } while (nextQuestionNum < questionCount - 1); // end вопрос должен всегда быть показан\r\n    return nextQuestionNum;\r\n  }\r\n\r\n  #initiateQuestions(questionList: Array<IdbQuestion>): void{\r\n    questionList.forEach(question => {\r\n      //создаём пустой ответ для вопроса.\r\n      const answer = this._interview.createAnswer(question.QuestionGuid, question.Version);\r\n\r\n      //такого быть не должно, это ошибка\r\n      if (!answer) {\r\n        throw new Error('answer not found');\r\n      }\r\n\r\n      this._logic.setLogicValue(question.SysName, MISSING_VALUE); // переменная самого вопроса\r\n\r\n      //добавим доп ответы и логику для них\r\n      question.AdditionalAnswerlist.forEach(aa => {\r\n        answer.AnswerAdditional[aa.Value] = null;\r\n        this._logic.setLogicValue(`${question.SysName}${aa.SysName}`, MISSING_VALUE); // логика для переменных доп ответа\r\n      });\r\n\r\n      // если в тексте postScript есть запись типа setVar('varName',  value);  нужно инициализировать переменную varName значением -97.\r\n      if (!!question.PostScript) {\r\n        (question.PostScript.match(/(\\bsetvar\\b)\\(['|\"](.*?)['|\"]/gim) ?? []).map(scriptVar => scriptVar.match(/['|\"](.+)['|\"]/i)![1]).forEach(varName => this._logic.setLogicValue(varName, null));\r\n      }\r\n\r\n      // для Вопросов SS/MS нужно прописать логику альтернатив sysname вопроса + sysname альтернативы\r\n      if (question.Sectionlist.length <= 0) {// для Вопросов SS/MS\r\n        question.Alternativelist.forEach(alt => this._logic.setLogicValue(`${question.SysName}${alt.SysName}`, MISSING_VALUE));\r\n      } else {\r\n        //для вопросов у которых есть секции проинициализируем каждую секцию\r\n        question.Sectionlist.forEach(section => {\r\n\r\n          //создаём пустой ответ для секции.\r\n          const answerForSection = this._interview.createAnswer(section.SectionGuid, 1);\r\n\r\n          //такого быть не должно, это ошибка\r\n          if (!answerForSection) {\r\n            throw new Error('answer not found');\r\n          }\r\n\r\n          // для Вопросов SL нужно прописать логику альтернатив и секций sysname секции + sysname альтернативы\r\n          this._logic.setLogicValue(section.SysName, MISSING_VALUE); // переменная самой секции\r\n\r\n          //логика альтернатив SL вопроса\r\n          question.Alternativelist.forEach(alt => this._logic.setLogicValue(`${section.SysName}${alt.SysName}`, MISSING_VALUE));\r\n        })\r\n      }\r\n    })\r\n  }\r\n\r\n  //ищет вопрос на guid\r\n  #findQuestionByGuid(guid: number): IdbQuestion | undefined {\r\n    if (!this.#questionary) {\r\n      return undefined;\r\n    }\r\n    return this.#questionary.Questions.find(q => q.QuestionGuid === guid);\r\n  }\r\n\r\n  #nextButtonExpressionIsTrue(): boolean{\r\n    const buttonNextExpression = this.#currentQuestion.value?.BtnNextExpression ?? '';\r\n    //если скрипта для кнопки далее нет, вернём true\r\n    if (buttonNextExpression.trim() === '') {\r\n      return true;\r\n    }\r\n    try {\r\n      return this._logic.executeLogic(buttonNextExpression);\r\n    } catch (e) {\r\n      console.error(e);\r\n      return false;\r\n    }\r\n  }\r\n\r\n  #calculateImplicitQuestion(question: IdbQuestion): void{\r\n    if (question.Implicit && (question.QuestionType === QuestionsType.eqSS || question.QuestionType === QuestionsType.eqMS) ) {\r\n      let questionAnswer: number | Array<number>;\r\n\r\n      if (question.QuestionType === QuestionsType.eqSS) {\r\n        const alt = question.Alternativelist.find(alt => this._logic.executeLogic(alt.EnableExpression));\r\n        questionAnswer = !!alt ? alt.Value : 0;\r\n      }\r\n\r\n      if (question.QuestionType === QuestionsType.eqMS) {\r\n        questionAnswer = question.Alternativelist\r\n          .filter(alt => this._logic.executeLogic(alt.EnableExpression))\r\n          .map(alt => alt.Value);\r\n      }\r\n\r\n      const answerChange: AnswerChange = {\r\n        guid: question.QuestionGuid,\r\n        isComplete: true,\r\n        answer: questionAnswer!,\r\n        additionalAnswers: {},\r\n        logicVars: {},\r\n        viewedAlternativeValues: [],\r\n        viewedSectionGuids: [],\r\n        alternativeSequenceClicks: []\r\n      }\r\n      this.updateAnswer(answerChange);\r\n    }\r\n\r\n  }\r\n}\r\n"]}
|
|
455
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tp.survey.core.service.js","sourceRoot":"","sources":["../../../../../../projects/tp.survey/src/lib/core/tp.survey.core/tp.survey.core.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {DestroyRef, inject, Inject, Injectable, PLATFORM_ID} from '@angular/core';\r\nimport {BehaviorSubject, filter, Observable, Subject} from 'rxjs';\r\nimport {SurveyQuotas} from '../tp.servey.quota/tp.survey.quota';\r\nimport {TpSurveyLogicService} from '../tp.survey.logic/tp.survey.logic.service';\r\nimport {isPlatformBrowser} from '@angular/common';\r\nimport {map} from 'rxjs/operators';\r\nimport {AnswerChange, IdbQuestion, IdbQuestionary, IInterviewData, IInterviewLogicVar, IQuestionAnswer} from '../../tp.survey.interface';\r\nimport {TpSurveyInterviewService} from '../tp.survey.interview/tp.survey.interview.service';\r\nimport {QuestionsType, ResultativeStatus, SurveyEvents, ValidStatus} from '../../tp.survey.enum';\r\nimport {Answer} from '../tp.survey.answer/tp.survey.answer';\r\nimport {AnswerSlType, LogicObject} from '../../tp.survey.type';\r\nimport {MISSING_VALUE} from '../../tp.survey.const';\r\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\r\n\r\ninterface BroadcastEvent {\r\n  key: string;\r\n  data?: any;\r\n}\r\n\r\ninterface ILogicVarContainer {\r\n  [questionGuid: string]: Set<string>;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class TpSurveyCoreService {\r\n  readonly #eventBus = new Subject<BroadcastEvent>();\r\n  readonly #currentQuestion = new BehaviorSubject<IdbQuestion | null>(null);\r\n  #surveyQuotas = new SurveyQuotas();\r\n  #questionary?: IdbQuestionary;\r\n  #displayedQuestions: Array<number> = []; // нужно для навигации по кнопке back\r\n  #questionLogicVarContainer: ILogicVarContainer = {}; //Нужно знать какая логическая переменная относится к какому вопросу. НапримерЮ когда нужно ее отчистить при переходе назад.\r\n  readonly #destroyRef = inject(DestroyRef);\r\n  constructor(@Inject(PLATFORM_ID) private readonly platformId: object, private readonly _logic: TpSurveyLogicService, private readonly _interview: TpSurveyInterviewService) {\r\n    // запускам проверку квот, после скрытия каждого(почти) вопроса (seQuestionHide)\r\n    this.#eventBus.asObservable().pipe(\r\n     filter(ev => ev.key === SurveyEvents.seQuestionHide ),\r\n     takeUntilDestroyed(this.#destroyRef),\r\n    ).subscribe({\r\n      next: () => this.#surveyQuotas.findFilledQuotas(this.#currentQuestion.value?.QuestionNum ?? -1, Object.assign({}, this._logic.getLogicVars()), this._interview.getAnswerList())\r\n    });\r\n  }\r\n\r\n  setupSurvey(questionaryRawData: IdbQuestionary, logicVars?: Array<IInterviewLogicVar>, answerList?: Array<IQuestionAnswer>, questionGuid?: number) {\r\n    //если анкеты нет, делать нечего.\r\n    if (!questionaryRawData) {\r\n      this.#currentQuestion.next(null);\r\n      return;\r\n    }\r\n\r\n    this.#questionary = questionaryRawData;\r\n\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      //узнаем есть ли вопрос который указали для перехода при заполнении квоты\r\n      const destinationQuestion = this.#findQuestionByGuid(questionaryRawData.BaseParams.QuotaLimitDestinationQuestionGuid);\r\n\r\n      //инициализируем квоты\r\n      this.#surveyQuotas.setQuotas(questionaryRawData.Quotas, questionaryRawData.Quotas2d, destinationQuestion?.QuestionGuid ?? -1, destinationQuestion?.QuestionNum ?? -1);\r\n    }\r\n\r\n    //создадим ответы и логические переменные\r\n    this.#initiateQuestions(this.#questionary.Questions)\r\n\r\n    // Может быть 2 варианта запуска интервью, 1-новое интервью, 2-перезапуск старого интервью. В первом случае начало это lang или первый вопрос. Во втором, с последнего вопроса на который ответили.\r\n    let startQuestionNum: number | undefined = undefined;\r\n    //если загружаем ранее не законченное интервью, продолжаем с места последнего вопроса (случай номер 2)\r\n    if (!!logicVars && !!answerList && !!questionGuid) {\r\n      //загружаем старые логические переменные и ответы\r\n      this.#fillOldInterviewData(logicVars, answerList);\r\n\r\n      //ищем номер вопроса, который был последний перед сохранением.\r\n      const questionLastSave = this.#findQuestionByGuid(questionGuid);\r\n\r\n      //Вопрос могли удалить. Тогда начнем сначала.\r\n      if (!!questionLastSave) {\r\n        startQuestionNum = questionLastSave.QuestionNum;\r\n      }\r\n    }\r\n    if (startQuestionNum === undefined) {\r\n      //Это новое интервью, без ранее сохранённых данных(случай 1). Запускаем опрос, первый экран это Lang вопрос, но если выбора языка нет, пропускаем и переходим на 1-й вопрос\r\n      startQuestionNum = this.#activeLangCount > 1 ? 0 : this.#nextQuestionNum();\r\n    }\r\n\r\n    //пошлём событие о начале опроса\r\n    this.#eventBus.next({key: SurveyEvents.seSurveyStart, data: {questionary: this.#questionary}});\r\n\r\n    //показываем вопрос\r\n    this.#changeCurrentQuestion(this.#questionary!.Questions[startQuestionNum]);\r\n  }\r\n\r\n  on(...key: Array<string>): Observable<any> {\r\n    return this.#eventBus.asObservable().pipe(\r\n     filter(event => key.includes(event.key)),\r\n     map(event => event.data)\r\n    );\r\n  }\r\n\r\n  goToNextQuestion(): void {\r\n    const currentQuestion = this.#currentQuestion.value;\r\n    if (!currentQuestion) {\r\n      return;\r\n    }\r\n    //если ответ на текущий вопрос не закончен, выходим\r\n    if (!this._interview.questionIsComplete(currentQuestion.QuestionGuid)) {\r\n      return ;\r\n    }\r\n\r\n    //если указан PostScript, выполним его\r\n    if (!!currentQuestion.PostScript) {\r\n      this._logic.executePostScript(currentQuestion.PostScript);\r\n    }\r\n    this.#changeCurrentQuestion(this.#questionary!.Questions[this.#nextQuestionNum()]);\r\n  }\r\n\r\n  goToBackQuestion(): void {\r\n    this.#displayedQuestions.pop(); // удалим информацию о том что показывали этот вопрос\r\n    const previousQuestionNum = this.#displayedQuestions.pop(); // узнаем номер вопроса на который нам надо перейти, и удалим информацию что-бы не дублировать\r\n    if (!!previousQuestionNum) {\r\n      const prevQuestion = this.#questionary?.Questions[previousQuestionNum];\r\n      if (!!prevQuestion) {\r\n        this.#clearAnswerForCurrentQuestion();// отчистим ответ на вопрос, если он уже есть.\r\n        this.#changeCurrentQuestion(prevQuestion);\r\n      }\r\n    }\r\n  }\r\n\r\n  questionIsComplete(guid: number): boolean {\r\n    const answer = this._interview.findAnswerByGuid(guid);\r\n    return answer?.answerIsComplete ?? false\r\n  }\r\n  \r\n  /*\r\n  * функция по номеру вопроса, ищет ответ на этот guid, и возвращает ответ на этот вопрос.\r\n  * если это ss ms, вместо номеров альтернатив подставляет текст.\r\n  * работает только для eqOL, eqSS, eqMS\r\n  * */\r\n  guidToTextAnswer(guid: number, languageId: number): string {\r\n    const answer = this.findAnswerByGuid(guid);\r\n    const question = this.#findQuestionByGuid(guid);\r\n    \r\n    //если ответа нет, ревнём пустую строку\r\n    if (!question || !answer || answer.AnswerValue === -97) {\r\n      return ''\r\n    }\r\n    const questionType = question.QuestionType;\r\n    \r\n    switch (questionType) {\r\n      \r\n      case QuestionsType.eqOL: return answer.AnswerValue;\r\n      case QuestionsType.eqSS: {\r\n        //находим альтернативу и подставляем заголовок альтернативы с указанным языком, если заголовка нет, вернём alias\r\n        const alternative = question.Alternativelist.find(alt => alt.Value === answer.AnswerValue);\r\n        const altText = alternative?.MultiLangCaptions.find(obj => obj.lang === languageId);\r\n\r\n        return (altText?.CaptionHtml || alternative?.alias) ?? '';\r\n      }\r\n      case QuestionsType.eqMS: {\r\n        //всё то же самое, что и ss, только тут массив ответов,\r\n        const answerArray = answer.AnswerValue as Array<number>;\r\n        const alternativeArray = question.Alternativelist.filter(alt => answerArray.indexOf(alt.Value) > 0);\r\n        const altTextArray = alternativeArray.map(alt => (alt.MultiLangCaptions.find(obj => obj.lang === languageId)?.CaptionHtml || alt.alias) ?? '');\r\n        return altTextArray.join(', ');\r\n      }\r\n      default: return '';\r\n    }\r\n  }\r\n\r\n  #fillOldInterviewData(logicVars: Array<IInterviewLogicVar>, answerList: Array<IQuestionAnswer>): void{\r\n    //пропишем старые логические переменные\r\n    if (Array.isArray(logicVars)) {\r\n      logicVars.forEach(logicVarObj => {\r\n        this._logic.setLogicValue(logicVarObj.nm, logicVarObj.vl);\r\n      })\r\n    }\r\n\r\n    //пропишем старые ответы\r\n    this._interview.overrideAnswers(answerList);\r\n\r\n    //нужно вернуть время начала интервью, отдельно оно не сохраняется, поэтому возьмём минимальное время начала из ответов\r\n    //фильтруем все вопросы у которых оно указано(>0) и трансформируем в число.\r\n    const startedAtArray =  answerList.filter(answer => answer.StartedAt > 0).map(answer => answer.StartedAt);\r\n\r\n    //берём минимальное\r\n    this._interview.startedAt = Math.min(...startedAtArray);\r\n  }\r\n\r\n  #changeCurrentQuestion(newQuestion: IdbQuestion): void {\r\n    const currentQuestion = this.#currentQuestion.value;\r\n    const currentTime = new Date().getTime();\r\n\r\n    if (!!currentQuestion) {\r\n      /*\r\n      Сохраним время начала интервью. Фиксировать будем не с самого начала, а с ответа на первый вопрос.\r\n      Потому что практика такая, открыть анкету на планшете и ходить искать респондентов и из-за этого длительность интервью может быть нерелевантно.\r\n      * */\r\n      if (this._interview.startedAt === 0 && currentQuestion.QuestionNum >= 0) {\r\n        this._interview.startedAt = currentTime;\r\n      }\r\n\r\n      //нужно сохранить FinishedAt в ответах для старого вопроса\r\n      const answer = this._interview.findAnswerByGuid(currentQuestion.QuestionGuid);\r\n      if (!!answer) {\r\n        answer.FinishedAt = currentTime;\r\n      }\r\n      this.#eventBus.next({key: SurveyEvents.seQuestionHide, data: currentQuestion});\r\n    }\r\n    this.#displayedQuestions.push(newQuestion.QuestionNum);\r\n    this.#currentQuestion.next(newQuestion);\r\n\r\n    //нужно сохранить StartedAt в ответах для нового вопроса\r\n    const answerNew = this._interview.findAnswerByGuid(newQuestion.QuestionGuid);\r\n    if (!!answerNew) {\r\n      answerNew.StartedAt = currentTime;\r\n    }\r\n    this.#eventBus.next({key: SurveyEvents.seQuestionShow, data: newQuestion});\r\n\r\n    //если новый экран это end тогда завершаем интервью\r\n    if (newQuestion.QuestionType === 'END') {\r\n      this.#finishInterview();\r\n    }\r\n    \r\n    //если это неявный вопрос и это ss или ms вопрос тогда посчитаем значения у этого вопроса\r\n    if (newQuestion.Implicit) {\r\n      this.#calculateImplicitQuestion(newQuestion);\r\n    }\r\n  }\r\n\r\n  currentQuestion$(): Observable<IdbQuestion | null> {\r\n    return this.#currentQuestion.asObservable();\r\n  }\r\n\r\n  findAnswerByGuid(guid: number): Answer | undefined {\r\n    return this._interview.findAnswerByGuid(guid);\r\n  }\r\n\r\n  setLogicValue(varName: string, value: any): void {\r\n    this._logic.setLogicValue(varName, value);\r\n  }\r\n\r\n  getLogicVars(): LogicObject {\r\n    return this._logic.getLogicVars();\r\n  }\r\n\r\n  getAnswerList(): Array<Answer> {\r\n    return this._interview.getAnswerList();\r\n  }\r\n\r\n  updateAnswer(eventData: AnswerChange): void {\r\n    const answer = this._interview.findAnswerByGuid(eventData.guid);\r\n    if (!answer) {\r\n      return;\r\n    }\r\n\r\n    //вопросы которые в своем составе имеет секции, хранят ответы чуть по другому, точнее ответ на каждую секцию это отдельный объект в  _interview.answerList\r\n    //поэтому ответ для таких типов вопросов это [key: string]: Array<number> | number, где key(guid секции) это ответ на секцию, если key нет, секция не была показана\r\n    if (this.#currentQuestion.value?.QuestionType === QuestionsType.eqSL) {\r\n      const sectionList = this.#currentQuestion.value?.Sectionlist;\r\n\r\n      //пройдёмся по каждой секции и если для неё есть ответ, запишем, в противном случае вместо ответа укажем -97(MISSING_VALUE)\r\n      if (!!sectionList) {\r\n        sectionList.forEach(section => {\r\n          const sectionInterviewAnswer =  this._interview.findAnswerByGuid(section.SectionGuid);\r\n          if (!sectionInterviewAnswer){\r\n            return\r\n          }\r\n          const sectionQuestionAnswer = eventData.answer ?  (eventData.answer as AnswerSlType)[section.SectionGuid.toString()] : undefined;\r\n          sectionInterviewAnswer.AnswerValue = sectionQuestionAnswer === undefined ? MISSING_VALUE : sectionQuestionAnswer;\r\n        });\r\n      }\r\n      //ответ самого исходного вопроса должен быть 1\r\n      answer.AnswerValue = 1;\r\n    } else {\r\n      //ответ на вопрос у которого нет секций\r\n      answer.AnswerValue = eventData.answer;\r\n    }\r\n\r\n    //на вопрос ответили тогда когда есть все ответы и выполняется логика кнопки далее(если её указали)\r\n    answer.answerIsComplete = eventData.isComplete && this.#nextButtonExpressionIsTrue();\r\n\r\n    //ViewedAlternativeValues и ViewedSectionGuids. Т.к. они readonly сначала удалим, потом добавим\r\n    answer.ViewedAlternativeValues.length = 0;\r\n    eventData.viewedAlternativeValues.forEach(oa => answer.ViewedAlternativeValues.push(oa));\r\n\r\n    answer.ViewedSectionGuids.length = 0;\r\n    eventData.viewedSectionGuids.forEach(oa => answer.ViewedSectionGuids.push(oa));\r\n\r\n    answer.AlternativeSequenceClicks.length = 0;\r\n    eventData.alternativeSequenceClicks.forEach(oa => answer.AlternativeSequenceClicks.push(oa));\r\n\r\n\r\n    Object.entries(answer.AnswerAdditional).forEach(([key]) => delete answer.AnswerAdditional[Number(key)]);\r\n    Object.entries(eventData.additionalAnswers).forEach(([key, value]) => answer.AnswerAdditional[Number(key)] = value);\r\n\r\n    this.updateLogic(eventData.logicVars);\r\n\r\n    this.#eventBus.next({key: SurveyEvents.seQuestionAnswersChange, data: {question: this.#currentQuestion.value, answerIsComplete: answer.answerIsComplete}});\r\n\r\n    //может возникнуть ситуация, когда нет ни одной альтернативы или секции для отображения.в этом случае вопрос показывать не нужно, как ответ записываем 0 и переходим на следующий\r\n    const question = this.#findQuestionByGuid(eventData.guid);\r\n    //ситуация для ss/ms вопросов и для неявных вопросов\r\n    if (!!question && answer.answerIsComplete && (question.QuestionType === 'SS' || question.QuestionType === 'MS') && ( (eventData.viewedAlternativeValues.length === 0) || question.Implicit)) {\r\n      this.goToNextQuestion()\r\n    }\r\n  }\r\n\r\n  updateLogic(logicVars: Record<string, any>): void{\r\n    Object.entries(logicVars).forEach(([varName, value]) => {\r\n      this.setLogicValue(varName, value)\r\n    })\r\n  }\r\n\r\n  //возвращает количество активных языков\r\n  get #activeLangCount(): number {\r\n    //если анкета по какой-то причине ну указана, вернём 0\r\n    return this.#questionary?.BaseParams.LanguageList.filter( langObj => langObj.isActive).length ?? 0;\r\n  }\r\n\r\n  //возвращает количество вопросов\r\n  get #questionsCount(): number {\r\n    return this.#questionary?.Questions.length ?? 0;\r\n  }\r\n\r\n  #finishInterview(): void {\r\n    const currentTime = new Date();\r\n    let resultDirty;\r\n    try {\r\n      resultDirty = this._logic.executeLogic(this.#questionary?.BaseParams.ResultativeScript ?? '');\r\n    }catch (e) {\r\n      resultDirty = false;\r\n    }\r\n\r\n    const interview: IInterviewData = {\r\n      Language: 0,    //todo lang\r\n      SpotId: 0,\r\n      DeviceId: 0,\r\n      DeviceType: 0,\r\n      Resultative: ResultativeStatus.rsUnset,\r\n      Valid: ValidStatus.vsUnset,\r\n      Wave: this.#questionary?.Info.current_wave ?? 0,\r\n      QuestionaryName: this.#questionary?.Info.name ?? '',\r\n      QuestionaryGuid: this.#questionary?.Info.questionary_guid ?? 0,\r\n      StartedAt: this._interview.startedAt,\r\n      FinishedAt: currentTime.getTime(),\r\n      TineZoneBias: currentTime.getTimezoneOffset(),\r\n      ResultatDirty: resultDirty,\r\n      Answers: {\r\n        AnsversList: Array.from(this._interview.getAnswerList())\r\n      },\r\n      additional_info: {\r\n        QuestionaryUpdateTimestamp: new Date(this.#questionary?.Info.last_update_timestamp ?? 0).getTime() ,\r\n      },\r\n      ResultQuotaGuids: [],\r\n      ResultQuota2dGuids: [],\r\n      LogicVars: [],\r\n    };\r\n\r\n    const logicVars = this._logic.getLogicVars();\r\n    Object.entries(logicVars).forEach(([key, value]) => interview.LogicVars.push({nm: key, vl: value}));\r\n\r\n    this.#eventBus.next({key: SurveyEvents.seSurveyEnd, data: interview});\r\n  }\r\n\r\n  #nextQuestionNum(): number {\r\n    const currentQuestion = this.#currentQuestion.value;\r\n    // проверим квоты\r\n    const currentGuid = currentQuestion?.QuestionGuid ?? 0;\r\n    if (this.#surveyQuotas.quotaIsCompleted && !this.#surveyQuotas.guidIsDestinationQuestion(currentGuid)) {\r\n      // у нас есть заполненная квота, найдем вопрос который нужно показать при заполнении квоты\r\n      const destinationQ = this.#findQuestionByGuid(this.#surveyQuotas.destinationQuestionGuid);\r\n      //если вопрос есть(его могли удалить из анкеты), показывает его\r\n      if (!!destinationQ) {\r\n        return destinationQ.QuestionNum;\r\n      }\r\n    }\r\n\r\n    // если нет заполненной квоты, переходим на следующий вопрос, ищем следующий вопрос который мы можем отобразить (до логика может запретить показ вопроса)\r\n    let nextQuestionNum = (currentQuestion?.QuestionNum ?? 0) + 1;\r\n\r\n    // но сначала проверим после логику\r\n    if (!!currentQuestion) {\r\n      const destQ = currentQuestion.logiclist.find(logObj => (logObj.DestinationQuestionGuid > 0) && this._logic.executeLogic(logObj.LogicText) );\r\n      if (!!destQ) {\r\n        const destQNum = this.#findQuestionByGuid(destQ.DestinationQuestionGuid)?.QuestionNum;\r\n        if (Number.isSafeInteger(destQNum)) { // вопрос мог быть удалён, а логику оставить\r\n          nextQuestionNum = destQNum!;\r\n        }\r\n      }\r\n    }\r\n\r\n    // общее количество вопросов, цикл будет считать до этого количества\r\n    const questionCount = this.#questionsCount;\r\n    do {\r\n      //потенциально следующий вопрос\r\n      const q = this.#questionary!.Questions[nextQuestionNum];\r\n\r\n      //если это последний вопрос или если выполняется BeforeExpression и этот вопрос не DestinationQuota, можем его показать\r\n      if (q.QuestionType === QuestionsType.eqEND || (this._logic.executeLogic(q.BeforeExpression) && !this.#surveyQuotas.guidIsDestinationQuestion(q.QuestionGuid)) ) {\r\n        return nextQuestionNum;\r\n      }\r\n      //вопрос нельзя показывать, увеличиваем счётчик текущего вопроса на 1\r\n      nextQuestionNum++;\r\n    } while (nextQuestionNum < questionCount - 1); // end вопрос должен всегда быть показан\r\n    return nextQuestionNum;\r\n  }\r\n\r\n  #initiateQuestions(questionList: Array<IdbQuestion>): void{\r\n    questionList.forEach(question => {\r\n      //создаём пустой ответ для вопроса.\r\n      const answer = this._interview.createAnswer(question.QuestionGuid, question.Version);\r\n\r\n      //такого быть не должно, это ошибка\r\n      if (!answer) {\r\n        throw new Error('answer not found');\r\n      }\r\n      \r\n      const logicVarSet = new Set<string>();\r\n      this.#questionLogicVarContainer[question.QuestionGuid.toString()] = logicVarSet;\r\n      \r\n      logicVarSet.add(question.SysName);//привязываем переменную к вопросу\r\n      this._logic.setLogicValue(question.SysName, MISSING_VALUE); // переменная самого вопроса\r\n\r\n      //добавим доп ответы и логику для них\r\n      question.AdditionalAnswerlist.forEach(aa => {\r\n        const aaSysName = `${question.SysName}${aa.SysName}`;\r\n        answer.AnswerAdditional[aa.Value] = null;\r\n        this._logic.setLogicValue(aaSysName, MISSING_VALUE); // логика для переменных доп ответа\r\n        logicVarSet.add(aaSysName); //привязываем переменную к вопросу\r\n      });\r\n\r\n      // если в тексте postScript есть запись типа setVar('varName',  value);  нужно инициализировать переменную varName значением -97.\r\n      if (!!question.PostScript) {\r\n        (question.PostScript.match(/(\\bsetvar\\b)\\(['|\"](.*?)['|\"]/gim) ?? []).map(scriptVar => scriptVar.match(/['|\"](.+)['|\"]/i)![1]).forEach(varName => {\r\n          logicVarSet.add(varName);//привязываем переменную к вопросу\r\n          this._logic.setLogicValue(varName, null)\r\n        });\r\n      }\r\n\r\n      // для Вопросов SS/MS нужно прописать логику альтернатив sysname вопроса + sysname альтернативы\r\n      if (question.Sectionlist.length <= 0) {// для Вопросов SS/MS\r\n        question.Alternativelist.forEach(alt => {\r\n          const aSysName = `${question.SysName}${alt.SysName}`\r\n          this._logic.setLogicValue(aSysName, MISSING_VALUE);\r\n          logicVarSet.add(aSysName); //привязываем переменную к вопросу\r\n        });\r\n      } else {\r\n        //для вопросов у которых есть секции проинициализируем каждую секцию\r\n        question.Sectionlist.forEach(section => {\r\n\r\n          //создаём пустой ответ для секции.\r\n          const answerForSection = this._interview.createAnswer(section.SectionGuid, 1);\r\n\r\n          //такого быть не должно, это ошибка\r\n          if (!answerForSection) {\r\n            throw new Error('answer not found');\r\n          }\r\n\r\n          // для Вопросов SL нужно прописать логику альтернатив и секций sysname секции + sysname альтернативы\r\n          this._logic.setLogicValue(section.SysName, MISSING_VALUE); // переменная самой секции\r\n          logicVarSet.add(section.SysName); //привязываем переменную к вопросу\r\n\r\n          //логика альтернатив SL вопроса\r\n          question.Alternativelist.forEach(alt => {\r\n            const aSysName = `${section.SysName}${alt.SysName}`;\r\n            this._logic.setLogicValue(aSysName, MISSING_VALUE);\r\n            logicVarSet.add(aSysName); //привязываем переменную к вопросу\r\n          });\r\n        })\r\n      }\r\n    })\r\n  }\r\n\r\n  //ищет вопрос на guid\r\n  #findQuestionByGuid(guid: number): IdbQuestion | undefined {\r\n    if (!this.#questionary) {\r\n      return undefined;\r\n    }\r\n    return this.#questionary.Questions.find(q => q.QuestionGuid === guid);\r\n  }\r\n\r\n  #nextButtonExpressionIsTrue(): boolean{\r\n    const buttonNextExpression = this.#currentQuestion.value?.BtnNextExpression ?? '';\r\n    //если скрипта для кнопки далее нет, вернём true\r\n    if (buttonNextExpression.trim() === '') {\r\n      return true;\r\n    }\r\n    try {\r\n      return this._logic.executeLogic(buttonNextExpression);\r\n    } catch (e) {\r\n      console.error(e);\r\n      return false;\r\n    }\r\n  }\r\n\r\n  #calculateImplicitQuestion(question: IdbQuestion): void {\r\n    if (question.Implicit && (question.QuestionType === QuestionsType.eqSS || question.QuestionType === QuestionsType.eqMS) ) {\r\n      let questionAnswer: number | Array<number>;\r\n\r\n      if (question.QuestionType === QuestionsType.eqSS) {\r\n        const alt = question.Alternativelist.find(alt => this._logic.executeLogic(alt.EnableExpression));\r\n        questionAnswer = !!alt ? alt.Value : 0;\r\n      }\r\n\r\n      if (question.QuestionType === QuestionsType.eqMS) {\r\n        questionAnswer = question.Alternativelist\r\n          .filter(alt => this._logic.executeLogic(alt.EnableExpression))\r\n          .map(alt => alt.Value);\r\n      }\r\n\r\n      const answerChange: AnswerChange = {\r\n        guid: question.QuestionGuid,\r\n        isComplete: true,\r\n        answer: questionAnswer!,\r\n        additionalAnswers: {},\r\n        logicVars: {},\r\n        viewedAlternativeValues: Array.isArray(questionAnswer!) ? questionAnswer : [questionAnswer!],\r\n        viewedSectionGuids: [],\r\n        alternativeSequenceClicks: []\r\n      }\r\n      this.updateAnswer(answerChange);\r\n    }\r\n  }\r\n  \r\n  #clearAnswerForCurrentQuestion(): void {\r\n    const currentQuestionGuid = this.#currentQuestion.value?.QuestionGuid ?? -1;\r\n    \r\n    //удаляем сам ответ на вопрос\r\n    this._interview.findAnswerByGuid(currentQuestionGuid)?.clear();\r\n\r\n    //удаляем переменные которые относятся к этому вопросу\r\n    const logicSet = this.#questionLogicVarContainer[currentQuestionGuid];\r\n    if (!!logicSet) {\r\n      logicSet.forEach(sysName =>this._logic.setLogicValue(sysName, MISSING_VALUE));\r\n    }\r\n  }\r\n}\r\n"]}
|
|
@@ -65,4 +65,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImpor
|
|
|
65
65
|
providedIn: 'root'
|
|
66
66
|
}]
|
|
67
67
|
}] });
|
|
68
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHAuc3VydmV5LmludGVydmlldy5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHAuc3VydmV5L3NyYy9saWIvY29yZS90cC5zdXJ2ZXkuaW50ZXJ2aWV3L3RwLnN1cnZleS5pbnRlcnZpZXcuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsTUFBTSxFQUFDLE1BQU0sc0NBQXNDLENBQUM7QUFFNUQsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLHVCQUF1QixDQUFDO0FBQ3BELE9BQU8sRUFBQyxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7O0FBS3pDLE1BQU0sT0FBTyx3QkFBd0I7SUFIckM7UUFJRSxnQkFBVyxHQUFtQixFQUFFLENBQUM7UUFDakMsY0FBUyxHQUFXLENBQUMsQ0FBQztLQTREdkI7SUE3REMsV0FBVyxDQUFzQjtJQUdqQyxZQUFZLENBQUMsSUFBWSxFQUFFLE9BQWU7UUFDeEMsK0NBQStDO1FBQy9DLElBQUksTUFBTSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUV6QyxtREFBbUQ7UUFDbkQsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDO1lBQ1osTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQTtZQUNsQyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUNoQyxDQUFDO1FBQ0QseUNBQXlDO1FBQ3pDLE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7SUFFRCxnQkFBZ0IsQ0FBQyxJQUFZO1FBQzNCLE9BQU8sSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsWUFBWSxLQUFLLElBQUksQ0FBQyxDQUFDO0lBQzdELENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxJQUFZO1FBQzdCLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMzQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQztJQUNuRCxDQUFDO0lBRUQsYUFBYTtRQUNYLE9BQU8sSUFBSSxDQUFDLFdBQVcsQ0FBQztJQUMxQixDQUFDO0lBRUQsZUFBZSxDQUFDLE9BQStCO1FBQzdDLHdDQUF3QztRQUN4QyxPQUFPLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLFdBQVcsS0FBSyxhQUFhLENBQUMsQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLEVBQUU7WUFDakYsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsWUFBWSxLQUFLLFNBQVMsQ0FBQyxZQUFZLENBQUMsQ0FBQztZQUMxRixJQUFJLENBQUMsQ0FBQyxTQUFTLEVBQUUsQ0FBQztnQkFDaEIsdUJBQXVCO2dCQUN2QixTQUFTLENBQUMsV0FBVyxHQUFHLFNBQVMsQ0FBQyxXQUFXLENBQUM7Z0JBRTlDLFNBQVMsQ0FBQyxTQUFTLEdBQUcsU0FBUyxDQUFDLFNBQVMsQ0FBQztnQkFDMUMsU0FBUyxDQUFDLFVBQVUsR0FBRyxTQUFTLENBQUMsVUFBVSxDQUFDO2dCQUU1QyxrR0FBa0c7Z0JBQ2xHLEtBQUssTUFBTSxJQUFJLElBQUksTUFBTSxDQUFDLG1CQUFtQixDQUFDLFNBQVMsQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFLENBQUM7b0JBQ3pFLE9BQU8sU0FBUyxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFDO2dCQUMzQyxDQUFDO2dCQUNELDJCQUEyQjtnQkFDM0IsS0FBSyxNQUFNLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxJQUFJLE1BQU0sQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLGdCQUFnQixDQUFDLEVBQUUsQ0FBQztvQkFDdEUsU0FBUyxDQUFDLGdCQUFnQixDQUFDLEdBQUcsQ0FBQyxHQUFHLEtBQUssQ0FBQztnQkFDMUMsQ0FBQztnQkFFRCx5SUFBeUk7Z0JBQ3pJLFNBQVMsQ0FBQyx1QkFBdUIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO2dCQUM3QyxTQUFTLENBQUMsdUJBQXVCLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsU0FBUyxDQUFDLHVCQUF1QixDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO2dCQUU1RixTQUFTLENBQUMsa0JBQWtCLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztnQkFDeEMsU0FBUyxDQUFDLGtCQUFrQixDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLFNBQVMsQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztnQkFFbEYsU0FBUyxDQUFDLHlCQUF5QixDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7Z0JBQy9DLFNBQVMsQ0FBQyx5QkFBeUIsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxTQUFTLENBQUMseUJBQXlCLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7WUFDbEcsQ0FBQztRQUNILENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs4R0E3RFUsd0JBQXdCO2tIQUF4Qix3QkFBd0IsY0FGdkIsTUFBTTs7MkZBRVAsd0JBQXdCO2tCQUhwQyxVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7QW5zd2VyfSBmcm9tICcuLi90cC5zdXJ2ZXkuYW5zd2VyL3RwLnN1cnZleS5hbnN3ZXInO1xyXG5pbXBvcnQge0lRdWVzdGlvbkFuc3dlcn0gZnJvbSAnLi4vLi4vdHAuc3VydmV5LmludGVyZmFjZSc7XHJcbmltcG9ydCB7TUlTU0lOR19WQUxVRX0gZnJvbSAnLi4vLi4vdHAuc3VydmV5LmNvbnN0JztcclxuaW1wb3J0IHtJbmplY3RhYmxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcbkBJbmplY3RhYmxlKHtcclxuICBwcm92aWRlZEluOiAncm9vdCdcclxufSlcclxuZXhwb3J0IGNsYXNzIFRwU3VydmV5SW50ZXJ2aWV3U2VydmljZSB7XHJcbiAgI2Fuc3dlckxpc3Q6IEFycmF5PEFuc3dlcj4gID0gW107XHJcbiAgc3RhcnRlZEF0OiBudW1iZXIgPSAwO1xyXG5cclxuICBjcmVhdGVBbnN3ZXIoZ3VpZDogbnVtYmVyLCB2ZXJzaW9uOiBudW1iZXIpOiBBbnN3ZXIge1xyXG4gICAgLy/QuNGI0LXQvCDQvtGC0LLQtdGCINGBINGC0LDQutC40Lwg0LbQtSDQvdC+0LzQtdGA0L7QvCDQutCw0Log0Lgg0LIg0LLQvtC/0YDQvtGB0LVcclxuICAgIGxldCBhbnN3ZXIgPSB0aGlzLmZpbmRBbnN3ZXJCeUd1aWQoZ3VpZCk7XHJcblxyXG4gICAgLy/QtdGB0LvQuCDQvtGC0LLQtdGC0LAg0YEg0YLQsNC60LjQvCDQvdC+0LzQtdGA0L7QvCDQstC+0L/RgNC+0YHQsCDQvdC10YIsINGB0L7Qt9C00LDQtNC40LxcclxuICAgIGlmICghYW5zd2VyKSB7XHJcbiAgICAgIGFuc3dlciA9IG5ldyBBbnN3ZXIoZ3VpZCwgdmVyc2lvbilcclxuICAgICAgdGhpcy4jYW5zd2VyTGlzdC5wdXNoKGFuc3dlcik7XHJcbiAgICB9XHJcbiAgICAvL9Cy0LXRgNC90ZHQvCDRgtC+LCDRh9GC0L4g0YHQvtC30LTQsNC70Lgg0LjQu9C4INGC0L4g0YfRgtC+INC90LDRiNC70LhcclxuICAgIHJldHVybiBhbnN3ZXI7XHJcbiAgfVxyXG5cclxuICBmaW5kQW5zd2VyQnlHdWlkKGd1aWQ6IG51bWJlcik6IEFuc3dlciB8IHVuZGVmaW5lZCB7XHJcbiAgICByZXR1cm4gdGhpcy4jYW5zd2VyTGlzdC5maW5kKGEgPT4gYS5RdWVzdGlvbkdVSUQgPT09IGd1aWQpO1xyXG4gIH1cclxuXHJcbiAgcXVlc3Rpb25Jc0NvbXBsZXRlKGd1aWQ6IG51bWJlcik6IGJvb2xlYW57XHJcbiAgICBjb25zdCBhbnN3ZXIgPSB0aGlzLmZpbmRBbnN3ZXJCeUd1aWQoZ3VpZCk7XHJcbiAgICByZXR1cm4gIWFuc3dlciA/IGZhbHNlIDogYW5zd2VyLmFuc3dlcklzQ29tcGxldGU7XHJcbiAgfVxyXG5cclxuICBnZXRBbnN3ZXJMaXN0KCk6IEFycmF5PEFuc3dlcj57XHJcbiAgICByZXR1cm4gdGhpcy4jYW5zd2VyTGlzdDtcclxuICB9XHJcblxyXG4gIG92ZXJyaWRlQW5zd2VycyhhbnN3ZXJzOiBBcnJheTxJUXVlc3Rpb25BbnN3ZXI+KTogdm9pZHtcclxuICAgIC8v0L3QtSDQsdGD0LTQtdC8INGC0YDQvtCz0LDRgtGMINCy0L7Qv9GA0L7RgdGLINC60L7RgtC+0YDRi9C1ID0gLTk3XHJcbiAgICBhbnN3ZXJzLmZpbHRlcihhbnN3ZXIgPT4gYW5zd2VyLkFuc3dlclZhbHVlICE9PSBNSVNTSU5HX1ZBTFVFKS5mb3JFYWNoKG9sZEFuc3dlciA9PiB7XHJcbiAgICAgIGNvbnN0IG5ld0Fuc3dlciA9IHRoaXMuI2Fuc3dlckxpc3QuZmluZChhYSA9PiBhYS5RdWVzdGlvbkdVSUQgPT09IG9sZEFuc3dlci5RdWVzdGlvbkdVSUQpO1xyXG4gICAgICBpZiAoISFuZXdBbnN3ZXIpIHtcclxuICAgICAgICAvL9Cy0LXRgNC90ZHQvCDQvtGB0L3QvtCy0L3QvtC5INC+0YLQstC10YJcclxuICAgICAgICBuZXdBbnN3ZXIuQW5zd2VyVmFsdWUgPSBvbGRBbnN3ZXIuQW5zd2VyVmFsdWU7XHJcblxyXG4gICAgICAgIG5ld0Fuc3dlci5TdGFydGVkQXQgPSBvbGRBbnN3ZXIuU3RhcnRlZEF0O1xyXG4gICAgICAgIG5ld0Fuc3dlci5GaW5pc2hlZEF0ID0gb2xkQW5zd2VyLkZpbmlzaGVkQXQ7XHJcblxyXG4gICAgICAgIC8v0KEg0LTQvtC/0L7Qu9C90LjRgtC10LvRjNC90YvQvNC4INC+0YLQstC10YLQsNC80Lgg0YHQu9C+0LbQvdC10LUsINC+0L3QuCDQv9C+0LzQtdGH0LXQvdGLIHJlYWRvbmx5LiDQn9C+0Y3RgtC+0LzRgyDRgdC90LDRh9Cw0LvQsCDRg9C00LDQu9C40LwsINC/0L7RgtC+0Lwg0LTQvtCx0LDQstC40LxcclxuICAgICAgICBmb3IgKGNvbnN0IHByb3Agb2YgT2JqZWN0LmdldE93blByb3BlcnR5TmFtZXMobmV3QW5zd2VyLkFuc3dlckFkZGl0aW9uYWwpKSB7XHJcbiAgICAgICAgICAgZGVsZXRlIG5ld0Fuc3dlci5BbnN3ZXJBZGRpdGlvbmFsW3Byb3BdO1xyXG4gICAgICAgIH1cclxuICAgICAgICAvL9C00L7QsdCw0LLQu9GP0LXQvCDRgdGC0LDRgNGL0LUg0LfQvdCw0YfQtdC90LjRj1xyXG4gICAgICAgIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKG9sZEFuc3dlci5BbnN3ZXJBZGRpdGlvbmFsKSkge1xyXG4gICAgICAgICAgbmV3QW5zd2VyLkFuc3dlckFkZGl0aW9uYWxba2V5XSA9IHZhbHVlO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy/QotC+INC20LUg0YHQsNC80L7QtSDRgSBWaWV3ZWRBbHRlcm5hdGl2ZVZhbHVlcyDQuCBWaWV3ZWRTZWN0aW9uR3VpZHMg0LggQWx0ZXJuYXRpdmVTZXF1ZW5jZUNsaWNrcy4g0KIu0LouINC+0L3QuCByZWFkb25seSDRgdC90LDRh9Cw0LvQsCDRg9C00LDQu9C40LwsINC/0L7RgtC+0Lwg0LTQvtCx0LDQstC40LxcclxuICAgICAgICBuZXdBbnN3ZXIuVmlld2VkQWx0ZXJuYXRpdmVWYWx1ZXMubGVuZ3RoID0gMDtcclxuICAgICAgICBvbGRBbnN3ZXIuVmlld2VkQWx0ZXJuYXRpdmVWYWx1ZXMuZm9yRWFjaChvYSA9PiBuZXdBbnN3ZXIuVmlld2VkQWx0ZXJuYXRpdmVWYWx1ZXMucHVzaChvYSkpO1xyXG5cclxuICAgICAgICBuZXdBbnN3ZXIuVmlld2VkU2VjdGlvbkd1aWRzLmxlbmd0aCA9IDA7XHJcbiAgICAgICAgb2xkQW5zd2VyLlZpZXdlZFNlY3Rpb25HdWlkcy5mb3JFYWNoKG9hID0+IG5ld0Fuc3dlci5WaWV3ZWRTZWN0aW9uR3VpZHMucHVzaChvYSkpO1xyXG5cclxuICAgICAgICBuZXdBbnN3ZXIuQWx0ZXJuYXRpdmVTZXF1ZW5jZUNsaWNrcy5sZW5ndGggPSAwO1xyXG4gICAgICAgIG9sZEFuc3dlci5BbHRlcm5hdGl2ZVNlcXVlbmNlQ2xpY2tzLmZvckVhY2gob2EgPT4gbmV3QW5zd2VyLkFsdGVybmF0aXZlU2VxdWVuY2VDbGlja3MucHVzaChvYSkpO1xyXG4gICAgICB9XHJcbiAgICB9KTtcclxuICB9XHJcbn1cclxuIl19
|