color-util-helpers 1.0.4 → 1.0.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/README.md +47 -0
  2. package/ng-package.json +8 -0
  3. package/package.json +2 -15
  4. package/src/lib/assets/picture.webp +0 -0
  5. package/src/lib/color-conversion.service.spec.ts +54 -0
  6. package/src/lib/color-conversion.service.ts +35 -0
  7. package/src/lib/color-extractor.directive.spec.ts +49 -0
  8. package/src/lib/color-extractor.directive.ts +28 -0
  9. package/src/lib/color-grab.directive.ts +204 -0
  10. package/src/lib/color-lighten-darken.service.spec.ts +61 -0
  11. package/src/lib/color-lighten-darken.service.ts +83 -0
  12. package/src/lib/color-pallette.service.spec.ts +85 -0
  13. package/src/lib/color-pallette.service.ts +191 -0
  14. package/src/lib/color-scheme.service.ts +123 -0
  15. package/src/lib/color-utilities-demo/color-utilities-demo.component.css +12 -0
  16. package/src/lib/color-utilities-demo/color-utilities-demo.component.html +109 -0
  17. package/src/lib/color-utilities-demo/color-utilities-demo.component.ts +57 -0
  18. package/src/lib/color-utils.module.ts +27 -0
  19. package/src/lib/text-color.service.spec.ts +75 -0
  20. package/src/lib/text-color.service.ts +101 -0
  21. package/{public-api.d.ts → src/public-api.ts} +7 -0
  22. package/tsconfig.lib.json +32 -0
  23. package/tsconfig.lib.prod.json +10 -0
  24. package/tsconfig.spec.json +14 -0
  25. package/color-util-helpers-1.0.4.tgz +0 -0
  26. package/esm2022/color-util-helpers.mjs +0 -5
  27. package/esm2022/lib/color-conversion.service.mjs +0 -39
  28. package/esm2022/lib/color-extractor.directive.mjs +0 -37
  29. package/esm2022/lib/color-lighten-darken.service.mjs +0 -79
  30. package/esm2022/lib/color-pallette.service.mjs +0 -172
  31. package/esm2022/lib/color-scheme.service.mjs +0 -113
  32. package/esm2022/lib/color-utilities-demo/color-utilities-demo.component.mjs +0 -41
  33. package/esm2022/lib/color-utils.module.mjs +0 -32
  34. package/esm2022/lib/text-color.service.mjs +0 -79
  35. package/esm2022/public-api.mjs +0 -12
  36. package/fesm2022/color-util-helpers.mjs +0 -580
  37. package/fesm2022/color-util-helpers.mjs.map +0 -1
  38. package/index.d.ts +0 -5
  39. package/lib/color-conversion.service.d.ts +0 -8
  40. package/lib/color-extractor.directive.d.ts +0 -13
  41. package/lib/color-lighten-darken.service.d.ts +0 -10
  42. package/lib/color-pallette.service.d.ts +0 -36
  43. package/lib/color-scheme.service.d.ts +0 -45
  44. package/lib/color-utilities-demo/color-utilities-demo.component.d.ts +0 -34
  45. package/lib/color-utils.module.d.ts +0 -10
  46. package/lib/text-color.service.d.ts +0 -18
@@ -0,0 +1,101 @@
1
+ import { Injectable } from '@angular/core'
2
+ import { ColorConversionService } from './color-conversion.service'
3
+
4
+ @Injectable({
5
+ providedIn: 'root'
6
+ })
7
+ export class TextColorService {
8
+
9
+ constructor(
10
+ private colorConversionService: ColorConversionService
11
+ ) {}
12
+
13
+ textColorForBgColor(bgColor: string, lightColor: string, darkColor: string) {
14
+
15
+ const UIColors = this.fixColor(bgColor)
16
+
17
+ const r = UIColors[0]
18
+ const g = UIColors[1]
19
+ const b = UIColors[2]
20
+
21
+ return ((r*0.299 + g*0.587 + b*0.114) > 149) ? darkColor : lightColor;
22
+
23
+ }
24
+
25
+ darkerColor(color1: string, color2: string) {
26
+ return this.isColorDarker(color1, color2) ? color1 : color2;
27
+ }
28
+
29
+ isColorDarker(color1: string, color2: string) {
30
+
31
+ const newColor1 = this.fixColor(color1);
32
+ const newColor2 = this.fixColor(color2);
33
+
34
+ const luminance1 = this.calculateLuminance(newColor1[0], newColor1[1], newColor1[2]);
35
+ const luminance2 = this.calculateLuminance(newColor2[0], newColor2[1], newColor2[2]);
36
+
37
+ return luminance1 < luminance2;
38
+ }
39
+
40
+
41
+ lighterColor(color1: string, color2: string): string {
42
+ return (this.isColorLighter(color1,color2)) ? color1 : color2;
43
+ }
44
+
45
+ isColorLighter(color1: string, color2: string) {
46
+
47
+ const newColor1 = this.fixColor(color1);
48
+ const newColor2 = this.fixColor(color2);
49
+
50
+ const luminance1 = this.calculateLuminance(newColor1[0], newColor1[1], newColor1[2]);
51
+ const luminance2 = this.calculateLuminance(newColor2[0], newColor2[1], newColor2[2]);
52
+
53
+ return (luminance1 > luminance2)
54
+
55
+ }
56
+
57
+ calculateLuminance(r: number, g: number, b: number): number {
58
+ return 0.2126 * r + 0.7152 * g + 0.0722 * b;
59
+ }
60
+
61
+ fixColor(color: string) {
62
+ // Remove leading hash if present
63
+ const sanitizedColor = color.startsWith('#') ? color.slice(1) : color;
64
+
65
+ // Validate if the color is a valid hex (3 or 6 characters)
66
+ if (!this.isValidHex(sanitizedColor)) {
67
+ return this.parseRgb(sanitizedColor); // If not hex, attempt to parse as RGB
68
+ }
69
+
70
+ // Convert hex to RGB
71
+ const rgb = this.hexToRgb(sanitizedColor);
72
+ return rgb;
73
+ }
74
+
75
+ // Helper function to validate if a string is a valid 3 or 6 digit hex code
76
+ isValidHex(color: string): boolean {
77
+ const hexRegex = /^[A-Fa-f0-9]{3}$|^[A-Fa-f0-9]{6}$/i;
78
+ return hexRegex.test(color);
79
+ }
80
+
81
+ // Helper function to convert a 3 or 6 digit hex color to RGB
82
+ hexToRgb(hex: string): number[] {
83
+ if (hex.length === 3) {
84
+ hex = hex.split('').map(c => c + c).join(''); // Expand shorthand hex to full
85
+ }
86
+
87
+ return [
88
+ parseInt(hex.slice(0, 2), 16),
89
+ parseInt(hex.slice(2, 4), 16),
90
+ parseInt(hex.slice(4, 6), 16),
91
+ ];
92
+ }
93
+
94
+ // Helper function to parse an RGB string (e.g., rgb(255, 0, 0)) into an array
95
+ parseRgb(rgb: string): number[] {
96
+ const match = rgb.match(/\d+/g);
97
+ return match ? match.map(num => parseInt(num)) : [];
98
+ }
99
+
100
+
101
+ }
@@ -1,8 +1,15 @@
1
+ /*
2
+ * Public API Surface of color-utils
3
+ */
4
+
1
5
  export * from './lib/color-utils.module';
2
6
  export * from './lib/color-utilities-demo/color-utilities-demo.component';
7
+
3
8
  export * from './lib/color-conversion.service';
4
9
  export * from './lib/color-extractor.directive';
10
+ export * from './lib/color-grab.directive';
5
11
  export * from './lib/color-pallette.service';
6
12
  export * from './lib/text-color.service';
13
+
7
14
  export * from './lib/color-scheme.service';
8
15
  export * from './lib/color-lighten-darken.service';
