@hestia-earth/ui-components 0.0.34 → 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (36) hide show
  1. package/bundles/hestia-earth-ui-components.umd.js +103 -64
  2. package/bundles/hestia-earth-ui-components.umd.js.map +1 -1
  3. package/cycles/cycles-emissions-chart/cycles-emissions-chart.component.d.ts +4 -1
  4. package/cycles/cycles-result/cycles-result.component.d.ts +3 -1
  5. package/esm2015/bibliographies/bibliographies-search-confirm/bibliographies-search-confirm.component.js +5 -4
  6. package/esm2015/common/blank-node-state/blank-node-state.component.js +5 -4
  7. package/esm2015/common/blank-node-state-notice/blank-node-state-notice.component.js +5 -4
  8. package/esm2015/common/blank-node-value-delta/blank-node-value-delta.component.js +5 -4
  9. package/esm2015/common/link-key-value/link-key-value.component.js +5 -4
  10. package/esm2015/common/schema-version-link/schema-version-link.component.js +5 -4
  11. package/esm2015/common/social-tags/social-tags.component.js +5 -4
  12. package/esm2015/common/unit-converter/unit-converter.component.js +5 -4
  13. package/esm2015/cycles/cycles-completeness/cycles-completeness.component.js +5 -4
  14. package/esm2015/cycles/cycles-emissions-chart/cycles-emissions-chart.component.js +9 -5
  15. package/esm2015/cycles/cycles-functional-unit-measure/cycles-functional-unit-measure.component.js +5 -4
  16. package/esm2015/cycles/cycles-result/cycles-result.component.js +10 -8
  17. package/esm2015/engine/engine-orchestrator-edit/engine-orchestrator-edit.component.js +5 -4
  18. package/esm2015/engine/engine-requirements-form/engine-requirements-form.component.js +5 -4
  19. package/esm2015/files/files-form/files-form.component.js +5 -4
  20. package/esm2015/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.js +9 -5
  21. package/esm2015/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.js +9 -5
  22. package/esm2015/node/node-csv-export-confirm/node-csv-export-confirm.component.js +5 -4
  23. package/esm2015/node/node-csv-select-headers/node-csv-select-headers.component.js +5 -4
  24. package/esm2015/node/node-diffs/node-diffs.component.js +5 -4
  25. package/esm2015/node/node-icon/node-icon.component.js +5 -4
  26. package/esm2015/node/node-link/node-link.component.js +5 -4
  27. package/esm2015/node/node-logs-file/node-logs-file.component.js +5 -4
  28. package/esm2015/node/node-missing-lookup-factors/node-missing-lookup-factors.component.js +5 -4
  29. package/esm2015/node/node-value-details/node-value-details.component.js +5 -4
  30. package/esm2015/search/search.model.js +3 -3
  31. package/esm2015/sites/sites-maps/sites-maps.component.js +5 -4
  32. package/fesm2015/hestia-earth-ui-components.js +101 -65
  33. package/fesm2015/hestia-earth-ui-components.js.map +1 -1
  34. package/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.d.ts +4 -2
  35. package/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.d.ts +4 -1
  36. package/package.json +6 -6
@@ -44,7 +44,8 @@ const toCsv = (logs) => [
44
44
  .map(v => v.join(','))
45
45
  ].join('\n');
46
46
  export class ImpactAssessmentsIndicatorBreakdownChartComponent {
47
- constructor(domSanitizer, searchService, nodeService) {
47
+ constructor(ngZone, domSanitizer, searchService, nodeService) {
48
+ this.ngZone = ngZone;
48
49
  this.domSanitizer = domSanitizer;
49
50
  this.searchService = searchService;
50
51
  this.nodeService = nodeService;
@@ -90,7 +91,7 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
90
91
  get id() {
91
92
  return this.impactAssessment['@id'] || this.impactAssessment.id;
92
93
  }
93
- updateChart() {
94
+ initChart() {
94
95
  var _a, _b, _c;
95
96
  const logs = this.logs
96
97
  .filter(({ indicator }) => indicator === this.selectedTerm['@id'])
@@ -169,8 +170,11 @@ export class ImpactAssessmentsIndicatorBreakdownChartComponent {
169
170
  }
170
171
  });
171
172
  }
173
+ updateChart() {
174
+ this.ngZone.runOutsideAngular(() => this.initChart());
175
+ }
172
176
  }
173
- ImpactAssessmentsIndicatorBreakdownChartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsIndicatorBreakdownChartComponent, deps: [{ token: i1.DomSanitizer }, { token: i2.HeSearchService }, { token: i3.HeNodeService }], target: i0.ɵɵFactoryTarget.Component });
177
+ ImpactAssessmentsIndicatorBreakdownChartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsIndicatorBreakdownChartComponent, deps: [{ token: i0.NgZone }, { token: i1.DomSanitizer }, { token: i2.HeSearchService }, { token: i3.HeNodeService }], target: i0.ɵɵFactoryTarget.Component });
174
178
  ImpactAssessmentsIndicatorBreakdownChartComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: ImpactAssessmentsIndicatorBreakdownChartComponent, selector: "he-impact-assessments-indicator-breakdown-chart", inputs: { impactAssessment: "impactAssessment", indicators: "indicators" }, viewQueries: [{ propertyName: "chartRef", first: true, predicate: ["chart"], descendants: true }], ngImport: i0, template: "<div class=\"p-3\" [class.is-hidden]=\"loading || !terms?.length\">\n <div class=\"columns\">\n <div class=\"column\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <span class=\"button is-small is-static\">Select an Indicator</span>\n </div>\n <div class=\"control is-expanded\">\n <div class=\"select is-fullwidth is-small\">\n <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{term.name}} ({{term.units}})</option>\n </select>\n </div>\n </div>\n </div>\n </div>\n <div class=\"column is-narrow\">\n <a class=\"button is-dark is-outlined is-small\" [href]=\"csvContent\" [download]=\"id + '-logs.csv'\">\n <fa-icon icon=\"download\"></fa-icon>\n <span class=\"pl-2\">Download All (CSV)</span>\n </a>\n </div>\n </div>\n\n <div class=\"mt-1\">\n <p *ngIf=\"noData\" class=\"has-text-centered\">No breakdown available for {{selectedTerm?.name}}.</p>\n <div class=\"chart-container h-100\">\n <canvas #chart></canvas>\n </div>\n </div>\n</div>\n\n<ng-container *ngIf=\"loading\">\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-container>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"], components: [{ type: i4.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i5.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { type: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i5.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i5.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
175
179
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsIndicatorBreakdownChartComponent, decorators: [{
176
180
  type: Component,
@@ -179,7 +183,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
179
183
  templateUrl: './impact-assessments-indicator-breakdown-chart.component.html',
180
184
  styleUrls: ['./impact-assessments-indicator-breakdown-chart.component.scss']
181
185
  }]
182
- }], ctorParameters: function () { return [{ type: i1.DomSanitizer }, { type: i2.HeSearchService }, { type: i3.HeNodeService }]; }, propDecorators: { chartRef: [{
186
+ }], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i1.DomSanitizer }, { type: i2.HeSearchService }, { type: i3.HeNodeService }]; }, propDecorators: { chartRef: [{
183
187
  type: ViewChild,
184
188
  args: ['chart']
185
189
  }], impactAssessment: [{
@@ -187,4 +191,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
187
191
  }], indicators: [{
188
192
  type: Input
189
193
  }] } });
