countries-map
Version:
World countries datamaps component for Angular
196 lines • 35.6 kB
JavaScript
import { Component, ElementRef, Input, Output, ViewChild, ChangeDetectionStrategy, EventEmitter } from '@angular/core';
import { CharErrorCode } from './chart-events.interface';
import { en as countriesEN } from '@jagomf/countrieslist';
import { scale } from 'chroma-js';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "./base-map.component";
const exists = item => typeof item !== 'undefined' && item !== null;
const countryNum = (item) => parseInt(item.value?.toString());
const countryClass = 'countryxx';
const oceanId = 'ocean';
const getStrokeWidth = (isHovered) => isHovered ? '0.2%' : '0.1%';
const getStrokeColor = (isHovered) => isHovered ? '#888' : '#afafaf';
const countryName = (countryCode) => {
return countriesEN[countryCode];
};
export class CountriesMapComponent {
get loading() {
return this.innerLoading;
}
get selectionValue() {
return this.data[this.selection.countryId].value;
}
constructor(cdRef) {
this.cdRef = cdRef;
this.countryLabel = 'Country';
this.valueLabel = 'Value';
this.showCaption = true;
this.captionBelow = true;
this.minColor = 'white';
this.maxColor = 'red';
this.backgroundColor = 'white';
this.noDataColor = '#CFCFCF';
this.exceptionColor = '#FFEE58';
this.chartReady = new EventEmitter();
this.chartError = new EventEmitter();
this.chartSelect = new EventEmitter();
this.selection = null;
this.innerLoading = true;
}
getExtraSelected(country) {
const { extra } = this.data[country];
return extra && Object.keys(extra).map(key => ({ key, val: extra[key] }));
}
selectCountry(country) {
this.selection = country ? {
countryId: country,
countryName: countryName(country),
extra: this.getExtraSelected(country)
} : null;
this.cdRef.detectChanges();
}
ngAfterViewInit() {
this.initializeMap();
}
ngOnChanges(changes) {
const changedMapValueButNotOnStart = ['data', 'minColor', 'maxColor', 'backgroundColor', 'noDataColor', 'exceptionColor']
.some(attr => changes[attr] && !changes[attr].firstChange);
if (changedMapValueButNotOnStart) {
this.initializeMap();
}
}
initializeMap() {
try {
// data is provided: might be able to paint countries in colors
if (this.data) {
// get highest value in range
const maxVal = exists(this.maxValue) ? this.maxValue : Object.values(this.data).reduce((acc, curr) => countryNum(curr) > acc || acc === null ? countryNum(curr) : acc, null);
// get lowest value in range
const minVal = exists(this.minValue) ? this.minValue : Object.values(this.data).reduce((acc, curr) => countryNum(curr) < acc || acc === null ? countryNum(curr) : acc, null);
// map values in range to colors
const valToCol = scale([this.minColor, this.maxColor]).colors((maxVal ?? 1) - (minVal ?? 0) + 1).reduce((acc, curr, i) => ({ ...acc, [i + minVal]: curr }), {});
// create local Map using provided data + calculated colors
this.mapData = Object.entries(this.data).reduce((acc, [countryId, countryVal]) => ({ ...acc,
[countryId.toLowerCase()]: {
...countryVal,
color: valToCol[countryNum(countryVal)] // value in between minVal and maxVal
|| (
// value below minVal
countryNum(countryVal) <= minVal ? valToCol[minVal] :
// value above maxVal
countryNum(countryVal) >= maxVal ? valToCol[maxVal]
// weird; should never get to here
: this.exceptionColor)
} }), {});
// no data provided: will paint plain map
}
else {
this.mapData = {};
}
const svgMap = this.mapContent.nativeElement.children[0];
svgMap.style.backgroundColor = this.backgroundColor;
svgMap.querySelectorAll(`.${countryClass}`).forEach(item => {
const mapItem = this.mapData[item.id.toLowerCase()];
const isException = mapItem ? !exists(mapItem.value) : false;
item.style.fill = mapItem ? isException ? this.exceptionColor : mapItem.color : this.noDataColor;
item.onmouseenter = this.countryHover.bind(this, item, true);
item.onmouseleave = this.countryHover.bind(this, item, false);
});
// this.innerLoading = false;
this.onChartReady();
this.cdRef.detectChanges();
}
catch (e) {
this.onCharterror({ id: CharErrorCode.loading, message: 'Could not load' });
}
}
countryHover(item, hovered) {
item.style.strokeWidth = getStrokeWidth(hovered);
item.style.stroke = getStrokeColor(hovered);
item.querySelectorAll('.landxx').forEach(i => {
i.style.strokeWidth = getStrokeWidth(hovered);
i.style.stroke = getStrokeColor(hovered);
});
}
onChartReady() {
if (this.innerLoading) {
this.innerLoading = false;
this.chartReady.emit();
}
}
onCharterror(error) {
this.chartError.emit(error);
}
onMapSelect(ev) {
const event = {
selected: false,
value: null,
extra: null,
country: null
};
let newItem;
if (ev.target?.id === oceanId) {
this.selectCountry(null);
}
else {
newItem = ev.target;
while (!newItem.classList.contains(countryClass)) {
newItem = newItem.parentNode;
}
}
const country = this.mapData[newItem?.id];
if (country) {
event.selected = true;
event.value = countryNum(country);
event.country = newItem.id.toUpperCase();
event.extra = country.extra;
this.selectCountry(event.country);
}
else {
this.selectCountry(null);
}
this.chartSelect.emit(event);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountriesMapComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CountriesMapComponent, selector: "countries-map", inputs: { data: "data", countryLabel: "countryLabel", valueLabel: "valueLabel", showCaption: "showCaption", captionBelow: "captionBelow", minValue: "minValue", maxValue: "maxValue", minColor: "minColor", maxColor: "maxColor", backgroundColor: "backgroundColor", noDataColor: "noDataColor", exceptionColor: "exceptionColor" }, outputs: { chartReady: "chartReady", chartError: "chartError", chartSelect: "chartSelect" }, viewQueries: [{ propertyName: "mapContent", first: true, predicate: ["mapContent"], descendants: true, read: ElementRef }], usesOnChanges: true, ngImport: i0, template: "@if (loading) {\n<div class=\"major-block loading\"><span class=\"text\">Loading map...</span></div>\n}\n\n<countries-map-base class=\"major-block cm-map-content\" #mapContent (click)=\"onMapSelect($event)\" [ngClass]=\"{'goes-first': captionBelow}\"/>\n\n@if (!loading && showCaption) {\n<div class=\"major-block cm-caption-container\" [ngClass]=\"{'goes-first': !captionBelow}\">\n <div class=\"cm-simple-caption\">\n <div class=\"cm-country-label\">\n @if (selection) {\n <span class=\"cm-country-name\">{{selection?.countryName}}</span>\n } @else {\n <span class=\"cm-default-label\">{{countryLabel}}</span>\n }\n </div>\n <div class=\"cm-value-label\">\n <span class=\"cm-value-text\"\n [ngClass]=\"{'has-value': selection}\">{{valueLabel}}@if (selection) {<span>: </span>}</span>\n @if (selection) {\n <span class=\"cm-value-content\">{{selectionValue}}</span>\n }\n </div>\n </div>\n @if (selection?.extra?.length > 0) {\n <div class=\"cm-extended-caption\">\n @for (item of selection?.extra; track item.key) {\n <div class=\"cm-extended-item\">\n <span class=\"cm-extended-label\">{{item.key}}</span>:\n <span class=\"cm-extended-value\">{{item.val}}</span>\n </div>\n }\n </div>\n }\n</div>\n}\n", styles: [":host{display:flex;flex-flow:column nowrap;justify-content:space-between;align-items:stretch;align-content:stretch}.major-block.loading{flex:0 1 auto;align-self:center}.major-block.loading .text{font-style:italic;font-family:sans-serif;color:gray}.major-block.cm-map-content{flex:0 1 auto}.major-block.goes-first{order:0}.major-block:not(.goes-first){order:1}.major-block.cm-caption-container{flex:0 1 auto;display:flex;flex-flow:column nowrap;justify-content:space-between}.cm-simple-caption{display:flex;flex-flow:row nowrap;justify-content:space-between}.cm-country-label{flex:0 1 auto;align-self:flex-start}.cm-value-label{flex:0 1 auto;align-self:flex-end}.cm-country-label,.cm-value-label{flex:0 1 auto}.cm-country-label .cm-country-name{font-weight:700}.cm-country-label .cm-country-name,.cm-value-label .cm-value-text{color:#333}.cm-country-label .cm-default-label,.cm-value-label .cm-value-text:not(.has-value){font-style:italic;color:#777}.cm-extended-caption{display:grid;grid-template-columns:repeat(auto-fill,minmax(120px,1fr));grid-gap:5px}.cm-extended-item{margin:5px auto}.cm-extended-item .cm-extended-label{font-weight:700}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: i2.CountriesMapBaseComponent, selector: "countries-map-base" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CountriesMapComponent, decorators: [{
type: Component,
args: [{ selector: 'countries-map', changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (loading) {\n<div class=\"major-block loading\"><span class=\"text\">Loading map...</span></div>\n}\n\n<countries-map-base class=\"major-block cm-map-content\" #mapContent (click)=\"onMapSelect($event)\" [ngClass]=\"{'goes-first': captionBelow}\"/>\n\n@if (!loading && showCaption) {\n<div class=\"major-block cm-caption-container\" [ngClass]=\"{'goes-first': !captionBelow}\">\n <div class=\"cm-simple-caption\">\n <div class=\"cm-country-label\">\n @if (selection) {\n <span class=\"cm-country-name\">{{selection?.countryName}}</span>\n } @else {\n <span class=\"cm-default-label\">{{countryLabel}}</span>\n }\n </div>\n <div class=\"cm-value-label\">\n <span class=\"cm-value-text\"\n [ngClass]=\"{'has-value': selection}\">{{valueLabel}}@if (selection) {<span>: </span>}</span>\n @if (selection) {\n <span class=\"cm-value-content\">{{selectionValue}}</span>\n }\n </div>\n </div>\n @if (selection?.extra?.length > 0) {\n <div class=\"cm-extended-caption\">\n @for (item of selection?.extra; track item.key) {\n <div class=\"cm-extended-item\">\n <span class=\"cm-extended-label\">{{item.key}}</span>:\n <span class=\"cm-extended-value\">{{item.val}}</span>\n </div>\n }\n </div>\n }\n</div>\n}\n", styles: [":host{display:flex;flex-flow:column nowrap;justify-content:space-between;align-items:stretch;align-content:stretch}.major-block.loading{flex:0 1 auto;align-self:center}.major-block.loading .text{font-style:italic;font-family:sans-serif;color:gray}.major-block.cm-map-content{flex:0 1 auto}.major-block.goes-first{order:0}.major-block:not(.goes-first){order:1}.major-block.cm-caption-container{flex:0 1 auto;display:flex;flex-flow:column nowrap;justify-content:space-between}.cm-simple-caption{display:flex;flex-flow:row nowrap;justify-content:space-between}.cm-country-label{flex:0 1 auto;align-self:flex-start}.cm-value-label{flex:0 1 auto;align-self:flex-end}.cm-country-label,.cm-value-label{flex:0 1 auto}.cm-country-label .cm-country-name{font-weight:700}.cm-country-label .cm-country-name,.cm-value-label .cm-value-text{color:#333}.cm-country-label .cm-default-label,.cm-value-label .cm-value-text:not(.has-value){font-style:italic;color:#777}.cm-extended-caption{display:grid;grid-template-columns:repeat(auto-fill,minmax(120px,1fr));grid-gap:5px}.cm-extended-item{margin:5px auto}.cm-extended-item .cm-extended-label{font-weight:700}\n"] }]
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { data: [{
type: Input,
args: [{ required: true }]
}], countryLabel: [{
type: Input
}], valueLabel: [{
type: Input
}], showCaption: [{
type: Input
}], captionBelow: [{
type: Input
}], minValue: [{
type: Input
}], maxValue: [{
type: Input
}], minColor: [{
type: Input
}], maxColor: [{
type: Input
}], backgroundColor: [{
type: Input
}], noDataColor: [{
type: Input
}], exceptionColor: [{
type: Input
}], chartReady: [{
type: Output
}], chartError: [{
type: Output
}], chartSelect: [{
type: Output
}], mapContent: [{
type: ViewChild,
args: ['mapContent', { static: false, read: ElementRef }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"countries-map.component.js","sourceRoot":"","sources":["../../../../projects/lib/src/lib/countries-map.component.ts","../../../../projects/lib/src/lib/countries-map.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  Input,\n  Output,\n  ViewChild,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  EventEmitter,\n  AfterViewInit,\n  OnChanges,\n  SimpleChanges\n} from '@angular/core';\nimport { CharErrorCode } from './chart-events.interface';\nimport type { ChartSelectEvent, ChartErrorEvent } from './chart-events.interface';\nimport type { CountriesData, SelectionExtra, DrawableCountries, Selection,\n  ValidExtraData, DrawableCountry, CountryData } from './data-types.interface';\nimport { en as countriesEN } from '@jagomf/countrieslist';\nimport { scale } from 'chroma-js';\n\nconst exists = item => typeof item !== 'undefined' && item !== null;\nconst countryNum = (item: CountryData) => parseInt(item.value?.toString());\n\nconst countryClass = 'countryxx';\nconst oceanId = 'ocean';\nconst getStrokeWidth = (isHovered: boolean) => isHovered ? '0.2%' : '0.1%';\nconst getStrokeColor = (isHovered: boolean) => isHovered ? '#888' : '#afafaf';\n\nconst countryName = (countryCode: string): string => {\n  return countriesEN[countryCode];\n};\n\n@Component({\n  selector: 'countries-map',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  templateUrl: './countries-map.component.html',\n  styleUrls: ['./countries-map.component.scss']\n})\nexport class CountriesMapComponent implements AfterViewInit, OnChanges {\n\n  @Input({ required: true }) data: CountriesData;\n  @Input() countryLabel = 'Country';\n  @Input() valueLabel = 'Value';\n  @Input() showCaption = true;\n  @Input() captionBelow = true;\n  @Input() minValue: number;\n  @Input() maxValue: number;\n  @Input() minColor = 'white';\n  @Input() maxColor = 'red';\n  @Input() backgroundColor = 'white';\n  @Input() noDataColor = '#CFCFCF';\n  @Input() exceptionColor = '#FFEE58';\n\n  @Output() private readonly chartReady = new EventEmitter<void>();\n  @Output() private readonly chartError = new EventEmitter<ChartErrorEvent>();\n  @Output() private readonly chartSelect = new EventEmitter<ChartSelectEvent>();\n\n  @ViewChild('mapContent', { static: false, read: ElementRef }) private readonly mapContent: ElementRef<HTMLElement>;\n\n  mapData: DrawableCountries;\n  selection: Selection | null = null;\n\n  private innerLoading = true;\n  get loading(): boolean {\n    return this.innerLoading;\n  }\n\n  get selectionValue(): ValidExtraData {\n    return this.data[this.selection.countryId].value;\n  }\n\n  constructor(\n    private readonly cdRef: ChangeDetectorRef,\n  ) { }\n\n  private getExtraSelected(country: string): SelectionExtra[] | null {\n    const { extra } = this.data[country];\n    return extra && Object.keys(extra).map(key => ({ key, val: extra[key] }));\n  }\n\n  private selectCountry(country?: string): void {\n    this.selection = country ? {\n      countryId: country,\n      countryName: countryName(country),\n      extra: this.getExtraSelected(country)\n    } : null;\n    this.cdRef.detectChanges();\n  }\n\n  ngAfterViewInit(): void {\n    this.initializeMap();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const changedMapValueButNotOnStart = ['data', 'minColor', 'maxColor', 'backgroundColor', 'noDataColor', 'exceptionColor']\n      .some(attr => changes[attr] && !changes[attr].firstChange);\n\n    if (changedMapValueButNotOnStart) {\n      this.initializeMap();\n    }\n  }\n\n  private initializeMap(): void {\n    try {\n      // data is provided: might be able to paint countries in colors\n      if (this.data) {\n        // get highest value in range\n        const maxVal = exists(this.maxValue) ? this.maxValue : Object.values(this.data).reduce(\n          (acc, curr) => countryNum(curr) > acc || acc === null ? countryNum(curr) : acc, null as number\n        );\n        // get lowest value in range\n        const minVal = exists(this.minValue) ? this.minValue : Object.values(this.data).reduce(\n          (acc, curr) => countryNum(curr) < acc || acc === null ? countryNum(curr) : acc, null as number\n        );\n\n        // map values in range to colors\n        const valToCol = scale([this.minColor, this.maxColor]).colors((maxVal ?? 1) - (minVal ?? 0) + 1).reduce((acc, curr, i) =>\n          ({ ...acc, [i + minVal]: curr }), {} as { [key: number]: string }\n        );\n\n        // create local Map using provided data + calculated colors\n        this.mapData = Object.entries(this.data).reduce((acc, [ countryId, countryVal ]) =>\n          ({ ...acc,\n            [countryId.toLowerCase()]: {\n              ...countryVal,\n              color: valToCol[countryNum(countryVal)] // value in between minVal and maxVal\n                || (\n                  // value below minVal\n                  countryNum(countryVal) <= minVal ? valToCol[minVal] :\n                  // value above maxVal\n                  countryNum(countryVal) >= maxVal ? valToCol[maxVal]\n                  // weird; should never get to here\n                    : this.exceptionColor\n                )\n            } as DrawableCountry }),\n          {} as DrawableCountries\n        );\n\n      // no data provided: will paint plain map\n      } else {\n        this.mapData = {};\n      }\n\n      const svgMap = this.mapContent.nativeElement.children[0] as SVGSVGElement;\n      svgMap.style.backgroundColor = this.backgroundColor;\n      svgMap.querySelectorAll<SVGSVGElement>(`.${countryClass}`).forEach(item => {\n        const mapItem = this.mapData[item.id.toLowerCase()];\n        const isException = mapItem ? !exists(mapItem.value) : false;\n        item.style.fill = mapItem ? isException ? this.exceptionColor : mapItem.color : this.noDataColor;\n        item.onmouseenter = this.countryHover.bind(this, item, true);\n        item.onmouseleave = this.countryHover.bind(this, item, false);\n      });\n\n      // this.innerLoading = false;\n      this.onChartReady();\n      this.cdRef.detectChanges();\n\n    } catch (e) {\n      this.onCharterror({ id: CharErrorCode.loading, message: 'Could not load' });\n    }\n  }\n\n  private countryHover(item: SVGElement, hovered: boolean): void {\n    item.style.strokeWidth = getStrokeWidth(hovered);\n    item.style.stroke = getStrokeColor(hovered);\n    item.querySelectorAll<SVGElement>('.landxx').forEach(i => {\n      i.style.strokeWidth = getStrokeWidth(hovered);\n      i.style.stroke = getStrokeColor(hovered);\n    });\n  }\n\n  private onChartReady(): void {\n    if (this.innerLoading) {\n      this.innerLoading = false;\n      this.chartReady.emit();\n    }\n  }\n\n  private onCharterror(error: ChartErrorEvent): void {\n    this.chartError.emit(error);\n  }\n\n  onMapSelect(ev: MouseEvent): void {\n    const event: ChartSelectEvent = {\n      selected: false,\n      value: null,\n      extra: null,\n      country: null\n    };\n\n    let newItem: SVGElement;\n    if ((ev.target as SVGElement)?.id === oceanId) {\n      this.selectCountry(null);\n\n    } else {\n      newItem = ev.target as SVGElement;\n      while (!newItem.classList.contains(countryClass)) {\n        newItem = newItem.parentNode as SVGElement;\n      }\n    }\n\n    const country = this.mapData[newItem?.id];\n    if (country) {\n      event.selected = true;\n      event.value = countryNum(country);\n      event.country = newItem.id.toUpperCase();\n      event.extra = country.extra;\n      this.selectCountry(event.country);\n    } else {\n      this.selectCountry(null);\n    }\n    this.chartSelect.emit(event);\n  }\n\n}\n","@if (loading) {\n<div class=\"major-block loading\"><span class=\"text\">Loading map...</span></div>\n}\n\n<countries-map-base class=\"major-block cm-map-content\" #mapContent (click)=\"onMapSelect($event)\" [ngClass]=\"{'goes-first': captionBelow}\"/>\n\n@if (!loading && showCaption) {\n<div class=\"major-block cm-caption-container\" [ngClass]=\"{'goes-first': !captionBelow}\">\n  <div class=\"cm-simple-caption\">\n    <div class=\"cm-country-label\">\n      @if (selection) {\n        <span class=\"cm-country-name\">{{selection?.countryName}}</span>\n      } @else {\n        <span class=\"cm-default-label\">{{countryLabel}}</span>\n      }\n    </div>\n    <div class=\"cm-value-label\">\n      <span class=\"cm-value-text\"\n        [ngClass]=\"{'has-value': selection}\">{{valueLabel}}@if (selection) {<span>: </span>}</span>\n      @if (selection) {\n        <span class=\"cm-value-content\">{{selectionValue}}</span>\n      }\n    </div>\n  </div>\n  @if (selection?.extra?.length > 0) {\n    <div class=\"cm-extended-caption\">\n      @for (item of selection?.extra; track item.key) {\n        <div class=\"cm-extended-item\">\n          <span class=\"cm-extended-label\">{{item.key}}</span>:\n          <span class=\"cm-extended-value\">{{item.val}}</span>\n        </div>\n      }\n    </div>\n  }\n</div>\n}\n"]}