@snabcentr/client-ui 0.17.3 → 0.18.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -1,73 +1,74 @@
1
- import { ChangeDetectorRef, OnInit } from '@angular/core';
2
- import { ScPriceHistory } from '@snabcentr/client-core';
3
- import { ECharts, EChartsOption } from 'echarts';
1
+ import { OnInit } from '@angular/core';
2
+ import { ScCatalogService, ScProduct } from '@snabcentr/client-core';
3
+ import { TuiDay, TuiDayRange, TuiStringHandler } from '@taiga-ui/cdk';
4
+ import { TuiDialogContext } from '@taiga-ui/core';
5
+ import { Observable } from 'rxjs';
4
6
  import * as i0 from "@angular/core";
5
7
  /**
6
8
  * График истории цен товара или услуги.
7
- * TODO: TASK[#7482] Перепроверить возможность использования TuiLineDaysChart.
8
9
  */
9
10
  export declare class ScPriceHistoryComponent implements OnInit {
10
- private readonly cdr;
11
+ private readonly catalogService;
12
+ private readonly months$;
13
+ private readonly context?;
11
14
  /**
12
- * История цен на товар/услугу.
15
+ * {@link Observable} временного промежутка истории цен.
13
16
  */
14
- history: ScPriceHistory;
17
+ range$: Observable<TuiDayRange>;
15
18
  /**
16
- * Максимальная цена товара.
17
- */
18
- maxPrice?: number;
19
- /**
20
- * Минимальная цена товара.
19
+ * Товар или услуга, для которого необходимо отобразить историю цен.
21
20
  */
22
- minPrice?: number;
21
+ product: ScProduct;
23
22
  /**
24
- * Настройки графика.
23
+ * {@link Observable} истории цен.
25
24
  */
26
- initOption: {
27
- devicePixelRatio?: number;
28
- renderer?: string;
29
- width?: number | string;
30
- height?: number | string;
31
- locale?: string;
32
- };
25
+ history$?: Observable<ReadonlyArray<[TuiDay, number]>>;
33
26
  /**
34
- * Параметры отрисовки графика истории цены.
27
+ * Максимальная цена товара.
35
28
  */
36
- chartOption: EChartsOption;
29
+ priceInfo$?: Observable<{
30
+ min: number;
31
+ max: number;
32
+ }>;
37
33
  /**
38
- * Данные об истории цены на товар или услугу.
34
+ * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.
39
35
  */
40
- private data;
36
+ readonly xStringify$: Observable<TuiStringHandler<TuiDay>>;
41
37
  /**
42
- * Экземпляр {@link ECharts} в шаблоне.
38
+ * Функция для преобразования значения number в строку в подсказке по оси Y.
39
+ *
40
+ * @param value Значение оси Y.
43
41
  */
44
- private eChartsInstance;
42
+ readonly yStringify: TuiStringHandler<number>;
45
43
  /**
46
44
  * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.
47
45
  *
48
- * @param cdr Объект для работы с обнаружением изменений.
46
+ * @param catalogService Сервис для работы с каталогом.
47
+ * @param months$ Перечисление месяцев.
48
+ * @param context Контекст диалогового окна, в котором открыт компонент.
49
49
  */
50
- constructor(cdr: ChangeDetectorRef);
50
+ constructor(catalogService: ScCatalogService, months$: Observable<readonly string[]>, context?: TuiDialogContext<void, {
51
+ product: ScProduct;
52
+ }> | undefined);
51
53
  /** @inheritDoc */
52
54
  ngOnInit(): void;
53
55
  /**
54
- * Перехватчик жизненного цикла {@link ECharts}, который вызывается при его инициализации.
56
+ * Вычисляет подписи даты к оси X.
55
57
  *
56
- * @param eChartsInstance Экземпляр {@link ECharts}.
57
- */
58
- onChartInit(eChartsInstance: ECharts): void;
59
- /**
60
- * Устанавливает новые данные {@link ScPriceHistoryComponent.eChartsInstance}.
58
+ * @param param0 {@link TuiDayRange} Временной промежуток истории цен.
59
+ * @param param0.from {@link TuiDay} Дата начала истории цен.
60
+ * @param param0.to {@link TuiDay} Дата конца истории цен.
61
61
  */
62
- private setChartData;
62
+ computeLabels$({ from, to }: TuiDayRange): Observable<readonly string[]>;
63
63
  /**
64
- * Добавляет значение цены товара в массив истории цен.
64
+ * Вычисляет данные для отображения на графике.
65
65
  *
66
- * @param value Цена товара или услуги.
67
- * @param date Дата установки цены.
68
- * @param nextDate Следующая дата установки цены.
66
+ * @param param0 {@link TuiDayRange} Временной промежуток истории цен.
67
+ * @param param0.from {@link TuiDay} Дата начала истории цен.
68
+ * @param param0.to {@link TuiDay} Дата конца истории цен.
69
+ * @param history История цен.
69
70
  */
70
- private pushDataItem;
71
- static ɵfac: i0.ɵɵFactoryDeclaration<ScPriceHistoryComponent, never>;
72
- static ɵcmp: i0.ɵɵComponentDeclaration<ScPriceHistoryComponent, "sc-price-history", never, { "history": "history"; }, {}, never, never, false>;
71
+ computeValue({ from, to }: TuiDayRange, history: ReadonlyArray<[TuiDay, number]>): ReadonlyArray<[TuiDay, number]>;
72
+ static ɵfac: i0.ɵɵFactoryDeclaration<ScPriceHistoryComponent, [null, null, { optional: true; }]>;
73
+ static ɵcmp: i0.ɵɵComponentDeclaration<ScPriceHistoryComponent, "sc-price-history", never, { "product": "product"; }, {}, never, never, false>;
73
74
  }
@@ -12,13 +12,13 @@ import * as i10 from "@angular/router";
12
12
  import * as i11 from "@taiga-ui/core";
13
13
  import * as i12 from "@taiga-ui/kit";
14
14
  import * as i13 from "@angular/forms";
15
- import * as i14 from "ngx-echarts";
16
- import * as i15 from "@taiga-ui/cdk";
15
+ import * as i14 from "@taiga-ui/cdk";
16
+ import * as i15 from "@taiga-ui/addon-charts";
17
17
  /**
18
18
  * Модуль каталога.
19
19
  */
20
20
  export declare class ScCatalogModule {
21
21
  static ɵfac: i0.ɵɵFactoryDeclaration<ScCatalogModule, never>;
22
- static ɵmod: i0.ɵɵNgModuleDeclaration<ScCatalogModule, [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent], [typeof i9.CommonModule, typeof i10.RouterModule, typeof i11.TuiButtonModule, typeof i11.TuiSvgModule, typeof i12.TuiIslandModule, typeof i12.TuiInputNumberModule, typeof i11.TuiLabelModule, typeof i11.TuiTextfieldControllerModule, typeof i13.FormsModule, typeof i13.ReactiveFormsModule, typeof i11.TuiHintModule, typeof i11.TuiModeModule, typeof i12.TuiFieldErrorPipeModule, typeof i11.TuiLoaderModule, typeof i11.TuiLinkModule, typeof i12.TuiElasticContainerModule, typeof i14.NgxEchartsModule, typeof i15.TuiLetModule, typeof i15.TuiRepeatTimesModule, typeof i12.TuiHighlightModule], [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent]>;
22
+ static ɵmod: i0.ɵɵNgModuleDeclaration<ScCatalogModule, [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent], [typeof i9.CommonModule, typeof i10.RouterModule, typeof i11.TuiButtonModule, typeof i11.TuiSvgModule, typeof i12.TuiIslandModule, typeof i12.TuiInputNumberModule, typeof i11.TuiLabelModule, typeof i11.TuiTextfieldControllerModule, typeof i13.FormsModule, typeof i13.ReactiveFormsModule, typeof i11.TuiHintModule, typeof i11.TuiModeModule, typeof i12.TuiFieldErrorPipeModule, typeof i11.TuiLoaderModule, typeof i11.TuiLinkModule, typeof i12.TuiElasticContainerModule, typeof i14.TuiLetModule, typeof i14.TuiRepeatTimesModule, typeof i12.TuiHighlightModule, typeof i15.TuiLineDaysChartModule, typeof i15.TuiAxesModule], [typeof i1.ScPriceListPaginationComponent, typeof i2.ScCategoryCardComponent, typeof i3.ScFavoriteBtnComponent, typeof i4.ScInputQuantityComponent, typeof i5.ScPriceCardComponent, typeof i6.ScPriceWarehouseStockComponent, typeof i7.ScPriceHistoryComponent, typeof i8.ScCategoriesListComponent]>;
23
23
  static ɵinj: i0.ɵɵInjectorDeclaration<ScCatalogModule>;
24
24
  }