190
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicator-breakdown-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, OnInit, ViewChild } from '@angular/core';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { of } from 'rxjs';\nimport { filter, groupBy, map, mergeAll, mergeMap, toArray } from 'rxjs/operators';\nimport { Chart } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\nimport { DataState } from '@hestia-earth/api';\nimport { IImpactAssessmentJSONLD, Indicator, ITermJSONLD, NodeType, Term, TermTermType } from '@hestia-earth/schema';\nimport { toPrecision } from '@hestia-earth/utils';\n\nimport { matchTermType, matchType } from '../../search/search.model';\nimport { HeSearchService } from '../../search/search.service';\nimport { HeNodeService } from '../../node/node.service';\nimport { Level, parseLines, parseMessage } from '../../common/logs-utils';\nimport { listColor } from '../../common/utils';\n\ninterface ILog {\n  indicator: string;\n  emission: string;\n  coefficient: number;\n  value: number;\n  // store all indicators with inputs\n  inputs?: {\n    name: string;\n    value: number;\n  }[];\n  inputsValue?: number;\n}\n\nconst parseLog = (data: any): ILog => ({\n  indicator: data['key/term'] || data.term || data.indicator,\n  emission: data.node,\n  coefficient: +data.coefficient,\n  value: +data.value * +data.coefficient\n});\n\nconst filterTermTypes = [\n  TermTermType.emission,\n  TermTermType.characterisedIndicator\n];\n\nconst toCsv = (logs: ILog[]) => [\n  [\n    'Model',\n    'Impact',\n    'Value',\n    'Inputs',\n    'Inputs value'\n  ].join(','),\n  ...logs\n    .sort((a, b) => a.indicator.localeCompare(b.indicator))\n    .flatMap(({ indicator, emission, value, inputs }) => [\n      [indicator, emission, value, '', ''],\n      ...(inputs.map(v => [indicator, emission, '', v.name, v.value]))\n    ])\n    .map(v => v.join(','))\n].join('\\n');\n\n@Component({\n  selector: 'he-impact-assessments-indicator-breakdown-chart',\n  templateUrl: './impact-assessments-indicator-breakdown-chart.component.html',\n  styleUrls: ['./impact-assessments-indicator-breakdown-chart.component.scss']\n})\nexport class ImpactAssessmentsIndicatorBreakdownChartComponent implements OnInit {\n  private logs: ILog[] = [];\n  private emissions: ITermJSONLD[] = [];\n\n  @ViewChild('chart')\n  private chartRef?: ElementRef;\n  private chart: any;\n\n  @Input()\n  public impactAssessment?: IImpactAssessmentJSONLD;\n  @Input()\n  public indicators: Indicator[] = [];\n\n  public loading = true;\n  public terms: Term[] = [];\n  public selectedTerm?: Term;\n  public noData = false;\n  public csvContent?: SafeResourceUrl;\n\n  constructor(\n    private domSanitizer: DomSanitizer,\n    private searchService: HeSearchService,\n    private nodeService: HeNodeService\n  ) { }\n\n  async ngOnInit() {\n    this.terms = this.indicators.map(({ term }) => term!);\n\n    this.logs = (await of({\n      ...this.impactAssessment,\n      dataState: DataState.recalculated\n    }).pipe(\n      mergeMap(node => this.nodeService.getLog(node)),\n      map(value => value ? parseLines(value) : []),\n      mergeAll(),\n      filter(({ data }) => data.logger === 'hestia_earth.models' && data.level === Level.debug),\n      map(({ data }) => parseMessage(data.message)),\n      filter(message => 'node' in message),\n      map(parseLog),\n      filter(log => !!log.indicator && !!log.emission && !isNaN(log.value) && log.value > 0),\n      groupBy(log => [log.indicator, log.emission].join('/')),\n      mergeMap(group => group.pipe(toArray())),\n      map((values: ILog[]) => {\n        const log = values[0];\n        const total = values.reduce((prev, curr) => prev + curr.value, 0);\n        const inputs = (this.impactAssessment.emissionsResourceUse || [])\n          .filter(v => v.term['@id'] === log.emission && v.inputs?.length)\n          .map(v => ({ name: v.inputs.map(i => i['@id']).join(';'), value: v.value * log.coefficient }));\n        const inputsValue = inputs.reduce((prev, curr) => prev + curr.value, 0);\n        return { ...log, value: total, inputs, inputsValue };\n      }),\n      toArray()\n    ).toPromise()) as ILog[];\n    this.csvContent = this.domSanitizer.bypassSecurityTrustResourceUrl(\n      `data:text/html;charset=utf-8,${encodeURIComponent(toCsv(this.logs))}`\n    );\n\n    const { results: emissions } = await this.searchService.search<ITermJSONLD, NodeType.Term>({\n      fields: ['@type', '@id', 'name'],\n      limit: 1000,\n      query: {\n        bool: {\n          must: [\n            matchType(NodeType.Term)\n          ],\n          should: filterTermTypes.map(termType => matchTermType(termType)),\n          minimum_should_match: 1\n        }\n      }\n    });\n    this.emissions = emissions;\n\n    this.loading = false;\n\n    this.selectedTerm = this.terms[0];\n    this.updateChart();\n  }\n\n  public get id() {\n    return this.impactAssessment!['@id'] || (this.impactAssessment as any).id;\n  }\n\n  public updateChart() {\n    const logs = this.logs\n      .filter(({ indicator }) => indicator === this.selectedTerm!['@id'])\n      .sort((a, b) => b.value - a.value);\n    this.noData = logs.length === 0;\n\n    const total = logs.reduce((prev, curr) => prev + curr.value, 0);\n    const labels = logs.map(({ emission }) => this.emissions.find(v => v['@id'] === emission)?.name || emission);\n\n    // display the breakdown by input\n    // const inputs = logs.flatMap(v => v.inputs.map(i => i.name));\n    // const datasets = [\n    //   ...inputs.map((input, index) => ({\n    //     label: input,\n    //     data: logs.map(log => log.inputs.find(({ name }) => name === input)?.value || 0),\n    //     backgroundColor: itemColor(index),\n    //     borderColor: itemColor(index)\n    //   })),\n    //   // add dataset for rest not linked to inputs\n    //   {\n    //     label: this.selectedTerm?.name ?? '',\n    //     data: logs.map(log => log.value - log.inputsValue),\n    //     backgroundColor: itemColor(-1),\n    //     borderColor: itemColor(-1)\n    //   }\n    // ];\n\n    const datasets = [{\n      label: this.selectedTerm?.name ?? '',\n      data: logs.map(({ value }) => value),\n      backgroundColor: logs.map(listColor),\n      borderColor: logs.map(listColor)\n    }];\n\n    if (this.chart) {\n      this.chart.destroy();\n    }\n    this.chart = new Chart(this.chartRef?.nativeElement, {\n      type: 'horizontalBar',\n      data: {\n        labels,\n        datasets\n      },\n      plugins: [ChartDataLabels],\n      options: {\n        plugins: {\n          datalabels: {\n            color: 'black',\n            formatter: (value) => {\n              const ratio = toPrecision(value * 100 / total, 2);\n              return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n            },\n            align: 'end'\n          }\n        },\n        responsive: true,\n        maintainAspectRatio: false,\n        legend: {\n          display: false\n        },\n        tooltips: {\n          enabled: false,\n          callbacks: {\n            title: (tooltipItems, data) => data.datasets![tooltipItems[0].datasetIndex!]?.label || '',\n            label: (tooltipItem) => {\n              const value = +(tooltipItem.value || '0');\n              const ratio = toPrecision(value * 100 / total, 2);\n              return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n            }\n          }\n        },\n        scales: {\n          xAxes: [{\n            display: true\n          }],\n          yAxes: [{\n            position: 'left'\n          }]\n        }\n      }\n    });\n  }\n}\n","<div class=\"p-3\" [class.is-hidden]=\"loading || !terms?.length\">\n  <div class=\"columns\">\n    <div class=\"column\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static\">Select an Indicator</span>\n        </div>\n        <div class=\"control is-expanded\">\n          <div class=\"select is-fullwidth is-small\">\n            <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n              <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{term.name}} ({{term.units}})</option>\n            </select>\n          </div>\n        </div>\n      </div>\n    </div>\n    <div class=\"column is-narrow\">\n      <a class=\"button is-dark is-outlined is-small\" [href]=\"csvContent\" [download]=\"id + '-logs.csv'\">\n        <fa-icon icon=\"download\"></fa-icon>\n        <span class=\"pl-2\">Download All (CSV)</span>\n      </a>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <p *ngIf=\"noData\" class=\"has-text-centered\">No breakdown available for {{selectedTerm?.name}}.</p>\n    <div class=\"chart-container h-100\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n\n<ng-container *ngIf=\"loading\">\n  <div class=\"has-text-center py-3\">\n    <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n</ng-container>\n"]}
194
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicator-breakdown-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicator-breakdown-chart/impact-assessments-indicator-breakdown-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, Input, NgZone, OnInit, ViewChild } from '@angular/core';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { of } from 'rxjs';\nimport { filter, groupBy, map, mergeAll, mergeMap, toArray } from 'rxjs/operators';\nimport { Chart } from 'chart.js';\nimport ChartDataLabels from 'chartjs-plugin-datalabels';\nimport { DataState } from '@hestia-earth/api';\nimport { IImpactAssessmentJSONLD, Indicator, ITermJSONLD, NodeType, Term, TermTermType } from '@hestia-earth/schema';\nimport { toPrecision } from '@hestia-earth/utils';\n\nimport { matchTermType, matchType } from '../../search/search.model';\nimport { HeSearchService } from '../../search/search.service';\nimport { HeNodeService } from '../../node/node.service';\nimport { Level, parseLines, parseMessage } from '../../common/logs-utils';\nimport { listColor } from '../../common/utils';\n\ninterface ILog {\n  indicator: string;\n  emission: string;\n  coefficient: number;\n  value: number;\n  // store all indicators with inputs\n  inputs?: {\n    name: string;\n    value: number;\n  }[];\n  inputsValue?: number;\n}\n\nconst parseLog = (data: any): ILog => ({\n  indicator: data['key/term'] || data.term || data.indicator,\n  emission: data.node,\n  coefficient: +data.coefficient,\n  value: +data.value * +data.coefficient\n});\n\nconst filterTermTypes = [\n  TermTermType.emission,\n  TermTermType.characterisedIndicator\n];\n\nconst toCsv = (logs: ILog[]) => [\n  [\n    'Model',\n    'Impact',\n    'Value',\n    'Inputs',\n    'Inputs value'\n  ].join(','),\n  ...logs\n    .sort((a, b) => a.indicator.localeCompare(b.indicator))\n    .flatMap(({ indicator, emission, value, inputs }) => [\n      [indicator, emission, value, '', ''],\n      ...(inputs.map(v => [indicator, emission, '', v.name, v.value]))\n    ])\n    .map(v => v.join(','))\n].join('\\n');\n\n@Component({\n  selector: 'he-impact-assessments-indicator-breakdown-chart',\n  templateUrl: './impact-assessments-indicator-breakdown-chart.component.html',\n  styleUrls: ['./impact-assessments-indicator-breakdown-chart.component.scss']\n})\nexport class ImpactAssessmentsIndicatorBreakdownChartComponent implements OnInit {\n  private logs: ILog[] = [];\n  private emissions: ITermJSONLD[] = [];\n\n  @ViewChild('chart')\n  private chartRef?: ElementRef;\n  private chart: any;\n\n  @Input()\n  public impactAssessment?: IImpactAssessmentJSONLD;\n  @Input()\n  public indicators: Indicator[] = [];\n\n  public loading = true;\n  public terms: Term[] = [];\n  public selectedTerm?: Term;\n  public noData = false;\n  public csvContent?: SafeResourceUrl;\n\n  constructor(\n    private ngZone: NgZone,\n    private domSanitizer: DomSanitizer,\n    private searchService: HeSearchService,\n    private nodeService: HeNodeService\n  ) { }\n\n  async ngOnInit() {\n    this.terms = this.indicators.map(({ term }) => term!);\n\n    this.logs = (await of({\n      ...this.impactAssessment,\n      dataState: DataState.recalculated\n    }).pipe(\n      mergeMap(node => this.nodeService.getLog(node)),\n      map(value => value ? parseLines(value) : []),\n      mergeAll(),\n      filter(({ data }) => data.logger === 'hestia_earth.models' && data.level === Level.debug),\n      map(({ data }) => parseMessage(data.message)),\n      filter(message => 'node' in message),\n      map(parseLog),\n      filter(log => !!log.indicator && !!log.emission && !isNaN(log.value) && log.value > 0),\n      groupBy(log => [log.indicator, log.emission].join('/')),\n      mergeMap(group => group.pipe(toArray())),\n      map((values: ILog[]) => {\n        const log = values[0];\n        const total = values.reduce((prev, curr) => prev + curr.value, 0);\n        const inputs = (this.impactAssessment.emissionsResourceUse || [])\n          .filter(v => v.term['@id'] === log.emission && v.inputs?.length)\n          .map(v => ({ name: v.inputs.map(i => i['@id']).join(';'), value: v.value * log.coefficient }));\n        const inputsValue = inputs.reduce((prev, curr) => prev + curr.value, 0);\n        return { ...log, value: total, inputs, inputsValue };\n      }),\n      toArray()\n    ).toPromise()) as ILog[];\n    this.csvContent = this.domSanitizer.bypassSecurityTrustResourceUrl(\n      `data:text/html;charset=utf-8,${encodeURIComponent(toCsv(this.logs))}`\n    );\n\n    const { results: emissions } = await this.searchService.search<ITermJSONLD, NodeType.Term>({\n      fields: ['@type', '@id', 'name'],\n      limit: 1000,\n      query: {\n        bool: {\n          must: [\n            matchType(NodeType.Term)\n          ],\n          should: filterTermTypes.map(termType => matchTermType(termType)),\n          minimum_should_match: 1\n        }\n      }\n    });\n    this.emissions = emissions;\n\n    this.loading = false;\n\n    this.selectedTerm = this.terms[0];\n    this.updateChart();\n  }\n\n  public get id() {\n    return this.impactAssessment!['@id'] || (this.impactAssessment as any).id;\n  }\n\n  private initChart() {\n    const logs = this.logs\n      .filter(({ indicator }) => indicator === this.selectedTerm!['@id'])\n      .sort((a, b) => b.value - a.value);\n    this.noData = logs.length === 0;\n\n    const total = logs.reduce((prev, curr) => prev + curr.value, 0);\n    const labels = logs.map(({ emission }) => this.emissions.find(v => v['@id'] === emission)?.name || emission);\n\n    // display the breakdown by input\n    // const inputs = logs.flatMap(v => v.inputs.map(i => i.name));\n    // const datasets = [\n    //   ...inputs.map((input, index) => ({\n    //     label: input,\n    //     data: logs.map(log => log.inputs.find(({ name }) => name === input)?.value || 0),\n    //     backgroundColor: itemColor(index),\n    //     borderColor: itemColor(index)\n    //   })),\n    //   // add dataset for rest not linked to inputs\n    //   {\n    //     label: this.selectedTerm?.name ?? '',\n    //     data: logs.map(log => log.value - log.inputsValue),\n    //     backgroundColor: itemColor(-1),\n    //     borderColor: itemColor(-1)\n    //   }\n    // ];\n\n    const datasets = [{\n      label: this.selectedTerm?.name ?? '',\n      data: logs.map(({ value }) => value),\n      backgroundColor: logs.map(listColor),\n      borderColor: logs.map(listColor)\n    }];\n\n    if (this.chart) {\n      this.chart.destroy();\n    }\n    this.chart = new Chart(this.chartRef?.nativeElement, {\n      type: 'horizontalBar',\n      data: {\n        labels,\n        datasets\n      },\n      plugins: [ChartDataLabels],\n      options: {\n        plugins: {\n          datalabels: {\n            color: 'black',\n            formatter: (value) => {\n              const ratio = toPrecision(value * 100 / total, 2);\n              return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n            },\n            align: 'end'\n          }\n        },\n        responsive: true,\n        maintainAspectRatio: false,\n        legend: {\n          display: false\n        },\n        tooltips: {\n          enabled: false,\n          callbacks: {\n            title: (tooltipItems, data) => data.datasets![tooltipItems[0].datasetIndex!]?.label || '',\n            label: (tooltipItem) => {\n              const value = +(tooltipItem.value || '0');\n              const ratio = toPrecision(value * 100 / total, 2);\n              return value > 0 ? `${toPrecision(value, 3)} (${ratio}%)` : '';\n            }\n          }\n        },\n        scales: {\n          xAxes: [{\n            display: true\n          }],\n          yAxes: [{\n            position: 'left'\n          }]\n        }\n      }\n    });\n  }\n\n  public updateChart() {\n    this.ngZone.runOutsideAngular(() => this.initChart());\n  }\n}\n","<div class=\"p-3\" [class.is-hidden]=\"loading || !terms?.length\">\n  <div class=\"columns\">\n    <div class=\"column\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static\">Select an Indicator</span>\n        </div>\n        <div class=\"control is-expanded\">\n          <div class=\"select is-fullwidth is-small\">\n            <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n              <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{term.name}} ({{term.units}})</option>\n            </select>\n          </div>\n        </div>\n      </div>\n    </div>\n    <div class=\"column is-narrow\">\n      <a class=\"button is-dark is-outlined is-small\" [href]=\"csvContent\" [download]=\"id + '-logs.csv'\">\n        <fa-icon icon=\"download\"></fa-icon>\n        <span class=\"pl-2\">Download All (CSV)</span>\n      </a>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <p *ngIf=\"noData\" class=\"has-text-centered\">No breakdown available for {{selectedTerm?.name}}.</p>\n    <div class=\"chart-container h-100\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n\n<ng-container *ngIf=\"loading\">\n  <div class=\"has-text-center py-3\">\n    <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n</ng-container>\n"]}
@@ -21,7 +21,8 @@ const impactAssessmentDataset = (values, impact, index) => {
21
21
  };
22
22
  };
