@ecodev/natural 55.2.0 → 55.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,4 @@
1
- import { Component, Input } from '@angular/core';
1
+ import { Component, HostBinding, Input } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "../service/avatar.service";
@@ -10,10 +10,11 @@ export class NaturalAvatarComponent {
10
10
  constructor(avatarService) {
11
11
  this.avatarService = avatarService;
12
12
  this.size = 50;
13
- this.textSizeRatio = 3;
13
+ this.decorated = true;
14
+ this.textSizeRatio = 2.25;
14
15
  this.fgColor = '#FFF';
15
16
  this.borderRadius = '';
16
- this.textMaximumLength = 0;
17
+ this.textMaximumLength = 2;
17
18
  this.avatarSrc = null;
18
19
  this.avatarText = null;
19
20
  this.avatarStyle = {};
@@ -108,6 +109,8 @@ export class NaturalAvatarComponent {
108
109
  backgroundColor: this.bgColor ? this.bgColor : this.avatarService.getRandomColor(avatarValue),
109
110
  font: Math.floor(+this.size / this.textSizeRatio) + 'px Helvetica, Arial, sans-serif',
110
111
  lineHeight: this.size + 'px',
112
+ width: this.size + 'px',
113
+ height: this.size + 'px',
111
114
  };
112
115
  }
113
116
  /**
@@ -122,7 +125,7 @@ export class NaturalAvatarComponent {
122
125
  };
123
126
  }
124
127
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: NaturalAvatarComponent, deps: [{ token: i1.AvatarService }], target: i0.ɵɵFactoryTarget.Component }); }
125
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: NaturalAvatarComponent, isStandalone: true, selector: "natural-avatar", inputs: { image: "image", initials: "initials", gravatar: "gravatar", size: "size", textSizeRatio: "textSizeRatio", bgColor: "bgColor", fgColor: "fgColor", borderRadius: "borderRadius", textMaximumLength: "textMaximumLength" }, usesOnChanges: true, ngImport: i0, template: `
128
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.1.2", type: NaturalAvatarComponent, isStandalone: true, selector: "natural-avatar", inputs: { image: "image", initials: "initials", gravatar: "gravatar", size: "size", decorated: "decorated", textSizeRatio: "textSizeRatio", bgColor: "bgColor", fgColor: "fgColor", borderRadius: "borderRadius", textMaximumLength: "textMaximumLength" }, host: { properties: { "style.height.px": "this.size", "style.width.px": "this.size", "class.decorated": "this.decorated" } }, usesOnChanges: true, ngImport: i0, template: `
126
129
  <div class="avatar-container" [ngStyle]="hostStyle">
127
130
  <img
128
131
  *ngIf="avatarSrc"
@@ -134,11 +137,16 @@ export class NaturalAvatarComponent {
134
137
  class="avatar-content"
135
138
  loading="lazy"
136
139
  />
137
- <div *ngIf="avatarText" class="avatar-content" [ngStyle]="avatarStyle">
140
+ <div
141
+ *ngIf="avatarText"
142
+ class="avatar-content"
143
+ [class.natural-elevation]="decorated"
144
+ [ngStyle]="avatarStyle"
145
+ >
138
146
  {{ avatarText }}
139
147
  </div>
140
148
  </div>
141
- `, isInline: true, styles: [":host{border-radius:50%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); }
149
+ `, isInline: true, styles: [":host{display:block}:host.decorated{position:relative}:host.decorated .avatar-container:before{content:\"\";position:absolute;inset:0;border-radius:50%;background:linear-gradient(345deg,rgba(255,255,255,0) 25%,rgba(255,255,255,.33) 100%)}:host.decorated .avatar-content{text-shadow:0 1px 0 rgba(0,0,0,.6)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); }
142
150
  }
143
151
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImport: i0, type: NaturalAvatarComponent, decorators: [{
144
152
  type: Component,
@@ -154,11 +162,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
154
162
  class="avatar-content"
155
163
  loading="lazy"
156
164
  />
157
- <div *ngIf="avatarText" class="avatar-content" [ngStyle]="avatarStyle">
165
+ <div
166
+ *ngIf="avatarText"
167
+ class="avatar-content"
168
+ [class.natural-elevation]="decorated"
169
+ [ngStyle]="avatarStyle"
170
+ >
158
171
  {{ avatarText }}
159
172
  </div>
160
173
  </div>
161
- `, standalone: true, imports: [CommonModule], styles: [":host{border-radius:50%}\n"] }]
174
+ `, standalone: true, imports: [CommonModule], styles: [":host{display:block}:host.decorated{position:relative}:host.decorated .avatar-container:before{content:\"\";position:absolute;inset:0;border-radius:50%;background:linear-gradient(345deg,rgba(255,255,255,0) 25%,rgba(255,255,255,.33) 100%)}:host.decorated .avatar-content{text-shadow:0 1px 0 rgba(0,0,0,.6)}\n"] }]
162
175
  }], ctorParameters: function () { return [{ type: i1.AvatarService }]; }, propDecorators: { image: [{
163
176
  type: Input
164
177
  }], initials: [{
@@ -166,6 +179,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
166
179
  }], gravatar: [{
167
180
  type: Input
168
181
  }], size: [{
182
+ type: HostBinding,
183
+ args: ['style.height.px']
184
+ }, {
185
+ type: HostBinding,
186
+ args: ['style.width.px']
187
+ }, {
188
+ type: Input
189
+ }], decorated: [{
190
+ type: HostBinding,
191
+ args: ['class.decorated']
192
+ }, {
169
193
  type: Input
170
194
  }], textSizeRatio: [{
171
195
  type: Input
@@ -178,4 +202,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
178
202
  }], textMaximumLength: [{
179
203
  type: Input
180
204
  }] } });
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/avatar/component/avatar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {Component, Input, OnChanges, SimpleChanges} from '@angular/core';\n\nimport {Source} from '../sources/source';\nimport {AvatarService} from '../service/avatar.service';\nimport {CommonModule} from '@angular/common';\n\ntype Style = Partial<CSSStyleDeclaration>;\n\n/**\n * Show an avatar from different sources\n */\n@Component({\n    selector: 'natural-avatar',\n    styles: [\n        `\n            :host {\n                border-radius: 50%;\n            }\n        `,\n    ],\n    template: `\n        <div class=\"avatar-container\" [ngStyle]=\"hostStyle\">\n            <img\n                *ngIf=\"avatarSrc\"\n                [src]=\"avatarSrc\"\n                [width]=\"size\"\n                [height]=\"size\"\n                [ngStyle]=\"avatarStyle\"\n                (error)=\"tryNextSource()\"\n                class=\"avatar-content\"\n                loading=\"lazy\"\n            />\n            <div *ngIf=\"avatarText\" class=\"avatar-content\" [ngStyle]=\"avatarStyle\">\n                {{ avatarText }}\n            </div>\n        </div>\n    `,\n    standalone: true,\n    imports: [CommonModule],\n})\nexport class NaturalAvatarComponent implements OnChanges {\n    @Input() public image?: string | null;\n    @Input() public initials?: string | null;\n    @Input() public gravatar?: string | null;\n\n    @Input() public size = 50;\n    @Input() public textSizeRatio = 3;\n    @Input() public bgColor: string | undefined;\n    @Input() public fgColor = '#FFF';\n    @Input() public borderRadius = '';\n    @Input() public textMaximumLength = 0;\n\n    public avatarSrc: string | null = null;\n    public avatarText: string | null = null;\n    public avatarStyle: Style = {};\n    public hostStyle: Style = {};\n\n    private currentIndex = -1;\n    private sources: Source[] = [];\n\n    public constructor(private readonly avatarService: AvatarService) {}\n\n    /**\n     * Detect inputs change\n     */\n    public ngOnChanges(changes: SimpleChanges): void {\n        this.sources.length = 0;\n        for (const [propName, creator] of this.avatarService.getCreators()) {\n            if (!changes[propName]) {\n                continue;\n            }\n\n            const currentValue = changes[propName].currentValue;\n            if (currentValue && typeof currentValue === 'string') {\n                this.sources.push(new creator(currentValue));\n            }\n        }\n\n        // reinitialize the avatar component when a source property value has changed\n        // the fallback system must be re-invoked with the new values.\n        this.initializeAvatar();\n    }\n\n    /**\n     * Fetch avatar source\n     */\n    public tryNextSource(): void {\n        const previousSource = this.sources[this.currentIndex];\n        if (previousSource) {\n            this.avatarService.markSourceAsFailed(previousSource);\n        }\n\n        const source = this.findNextSource();\n        if (!source) {\n            this.clearAvatar();\n            return;\n        }\n\n        if (source.isTextual()) {\n            this.buildTextAvatar(source);\n        } else {\n            this.buildImageAvatar(source);\n        }\n    }\n\n    private findNextSource(): Source | null {\n        while (++this.currentIndex < this.sources.length) {\n            const source = this.sources[this.currentIndex];\n            if (source && !this.avatarService.sourceHasFailedBefore(source)) {\n                return source;\n            }\n        }\n\n        return null;\n    }\n\n    /**\n     * Initialize the avatar component and its fallback system\n     */\n    private initializeAvatar(): void {\n        this.currentIndex = -1;\n        if (this.sources.length > 0) {\n            this.tryNextSource();\n            this.hostStyle = {\n                width: this.size + 'px',\n                height: this.size + 'px',\n            };\n        }\n    }\n\n    private clearAvatar(): void {\n        this.avatarSrc = null;\n        this.avatarText = null;\n        this.avatarStyle = {};\n    }\n\n    private buildTextAvatar(avatarSource: Source): void {\n        this.clearAvatar();\n        this.avatarText = avatarSource.getAvatar(+this.textMaximumLength);\n        this.avatarStyle = this.getTextualStyle(avatarSource.getValue());\n    }\n\n    private buildImageAvatar(avatarSource: Source): void {\n        this.clearAvatar();\n        this.avatarSrc = avatarSource.getAvatar(+this.size);\n        this.avatarStyle = this.getImageStyle();\n    }\n\n    /**\n     * Returns initials style\n     */\n    private getTextualStyle(avatarValue: string): Style {\n        return {\n            textAlign: 'center',\n            borderRadius: this.borderRadius || '100%',\n            textTransform: 'uppercase',\n            color: this.fgColor,\n            backgroundColor: this.bgColor ? this.bgColor : this.avatarService.getRandomColor(avatarValue),\n            font: Math.floor(+this.size / this.textSizeRatio) + 'px Helvetica, Arial, sans-serif',\n            lineHeight: this.size + 'px',\n        };\n    }\n\n    /**\n     * Returns image style\n     */\n    private getImageStyle(): Style {\n        return {\n            maxWidth: '100%',\n            borderRadius: this.borderRadius || '100%',\n            width: this.size + 'px',\n            height: this.size + 'px',\n        };\n    }\n}\n"]}
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar.component.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/avatar/component/avatar.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {Component, HostBinding, Input, OnChanges, SimpleChanges} from '@angular/core';\n\nimport {Source} from '../sources/source';\nimport {AvatarService} from '../service/avatar.service';\nimport {CommonModule} from '@angular/common';\n\ntype Style = Partial<CSSStyleDeclaration>;\n\n/**\n * Show an avatar from different sources\n */\n@Component({\n    selector: 'natural-avatar',\n    styles: [\n        `\n            :host {\n                display: block;\n\n                &.decorated {\n                    position: relative;\n\n                    .avatar-container::before {\n                        content: '';\n                        position: absolute;\n                        left: 0;\n                        top: 0;\n                        right: 0;\n                        bottom: 0;\n                        border-radius: 50%;\n                        background: linear-gradient(345deg, rgba(255, 255, 255, 0) 25%, rgba(255, 255, 255, 0.33) 100%);\n                    }\n\n                    .avatar-content {\n                        text-shadow: 0 1px 0 rgba(0, 0, 0, 0.6);\n                    }\n                }\n            }\n        `,\n    ],\n    template: `\n        <div class=\"avatar-container\" [ngStyle]=\"hostStyle\">\n            <img\n                *ngIf=\"avatarSrc\"\n                [src]=\"avatarSrc\"\n                [width]=\"size\"\n                [height]=\"size\"\n                [ngStyle]=\"avatarStyle\"\n                (error)=\"tryNextSource()\"\n                class=\"avatar-content\"\n                loading=\"lazy\"\n            />\n            <div\n                *ngIf=\"avatarText\"\n                class=\"avatar-content\"\n                [class.natural-elevation]=\"decorated\"\n                [ngStyle]=\"avatarStyle\"\n            >\n                {{ avatarText }}\n            </div>\n        </div>\n    `,\n    standalone: true,\n    imports: [CommonModule],\n})\nexport class NaturalAvatarComponent implements OnChanges {\n    @Input() public image?: string | null;\n    @Input() public initials?: string | null;\n    @Input() public gravatar?: string | null;\n\n    @HostBinding('style.height.px')\n    @HostBinding('style.width.px')\n    @Input()\n    public size = 50;\n\n    @HostBinding('class.decorated')\n    @Input()\n    public decorated: boolean = true;\n\n    @Input() public textSizeRatio = 2.25;\n    @Input() public bgColor: string | undefined;\n    @Input() public fgColor = '#FFF';\n    @Input() public borderRadius = '';\n    @Input() public textMaximumLength = 2;\n\n    public avatarSrc: string | null = null;\n    public avatarText: string | null = null;\n    public avatarStyle: Style = {};\n    public hostStyle: Style = {};\n\n    private currentIndex = -1;\n    private sources: Source[] = [];\n\n    public constructor(private readonly avatarService: AvatarService) {}\n\n    /**\n     * Detect inputs change\n     */\n    public ngOnChanges(changes: SimpleChanges): void {\n        this.sources.length = 0;\n        for (const [propName, creator] of this.avatarService.getCreators()) {\n            if (!changes[propName]) {\n                continue;\n            }\n\n            const currentValue = changes[propName].currentValue;\n            if (currentValue && typeof currentValue === 'string') {\n                this.sources.push(new creator(currentValue));\n            }\n        }\n\n        // reinitialize the avatar component when a source property value has changed\n        // the fallback system must be re-invoked with the new values.\n        this.initializeAvatar();\n    }\n\n    /**\n     * Fetch avatar source\n     */\n    public tryNextSource(): void {\n        const previousSource = this.sources[this.currentIndex];\n        if (previousSource) {\n            this.avatarService.markSourceAsFailed(previousSource);\n        }\n\n        const source = this.findNextSource();\n        if (!source) {\n            this.clearAvatar();\n            return;\n        }\n\n        if (source.isTextual()) {\n            this.buildTextAvatar(source);\n        } else {\n            this.buildImageAvatar(source);\n        }\n    }\n\n    private findNextSource(): Source | null {\n        while (++this.currentIndex < this.sources.length) {\n            const source = this.sources[this.currentIndex];\n            if (source && !this.avatarService.sourceHasFailedBefore(source)) {\n                return source;\n            }\n        }\n\n        return null;\n    }\n\n    /**\n     * Initialize the avatar component and its fallback system\n     */\n    private initializeAvatar(): void {\n        this.currentIndex = -1;\n        if (this.sources.length > 0) {\n            this.tryNextSource();\n            this.hostStyle = {\n                width: this.size + 'px',\n                height: this.size + 'px',\n            };\n        }\n    }\n\n    private clearAvatar(): void {\n        this.avatarSrc = null;\n        this.avatarText = null;\n        this.avatarStyle = {};\n    }\n\n    private buildTextAvatar(avatarSource: Source): void {\n        this.clearAvatar();\n        this.avatarText = avatarSource.getAvatar(+this.textMaximumLength);\n        this.avatarStyle = this.getTextualStyle(avatarSource.getValue());\n    }\n\n    private buildImageAvatar(avatarSource: Source): void {\n        this.clearAvatar();\n        this.avatarSrc = avatarSource.getAvatar(+this.size);\n        this.avatarStyle = this.getImageStyle();\n    }\n\n    /**\n     * Returns initials style\n     */\n    private getTextualStyle(avatarValue: string): Style {\n        return {\n            textAlign: 'center',\n            borderRadius: this.borderRadius || '100%',\n            textTransform: 'uppercase',\n            color: this.fgColor,\n            backgroundColor: this.bgColor ? this.bgColor : this.avatarService.getRandomColor(avatarValue),\n            font: Math.floor(+this.size / this.textSizeRatio) + 'px Helvetica, Arial, sans-serif',\n            lineHeight: this.size + 'px',\n            width: this.size + 'px',\n            height: this.size + 'px',\n        };\n    }\n\n    /**\n     * Returns image style\n     */\n    private getImageStyle(): Style {\n        return {\n            maxWidth: '100%',\n            borderRadius: this.borderRadius || '100%',\n            width: this.size + 'px',\n            height: this.size + 'px',\n        };\n    }\n}\n"]}
@@ -18,14 +18,15 @@ export class AvatarService {
18
18
  ['initials', Initials],
19
19
  ]);
20
20
  this.avatarColors = [
21
- '#1abc9c',
22
- '#3498db',
23
- '#f1c40f',
24
- '#8e44ad',
25
- '#e74c3c',
26
- '#d35400',
27
- '#2c3e50',
28
- '#7f8c8d',
21
+ '#ff0000',
22
+ '#ff8800',
23
+ '#dabb00',
24
+ '#00c200',
25
+ '#01cbcb',
26
+ '#008cff',
27
+ '#ff00d8',
28
+ '#c800ff',
29
+ '#3b3b3b',
29
30
  ];
30
31
  this.failedSources = new Map();
31
32
  }
@@ -63,4 +64,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
63
64
  providedIn: 'root',
64
65
  }]
65
66
  }] });
