@touchpoll/tp-survey 0.0.23 → 0.0.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (60) hide show
  1. package/fesm2022/touchpoll-tp-survey.mjs +202 -150
  2. package/fesm2022/touchpoll-tp-survey.mjs.map +1 -1
  3. package/lib/component/answer/answer.ms/answer.ms.component.d.ts +3 -3
  4. package/lib/component/answer/answer.sl/answer.sl.component.d.ts +3 -3
  5. package/lib/component/survey.play/survey.play.component.d.ts +2 -2
  6. package/lib/pipe/get.question.image.background.size/get.question.image.background.css.size.pipe.d.ts +12 -0
  7. package/lib/tp.survey.interface.d.ts +4 -4
  8. package/package.json +1 -3
  9. package/styles/platform/component/survey.checkbox.scss +5 -5
  10. package/styles/platform/component/survey.radio.button.scss +3 -3
  11. package/styles/platform/survey.mixins.desktop.scss +9 -9
  12. package/styles/platform/survey.mixins.tablet.scss +3 -3
  13. package/styles/survey.mixins.scss +4 -3
  14. package/styles/survey.themes.scss +5 -2
  15. package/styles/theme/azure.blue.scss +3 -2
  16. package/styles/theme/cyan.orange.scss +4 -2
  17. package/styles/theme/rose.red.scss +3 -2
  18. package/esm2022/lib/component/answer/answer.additional/answer.additional.component.mjs +0 -49
  19. package/esm2022/lib/component/answer/answer.additional/directive/additional.answer.caption.position/additional.answer.caption.position.directive.mjs +0 -61
  20. package/esm2022/lib/component/answer/answer.end/answer.end.component.mjs +0 -30
  21. package/esm2022/lib/component/answer/answer.info/answer.info.component.mjs +0 -30
  22. package/esm2022/lib/component/answer/answer.lang/answer.lang.component.mjs +0 -22
  23. package/esm2022/lib/component/answer/answer.master/answer.master.component.mjs +0 -57
  24. package/esm2022/lib/component/answer/answer.ms/answer.ms.component.mjs +0 -106
  25. package/esm2022/lib/component/answer/answer.od/answer.od.component.mjs +0 -39
  26. package/esm2022/lib/component/answer/answer.ol/answer.ol.component.mjs +0 -40
  27. package/esm2022/lib/component/answer/answer.sl/answer.sl.component.mjs +0 -131
  28. package/esm2022/lib/component/answer/answer.sl/pipe/as.form.array/as.form.array.pipe.mjs +0 -17
  29. package/esm2022/lib/component/answer/answer.sl/pipe/as.form.control/as.form.control.pipe.mjs +0 -17
  30. package/esm2022/lib/component/answer/answer.ss/answer.ss.component.mjs +0 -51
  31. package/esm2022/lib/component/answer/container.answer/container.answer.component.mjs +0 -319
  32. package/esm2022/lib/component/answer/container.answer/container.answer.validator.mjs +0 -130
  33. package/esm2022/lib/component/navigation.button/navigation.button.component.mjs +0 -21
  34. package/esm2022/lib/component/question.caption/question.caption.component.mjs +0 -15
  35. package/esm2022/lib/component/survey.play/survey.play.component.mjs +0 -126
  36. package/esm2022/lib/component/survey.question.preview/survey.question.preview.component.mjs +0 -60
  37. package/esm2022/lib/core/tp.servey.quota/tp.survey.quota.mjs +0 -105
  38. package/esm2022/lib/core/tp.survey.answer/tp.survey.answer.mjs +0 -27
  39. package/esm2022/lib/core/tp.survey.core/tp.survey.core.service.mjs +0 -455
  40. package/esm2022/lib/core/tp.survey.interview/tp.survey.interview.service.mjs +0 -68
  41. package/esm2022/lib/core/tp.survey.logic/tp.survey.logic.service.mjs +0 -42
  42. package/esm2022/lib/directive/alternatives.container/alternatives.container.directive.mjs +0 -130
  43. package/esm2022/lib/directive/focus.and.select/focus.and.select.directive.mjs +0 -27
  44. package/esm2022/lib/directive/survey.theme/survey.theme.directive.mjs +0 -53
  45. package/esm2022/lib/directive/update.font.size/update.font.size.directive.mjs +0 -28
  46. package/esm2022/lib/icons/default.icons/default.icons.mjs +0 -15
  47. package/esm2022/lib/icons/default.icons/default.icons.module.mjs +0 -31
  48. package/esm2022/lib/pipe/get.active.language/get.active.language.pipe.mjs +0 -17
  49. package/esm2022/lib/pipe/get.additional.value.by.value/get.additional.value.by.value.pipe.mjs +0 -18
  50. package/esm2022/lib/pipe/key.values/key.values.pipe.mjs +0 -20
  51. package/esm2022/lib/pipe/multi.lang.object.to.html/multi.lang.object.to.html.pipe.mjs +0 -48
  52. package/esm2022/lib/tp.survey.config.mjs +0 -8
  53. package/esm2022/lib/tp.survey.const.mjs +0 -28
  54. package/esm2022/lib/tp.survey.enum.mjs +0 -50
  55. package/esm2022/lib/tp.survey.interface.mjs +0 -2
  56. package/esm2022/lib/tp.survey.method.mjs +0 -5
  57. package/esm2022/lib/tp.survey.module.mjs +0 -27
  58. package/esm2022/lib/tp.survey.type.mjs +0 -2
  59. package/esm2022/public-api.mjs +0 -14
  60. package/esm2022/touchpoll-tp-survey.mjs +0 -5