23
23
  export class ImpactAssessmentsIndicatorsChartComponent {
24
- constructor() {
24
+ constructor(ngZone) {
25
+ this.ngZone = ngZone;
25
26
  this.impactAssessments = [];
26
27
  this.selected = [];
27
28
  this.key = 'impacts';
@@ -52,7 +53,7 @@ export class ImpactAssessmentsIndicatorsChartComponent {
52
53
  this.selectedTerm = this.terms.includes(this.selectedTerm) ? this.selectedTerm : this.terms[0];
53
54
  return this.selectedTerm ? this.updateChart() : null;
54
55
  }
55
- updateChart() {
56
+ initChart() {
56
57
  var _a, _b, _c;
57
58
  const labels = [(_b = (_a = this.selectedTerm) === null || _a === void 0 ? void 0 : _a.name) !== null && _b !== void 0 ? _b : ''];
58
59
  const { values } = this.indicatorPerImpactAssessment[this.selectedTerm['@id']];
@@ -88,8 +89,11 @@ export class ImpactAssessmentsIndicatorsChartComponent {
88
89
  }
89
90
  });
90
91
  }
92
+ updateChart() {
93
+ this.ngZone.runOutsideAngular(() => this.initChart());
94
+ }
91
95
  }
92
- ImpactAssessmentsIndicatorsChartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsIndicatorsChartComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
96
+ ImpactAssessmentsIndicatorsChartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsIndicatorsChartComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
93
97
  ImpactAssessmentsIndicatorsChartComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: ImpactAssessmentsIndicatorsChartComponent, selector: "he-impact-assessments-indicators-chart", inputs: { impactAssessments: "impactAssessments", selected: "selected", key: "key", filterTermTypes: "filterTermTypes" }, viewQueries: [{ propertyName: "chartRef", first: true, predicate: ["chart"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"p-3\" [class.is-hidden]=\"!terms?.length\">\n <div class=\"field is-horizontal\">\n <div class=\"field-label is-normal\">\n <label class=\"label\" for=\"selectedTerm\">\n <span>Select a column</span>\n </label>\n </div>\n <div class=\"field-body\">\n <div class=\"control\">\n <div class=\"select is-small\">\n <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{term.name}}</option>\n </select>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"mt-1\">\n <div class=\"chart-container\">\n <canvas #chart></canvas>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;overflow:visible}.chart-container{height:400px;position:relative}\n"], directives: [{ type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }] });
94
98
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsIndicatorsChartComponent, decorators: [{
95
99
  type: Component,
@@ -98,7 +102,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
98
102
  templateUrl: './impact-assessments-indicators-chart.component.html',
99
103
  styleUrls: ['./impact-assessments-indicators-chart.component.scss']
100
104
  }]
101
- }], propDecorators: { chartRef: [{
105
+ }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { chartRef: [{
102
106
  type: ViewChild,
103
107
  args: ['chart']
104
108
  }], impactAssessments: [{
@@ -110,4 +114,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
110
114
  }], filterTermTypes: [{
111
115
  type: Input
112
116
  }] } });
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicators-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, AfterViewInit, OnChanges, ViewChild, SimpleChanges, ElementRef } from '@angular/core';\nimport { IImpactAssessmentJSONLD, Indicator, Term, TermTermType } from '@hestia-earth/schema';\nimport { Chart } from 'chart.js';\n\nimport { groupNodesByTerm, IGroupedNodes, IGroupedNodesValues } from '../../common/node-utils';\nimport { itemColor, defaultLabel } from '../../common/utils';\n\nconst isSelected = (selected: string[], v: IImpactAssessmentJSONLD) =>\n  selected.length === 0 || selected.includes(v['@id']);\n\nconst impactValue = (impact: IImpactAssessmentJSONLD, values: IGroupedNodesValues<Indicator>) =>\n  (values[impact['@id']]?.nodes[0] || { value: 0 }).value;\n\nconst impactName = (impact: IImpactAssessmentJSONLD, index: number) => `${index + 1}. ${defaultLabel(impact)}`;\n\nconst impactAssessmentDataset = (values: IGroupedNodesValues<Indicator>, impact: IImpactAssessmentJSONLD, index: number) => {\n  const label = impactName(impact, index);\n  const color = itemColor(index);\n  const data = [impactValue(impact, values)];\n  return {\n    label,\n    axis: 'y',\n    data,\n    backgroundColor: color,\n    borderColor: color\n  };\n};\n\n@Component({\n  selector: 'he-impact-assessments-indicators-chart',\n  templateUrl: './impact-assessments-indicators-chart.component.html',\n  styleUrls: ['./impact-assessments-indicators-chart.component.scss']\n})\nexport class ImpactAssessmentsIndicatorsChartComponent implements AfterViewInit, OnChanges {\n  @ViewChild('chart')\n  private chartRef?: ElementRef;\n  private chart: any;\n\n  @Input()\n  private impactAssessments: IImpactAssessmentJSONLD[] = [];\n  @Input()\n  private selected: string[] = [];\n  @Input()\n  private key: 'impacts' | 'emissionsResourceUse' = 'impacts';\n  @Input()\n  public filterTermTypes?: TermTermType[];\n\n  public indicatorPerImpactAssessment: IGroupedNodes<Indicator> = {};\n  public terms: Term[] = [];\n  public selectedTerm?: Term;\n\n  ngAfterViewInit() {\n    return this.init();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('impactAssessments' in changes && !changes.impactAssessments.firstChange) {\n      return this.init();\n    }\n    if ('selected' in changes && !changes.selected.firstChange) {\n      return this.init();\n    }\n  }\n\n  private termAllowed(term?: Term) {\n    return !this.filterTermTypes?.length || (this.filterTermTypes || []).includes(term?.termType!);\n  }\n\n  private init() {\n    const impacts = this.impactAssessments.filter(v => isSelected(this.selected, v));\n    this.indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      impacts, this.key\n    );\n    this.terms = Object.values(this.indicatorPerImpactAssessment)\n      .filter(({ term, values }: any) =>\n        this.termAllowed(term) && Object.values(values).some(({ nodes: [{ value }] }: any) => value >= 0)\n      )\n      .map(({ term }: any) => term);\n    this.selectedTerm = this.terms.includes(this.selectedTerm!) ? this.selectedTerm : this.terms[0];\n    return this.selectedTerm ? this.updateChart() : null;\n  }\n\n  public updateChart() {\n    const labels = [this.selectedTerm?.name ?? ''];\n    const { values } = this.indicatorPerImpactAssessment[this.selectedTerm!['@id']];\n    const datasets = this.impactAssessments\n      .map((impact, index) => isSelected(this.selected, impact) ? impactAssessmentDataset(values, impact, index) : null)\n      .filter(Boolean);\n\n    if (this.chart) {\n      this.chart.destroy();\n    }\n    this.chart = new Chart(this.chartRef?.nativeElement, {\n      type: 'bar',\n      data: {\n        labels,\n        datasets\n      },\n      options: {\n        responsive: true,\n        maintainAspectRatio: false,\n        legend: {\n          display: false\n        },\n        scales: {\n          xAxes: [{\n            display: false\n          }],\n          yAxes: [{\n            position: 'left',\n            ticks: {\n              min: 0\n            }\n          }]\n        }\n      }\n    });\n  }\n}\n","<div class=\"p-3\" [class.is-hidden]=\"!terms?.length\">\n  <div class=\"field is-horizontal\">\n    <div class=\"field-label is-normal\">\n      <label class=\"label\" for=\"selectedTerm\">\n        <span>Select a column</span>\n      </label>\n    </div>\n    <div class=\"field-body\">\n      <div class=\"control\">\n        <div class=\"select is-small\">\n          <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n            <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{term.name}}</option>\n          </select>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <div class=\"chart-container\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n"]}
117
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-indicators-chart.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.ts","../../../../src/impact-assessments/impact-assessments-indicators-chart/impact-assessments-indicators-chart.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component, Input, AfterViewInit, OnChanges, ViewChild, SimpleChanges, ElementRef, NgZone\n} from '@angular/core';\nimport { IImpactAssessmentJSONLD, Indicator, Term, TermTermType } from '@hestia-earth/schema';\nimport { Chart } from 'chart.js';\n\nimport { groupNodesByTerm, IGroupedNodes, IGroupedNodesValues } from '../../common/node-utils';\nimport { itemColor, defaultLabel } from '../../common/utils';\n\nconst isSelected = (selected: string[], v: IImpactAssessmentJSONLD) =>\n  selected.length === 0 || selected.includes(v['@id']);\n\nconst impactValue = (impact: IImpactAssessmentJSONLD, values: IGroupedNodesValues<Indicator>) =>\n  (values[impact['@id']]?.nodes[0] || { value: 0 }).value;\n\nconst impactName = (impact: IImpactAssessmentJSONLD, index: number) => `${index + 1}. ${defaultLabel(impact)}`;\n\nconst impactAssessmentDataset = (values: IGroupedNodesValues<Indicator>, impact: IImpactAssessmentJSONLD, index: number) => {\n  const label = impactName(impact, index);\n  const color = itemColor(index);\n  const data = [impactValue(impact, values)];\n  return {\n    label,\n    axis: 'y',\n    data,\n    backgroundColor: color,\n    borderColor: color\n  };\n};\n\n@Component({\n  selector: 'he-impact-assessments-indicators-chart',\n  templateUrl: './impact-assessments-indicators-chart.component.html',\n  styleUrls: ['./impact-assessments-indicators-chart.component.scss']\n})\nexport class ImpactAssessmentsIndicatorsChartComponent implements AfterViewInit, OnChanges {\n  @ViewChild('chart')\n  private chartRef?: ElementRef;\n  private chart: any;\n\n  @Input()\n  private impactAssessments: IImpactAssessmentJSONLD[] = [];\n  @Input()\n  private selected: string[] = [];\n  @Input()\n  private key: 'impacts' | 'emissionsResourceUse' = 'impacts';\n  @Input()\n  public filterTermTypes?: TermTermType[];\n\n  public indicatorPerImpactAssessment: IGroupedNodes<Indicator> = {};\n  public terms: Term[] = [];\n  public selectedTerm?: Term;\n\n  constructor(\n    private ngZone: NgZone\n  ) { }\n\n  ngAfterViewInit() {\n    return this.init();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('impactAssessments' in changes && !changes.impactAssessments.firstChange) {\n      return this.init();\n    }\n    if ('selected' in changes && !changes.selected.firstChange) {\n      return this.init();\n    }\n  }\n\n  private termAllowed(term?: Term) {\n    return !this.filterTermTypes?.length || (this.filterTermTypes || []).includes(term?.termType!);\n  }\n\n  private init() {\n    const impacts = this.impactAssessments.filter(v => isSelected(this.selected, v));\n    this.indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      impacts, this.key\n    );\n    this.terms = Object.values(this.indicatorPerImpactAssessment)\n      .filter(({ term, values }: any) =>\n        this.termAllowed(term) && Object.values(values).some(({ nodes: [{ value }] }: any) => value >= 0)\n      )\n      .map(({ term }: any) => term);\n    this.selectedTerm = this.terms.includes(this.selectedTerm!) ? this.selectedTerm : this.terms[0];\n    return this.selectedTerm ? this.updateChart() : null;\n  }\n\n  private initChart() {\n    const labels = [this.selectedTerm?.name ?? ''];\n    const { values } = this.indicatorPerImpactAssessment[this.selectedTerm!['@id']];\n    const datasets = this.impactAssessments\n      .map((impact, index) => isSelected(this.selected, impact) ? impactAssessmentDataset(values, impact, index) : null)\n      .filter(Boolean);\n\n    if (this.chart) {\n      this.chart.destroy();\n    }\n    this.chart = new Chart(this.chartRef?.nativeElement, {\n      type: 'bar',\n      data: {\n        labels,\n        datasets\n      },\n      options: {\n        responsive: true,\n        maintainAspectRatio: false,\n        legend: {\n          display: false\n        },\n        scales: {\n          xAxes: [{\n            display: false\n          }],\n          yAxes: [{\n            position: 'left',\n            ticks: {\n              min: 0\n            }\n          }]\n        }\n      }\n    });\n  }\n\n  public updateChart() {\n    this.ngZone.runOutsideAngular(() => this.initChart());\n  }\n}\n","<div class=\"p-3\" [class.is-hidden]=\"!terms?.length\">\n  <div class=\"field is-horizontal\">\n    <div class=\"field-label is-normal\">\n      <label class=\"label\" for=\"selectedTerm\">\n        <span>Select a column</span>\n      </label>\n    </div>\n    <div class=\"field-body\">\n      <div class=\"control\">\n        <div class=\"select is-small\">\n          <select (change)=\"updateChart()\" [(ngModel)]=\"selectedTerm\" id=\"selectedTerm\">\n            <option *ngFor=\"let term of terms\" [ngValue]=\"term\">{{term.name}}</option>\n          </select>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"mt-1\">\n    <div class=\"chart-container\">\n      <canvas #chart></canvas>\n    </div>\n  </div>\n</div>\n"]}
@@ -1,4 +1,4 @@
1
- import { Component, EventEmitter, Input, Output } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
2
2
  import { fileToExt, SupportedExtensions } from '@hestia-earth/api';
3
3
  import { toCsv } from '@hestia-earth/schema-convert';
4
4
  import * as i0 from "@angular/core";
@@ -41,13 +41,14 @@ export class NodeCsvExportConfirmComponent {
41
41
  }
42
42
  }
43
43
  NodeCsvExportConfirmComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: NodeCsvExportConfirmComponent, deps: [{ token: i1.DomSanitizer }], target: i0.ɵɵFactoryTarget.Component });