66
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXZhdGFyLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9hdmF0YXIvc2VydmljZS9hdmF0YXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRXpDLE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsUUFBUSxFQUFDLE1BQU0scUJBQXFCLENBQUM7QUFDN0MsT0FBTyxFQUFDLEtBQUssRUFBQyxNQUFNLGtCQUFrQixDQUFDOztBQUd2Qzs7R0FFRztBQUlILE1BQU0sT0FBTyxhQUFhO0lBSDFCO1FBSUk7OztXQUdHO1FBQ2MsbUJBQWMsR0FBRyxJQUFJLEdBQUcsQ0FBOEM7WUFDbkYsQ0FBQyxVQUFVLEVBQUUsUUFBUSxDQUFDO1lBQ3RCLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQztZQUNoQixDQUFDLFVBQVUsRUFBRSxRQUFRLENBQUM7U0FDekIsQ0FBQyxDQUFDO1FBRWMsaUJBQVksR0FBRztZQUM1QixTQUFTO1lBQ1QsU0FBUztZQUNULFNBQVM7WUFDVCxTQUFTO1lBQ1QsU0FBUztZQUNULFNBQVM7WUFDVCxTQUFTO1lBQ1QsU0FBUztTQUNaLENBQUM7UUFFZSxrQkFBYSxHQUFHLElBQUksR0FBRyxFQUFrQixDQUFDO0tBaUM5RDtJQS9CVSxjQUFjLENBQUMsVUFBa0I7UUFDcEMsSUFBSSxDQUFDLFVBQVUsRUFBRTtZQUNiLE9BQU8sYUFBYSxDQUFDO1NBQ3hCO1FBQ0QsTUFBTSxZQUFZLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBRXpELE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUN0RSxDQUFDO0lBRU0sV0FBVztRQUNkLE9BQU8sSUFBSSxDQUFDLGNBQWMsQ0FBQyxPQUFPLEVBQUUsQ0FBQztJQUN6QyxDQUFDO0lBRU8sWUFBWSxDQUFDLE1BQWM7UUFDL0IsT0FBTyxNQUFNLENBQUMsV0FBVyxDQUFDLElBQUksR0FBRyxHQUFHLEdBQUcsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQzdELENBQUM7SUFFTSxxQkFBcUIsQ0FBQyxNQUFjO1FBQ3ZDLE9BQU8sSUFBSSxDQUFDLGFBQWEsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDO0lBQzdELENBQUM7SUFFTSxrQkFBa0IsQ0FBQyxNQUFjO1FBQ3BDLElBQUksQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDOUQsQ0FBQztJQUVPLGtCQUFrQixDQUFDLEtBQWE7UUFDcEMsT0FBTyxLQUFLO2FBQ1AsS0FBSyxDQUFDLEVBQUUsQ0FBQzthQUNULEdBQUcsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUM7YUFDbkMsTUFBTSxDQUFDLENBQUMsUUFBUSxFQUFFLE9BQU8sRUFBRSxFQUFFLENBQUMsUUFBUSxHQUFHLE9BQU8sQ0FBQyxDQUFDO0lBQzNELENBQUM7OEdBdERRLGFBQWE7a0hBQWIsYUFBYSxjQUZWLE1BQU07OzJGQUVULGFBQWE7a0JBSHpCLFVBQVU7bUJBQUM7b0JBQ1IsVUFBVSxFQUFFLE1BQU07aUJBQ3JCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtJbmplY3RhYmxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7U291cmNlLCBTb3VyY2VDcmVhdG9yfSBmcm9tICcuLi9zb3VyY2VzL3NvdXJjZSc7XG5pbXBvcnQge0dyYXZhdGFyfSBmcm9tICcuLi9zb3VyY2VzL2dyYXZhdGFyJztcbmltcG9ydCB7SW5pdGlhbHN9IGZyb20gJy4uL3NvdXJjZXMvaW5pdGlhbHMnO1xuaW1wb3J0IHtJbWFnZX0gZnJvbSAnLi4vc291cmNlcy9pbWFnZSc7XG5pbXBvcnQge05hdHVyYWxBdmF0YXJDb21wb25lbnR9IGZyb20gJy4uL2NvbXBvbmVudC9hdmF0YXIuY29tcG9uZW50JztcblxuLyoqXG4gKiBQcm92aWRlcyB1dGlsaXRpZXMgbWV0aG9kcyByZWxhdGVkIHRvIEF2YXRhciBjb21wb25lbnRcbiAqL1xuQEluamVjdGFibGUoe1xuICAgIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgQXZhdGFyU2VydmljZSB7XG4gICAgLyoqXG4gICAgICogT3JkZXJlZCBwYWlycyBvZiBwb3NzaWJsZSBzb3VyY2VzLiBGaXJzdCBpbiBsaXN0IGlzIHRoZSBoaWdoZXN0IHByaW9yaXR5LlxuICAgICAqIEFuZCBrZXkgbXVzdCBtYXRjaCBvbmUgdGhlIGlucHV0IG9mIEF2YXRhckNvbXBvbmVudC5cbiAgICAgKi9cbiAgICBwcml2YXRlIHJlYWRvbmx5IHNvdXJjZUNyZWF0b3JzID0gbmV3IE1hcDxrZXlvZiBOYXR1cmFsQXZhdGFyQ29tcG9uZW50LCBTb3VyY2VDcmVhdG9yPihbXG4gICAgICAgIFsnZ3JhdmF0YXInLCBHcmF2YXRhcl0sXG4gICAgICAgIFsnaW1hZ2UnLCBJbWFnZV0sXG4gICAgICAgIFsnaW5pdGlhbHMnLCBJbml0aWFsc10sXG4gICAgXSk7XG5cbiAgICBwcml2YXRlIHJlYWRvbmx5IGF2YXRhckNvbG9ycyA9IFtcbiAgICAgICAgJyMxYWJjOWMnLFxuICAgICAgICAnIzM0OThkYicsXG4gICAgICAgICcjZjFjNDBmJyxcbiAgICAgICAgJyM4ZTQ0YWQnLFxuICAgICAgICAnI2U3NGMzYycsXG4gICAgICAgICcjZDM1NDAwJyxcbiAgICAgICAgJyMyYzNlNTAnLFxuICAgICAgICAnIzdmOGM4ZCcsXG4gICAgXTtcblxuICAgIHByaXZhdGUgcmVhZG9ubHkgZmFpbGVkU291cmNlcyA9IG5ldyBNYXA8c3RyaW5nLCBTb3VyY2U+KCk7XG5cbiAgICBwdWJsaWMgZ2V0UmFuZG9tQ29sb3IoYXZhdGFyVGV4dDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICAgICAgaWYgKCFhdmF0YXJUZXh0KSB7XG4gICAgICAgICAgICByZXR1cm4gJ3RyYW5zcGFyZW50JztcbiAgICAgICAgfVxuICAgICAgICBjb25zdCBhc2NpaUNvZGVTdW0gPSB0aGlzLmNhbGN1bGF0ZUFzY2lpQ29kZShhdmF0YXJUZXh0KTtcblxuICAgICAgICByZXR1cm4gdGhpcy5hdmF0YXJDb2xvcnNbYXNjaWlDb2RlU3VtICUgdGhpcy5hdmF0YXJDb2xvcnMubGVuZ3RoXTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0Q3JlYXRvcnMoKTogSXRlcmFibGVJdGVyYXRvcjxba2V5b2YgTmF0dXJhbEF2YXRhckNvbXBvbmVudCwgU291cmNlQ3JlYXRvcl0+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuc291cmNlQ3JlYXRvcnMuZW50cmllcygpO1xuICAgIH1cblxuICAgIHByaXZhdGUgZ2V0U291cmNlS2V5KHNvdXJjZTogU291cmNlKTogc3RyaW5nIHtcbiAgICAgICAgcmV0dXJuIHNvdXJjZS5jb25zdHJ1Y3Rvci5uYW1lICsgJy0nICsgc291cmNlLmdldFZhbHVlKCk7XG4gICAgfVxuXG4gICAgcHVibGljIHNvdXJjZUhhc0ZhaWxlZEJlZm9yZShzb3VyY2U6IFNvdXJjZSk6IGJvb2xlYW4ge1xuICAgICAgICByZXR1cm4gdGhpcy5mYWlsZWRTb3VyY2VzLmhhcyh0aGlzLmdldFNvdXJjZUtleShzb3VyY2UpKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgbWFya1NvdXJjZUFzRmFpbGVkKHNvdXJjZTogU291cmNlKTogdm9pZCB7XG4gICAgICAgIHRoaXMuZmFpbGVkU291cmNlcy5zZXQodGhpcy5nZXRTb3VyY2VLZXkoc291cmNlKSwgc291cmNlKTtcbiAgICB9XG5cbiAgICBwcml2YXRlIGNhbGN1bGF0ZUFzY2lpQ29kZSh2YWx1ZTogc3RyaW5nKTogbnVtYmVyIHtcbiAgICAgICAgcmV0dXJuIHZhbHVlXG4gICAgICAgICAgICAuc3BsaXQoJycpXG4gICAgICAgICAgICAubWFwKGxldHRlciA9PiBsZXR0ZXIuY2hhckNvZGVBdCgwKSlcbiAgICAgICAgICAgIC5yZWR1Y2UoKHByZXZpb3VzLCBjdXJyZW50KSA9PiBwcmV2aW91cyArIGN1cnJlbnQpO1xuICAgIH1cbn1cbiJdfQ==
67
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXZhdGFyLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9hdmF0YXIvc2VydmljZS9hdmF0YXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsVUFBVSxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRXpDLE9BQU8sRUFBQyxRQUFRLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUM3QyxPQUFPLEVBQUMsUUFBUSxFQUFDLE1BQU0scUJBQXFCLENBQUM7QUFDN0MsT0FBTyxFQUFDLEtBQUssRUFBQyxNQUFNLGtCQUFrQixDQUFDOztBQUd2Qzs7R0FFRztBQUlILE1BQU0sT0FBTyxhQUFhO0lBSDFCO1FBSUk7OztXQUdHO1FBQ2MsbUJBQWMsR0FBRyxJQUFJLEdBQUcsQ0FBOEM7WUFDbkYsQ0FBQyxVQUFVLEVBQUUsUUFBUSxDQUFDO1lBQ3RCLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQztZQUNoQixDQUFDLFVBQVUsRUFBRSxRQUFRLENBQUM7U0FDekIsQ0FBQyxDQUFDO1FBRWMsaUJBQVksR0FBRztZQUM1QixTQUFTO1lBQ1QsU0FBUztZQUNULFNBQVM7WUFDVCxTQUFTO1lBQ1QsU0FBUztZQUNULFNBQVM7WUFDVCxTQUFTO1lBQ1QsU0FBUztZQUNULFNBQVM7U0FDWixDQUFDO1FBRWUsa0JBQWEsR0FBRyxJQUFJLEdBQUcsRUFBa0IsQ0FBQztLQWlDOUQ7SUEvQlUsY0FBYyxDQUFDLFVBQWtCO1FBQ3BDLElBQUksQ0FBQyxVQUFVLEVBQUU7WUFDYixPQUFPLGFBQWEsQ0FBQztTQUN4QjtRQUNELE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUV6RCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDdEUsQ0FBQztJQUVNLFdBQVc7UUFDZCxPQUFPLElBQUksQ0FBQyxjQUFjLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDekMsQ0FBQztJQUVPLFlBQVksQ0FBQyxNQUFjO1FBQy9CLE9BQU8sTUFBTSxDQUFDLFdBQVcsQ0FBQyxJQUFJLEdBQUcsR0FBRyxHQUFHLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUM3RCxDQUFDO0lBRU0scUJBQXFCLENBQUMsTUFBYztRQUN2QyxPQUFPLElBQUksQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQztJQUM3RCxDQUFDO0lBRU0sa0JBQWtCLENBQUMsTUFBYztRQUNwQyxJQUFJLENBQUMsYUFBYSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLE1BQU0sQ0FBQyxFQUFFLE1BQU0sQ0FBQyxDQUFDO0lBQzlELENBQUM7SUFFTyxrQkFBa0IsQ0FBQyxLQUFhO1FBQ3BDLE9BQU8sS0FBSzthQUNQLEtBQUssQ0FBQyxFQUFFLENBQUM7YUFDVCxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxDQUFDO2FBQ25DLE1BQU0sQ0FBQyxDQUFDLFFBQVEsRUFBRSxPQUFPLEVBQUUsRUFBRSxDQUFDLFFBQVEsR0FBRyxPQUFPLENBQUMsQ0FBQztJQUMzRCxDQUFDOzhHQXZEUSxhQUFhO2tIQUFiLGFBQWEsY0FGVixNQUFNOzsyRkFFVCxhQUFhO2tCQUh6QixVQUFVO21CQUFDO29CQUNSLFVBQVUsRUFBRSxNQUFNO2lCQUNyQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SW5qZWN0YWJsZX0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge1NvdXJjZSwgU291cmNlQ3JlYXRvcn0gZnJvbSAnLi4vc291cmNlcy9zb3VyY2UnO1xuaW1wb3J0IHtHcmF2YXRhcn0gZnJvbSAnLi4vc291cmNlcy9ncmF2YXRhcic7XG5pbXBvcnQge0luaXRpYWxzfSBmcm9tICcuLi9zb3VyY2VzL2luaXRpYWxzJztcbmltcG9ydCB7SW1hZ2V9IGZyb20gJy4uL3NvdXJjZXMvaW1hZ2UnO1xuaW1wb3J0IHtOYXR1cmFsQXZhdGFyQ29tcG9uZW50fSBmcm9tICcuLi9jb21wb25lbnQvYXZhdGFyLmNvbXBvbmVudCc7XG5cbi8qKlxuICogUHJvdmlkZXMgdXRpbGl0aWVzIG1ldGhvZHMgcmVsYXRlZCB0byBBdmF0YXIgY29tcG9uZW50XG4gKi9cbkBJbmplY3RhYmxlKHtcbiAgICBwcm92aWRlZEluOiAncm9vdCcsXG59KVxuZXhwb3J0IGNsYXNzIEF2YXRhclNlcnZpY2Uge1xuICAgIC8qKlxuICAgICAqIE9yZGVyZWQgcGFpcnMgb2YgcG9zc2libGUgc291cmNlcy4gRmlyc3QgaW4gbGlzdCBpcyB0aGUgaGlnaGVzdCBwcmlvcml0eS5cbiAgICAgKiBBbmQga2V5IG11c3QgbWF0Y2ggb25lIHRoZSBpbnB1dCBvZiBBdmF0YXJDb21wb25lbnQuXG4gICAgICovXG4gICAgcHJpdmF0ZSByZWFkb25seSBzb3VyY2VDcmVhdG9ycyA9IG5ldyBNYXA8a2V5b2YgTmF0dXJhbEF2YXRhckNvbXBvbmVudCwgU291cmNlQ3JlYXRvcj4oW1xuICAgICAgICBbJ2dyYXZhdGFyJywgR3JhdmF0YXJdLFxuICAgICAgICBbJ2ltYWdlJywgSW1hZ2VdLFxuICAgICAgICBbJ2luaXRpYWxzJywgSW5pdGlhbHNdLFxuICAgIF0pO1xuXG4gICAgcHJpdmF0ZSByZWFkb25seSBhdmF0YXJDb2xvcnMgPSBbXG4gICAgICAgICcjZmYwMDAwJyxcbiAgICAgICAgJyNmZjg4MDAnLFxuICAgICAgICAnI2RhYmIwMCcsXG4gICAgICAgICcjMDBjMjAwJyxcbiAgICAgICAgJyMwMWNiY2InLFxuICAgICAgICAnIzAwOGNmZicsXG4gICAgICAgICcjZmYwMGQ4JyxcbiAgICAgICAgJyNjODAwZmYnLFxuICAgICAgICAnIzNiM2IzYicsXG4gICAgXTtcblxuICAgIHByaXZhdGUgcmVhZG9ubHkgZmFpbGVkU291cmNlcyA9IG5ldyBNYXA8c3RyaW5nLCBTb3VyY2U+KCk7XG5cbiAgICBwdWJsaWMgZ2V0UmFuZG9tQ29sb3IoYXZhdGFyVGV4dDogc3RyaW5nKTogc3RyaW5nIHtcbiAgICAgICAgaWYgKCFhdmF0YXJUZXh0KSB7XG4gICAgICAgICAgICByZXR1cm4gJ3RyYW5zcGFyZW50JztcbiAgICAgICAgfVxuICAgICAgICBjb25zdCBhc2NpaUNvZGVTdW0gPSB0aGlzLmNhbGN1bGF0ZUFzY2lpQ29kZShhdmF0YXJUZXh0KTtcblxuICAgICAgICByZXR1cm4gdGhpcy5hdmF0YXJDb2xvcnNbYXNjaWlDb2RlU3VtICUgdGhpcy5hdmF0YXJDb2xvcnMubGVuZ3RoXTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0Q3JlYXRvcnMoKTogSXRlcmFibGVJdGVyYXRvcjxba2V5b2YgTmF0dXJhbEF2YXRhckNvbXBvbmVudCwgU291cmNlQ3JlYXRvcl0+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuc291cmNlQ3JlYXRvcnMuZW50cmllcygpO1xuICAgIH1cblxuICAgIHByaXZhdGUgZ2V0U291cmNlS2V5KHNvdXJjZTogU291cmNlKTogc3RyaW5nIHtcbiAgICAgICAgcmV0dXJuIHNvdXJjZS5jb25zdHJ1Y3Rvci5uYW1lICsgJy0nICsgc291cmNlLmdldFZhbHVlKCk7XG4gICAgfVxuXG4gICAgcHVibGljIHNvdXJjZUhhc0ZhaWxlZEJlZm9yZShzb3VyY2U6IFNvdXJjZSk6IGJvb2xlYW4ge1xuICAgICAgICByZXR1cm4gdGhpcy5mYWlsZWRTb3VyY2VzLmhhcyh0aGlzLmdldFNvdXJjZUtleShzb3VyY2UpKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgbWFya1NvdXJjZUFzRmFpbGVkKHNvdXJjZTogU291cmNlKTogdm9pZCB7XG4gICAgICAgIHRoaXMuZmFpbGVkU291cmNlcy5zZXQodGhpcy5nZXRTb3VyY2VLZXkoc291cmNlKSwgc291cmNlKTtcbiAgICB9XG5cbiAgICBwcml2YXRlIGNhbGN1bGF0ZUFzY2lpQ29kZSh2YWx1ZTogc3RyaW5nKTogbnVtYmVyIHtcbiAgICAgICAgcmV0dXJuIHZhbHVlXG4gICAgICAgICAgICAuc3BsaXQoJycpXG4gICAgICAgICAgICAubWFwKGxldHRlciA9PiBsZXR0ZXIuY2hhckNvZGVBdCgwKSlcbiAgICAgICAgICAgIC5yZWR1Y2UoKHByZXZpb3VzLCBjdXJyZW50KSA9PiBwcmV2aW91cyArIGN1cnJlbnQpO1xuICAgIH1cbn1cbiJdfQ==
@@ -22,10 +22,10 @@ function getInitials(name, size) {
22
22
  */
23
23
  export class Initials extends Source {
24
24
  getAvatar(size) {
25
- return getInitials(this.getValue(), size);
25
+ return getInitials(this.getValue().replace(/[^a-zA-Z0-9\s]/g, ''), size); // only letters, numbers and space
26
26
  }
27
27
  isTextual() {
28
28
  return true;
29
29
  }
30
30
  }
31
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5pdGlhbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9hdmF0YXIvc291cmNlcy9pbml0aWFscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsTUFBTSxFQUFDLE1BQU0sVUFBVSxDQUFDO0FBRWhDOztHQUVHO0FBQ0gsU0FBUyxXQUFXLENBQUMsSUFBWSxFQUFFLElBQVk7SUFDM0MsSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUVuQixJQUFJLENBQUMsSUFBSSxFQUFFO1FBQ1AsT0FBTyxFQUFFLENBQUM7S0FDYjtJQUVELElBQUksS0FBSyxHQUFhLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFdEMsSUFBSSxJQUFJLElBQUksSUFBSSxHQUFHLEtBQUssQ0FBQyxNQUFNLEVBQUU7UUFDN0IsS0FBSyxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxDQUFDO0tBQ2hDO0lBRUQsT0FBTyxLQUFLO1NBQ1AsTUFBTSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7U0FDckMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO1NBQzFCLElBQUksQ0FBQyxFQUFFLENBQUM7U0FDUixXQUFXLEVBQUUsQ0FBQztBQUN2QixDQUFDO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sUUFBUyxTQUFRLE1BQU07SUFDekIsU0FBUyxDQUFDLElBQVk7UUFDekIsT0FBTyxXQUFXLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQzlDLENBQUM7SUFFTSxTQUFTO1FBQ1osT0FBTyxJQUFJLENBQUM7SUFDaEIsQ0FBQztDQUNKIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtTb3VyY2V9IGZyb20gJy4vc291cmNlJztcblxuLyoqXG4gKiBJdGVyYXRlcyBhIHBlcnNvbidzIG5hbWUgc3RyaW5nIHRvIGdldCB0aGUgaW5pdGlhbHMgb2YgZWFjaCB3b3JkIGluIHVwcGVyY2FzZS5cbiAqL1xuZnVuY3Rpb24gZ2V0SW5pdGlhbHMobmFtZTogc3RyaW5nLCBzaXplOiBudW1iZXIpOiBzdHJpbmcge1xuICAgIG5hbWUgPSBuYW1lLnRyaW0oKTtcblxuICAgIGlmICghbmFtZSkge1xuICAgICAgICByZXR1cm4gJyc7XG4gICAgfVxuXG4gICAgbGV0IHdvcmRzOiBzdHJpbmdbXSA9IG5hbWUuc3BsaXQoJyAnKTtcblxuICAgIGlmIChzaXplICYmIHNpemUgPCB3b3Jkcy5sZW5ndGgpIHtcbiAgICAgICAgd29yZHMgPSB3b3Jkcy5zbGljZSgwLCBzaXplKTtcbiAgICB9XG5cbiAgICByZXR1cm4gd29yZHNcbiAgICAgICAgLmZpbHRlcihlbGVtZW50ID0+IGVsZW1lbnQubGVuZ3RoID4gMClcbiAgICAgICAgLm1hcChlbGVtZW50ID0+IGVsZW1lbnRbMF0pXG4gICAgICAgIC5qb2luKCcnKVxuICAgICAgICAudG9VcHBlckNhc2UoKTtcbn1cblxuLyoqXG4gKiBSZXR1cm4gdGhlIGluaXRpYWxzIG9mIHRoZSBnaXZlbiB2YWx1ZSBhcyBhdmF0YXJcbiAqL1xuZXhwb3J0IGNsYXNzIEluaXRpYWxzIGV4dGVuZHMgU291cmNlIHtcbiAgICBwdWJsaWMgZ2V0QXZhdGFyKHNpemU6IG51bWJlcik6IHN0cmluZyB7XG4gICAgICAgIHJldHVybiBnZXRJbml0aWFscyh0aGlzLmdldFZhbHVlKCksIHNpemUpO1xuICAgIH1cblxuICAgIHB1YmxpYyBpc1RleHR1YWwoKTogYm9vbGVhbiB7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbn1cbiJdfQ==
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5pdGlhbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uYXR1cmFsL3NyYy9saWIvbW9kdWxlcy9hdmF0YXIvc291cmNlcy9pbml0aWFscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUMsTUFBTSxFQUFDLE1BQU0sVUFBVSxDQUFDO0FBRWhDOztHQUVHO0FBQ0gsU0FBUyxXQUFXLENBQUMsSUFBWSxFQUFFLElBQVk7SUFDM0MsSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUVuQixJQUFJLENBQUMsSUFBSSxFQUFFO1FBQ1AsT0FBTyxFQUFFLENBQUM7S0FDYjtJQUVELElBQUksS0FBSyxHQUFhLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFdEMsSUFBSSxJQUFJLElBQUksSUFBSSxHQUFHLEtBQUssQ0FBQyxNQUFNLEVBQUU7UUFDN0IsS0FBSyxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxDQUFDO0tBQ2hDO0lBRUQsT0FBTyxLQUFLO1NBQ1AsTUFBTSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7U0FDckMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO1NBQzFCLElBQUksQ0FBQyxFQUFFLENBQUM7U0FDUixXQUFXLEVBQUUsQ0FBQztBQUN2QixDQUFDO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sUUFBUyxTQUFRLE1BQU07SUFDekIsU0FBUyxDQUFDLElBQVk7UUFDekIsT0FBTyxXQUFXLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLE9BQU8sQ0FBQyxpQkFBaUIsRUFBRSxFQUFFLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQyxDQUFDLGtDQUFrQztJQUNoSCxDQUFDO0lBRU0sU0FBUztRQUNaLE9BQU8sSUFBSSxDQUFDO0lBQ2hCLENBQUM7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7U291cmNlfSBmcm9tICcuL3NvdXJjZSc7XG5cbi8qKlxuICogSXRlcmF0ZXMgYSBwZXJzb24ncyBuYW1lIHN0cmluZyB0byBnZXQgdGhlIGluaXRpYWxzIG9mIGVhY2ggd29yZCBpbiB1cHBlcmNhc2UuXG4gKi9cbmZ1bmN0aW9uIGdldEluaXRpYWxzKG5hbWU6IHN0cmluZywgc2l6ZTogbnVtYmVyKTogc3RyaW5nIHtcbiAgICBuYW1lID0gbmFtZS50cmltKCk7XG5cbiAgICBpZiAoIW5hbWUpIHtcbiAgICAgICAgcmV0dXJuICcnO1xuICAgIH1cblxuICAgIGxldCB3b3Jkczogc3RyaW5nW10gPSBuYW1lLnNwbGl0KCcgJyk7XG5cbiAgICBpZiAoc2l6ZSAmJiBzaXplIDwgd29yZHMubGVuZ3RoKSB7XG4gICAgICAgIHdvcmRzID0gd29yZHMuc2xpY2UoMCwgc2l6ZSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHdvcmRzXG4gICAgICAgIC5maWx0ZXIoZWxlbWVudCA9PiBlbGVtZW50Lmxlbmd0aCA+IDApXG4gICAgICAgIC5tYXAoZWxlbWVudCA9PiBlbGVtZW50WzBdKVxuICAgICAgICAuam9pbignJylcbiAgICAgICAgLnRvVXBwZXJDYXNlKCk7XG59XG5cbi8qKlxuICogUmV0dXJuIHRoZSBpbml0aWFscyBvZiB0aGUgZ2l2ZW4gdmFsdWUgYXMgYXZhdGFyXG4gKi9cbmV4cG9ydCBjbGFzcyBJbml0aWFscyBleHRlbmRzIFNvdXJjZSB7XG4gICAgcHVibGljIGdldEF2YXRhcihzaXplOiBudW1iZXIpOiBzdHJpbmcge1xuICAgICAgICByZXR1cm4gZ2V0SW5pdGlhbHModGhpcy5nZXRWYWx1ZSgpLnJlcGxhY2UoL1teYS16QS1aMC05XFxzXS9nLCAnJyksIHNpemUpOyAvLyBvbmx5IGxldHRlcnMsIG51bWJlcnMgYW5kIHNwYWNlXG4gICAgfVxuXG4gICAgcHVibGljIGlzVGV4dHVhbCgpOiBib29sZWFuIHtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxufVxuIl19
@@ -1,8 +1,9 @@
1
1
  import { DOCUMENT } from '@angular/common';
2
2
  import { Inject, Injectable, InjectionToken, LOCALE_ID } from '@angular/core';
3
3
  import { NavigationEnd, PRIMARY_OUTLET } from '@angular/router';
4
- import { filter } from 'rxjs/operators';
4
+ import { filter, startWith } from 'rxjs/operators';
5
5
  import { NaturalDialogTriggerComponent } from '../../dialog-trigger/dialog-trigger.component';
6
+ import { combineLatest, Observable, of } from 'rxjs';
6
7
  import * as i0 from "@angular/core";
7
8
  import * as i1 from "@angular/router";
8
9
  import * as i2 from "@angular/platform-browser";
@@ -25,14 +26,20 @@ export function stripTags(str) {
25
26
  * configured for it in the routing.
26
27
  */
27
28
  export class NaturalSeoService {
28
- constructor(config, router, titleService, metaTagService, document, locale) {
29
- this.config = config;
29
+ constructor(configToken, router, titleService, metaTagService, document, locale) {
30
30
  this.router = router;
31
31
  this.titleService = titleService;
32
32
  this.metaTagService = metaTagService;
33
33
  this.document = document;
34
34
  this.locale = locale;
35
- this.router.events.pipe(filter(event => event instanceof NavigationEnd)).subscribe(() => {
35
+ this.config = {
36
+ applicationName: '',
37
+ };
38
+ combineLatest({
39
+ config: configToken instanceof Observable ? configToken.pipe(startWith(this.config)) : of(configToken),
40
+ navigationEnd: this.router.events.pipe(filter(event => event instanceof NavigationEnd)),
41
+ }).subscribe(({ config }) => {
42
+ this.config = config;
36
43
  const root = this.router.routerState.root.snapshot;
37
44
  this.routeData = this.getRouteData(root);
38
45
  const seo = this.routeData.seo ?? { title: '' };
@@ -52,11 +59,11 @@ export class NaturalSeoService {
52
59
  /**
53
60
  * Update the SEO with given info. The extra part and app name will be appended automatically.
54
61
  *
55
- * In most cases this should not be used, and instead the SEO should be configured in the routing,
62
+ * In most cases, this should not be used. And instead, the SEO should be configured in the routing,
56
63
  * possibly with the callback variant for some dynamism.
57
64
  *
58
- * But in rare cases only the Component is able to build a proper page title, after it gather everything it
59
- * needed. For those cases the Component can inject this service and update the SEO directly.
65
+ * But in rare cases, only the Component is able to build a proper page title, after it gathered everything it
66
+ * needed. For those cases, the Component can inject this service and update the SEO directly.
60
67
  */
61
68
  update(seo) {
62
69
  // Title
@@ -76,7 +83,7 @@ export class NaturalSeoService {
76
83
  // Canonical
77
84
  // Add language in url (after domain) if some languages are provided only
78
85
  const language = this.config.languages?.length && this.locale ? this.locale.split('-')[0] : '';
79
- const urlParts = this.getUrlParts(seo.canonicalQueryParamsWhitelist || []);
86
+ const urlParts = this.getUrlParts(seo.canonicalQueryParamsWhitelist ?? []);
80
87
  this.updateLinkTag({ rel: 'canonical', href: this.getUrl(urlParts, language) });
81
88
  this.updateAlternates(urlParts);
82
89
  }
@@ -97,10 +104,10 @@ export class NaturalSeoService {
97
104
  }
98
105
  // Query Params
99
106
  let params = '';
100
- for (const param in urlTree.queryParams) {
101
- if (whiteListedParams.includes(param)) {
102
- const key = encodeURIComponent(param);
103
- const value = encodeURIComponent(urlTree.queryParams[param]);
107
+ for (const whiteListedParam of whiteListedParams) {
108
+ if (whiteListedParam in urlTree.queryParams) {
109
+ const key = encodeURIComponent(whiteListedParam);
110
+ const value = encodeURIComponent(urlTree.queryParams[whiteListedParam]);
104
111
  if (params.length) {
105
112
  params += '&';
106
113
  }
@@ -114,9 +121,6 @@ export class NaturalSeoService {
114
121
  }
115
122
  /**
116
123
  * Add language between domain and uri https://example.com/fr/folder/page
117
- * @param urlParts
118
- * @param language
119
- * @private
120
124
  */
121
125
  getUrl(urlParts, language) {
122
126
  let url = urlParts.url;
@@ -148,7 +152,7 @@ export class NaturalSeoService {
148
152
  }
149
153
  }
150
154
  updateLinkTag(definition) {
151
- const linkElement = this.document.head.querySelector(this._parseSelector(definition)) ||
155
+ const linkElement = this.document.head.querySelector(this.parseSelector(definition)) ??
152
156
  this.document.head.appendChild(this.document.createElement('link'));
153
157
  if (linkElement) {
154
158
  Object.keys(definition).forEach((attribute) => {
@@ -157,13 +161,11 @@ export class NaturalSeoService {
157
161
  }
158
162
  }
159
163
  /**
160
- * Parse tag to create a selector
161
- * @param definition
162
- * @return {string} selector to use in querySelector
164
+ * Returns selector to use in querySelector to get the given link
163
165
  */
164
- _parseSelector(definition) {
166
+ parseSelector(definition) {
165
167
  let attributes = 'link';
166
- Object.keys(definition).forEach((attr) => {
168
+ Object.keys(definition).forEach(attr => {
167
169
  if (attr !== 'href') {
168
170
  attributes += `[${attr}="${definition[attr]}"]`;
169
171
  }
@@ -231,4 +233,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.2", ngImpor
231
233
  type: Inject,
232
234
  args: [LOCALE_ID]
233
235
  }] }]; } });
234
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"seo.service.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/common/services/seo.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {DOCUMENT} from '@angular/common';\nimport {Inject, Injectable, InjectionToken, LOCALE_ID} from '@angular/core';\nimport {Meta, Title} from '@angular/platform-browser';\nimport {ActivatedRouteSnapshot, Data, NavigationEnd, PRIMARY_OUTLET, Router} from '@angular/router';\nimport {filter} from 'rxjs/operators';\nimport {NaturalDialogTriggerComponent} from '../../dialog-trigger/dialog-trigger.component';\n\nexport type NaturalSeo = NaturalSeoBasic | NaturalSeoCallback | NaturalSeoResolve;\n\n/**\n * Typically used for \"static\" pages where there is not a single resolved object. So\n * all pages such as \"About\", or list of objects (list of risks, etc.).\n *\n * This should be the most common used variant.\n */\nexport type NaturalSeoBasic = Robots & {\n    /**\n     * The page title, that will be concatenated with application name\n     */\n    title: string;\n\n    /**\n     * If given will be used as page description, otherwise fallback on default value\n     */\n    description?: string;\n\n    /**\n     * List of parameters included in the canonical tag's url\n     */\n    canonicalQueryParamsWhitelist?: string[];\n};\n\nexport declare type NaturalLinkDefinition = {\n    charset?: string;\n    crossorigin?: string;\n    href?: string;\n    hreflang?: string;\n    media?: string;\n    rel?: string;\n    rev?: string;\n    sizes?: string;\n    target?: string;\n    type?: string;\n} & {\n    [prop: string]: string;\n};\n/**\n * Typically used for a \"dynamic\" page where a single object is resolved. So a detail page, such\n * as the detail of a risk and so on.\n */\nexport type NaturalSeoResolve = Robots & {\n    /**\n     * The key to be used in the resolved data to find the resolved object. The fullName\n     * or name of the object will be used for page title, and the object description, if any,\n     * will be used for page description.\n     */\n    resolveKey: string;\n};\n\n/**\n * Rarely used for a page that has very specific needs and need to build title and description in a custom way.\n * The callback back will be given the resolved data, and it is up to the callback to find whatever it needs.\n */\nexport type NaturalSeoCallback = (routeData: Data) => NaturalSeoBasic;\n\n/**\n * Typically used to type the routing data received in the component, eg:\n *\n * ```ts\n * class MyComponent extends NaturalAbstractDetail<MyService, NaturalSeoResolveData> {}\n * ```\n */\nexport type NaturalSeoResolveData = {\n    seo: NaturalSeoBasic;\n};\n\ninterface Robots {\n    /**\n     * If given will be used as robots meta tag, otherwise fallback on default value\n     */\n    robots?: string;\n}\n\nexport interface NaturalSeoConfig {\n    /**\n     * The name of the application that will always appear in the page title\n     */\n    readonly applicationName: string;\n\n    /**\n     * Default value for description meta that is used for pages without value\n     */\n    readonly defaultDescription?: string;\n\n    /**\n     * Default value for robots meta that is used for pages without value\n     */\n    readonly defaultRobots?: string;\n\n    /**\n     * If given, the callback will be called for each route and must return a string that will\n     * be inserted between the page title and the application name.\n     *\n     * It should be used to complete the title with info that are often, but not necessarily always,\n     * available throughout the entire application. Typically used for the site/state in OKpilot.\n     */\n    readonly extraPart?: (routeData: Data) => string;\n\n    /**\n     * Used to generate alternative tags\n     * <link rel=\"alternate\" hreflang=\"en\" href=\"https://www.example.com/en/page\">\n     */\n    readonly languages?: Readonly<string[]>;\n}\n\nexport const NATURAL_SEO_CONFIG = new InjectionToken<NaturalSeoConfig>('Configuration for SEO service');\n\nexport function stripTags(str: string): string {\n    return str.replace(/<\\/?[^>]+>/g, '');\n}\n\ntype ResolvedData = {\n    model?: {\n        name?: string;\n        fullName?: string;\n        description?: string;\n    };\n};\n\n/**\n * This service is responsible to keep up to date the page title and page description according\n * to what is configured in routing or default values.\n *\n * It **must** be injected in the root module of the application to have effects in all modules. And it\n * must be provided a configuration.\n *\n * The full title has the following structure:\n *\n *     dialog title - page title - extra part - app name\n *\n * `dialog title` only exists if a `NaturalDialogTriggerComponent` is currently open, and that some SEO is\n * configured for it in the routing.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalSeoService {\n    private routeData?: Data;\n\n    public constructor(\n        @Inject(NATURAL_SEO_CONFIG) private readonly config: NaturalSeoConfig,\n        private readonly router: Router,\n        private readonly titleService: Title,\n        private readonly metaTagService: Meta,\n        @Inject(DOCUMENT) private readonly document: Document,\n        @Inject(LOCALE_ID) private locale: string,\n    ) {\n        this.router.events.pipe(filter(event => event instanceof NavigationEnd)).subscribe(() => {\n            const root = this.router.routerState.root.snapshot;\n            this.routeData = this.getRouteData(root);\n\n            const seo = this.routeData.seo ?? {title: ''};\n            const dialogRouteData = this.getDialogRouteData(root);\n            const dialogSeo: NaturalSeo = dialogRouteData?.seo;\n\n            let basic = this.toBasic(seo, this.routeData);\n            if (dialogRouteData && dialogSeo) {\n                const dialogBasic = this.toBasic(dialogSeo, dialogRouteData);\n                basic = {\n                    ...dialogBasic,\n                    title: this.join([dialogBasic.title, basic.title]),\n                };\n            }\n\n            this.update(basic);\n        });\n    }\n\n    /**\n     * Update the SEO with given info. The extra part and app name will be appended automatically.\n     *\n     * In most cases this should not be used, and instead the SEO should be configured in the routing,\n     * possibly with the callback variant for some dynamism.\n     *\n     * But in rare cases only the Component is able to build a proper page title, after it gather everything it\n     * needed. For those cases the Component can inject this service and update the SEO directly.\n     */\n    public update(seo: NaturalSeoBasic): void {\n        // Title\n        const parts = [\n            seo.title,\n            this.config.extraPart && this.routeData ? this.config.extraPart(this.routeData) : '',\n            this.config.applicationName,\n        ];\n\n        const title = this.join(parts);\n        this.titleService.setTitle(title);\n\n        // Description\n        const description = seo?.description ?? this.config.defaultDescription;\n        this.updateTag('description', description);\n\n        // Robots\n        const robots = seo?.robots ?? this.config.defaultRobots;\n        this.updateTag('robots', robots);\n\n        // Canonical\n        // Add language in url (after domain) if some languages are provided only\n        const language = this.config.languages?.length && this.locale ? this.locale.split('-')[0] : '';\n        const urlParts = this.getUrlParts(seo.canonicalQueryParamsWhitelist || []);\n        this.updateLinkTag({rel: 'canonical', href: this.getUrl(urlParts, language)});\n        this.updateAlternates(urlParts);\n    }\n\n    public updateAlternates(urlParts: {url: string; params: string}): void {\n        this.config.languages?.forEach(language => {\n            this.updateLinkTag({rel: 'alternate', href: this.getUrl(urlParts, language), hreflang: language});\n        });\n    }\n\n    private getUrlParts(whiteListedParams: string[]): {url: string; params: string} {\n        let url = 'https://' + this.document.defaultView?.window.location.hostname;\n        const urlTree = this.router.parseUrl(this.router.url);\n\n        // need better like something recursive ?\n        if (urlTree.root.hasChildren()) {\n            const segments = urlTree.root.children['primary'].segments;\n            if (segments && segments.length > 0) {\n                url += '/' + segments.map(segment => segment.path).join('/');\n            }\n        }\n\n        // Query Params\n        let params = '';\n        for (const param in urlTree.queryParams) {\n            if (whiteListedParams.includes(param)) {\n                const key = encodeURIComponent(param);\n                const value = encodeURIComponent(urlTree.queryParams[param]);\n                if (params.length) {\n                    params += '&';\n                }\n                params += `${key}=${value}`;\n            }\n        }\n\n        if (params.length) {\n            params = '?' + params;\n        }\n\n        return {url, params};\n    }\n\n    /**\n     * Add language between domain and uri https://example.com/fr/folder/page\n     * @param urlParts\n     * @param language\n     * @private\n     */\n    private getUrl(urlParts: {url: string; params: string}, language?: string): string {\n        let url = urlParts.url;\n\n        if (language) {\n            url = this.addLanguageSegment(url, language);\n        }\n\n        if (urlParts.params) {\n            url += urlParts.params;\n        }\n\n        return url;\n    }\n\n    private addLanguageSegment(url: string, language: string): string {\n        const urlObj = new URL(url);\n        const newPath = urlObj.pathname === '/' ? `/${language}` : `/${language}${urlObj.pathname}`;\n        return urlObj.origin + newPath + urlObj.search + urlObj.hash;\n    }\n\n    private join(parts: string[]): string {\n        return parts.filter(s => !!s).join(' - ');\n    }\n\n    private updateTag(name: string, value?: string): void {\n        if (value) {\n            this.metaTagService.updateTag({\n                name: name,\n                value: stripTags(value),\n            });\n        } else {\n            this.metaTagService.removeTag(`name=\"${name}\"`);\n        }\n    }\n\n    private updateLinkTag(definition: NaturalLinkDefinition): void {\n        const linkElement =\n            <HTMLLinkElement>this.document.head.querySelector(this._parseSelector(definition)) ||\n            this.document.head.appendChild(this.document.createElement('link'));\n\n        if (linkElement) {\n            Object.keys(definition).forEach((attribute: string) => {\n                linkElement.setAttribute(attribute, definition[attribute]);\n            });\n        }\n    }\n\n    /**\n     * Parse tag to create a selector\n     * @param  definition\n     * @return {string} selector to use in querySelector\n     */\n    private _parseSelector(definition: NaturalLinkDefinition): string {\n        let attributes = 'link';\n        Object.keys(definition).forEach((attr: string) => {\n            if (attr !== 'href') {\n                attributes += `[${attr}=\"${definition[attr]}\"]`;\n            }\n        });\n\n        return attributes;\n    }\n\n    /**\n     * Returns the data from the most deep/specific activated route\n     */\n    private getRouteData(route: ActivatedRouteSnapshot): Data {\n        if (route.firstChild) {\n            return this.getRouteData(route.firstChild);\n        } else {\n            return route.data;\n        }\n    }\n\n    /**\n     * Returns the data from the `NaturalDialogTriggerComponent` if one is open\n     */\n    private getDialogRouteData(route: ActivatedRouteSnapshot): Data | null {\n        if (route.component === NaturalDialogTriggerComponent && route.outlet !== PRIMARY_OUTLET) {\n            return route.data;\n        }\n\n        for (const child of route.children) {\n            const data = this.getDialogRouteData(child);\n            if (data) {\n                return data;\n            }\n        }\n\n        return null;\n    }\n\n    private toBasic(seo: NaturalSeo, routeData: Data): NaturalSeoBasic {\n        if (typeof seo === 'function') {\n            return seo(routeData);\n        } else if ('resolveKey' in seo) {\n            const data: ResolvedData | undefined = routeData[seo.resolveKey];\n            if (!data) {\n                throw new Error('Could not find resolved data for SEO service with key: ' + seo.resolveKey);\n            }\n\n            return {\n                title: data.model?.fullName ?? data.model?.name ?? '',\n                description: data.model?.description,\n                robots: seo.robots,\n            };\n        }\n\n        return seo;\n    }\n}\n"]}
236
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"seo.service.js","sourceRoot":"","sources":["../../../../../../../projects/natural/src/lib/modules/common/services/seo.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {DOCUMENT} from '@angular/common';\nimport {Inject, Injectable, InjectionToken, LOCALE_ID} from '@angular/core';\nimport {Meta, Title} from '@angular/platform-browser';\nimport {ActivatedRouteSnapshot, Data, NavigationEnd, PRIMARY_OUTLET, Router} from '@angular/router';\nimport {filter, startWith} from 'rxjs/operators';\nimport {NaturalDialogTriggerComponent} from '../../dialog-trigger/dialog-trigger.component';\nimport {combineLatest, Observable, of} from 'rxjs';\n\nexport type NaturalSeo = NaturalSeoBasic | NaturalSeoCallback | NaturalSeoResolve;\n\n/**\n * Typically used for \"static\" pages where there is not a single resolved object. So\n * all pages such as \"About\", or list of objects (list of risks, etc.).\n *\n * This should be the most common used variant.\n */\nexport type NaturalSeoBasic = Robots & {\n    /**\n     * The page title, that will be concatenated with application name\n     */\n    title: string;\n\n    /**\n     * If given will be used as page description, otherwise fallback on default value\n     */\n    description?: string;\n\n    /**\n     * List of parameters included in the canonical tag's url\n     */\n    canonicalQueryParamsWhitelist?: string[];\n};\n\ntype NaturalLinkDefinition = {\n    charset?: string;\n    crossorigin?: string;\n    href?: string;\n    hreflang?: string;\n    media?: string;\n    rel?: string;\n    rev?: string;\n    sizes?: string;\n    target?: string;\n    type?: string;\n} & {\n    [prop: string]: string;\n};\n/**\n * Typically used for a \"dynamic\" page where a single object is resolved. So a detail page, such\n * as the detail of a risk and so on.\n */\nexport type NaturalSeoResolve = Robots & {\n    /**\n     * The key to be used in the resolved data to find the resolved object. The fullName\n     * or name of the object will be used for page title, and the object description, if any,\n     * will be used for page description.\n     */\n    resolveKey: string;\n};\n\n/**\n * Rarely used for a page that has very specific needs and need to build title and description in a custom way.\n * The callback back will be given the resolved data, and it is up to the callback to find whatever it needs.\n */\nexport type NaturalSeoCallback = (routeData: Data) => NaturalSeoBasic;\n\n/**\n * Typically used to type the routing data received in the component, eg:\n *\n * ```ts\n * class MyComponent extends NaturalAbstractDetail<MyService, NaturalSeoResolveData> {}\n * ```\n */\nexport type NaturalSeoResolveData = {\n    seo: NaturalSeoBasic;\n};\n\ninterface Robots {\n    /**\n     * If given will be used as robots meta tag, otherwise fallback on default value\n     */\n    robots?: string;\n}\n\ninterface NaturalSeoConfigPlain {\n    /**\n     * The name of the application that will always appear in the page title\n     */\n    readonly applicationName: string;\n\n    /**\n     * Default value for description meta that is used for pages without value\n     */\n    readonly defaultDescription?: string;\n\n    /**\n     * Default value for robots meta that is used for pages without value\n     */\n    readonly defaultRobots?: string;\n\n    /**\n     * If given, the callback will be called for each route and must return a string that will\n     * be inserted between the page title and the application name.\n     *\n     * It should be used to complete the title with info that are often, but not necessarily always,\n     * available throughout the entire application. Typically used for the site/state in OKpilot.\n     */\n    readonly extraPart?: (routeData: Data) => string;\n\n    /**\n     * Used to generate alternative tags\n     * <link rel=\"alternate\" hreflang=\"en\" href=\"https://www.example.com/en/page\">\n     */\n    readonly languages?: Readonly<string[]>;\n}\n\nexport type NaturalSeoConfig = NaturalSeoConfigPlain | Observable<NaturalSeoConfigPlain>;\nexport const NATURAL_SEO_CONFIG = new InjectionToken<NaturalSeoConfig>('Configuration for SEO service');\n\nexport function stripTags(str: string): string {\n    return str.replace(/<\\/?[^>]+>/g, '');\n}\n\ntype ResolvedData = {\n    model?: {\n        name?: string;\n        fullName?: string;\n        description?: string;\n    };\n};\n\n/**\n * This service is responsible to keep up to date the page title and page description according\n * to what is configured in routing or default values.\n *\n * It **must** be injected in the root module of the application to have effects in all modules. And it\n * must be provided a configuration.\n *\n * The full title has the following structure:\n *\n *     dialog title - page title - extra part - app name\n *\n * `dialog title` only exists if a `NaturalDialogTriggerComponent` is currently open, and that some SEO is\n * configured for it in the routing.\n */\n@Injectable({\n    providedIn: 'root',\n})\nexport class NaturalSeoService {\n    private routeData?: Data;\n    private config: NaturalSeoConfigPlain = {\n        applicationName: '',\n    };\n\n    public constructor(\n        @Inject(NATURAL_SEO_CONFIG) configToken: NaturalSeoConfig,\n        private readonly router: Router,\n        private readonly titleService: Title,\n        private readonly metaTagService: Meta,\n        @Inject(DOCUMENT) private readonly document: Document,\n        @Inject(LOCALE_ID) private locale: string,\n    ) {\n        combineLatest({\n            config: configToken instanceof Observable ? configToken.pipe(startWith(this.config)) : of(configToken),\n            navigationEnd: this.router.events.pipe(filter(event => event instanceof NavigationEnd)),\n        }).subscribe(({config}) => {\n            this.config = config;\n            const root = this.router.routerState.root.snapshot;\n            this.routeData = this.getRouteData(root);\n\n            const seo: NaturalSeo = this.routeData.seo ?? {title: ''};\n            const dialogRouteData = this.getDialogRouteData(root);\n            const dialogSeo: NaturalSeo = dialogRouteData?.seo;\n\n            let basic = this.toBasic(seo, this.routeData);\n            if (dialogRouteData && dialogSeo) {\n                const dialogBasic = this.toBasic(dialogSeo, dialogRouteData);\n                basic = {\n                    ...dialogBasic,\n                    title: this.join([dialogBasic.title, basic.title]),\n                };\n            }\n\n            this.update(basic);\n        });\n    }\n\n    /**\n     * Update the SEO with given info. The extra part and app name will be appended automatically.\n     *\n     * In most cases, this should not be used. And instead, the SEO should be configured in the routing,\n     * possibly with the callback variant for some dynamism.\n     *\n     * But in rare cases, only the Component is able to build a proper page title, after it gathered everything it\n     * needed. For those cases, the Component can inject this service and update the SEO directly.\n     */\n    public update(seo: NaturalSeoBasic): void {\n        // Title\n        const parts = [\n            seo.title,\n            this.config.extraPart && this.routeData ? this.config.extraPart(this.routeData) : '',\n            this.config.applicationName,\n        ];\n\n        const title = this.join(parts);\n        this.titleService.setTitle(title);\n\n        // Description\n        const description = seo?.description ?? this.config.defaultDescription;\n        this.updateTag('description', description);\n\n        // Robots\n        const robots = seo?.robots ?? this.config.defaultRobots;\n        this.updateTag('robots', robots);\n\n        // Canonical\n        // Add language in url (after domain) if some languages are provided only\n        const language = this.config.languages?.length && this.locale ? this.locale.split('-')[0] : '';\n        const urlParts = this.getUrlParts(seo.canonicalQueryParamsWhitelist ?? []);\n        this.updateLinkTag({rel: 'canonical', href: this.getUrl(urlParts, language)});\n        this.updateAlternates(urlParts);\n    }\n\n    private updateAlternates(urlParts: {url: string; params: string}): void {\n        this.config.languages?.forEach(language => {\n            this.updateLinkTag({rel: 'alternate', href: this.getUrl(urlParts, language), hreflang: language});\n        });\n    }\n\n    private getUrlParts(whiteListedParams: string[]): {url: string; params: string} {\n        let url = 'https://' + this.document.defaultView?.window.location.hostname;\n        const urlTree = this.router.parseUrl(this.router.url);\n\n        // need better like something recursive ?\n        if (urlTree.root.hasChildren()) {\n            const segments = urlTree.root.children['primary'].segments;\n            if (segments && segments.length > 0) {\n                url += '/' + segments.map(segment => segment.path).join('/');\n            }\n        }\n\n        // Query Params\n        let params = '';\n        for (const whiteListedParam of whiteListedParams) {\n            if (whiteListedParam in urlTree.queryParams) {\n                const key = encodeURIComponent(whiteListedParam);\n                const value = encodeURIComponent(urlTree.queryParams[whiteListedParam]);\n                if (params.length) {\n                    params += '&';\n                }\n                params += `${key}=${value}`;\n            }\n        }\n\n        if (params.length) {\n            params = '?' + params;\n        }\n\n        return {url, params};\n    }\n\n    /**\n     * Add language between domain and uri https://example.com/fr/folder/page\n     */\n    private getUrl(urlParts: {url: string; params: string}, language?: string): string {\n        let url = urlParts.url;\n\n        if (language) {\n            url = this.addLanguageSegment(url, language);\n        }\n\n        if (urlParts.params) {\n            url += urlParts.params;\n        }\n\n        return url;\n    }\n\n    private addLanguageSegment(url: string, language: string): string {\n        const urlObj = new URL(url);\n        const newPath = urlObj.pathname === '/' ? `/${language}` : `/${language}${urlObj.pathname}`;\n        return urlObj.origin + newPath + urlObj.search + urlObj.hash;\n    }\n\n    private join(parts: string[]): string {\n        return parts.filter(s => !!s).join(' - ');\n    }\n\n    private updateTag(name: string, value?: string): void {\n        if (value) {\n            this.metaTagService.updateTag({\n                name: name,\n                value: stripTags(value),\n            });\n        } else {\n            this.metaTagService.removeTag(`name=\"${name}\"`);\n        }\n    }\n\n    private updateLinkTag(definition: NaturalLinkDefinition): void {\n        const linkElement =\n            this.document.head.querySelector<HTMLLinkElement>(this.parseSelector(definition)) ??\n            this.document.head.appendChild(this.document.createElement('link'));\n\n        if (linkElement) {\n            Object.keys(definition).forEach((attribute: string) => {\n                linkElement.setAttribute(attribute, definition[attribute]);\n            });\n        }\n    }\n\n    /**\n     * Returns selector to use in querySelector to get the given link\n     */\n    private parseSelector(definition: NaturalLinkDefinition): string {\n        let attributes = 'link';\n        Object.keys(definition).forEach(attr => {\n            if (attr !== 'href') {\n                attributes += `[${attr}=\"${definition[attr]}\"]`;\n            }\n        });\n\n        return attributes;\n    }\n\n    /**\n     * Returns the data from the most deep/specific activated route\n     */\n    private getRouteData(route: ActivatedRouteSnapshot): Data {\n        if (route.firstChild) {\n            return this.getRouteData(route.firstChild);\n        } else {\n            return route.data;\n        }\n    }\n\n    /**\n     * Returns the data from the `NaturalDialogTriggerComponent` if one is open\n     */\n    private getDialogRouteData(route: ActivatedRouteSnapshot): Data | null {\n        if (route.component === NaturalDialogTriggerComponent && route.outlet !== PRIMARY_OUTLET) {\n            return route.data;\n        }\n\n        for (const child of route.children) {\n            const data = this.getDialogRouteData(child);\n            if (data) {\n                return data;\n            }\n        }\n\n        return null;\n    }\n\n    private toBasic(seo: NaturalSeo, routeData: Data): NaturalSeoBasic {\n        if (typeof seo === 'function') {\n            return seo(routeData);\n        } else if ('resolveKey' in seo) {\n            const data: ResolvedData | undefined = routeData[seo.resolveKey];\n            if (!data) {\n                throw new Error('Could not find resolved data for SEO service with key: ' + seo.resolveKey);\n            }\n\n            return {\n                title: data.model?.fullName ?? data.model?.name ?? '',\n                description: data.model?.description,\n                robots: seo.robots,\n            };\n        }\n\n        return seo;\n    }\n}\n"]}