UNPKG

ngx-input-color

Version:

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

245 lines 47.7 kB
import { CommonModule } from '@angular/common'; import { Component, ChangeDetectionStrategy, forwardRef, HostListener, ViewChild, Input, Output, EventEmitter, } from '@angular/core'; import { NG_VALUE_ACCESSOR, NG_VALIDATORS, FormsModule, } from '@angular/forms'; import { getOffsetPosition } from '../../utils/get-offset-position'; import { parseBoxShadowToPx, stringifyBoxShadow } from '../../utils/box-shadow'; import { NgxInputColorModule } from '../../ngx-input-color.module'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; import * as i2 from "../../directives/ngx-input-color.directive"; export class NgxBoxShadowComponent { set setTheme(val) { if (!val || val == 'auto') { this.theme = window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light'; } else { this.theme = val; } } constructor(cd) { this.cd = cd; this.theme = 'light'; /** * The maximum range of the box shadow. * @default 25 */ this.maxRange = 25; this.change = new EventEmitter(); this.isDisabled = false; this.isDragging = false; this.value = { x: 0, y: 0 }; this.x = 0; this.y = 0; this.blur = 0; this.spread = 0; this.color = 'black'; this.line = { x1: 0, y1: 0, x2: 0, y2: 0 }; this.center = { x: 0, y: 0 }; this._onChange = (value) => { }; this._onTouched = () => { }; this._onValidateChange = () => { }; } ngOnInit() { } ngAfterViewInit() { this.updateRects(); } ngOnDestroy() { } registerOnChange(fn) { this._onChange = fn; } registerOnTouched(fn) { this._onTouched = fn; } setDisabledState(disabled) { this.isDisabled = disabled; } registerOnValidatorChange(fn) { this._onValidateChange = fn; } validate(control) { return null; } writeValue(value) { this.resetPosition(); if (value) { const boxShadow = parseBoxShadowToPx(value); console.log(boxShadow); if (boxShadow) { this.value = { x: boxShadow.offsetX, y: boxShadow.offsetY }; this.blur = boxShadow.blurRadius; this.spread = boxShadow.spreadRadius; this.color = boxShadow.color; this.convertValueToPosition(boxShadow.offsetX, boxShadow.offsetY); } } } dragStart(ev) { ev.stopPropagation(); ev.preventDefault(); this.isDragging = true; this.updatePosition(ev); this.updateRects(); } updateRects() { this.padRect = this.pad.nativeElement.getBoundingClientRect(); this.thumbRect = this.thumb.nativeElement.getBoundingClientRect(); } resetPosition() { if (!this.padRect || !this.thumbRect) this.updateRects(); this.center = { x: this.padRect.width / 2, y: this.padRect.height / 2 }; this.x = this.center.x - this.thumbRect.width / 2; this.y = this.center.y - this.thumbRect.height / 2; this.line = { x1: this.center.x, y1: this.center.y, x2: this.center.x, y2: this.center.y }; this.cd.detectChanges(); } onResize() { this.writeValue(this.value); } onDrag(ev) { if (!this.isDragging) return; this.updatePosition(ev); } updatePosition(ev) { if (!this.isDragging) return; if (!this.padRect || !this.thumbRect) this.updateRects(); const position = getOffsetPosition(ev, this.pad.nativeElement); const padRec = this.padRect; const thumbRec = this.thumbRect; const minX = thumbRec.width / 2; const maxX = padRec.width - thumbRec.width / 2; const minY = thumbRec.height / 2; const maxY = padRec.height - thumbRec.height / 2; const clampedX = Math.max(minX, Math.min(position.x, maxX)); const clampedY = Math.max(minY, Math.min(position.y, maxY)); // فقط یکبار از وسط اصلاح کن this.x = clampedX - thumbRec.width / 2; this.y = clampedY - thumbRec.height / 2; this.line = { x1: this.center.x, y1: this.center.y, x2: this.x + thumbRec.width / 2, y2: this.y + thumbRec.height / 2, }; this.setValueByPosition(thumbRec, padRec); } onDragEnd(ev) { this.isDragging = false; } setValueByPosition(thumbRec, padRec) { const padCenterX = (padRec.width - thumbRec.width) / 2; const padCenterY = (padRec.height - thumbRec.height) / 2; // فاصله thumb از مرکز const dx = this.x - padCenterX; const dy = this.y - padCenterY; // مقیاس تبدیل به -50 تا +50 (یا هرچقدر بخوای) const halfRangeX = (padRec.width - thumbRec.width) / 2; const halfRangeY = (padRec.height - thumbRec.height) / 2; let valueX = (dx / halfRangeX) * this.maxRange; // -50 تا +50 let valueY = (dy / halfRangeY) * this.maxRange; // رُند کردن valueX = Math.round(valueX); valueY = Math.round(valueY); // محدود کردن به -50 تا +50 یا هر مقدار دلخواه valueX = Math.min(Math.max(valueX, -this.maxRange), this.maxRange); valueY = Math.min(Math.max(valueY, -this.maxRange), this.maxRange); const newValue = { x: valueX, y: valueY }; if (!this.value || this.value.x !== valueX || this.value.y !== valueY) { this.value = newValue; this.onChangeData(); } } onChangeValue() { this.convertValueToPosition(this.value.x, this.value.y); this.onChangeData(); } convertValueToPosition(offsetX, offsetY) { if (!this.padRect || !this.thumbRect) this.updateRects(); // موقعیت پیشنهادی بر اساس offset let proposedX = this.padRect.width / 2 + offsetX - this.thumbRect.width / 2; let proposedY = this.padRect.height / 2 + offsetY - this.thumbRect.height / 2; // محدود کردن (clamp) تا خارج از pad نره const minX = 0; const maxX = this.padRect.width - this.thumbRect.width; const minY = 0; const maxY = this.padRect.height - this.thumbRect.height; this.x = Math.min(Math.max(proposedX, minX), maxX); this.y = Math.min(Math.max(proposedY, minY), maxY); // رسم خط راهنما از مرکز به thumb this.line = { x1: this.center.x, y1: this.center.y, x2: this.x + this.thumbRect.width / 2, y2: this.y + this.thumbRect.height / 2, }; this.cd.detectChanges(); } onChangeData() { const boxShadow = stringifyBoxShadow({ inset: false, offsetX: this.value.x, offsetY: this.value.y, blurRadius: this.blur, spreadRadius: this.spread, color: this.color, }); this._onChange(boxShadow); this.change.emit(boxShadow); } stopPropagation(ev) { ev.stopPropagation(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxBoxShadowComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: NgxBoxShadowComponent, isStandalone: true, selector: "ngx-box-shadow", inputs: { setTheme: ["theme", "setTheme"], maxRange: "maxRange" }, 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(() => NgxBoxShadowComponent), multi: true }, { provide: NG_VALIDATORS, multi: true, useExisting: NgxBoxShadowComponent, }, ], viewQueries: [{ propertyName: "pad", first: true, predicate: ["pad"], descendants: true, static: true }, { propertyName: "thumb", first: true, predicate: ["thumb"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"ngx-box-shadow-container\" (click)=\"stopPropagation($event)\">\r\n <div class=\"pad-container\" #pad>\r\n <svg viewBox=\"0 0 100 100\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <line\r\n [attr.x1]=\"line.x1\"\r\n [attr.y1]=\"line.y1\"\r\n [attr.x2]=\"line.x2\"\r\n [attr.y2]=\"line.y2\"\r\n stroke=\"blue\"\r\n stroke-width=\"2\" />\r\n </svg>\r\n <div\r\n class=\"thumb\"\r\n #thumb\r\n [style.left.px]=\"x\"\r\n [style.top.px]=\"y\"\r\n (mousedown)=\"dragStart($event)\"\r\n (touchstart)=\"dragStart($event)\"></div>\r\n </div>\r\n\r\n <div class=\"ngx-settings\">\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Offset X</span>\r\n <input type=\"number\" id=\"offsetX\" [(ngModel)]=\"value.x\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input\r\n type=\"range\"\r\n id=\"offsetX\"\r\n [(ngModel)]=\"value.x\"\r\n (ngModelChange)=\"onChangeValue()\"\r\n step=\"1\"\r\n [min]=\"-maxRange\"\r\n [max]=\"maxRange\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Offset Y</span>\r\n <input type=\"number\" id=\"offsetY\" [(ngModel)]=\"value.y\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input\r\n type=\"range\"\r\n id=\"offsetY\"\r\n [(ngModel)]=\"value.y\"\r\n (ngModelChange)=\"onChangeValue()\"\r\n step=\"1\"\r\n [min]=\"-maxRange\"\r\n [max]=\"maxRange\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Blur</span>\r\n <input type=\"number\" id=\"blur\" [(ngModel)]=\"blur\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input type=\"range\" id=\"blur\" [(ngModel)]=\"blur\" (ngModelChange)=\"onChangeValue()\" step=\"1\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Spread</span>\r\n <input type=\"number\" id=\"spread\" [(ngModel)]=\"spread\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input type=\"range\" id=\"spread\" [(ngModel)]=\"spread\" (ngModelChange)=\"onChangeValue()\" step=\"1\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>color</span>\r\n <input\r\n type=\"text\"\r\n readonly\r\n id=\"color\"\r\n [(ngModel)]=\"color\"\r\n ngxInputColor\r\n [simpleMode]=\"true\"\r\n (ngModelChange)=\"onChangeData()\"\r\n autocomplete=\"off\" />\r\n </div>\r\n </div>\r\n </div>\r\n\r\n</div>\r\n", styles: [".ngx-box-shadow-container{width:180px;max-width:100%;border:1px #bfbfbf solid;border-radius:15px;background-color:#fff;overflow:hidden;box-shadow:0 0 20px #0000004f;direction:ltr;padding:12px}.ngx-box-shadow-container *{box-sizing:border-box}.pad-container{position:relative;width:100px;height:100px;border:1px #545454 solid;box-sizing:border-box;direction:ltr;border-radius:3px}.pad-container:after,.pad-container:before{content:\"\";position:absolute;border:1px rgba(145,145,145,.4117647059) solid}.pad-container:after{width:100%;top:calc(50% - 1px);left:0}.pad-container:before{height:100%;top:0;left:calc(50% - 1px)}.pad-container .thumb{box-shadow:inset 0 0 0 2px #fff;width:15px;height:15px;display:block;border-radius:10px;position:absolute;cursor:pointer;background:#00f}.ngx-settings .ngx-row{display:flex;align-items:flex-end;justify-content:space-between}.ngx-settings input:not([type=range]){width:46px;border:none;border:1px solid rgba(145,145,145,.4117647059);padding:2px 4px;border-radius:5px;background-color:transparent;color:#000;font-size:14px;font-weight:500;text-align:center;direction:ltr}.ngx-settings input:not([type=range]):focus{outline:none;border:1px solid #09218d}.ngx-settings input[type=range]{width:100%}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.RangeValueAccessor, selector: "input[type=range][formControlName],input[type=range][formControl],input[type=range][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: NgxInputColorModule }, { kind: "directive", type: i2.NgxInputColorDirective, selector: "[ngxInputColor]", inputs: ["setInputBackgroundColor", "defaultInspector", "simpleMode", "outputType", "theme", "ngxInputColor"], outputs: ["change"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxBoxShadowComponent, decorators: [{ type: Component, args: [{ standalone: true, selector: 'ngx-box-shadow', changeDetection: ChangeDetectionStrategy.OnPush, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxBoxShadowComponent), multi: true }, { provide: NG_VALIDATORS, multi: true, useExisting: NgxBoxShadowComponent, }, ], imports: [CommonModule, FormsModule, NgxInputColorModule], template: "<div class=\"ngx-box-shadow-container\" (click)=\"stopPropagation($event)\">\r\n <div class=\"pad-container\" #pad>\r\n <svg viewBox=\"0 0 100 100\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <line\r\n [attr.x1]=\"line.x1\"\r\n [attr.y1]=\"line.y1\"\r\n [attr.x2]=\"line.x2\"\r\n [attr.y2]=\"line.y2\"\r\n stroke=\"blue\"\r\n stroke-width=\"2\" />\r\n </svg>\r\n <div\r\n class=\"thumb\"\r\n #thumb\r\n [style.left.px]=\"x\"\r\n [style.top.px]=\"y\"\r\n (mousedown)=\"dragStart($event)\"\r\n (touchstart)=\"dragStart($event)\"></div>\r\n </div>\r\n\r\n <div class=\"ngx-settings\">\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Offset X</span>\r\n <input type=\"number\" id=\"offsetX\" [(ngModel)]=\"value.x\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input\r\n type=\"range\"\r\n id=\"offsetX\"\r\n [(ngModel)]=\"value.x\"\r\n (ngModelChange)=\"onChangeValue()\"\r\n step=\"1\"\r\n [min]=\"-maxRange\"\r\n [max]=\"maxRange\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Offset Y</span>\r\n <input type=\"number\" id=\"offsetY\" [(ngModel)]=\"value.y\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input\r\n type=\"range\"\r\n id=\"offsetY\"\r\n [(ngModel)]=\"value.y\"\r\n (ngModelChange)=\"onChangeValue()\"\r\n step=\"1\"\r\n [min]=\"-maxRange\"\r\n [max]=\"maxRange\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Blur</span>\r\n <input type=\"number\" id=\"blur\" [(ngModel)]=\"blur\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input type=\"range\" id=\"blur\" [(ngModel)]=\"blur\" (ngModelChange)=\"onChangeValue()\" step=\"1\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>Spread</span>\r\n <input type=\"number\" id=\"spread\" [(ngModel)]=\"spread\" (ngModelChange)=\"onChangeValue()\" />\r\n </div>\r\n <input type=\"range\" id=\"spread\" [(ngModel)]=\"spread\" (ngModelChange)=\"onChangeValue()\" step=\"1\" />\r\n </div>\r\n <div class=\"ngx-setting-item\">\r\n <div class=\"ngx-row\">\r\n <span>color</span>\r\n <input\r\n type=\"text\"\r\n readonly\r\n id=\"color\"\r\n [(ngModel)]=\"color\"\r\n ngxInputColor\r\n [simpleMode]=\"true\"\r\n (ngModelChange)=\"onChangeData()\"\r\n autocomplete=\"off\" />\r\n </div>\r\n </div>\r\n </div>\r\n\r\n</div>\r\n", styles: [".ngx-box-shadow-container{width:180px;max-width:100%;border:1px #bfbfbf solid;border-radius:15px;background-color:#fff;overflow:hidden;box-shadow:0 0 20px #0000004f;direction:ltr;padding:12px}.ngx-box-shadow-container *{box-sizing:border-box}.pad-container{position:relative;width:100px;height:100px;border:1px #545454 solid;box-sizing:border-box;direction:ltr;border-radius:3px}.pad-container:after,.pad-container:before{content:\"\";position:absolute;border:1px rgba(145,145,145,.4117647059) solid}.pad-container:after{width:100%;top:calc(50% - 1px);left:0}.pad-container:before{height:100%;top:0;left:calc(50% - 1px)}.pad-container .thumb{box-shadow:inset 0 0 0 2px #fff;width:15px;height:15px;display:block;border-radius:10px;position:absolute;cursor:pointer;background:#00f}.ngx-settings .ngx-row{display:flex;align-items:flex-end;justify-content:space-between}.ngx-settings input:not([type=range]){width:46px;border:none;border:1px solid rgba(145,145,145,.4117647059);padding:2px 4px;border-radius:5px;background-color:transparent;color:#000;font-size:14px;font-weight:500;text-align:center;direction:ltr}.ngx-settings input:not([type=range]):focus{outline:none;border:1px solid #09218d}.ngx-settings input[type=range]{width:100%}\n"] }] }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { setTheme: [{ type: Input, args: ['theme'] }], maxRange: [{ type: Input }], change: [{ type: Output }], pad: [{ type: ViewChild, args: ['pad', { 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":"ngx-box-shadow.component.js","sourceRoot":"","sources":["../../../../../projects/ngx-input-color/src/lib/ngx-box-shadow/ngx-box-shadow.component.ts","../../../../../projects/ngx-input-color/src/lib/ngx-box-shadow/ngx-box-shadow.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport {\r\n  Component,\r\n  ChangeDetectionStrategy,\r\n  forwardRef,\r\n  OnInit,\r\n  OnDestroy,\r\n  ChangeDetectorRef,\r\n  HostListener,\r\n  ElementRef,\r\n  ViewChild,\r\n  AfterViewInit,\r\n  Input,\r\n  Output,\r\n  EventEmitter,\r\n} from '@angular/core';\r\nimport {\r\n  NG_VALUE_ACCESSOR,\r\n  NG_VALIDATORS,\r\n  FormsModule,\r\n  ControlValueAccessor,\r\n  Validator,\r\n  AbstractControl,\r\n  ValidationErrors,\r\n} from '@angular/forms';\r\nimport { getOffsetPosition } from '../../utils/get-offset-position';\r\nimport { IPosition } from '../../models/IPosition';\r\nimport { parseBoxShadowToPx, stringifyBoxShadow } from '../../utils/box-shadow';\r\nimport { NgxInputColorModule } from '../../ngx-input-color.module';\r\n\r\n@Component({\r\n  standalone: true,\r\n  selector: 'ngx-box-shadow',\r\n  templateUrl: './ngx-box-shadow.component.html',\r\n  styleUrls: ['./ngx-box-shadow.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n  providers: [\r\n    { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxBoxShadowComponent), multi: true },\r\n    {\r\n      provide: NG_VALIDATORS,\r\n      multi: true,\r\n      useExisting: NgxBoxShadowComponent,\r\n    },\r\n  ],\r\n  imports: [CommonModule, FormsModule, NgxInputColorModule],\r\n})\r\nexport class NgxBoxShadowComponent implements OnInit, AfterViewInit, OnDestroy, ControlValueAccessor, Validator {\r\n  theme: 'light' | 'dark' | 'auto' = 'light';\r\n  @Input('theme') set setTheme(val: 'light' | 'dark' | 'auto') {\r\n    if (!val || val == 'auto') {\r\n      this.theme = window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light';\r\n    } else {\r\n      this.theme = val;\r\n    }\r\n  }\r\n  /**\r\n   * The maximum range of the box shadow.\r\n   * @default 25\r\n   */\r\n  @Input() maxRange = 25;\r\n  @Output() change = new EventEmitter<string>();\r\n  isDisabled = false;\r\n  isDragging = false;\r\n  value: IPosition = { x: 0, y: 0 };\r\n  x = 0;\r\n  y = 0;\r\n  blur = 0;\r\n  spread = 0;\r\n  color = 'black';\r\n  line: { x1: number; y1: number; x2: number; y2: number } = { x1: 0, y1: 0, x2: 0, y2: 0 };\r\n  center: { x: number; y: number } = { x: 0, y: 0 };\r\n\r\n  padRect?: DOMRect;\r\n  thumbRect?: DOMRect;\r\n  _onChange = (value: string) => {};\r\n  _onTouched = () => {};\r\n  _onValidateChange = () => {};\r\n\r\n  @ViewChild('pad', { static: true }) pad!: ElementRef<HTMLDivElement>;\r\n  @ViewChild('thumb', { static: true }) thumb!: ElementRef<HTMLDivElement>;\r\n\r\n  constructor(private cd: ChangeDetectorRef) {}\r\n\r\n  ngOnInit(): void {}\r\n  ngAfterViewInit(): void {\r\n    this.updateRects();\r\n  }\r\n  ngOnDestroy(): void {}\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  }\r\n  registerOnValidatorChange(fn: () => void): void {\r\n    this._onValidateChange = fn;\r\n  }\r\n  validate(control: AbstractControl): ValidationErrors | null {\r\n    return null;\r\n  }\r\n\r\n  writeValue(value: any): void {\r\n    this.resetPosition();\r\n    if (value) {\r\n      const boxShadow = parseBoxShadowToPx(value);\r\n      console.log(boxShadow);\r\n      if (boxShadow) {\r\n        this.value = { x: boxShadow.offsetX, y: boxShadow.offsetY };\r\n        this.blur = boxShadow.blurRadius;\r\n        this.spread = boxShadow.spreadRadius;\r\n        this.color = boxShadow.color;\r\n        this.convertValueToPosition(boxShadow.offsetX, boxShadow.offsetY);\r\n      }\r\n    }\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.updatePosition(ev);\r\n    this.updateRects();\r\n  }\r\n  private updateRects() {\r\n    this.padRect = this.pad.nativeElement.getBoundingClientRect();\r\n    this.thumbRect = this.thumb.nativeElement.getBoundingClientRect();\r\n  }\r\n  private resetPosition() {\r\n    if (!this.padRect || !this.thumbRect) this.updateRects();\r\n    this.center = { x: this.padRect!.width / 2, y: this.padRect!.height / 2 };\r\n    this.x = this.center.x - this.thumbRect!.width / 2;\r\n    this.y = this.center.y - this.thumbRect!.height / 2;\r\n    this.line = { x1: this.center.x, y1: this.center.y, x2: this.center.x, y2: this.center.y };\r\n    this.cd.detectChanges();\r\n  }\r\n\r\n  @HostListener('window:resize', ['$event'])\r\n  onResize() {\r\n    this.writeValue(this.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.padRect || !this.thumbRect) this.updateRects();\r\n    const position = getOffsetPosition(ev, this.pad.nativeElement);\r\n\r\n    const padRec = this.padRect!;\r\n    const thumbRec = this.thumbRect!;\r\n\r\n    const minX = thumbRec.width / 2;\r\n    const maxX = padRec.width - thumbRec.width / 2;\r\n    const minY = thumbRec.height / 2;\r\n    const maxY = padRec.height - thumbRec.height / 2;\r\n\r\n    const clampedX = Math.max(minX, Math.min(position.x, maxX));\r\n    const clampedY = Math.max(minY, Math.min(position.y, maxY));\r\n\r\n    // فقط یکبار از وسط اصلاح کن\r\n    this.x = clampedX - thumbRec.width / 2;\r\n    this.y = clampedY - thumbRec.height / 2;\r\n\r\n    this.line = {\r\n      x1: this.center.x,\r\n      y1: this.center.y,\r\n      x2: this.x + thumbRec.width / 2,\r\n      y2: this.y + thumbRec.height / 2,\r\n    };\r\n\r\n    this.setValueByPosition(thumbRec, padRec);\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, padRec: DOMRect) {\r\n    const padCenterX = (padRec.width - thumbRec.width) / 2;\r\n    const padCenterY = (padRec.height - thumbRec.height) / 2;\r\n\r\n    // فاصله thumb از مرکز\r\n    const dx = this.x - padCenterX;\r\n    const dy = this.y - padCenterY;\r\n\r\n    // مقیاس تبدیل به -50 تا +50 (یا هرچقدر بخوای)\r\n    const halfRangeX = (padRec.width - thumbRec.width) / 2;\r\n    const halfRangeY = (padRec.height - thumbRec.height) / 2;\r\n\r\n    let valueX = (dx / halfRangeX) * this.maxRange; // -50 تا +50\r\n    let valueY = (dy / halfRangeY) * this.maxRange;\r\n\r\n    // رُند کردن\r\n    valueX = Math.round(valueX);\r\n    valueY = Math.round(valueY);\r\n\r\n    // محدود کردن به -50 تا +50 یا هر مقدار دلخواه\r\n    valueX = Math.min(Math.max(valueX, -this.maxRange), this.maxRange);\r\n    valueY = Math.min(Math.max(valueY, -this.maxRange), this.maxRange);\r\n\r\n    const newValue = { x: valueX, y: valueY };\r\n\r\n    if (!this.value || this.value.x !== valueX || this.value.y !== valueY) {\r\n      this.value = newValue;\r\n      this.onChangeData();\r\n    }\r\n  }\r\n\r\n  public onChangeValue() {\r\n    this.convertValueToPosition(this.value.x, this.value.y);\r\n    this.onChangeData();\r\n  }\r\n  private convertValueToPosition(offsetX: number, offsetY: number) {\r\n    if (!this.padRect || !this.thumbRect) this.updateRects();\r\n\r\n    // موقعیت پیشنهادی بر اساس offset\r\n    let proposedX = this.padRect!.width / 2 + offsetX - this.thumbRect!.width / 2;\r\n    let proposedY = this.padRect!.height / 2 + offsetY - this.thumbRect!.height / 2;\r\n\r\n    // محدود کردن (clamp) تا خارج از pad نره\r\n    const minX = 0;\r\n    const maxX = this.padRect!.width - this.thumbRect!.width;\r\n    const minY = 0;\r\n    const maxY = this.padRect!.height - this.thumbRect!.height;\r\n\r\n    this.x = Math.min(Math.max(proposedX, minX), maxX);\r\n    this.y = Math.min(Math.max(proposedY, minY), maxY);\r\n\r\n    // رسم خط راهنما از مرکز به thumb\r\n    this.line = {\r\n      x1: this.center.x,\r\n      y1: this.center.y,\r\n      x2: this.x + this.thumbRect!.width / 2,\r\n      y2: this.y + this.thumbRect!.height / 2,\r\n    };\r\n\r\n    this.cd.detectChanges();\r\n  }\r\n\r\n  onChangeData() {\r\n    const boxShadow = stringifyBoxShadow({\r\n      inset: false,\r\n      offsetX: this.value.x,\r\n      offsetY: this.value.y,\r\n      blurRadius: this.blur,\r\n      spreadRadius: this.spread,\r\n      color: this.color,\r\n    });\r\n    this._onChange(boxShadow);\r\n    this.change.emit(boxShadow);\r\n  }\r\n\r\n  stopPropagation(ev: Event) {\r\n    ev.stopPropagation();\r\n  }\r\n}\r\n","<div class=\"ngx-box-shadow-container\" (click)=\"stopPropagation($event)\">\r\n  <div class=\"pad-container\" #pad>\r\n    <svg viewBox=\"0 0 100 100\" xmlns=\"http://www.w3.org/2000/svg\">\r\n      <line\r\n        [attr.x1]=\"line.x1\"\r\n        [attr.y1]=\"line.y1\"\r\n        [attr.x2]=\"line.x2\"\r\n        [attr.y2]=\"line.y2\"\r\n        stroke=\"blue\"\r\n        stroke-width=\"2\" />\r\n    </svg>\r\n    <div\r\n      class=\"thumb\"\r\n      #thumb\r\n      [style.left.px]=\"x\"\r\n      [style.top.px]=\"y\"\r\n      (mousedown)=\"dragStart($event)\"\r\n      (touchstart)=\"dragStart($event)\"></div>\r\n  </div>\r\n\r\n  <div class=\"ngx-settings\">\r\n    <div class=\"ngx-setting-item\">\r\n      <div class=\"ngx-row\">\r\n        <span>Offset X</span>\r\n        <input type=\"number\" id=\"offsetX\" [(ngModel)]=\"value.x\" (ngModelChange)=\"onChangeValue()\" />\r\n      </div>\r\n      <input\r\n        type=\"range\"\r\n        id=\"offsetX\"\r\n        [(ngModel)]=\"value.x\"\r\n        (ngModelChange)=\"onChangeValue()\"\r\n        step=\"1\"\r\n        [min]=\"-maxRange\"\r\n        [max]=\"maxRange\" />\r\n    </div>\r\n    <div class=\"ngx-setting-item\">\r\n      <div class=\"ngx-row\">\r\n        <span>Offset Y</span>\r\n        <input type=\"number\" id=\"offsetY\" [(ngModel)]=\"value.y\" (ngModelChange)=\"onChangeValue()\" />\r\n      </div>\r\n      <input\r\n        type=\"range\"\r\n        id=\"offsetY\"\r\n        [(ngModel)]=\"value.y\"\r\n        (ngModelChange)=\"onChangeValue()\"\r\n        step=\"1\"\r\n        [min]=\"-maxRange\"\r\n        [max]=\"maxRange\" />\r\n    </div>\r\n    <div class=\"ngx-setting-item\">\r\n      <div class=\"ngx-row\">\r\n        <span>Blur</span>\r\n        <input type=\"number\" id=\"blur\" [(ngModel)]=\"blur\" (ngModelChange)=\"onChangeValue()\" />\r\n      </div>\r\n      <input type=\"range\" id=\"blur\" [(ngModel)]=\"blur\" (ngModelChange)=\"onChangeValue()\" step=\"1\" />\r\n    </div>\r\n    <div class=\"ngx-setting-item\">\r\n      <div class=\"ngx-row\">\r\n        <span>Spread</span>\r\n        <input type=\"number\" id=\"spread\" [(ngModel)]=\"spread\" (ngModelChange)=\"onChangeValue()\" />\r\n      </div>\r\n      <input type=\"range\" id=\"spread\" [(ngModel)]=\"spread\" (ngModelChange)=\"onChangeValue()\" step=\"1\" />\r\n    </div>\r\n    <div class=\"ngx-setting-item\">\r\n      <div class=\"ngx-row\">\r\n        <span>color</span>\r\n        <input\r\n          type=\"text\"\r\n          readonly\r\n          id=\"color\"\r\n          [(ngModel)]=\"color\"\r\n          ngxInputColor\r\n          [simpleMode]=\"true\"\r\n          (ngModelChange)=\"onChangeData()\"\r\n          autocomplete=\"off\" />\r\n      </div>\r\n    </div>\r\n  </div>\r\n\r\n</div>\r\n"]}