44
- NodeCsvExportConfirmComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: NodeCsvExportConfirmComponent, selector: "he-node-csv-export-confirm", inputs: { nodes: "nodes", filename: "filename", headerKeys: "headerKeys", extension: "extension", isUpload: "isUpload" }, outputs: { closed: "closed" }, ngImport: i0, template: "<div class=\"modal is-active\">\n <div class=\"modal-background\"></div>\n <div class=\"modal-card\">\n <header class=\"modal-card-head\">\n <p class=\"modal-card-title\">Export as CSV</p>\n <button class=\"delete\" aria-label=\"close\" (click)=\"closed.next(true)\"></button>\n </header>\n <section class=\"modal-card-body\">\n <div class=\"notification is-info\" role=\"alert\" *ngIf=\"isUpload\">\n <span>After Download, you can edit and</span>\n <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n\n <p>\n <span>Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and selecting the exported file.</span>\n </p>\n </div>\n\n <ng-container *ngIf=\"!isUpload && includedNodes.length > 1\">\n <p class=\"mb-2\">\n <b>{{includedNodes.length}}</b>\n <span class=\"px-1\">Nodes will be included in your download.</span>\n <a (click)=\"showIncludeNodes = !showIncludeNodes\">\n <ng-container *ngIf=\"!showIncludeNodes\">Show list</ng-container>\n <ng-container *ngIf=\"showIncludeNodes\">Hide list</ng-container>\n </a>\n </p>\n\n <div class=\"table-container\" *ngIf=\"showIncludeNodes\">\n <table class=\"table is-fullwidth is-hoverable mb-0\">\n <thead class=\"has-background-black\">\n <tr>\n <th>\n <span class=\"has-text-white\">@type</span>\n </th>\n <th>\n <span class=\"has-text-white\">@id</span>\n </th>\n <th>\n <span class=\"has-text-white\">Name</span>\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let node of includedNodes\">\n <td>\n <span *bindOnce=\"node\">{{node.node['@type']}}</span>\n </td>\n <td>\n <span *bindOnce=\"node\">{{node.node['@id']}}</span>\n </td>\n <td>\n <span *bindOnce=\"node\">{{node.node.name}}</span>\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n </ng-container>\n\n <he-node-csv-select-headers [class.is-hidden]=\"isUpload\"\n [csv]=\"csvData\" [keys]=\"headerKeys\" [includeDefaultCSV]=\"isUpload\"\n (headersChanged)=\"headersUpdated($event)\"\n ></he-node-csv-select-headers>\n </section>\n <footer class=\"modal-card-foot\">\n <a class=\"button is-success\" target=\"_blank\"\n [attr.disabled]=\"csvContent ? null : true\"\n [href]=\"csvContent\"\n [attr.download]=\"csvContent ? downloadFilename : null\"\n (click)=\"closed.next(true)\"\n >\n <fa-icon class=\"mr-2\" icon=\"download\"></fa-icon>\n <span>Download CSV</span>\n </a>\n <button class=\"button\" (click)=\"closed.next(true)\">\n <span>Cancel</span>\n </button>\n </footer>\n </div>\n</div>\n", styles: [""], components: [{ type: i2.NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: ["csv", "keys", "includeDefaultCSV"], outputs: ["headersChanged"] }, { type: i3.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.RouterLinkWithHref, selector: "a[routerLink],area[routerLink]", inputs: ["routerLink", "target", "queryParams", "fragment", "queryParamsHandling", "preserveFragment", "skipLocationChange", "replaceUrl", "state", "relativeTo"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i6.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }] });
44
+ NodeCsvExportConfirmComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: NodeCsvExportConfirmComponent, selector: "he-node-csv-export-confirm", inputs: { nodes: "nodes", filename: "filename", headerKeys: "headerKeys", extension: "extension", isUpload: "isUpload" }, outputs: { closed: "closed" }, ngImport: i0, template: "<div class=\"modal is-active\">\n <div class=\"modal-background\"></div>\n <div class=\"modal-card\">\n <header class=\"modal-card-head\">\n <p class=\"modal-card-title\">Export as CSV</p>\n <button class=\"delete\" aria-label=\"close\" (click)=\"closed.next(true)\"></button>\n </header>\n <section class=\"modal-card-body\">\n <div class=\"notification is-info\" role=\"alert\" *ngIf=\"isUpload\">\n <span>After Download, you can edit and</span>\n <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n\n <p>\n <span>Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and selecting the exported file.</span>\n </p>\n </div>\n\n <ng-container *ngIf=\"!isUpload && includedNodes.length > 1\">\n <p class=\"mb-2\">\n <b>{{includedNodes.length}}</b>\n <span class=\"px-1\">Nodes will be included in your download.</span>\n <a (click)=\"showIncludeNodes = !showIncludeNodes\">\n <ng-container *ngIf=\"!showIncludeNodes\">Show list</ng-container>\n <ng-container *ngIf=\"showIncludeNodes\">Hide list</ng-container>\n </a>\n </p>\n\n <div class=\"table-container\" *ngIf=\"showIncludeNodes\">\n <table class=\"table is-fullwidth is-hoverable mb-0\">\n <thead class=\"has-background-black\">\n <tr>\n <th>\n <span class=\"has-text-white\">@type</span>\n </th>\n <th>\n <span class=\"has-text-white\">@id</span>\n </th>\n <th>\n <span class=\"has-text-white\">Name</span>\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let node of includedNodes\">\n <td>\n <span *bindOnce=\"node\">{{node.node['@type']}}</span>\n </td>\n <td>\n <span *bindOnce=\"node\">{{node.node['@id']}}</span>\n </td>\n <td>\n <span *bindOnce=\"node\">{{node.node.name}}</span>\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n </ng-container>\n\n <he-node-csv-select-headers [class.is-hidden]=\"isUpload\"\n [csv]=\"csvData\" [keys]=\"headerKeys\" [includeDefaultCSV]=\"isUpload\"\n (headersChanged)=\"headersUpdated($event)\"\n ></he-node-csv-select-headers>\n </section>\n <footer class=\"modal-card-foot\">\n <a class=\"button is-success\" target=\"_blank\"\n [attr.disabled]=\"csvContent ? null : true\"\n [href]=\"csvContent\"\n [attr.download]=\"csvContent ? downloadFilename : null\"\n (click)=\"closed.next(true)\"\n >\n <fa-icon class=\"mr-2\" icon=\"download\"></fa-icon>\n <span>Download CSV</span>\n </a>\n <button class=\"button\" (click)=\"closed.next(true)\">\n <span>Cancel</span>\n </button>\n </footer>\n </div>\n</div>\n", styles: [""], components: [{ type: i2.NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: ["csv", "keys", "includeDefaultCSV"], outputs: ["headersChanged"] }, { type: i3.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i5.RouterLinkWithHref, selector: "a[routerLink],area[routerLink]", inputs: ["routerLink", "target", "queryParams", "fragment", "queryParamsHandling", "preserveFragment", "skipLocationChange", "replaceUrl", "state", "relativeTo"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i6.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
45
45
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: NodeCsvExportConfirmComponent, decorators: [{
46
46
  type: Component,
47
47
  args: [{
48
48
  selector: 'he-node-csv-export-confirm',
49
49
  templateUrl: './node-csv-export-confirm.component.html',
50
- styleUrls: ['./node-csv-export-confirm.component.scss']
50
+ styleUrls: ['./node-csv-export-confirm.component.scss'],
51
+ changeDetection: ChangeDetectionStrategy.OnPush
51
52
  }]
52
53
  }], ctorParameters: function () { return [{ type: i1.DomSanitizer }]; }, propDecorators: { nodes: [{
53
54
  type: Input
@@ -62,4 +63,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
62
63
  }], closed: [{
63
64
  type: Output
64
65
  }] } });
65
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-export-confirm.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.ts","../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAE/E,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEnE,OAAO,EAAE,KAAK,EAAE,MAAM,8BAA8B,CAAC;;;;;;;;AAYrD,MAAM,OAAO,6BAA6B;IAsBxC,YACU,YAA0B;QAA1B,iBAAY,GAAZ,YAAY,CAAc;QAtB5B,YAAO,GAAa,EAAE,CAAC;QAGvB,UAAK,GAAsB,EAAE,CAAC;QAE9B,aAAQ,GAAG,UAAU,CAAC;QAIvB,cAAS,GAAG,mBAAmB,CAAC,GAAG,CAAC;QAEpC,aAAQ,GAAG,IAAI,CAAC;QAGhB,WAAM,GAAG,IAAI,YAAY,EAAW,CAAC;QAIrC,qBAAgB,GAAG,KAAK,CAAC;QACzB,kBAAa,GAAoB,EAAE,CAAC;IAIvC,CAAC;IAEL,QAAQ;QACN,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;QACxE,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACzD,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC;IAC/D,CAAC;IAEM,cAAc,CAAC,OAAiB;QACrC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,OAAO,IAAI,CAAC,gBAAgB,EAAE,CAAC;IACjC,CAAC;IAEM,gBAAgB;QACrB,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC;QACpD,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAC/D,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,8BAA8B,CACtF,gCAAgC,kBAAkB,CAAC,IAAI,CAAC,EAAE,CAC3D,CAAC,CAAC,CAAC,SAAS,CAAC;IAChB,CAAC;IAED,IAAW,gBAAgB;QACzB,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;IAClD,CAAC;;2HAlDU,6BAA6B;+GAA7B,6BAA6B,2NChB1C,2qGAiFA;4FDjEa,6BAA6B;kBALzC,SAAS;mBAAC;oBACT,QAAQ,EAAE,4BAA4B;oBACtC,WAAW,EAAE,0CAA0C;oBACvD,SAAS,EAAE,CAAC,0CAA0C,CAAC;iBACxD;mGAKS,KAAK;sBADZ,KAAK;gBAGE,QAAQ;sBADf,KAAK;gBAGC,UAAU;sBADhB,KAAK;gBAGC,SAAS;sBADf,KAAK;gBAGC,QAAQ;sBADd,KAAK;gBAIC,MAAM;sBADZ,MAAM","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { fileToExt, SupportedExtensions } from '@hestia-earth/api';\nimport { JSON as HestiaJson } from '@hestia-earth/schema';\nimport { toCsv } from '@hestia-earth/schema-convert';\n\ninterface INodeIncluded {\n  node: HestiaJson<any>;\n  included: boolean;\n}\n\n@Component({\n  selector: 'he-node-csv-export-confirm',\n  templateUrl: './node-csv-export-confirm.component.html',\n  styleUrls: ['./node-csv-export-confirm.component.scss']\n})\nexport class NodeCsvExportConfirmComponent implements OnInit {\n  private headers: string[] = [];\n\n  @Input()\n  private nodes: HestiaJson<any>[] = [];\n  @Input()\n  private filename = 'download';\n  @Input()\n  public headerKeys?: string[];\n  @Input()\n  public extension = SupportedExtensions.csv;\n  @Input()\n  public isUpload = true;\n\n  @Output()\n  public closed = new EventEmitter<boolean>();\n\n  public csvData?: string;\n  public csvContent?: SafeResourceUrl;\n  public showIncludeNodes = false;\n  public includedNodes: INodeIncluded[] = [];\n\n  constructor(\n    private domSanitizer: DomSanitizer\n  ) { }\n\n  ngOnInit() {\n    this.includedNodes = this.nodes.map(node => ({ node, included: true }));\n    this.csvData = toCsv(this.nodes || [], !this.isUpload);\n  }\n\n  public get included() {\n    return this.includedNodes.filter(({ included }) => included);\n  }\n\n  public headersUpdated(headers: string[]) {\n    this.headers = headers;\n    return this.updateCsvContent();\n  }\n\n  public updateCsvContent() {\n    const nodes = this.included.map(({ node }) => node);\n    const data = toCsv(nodes, !this.isUpload, false, this.headers);\n    this.csvContent = this.headers.length ? this.domSanitizer.bypassSecurityTrustResourceUrl(\n      `data:text/html;charset=utf-8,${encodeURIComponent(data)}`\n    ) : undefined;\n  }\n\n  public get downloadFilename() {\n    return fileToExt(this.filename, this.extension);\n  }\n}\n","<div class=\"modal is-active\">\n  <div class=\"modal-background\"></div>\n  <div class=\"modal-card\">\n    <header class=\"modal-card-head\">\n      <p class=\"modal-card-title\">Export as CSV</p>\n      <button class=\"delete\" aria-label=\"close\" (click)=\"closed.next(true)\"></button>\n    </header>\n    <section class=\"modal-card-body\">\n      <div class=\"notification is-info\" role=\"alert\" *ngIf=\"isUpload\">\n        <span>After Download, you can edit and</span>\n        <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n        <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n\n        <p>\n          <span>Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and selecting the exported file.</span>\n        </p>\n      </div>\n\n      <ng-container *ngIf=\"!isUpload && includedNodes.length > 1\">\n        <p class=\"mb-2\">\n          <b>{{includedNodes.length}}</b>\n          <span class=\"px-1\">Nodes will be included in your download.</span>\n          <a (click)=\"showIncludeNodes = !showIncludeNodes\">\n            <ng-container *ngIf=\"!showIncludeNodes\">Show list</ng-container>\n            <ng-container *ngIf=\"showIncludeNodes\">Hide list</ng-container>\n          </a>\n        </p>\n\n        <div class=\"table-container\" *ngIf=\"showIncludeNodes\">\n          <table class=\"table is-fullwidth is-hoverable mb-0\">\n            <thead class=\"has-background-black\">\n              <tr>\n                <th>\n                  <span class=\"has-text-white\">@type</span>\n                </th>\n                <th>\n                  <span class=\"has-text-white\">@id</span>\n                </th>\n                <th>\n                  <span class=\"has-text-white\">Name</span>\n                </th>\n              </tr>\n            </thead>\n            <tbody>\n              <tr *ngFor=\"let node of includedNodes\">\n                <td>\n                  <span *bindOnce=\"node\">{{node.node['@type']}}</span>\n                </td>\n                <td>\n                  <span *bindOnce=\"node\">{{node.node['@id']}}</span>\n                </td>\n                <td>\n                  <span *bindOnce=\"node\">{{node.node.name}}</span>\n                </td>\n              </tr>\n            </tbody>\n          </table>\n        </div>\n      </ng-container>\n\n      <he-node-csv-select-headers [class.is-hidden]=\"isUpload\"\n        [csv]=\"csvData\" [keys]=\"headerKeys\" [includeDefaultCSV]=\"isUpload\"\n        (headersChanged)=\"headersUpdated($event)\"\n      ></he-node-csv-select-headers>\n    </section>\n    <footer class=\"modal-card-foot\">\n      <a class=\"button is-success\" target=\"_blank\"\n        [attr.disabled]=\"csvContent ? null : true\"\n        [href]=\"csvContent\"\n        [attr.download]=\"csvContent ? downloadFilename : null\"\n        (click)=\"closed.next(true)\"\n      >\n        <fa-icon class=\"mr-2\" icon=\"download\"></fa-icon>\n        <span>Download CSV</span>\n      </a>\n      <button class=\"button\" (click)=\"closed.next(true)\">\n        <span>Cancel</span>\n      </button>\n    </footer>\n  </div>\n</div>\n"]}
66
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-export-confirm.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.ts","../../../../src/node/node-csv-export-confirm/node-csv-export-confirm.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAExG,OAAO,EAAE,SAAS,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEnE,OAAO,EAAE,KAAK,EAAE,MAAM,8BAA8B,CAAC;;;;;;;;AAarD,MAAM,OAAO,6BAA6B;IAsBxC,YACU,YAA0B;QAA1B,iBAAY,GAAZ,YAAY,CAAc;QAtB5B,YAAO,GAAa,EAAE,CAAC;QAGvB,UAAK,GAAsB,EAAE,CAAC;QAE9B,aAAQ,GAAG,UAAU,CAAC;QAIvB,cAAS,GAAG,mBAAmB,CAAC,GAAG,CAAC;QAEpC,aAAQ,GAAG,IAAI,CAAC;QAGhB,WAAM,GAAG,IAAI,YAAY,EAAW,CAAC;QAIrC,qBAAgB,GAAG,KAAK,CAAC;QACzB,kBAAa,GAAoB,EAAE,CAAC;IAIvC,CAAC;IAEL,QAAQ;QACN,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;QACxE,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACzD,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC;IAC/D,CAAC;IAEM,cAAc,CAAC,OAAiB;QACrC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,OAAO,IAAI,CAAC,gBAAgB,EAAE,CAAC;IACjC,CAAC;IAEM,gBAAgB;QACrB,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC;QACpD,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAC/D,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,8BAA8B,CACtF,gCAAgC,kBAAkB,CAAC,IAAI,CAAC,EAAE,CAC3D,CAAC,CAAC,CAAC,SAAS,CAAC;IAChB,CAAC;IAED,IAAW,gBAAgB;QACzB,OAAO,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;IAClD,CAAC;;2HAlDU,6BAA6B;+GAA7B,6BAA6B,2NCjB1C,2qGAiFA;4FDhEa,6BAA6B;kBANzC,SAAS;mBAAC;oBACT,QAAQ,EAAE,4BAA4B;oBACtC,WAAW,EAAE,0CAA0C;oBACvD,SAAS,EAAE,CAAC,0CAA0C,CAAC;oBACvD,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD;mGAKS,KAAK;sBADZ,KAAK;gBAGE,QAAQ;sBADf,KAAK;gBAGC,UAAU;sBADhB,KAAK;gBAGC,SAAS;sBADf,KAAK;gBAGC,QAAQ;sBADd,KAAK;gBAIC,MAAM;sBADZ,MAAM","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { fileToExt, SupportedExtensions } from '@hestia-earth/api';\nimport { JSON as HestiaJson } from '@hestia-earth/schema';\nimport { toCsv } from '@hestia-earth/schema-convert';\n\ninterface INodeIncluded {\n  node: HestiaJson<any>;\n  included: boolean;\n}\n\n@Component({\n  selector: 'he-node-csv-export-confirm',\n  templateUrl: './node-csv-export-confirm.component.html',\n  styleUrls: ['./node-csv-export-confirm.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class NodeCsvExportConfirmComponent implements OnInit {\n  private headers: string[] = [];\n\n  @Input()\n  private nodes: HestiaJson<any>[] = [];\n  @Input()\n  private filename = 'download';\n  @Input()\n  public headerKeys?: string[];\n  @Input()\n  public extension = SupportedExtensions.csv;\n  @Input()\n  public isUpload = true;\n\n  @Output()\n  public closed = new EventEmitter<boolean>();\n\n  public csvData?: string;\n  public csvContent?: SafeResourceUrl;\n  public showIncludeNodes = false;\n  public includedNodes: INodeIncluded[] = [];\n\n  constructor(\n    private domSanitizer: DomSanitizer\n  ) { }\n\n  ngOnInit() {\n    this.includedNodes = this.nodes.map(node => ({ node, included: true }));\n    this.csvData = toCsv(this.nodes || [], !this.isUpload);\n  }\n\n  public get included() {\n    return this.includedNodes.filter(({ included }) => included);\n  }\n\n  public headersUpdated(headers: string[]) {\n    this.headers = headers;\n    return this.updateCsvContent();\n  }\n\n  public updateCsvContent() {\n    const nodes = this.included.map(({ node }) => node);\n    const data = toCsv(nodes, !this.isUpload, false, this.headers);\n    this.csvContent = this.headers.length ? this.domSanitizer.bypassSecurityTrustResourceUrl(\n      `data:text/html;charset=utf-8,${encodeURIComponent(data)}`\n    ) : undefined;\n  }\n\n  public get downloadFilename() {\n    return fileToExt(this.filename, this.extension);\n  }\n}\n","<div class=\"modal is-active\">\n  <div class=\"modal-background\"></div>\n  <div class=\"modal-card\">\n    <header class=\"modal-card-head\">\n      <p class=\"modal-card-title\">Export as CSV</p>\n      <button class=\"delete\" aria-label=\"close\" (click)=\"closed.next(true)\"></button>\n    </header>\n    <section class=\"modal-card-body\">\n      <div class=\"notification is-info\" role=\"alert\" *ngIf=\"isUpload\">\n        <span>After Download, you can edit and</span>\n        <a class=\"px-1\" routerLink=\"../\">upload the CSV file</a>\n        <span>to submit your content on the Hestia platform, and your draft will remain unchanged.</span>\n\n        <p>\n          <span>Alternatively, you can import the CSV file right back by clicking on the \"Import from CSV\" button and selecting the exported file.</span>\n        </p>\n      </div>\n\n      <ng-container *ngIf=\"!isUpload && includedNodes.length > 1\">\n        <p class=\"mb-2\">\n          <b>{{includedNodes.length}}</b>\n          <span class=\"px-1\">Nodes will be included in your download.</span>\n          <a (click)=\"showIncludeNodes = !showIncludeNodes\">\n            <ng-container *ngIf=\"!showIncludeNodes\">Show list</ng-container>\n            <ng-container *ngIf=\"showIncludeNodes\">Hide list</ng-container>\n          </a>\n        </p>\n\n        <div class=\"table-container\" *ngIf=\"showIncludeNodes\">\n          <table class=\"table is-fullwidth is-hoverable mb-0\">\n            <thead class=\"has-background-black\">\n              <tr>\n                <th>\n                  <span class=\"has-text-white\">@type</span>\n                </th>\n                <th>\n                  <span class=\"has-text-white\">@id</span>\n                </th>\n                <th>\n                  <span class=\"has-text-white\">Name</span>\n                </th>\n              </tr>\n            </thead>\n            <tbody>\n              <tr *ngFor=\"let node of includedNodes\">\n                <td>\n                  <span *bindOnce=\"node\">{{node.node['@type']}}</span>\n                </td>\n                <td>\n                  <span *bindOnce=\"node\">{{node.node['@id']}}</span>\n                </td>\n                <td>\n                  <span *bindOnce=\"node\">{{node.node.name}}</span>\n                </td>\n              </tr>\n            </tbody>\n          </table>\n        </div>\n      </ng-container>\n\n      <he-node-csv-select-headers [class.is-hidden]=\"isUpload\"\n        [csv]=\"csvData\" [keys]=\"headerKeys\" [includeDefaultCSV]=\"isUpload\"\n        (headersChanged)=\"headersUpdated($event)\"\n      ></he-node-csv-select-headers>\n    </section>\n    <footer class=\"modal-card-foot\">\n      <a class=\"button is-success\" target=\"_blank\"\n        [attr.disabled]=\"csvContent ? null : true\"\n        [href]=\"csvContent\"\n        [attr.download]=\"csvContent ? downloadFilename : null\"\n        (click)=\"closed.next(true)\"\n      >\n        <fa-icon class=\"mr-2\" icon=\"download\"></fa-icon>\n        <span>Download CSV</span>\n      </a>\n      <button class=\"button\" (click)=\"closed.next(true)\">\n        <span>Cancel</span>\n      </button>\n    </footer>\n  </div>\n</div>\n"]}
@@ -1,5 +1,5 @@
1
1
  import { __awaiter } from "tslib";
2
- import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
3
3
  import { moveItemInArray } from '@angular/cdk/drag-drop';
4
4
  import { isDefaultCSVSelected, isCSVIncluded } from '@hestia-earth/json-schema/schema-utils';
5
5
  import * as i0 from "@angular/core";
@@ -104,13 +104,14 @@ export class NodeCsvSelectHeadersComponent {
104
104
  }
105
105
  }
106
106
  NodeCsvSelectHeadersComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: NodeCsvSelectHeadersComponent, deps: [{ token: i1.HeSchemaService }], target: i0.ɵɵFactoryTarget.Component });
