├── projects
├── app
│ ├── src
│ │ ├── polyfills.ts
│ │ ├── environments
│ │ │ ├── environment.prod.ts
│ │ │ └── environment.ts
│ │ ├── favicon.ico
│ │ ├── styles.css
│ │ ├── main.ts
│ │ ├── app
│ │ │ ├── app.component.css
│ │ │ ├── app.component.ts
│ │ │ └── app.component.html
│ │ └── index.html
│ ├── tsconfig.json
│ ├── browserslist
│ └── stylelint.json
└── lib
│ ├── ng-package.json
│ ├── src
│ ├── public-api.ts
│ └── lib
│ │ ├── formats.ts
│ │ ├── helpers.ts
│ │ ├── color-picker.component.html
│ │ ├── color-picker.service.ts
│ │ ├── color-picker.directive.ts
│ │ ├── color-picker.component.css
│ │ └── color-picker.component.ts
│ ├── package.json
│ ├── tsconfig.json
│ ├── README.md
│ └── stylelint.json
├── .prettierrc.json
├── .npmignore
├── .gitignore
├── tailwind.config.js
├── CONTRIBUTING.md
├── tsconfig.json
├── .eslintrc.json
├── LICENSE.md
├── package.json
├── angular.json
├── README.md
└── .stylelintrc.json
/projects/app/src/polyfills.ts:
--------------------------------------------------------------------------------
1 | import 'zone.js';
2 |
--------------------------------------------------------------------------------
/projects/app/src/environments/environment.prod.ts:
--------------------------------------------------------------------------------
1 | export const environment = {
2 | production: true
3 | };
4 |
--------------------------------------------------------------------------------
/projects/app/src/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/zefoy/ngx-color-picker/HEAD/projects/app/src/favicon.ico
--------------------------------------------------------------------------------
/.prettierrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "semi": false,
3 | "tabWidth": 2,
4 | "singleQuote": true,
5 | "trailingComma": "none"
6 | }
7 |
--------------------------------------------------------------------------------
/projects/app/src/styles.css:
--------------------------------------------------------------------------------
1 | /* You can add global styles to this file, and also import other style files */
2 |
3 | @tailwind base;
4 | @tailwind utilities;
5 | @tailwind components;
6 |
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | .*
2 | *.tgz
3 |
4 | src/
5 | config/
6 | example/
7 |
8 | package/
9 | npm-debug.log
10 | node_modules/
11 |
12 | tslint.json
13 | tsconfig.json
14 | stylelint.json
15 |
--------------------------------------------------------------------------------
/projects/app/src/main.ts:
--------------------------------------------------------------------------------
1 | import { bootstrapApplication } from '@angular/platform-browser';
2 |
3 | import { AppComponent } from './app/app.component';
4 |
5 | bootstrapApplication(AppComponent);
6 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | dist/
2 | bundles/
3 |
4 | package/
5 | npm-debug.log
6 | node_modules/
7 | .angular/
8 | .vs/
9 |
10 | example/dist/
11 | example/npm-debug.log
12 | example/node_modules/
13 | .idea
14 |
--------------------------------------------------------------------------------
/projects/lib/ng-package.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "../../node_modules/ng-packagr/ng-package.schema.json",
3 | "dest": "../../dist/lib",
4 | "lib": {
5 | "entryFile": "src/public-api.ts"
6 | }
7 | }
8 |
--------------------------------------------------------------------------------
/tailwind.config.js:
--------------------------------------------------------------------------------
1 | /** @type {import('tailwindcss').Config} */
2 | module.exports = {
3 | content: [
4 | "./projects/app/src/**/*.{html,ts}",
5 | ],
6 | theme: {
7 | extend: {},
8 | },
9 | plugins: [],
10 | }
11 |
12 |
--------------------------------------------------------------------------------
/projects/app/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "../../tsconfig.json",
3 | "compilerOptions": {
4 | "outDir": "../../out-tsc/app",
5 | "types": []
6 | },
7 | "files": [
8 | "src/main.ts",
9 | "src/polyfills.ts"
10 | ],
11 | "include": [
12 | "src/**/*.d.ts"
13 | ]
14 | }
15 |
--------------------------------------------------------------------------------
/projects/lib/src/public-api.ts:
--------------------------------------------------------------------------------
1 | export { Cmyk, Hsla, Hsva, Rgba } from "./lib/formats";
2 | export {
3 | AlphaChannel,
4 | ColorMode,
5 | OutputFormat,
6 | TextDirective,
7 | SliderDirective,
8 | } from "./lib/helpers";
9 |
10 | export { ColorPickerComponent } from "./lib/color-picker.component";
11 | export { ColorPickerDirective } from "./lib/color-picker.directive";
12 | export { ColorPickerService } from "./lib/color-picker.service";
13 |
--------------------------------------------------------------------------------
/projects/app/browserslist:
--------------------------------------------------------------------------------
1 | # This file is used by the build system to adjust CSS and JS output to support the specified browsers below.
2 | # For additional information regarding the format and rule options, please see:
3 | # https://github.com/browserslist/browserslist#queries
4 |
5 | # You can see what browsers were selected by your queries by running:
6 | # npx browserslist
7 |
8 | > 0.5%
9 | last 2 versions
10 | Firefox ESR
11 | not dead
12 | not IE 9-11 # For IE 9-11 support, remove 'not'.
--------------------------------------------------------------------------------
/projects/app/src/app/app.component.css:
--------------------------------------------------------------------------------
1 | input {
2 | width: 150px;
3 | margin-bottom: 16px;
4 | }
5 |
6 | .cmyk-text {
7 | float: left;
8 |
9 | width: 72px;
10 | height: 72px;
11 |
12 | font-weight: bolder;
13 | line-height: 72px;
14 | text-align: center;
15 | text-shadow: 1px 1px 2px #bbb;
16 | }
17 |
18 | .color-box {
19 | width: 100px;
20 | height: 25px;
21 | margin: 16px auto;
22 |
23 | cursor: pointer;
24 | }
25 |
26 | .change-me {
27 | cursor: pointer;
28 | font-size: 30px;
29 | font-weight: bolder;
30 | }
31 |
--------------------------------------------------------------------------------
/projects/lib/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "license": "MIT",
3 | "version": "20.1.1",
4 | "name": "ngx-color-picker",
5 | "description": "Color picker widget for Angular",
6 | "bugs": "https://github.com/zefoy/ngx-color-picker/issues",
7 | "repository": {
8 | "type": "git",
9 | "url": "https://github.com/zefoy/ngx-color-picker.git",
10 | "directory": "projects/lib"
11 | },
12 | "peerDependencies": {
13 | "@angular/common": ">=19.0.0",
14 | "@angular/core": ">=19.0.0",
15 | "@angular/forms": ">=19.0.0"
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/projects/lib/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "../../tsconfig.json",
3 | "compilerOptions": {
4 | "outDir": "../../out-tsc/lib",
5 | "target": "es2022",
6 | "declaration": true,
7 | "inlineSources": true,
8 | "stripInternal": true,
9 | "types": [],
10 | "lib": [
11 | "dom",
12 | "es2022"
13 | ]
14 | },
15 | "angularCompilerOptions": {
16 | "skipTemplateCodegen": true,
17 | "strictMetadataEmit": true,
18 | "enableResourceInlining": true,
19 | "compilationMode": "partial"
20 | },
21 | "exclude": [
22 | "src/test.ts",
23 | "**/*.spec.ts"
24 | ]
25 | }
26 |
--------------------------------------------------------------------------------
/CONTRIBUTING.md:
--------------------------------------------------------------------------------
1 | # Contributing guide
2 |
3 | Want to contribute to this project? Awesome! There are many ways you can contribute, see below.
4 |
5 | ### Opening issues
6 |
7 | Open an issue to report bugs or to propose new features. If you have a general usage question please note that this is just a wrapper and most questions should be directed to the actual library in Stack Overflow etc.
8 |
9 | ### Proposing pull requests
10 |
11 | Pull requests are more than welcome. Note that if you are going to propose drastic changes or new features, be sure to open an issue for discussion first, to make sure that your PR will be accepted before you spend effort coding it.
12 |
--------------------------------------------------------------------------------
/projects/app/src/environments/environment.ts:
--------------------------------------------------------------------------------
1 | // This file can be replaced during build by using the `fileReplacements` array.
2 | // `ng build --prod` replaces `environment.ts` with `environment.prod.ts`.
3 | // The list of file replacements can be found in `angular.json`.
4 |
5 | export const environment = {
6 | production: false
7 | };
8 |
9 | /*
10 | * For easier debugging in development mode, you can import the following file
11 | * to ignore zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`.
12 | *
13 | * This import should be commented out in production mode because it will have a negative impact
14 | * on performance if an error is thrown.
15 | */
16 | // import 'zone.js/dist/zone-error'; // Included with Angular CLI.
17 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compileOnSave": false,
3 | "compilerOptions": {
4 | "baseUrl": "./",
5 | "outDir": "./dist",
6 | "sourceMap": true,
7 | "declaration": false,
8 | "downlevelIteration": true,
9 | "experimentalDecorators": true,
10 | "module": "esnext",
11 | "moduleResolution": "node",
12 | "importHelpers": true,
13 | "target": "es2022",
14 | "typeRoots": [
15 | "node_modules/@types"
16 | ],
17 | "lib": [
18 | "dom",
19 | "es2022"
20 | ],
21 | "paths": {
22 | "ngx-color-picker": [
23 | "./projects/lib/src/public-api.ts"
24 | ]
25 | }
26 | },
27 | "angularCompilerOptions": {
28 | "fullTemplateTypeCheck": true,
29 | "strictInjectionParameters": true
30 | }
31 | }
32 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/formats.ts:
--------------------------------------------------------------------------------
1 | export enum ColorFormats {
2 | HEX,
3 | RGBA,
4 | HSLA,
5 | CMYK,
6 | }
7 |
8 | export class Rgba {
9 | constructor(
10 | public r: number,
11 | public g: number,
12 | public b: number,
13 | public a: number
14 | ) {}
15 | }
16 |
17 | export class Hsva {
18 | constructor(
19 | public h: number,
20 | public s: number,
21 | public v: number,
22 | public a: number
23 | ) {}
24 | }
25 |
26 | export class Hsla {
27 | constructor(
28 | public h: number,
29 | public s: number,
30 | public l: number,
31 | public a: number
32 | ) {}
33 | }
34 |
35 | export class Cmyk {
36 | constructor(
37 | public c: number,
38 | public m: number,
39 | public y: number,
40 | public k: number,
41 | public a: number = 1
42 | ) {}
43 | }
44 |
--------------------------------------------------------------------------------
/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "root": true,
3 | "ignorePatterns": [
4 | ],
5 | "overrides": [
6 | {
7 | "files": [
8 | "*.ts"
9 | ],
10 | "parserOptions": {
11 | "project": [
12 | "tsconfig.json",
13 | "e2e/tsconfig.json"
14 | ],
15 | "createDefaultProgram": true
16 | },
17 | "extends": [
18 | "plugin:@angular-eslint/recommended",
19 | "plugin:@angular-eslint/template/process-inline-templates"
20 | ],
21 | "rules": {
22 | "@angular-eslint/component-selector": [
23 | "off"
24 | ],
25 | "@angular-eslint/directive-selector": [
26 | "off"
27 | ]
28 | }
29 | },
30 | {
31 | "files": [
32 | "*.html"
33 | ],
34 | "extends": [
35 | "plugin:@angular-eslint/template/recommended"
36 | ],
37 | "rules": {}
38 | }
39 | ]
40 | }
41 |
--------------------------------------------------------------------------------
/LICENSE.md:
--------------------------------------------------------------------------------
1 | The MIT License
2 |
3 | Copyright (c) 2016 Zef Oy
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in
13 | all copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21 | THE SOFTWARE.
22 |
--------------------------------------------------------------------------------
/projects/app/src/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 | Example app
8 |
9 |
10 |
11 |
12 |
13 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "ngx-color-picker",
3 | "description": "Color picker widget for Angular",
4 | "bugs": "https://github.com/zefoy/ngx-color-picker/issues",
5 | "version": "20.1.1",
6 | "license": "MIT",
7 | "private": true,
8 | "scripts": {
9 | "ng": "ng",
10 | "lint": "ng lint",
11 | "start": "ng serve app",
12 | "build": "ng build lib",
13 | "deploy": "deploy-to-git",
14 | "prepare": "ng build lib --configuration production",
15 | "publish": "npm publish ./dist/lib",
16 | "predeploy": "rimraf ./dist/app && mkdirp ./dist/app"
17 | },
18 | "repository": {
19 | "type": "git",
20 | "url": "https://github.com/zefoy/ngx-color-picker.git"
21 | },
22 | "config": {
23 | "deployToGit": {
24 | "repository": "git@github.com:zefoy/ngx-color-picker.git",
25 | "branch": "gh-pages",
26 | "folder": "dist/app",
27 | "script": "ng build app --configuration production --base-href=ngx-color-picker --delete-output-path=false",
28 | "commit": "Publishing $npm_package_version",
29 | "user": {
30 | "name": "ZEF Devel",
31 | "email": "devel@zef.fi"
32 | }
33 | }
34 | },
35 | "dependencies": {
36 | "@angular/cdk": "^20.0.0",
37 | "@angular/common": "^20.0.0",
38 | "@angular/compiler": "^20.0.0",
39 | "@angular/core": "^20.0.0",
40 | "@angular/forms": "^20.0.0",
41 | "@angular/platform-browser": "^20.0.0",
42 | "@angular/platform-browser-dynamic": "^20.0.0",
43 | "rxjs": "^7.8.0",
44 | "tailwindcss": "~3.4.0",
45 | "zone.js": "^0.15.0"
46 | },
47 | "devDependencies": {
48 | "@angular-devkit/build-angular": "^20.0.0",
49 | "@angular-eslint/builder": "^20.0.0",
50 | "@angular-eslint/eslint-plugin": "^20.0.0",
51 | "@angular-eslint/eslint-plugin-template": "^20.0.0",
52 | "@angular-eslint/schematics": "^20.0.0",
53 | "@angular-eslint/template-parser": "^20.0.0",
54 | "@angular/cli": "^20.0.0",
55 | "@angular/compiler-cli": "^20.0.0",
56 | "@typescript-eslint/eslint-plugin": "^8.26.0",
57 | "@typescript-eslint/parser": "^8.26.0",
58 | "cpx": "^1.5.0",
59 | "deploy-to-git": "^0.4.0",
60 | "eslint": "^8.57.0",
61 | "mkdirp": "^3.0.0",
62 | "ng-packagr": "^20.0.0",
63 | "rimraf": "^6.0.0",
64 | "stylelint": "^16.15.0",
65 | "stylelint-config-standard": "^37.0.0",
66 | "stylelint-order": "^6.0.0",
67 | "terser": "^5.43.0",
68 | "tslib": "^2.8.0",
69 | "typescript": "~5.8.0",
70 | "watch": "^1.0.0"
71 | }
72 | }
73 |
--------------------------------------------------------------------------------
/projects/app/src/app/app.component.ts:
--------------------------------------------------------------------------------
1 | import { Component, inject } from '@angular/core'
2 | import { FormsModule } from '@angular/forms'
3 |
4 | import {
5 | ColorPickerService,
6 | Cmyk,
7 | ColorPickerDirective,
8 | } from 'ngx-color-picker'
9 |
10 | @Component({
11 | selector: 'app',
12 | styleUrls: ['app.component.css'],
13 | templateUrl: 'app.component.html',
14 | imports: [FormsModule, ColorPickerDirective],
15 | })
16 | export class AppComponent {
17 | private cpService = inject(ColorPickerService)
18 |
19 | public toggle: boolean = false
20 |
21 | public rgbaText: string = 'rgba(165, 26, 214, 0.2)'
22 |
23 | public arrayColors: any = {
24 | color1: '#2883e9',
25 | color2: '#e920e9',
26 | color3: 'rgb(255,245,0)',
27 | color4: 'rgb(236,64,64)',
28 | color5: 'rgba(45,208,45,1)',
29 | }
30 |
31 | public selectedColor: string = 'color1'
32 |
33 | public color1: string = '#2889e9'
34 | public color2: string = '#e920e9'
35 | public color3: string = '#fff500'
36 | public color4: string = 'rgb(236,64,64)'
37 | public color5: string = 'rgba(45,208,45,1)'
38 | public color6: string = '#1973c0'
39 | public color7: string = '#f200bd'
40 | public color8: string = '#a8ff00'
41 | public color9: string = '#278ce2'
42 | public color10: string = '#0a6211'
43 | public color11: string = '#f2ff00'
44 | public color12: string = '#f200bd'
45 | public color13: string = 'rgba(0,255,0,0.5)'
46 | public color14: string = 'rgb(0,255,255)'
47 | public color15: string = 'rgb(255,0,0)'
48 | public color16: string = '#a51ad633'
49 | public color17: string = '#666666'
50 | public color18: string = '#fa8072'
51 | public color19: string = '#f88888'
52 | public color20: string = '#ff0000'
53 |
54 | public cmykValue: string = ''
55 |
56 | public cmykColor: Cmyk = new Cmyk(0, 0, 0, 0)
57 |
58 | public alphaEnabled = false
59 |
60 | public onEventLog(event: string, data: any): void {
61 | console.log(event, data)
62 | }
63 |
64 | public onChangeColor(color: string): void {
65 | console.log('Color changed:', color)
66 | }
67 |
68 | public onChangeColorCmyk(color: string): Cmyk {
69 | const hsva = this.cpService.stringToHsva(color)
70 |
71 | if (hsva) {
72 | const rgba = this.cpService.hsvaToRgba(hsva)
73 |
74 | return this.cpService.rgbaToCmyk(rgba)
75 | }
76 |
77 | return new Cmyk(0, 0, 0, 0)
78 | }
79 |
80 | public onChangeColorHex8(color: string): string {
81 | const hsva = this.cpService.stringToHsva(color, true)
82 |
83 | if (hsva) {
84 | return this.cpService.outputFormat(hsva, 'rgba', null)
85 | }
86 |
87 | return ''
88 | }
89 | }
90 |
--------------------------------------------------------------------------------
/angular.json:
--------------------------------------------------------------------------------
1 | {
2 | "$schema": "./node_modules/@angular/cli/lib/config/schema.json",
3 | "cli": {
4 | "analytics": false
5 | },
6 | "version": 1,
7 | "newProjectRoot": "projects",
8 | "projects": {
9 | "app": {
10 | "projectType": "application",
11 | "schematics": {},
12 | "root": "",
13 | "sourceRoot": "projects/app/src",
14 | "prefix": "app",
15 | "architect": {
16 | "build": {
17 | "builder": "@angular-devkit/build-angular:application",
18 | "options": {
19 | "outputPath": "dist/app",
20 | "index": "projects/app/src/index.html",
21 | "browser": "projects/app/src/main.ts",
22 | "polyfills": ["projects/app/src/polyfills.ts"],
23 | "tsConfig": "projects/app/tsconfig.json",
24 | "assets": [
25 | "projects/app/src/favicon.ico",
26 | "projects/app/src/assets"
27 | ],
28 | "styles": [
29 | "projects/app/src/styles.css"
30 | ],
31 | "scripts": []
32 | },
33 | "configurations": {
34 | "production": {
35 | "fileReplacements": [
36 | {
37 | "replace": "projects/app/src/environments/environment.ts",
38 | "with": "projects/app/src/environments/environment.prod.ts"
39 | }
40 | ],
41 | "budgets": [
42 | {
43 | "type": "initial",
44 | "maximumWarning": "2mb",
45 | "maximumError": "5mb"
46 | },
47 | {
48 | "type": "anyComponentStyle",
49 | "maximumWarning": "30kb",
50 | "maximumError": "50kb"
51 | }
52 | ]
53 | },
54 | "development": {
55 | "optimization": false,
56 | "extractLicenses": false,
57 | "sourceMap": true,
58 | "namedChunks": true
59 | }
60 | }
61 | },
62 | "serve": {
63 | "builder": "@angular-devkit/build-angular:dev-server",
64 | "options": {
65 | "buildTarget": "app:build"
66 | },
67 | "configurations": {
68 | "production": {
69 | "buildTarget": "app:build:production"
70 | },
71 | "development": {
72 | "buildTarget": "app:build:development"
73 | }
74 | },
75 | "defaultConfiguration": "development"
76 | },
77 | "extract-i18n": {
78 | "builder": "@angular-devkit/build-angular:extract-i18n",
79 | "options": {
80 | "buildTarget": "app:build"
81 | }
82 | },
83 | "lint": {
84 | "builder": "@angular-eslint/builder:lint",
85 | "options": {
86 | "lintFilePatterns": [
87 | "projects/app/src/**/*.ts",
88 | "projects/app/src/**/*.html"
89 | ]
90 | }
91 | }
92 | }
93 | },
94 | "lib": {
95 | "projectType": "library",
96 | "root": "projects/lib",
97 | "sourceRoot": "projects/lib/src",
98 | "prefix": "lib",
99 | "architect": {
100 | "build": {
101 | "builder": "@angular-devkit/build-angular:ng-packagr",
102 | "options": {
103 | "tsConfig": "projects/lib/tsconfig.json",
104 | "project": "projects/lib/ng-package.json"
105 | },
106 | "configurations": {
107 | "production": {}
108 | }
109 | },
110 | "lint": {
111 | "builder": "@angular-eslint/builder:lint",
112 | "options": {
113 | "lintFilePatterns": [
114 | "projects/lib/src/**/*.ts",
115 | "projects/lib/src/**/*.html"
116 | ]
117 | }
118 | }
119 | }
120 | }
121 | }
122 | }
123 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/helpers.ts:
--------------------------------------------------------------------------------
1 | import { DOCUMENT } from '@angular/common'
2 | import {
3 | Directive,
4 | Input,
5 | Output,
6 | EventEmitter,
7 | HostListener,
8 | ElementRef,
9 | inject,
10 | } from '@angular/core'
11 |
12 | export type ColorMode =
13 | | 'color'
14 | | 'c'
15 | | '1'
16 | | 'grayscale'
17 | | 'g'
18 | | '2'
19 | | 'presets'
20 | | 'p'
21 | | '3'
22 |
23 | export type AlphaChannel = 'enabled' | 'disabled' | 'always' | 'forced'
24 |
25 | export type BoundingRectangle = {
26 | top: number
27 | bottom: number
28 | left: number
29 | right: number
30 | height: number
31 | width: number
32 | }
33 |
34 | export type OutputFormat = 'auto' | 'hex' | 'rgba' | 'hsla'
35 |
36 | export function calculateAutoPositioning(
37 | elBounds: BoundingRectangle,
38 | triggerElBounds: BoundingRectangle,
39 | window: Window
40 | ): string {
41 | // Defaults
42 | let usePositionX = 'right'
43 | let usePositionY = 'bottom'
44 | // Calculate collisions
45 | const { height, width } = elBounds
46 | const { top, left } = triggerElBounds
47 | const bottom = top + triggerElBounds.height
48 | const right = left + triggerElBounds.width
49 |
50 | const collisionTop = top - height < 0
51 | const collisionBottom =
52 | bottom + height >
53 | (window.innerHeight || document.documentElement.clientHeight)
54 | const collisionLeft = left - width < 0
55 | const collisionRight =
56 | right + width > (window.innerWidth || document.documentElement.clientWidth)
57 | const collisionAll =
58 | collisionTop && collisionBottom && collisionLeft && collisionRight
59 |
60 | // Generate X & Y position values
61 | if (collisionBottom) {
62 | usePositionY = 'top'
63 | }
64 |
65 | if (collisionTop) {
66 | usePositionY = 'bottom'
67 | }
68 |
69 | if (collisionLeft) {
70 | usePositionX = 'right'
71 | }
72 |
73 | if (collisionRight) {
74 | usePositionX = 'left'
75 | }
76 |
77 | // Choose the largest gap available
78 | if (collisionAll) {
79 | const postions = ['left', 'right', 'top', 'bottom']
80 | return postions.reduce((prev, next) =>
81 | elBounds[prev] > elBounds[next] ? prev : next
82 | )
83 | }
84 |
85 | if (collisionLeft && collisionRight) {
86 | if (collisionTop) {
87 | return 'bottom'
88 | }
89 | if (collisionBottom) {
90 | return 'top'
91 | }
92 | return top > bottom ? 'top' : 'bottom'
93 | }
94 |
95 | if (collisionTop && collisionBottom) {
96 | if (collisionLeft) {
97 | return 'right'
98 | }
99 | if (collisionRight) {
100 | return 'left'
101 | }
102 | return left > right ? 'left' : 'right'
103 | }
104 |
105 | return `${usePositionY}-${usePositionX}`
106 | }
107 |
108 | @Directive({
109 | selector: '[text]',
110 | })
111 | export class TextDirective {
112 | @Input() rg: number
113 | @Input() text: any
114 |
115 | @Output() newValue = new EventEmitter()
116 |
117 | @HostListener('input', ['$event']) inputChange(event: any): void {
118 | const value = event.target.value
119 |
120 | if (this.rg === undefined) {
121 | this.newValue.emit(value)
122 | } else {
123 | const numeric = parseFloat(value)
124 |
125 | this.newValue.emit({ v: numeric, rg: this.rg })
126 | }
127 | }
128 | }
129 |
130 | @Directive({
131 | selector: '[slider]',
132 | })
133 | export class SliderDirective {
134 | private elRef = inject(ElementRef)
135 | private document = inject(DOCUMENT)
136 |
137 | private readonly listenerMove: (event: Event) => void
138 | private readonly listenerStop: () => void
139 |
140 | @Input() rgX: number
141 | @Input() rgY: number
142 |
143 | @Output() dragEnd = new EventEmitter()
144 | @Output() dragStart = new EventEmitter()
145 |
146 | @Output() newValue = new EventEmitter()
147 |
148 | @HostListener('mousedown', ['$event']) mouseDown(event: any): void {
149 | this.start(event)
150 | }
151 |
152 | @HostListener('touchstart', ['$event']) touchStart(event: any): void {
153 | this.start(event)
154 | }
155 |
156 | constructor() {
157 | this.listenerMove = (event: Event) => this.move(event)
158 |
159 | this.listenerStop = () => this.stop()
160 | }
161 |
162 | private move(event: Event): void {
163 | event.preventDefault()
164 |
165 | this.setCursor(event)
166 | }
167 |
168 | private start(event: Event): void {
169 | this.setCursor(event)
170 |
171 | event.stopPropagation()
172 |
173 | this.document.addEventListener('mouseup', this.listenerStop)
174 | this.document.addEventListener('touchend', this.listenerStop)
175 | this.document.addEventListener('mousemove', this.listenerMove)
176 | this.document.addEventListener('touchmove', this.listenerMove)
177 |
178 | this.dragStart.emit()
179 | }
180 |
181 | private stop(): void {
182 | this.document.removeEventListener('mouseup', this.listenerStop)
183 | this.document.removeEventListener('touchend', this.listenerStop)
184 | this.document.removeEventListener('mousemove', this.listenerMove)
185 | this.document.removeEventListener('touchmove', this.listenerMove)
186 |
187 | this.dragEnd.emit()
188 | }
189 |
190 | private getX(event: any): number {
191 | const position = this.elRef.nativeElement.getBoundingClientRect()
192 |
193 | const pageX =
194 | event.pageX !== undefined ? event.pageX : event.touches[0].pageX
195 |
196 | return pageX - position.left - window.pageXOffset
197 | }
198 |
199 | private getY(event: any): number {
200 | const position = this.elRef.nativeElement.getBoundingClientRect()
201 |
202 | const pageY =
203 | event.pageY !== undefined ? event.pageY : event.touches[0].pageY
204 |
205 | return pageY - position.top - window.pageYOffset
206 | }
207 |
208 | private setCursor(event: any): void {
209 | const width = this.elRef.nativeElement.offsetWidth
210 | const height = this.elRef.nativeElement.offsetHeight
211 |
212 | const x = Math.max(0, Math.min(this.getX(event), width))
213 | const y = Math.max(0, Math.min(this.getY(event), height))
214 |
215 | if (this.rgX !== undefined && this.rgY !== undefined) {
216 | this.newValue.emit({
217 | s: x / width,
218 | v: 1 - y / height,
219 | rgX: this.rgX,
220 | rgY: this.rgY,
221 | })
222 | } else if (this.rgX === undefined && this.rgY !== undefined) {
223 | this.newValue.emit({ v: y / height, rgY: this.rgY })
224 | } else if (this.rgX !== undefined && this.rgY === undefined) {
225 | this.newValue.emit({ v: x / width, rgX: this.rgX })
226 | }
227 | }
228 | }
229 |
230 | export class SliderPosition {
231 | constructor(
232 | public h: number,
233 | public s: number,
234 | public v: number,
235 | public a: number
236 | ) {}
237 | }
238 |
239 | export class SliderDimension {
240 | constructor(
241 | public h: number,
242 | public s: number,
243 | public v: number,
244 | public a: number
245 | ) {}
246 | }
247 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Angular Color Picker
2 |
3 |
4 |
5 | This is a simple color picker based on the cool angular2-color-picker by Alberplz.
6 |
7 | This documentation is for the latest version which requires Angular version newer than 2 major versions prior to the current latest version. For older Angular versions you need to use an older version of this library.
8 |
9 | ### Quick links
10 |
11 | [Example application](https://zefoy.github.io/ngx-color-picker/)
12 | |
13 | [StackBlitz example](https://stackblitz.com/github/zefoy/ngx-color-picker/tree/main)
14 |
15 | ### Building the library
16 |
17 | ```bash
18 | npm install
19 | npm run build
20 | ```
21 |
22 | ### Running the example
23 |
24 | ```bash
25 | npm install
26 | npm run start
27 | ```
28 |
29 | ### Installing and usage
30 |
31 | ```bash
32 | npm install ngx-color-picker --save
33 | ```
34 |
35 | ##### Import the library into your component:
36 |
37 | ```javascript
38 | import { ColorPickerDirective } from 'ngx-color-picker';
39 |
40 | @Component({
41 | // ...
42 | imports: [
43 | // ...
44 | ColorPickerDirective,
45 | // ...
46 | ]
47 | })
48 | ```
49 |
50 | ##### Use it in your HTML template:
51 |
52 | ```html
53 |
54 | ```
55 |
56 | ```javascript
57 | [colorPicker] // The color to show in the color picker dialog.
58 |
59 | [cpWidth] // Use this option to set color picker dialog width ('230px').
60 | [cpHeight] // Use this option to force color picker dialog height ('auto').
61 |
62 | [cpToggle] // Sets the default open / close state of the color picker (false).
63 | [cpDisabled] // Disables opening of the color picker dialog via toggle / events.
64 |
65 | [cpColorMode] // Dialog color mode: 'color', 'grayscale', 'presets' ('color').
66 |
67 | [cpCmykEnabled] // Enables CMYK input format and color change event (false).
68 |
69 | [cpOutputFormat] // Output color format: 'auto', 'hex', 'rgba', 'hsla' ('auto').
70 | [cpAlphaChannel] // Alpha mode: 'enabled', 'disabled', 'always', 'forced' ('enabled').
71 |
72 | [cpFallbackColor] // Used when the color is not well-formed or is undefined ('#000').
73 |
74 | [cpPosition] // Dialog position: 'auto', 'top', 'bottom', 'left', 'right',
75 | // 'top-left', 'top-right', 'bottom-left', 'bottom-right' ('auto').
76 | [cpPositionOffset] // Dialog offset percentage relative to the directive element (0%).
77 | [cpPositionRelativeToArrow] // Dialog position is calculated relative to dialog arrow (false).
78 |
79 | [cpPresetLabel] // Label text for the preset colors if any provided ('Preset colors').
80 | [cpPresetColors] // Array of preset colors to show in the color picker dialog ([]).
81 |
82 | [cpDisableInput] // Disables / hides the color input field from the dialog (false).
83 |
84 | [cpDialogDisplay] // Dialog positioning mode: 'popup', 'inline' ('popup').
85 | // popup: dialog is shown as popup (fixed positioning).
86 | // inline: dialog is shown permanently (static positioning).
87 |
88 | [cpIgnoredElements] // Array of HTML elements that will be ignored when clicked ([]).
89 |
90 | [cpSaveClickOutside] // Save currently selected color when user clicks outside (true).
91 | [cpCloseClickOutside] // Close the color picker dialog when user clicks outside (true).
92 |
93 | [cpOKButton] // Show an OK / Apply button which saves the color (false).
94 | [cpOKButtonText] // Button label text shown inside the OK / Apply button ('OK').
95 | [cpOKButtonClass] // Additional class for customizing the OK / Apply button ('').
96 |
97 | [cpCancelButton] // Show a Cancel / Reset button which resets the color (false).
98 | [cpCancelButtonText] // Button label text shown inside the Cancel / Reset button ('Cancel').
99 | [cpCancelButtonClass] // Additional class for customizing the Cancel / Reset button ('').
100 |
101 | [cpAddColorButton] // Show an Add Color button which add the color into preset (false).
102 | [cpAddColorButtonText] // Button label text shown inside the Add Color button ('Add color').
103 | [cpAddColorButtonClass] // Additional class for customizing the Add Color button ('').
104 |
105 | [cpRemoveColorButtonClass] // Additional class for customizing the Remove Color button ('').
106 |
107 | [cpPresetColorsClass] // Additional class for customizing the Preset Colors container ('').
108 |
109 | [cpMaxPresetColorsLength] // Use this option to set the max colors allowed in presets (null).
110 |
111 | [cpPresetEmptyMessage] // Message for empty colors if any provided used ('No colors added').
112 | [cpPresetEmptyMessageClass] // Additional class for customizing the empty colors message ('').
113 |
114 | [cpUseRootViewContainer] // Create dialog component in the root view container (false).
115 | // Note: The root component needs to have public viewContainerRef.
116 |
117 | (colorPickerOpen) // Current color value, send when dialog is opened (value: string).
118 | (colorPickerClose) // Current color value, send when dialog is closed (value: string).
119 |
120 | (colorPickerChange) // Changed color value, send when color is changed (value: string).
121 | (colorPickerCancel) // Color select canceled, send when Cancel button is pressed (void).
122 | (colorPickerSelect) // Selected color value, send when OK button is pressed (value: string).
123 |
124 | (cpToggleChange) // Status of the dialog, send when dialog is opened / closed (open: boolean).
125 |
126 | (cpInputChange) // Input name and its value, send when user changes color through inputs
127 | // ({input: string, value: number | string, color: string})
128 |
129 | (cpSliderChange) // Slider name and its value, send when user changes color through slider
130 | // ({slider: string, value: number | string, color: string})
131 | (cpSliderDragStart) // Slider name and current color, send when slider dragging starts (mousedown,touchstart)
132 | // ({slider: string, color: string})
133 | (cpSliderDragEnd) // Slider name and current color, send when slider dragging ends (mouseup,touchend)
134 | // ({slider: string, color: string})
135 |
136 | (cpCmykColorChange) // Outputs the color as CMYK string if CMYK is enabled (value: string).
137 |
138 | (cpPresetColorsChange) // Preset colors, send when 'Add Color' button is pressed (value: array).
139 | ```
140 |
141 | ##### Available control / helper functions (provided by the directive):
142 |
143 | ```javascript
144 | openDialog() // Opens the color picker dialog if not already open.
145 | closeDialog() // Closes the color picker dialog if not already closed.
146 | ```
147 |
--------------------------------------------------------------------------------
/projects/lib/README.md:
--------------------------------------------------------------------------------
1 | # Angular Color Picker
2 |
3 |
4 |
5 | This is a simple color picker based on the cool angular2-color-picker by Alberplz.
6 |
7 | This documentation is for the latest 5/6.x.x version which requires Angular 5 or newer. For Angular 4 you need to use the latest 4.x.x version. Documentation for the 4.x.x can be found from here .
8 |
9 | ### Quick links
10 |
11 | [Example application](https://zefoy.github.io/ngx-color-picker/)
12 | |
13 | [StackBlitz example](https://stackblitz.com/github/zefoy/ngx-color-picker/tree/master)
14 |
15 | ### Building the library
16 |
17 | ```bash
18 | npm install
19 | npm run build
20 | ```
21 |
22 | ### Running the example
23 |
24 | ```bash
25 | npm install
26 | npm run start
27 | ```
28 |
29 | ### Installing and usage
30 |
31 | ```bash
32 | npm install ngx-color-picker --save
33 | ```
34 |
35 | ##### Load the module for your app:
36 |
37 | ```javascript
38 | import { ColorPickerModule } from 'ngx-color-picker';
39 |
40 | @NgModule({
41 | ...
42 | imports: [
43 | ...
44 | ColorPickerModule
45 | ]
46 | })
47 | ```
48 |
49 | ##### Use it in your HTML template:
50 |
51 | ```html
52 |
53 | ```
54 |
55 | ```javascript
56 | [colorPicker] // The color to show in the color picker dialog.
57 |
58 | [cpWidth] // Use this option to set color picker dialog width ('230px').
59 | [cpHeight] // Use this option to force color picker dialog height ('auto').
60 |
61 | [cpToggle] // Sets the default open / close state of the color picker (false).
62 | [cpDisabled] // Disables opening of the color picker dialog via toggle / events.
63 |
64 | [cpColorMode] // Dialog color mode: 'color', 'grayscale', 'presets' ('color').
65 |
66 | [cpCmykEnabled] // Enables CMYK input format and color change event (false).
67 |
68 | [cpOutputFormat] // Output color format: 'auto', 'hex', 'rgba', 'hsla' ('auto').
69 | [cpAlphaChannel] // Alpha mode: 'enabled', 'disabled', 'always', 'forced' ('enabled').
70 |
71 | [cpFallbackColor] // Used when the color is not well-formed or is undefined ('#000').
72 |
73 | [cpPosition] // Dialog position: 'auto', 'top', 'bottom', 'left', 'right',
74 | // 'top-left', 'top-right', 'bottom-left', 'bottom-right' ('auto').
75 | [cpPositionOffset] // Dialog offset percentage relative to the directive element (0%).
76 | [cpPositionRelativeToArrow] // Dialog position is calculated relative to dialog arrow (false).
77 |
78 | [cpPresetLabel] // Label text for the preset colors if any provided ('Preset colors').
79 | [cpPresetColors] // Array of preset colors to show in the color picker dialog ([]).
80 |
81 | [cpDisableInput] // Disables / hides the color input field from the dialog (false).
82 |
83 | [cpDialogDisplay] // Dialog positioning mode: 'popup', 'inline' ('popup').
84 | // popup: dialog is shown as popup (fixed positioning).
85 | // inline: dialog is shown permanently (static positioning).
86 |
87 | [cpIgnoredElements] // Array of HTML elements that will be ignored when clicked ([]).
88 |
89 | [cpSaveClickOutside] // Save currently selected color when user clicks outside (true).
90 | [cpCloseClickOutside] // Close the color picker dialog when user clicks outside (true).
91 |
92 | [cpOKButton] // Show an OK / Apply button which saves the color (false).
93 | [cpOKButtonText] // Button label text shown inside the OK / Apply button ('OK').
94 | [cpOKButtonClass] // Additional class for customizing the OK / Apply button ('').
95 |
96 | [cpCancelButton] // Show a Cancel / Reset button which resets the color (false).
97 | [cpCancelButtonText] // Button label text shown inside the Cancel / Reset button ('Cancel').
98 | [cpCancelButtonClass] // Additional class for customizing the Cancel / Reset button ('').
99 |
100 | [cpAddColorButton] // Show an Add Color button which add the color into preset (false).
101 | [cpAddColorButtonText] // Button label text shown inside the Add Color button ('Add color').
102 | [cpAddColorButtonClass] // Additional class for customizing the Add Color button ('').
103 |
104 | [cpRemoveColorButtonClass] // Additional class for customizing the Remove Color button ('').
105 |
106 | [cpPresetColorsClass] // Additional class for customizing the Preset Colors container ('').
107 |
108 | [cpMaxPresetColorsLength] // Use this option to set the max colors allowed in presets (null).
109 |
110 | [cpPresetEmptyMessage] // Message for empty colors if any provided used ('No colors added').
111 | [cpPresetEmptyMessageClass] // Additional class for customizing the empty colors message ('').
112 |
113 | [cpUseRootViewContainer] // Create dialog component in the root view container (false).
114 | // Note: The root component needs to have public viewContainerRef.
115 |
116 | (colorPickerOpen) // Current color value, send when dialog is opened (value: string).
117 | (colorPickerClose) // Current color value, send when dialog is closed (value: string).
118 |
119 | (colorPickerChange) // Changed color value, send when color is changed (value: string).
120 | (colorPickerCancel) // Color select canceled, send when Cancel button is pressed (void).
121 | (colorPickerSelect) // Selected color value, send when OK button is pressed (value: string).
122 |
123 | (cpToggleChange) // Status of the dialog, send when dialog is opened / closed (open: boolean).
124 |
125 | (cpInputChange) // Input name and its value, send when user changes color through inputs
126 | // ({input: string, value: number | string, color: string})
127 |
128 | (cpSliderChange) // Slider name and its value, send when user changes color through slider
129 | // ({slider: string, value: number | string, color: string})
130 | (cpSliderDragStart) // Slider name and current color, send when slider dragging starts (mousedown,touchstart)
131 | // ({slider: string, color: string})
132 | (cpSliderDragEnd) // Slider name and current color, send when slider dragging ends (mouseup,touchend)
133 | // ({slider: string, color: string})
134 |
135 | (cpCmykColorChange) // Outputs the color as CMYK string if CMYK is enabled (value: string).
136 |
137 | (cpPresetColorsChange) // Preset colors, send when 'Add Color' button is pressed (value: array).
138 | ```
139 |
140 | ##### Available control / helper functions (provided by the directive):
141 |
142 | ```javascript
143 | openDialog() // Opens the color picker dialog if not already open.
144 | closeDialog() // Closes the color picker dialog if not already closed.
145 | ```
146 |
--------------------------------------------------------------------------------
/.stylelintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "plugins": [
3 | "stylelint-order"
4 | ],
5 | "extends": [
6 | "stylelint-config-standard"
7 | ],
8 | "rules": {
9 | "color-hex-case": "lower",
10 | "color-no-invalid-hex": true,
11 |
12 | "font-family-no-missing-generic-family-keyword": null,
13 | "function-calc-no-unspaced-operator": true,
14 | "function-comma-space-after": "always-single-line",
15 | "function-comma-space-before": "never",
16 | "function-name-case": "lower",
17 | "function-url-quotes": "always",
18 | "function-whitespace-after": "always",
19 |
20 | "number-leading-zero": "always",
21 | "number-no-trailing-zeros": true,
22 | "length-zero-no-unit": true,
23 |
24 | "string-no-newline": true,
25 | "string-quotes": "single",
26 |
27 | "unit-case": "lower",
28 | "unit-no-unknown": true,
29 | "unit-whitelist": [
30 | "px",
31 | "%",
32 | "deg",
33 | "ms",
34 | "em"
35 | ],
36 |
37 | "value-list-comma-space-after": "always-single-line",
38 | "value-list-comma-space-before": "never",
39 |
40 | "shorthand-property-no-redundant-values": true,
41 |
42 | "property-case": "lower",
43 |
44 | "at-rule-empty-line-before": [
45 | "always",
46 | {
47 | "ignore": [
48 | "blockless-after-blockless"
49 | ]
50 | }
51 | ],
52 |
53 | "at-rule-no-unknown": null,
54 |
55 | "declaration-block-no-duplicate-properties": true,
56 | "declaration-block-trailing-semicolon": "always",
57 | "declaration-block-single-line-max-declarations": 1,
58 | "declaration-block-semicolon-space-before": "never",
59 | "declaration-block-semicolon-space-after": "always-single-line",
60 | "declaration-block-semicolon-newline-before": "never-multi-line",
61 | "declaration-block-semicolon-newline-after": "always-multi-line",
62 |
63 | "block-closing-brace-newline-after": "always",
64 | "block-closing-brace-newline-before": "always-multi-line",
65 | "block-no-empty": true,
66 | "block-opening-brace-newline-after": "always-multi-line",
67 | "block-opening-brace-space-before": "always-multi-line",
68 |
69 | "selector-attribute-brackets-space-inside": "never",
70 | "selector-attribute-operator-space-after": "never",
71 | "selector-attribute-operator-space-before": "never",
72 | "selector-combinator-space-after": "always",
73 | "selector-combinator-space-before": "always",
74 | "selector-pseudo-class-case": "lower",
75 | "selector-pseudo-class-parentheses-space-inside": "never",
76 | "selector-pseudo-element-case": "lower",
77 | "selector-pseudo-element-colon-notation": "double",
78 | "selector-pseudo-element-no-unknown": true,
79 | "selector-type-no-unknown": null,
80 | "selector-type-case": "lower",
81 | "selector-max-id": 0,
82 |
83 | "declaration-empty-line-before": null,
84 |
85 | "no-descending-specificity": null,
86 |
87 | "order/properties-order": [
88 | [
89 | {
90 | "properties": [
91 | "content",
92 | "direction"
93 | ]
94 | }, {
95 | "emptyLineBefore": "always",
96 | "properties": [
97 | "float",
98 | "position",
99 | "z-index",
100 | "top",
101 | "right",
102 | "bottom",
103 | "left"
104 | ]
105 | }, {
106 | "emptyLineBefore": "always",
107 | "properties": [
108 | "visibility",
109 | "opacity",
110 | "display",
111 | "overflow",
112 | "box-sizing",
113 | "flex",
114 | "flex-basis",
115 | "flex-direction",
116 | "flex-flow",
117 | "flex-grow",
118 | "flex-shrink",
119 | "flex-wrap",
120 | "align-self",
121 | "align-items",
122 | "align-content",
123 | "justify-content",
124 | "order",
125 | "width",
126 | "height",
127 | "min-width",
128 | "min-height",
129 | "max-width",
130 | "max-height",
131 | "padding",
132 | "padding-top",
133 | "padding-right",
134 | "padding-bottom",
135 | "padding-left",
136 | "margin",
137 | "margin-top",
138 | "margin-right",
139 | "margin-bottom",
140 | "margin-left",
141 | "border",
142 | "border-top",
143 | "border-right",
144 | "border-bottom",
145 | "border-left",
146 | "border-width",
147 | "border-top-width",
148 | "border-right-width",
149 | "border-bottom-width",
150 | "border-left-width",
151 | "border-style",
152 | "border-top-style",
153 | "border-right-style",
154 | "border-bottom-style",
155 | "border-left-style",
156 | "border-color",
157 | "border-top-color",
158 | "border-right-color",
159 | "border-bottom-color",
160 | "border-left-color",
161 | "border-radius",
162 | "border-top-left-radius",
163 | "border-top-right-radius",
164 | "border-bottom-right-radius",
165 | "border-bottom-left-radius",
166 | "outline"
167 | ]
168 | }, {
169 | "emptyLineBefore": "always",
170 | "properties": [
171 | "cursor",
172 | "resize",
173 | "user-select",
174 | "touch-action",
175 | "pointer-events",
176 | "font-size",
177 | "font-style",
178 | "font-weight",
179 | "font-family",
180 | "line-height",
181 | "white-space",
182 | "text-align",
183 | "text-shadow",
184 | "text-decoration",
185 | "text-transform",
186 | "text-overflow",
187 | "letter-spacing",
188 | "list-style-type",
189 | "object-fit",
190 | "vertical-align",
191 | "color",
192 | "background",
193 | "background-size",
194 | "background-color",
195 | "background-image",
196 | "background-repeat",
197 | "background-position",
198 | "background-attachment",
199 | "box-shadow"
200 | ]
201 | }, {
202 | "emptyLineBefore": "always",
203 | "properties": [
204 | "filter",
205 | "animation",
206 | "animation-name",
207 | "animation-delay",
208 | "animation-duration",
209 | "animation-direction",
210 | "animation-fill-mode",
211 | "animation-play-state",
212 | "animation-iteration-count",
213 | "animation-timing-function",
214 | "transform",
215 | "transform-style",
216 | "transform-origin",
217 | "transition",
218 | "transition-delay",
219 | "transition-duration",
220 | "transition-property",
221 | "transition-timing-function"
222 | ]
223 | }
224 | ],
225 | {
226 | "unspecified": "bottomAlphabetical"
227 | }
228 | ]
229 | }
230 | }
231 |
--------------------------------------------------------------------------------
/projects/app/stylelint.json:
--------------------------------------------------------------------------------
1 | {
2 | "plugins": [
3 | "stylelint-order"
4 | ],
5 | "extends": [
6 | "stylelint-config-standard"
7 | ],
8 | "rules": {
9 | "color-hex-case": "lower",
10 | "color-no-invalid-hex": true,
11 |
12 | "font-family-no-missing-generic-family-keyword": null,
13 | "function-calc-no-unspaced-operator": true,
14 | "function-comma-space-after": "always-single-line",
15 | "function-comma-space-before": "never",
16 | "function-name-case": "lower",
17 | "function-url-quotes": "always",
18 | "function-whitespace-after": "always",
19 |
20 | "number-leading-zero": "always",
21 | "number-no-trailing-zeros": true,
22 | "length-zero-no-unit": true,
23 |
24 | "string-no-newline": true,
25 | "string-quotes": "single",
26 |
27 | "unit-case": "lower",
28 | "unit-no-unknown": true,
29 | "unit-whitelist": [
30 | "px",
31 | "%",
32 | "deg",
33 | "ms",
34 | "em"
35 | ],
36 |
37 | "value-list-comma-space-after": "always-single-line",
38 | "value-list-comma-space-before": "never",
39 |
40 | "shorthand-property-no-redundant-values": true,
41 |
42 | "property-case": "lower",
43 |
44 | "at-rule-empty-line-before": [
45 | "always",
46 | {
47 | "ignore": [
48 | "blockless-after-blockless"
49 | ]
50 | }
51 | ],
52 |
53 | "at-rule-no-unknown": null,
54 |
55 | "declaration-block-no-duplicate-properties": true,
56 | "declaration-block-trailing-semicolon": "always",
57 | "declaration-block-single-line-max-declarations": 1,
58 | "declaration-block-semicolon-space-before": "never",
59 | "declaration-block-semicolon-space-after": "always-single-line",
60 | "declaration-block-semicolon-newline-before": "never-multi-line",
61 | "declaration-block-semicolon-newline-after": "always-multi-line",
62 |
63 | "block-closing-brace-newline-after": "always",
64 | "block-closing-brace-newline-before": "always-multi-line",
65 | "block-no-empty": true,
66 | "block-opening-brace-newline-after": "always-multi-line",
67 | "block-opening-brace-space-before": "always-multi-line",
68 |
69 | "selector-attribute-brackets-space-inside": "never",
70 | "selector-attribute-operator-space-after": "never",
71 | "selector-attribute-operator-space-before": "never",
72 | "selector-combinator-space-after": "always",
73 | "selector-combinator-space-before": "always",
74 | "selector-pseudo-class-case": "lower",
75 | "selector-pseudo-class-parentheses-space-inside": "never",
76 | "selector-pseudo-element-case": "lower",
77 | "selector-pseudo-element-colon-notation": "double",
78 | "selector-pseudo-element-no-unknown": true,
79 | "selector-type-no-unknown": null,
80 | "selector-type-case": "lower",
81 | "selector-max-id": 0,
82 |
83 | "declaration-empty-line-before": null,
84 |
85 | "no-descending-specificity": null,
86 |
87 | "order/properties-order": [
88 | [
89 | {
90 | "properties": [
91 | "content",
92 | "direction"
93 | ]
94 | }, {
95 | "emptyLineBefore": "always",
96 | "properties": [
97 | "float",
98 | "position",
99 | "z-index",
100 | "top",
101 | "right",
102 | "bottom",
103 | "left"
104 | ]
105 | }, {
106 | "emptyLineBefore": "always",
107 | "properties": [
108 | "visibility",
109 | "opacity",
110 | "display",
111 | "overflow",
112 | "box-sizing",
113 | "flex",
114 | "flex-basis",
115 | "flex-direction",
116 | "flex-flow",
117 | "flex-grow",
118 | "flex-shrink",
119 | "flex-wrap",
120 | "align-self",
121 | "align-items",
122 | "align-content",
123 | "justify-content",
124 | "order",
125 | "width",
126 | "height",
127 | "min-width",
128 | "min-height",
129 | "max-width",
130 | "max-height",
131 | "padding",
132 | "padding-top",
133 | "padding-right",
134 | "padding-bottom",
135 | "padding-left",
136 | "margin",
137 | "margin-top",
138 | "margin-right",
139 | "margin-bottom",
140 | "margin-left",
141 | "border",
142 | "border-top",
143 | "border-right",
144 | "border-bottom",
145 | "border-left",
146 | "border-width",
147 | "border-top-width",
148 | "border-right-width",
149 | "border-bottom-width",
150 | "border-left-width",
151 | "border-style",
152 | "border-top-style",
153 | "border-right-style",
154 | "border-bottom-style",
155 | "border-left-style",
156 | "border-color",
157 | "border-top-color",
158 | "border-right-color",
159 | "border-bottom-color",
160 | "border-left-color",
161 | "border-radius",
162 | "border-top-left-radius",
163 | "border-top-right-radius",
164 | "border-bottom-right-radius",
165 | "border-bottom-left-radius",
166 | "outline"
167 | ]
168 | }, {
169 | "emptyLineBefore": "always",
170 | "properties": [
171 | "cursor",
172 | "resize",
173 | "user-select",
174 | "touch-action",
175 | "pointer-events",
176 | "font-size",
177 | "font-style",
178 | "font-weight",
179 | "font-family",
180 | "line-height",
181 | "white-space",
182 | "text-align",
183 | "text-shadow",
184 | "text-decoration",
185 | "text-transform",
186 | "text-overflow",
187 | "letter-spacing",
188 | "list-style-type",
189 | "object-fit",
190 | "vertical-align",
191 | "color",
192 | "background",
193 | "background-size",
194 | "background-color",
195 | "background-image",
196 | "background-repeat",
197 | "background-position",
198 | "background-attachment",
199 | "box-shadow"
200 | ]
201 | }, {
202 | "emptyLineBefore": "always",
203 | "properties": [
204 | "filter",
205 | "animation",
206 | "animation-name",
207 | "animation-delay",
208 | "animation-duration",
209 | "animation-direction",
210 | "animation-fill-mode",
211 | "animation-play-state",
212 | "animation-iteration-count",
213 | "animation-timing-function",
214 | "transform",
215 | "transform-style",
216 | "transform-origin",
217 | "transition",
218 | "transition-delay",
219 | "transition-duration",
220 | "transition-property",
221 | "transition-timing-function"
222 | ]
223 | }
224 | ],
225 | {
226 | "unspecified": "bottomAlphabetical"
227 | }
228 | ]
229 | }
230 | }
231 |
--------------------------------------------------------------------------------
/projects/lib/stylelint.json:
--------------------------------------------------------------------------------
1 | {
2 | "plugins": [
3 | "stylelint-order"
4 | ],
5 | "extends": [
6 | "stylelint-config-standard"
7 | ],
8 | "rules": {
9 | "color-hex-case": "lower",
10 | "color-no-invalid-hex": true,
11 |
12 | "font-family-no-missing-generic-family-keyword": null,
13 | "function-calc-no-unspaced-operator": true,
14 | "function-comma-space-after": "always-single-line",
15 | "function-comma-space-before": "never",
16 | "function-name-case": "lower",
17 | "function-url-quotes": "always",
18 | "function-whitespace-after": "always",
19 |
20 | "number-leading-zero": "always",
21 | "number-no-trailing-zeros": true,
22 | "length-zero-no-unit": true,
23 |
24 | "string-no-newline": true,
25 | "string-quotes": "single",
26 |
27 | "unit-case": "lower",
28 | "unit-no-unknown": true,
29 | "unit-whitelist": [
30 | "px",
31 | "%",
32 | "deg",
33 | "ms",
34 | "em"
35 | ],
36 |
37 | "value-list-comma-space-after": "always-single-line",
38 | "value-list-comma-space-before": "never",
39 |
40 | "shorthand-property-no-redundant-values": true,
41 |
42 | "property-case": "lower",
43 |
44 | "at-rule-empty-line-before": [
45 | "always",
46 | {
47 | "ignore": [
48 | "blockless-after-blockless"
49 | ]
50 | }
51 | ],
52 |
53 | "at-rule-no-unknown": null,
54 |
55 | "declaration-block-no-duplicate-properties": true,
56 | "declaration-block-trailing-semicolon": "always",
57 | "declaration-block-single-line-max-declarations": 1,
58 | "declaration-block-semicolon-space-before": "never",
59 | "declaration-block-semicolon-space-after": "always-single-line",
60 | "declaration-block-semicolon-newline-before": "never-multi-line",
61 | "declaration-block-semicolon-newline-after": "always-multi-line",
62 |
63 | "block-closing-brace-newline-after": "always",
64 | "block-closing-brace-newline-before": "always-multi-line",
65 | "block-no-empty": true,
66 | "block-opening-brace-newline-after": "always-multi-line",
67 | "block-opening-brace-space-before": "always-multi-line",
68 |
69 | "selector-attribute-brackets-space-inside": "never",
70 | "selector-attribute-operator-space-after": "never",
71 | "selector-attribute-operator-space-before": "never",
72 | "selector-combinator-space-after": "always",
73 | "selector-combinator-space-before": "always",
74 | "selector-pseudo-class-case": "lower",
75 | "selector-pseudo-class-parentheses-space-inside": "never",
76 | "selector-pseudo-element-case": "lower",
77 | "selector-pseudo-element-colon-notation": "double",
78 | "selector-pseudo-element-no-unknown": true,
79 | "selector-type-no-unknown": null,
80 | "selector-type-case": "lower",
81 | "selector-max-id": 0,
82 |
83 | "declaration-empty-line-before": null,
84 |
85 | "no-descending-specificity": null,
86 |
87 | "order/properties-order": [
88 | [
89 | {
90 | "properties": [
91 | "content",
92 | "direction"
93 | ]
94 | }, {
95 | "emptyLineBefore": "always",
96 | "properties": [
97 | "float",
98 | "position",
99 | "z-index",
100 | "top",
101 | "right",
102 | "bottom",
103 | "left"
104 | ]
105 | }, {
106 | "emptyLineBefore": "always",
107 | "properties": [
108 | "visibility",
109 | "opacity",
110 | "display",
111 | "overflow",
112 | "box-sizing",
113 | "flex",
114 | "flex-basis",
115 | "flex-direction",
116 | "flex-flow",
117 | "flex-grow",
118 | "flex-shrink",
119 | "flex-wrap",
120 | "align-self",
121 | "align-items",
122 | "align-content",
123 | "justify-content",
124 | "order",
125 | "width",
126 | "height",
127 | "min-width",
128 | "min-height",
129 | "max-width",
130 | "max-height",
131 | "padding",
132 | "padding-top",
133 | "padding-right",
134 | "padding-bottom",
135 | "padding-left",
136 | "margin",
137 | "margin-top",
138 | "margin-right",
139 | "margin-bottom",
140 | "margin-left",
141 | "border",
142 | "border-top",
143 | "border-right",
144 | "border-bottom",
145 | "border-left",
146 | "border-width",
147 | "border-top-width",
148 | "border-right-width",
149 | "border-bottom-width",
150 | "border-left-width",
151 | "border-style",
152 | "border-top-style",
153 | "border-right-style",
154 | "border-bottom-style",
155 | "border-left-style",
156 | "border-color",
157 | "border-top-color",
158 | "border-right-color",
159 | "border-bottom-color",
160 | "border-left-color",
161 | "border-radius",
162 | "border-top-left-radius",
163 | "border-top-right-radius",
164 | "border-bottom-right-radius",
165 | "border-bottom-left-radius",
166 | "outline"
167 | ]
168 | }, {
169 | "emptyLineBefore": "always",
170 | "properties": [
171 | "cursor",
172 | "resize",
173 | "user-select",
174 | "touch-action",
175 | "pointer-events",
176 | "font-size",
177 | "font-style",
178 | "font-weight",
179 | "font-family",
180 | "line-height",
181 | "white-space",
182 | "text-align",
183 | "text-shadow",
184 | "text-decoration",
185 | "text-transform",
186 | "text-overflow",
187 | "letter-spacing",
188 | "list-style-type",
189 | "object-fit",
190 | "vertical-align",
191 | "color",
192 | "background",
193 | "background-size",
194 | "background-color",
195 | "background-image",
196 | "background-repeat",
197 | "background-position",
198 | "background-attachment",
199 | "box-shadow"
200 | ]
201 | }, {
202 | "emptyLineBefore": "always",
203 | "properties": [
204 | "filter",
205 | "animation",
206 | "animation-name",
207 | "animation-delay",
208 | "animation-duration",
209 | "animation-direction",
210 | "animation-fill-mode",
211 | "animation-play-state",
212 | "animation-iteration-count",
213 | "animation-timing-function",
214 | "transform",
215 | "transform-style",
216 | "transform-origin",
217 | "transition",
218 | "transition-delay",
219 | "transition-duration",
220 | "transition-property",
221 | "transition-timing-function"
222 | ]
223 | }
224 | ],
225 | {
226 | "unspecified": "bottomAlphabetical"
227 | }
228 | ]
229 | }
230 | }
231 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/color-picker.component.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
7 |
8 |
9 |
10 |
11 |
12 |
15 |
16 |
= cpMaxPresetColorsLength" (click)="onAddPresetColor($event, selectedColor)">
17 | {{cpAddColorButtonText}}
18 |
19 |
20 |
21 |
22 |
23 |
24 |
27 |
28 |
31 |
32 |
35 |
36 |
37 |
38 |
51 |
52 |
64 |
65 |
77 |
78 |
90 |
91 |
101 |
102 |
103 |
104 |
105 |
106 |
107 |
108 |
109 |
110 |
{{cpPresetLabel}}
111 |
112 |
117 |
118 |
{{cpPresetEmptyMessage}}
119 |
120 |
121 |
122 | {{cpCancelButtonText}}
123 |
124 | {{cpOKButtonText}}
125 |
126 |
127 |
130 |
131 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/color-picker.service.ts:
--------------------------------------------------------------------------------
1 | import { Injectable } from '@angular/core'
2 |
3 | import { Cmyk, Rgba, Hsla, Hsva } from './formats'
4 |
5 | import { ColorPickerComponent } from './color-picker.component'
6 |
7 | @Injectable({
8 | providedIn: 'root',
9 | })
10 | export class ColorPickerService {
11 | private active: ColorPickerComponent | null = null
12 |
13 | public setActive(active: ColorPickerComponent | null): void {
14 | if (
15 | this.active &&
16 | this.active !== active &&
17 | this.active.cpDialogDisplay !== 'inline'
18 | ) {
19 | this.active.closeDialog()
20 | }
21 |
22 | this.active = active
23 | }
24 |
25 | public hsva2hsla(hsva: Hsva): Hsla {
26 | const h = hsva.h,
27 | s = hsva.s,
28 | v = hsva.v,
29 | a = hsva.a
30 |
31 | if (v === 0) {
32 | return new Hsla(h, 0, 0, a)
33 | } else if (s === 0 && v === 1) {
34 | return new Hsla(h, 1, 1, a)
35 | } else {
36 | const l = (v * (2 - s)) / 2
37 |
38 | return new Hsla(h, (v * s) / (1 - Math.abs(2 * l - 1)), l, a)
39 | }
40 | }
41 |
42 | public hsla2hsva(hsla: Hsla): Hsva {
43 | const h = Math.min(hsla.h, 1),
44 | s = Math.min(hsla.s, 1)
45 | const l = Math.min(hsla.l, 1),
46 | a = Math.min(hsla.a, 1)
47 |
48 | if (l === 0) {
49 | return new Hsva(h, 0, 0, a)
50 | } else {
51 | const v = l + (s * (1 - Math.abs(2 * l - 1))) / 2
52 |
53 | return new Hsva(h, (2 * (v - l)) / v, v, a)
54 | }
55 | }
56 |
57 | public hsvaToRgba(hsva: Hsva): Rgba {
58 | let r: number, g: number, b: number
59 |
60 | const h = hsva.h,
61 | s = hsva.s,
62 | v = hsva.v,
63 | a = hsva.a
64 |
65 | const i = Math.floor(h * 6)
66 | const f = h * 6 - i
67 | const p = v * (1 - s)
68 | const q = v * (1 - f * s)
69 | const t = v * (1 - (1 - f) * s)
70 |
71 | switch (i % 6) {
72 | case 0:
73 | ;(r = v), (g = t), (b = p)
74 | break
75 | case 1:
76 | ;(r = q), (g = v), (b = p)
77 | break
78 | case 2:
79 | ;(r = p), (g = v), (b = t)
80 | break
81 | case 3:
82 | ;(r = p), (g = q), (b = v)
83 | break
84 | case 4:
85 | ;(r = t), (g = p), (b = v)
86 | break
87 | case 5:
88 | ;(r = v), (g = p), (b = q)
89 | break
90 | default:
91 | ;(r = 0), (g = 0), (b = 0)
92 | }
93 |
94 | return new Rgba(r, g, b, a)
95 | }
96 |
97 | public cmykToRgb(cmyk: Cmyk): Rgba {
98 | const r = (1 - cmyk.c) * (1 - cmyk.k)
99 | const g = (1 - cmyk.m) * (1 - cmyk.k)
100 | const b = (1 - cmyk.y) * (1 - cmyk.k)
101 |
102 | return new Rgba(r, g, b, cmyk.a)
103 | }
104 |
105 | public rgbaToCmyk(rgba: Rgba): Cmyk {
106 | const k: number = 1 - Math.max(rgba.r, rgba.g, rgba.b)
107 |
108 | if (k === 1) {
109 | return new Cmyk(0, 0, 0, 1, rgba.a)
110 | } else {
111 | const c = (1 - rgba.r - k) / (1 - k)
112 | const m = (1 - rgba.g - k) / (1 - k)
113 | const y = (1 - rgba.b - k) / (1 - k)
114 |
115 | return new Cmyk(c, m, y, k, rgba.a)
116 | }
117 | }
118 |
119 | public rgbaToHsva(rgba: Rgba): Hsva {
120 | let h: number, s: number
121 |
122 | const r = Math.min(rgba.r, 1),
123 | g = Math.min(rgba.g, 1)
124 | const b = Math.min(rgba.b, 1),
125 | a = Math.min(rgba.a, 1)
126 |
127 | const max = Math.max(r, g, b),
128 | min = Math.min(r, g, b)
129 |
130 | const v: number = max,
131 | d = max - min
132 |
133 | s = max === 0 ? 0 : d / max
134 |
135 | if (max === min) {
136 | h = 0
137 | } else {
138 | switch (max) {
139 | case r:
140 | h = (g - b) / d + (g < b ? 6 : 0)
141 | break
142 | case g:
143 | h = (b - r) / d + 2
144 | break
145 | case b:
146 | h = (r - g) / d + 4
147 | break
148 | default:
149 | h = 0
150 | }
151 |
152 | h /= 6
153 | }
154 |
155 | return new Hsva(h, s, v, a)
156 | }
157 |
158 | public rgbaToHex(rgba: Rgba, allowHex8?: boolean): string {
159 | /* eslint-disable no-bitwise */
160 | let hex =
161 | '#' +
162 | ((1 << 24) | (rgba.r << 16) | (rgba.g << 8) | rgba.b)
163 | .toString(16)
164 | .substr(1)
165 |
166 | if (allowHex8) {
167 | hex += ((1 << 8) | Math.round(rgba.a * 255)).toString(16).substr(1)
168 | }
169 | /* eslint-enable no-bitwise */
170 |
171 | return hex
172 | }
173 |
174 | public normalizeCMYK(cmyk: Cmyk): Cmyk {
175 | return new Cmyk(
176 | cmyk.c / 100,
177 | cmyk.m / 100,
178 | cmyk.y / 100,
179 | cmyk.k / 100,
180 | cmyk.a
181 | )
182 | }
183 |
184 | public denormalizeCMYK(cmyk: Cmyk): Cmyk {
185 | return new Cmyk(
186 | Math.floor(cmyk.c * 100),
187 | Math.floor(cmyk.m * 100),
188 | Math.floor(cmyk.y * 100),
189 | Math.floor(cmyk.k * 100),
190 | cmyk.a
191 | )
192 | }
193 |
194 | public denormalizeRGBA(rgba: Rgba): Rgba {
195 | return new Rgba(
196 | Math.round(rgba.r * 255),
197 | Math.round(rgba.g * 255),
198 | Math.round(rgba.b * 255),
199 | rgba.a
200 | )
201 | }
202 |
203 | public stringToHsva(
204 | colorString: string = '',
205 | allowHex8: boolean = false
206 | ): Hsva | null {
207 | let hsva: Hsva | null = null
208 |
209 | colorString = (colorString || '').toLowerCase()
210 |
211 | const stringParsers = [
212 | {
213 | re: /(rgb)a?\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*%?,\s*(\d{1,3})\s*%?(?:,\s*(\d+(?:\.\d+)?)\s*)?\)/,
214 | parse: function (execResult: any) {
215 | return new Rgba(
216 | parseInt(execResult[2], 10) / 255,
217 | parseInt(execResult[3], 10) / 255,
218 | parseInt(execResult[4], 10) / 255,
219 | isNaN(parseFloat(execResult[5])) ? 1 : parseFloat(execResult[5])
220 | )
221 | },
222 | },
223 | {
224 | re: /(hsl)a?\(\s*(\d{1,3})\s*,\s*(\d{1,3})%\s*,\s*(\d{1,3})%\s*(?:,\s*(\d+(?:\.\d+)?)\s*)?\)/,
225 | parse: function (execResult: any) {
226 | return new Hsla(
227 | parseInt(execResult[2], 10) / 360,
228 | parseInt(execResult[3], 10) / 100,
229 | parseInt(execResult[4], 10) / 100,
230 | isNaN(parseFloat(execResult[5])) ? 1 : parseFloat(execResult[5])
231 | )
232 | },
233 | },
234 | ]
235 |
236 | if (allowHex8) {
237 | stringParsers.push({
238 | re: /#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})?$/,
239 | parse: function (execResult: any) {
240 | return new Rgba(
241 | parseInt(execResult[1], 16) / 255,
242 | parseInt(execResult[2], 16) / 255,
243 | parseInt(execResult[3], 16) / 255,
244 | parseInt(execResult[4] || 'FF', 16) / 255
245 | )
246 | },
247 | })
248 | } else {
249 | stringParsers.push({
250 | re: /#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})$/,
251 | parse: function (execResult: any) {
252 | return new Rgba(
253 | parseInt(execResult[1], 16) / 255,
254 | parseInt(execResult[2], 16) / 255,
255 | parseInt(execResult[3], 16) / 255,
256 | 1
257 | )
258 | },
259 | })
260 | }
261 |
262 | stringParsers.push({
263 | re: /#([a-fA-F0-9])([a-fA-F0-9])([a-fA-F0-9])$/,
264 | parse: function (execResult: any) {
265 | return new Rgba(
266 | parseInt(execResult[1] + execResult[1], 16) / 255,
267 | parseInt(execResult[2] + execResult[2], 16) / 255,
268 | parseInt(execResult[3] + execResult[3], 16) / 255,
269 | 1
270 | )
271 | },
272 | })
273 |
274 | for (const key in stringParsers) {
275 | if (stringParsers.hasOwnProperty(key)) {
276 | const parser = stringParsers[key]
277 |
278 | const match = parser.re.exec(colorString),
279 | color: any = match && parser.parse(match)
280 |
281 | if (color) {
282 | if (color instanceof Rgba) {
283 | hsva = this.rgbaToHsva(color)
284 | } else if (color instanceof Hsla) {
285 | hsva = this.hsla2hsva(color)
286 | }
287 |
288 | return hsva
289 | }
290 | }
291 | }
292 |
293 | return hsva
294 | }
295 |
296 | public outputFormat(
297 | hsva: Hsva,
298 | outputFormat: string,
299 | alphaChannel: string | null
300 | ): string {
301 | if (outputFormat === 'auto') {
302 | outputFormat = hsva.a < 1 ? 'rgba' : 'hex'
303 | }
304 |
305 | switch (outputFormat) {
306 | case 'hsla':
307 | const hsla = this.hsva2hsla(hsva)
308 |
309 | const hslaText = new Hsla(
310 | Math.round(hsla.h * 360),
311 | Math.round(hsla.s * 100),
312 | Math.round(hsla.l * 100),
313 | Math.round(hsla.a * 100) / 100
314 | )
315 |
316 | if (hsva.a < 1 || alphaChannel === 'always') {
317 | return (
318 | 'hsla(' +
319 | hslaText.h +
320 | ',' +
321 | hslaText.s +
322 | '%,' +
323 | hslaText.l +
324 | '%,' +
325 | hslaText.a +
326 | ')'
327 | )
328 | } else {
329 | return (
330 | 'hsl(' + hslaText.h + ',' + hslaText.s + '%,' + hslaText.l + '%)'
331 | )
332 | }
333 |
334 | case 'rgba':
335 | const rgba = this.denormalizeRGBA(this.hsvaToRgba(hsva))
336 |
337 | if (hsva.a < 1 || alphaChannel === 'always') {
338 | return (
339 | 'rgba(' +
340 | rgba.r +
341 | ',' +
342 | rgba.g +
343 | ',' +
344 | rgba.b +
345 | ',' +
346 | Math.round(rgba.a * 100) / 100 +
347 | ')'
348 | )
349 | } else {
350 | return 'rgb(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ')'
351 | }
352 |
353 | default:
354 | const allowHex8 = alphaChannel === 'always' || alphaChannel === 'forced'
355 |
356 | return this.rgbaToHex(
357 | this.denormalizeRGBA(this.hsvaToRgba(hsva)),
358 | allowHex8
359 | )
360 | }
361 | }
362 | }
363 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/color-picker.directive.ts:
--------------------------------------------------------------------------------
1 | import {
2 | Directive,
3 | OnChanges,
4 | OnDestroy,
5 | Input,
6 | Output,
7 | EventEmitter,
8 | HostListener,
9 | ApplicationRef,
10 | ComponentRef,
11 | ElementRef,
12 | ViewContainerRef,
13 | Injector,
14 | EmbeddedViewRef,
15 | TemplateRef,
16 | isDevMode,
17 | inject,
18 | } from '@angular/core'
19 |
20 | import { ColorPickerComponent } from './color-picker.component'
21 |
22 | import { AlphaChannel, ColorMode, OutputFormat } from './helpers'
23 |
24 | @Directive({
25 | selector: '[colorPicker]',
26 | exportAs: 'ngxColorPicker',
27 | })
28 | export class ColorPickerDirective implements OnChanges, OnDestroy {
29 | private readonly injector = inject(Injector)
30 | private readonly appRef = inject(ApplicationRef)
31 | private readonly vcRef = inject(ViewContainerRef)
32 | private readonly elRef = inject(ElementRef)
33 |
34 | private dialog: any
35 |
36 | private dialogCreated: boolean = false
37 | private ignoreChanges: boolean = false
38 |
39 | private cmpRef: ComponentRef
40 | private viewAttachedToAppRef: boolean = false
41 |
42 | @Input() colorPicker: string
43 |
44 | @Input() cpWidth: string = '230px'
45 | @Input() cpHeight: string = 'auto'
46 |
47 | @Input() cpToggle: boolean = false
48 | @Input() cpDisabled: boolean = false
49 |
50 | @Input() cpIgnoredElements: any = []
51 |
52 | @Input() cpFallbackColor: string = ''
53 |
54 | @Input() cpColorMode: ColorMode = 'color'
55 |
56 | @Input() cpCmykEnabled: boolean = false
57 |
58 | @Input() cpOutputFormat: OutputFormat = 'auto'
59 | @Input() cpAlphaChannel: AlphaChannel = 'enabled'
60 |
61 | @Input() cpDisableInput: boolean = false
62 |
63 | @Input() cpDialogDisplay: string = 'popup'
64 |
65 | @Input() cpSaveClickOutside: boolean = true
66 | @Input() cpCloseClickOutside: boolean = true
67 |
68 | @Input() cpUseRootViewContainer: boolean = false
69 |
70 | @Input() cpPosition: string = 'auto'
71 | @Input() cpPositionOffset: string = '0%'
72 | @Input() cpPositionRelativeToArrow: boolean = false
73 |
74 | @Input() cpOKButton: boolean = false
75 | @Input() cpOKButtonText: string = 'OK'
76 | @Input() cpOKButtonClass: string = 'cp-ok-button-class'
77 |
78 | @Input() cpCancelButton: boolean = false
79 | @Input() cpCancelButtonText: string = 'Cancel'
80 | @Input() cpCancelButtonClass: string = 'cp-cancel-button-class'
81 |
82 | @Input() cpEyeDropper: boolean = false
83 |
84 | @Input() cpPresetLabel: string = 'Preset colors'
85 | @Input() cpPresetColors: string[]
86 | @Input() cpPresetColorsClass: string = 'cp-preset-colors-class'
87 | @Input() cpMaxPresetColorsLength: number = 6
88 |
89 | @Input() cpPresetEmptyMessage: string = 'No colors added'
90 | @Input() cpPresetEmptyMessageClass: string = 'preset-empty-message'
91 |
92 | @Input() cpAddColorButton: boolean = false
93 | @Input() cpAddColorButtonText: string = 'Add color'
94 | @Input() cpAddColorButtonClass: string = 'cp-add-color-button-class'
95 |
96 | @Input() cpRemoveColorButtonClass: string = 'cp-remove-color-button-class'
97 | @Input() cpArrowPosition: number = 0
98 |
99 | @Input() cpExtraTemplate: TemplateRef
100 |
101 | @Output() cpInputChange = new EventEmitter<{
102 | input: string
103 | value: number | string
104 | color: string
105 | }>(true)
106 |
107 | @Output() cpToggleChange = new EventEmitter(true)
108 |
109 | @Output() cpSliderChange = new EventEmitter<{
110 | slider: string
111 | value: string | number
112 | color: string
113 | }>(true)
114 | @Output() cpSliderDragEnd = new EventEmitter<{
115 | slider: string
116 | color: string
117 | }>(true)
118 | @Output() cpSliderDragStart = new EventEmitter<{
119 | slider: string
120 | color: string
121 | }>(true)
122 |
123 | @Output() colorPickerOpen = new EventEmitter(true)
124 | @Output() colorPickerClose = new EventEmitter(true)
125 |
126 | @Output() colorPickerCancel = new EventEmitter(true)
127 | @Output() colorPickerSelect = new EventEmitter(true)
128 | @Output() colorPickerChange = new EventEmitter(false)
129 |
130 | @Output() cpCmykColorChange = new EventEmitter(true)
131 |
132 | @Output() cpPresetColorsChange = new EventEmitter(true)
133 |
134 | @HostListener('click') handleClick(): void {
135 | this.inputFocus()
136 | }
137 |
138 | @HostListener('focus') handleFocus(): void {
139 | this.inputFocus()
140 | }
141 |
142 | @HostListener('input', ['$event']) handleInput(event: any): void {
143 | this.inputChange(event)
144 | }
145 |
146 | ngOnDestroy(): void {
147 | if (this.cmpRef != null) {
148 | if (this.viewAttachedToAppRef) {
149 | this.appRef.detachView(this.cmpRef.hostView)
150 | }
151 |
152 | this.cmpRef.destroy()
153 |
154 | this.cmpRef = null
155 | this.dialog = null
156 | }
157 | }
158 |
159 | ngOnChanges(changes: any): void {
160 | if (changes.cpToggle && !this.cpDisabled) {
161 | if (changes.cpToggle.currentValue) {
162 | this.openDialog()
163 | } else if (!changes.cpToggle.currentValue) {
164 | this.closeDialog()
165 | }
166 | }
167 |
168 | if (changes.colorPicker) {
169 | if (this.dialog && !this.ignoreChanges) {
170 | if (this.cpDialogDisplay === 'inline') {
171 | this.dialog.setInitialColor(changes.colorPicker.currentValue)
172 | }
173 |
174 | this.dialog.setColorFromString(changes.colorPicker.currentValue, false)
175 |
176 | if (this.cpUseRootViewContainer && this.cpDialogDisplay !== 'inline') {
177 | this.cmpRef.changeDetectorRef.detectChanges()
178 | }
179 | }
180 |
181 | this.ignoreChanges = false
182 | }
183 |
184 | if (changes.cpPresetLabel || changes.cpPresetColors) {
185 | if (this.dialog) {
186 | this.dialog.setPresetConfig(this.cpPresetLabel, this.cpPresetColors)
187 | }
188 | }
189 | }
190 |
191 | public openDialog(): void {
192 | if (!this.dialogCreated) {
193 | let vcRef = this.vcRef
194 |
195 | this.dialogCreated = true
196 | this.viewAttachedToAppRef = false
197 |
198 | if (this.cpUseRootViewContainer && this.cpDialogDisplay !== 'inline') {
199 | const classOfRootComponent = this.appRef.componentTypes[0]
200 | const appInstance = this.injector.get(
201 | classOfRootComponent,
202 | Injector.NULL
203 | )
204 |
205 | if (appInstance !== Injector.NULL) {
206 | vcRef =
207 | appInstance.vcRef || appInstance.viewContainerRef || this.vcRef
208 |
209 | if (isDevMode() && vcRef === this.vcRef) {
210 | console.warn(
211 | 'You are using cpUseRootViewContainer, ' +
212 | 'but the root component is not exposing viewContainerRef!' +
213 | "Please expose it by adding 'public vcRef: ViewContainerRef' to the constructor."
214 | )
215 | }
216 | } else {
217 | this.viewAttachedToAppRef = true
218 | }
219 | }
220 |
221 | if (this.viewAttachedToAppRef) {
222 | this.cmpRef = vcRef.createComponent(ColorPickerComponent, {
223 | injector: this.injector,
224 | })
225 | document.body.appendChild(
226 | (this.cmpRef.hostView as EmbeddedViewRef)
227 | .rootNodes[0] as HTMLElement
228 | )
229 | } else {
230 | const injector = Injector.create({
231 | providers: [],
232 | // We shouldn't use `vcRef.parentInjector` since it's been deprecated long time ago and might be removed
233 | // in newer Angular versions: https://github.com/angular/angular/pull/25174.
234 | parent: vcRef.injector,
235 | })
236 |
237 | this.cmpRef = vcRef.createComponent(ColorPickerComponent, {
238 | injector,
239 | index: 0,
240 | })
241 | }
242 |
243 | this.cmpRef.instance.setupDialog(
244 | this,
245 | this.elRef,
246 | this.colorPicker,
247 | this.cpWidth,
248 | this.cpHeight,
249 | this.cpDialogDisplay,
250 | this.cpFallbackColor,
251 | this.cpColorMode,
252 | this.cpCmykEnabled,
253 | this.cpAlphaChannel,
254 | this.cpOutputFormat,
255 | this.cpDisableInput,
256 | this.cpIgnoredElements,
257 | this.cpSaveClickOutside,
258 | this.cpCloseClickOutside,
259 | this.cpUseRootViewContainer,
260 | this.cpPosition,
261 | this.cpPositionOffset,
262 | this.cpPositionRelativeToArrow,
263 | this.cpPresetLabel,
264 | this.cpPresetColors,
265 | this.cpPresetColorsClass,
266 | this.cpMaxPresetColorsLength,
267 | this.cpPresetEmptyMessage,
268 | this.cpPresetEmptyMessageClass,
269 | this.cpOKButton,
270 | this.cpOKButtonClass,
271 | this.cpOKButtonText,
272 | this.cpCancelButton,
273 | this.cpCancelButtonClass,
274 | this.cpCancelButtonText,
275 | this.cpAddColorButton,
276 | this.cpAddColorButtonClass,
277 | this.cpAddColorButtonText,
278 | this.cpRemoveColorButtonClass,
279 | this.cpEyeDropper,
280 | this.elRef,
281 | this.cpExtraTemplate
282 | )
283 |
284 | this.dialog = this.cmpRef.instance
285 |
286 | if (this.vcRef !== vcRef) {
287 | this.cmpRef.changeDetectorRef.detectChanges()
288 | }
289 | } else if (this.dialog) {
290 | // Update properties.
291 | this.cmpRef.instance.cpAlphaChannel = this.cpAlphaChannel
292 |
293 | // Open dialog.
294 | this.dialog.openDialog(this.colorPicker)
295 | }
296 | }
297 |
298 | public closeDialog(): void {
299 | if (this.dialog && this.cpDialogDisplay === 'popup') {
300 | this.dialog.closeDialog()
301 | }
302 | }
303 |
304 | public cmykChanged(value: string): void {
305 | this.cpCmykColorChange.emit(value)
306 | }
307 |
308 | public stateChanged(state: boolean): void {
309 | this.cpToggleChange.emit(state)
310 |
311 | if (state) {
312 | this.colorPickerOpen.emit(this.colorPicker)
313 | } else {
314 | this.colorPickerClose.emit(this.colorPicker)
315 | }
316 | }
317 |
318 | public colorChanged(value: string, ignore: boolean = true): void {
319 | this.ignoreChanges = ignore
320 |
321 | this.colorPickerChange.emit(value)
322 | }
323 |
324 | public colorSelected(value: string): void {
325 | this.colorPickerSelect.emit(value)
326 | }
327 |
328 | public colorCanceled(): void {
329 | this.colorPickerCancel.emit()
330 | }
331 |
332 | public inputFocus(): void {
333 | const element = this.elRef.nativeElement
334 |
335 | const ignored = this.cpIgnoredElements.filter(
336 | (item: any) => item === element
337 | )
338 |
339 | if (!this.cpDisabled && !ignored.length) {
340 | if (
341 | typeof document !== 'undefined' &&
342 | element === document.activeElement
343 | ) {
344 | this.openDialog()
345 | } else if (!this.dialog || !this.dialog.show) {
346 | this.openDialog()
347 | } else {
348 | this.closeDialog()
349 | }
350 | }
351 | }
352 |
353 | public inputChange(event: any): void {
354 | if (this.dialog) {
355 | this.dialog.setColorFromString(event.target.value, true)
356 | } else {
357 | this.colorPicker = event.target.value
358 |
359 | this.colorPickerChange.emit(this.colorPicker)
360 | }
361 | }
362 |
363 | public inputChanged(event: any): void {
364 | this.cpInputChange.emit(event)
365 | }
366 |
367 | public sliderChanged(event: any): void {
368 | this.cpSliderChange.emit(event)
369 | }
370 |
371 | public sliderDragEnd(event: { slider: string; color: string }): void {
372 | this.cpSliderDragEnd.emit(event)
373 | }
374 |
375 | public sliderDragStart(event: { slider: string; color: string }): void {
376 | this.cpSliderDragStart.emit(event)
377 | }
378 |
379 | public presetColorsChanged(value: any[]): void {
380 | this.cpPresetColorsChange.emit(value)
381 | }
382 | }
383 |
--------------------------------------------------------------------------------
/projects/app/src/app/app.component.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
Angular Color Picker Directive
5 |
A Color Picker Directive for Angular with no dependencies.
6 |
based on angular2-color-picker by Alberto Pujante
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
Usage:
19 |
20 | <input [(colorPicker)]="color"
21 | [style.background]="color"/>
22 |
23 |
Or:
24 |
25 | <input [style.background]="color"
26 | [colorPicker]="color"
27 | (colorPickerChange)="color=$event"/>
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
Grayscale color mode:
41 |
42 | <input [(colorPicker)]="color" [cpColorMode]="'grayscale'"
43 | [style.background]="color"/>
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
Show the color in the input field:
57 |
58 | <input [value]="color"
59 | [style.background]="color"
60 | [(colorPicker)]="color"/>
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
Output format:
78 |
79 | <input [value]="color"
80 | [style.background]="color"
81 | [cpOutputFormat]="'rgba'"
82 | [(colorPicker)]="color"/>
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
Changing dialog position:
96 |
97 | <input [value]="color"
98 | [style.background]="color"
99 | [cpPosition]="'top-right'"
100 | [(colorPicker)]="color"/>
101 |
102 |
103 |
104 |
105 |
106 |
107 |
108 |
109 | Change me!
110 |
111 |
112 |
113 |
You can introduce a offset of the color picker relative to the html element:
114 |
115 | <span [style.color]="color"
116 | [cpPosition]="'bottom'"
117 | [cpPositionOffset]="'50%'"
118 | [cpPositionRelativeToArrow]="true"
119 | [(colorPicker)]="color">Change me!</span>
120 |
121 |
122 |
123 |
124 |
125 |
126 |
127 |
128 |
129 |
130 |
131 |
132 |
Show cancel button:
133 |
134 | <input [value]="color"
135 | [style.background]="color"
136 | [cpCancelButton]="true"
137 | [(colorPicker)]="color"/>
138 |
139 |
140 |
141 |
142 |
143 |
144 |
145 |
146 |
147 |
148 |
149 |
150 |
Change cancel button class, in this example we are using a bootstrap button:
151 |
152 | <input [value]="color"
153 | [style.background]="color"
154 | [cpCancelButton]="true"
155 | [cpCancelButtonClass]= "'btn btn-primary btn-xs'"
156 | [(colorPicker)]="color"/>
157 |
158 |
159 |
160 |
161 |
162 |
163 |
164 |
165 |
166 |
167 |
168 |
169 |
Show OK button:
170 |
171 | <input [value]="color"
172 | [style.background]="color"
173 | [cpOKButton]="true"
174 | [cpSaveClickOutside]="false"
175 | [cpOKButtonClass]= "'btn btn-primary btn-xs'"
176 | [(colorPicker)]="color"/>
177 |
178 |
179 |
180 |
181 |
182 |
183 |
184 |
185 |
186 |
187 |
188 |
189 |
Enable Eye Dropper:
190 |
You can open the eye dropper by clicking the colored circle.
191 |
192 | <input [value]="color"
193 | [style.background]="color"
194 | [cpEyeDropper]="true"
195 | [cpSaveClickOutside]="false"
196 | [cpOKButtonClass]= "'btn btn-primary btn-xs'"
197 | [(colorPicker)]="color"/>
198 |
199 |
200 |
201 |
202 |
203 |
204 |
205 |
206 |
207 |
208 |
209 |
210 |
{{cmykValue}}
211 |
212 |
213 | C
214 |
215 | M
216 |
217 |
218 |
219 |
220 |
221 | Y
222 |
223 | K
224 |
225 |
226 |
227 |
228 |
229 |
230 |
Change event color:
231 |
232 | <input [style.background]="color"
233 | [colorPicker]="color"
234 | [cpCmykEnabled]="true"
235 | (cpCmykColorChange)="cmykValue=$event"
236 | (colorPickerChange)="cmykColor=onChangeColorCmyk($event);color=$event"/>
237 |
238 | <span [style.font-size.px]="100 * cmykColor.c"/>C</span/>
239 | <span [style.font-size.px]="100 * cmykColor.m"/>M</span/>
240 | <span [style.font-size.px]="100 * cmykColor.y"/>Y</span/>
241 | <span [style.font-size.px]="100 * cmykColor.k"/>K</span/>
242 |
243 |
244 |
245 |
246 |
247 |
248 |
249 |
250 |
251 |
252 |
253 |
254 |
With preset colors:
255 |
256 | <input [style.background]="color"
257 | [cpPresetColors]="['#fff', '#000', '#2889e9', '#e920e9', '#fff500', 'rgb(236,64,64)']"
258 | [(colorPicker)]="color"/>
259 |
260 |
261 |
262 |
263 |
264 |
265 |
266 |
267 |
268 |
269 |
270 |
271 |
Add and remove preset colors:
272 |
273 | <input [style.background]="color"
274 | [cpAlphaChannel]="'always'"
275 | [cpOutputFormat]="'rgba'"
276 | [cpPresetColors]="['#fff', '#2889e9']"
277 | [cpAddColorButton]="true"
278 | [(colorPicker)]="color"/>
279 |
280 |
281 |
282 |
283 |
284 |
285 |
286 |
287 |
288 |
289 |
290 |
291 |
Toggle
292 |
293 |
294 |
295 |
Toggle status: {{toggle}}
296 |
297 |
298 |
299 |
Use cpToggle with cpIgnoredElements:
300 |
301 | <input #ignoredInput
302 | [style.background]="color"
303 | [cpIgnoredElements]="[ignoredButton, ignoredInput]"
304 | [(cpToggle)]="toggle"
305 | [(colorPicker)]="color"/>
306 |
307 | <button #ignoredButton (click)="toggle=!toggle"></button>
308 |
309 |
310 |
311 |
312 |
313 |
314 |
315 |
316 |
317 |
318 |
319 |
320 |
Auto positioning:
321 |
322 | <input [value]="color"
323 | [style.background]="color"
324 | cpPosition="auto"
325 | [(colorPicker)]="color"/>
326 |
327 |
328 |
329 |
330 |
331 |
332 |
394 |
395 |
396 |
397 |
398 |
399 |
400 |
401 |
402 |
403 |
404 |
405 |
406 |
407 |
408 |
409 |
410 |
411 |
412 |
413 |
414 |
415 |
Show the dialog permanently:
416 |
417 | <span [style.background]="arrayColors[selectedColor]"
418 | [cpToggle]="true"
419 | [cpDialogDisplay]="'inline'"
420 | [cpCancelButton]="true"
421 | [(colorPicker)]="arrayColors[selectedColor]"></span>
422 |
423 | <div [style.background]="arrayColors['color1']"
424 | (click)="selectedColor='color1'"></div>
425 |
426 | <div [style.background]="arrayColors['color2']"
427 | (click)="selectedColor='color2'"></div>
428 |
429 |
430 |
431 |
432 |
433 |
434 |
435 |
436 |
437 |
438 |
449 |
450 |
451 |
452 |
Custom template content here.
453 |
454 |
455 |
456 |
457 |
458 |
Custom template:
459 |
460 | <input (colorPickerClose)="onEventLog('colorPickerClose', $event)"
461 | (colorPickerOpen)="onEventLog('colorPickerOpen', $event)"
462 | (cpInputChange)="onEventLog('cpInputChange', $event)"
463 | (cpSliderDragEnd)="onEventLog('cpSliderDragEnd', $event)"
464 | (cpSliderDragStart)="onEventLog('cpSliderDragStart', $event)"
465 | [cpExtraTemplate]="templateTest"
466 | [style.background]="color"
467 | />
468 |
469 | <ng-template #customTemplate>
470 | <div class="">
471 | <h3>Custom template.</h3>
472 | </div>
473 | </ng-template>
474 |
475 |
476 |
477 |
478 |
479 |
480 |
481 |
482 | Options
483 | Values (default values in bold)
484 |
485 |
486 |
487 |
488 |
489 | cpOutputFormat
490 |
491 | 'auto' , 'hex', 'rgba', 'hsla'
492 |
493 |
494 |
495 |
496 | cpPosition
497 |
498 | 'auto' , 'top', 'bottom', 'top-right', 'top-left', 'bottom-left', 'bottom-right'
499 |
500 |
501 |
502 |
503 | cpPositionOffset
504 |
505 | '0%'
506 | Dialog offset (percent) relative to the element that contains the directive.
507 |
508 |
509 |
510 |
511 | cpPositionRelativeToArrow
512 |
513 | false , true
514 | Dialog position is calculated relative to the dialog (false) or relative to the dialog arrow (true).
515 |
516 |
517 |
518 |
519 | cpWidth
520 |
521 | '230px'
522 | Use this option to set color picker dialog width (pixels).
523 |
524 |
525 |
526 |
527 | cpHeight
528 |
529 | 'auto'
530 | Use this option to force color picker dialog height (pixels).
531 |
532 |
533 |
534 |
535 | cpSaveClickOutside
536 |
537 | true , false
538 | If true the initial color is restored when user clicks outside.
539 |
540 |
541 |
542 |
543 | cpOKButton
544 |
545 | false , true
546 | Shows the Ok button. Saves the selected color.
547 |
548 |
549 |
550 |
551 | cpOKButtonText
552 |
553 | 'OK'
554 |
555 |
556 |
557 |
558 | cpOKButtonClass
559 |
560 | Class to customize the OK button.
561 |
562 |
563 |
564 |
565 | cpCancelButton
566 |
567 | false , true
568 | Shows the Cancel button. Cancel the selected color.
569 |
570 |
571 |
572 |
573 | cpCancelButtonText
574 |
575 | 'Cancel'
576 |
577 |
578 |
579 |
580 | cpCancelButtonClass
581 |
582 | Class to customize the Cancel button.
583 |
584 |
585 |
586 |
587 | cpFallbackColor
588 |
589 | '#fff'
590 | Is used when the color is not well-formed or not defined.
591 |
592 |
593 |
594 |
595 | cpPresetLabel
596 |
597 | 'Preset colors'
598 | Label for preset colors if any provided used.
599 |
600 |
601 |
602 |
603 | cpPresetColors
604 |
605 | []
606 | Array of preset colors to show in the color picker dialog.
607 |
608 |
609 |
610 |
611 | cpToggle
612 |
613 | false , true
614 | Input/ouput to open/close the color picker.
615 |
616 |
617 |
618 |
619 | cpIgnoredElements
620 |
621 | []
622 | Array of HTML elements that will be ignored by the color picker when they are clicked.
623 |
624 |
625 |
626 |
627 | cpDialogDisplay
628 |
629 | 'popup' , 'inline'
630 | popup: dialog is showed when user clicks in the directive.
631 | inline: dialog is showed permanently. You can show/hide the dialog with cpToggle.
632 |
633 |
634 |
635 |
636 | cpDisableInput
637 |
638 | false , true
639 | Disables / hides the color input field from the dialog.
640 |
641 |
642 |
643 |
644 | cpAlphaChannel
645 |
646 | 'enabled' , 'disabled', 'always', 'forced'
647 | enabled: alpha channel is not included for hexadecimal (hex6) values or for values without alpha (alpha = 1).
648 | disabled: alpha channel is completely disabled.
649 | always: alpha channel is included for hexadecimal (hex6) values and values without alpha (alpha = 1).
650 | forced: alpha channel field is added for hexadecimal (hex6) values.
651 |
652 |
653 |
654 |
655 | cpCmykEnabled
656 |
657 | false , true
658 | Enables CMYK color input and selected CMYK color event sending on color change.
659 |
660 |
661 |
662 |
663 | cpUseRootViewContainer
664 |
665 | false , true
666 | Create dialog component in the root view container instead the elements view container.
667 |
668 |
669 |
670 |
671 | cpAddColorButton
672 |
673 | false , true
674 | Add or remove colors into your preset panel. The [cpPresetColors] is needed
675 |
676 |
677 |
678 |
679 | cpAddColorButtonText
680 |
681 | 'Add color'
682 |
683 |
684 |
685 |
686 | cpAddColorButtonClass
687 |
688 | Class to customize the add color button.
689 |
690 |
691 |
692 |
693 | cpRemoveColorButtonClass
694 |
695 | Class to customize the remove color button.
696 |
697 |
698 |
699 |
700 | cpPresetColorsClass
701 |
702 | Class to customize the preset colors container.
703 |
704 |
705 |
706 |
707 | cpMaxPresetColorsLength
708 |
709 | 8 (number)
710 | Use this option to set the max colors allowed into preset panel.
711 |
712 |
713 |
714 |
715 | cpPresetEmptyMessage
716 |
717 | 'No colors added'
718 | Message for empty colors if any provided used.
719 |
720 |
721 |
722 |
723 | cpPresetEmptyMessageClass
724 |
725 | Class to customize the empty colors message.
726 |
727 |
728 |
729 | cpEyeDropper
730 |
731 | Enable eye dropper on click of colored circle. Click again to pick a color.
732 |
733 |
734 |
735 |
736 |
737 |
738 |
739 |
740 |
741 |
742 |
743 |
744 |
745 |
746 | Events
747 | Description (data format in bold)
748 |
749 |
750 |
751 |
752 |
753 | colorPickerChange
754 |
755 | Changed color value, send when color is changed. (value: string)
756 |
757 |
758 |
759 |
760 | colorPickerSelect
761 |
762 | Selected color value, send when user presses the OK button. (value: string)
763 |
764 |
765 |
766 |
767 | cpToggleChange
768 |
769 | Status of the dialog, send when dialog is opened / closed. (open: boolean)
770 |
771 |
772 |
773 |
774 | cpInputChange
775 |
776 | Input name and its value, send when user changes color through inputs. ({{ '{' }}input: string, value: string{{ '}' }})
777 |
778 |
779 |
780 |
781 | cpSliderChange
782 |
783 | Slider name and its value, send when user changes color through slider. ({{ '{' }}slider: string, value: Object{{ '}' }})
784 |
785 |
786 |
787 |
788 | cpCmykColorChange
789 |
790 | CMYK color value, send when on color change if cpCmykEnabled is true. (value: string)
791 |
792 |
793 |
794 |
795 | cpPresetColorsChange
796 |
797 | Preset colors value, send when Add Color button is pressed. (value: array)
798 |
799 |
800 |
801 |
802 |
803 |
804 |
805 |
806 |
807 |
808 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/color-picker.component.css:
--------------------------------------------------------------------------------
1 | .color-picker {
2 | position: absolute;
3 | z-index: 1000;
4 |
5 | width: 230px;
6 | height: auto;
7 | border: #777 solid 1px;
8 |
9 | cursor: default;
10 |
11 | -webkit-user-select: none;
12 | -khtml-user-select: none;
13 | -moz-user-select: none;
14 | -ms-user-select: none;
15 |
16 | user-select: none;
17 | background-color: #fff;
18 | }
19 |
20 | .color-picker * {
21 | -webkit-box-sizing: border-box;
22 | -moz-box-sizing: border-box;
23 |
24 | box-sizing: border-box;
25 | margin: 0;
26 |
27 | font-size: 11px;
28 | }
29 |
30 | .color-picker input {
31 | width: 0;
32 | height: 26px;
33 | min-width: 0;
34 |
35 | font-size: 13px;
36 | text-align: center;
37 | color: #000;
38 | }
39 |
40 | .color-picker input:invalid,
41 | .color-picker input:-moz-ui-invalid,
42 | .color-picker input:-moz-submit-invalid {
43 | box-shadow: none;
44 | }
45 |
46 | .color-picker input::-webkit-inner-spin-button,
47 | .color-picker input::-webkit-outer-spin-button {
48 | margin: 0;
49 |
50 | -webkit-appearance: none;
51 | }
52 |
53 | .color-picker .arrow {
54 | position: absolute;
55 | z-index: 999999;
56 |
57 | width: 0;
58 | height: 0;
59 | border-style: solid;
60 | }
61 |
62 | .color-picker .arrow.arrow-top {
63 | left: 8px;
64 |
65 | border-width: 10px 5px;
66 | border-color: #777 rgba(0, 0, 0, 0) rgba(0, 0, 0, 0) rgba(0, 0, 0, 0);
67 | }
68 |
69 | .color-picker .arrow.arrow-bottom {
70 | top: -20px;
71 | left: 8px;
72 |
73 | border-width: 10px 5px;
74 | border-color: rgba(0, 0, 0, 0) rgba(0, 0, 0, 0) #777 rgba(0, 0, 0, 0);
75 | }
76 |
77 | .color-picker .arrow.arrow-top-left,
78 | .color-picker .arrow.arrow-left-top {
79 | right: -21px;
80 | bottom: 8px;
81 |
82 | border-width: 5px 10px;
83 | border-color: rgba(0, 0, 0, 0) rgba(0, 0, 0, 0) rgba(0, 0, 0, 0) #777;
84 | }
85 |
86 | .color-picker .arrow.arrow-top-right,
87 | .color-picker .arrow.arrow-right-top {
88 | bottom: 8px;
89 | left: -20px;
90 |
91 | border-width: 5px 10px;
92 | border-color: rgba(0, 0, 0, 0) #777 rgba(0, 0, 0, 0) rgba(0, 0, 0, 0);
93 | }
94 |
95 | .color-picker .arrow.arrow-left,
96 | .color-picker .arrow.arrow-left-bottom,
97 | .color-picker .arrow.arrow-bottom-left {
98 | top: 8px;
99 | right: -21px;
100 |
101 | border-width: 5px 10px;
102 | border-color: rgba(0, 0, 0, 0) rgba(0, 0, 0, 0) rgba(0, 0, 0, 0) #777;
103 | }
104 |
105 | .color-picker .arrow.arrow-right,
106 | .color-picker .arrow.arrow-right-bottom,
107 | .color-picker .arrow.arrow-bottom-right {
108 | top: 8px;
109 | left: -20px;
110 |
111 | border-width: 5px 10px;
112 | border-color: rgba(0, 0, 0, 0) #777 rgba(0, 0, 0, 0) rgba(0, 0, 0, 0);
113 | }
114 |
115 | .color-picker .cursor {
116 | position: relative;
117 |
118 | width: 16px;
119 | height: 16px;
120 | border: #222 solid 2px;
121 | border-radius: 50%;
122 |
123 | cursor: default;
124 | }
125 |
126 | .color-picker .box {
127 | display: flex;
128 | padding: 4px 8px;
129 | }
130 |
131 | .color-picker .left {
132 | position: relative;
133 |
134 | padding: 16px 8px;
135 | }
136 |
137 | .color-picker .right {
138 | -webkit-flex: 1 1 auto;
139 | -ms-flex: 1 1 auto;
140 |
141 | flex: 1 1 auto;
142 |
143 | padding: 12px 8px;
144 | }
145 |
146 | .color-picker .button-area {
147 | padding: 0 16px 16px;
148 |
149 | text-align: right;
150 | }
151 |
152 | .color-picker .button-area button {
153 | margin-left: 8px;
154 | }
155 |
156 | .color-picker .preset-area {
157 | padding: 4px 15px;
158 | }
159 |
160 | .color-picker .preset-area .preset-label {
161 | overflow: hidden;
162 | width: 100%;
163 | padding: 4px;
164 |
165 | font-size: 11px;
166 | white-space: nowrap;
167 | text-align: left;
168 | text-overflow: ellipsis;
169 | color: #555;
170 | }
171 |
172 | .color-picker .preset-area .preset-color {
173 | position: relative;
174 |
175 | display: inline-block;
176 | width: 18px;
177 | height: 18px;
178 | margin: 4px 6px 8px;
179 | border: #a9a9a9 solid 1px;
180 | border-radius: 25%;
181 |
182 | cursor: pointer;
183 | }
184 |
185 | .color-picker .preset-area .preset-empty-message {
186 | min-height: 18px;
187 | margin-top: 4px;
188 | margin-bottom: 8px;
189 |
190 | font-style: italic;
191 | text-align: center;
192 | }
193 |
194 | .color-picker .hex-text {
195 | width: 100%;
196 | padding: 4px 8px;
197 |
198 | font-size: 11px;
199 | }
200 |
201 | .color-picker .hex-text .box {
202 | padding: 0 24px 8px 8px;
203 | }
204 |
205 | .color-picker .hex-text .box div {
206 | float: left;
207 |
208 | -webkit-flex: 1 1 auto;
209 | -ms-flex: 1 1 auto;
210 |
211 | flex: 1 1 auto;
212 |
213 | text-align: center;
214 | color: #555;
215 | clear: left;
216 | }
217 |
218 | .color-picker .hex-text .box input {
219 | -webkit-flex: 1 1 auto;
220 | -ms-flex: 1 1 auto;
221 |
222 | flex: 1 1 auto;
223 | padding: 1px;
224 | border: #a9a9a9 solid 1px;
225 | }
226 |
227 | .color-picker .hex-alpha .box div:first-child,
228 | .color-picker .hex-alpha .box input:first-child {
229 | flex-grow: 3;
230 | margin-right: 8px;
231 | }
232 |
233 | .color-picker .cmyk-text,
234 | .color-picker .hsla-text,
235 | .color-picker .rgba-text,
236 | .color-picker .value-text {
237 | width: 100%;
238 | padding: 4px 8px;
239 |
240 | font-size: 11px;
241 | }
242 |
243 | .color-picker .cmyk-text .box,
244 | .color-picker .hsla-text .box,
245 | .color-picker .rgba-text .box {
246 | padding: 0 24px 8px 8px;
247 | }
248 |
249 | .color-picker .value-text .box {
250 | padding: 0 8px 8px;
251 | }
252 |
253 | .color-picker .cmyk-text .box div,
254 | .color-picker .hsla-text .box div,
255 | .color-picker .rgba-text .box div,
256 | .color-picker .value-text .box div {
257 | -webkit-flex: 1 1 auto;
258 | -ms-flex: 1 1 auto;
259 |
260 | flex: 1 1 auto;
261 | margin-right: 8px;
262 |
263 | text-align: center;
264 | color: #555;
265 | }
266 |
267 | .color-picker .cmyk-text .box div:last-child,
268 | .color-picker .hsla-text .box div:last-child,
269 | .color-picker .rgba-text .box div:last-child,
270 | .color-picker .value-text .box div:last-child {
271 | margin-right: 0;
272 | }
273 |
274 | .color-picker .cmyk-text .box input,
275 | .color-picker .hsla-text .box input,
276 | .color-picker .rgba-text .box input,
277 | .color-picker .value-text .box input {
278 | float: left;
279 |
280 | -webkit-flex: 1;
281 | -ms-flex: 1;
282 |
283 | flex: 1;
284 | padding: 1px;
285 | margin: 0 8px 0 0;
286 | border: #a9a9a9 solid 1px;
287 | }
288 |
289 | .color-picker .cmyk-text .box input:last-child,
290 | .color-picker .hsla-text .box input:last-child,
291 | .color-picker .rgba-text .box input:last-child,
292 | .color-picker .value-text .box input:last-child {
293 | margin-right: 0;
294 | }
295 |
296 | .color-picker .hue-alpha {
297 | align-items: center;
298 | margin-bottom: 3px;
299 | }
300 |
301 | .color-picker .hue {
302 | direction: ltr;
303 |
304 | width: 100%;
305 | height: 16px;
306 | margin-bottom: 16px;
307 | border: none;
308 |
309 | cursor: pointer;
310 | background-size: 100% 100%;
311 | background-image: url('');
312 | }
313 |
314 | .color-picker .value {
315 | direction: rtl;
316 |
317 | width: 100%;
318 | height: 16px;
319 | margin-bottom: 16px;
320 | border: none;
321 |
322 | cursor: pointer;
323 | background-size: 100% 100%;
324 | background-image: url('');
325 | }
326 |
327 | .color-picker .alpha {
328 | direction: ltr;
329 |
330 | width: 100%;
331 | height: 16px;
332 | border: none;
333 |
334 | cursor: pointer;
335 | background-size: 100% 100%;
336 | background-image: url('');
337 | }
338 |
339 | .color-picker .type-policy {
340 | position: absolute;
341 | top: 218px;
342 | right: 12px;
343 |
344 | width: 16px;
345 | height: 24px;
346 |
347 | background-size: 8px 16px;
348 | background-image: url('');
349 | background-repeat: no-repeat;
350 | background-position: center;
351 | }
352 |
353 | .color-picker .type-policy .type-policy-arrow {
354 | display: block;
355 |
356 | width: 100%;
357 | height: 50%;
358 | }
359 |
360 | .color-picker .selected-color {
361 | position: absolute;
362 | top: 16px;
363 | left: 8px;
364 |
365 | width: 40px;
366 | height: 40px;
367 | border: 1px solid #a9a9a9;
368 | border-radius: 50%;
369 | }
370 |
371 | .color-picker .selected-color-background {
372 | width: 40px;
373 | height: 40px;
374 | border-radius: 50%;
375 |
376 | background-image: url('');
377 | }
378 |
379 | .color-picker .saturation-lightness {
380 | direction: ltr;
381 |
382 | width: 100%;
383 | height: 130px;
384 | border: none;
385 |
386 | cursor: pointer;
387 | touch-action: manipulation;
388 | background-size: 100% 100%;
389 | background-image: url('');
390 | }
391 |
392 | .color-picker .cp-add-color-button-class {
393 | position: absolute;
394 |
395 | display: inline;
396 | padding: 0;
397 | margin: 3px -3px;
398 | border: 0;
399 |
400 | cursor: pointer;
401 | background: transparent;
402 | }
403 |
404 | .color-picker .cp-add-color-button-class:hover {
405 | text-decoration: underline;
406 | }
407 |
408 | .color-picker .cp-add-color-button-class:disabled {
409 | cursor: not-allowed;
410 | color: #999;
411 | }
412 |
413 | .color-picker .cp-add-color-button-class:disabled:hover {
414 | text-decoration: none;
415 | }
416 |
417 | .color-picker .cp-remove-color-button-class {
418 | position: absolute;
419 | top: -5px;
420 | right: -5px;
421 |
422 | display: block;
423 | width: 10px;
424 | height: 10px;
425 | border-radius: 50%;
426 |
427 | cursor: pointer;
428 | text-align: center;
429 | background: #fff;
430 |
431 | box-shadow: 1px 1px 5px #333;
432 | }
433 |
434 | .color-picker .cp-remove-color-button-class::before {
435 | content: 'x';
436 |
437 | position: relative;
438 | bottom: 3.5px;
439 |
440 | display: inline-block;
441 |
442 | font-size: 10px;
443 | }
444 |
445 | .color-picker .eyedropper-icon {
446 | position: absolute;
447 | top: 50%;
448 | left: 50%;
449 | transform: translate(-50%, -50%);
450 | fill: white;mix-blend-mode: exclusion;
451 | }
452 |
--------------------------------------------------------------------------------
/projects/lib/src/lib/color-picker.component.ts:
--------------------------------------------------------------------------------
1 | import {
2 | Component,
3 | OnInit,
4 | OnDestroy,
5 | AfterViewInit,
6 | ViewChild,
7 | HostListener,
8 | ViewEncapsulation,
9 | ElementRef,
10 | ChangeDetectorRef,
11 | TemplateRef,
12 | NgZone,
13 | PLATFORM_ID,
14 | inject
15 | } from '@angular/core'
16 |
17 | import {
18 | DOCUMENT,
19 | isPlatformBrowser,
20 | NgForOf,
21 | NgIf,
22 | NgTemplateOutlet
23 | } from '@angular/common'
24 |
25 | import {
26 | calculateAutoPositioning,
27 | SliderDirective,
28 | TextDirective
29 | } from './helpers'
30 |
31 | import { ColorFormats, Cmyk, Hsla, Hsva, Rgba } from './formats'
32 | import {
33 | AlphaChannel,
34 | OutputFormat,
35 | SliderDimension,
36 | SliderPosition
37 | } from './helpers'
38 |
39 | import { ColorPickerService } from './color-picker.service'
40 |
41 | // Do not store that on the class instance since the condition will be run
42 | // every time the class is created.
43 | const SUPPORTS_TOUCH = typeof window !== 'undefined' && 'ontouchstart' in window
44 |
45 | @Component({
46 | selector: 'color-picker',
47 | templateUrl: './color-picker.component.html',
48 | styleUrls: ['./color-picker.component.css'],
49 | encapsulation: ViewEncapsulation.None,
50 | imports: [SliderDirective, TextDirective, NgIf, NgForOf, NgTemplateOutlet]
51 | })
52 | export class ColorPickerComponent implements OnInit, OnDestroy, AfterViewInit {
53 | private ngZone = inject(NgZone)
54 |
55 | private elRef = inject(ElementRef)
56 | private cdRef = inject(ChangeDetectorRef)
57 |
58 | private platformId = inject(PLATFORM_ID)
59 |
60 | private service = inject(ColorPickerService)
61 |
62 | private document = inject(DOCUMENT)
63 |
64 | private cmyk: Cmyk
65 | private hsva: Hsva
66 |
67 | private width: number
68 | private height: number
69 |
70 | private cmykColor: string
71 | private outputColor: string
72 | private initialColor: string
73 | private fallbackColor: string
74 |
75 | private listenerResize: any
76 | private listenerMouseDown: EventListener
77 |
78 | private directiveInstance: any
79 |
80 | private sliderH: number
81 | private sliderDimMax: SliderDimension
82 | private directiveElementRef: ElementRef
83 |
84 | private dialogArrowSize: number = 10
85 | private dialogArrowOffset: number = 15
86 |
87 | private dialogInputFields: ColorFormats[] = [
88 | ColorFormats.HEX,
89 | ColorFormats.RGBA,
90 | ColorFormats.HSLA,
91 | ColorFormats.CMYK
92 | ]
93 |
94 | private useRootViewContainer: boolean = false
95 |
96 | private readonly window: Window
97 |
98 | public show: boolean
99 | public hidden: boolean
100 |
101 | public top: number
102 | public left: number
103 | public position: string
104 |
105 | public format: ColorFormats
106 | public slider: SliderPosition
107 |
108 | public hexText: string
109 | public hexAlpha: number
110 |
111 | public cmykText: Cmyk
112 | public hslaText: Hsla
113 | public rgbaText: Rgba
114 |
115 | public arrowTop: number
116 |
117 | public selectedColor: string
118 | public hueSliderColor: string
119 | public alphaSliderColor: string
120 |
121 | public cpWidth: number
122 | public cpHeight: number
123 |
124 | public cpColorMode: number
125 |
126 | public cpCmykEnabled: boolean
127 |
128 | public cpAlphaChannel: AlphaChannel
129 | public cpOutputFormat: OutputFormat
130 |
131 | public cpDisableInput: boolean
132 | public cpDialogDisplay: string
133 |
134 | public cpIgnoredElements: any
135 |
136 | public cpSaveClickOutside: boolean
137 | public cpCloseClickOutside: boolean
138 |
139 | public cpPosition: string
140 | public cpUsePosition: string
141 | public cpPositionOffset: number
142 |
143 | public cpOKButton: boolean
144 | public cpOKButtonText: string
145 | public cpOKButtonClass: string
146 |
147 | public cpCancelButton: boolean
148 | public cpCancelButtonText: string
149 | public cpCancelButtonClass: string
150 |
151 | public cpEyeDropper: boolean
152 | public eyeDropperSupported: boolean
153 |
154 | public cpPresetLabel: string
155 | public cpPresetColors: string[]
156 | public cpPresetColorsClass: string
157 | public cpMaxPresetColorsLength: number
158 |
159 | public cpPresetEmptyMessage: string
160 | public cpPresetEmptyMessageClass: string
161 |
162 | public cpAddColorButton: boolean
163 | public cpAddColorButtonText: string
164 | public cpAddColorButtonClass: string
165 | public cpRemoveColorButtonClass: string
166 | public cpArrowPosition: number
167 |
168 | public cpTriggerElement: ElementRef
169 |
170 | public cpExtraTemplate: TemplateRef
171 |
172 | @ViewChild('dialogPopup', { static: true }) dialogElement: ElementRef
173 |
174 | @ViewChild('hueSlider', { static: true }) hueSlider: ElementRef
175 | @ViewChild('alphaSlider', { static: true }) alphaSlider: ElementRef
176 |
177 | @HostListener('document:keyup.esc', ['$event']) handleEsc(event: any): void {
178 | if (this.show && this.cpDialogDisplay === 'popup') {
179 | this.onCancelColor(event)
180 | }
181 | }
182 |
183 | @HostListener('document:keyup.enter', ['$event']) handleEnter(
184 | event: any
185 | ): void {
186 | if (this.show && this.cpDialogDisplay === 'popup') {
187 | this.onAcceptColor(event)
188 | }
189 | }
190 |
191 | constructor() {
192 | this.window = this.document.defaultView
193 | this.eyeDropperSupported =
194 | isPlatformBrowser(this.platformId) && 'EyeDropper' in this.window
195 | }
196 |
197 | ngOnInit(): void {
198 | this.slider = new SliderPosition(0, 0, 0, 0)
199 |
200 | const hueWidth = this.hueSlider.nativeElement.offsetWidth || 140
201 | const alphaWidth = this.alphaSlider.nativeElement.offsetWidth || 140
202 |
203 | this.sliderDimMax = new SliderDimension(
204 | hueWidth,
205 | this.cpWidth,
206 | 130,
207 | alphaWidth
208 | )
209 |
210 | if (this.cpCmykEnabled) {
211 | this.format = ColorFormats.CMYK
212 | } else if (this.cpOutputFormat === 'rgba') {
213 | this.format = ColorFormats.RGBA
214 | } else if (this.cpOutputFormat === 'hsla') {
215 | this.format = ColorFormats.HSLA
216 | } else {
217 | this.format = ColorFormats.HEX
218 | }
219 |
220 | this.listenerMouseDown = (event: MouseEvent) => {
221 | this.onMouseDown(event)
222 | }
223 | this.listenerResize = () => {
224 | this.onResize()
225 | }
226 |
227 | this.openDialog(this.initialColor, false)
228 | }
229 |
230 | ngOnDestroy(): void {
231 | this.closeDialog()
232 | }
233 |
234 | ngAfterViewInit(): void {
235 | if (this.cpWidth !== 230 || this.cpDialogDisplay === 'inline') {
236 | const hueWidth = this.hueSlider.nativeElement.offsetWidth || 140
237 | const alphaWidth = this.alphaSlider.nativeElement.offsetWidth || 140
238 |
239 | this.sliderDimMax = new SliderDimension(
240 | hueWidth,
241 | this.cpWidth,
242 | 130,
243 | alphaWidth
244 | )
245 |
246 | this.updateColorPicker(false)
247 |
248 | this.cdRef.detectChanges()
249 | }
250 | }
251 |
252 | public openDialog(color: any, emit: boolean = true): void {
253 | this.service.setActive(this)
254 |
255 | if (!this.width) {
256 | this.cpWidth = this.directiveElementRef.nativeElement.offsetWidth
257 | }
258 |
259 | if (!this.height) {
260 | this.height = 320
261 | }
262 |
263 | this.setInitialColor(color)
264 |
265 | this.setColorFromString(color, emit)
266 |
267 | this.openColorPicker()
268 | }
269 |
270 | public closeDialog(): void {
271 | this.closeColorPicker()
272 | }
273 |
274 | public setupDialog(
275 | instance: any,
276 | elementRef: ElementRef,
277 | color: any,
278 | cpWidth: string,
279 | cpHeight: string,
280 | cpDialogDisplay: string,
281 | cpFallbackColor: string,
282 | cpColorMode: string,
283 | cpCmykEnabled: boolean,
284 | cpAlphaChannel: AlphaChannel,
285 | cpOutputFormat: OutputFormat,
286 | cpDisableInput: boolean,
287 | cpIgnoredElements: any,
288 | cpSaveClickOutside: boolean,
289 | cpCloseClickOutside: boolean,
290 | cpUseRootViewContainer: boolean,
291 | cpPosition: string,
292 | cpPositionOffset: string,
293 | cpPositionRelativeToArrow: boolean,
294 | cpPresetLabel: string,
295 | cpPresetColors: string[],
296 | cpPresetColorsClass: string,
297 | cpMaxPresetColorsLength: number,
298 | cpPresetEmptyMessage: string,
299 | cpPresetEmptyMessageClass: string,
300 | cpOKButton: boolean,
301 | cpOKButtonClass: string,
302 | cpOKButtonText: string,
303 | cpCancelButton: boolean,
304 | cpCancelButtonClass: string,
305 | cpCancelButtonText: string,
306 | cpAddColorButton: boolean,
307 | cpAddColorButtonClass: string,
308 | cpAddColorButtonText: string,
309 | cpRemoveColorButtonClass: string,
310 | cpEyeDropper: boolean,
311 | cpTriggerElement: ElementRef,
312 | cpExtraTemplate: TemplateRef
313 | ): void {
314 | this.setInitialColor(color)
315 |
316 | this.setColorMode(cpColorMode)
317 |
318 | this.directiveInstance = instance
319 | this.directiveElementRef = elementRef
320 |
321 | this.cpDisableInput = cpDisableInput
322 |
323 | this.cpCmykEnabled = cpCmykEnabled
324 | this.cpAlphaChannel = cpAlphaChannel
325 | this.cpOutputFormat = cpOutputFormat
326 |
327 | this.cpDialogDisplay = cpDialogDisplay
328 |
329 | this.cpIgnoredElements = cpIgnoredElements
330 |
331 | this.cpSaveClickOutside = cpSaveClickOutside
332 | this.cpCloseClickOutside = cpCloseClickOutside
333 |
334 | this.useRootViewContainer = cpUseRootViewContainer
335 |
336 | this.width = this.cpWidth = parseInt(cpWidth, 10)
337 | this.height = this.cpHeight = parseInt(cpHeight, 10)
338 |
339 | this.cpPosition = cpPosition
340 | this.cpPositionOffset = parseInt(cpPositionOffset, 10)
341 |
342 | this.cpOKButton = cpOKButton
343 | this.cpOKButtonText = cpOKButtonText
344 | this.cpOKButtonClass = cpOKButtonClass
345 |
346 | this.cpCancelButton = cpCancelButton
347 | this.cpCancelButtonText = cpCancelButtonText
348 | this.cpCancelButtonClass = cpCancelButtonClass
349 |
350 | this.cpEyeDropper = cpEyeDropper
351 |
352 | this.fallbackColor = cpFallbackColor || '#fff'
353 |
354 | this.setPresetConfig(cpPresetLabel, cpPresetColors)
355 |
356 | this.cpPresetColorsClass = cpPresetColorsClass
357 | this.cpMaxPresetColorsLength = cpMaxPresetColorsLength
358 | this.cpPresetEmptyMessage = cpPresetEmptyMessage
359 | this.cpPresetEmptyMessageClass = cpPresetEmptyMessageClass
360 |
361 | this.cpAddColorButton = cpAddColorButton
362 | this.cpAddColorButtonText = cpAddColorButtonText
363 | this.cpAddColorButtonClass = cpAddColorButtonClass
364 | this.cpRemoveColorButtonClass = cpRemoveColorButtonClass
365 |
366 | this.cpTriggerElement = cpTriggerElement
367 | this.cpExtraTemplate = cpExtraTemplate
368 |
369 | if (!cpPositionRelativeToArrow) {
370 | this.dialogArrowOffset = 0
371 | }
372 |
373 | if (cpDialogDisplay === 'inline') {
374 | this.dialogArrowSize = 0
375 | this.dialogArrowOffset = 0
376 | }
377 |
378 | if (
379 | cpOutputFormat === 'hex' &&
380 | cpAlphaChannel !== 'always' &&
381 | cpAlphaChannel !== 'forced'
382 | ) {
383 | this.cpAlphaChannel = 'disabled'
384 | }
385 | }
386 |
387 | public setColorMode(mode: string): void {
388 | switch (mode.toString().toUpperCase()) {
389 | case '1':
390 | case 'C':
391 | case 'COLOR':
392 | this.cpColorMode = 1
393 | break
394 | case '2':
395 | case 'G':
396 | case 'GRAYSCALE':
397 | this.cpColorMode = 2
398 | break
399 | case '3':
400 | case 'P':
401 | case 'PRESETS':
402 | this.cpColorMode = 3
403 | break
404 | default:
405 | this.cpColorMode = 1
406 | }
407 | }
408 |
409 | public setInitialColor(color: any): void {
410 | this.initialColor = color
411 | }
412 |
413 | public setPresetConfig(
414 | cpPresetLabel: string,
415 | cpPresetColors: string[]
416 | ): void {
417 | this.cpPresetLabel = cpPresetLabel
418 | this.cpPresetColors = cpPresetColors
419 | }
420 |
421 | public setColorFromString(
422 | value: string,
423 | emit: boolean = true,
424 | update: boolean = true
425 | ): void {
426 | let hsva: Hsva | null
427 |
428 | if (this.cpAlphaChannel === 'always' || this.cpAlphaChannel === 'forced') {
429 | hsva = this.service.stringToHsva(value, true)
430 |
431 | if (!hsva && !this.hsva) {
432 | hsva = this.service.stringToHsva(value, false)
433 | }
434 | } else {
435 | hsva = this.service.stringToHsva(value, false)
436 | }
437 |
438 | if (!hsva && !this.hsva) {
439 | hsva = this.service.stringToHsva(this.fallbackColor, false)
440 | }
441 |
442 | if (hsva) {
443 | this.hsva = hsva
444 |
445 | this.sliderH = this.hsva.h
446 |
447 | if (this.cpOutputFormat === 'hex' && this.cpAlphaChannel === 'disabled') {
448 | this.hsva.a = 1
449 | }
450 |
451 | this.updateColorPicker(emit, update)
452 | }
453 | }
454 |
455 | public onResize(): void {
456 | if (this.position === 'fixed') {
457 | this.setDialogPosition()
458 | } else if (this.cpDialogDisplay !== 'inline') {
459 | this.closeColorPicker()
460 | }
461 | }
462 |
463 | public onDragEnd(slider: string): void {
464 | this.directiveInstance.sliderDragEnd({
465 | slider: slider,
466 | color: this.outputColor
467 | })
468 | }
469 |
470 | public onDragStart(slider: string): void {
471 | this.directiveInstance.sliderDragStart({
472 | slider: slider,
473 | color: this.outputColor
474 | })
475 | }
476 |
477 | public onMouseDown(event: MouseEvent): void {
478 | if (
479 | this.show &&
480 | this.cpDialogDisplay === 'popup' &&
481 | event.target !== this.directiveElementRef.nativeElement &&
482 | !this.isDescendant(this.elRef.nativeElement, event.target) &&
483 | !this.isDescendant(
484 | this.directiveElementRef.nativeElement,
485 | event.target
486 | ) &&
487 | this.cpIgnoredElements.filter((item: any) => item === event.target)
488 | .length === 0
489 | ) {
490 | this.ngZone.run(() => {
491 | if (this.cpSaveClickOutside) {
492 | this.directiveInstance.colorSelected(this.outputColor)
493 | } else {
494 | this.hsva = null
495 |
496 | this.setColorFromString(this.initialColor, false)
497 |
498 | if (this.cpCmykEnabled) {
499 | this.directiveInstance.cmykChanged(this.cmykColor)
500 | }
501 |
502 | this.directiveInstance.colorChanged(this.initialColor)
503 |
504 | this.directiveInstance.colorCanceled()
505 | }
506 |
507 | if (this.cpCloseClickOutside) {
508 | this.closeColorPicker()
509 | }
510 | })
511 | }
512 | }
513 |
514 | public onAcceptColor(event: Event): void {
515 | event.stopPropagation()
516 |
517 | if (this.outputColor) {
518 | this.directiveInstance.colorSelected(this.outputColor)
519 | }
520 |
521 | if (this.cpDialogDisplay === 'popup') {
522 | this.closeColorPicker()
523 | }
524 | }
525 |
526 | public onCancelColor(event: Event): void {
527 | this.hsva = null
528 |
529 | event.stopPropagation()
530 |
531 | this.directiveInstance.colorCanceled()
532 |
533 | this.setColorFromString(this.initialColor, true)
534 |
535 | if (this.cpDialogDisplay === 'popup') {
536 | if (this.cpCmykEnabled) {
537 | this.directiveInstance.cmykChanged(this.cmykColor)
538 | }
539 |
540 | this.directiveInstance.colorChanged(this.initialColor, true)
541 |
542 | this.closeColorPicker()
543 | }
544 | }
545 |
546 | public onEyeDropper(): void {
547 | if (!this.eyeDropperSupported) return
548 | const eyeDropper = new (window as any).EyeDropper()
549 | eyeDropper.open().then((eyeDropperResult: { sRGBHex: string }) => {
550 | this.setColorFromString(eyeDropperResult.sRGBHex, true)
551 | })
552 | }
553 |
554 | public onFormatToggle(change: number): void {
555 | const availableFormats =
556 | this.dialogInputFields.length - (this.cpCmykEnabled ? 0 : 1)
557 |
558 | const nextFormat =
559 | (((this.dialogInputFields.indexOf(this.format) + change) %
560 | availableFormats) +
561 | availableFormats) %
562 | availableFormats
563 |
564 | this.format = this.dialogInputFields[nextFormat]
565 | }
566 |
567 | public onColorChange(value: {
568 | s: number
569 | v: number
570 | rgX: number
571 | rgY: number
572 | }): void {
573 | this.hsva.s = value.s / value.rgX
574 | this.hsva.v = value.v / value.rgY
575 |
576 | this.updateColorPicker()
577 |
578 | this.directiveInstance.sliderChanged({
579 | slider: 'lightness',
580 | value: this.hsva.v,
581 | color: this.outputColor
582 | })
583 |
584 | this.directiveInstance.sliderChanged({
585 | slider: 'saturation',
586 | value: this.hsva.s,
587 | color: this.outputColor
588 | })
589 | }
590 |
591 | public onHueChange(value: { v: number; rgX: number }): void {
592 | this.hsva.h = value.v / value.rgX
593 | this.sliderH = this.hsva.h
594 |
595 | this.updateColorPicker()
596 |
597 | this.directiveInstance.sliderChanged({
598 | slider: 'hue',
599 | value: this.hsva.h,
600 | color: this.outputColor
601 | })
602 | }
603 |
604 | public onValueChange(value: { v: number; rgX: number }): void {
605 | this.hsva.v = value.v / value.rgX
606 |
607 | this.updateColorPicker()
608 |
609 | this.directiveInstance.sliderChanged({
610 | slider: 'value',
611 | value: this.hsva.v,
612 | color: this.outputColor
613 | })
614 | }
615 |
616 | public onAlphaChange(value: { v: number; rgX: number }): void {
617 | this.hsva.a = value.v / value.rgX
618 |
619 | this.updateColorPicker()
620 |
621 | this.directiveInstance.sliderChanged({
622 | slider: 'alpha',
623 | value: this.hsva.a,
624 | color: this.outputColor
625 | })
626 | }
627 |
628 | public onHexInput(value: string | null): void {
629 | if (value === null) {
630 | this.updateColorPicker()
631 | } else {
632 | if (value && value[0] !== '#') {
633 | value = '#' + value
634 | }
635 |
636 | let validHex = /^#[a-f0-9]{6}$/gi
637 |
638 | if (this.cpAlphaChannel === 'always') {
639 | validHex = /^#([a-f0-9]{6}|[a-f0-9]{8})$/gi
640 | }
641 |
642 | const valid = validHex.test(value)
643 |
644 | if (valid) {
645 | if (this.cpAlphaChannel === 'forced') {
646 | value += Math.round(this.hsva.a * 255).toString(16)
647 | }
648 |
649 | this.setColorFromString(value, true, false)
650 | }
651 |
652 | this.directiveInstance.inputChanged({
653 | input: 'hex',
654 | valid: valid,
655 | value: value,
656 | color: this.outputColor
657 | })
658 | }
659 | }
660 |
661 | public onRedInput(value: { v: number; rg: number }): void {
662 | const rgba = this.service.hsvaToRgba(this.hsva)
663 |
664 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
665 |
666 | if (valid) {
667 | rgba.r = value.v / value.rg
668 |
669 | this.hsva = this.service.rgbaToHsva(rgba)
670 |
671 | this.sliderH = this.hsva.h
672 |
673 | this.updateColorPicker()
674 | }
675 |
676 | this.directiveInstance.inputChanged({
677 | input: 'red',
678 | valid: valid,
679 | value: rgba.r,
680 | color: this.outputColor
681 | })
682 | }
683 |
684 | public onBlueInput(value: { v: number; rg: number }): void {
685 | const rgba = this.service.hsvaToRgba(this.hsva)
686 |
687 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
688 |
689 | if (valid) {
690 | rgba.b = value.v / value.rg
691 |
692 | this.hsva = this.service.rgbaToHsva(rgba)
693 |
694 | this.sliderH = this.hsva.h
695 |
696 | this.updateColorPicker()
697 | }
698 |
699 | this.directiveInstance.inputChanged({
700 | input: 'blue',
701 | valid: valid,
702 | value: rgba.b,
703 | color: this.outputColor
704 | })
705 | }
706 |
707 | public onGreenInput(value: { v: number; rg: number }): void {
708 | const rgba = this.service.hsvaToRgba(this.hsva)
709 |
710 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
711 |
712 | if (valid) {
713 | rgba.g = value.v / value.rg
714 |
715 | this.hsva = this.service.rgbaToHsva(rgba)
716 |
717 | this.sliderH = this.hsva.h
718 |
719 | this.updateColorPicker()
720 | }
721 |
722 | this.directiveInstance.inputChanged({
723 | input: 'green',
724 | valid: valid,
725 | value: rgba.g,
726 | color: this.outputColor
727 | })
728 | }
729 |
730 | public onHueInput(value: { v: number; rg: number }) {
731 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
732 |
733 | if (valid) {
734 | this.hsva.h = value.v / value.rg
735 |
736 | this.sliderH = this.hsva.h
737 |
738 | this.updateColorPicker()
739 | }
740 |
741 | this.directiveInstance.inputChanged({
742 | input: 'hue',
743 | valid: valid,
744 | value: this.hsva.h,
745 | color: this.outputColor
746 | })
747 | }
748 |
749 | public onValueInput(value: { v: number; rg: number }): void {
750 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
751 |
752 | if (valid) {
753 | this.hsva.v = value.v / value.rg
754 |
755 | this.updateColorPicker()
756 | }
757 |
758 | this.directiveInstance.inputChanged({
759 | input: 'value',
760 | valid: valid,
761 | value: this.hsva.v,
762 | color: this.outputColor
763 | })
764 | }
765 |
766 | public onAlphaInput(value: { v: number; rg: number }): void {
767 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
768 |
769 | if (valid) {
770 | this.hsva.a = value.v / value.rg
771 |
772 | this.updateColorPicker()
773 | }
774 |
775 | this.directiveInstance.inputChanged({
776 | input: 'alpha',
777 | valid: valid,
778 | value: this.hsva.a,
779 | color: this.outputColor
780 | })
781 | }
782 |
783 | public onLightnessInput(value: { v: number; rg: number }): void {
784 | const hsla = this.service.hsva2hsla(this.hsva)
785 |
786 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
787 |
788 | if (valid) {
789 | hsla.l = value.v / value.rg
790 |
791 | this.hsva = this.service.hsla2hsva(hsla)
792 |
793 | this.sliderH = this.hsva.h
794 |
795 | this.updateColorPicker()
796 | }
797 |
798 | this.directiveInstance.inputChanged({
799 | input: 'lightness',
800 | valid: valid,
801 | value: hsla.l,
802 | color: this.outputColor
803 | })
804 | }
805 |
806 | public onSaturationInput(value: { v: number; rg: number }): void {
807 | const hsla = this.service.hsva2hsla(this.hsva)
808 |
809 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
810 |
811 | if (valid) {
812 | hsla.s = value.v / value.rg
813 |
814 | this.hsva = this.service.hsla2hsva(hsla)
815 |
816 | this.sliderH = this.hsva.h
817 |
818 | this.updateColorPicker()
819 | }
820 |
821 | this.directiveInstance.inputChanged({
822 | input: 'saturation',
823 | valid: valid,
824 | value: hsla.s,
825 | color: this.outputColor
826 | })
827 | }
828 |
829 | public onCyanInput(value: { v: number; rg: number }): void {
830 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
831 |
832 | if (valid) {
833 | this.cmyk.c = value.v
834 |
835 | this.updateColorPicker(false, true, true)
836 | }
837 |
838 | this.directiveInstance.inputChanged({
839 | input: 'cyan',
840 | valid: true,
841 | value: this.cmyk.c,
842 | color: this.outputColor
843 | })
844 | }
845 |
846 | public onMagentaInput(value: { v: number; rg: number }): void {
847 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
848 |
849 | if (valid) {
850 | this.cmyk.m = value.v
851 |
852 | this.updateColorPicker(false, true, true)
853 | }
854 |
855 | this.directiveInstance.inputChanged({
856 | input: 'magenta',
857 | valid: true,
858 | value: this.cmyk.m,
859 | color: this.outputColor
860 | })
861 | }
862 |
863 | public onYellowInput(value: { v: number; rg: number }): void {
864 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
865 |
866 | if (valid) {
867 | this.cmyk.y = value.v
868 |
869 | this.updateColorPicker(false, true, true)
870 | }
871 |
872 | this.directiveInstance.inputChanged({
873 | input: 'yellow',
874 | valid: true,
875 | value: this.cmyk.y,
876 | color: this.outputColor
877 | })
878 | }
879 |
880 | public onBlackInput(value: { v: number; rg: number }): void {
881 | const valid = !isNaN(value.v) && value.v >= 0 && value.v <= value.rg
882 |
883 | if (valid) {
884 | this.cmyk.k = value.v
885 |
886 | this.updateColorPicker(false, true, true)
887 | }
888 |
889 | this.directiveInstance.inputChanged({
890 | input: 'black',
891 | valid: true,
892 | value: this.cmyk.k,
893 | color: this.outputColor
894 | })
895 | }
896 |
897 | public onAddPresetColor(event: any, value: string): void {
898 | event.stopPropagation()
899 |
900 | if (!this.cpPresetColors.filter((color) => color === value).length) {
901 | this.cpPresetColors = this.cpPresetColors.concat(value)
902 |
903 | this.directiveInstance.presetColorsChanged(this.cpPresetColors)
904 | }
905 | }
906 |
907 | public onRemovePresetColor(event: any, value: string): void {
908 | event.stopPropagation()
909 |
910 | this.cpPresetColors = this.cpPresetColors.filter((color) => color !== value)
911 |
912 | this.directiveInstance.presetColorsChanged(this.cpPresetColors)
913 | }
914 |
915 | // Private helper functions for the color picker dialog status
916 |
917 | private openColorPicker(): void {
918 | if (!this.show) {
919 | this.show = true
920 | this.hidden = true
921 |
922 | setTimeout(() => {
923 | this.hidden = false
924 |
925 | this.setDialogPosition()
926 |
927 | this.cdRef.detectChanges()
928 | }, 0)
929 |
930 | this.directiveInstance.stateChanged(true)
931 |
932 | // The change detection should be run on `mousedown` event only when the condition
933 | // is met within the `onMouseDown` method.
934 | this.ngZone.runOutsideAngular(() => {
935 | // There's no sense to add both event listeners on touch devices since the `touchstart`
936 | // event is handled earlier than `mousedown`, so we'll get 2 change detections and the
937 | // second one will be unnecessary.
938 | if (SUPPORTS_TOUCH) {
939 | this.document.addEventListener('touchstart', this.listenerMouseDown)
940 | } else {
941 | this.document.addEventListener('mousedown', this.listenerMouseDown)
942 | }
943 | })
944 |
945 | this.window.addEventListener('resize', this.listenerResize)
946 | }
947 | }
948 |
949 | private closeColorPicker(): void {
950 | if (this.show) {
951 | this.show = false
952 |
953 | this.directiveInstance.stateChanged(false)
954 |
955 | if (SUPPORTS_TOUCH) {
956 | this.document.removeEventListener('touchstart', this.listenerMouseDown)
957 | } else {
958 | this.document.removeEventListener('mousedown', this.listenerMouseDown)
959 | }
960 |
961 | this.window.removeEventListener('resize', this.listenerResize)
962 |
963 | if (!this.cdRef['destroyed']) {
964 | this.cdRef.detectChanges()
965 | }
966 | }
967 | }
968 |
969 | private updateColorPicker(
970 | emit: boolean = true,
971 | update: boolean = true,
972 | cmykInput: boolean = false
973 | ): void {
974 | if (this.sliderDimMax) {
975 | if (this.cpColorMode === 2) {
976 | this.hsva.s = 0
977 | }
978 |
979 | let hue: Rgba, hsla: Hsla, rgba: Rgba
980 |
981 | const lastOutput = this.outputColor
982 |
983 | hsla = this.service.hsva2hsla(this.hsva)
984 |
985 | if (!this.cpCmykEnabled) {
986 | rgba = this.service.denormalizeRGBA(this.service.hsvaToRgba(this.hsva))
987 | } else {
988 | if (!cmykInput) {
989 | rgba = this.service.hsvaToRgba(this.hsva)
990 |
991 | this.cmyk = this.service.denormalizeCMYK(
992 | this.service.rgbaToCmyk(rgba)
993 | )
994 | } else {
995 | rgba = this.service.cmykToRgb(this.service.normalizeCMYK(this.cmyk))
996 |
997 | this.hsva = this.service.rgbaToHsva(rgba)
998 | }
999 |
1000 | rgba = this.service.denormalizeRGBA(rgba)
1001 |
1002 | this.sliderH = this.hsva.h
1003 | }
1004 |
1005 | hue = this.service.denormalizeRGBA(
1006 | this.service.hsvaToRgba(new Hsva(this.sliderH || this.hsva.h, 1, 1, 1))
1007 | )
1008 |
1009 | if (update) {
1010 | this.hslaText = new Hsla(
1011 | Math.round(hsla.h * 360),
1012 | Math.round(hsla.s * 100),
1013 | Math.round(hsla.l * 100),
1014 | Math.round(hsla.a * 100) / 100
1015 | )
1016 |
1017 | this.rgbaText = new Rgba(
1018 | rgba.r,
1019 | rgba.g,
1020 | rgba.b,
1021 | Math.round(rgba.a * 100) / 100
1022 | )
1023 |
1024 | if (this.cpCmykEnabled) {
1025 | this.cmykText = new Cmyk(
1026 | this.cmyk.c,
1027 | this.cmyk.m,
1028 | this.cmyk.y,
1029 | this.cmyk.k,
1030 | Math.round(this.cmyk.a * 100) / 100
1031 | )
1032 | }
1033 |
1034 | const allowHex8 = this.cpAlphaChannel === 'always'
1035 |
1036 | this.hexText = this.service.rgbaToHex(rgba, allowHex8)
1037 | this.hexAlpha = this.rgbaText.a
1038 | }
1039 |
1040 | if (this.cpOutputFormat === 'auto') {
1041 | if (
1042 | this.format !== ColorFormats.RGBA &&
1043 | this.format !== ColorFormats.CMYK &&
1044 | this.format !== ColorFormats.HSLA
1045 | ) {
1046 | if (this.hsva.a < 1) {
1047 | this.format = this.hsva.a < 1 ? ColorFormats.RGBA : ColorFormats.HEX
1048 | }
1049 | }
1050 | }
1051 |
1052 | this.hueSliderColor = 'rgb(' + hue.r + ',' + hue.g + ',' + hue.b + ')'
1053 | this.alphaSliderColor =
1054 | 'rgb(' + rgba.r + ',' + rgba.g + ',' + rgba.b + ')'
1055 |
1056 | this.outputColor = this.service.outputFormat(
1057 | this.hsva,
1058 | this.cpOutputFormat,
1059 | this.cpAlphaChannel
1060 | )
1061 | this.selectedColor = this.service.outputFormat(this.hsva, 'rgba', null)
1062 |
1063 | if (this.format !== ColorFormats.CMYK) {
1064 | this.cmykColor = ''
1065 | } else {
1066 | if (
1067 | this.cpAlphaChannel === 'always' ||
1068 | this.cpAlphaChannel === 'enabled' ||
1069 | this.cpAlphaChannel === 'forced'
1070 | ) {
1071 | const alpha = Math.round(this.cmyk.a * 100) / 100
1072 |
1073 | this.cmykColor = `cmyka(${this.cmyk.c},${this.cmyk.m},${this.cmyk.y},${this.cmyk.k},${alpha})`
1074 | } else {
1075 | this.cmykColor = `cmyk(${this.cmyk.c},${this.cmyk.m},${this.cmyk.y},${this.cmyk.k})`
1076 | }
1077 | }
1078 |
1079 | this.slider = new SliderPosition(
1080 | (this.sliderH || this.hsva.h) * this.sliderDimMax.h - 8,
1081 | this.hsva.s * this.sliderDimMax.s - 8,
1082 | (1 - this.hsva.v) * this.sliderDimMax.v - 8,
1083 | this.hsva.a * this.sliderDimMax.a - 8
1084 | )
1085 |
1086 | if (emit && lastOutput !== this.outputColor) {
1087 | if (this.cpCmykEnabled) {
1088 | this.directiveInstance.cmykChanged(this.cmykColor)
1089 | }
1090 |
1091 | this.directiveInstance.colorChanged(this.outputColor)
1092 | }
1093 | }
1094 | }
1095 |
1096 | // Private helper functions for the color picker dialog positioning
1097 |
1098 | private setDialogPosition(): void {
1099 | if (this.cpDialogDisplay === 'inline') {
1100 | this.position = 'relative'
1101 | } else {
1102 | let position = 'static',
1103 | transform = '',
1104 | style
1105 |
1106 | let parentNode: any = null,
1107 | transformNode: any = null
1108 |
1109 | let node = this.directiveElementRef.nativeElement.parentNode
1110 |
1111 | const dialogHeight = this.dialogElement.nativeElement.offsetHeight
1112 |
1113 | while (node !== null && node.tagName !== 'HTML') {
1114 | style = this.window.getComputedStyle(node)
1115 | position = style.getPropertyValue('position')
1116 | transform = style.getPropertyValue('transform')
1117 |
1118 | if (position !== 'static' && parentNode === null) {
1119 | parentNode = node
1120 | }
1121 |
1122 | if (transform && transform !== 'none' && transformNode === null) {
1123 | transformNode = node
1124 | }
1125 |
1126 | if (position === 'fixed') {
1127 | parentNode = transformNode
1128 |
1129 | break
1130 | }
1131 |
1132 | node = node.parentNode
1133 | }
1134 |
1135 | const boxDirective = this.createDialogBox(
1136 | this.directiveElementRef.nativeElement,
1137 | position !== 'fixed'
1138 | )
1139 |
1140 | if (
1141 | this.useRootViewContainer ||
1142 | (position === 'fixed' &&
1143 | (!parentNode || parentNode instanceof HTMLUnknownElement))
1144 | ) {
1145 | this.top = boxDirective.top
1146 | this.left = boxDirective.left
1147 | } else {
1148 | if (parentNode === null) {
1149 | parentNode = node
1150 | }
1151 |
1152 | const boxParent = this.createDialogBox(parentNode, position !== 'fixed')
1153 |
1154 | this.top = boxDirective.top - boxParent.top
1155 | this.left = boxDirective.left - boxParent.left
1156 | }
1157 |
1158 | if (position === 'fixed') {
1159 | this.position = 'fixed'
1160 | }
1161 |
1162 | let usePosition = this.cpPosition
1163 |
1164 | const dialogBounds =
1165 | this.dialogElement.nativeElement.getBoundingClientRect()
1166 | if (this.cpPosition === 'auto') {
1167 | const triggerBounds =
1168 | this.cpTriggerElement.nativeElement.getBoundingClientRect()
1169 | usePosition = calculateAutoPositioning(
1170 | dialogBounds,
1171 | triggerBounds,
1172 | this.window
1173 | )
1174 | }
1175 |
1176 | this.arrowTop = usePosition === 'top' ? dialogHeight - 1 : undefined
1177 | this.cpArrowPosition = undefined
1178 |
1179 | switch (usePosition) {
1180 | case 'top':
1181 | this.top -= dialogHeight + this.dialogArrowSize
1182 | this.left +=
1183 | (this.cpPositionOffset / 100) * boxDirective.width -
1184 | this.dialogArrowOffset
1185 | break
1186 | case 'bottom':
1187 | this.top += boxDirective.height + this.dialogArrowSize
1188 | this.left +=
1189 | (this.cpPositionOffset / 100) * boxDirective.width -
1190 | this.dialogArrowOffset
1191 | break
1192 | case 'top-left':
1193 | case 'left-top':
1194 | this.top -=
1195 | dialogHeight -
1196 | boxDirective.height +
1197 | (boxDirective.height * this.cpPositionOffset) / 100
1198 | this.left -=
1199 | this.cpWidth + this.dialogArrowSize - 2 - this.dialogArrowOffset
1200 | break
1201 | case 'top-right':
1202 | case 'right-top':
1203 | this.top -=
1204 | dialogHeight -
1205 | boxDirective.height +
1206 | (boxDirective.height * this.cpPositionOffset) / 100
1207 | this.left +=
1208 | boxDirective.width +
1209 | this.dialogArrowSize -
1210 | 2 -
1211 | this.dialogArrowOffset
1212 | break
1213 | case 'left':
1214 | case 'bottom-left':
1215 | case 'left-bottom':
1216 | this.top +=
1217 | (boxDirective.height * this.cpPositionOffset) / 100 -
1218 | this.dialogArrowOffset
1219 | this.left -= this.cpWidth + this.dialogArrowSize - 2
1220 | break
1221 | case 'right':
1222 | case 'bottom-right':
1223 | case 'right-bottom':
1224 | default:
1225 | this.top +=
1226 | (boxDirective.height * this.cpPositionOffset) / 100 -
1227 | this.dialogArrowOffset
1228 | this.left += boxDirective.width + this.dialogArrowSize - 2
1229 | break
1230 | }
1231 |
1232 | const windowInnerHeight = this.window.innerHeight
1233 | const windowInnerWidth = this.window.innerWidth
1234 | const elRefClientRect = this.elRef.nativeElement.getBoundingClientRect()
1235 | const bottom = this.top + dialogBounds.height
1236 | if (bottom > windowInnerHeight) {
1237 | this.top = windowInnerHeight - dialogBounds.height
1238 | this.cpArrowPosition = elRefClientRect.x / 2 - 20
1239 | }
1240 | const right = this.left + dialogBounds.width
1241 | if (right > windowInnerWidth) {
1242 | this.left = windowInnerWidth - dialogBounds.width
1243 | this.cpArrowPosition = elRefClientRect.x / 2 - 20
1244 | }
1245 |
1246 | this.cpUsePosition = usePosition
1247 | }
1248 | }
1249 |
1250 | // Private helper functions for the color picker dialog positioning and opening
1251 |
1252 | private isDescendant(parent: any, child: any): boolean {
1253 | let node: any = child.parentNode
1254 |
1255 | while (node !== null) {
1256 | if (node === parent) {
1257 | return true
1258 | }
1259 |
1260 | node = node.parentNode
1261 | }
1262 |
1263 | return false
1264 | }
1265 |
1266 | private createDialogBox(element: any, offset: boolean): any {
1267 | const { top, left } = element.getBoundingClientRect()
1268 | return {
1269 | top: top + (offset ? this.window.pageYOffset : 0),
1270 | left: left + (offset ? this.window.pageXOffset : 0),
1271 | width: element.offsetWidth,
1272 | height: element.offsetHeight
1273 | }
1274 | }
1275 | }
1276 |
--------------------------------------------------------------------------------