@hestia-earth/ui-components 0.0.4 → 0.0.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/hestia-earth-ui-components.umd.js +392 -506
- package/bundles/hestia-earth-ui-components.umd.js.map +1 -1
- package/common/common.service.d.ts +1 -2
- package/common/public-api.d.ts +1 -0
- package/common/toast/toast.component.d.ts +2 -2
- package/common/toast.service.d.ts +3 -3
- package/common/utils.d.ts +3 -11
- package/cycles/public-api.d.ts +1 -0
- package/engine/aggregation-engine.service.d.ts +8 -20
- package/engine/engine.service.d.ts +2 -6
- package/esm2015/common/common.service.js +1 -7
- package/esm2015/common/public-api.js +2 -1
- package/esm2015/common/toast/toast.component.js +3 -3
- package/esm2015/common/toast.service.js +5 -5
- package/esm2015/common/utils.js +3 -61
- package/esm2015/cycles/public-api.js +2 -1
- package/esm2015/engine/aggregation-engine.service.js +11 -47
- package/esm2015/engine/engine.service.js +9 -15
- package/esm2015/impact-assessments/impact-assessments-products/impact-assessments-products.component.js +3 -3
- package/esm2015/mendeley/mendeley.service.js +1 -1
- package/esm2015/node/node-csv.service.js +1 -1
- package/esm2015/node/node.service.js +26 -16
- package/esm2015/schema/schema.service.js +1 -1
- package/esm2015/search/search.service.js +1 -1
- package/fesm2015/hestia-earth-ui-components.js +281 -372
- package/fesm2015/hestia-earth-ui-components.js.map +1 -1
- package/impact-assessments/impact-assessments-products/impact-assessments-products.component.d.ts +2 -2
- package/mendeley/mendeley.service.d.ts +2 -2
- package/node/node-csv.service.d.ts +1 -1
- package/node/node.service.d.ts +16 -7
- package/package.json +1 -1
- package/schema/schema.service.d.ts +1 -1
- package/search/search.service.d.ts +2 -2
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
export * from './cycles.model';
|
|
1
2
|
export * from './cycles.module';
|
|
2
3
|
export { CyclesActivityComponent } from './cycles-activity/cycles-activity.component';
|
|
3
4
|
export { CyclesActivityLogsComponent } from './cycles-activity-logs/cycles-activity-logs.component';
|
|
@@ -8,4 +9,4 @@ export { CyclesEmissionsComponent } from './cycles-emissions/cycles-emissions.co
|
|
|
8
9
|
export { CyclesFunctionalUnitMeasureComponent } from './cycles-functional-unit-measure/cycles-functional-unit-measure.component';
|
|
9
10
|
export { CyclesResultComponent } from './cycles-result/cycles-result.component';
|
|
10
11
|
export { CyclesSuggestFormComponent } from './cycles-suggest-form/cycles-suggest-form.component';
|
|
11
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
12
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jeWNsZXMvcHVibGljLWFwaS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsaUJBQWlCLENBQUM7QUFFaEMsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sNkNBQTZDLENBQUM7QUFDdEYsT0FBTyxFQUFFLDJCQUEyQixFQUFFLE1BQU0sdURBQXVELENBQUM7QUFDcEcsT0FBTyxFQUFFLDJCQUEyQixFQUFFLE1BQU0scURBQXFELENBQUM7QUFDbEcsT0FBTyxFQUFFLDZCQUE2QixFQUFFLE1BQU0sMkRBQTJELENBQUM7QUFDMUcsT0FBTyxFQUFFLDRCQUE0QixFQUFFLE1BQU0seURBQXlELENBQUM7QUFDdkcsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sK0NBQStDLENBQUM7QUFDekYsT0FBTyxFQUFFLG9DQUFvQyxFQUFFLE1BQU0sMkVBQTJFLENBQUM7QUFDakksT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0seUNBQXlDLENBQUM7QUFDaEYsT0FBTyxFQUFFLDBCQUEwQixFQUFFLE1BQU0scURBQXFELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2N5Y2xlcy5tb2RlbCc7XG5leHBvcnQgKiBmcm9tICcuL2N5Y2xlcy5tb2R1bGUnO1xuXG5leHBvcnQgeyBDeWNsZXNBY3Rpdml0eUNvbXBvbmVudCB9IGZyb20gJy4vY3ljbGVzLWFjdGl2aXR5L2N5Y2xlcy1hY3Rpdml0eS5jb21wb25lbnQnO1xuZXhwb3J0IHsgQ3ljbGVzQWN0aXZpdHlMb2dzQ29tcG9uZW50IH0gZnJvbSAnLi9jeWNsZXMtYWN0aXZpdHktbG9ncy9jeWNsZXMtYWN0aXZpdHktbG9ncy5jb21wb25lbnQnO1xuZXhwb3J0IHsgQ3ljbGVzQ29tcGxldGVuZXNzQ29tcG9uZW50IH0gZnJvbSAnLi9jeWNsZXMtY29tcGxldGVuZXNzL2N5Y2xlcy1jb21wbGV0ZW5lc3MuY29tcG9uZW50JztcbmV4cG9ydCB7IEN5Y2xlc0VtaXNzaW9uc0NoYXJ0Q29tcG9uZW50IH0gZnJvbSAnLi9jeWNsZXMtZW1pc3Npb25zLWNoYXJ0L2N5Y2xlcy1lbWlzc2lvbnMtY2hhcnQuY29tcG9uZW50JztcbmV4cG9ydCB7IEN5Y2xlc0VtaXNzaW9uc0xvZ3NDb21wb25lbnQgfSBmcm9tICcuL2N5Y2xlcy1lbWlzc2lvbnMtbG9ncy9jeWNsZXMtZW1pc3Npb25zLWxvZ3MuY29tcG9uZW50JztcbmV4cG9ydCB7IEN5Y2xlc0VtaXNzaW9uc0NvbXBvbmVudCB9IGZyb20gJy4vY3ljbGVzLWVtaXNzaW9ucy9jeWNsZXMtZW1pc3Npb25zLmNvbXBvbmVudCc7XG5leHBvcnQgeyBDeWNsZXNGdW5jdGlvbmFsVW5pdE1lYXN1cmVDb21wb25lbnQgfSBmcm9tICcuL2N5Y2xlcy1mdW5jdGlvbmFsLXVuaXQtbWVhc3VyZS9jeWNsZXMtZnVuY3Rpb25hbC11bml0LW1lYXN1cmUuY29tcG9uZW50JztcbmV4cG9ydCB7IEN5Y2xlc1Jlc3VsdENvbXBvbmVudCB9IGZyb20gJy4vY3ljbGVzLXJlc3VsdC9jeWNsZXMtcmVzdWx0LmNvbXBvbmVudCc7XG5leHBvcnQgeyBDeWNsZXNTdWdnZXN0Rm9ybUNvbXBvbmVudCB9IGZyb20gJy4vY3ljbGVzLXN1Z2dlc3QtZm9ybS9jeWNsZXMtc3VnZ2VzdC1mb3JtLmNvbXBvbmVudCc7XG4iXX0=
|
|
@@ -4,49 +4,19 @@ import { forkJoin, from, of, ReplaySubject } from 'rxjs';
|
|
|
4
4
|
import { catchError, map, take } from 'rxjs/operators';
|
|
5
5
|
import { TermTermType } from '@hestia-earth/schema';
|
|
6
6
|
import * as csvtojson from 'csvtojson';
|
|
7
|
-
import {
|
|
8
|
-
import { primaryProduct } from '../cycles/cycles.model';
|
|
9
|
-
import { propertyValue, delta, gitHome, gitRawBaseUrl } from '../common/utils';
|
|
7
|
+
import { gitBranch, gitHome, gitRawBaseUrl } from '../common/utils';
|
|
10
8
|
import * as i0 from "@angular/core";
|
|
11
9
|
import * as i1 from "@angular/common/http";
|
|
12
|
-
import * as i2 from "../
|
|
13
|
-
|
|
14
|
-
const
|
|
15
|
-
const lookups = Object.freeze({
|
|
10
|
+
import * as i2 from "../node/node.service";
|
|
11
|
+
const gitUrl = () => `${gitHome}/hestia-aggregation-engine/-/blob/${gitBranch()}`;
|
|
12
|
+
const rawUrl = () => `${gitRawBaseUrl}/hestia-aggregation-engine/-/raw/${gitBranch()}`;
|
|
13
|
+
export const lookups = Object.freeze({
|
|
16
14
|
cropYield: 'region-crop-cropGroupingFaostatProduction-yield',
|
|
17
15
|
cropGroupingColumn: 'cropGroupingFaostatProduction'
|
|
18
16
|
});
|
|
19
|
-
const extractGroupedDataClosesDate = (data, year) => {
|
|
20
|
-
const dataByDate = (data || '').split(';').reduce((prev, curr) => (Object.assign(Object.assign({}, prev), (curr.length > 0 && curr.split(':')[1] !== '-' ? { [curr.split(':')[0]]: curr.split(':')[1] } : {}))), {});
|
|
21
|
-
const years = Object.keys(dataByDate).map(v => +v);
|
|
22
|
-
const closestYear = years.reduce((prev, curr) => Math.abs(curr - year) < Math.abs(prev - year) ? curr : prev, 0);
|
|
23
|
-
return closestYear ? +dataByDate[closestYear] / 10 : null;
|
|
24
|
-
};
|
|
25
|
-
const cropGrouping = (lookup, product) => (lookup.find(({ term }) => term.id === (product === null || product === void 0 ? void 0 : product.term['@id'])) || {})[lookups.cropGroupingColumn];
|
|
26
|
-
const cropYieldCountry = (cropYield, cycle) => cropYield.find(({ term }) => term.id === get(cycle, 'site.country.@id', worldRegion['@id'])) || {};
|
|
27
|
-
export const extendCycle = ({ crop, cropYield }) => (cycle) => {
|
|
28
|
-
var _a, _b, _c;
|
|
29
|
-
const product = primaryProduct(cycle);
|
|
30
|
-
const grouping = cropGrouping(crop, product);
|
|
31
|
-
const countryData = cropYieldCountry(cropYield, cycle)[grouping];
|
|
32
|
-
const countryValue = extractGroupedDataClosesDate(countryData, +(cycle.endDate || ''));
|
|
33
|
-
const productValue = propertyValue(product === null || product === void 0 ? void 0 : product.value);
|
|
34
|
-
return {
|
|
35
|
-
cycle,
|
|
36
|
-
name: (_a = product === null || product === void 0 ? void 0 : product.term) === null || _a === void 0 ? void 0 : _a.name,
|
|
37
|
-
startDate: cycle.startDate,
|
|
38
|
-
endDate: cycle.endDate,
|
|
39
|
-
country: ((_c = (_b = cycle.site) === null || _b === void 0 ? void 0 : _b.country) === null || _c === void 0 ? void 0 : _c.name) || worldRegion.name,
|
|
40
|
-
observations: propertyValue(product === null || product === void 0 ? void 0 : product.observations) || 0,
|
|
41
|
-
yield: productValue,
|
|
42
|
-
faoYield: countryValue || undefined,
|
|
43
|
-
diff: delta(productValue, countryValue)
|
|
44
|
-
};
|
|
45
|
-
};
|
|
46
17
|
export class HeAggregationEngineService {
|
|
47
|
-
constructor(http,
|
|
18
|
+
constructor(http, nodeService) {
|
|
48
19
|
this.http = http;
|
|
49
|
-
this.commonService = commonService;
|
|
50
20
|
this.nodeService = nodeService;
|
|
51
21
|
this.modelsLoading = false;
|
|
52
22
|
this.modelsLoaded = false;
|
|
@@ -61,18 +31,12 @@ export class HeAggregationEngineService {
|
|
|
61
31
|
yield this.loadModels().toPromise();
|
|
62
32
|
});
|
|
63
33
|
}
|
|
64
|
-
get gitUrl() {
|
|
65
|
-
return `${gitHome}/hestia-aggregation-engine/-/blob/${this.commonService.gitBranch}`;
|
|
66
|
-
}
|
|
67
|
-
get rawUrl() {
|
|
68
|
-
return `${gitRawBaseUrl}/hestia-aggregation-engine/-/raw/${this.commonService.gitBranch}`;
|
|
69
|
-
}
|
|
70
34
|
loadModels() {
|
|
71
35
|
this.modelsLoading = true;
|
|
72
|
-
return this.http.get(`${
|
|
36
|
+
return this.http.get(`${rawUrl()}/model-links.json`).pipe(catchError(() => of({ links: [] })), map(({ links }) => {
|
|
73
37
|
this._models.next(links.map((_a) => {
|
|
74
38
|
var { path: modelPath, docPath } = _a, link = __rest(_a, ["path", "docPath"]);
|
|
75
|
-
return (Object.assign(Object.assign({}, link), { path: `${
|
|
39
|
+
return (Object.assign(Object.assign({}, link), { path: `${gitUrl()}/${modelPath}`, docPath: `${gitUrl()}/${docPath}` }));
|
|
76
40
|
}));
|
|
77
41
|
this.modelsLoading = false;
|
|
78
42
|
this.modelsLoaded = true;
|
|
@@ -115,12 +79,12 @@ export class HeAggregationEngineService {
|
|
|
115
79
|
return this.lookups$.pipe(take(1)).toPromise();
|
|
116
80
|
}
|
|
117
81
|
}
|
|
118
|
-
HeAggregationEngineService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeAggregationEngineService, deps: [{ token: i1.HttpClient }, { token: i2.
|
|
82
|
+
HeAggregationEngineService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeAggregationEngineService, deps: [{ token: i1.HttpClient }, { token: i2.HeNodeService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
119
83
|
HeAggregationEngineService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeAggregationEngineService, providedIn: 'root' });
|
|
120
84
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeAggregationEngineService, decorators: [{
|
|
121
85
|
type: Injectable,
|
|
122
86
|
args: [{
|
|
123
87
|
providedIn: 'root'
|
|
124
88
|
}]
|
|
125
|
-
}], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.
|
|
126
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aggregation-engine.service.js","sourceRoot":"","sources":["../../../src/engine/aggregation-engine.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { forkJoin, from, of, ReplaySubject } from 'rxjs';\nimport { catchError, map, take } from 'rxjs/operators';\nimport { ICycleJSONLD, NodeType, Product, Term, TermTermType } from '@hestia-earth/schema';\nimport * as csvtojson from 'csvtojson';\nconst get = require('lodash.get');\n\nimport { HeNodeService } from '../node/node.service';\nimport { worldRegion } from '../search/search.model';\nimport { primaryProduct } from '../cycles/cycles.model';\nimport { propertyValue, delta, gitHome, gitRawBaseUrl } from '../common/utils';\nimport { HeCommonService } from '../common/common.service';\n\nconst lookups = Object.freeze({\n  cropYield: 'region-crop-cropGroupingFaostatProduction-yield',\n  cropGroupingColumn: 'cropGroupingFaostatProduction'\n});\n\nexport interface IModel {\n  type: NodeType;\n  /**\n   * Path to the implementation (code) of the model.\n   */\n  path: string;\n  /**\n   * Path to the documentation of the model.\n   */\n  docPath: string;\n  dataPath: string;\n  '@id'?: string;\n}\n\ninterface IModelLinks {\n  links: IModel[];\n}\n\nexport interface ILookups {\n  crop: any[];\n  cropYield: any[];\n}\n\nconst extractGroupedDataClosesDate = (data: string, year: number) => {\n  const dataByDate = (data || '').split(';').reduce((prev, curr) => ({\n    ...prev,\n    ...(curr.length > 0 && curr.split(':')[1] !== '-' ? {[curr.split(':')[0]]: curr.split(':')[1]} : {})\n  }), {});\n  const years = Object.keys(dataByDate).map(v => +v);\n  const closestYear = years.reduce((prev, curr) => Math.abs(curr - year) < Math.abs(prev - year) ? curr : prev, 0);\n  return closestYear ? +dataByDate[closestYear] / 10 : null;\n};\n\nconst cropGrouping = (lookup: any[], product: Product) =>\n  (lookup.find(({ term }) => term.id === product?.term!['@id']) || {})[lookups.cropGroupingColumn];\n\nexport interface ICycle {\n  cycle: ICycleJSONLD;\n  name?: string;\n  startDate?: string;\n  endDate?: string;\n  country: string;\n  observations?: number;\n  yield?: number;\n  faoYield?: number;\n  diff?: number;\n}\n\nconst cropYieldCountry = (cropYield: any[], cycle: ICycleJSONLD) =>\n  cropYield.find(({ term }) => term.id === get(cycle, 'site.country.@id', worldRegion['@id'])) || {};\n\nexport const extendCycle = ({ crop, cropYield }: ILookups) => (cycle: ICycleJSONLD): ICycle => {\n  const product = primaryProduct(cycle);\n  const grouping = cropGrouping(crop, product!);\n  const countryData = cropYieldCountry(cropYield, cycle)[grouping];\n  const countryValue = extractGroupedDataClosesDate(countryData, +(cycle.endDate || ''));\n  const productValue = propertyValue(product?.value!);\n\n  return {\n    cycle,\n    name: product?.term?.name,\n    startDate: cycle.startDate,\n    endDate: cycle.endDate,\n    country: cycle.site?.country?.name || worldRegion.name,\n    observations: propertyValue(product?.observations!) || 0,\n    yield: productValue,\n    faoYield: countryValue || undefined,\n    diff: delta(productValue, countryValue)\n  };\n};\n\nexport interface ITriggerCountryParams {\n  country: Term;\n  product: Product;\n  '@type'?: NodeType;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class HeAggregationEngineService {\n  private modelsLoading = false;\n  private modelsLoaded = false;\n  private _models = new ReplaySubject<IModel[]>(1);\n\n  private lookupsLoading = false;\n  private lookupsLoaded = false;\n  private _lookups = new ReplaySubject<ILookups>(1);\n\n  constructor(\n    private http: HttpClient,\n    private commonService: HeCommonService,\n    private nodeService: HeNodeService\n  ) {\n    this.init();\n  }\n\n  private async init() {\n    await this.loadModels().toPromise();\n  }\n\n  private get gitUrl() {\n    return `${gitHome}/hestia-aggregation-engine/-/blob/${this.commonService.gitBranch}`;\n  }\n\n  private get rawUrl() {\n    return `${gitRawBaseUrl}/hestia-aggregation-engine/-/raw/${this.commonService.gitBranch}`;\n  }\n\n  private loadModels() {\n    this.modelsLoading = true;\n    return this.http.get<IModelLinks>(`${this.rawUrl}/model-links.json`).pipe(\n      catchError(() => of({ links: [] } as IModelLinks)),\n      map(({ links }) => {\n        this._models.next(links.map(({ path: modelPath, docPath, ...link }) => ({\n          ...link,\n          path: `${this.gitUrl}/${modelPath}`,\n          docPath: `${this.gitUrl}/${docPath}`\n        })));\n        this.modelsLoading = false;\n        this.modelsLoaded = true;\n        return links;\n      })\n    );\n  }\n\n  public get models$() {\n    return this.modelsLoading || this.modelsLoaded ? this._models.asObservable() : this.loadModels();\n  }\n\n  public models() {\n    return this.models$.pipe(take(1)).toPromise();\n  }\n\n  public model$(model: Partial<IModel>) {\n    return this.models$.pipe(\n      map(models => models.find(m => Object.keys(model).every(key => model[key] === m[key])))\n    );\n  }\n\n  public model(model: Partial<IModel>) {\n    return this.model$(model).pipe(take(1)).toPromise();\n  }\n\n  private async loadLookup(filename: string) {\n    const data = await this.nodeService.downloadLookup(filename);\n    return csvtojson({ delimiter: 'auto' }).fromString(data!);\n  }\n\n  private loadLookups() {\n    return forkJoin({\n      cropYield: from(this.loadLookup(lookups.cropYield)),\n      crop: from(this.loadLookup(TermTermType.crop))\n    }).pipe(\n      map(values => {\n        this._lookups.next(values);\n        this.lookupsLoading = false;\n        this.lookupsLoaded = true;\n        return values;\n      })\n    );\n  }\n\n  public get lookups$() {\n    return this.lookupsLoading || this.lookupsLoaded ? this._lookups.asObservable() : this.loadLookups();\n  }\n\n  public lookups() {\n    return this.lookups$.pipe(take(1)).toPromise();\n  }\n}\n"]}
|
|
89
|
+
}], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.HeNodeService }]; } });
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aggregation-engine.service.js","sourceRoot":"","sources":["../../../src/engine/aggregation-engine.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { forkJoin, from, of, ReplaySubject } from 'rxjs';\nimport { catchError, map, take } from 'rxjs/operators';\nimport { NodeType, Product, Term, TermTermType } from '@hestia-earth/schema';\nimport * as csvtojson from 'csvtojson';\n\nimport { HeNodeService } from '../node/node.service';\nimport { gitBranch, gitHome, gitRawBaseUrl } from '../common/utils';\n\nconst gitUrl = () => `${gitHome}/hestia-aggregation-engine/-/blob/${gitBranch()}`;\nconst rawUrl = () => `${gitRawBaseUrl}/hestia-aggregation-engine/-/raw/${gitBranch()}`;\n\nexport const lookups = Object.freeze({\n  cropYield: 'region-crop-cropGroupingFaostatProduction-yield',\n  cropGroupingColumn: 'cropGroupingFaostatProduction'\n});\n\nexport interface IModel {\n  type: NodeType;\n  /**\n   * Path to the implementation (code) of the model.\n   */\n  path: string;\n  /**\n   * Path to the documentation of the model.\n   */\n  docPath: string;\n  dataPath: string;\n  '@id'?: string;\n}\n\ninterface IModelLinks {\n  links: IModel[];\n}\n\nexport interface ILookups {\n  crop: any[];\n  cropYield: any[];\n}\n\nexport interface ITriggerCountryParams {\n  country: Term;\n  product: Product;\n  '@type'?: NodeType;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class HeAggregationEngineService {\n  private modelsLoading = false;\n  private modelsLoaded = false;\n  private _models = new ReplaySubject<IModel[]>(1);\n\n  private lookupsLoading = false;\n  private lookupsLoaded = false;\n  private _lookups = new ReplaySubject<ILookups>(1);\n\n  constructor(\n    protected http: HttpClient,\n    protected nodeService: HeNodeService\n  ) {\n    this.init();\n  }\n\n  private async init() {\n    await this.loadModels().toPromise();\n  }\n\n  private loadModels() {\n    this.modelsLoading = true;\n    return this.http.get<IModelLinks>(`${rawUrl()}/model-links.json`).pipe(\n      catchError(() => of({ links: [] } as IModelLinks)),\n      map(({ links }) => {\n        this._models.next(links.map(({ path: modelPath, docPath, ...link }) => ({\n          ...link,\n          path: `${gitUrl()}/${modelPath}`,\n          docPath: `${gitUrl()}/${docPath}`\n        })));\n        this.modelsLoading = false;\n        this.modelsLoaded = true;\n        return links;\n      })\n    );\n  }\n\n  public get models$() {\n    return this.modelsLoading || this.modelsLoaded ? this._models.asObservable() : this.loadModels();\n  }\n\n  public models() {\n    return this.models$.pipe(take(1)).toPromise();\n  }\n\n  public model$(model: Partial<IModel>) {\n    return this.models$.pipe(\n      map(models => models.find(m => Object.keys(model).every(key => model[key] === m[key])))\n    );\n  }\n\n  public model(model: Partial<IModel>) {\n    return this.model$(model).pipe(take(1)).toPromise();\n  }\n\n  private async loadLookup(filename: string) {\n    const data = await this.nodeService.downloadLookup(filename);\n    return csvtojson({ delimiter: 'auto' }).fromString(data!);\n  }\n\n  private loadLookups() {\n    return forkJoin({\n      cropYield: from(this.loadLookup(lookups.cropYield)),\n      crop: from(this.loadLookup(TermTermType.crop))\n    }).pipe(\n      map(values => {\n        this._lookups.next(values);\n        this.lookupsLoading = false;\n        this.lookupsLoaded = true;\n        return values;\n      })\n    );\n  }\n\n  public get lookups$() {\n    return this.lookupsLoading || this.lookupsLoaded ? this._lookups.asObservable() : this.loadLookups();\n  }\n\n  public lookups() {\n    return this.lookups$.pipe(take(1)).toPromise();\n  }\n}\n"]}
|
|
@@ -2,11 +2,12 @@ import { __awaiter, __rest } from "tslib";
|
|
|
2
2
|
import { Injectable, InjectionToken } from '@angular/core';
|
|
3
3
|
import { of, ReplaySubject } from 'rxjs';
|
|
4
4
|
import { catchError, map, take } from 'rxjs/operators';
|
|
5
|
-
import { baseUrl, toDashCase, gitHome, gitRawBaseUrl } from '../common/utils';
|
|
5
|
+
import { baseUrl, toDashCase, gitHome, gitRawBaseUrl, gitBranch } from '../common/utils';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
7
7
|
import * as i1 from "@angular/common/http";
|
|
8
|
-
import * as i2 from "../common/common.service";
|
|
9
8
|
export const HE_ORCHESTRATOR_BASE_URL = new InjectionToken('HE_ORCHESTRATOR_BASE_URL');
|
|
9
|
+
const gitUrl = () => `${gitHome}/hestia-engine-models/-/blob/${gitBranch()}`;
|
|
10
|
+
const rawUrl = () => `${gitRawBaseUrl}/hestia-engine-models/-/raw/${gitBranch()}`;
|
|
10
11
|
export const findConfigModels = ({ models }, modelId, modelKey) => models.flat().filter(({ value, key }) => modelId.startsWith(value) && (!modelKey || key === modelKey));
|
|
11
12
|
export const pathToApiDocsPath = (model, term) => [
|
|
12
13
|
baseUrl(),
|
|
@@ -18,9 +19,8 @@ export const pathToApiDocsPath = (model, term) => [
|
|
|
18
19
|
].filter(Boolean).join('-')
|
|
19
20
|
].join('/');
|
|
20
21
|
export class HeEngineService {
|
|
21
|
-
constructor(http
|
|
22
|
+
constructor(http) {
|
|
22
23
|
this.http = http;
|
|
23
|
-
this.commonService = commonService;
|
|
24
24
|
this.modelsLoading = false;
|
|
25
25
|
this.modelsLoaded = false;
|
|
26
26
|
this._models = new ReplaySubject(1);
|
|
@@ -31,18 +31,12 @@ export class HeEngineService {
|
|
|
31
31
|
yield this.loadModels().toPromise();
|
|
32
32
|
});
|
|
33
33
|
}
|
|
34
|
-
get gitUrl() {
|
|
35
|
-
return `${gitHome}/hestia-engine-models/-/blob/${this.commonService.gitBranch}`;
|
|
36
|
-
}
|
|
37
|
-
get rawUrl() {
|
|
38
|
-
return `${gitRawBaseUrl}/hestia-engine-models/-/raw/${this.commonService.gitBranch}`;
|
|
39
|
-
}
|
|
40
34
|
loadModels() {
|
|
41
35
|
this.modelsLoading = true;
|
|
42
|
-
return this.http.get(`${
|
|
36
|
+
return this.http.get(`${rawUrl()}/model-links.json`).pipe(catchError(() => of({ links: [] })), map(({ links }) => {
|
|
43
37
|
this._models.next(links.map((_a) => {
|
|
44
38
|
var { path, docPath } = _a, link = __rest(_a, ["path", "docPath"]);
|
|
45
|
-
return (Object.assign(Object.assign({}, link), { path: `${
|
|
39
|
+
return (Object.assign(Object.assign({}, link), { path: `${gitUrl()}/${path}`, docPath: `${gitUrl()}/${docPath}`, apiDocsPath: pathToApiDocsPath(link.model, link.term || link.modelKey) }));
|
|
46
40
|
}));
|
|
47
41
|
this.modelsLoading = false;
|
|
48
42
|
this.modelsLoaded = true;
|
|
@@ -67,12 +61,12 @@ export class HeEngineService {
|
|
|
67
61
|
}).toPromise();
|
|
68
62
|
}
|
|
69
63
|
}
|
|
70
|
-
HeEngineService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeEngineService, deps: [{ token: i1.HttpClient }
|
|
64
|
+
HeEngineService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeEngineService, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
71
65
|
HeEngineService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeEngineService, providedIn: 'root' });
|
|
72
66
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: HeEngineService, decorators: [{
|
|
73
67
|
type: Injectable,
|
|
74
68
|
args: [{
|
|
75
69
|
providedIn: 'root'
|
|
76
70
|
}]
|
|
77
|
-
}], ctorParameters: function () { return [{ type: i1.HttpClient }
|
|
78
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
71
|
+
}], ctorParameters: function () { return [{ type: i1.HttpClient }]; } });
|
|
72
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.service.js","sourceRoot":"","sources":["../../../src/engine/engine.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, InjectionToken } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { of, ReplaySubject } from 'rxjs';\nimport { catchError, map, take } from 'rxjs/operators';\nimport { NodeType } from '@hestia-earth/schema';\n\nimport { baseUrl, toDashCase, gitHome, gitRawBaseUrl, gitBranch } from '../common/utils';\n\nexport const HE_ORCHESTRATOR_BASE_URL = new InjectionToken<string>('HE_ORCHESTRATOR_BASE_URL');\n\nconst gitUrl = () => `${gitHome}/hestia-engine-models/-/blob/${gitBranch()}`;\nconst rawUrl = () => `${gitRawBaseUrl}/hestia-engine-models/-/raw/${gitBranch()}`;\n\nexport const findConfigModels = ({ models }: IOrchestratorConfig, modelId: string, modelKey: string) =>\n  models.flat().filter(({ value, key }) => modelId.startsWith(value) && (!modelKey || key === modelKey));\n\nexport const pathToApiDocsPath = (model: string, term?: string) =>\n  [\n    baseUrl(),\n    'docs',\n    [\n      '#hestia-calculation-models',\n      toDashCase(model),\n      toDashCase(term)\n    ].filter(Boolean).join('-')\n  ].join('/');\n\nexport interface IModel {\n  /**\n   * Path to the implementation (code) of the model.\n   */\n  path: string;\n  /**\n   * Path to the documentation of the model.\n   */\n  docPath: string;\n  /**\n   * Path to the API Documentation.\n   */\n  apiDocsPath?: string;\n  /**\n   * The name of the model used as `methodModel`.\n   */\n  model: string;\n  /**\n   * The term the model is associated with.\n   */\n  term?: string;\n  /**\n   * A key in the model if term does not exist.\n   */\n  modelKey?: string;\n}\n\ninterface IModelLinks {\n  links: IModel[];\n}\n\nexport interface IOrchestratorModelConfig {\n  key: string;\n  model: string;\n  value: string;\n}\n\nexport interface IOrchestratorConfig {\n  models: IOrchestratorModelConfig[];\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class HeEngineService {\n  private modelsLoading = false;\n  private modelsLoaded = false;\n  private _models = new ReplaySubject<IModel[]>(1);\n\n  constructor(\n    protected http: HttpClient\n  ) {\n    this.init();\n  }\n\n  private async init() {\n    await this.loadModels().toPromise();\n  }\n\n  private loadModels() {\n    this.modelsLoading = true;\n    return this.http.get<IModelLinks>(`${rawUrl()}/model-links.json`).pipe(\n      catchError(() => of({ links: [] } as IModelLinks)),\n      map(({ links }) => {\n        this._models.next(links.map(({ path, docPath, ...link }) => ({\n          ...link,\n          path: `${gitUrl()}/${path}`,\n          docPath: `${gitUrl()}/${docPath}`,\n          apiDocsPath: pathToApiDocsPath(link.model, link.term || link.modelKey)\n        })));\n        this.modelsLoading = false;\n        this.modelsLoaded = true;\n        return links;\n      })\n    );\n  }\n\n  public get models$() {\n    return this.modelsLoading || this.modelsLoaded ? this._models.asObservable() : this.loadModels();\n  }\n\n  public models() {\n    return this.models$.pipe(take(1)).toPromise();\n  }\n\n  public model$(model: Partial<IModel>) {\n    return this.models$.pipe(\n      map(models => models.find(m => Object.keys(model).every(key => model[key] === m[key])))\n    );\n  }\n\n  public model(model: Partial<IModel>) {\n    return this.model$(model).pipe(take(1)).toPromise();\n  }\n\n  public ochestratorConfig(type: NodeType) {\n    return this.http.get<IOrchestratorConfig>(`${HE_ORCHESTRATOR_BASE_URL}/${type}.json`, {\n      params: { inline: false }\n    }).toPromise();\n  }\n}\n"]}
|
|
@@ -225,7 +225,7 @@ export class ImpactAssessmentsProductsComponent {
|
|
|
225
225
|
});
|
|
226
226
|
}
|
|
227
227
|
}
|
|
228
|
-
ImpactAssessmentsProductsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsProductsComponent, deps: [{ token: i1.FormBuilder }, { token: i2.HeNodeService }, { token: i3.HeSearchService }, { token: i4.
|
|
228
|
+
ImpactAssessmentsProductsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsProductsComponent, deps: [{ token: i1.FormBuilder }, { token: i2.HeNodeService }, { token: i3.HeSearchService }, { token: i4.HeToastService }], target: i0.ɵɵFactoryTarget.Component });
|
|
229
229
|
ImpactAssessmentsProductsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.16", type: ImpactAssessmentsProductsComponent, selector: "he-impact-assessments-products", inputs: { cycles: "cycles", impactAssessments: "impactAssessments", key: "key", dataState: "dataState", enableCompare: "enableCompare", filterTermTypes: "filterTermTypes", enableFilterMethodModel: "enableFilterMethodModel" }, usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"indicators.length; else emptyTable\">\n <div class=\"columns is-variable is-2 m-0\">\n <div class=\"column is-hidden-mobile\"></div>\n <div class=\"column is-narrow\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <button class=\"button is-small\" [class.is-active]=\"selectedView === View.table\" (click)=\"selectedView = View.table\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span>Table view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"impactAssessments.length > 1\">\n <button class=\"button is-small\" [class.is-active]=\"selectedView === View.chart\" (click)=\"selectedView = View.chart\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span>Chart view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"enableBreakdown\">\n <button class=\"button is-small\" [class.is-active]=\"selectedView === View.breakdown\" (click)=\"selectedView = View.breakdown\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span>Breakdown view</span>\n </button>\n </div>\n <div class=\"control\" *ngIf=\"!isOriginal && impactAssessments.length === 1\">\n <button class=\"button is-small\" [class.is-active]=\"selectedView === View.logs\" (click)=\"showRecalculationLogs()\">\n <span class=\"icon is-small\">\n <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n </span>\n <span>Recalculations logs</span>\n </button>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n <div class=\"has-text-right mb-2\">\n <button class=\"button is-dark is-outlined is-small\" (click)=\"showDownload = true\">\n <fa-icon icon=\"download\"></fa-icon>\n <span class=\"pl-2\">Download (CSV)</span>\n </button>\n </div>\n\n <div class=\"table-container data-table-container mb-1\" *bindOnce=\"indicators\">\n <table class=\"table is-narrow data-table has-children-{{indicators.length + 1}}\">\n <thead>\n <tr>\n <th class=\"width-auto\">\n <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n <select name=\"selectedMethodModel\"\n (change)=\"updateImpacts()\" [(ngModel)]=\"selectedMethodModel\"\n >\n <option [ngValue]=\"undefined\">Filter Model</option>\n <option *ngFor=\"let term of methodModels\" [ngValue]=\"term\">{{term.name}}</option>\n </select>\n </div>\n </th>\n <th></th>\n <th *ngFor=\"let indicator of indicators\"\n [attr.title]=\"indicator.value.term.name\"\n >\n <he-node-link [node]=\"indicator.value.term\">\n <span>{{indicator.value.term.name | ellipsis:30}}</span>\n </he-node-link>\n </th>\n </tr>\n <tr>\n <th class=\"width-auto\">\n <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit:</a>\n <span class=\"pl-1\">1 kg</span>\n </th>\n <th>Product</th>\n <th *ngFor=\"let indicator of indicators\"\n [attr.title]=\"indicator.value.term.units\"\n >{{indicator.value.term.units}}</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let impactAssessment of impactAssessments; trackBy: trackById; let i = index\">\n <td class=\"width-auto\" [attr.title]=\"impactName(impactAssessment)\" [style.border-left-color]=\"itemColor(i)\">\n <label *ngIf=\"enableCompare\" class=\"is-inline-block checkbox\">\n <input type=\"checkbox\" class=\"selector\"\n (change)=\"toggleImpact(impactAssessment)\"\n [checked]=\"isSelected(impactAssessment)\"\n >\n </label>\n <he-node-link class=\"is-inline-block\" [node]=\"impactAssessment\">\n <span class=\"is-nowrap has-text-ellipsis\">{{i + 1}}. {{impactName(impactAssessment)}}</span>\n </he-node-link>\n </td>\n <td [attr.title]=\"impactAssessment.product?.name\">\n <he-node-link *ngIf=\"impactAssessment.product\" [node]=\"impactAssessment.product\">\n <span>{{impactAssessment.product.name | ellipsis:30}}</span>\n </he-node-link>\n </td>\n <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators\">\n <span *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n class=\"trigger-popover\"\n [ngbPopover]=\"details\" [autoClose]=\"'outside'\"\n triggers=\"manual\" #p=\"ngbPopover\" placement=\"left\" container=\"body\"\n (click)=\"togglePopover(p, { data: indicator.value.values[impactAssessment['@id']], impactAssessment: impactAssessment, key: key })\"\n >\n <span pointer>{{propertyValue(indicator.value.values[impactAssessment['@id']].value, key === 'impacts') | precision:3 | default:'-'}}</span>\n <he-blank-node-state class=\"ml-1\"\n [node]=\"indicator.value.values[impactAssessment['@id']].nodes[0]\"\n key=\"value\"\n [state]=\"impactAssessment.aggregated ? 'aggregated' : null\"\n ></he-blank-node-state>\n </span>\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n\n <form *ngIf=\"enableCompare\" class=\"mt-2\" [formGroup]=\"form\" (submit)=\"form.valid && addImpact()\">\n <div class=\"field has-addons\">\n <div class=\"control\">\n <span class=\"button is-small is-static\">Compare with Aggregated Impact Assessment</span>\n </div>\n <div class=\"control is-expanded\" [class.has-icons-right]=\"suggesting || loading\">\n <input class=\"input is-small\"\n placeholder=\"Search by name or id\"\n formControlName=\"search\" name=\"search\"\n\n [ngbTypeahead]=\"suggestImpactAssessment\"\n [inputFormatter]=\"formatter\"\n [resultTemplate]=\"suggestion\"\n [focusFirst]=\"true\"\n >\n <span class=\"icon is-small is-right has-text-grey-dark\" [class.is-hidden]=\"!(suggesting || loading)\">\n <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"sm\"></fa-icon>\n </span>\n </div>\n <div class=\"control\">\n <button class=\"button is-small\" type=\"submit\" [disabled]=\"suggesting || loading\">\n <fa-icon icon=\"plus\"></fa-icon>\n </button>\n </div>\n </div>\n </form>\n </div>\n\n <he-impact-assessments-indicator-breakdown-chart *ngIf=\"selectedView === View.breakdown\"\n [impactAssessment]=\"impactAssessments[0]\"\n [indicators]=\"impactAssessments[0][key]\"\n ></he-impact-assessments-indicator-breakdown-chart>\n\n <he-impact-assessments-indicators-chart *ngIf=\"impactAssessments.length > 1\" [class.is-hidden]=\"selectedView !== View.chart\"\n [key]=\"key\"\n [impactAssessments]=\"selectedImpactAssessments\"\n [filterTermTypes]=\"filterTermTypes\"\n ></he-impact-assessments-indicators-chart>\n\n <he-impact-assessments-products-logs *ngIf=\"selectedView === View.logs && !isOriginal\"\n [key]=\"key\"\n [impactAssessment]=\"impactAssessments[0]\"\n [filterTermTypes]=\"filterTermTypes\"\n [originalValues]=\"originalValues[0][key]\"\n [recalculatedValues]=\"impactAssessments[0][key]\"\n ></he-impact-assessments-products-logs>\n</ng-container>\n\n<he-node-csv-export-confirm *ngIf=\"showDownload\"\n [nodes]=\"impactAssessments\" [filename]=\"'impact-' + key + '.csv'\" [isUpload]=\"false\"\n [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + key + '.']\"\n (closed)=\"showDownload = false\"\n></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n <div class=\"panel-block\">\n <span>No data</span>\n </div>\n</ng-template>\n\n<ng-template #emptyValue>\n <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\" let-key=\"key\">\n <p *bindOnce=\"node\">\n <b>\n <span *ngIf=\"data.cycle\">{{cycleLabel(node.cycle)}}</span>\n <span *ngIf=\"!data.cycle\">{{data.name}}</span>\n </b>\n </p>\n <he-node-value-details\n [data]=\"data\" [nodeType]=\"node['@type']\" [dataKey]=\"key\"\n ></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n <div class=\"is-block\">\n <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n </div>\n <div class=\"columns is-flex\">\n <div class=\"column\" *ngIf=\"impact.country\">\n <span class=\"pr-1 has-text-underline\">Country:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n </div>\n <div class=\"column\" *ngIf=\"impact.product\">\n <span class=\"pr-1 has-text-underline\">Product:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.product.name\" [term]=\"t\"></ngb-highlight></span>\n </div>\n <div class=\"column\" *ngIf=\"impact.endDate\">\n <span class=\"pr-1 has-text-underline\">Date:</span>\n <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n </div>\n </div>\n</ng-template>\n", styles: ["label.checkbox{width:20px}td he-node-link{width:160px}table.data-table td:first-child{border-left-width:8px}\n"], components: [{ type: i5.FaIconComponent, selector: "fa-icon", inputs: ["classes", "icon", "title", "spin", "pulse", "mask", "styles", "flip", "size", "pull", "border", "inverse", "symbol", "rotate", "fixedWidth", "transform", "a11yRole"] }, { type: i6.NodeLinkComponent, selector: "he-node-link", inputs: ["node", "showExternalLink"] }, { type: i7.BlankNodeStateComponent, selector: "he-blank-node-state", inputs: ["nodeType", "dataKey", "key", "node", "state"] }, { type: i8.BlankNodeStateNoticeComponent, selector: "he-blank-node-state-notice", inputs: ["dataState", "showDeleted"] }, { type: i9.ImpactAssessmentsIndicatorBreakdownChartComponent, selector: "he-impact-assessments-indicator-breakdown-chart", inputs: ["impactAssessment", "indicators"] }, { type: i10.ImpactAssessmentsIndicatorsChartComponent, selector: "he-impact-assessments-indicators-chart", inputs: ["impactAssessments", "key", "filterTermTypes"] }, { type: i11.ImpactAssessmentsProductsLogsComponent, selector: "he-impact-assessments-products-logs", inputs: ["impactAssessment", "key", "filterTermTypes", "originalValues", "recalculatedValues"] }, { type: i12.NodeCsvExportConfirmComponent, selector: "he-node-csv-export-confirm", inputs: ["nodes", "filename", "headerKeys", "extension", "isUpload"], outputs: ["closed"] }, { type: i13.NodeValueDetailsComponent, selector: "he-node-value-details", inputs: ["data", "nodeType", "dataKey"] }, { type: i14.NgbHighlight, selector: "ngb-highlight", inputs: ["highlightClass", "result", "term"] }], directives: [{ type: i15.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i16.BindOnceDirective, selector: "[bindOnce]", inputs: ["bindOnce"] }, { 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: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i15.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i14.NgbPopover, selector: "[ngbPopover]", inputs: ["animation", "autoClose", "placement", "triggers", "container", "disablePopover", "popoverClass", "openDelay", "closeDelay", "ngbPopover", "popoverTitle"], outputs: ["shown", "hidden"], exportAs: ["ngbPopover"] }, { type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i14.NgbTypeahead, selector: "input[ngbTypeahead]", inputs: ["autocomplete", "placement", "container", "editable", "focusFirst", "showHint", "inputFormatter", "ngbTypeahead", "resultFormatter", "resultTemplate"], outputs: ["selectItem"], exportAs: ["ngbTypeahead"] }, { type: i1.FormControlName, selector: "[formControlName]", inputs: ["disabled", "formControlName", "ngModel"], outputs: ["ngModelChange"] }], pipes: { "ellipsis": i17.EllipsisPipe, "default": i18.DefaultPipe, "precision": i19.PrecisionPipe } });
|
|
230
230
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImport: i0, type: ImpactAssessmentsProductsComponent, decorators: [{
|
|
231
231
|
type: Component,
|
|
@@ -234,7 +234,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
|
|
|
234
234
|
templateUrl: './impact-assessments-products.component.html',
|
|
235
235
|
styleUrls: ['./impact-assessments-products.component.scss']
|
|
236
236
|
}]
|
|
237
|
-
}], ctorParameters: function () { return [{ type: i1.FormBuilder }, { type: i2.HeNodeService }, { type: i3.HeSearchService }, { type: i4.
|
|
237
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }, { type: i2.HeNodeService }, { type: i3.HeSearchService }, { type: i4.HeToastService }]; }, propDecorators: { cycles: [{
|
|
238
238
|
type: Input
|
|
239
239
|
}], impactAssessments: [{
|
|
240
240
|
type: Input
|
|
@@ -249,4 +249,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
|
|
|
249
249
|
}], enableFilterMethodModel: [{
|
|
250
250
|
type: Input
|
|
251
251
|
}] } });
|
|
252
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-products.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.ts","../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { FormBuilder, Validators } from '@angular/forms';\nimport { Observable, of } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, switchMap, tap } from 'rxjs/operators';\nimport { DataState } from '@hestia-earth/api';\nimport {\n  NodeType, ICycleJSONLD, IImpactAssessmentJSONLD, Indicator, Cycle, TermTermType, Term\n} from '@hestia-earth/schema';\nconst orderBy = require('lodash.orderby');\nimport { unique } from '@hestia-earth/utils';\n\nimport { HeNodeService } from '../../node/node.service';\nimport {\n  matchType, matchExactQuery, matchPhraseQuery, matchPhrasePrefixQuery, matchBoolPrefixQuery, matchAggregatedQuery\n} from '../../search/search.model';\nimport { HeSearchService } from '../../search/search.service';\nimport { ToastService } from '../../common/toast.service';\nimport { propertyValue, groupNodesByTerm, itemColor, IGroupedKeys, grouppedKeys, baseUrl } from '../../common/utils';\n\nconst MIN_TYPEAHEAD_LENGTH = 1;\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  breakdown = 'breakdown',\n  logs = 'logs'\n}\n\n@Component({\n  selector: 'he-impact-assessments-products',\n  templateUrl: './impact-assessments-products.component.html',\n  styleUrls: ['./impact-assessments-products.component.scss']\n})\nexport class ImpactAssessmentsProductsComponent implements OnChanges {\n  private dataStateValues: {\n    [dataState in DataState]?: IImpactAssessmentJSONLD[];\n  } = {};\n  @Input()\n  private cycles?: ICycleJSONLD[];\n\n  @Input()\n  public impactAssessments: IImpactAssessmentJSONLD[] = [];\n  @Input()\n  public key: 'impacts' | 'emissionsResourceUse' = 'impacts';\n  @Input()\n  public dataState?: DataState;\n  @Input()\n  public enableCompare = true;\n  @Input()\n  public filterTermTypes?: TermTermType[];\n  @Input()\n  public enableFilterMethodModel = false;\n\n  public propertyValue = propertyValue;\n  public itemColor = itemColor;\n  public baseUrl = baseUrl();\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n\n  public methodModels: Term[] = [];\n  public selectedMethodModel?: Term;\n\n  public indicators: IGroupedKeys<Indicator>[] = [];\n\n  // Adding impacts to compare\n  public form = this.formBuilder.group({\n    search: ['', Validators.required]\n  });\n  public selectedImpactAssessments: IImpactAssessmentJSONLD[] = [];\n  public loading = false;\n  public suggesting = false;\n  public formatter = ({ '@id': id }) => id;\n  public suggestImpactAssessment = (text$: Observable<string>) =>\n    text$.pipe(\n      debounceTime(300),\n      distinctUntilChanged(),\n      tap(() => this.suggesting = true),\n      switchMap(term => this.suggest(term)),\n      tap(() => this.suggesting = false)\n    );\n\n  constructor(\n    private formBuilder: FormBuilder,\n    private nodeService: HeNodeService,\n    private searchService: HeSearchService,\n    private toastService: ToastService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('impactAssessments' in changes) {\n      this.dataStateValues[this.dataState!] = changes.impactAssessments.currentValue.slice();\n      this.selectAllImpacts();\n      return this.update();\n    }\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n      return this.updateDataState();\n    }\n  }\n\n  public trackById(_index: number, item: IImpactAssessmentJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  public get originalValues() {\n    return (this.isOriginal ? null : this.dataStateValues[DataState.original]) || [];\n  }\n\n  private termAllowed(term?: Term) {\n    return !this.filterTermTypes?.length || (this.filterTermTypes || []).includes(term?.termType!);\n  }\n\n  private update() {\n    this.updateImpacts();\n    this.enableFilterMethodModel && this.updateMethodModels();\n  }\n\n  private updateMethodModels() {\n    const methodModels = this.impactAssessments\n      .flatMap(impact => (impact[this.key] || []).map(v => v.methodModel))\n      .filter(Boolean);\n    this.methodModels = unique(methodModels as Term[]);\n  }\n\n  public updateImpacts() {\n    const indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      this.impactAssessments, this.key, this.originalValues, this.selectedMethodModel\n    );\n    this.indicators = orderBy(grouppedKeys(indicatorPerImpactAssessment), ['value.methodTierOrder', 'key'], ['asc', 'asc'])\n      .filter(({ value }) => this.termAllowed(value?.term));\n  }\n\n  private async loadDataState(dataState: DataState) {\n    this.dataStateValues[dataState] = this.dataStateValues[dataState] || await Promise.all(\n      this.impactAssessments.map(node =>\n        this.nodeService.get<IImpactAssessmentJSONLD>({\n          ...node,\n          dataState\n        })\n      )\n    );\n  }\n\n  private async updateDataState() {\n    await this.loadDataState(this.dataState!);\n    this.impactAssessments = this.dataStateValues[this.dataState!]!.slice();\n    return this.update();\n  }\n\n  public togglePopover(popover, context) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public impactName(impact: IImpactAssessmentJSONLD) {\n    return impact.name || (impact.cycle ? this.cycleLabel(impact.cycle) : impact['@id']) || (impact as any).id;\n  }\n\n  public cycleLabel({ '@id': id, name }: any) {\n    return name || (this.cycles || []).find(v => v['@id'] === id)?.name;\n  }\n\n  public get enableBreakdown() {\n    return !this.isOriginal && this.impactAssessments.length === 1 && this.key === 'impacts';\n  }\n\n  // Recalculation logs\n\n  public async showRecalculationLogs() {\n    this.loading = true;\n    // make sure original data is also loaded\n    await this.loadDataState(DataState.original);\n    this.loading = false;\n    this.selectedView = View.logs;\n  }\n\n  // Compare\n\n  private selectAllImpacts() {\n    this.selectedImpactAssessments = this.impactAssessments.slice();\n  }\n\n  private unselectAllImpacts() {\n    this.selectedImpactAssessments = [];\n  }\n\n  public toggleAllImpacts() {\n    return this.selectedImpactAssessments.length ? this.selectAllImpacts() : this.unselectAllImpacts();\n  }\n\n  private selectImpact(impact: IImpactAssessmentJSONLD) {\n    this.selectedImpactAssessments = [\n      ...this.selectedImpactAssessments,\n      impact\n    ];\n  }\n\n  private unselectImpact(index: number) {\n    this.selectedImpactAssessments.splice(index, 1);\n    this.selectedImpactAssessments = this.selectedImpactAssessments.slice();\n  }\n\n  public toggleImpact(impact: IImpactAssessmentJSONLD) {\n    const index = this.selectedImpactAssessments.indexOf(impact);\n    return index >= 0 ? this.unselectImpact(index) : this.selectImpact(impact);\n  }\n\n  public isSelected(impact: IImpactAssessmentJSONLD) {\n    return this.selectedImpactAssessments.includes(impact);\n  }\n\n  private suggest(term: string) {\n    return term.length < MIN_TYPEAHEAD_LENGTH ?\n      of([]) :\n      this.searchService.suggest(term, NodeType.ImpactAssessment, [], {\n        bool: {\n          must: [\n            matchType(NodeType.ImpactAssessment),\n            matchAggregatedQuery\n          ],\n          must_not: this.impactAssessments.map(({ '@id': id }) => matchExactQuery('@id', id)),\n          should: [\n            matchPhraseQuery(term, 100, '@id', 'name', 'cycle.name'),\n            matchPhrasePrefixQuery(term, 20, '@id', 'name', 'cycle.name'),\n            matchBoolPrefixQuery(term, 10, '@id', 'name', 'cycle.name')\n          ],\n          minimum_should_match: 1\n        }\n      }, 10, ['cycle.name', 'country.name', 'product.name', 'endDate']);\n  }\n\n  private async loadImpact(id: string) {\n    const impact = await this.nodeService.get<IImpactAssessmentJSONLD>({\n      '@type': NodeType.ImpactAssessment,\n      '@id': id,\n      dataState: DataState.recalculated\n    }, false);\n    return impact['@id'] ? {\n      ...impact,\n      cycle: await this.fetchCycle(impact)\n    } : null;\n  }\n\n  private fetchCycle({ cycle }: IImpactAssessmentJSONLD) {\n    return !cycle || cycle.name ? cycle: this.nodeService.get<Cycle>({\n      '@type': NodeType.Cycle,\n      '@id': cycle['@id']\n    });\n  }\n\n  public async addImpact() {\n    this.loading = true;\n    const { search: { '@id': id, name } } = this.form.value;\n    const impact = await this.loadImpact(id);\n    if (impact) {\n      this.impactAssessments = [\n        ...this.impactAssessments,\n        impact\n      ];\n      this.selectImpact(impact);\n    }\n    else {\n      this.toastService.error(`Error while loading ${name || id}`);\n    }\n\n    this.loading = false;\n\n    this.form.get('search')?.setValue('');\n    this.form.updateValueAndValidity();\n    return this.update();\n  }\n}\n","<ng-container *ngIf=\"indicators.length; else emptyTable\">\n  <div class=\"columns is-variable is-2 m-0\">\n    <div class=\"column is-hidden-mobile\"></div>\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.table\" (click)=\"selectedView = View.table\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Table view</span>\n          </button>\n        </div>\n        <div class=\"control\" *ngIf=\"impactAssessments.length > 1\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.chart\" (click)=\"selectedView = View.chart\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Chart view</span>\n          </button>\n        </div>\n        <div class=\"control\" *ngIf=\"enableBreakdown\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.breakdown\" (click)=\"selectedView = View.breakdown\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Breakdown view</span>\n          </button>\n        </div>\n        <div class=\"control\" *ngIf=\"!isOriginal && impactAssessments.length === 1\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.logs\" (click)=\"showRecalculationLogs()\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Recalculations logs</span>\n          </button>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n    <div class=\"has-text-right mb-2\">\n      <button class=\"button is-dark is-outlined is-small\" (click)=\"showDownload = true\">\n        <fa-icon icon=\"download\"></fa-icon>\n        <span class=\"pl-2\">Download (CSV)</span>\n      </button>\n    </div>\n\n    <div class=\"table-container data-table-container mb-1\" *bindOnce=\"indicators\">\n      <table class=\"table is-narrow data-table has-children-{{indicators.length + 1}}\">\n        <thead>\n          <tr>\n            <th class=\"width-auto\">\n              <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n                <select name=\"selectedMethodModel\"\n                  (change)=\"updateImpacts()\" [(ngModel)]=\"selectedMethodModel\"\n                >\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  <option *ngFor=\"let term of methodModels\" [ngValue]=\"term\">{{term.name}}</option>\n                </select>\n              </div>\n            </th>\n            <th></th>\n            <th *ngFor=\"let indicator of indicators\"\n              [attr.title]=\"indicator.value.term.name\"\n            >\n              <he-node-link [node]=\"indicator.value.term\">\n                <span>{{indicator.value.term.name | ellipsis:30}}</span>\n              </he-node-link>\n            </th>\n          </tr>\n          <tr>\n            <th class=\"width-auto\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit:</a>\n              <span class=\"pl-1\">1 kg</span>\n            </th>\n            <th>Product</th>\n            <th *ngFor=\"let indicator of indicators\"\n              [attr.title]=\"indicator.value.term.units\"\n            >{{indicator.value.term.units}}</th>\n          </tr>\n        </thead>\n        <tbody>\n          <tr *ngFor=\"let impactAssessment of impactAssessments; trackBy: trackById; let i = index\">\n            <td class=\"width-auto\" [attr.title]=\"impactName(impactAssessment)\" [style.border-left-color]=\"itemColor(i)\">\n              <label *ngIf=\"enableCompare\" class=\"is-inline-block checkbox\">\n                <input type=\"checkbox\" class=\"selector\"\n                  (change)=\"toggleImpact(impactAssessment)\"\n                  [checked]=\"isSelected(impactAssessment)\"\n                >\n              </label>\n              <he-node-link class=\"is-inline-block\" [node]=\"impactAssessment\">\n                <span class=\"is-nowrap has-text-ellipsis\">{{i + 1}}. {{impactName(impactAssessment)}}</span>\n              </he-node-link>\n            </td>\n            <td [attr.title]=\"impactAssessment.product?.name\">\n              <he-node-link *ngIf=\"impactAssessment.product\" [node]=\"impactAssessment.product\">\n                <span>{{impactAssessment.product.name | ellipsis:30}}</span>\n              </he-node-link>\n            </td>\n            <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators\">\n              <span *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n                class=\"trigger-popover\"\n                [ngbPopover]=\"details\" [autoClose]=\"'outside'\"\n                triggers=\"manual\" #p=\"ngbPopover\" placement=\"left\" container=\"body\"\n                (click)=\"togglePopover(p, { data: indicator.value.values[impactAssessment['@id']], impactAssessment: impactAssessment, key: key })\"\n              >\n                <span pointer>{{propertyValue(indicator.value.values[impactAssessment['@id']].value, key === 'impacts') | precision:3 | default:'-'}}</span>\n                <he-blank-node-state class=\"ml-1\"\n                  [node]=\"indicator.value.values[impactAssessment['@id']].nodes[0]\"\n                  key=\"value\"\n                  [state]=\"impactAssessment.aggregated ? 'aggregated' : null\"\n                ></he-blank-node-state>\n              </span>\n            </td>\n          </tr>\n        </tbody>\n      </table>\n    </div>\n\n    <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n\n    <form *ngIf=\"enableCompare\" class=\"mt-2\" [formGroup]=\"form\" (submit)=\"form.valid && addImpact()\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static\">Compare with Aggregated Impact Assessment</span>\n        </div>\n        <div class=\"control is-expanded\" [class.has-icons-right]=\"suggesting || loading\">\n          <input class=\"input is-small\"\n            placeholder=\"Search by name or id\"\n            formControlName=\"search\" name=\"search\"\n\n            [ngbTypeahead]=\"suggestImpactAssessment\"\n            [inputFormatter]=\"formatter\"\n            [resultTemplate]=\"suggestion\"\n            [focusFirst]=\"true\"\n          >\n          <span class=\"icon is-small is-right has-text-grey-dark\" [class.is-hidden]=\"!(suggesting || loading)\">\n            <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"sm\"></fa-icon>\n          </span>\n        </div>\n        <div class=\"control\">\n          <button class=\"button is-small\" type=\"submit\" [disabled]=\"suggesting || loading\">\n            <fa-icon icon=\"plus\"></fa-icon>\n          </button>\n        </div>\n      </div>\n    </form>\n  </div>\n\n  <he-impact-assessments-indicator-breakdown-chart *ngIf=\"selectedView === View.breakdown\"\n    [impactAssessment]=\"impactAssessments[0]\"\n    [indicators]=\"impactAssessments[0][key]\"\n  ></he-impact-assessments-indicator-breakdown-chart>\n\n  <he-impact-assessments-indicators-chart *ngIf=\"impactAssessments.length > 1\" [class.is-hidden]=\"selectedView !== View.chart\"\n    [key]=\"key\"\n    [impactAssessments]=\"selectedImpactAssessments\"\n    [filterTermTypes]=\"filterTermTypes\"\n  ></he-impact-assessments-indicators-chart>\n\n  <he-impact-assessments-products-logs *ngIf=\"selectedView === View.logs && !isOriginal\"\n    [key]=\"key\"\n    [impactAssessment]=\"impactAssessments[0]\"\n    [filterTermTypes]=\"filterTermTypes\"\n    [originalValues]=\"originalValues[0][key]\"\n    [recalculatedValues]=\"impactAssessments[0][key]\"\n  ></he-impact-assessments-products-logs>\n</ng-container>\n\n<he-node-csv-export-confirm *ngIf=\"showDownload\"\n  [nodes]=\"impactAssessments\" [filename]=\"'impact-' + key + '.csv'\" [isUpload]=\"false\"\n  [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + key + '.']\"\n  (closed)=\"showDownload = false\"\n></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n  <div class=\"panel-block\">\n    <span>No data</span>\n  </div>\n</ng-template>\n\n<ng-template #emptyValue>\n  <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\" let-key=\"key\">\n  <p *bindOnce=\"node\">\n    <b>\n      <span *ngIf=\"data.cycle\">{{cycleLabel(node.cycle)}}</span>\n      <span *ngIf=\"!data.cycle\">{{data.name}}</span>\n    </b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\" [nodeType]=\"node['@type']\" [dataKey]=\"key\"\n  ></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n  <div class=\"is-block\">\n    <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n  </div>\n  <div class=\"columns is-flex\">\n    <div class=\"column\" *ngIf=\"impact.country\">\n      <span class=\"pr-1 has-text-underline\">Country:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.product\">\n      <span class=\"pr-1 has-text-underline\">Product:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.product.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.endDate\">\n      <span class=\"pr-1 has-text-underline\">Date:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
252
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"impact-assessments-products.component.js","sourceRoot":"","sources":["../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.ts","../../../../src/impact-assessments/impact-assessments-products/impact-assessments-products.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport { FormBuilder, Validators } from '@angular/forms';\nimport { Observable, of } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, switchMap, tap } from 'rxjs/operators';\nimport { DataState } from '@hestia-earth/api';\nimport {\n  NodeType, ICycleJSONLD, IImpactAssessmentJSONLD, Indicator, Cycle, TermTermType, Term\n} from '@hestia-earth/schema';\nconst orderBy = require('lodash.orderby');\nimport { unique } from '@hestia-earth/utils';\n\nimport { HeNodeService } from '../../node/node.service';\nimport {\n  matchType, matchExactQuery, matchPhraseQuery, matchPhrasePrefixQuery, matchBoolPrefixQuery, matchAggregatedQuery\n} from '../../search/search.model';\nimport { HeSearchService } from '../../search/search.service';\nimport { HeToastService } from '../../common/toast.service';\nimport { propertyValue, groupNodesByTerm, itemColor, IGroupedKeys, grouppedKeys, baseUrl } from '../../common/utils';\n\nconst MIN_TYPEAHEAD_LENGTH = 1;\n\nenum View {\n  table = 'table',\n  chart = 'chart',\n  breakdown = 'breakdown',\n  logs = 'logs'\n}\n\n@Component({\n  selector: 'he-impact-assessments-products',\n  templateUrl: './impact-assessments-products.component.html',\n  styleUrls: ['./impact-assessments-products.component.scss']\n})\nexport class ImpactAssessmentsProductsComponent implements OnChanges {\n  private dataStateValues: {\n    [dataState in DataState]?: IImpactAssessmentJSONLD[];\n  } = {};\n  @Input()\n  private cycles?: ICycleJSONLD[];\n\n  @Input()\n  public impactAssessments: IImpactAssessmentJSONLD[] = [];\n  @Input()\n  public key: 'impacts' | 'emissionsResourceUse' = 'impacts';\n  @Input()\n  public dataState?: DataState;\n  @Input()\n  public enableCompare = true;\n  @Input()\n  public filterTermTypes?: TermTermType[];\n  @Input()\n  public enableFilterMethodModel = false;\n\n  public propertyValue = propertyValue;\n  public itemColor = itemColor;\n  public baseUrl = baseUrl();\n  public showDownload = false;\n  public View = View;\n  public selectedView = View.table;\n\n  public methodModels: Term[] = [];\n  public selectedMethodModel?: Term;\n\n  public indicators: IGroupedKeys<Indicator>[] = [];\n\n  // Adding impacts to compare\n  public form = this.formBuilder.group({\n    search: ['', Validators.required]\n  });\n  public selectedImpactAssessments: IImpactAssessmentJSONLD[] = [];\n  public loading = false;\n  public suggesting = false;\n  public formatter = ({ '@id': id }) => id;\n  public suggestImpactAssessment = (text$: Observable<string>) =>\n    text$.pipe(\n      debounceTime(300),\n      distinctUntilChanged(),\n      tap(() => this.suggesting = true),\n      switchMap(term => this.suggest(term)),\n      tap(() => this.suggesting = false)\n    );\n\n  constructor(\n    private formBuilder: FormBuilder,\n    private nodeService: HeNodeService,\n    private searchService: HeSearchService,\n    private toastService: HeToastService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if ('impactAssessments' in changes) {\n      this.dataStateValues[this.dataState!] = changes.impactAssessments.currentValue.slice();\n      this.selectAllImpacts();\n      return this.update();\n    }\n    if ('dataState' in changes) {\n      this.selectedView = View.table;\n      return this.updateDataState();\n    }\n  }\n\n  public trackById(_index: number, item: IImpactAssessmentJSONLD) {\n    return item['@id'];\n  }\n\n  public get isOriginal() {\n    return this.dataState === DataState.original;\n  }\n\n  public get originalValues() {\n    return (this.isOriginal ? null : this.dataStateValues[DataState.original]) || [];\n  }\n\n  private termAllowed(term?: Term) {\n    return !this.filterTermTypes?.length || (this.filterTermTypes || []).includes(term?.termType!);\n  }\n\n  private update() {\n    this.updateImpacts();\n    this.enableFilterMethodModel && this.updateMethodModels();\n  }\n\n  private updateMethodModels() {\n    const methodModels = this.impactAssessments\n      .flatMap(impact => (impact[this.key] || []).map(v => v.methodModel))\n      .filter(Boolean);\n    this.methodModels = unique(methodModels as Term[]);\n  }\n\n  public updateImpacts() {\n    const indicatorPerImpactAssessment = groupNodesByTerm<IImpactAssessmentJSONLD, Indicator>(\n      this.impactAssessments, this.key, this.originalValues, this.selectedMethodModel\n    );\n    this.indicators = orderBy(grouppedKeys(indicatorPerImpactAssessment), ['value.methodTierOrder', 'key'], ['asc', 'asc'])\n      .filter(({ value }) => this.termAllowed(value?.term));\n  }\n\n  private async loadDataState(dataState: DataState) {\n    this.dataStateValues[dataState] = this.dataStateValues[dataState] || await Promise.all(\n      this.impactAssessments.map(node =>\n        this.nodeService.get<IImpactAssessmentJSONLD>({\n          ...node,\n          dataState\n        })\n      )\n    );\n  }\n\n  private async updateDataState() {\n    await this.loadDataState(this.dataState!);\n    this.impactAssessments = this.dataStateValues[this.dataState!]!.slice();\n    return this.update();\n  }\n\n  public togglePopover(popover, context) {\n    return popover.isOpen() ? popover.close() : popover.open(context);\n  }\n\n  public impactName(impact: IImpactAssessmentJSONLD) {\n    return impact.name || (impact.cycle ? this.cycleLabel(impact.cycle) : impact['@id']) || (impact as any).id;\n  }\n\n  public cycleLabel({ '@id': id, name }: any) {\n    return name || (this.cycles || []).find(v => v['@id'] === id)?.name;\n  }\n\n  public get enableBreakdown() {\n    return !this.isOriginal && this.impactAssessments.length === 1 && this.key === 'impacts';\n  }\n\n  // Recalculation logs\n\n  public async showRecalculationLogs() {\n    this.loading = true;\n    // make sure original data is also loaded\n    await this.loadDataState(DataState.original);\n    this.loading = false;\n    this.selectedView = View.logs;\n  }\n\n  // Compare\n\n  private selectAllImpacts() {\n    this.selectedImpactAssessments = this.impactAssessments.slice();\n  }\n\n  private unselectAllImpacts() {\n    this.selectedImpactAssessments = [];\n  }\n\n  public toggleAllImpacts() {\n    return this.selectedImpactAssessments.length ? this.selectAllImpacts() : this.unselectAllImpacts();\n  }\n\n  private selectImpact(impact: IImpactAssessmentJSONLD) {\n    this.selectedImpactAssessments = [\n      ...this.selectedImpactAssessments,\n      impact\n    ];\n  }\n\n  private unselectImpact(index: number) {\n    this.selectedImpactAssessments.splice(index, 1);\n    this.selectedImpactAssessments = this.selectedImpactAssessments.slice();\n  }\n\n  public toggleImpact(impact: IImpactAssessmentJSONLD) {\n    const index = this.selectedImpactAssessments.indexOf(impact);\n    return index >= 0 ? this.unselectImpact(index) : this.selectImpact(impact);\n  }\n\n  public isSelected(impact: IImpactAssessmentJSONLD) {\n    return this.selectedImpactAssessments.includes(impact);\n  }\n\n  private suggest(term: string) {\n    return term.length < MIN_TYPEAHEAD_LENGTH ?\n      of([]) :\n      this.searchService.suggest(term, NodeType.ImpactAssessment, [], {\n        bool: {\n          must: [\n            matchType(NodeType.ImpactAssessment),\n            matchAggregatedQuery\n          ],\n          must_not: this.impactAssessments.map(({ '@id': id }) => matchExactQuery('@id', id)),\n          should: [\n            matchPhraseQuery(term, 100, '@id', 'name', 'cycle.name'),\n            matchPhrasePrefixQuery(term, 20, '@id', 'name', 'cycle.name'),\n            matchBoolPrefixQuery(term, 10, '@id', 'name', 'cycle.name')\n          ],\n          minimum_should_match: 1\n        }\n      }, 10, ['cycle.name', 'country.name', 'product.name', 'endDate']);\n  }\n\n  private async loadImpact(id: string) {\n    const impact = await this.nodeService.get<IImpactAssessmentJSONLD>({\n      '@type': NodeType.ImpactAssessment,\n      '@id': id,\n      dataState: DataState.recalculated\n    }, false);\n    return impact['@id'] ? {\n      ...impact,\n      cycle: await this.fetchCycle(impact)\n    } : null;\n  }\n\n  private fetchCycle({ cycle }: IImpactAssessmentJSONLD) {\n    return !cycle || cycle.name ? cycle: this.nodeService.get<Cycle>({\n      '@type': NodeType.Cycle,\n      '@id': cycle['@id']\n    });\n  }\n\n  public async addImpact() {\n    this.loading = true;\n    const { search: { '@id': id, name } } = this.form.value;\n    const impact = await this.loadImpact(id);\n    if (impact) {\n      this.impactAssessments = [\n        ...this.impactAssessments,\n        impact\n      ];\n      this.selectImpact(impact);\n    }\n    else {\n      this.toastService.error(`Error while loading ${name || id}`);\n    }\n\n    this.loading = false;\n\n    this.form.get('search')?.setValue('');\n    this.form.updateValueAndValidity();\n    return this.update();\n  }\n}\n","<ng-container *ngIf=\"indicators.length; else emptyTable\">\n  <div class=\"columns is-variable is-2 m-0\">\n    <div class=\"column is-hidden-mobile\"></div>\n    <div class=\"column is-narrow\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.table\" (click)=\"selectedView = View.table\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"list\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Table view</span>\n          </button>\n        </div>\n        <div class=\"control\" *ngIf=\"impactAssessments.length > 1\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.chart\" (click)=\"selectedView = View.chart\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Chart view</span>\n          </button>\n        </div>\n        <div class=\"control\" *ngIf=\"enableBreakdown\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.breakdown\" (click)=\"selectedView = View.breakdown\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"chart-bar\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Breakdown view</span>\n          </button>\n        </div>\n        <div class=\"control\" *ngIf=\"!isOriginal && impactAssessments.length === 1\">\n          <button class=\"button is-small\" [class.is-active]=\"selectedView === View.logs\" (click)=\"showRecalculationLogs()\">\n            <span class=\"icon is-small\">\n              <fa-icon icon=\"calculator\" aria-hidden=\"true\"></fa-icon>\n            </span>\n            <span>Recalculations logs</span>\n          </button>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <div class=\"px-3 pb-3\" [class.is-hidden]=\"selectedView !== View.table\">\n    <div class=\"has-text-right mb-2\">\n      <button class=\"button is-dark is-outlined is-small\" (click)=\"showDownload = true\">\n        <fa-icon icon=\"download\"></fa-icon>\n        <span class=\"pl-2\">Download (CSV)</span>\n      </button>\n    </div>\n\n    <div class=\"table-container data-table-container mb-1\" *bindOnce=\"indicators\">\n      <table class=\"table is-narrow data-table has-children-{{indicators.length + 1}}\">\n        <thead>\n          <tr>\n            <th class=\"width-auto\">\n              <div class=\"select is-small\" *ngIf=\"enableFilterMethodModel\">\n                <select name=\"selectedMethodModel\"\n                  (change)=\"updateImpacts()\" [(ngModel)]=\"selectedMethodModel\"\n                >\n                  <option [ngValue]=\"undefined\">Filter Model</option>\n                  <option *ngFor=\"let term of methodModels\" [ngValue]=\"term\">{{term.name}}</option>\n                </select>\n              </div>\n            </th>\n            <th></th>\n            <th *ngFor=\"let indicator of indicators\"\n              [attr.title]=\"indicator.value.term.name\"\n            >\n              <he-node-link [node]=\"indicator.value.term\">\n                <span>{{indicator.value.term.name | ellipsis:30}}</span>\n              </he-node-link>\n            </th>\n          </tr>\n          <tr>\n            <th class=\"width-auto\">\n              <a [href]=\"baseUrl + '/schema/ImpactAssessment#functionalUnit'\" target=\"_blank\">Functional unit:</a>\n              <span class=\"pl-1\">1 kg</span>\n            </th>\n            <th>Product</th>\n            <th *ngFor=\"let indicator of indicators\"\n              [attr.title]=\"indicator.value.term.units\"\n            >{{indicator.value.term.units}}</th>\n          </tr>\n        </thead>\n        <tbody>\n          <tr *ngFor=\"let impactAssessment of impactAssessments; trackBy: trackById; let i = index\">\n            <td class=\"width-auto\" [attr.title]=\"impactName(impactAssessment)\" [style.border-left-color]=\"itemColor(i)\">\n              <label *ngIf=\"enableCompare\" class=\"is-inline-block checkbox\">\n                <input type=\"checkbox\" class=\"selector\"\n                  (change)=\"toggleImpact(impactAssessment)\"\n                  [checked]=\"isSelected(impactAssessment)\"\n                >\n              </label>\n              <he-node-link class=\"is-inline-block\" [node]=\"impactAssessment\">\n                <span class=\"is-nowrap has-text-ellipsis\">{{i + 1}}. {{impactName(impactAssessment)}}</span>\n              </he-node-link>\n            </td>\n            <td [attr.title]=\"impactAssessment.product?.name\">\n              <he-node-link *ngIf=\"impactAssessment.product\" [node]=\"impactAssessment.product\">\n                <span>{{impactAssessment.product.name | ellipsis:30}}</span>\n              </he-node-link>\n            </td>\n            <td class=\"is-nowrap\" *ngFor=\"let indicator of indicators\">\n              <span *ngIf=\"indicator.value.values[impactAssessment['@id']]; else emptyValue\"\n                class=\"trigger-popover\"\n                [ngbPopover]=\"details\" [autoClose]=\"'outside'\"\n                triggers=\"manual\" #p=\"ngbPopover\" placement=\"left\" container=\"body\"\n                (click)=\"togglePopover(p, { data: indicator.value.values[impactAssessment['@id']], impactAssessment: impactAssessment, key: key })\"\n              >\n                <span pointer>{{propertyValue(indicator.value.values[impactAssessment['@id']].value, key === 'impacts') | precision:3 | default:'-'}}</span>\n                <he-blank-node-state class=\"ml-1\"\n                  [node]=\"indicator.value.values[impactAssessment['@id']].nodes[0]\"\n                  key=\"value\"\n                  [state]=\"impactAssessment.aggregated ? 'aggregated' : null\"\n                ></he-blank-node-state>\n              </span>\n            </td>\n          </tr>\n        </tbody>\n      </table>\n    </div>\n\n    <he-blank-node-state-notice [dataState]=\"dataState\"></he-blank-node-state-notice>\n\n    <form *ngIf=\"enableCompare\" class=\"mt-2\" [formGroup]=\"form\" (submit)=\"form.valid && addImpact()\">\n      <div class=\"field has-addons\">\n        <div class=\"control\">\n          <span class=\"button is-small is-static\">Compare with Aggregated Impact Assessment</span>\n        </div>\n        <div class=\"control is-expanded\" [class.has-icons-right]=\"suggesting || loading\">\n          <input class=\"input is-small\"\n            placeholder=\"Search by name or id\"\n            formControlName=\"search\" name=\"search\"\n\n            [ngbTypeahead]=\"suggestImpactAssessment\"\n            [inputFormatter]=\"formatter\"\n            [resultTemplate]=\"suggestion\"\n            [focusFirst]=\"true\"\n          >\n          <span class=\"icon is-small is-right has-text-grey-dark\" [class.is-hidden]=\"!(suggesting || loading)\">\n            <fa-icon icon=\"spinner\" [pulse]=\"true\" size=\"sm\"></fa-icon>\n          </span>\n        </div>\n        <div class=\"control\">\n          <button class=\"button is-small\" type=\"submit\" [disabled]=\"suggesting || loading\">\n            <fa-icon icon=\"plus\"></fa-icon>\n          </button>\n        </div>\n      </div>\n    </form>\n  </div>\n\n  <he-impact-assessments-indicator-breakdown-chart *ngIf=\"selectedView === View.breakdown\"\n    [impactAssessment]=\"impactAssessments[0]\"\n    [indicators]=\"impactAssessments[0][key]\"\n  ></he-impact-assessments-indicator-breakdown-chart>\n\n  <he-impact-assessments-indicators-chart *ngIf=\"impactAssessments.length > 1\" [class.is-hidden]=\"selectedView !== View.chart\"\n    [key]=\"key\"\n    [impactAssessments]=\"selectedImpactAssessments\"\n    [filterTermTypes]=\"filterTermTypes\"\n  ></he-impact-assessments-indicators-chart>\n\n  <he-impact-assessments-products-logs *ngIf=\"selectedView === View.logs && !isOriginal\"\n    [key]=\"key\"\n    [impactAssessment]=\"impactAssessments[0]\"\n    [filterTermTypes]=\"filterTermTypes\"\n    [originalValues]=\"originalValues[0][key]\"\n    [recalculatedValues]=\"impactAssessments[0][key]\"\n  ></he-impact-assessments-products-logs>\n</ng-container>\n\n<he-node-csv-export-confirm *ngIf=\"showDownload\"\n  [nodes]=\"impactAssessments\" [filename]=\"'impact-' + key + '.csv'\" [isUpload]=\"false\"\n  [headerKeys]=\"['impactAssessment.id', 'impactAssessment.@id', 'impactAssessment.' + key + '.']\"\n  (closed)=\"showDownload = false\"\n></he-node-csv-export-confirm>\n\n<ng-template #emptyTable>\n  <div class=\"panel-block\">\n    <span>No data</span>\n  </div>\n</ng-template>\n\n<ng-template #emptyValue>\n  <span>-</span>\n</ng-template>\n\n<ng-template #details let-node=\"impactAssessment\" let-data=\"data\" let-key=\"key\">\n  <p *bindOnce=\"node\">\n    <b>\n      <span *ngIf=\"data.cycle\">{{cycleLabel(node.cycle)}}</span>\n      <span *ngIf=\"!data.cycle\">{{data.name}}</span>\n    </b>\n  </p>\n  <he-node-value-details\n    [data]=\"data\" [nodeType]=\"node['@type']\" [dataKey]=\"key\"\n  ></he-node-value-details>\n</ng-template>\n\n<ng-template #suggestion let-impact=\"result\" let-t=\"term\">\n  <div class=\"is-block\">\n    <ngb-highlight [result]=\"impact.name || impact.cycle.name\" [term]=\"t\"></ngb-highlight>\n  </div>\n  <div class=\"columns is-flex\">\n    <div class=\"column\" *ngIf=\"impact.country\">\n      <span class=\"pr-1 has-text-underline\">Country:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.country.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.product\">\n      <span class=\"pr-1 has-text-underline\">Product:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.product.name\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n    <div class=\"column\" *ngIf=\"impact.endDate\">\n      <span class=\"pr-1 has-text-underline\">Date:</span>\n      <span class=\"is-inline-flex\"><ngb-highlight [result]=\"impact.endDate\" [term]=\"t\"></ngb-highlight></span>\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
@@ -32,4 +32,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
|
|
|
32
32
|
providedIn: 'root'
|
|
33
33
|
}]
|
|
34
34
|
}], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.HeCommonService }]; } });
|
|
35
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVuZGVsZXkuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9tZW5kZWxleS9tZW5kZWxleS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFJM0MsT0FBTyxFQUFFLFlBQVksRUFBRSxjQUFjLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7OztBQVUvRCxNQUFNLE9BQU8sb0JBQW9CO0lBQWpDO1FBQ0UsVUFBSyxHQUFHLENBQUMsQ0FBQztRQUNWLFlBQU8sR0FBNEIsRUFBRSxDQUFDO0lBQ3hDLENBQUM7Q0FBQTtBQUtELE1BQU0sT0FBTyxpQkFBaUI7SUFDNUIsWUFDWSxJQUFnQixFQUNoQixhQUE4QjtRQUQ5QixTQUFJLEdBQUosSUFBSSxDQUFZO1FBQ2hCLGtCQUFhLEdBQWIsYUFBYSxDQUFpQjtJQUN2QyxDQUFDO0lBRUosSUFBWSxJQUFJO1FBQ2QsT0FBTyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsVUFBVSxXQUFXLENBQUM7SUFDckQsQ0FBQztJQUVELElBQUksQ0FBQyxNQUE2QjtRQUNoQyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUF1QixJQUFJLENBQUMsSUFBSSxFQUFFLEVBQUUsTUFBTSxFQUFFLFlBQVksQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDMUYsQ0FBQztJQUVELEdBQUcsQ0FBQyxFQUFVO1FBQ1osT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBZSxHQUFHLElBQUksQ0FBQyxJQUFJLElBQUksRUFBRSxFQUFFLENBQUMsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLENBQUM7SUFDN0YsQ0FBQzs7K0dBaEJVLGlCQUFpQjttSEFBakIsaUJBQWlCLGNBRmhCLE1BQU07NEZBRVAsaUJBQWlCO2tCQUg3QixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEh0dHBDbGllbnQgfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XG5pbXBvcnQgeyBCaWJsaW9ncmFwaHkgfSBmcm9tICdAaGVzdGlhLWVhcnRoL3NjaGVtYSc7XG5cbmltcG9ydCB7IGZpbHRlclBhcmFtcywgaGFuZGxlQVBJRXJyb3IgfSBmcm9tICcuLi9jb21tb24vdXRpbHMnO1xuaW1wb3J0IHsgSGVDb21tb25TZXJ2aWNlIH0gZnJvbSAnLi4vY29tbW9uL2NvbW1vbi5zZXJ2aWNlJztcblxuaW50ZXJmYWNlIElNZW5kZWxleVNlYXJjaFBhcmFtcyB7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBsaW1pdD86IG51bWJlcjtcbiAgZG9pPzogc3RyaW5nO1xuICBzY29wdXM/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBjbGFzcyBNZW5kZWxleVNlYXJjaFJlc3VsdCB7XG4gIGNvdW50ID0gMDtcbiAgcmVzdWx0czogUGFydGlhbDxCaWJsaW9ncmFwaHk+W10gPSBbXTtcbn1cblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgSGVNZW5kZWxleVNlcnZpY2Uge1xuICBjb25zdHJ1Y3RvcihcbiAgICBwcm90ZWN0ZWQgaHR0cDogSHR0cENsaWVudCxcbiAgICBwcm90ZWN0ZWQgY29tbW9uU2VydmljZTogSGVDb21tb25TZXJ2aWNlXG4gICkge31cblxuICBwcml2YXRlIGdldCBwYXRoKCkge1xuICAgIHJldHVybiBgJHt0aGlzLmNvbW1vblNlcnZpY2UuYXBpQmFzZVVybH0vbWVuZGVsZXlgO1xuICB9XG5cbiAgZmluZChwYXJhbXM6IElNZW5kZWxleVNlYXJjaFBhcmFtcykge1xuICAgIHJldHVybiB0aGlzLmh0dHAuZ2V0PE1lbmRlbGV5U2VhcmNoUmVzdWx0Pih0aGlzLnBhdGgsIHsgcGFyYW1zOiBmaWx0ZXJQYXJhbXMocGFyYW1zKSB9KTtcbiAgfVxuXG4gIGdldChpZDogc3RyaW5nKSB7XG4gICAgcmV0dXJuIHRoaXMuaHR0cC5nZXQ8QmlibGlvZ3JhcGh5PihgJHt0aGlzLnBhdGh9LyR7aWR9YCkudG9Qcm9taXNlKCkuY2F0Y2goaGFuZGxlQVBJRXJyb3IpO1xuICB9XG59XG4iXX0=
|
|
@@ -21,4 +21,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.16", ngImpo
|
|
|
21
21
|
providedIn: 'root'
|
|
22
22
|
}]
|
|
23
23
|
}], ctorParameters: function () { return [{ type: i1.HeSchemaService }]; } });
|
|
24
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS1jc3Yuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9ub2RlL25vZGUtY3N2LnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLDhCQUE4QixDQUFDOzs7QUFPdEQsTUFBTSxPQUFPLGdCQUFnQjtJQUMzQixZQUNZLGFBQThCO1FBQTlCLGtCQUFhLEdBQWIsYUFBYSxDQUFpQjtJQUN0QyxDQUFDO0lBRUMsU0FBUyxDQUFDLE9BQWU7O1lBQzdCLE9BQU8sTUFBTSxDQUFDLE1BQU0sSUFBSSxDQUFDLGFBQWEsQ0FBQyxPQUFPLEVBQUUsRUFBRSxPQUFPLENBQUMsQ0FBQztRQUM3RCxDQUFDO0tBQUE7OzhHQVBVLGdCQUFnQjtrSEFBaEIsZ0JBQWdCLGNBRmYsTUFBTTs0RkFFUCxnQkFBZ0I7a0JBSDVCLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgdG9Kc29uIH0gZnJvbSAnQGhlc3RpYS1lYXJ0aC9zY2hlbWEtY29udmVydCc7XG5cbmltcG9ydCB7IEhlU2NoZW1hU2VydmljZSB9IGZyb20gJy4uL3NjaGVtYS9zY2hlbWEuc2VydmljZSc7XG5cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIEhlTm9kZUNzdlNlcnZpY2Uge1xuICBjb25zdHJ1Y3RvcihcbiAgICBwcm90ZWN0ZWQgc2NoZW1hU2VydmljZTogSGVTY2hlbWFTZXJ2aWNlXG4gICkgeyB9XG5cbiAgYXN5bmMgY3N2VG9Kc29uKGNvbnRlbnQ6IHN0cmluZykge1xuICAgIHJldHVybiB0b0pzb24oYXdhaXQgdGhpcy5zY2hlbWFTZXJ2aWNlLnNjaGVtYXMoKSwgY29udGVudCk7XG4gIH1cbn1cbiJdfQ==
|