107
- NodeCsvSelectHeadersComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: { csv: "csv", keys: "keys", includeDefaultCSV: "includeDefaultCSV" }, outputs: { headersChanged: "headersChanged" }, ngImport: i0, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\"><i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i></p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"allSelected\"\n (change)=\"toggleAll(); refresh()\"\n >\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div class=\"dropdown is-right has-text-left\"\n [class.is-active]=\"showSelectTermFields\"\n (clickOutside)=\"showSelectTermFields = false\"\n >\n <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n <button class=\"button is-small\"\n aria-haspopup=\"true\" aria-controls=\"select-menu\"\n >\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n <div class=\"dropdown-content\">\n <a class=\"dropdown-item\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"showNonIncluded\"\n (change)=\"refresh()\"\n >\n <span class=\"ml-2\">Include <code>internal</code> fields</span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n <a class=\"dropdown-item\"\n *ngFor=\"let field of termFields\"\n >\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"termFieldSelected[field]\"\n (change)=\"toggleTermField(field)\"\n >\n <span class=\"ml-2\">{{field}}</span>\n </label>\n </a>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n <div class=\"card\" *ngFor=\"let group of headers | keys\">\n <header class=\"card-header\" *ngIf=\"group.key\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [indeterminate]=\"group.value.partialSelected\"\n [(ngModel)]=\"group.value.selected\"\n (change)=\"updateGroup(group.key); refresh()\"\n >\n <span class=\"ml-2\">{{group.key}}</span>\n </label>\n </div>\n <span class=\"card-header-icon has-text-link\" aria-label=\"open / close\"\n (click)=\"group.value.open = !group.value.open\" pointer\n >\n <span class=\"icon\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n </span>\n </span>\n </header>\n\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n <ng-container *ngFor=\"let header of group.value.headers\">\n <ng-container *ngIf=\"header.included || showNonIncluded\">\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"header.selected\"\n (change)=\"updateGroupHeader(group.key); refresh()\"\n >\n <span class=\"ml-2\">{{header.header}}</span>\n </label>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n </div>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [".toggle-all{border-bottom:1px solid #7a7a7a}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:2px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"], components: [{ type: i2.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i3.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i4.ClickOutsideDirective, selector: "[clickOutside]", outputs: ["clickOutside"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "id", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListAutoScrollDisabled", "cdkDropListOrientation", "cdkDropListLockAxis", "cdkDropListData", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { type: i6.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragDisabled", "cdkDragStartDelay", "cdkDragLockAxis", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragBoundary", "cdkDragRootElement", "cdkDragPreviewContainer", "cdkDragData", "cdkDragFreeDragPosition"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }], pipes: { "keys": i7.KeysPipe } });
107
+ NodeCsvSelectHeadersComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: NodeCsvSelectHeadersComponent, selector: "he-node-csv-select-headers", inputs: { csv: "csv", keys: "keys", includeDefaultCSV: "includeDefaultCSV" }, outputs: { headersChanged: "headersChanged" }, ngImport: i0, template: "<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\"><i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i></p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n <div class=\"column\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"allSelected\"\n (change)=\"toggleAll(); refresh()\"\n >\n <span class=\"ml-2\">Toggle All</span>\n </label>\n </div>\n\n <div class=\"column is-narrow has-text-right\">\n <div class=\"dropdown is-right has-text-left\"\n [class.is-active]=\"showSelectTermFields\"\n (clickOutside)=\"showSelectTermFields = false\"\n >\n <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n <button class=\"button is-small\"\n aria-haspopup=\"true\" aria-controls=\"select-menu\"\n >\n <span>Advanced Filters</span>\n <span class=\"icon is-small\">\n <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n </span>\n </button>\n </div>\n <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n <div class=\"dropdown-content\">\n <a class=\"dropdown-item\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"showNonIncluded\"\n (change)=\"refresh()\"\n >\n <span class=\"ml-2\">Include <code>internal</code> fields</span>\n </label>\n </a>\n <div class=\"dropdown-item\">\n <p>Toggle Term Fields</p>\n </div>\n <a class=\"dropdown-item\"\n *ngFor=\"let field of termFields\"\n >\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"termFieldSelected[field]\"\n (change)=\"toggleTermField(field)\"\n >\n <span class=\"ml-2\">{{field}}</span>\n </label>\n </a>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n <div class=\"card\" *ngFor=\"let group of headers | keys\">\n <header class=\"card-header\" *ngIf=\"group.key\">\n <div class=\"card-header-title\">\n <label class=\"checkbox ml-1\">\n <input type=\"checkbox\" class=\"selector\"\n [indeterminate]=\"group.value.partialSelected\"\n [(ngModel)]=\"group.value.selected\"\n (change)=\"updateGroup(group.key); refresh()\"\n >\n <span class=\"ml-2\">{{group.key}}</span>\n </label>\n </div>\n <span class=\"card-header-icon has-text-link\" aria-label=\"open / close\"\n (click)=\"group.value.open = !group.value.open\" pointer\n >\n <span class=\"icon\">\n <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n </span>\n </span>\n </header>\n\n <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n <ng-container *ngFor=\"let header of group.value.headers\">\n <ng-container *ngIf=\"header.included || showNonIncluded\">\n <li cdkDrag>\n <label class=\"checkbox ml-2\">\n <input type=\"checkbox\" class=\"selector\"\n [(ngModel)]=\"header.selected\"\n (change)=\"updateGroupHeader(group.key); refresh()\"\n >\n <span class=\"ml-2\">{{header.header}}</span>\n </label>\n </li>\n </ng-container>\n </ng-container>\n </ul>\n </div>\n </div>\n</div>\n\n<ng-template #loader>\n <div class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n </div>\n</ng-template>\n", styles: [".toggle-all{border-bottom:1px solid #7a7a7a}.drag-container,.table-container{max-height:45vh;overflow-y:auto}.cdk-drag-preview{border-radius:2px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f;list-style:none}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}\n"], components: [{ type: i2.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i3.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i4.ClickOutsideDirective, selector: "[clickOutside]", outputs: ["clickOutside"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "id", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListAutoScrollDisabled", "cdkDropListOrientation", "cdkDropListLockAxis", "cdkDropListData", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { type: i6.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragDisabled", "cdkDragStartDelay", "cdkDragLockAxis", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragBoundary", "cdkDragRootElement", "cdkDragPreviewContainer", "cdkDragData", "cdkDragFreeDragPosition"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }], pipes: { "keys": i7.KeysPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
108
108
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: NodeCsvSelectHeadersComponent, decorators: [{
109
109
  type: Component,
110
110
  args: [{
111
111
  selector: 'he-node-csv-select-headers',
112
112
  templateUrl: './node-csv-select-headers.component.html',
113
- styleUrls: ['./node-csv-select-headers.component.scss']
113
+ styleUrls: ['./node-csv-select-headers.component.scss'],
114
+ changeDetection: ChangeDetectionStrategy.OnPush
114
115
  }]
115
116
  }], ctorParameters: function () { return [{ type: i1.HeSchemaService }]; }, propDecorators: { csv: [{
116
117
  type: Input
@@ -121,4 +122,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
121
122
  }], headersChanged: [{
122
123
  type: Output
123
124
  }] } });
124
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-select-headers.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.ts","../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';\nimport { definitions } from '@hestia-earth/json-schema';\nimport { isDefaultCSVSelected, isCSVIncluded } from '@hestia-earth/json-schema/schema-utils';\n\nimport { HeSchemaService } from '../../schema/schema.service';\n\ninterface IHeader {\n  header: string;\n  selected: boolean;\n  /**\n   * `false` if it is an `internal` field.\n   */\n  included: boolean;\n}\n\ninterface IGroupedHeaders {\n  [group: string]: {\n    selected: boolean;\n    partialSelected: boolean;\n    open: boolean;\n    headers: IHeader[];\n  };\n}\n\nconst headerGroup = (header: string) => {\n  const parts = header.split('.');\n  return parts.length === 2 ? '' : [parts[0], parts[1]].join('.');\n};\n\nconst termFields = [\n  '@id', 'name', 'units', 'termType'\n];\n\nconst isTermField = (field: string) => ({ header }: IHeader) => [\n  'term', 'methodModel', 'country', 'product', 'operation', 'key'\n].some(parent => header.endsWith([parent, field].join('.')));\n\n@Component({\n  selector: 'he-node-csv-select-headers',\n  templateUrl: './node-csv-select-headers.component.html',\n  styleUrls: ['./node-csv-select-headers.component.scss']\n})\nexport class NodeCsvSelectHeadersComponent implements OnInit {\n  @Input()\n  private csv = '';\n  @Input()\n  private keys?: string[];\n  @Input()\n  private includeDefaultCSV = false;\n\n  @Output()\n  private headersChanged = new EventEmitter<string[]>();\n\n  public loading = true;\n  public headers: IGroupedHeaders = {};\n  public allSelected = false;\n\n  public showSelectTermFields = false;\n  public termFields = termFields;\n  public termFieldSelected: any = {};\n  public showNonIncluded = false;\n\n  constructor(\n    private schemaService: HeSchemaService\n  ) {}\n\n  async ngOnInit() {\n    const schemas = await this.schemaService.schemas();\n    const isSelected = await this.isSelected(schemas);\n    const isIncluded = isCSVIncluded(schemas);\n    const headers = await this.schemaService.parseHeaders(this.csv);\n    this.headers = headers.reduce((prev, header) => {\n      const group = headerGroup(header);\n      prev[group] = prev[group] || { headers: [], selected: false, partialSelected: false, open: true };\n      prev[group].headers.push({ header, selected: isSelected(header), included: isIncluded(header) });\n      return prev;\n    }, {} as IGroupedHeaders);\n    Object.keys(this.headers).map(groupKey => this.updateGroupHeader(groupKey));\n    this.termFieldSelected = termFields.reduce((prev, curr) => ({ ...prev, [curr]: curr === '@id' }), {});\n    this.refresh();\n    this.loading = false;\n  }\n\n  private async isSelected(schemas: definitions) {\n    const isSelected = this.includeDefaultCSV ? () => true : isDefaultCSVSelected(schemas);\n    return (key: string) => (!this.keys?.length || this.keys.some(v => key.startsWith(v))) && isSelected(key);\n  }\n\n  public dropHeader(event: CdkDragDrop<IHeader[]>, groupKey: string) {\n    moveItemInArray(this.headers[groupKey].headers, event.previousIndex, event.currentIndex);\n    this.headersChanged.emit(this.selectedHeaders);\n  }\n\n  private updateAllSelected() {\n    this.allSelected = Object.keys(this.headers).every(key => this.headers[key].selected);\n  }\n\n  public toggleAll() {\n    // select all term fields\n    Object.keys(this.termFieldSelected).map(key => (this.termFieldSelected[key] = true));\n    // select all keys\n    return Object.entries(this.headers).map(([key, group]) => {\n      group.selected = this.allSelected;\n      return this.updateGroup(key);\n    });\n  }\n\n  public updateGroup(groupKey: string) {\n    const group = this.headers[groupKey];\n    group.headers.map(header => header.selected = group.selected);\n  }\n\n  public updateGroupHeader(groupKey: string) {\n    const group = this.headers[groupKey];\n    const allSelected = group.headers.every(({ selected }) => selected);\n    group.selected = allSelected;\n    group.partialSelected = !allSelected && group.headers.some(({ selected }) => selected);\n  }\n\n  public get selectedHeaders() {\n    return Object.values(this.headers)\n      .flatMap(({ headers }) => headers)\n      .filter(({ selected, included }) => selected && (included || this.showNonIncluded))\n      .map(({ header }) => header);\n  }\n\n  public refresh() {\n    this.updateAllSelected();\n    this.headersChanged.emit(this.selectedHeaders);\n  }\n\n  // Term fields\n\n  public toggleTermField(field: string) {\n    const selected = this.termFieldSelected[field];\n    return Object.entries(this.headers).map(([key, group]) => {\n      group.headers.filter(isTermField(field)).map(header => header.selected = selected);\n      return this.updateGroupHeader(key);\n    });\n  }\n}\n","<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\"><i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i></p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n  <div class=\"column\">\n    <label class=\"checkbox ml-1\">\n      <input type=\"checkbox\" class=\"selector\"\n        [(ngModel)]=\"allSelected\"\n        (change)=\"toggleAll(); refresh()\"\n      >\n      <span class=\"ml-2\">Toggle All</span>\n    </label>\n  </div>\n\n  <div class=\"column is-narrow has-text-right\">\n    <div class=\"dropdown is-right has-text-left\"\n      [class.is-active]=\"showSelectTermFields\"\n      (clickOutside)=\"showSelectTermFields = false\"\n    >\n      <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n        <button class=\"button is-small\"\n          aria-haspopup=\"true\" aria-controls=\"select-menu\"\n        >\n          <span>Advanced Filters</span>\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n          </span>\n        </button>\n      </div>\n      <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n        <div class=\"dropdown-content\">\n          <a class=\"dropdown-item\">\n            <label class=\"checkbox ml-1\">\n              <input type=\"checkbox\" class=\"selector\"\n                [(ngModel)]=\"showNonIncluded\"\n                (change)=\"refresh()\"\n              >\n              <span class=\"ml-2\">Include <code>internal</code> fields</span>\n            </label>\n          </a>\n          <div class=\"dropdown-item\">\n            <p>Toggle Term Fields</p>\n          </div>\n          <a class=\"dropdown-item\"\n            *ngFor=\"let field of termFields\"\n          >\n            <label class=\"checkbox ml-1\">\n              <input type=\"checkbox\" class=\"selector\"\n                [(ngModel)]=\"termFieldSelected[field]\"\n                (change)=\"toggleTermField(field)\"\n              >\n              <span class=\"ml-2\">{{field}}</span>\n            </label>\n          </a>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n  <div class=\"card\" *ngFor=\"let group of headers | keys\">\n    <header class=\"card-header\" *ngIf=\"group.key\">\n      <div class=\"card-header-title\">\n        <label class=\"checkbox ml-1\">\n          <input type=\"checkbox\" class=\"selector\"\n            [indeterminate]=\"group.value.partialSelected\"\n            [(ngModel)]=\"group.value.selected\"\n            (change)=\"updateGroup(group.key); refresh()\"\n          >\n          <span class=\"ml-2\">{{group.key}}</span>\n        </label>\n      </div>\n      <span class=\"card-header-icon has-text-link\" aria-label=\"open / close\"\n        (click)=\"group.value.open = !group.value.open\" pointer\n      >\n        <span class=\"icon\">\n          <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n          <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n        </span>\n      </span>\n    </header>\n\n    <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n      <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n        <ng-container *ngFor=\"let header of group.value.headers\">\n          <ng-container *ngIf=\"header.included || showNonIncluded\">\n            <li cdkDrag>\n              <label class=\"checkbox ml-2\">\n                <input type=\"checkbox\" class=\"selector\"\n                  [(ngModel)]=\"header.selected\"\n                  (change)=\"updateGroupHeader(group.key); refresh()\"\n                >\n                <span class=\"ml-2\">{{header.header}}</span>\n              </label>\n            </li>\n          </ng-container>\n        </ng-container>\n      </ul>\n    </div>\n  </div>\n</div>\n\n<ng-template #loader>\n  <div class=\"has-text-center py-3\">\n    <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n</ng-template>\n"]}
125
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-csv-select-headers.component.js","sourceRoot":"","sources":["../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.ts","../../../../src/node/node-csv-select-headers/node-csv-select-headers.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';\nimport { definitions } from '@hestia-earth/json-schema';\nimport { isDefaultCSVSelected, isCSVIncluded } from '@hestia-earth/json-schema/schema-utils';\n\nimport { HeSchemaService } from '../../schema/schema.service';\n\ninterface IHeader {\n  header: string;\n  selected: boolean;\n  /**\n   * `false` if it is an `internal` field.\n   */\n  included: boolean;\n}\n\ninterface IGroupedHeaders {\n  [group: string]: {\n    selected: boolean;\n    partialSelected: boolean;\n    open: boolean;\n    headers: IHeader[];\n  };\n}\n\nconst headerGroup = (header: string) => {\n  const parts = header.split('.');\n  return parts.length === 2 ? '' : [parts[0], parts[1]].join('.');\n};\n\nconst termFields = [\n  '@id', 'name', 'units', 'termType'\n];\n\nconst isTermField = (field: string) => ({ header }: IHeader) => [\n  'term', 'methodModel', 'country', 'product', 'operation', 'key'\n].some(parent => header.endsWith([parent, field].join('.')));\n\n@Component({\n  selector: 'he-node-csv-select-headers',\n  templateUrl: './node-csv-select-headers.component.html',\n  styleUrls: ['./node-csv-select-headers.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class NodeCsvSelectHeadersComponent implements OnInit {\n  @Input()\n  private csv = '';\n  @Input()\n  private keys?: string[];\n  @Input()\n  private includeDefaultCSV = false;\n\n  @Output()\n  private headersChanged = new EventEmitter<string[]>();\n\n  public loading = true;\n  public headers: IGroupedHeaders = {};\n  public allSelected = false;\n\n  public showSelectTermFields = false;\n  public termFields = termFields;\n  public termFieldSelected: any = {};\n  public showNonIncluded = false;\n\n  constructor(\n    private schemaService: HeSchemaService\n  ) {}\n\n  async ngOnInit() {\n    const schemas = await this.schemaService.schemas();\n    const isSelected = await this.isSelected(schemas);\n    const isIncluded = isCSVIncluded(schemas);\n    const headers = await this.schemaService.parseHeaders(this.csv);\n    this.headers = headers.reduce((prev, header) => {\n      const group = headerGroup(header);\n      prev[group] = prev[group] || { headers: [], selected: false, partialSelected: false, open: true };\n      prev[group].headers.push({ header, selected: isSelected(header), included: isIncluded(header) });\n      return prev;\n    }, {} as IGroupedHeaders);\n    Object.keys(this.headers).map(groupKey => this.updateGroupHeader(groupKey));\n    this.termFieldSelected = termFields.reduce((prev, curr) => ({ ...prev, [curr]: curr === '@id' }), {});\n    this.refresh();\n    this.loading = false;\n  }\n\n  private async isSelected(schemas: definitions) {\n    const isSelected = this.includeDefaultCSV ? () => true : isDefaultCSVSelected(schemas);\n    return (key: string) => (!this.keys?.length || this.keys.some(v => key.startsWith(v))) && isSelected(key);\n  }\n\n  public dropHeader(event: CdkDragDrop<IHeader[]>, groupKey: string) {\n    moveItemInArray(this.headers[groupKey].headers, event.previousIndex, event.currentIndex);\n    this.headersChanged.emit(this.selectedHeaders);\n  }\n\n  private updateAllSelected() {\n    this.allSelected = Object.keys(this.headers).every(key => this.headers[key].selected);\n  }\n\n  public toggleAll() {\n    // select all term fields\n    Object.keys(this.termFieldSelected).map(key => (this.termFieldSelected[key] = true));\n    // select all keys\n    return Object.entries(this.headers).map(([key, group]) => {\n      group.selected = this.allSelected;\n      return this.updateGroup(key);\n    });\n  }\n\n  public updateGroup(groupKey: string) {\n    const group = this.headers[groupKey];\n    group.headers.map(header => header.selected = group.selected);\n  }\n\n  public updateGroupHeader(groupKey: string) {\n    const group = this.headers[groupKey];\n    const allSelected = group.headers.every(({ selected }) => selected);\n    group.selected = allSelected;\n    group.partialSelected = !allSelected && group.headers.some(({ selected }) => selected);\n  }\n\n  public get selectedHeaders() {\n    return Object.values(this.headers)\n      .flatMap(({ headers }) => headers)\n      .filter(({ selected, included }) => selected && (included || this.showNonIncluded))\n      .map(({ header }) => header);\n  }\n\n  public refresh() {\n    this.updateAllSelected();\n    this.headersChanged.emit(this.selectedHeaders);\n  }\n\n  // Term fields\n\n  public toggleTermField(field: string) {\n    const selected = this.termFieldSelected[field];\n    return Object.entries(this.headers).map(([key, group]) => {\n      group.headers.filter(isTermField(field)).map(header => header.selected = selected);\n      return this.updateGroupHeader(key);\n    });\n  }\n}\n","<p class=\"mb-2\">Please select which columns you would like to include:</p>\n\n<p class=\"my-2 is-size-7\"><i>You can drag and drop the headers to sort them as they would appear in the CSV file.</i></p>\n\n<div class=\"columns toggle-all mx-4 pb-1 mb-0\">\n  <div class=\"column\">\n    <label class=\"checkbox ml-1\">\n      <input type=\"checkbox\" class=\"selector\"\n        [(ngModel)]=\"allSelected\"\n        (change)=\"toggleAll(); refresh()\"\n      >\n      <span class=\"ml-2\">Toggle All</span>\n    </label>\n  </div>\n\n  <div class=\"column is-narrow has-text-right\">\n    <div class=\"dropdown is-right has-text-left\"\n      [class.is-active]=\"showSelectTermFields\"\n      (clickOutside)=\"showSelectTermFields = false\"\n    >\n      <div class=\"dropdown-trigger\" (click)=\"showSelectTermFields = !showSelectTermFields\">\n        <button class=\"button is-small\"\n          aria-haspopup=\"true\" aria-controls=\"select-menu\"\n        >\n          <span>Advanced Filters</span>\n          <span class=\"icon is-small\">\n            <fa-icon icon=\"filter\" aria-hidden=\"true\"></fa-icon>\n          </span>\n        </button>\n      </div>\n      <div class=\"dropdown-menu\" id=\"select-menu\" role=\"menu\">\n        <div class=\"dropdown-content\">\n          <a class=\"dropdown-item\">\n            <label class=\"checkbox ml-1\">\n              <input type=\"checkbox\" class=\"selector\"\n                [(ngModel)]=\"showNonIncluded\"\n                (change)=\"refresh()\"\n              >\n              <span class=\"ml-2\">Include <code>internal</code> fields</span>\n            </label>\n          </a>\n          <div class=\"dropdown-item\">\n            <p>Toggle Term Fields</p>\n          </div>\n          <a class=\"dropdown-item\"\n            *ngFor=\"let field of termFields\"\n          >\n            <label class=\"checkbox ml-1\">\n              <input type=\"checkbox\" class=\"selector\"\n                [(ngModel)]=\"termFieldSelected[field]\"\n                (change)=\"toggleTermField(field)\"\n              >\n              <span class=\"ml-2\">{{field}}</span>\n            </label>\n          </a>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"drag-container\" *ngIf=\"!loading; else loader\">\n  <div class=\"card\" *ngFor=\"let group of headers | keys\">\n    <header class=\"card-header\" *ngIf=\"group.key\">\n      <div class=\"card-header-title\">\n        <label class=\"checkbox ml-1\">\n          <input type=\"checkbox\" class=\"selector\"\n            [indeterminate]=\"group.value.partialSelected\"\n            [(ngModel)]=\"group.value.selected\"\n            (change)=\"updateGroup(group.key); refresh()\"\n          >\n          <span class=\"ml-2\">{{group.key}}</span>\n        </label>\n      </div>\n      <span class=\"card-header-icon has-text-link\" aria-label=\"open / close\"\n        (click)=\"group.value.open = !group.value.open\" pointer\n      >\n        <span class=\"icon\">\n          <fa-icon icon=\"angle-down\" [class.is-hidden]=\"!group.value.open\"></fa-icon>\n          <fa-icon icon=\"angle-left\" [class.is-hidden]=\"group.value.open\"></fa-icon>\n        </span>\n      </span>\n    </header>\n\n    <div class=\"card-content p-3\" [class.is-hidden]=\"!group.value.open\">\n      <ul cdkDropList (cdkDropListDropped)=\"dropHeader($event, group.key)\">\n        <ng-container *ngFor=\"let header of group.value.headers\">\n          <ng-container *ngIf=\"header.included || showNonIncluded\">\n            <li cdkDrag>\n              <label class=\"checkbox ml-2\">\n                <input type=\"checkbox\" class=\"selector\"\n                  [(ngModel)]=\"header.selected\"\n                  (change)=\"updateGroupHeader(group.key); refresh()\"\n                >\n                <span class=\"ml-2\">{{header.header}}</span>\n              </label>\n            </li>\n          </ng-container>\n        </ng-container>\n      </ul>\n    </div>\n  </div>\n</div>\n\n<ng-template #loader>\n  <div class=\"has-text-center py-3\">\n    <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n  </div>\n</ng-template>\n"]}
@@ -1,5 +1,5 @@
1
1
  import { __awaiter } from "tslib";
2
- import { Component, Input, Output, EventEmitter } from '@angular/core';
2
+ import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy } from '@angular/core';
3
3
  import { create, formatters } from 'jsondiffpatch';
4
4
  import { DiffsDisplayType } from './node-diffs.model';
5
5
  import * as i0 from "@angular/core";
@@ -45,13 +45,14 @@ export class NodeDiffsComponent {
45
45
  }
46
46
  }
47
47
  NodeDiffsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: NodeDiffsComponent, deps: [{ token: i1.HeNodeService }], target: i0.ɵɵFactoryTarget.Component });