@@ -0,0 +1,32 @@
1
+ /* To learn more about this file see: https://angular.io/config/tsconfig. */
2
+ {
3
+ "extends": "../../tsconfig.json",
4
+ "compilerOptions": {
5
+ "outDir": "../../dist/color-utils",
6
+ "declaration": true,
7
+ "declarationMap": true,
8
+ "inlineSources": true,
9
+ "types": [],
10
+ "baseUrl": "./",
11
+ "paths": {
12
+ "@angular/*": ["node_modules/@angular/*"],
13
+ "color-utils": ["node_modules/color-utils"],
14
+ },
15
+ "target": "ES2022",
16
+ "lib": [
17
+ "dom",
18
+ "ES2022"
19
+ ]
20
+ },
21
+ "angularCompilerOptions": {
22
+ "skipTemplateCodegen": true,
23
+ "strictMetadataEmit": true,
24
+ "enableResourceInlining": true,
25
+ "strictInjectionParameters": true,
26
+ "strictTemplates": true
27
+ },
28
+ "exclude": [
29
+ "src/test.ts",
30
+ "**/*.spec.ts"
31
+ ]
32
+ }
@@ -0,0 +1,10 @@
1
+ /* To learn more about this file see: https://angular.io/config/tsconfig. */
2
+ {
3
+ "extends": "./tsconfig.lib.json",
4
+ "compilerOptions": {
5
+ "declarationMap": false
6
+ },
7
+ "angularCompilerOptions": {
8
+ "compilationMode": "partial"
9
+ }
10
+ }
@@ -0,0 +1,14 @@
1
+ /* To learn more about this file see: https://angular.io/config/tsconfig. */
2
+ {
3
+ "extends": "../../tsconfig.json",
4
+ "compilerOptions": {
5
+ "outDir": "../../out-tsc/spec",
6
+ "types": [
7
+ "jasmine"
8
+ ]
9
+ },
10
+ "include": [
11
+ "**/*.spec.ts",
12
+ "**/*.d.ts"
13
+ ]
14
+ }
Binary file
@@ -1,5 +0,0 @@
1
- /**
2
- * Generated bundle index. Do not edit.
3
- */
4
- export * from './public-api';
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sb3ItdXRpbC1oZWxwZXJzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vcHJvamVjdHMvY29sb3ItdXRpbHMvc3JjL2NvbG9yLXV0aWwtaGVscGVycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -1,39 +0,0 @@
1
- import { Injectable } from '@angular/core';
2
- import * as i0 from "@angular/core";
3
- export class ColorConversionService {
4
- constructor() {
5
- this.componentToHex = (c) => {
6
- const hex = c.toString(16);
7
- return hex.length === 1 ? "0" + hex : hex;
8
- };
9
- }
10
- rgbToHex(rgb) {
11
- if (rgb === null || rgb.length !== 3 || rgb.some(value => value < 0 || value > 255))
12
- return '';
13
- const [r, g, b] = rgb;
14
- const hexR = this.componentToHex(r);
15
- const hexG = this.componentToHex(g);
16
- const hexB = this.componentToHex(b);
17
- return "#" + hexR + hexG + hexB;
18
- }
19
- hexToRgb(hex) {
20
- if (hex === null || hex === undefined)
21
- return [];
22
- hex = (hex.length === 3) ? hex + hex : hex;
23
- const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
24
- return result ? [
25
- parseInt(result[1], 16),
26
- parseInt(result[2], 16),
27
- parseInt(result[3], 16)
28
- ] : [];
29
- }
30
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorConversionService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
31
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorConversionService, providedIn: 'root' }); }
32
- }
33
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorConversionService, decorators: [{
34
- type: Injectable,
35
- args: [{
36
- providedIn: 'root'
37
- }]
38
- }] });
39
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sb3ItY29udmVyc2lvbi5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvY29sb3ItdXRpbHMvc3JjL2xpYi9jb2xvci1jb252ZXJzaW9uLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFLM0MsTUFBTSxPQUFPLHNCQUFzQjtJQUhuQztRQWVVLG1CQUFjLEdBQUcsQ0FBQyxDQUFTLEVBQUUsRUFBRTtZQUNyQyxNQUFNLEdBQUcsR0FBRyxDQUFDLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQzNCLE9BQU8sR0FBRyxDQUFDLE1BQU0sS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQztRQUM1QyxDQUFDLENBQUE7S0FjRjtJQTNCQyxRQUFRLENBQUMsR0FBb0I7UUFDM0IsSUFBSSxHQUFHLEtBQUssSUFBSSxJQUFJLEdBQUcsQ0FBQyxNQUFNLEtBQUssQ0FBQyxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLEdBQUcsQ0FBQyxJQUFJLEtBQUssR0FBRyxHQUFHLENBQUM7WUFBRSxPQUFPLEVBQUUsQ0FBQztRQUUvRixNQUFNLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsR0FBRyxHQUFHLENBQUM7UUFDdEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNwQyxNQUFNLElBQUksR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3BDLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDcEMsT0FBTyxHQUFHLEdBQUcsSUFBSSxHQUFHLElBQUksR0FBRyxJQUFJLENBQUM7SUFDbEMsQ0FBQztJQU9ELFFBQVEsQ0FBQyxHQUE4QjtRQUNyQyxJQUFJLEdBQUcsS0FBSyxJQUFJLElBQUksR0FBRyxLQUFLLFNBQVM7WUFBRSxPQUFPLEVBQUUsQ0FBQztRQUVqRCxHQUFHLEdBQUcsQ0FBQyxHQUFHLENBQUMsTUFBTSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUM7UUFDM0MsTUFBTSxNQUFNLEdBQUcsMkNBQTJDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBRXJFLE9BQU8sTUFBTSxDQUFDLENBQUMsQ0FBQztZQUNkLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDO1lBQ3ZCLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDO1lBQ3ZCLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDO1NBQ3hCLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztJQUNULENBQUM7K0dBNUJVLHNCQUFzQjttSEFBdEIsc0JBQXNCLGNBRnJCLE1BQU07OzRGQUVQLHNCQUFzQjtrQkFIbEMsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcblxyXG5ASW5qZWN0YWJsZSh7XHJcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBDb2xvckNvbnZlcnNpb25TZXJ2aWNlIHtcclxuXHJcbiAgcmdiVG9IZXgocmdiOiBudW1iZXJbXSB8IG51bGwpOiBzdHJpbmcge1xyXG4gICAgaWYgKHJnYiA9PT0gbnVsbCB8fCByZ2IubGVuZ3RoICE9PSAzIHx8IHJnYi5zb21lKHZhbHVlID0+IHZhbHVlIDwgMCB8fCB2YWx1ZSA+IDI1NSkpIHJldHVybiAnJztcclxuXHJcbiAgICBjb25zdCBbciwgZywgYl0gPSByZ2I7XHJcbiAgICBjb25zdCBoZXhSID0gdGhpcy5jb21wb25lbnRUb0hleChyKTtcclxuICAgIGNvbnN0IGhleEcgPSB0aGlzLmNvbXBvbmVudFRvSGV4KGcpO1xyXG4gICAgY29uc3QgaGV4QiA9IHRoaXMuY29tcG9uZW50VG9IZXgoYik7XHJcbiAgICByZXR1cm4gXCIjXCIgKyBoZXhSICsgaGV4RyArIGhleEI7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGNvbXBvbmVudFRvSGV4ID0gKGM6IG51bWJlcikgPT4ge1xyXG4gICAgY29uc3QgaGV4ID0gYy50b1N0cmluZygxNik7XHJcbiAgICByZXR1cm4gaGV4Lmxlbmd0aCA9PT0gMSA/IFwiMFwiICsgaGV4IDogaGV4O1xyXG4gIH1cclxuXHJcbiAgaGV4VG9SZ2IoaGV4OiBzdHJpbmcgfCBudWxsIHwgdW5kZWZpbmVkKTogbnVtYmVyW10ge1xyXG4gICAgaWYgKGhleCA9PT0gbnVsbCB8fCBoZXggPT09IHVuZGVmaW5lZCkgcmV0dXJuIFtdO1xyXG5cclxuICAgIGhleCA9IChoZXgubGVuZ3RoID09PSAzKSA/IGhleCArIGhleCA6IGhleDtcclxuICAgIGNvbnN0IHJlc3VsdCA9IC9eIz8oW2EtZlxcZF17Mn0pKFthLWZcXGRdezJ9KShbYS1mXFxkXXsyfSkkL2kuZXhlYyhoZXgpO1xyXG5cclxuICAgIHJldHVybiByZXN1bHQgPyBbXHJcbiAgICAgIHBhcnNlSW50KHJlc3VsdFsxXSwgMTYpLFxyXG4gICAgICBwYXJzZUludChyZXN1bHRbMl0sIDE2KSxcclxuICAgICAgcGFyc2VJbnQocmVzdWx0WzNdLCAxNilcclxuICAgIF0gOiBbXTtcclxuICB9XHJcbn1cclxuIl19
@@ -1,37 +0,0 @@
1
- import { Directive, HostListener, EventEmitter, Output } from '@angular/core';
2
- import * as i0 from "@angular/core";
3
- export class ColorExtractorDirective {
4
- constructor(elementRef, renderer) {
5
- this.elementRef = elementRef;
6
- this.renderer = renderer;
7
- this.colorValue = new EventEmitter();
8
- }
9
- get currentElement() {
10
- return window.getComputedStyle(this.elementRef.nativeElement);
11
- }
12
- onMouseEnter() {
13
- // console.log('ENTER', this.currentElement)
14
- this.colorValue.emit(this.currentElement.getPropertyValue('background-color'));
15
- }
16
- onMouseLeave() {
17
- // console.log('LEAVE', this.currentElement.getPropertyValue('background-color'))
18
- this.colorValue.emit('white');
19
- }
20
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorExtractorDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); }
21
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: ColorExtractorDirective, selector: "[getColor]", outputs: { colorValue: "colorValue" }, host: { listeners: { "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()" } }, ngImport: i0 }); }
22
- }
23
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorExtractorDirective, decorators: [{
24
- type: Directive,
25
- args: [{
26
- selector: '[getColor]'
27
- }]
28
- }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }]; }, propDecorators: { colorValue: [{
29
- type: Output
30
- }], onMouseEnter: [{
31
- type: HostListener,
32
- args: ['mouseenter']
33
- }], onMouseLeave: [{
34
- type: HostListener,
35
- args: ['mouseleave']
36
- }] } });
37
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sb3ItZXh0cmFjdG9yLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbG9yLXV0aWxzL3NyYy9saWIvY29sb3ItZXh0cmFjdG9yLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUF5QixZQUFZLEVBQUUsWUFBWSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFLckcsTUFBTSxPQUFPLHVCQUF1QjtJQUlsQyxZQUFvQixVQUFzQixFQUFVLFFBQW1CO1FBQW5ELGVBQVUsR0FBVixVQUFVLENBQVk7UUFBVSxhQUFRLEdBQVIsUUFBUSxDQUFXO1FBRjdELGVBQVUsR0FBeUIsSUFBSSxZQUFZLEVBQVUsQ0FBQztJQUVFLENBQUM7SUFFM0UsSUFBSSxjQUFjO1FBQ2hCLE9BQU8sTUFBTSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDaEUsQ0FBQztJQUdELFlBQVk7UUFDViw0Q0FBNEM7UUFDNUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxnQkFBZ0IsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLENBQUE7SUFDaEYsQ0FBQztJQUdELFlBQVk7UUFDVixpRkFBaUY7UUFDakYsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUE7SUFDL0IsQ0FBQzsrR0FwQlUsdUJBQXVCO21HQUF2Qix1QkFBdUI7OzRGQUF2Qix1QkFBdUI7a0JBSG5DLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLFlBQVk7aUJBQ3ZCO3lIQUdXLFVBQVU7c0JBQW5CLE1BQU07Z0JBU1AsWUFBWTtzQkFEWCxZQUFZO3VCQUFDLFlBQVk7Z0JBTzFCLFlBQVk7c0JBRFgsWUFBWTt1QkFBQyxZQUFZIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBFbGVtZW50UmVmLCBSZW5kZXJlcjIsIEhvc3RMaXN0ZW5lciwgRXZlbnRFbWl0dGVyLCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2dldENvbG9yXSdcbn0pXG5leHBvcnQgY2xhc3MgQ29sb3JFeHRyYWN0b3JEaXJlY3RpdmUge1xuXG4gIEBPdXRwdXQoKSBjb2xvclZhbHVlOiBFdmVudEVtaXR0ZXI8c3RyaW5nPiA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgZWxlbWVudFJlZjogRWxlbWVudFJlZiwgcHJpdmF0ZSByZW5kZXJlcjogUmVuZGVyZXIyKSB7fVxuXG4gIGdldCBjdXJyZW50RWxlbWVudCgpIHtcbiAgICByZXR1cm4gd2luZG93LmdldENvbXB1dGVkU3R5bGUodGhpcy5lbGVtZW50UmVmLm5hdGl2ZUVsZW1lbnQpO1xuICB9XG5cbiAgQEhvc3RMaXN0ZW5lcignbW91c2VlbnRlcicpXG4gIG9uTW91c2VFbnRlcigpIHtcbiAgICAvLyBjb25zb2xlLmxvZygnRU5URVInLCB0aGlzLmN1cnJlbnRFbGVtZW50KVxuICAgIHRoaXMuY29sb3JWYWx1ZS5lbWl0KHRoaXMuY3VycmVudEVsZW1lbnQuZ2V0UHJvcGVydHlWYWx1ZSgnYmFja2dyb3VuZC1jb2xvcicpKVxuICB9XG5cbiAgQEhvc3RMaXN0ZW5lcignbW91c2VsZWF2ZScpXG4gIG9uTW91c2VMZWF2ZSgpIHtcbiAgICAvLyBjb25zb2xlLmxvZygnTEVBVkUnLCB0aGlzLmN1cnJlbnRFbGVtZW50LmdldFByb3BlcnR5VmFsdWUoJ2JhY2tncm91bmQtY29sb3InKSlcbiAgICB0aGlzLmNvbG9yVmFsdWUuZW1pdCgnd2hpdGUnKVxuICB9XG5cbn1cbiJdfQ==
@@ -1,79 +0,0 @@
1
- import { Injectable, inject } from '@angular/core';
2
- import { TextColorService } from './text-color.service';
3
- import * as i0 from "@angular/core";
4
- export class ColorLightenDarkenService {
5
- // const color = '#3498db'; // Your color
6
- // const lighterColor = lighten(color, 0.2); // 20% lighter
7
- // const darkerColor = darken(color, 0.2); // 20% darker
8
- // console.log(lighterColor, darkerColor);
9
- constructor() {
10
- this.colors = inject(TextColorService);
11
- }
12
- lighten(color, amount) {
13
- const rgb = this.colors.fixColor(color);
14
- // const rgb = color.match(/\w\w/g)?.map((x) => parseInt(x, 16)) || [];
15
- // Convert RGB to HSL
16
- let [r, g, b] = rgb.map((c) => c / 255);
17
- const max = Math.max(r, g, b), min = Math.min(r, g, b);
18
- let h = 0, s = 0, l = (max + min) / 2;
19
- if (max !== min) {
20
- const d = max - min;
21
- s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
22
- switch (max) {
23
- case r:
24
- h = (g - b) / d + (g < b ? 6 : 0);
25
- break;
26
- case g:
27
- h = (b - r) / d + 2;
28
- break;
29
- case b:
30
- h = (r - g) / d + 4;
31
- break;
32
- }
33
- h /= 6;
34
- }
35
- // Modify the lightness and clamp it to [0, 1]
36
- l = Math.min(1, l + amount);
37
- // Convert HSL back to RGB
38
- if (s === 0) {
39
- r = g = b = l; // achromatic
40
- }
41
- else {
42
- const hue2rgb = (p, q, t) => {
43
- if (t < 0)
44
- t += 1;
45
- if (t > 1)
46
- t -= 1;
47
- if (t < 1 / 6)
48
- return p + (q - p) * 6 * t;
49
- if (t < 1 / 2)
50
- return q;
51
- if (t < 2 / 3)
52
- return p + (q - p) * (2 / 3 - t) * 6;
53
- return p;
54
- };
55
- const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
56
- const p = 2 * l - q;
57
- r = hue2rgb(p, q, h + 1 / 3);
58
- g = hue2rgb(p, q, h);
59
- b = hue2rgb(p, q, h - 1 / 3);
60
- }
61
- // Convert RGB back to hexadecimal color
62
- const toHex = (x) => Math.round(x * 255)
63
- .toString(16)
64
- .padStart(2, '0');
65
- return `#${toHex(r)}${toHex(g)}${toHex(b)}`;
66
- }
67
- darken(color, amount) {
68
- return this.lighten(color, -amount);
69
- }
70
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorLightenDarkenService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
71
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorLightenDarkenService, providedIn: 'root' }); }
72
- }
73
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorLightenDarkenService, decorators: [{
74
- type: Injectable,
75
- args: [{
76
- providedIn: 'root'
77
- }]
78
- }], ctorParameters: function () { return []; } });
79
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sb3ItbGlnaHRlbi1kYXJrZW4uc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbG9yLXV0aWxzL3NyYy9saWIvY29sb3ItbGlnaHRlbi1kYXJrZW4uc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNuRCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQzs7QUFLeEQsTUFBTSxPQUFPLHlCQUF5QjtJQUlwQyx5Q0FBeUM7SUFDekMsMkRBQTJEO0lBQzNELDBEQUEwRDtJQUUxRCwwQ0FBMEM7SUFFNUM7UUFSRSxXQUFNLEdBQUcsTUFBTSxDQUFDLGdCQUFnQixDQUFDLENBQUE7SUFRbkIsQ0FBQztJQUVmLE9BQU8sQ0FBQyxLQUFhLEVBQUUsTUFBYztRQUVuQyxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQTtRQUN2Qyx1RUFBdUU7UUFFdkUscUJBQXFCO1FBQ3JCLElBQUksQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsR0FBRyxHQUFHLENBQUMsQ0FBQztRQUN4QyxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLEVBQzNCLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDMUIsSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUNQLENBQUMsR0FBRyxDQUFDLEVBQ0wsQ0FBQyxHQUFHLENBQUMsR0FBRyxHQUFHLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUV0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S0FBSyxDQUFDLENBQUMsR0FBRyxHQUFHLENBQUM7YUFDaEIsUUFBUSxDQUFDLEVBQUUsQ0FBQzthQUNaLFFBQVEsQ0FBQyxDQUFDLEVBQUUsR0FBRyxDQUFDLENBQUM7UUFDdEIsT0FBTyxJQUFJLEtBQUssQ0FBQyxDQUFDLENBQUMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7SUFDOUMsQ0FBQztJQUVELE1BQU0sQ0FBQyxLQUFhLEVBQUUsTUFBYztRQUNsQyxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsS0FBSyxFQUFFLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDdEMsQ0FBQzsrR0ExRVUseUJBQXlCO21IQUF6Qix5QkFBeUIsY0FGeEIsTUFBTTs7NEZBRVAseUJBQXlCO2tCQUhyQyxVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUsIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgVGV4dENvbG9yU2VydmljZSB9IGZyb20gJy4vdGV4dC1jb2xvci5zZXJ2aWNlJztcblxuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgQ29sb3JMaWdodGVuRGFya2VuU2VydmljZSB7XG5cbiAgY29sb3JzID0gaW5qZWN0KFRleHRDb2xvclNlcnZpY2UpXG5cbiAgLy8gY29uc3QgY29sb3IgPSAnIzM0OThkYic7IC8vIFlvdXIgY29sb3JcbiAgLy8gY29uc3QgbGlnaHRlckNvbG9yID0gbGlnaHRlbihjb2xvciwgMC4yKTsgLy8gMjAlIGxpZ2h0ZXJcbiAgLy8gY29uc3QgZGFya2VyQ29sb3IgPSBkYXJrZW4oY29sb3IsIDAuMik7ICAgLy8gMjAlIGRhcmtlclxuXG4gIC8vIGNvbnNvbGUubG9nKGxpZ2h0ZXJDb2xvciwgZGFya2VyQ29sb3IpO1xuXG5jb25zdHJ1Y3RvcigpIHsgfVxuXG4gIGxpZ2h0ZW4oY29sb3I6IHN0cmluZywgYW1vdW50OiBudW1iZXIpIHtcblxuICAgIGNvbnN0IHJnYiA9IHRoaXMuY29sb3JzLmZpeENvbG9yKGNvbG9yKVxuICAgIC8vIGNvbnN0IHJnYiA9IGNvbG9yLm1hdGNoKC9cXHdcXHcvZyk/Lm1hcCgoeCkgPT4gcGFyc2VJbnQoeCwgMTYpKSB8fCBbXTtcblxuICAgIC8vIENvbnZlcnQgUkdCIHRvIEhTTFxuICAgIGxldCBbciwgZywgYl0gPSByZ2IubWFwKChjKSA9PiBjIC8gMjU1KTtcbiAgICBjb25zdCBtYXggPSBNYXRoLm1heChyLCBnLCBiKSxcbiAgICAgIG1pbiA9IE1hdGgubWluKHIsIGcsIGIpO1xuICAgIGxldCBoID0gMCxcbiAgICAgIHMgPSAwLFxuICAgICAgbCA9IChtYXggKyBtaW4pIC8gMjtcblxuICAgIGlmIChtYXggIT09IG1pbikge1xuICAgICAgY29uc3QgZCA9IG1heCAtIG1pbjtcbiAgICAgIHMgPSBsID4gMC41ID8gZCAvICgyIC0gbWF4IC0gbWluKSA6IGQgLyAobWF4ICsgbWluKTtcbiAgICAgIHN3aXRjaCAobWF4KSB7XG4gICAgICAgIGNhc2UgcjpcbiAgICAgICAgICBoID0gKGcgLSBiKSAvIGQgKyAoZyA8IGIgPyA2IDogMCk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgZzpcbiAgICAgICAgICBoID0gKGIgLSByKSAvIGQgKyAyO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIGI6XG4gICAgICAgICAgaCA9IChyIC0gZykgLyBkICsgNDtcbiAgICAgICAgICBicmVhaztcbiAgICAgIH1cbiAgICAgIGggLz0gNjtcbiAgICB9XG5cbiAgICAvLyBNb2RpZnkgdGhlIGxpZ2h0bmVzcyBhbmQgY2xhbXAgaXQgdG8gWzAsIDFdXG4gICAgbCA9IE1hdGgubWluKDEsIGwgKyBhbW91bnQpO1xuXG4gICAgLy8gQ29udmVydCBIU0wgYmFjayB0byBSR0JcbiAgICBpZiAocyA9PT0gMCkge1xuICAgICAgciA9IGcgPSBiID0gbDsgLy8gYWNocm9tYXRpY1xuICAgIH0gZWxzZSB7XG4gICAgICBjb25zdCBodWUycmdiID0gKHA6IG51bWJlciwgcTogbnVtYmVyLCB0OiBudW1iZXIpID0+IHtcbiAgICAgICAgaWYgKHQgPCAwKSB0ICs9IDE7XG4gICAgICAgIGlmICh0ID4gMSkgdCAtPSAxO1xuICAgICAgICBpZiAodCA8IDEgLyA2KSByZXR1cm4gcCArIChxIC0gcCkgKiA2ICogdDtcbiAgICAgICAgaWYgKHQgPCAxIC8gMikgcmV0dXJuIHE7XG4gICAgICAgIGlmICh0IDwgMiAvIDMpIHJldHVybiBwICsgKHEgLSBwKSAqICgyIC8gMyAtIHQpICogNjtcbiAgICAgICAgcmV0dXJuIHA7XG4gICAgICB9O1xuICAgICAgY29uc3QgcSA9IGwgPCAwLjUgPyBsICogKDEgKyBzKSA6IGwgKyBzIC0gbCAqIHM7XG4gICAgICBjb25zdCBwID0gMiAqIGwgLSBxO1xuICAgICAgciA9IGh1ZTJyZ2IocCwgcSwgaCArIDEgLyAzKTtcbiAgICAgIGcgPSBodWUycmdiKHAsIHEsIGgpO1xuICAgICAgYiA9IGh1ZTJyZ2IocCwgcSwgaCAtIDEgLyAzKTtcbiAgICB9XG5cbiAgICAvLyBDb252ZXJ0IFJHQiBiYWNrIHRvIGhleGFkZWNpbWFsIGNvbG9yXG4gICAgY29uc3QgdG9IZXggPSAoeDogbnVtYmVyKSA9PlxuICAgICAgTWF0aC5yb3VuZCh4ICogMjU1KVxuICAgICAgICAudG9TdHJpbmcoMTYpXG4gICAgICAgIC5wYWRTdGFydCgyLCAnMCcpO1xuICAgIHJldHVybiBgIyR7dG9IZXgocil9JHt0b0hleChnKX0ke3RvSGV4KGIpfWA7XG4gIH1cblxuICBkYXJrZW4oY29sb3I6IHN0cmluZywgYW1vdW50OiBudW1iZXIpIHtcbiAgICByZXR1cm4gdGhpcy5saWdodGVuKGNvbG9yLCAtYW1vdW50KTtcbiAgfVxuXG59XG4iXX0=
@@ -1,172 +0,0 @@
1
- import { Injectable } from '@angular/core';
2
- import { BehaviorSubject } from 'rxjs';
3
- import * as i0 from "@angular/core";
4
- import * as i1 from "./color-conversion.service";
5
- export class ColorPalletteService {
6
- constructor(colorConversionService) {
7
- this.colorConversionService = colorConversionService;
8
- // define image path
9
- // this.colorSelectionService.getColorsFromImage('../assets/sample2.jpg')
10
- // get colors
11
- // this.colorSelectionService.palette.subscribe(data => this.palette = data)
12
- // sample html
13
- // <div *ngFor="let color of palette">
14
- // <div style="display: flex;">
15
- // <div
16
- // class="box"
17
- // [style.background-color]="color.color"
18
- // >
19
- // Color
20
- // </div>
21
- // <div
22
- // class="box"
23
- // [style.background-color]="color.complementaryColor"
24
- // >
25
- // Complementary
26
- // </div>
27
- // </div>
28
- // </div>
29
- // CSS
30
- // .box {
31
- // width: 100px;
32
- // height: 100px;
33
- // border: solid thin black;
34
- // color: black;
35
- // margin: 4px;
36
- // padding: 16px;
37
- // display: flex;
38
- // flex-wrap: wrap;
39
- // align-content: center;
40
- // justify-content: center;
41
- // }
42
- this.palette = new BehaviorSubject([]);
43
- this.palette$ = this.palette.asObservable();
44
- }
45
- /**
46
- * Retrieves a color palette from an image at the specified path.
47
- *
48
- * @param imagePath - The path to the image to extract the color palette from.
49
- * @param colors - The number of colors to include in the palette (default is 3).
50
- * @returns An observable that emits the generated color palette.
51
- */
52
- getColorsFromImage(imagePath, colors = 3) {
53
- const image = new Image();
54
- image.src = imagePath;
55
- image.onload = () => {
56
- const data = this.generateColorPalette(image, colors) || [];
57
- this.palette.next(data);
58
- };
59
- }
60
- /**
61
- * Generates a color palette from an image.
62
- *
63
- * @param image - The HTML image element to extract the color palette from.
64
- * @param colorCount - The number of colors to include in the palette (default is 6).
65
- * @returns An array of color objects, each with a hex color and a complementary hex color.
66
- */
67
- generateColorPalette(image, colorCount = 6) {
68
- const canvas = document.createElement("canvas");
69
- const context = canvas.getContext("2d");
70
- if (!context)
71
- return;
72
- canvas.width = image.width;
73
- canvas.height = image.height;
74
- context.drawImage(image, 0, 0);
75
- // Get the image data
76
- const imageData = context.getImageData(0, 0, canvas.width, canvas.height);
77
- const pixels = imageData.data;
78
- const pixelCount = imageData.width * imageData.height;
79
- // Build an array of RGB colors
80
- const colors = [];
81
- for (let i = 0; i < pixelCount; i++) {
82
- const offset = i * 4;
83
- const r = pixels[offset];
84
- const g = pixels[offset + 1];
85
- const b = pixels[offset + 2];
86
- colors.push([r, g, b]);
87
- }
88
- // Apply color quantization using k-means clustering
89
- const quantizedColors = this.kMeansColorQuantization(colors, colorCount);
90
- // Order colors by luminance
91
- quantizedColors.sort((color1, color2) => {
92
- const luminance1 = this.getLuminance(color1);
93
- const luminance2 = this.getLuminance(color2);
94
- return luminance2 - luminance1;
95
- });
96
- const palette = quantizedColors.map((color) => {
97
- const complementaryColor = color.map((component) => 255 - component);
98
- const hexColor = this.colorConversionService.rgbToHex(color);
99
- const hexComplementaryColor = this.colorConversionService.rgbToHex(complementaryColor);
100
- return { color: hexColor, complementaryColor: hexComplementaryColor };
101
- });
102
- return palette;
103
- }
104
- getLuminance(color) {
105
- const [r, g, b] = color;
106
- return 0.299 * r + 0.587 * g + 0.114 * b;
107
- }
108
- calculateColorDistance(color1, color2) {
109
- const [r1, g1, b1] = color1;
110
- const [r2, g2, b2] = color2;
111
- const dr = r2 - r1;
112
- const dg = g2 - g1;
113
- const db = b2 - b1;
114
- return Math.sqrt(dr * dr + dg * dg + db * db);
115
- }
116
- calculateMeanColor(colors) {
117
- let sumR = 0;
118
- let sumG = 0;
119
- let sumB = 0;
120
- for (let i = 0; i < colors.length; i++) {
121
- const [r, g, b] = colors[i];
122
- sumR += r;
123
- sumG += g;
124
- sumB += b;
125
- }
126
- const meanR = Math.round(sumR / colors.length);
127
- const meanG = Math.round(sumG / colors.length);
128
- const meanB = Math.round(sumB / colors.length);
129
- return [meanR, meanG, meanB];
130
- }
131
- kMeansColorQuantization(colors, k) {
132
- let clusterCenters = [];
133
- for (let i = 0; i < k; i++) {
134
- const randomColor = colors[Math.floor(Math.random() * colors.length)];
135
- clusterCenters.push(randomColor);
136
- }
137
- let clusters = [];
138
- for (let i = 0; i < colors.length; i++) {
139
- const color = colors[i];
140
- let minDistance = Infinity;
141
- let nearestCenter = null;
142
- for (let j = 0; j < clusterCenters.length; j++) {
143
- const center = clusterCenters[j];
144
- const distance = this.calculateColorDistance(color, center);
145
- if (distance < minDistance) {
146
- minDistance = distance;
147
- nearestCenter = center;
148
- }
149
- }
150
- clusters.push({ color, center: nearestCenter });
151
- }
152
- let updatedCenters = [];
153
- for (let i = 0; i < clusterCenters.length; i++) {
154
- const center = clusterCenters[i];
155
- const clusterColors = clusters.filter(c => c.center === center).map(c => c.color);
156
- if (clusterColors.length > 0) {
157
- const updatedCenter = this.calculateMeanColor(clusterColors);
158
- updatedCenters.push(updatedCenter);
159
- }
160
- }
161
- return updatedCenters;
162
- }
163
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorPalletteService, deps: [{ token: i1.ColorConversionService }], target: i0.ɵɵFactoryTarget.Injectable }); }
164
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorPalletteService, providedIn: 'root' }); }
165
- }
166
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ColorPalletteService, decorators: [{
167
- type: Injectable,
168
- args: [{
169
- providedIn: 'root'
170
- }]
171
- }], ctorParameters: function () { return [{ type: i1.ColorConversionService }]; } });
172
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29sb3ItcGFsbGV0dGUuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbG9yLXV0aWxzL3NyYy9saWIvY29sb3ItcGFsbGV0dGUuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFBO0FBQzFDLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxNQUFNLENBQUE7OztBQU10QyxNQUFNLE9BQU8sb0JBQW9CO0lBMkMvQixZQUNVLHNCQUE4QztRQUE5QywyQkFBc0IsR0FBdEIsc0JBQXNCLENBQXdCO1FBMUN4RCxvQkFBb0I7UUFDcEIseUVBQXlFO1FBRXpFLGFBQWE7UUFDYiw0RUFBNEU7UUFFNUUsY0FBYztRQUNkLHdDQUF3QztRQUN4QyxtQ0FBbUM7UUFDbkMsZUFBZTtRQUNmLDBCQUEwQjtRQUMxQixxREFBcUQ7UUFDckQsWUFBWTtRQUNaLGdCQUFnQjtRQUNoQixpQkFBaUI7UUFDakIsZUFBZTtRQUNmLDBCQUEwQjtRQUMxQixrRUFBa0U7UUFDbEUsWUFBWTtRQUNaLHdCQUF3QjtRQUN4QixpQkFBaUI7UUFDakIsYUFBYTtRQUNiLFNBQVM7UUFFVCxNQUFNO1FBQ04sV0FBVztRQUNYLG9CQUFvQjtRQUNwQixxQkFBcUI7UUFDckIsZ0NBQWdDO1FBQ2hDLG9CQUFvQjtRQUNwQixtQkFBbUI7UUFDbkIscUJBQXFCO1FBQ3JCLHFCQUFxQjtRQUNyQix1QkFBdUI7UUFDdkIsNkJBQTZCO1FBQzdCLCtCQUErQjtRQUMvQixJQUFJO1FBRUksWUFBTyxHQUFHLElBQUksZUFBZSxDQUFrRCxFQUFFLENBQUMsQ0FBQTtRQUMxRixhQUFRLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsQ0FBQTtJQUlsQyxDQUFDO0lBRUw7Ozs7OztPQU1HO0lBQ0gsa0JBQWtCLENBQUMsU0FBaUIsRUFBRSxNQUFNLEdBQUcsQ0FB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QUU7WUFDdEMsTUFBTSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFBO1lBQzNCLElBQUksSUFBSSxDQUFDLENBQUE7WUFDVCxJQUFJLElBQUksQ0FBQyxDQUFBO1lBQ1QsSUFBSSxJQUFJLENBQUMsQ0FBQTtTQUNWO1FBQ0QsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFBO1FBQzlDLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxHQUFHLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQTtRQUM5QyxNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUE7UUFDOUMsT0FBTyxDQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUE7SUFDOUIsQ0FBQztJQUVPLHVCQUF1QixDQUFDLE1BQWtCLEVBQUUsQ0FBUztRQUMzRCxJQUFJLGNBQWMsR0FBRyxFQUFFLENBQUE7UUFDdkIsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUMxQixNQUFNLFdBQVcsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUE7WUFDckUsY0FBYyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQTtTQUNqQztRQUVELElBQUksUUFBUSxHQUFHLEVBQUUsQ0FBQTtRQUNqQixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtZQUN0QyxNQUFNLEtBQUssR0FBRyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUE7WUFDdkIsSUFBSSxXQUFXLEdBQUcsUUFBUSxDQUFBO1lBQzFCLElBQUksYUFBYSxHQUFHLElBQUksQ0FBQTtZQUN4QixLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsY0FBYyxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtnQkFDOUMsTUFBTSxNQUFNLEdBQUcsY0FBYyxDQUFDLENBQUMsQ0FBQyxDQUFBO2dCQUNoQyxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsc0JBQXNCLENBQUMsS0FBSyxFQUFFLE1BQU0sQ0FBQyxDQUFBO2dCQUMzRCxJQUFJLFFBQVEsR0FBRyxXQUFXLEVBQUU7b0JBQzFCLFdBQVcsR0FBRyxRQUFRLENBQUE7b0JBQ3RCLGFBQWEsR0FBRyxNQUFNLENBQUE7aUJBQ3ZCO2FBQ0Y7WUFDRCxRQUFRLENBQUMsSUFBSSxDQUFDLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxhQUFhLEVBQUUsQ0FBQyxDQUFBO1NBQ2hEO1FBRUQsSUFBSSxjQUFjLEdBQUcsRUFBRSxDQUFBO1FBQ3ZCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxjQUFjLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO1lBQzlDLE1BQU0sTUFBTSxHQUFHLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQTtZQUNoQyxNQUFNLGFBQWEsR0FBRyxRQUFRLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLE1BQU0sS0FBSyxNQUFNLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUE7WUFDakYsSUFBSSxhQUFhLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtnQkFDNUIsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixDQUFDLGFBQWEsQ0FBQyxDQUFBO2dCQUM1RCxjQUFjLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFBO2FBQ25DO1NBQ0Y7UUFFRCxPQUFPLGNBQWMsQ0FBQTtJQUN2QixDQUFDOytHQXJMVSxvQkFBb0I7bUhBQXBCLG9CQUFvQixjQUZuQixNQUFNOzs0RkFFUCxvQkFBb0I7a0JBSGhDLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnXHJcbmltcG9ydCB7IEJlaGF2aW9yU3ViamVjdCB9IGZyb20gJ3J4anMnXHJcbmltcG9ydCB7IENvbG9yQ29udmVyc2lvblNlcnZpY2UgfSBmcm9tICcuL2NvbG9yLWNvbnZlcnNpb24uc2VydmljZSdcclxuXHJcbkBJbmplY3RhYmxlKHtcclxuICBwcm92aWRlZEluOiAncm9vdCdcclxufSlcclxuZXhwb3J0IGNsYXNzIENvbG9yUGFsbGV0dGVTZXJ2aWNlIHtcclxuXHJcbiAgLy8gZGVmaW5lIGltYWdlIHBhdGhcclxuICAvLyB0aGlzLmNvbG9yU2VsZWN0aW9uU2VydmljZS5nZXRDb2xvcnNGcm9tSW1hZ2UoJy4uL2Fzc2V0cy9zYW1wbGUyLmpwZycpXHJcblxyXG4gIC8vIGdldCBjb2xvcnNcclxuICAvLyB0aGlzLmNvbG9yU2VsZWN0aW9uU2VydmljZS5wYWxldHRlLnN1YnNjcmliZShkYXRhID0+IHRoaXMucGFsZXR0ZSA9IGRhdGEpXHJcblxyXG4gIC8vIHNhbXBsZSBodG1sXHJcbiAgLy8gICA8ZGl2ICpuZ0Zvcj1cImxldCBjb2xvciBvZiBwYWxldHRlXCI+XHJcbiAgLy8gICAgIDxkaXYgc3R5bGU9XCJkaXNwbGF5OiBmbGV4O1wiPlxyXG4gIC8vICAgICAgICAgPGRpdlxyXG4gIC8vICAgICAgICAgICAgIGNsYXNzPVwiYm94XCJcclxuICAvLyAgICAgICAgICAgICBbc3R5bGUuYmFja2dyb3VuZC1jb2xvcl09XCJjb2xvci5jb2xvclwiXHJcbiAgLy8gICAgICAgICA+XHJcbiAgLy8gICAgICAgICBDb2xvclxyXG4gIC8vICAgICAgICAgPC9kaXY+XHJcbiAgLy8gICAgICAgICA8ZGl2XHJcbiAgLy8gICAgICAgICAgICAgY2xhc3M9XCJib3hcIlxyXG4gIC8vICAgICAgICAgICAgIFtzdHlsZS5iYWNrZ3JvdW5kLWNvbG9yXT1cImNvbG9yLmNvbXBsZW1lbnRhcnlDb2xvclwiXHJcbiAgLy8gICAgICAgICA+XHJcbiAgLy8gICAgICAgICBDb21wbGVtZW50YXJ5XHJcbiAgLy8gICAgICAgICA8L2Rpdj5cclxuICAvLyAgICAgPC9kaXY+XHJcbiAgLy8gPC9kaXY+XHJcblxyXG4gIC8vIENTU1xyXG4gIC8vICAgLmJveCB7XHJcbiAgLy8gICAgIHdpZHRoOiAxMDBweDtcclxuICAvLyAgICAgaGVpZ2h0OiAxMDBweDtcclxuICAvLyAgICAgYm9yZGVyOiBzb2xpZCB0aGluIGJsYWNrO1xyXG4gIC8vICAgICBjb2xvcjogYmxhY2s7XHJcbiAgLy8gICAgIG1hcmdpbjogNHB4O1xyXG4gIC8vICAgICBwYWRkaW5nOiAxNnB4O1xyXG4gIC8vICAgICBkaXNwbGF5OiBmbGV4O1xyXG4gIC8vICAgICBmbGV4LXdyYXA6IHdyYXA7XHJcbiAgLy8gICAgIGFsaWduLWNvbnRlbnQ6IGNlbnRlcjtcclxuICAvLyAgICAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XHJcbiAgLy8gfVxyXG5cclxuICBwcml2YXRlIHBhbGV0dGUgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PHsgY29sb3I6IHN0cmluZywgY29tcGxlbWVudGFyeUNvbG9yOiBzdHJpbmcgfVtdPihbXSlcclxuICBwYWxldHRlJCA9IHRoaXMucGFsZXR0ZS5hc09ic2VydmFibGUoKVxyXG5cclxuICBjb25zdHJ1Y3RvcihcclxuICAgIHByaXZhdGUgY29sb3JDb252ZXJzaW9uU2VydmljZTogQ29sb3JDb252ZXJzaW9uU2VydmljZVxyXG4gICkgeyB9XHJcblxyXG4gIC8qKlxyXG4gICAqIFJldHJpZXZlcyBhIGNvbG9yIHBhbGV0dGUgZnJvbSBhbiBpbWFnZSBhdCB0aGUgc3BlY2lmaWVkIHBhdGguXHJcbiAgICpcclxuICAgKiBAcGFyYW0gaW1hZ2VQYXRoIC0gVGhlIHBhdGggdG8gdGhlIGltYWdlIHRvIGV4dHJhY3QgdGhlIGNvbG9yIHBhbGV0dGUgZnJvbS5cclxuICAgKiBAcGFyYW0gY29sb3JzIC0gVGhlIG51bWJlciBvZiBjb2xvcnMgdG8gaW5jbHVkZSBpbiB0aGUgcGFsZXR0ZSAoZGVmYXVsdCBpcyAzKS5cclxuICAgKiBAcmV0dXJucyBBbiBvYnNlcnZhYmxlIHRoYXQgZW1pdHMgdGhlIGdlbmVyYXRlZCBjb2xvciBwYWxldHRlLlxyXG4gICAqL1xyXG4gIGdldENvbG9yc0Zyb21JbWFnZShpbWFnZVBhdGg6IHN0cmluZywgY29sb3JzID0gMykge1xyXG4gICAgY29uc3QgaW1hZ2UgPSBuZXcgSW1hZ2UoKTtcclxuICAgIGltYWdlLnNyYyA9IGltYWdlUGF0aDtcclxuXHJcbiAgICBpbWFnZS5vbmxvYWQgPSAoKSA9PiB7XHJcbiAgICAgIGNvbnN0IGRhdGEgPSB0aGlzLmdlbmVyYXRlQ29sb3JQYWxldHRlKGltYWdlLCBjb2xvcnMpIHx8IFtdO1xyXG4gICAgICB0aGlzLnBhbGV0dGUubmV4dChkYXRhKTtcclxuICAgIH07XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBHZW5lcmF0ZXMgYSBjb2xvciBwYWxldHRlIGZyb20gYW4gaW1hZ2UuXHJcbiAgICpcclxuICAgKiBAcGFyYW0gaW1hZ2UgLSBUaGUgSFRNTCBpbWFnZSBlbGVtZW50IHRvIGV4dHJhY3QgdGhlIGNvbG9yIHBhbGV0dGUgZnJvbS5cclxuICAgKiBAcGFyYW0gY29sb3JDb3VudCAtIFRoZSBudW1iZXIgb2YgY29sb3JzIHRvIGluY2x1ZGUgaW4gdGhlIHBhbGV0dGUgKGRlZmF1bHQgaXMgNikuXHJcbiAgICogQHJldHVybnMgQW4gYXJyYXkgb2YgY29sb3Igb2JqZWN0cywgZWFjaCB3aXRoIGEgaGV4IGNvbG9yIGFuZCBhIGNvbXBsZW1lbnRhcnkgaGV4IGNvbG9yLlxyXG4gICAqL1xyXG4gIGdlbmVyYXRlQ29sb3JQYWxldHRlKGltYWdlOiBIVE1MSW1hZ2VFbGVtZW50LCBjb2xvckNvdW50ID0gNikge1xyXG4gICAgY29uc3QgY2FudmFzID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcImNhbnZhc1wiKTtcclxuICAgIGNvbnN0IGNvbnRleHQgPSBjYW52YXMuZ2V0Q29udGV4dChcIjJkXCIpO1xyXG5cclxuICAgIGlmICghY29udGV4dCkgcmV0dXJuO1xyXG5cclxuICAgIGNhbnZhcy53aWR0aCA9IGltYWdlLndpZHRoO1xyXG4gICAgY2FudmFzLmhlaWdodCA9IGltYWdlLmhlaWdodDtcclxuICAgIGNvbnRleHQuZHJhd0ltYWdlKGltYWdlLCAwLCAwKTtcclxuXHJcbiAgICAvLyBHZXQgdGhlIGltYWdlIGRhdGFcclxuICAgIGNvbnN0IGltYWdlRGF0YSA9IGNvbnRleHQuZ2V0SW1hZ2VEYXRhKDAsIDAsIGNhbnZhcy53aWR0aCwgY2FudmFzLmhlaWdodCk7XHJcbiAgICBjb25zdCBwaXhlbHMgPSBpbWFnZURhdGEuZGF0YTtcclxuICAgIGNvbnN0IHBpeGVsQ291bnQgPSBpbWFnZURhdGEud2lkdGggKiBpbWFnZURhdGEuaGVpZ2h0O1xyXG5cclxuICAgIC8vIEJ1aWxkIGFuIGFycmF5IG9mIFJHQiBjb2xvcnNcclxuICAgIGNvbnN0IGNvbG9ycyA9IFtdO1xyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBwaXhlbENvdW50OyBpKyspIHtcclxuICAgICAgY29uc3Qgb2Zmc2V0ID0gaSAqIDQ7XHJcbiAgICAgIGNvbnN0IHIgPSBwaXhlbHNbb2Zmc2V0XTtcclxuICAgICAgY29uc3QgZyA9IHBpeGVsc1tvZmZzZXQgKyAxXTtcclxuICAgICAgY29uc3QgYiA9IHBpeGVsc1tvZmZzZXQgKyAyXTtcclxuICAgICAgY29sb3JzLnB1c2goW3IsIGcsIGJdKTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBBcHBseSBjb2xvciBxdWFudGl6YXRpb24gdXNpbmcgay1tZWFucyBjbHVzdGVyaW5nXHJcbiAgICBjb25zdCBxdWFudGl6ZWRDb2xvcnMgPSB0aGlzLmtNZWFuc0NvbG9yUXVhbnRpemF0aW9uKGNvbG9ycywgY29sb3JDb3VudCk7XHJcblxyXG4gICAgLy8gT3JkZXIgY29sb3JzIGJ5IGx1bWluYW5jZVxyXG4gICAgcXVhbnRpemVkQ29sb3JzLnNvcnQoKGNvbG9yMSwgY29sb3IyKSA9PiB7XHJcbiAgICAgIGNvbnN0IGx1bWluYW5jZTEgPSB0aGlzLmdldEx1bWluYW5jZShjb2xvcjEpO1xyXG4gICAgICBjb25zdCBsdW1pbmFuY2UyID0gdGhpcy5nZXRMdW1pbmFuY2UoY29sb3IyKTtcclxuICAgICAgcmV0dXJuIGx1bWluYW5jZTIgLSBsdW1pbmFuY2UxO1xyXG4gICAgfSk7XHJcblxyXG4gICAgY29uc3QgcGFsZXR0ZSA9IHF1YW50aXplZENvbG9ycy5tYXAoKGNvbG9yKSA9PiB7XHJcbiAgICAgIGNvbnN0IGNvbXBsZW1lbnRhcnlDb2xvciA9IGNvbG9yLm1hcCgoY29tcG9uZW50KSA9PiAyNTUgLSBjb21wb25lbnQpO1xyXG4gICAgICBjb25zdCBoZXhDb2xvciA9IHRoaXMuY29sb3JDb252ZXJzaW9uU2VydmljZS5yZ2JUb0hleChjb2xvcik7XHJcbiAgICAgIGNvbnN0IGhleENvbXBsZW1lbnRhcnlDb2xvciA9XHJcbiAgICAgICAgdGhpcy5jb2xvckNvbnZlcnNpb25TZXJ2aWNlLnJnYlRvSGV4KGNvbXBsZW1lbnRhcnlDb2xvcik7XHJcbiAgICAgIHJldHVybiB7IGNvbG9yOiBoZXhDb2xvciwgY29tcGxlbWVudGFyeUNvbG9yOiBoZXhDb21wbGVtZW50YXJ5Q29sb3IgfTtcclxuICAgIH0pO1xyXG5cclxuICAgIHJldHVybiBwYWxldHRlO1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBnZXRMdW1pbmFuY2UoY29sb3I6IG51bWJlcltdKSB7XHJcbiAgICBjb25zdCBbciwgZywgYl0gPSBjb2xvclxyXG4gICAgcmV0dXJuIDAuMjk5ICogciArIDAuNTg3ICogZyArIDAuMTE0ICogYlxyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBjYWxjdWxhdGVDb2xvckRpc3RhbmNlKGNvbG9yMTogbnVtYmVyW10sIGNvbG9yMjogbnVtYmVyW10pIHtcclxuICAgIGNvbnN0IFtyMSwgZzEsIGIxXSA9IGNvbG9yMVxyXG4gICAgY29uc3QgW3IyLCBnMiwgYjJdID0gY29sb3IyXHJcbiAgICBjb25zdCBkciA9IHIyIC0gcjFcclxuICAgIGNvbnN0IGRnID0gZzIgLSBnMVxyXG4gICAgY29uc3QgZGIgPSBiMiAtIGIxXHJcbiAgICByZXR1cm4gTWF0aC5zcXJ0KGRyICogZHIgKyBkZyAqIGRnICsgZGIgKiBkYilcclxuICB9XHJcblxyXG4gIHByaXZhdGUgY2FsY3VsYXRlTWVhbkNvbG9yKGNvbG9yczogbnVtYmVyW11bXSkge1xyXG4gICAgbGV0IHN1bVIgPSAwXHJcbiAgICBsZXQgc3VtRyA9IDBcclxuICAgIGxldCBzdW1CID0gMFxyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBjb2xvcnMubGVuZ3RoOyBpKyspIHtcclxuICAgICAgY29uc3QgW3IsIGcsIGJdID0gY29sb3JzW2ldXHJcbiAgICAgIHN1bVIgKz0gclxyXG4gICAgICBzdW1HICs9IGdcclxuICAgICAgc3VtQiArPSBiXHJcbiAgICB9XHJcbiAgICBjb25zdCBtZWFuUiA9IE1hdGgucm91bmQoc3VtUiAvIGNvbG9ycy5sZW5ndGgpXHJcbiAgICBjb25zdCBtZWFuRyA9IE1hdGgucm91bmQoc3VtRyAvIGNvbG9ycy5sZW5ndGgpXHJcbiAgICBjb25zdCBtZWFuQiA9IE1hdGgucm91bmQoc3VtQiAvIGNvbG9ycy5sZW5ndGgpXHJcbiAgICByZXR1cm4gW21lYW5SLCBtZWFuRywgbWVhbkJdXHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGtNZWFuc0NvbG9yUXVhbnRpemF0aW9uKGNvbG9yczogbnVtYmVyW11bXSwgazogbnVtYmVyKSB7XHJcbiAgICBsZXQgY2x1c3RlckNlbnRlcnMgPSBbXVxyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBrOyBpKyspIHtcclxuICAgICAgY29uc3QgcmFuZG9tQ29sb3IgPSBjb2xvcnNbTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogY29sb3JzLmxlbmd0aCldXHJcbiAgICAgIGNsdXN0ZXJDZW50ZXJzLnB1c2gocmFuZG9tQ29sb3IpXHJcbiAgICB9XHJcblxyXG4gICAgbGV0IGNsdXN0ZXJzID0gW11cclxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgY29sb3JzLmxlbmd0aDsgaSsrKSB7XHJcbiAgICAgIGNvbnN0IGNvbG9yID0gY29sb3JzW2ldXHJcbiAgICAgIGxldCBtaW5EaXN0YW5jZSA9IEluZmluaXR5XHJcbiAgICAgIGxldCBuZWFyZXN0Q2VudGVyID0gbnVsbFxyXG4gICAgICBmb3IgKGxldCBqID0gMDsgaiA8IGNsdXN0ZXJDZW50ZXJzLmxlbmd0aDsgaisrKSB7XHJcbiAgICAgICAgY29uc3QgY2VudGVyID0gY2x1c3RlckNlbnRlcnNbal1cclxuICAgICAgICBjb25zdCBkaXN0YW5jZSA9IHRoaXMuY2FsY3VsYXRlQ29sb3JEaXN0YW5jZShjb2xvciwgY2VudGVyKVxyXG4gICAgICAgIGlmIChkaXN0YW5jZSA8IG1pbkRpc3RhbmNlKSB7XHJcbiAgICAgICAgICBtaW5EaXN0YW5jZSA9IGRpc3RhbmNlXHJcbiAgICAgICAgICBuZWFyZXN0Q2VudGVyID0gY2VudGVyXHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcbiAgICAgIGNsdXN0ZXJzLnB1c2goeyBjb2xvciwgY2VudGVyOiBuZWFyZXN0Q2VudGVyIH0pXHJcbiAgICB9XHJcblxyXG4gICAgbGV0IHVwZGF0ZWRDZW50ZXJzID0gW11cclxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgY2x1c3RlckNlbnRlcnMubGVuZ3RoOyBpKyspIHtcclxuICAgICAgY29uc3QgY2VudGVyID0gY2x1c3RlckNlbnRlcnNbaV1cclxuICAgICAgY29uc3QgY2x1c3RlckNvbG9ycyA9IGNsdXN0ZXJzLmZpbHRlcihjID0+IGMuY2VudGVyID09PSBjZW50ZXIpLm1hcChjID0+IGMuY29sb3IpXHJcbiAgICAgIGlmIChjbHVzdGVyQ29sb3JzLmxlbmd0aCA+IDApIHtcclxuICAgICAgICBjb25zdCB1cGRhdGVkQ2VudGVyID0gdGhpcy5jYWxjdWxhdGVNZWFuQ29sb3IoY2x1c3RlckNvbG9ycylcclxuICAgICAgICB1cGRhdGVkQ2VudGVycy5wdXNoKHVwZGF0ZWRDZW50ZXIpXHJcbiAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gdXBkYXRlZENlbnRlcnNcclxuICB9XHJcblxyXG59XHJcbiJdfQ==