@@ -1,105 +0,0 @@
1
- import { executeLogicExpression } from '../../tp.survey.method';
2
- export class SurveyQuotas {
3
- #completedQuota = { name: null, guid: null, guid2d: null };
4
- #quotas = [];
5
- #quotas2d = [];
6
- #quotaLimitDestinationQuestionGuid = -1;
7
- #quotaLimitDestinationQuestionNum = -1;
8
- setQuotas(quotas1d = [], quotas2d = [], quotaLimitDestinationQuestionGuid = -1, destinationQuestionNum = -1) {
9
- this.#quotas = quotas1d;
10
- this.#quotas2d = quotas2d;
11
- this.#quotaLimitDestinationQuestionGuid = quotaLimitDestinationQuestionGuid;
12
- this.#quotaLimitDestinationQuestionNum = destinationQuestionNum;
13
- }
14
- guidIsDestinationQuestion(guid) {
15
- return this.#quotaLimitDestinationQuestionGuid === guid;
16
- }
17
- get quotaIsCompleted() {
18
- return this.#completedQuota && (!!this.#completedQuota.guid || !!this.#completedQuota.guid2d);
19
- }
20
- get destinationQuestionGuid() {
21
- return this.#quotaLimitDestinationQuestionGuid;
22
- }
23
- get quotaCount() {
24
- const quotas = this.#quotas && Array.isArray(this.#quotas) ? this.#quotas.length : 0;
25
- const quotas2d = this.#quotas2d && Array.isArray(this.#quotas2d) ? this.#quotas2d.length : 0;
26
- return quotas + quotas2d;
27
- }
28
- findFilledQuotas(currentQuestionNum, vars, interviewAnswers) {
29
- /*
30
- * Поиск заполненной квоты это очень дорогой процесс. Его нужно оптимизировать по максимуму.
31
- * Перед поиском нужно выполнить ряд проверок.
32
- * Проверка currentQuestionNum <= this.#quotaLimitDestinationQuestionNum Нужна для оптимизации.
33
- * DestinationQuestion Желательно располагать после последнего квотного вопроса, потому что нет смысла постоянно проверять квоты, если вопросы которые используются для расчёта квоты меняться не будут.
34
- * например Квота Q1 + Q2 <= 25. DestinationQuestion нужно делать номером 3 или 4. Потому что как бы респондент не отвечал на последующие вопросы, результат квот не измениться и каждый раз их пересчитывать смысла нет.
35
- */
36
- if ((this.destinationQuestionGuid > 0) && //Глобально в настройках анкеты должен быть указан вопрос который должен отобразиться при достижении квоты, если его нет, то и искать квоты смысла нет.
37
- (this.quotaCount > 0) && //проверим если у нас вообще квоты
38
- (currentQuestionNum <= this.#quotaLimitDestinationQuestionNum) && // номер текущий вопроса меньше номера вопроса для перехода по достижении квот
39
- (!this.quotaIsCompleted) //и на финал проверим если ли уже заполненная квота, повторно не запускаем
40
- ) {
41
- new Promise((resolve, reject) => {
42
- if (Array.isArray(this.#quotas)) {
43
- const completedQuota = this.#quotas.find(quota => executeLogicExpression(quota.expression, vars));
44
- if (!!completedQuota) {
45
- return resolve({ name: completedQuota.name, guid: completedQuota.quota_guid });
46
- }
47
- }
48
- if (Array.isArray(this.#quotas2d)) {
49
- const answers = this.#quotas2d.length > 0 ? interviewAnswers.map(a => ({ av: a.AnswerValue, qg: a.QuestionGUID })) : [];
50
- const filledQuota2dArray = this.#quotas2d.filter(quotaValues => {
51
- /*
52
- * 2d квоты это всегда таблица. Но может быть иерархия, когда одно значение ответа может включать в себя один или несколько других ответов. Например "Обл центр" это 1, а город или село это другой ответ, хотя город и село это разные альтернативы
53
- * */
54
- if (Array.isArray(quotaValues.cqv) && quotaValues.cqv.length > 0) {
55
- /*
56
- * это ветка для случая с иерархией. Тут будет 2 массива.
57
- * В первом будут контролироваться те ответы которое должны совпадать полностью arrayEveryValues
58
- * Во втором arraySomeValue будет проверяться совпадение хотя бы 1 ответа
59
- * */
60
- const questionGuid = quotaValues.cqv[0].guid;
61
- const masterQuestion = quotaValues.qs.find(q => q.guid === questionGuid);
62
- const arrayEveryValues = quotaValues.qs.filter(q => q.guid != questionGuid);
63
- const arraySomeValue = [...quotaValues.cqv, ...[masterQuestion]];
64
- const arrayEveryResult = arrayEveryValues.every(qvq => !!answers.find(intAnswer => intAnswer.qg === qvq.guid && intAnswer.av === qvq.value));
65
- const arraySomeResult = arraySomeValue.some(qvq => !!answers.find(intAnswer => intAnswer.qg === qvq.guid && intAnswer.av === qvq.value));
66
- //если в обоих массивах совпадают значения значит квота выполнилась
67
- return arrayEveryResult && arraySomeResult;
68
- }
69
- else {
70
- //обычный массив значений, в котором должны совпадать все значения
71
- return quotaValues.qs.every(qvq => !!answers.find(intAnswer => intAnswer.qg === qvq.guid && intAnswer.av === qvq.value));
72
- }
73
- });
74
- //если не нашли заполненных квот выходим
75
- if (filledQuota2dArray.length <= 0) {
76
- return;
77
- }
78
- //ищем квоту которая заполнена и у нее есть условие и оно выполняется.
79
- const quota2dWithExpression = filledQuota2dArray.find((q) => !!q.exp && executeLogicExpression(q.exp, vars));
80
- //если квота есть сообщим
81
- if (!!quota2dWithExpression) {
82
- return resolve({ name: null, guid2d: quota2dWithExpression.qvg });
83
- }
84
- //если квоты с Expression нет, ищем квоту которая заполнена и у нее нет условия
85
- const quota2dWithoutExpression = filledQuota2dArray.find((q) => !q.exp || (typeof q.exp === 'string' && q.exp.trim().length === 0));
86
- if (!!quota2dWithoutExpression) {
87
- return resolve({ name: null, guid2d: quota2dWithoutExpression.qvg });
88
- }
89
- }
90
- reject(null);
91
- }).then(quota => {
92
- if (!!quota) {
93
- this.#completedQuota = {
94
- name: quota.name,
95
- guid: 'guid' in quota ? quota.guid : null,
96
- guid2d: 'guid2d' in quota ? quota.guid2d : null,
97
- };
98
- }
99
- }).catch(err => {
100
- console.error(err);
101
- });
102
- }
103
- }
104
- }
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tp.survey.quota.js","sourceRoot":"","sources":["../../../../../../projects/tp.survey/src/lib/core/tp.servey.quota/tp.survey.quota.ts"],"names":[],"mappings":"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","sourcesContent":["import {IdbQuestionaryQuota, IdbQuestionaryQuota2dObject} from '../../tp.survey.interface';\r\nimport {Answer} from '../tp.survey.answer/tp.survey.answer';\r\nimport {executeLogicExpression} from '../../tp.survey.method';\r\nimport {LogicObject} from '../../tp.survey.type';\r\n\r\ntype QuotaCalculationResult = {\r\n  name: string;\r\n  guid: number;\r\n} | {\r\n  name: null;\r\n  guid2d: number;\r\n} | null;\r\n\r\nexport class SurveyQuotas {\r\n  #completedQuota: {name: string | null; guid: number| null; guid2d: number| null} = {name: null, guid: null, guid2d: null};\r\n  #quotas: Array<IdbQuestionaryQuota> = [];\r\n  #quotas2d: Array<IdbQuestionaryQuota2dObject> = [];\r\n  #quotaLimitDestinationQuestionGuid: number = -1;\r\n  #quotaLimitDestinationQuestionNum: number = -1;\r\n\r\n  setQuotas(quotas1d: Array<IdbQuestionaryQuota> = [], quotas2d: Array<IdbQuestionaryQuota2dObject> = [], quotaLimitDestinationQuestionGuid: number = -1, destinationQuestionNum: number = -1){\r\n    this.#quotas = quotas1d;\r\n    this.#quotas2d = quotas2d;\r\n    this.#quotaLimitDestinationQuestionGuid = quotaLimitDestinationQuestionGuid;\r\n    this.#quotaLimitDestinationQuestionNum = destinationQuestionNum;\r\n  }\r\n\r\n  guidIsDestinationQuestion(guid: number): boolean{\r\n    return this.#quotaLimitDestinationQuestionGuid === guid;\r\n  }\r\n\r\n\r\n  get quotaIsCompleted(): boolean {\r\n    return this.#completedQuota && (!!this.#completedQuota.guid || !!this.#completedQuota.guid2d)  ;\r\n  }\r\n\r\n  get destinationQuestionGuid(): number {\r\n    return this.#quotaLimitDestinationQuestionGuid\r\n  }\r\n\r\n  get quotaCount(): number {\r\n    const quotas   = this.#quotas &&  Array.isArray(this.#quotas) ? this.#quotas.length : 0;\r\n    const quotas2d = this.#quotas2d &&  Array.isArray(this.#quotas2d) ? this.#quotas2d.length : 0;\r\n    return quotas + quotas2d;\r\n  }\r\n\r\n  findFilledQuotas(currentQuestionNum: number, vars: LogicObject, interviewAnswers: Array<Answer>): void {\r\n    /*\r\n    * Поиск заполненной квоты это очень дорогой процесс. Его нужно оптимизировать по максимуму.\r\n    * Перед поиском нужно выполнить ряд проверок.\r\n    * Проверка currentQuestionNum <= this.#quotaLimitDestinationQuestionNum Нужна для оптимизации.\r\n    * DestinationQuestion Желательно располагать после последнего квотного вопроса, потому что нет смысла постоянно проверять квоты, если вопросы которые используются для расчёта квоты меняться не будут.\r\n    * например Квота Q1 + Q2 <= 25. DestinationQuestion нужно делать номером 3 или 4. Потому что как бы респондент не отвечал на последующие вопросы, результат квот не измениться и каждый раз их пересчитывать смысла нет.\r\n    */\r\n    if (\r\n     (this.destinationQuestionGuid > 0) && //Глобально в настройках анкеты должен быть указан вопрос который должен отобразиться при достижении квоты, если его нет, то и искать квоты смысла нет.\r\n     (this.quotaCount > 0) && //проверим если у нас вообще квоты\r\n     (currentQuestionNum <= this.#quotaLimitDestinationQuestionNum) && // номер текущий вопроса меньше номера вопроса для перехода по достижении квот\r\n     (!this.quotaIsCompleted) //и на финал проверим если ли уже заполненная квота, повторно не запускаем\r\n    ) {\r\n\r\n      new Promise<QuotaCalculationResult>((resolve, reject) => {\r\n        if (Array.isArray(this.#quotas)) {\r\n          const completedQuota = this.#quotas.find(quota => executeLogicExpression(quota.expression, vars));\r\n          if (!!completedQuota) {\r\n            return resolve({name: completedQuota.name, guid: completedQuota.quota_guid});\r\n          }\r\n        }\r\n\r\n        if (Array.isArray(this.#quotas2d)) {\r\n          const answers = this.#quotas2d.length > 0 ? interviewAnswers.map(a => ({av: a.AnswerValue, qg: a.QuestionGUID}) ) : [];\r\n\r\n          const filledQuota2dArray = this.#quotas2d.filter(quotaValues => {\r\n            /*\r\n            * 2d квоты это всегда таблица. Но может быть иерархия, когда одно значение ответа может включать в себя один или несколько других ответов. Например \"Обл центр\" это 1, а город или село это другой ответ, хотя город и село это разные альтернативы\r\n            * */\r\n            if (Array.isArray(quotaValues.cqv) && quotaValues.cqv.length > 0) {\r\n              /*\r\n              * это ветка для случая с иерархией. Тут будет 2 массива.\r\n              * В первом будут контролироваться те ответы которое должны совпадать полностью arrayEveryValues\r\n              * Во втором arraySomeValue будет проверяться совпадение хотя бы 1 ответа\r\n              * */\r\n\r\n              const questionGuid = quotaValues.cqv[0].guid;\r\n\r\n              const masterQuestion = quotaValues.qs.find( q => q.guid === questionGuid);\r\n              const arrayEveryValues = quotaValues.qs.filter(q => q.guid != questionGuid);\r\n\r\n              const arraySomeValue = [...quotaValues.cqv, ...[masterQuestion!]];\r\n\r\n              const arrayEveryResult = arrayEveryValues.every(qvq => !!answers.find(intAnswer => intAnswer.qg === qvq.guid && intAnswer.av === qvq.value))\r\n              const arraySomeResult = arraySomeValue.some(qvq => !!answers.find(intAnswer => intAnswer.qg === qvq.guid && intAnswer.av === qvq.value))\r\n              //если в обоих массивах совпадают значения значит квота выполнилась\r\n              return arrayEveryResult && arraySomeResult;\r\n            } else {\r\n              //обычный массив значений, в котором должны совпадать все значения\r\n              return quotaValues.qs.every(qvq => !!answers.find(intAnswer => intAnswer.qg === qvq.guid && intAnswer.av === qvq.value))\r\n            }\r\n          });\r\n          //если не нашли заполненных квот выходим\r\n          if (filledQuota2dArray.length <= 0) {\r\n            return;\r\n          }\r\n\r\n          //ищем квоту которая заполнена и у нее есть условие и оно выполняется.\r\n          const quota2dWithExpression = filledQuota2dArray.find((q: IdbQuestionaryQuota2dObject) => !!q.exp && executeLogicExpression(q.exp, vars));\r\n\r\n          //если квота есть сообщим\r\n          if (!!quota2dWithExpression) {\r\n            return resolve({name: null, guid2d: quota2dWithExpression.qvg});\r\n          }\r\n\r\n          //если квоты с Expression нет, ищем квоту которая заполнена и у нее нет условия\r\n          const quota2dWithoutExpression = filledQuota2dArray.find((q: IdbQuestionaryQuota2dObject) => !q.exp  || (typeof q.exp  === 'string' && q.exp.trim().length === 0));\r\n          if (!!quota2dWithoutExpression) {\r\n            return resolve({name: null, guid2d: quota2dWithoutExpression.qvg});\r\n          }\r\n        }\r\n        reject(null)\r\n      }).then(quota => {\r\n        if (!!quota) {\r\n          this.#completedQuota = {\r\n            name: quota.name,\r\n            guid: 'guid' in  quota ? quota.guid : null,\r\n            guid2d: 'guid2d' in  quota ? quota.guid2d : null,\r\n          }\r\n        }\r\n      }).catch(err => {\r\n        console.error(err);\r\n      });\r\n    }\r\n  }\r\n}\r\n"]}
@@ -1,27 +0,0 @@
1
- import { MISSING_VALUE } from '../../tp.survey.const';
2
- export class Answer {
3
- constructor(guid, version = -1) {
4
- this.AnswerAdditional = {};
5
- this.ViewedAlternativeValues = [];
6
- this.ViewedSectionGuids = [];
7
- this.AlternativeSequenceClicks = [];
8
- this.StartedAt = 0;
9
- this.FinishedAt = 0;
10
- this.AnswerValue = MISSING_VALUE;
11
- this.answerIsComplete = false;
12
- this.QuestionGUID = guid;
13
- this.QuestionVersion = version;
14
- }
15
- clear() {
16
- this.StartedAt = 0;
17
- this.FinishedAt = 0;
18
- this.AnswerValue = MISSING_VALUE;
19
- this.ViewedAlternativeValues.length = 0;
20
- this.ViewedSectionGuids.length = 0;
21
- this.AlternativeSequenceClicks.length = 0;
22
- for (const key of Object.keys(this.AnswerAdditional)) {
23
- this.AnswerAdditional[Number(key)] = null;
24
- }
25
- }
26
- }
27
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHAuc3VydmV5LmFuc3dlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RwLnN1cnZleS9zcmMvbGliL2NvcmUvdHAuc3VydmV5LmFuc3dlci90cC5zdXJ2ZXkuYW5zd2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQztBQUVwRCxNQUFNLE9BQU8sTUFBTTtJQVdqQixZQUFZLElBQVksRUFBRSxVQUFrQixDQUFDLENBQUM7UUFSckMscUJBQWdCLEdBQXFDLEVBQUUsQ0FBQztRQUN4RCw0QkFBdUIsR0FBa0IsRUFBRSxDQUFDO1FBQzVDLHVCQUFrQixHQUFrQixFQUFFLENBQUM7UUFDdkMsOEJBQXlCLEdBQTRCLEVBQUUsQ0FBQztRQUNqRSxjQUFTLEdBQUcsQ0FBQyxDQUFDO1FBQ2QsZUFBVSxHQUFHLENBQUMsQ0FBQztRQUNmLGdCQUFXLEdBQVEsYUFBYSxDQUFDO1FBQ2pDLHFCQUFnQixHQUFZLEtBQUssQ0FBQztRQUVoQyxJQUFJLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQztRQUN6QixJQUFJLENBQUMsZUFBZSxHQUFHLE9BQU8sQ0FBQztJQUNqQyxDQUFDO0lBQ0QsS0FBSztRQUNILElBQUksQ0FBQyxTQUFTLEdBQUcsQ0FBQyxDQUFDO1FBQ25CLElBQUksQ0FBQyxVQUFVLEdBQUcsQ0FBQyxDQUFDO1FBQ3BCLElBQUksQ0FBQyxXQUFXLEdBQUcsYUFBYSxDQUFDO1FBQ2pDLElBQUksQ0FBQyx1QkFBdUIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQ3hDLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQ25DLElBQUksQ0FBQyx5QkFBeUIsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBQzFDLEtBQUssTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsRUFBRSxDQUFDO1lBQ3JELElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUM7UUFDNUMsQ0FBQztJQUNILENBQUM7Q0FLRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SVF1ZXN0aW9uQW5zd2VyfSBmcm9tICcuLi8uLi90cC5zdXJ2ZXkuaW50ZXJmYWNlJztcclxuaW1wb3J0IHtBZGRpdGlvbmFsQW5zd2VyfSBmcm9tICcuLi8uLi90cC5zdXJ2ZXkudHlwZSc7XHJcbmltcG9ydCB7TUlTU0lOR19WQUxVRX0gZnJvbSAnLi4vLi4vdHAuc3VydmV5LmNvbnN0JztcclxuXHJcbmV4cG9ydCBjbGFzcyBBbnN3ZXIgaW1wbGVtZW50cyBJUXVlc3Rpb25BbnN3ZXIge1xyXG4gIHJlYWRvbmx5IFF1ZXN0aW9uR1VJRDogbnVtYmVyO1xyXG4gIHJlYWRvbmx5IFF1ZXN0aW9uVmVyc2lvbjogbnVtYmVyO1xyXG4gIHJlYWRvbmx5IEFuc3dlckFkZGl0aW9uYWw6IFJlY29yZDxzdHJpbmcsIEFkZGl0aW9uYWxBbnN3ZXI+ID0ge307XHJcbiAgcmVhZG9ubHkgVmlld2VkQWx0ZXJuYXRpdmVWYWx1ZXM6IEFycmF5PG51bWJlcj4gPSBbXTtcclxuICByZWFkb25seSBWaWV3ZWRTZWN0aW9uR3VpZHM6IEFycmF5PG51bWJlcj4gPSBbXTtcclxuICByZWFkb25seSBBbHRlcm5hdGl2ZVNlcXVlbmNlQ2xpY2tzOiBBcnJheTxbbnVtYmVyLCBudW1iZXJdPiA9IFtdO1xyXG4gIFN0YXJ0ZWRBdCA9IDA7XHJcbiAgRmluaXNoZWRBdCA9IDA7XHJcbiAgQW5zd2VyVmFsdWU6IGFueSA9IE1JU1NJTkdfVkFMVUU7XHJcbiAgYW5zd2VySXNDb21wbGV0ZTogYm9vbGVhbiA9IGZhbHNlO1xyXG4gIGNvbnN0cnVjdG9yKGd1aWQ6IG51bWJlciwgdmVyc2lvbjogbnVtYmVyID0gLTEpIHtcclxuICAgIHRoaXMuUXVlc3Rpb25HVUlEID0gZ3VpZDtcclxuICAgIHRoaXMuUXVlc3Rpb25WZXJzaW9uID0gdmVyc2lvbjtcclxuICB9XHJcbiAgY2xlYXIoKTogdm9pZCB7XHJcbiAgICB0aGlzLlN0YXJ0ZWRBdCA9IDA7XHJcbiAgICB0aGlzLkZpbmlzaGVkQXQgPSAwO1xyXG4gICAgdGhpcy5BbnN3ZXJWYWx1ZSA9IE1JU1NJTkdfVkFMVUU7XHJcbiAgICB0aGlzLlZpZXdlZEFsdGVybmF0aXZlVmFsdWVzLmxlbmd0aCA9IDA7XHJcbiAgICB0aGlzLlZpZXdlZFNlY3Rpb25HdWlkcy5sZW5ndGggPSAwO1xyXG4gICAgdGhpcy5BbHRlcm5hdGl2ZVNlcXVlbmNlQ2xpY2tzLmxlbmd0aCA9IDA7XHJcbiAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyh0aGlzLkFuc3dlckFkZGl0aW9uYWwpKSB7XHJcbiAgICAgIHRoaXMuQW5zd2VyQWRkaXRpb25hbFtOdW1iZXIoa2V5KV0gPSBudWxsO1xyXG4gICAgfVxyXG4gIH1cclxuICAvLyBhZGRBbnN3ZXIoYW5zd2VyVmFsdWU6IHVua25vd24sIGFuc3dlckFkZGl0aW9uYWw6IExvZ2ljT2JqZWN0LCBhc0NvbXBsZXRlOiBib29sZWFuKTogdm9pZCB7XHJcbiAgLy8gICB0aGlzLkFuc3dlclZhbHVlID0gYW5zd2VyVmFsdWU7XHJcbiAgLy8gfVxyXG5cclxufVxyXG4iXX0=