@@ -1,108 +1,114 @@
1
- import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
2
- import { scChartOption } from './sc-chart-option';
1
+ import { __decorate } from "tslib";
2
+ /* eslint-disable class-methods-use-this,lodash/prefer-lodash-method */
3
+ import { ChangeDetectionStrategy, Component, Inject, Input, Optional } from '@angular/core';
4
+ import { TuiDay, TuiDayRange, TuiMonth, tuiPure } from '@taiga-ui/cdk';
5
+ import { TUI_MONTHS } from '@taiga-ui/core';
6
+ import { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';
7
+ import { filter, map, shareReplay } from 'rxjs';
3
8
  import * as i0 from "@angular/core";
4
- import * as i1 from "@angular/common";
5
- import * as i2 from "ngx-echarts";
9
+ import * as i1 from "@snabcentr/client-core";
10
+ import * as i2 from "@angular/common";
11
+ import * as i3 from "@taiga-ui/addon-charts";
12
+ import * as i4 from "rxjs";
6
13
  /**
7
14
  * График истории цен товара или услуги.
8
- * TODO: TASK[#7482] Перепроверить возможность использования TuiLineDaysChart.
9
15
  */
10
16
  export class ScPriceHistoryComponent {
11
17
  /**
12
18
  * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.
13
19
  *
14
- * @param cdr Объект для работы с обнаружением изменений.
20
+ * @param catalogService Сервис для работы с каталогом.
21
+ * @param months$ Перечисление месяцев.
22
+ * @param context Контекст диалогового окна, в котором открыт компонент.
15
23
  */
16
- constructor(cdr) {
17
- this.cdr = cdr;
24
+ constructor(catalogService, months$, context) {
25
+ this.catalogService = catalogService;
26
+ this.months$ = months$;
27
+ this.context = context;
18
28
  /**
19
- * Настройки графика.
29
+ * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.
20
30
  */
21
- this.initOption = { locale: 'RU' };
31
+ this.xStringify$ = this.months$.pipe(map((months) => ({ month, day }) =>
32
+ // eslint-disable-next-line security/detect-object-injection
33
+ `${months[month]}, ${day}`));
22
34
  /**
23
- * Параметры отрисовки графика истории цены.
35
+ * Функция для преобразования значения number в строку в подсказке по оси Y.
36
+ *
37
+ * @param value Значение оси Y.
24
38
  */
25
- this.chartOption = scChartOption;
26
- /**
27
- * Данные об истории цены на товар или услугу.
28
- */
29
- this.data = [];
39
+ this.yStringify = (value) => `${value.toLocaleString('ru-RU')} ₽`;
40
+ if (context) {
41
+ this.product = context.data.product;
42
+ }
30
43
  }
31
44
  /** @inheritDoc */
32
45
  ngOnInit() {
33
- if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {
34
- this.chartOption.series.data = [];
35
- }
36
- Object.keys(this.history)
37
- .map((key) => {
38
- // ? Можно избежать переведя даты в api в формат ECMAScript® 2023: https://tc39.es/ecma262/#sec-date-time-string-format
39
- const dataString = key.split('.').reverse();
40
- const data = new Date(+dataString[0], +dataString[1] - 1, +dataString[2]);
41
- return { data: data, cost: this.history[String(key)].cost };
42
- })
43
- .sort((a, b) => +a.data - +b.data)
44
- .forEach((item, index, array) => {
45
- let nextDate;
46
- // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
47
- if (array[index + 1]?.data) {
48
- nextDate = new Date(array[index + 1]?.data);
49
- nextDate.setDate(nextDate.getDate() - 1);
50
- }
51
- else {
52
- nextDate = new Date();
53
- }
54
- this.pushDataItem(item.cost, item.data, nextDate);
55
- });
56
- // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
57
- if (this.eChartsInstance) {
58
- this.setChartData();
59
- }
46
+ this.history$ = this.catalogService.getPriceHistory$(this.product).pipe(map((history) => Object.keys(history)
47
+ .map((item) => [TuiDay.normalizeParse(item, 'DMY'), history[String(item)].cost])
48
+ .sort((a, b) => (a[0] > b[0] ? 1 : -1))), shareReplay({ bufferSize: 1, refCount: true }), filter((history) => history.length > 1));
49
+ this.priceInfo$ = this.history$.pipe(map((history) => history.map((item) => item[1])), map((values) => ({
50
+ min: Math.min(...values),
51
+ max: Math.max(...values),
52
+ })));
53
+ this.range$ = this.history$.pipe(map((value) => ({ from: value.at(0)?.[0], to: value.at(-1)?.[0] })), filter((range) => !!range.from && !!range.to), map((range) => {
54
+ return new TuiDayRange(range.from, range.to);
55
+ }));
60
56
  }
61
57
  /**
62
- * Перехватчик жизненного цикла {@link ECharts}, который вызывается при его инициализации.
58
+ * Вычисляет подписи даты к оси X.
63
59
  *
64
- * @param eChartsInstance Экземпляр {@link ECharts}.
60
+ * @param param0 {@link TuiDayRange} Временной промежуток истории цен.
61
+ * @param param0.from {@link TuiDay} Дата начала истории цен.
62
+ * @param param0.to {@link TuiDay} Дата конца истории цен.
65
63
  */
66
- // eslint-disable-next-line unicorn/prevent-abbreviations
67
- onChartInit(eChartsInstance) {
68
- this.eChartsInstance = eChartsInstance;
69
- if (this.data.length > 0) {
70
- this.setChartData();
71
- }
72
- }
73
- /**
74
- * Устанавливает новые данные {@link ScPriceHistoryComponent.eChartsInstance}.
75
- */
76
- setChartData() {
77
- if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {
78
- this.maxPrice = Math.max(...this.data.map((item) => item.value[1]));
79
- this.minPrice = Math.min(...this.data.map((item) => item.value[1]));
80
- this.chartOption.series.data = this.data;
81
- this.eChartsInstance.clear();
82
- this.eChartsInstance.setOption(this.chartOption, true, true);
83
- this.cdr.markForCheck();
84
- }
64
+ computeLabels$({ from, to }) {
65
+ return this.months$.pipe(map((months) => Array.from({ length: TuiMonth.lengthBetween(from, to) + 1 }, (_, index) => months[from.append({ month: index }).month])));
85
66
  }
86
67
  /**
87
- * Добавляет значение цены товара в массив истории цен.
68
+ * Вычисляет данные для отображения на графике.
88
69
  *
89
- * @param value Цена товара или услуги.
90
- * @param date Дата установки цены.
91
- * @param nextDate Следующая дата установки цены.
70
+ * @param param0 {@link TuiDayRange} Временной промежуток истории цен.
71
+ * @param param0.from {@link TuiDay} Дата начала истории цен.
72
+ * @param param0.to {@link TuiDay} Дата конца истории цен.
73
+ * @param history История цен.
92
74
  */
93
- pushDataItem(value, date, nextDate) {
94
- this.data.push({
95
- name: date.toString(),
96
- value: [date, value, nextDate],
97
- });
75
+ computeValue({ from, to }, history) {
76
+ return (Array.from({ length: TuiDay.lengthBetween(from, to) + 1 })
77
+ .fill(0)
78
+ // eslint-disable-next-line unicorn/no-array-reduce
79
+ .reduce((array, _, index) => {
80
+ const data = from.append({ day: index });
81
+ return [
82
+ ...array,
83
+ [
84
+ data,
85
+ (history.find((valueHistory, indexHistory, arrayHistory) => data.daySame(valueHistory[0]) || data.dayBefore(arrayHistory[indexHistory + 1][0])) ??
86
+ history[0])[1],
87
+ ],
88
+ ];
89
+ }, []));
98
90
  }
99
91
  }
100
- ScPriceHistoryComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScPriceHistoryComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
101
- ScPriceHistoryComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScPriceHistoryComponent, selector: "sc-price-history", inputs: { history: "history" }, ngImport: i0, template: "<div class=\"flex flex-col items-center\">\n <div *ngIf=\"maxPrice && minPrice\" class=\"w-full font-bold text-end text-lg mb-1\">\u043E\u0442 {{ minPrice.toLocaleString() }} \u20BD \u0434\u043E {{ maxPrice.toLocaleString() }} \u20BD</div>\n <div class=\"relative w-full h-56\">\n <div class=\"h-48 bg-tui-base-02 mt-2 absolute rounded right-0 left-16\"></div>\n <div echarts [initOpts]=\"initOption\" (chartInit)=\"onChartInit($event)\" [options]=\"chartOption\" class=\"w-full !h-full touch-none\"></div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgxEchartsDirective, selector: "echarts, [echarts]", inputs: ["options", "theme", "loading", "initOpts", "merge", "autoResize", "loadingType", "loadingOpts"], outputs: ["chartInit", "optionsError", "chartClick", "chartDblClick", "chartMouseDown", "chartMouseMove", "chartMouseUp", "chartMouseOver", "chartMouseOut", "chartGlobalOut", "chartContextMenu", "chartLegendSelectChanged", "chartLegendSelected", "chartLegendUnselected", "chartLegendScroll", "chartDataZoom", "chartDataRangeSelected", "chartTimelineChanged", "chartTimelinePlayChanged", "chartRestore", "chartDataViewChanged", "chartMagicTypeChanged", "chartPieSelectChanged", "chartPieSelected", "chartPieUnselected", "chartMapSelectChanged", "chartMapSelected", "chartMapUnselected", "chartAxisAreaSelected", "chartFocusNodeAdjacency", "chartUnfocusNodeAdjacency", "chartBrush", "chartBrushEnd", "chartBrushSelected", "chartRendered", "chartFinished"], exportAs: ["echarts"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
92
+ ScPriceHistoryComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScPriceHistoryComponent, deps: [{ token: i1.ScCatalogService }, { token: TUI_MONTHS }, { token: POLYMORPHEUS_CONTEXT, optional: true }], target: i0.ɵɵFactoryTarget.Component });
93
+ ScPriceHistoryComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: ScPriceHistoryComponent, selector: "sc-price-history", inputs: { product: "product" }, ngImport: i0, template: "<div class=\"flex flex-col items-center\">\n <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n <div class=\"w-174 mb-1 text-end text-lg font-bold\">\u043E\u0442 {{ priceInfo.min.toLocaleString() }} \u20BD \u0434\u043E {{ priceInfo.max.toLocaleString() }} \u20BD</div>\n\n <ng-container *ngIf=\"history$ | async as history\">\n <div\n *ngIf=\"range$ | async as range\"\n class=\"relative h-56 w-full p-5 text-tui-primary\"\n >\n <tui-axes\n *ngIf=\"computeLabels$(range) | async as labels\"\n class=\"h-full\"\n [axisXLabels]=\"labels\"\n [horizontalLines]=\"4\"\n [verticalLines]=\"labels.length\"\n >\n <tui-line-days-chart\n class=\"chart\"\n [height]=\"priceInfo.max\"\n [smoothingFactor]=\"10\"\n [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n [value]=\"computeValue(range, history)\"\n [xStringify]=\"xStringify$ | async\"\n [yStringify]=\"yStringify\"\n [hintContent]=\"hint\"\n ></tui-line-days-chart>\n </tui-axes>\n </div>\n </ng-container>\n </ng-container>\n <ng-template #isNotExist>\n <p class=\"text-base font-bold\">\u0426\u0435\u043D\u0430 \u043D\u0430 \u0442\u043E\u0432\u0430\u0440 \u043D\u0435 \u0438\u0437\u043C\u0435\u043D\u044F\u043B\u0430\u0441\u044C</p>\n </ng-template>\n <ng-template\n #hint\n let-data\n >\n <div class=\"font-bold\">{{ data[1].toLocaleString() }} \u20BD</div>\n <div>{{ data[0] }}</div>\n </ng-template>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.TuiLineDaysChartComponent, selector: "tui-line-days-chart", inputs: ["value", "y", "height", "smoothingFactor", "hintContent", "xStringify", "yStringify", "dots"] }, { kind: "component", type: i3.TuiAxesComponent, selector: "tui-axes", inputs: ["axisX", "axisXLabels", "axisY", "axisYInset", "axisYLabels", "axisYName", "axisYSecondaryInset", "axisYSecondaryLabels", "axisYSecondaryName", "horizontalLines", "horizontalLinesHandler", "verticalLines", "verticalLinesHandler"] }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
94
+ __decorate([
95
+ tuiPure
96
+ ], ScPriceHistoryComponent.prototype, "computeLabels$", null);
97
+ __decorate([
98
+ tuiPure
99
+ ], ScPriceHistoryComponent.prototype, "computeValue", null);
102
100
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScPriceHistoryComponent, decorators: [{
103
101
  type: Component,
104
- args: [{ selector: 'sc-price-history', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center\">\n <div *ngIf=\"maxPrice && minPrice\" class=\"w-full font-bold text-end text-lg mb-1\">\u043E\u0442 {{ minPrice.toLocaleString() }} \u20BD \u0434\u043E {{ maxPrice.toLocaleString() }} \u20BD</div>\n <div class=\"relative w-full h-56\">\n <div class=\"h-48 bg-tui-base-02 mt-2 absolute rounded right-0 left-16\"></div>\n <div echarts [initOpts]=\"initOption\" (chartInit)=\"onChartInit($event)\" [options]=\"chartOption\" class=\"w-full !h-full touch-none\"></div>\n </div>\n</div>\n" }]
105
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { history: [{
102
+ args: [{ selector: 'sc-price-history', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"flex flex-col items-center\">\n <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n <div class=\"w-174 mb-1 text-end text-lg font-bold\">\u043E\u0442 {{ priceInfo.min.toLocaleString() }} \u20BD \u0434\u043E {{ priceInfo.max.toLocaleString() }} \u20BD</div>\n\n <ng-container *ngIf=\"history$ | async as history\">\n <div\n *ngIf=\"range$ | async as range\"\n class=\"relative h-56 w-full p-5 text-tui-primary\"\n >\n <tui-axes\n *ngIf=\"computeLabels$(range) | async as labels\"\n class=\"h-full\"\n [axisXLabels]=\"labels\"\n [horizontalLines]=\"4\"\n [verticalLines]=\"labels.length\"\n >\n <tui-line-days-chart\n class=\"chart\"\n [height]=\"priceInfo.max\"\n [smoothingFactor]=\"10\"\n [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n [value]=\"computeValue(range, history)\"\n [xStringify]=\"xStringify$ | async\"\n [yStringify]=\"yStringify\"\n [hintContent]=\"hint\"\n ></tui-line-days-chart>\n </tui-axes>\n </div>\n </ng-container>\n </ng-container>\n <ng-template #isNotExist>\n <p class=\"text-base font-bold\">\u0426\u0435\u043D\u0430 \u043D\u0430 \u0442\u043E\u0432\u0430\u0440 \u043D\u0435 \u0438\u0437\u043C\u0435\u043D\u044F\u043B\u0430\u0441\u044C</p>\n </ng-template>\n <ng-template\n #hint\n let-data\n >\n <div class=\"font-bold\">{{ data[1].toLocaleString() }} \u20BD</div>\n <div>{{ data[0] }}</div>\n </ng-template>\n</div>\n" }]
103
+ }], ctorParameters: function () { return [{ type: i1.ScCatalogService }, { type: i4.Observable, decorators: [{
104
+ type: Inject,
105
+ args: [TUI_MONTHS]
106
+ }] }, { type: undefined, decorators: [{
107
+ type: Optional
108
+ }, {
109
+ type: Inject,
110
+ args: [POLYMORPHEUS_CONTEXT]
111
+ }] }]; }, propDecorators: { product: [{
106
112
  type: Input
107
- }] } });
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-history.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.ts","../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport { ScPriceHistory } from '@snabcentr/client-core';\nimport { ECharts, EChartsOption } from 'echarts';\n\nimport { scChartOption } from './sc-chart-option';\nimport { ScIChartDataItem } from './sc-i-chart-data-item';\n\n/**\n * График истории цен товара или услуги.\n * TODO: TASK[#7482] Перепроверить возможность использования TuiLineDaysChart.\n */\n@Component({\n    selector: 'sc-price-history',\n    templateUrl: './sc-price-history.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceHistoryComponent implements OnInit {\n    /**\n     * История цен на товар/услугу.\n     */\n    @Input() public history: ScPriceHistory;\n\n    /**\n     * Максимальная цена товара.\n     */\n    public maxPrice?: number;\n\n    /**\n     * Минимальная цена товара.\n     */\n    public minPrice?: number;\n\n    /**\n     * Настройки графика.\n     */\n    public initOption: {\n        devicePixelRatio?: number;\n        renderer?: string;\n        width?: number | string;\n        height?: number | string;\n        locale?: string;\n    } = { locale: 'RU' };\n\n    /**\n     * Параметры отрисовки графика истории цены.\n     */\n    public chartOption: EChartsOption = scChartOption;\n\n    /**\n     * Данные об истории цены на товар или услугу.\n     */\n    private data: ScIChartDataItem[] = [];\n\n    /**\n     * Экземпляр {@link ECharts} в шаблоне.\n     */\n    private eChartsInstance: ECharts;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.\n     *\n     * @param cdr Объект для работы с обнаружением изменений.\n     */\n    public constructor(private readonly cdr: ChangeDetectorRef) {}\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {\n            this.chartOption.series.data = [];\n        }\n\n        Object.keys(this.history)\n            .map((key) => {\n                // ? Можно избежать переведя даты в api в формат ECMAScript® 2023: https://tc39.es/ecma262/#sec-date-time-string-format\n                const dataString = key.split('.').reverse();\n                const data = new Date(+dataString[0], +dataString[1] - 1, +dataString[2]);\n\n                return { data: data, cost: this.history[String(key)].cost };\n            })\n            .sort((a, b) => +a.data - +b.data)\n            .forEach((item, index, array) => {\n                let nextDate: Date;\n\n                // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n                if (array[index + 1]?.data) {\n                    nextDate = new Date(array[index + 1]?.data);\n                    nextDate.setDate(nextDate.getDate() - 1);\n                } else {\n                    nextDate = new Date();\n                }\n\n                this.pushDataItem(item.cost, item.data, nextDate);\n            });\n\n        // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n        if (this.eChartsInstance) {\n            this.setChartData();\n        }\n    }\n\n    /**\n     * Перехватчик жизненного цикла {@link ECharts}, который вызывается при его инициализации.\n     *\n     * @param eChartsInstance Экземпляр {@link ECharts}.\n     */\n    // eslint-disable-next-line unicorn/prevent-abbreviations\n    public onChartInit(eChartsInstance: ECharts) {\n        this.eChartsInstance = eChartsInstance;\n\n        if (this.data.length > 0) {\n            this.setChartData();\n        }\n    }\n\n    /**\n     * Устанавливает новые данные {@link ScPriceHistoryComponent.eChartsInstance}.\n     */\n    private setChartData(): void {\n        if (this.chartOption.series && !Array.isArray(this.chartOption.series)) {\n            this.maxPrice = Math.max(...this.data.map((item) => item.value[1]));\n            this.minPrice = Math.min(...this.data.map((item) => item.value[1]));\n            this.chartOption.series.data = this.data;\n            this.eChartsInstance.clear();\n            this.eChartsInstance.setOption(this.chartOption, true, true);\n            this.cdr.markForCheck();\n        }\n    }\n\n    /**\n     * Добавляет значение цены товара в массив истории цен.\n     *\n     * @param value Цена товара или услуги.\n     * @param date Дата установки цены.\n     * @param nextDate Следующая дата установки цены.\n     */\n    private pushDataItem(value: number, date: Date, nextDate: Date): void {\n        this.data.push({\n            name: date.toString(),\n            value: [date, value, nextDate],\n        });\n    }\n}\n","<div class=\"flex flex-col items-center\">\n    <div *ngIf=\"maxPrice && minPrice\" class=\"w-full font-bold text-end text-lg mb-1\">от {{ minPrice.toLocaleString() }} ₽ до {{ maxPrice.toLocaleString() }} ₽</div>\n    <div class=\"relative w-full h-56\">\n        <div class=\"h-48 bg-tui-base-02 mt-2 absolute rounded right-0 left-16\"></div>\n        <div echarts [initOpts]=\"initOption\" (chartInit)=\"onChartInit($event)\" [options]=\"chartOption\" class=\"w-full !h-full touch-none\"></div>\n    </div>\n</div>\n"]}
113
+ }], computeLabels$: [], computeValue: [] } });
114
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sc-price-history.component.js","sourceRoot":"","sources":["../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.ts","../../../../../projects/client-ui/catalog/price-history/sc-price-history.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable class-methods-use-this,lodash/prefer-lodash-method */\nimport { ChangeDetectionStrategy, Component, Inject, Input, OnInit, Optional } from '@angular/core';\nimport { ScCatalogService, ScProduct } from '@snabcentr/client-core';\nimport { TuiDay, TuiDayRange, TuiMonth, tuiPure, TuiStringHandler } from '@taiga-ui/cdk';\nimport { TUI_MONTHS, TuiDialogContext } from '@taiga-ui/core';\nimport { POLYMORPHEUS_CONTEXT } from '@tinkoff/ng-polymorpheus';\nimport { filter, map, Observable, shareReplay } from 'rxjs';\n\n/**\n * График истории цен товара или услуги.\n */\n@Component({\n    selector: 'sc-price-history',\n    templateUrl: './sc-price-history.component.html',\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ScPriceHistoryComponent implements OnInit {\n    /**\n     * {@link Observable} временного промежутка истории цен.\n     */\n    public range$: Observable<TuiDayRange>;\n\n    /**\n     * Товар или услуга, для которого необходимо отобразить историю цен.\n     */\n    @Input() public product: ScProduct;\n\n    /**\n     * {@link Observable} истории цен.\n     */\n    public history$?: Observable<ReadonlyArray<[TuiDay, number]>>;\n\n    /**\n     * Максимальная цена товара.\n     */\n    public priceInfo$?: Observable<{\n        min: number;\n        max: number;\n    }>;\n\n    /**\n     * {@link Observable} функция для преобразования значения number в строку в подсказке по оси X.\n     */\n    public readonly xStringify$: Observable<TuiStringHandler<TuiDay>> = this.months$.pipe(\n        map(\n            (months) =>\n                ({ month, day }) =>\n                    // eslint-disable-next-line security/detect-object-injection\n                    `${months[month]}, ${day}`\n        )\n    );\n\n    /**\n     * Функция для преобразования значения number в строку в подсказке по оси Y.\n     *\n     * @param value Значение оси Y.\n     */\n    public readonly yStringify: TuiStringHandler<number> = (value) => `${value.toLocaleString('ru-RU')} ₽`;\n\n    /**\n     * Инициализирует экземпляр класса {@link ScPriceHistoryComponent}.\n     *\n     * @param catalogService Сервис для работы с каталогом.\n     * @param months$ Перечисление месяцев.\n     * @param context Контекст диалогового окна, в котором открыт компонент.\n     */\n    public constructor(\n        private readonly catalogService: ScCatalogService,\n        @Inject(TUI_MONTHS) private readonly months$: Observable<readonly string[]>,\n        @Optional()\n        @Inject(POLYMORPHEUS_CONTEXT)\n        private readonly context?: TuiDialogContext<void, { product: ScProduct }>\n    ) {\n        if (context) {\n            this.product = context.data.product;\n        }\n    }\n\n    /** @inheritDoc */\n    public ngOnInit(): void {\n        this.history$ = this.catalogService.getPriceHistory$(this.product).pipe(\n            map((history) =>\n                Object.keys(history)\n                    .map((item): [TuiDay, number] => [TuiDay.normalizeParse(item, 'DMY'), history[String(item)].cost])\n                    .sort((a, b) => (a[0] > b[0] ? 1 : -1))\n            ),\n            shareReplay({ bufferSize: 1, refCount: true }),\n            filter((history) => history.length > 1)\n        );\n\n        this.priceInfo$ = this.history$.pipe(\n            map((history) => history.map((item) => item[1])),\n            map((values) => ({\n                min: Math.min(...values),\n                max: Math.max(...values),\n            }))\n        );\n\n        this.range$ = this.history$.pipe(\n            map((value: ReadonlyArray<[TuiDay, number]>) => ({ from: value.at(0)?.[0], to: value.at(-1)?.[0] })),\n            filter((range): range is { from: TuiDay; to: TuiDay } => !!range.from && !!range.to),\n            map((range) => {\n                return new TuiDayRange(range.from, range.to);\n            })\n        );\n    }\n\n    /**\n     * Вычисляет подписи даты к оси X.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     */\n    @tuiPure\n    public computeLabels$({ from, to }: TuiDayRange): Observable<readonly string[]> {\n        return this.months$.pipe(map((months) => Array.from({ length: TuiMonth.lengthBetween(from, to) + 1 }, (_, index) => months[from.append({ month: index }).month])));\n    }\n\n    /**\n     * Вычисляет данные для отображения на графике.\n     *\n     * @param param0 {@link TuiDayRange} Временной промежуток истории цен.\n     * @param param0.from {@link TuiDay} Дата начала истории цен.\n     * @param param0.to {@link TuiDay} Дата конца истории цен.\n     * @param history История цен.\n     */\n    @tuiPure\n    public computeValue({ from, to }: TuiDayRange, history: ReadonlyArray<[TuiDay, number]>): ReadonlyArray<[TuiDay, number]> {\n        return (\n            Array.from({ length: TuiDay.lengthBetween(from, to) + 1 })\n                .fill(0)\n                // eslint-disable-next-line unicorn/no-array-reduce\n                .reduce<ReadonlyArray<[TuiDay, number]>>((array, _, index) => {\n                    const data = from.append({ day: index });\n                    return [\n                        ...array,\n                        [\n                            data,\n                            (history.find((valueHistory, indexHistory, arrayHistory) => data.daySame(valueHistory[0]) || data.dayBefore(arrayHistory[indexHistory + 1][0])) ??\n                                history[0])[1],\n                        ],\n                    ];\n                }, [])\n        );\n    }\n}\n","<div class=\"flex flex-col items-center\">\n    <ng-container *ngIf=\"priceInfo$ | async as priceInfo; else isNotExist\">\n        <div class=\"w-174 mb-1 text-end text-lg font-bold\">от {{ priceInfo.min.toLocaleString() }} ₽ до {{ priceInfo.max.toLocaleString() }} ₽</div>\n\n        <ng-container *ngIf=\"history$ | async as history\">\n            <div\n                *ngIf=\"range$ | async as range\"\n                class=\"relative h-56 w-full p-5 text-tui-primary\"\n            >\n                <tui-axes\n                    *ngIf=\"computeLabels$(range) | async as labels\"\n                    class=\"h-full\"\n                    [axisXLabels]=\"labels\"\n                    [horizontalLines]=\"4\"\n                    [verticalLines]=\"labels.length\"\n                >\n                    <tui-line-days-chart\n                        class=\"chart\"\n                        [height]=\"priceInfo.max\"\n                        [smoothingFactor]=\"10\"\n                        [y]=\"priceInfo.min - (priceInfo.min * 1) / 4\"\n                        [value]=\"computeValue(range, history)\"\n                        [xStringify]=\"xStringify$ | async\"\n                        [yStringify]=\"yStringify\"\n                        [hintContent]=\"hint\"\n                    ></tui-line-days-chart>\n                </tui-axes>\n            </div>\n        </ng-container>\n    </ng-container>\n    <ng-template #isNotExist>\n        <p class=\"text-base font-bold\">Цена на товар не изменялась</p>\n    </ng-template>\n    <ng-template\n        #hint\n        let-data\n    >\n        <div class=\"font-bold\">{{ data[1].toLocaleString() }} ₽</div>\n        <div>{{ data[0] }}</div>\n    </ng-template>\n</div>\n"]}
@@ -2,27 +2,19 @@ import { CommonModule } from '@angular/common';
2
2
  import { NgModule } from '@angular/core';
3
3
  import { FormsModule, ReactiveFormsModule } from '@angular/forms';
4
4
  import { RouterModule } from '@angular/router';
5
+ import { TuiAxesModule, TuiLineDaysChartModule } from '@taiga-ui/addon-charts';
5
6
  import { TuiLetModule, TuiRepeatTimesModule } from '@taiga-ui/cdk';
6
7
  import { TuiButtonModule, TuiHintModule, TuiLabelModule, TuiLinkModule, TuiLoaderModule, TuiModeModule, TuiSvgModule, TuiTextfieldControllerModule } from '@taiga-ui/core';
7
8
  import { TuiElasticContainerModule, TuiFieldErrorPipeModule, TuiHighlightModule, TuiInputNumberModule, TuiIslandModule } from '@taiga-ui/kit';
8
- import { LineChart } from 'echarts/charts';
9
- import { GridComponent, TitleComponent, TooltipComponent } from 'echarts/components';
10
- import * as echarts from 'echarts/core';
11
- import { SVGRenderer } from 'echarts/renderers';
12
- import { NgxEchartsModule } from 'ngx-echarts';
13
9
  import { ScCategoriesListComponent } from './categories-list/sc-categories-list.component';
14
10
  import { ScCategoryCardComponent } from './category-card/sc-category-card.component';
15
11
  import { ScInputQuantityComponent } from './input-quantity/sc-input-quantity.component';
16
12
  import { ScPriceCardComponent } from './price-card/sc-price-card.component';
17
- import scLangRU from './price-history/sc-lang-ru';
18
13
  import { ScPriceHistoryComponent } from './price-history/sc-price-history.component';
19
14
  import { ScPriceListPaginationComponent } from './price-list-pagination/sc-price-list-pagination.component';
20
15
  import { ScPriceWarehouseStockComponent } from './price-warehouse-stock/sc-price-warehouse-stock.component';
21
16
  import { ScFavoriteBtnComponent as ScFavoriteButtonComponent } from './sc-favorite-btn/sc-favorite-btn.component';
22
17
  import * as i0 from "@angular/core";
23
- import * as i1 from "ngx-echarts";
24
- echarts.registerLocale('RU', scLangRU);
25
- echarts.use([TitleComponent, TooltipComponent, GridComponent, LineChart, SVGRenderer]);
26
18
  /**
27
19
  * Модуль каталога.
28
20
  */
@@ -51,9 +43,12 @@ ScCatalogModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version
51
43
  TuiFieldErrorPipeModule,
52
44
  TuiLoaderModule,
53
45
  TuiLinkModule,
54
- TuiElasticContainerModule, i1.NgxEchartsModule, TuiLetModule,
46
+ TuiElasticContainerModule,
47
+ TuiLetModule,
55
48
  TuiRepeatTimesModule,
56
- TuiHighlightModule], exports: [ScPriceListPaginationComponent,
49
+ TuiHighlightModule,
50
+ TuiLineDaysChartModule,
51
+ TuiAxesModule], exports: [ScPriceListPaginationComponent,
57
52
  ScCategoryCardComponent,
58
53
  ScFavoriteButtonComponent,
59
54
  ScInputQuantityComponent,
@@ -77,10 +72,11 @@ ScCatalogModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version
77
72
  TuiLoaderModule,
78
73
  TuiLinkModule,
79
74
  TuiElasticContainerModule,
80
- NgxEchartsModule.forRoot({ echarts }),
81
75
  TuiLetModule,
82
76
  TuiRepeatTimesModule,
83
- TuiHighlightModule] });
77
+ TuiHighlightModule,
78
+ TuiLineDaysChartModule,
79
+ TuiAxesModule] });
84
80
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: ScCatalogModule, decorators: [{
85
81
  type: NgModule,
86
82
  args: [{
@@ -121,11 +117,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
121
117
  TuiLoaderModule,
122
118
  TuiLinkModule,
123
119
  TuiElasticContainerModule,
124
- NgxEchartsModule.forRoot({ echarts }),
125
120
  TuiLetModule,
126
121
  TuiRepeatTimesModule,
127
122
  TuiHighlightModule,
123
+ TuiLineDaysChartModule,
124
+ TuiAxesModule,
128
125
  ],
129
126
  }]
130
127
  }] });