48
- NodeDiffsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: NodeDiffsComponent, selector: "he-node-diffs", inputs: { data: "data", id: "id", type: "type", displayType: "displayType" }, outputs: { diffsLoaded: "diffsLoaded" }, ngImport: i0, template: "<div *ngIf=\"loading\" class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n</div>\n\n<ng-container *ngIf=\"left && right\">\n <ng-container [ngSwitch]=\"displayType\">\n <div *ngSwitchCase=\"DiffsDisplayType.diffs\" [innerHTML]=\"diffHtml\"></div>\n\n <div *ngSwitchCase=\"DiffsDisplayType.sideBySide\" class=\"columns\">\n <div class=\"column is-6\">\n <pre class=\"has-background-black has-text-white\"><code>{{left | json}}</code></pre>\n </div>\n <div class=\"column is-6\">\n <pre class=\"has-background-black has-text-white\"><code>{{right | json}}</code></pre>\n </div>\n </div>\n </ng-container>\n</ng-container>\n", styles: [""], components: [{ type: i2.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i3.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }], pipes: { "json": i3.JsonPipe } });
48
+ NodeDiffsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: NodeDiffsComponent, selector: "he-node-diffs", inputs: { data: "data", id: "id", type: "type", displayType: "displayType" }, outputs: { diffsLoaded: "diffsLoaded" }, ngImport: i0, template: "<div *ngIf=\"loading\" class=\"has-text-center py-3\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"lg\"></fa-icon>\n</div>\n\n<ng-container *ngIf=\"left && right\">\n <ng-container [ngSwitch]=\"displayType\">\n <div *ngSwitchCase=\"DiffsDisplayType.diffs\" [innerHTML]=\"diffHtml\"></div>\n\n <div *ngSwitchCase=\"DiffsDisplayType.sideBySide\" class=\"columns\">\n <div class=\"column is-6\">\n <pre class=\"has-background-black has-text-white\"><code>{{left | json}}</code></pre>\n </div>\n <div class=\"column is-6\">\n <pre class=\"has-background-black has-text-white\"><code>{{right | json}}</code></pre>\n </div>\n </div>\n </ng-container>\n</ng-container>\n", styles: [""], components: [{ type: i2.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i3.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }], pipes: { "json": i3.JsonPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
49
49
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: NodeDiffsComponent, decorators: [{
50
50
  type: Component,
51
51
  args: [{
52
52
  selector: 'he-node-diffs',
53
53
  templateUrl: './node-diffs.component.html',
54
- styleUrls: ['./node-diffs.component.scss']
54
+ styleUrls: ['./node-diffs.component.scss'],
55
+ changeDetection: ChangeDetectionStrategy.OnPush
55
56
  }]
56
57
  }], ctorParameters: function () { return [{ type: i1.HeNodeService }]; }, propDecorators: { data: [{
57
58
  type: Input
@@ -64,4 +65,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
64
65
  }], diffsLoaded: [{
65
66
  type: Output
66
67
  }] } });
