UNPKG

ngx-input-color

Version:

Angular color input component and color picker (with HSL, HSV, RGB, CMYK, HEX, alpha, eye-dropper, etc)

181 lines 26.9 kB
import { CommonModule } from '@angular/common'; import { Component, EventEmitter, HostListener, Input, Output, ViewChild, forwardRef } from '@angular/core'; import { getOffsetPosition } from '../utils/get-offset-position'; import { NG_VALUE_ACCESSOR, FormControl, } from '@angular/forms'; import * as i0 from "@angular/core"; export class SaturationComponent { constructor() { this.color = 'red'; this.step = 1; this.min = { x: 0, y: 0 }; this.max = { x: 100, y: 100 }; this.change = new EventEmitter(); this.isDragging = false; this.x = 0; this.y = 0; this.myControl = new FormControl(null); this.isDisabled = false; this._onChange = (value) => { }; this._onTouched = () => { }; this._validatorOnChange = () => { }; } updateRects() { this.saturationRect = this.saturation.nativeElement.getBoundingClientRect(); this.thumbRect = this.thumb.nativeElement.getBoundingClientRect(); } writeValue(val) { if (!val) val = { x: 0, y: 0 }; let value = val; this.myControl.setValue(value, { emitEvent: false }); this.updateRects(); const saturationRec = this.saturationRect; const thumbRec = this.thumbRect; this.x = ((value.x - this.min.x) * (saturationRec.width - thumbRec.width / 2)) / (this.max.x - this.min.x); this.y = ((value.y - this.min.y) * (saturationRec.height - thumbRec.height / 2)) / (this.max.y - this.min.y); if (val !== value) { this.valueChanged(value); } } validate(control) { return this.myControl.errors; } registerOnValidatorChange(fn) { this._validatorOnChange = fn; } registerOnChange(fn) { this._onChange = fn; } registerOnTouched(fn) { this._onTouched = fn; } setDisabledState(disabled) { this.isDisabled = disabled; if (disabled) this.myControl.disable(); else this.myControl.enable(); } dragStart(ev) { ev.stopPropagation(); ev.preventDefault(); this.isDragging = true; this.updateRects(); this.updatePosition(ev); } onResize() { this.writeValue(this.myControl.value); } onDrag(ev) { if (!this.isDragging) return; this.updatePosition(ev); } updatePosition(ev) { if (!this.isDragging) return; if (!this.saturationRect || !this.thumbRect) this.updateRects(); let position = getOffsetPosition(ev, this.saturation.nativeElement); let thumbRec = this.thumbRect; let saturationRec = this.saturationRect; if (position.x < 0) { this.x = 0; } else if (position.x > saturationRec.width - (thumbRec.width / 2 - 3)) { this.x = saturationRec.width - (thumbRec.width / 2 - 3); } else { this.x = position.x; } // this.x = this.x - thumbRec.width / 2; if (position.y < 0) { this.y = 0; } else if (position.y > saturationRec.height - (thumbRec.height / 2 - 3)) { this.y = saturationRec.height - (thumbRec.height / 2 - 3); } else { this.y = position.y; } // this.y = this.y - thumbRec.height / 2; this.setValueByPosition(thumbRec, saturationRec); } onDragEnd(ev) { this.isDragging = false; } setValueByPosition(thumbRec, saturationRec) { const percentageX = this.x / (saturationRec.width - thumbRec.width); let newValueX = this.min.x + percentageX * (this.max.x - this.min.x); newValueX = Math.round(newValueX / this.step) * this.step; let valueX = Math.min(Math.max(newValueX, this.min.x), this.max.x); //----------------------------- const percentageY = this.y / (saturationRec.height - thumbRec.height); let newValueY = this.min.y + percentageY * (this.max.y - this.min.y); newValueY = Math.round(newValueY / this.step) * this.step; let valueY = Math.min(Math.max(newValueY, this.min.y), this.max.y); const newValue = { x: valueX, y: valueY }; if (!this.myControl.value || this.myControl.value.x !== valueX || this.myControl.value.y !== valueY) { this.valueChanged(newValue); } } valueChanged(value) { this.myControl.setValue(value, { emitEvent: false }); this._onChange(value); this.change.emit(value); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SaturationComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: SaturationComponent, isStandalone: true, selector: "saturation", inputs: { width: "width", height: "height", color: "color", step: "step", min: "min", max: "max" }, outputs: { change: "change" }, host: { listeners: { "window:resize": "onResize($event)", "document:mousemove": "onDrag($event)", "document:touchmove": "onDrag($event)", "document:mouseup": "onDragEnd($event)", "document:touchend": "onDragEnd($event)" } }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SaturationComponent), multi: true, }, ], viewQueries: [{ propertyName: "saturation", first: true, predicate: ["saturation"], descendants: true, static: true }, { propertyName: "thumb", first: true, predicate: ["thumb"], descendants: true, static: true }], ngImport: i0, template: "<div\r\n class=\"saturation-container\"\r\n [style.width.px]=\"width\"\r\n [style.height.px]=\"height\"\r\n (mousedown)=\"dragStart($event)\"\r\n (touchstart)=\"dragStart($event)\">\r\n <div class=\"saturation\" #saturation>\r\n <div class=\"s-bg\" [style.background]=\"color\"></div>\r\n <div class=\"s-white\"></div>\r\n <div class=\"s-black\"></div>\r\n </div>\r\n <div class=\"thumb\" #thumb [style.left.px]=\"x\" [style.top.px]=\"y\"></div>\r\n</div>\r\n", styles: [".saturation-container{position:relative;width:100%;height:200px}.saturation-container .thumb{border:3px #000000 solid;box-shadow:inset 0 0 0 2px #fff;width:12px;height:12px;display:block;border-radius:10px;position:absolute;pointer-events:none;cursor:pointer}.saturation{position:absolute;inset:6px;border-radius:4px;overflow:hidden}.saturation .s-white,.saturation .s-black,.saturation .s-bg{position:absolute;inset:0}.saturation .s-white{background:linear-gradient(to right,#fff,#fff0)}.saturation .s-black{background:linear-gradient(to bottom,#0000,#000)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SaturationComponent, decorators: [{ type: Component, args: [{ selector: 'saturation', standalone: true, imports: [CommonModule], providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SaturationComponent), multi: true, }, ], template: "<div\r\n class=\"saturation-container\"\r\n [style.width.px]=\"width\"\r\n [style.height.px]=\"height\"\r\n (mousedown)=\"dragStart($event)\"\r\n (touchstart)=\"dragStart($event)\">\r\n <div class=\"saturation\" #saturation>\r\n <div class=\"s-bg\" [style.background]=\"color\"></div>\r\n <div class=\"s-white\"></div>\r\n <div class=\"s-black\"></div>\r\n </div>\r\n <div class=\"thumb\" #thumb [style.left.px]=\"x\" [style.top.px]=\"y\"></div>\r\n</div>\r\n", styles: [".saturation-container{position:relative;width:100%;height:200px}.saturation-container .thumb{border:3px #000000 solid;box-shadow:inset 0 0 0 2px #fff;width:12px;height:12px;display:block;border-radius:10px;position:absolute;pointer-events:none;cursor:pointer}.saturation{position:absolute;inset:6px;border-radius:4px;overflow:hidden}.saturation .s-white,.saturation .s-black,.saturation .s-bg{position:absolute;inset:0}.saturation .s-white{background:linear-gradient(to right,#fff,#fff0)}.saturation .s-black{background:linear-gradient(to bottom,#0000,#000)}\n"] }] }], ctorParameters: () => [], propDecorators: { width: [{ type: Input }], height: [{ type: Input }], color: [{ type: Input }], step: [{ type: Input }], min: [{ type: Input }], max: [{ type: Input }], change: [{ type: Output }], saturation: [{ type: ViewChild, args: ['saturation', { static: true }] }], thumb: [{ type: ViewChild, args: ['thumb', { static: true }] }], onResize: [{ type: HostListener, args: ['window:resize', ['$event']] }], onDrag: [{ type: HostListener, args: ['document:mousemove', ['$event']] }, { type: HostListener, args: ['document:touchmove', ['$event']] }], onDragEnd: [{ type: HostListener, args: ['document:mouseup', ['$event']] }, { type: HostListener, args: ['document:touchend', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saturation.component.js","sourceRoot":"","sources":["../../../../projects/ngx-input-color/src/saturation/saturation.component.ts","../../../../projects/ngx-input-color/src/saturation/saturation.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, ElementRef, EventEmitter, HostListener, Input, Output, ViewChild, forwardRef } from '@angular/core';\r\nimport { getOffsetPosition } from '../utils/get-offset-position';\r\nimport {\r\n  NG_VALUE_ACCESSOR,\r\n  ControlValueAccessor,\r\n  FormControl,\r\n  AbstractControl,\r\n  ValidationErrors,\r\n} from '@angular/forms';\r\nimport { IPosition } from '../models/IPosition';\r\n\r\n@Component({\r\n  selector: 'saturation',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  templateUrl: './saturation.component.html',\r\n  styleUrl: './saturation.component.scss',\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => SaturationComponent),\r\n      multi: true,\r\n    },\r\n  ],\r\n})\r\nexport class SaturationComponent implements ControlValueAccessor {\r\n  @Input() width?: number;\r\n  @Input() height?: number;\r\n  @Input() color = 'red';\r\n  @Input() step = 1;\r\n  @Input() min: IPosition = { x: 0, y: 0 };\r\n  @Input() max = { x: 100, y: 100 };\r\n  @Output() change = new EventEmitter<IPosition>();\r\n\r\n  isDragging = false;\r\n  @ViewChild('saturation', { static: true }) saturation!: ElementRef<HTMLDivElement>;\r\n  @ViewChild('thumb', { static: true }) thumb!: ElementRef<HTMLDivElement>;\r\n  x = 0;\r\n  y = 0;\r\n  myControl = new FormControl<IPosition | null>(null);\r\n  isDisabled = false;\r\n  _onChange = (value: any) => {};\r\n  _onTouched = () => {};\r\n  _validatorOnChange = () => {};\r\n\r\n  private saturationRect?: DOMRect;\r\n  private thumbRect?: DOMRect;\r\n\r\n  constructor() {}\r\n\r\n  private updateRects() {\r\n    this.saturationRect = this.saturation.nativeElement.getBoundingClientRect();\r\n    this.thumbRect = this.thumb.nativeElement.getBoundingClientRect();\r\n  }\r\n\r\n  writeValue(val?: IPosition | null): void {\r\n    if (!val) val = { x: 0, y: 0 };\r\n    let value: IPosition = val;\r\n    this.myControl.setValue(value, { emitEvent: false });\r\n    this.updateRects();\r\n    const saturationRec = this.saturationRect!;\r\n    const thumbRec = this.thumbRect!;\r\n    this.x = ((value.x - this.min.x) * (saturationRec.width - thumbRec.width / 2)) / (this.max.x - this.min.x);\r\n    this.y = ((value.y - this.min.y) * (saturationRec.height - thumbRec.height / 2)) / (this.max.y - this.min.y);\r\n    if (val !== value) {\r\n      this.valueChanged(value);\r\n    }\r\n  }\r\n  validate(control: AbstractControl): ValidationErrors | null {\r\n    return this.myControl.errors;\r\n  }\r\n  registerOnValidatorChange?(fn: () => void): void {\r\n    this._validatorOnChange = fn;\r\n  }\r\n\r\n  registerOnChange(fn: any): void {\r\n    this._onChange = fn;\r\n  }\r\n  registerOnTouched(fn: any): void {\r\n    this._onTouched = fn;\r\n  }\r\n  setDisabledState?(disabled: boolean): void {\r\n    this.isDisabled = disabled;\r\n    if (disabled) this.myControl.disable();\r\n    else this.myControl.enable();\r\n  }\r\n\r\n  dragStart(ev: MouseEvent | TouchEvent) {\r\n    ev.stopPropagation();\r\n    ev.preventDefault();\r\n    this.isDragging = true;\r\n    this.updateRects();\r\n    this.updatePosition(ev);\r\n  }\r\n\r\n  @HostListener('window:resize', ['$event'])\r\n  onResize() {\r\n    this.writeValue(this.myControl.value);\r\n  }\r\n\r\n  @HostListener('document:mousemove', ['$event'])\r\n  @HostListener('document:touchmove', ['$event'])\r\n  onDrag(ev: MouseEvent | TouchEvent) {\r\n    if (!this.isDragging) return;\r\n    this.updatePosition(ev);\r\n  }\r\n\r\n  private updatePosition(ev: MouseEvent | TouchEvent) {\r\n    if (!this.isDragging) return;\r\n    if (!this.saturationRect || !this.thumbRect) this.updateRects();\r\n    let position = getOffsetPosition(ev, this.saturation.nativeElement);\r\n    let thumbRec = this.thumbRect!;\r\n    let saturationRec = this.saturationRect!;\r\n    if (position.x < 0) {\r\n      this.x = 0;\r\n    } else if (position.x > saturationRec.width - (thumbRec.width / 2 - 3)) {\r\n      this.x = saturationRec.width - (thumbRec.width / 2 - 3);\r\n    } else {\r\n      this.x = position.x;\r\n    }\r\n    // this.x = this.x - thumbRec.width / 2;\r\n\r\n    if (position.y < 0) {\r\n      this.y = 0;\r\n    } else if (position.y > saturationRec.height - (thumbRec.height / 2 - 3)) {\r\n      this.y = saturationRec.height - (thumbRec.height / 2 - 3);\r\n    } else {\r\n      this.y = position.y;\r\n    }\r\n    //  this.y = this.y - thumbRec.height / 2;\r\n    this.setValueByPosition(thumbRec, saturationRec);\r\n  }\r\n\r\n  @HostListener('document:mouseup', ['$event'])\r\n  @HostListener('document:touchend', ['$event'])\r\n  onDragEnd(ev: MouseEvent | TouchEvent) {\r\n    this.isDragging = false;\r\n  }\r\n\r\n  setValueByPosition(thumbRec: DOMRect, saturationRec: DOMRect) {\r\n    const percentageX = this.x / (saturationRec.width - thumbRec.width);\r\n    let newValueX = this.min.x + percentageX * (this.max.x - this.min.x);\r\n    newValueX = Math.round(newValueX / this.step) * this.step;\r\n    let valueX = Math.min(Math.max(newValueX, this.min.x), this.max.x);\r\n    //-----------------------------\r\n    const percentageY = this.y / (saturationRec.height - thumbRec.height);\r\n    let newValueY = this.min.y + percentageY * (this.max.y - this.min.y);\r\n    newValueY = Math.round(newValueY / this.step) * this.step;\r\n    let valueY = Math.min(Math.max(newValueY, this.min.y), this.max.y);\r\n    const newValue = { x: valueX, y: valueY };\r\n    if (!this.myControl.value || this.myControl.value.x !== valueX || this.myControl.value.y !== valueY) {\r\n      this.valueChanged(newValue);\r\n    }\r\n  }\r\n\r\n  valueChanged(value: IPosition) {\r\n    this.myControl.setValue(value, { emitEvent: false });\r\n    this._onChange(value);\r\n    this.change.emit(value);\r\n  }\r\n}\r\n","<div\r\n  class=\"saturation-container\"\r\n  [style.width.px]=\"width\"\r\n  [style.height.px]=\"height\"\r\n  (mousedown)=\"dragStart($event)\"\r\n  (touchstart)=\"dragStart($event)\">\r\n  <div class=\"saturation\" #saturation>\r\n    <div class=\"s-bg\" [style.background]=\"color\"></div>\r\n    <div class=\"s-white\"></div>\r\n    <div class=\"s-black\"></div>\r\n  </div>\r\n  <div class=\"thumb\" #thumb [style.left.px]=\"x\" [style.top.px]=\"y\"></div>\r\n</div>\r\n"]}