131
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2F0YWxvZy5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9zYy1jYXRhbG9nLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxZQUFZLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkUsT0FBTyxFQUFFLGVBQWUsRUFBRSxhQUFhLEVBQUUsY0FBYyxFQUFFLGFBQWEsRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLFlBQVksRUFBRSw0QkFBNEIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzNLLE9BQU8sRUFBRSx5QkFBeUIsRUFBRSx1QkFBdUIsRUFBRSxrQkFBa0IsRUFBRSxvQkFBb0IsRUFBRSxlQUFlLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDOUksT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzNDLE9BQU8sRUFBRSxhQUFhLEVBQUUsY0FBYyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDckYsT0FBTyxLQUFLLE9BQU8sTUFBTSxjQUFjLENBQUM7QUFDeEMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQ2hELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUUvQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxnREFBZ0QsQ0FBQztBQUMzRixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFB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QUFvQjt3QkFDcEIsa0JBQWtCO3FCQUNyQjtpQkFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBSb3V0ZXJNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgVHVpTGV0TW9kdWxlLCBUdWlSZXBlYXRUaW1lc01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jZGsnO1xuaW1wb3J0IHsgVHVpQnV0dG9uTW9kdWxlLCBUdWlIaW50TW9kdWxlLCBUdWlMYWJlbE1vZHVsZSwgVHVpTGlua01vZHVsZSwgVHVpTG9hZGVyTW9kdWxlLCBUdWlNb2RlTW9kdWxlLCBUdWlTdmdNb2R1bGUsIFR1aVRleHRmaWVsZENvbnRyb2xsZXJNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY29yZSc7XG5pbXBvcnQgeyBUdWlFbGFzdGljQ29udGFpbmVyTW9kdWxlLCBUdWlGaWVsZEVycm9yUGlwZU1vZHVsZSwgVHVpSGlnaGxpZ2h0TW9kdWxlLCBUdWlJbnB1dE51bWJlck1vZHVsZSwgVHVpSXNsYW5kTW9kdWxlIH0gZnJvbSAnQHRhaWdhLXVpL2tpdCc7XG5pbXBvcnQgeyBMaW5lQ2hhcnQgfSBmcm9tICdlY2hhcnRzL2NoYXJ0cyc7XG5pbXBvcnQgeyBHcmlkQ29tcG9uZW50LCBUaXRsZUNvbXBvbmVudCwgVG9vbHRpcENvbXBvbmVudCB9IGZyb20gJ2VjaGFydHMvY29tcG9uZW50cyc7XG5pbXBvcnQgKiBhcyBlY2hhcnRzIGZyb20gJ2VjaGFydHMvY29yZSc7XG5pbXBvcnQgeyBTVkdSZW5kZXJlciB9IGZyb20gJ2VjaGFydHMvcmVuZGVyZXJzJztcbmltcG9ydCB7IE5neEVjaGFydHNNb2R1bGUgfSBmcm9tICduZ3gtZWNoYXJ0cyc7XG5cbmltcG9ydCB7IFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQgfSBmcm9tICcuL2NhdGVnb3JpZXMtbGlzdC9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IFNjQ2F0ZWdvcnlDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9jYXRlZ29yeS1jYXJkL3NjLWNhdGVnb3J5LWNhcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCB9IGZyb20gJy4vaW5wdXQtcXVhbnRpdHkvc2MtaW5wdXQtcXVhbnRpdHkuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS1jYXJkL3NjLXByaWNlLWNhcmQuY29tcG9uZW50JztcbmltcG9ydCBzY0xhbmdSVSBmcm9tICcuL3ByaWNlLWhpc3Rvcnkvc2MtbGFuZy1ydSc7XG5pbXBvcnQgeyBTY1ByaWNlSGlzdG9yeUNvbXBvbmVudCB9IGZyb20gJy4vcHJpY2UtaGlzdG9yeS9zYy1wcmljZS1oaXN0b3J5LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTY1ByaWNlTGlzdFBhZ2luYXRpb25Db21wb25lbnQgfSBmcm9tICcuL3ByaWNlLWxpc3QtcGFnaW5hdGlvbi9zYy1wcmljZS1saXN0LXBhZ2luYXRpb24uY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VXYXJlaG91c2VTdG9ja0NvbXBvbmVudCB9IGZyb20gJy4vcHJpY2Utd2FyZWhvdXNlLXN0b2NrL3NjLXByaWNlLXdhcmVob3VzZS1zdG9jay5jb21wb25lbnQnO1xuaW1wb3J0IHsgU2NGYXZvcml0ZUJ0bkNvbXBvbmVudCBhcyBTY0Zhdm9yaXRlQnV0dG9uQ29tcG9uZW50IH0gZnJvbSAnLi9zYy1mYXZvcml0ZS1idG4vc2MtZmF2b3JpdGUtYnRuLmNvbXBvbmVudCc7XG5cbmVjaGFydHMucmVnaXN0ZXJMb2NhbGUoJ1JVJywgc2NMYW5nUlUpO1xuZWNoYXJ0cy51c2UoW1RpdGxlQ29tcG9uZW50LCBUb29sdGlwQ29tcG9uZW50LCBHcmlkQ29tcG9uZW50LCBMaW5lQ2hhcnQsIFNWR1JlbmRlcmVyXSk7XG5cbi8qKlxuICog0JzQvtC00YPQu9GMINC60LDRgtCw0LvQvtCz0LAuXG4gKi9cbkBOZ01vZHVsZSh7XG4gICAgZGVjbGFyYXRpb25zOiBbXG4gICAgICAgIFNjUHJpY2VMaXN0UGFnaW5hdGlvbkNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yeUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjRmF2b3JpdGVCdXR0b25Db21wb25lbnQsXG4gICAgICAgIFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjUHJpY2VXYXJlaG91c2VTdG9ja0NvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUhpc3RvcnlDb21wb25lbnQsXG4gICAgICAgIFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQsXG4gICAgXSxcbiAgICBleHBvcnRzOiBbXG4gICAgICAgIFNjUHJpY2VMaXN0UGFnaW5hdGlvbkNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yeUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjRmF2b3JpdGVCdXR0b25Db21wb25lbnQsXG4gICAgICAgIFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUNhcmRDb21wb25lbnQsXG4gICAgICAgIFNjUHJpY2VXYXJlaG91c2VTdG9ja0NvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZUhpc3RvcnlDb21wb25lbnQsXG4gICAgICAgIFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQsXG4gICAgXSxcbiAgICBpbXBvcnRzOiBbXG4gICAgICAgIENvbW1vbk1vZHVsZSxcbiAgICAgICAgUm91dGVyTW9kdWxlLFxuICAgICAgICBUdWlCdXR0b25Nb2R1bGUsXG4gICAgICAgIFR1aVN2Z01vZHVsZSxcbiAgICAgICAgVHVpSXNsYW5kTW9kdWxlLFxuICAgICAgICBUdWlJbnB1dE51bWJlck1vZHVsZSxcbiAgICAgICAgVHVpTGFiZWxNb2R1bGUsXG4gICAgICAgIFR1aVRleHRmaWVsZENvbnRyb2xsZXJNb2R1bGUsXG4gICAgICAgIEZvcm1zTW9kdWxlLFxuICAgICAgICBSZWFjdGl2ZUZvcm1zTW9kdWxlLFxuICAgICAgICBUdWlIaW50TW9kdWxlLFxuICAgICAgICBUdWlNb2RlTW9kdWxlLFxuICAgICAgICBUdWlGaWVsZEVycm9yUGlwZU1vZHVsZSxcbiAgICAgICAgVHVpTG9hZGVyTW9kdWxlLFxuICAgICAgICBUdWlMaW5rTW9kdWxlLFxuICAgICAgICBUdWlFbGFzdGljQ29udGFpbmVyTW9kdWxlLFxuICAgICAgICBOZ3hFY2hhcnRzTW9kdWxlLmZvclJvb3QoeyBlY2hhcnRzIH0pLFxuICAgICAgICBUdWlMZXRNb2R1bGUsXG4gICAgICAgIFR1aVJlcGVhdFRpbWVzTW9kdWxlLFxuICAgICAgICBUdWlIaWdobGlnaHRNb2R1bGUsXG4gICAgXSxcbn0pXG5leHBvcnQgY2xhc3MgU2NDYXRhbG9nTW9kdWxlIHt9XG4iXX0=
128
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2MtY2F0YWxvZy5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9jbGllbnQtdWkvY2F0YWxvZy9zYy1jYXRhbG9nLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxhQUFhLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUMvRSxPQUFPLEVBQUUsWUFBWSxFQUFFLG9CQUFvQixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25FLE9BQU8sRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLGNBQWMsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLGFBQWEsRUFBRSxZQUFZLEVBQUUsNEJBQTRCLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUMzSyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsdUJBQXVCLEVBQUUsa0JBQWtCLEVBQUUsb0JBQW9CLEVBQUUsZUFBZSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRTlJLE9BQU8sRUFBRSx5QkFBeUIsRUFBRSxNQUFNLGdEQUFnRCxDQUFDO0FBQzNGLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLDRDQUE0QyxDQUFDO0FBQ3JGLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLDhDQUE4QyxDQUFDO0FBQ3hGLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHNDQUFzQyxDQUFDO0FBQzVFLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxNQUFNLDRDQUE0QyxDQUFDO0FBQ3JGLE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLDREQUE0RCxDQUFDO0FBQzVHLE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLDREQUE0RCxDQUFDO0FBQzVHLE9BQU8sRUFBRSxzQkFBc0IsSUFBSSx5QkFBeUIsRUFBRSxNQUFNLDZDQUE2QyxDQUFDOztBQUVsSDs7R0FFRztBQThDSCxNQUFNLE9BQU8sZUFBZTs7NEdBQWYsZUFBZTs2R0FBZixlQUFlLGlCQTNDcEIsOEJBQThCO1FBQzlCLHVCQUF1QjtRQUN2Qix5QkFBeUI7UUFDekIsd0JBQXdCO1FBQ3hCLG9CQUFvQjtRQUNwQiw4QkFBOEI7UUFDOUIsdUJBQXVCO1FBQ3ZCLHlCQUF5QixhQWF6QixZQUFZO1FBQ1osWUFBWTtRQUNaLGVBQWU7UUFDZixZQUFZO1FBQ1osZUFBZTtRQUNmLG9CQUFvQjtRQUNwQixjQUFjO1FBQ2QsNEJBQTRCO1FBQzVCLFdBQVc7UUFDWCxtQkFBbUI7UUFDbkIsYUFBYTtRQUNiLGFBQWE7UUFDYix1QkFBdUI7UUFDdkIsZUFBZTtRQUNmLGFBQWE7UUFDYix5QkFBeUI7UUFDekIsWUFBWTtRQUNaLG9CQUFvQjtRQUNwQixrQkFBa0I7UUFDbEIsc0JBQXNCO1FBQ3RCLGFBQWEsYUE5QmIsOEJBQThCO1FBQzlCLHVCQUF1QjtRQUN2Qix5QkFBeUI7UUFDekIsd0JBQXdCO1FBQ3hCLG9CQUFvQjtRQUNwQiw4QkFBOEI7UUFDOUIsdUJBQXVCO1FBQ3ZCLHlCQUF5Qjs2R0EwQnBCLGVBQWUsWUF2QnBCLFlBQVk7UUFDWixZQUFZO1FBQ1osZUFBZTtRQUNmLFlBQVk7UUFDWixlQUFlO1FBQ2Ysb0JBQW9CO1FBQ3BCLGNBQWM7UUFDZCw0QkFBNEI7UUFDNUIsV0FBVztRQUNYLG1CQUFtQjtRQUNuQixhQUFhO1FBQ2IsYUFBYTtRQUNiLHVCQUF1QjtRQUN2QixlQUFlO1FBQ2YsYUFBYTtRQUNiLHlCQUF5QjtRQUN6QixZQUFZO1FBQ1osb0JBQW9CO1FBQ3BCLGtCQUFrQjtRQUNsQixzQkFBc0I7UUFDdEIsYUFBYTsyRkFHUixlQUFlO2tCQTdDM0IsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUU7d0JBQ1YsOEJBQThCO3dCQUM5Qix1QkFBdUI7d0JBQ3ZCLHlCQUF5Qjt3QkFDekIsd0JBQXdCO3dCQUN4QixvQkFBb0I7d0JBQ3BCLDhCQUE4Qjt3QkFDOUIsdUJBQXVCO3dCQUN2Qix5QkFBeUI7cUJBQzVCO29CQUNELE9BQU8sRUFBRTt3QkFDTCw4QkFBOEI7d0JBQzlCLHVCQUF1Qjt3QkFDdkIseUJBQXlCO3dCQUN6Qix3QkFBd0I7d0JBQ3hCLG9CQUFvQjt3QkFDcEIsOEJBQThCO3dCQUM5Qix1QkFBdUI7d0JBQ3ZCLHlCQUF5QjtxQkFDNUI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNMLFlBQVk7d0JBQ1osWUFBWTt3QkFDWixlQUFlO3dCQUNmLFlBQVk7d0JBQ1osZUFBZTt3QkFDZixvQkFBb0I7d0JBQ3BCLGNBQWM7d0JBQ2QsNEJBQTRCO3dCQUM1QixXQUFXO3dCQUNYLG1CQUFtQjt3QkFDbkIsYUFBYTt3QkFDYixhQUFhO3dCQUNiLHVCQUF1Qjt3QkFDdkIsZUFBZTt3QkFDZixhQUFhO3dCQUNiLHlCQUF5Qjt3QkFDekIsWUFBWTt3QkFDWixvQkFBb0I7d0JBQ3BCLGtCQUFrQjt3QkFDbEIsc0JBQXNCO3dCQUN0QixhQUFhO3FCQUNoQjtpQkFDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBSb3V0ZXJNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgVHVpQXhlc01vZHVsZSwgVHVpTGluZURheXNDaGFydE1vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9hZGRvbi1jaGFydHMnO1xuaW1wb3J0IHsgVHVpTGV0TW9kdWxlLCBUdWlSZXBlYXRUaW1lc01vZHVsZSB9IGZyb20gJ0B0YWlnYS11aS9jZGsnO1xuaW1wb3J0IHsgVHVpQnV0dG9uTW9kdWxlLCBUdWlIaW50TW9kdWxlLCBUdWlMYWJlbE1vZHVsZSwgVHVpTGlua01vZHVsZSwgVHVpTG9hZGVyTW9kdWxlLCBUdWlNb2RlTW9kdWxlLCBUdWlTdmdNb2R1bGUsIFR1aVRleHRmaWVsZENvbnRyb2xsZXJNb2R1bGUgfSBmcm9tICdAdGFpZ2EtdWkvY29yZSc7XG5pbXBvcnQgeyBUdWlFbGFzdGljQ29udGFpbmVyTW9kdWxlLCBUdWlGaWVsZEVycm9yUGlwZU1vZHVsZSwgVHVpSGlnaGxpZ2h0TW9kdWxlLCBUdWlJbnB1dE51bWJlck1vZHVsZSwgVHVpSXNsYW5kTW9kdWxlIH0gZnJvbSAnQHRhaWdhLXVpL2tpdCc7XG5cbmltcG9ydCB7IFNjQ2F0ZWdvcmllc0xpc3RDb21wb25lbnQgfSBmcm9tICcuL2NhdGVnb3JpZXMtbGlzdC9zYy1jYXRlZ29yaWVzLWxpc3QuY29tcG9uZW50JztcbmltcG9ydCB7IFNjQ2F0ZWdvcnlDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9jYXRlZ29yeS1jYXJkL3NjLWNhdGVnb3J5LWNhcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFNjSW5wdXRRdWFudGl0eUNvbXBvbmVudCB9IGZyb20gJy4vaW5wdXQtcXVhbnRpdHkvc2MtaW5wdXQtcXVhbnRpdHkuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS1jYXJkL3NjLXByaWNlLWNhcmQuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VIaXN0b3J5Q29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS1oaXN0b3J5L3NjLXByaWNlLWhpc3RvcnkuY29tcG9uZW50JztcbmltcG9ydCB7IFNjUHJpY2VMaXN0UGFnaW5hdGlvbkNvbXBvbmVudCB9IGZyb20gJy4vcHJpY2UtbGlzdC1wYWdpbmF0aW9uL3NjLXByaWNlLWxpc3QtcGFnaW5hdGlvbi5jb21wb25lbnQnO1xuaW1wb3J0IHsgU2NQcmljZVdhcmVob3VzZVN0b2NrQ29tcG9uZW50IH0gZnJvbSAnLi9wcmljZS13YXJlaG91c2Utc3RvY2svc2MtcHJpY2Utd2FyZWhvdXNlLXN0b2NrLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTY0Zhdm9yaXRlQnRuQ29tcG9uZW50IGFzIFNjRmF2b3JpdGVCdXR0b25Db21wb25lbnQgfSBmcm9tICcuL3NjLWZhdm9yaXRlLWJ0bi9zYy1mYXZvcml0ZS1idG4uY29tcG9uZW50JztcblxuLyoqXG4gKiDQnNC+0LTRg9C70Ywg0LrQsNGC0LDQu9C+0LPQsC5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgICBkZWNsYXJhdGlvbnM6IFtcbiAgICAgICAgU2NQcmljZUxpc3RQYWdpbmF0aW9uQ29tcG9uZW50LFxuICAgICAgICBTY0NhdGVnb3J5Q2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NGYXZvcml0ZUJ1dHRvbkNvbXBvbmVudCxcbiAgICAgICAgU2NJbnB1dFF1YW50aXR5Q29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlQ2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZVdhcmVob3VzZVN0b2NrQ29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlSGlzdG9yeUNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yaWVzTGlzdENvbXBvbmVudCxcbiAgICBdLFxuICAgIGV4cG9ydHM6IFtcbiAgICAgICAgU2NQcmljZUxpc3RQYWdpbmF0aW9uQ29tcG9uZW50LFxuICAgICAgICBTY0NhdGVnb3J5Q2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NGYXZvcml0ZUJ1dHRvbkNvbXBvbmVudCxcbiAgICAgICAgU2NJbnB1dFF1YW50aXR5Q29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlQ2FyZENvbXBvbmVudCxcbiAgICAgICAgU2NQcmljZVdhcmVob3VzZVN0b2NrQ29tcG9uZW50LFxuICAgICAgICBTY1ByaWNlSGlzdG9yeUNvbXBvbmVudCxcbiAgICAgICAgU2NDYXRlZ29yaWVzTGlzdENvbXBvbmVudCxcbiAgICBdLFxuICAgIGltcG9ydHM6IFtcbiAgICAgICAgQ29tbW9uTW9kdWxlLFxuICAgICAgICBSb3V0ZXJNb2R1bGUsXG4gICAgICAgIFR1aUJ1dHRvbk1vZHVsZSxcbiAgICAgICAgVHVpU3ZnTW9kdWxlLFxuICAgICAgICBUdWlJc2xhbmRNb2R1bGUsXG4gICAgICAgIFR1aUlucHV0TnVtYmVyTW9kdWxlLFxuICAgICAgICBUdWlMYWJlbE1vZHVsZSxcbiAgICAgICAgVHVpVGV4dGZpZWxkQ29udHJvbGxlck1vZHVsZSxcbiAgICAgICAgRm9ybXNNb2R1bGUsXG4gICAgICAgIFJlYWN0aXZlRm9ybXNNb2R1bGUsXG4gICAgICAgIFR1aUhpbnRNb2R1bGUsXG4gICAgICAgIFR1aU1vZGVNb2R1bGUsXG4gICAgICAgIFR1aUZpZWxkRXJyb3JQaXBlTW9kdWxlLFxuICAgICAgICBUdWlMb2FkZXJNb2R1bGUsXG4gICAgICAgIFR1aUxpbmtNb2R1bGUsXG4gICAgICAgIFR1aUVsYXN0aWNDb250YWluZXJNb2R1bGUsXG4gICAgICAgIFR1aUxldE1vZHVsZSxcbiAgICAgICAgVHVpUmVwZWF0VGltZXNNb2R1bGUsXG4gICAgICAgIFR1aUhpZ2hsaWdodE1vZHVsZSxcbiAgICAgICAgVHVpTGluZURheXNDaGFydE1vZHVsZSxcbiAgICAgICAgVHVpQXhlc01vZHVsZSxcbiAgICBdLFxufSlcbmV4cG9ydCBjbGFzcyBTY0NhdGFsb2dNb2R1bGUge31cbiJdfQ==