67
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS1kaWZmcy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbm9kZS9ub2RlLWRpZmZzL25vZGUtZGlmZnMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vc3JjL25vZGUvbm9kZS1kaWZmcy9ub2RlLWRpZmZzLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBVSxNQUFNLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRS9FLE9BQU8sRUFBRSxNQUFNLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBR25ELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLG9CQUFvQixDQUFDOzs7OztBQUZ0RCxNQUFNLElBQUksR0FBRyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUM7QUFLcEMsTUFBTSxnQkFBZ0IsR0FBRztJQUN2QixPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxJQUFJO0NBQzdCLENBQUM7QUFFRixNQUFNLGlCQUFpQixHQUFHO0lBQ3hCLDhCQUE4QjtJQUM5QixVQUFVLEVBQUUsWUFBWSxFQUFFLGdCQUFnQixFQUFFLGNBQWMsRUFBRSxlQUFlLEVBQUUsWUFBWSxFQUFFLGFBQWE7Q0FDekcsQ0FBQztBQUVGLE1BQU0sVUFBVSxHQUFHLE1BQU0sQ0FBQztJQUN4QixjQUFjLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLGdCQUFnQixDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUM7Q0FDOUYsQ0FBQyxDQUFDO0FBT0gsTUFBTSxPQUFPLGtCQUFrQjtJQW1CN0IsWUFDVSxXQUEwQjtRQUExQixnQkFBVyxHQUFYLFdBQVcsQ0FBZTtRQVo3QixnQkFBVyxHQUFHLGdCQUFnQixDQUFDLEtBQUssQ0FBQztRQUdwQyxnQkFBVyxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFFMUMscUJBQWdCLEdBQUcsZ0JBQWdCLENBQUM7UUFDcEMsWUFBTyxHQUFHLElBQUksQ0FBQztJQU9sQixDQUFDO0lBRUMsUUFBUTs7O1lBQ1osSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO1lBQ3ZCLE1BQU0sSUFBSSxHQUFHLENBQUEsTUFBQSxJQUFJLENBQUMsSUFBSSwwQ0FBRSxJQUFJLEtBQUksSUFBSSxDQUFDLElBQUksQ0FBQztZQUMxQyxJQUFJLENBQUMsSUFBSSxHQUFHLE1BQU0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQW1CLEVBQUUsT0FBTyxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDNUYsSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQ2pCLElBQUksQ0FBQyxPQUFPLEdBQUcsS0FBSyxDQUFDOztLQUN0QjtJQUVPLFNBQVM7UUFDZixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLGlCQUFpQixDQUFDLENBQUM7UUFDL0MsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxpQkFBaUIsQ0FBQyxDQUFDO1FBQ2pELE1BQU0sS0FBSyxHQUFHLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDckQsTUFBTSxLQUFLLEdBQUcsVUFBVSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBTSxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUN4RCxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM3QixJQUFJLENBQUMsUUFBUSxHQUFHLEtBQUssSUFBSSx1QkFBdUIsQ0FBQztJQUNuRCxDQUFDOztnSEF0Q1Usa0JBQWtCO29HQUFsQixrQkFBa0IsNEtDMUIvQiwrc0JBa0JBOzRGRFFhLGtCQUFrQjtrQkFMOUIsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsZUFBZTtvQkFDekIsV0FBVyxFQUFFLDZCQUE2QjtvQkFDMUMsU0FBUyxFQUFFLENBQUMsNkJBQTZCLENBQUM7aUJBQzNDO29HQUdTLElBQUk7c0JBRFgsS0FBSztnQkFHRSxFQUFFO3NCQURULEtBQUs7Z0JBR0UsSUFBSTtzQkFEWCxLQUFLO2dCQUdDLFdBQVc7c0JBRGpCLEtBQUs7Z0JBSUUsV0FBVztzQkFEbEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIE9uSW5pdCwgT3V0cHV0LCBFdmVudEVtaXR0ZXIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5vZGVUeXBlLCBKU09OTEQgfSBmcm9tICdAaGVzdGlhLWVhcnRoL3NjaGVtYSc7XG5pbXBvcnQgeyBjcmVhdGUsIGZvcm1hdHRlcnMgfSBmcm9tICdqc29uZGlmZnBhdGNoJztcbmNvbnN0IG9taXQgPSByZXF1aXJlKCdsb2Rhc2gub21pdCcpO1xuXG5pbXBvcnQgeyBEaWZmc0Rpc3BsYXlUeXBlIH0gZnJvbSAnLi9ub2RlLWRpZmZzLm1vZGVsJztcbmltcG9ydCB7IEhlTm9kZVNlcnZpY2UgfSBmcm9tICcuLi9ub2RlLnNlcnZpY2UnO1xuXG5jb25zdCBpZ25vcmVQcm9wZXJ0aWVzID0gW1xuICAnQHR5cGUnLCAnQGlkJywgJ3R5cGUnLCAnaWQnXG5dO1xuXG5jb25zdCBleGNsdWRlUHJvcGVydGllcyA9IFtcbiAgLy8gcHJvcGVydGllcyBmb3IgaW5kZXhlZCBOb2RlXG4gICdAY29udGV4dCcsICd1cGxvYWREYXRlJywgJ3ZhbGlkYXRpb25EYXRlJywgJ3ZhbGlkYXRpb25CeScsICdzY2hlbWFWZXJzaW9uJywgJ29yaWdpbmFsSWQnLCAnZGF0YVByaXZhdGUnXG5dO1xuXG5jb25zdCBjdXN0b21EaWZmID0gY3JlYXRlKHtcbiAgcHJvcGVydHlGaWx0ZXI6IG5hbWUgPT4gIWlnbm9yZVByb3BlcnRpZXMuaW5jbHVkZXMobmFtZSkgJiYgIWV4Y2x1ZGVQcm9wZXJ0aWVzLmluY2x1ZGVzKG5hbWUpXG59KTtcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnaGUtbm9kZS1kaWZmcycsXG4gIHRlbXBsYXRlVXJsOiAnLi9ub2RlLWRpZmZzLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vbm9kZS1kaWZmcy5jb21wb25lbnQuc2NzcyddXG59KVxuZXhwb3J0IGNsYXNzIE5vZGVEaWZmc0NvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIEBJbnB1dCgpXG4gIHByaXZhdGUgZGF0YTogYW55O1xuICBASW5wdXQoKVxuICBwcml2YXRlIGlkPzogc3RyaW5nO1xuICBASW5wdXQoKVxuICBwcml2YXRlIHR5cGU/OiBOb2RlVHlwZTtcbiAgQElucHV0KClcbiAgcHVibGljIGRpc3BsYXlUeXBlID0gRGlmZnNEaXNwbGF5VHlwZS5kaWZmcztcblxuICBAT3V0cHV0KClcbiAgcHJpdmF0ZSBkaWZmc0xvYWRlZCA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xuXG4gIHB1YmxpYyBEaWZmc0Rpc3BsYXlUeXBlID0gRGlmZnNEaXNwbGF5VHlwZTtcbiAgcHVibGljIGxvYWRpbmcgPSB0cnVlO1xuICBwdWJsaWMgbGVmdD86IEpTT05MRDxOb2RlVHlwZT47XG4gIHB1YmxpYyByaWdodDogYW55O1xuICBwdWJsaWMgZGlmZkh0bWw/OiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBub2RlU2VydmljZTogSGVOb2RlU2VydmljZVxuICApIHsgfVxuXG4gIGFzeW5jIG5nT25Jbml0KCkge1xuICAgIHRoaXMucmlnaHQgPSB0aGlzLmRhdGE7XG4gICAgY29uc3QgdHlwZSA9IHRoaXMuZGF0YT8udHlwZSB8fCB0aGlzLnR5cGU7XG4gICAgdGhpcy5sZWZ0ID0gYXdhaXQgdGhpcy5ub2RlU2VydmljZS5nZXQ8SlNPTkxEPE5vZGVUeXBlPj4oeyAnQHR5cGUnOiB0eXBlLCAnQGlkJzogdGhpcy5pZCB9KTtcbiAgICB0aGlzLmxvYWREaWZmcygpO1xuICAgIHRoaXMubG9hZGluZyA9IGZhbHNlO1xuICB9XG5cbiAgcHJpdmF0ZSBsb2FkRGlmZnMoKSB7XG4gICAgdGhpcy5sZWZ0ID0gb21pdCh0aGlzLmxlZnQsIGV4Y2x1ZGVQcm9wZXJ0aWVzKTtcbiAgICB0aGlzLnJpZ2h0ID0gb21pdCh0aGlzLnJpZ2h0LCBleGNsdWRlUHJvcGVydGllcyk7XG4gICAgY29uc3QgZGVsdGEgPSBjdXN0b21EaWZmLmRpZmYodGhpcy5sZWZ0LCB0aGlzLnJpZ2h0KTtcbiAgICBjb25zdCBkaWZmcyA9IGZvcm1hdHRlcnMuaHRtbC5mb3JtYXQoZGVsdGEhLCB0aGlzLmxlZnQpO1xuICAgIHRoaXMuZGlmZnNMb2FkZWQuZW1pdChkaWZmcyk7XG4gICAgdGhpcy5kaWZmSHRtbCA9IGRpZmZzIHx8ICc8cD5ObyBkaWZmZXJlbmNlLjwvcD4nO1xuICB9XG59XG4iLCI8ZGl2ICpuZ0lmPVwibG9hZGluZ1wiIGNsYXNzPVwiaGFzLXRleHQtY2VudGVyIHB5LTNcIj5cbiAgPGZhLWljb24gaWNvbj1cInNwaW5uZXJcIiBbcHVsc2VdPVwidHJ1ZVwiIHNpemU9XCJsZ1wiPjwvZmEtaWNvbj5cbjwvZGl2PlxuXG48bmctY29udGFpbmVyICpuZ0lmPVwibGVmdCAmJiByaWdodFwiPlxuICA8bmctY29udGFpbmVyIFtuZ1N3aXRjaF09XCJkaXNwbGF5VHlwZVwiPlxuICAgIDxkaXYgKm5nU3dpdGNoQ2FzZT1cIkRpZmZzRGlzcGxheVR5cGUuZGlmZnNcIiBbaW5uZXJIVE1MXT1cImRpZmZIdG1sXCI+PC9kaXY+XG5cbiAgICA8ZGl2ICpuZ1N3aXRjaENhc2U9XCJEaWZmc0Rpc3BsYXlUeXBlLnNpZGVCeVNpZGVcIiBjbGFzcz1cImNvbHVtbnNcIj5cbiAgICAgIDxkaXYgY2xhc3M9XCJjb2x1bW4gaXMtNlwiPlxuICAgICAgICA8cHJlIGNsYXNzPVwiaGFzLWJhY2tncm91bmQtYmxhY2sgaGFzLXRleHQtd2hpdGVcIj48Y29kZT57e2xlZnQgfCBqc29ufX08L2NvZGU+PC9wcmU+XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJjb2x1bW4gaXMtNlwiPlxuICAgICAgICA8cHJlIGNsYXNzPVwiaGFzLWJhY2tncm91bmQtYmxhY2sgaGFzLXRleHQtd2hpdGVcIj48Y29kZT57e3JpZ2h0IHwganNvbn19PC9jb2RlPjwvcHJlPlxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIDwvbmctY29udGFpbmVyPlxuPC9uZy1jb250YWluZXI+XG4iXX0=
68
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS1kaWZmcy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbm9kZS9ub2RlLWRpZmZzL25vZGUtZGlmZnMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vc3JjL25vZGUvbm9kZS1kaWZmcy9ub2RlLWRpZmZzLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBVSxNQUFNLEVBQUUsWUFBWSxFQUFFLHVCQUF1QixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRXhHLE9BQU8sRUFBRSxNQUFNLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBR25ELE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLG9CQUFvQixDQUFDOzs7OztBQUZ0RCxNQUFNLElBQUksR0FBRyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUM7QUFLcEMsTUFBTSxnQkFBZ0IsR0FBRztJQUN2QixPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxJQUFJO0NBQzdCLENBQUM7QUFFRixNQUFNLGlCQUFpQixHQUFHO0lBQ3hCLDhCQUE4QjtJQUM5QixVQUFVLEVBQUUsWUFBWSxFQUFFLGdCQUFnQixFQUFFLGNBQWMsRUFBRSxlQUFlLEVBQUUsWUFBWSxFQUFFLGFBQWE7Q0FDekcsQ0FBQztBQUVGLE1BQU0sVUFBVSxHQUFHLE1BQU0sQ0FBQztJQUN4QixjQUFjLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLGdCQUFnQixDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUM7Q0FDOUYsQ0FBQyxDQUFDO0FBUUgsTUFBTSxPQUFPLGtCQUFrQjtJQW1CN0IsWUFDVSxXQUEwQjtRQUExQixnQkFBVyxHQUFYLFdBQVcsQ0FBZTtRQVo3QixnQkFBVyxHQUFHLGdCQUFnQixDQUFDLEtBQUssQ0FBQztRQUdwQyxnQkFBVyxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFFMUMscUJBQWdCLEdBQUcsZ0JBQWdCLENBQUM7UUFDcEMsWUFBTyxHQUFHLElBQUksQ0FBQztJQU9sQixDQUFDO0lBRUMsUUFBUTs7O1lBQ1osSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO1lBQ3ZCLE1BQU0sSUFBSSxHQUFHLENBQUEsTUFBQSxJQUFJLENBQUMsSUFBSSwwQ0FBRSxJQUFJLEtBQUksSUFBSSxDQUFDLElBQUksQ0FBQztZQUMxQyxJQUFJLENBQUMsSUFBSSxHQUFHLE1BQU0sSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQW1CLEVBQUUsT0FBTyxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDNUYsSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO1lBQ2pCLElBQUksQ0FBQyxPQUFPLEdBQUcsS0FBSyxDQUFDOztLQUN0QjtJQUVPLFNBQVM7UUFDZixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLGlCQUFpQixDQUFDLENBQUM7UUFDL0MsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxpQkFBaUIsQ0FBQyxDQUFDO1FBQ2pELE1BQU0sS0FBSyxHQUFHLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDckQsTUFBTSxLQUFLLEdBQUcsVUFBVSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBTSxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUN4RCxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM3QixJQUFJLENBQUMsUUFBUSxHQUFHLEtBQUssSUFBSSx1QkFBdUIsQ0FBQztJQUNuRCxDQUFDOztnSEF0Q1Usa0JBQWtCO29HQUFsQixrQkFBa0IsNEtDM0IvQiwrc0JBa0JBOzRGRFNhLGtCQUFrQjtrQkFOOUIsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsZUFBZTtvQkFDekIsV0FBVyxFQUFFLDZCQUE2QjtvQkFDMUMsU0FBUyxFQUFFLENBQUMsNkJBQTZCLENBQUM7b0JBQzFDLGVBQWUsRUFBRSx1QkFBdUIsQ0FBQyxNQUFNO2lCQUNoRDtvR0FHUyxJQUFJO3NCQURYLEtBQUs7Z0JBR0UsRUFBRTtzQkFEVCxLQUFLO2dCQUdFLElBQUk7c0JBRFgsS0FBSztnQkFHQyxXQUFXO3NCQURqQixLQUFLO2dCQUlFLFdBQVc7c0JBRGxCLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPbkluaXQsIE91dHB1dCwgRXZlbnRFbWl0dGVyLCBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTm9kZVR5cGUsIEpTT05MRCB9IGZyb20gJ0BoZXN0aWEtZWFydGgvc2NoZW1hJztcbmltcG9ydCB7IGNyZWF0ZSwgZm9ybWF0dGVycyB9IGZyb20gJ2pzb25kaWZmcGF0Y2gnO1xuY29uc3Qgb21pdCA9IHJlcXVpcmUoJ2xvZGFzaC5vbWl0Jyk7XG5cbmltcG9ydCB7IERpZmZzRGlzcGxheVR5cGUgfSBmcm9tICcuL25vZGUtZGlmZnMubW9kZWwnO1xuaW1wb3J0IHsgSGVOb2RlU2VydmljZSB9IGZyb20gJy4uL25vZGUuc2VydmljZSc7XG5cbmNvbnN0IGlnbm9yZVByb3BlcnRpZXMgPSBbXG4gICdAdHlwZScsICdAaWQnLCAndHlwZScsICdpZCdcbl07XG5cbmNvbnN0IGV4Y2x1ZGVQcm9wZXJ0aWVzID0gW1xuICAvLyBwcm9wZXJ0aWVzIGZvciBpbmRleGVkIE5vZGVcbiAgJ0Bjb250ZXh0JywgJ3VwbG9hZERhdGUnLCAndmFsaWRhdGlvbkRhdGUnLCAndmFsaWRhdGlvbkJ5JywgJ3NjaGVtYVZlcnNpb24nLCAnb3JpZ2luYWxJZCcsICdkYXRhUHJpdmF0ZSdcbl07XG5cbmNvbnN0IGN1c3RvbURpZmYgPSBjcmVhdGUoe1xuICBwcm9wZXJ0eUZpbHRlcjogbmFtZSA9PiAhaWdub3JlUHJvcGVydGllcy5pbmNsdWRlcyhuYW1lKSAmJiAhZXhjbHVkZVByb3BlcnRpZXMuaW5jbHVkZXMobmFtZSlcbn0pO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdoZS1ub2RlLWRpZmZzJyxcbiAgdGVtcGxhdGVVcmw6ICcuL25vZGUtZGlmZnMuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9ub2RlLWRpZmZzLmNvbXBvbmVudC5zY3NzJ10sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoXG59KVxuZXhwb3J0IGNsYXNzIE5vZGVEaWZmc0NvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIEBJbnB1dCgpXG4gIHByaXZhdGUgZGF0YTogYW55O1xuICBASW5wdXQoKVxuICBwcml2YXRlIGlkPzogc3RyaW5nO1xuICBASW5wdXQoKVxuICBwcml2YXRlIHR5cGU/OiBOb2RlVHlwZTtcbiAgQElucHV0KClcbiAgcHVibGljIGRpc3BsYXlUeXBlID0gRGlmZnNEaXNwbGF5VHlwZS5kaWZmcztcblxuICBAT3V0cHV0KClcbiAgcHJpdmF0ZSBkaWZmc0xvYWRlZCA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xuXG4gIHB1YmxpYyBEaWZmc0Rpc3BsYXlUeXBlID0gRGlmZnNEaXNwbGF5VHlwZTtcbiAgcHVibGljIGxvYWRpbmcgPSB0cnVlO1xuICBwdWJsaWMgbGVmdD86IEpTT05MRDxOb2RlVHlwZT47XG4gIHB1YmxpYyByaWdodDogYW55O1xuICBwdWJsaWMgZGlmZkh0bWw/OiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBub2RlU2VydmljZTogSGVOb2RlU2VydmljZVxuICApIHsgfVxuXG4gIGFzeW5jIG5nT25Jbml0KCkge1xuICAgIHRoaXMucmlnaHQgPSB0aGlzLmRhdGE7XG4gICAgY29uc3QgdHlwZSA9IHRoaXMuZGF0YT8udHlwZSB8fCB0aGlzLnR5cGU7XG4gICAgdGhpcy5sZWZ0ID0gYXdhaXQgdGhpcy5ub2RlU2VydmljZS5nZXQ8SlNPTkxEPE5vZGVUeXBlPj4oeyAnQHR5cGUnOiB0eXBlLCAnQGlkJzogdGhpcy5pZCB9KTtcbiAgICB0aGlzLmxvYWREaWZmcygpO1xuICAgIHRoaXMubG9hZGluZyA9IGZhbHNlO1xuICB9XG5cbiAgcHJpdmF0ZSBsb2FkRGlmZnMoKSB7XG4gICAgdGhpcy5sZWZ0ID0gb21pdCh0aGlzLmxlZnQsIGV4Y2x1ZGVQcm9wZXJ0aWVzKTtcbiAgICB0aGlzLnJpZ2h0ID0gb21pdCh0aGlzLnJpZ2h0LCBleGNsdWRlUHJvcGVydGllcyk7XG4gICAgY29uc3QgZGVsdGEgPSBjdXN0b21EaWZmLmRpZmYodGhpcy5sZWZ0LCB0aGlzLnJpZ2h0KTtcbiAgICBjb25zdCBkaWZmcyA9IGZvcm1hdHRlcnMuaHRtbC5mb3JtYXQoZGVsdGEhLCB0aGlzLmxlZnQpO1xuICAgIHRoaXMuZGlmZnNMb2FkZWQuZW1pdChkaWZmcyk7XG4gICAgdGhpcy5kaWZmSHRtbCA9IGRpZmZzIHx8ICc8cD5ObyBkaWZmZXJlbmNlLjwvcD4nO1xuICB9XG59XG4iLCI8ZGl2ICpuZ0lmPVwibG9hZGluZ1wiIGNsYXNzPVwiaGFzLXRleHQtY2VudGVyIHB5LTNcIj5cbiAgPGZhLWljb24gaWNvbj1cInNwaW5uZXJcIiBbcHVsc2VdPVwidHJ1ZVwiIHNpemU9XCJsZ1wiPjwvZmEtaWNvbj5cbjwvZGl2PlxuXG48bmctY29udGFpbmVyICpuZ0lmPVwibGVmdCAmJiByaWdodFwiPlxuICA8bmctY29udGFpbmVyIFtuZ1N3aXRjaF09XCJkaXNwbGF5VHlwZVwiPlxuICAgIDxkaXYgKm5nU3dpdGNoQ2FzZT1cIkRpZmZzRGlzcGxheVR5cGUuZGlmZnNcIiBbaW5uZXJIVE1MXT1cImRpZmZIdG1sXCI+PC9kaXY+XG5cbiAgICA8ZGl2ICpuZ1N3aXRjaENhc2U9XCJEaWZmc0Rpc3BsYXlUeXBlLnNpZGVCeVNpZGVcIiBjbGFzcz1cImNvbHVtbnNcIj5cbiAgICAgIDxkaXYgY2xhc3M9XCJjb2x1bW4gaXMtNlwiPlxuICAgICAgICA8cHJlIGNsYXNzPVwiaGFzLWJhY2tncm91bmQtYmxhY2sgaGFzLXRleHQtd2hpdGVcIj48Y29kZT57e2xlZnQgfCBqc29ufX08L2NvZGU+PC9wcmU+XG4gICAgICA8L2Rpdj5cbiAgICAgIDxkaXYgY2xhc3M9XCJjb2x1bW4gaXMtNlwiPlxuICAgICAgICA8cHJlIGNsYXNzPVwiaGFzLWJhY2tncm91bmQtYmxhY2sgaGFzLXRleHQtd2hpdGVcIj48Y29kZT57e3JpZ2h0IHwganNvbn19PC9jb2RlPjwvcHJlPlxuICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG4gIDwvbmctY29udGFpbmVyPlxuPC9uZy1jb250YWluZXI+XG4iXX0=