UNPKG

ngx-input-color

Version:

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

156 lines 20.2 kB
import { Directive, HostListener, Inject, Input, forwardRef, } from '@angular/core'; import { NG_VALIDATORS, NG_VALUE_ACCESSOR, } from '@angular/forms'; import { DOCUMENT } from '@angular/common'; import { NgxBoxShadowComponent } from '../lib/ngx-box-shadow/ngx-box-shadow.component'; import * as i0 from "@angular/core"; export class NgxInputBoxShadowDirective { constructor(_doc, el, renderer, viewContainerRef) { this._doc = _doc; this.el = el; this.renderer = renderer; this.viewContainerRef = viewContainerRef; this.setInputBackground = true; this.isDisabled = false; this.value = ''; this._onChange = (value) => { }; this._onTouched = () => { }; this._onValidateChange = () => { }; } onClick(ev) { ev.stopPropagation(); ev.preventDefault(); this.toggleColorPicker(); } registerOnChange(fn) { this._onChange = fn; } registerOnTouched(fn) { this._onTouched = fn; } setDisabledState(disabled) { this.isDisabled = disabled; } registerOnValidatorChange(fn) { this._onValidateChange = fn; } validate(control) { return null; } ngOnDestroy() { this.destroyPicker(); } writeValue(value) { this.value = value; } toggleColorPicker() { if (this.pickerComponentRef) { this.destroyPicker(); return; } // ایجاد کامپوننت this.pickerComponentRef = this.viewContainerRef.createComponent(NgxBoxShadowComponent); const instance = this.pickerComponentRef.instance; instance.writeValue(this.value); // بک‌دراپ this.backdrop = this.renderer.createElement('div'); if (this.backdrop) { this.backdrop.style.cssText = ` background: #5e5e5e1e; position: fixed; top: 0; left: 0; right: 0; bottom: 0; overflow: auto; transition: all 300ms; z-index: 1000; `; this.backdrop.onclick = () => this.destroyPicker(); } // گرفتن المنت کامپوننت واقعی this.pickerEl = this.pickerComponentRef.hostView.rootNodes[0]; this.renderer.appendChild(this.backdrop, this.pickerEl); this.renderer.appendChild(this._doc.body, this.backdrop); this.setPosition(); } setPosition() { setTimeout(() => { if (!this.pickerEl || !this.pickerComponentRef) return; const hostRect = this.el.nativeElement.getBoundingClientRect(); const pickerEl = this.pickerEl; // اعمال موقتی برای گرفتن سایز دقیق this.renderer.setStyle(pickerEl, 'position', 'absolute'); this.renderer.setStyle(pickerEl, 'z-index', '9999'); this._doc.body.appendChild(pickerEl); // لازم برای محاسبه دقیق اندازه const pickerRect = pickerEl.getBoundingClientRect(); // وسط‌چین کردن افقی let left = hostRect.left + hostRect.width / 2 - pickerRect.width / 2; let top = hostRect.bottom; // جلوگیری از بیرون زدن از راست if (left + pickerRect.width > window.innerWidth) { left = window.innerWidth - pickerRect.width - 8; } // جلوگیری از بیرون زدن از چپ if (left < 8) { left = 8; } // اگر از پایین بیرون زد، ببر بالا if (top + pickerRect.height > window.innerHeight) { top = hostRect.top - pickerRect.height; } // جلوگیری از بیرون زدن از بالا if (top < 8) { top = 8; } this.renderer.setStyle(pickerEl, 'top', `${top}px`); this.renderer.setStyle(pickerEl, 'left', `${left}px`); }); } destroyPicker() { if (this.pickerComponentRef) { this.pickerComponentRef.destroy(); this.pickerComponentRef = undefined; } if (this.backdrop && this.backdrop.parentNode) { this.renderer.removeChild(this._doc.body, this.backdrop); this.backdrop = undefined; } this.pickerEl = undefined; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxInputBoxShadowDirective, deps: [{ token: DOCUMENT }, { token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.ViewContainerRef }], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: NgxInputBoxShadowDirective, selector: "[ngxInputBoxShadow]", inputs: { setInputBackground: "setInputBackground" }, host: { listeners: { "click": "onClick($event)", "window:resize": "setPosition($event)" } }, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxInputBoxShadowDirective), multi: true }, { provide: NG_VALIDATORS, multi: true, useExisting: NgxInputBoxShadowDirective, }, ], ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxInputBoxShadowDirective, decorators: [{ type: Directive, args: [{ selector: '[ngxInputBoxShadow]', providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxInputBoxShadowDirective), multi: true }, { provide: NG_VALIDATORS, multi: true, useExisting: NgxInputBoxShadowDirective, }, ], }] }], ctorParameters: () => [{ type: Document, decorators: [{ type: Inject, args: [DOCUMENT] }] }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ViewContainerRef }], propDecorators: { setInputBackground: [{ type: Input }], onClick: [{ type: HostListener, args: ['click', ['$event']] }], setPosition: [{ type: HostListener, args: ['window:resize', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-input-box-shadow.directive.js","sourceRoot":"","sources":["../../../../projects/ngx-input-color/src/directives/ngx-input-box-shadow.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  ComponentRef,\r\n  Directive,\r\n  ElementRef,\r\n  HostListener,\r\n  Inject,\r\n  Input,\r\n  OnDestroy,\r\n  Renderer2,\r\n  ViewContainerRef,\r\n  forwardRef,\r\n} from '@angular/core';\r\nimport {\r\n  AbstractControl,\r\n  ControlValueAccessor,\r\n  NG_VALIDATORS,\r\n  NG_VALUE_ACCESSOR,\r\n  ValidationErrors,\r\n  Validator,\r\n} from '@angular/forms';\r\nimport { ColorInspector } from '../models/ColorInspector.enum'; \r\nimport { DOCUMENT } from '@angular/common'; \r\nimport { NgxBoxShadowComponent } from '../lib/ngx-box-shadow/ngx-box-shadow.component';\r\n\r\n@Directive({\r\n  selector: '[ngxInputBoxShadow]',\r\n  providers: [\r\n    { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxInputBoxShadowDirective), multi: true },\r\n    {\r\n      provide: NG_VALIDATORS,\r\n      multi: true,\r\n      useExisting: NgxInputBoxShadowDirective,\r\n    },\r\n  ],\r\n})\r\nexport class NgxInputBoxShadowDirective implements OnDestroy, ControlValueAccessor, Validator {\r\n  @Input() setInputBackground = true;\r\n\r\n  private pickerComponentRef?: ComponentRef<NgxBoxShadowComponent>;\r\n  private backdrop?: HTMLDivElement;\r\n  private pickerEl?: HTMLElement;\r\n  isDisabled = false;\r\n\r\n  value = '';\r\n\r\n  _onChange = (value: string) => {};\r\n  _onTouched = () => {};\r\n  _onValidateChange = () => {};\r\n  constructor(\r\n    @Inject(DOCUMENT) private _doc: Document,\r\n    private el: ElementRef,\r\n    private renderer: Renderer2,\r\n    private viewContainerRef: ViewContainerRef\r\n  ) {}\r\n\r\n  @HostListener('click', ['$event']) onClick(ev: Event) {\r\n    ev.stopPropagation();\r\n    ev.preventDefault();\r\n    this.toggleColorPicker();\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  }\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  ngOnDestroy(): void {\r\n    this.destroyPicker();\r\n  }\r\n\r\n  writeValue(value: any): void {\r\n    this.value = value;\r\n   \r\n  }\r\n\r\n  toggleColorPicker() {\r\n    if (this.pickerComponentRef) {\r\n      this.destroyPicker();\r\n      return;\r\n    }\r\n\r\n    // ایجاد کامپوننت\r\n    this.pickerComponentRef = this.viewContainerRef.createComponent(NgxBoxShadowComponent);\r\n\r\n    const instance = this.pickerComponentRef.instance;\r\n    instance.writeValue(this.value);\r\n\r\n    // بک‌دراپ\r\n    this.backdrop = this.renderer.createElement('div');\r\n    if (this.backdrop) {\r\n      this.backdrop.style.cssText = `\r\n          background: #5e5e5e1e;\r\n          position: fixed;\r\n          top: 0;\r\n          left: 0;\r\n          right: 0;\r\n          bottom: 0;\r\n          overflow: auto;\r\n          transition: all 300ms;\r\n          z-index: 1000;\r\n        `;\r\n      this.backdrop.onclick = () => this.destroyPicker();\r\n    }\r\n    // گرفتن المنت کامپوننت واقعی\r\n    this.pickerEl = (this.pickerComponentRef.hostView as any).rootNodes[0] as HTMLElement;\r\n    this.renderer.appendChild(this.backdrop, this.pickerEl);\r\n    this.renderer.appendChild(this._doc.body, this.backdrop);\r\n    this.setPosition();\r\n  }\r\n\r\n  @HostListener('window:resize', ['$event'])\r\n  setPosition() {\r\n    setTimeout(() => {\r\n      if (!this.pickerEl || !this.pickerComponentRef) return;\r\n      const hostRect = this.el.nativeElement.getBoundingClientRect();\r\n      const pickerEl = this.pickerEl;\r\n\r\n      // اعمال موقتی برای گرفتن سایز دقیق\r\n      this.renderer.setStyle(pickerEl, 'position', 'absolute');\r\n      this.renderer.setStyle(pickerEl, 'z-index', '9999');\r\n\r\n      this._doc.body.appendChild(pickerEl); // لازم برای محاسبه دقیق اندازه\r\n\r\n      const pickerRect = pickerEl.getBoundingClientRect();\r\n\r\n      // وسط‌چین کردن افقی\r\n      let left = hostRect.left + hostRect.width / 2 - pickerRect.width / 2;\r\n      let top = hostRect.bottom;\r\n\r\n      // جلوگیری از بیرون زدن از راست\r\n      if (left + pickerRect.width > window.innerWidth) {\r\n        left = window.innerWidth - pickerRect.width - 8;\r\n      }\r\n\r\n      // جلوگیری از بیرون زدن از چپ\r\n      if (left < 8) {\r\n        left = 8;\r\n      }\r\n\r\n      // اگر از پایین بیرون زد، ببر بالا\r\n      if (top + pickerRect.height > window.innerHeight) {\r\n        top = hostRect.top - pickerRect.height;\r\n      }\r\n\r\n      // جلوگیری از بیرون زدن از بالا\r\n      if (top < 8) {\r\n        top = 8;\r\n      }\r\n\r\n      this.renderer.setStyle(pickerEl, 'top', `${top}px`);\r\n      this.renderer.setStyle(pickerEl, 'left', `${left}px`);\r\n    });\r\n  }\r\n\r\n  destroyPicker() {\r\n    if (this.pickerComponentRef) {\r\n      this.pickerComponentRef.destroy();\r\n      this.pickerComponentRef = undefined;\r\n    }\r\n    if (this.backdrop && this.backdrop.parentNode) {\r\n      this.renderer.removeChild(this._doc.body, this.backdrop);\r\n      this.backdrop = undefined;\r\n    }\r\n    this.pickerEl = undefined;\r\n  }\r\n}\r\n"]}