UNPKG

ngx-input-color

Version:

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

182 lines 26.4 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output, ViewChild, forwardRef, } from '@angular/core'; import { getOffsetPosition } from '../utils/get-offset-position'; import { FormControl, NG_VALIDATORS, NG_VALUE_ACCESSOR, Validators, } from '@angular/forms'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class SliderComponent { constructor() { this.step = 1; this.min = 0; this.max = 100; this.isBgTransparent = false; this.change = new EventEmitter(); this.isDragging = false; this.x = 0; this.myControl = new FormControl(null); this.isDisabled = false; this._onChange = (value) => { }; this._onTouched = () => { }; this._validatorOnChange = () => { }; } ngOnInit() { this.myControl.setValidators([Validators.min(this.min), Validators.max(this.max)]); } updateRects() { this.sliderRect = this.slider.nativeElement.getBoundingClientRect(); this.thumbRect = this.thumb.nativeElement.getBoundingClientRect(); } writeValue(val) { let value = 0; if (!val) value = 0; else if (+val < +this.min) value = +this.min; else if (+val > +this.max) value = +this.max; else value = +val; this.myControl.setValue(value, { emitEvent: false }); this.updateRects(); const sliderRec = this.sliderRect; const thumbRec = this.thumbRect; this.x = ((value - this.min) * (sliderRec.width - thumbRec.width)) / (this.max - this.min); 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); } onDrag(ev) { if (!this.isDragging) return; this.updatePosition(ev); } onResize() { this.writeValue(this.myControl.value); } updatePosition(ev) { if (!this.isDragging) return; if (!this.sliderRect || !this.thumbRect) this.updateRects(); let position = getOffsetPosition(ev, this.slider.nativeElement); let thumbRec = this.thumbRect; position.x -= thumbRec.width / 2; let sliderRec = this.sliderRect; if (position.x < 0) { this.x = 0; } else if (position.x > sliderRec.width - thumbRec.width) { this.x = sliderRec.width - thumbRec.width; } else { this.x = position.x; } this.setValueByPosition(thumbRec, sliderRec); } setValueByPosition(thumbRec, sliderRec) { const percentage = this.x / (sliderRec.width - thumbRec.width); let newValue = this.min + percentage * (this.max - this.min); const stepDecimalPlaces = (this.step.toString().split('.')[1] || '').length; newValue = parseFloat((Math.round(newValue / this.step) * this.step).toFixed(stepDecimalPlaces)); let value = Math.min(Math.max(newValue, this.min), this.max); if (this.myControl.value !== value) { this.valueChanged(value); } } onDragEnd(ev) { this.isDragging = false; } 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: SliderComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: SliderComponent, isStandalone: true, selector: "slider", inputs: { step: "step", min: "min", max: "max", background: "background", isBgTransparent: "isBgTransparent" }, outputs: { change: "change" }, host: { listeners: { "document:mousemove": "onDrag($event)", "document:touchmove": "onDrag($event)", "window:resize": "onResize($event)", "document:mouseup": "onDragEnd($event)", "document:touchend": "onDragEnd($event)" } }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SliderComponent), multi: true, }, { provide: NG_VALIDATORS, useExisting: forwardRef(() => SliderComponent), multi: true, }, ], viewQueries: [{ propertyName: "slider", first: true, predicate: ["slider"], descendants: true, static: true }, { propertyName: "thumb", first: true, predicate: ["thumb"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"slider-container\">\r\n <ng-content></ng-content>\r\n <div\r\n #slider\r\n class=\"slider\"\r\n (mousedown)=\"dragStart($event)\"\r\n (touchstart)=\"dragStart($event)\"\r\n [ngStyle]=\"{ '--ngx-slider-bg': background }\"\r\n [class.bg-transparent]=\"isBgTransparent\">\r\n <div class=\"thumb\" #thumb [style.left.px]=\"x\" [title]=\"myControl.value\"></div>\r\n </div>\r\n</div>\r\n", styles: [".slider-container{max-width:100%;padding:1px 0}.slider-container .slider{position:relative;box-shadow:inset #00000013 0 0 0 1px;border-radius:10px;height:12px;width:100%;background:var(--ngx-slider-bg, rgb(140, 51, 250));margin:10px 0}.slider-container .slider.bg-transparent{background:transparent}.slider-container .slider.bg-transparent:before,.slider-container .slider.bg-transparent:after{position:absolute;inset:1px;border-radius:9px}.slider-container .slider.bg-transparent:before{content:\" \";background-image:linear-gradient(45deg,#ccc 25%,transparent 25%),linear-gradient(-45deg,#ccc 25%,transparent 25%),linear-gradient(45deg,transparent 75%,#ccc 75%),linear-gradient(-45deg,transparent 75%,#ccc 75%);background-size:16px 16px;background-position:0 0,0 8px,8px -8px,-8px 0px}.slider-container .slider.bg-transparent:after{content:\" \";background:var(--ngx-slider-bg)}.slider-container .thumb{box-shadow:#00000026 0 0 0 1px,#0000000d 0 10px 10px -5px,inset #fff 0 0 0 6px;background:var(--ngx-slider-bg, rgb(140, 51, 250));height:var(--ngx-thumb-size, 30px);width:var(--ngx-thumb-size, 30px);display:block;border-radius:100%;top:calc(6px - var(--ngx-thumb-size, 30px) / 2);position:absolute;cursor:grab;z-index:100}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SliderComponent, decorators: [{ type: Component, args: [{ selector: 'slider', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SliderComponent), multi: true, }, { provide: NG_VALIDATORS, useExisting: forwardRef(() => SliderComponent), multi: true, }, ], template: "<div class=\"slider-container\">\r\n <ng-content></ng-content>\r\n <div\r\n #slider\r\n class=\"slider\"\r\n (mousedown)=\"dragStart($event)\"\r\n (touchstart)=\"dragStart($event)\"\r\n [ngStyle]=\"{ '--ngx-slider-bg': background }\"\r\n [class.bg-transparent]=\"isBgTransparent\">\r\n <div class=\"thumb\" #thumb [style.left.px]=\"x\" [title]=\"myControl.value\"></div>\r\n </div>\r\n</div>\r\n", styles: [".slider-container{max-width:100%;padding:1px 0}.slider-container .slider{position:relative;box-shadow:inset #00000013 0 0 0 1px;border-radius:10px;height:12px;width:100%;background:var(--ngx-slider-bg, rgb(140, 51, 250));margin:10px 0}.slider-container .slider.bg-transparent{background:transparent}.slider-container .slider.bg-transparent:before,.slider-container .slider.bg-transparent:after{position:absolute;inset:1px;border-radius:9px}.slider-container .slider.bg-transparent:before{content:\" \";background-image:linear-gradient(45deg,#ccc 25%,transparent 25%),linear-gradient(-45deg,#ccc 25%,transparent 25%),linear-gradient(45deg,transparent 75%,#ccc 75%),linear-gradient(-45deg,transparent 75%,#ccc 75%);background-size:16px 16px;background-position:0 0,0 8px,8px -8px,-8px 0px}.slider-container .slider.bg-transparent:after{content:\" \";background:var(--ngx-slider-bg)}.slider-container .thumb{box-shadow:#00000026 0 0 0 1px,#0000000d 0 10px 10px -5px,inset #fff 0 0 0 6px;background:var(--ngx-slider-bg, rgb(140, 51, 250));height:var(--ngx-thumb-size, 30px);width:var(--ngx-thumb-size, 30px);display:block;border-radius:100%;top:calc(6px - var(--ngx-thumb-size, 30px) / 2);position:absolute;cursor:grab;z-index:100}\n"] }] }], ctorParameters: () => [], propDecorators: { step: [{ type: Input }], min: [{ type: Input }], max: [{ type: Input }], background: [{ type: Input }], isBgTransparent: [{ type: Input }], change: [{ type: Output }], slider: [{ type: ViewChild, args: ['slider', { static: true }] }], thumb: [{ type: ViewChild, args: ['thumb', { static: true }] }], onDrag: [{ type: HostListener, args: ['document:mousemove', ['$event']] }, { type: HostListener, args: ['document:touchmove', ['$event']] }], onResize: [{ type: HostListener, args: ['window:resize', ['$event']] }], onDragEnd: [{ type: HostListener, args: ['document:mouseup', ['$event']] }, { type: HostListener, args: ['document:touchend', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slider.component.js","sourceRoot":"","sources":["../../../../projects/ngx-input-color/src/slider/slider.component.ts","../../../../projects/ngx-input-color/src/slider/slider.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport {\r\n  ChangeDetectionStrategy,\r\n  Component,\r\n  ElementRef,\r\n  EventEmitter,\r\n  HostListener,\r\n  Input,\r\n  Output,\r\n  ViewChild,\r\n  forwardRef,\r\n  type OnInit,\r\n} from '@angular/core';\r\nimport { getOffsetPosition } from '../utils/get-offset-position';\r\nimport {\r\n  AbstractControl,\r\n  ControlValueAccessor,\r\n  FormControl,\r\n  NG_VALIDATORS,\r\n  NG_VALUE_ACCESSOR,\r\n  ValidationErrors,\r\n  Validator,\r\n  Validators,\r\n} from '@angular/forms';\r\n\r\n@Component({\r\n  selector: 'slider',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  templateUrl: './slider.component.html',\r\n  styleUrls: ['./slider.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => SliderComponent),\r\n      multi: true,\r\n    },\r\n    {\r\n      provide: NG_VALIDATORS,\r\n      useExisting: forwardRef(() => SliderComponent),\r\n      multi: true,\r\n    },\r\n  ],\r\n})\r\nexport class SliderComponent implements OnInit, ControlValueAccessor, Validator {\r\n  @Input() step = 1;\r\n  @Input() min = 0;\r\n  @Input() max = 100;\r\n  @Input() background?: string;\r\n  @Input() isBgTransparent = false;\r\n  @Output() change = new EventEmitter<number>();\r\n  isDragging = false;\r\n  @ViewChild('slider', { static: true }) slider!: ElementRef<HTMLDivElement>;\r\n  @ViewChild('thumb', { static: true }) thumb!: ElementRef<HTMLDivElement>;\r\n  x = 0;\r\n  myControl = new FormControl<number | null>(null);\r\n  isDisabled = false;\r\n  _onChange = (value: any) => {};\r\n  _onTouched = () => {};\r\n  _validatorOnChange = () => {};\r\n  private sliderRect?: DOMRect;\r\n  private thumbRect?: DOMRect;\r\n  constructor() {}\r\n  ngOnInit(): void {\r\n    this.myControl.setValidators([Validators.min(this.min), Validators.max(this.max)]);\r\n  }\r\n\r\n  private updateRects() {\r\n    this.sliderRect = this.slider.nativeElement.getBoundingClientRect();\r\n    this.thumbRect = this.thumb.nativeElement.getBoundingClientRect();\r\n  }\r\n\r\n  writeValue(val?: number | string | null): void {\r\n    let value = 0;\r\n    if (!val) value = 0;\r\n    else if (+val < +this.min) value = +this.min;\r\n    else if (+val > +this.max) value = +this.max;\r\n    else value = +val;\r\n    this.myControl.setValue(value, { emitEvent: false });\r\n    this.updateRects();\r\n    const sliderRec = this.sliderRect!;\r\n    const thumbRec = this.thumbRect!;\r\n    this.x = ((value - this.min) * (sliderRec.width - thumbRec.width)) / (this.max - this.min);\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('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  @HostListener('window:resize', ['$event'])\r\n  onResize() {\r\n    this.writeValue(this.myControl.value);\r\n  }\r\n\r\n  private updatePosition(ev: MouseEvent | TouchEvent) {\r\n    if (!this.isDragging) return;\r\n    if (!this.sliderRect || !this.thumbRect) this.updateRects();\r\n    let position = getOffsetPosition(ev, this.slider.nativeElement);\r\n    let thumbRec = this.thumbRect!;\r\n    position.x -= thumbRec.width / 2;\r\n    let sliderRec = this.sliderRect!;\r\n    if (position.x < 0) {\r\n      this.x = 0;\r\n    } else if (position.x > sliderRec.width - thumbRec.width) {\r\n      this.x = sliderRec.width - thumbRec.width;\r\n    } else {\r\n      this.x = position.x;\r\n    }\r\n    this.setValueByPosition(thumbRec, sliderRec);\r\n  }\r\n\r\n  setValueByPosition(thumbRec: DOMRect, sliderRec: DOMRect) {\r\n    const percentage = this.x / (sliderRec.width - thumbRec.width);\r\n    let newValue = this.min + percentage * (this.max - this.min);\r\n    const stepDecimalPlaces = (this.step.toString().split('.')[1] || '').length;\r\n    newValue = parseFloat((Math.round(newValue / this.step) * this.step).toFixed(stepDecimalPlaces));\r\n    let value = Math.min(Math.max(newValue, this.min), this.max);\r\n    if (this.myControl.value !== value) {\r\n      this.valueChanged(value);\r\n    }\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  valueChanged(value: number) {\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 class=\"slider-container\">\r\n  <ng-content></ng-content>\r\n  <div\r\n    #slider\r\n    class=\"slider\"\r\n    (mousedown)=\"dragStart($event)\"\r\n    (touchstart)=\"dragStart($event)\"\r\n    [ngStyle]=\"{ '--ngx-slider-bg': background }\"\r\n    [class.bg-transparent]=\"isBgTransparent\">\r\n    <div class=\"thumb\" #thumb [style.left.px]=\"x\" [title]=\"myControl.value\"></div>\r\n  </div>\r\n</div>\r\n"]}