ngx-input-color
Version:
Angular color input component and color picker (with HSL, HSV, RGB, CMYK, HEX, alpha, eye-dropper, etc)
228 lines • 29.4 kB
JavaScript
import { Directive, ElementRef, EventEmitter, HostListener, Inject, Input, Output, forwardRef, } from '@angular/core';
import { NG_VALIDATORS, NG_VALUE_ACCESSOR, } from '@angular/forms';
import { NgxInputGradientComponent } from '../lib/ngx-input-gradient/ngx-input-gradient.component';
import { DOCUMENT } from '@angular/common';
import { isValidGradient, parseGradient } from '../utils/build-gradient';
import * as i0 from "@angular/core";
export class NgxInputGradientDirective {
set ngxInputColor(el) {
this.isHostInput = false;
if (el instanceof ElementRef) {
this._targetInput = el.nativeElement;
}
else if (el instanceof HTMLInputElement) {
this.isHostInput = true;
this._targetInput = el;
}
else {
this._targetInput = undefined;
}
if (this._targetInput) {
this._targetInput.addEventListener('input', this.boundInputHandler);
}
}
constructor(_doc, el, renderer, viewContainerRef) {
this._doc = _doc;
this.el = el;
this.renderer = renderer;
this.viewContainerRef = viewContainerRef;
this.setInputBackground = true;
this.theme = 'auto';
this.change = new EventEmitter();
this.boundInputHandler = (e) => {
this.writeValue(e.target.value);
};
this.isHostInput = false;
this.isDisabled = false;
this.value = '';
this._onChange = (value) => { };
this._onTouched = () => { };
this._onValidateChange = () => { };
}
onClick(ev) {
ev.stopPropagation();
ev.preventDefault();
this.toggleColorPicker();
}
ngAfterViewInit() {
setTimeout(() => {
if (this._targetInput && this._targetInput.tagName.toLowerCase() === 'input') {
this.writeValue(this._targetInput.value);
}
});
}
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;
if (value && isValidGradient(value)) {
const parsed = parseGradient(value);
if (parsed.valid && this.setInputBackground) {
this.renderer.setStyle(this.el.nativeElement, 'background', value);
}
// اگر دایرکتیو روی input باشه (ControlValueAccessor)
if (this.isHostInput) {
const input = this.el.nativeElement;
input.value = this.value;
}
this._onValidateChange();
}
}
toggleColorPicker() {
if (this.pickerComponentRef) {
this.destroyPicker();
return;
}
// ایجاد کامپوننت
this.pickerComponentRef = this.viewContainerRef.createComponent(NgxInputGradientComponent);
const instance = this.pickerComponentRef.instance;
instance.setTheme = this.theme;
instance.writeValue(this.value);
instance.change.subscribe((c) => {
this.emitChange(c);
});
// بکدراپ
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', '1001');
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;
}
async emitChange(c) {
if (this.setInputBackground) {
this.renderer.setStyle(this.el.nativeElement, 'background', c);
}
// اگر روی input باشیم، مقدار رو در input قرار بده
if (this.isHostInput) {
const input = this.el.nativeElement;
input.value = c;
}
// اگر targetInput وجود داره، در اونم مقدار ست کن
if (this._targetInput instanceof HTMLInputElement) {
this._targetInput.value = c;
const event = new Event('input', { bubbles: true });
this._targetInput.dispatchEvent(event);
}
this._onChange(c);
this.change.emit(c);
this._onTouched();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxInputGradientDirective, 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: NgxInputGradientDirective, selector: "[ngxInputGradient]", inputs: { setInputBackground: "setInputBackground", theme: "theme", ngxInputColor: "ngxInputColor" }, outputs: { change: "change" }, host: { listeners: { "click": "onClick($event)", "window:resize": "setPosition($event)" } }, providers: [
{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxInputGradientDirective), multi: true },
{
provide: NG_VALIDATORS,
multi: true,
useExisting: NgxInputGradientDirective,
},
], ngImport: i0 }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxInputGradientDirective, decorators: [{
type: Directive,
args: [{
selector: '[ngxInputGradient]',
providers: [
{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxInputGradientDirective), multi: true },
{
provide: NG_VALIDATORS,
multi: true,
useExisting: NgxInputGradientDirective,
},
],
}]
}], ctorParameters: () => [{ type: Document, decorators: [{
type: Inject,
args: [DOCUMENT]
}] }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ViewContainerRef }], propDecorators: { setInputBackground: [{
type: Input
}], theme: [{
type: Input
}], ngxInputColor: [{
type: Input,
args: ['ngxInputColor']
}], change: [{
type: Output
}], onClick: [{
type: HostListener,
args: ['click', ['$event']]
}], setPosition: [{
type: HostListener,
args: ['window:resize', ['$event']]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-input-gradient.directive.js","sourceRoot":"","sources":["../../../../projects/ngx-input-color/src/directives/ngx-input-gradient.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  AfterViewInit,\r\n  ComponentRef,\r\n  Directive,\r\n  ElementRef,\r\n  EventEmitter,\r\n  HostListener,\r\n  Inject,\r\n  Input,\r\n  OnDestroy,\r\n  Output,\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 { NgxInputGradientComponent } from '../lib/ngx-input-gradient/ngx-input-gradient.component';\r\nimport { DOCUMENT } from '@angular/common';\r\nimport { isValidGradient, parseGradient } from '../utils/build-gradient';\r\n\r\n@Directive({\r\n  selector: '[ngxInputGradient]',\r\n  providers: [\r\n    { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NgxInputGradientDirective), multi: true },\r\n    {\r\n      provide: NG_VALIDATORS,\r\n      multi: true,\r\n      useExisting: NgxInputGradientDirective,\r\n    },\r\n  ],\r\n})\r\nexport class NgxInputGradientDirective implements AfterViewInit, OnDestroy, ControlValueAccessor, Validator {\r\n  @Input() setInputBackground = true;\r\n  @Input() theme: 'light' | 'dark' | 'auto' = 'auto';\r\n  private _targetInput?: HTMLInputElement;\r\n  @Input('ngxInputColor') set ngxInputColor(\r\n    el: HTMLInputElement | ElementRef<HTMLInputElement> | null | undefined | ''\r\n  ) {\r\n    this.isHostInput = false;\r\n    if (el instanceof ElementRef) {\r\n      this._targetInput = el.nativeElement;\r\n    } else if (el instanceof HTMLInputElement) {\r\n      this.isHostInput = true;\r\n      this._targetInput = el;\r\n    } else {\r\n      this._targetInput = undefined;\r\n    }\r\n\r\n    if (this._targetInput) {\r\n      this._targetInput.addEventListener('input', this.boundInputHandler);\r\n    }\r\n  }\r\n  @Output() change = new EventEmitter<string>();\r\n\r\n  private boundInputHandler = (e: Event) => {\r\n    this.writeValue((e.target as HTMLInputElement).value);\r\n  };\r\n  private isHostInput = false;\r\n  private pickerComponentRef?: ComponentRef<NgxInputGradientComponent>;\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  ngAfterViewInit(): void {\r\n    setTimeout(() => {\r\n      if (this._targetInput && this._targetInput.tagName.toLowerCase() === 'input') {\r\n        this.writeValue(this._targetInput.value);\r\n      }\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    if (value && isValidGradient(value)) {\r\n      const parsed = parseGradient(value);\r\n      if (parsed.valid && this.setInputBackground) {\r\n        this.renderer.setStyle(this.el.nativeElement, 'background', value);\r\n      }\r\n      // اگر دایرکتیو روی input باشه (ControlValueAccessor)\r\n      if (this.isHostInput) {\r\n        const input = this.el.nativeElement as HTMLInputElement;\r\n        input.value = this.value;\r\n      }\r\n      this._onValidateChange();\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(NgxInputGradientComponent);\r\n\r\n    const instance = this.pickerComponentRef.instance;\r\n    instance.setTheme = this.theme;\r\n    instance.writeValue(this.value);\r\n    instance.change.subscribe((c: string) => {\r\n      this.emitChange(c);\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', '1001');\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  private async emitChange(c: string) {\r\n    if (this.setInputBackground) {\r\n      this.renderer.setStyle(this.el.nativeElement, 'background', c);\r\n    }\r\n\r\n    // اگر روی input باشیم، مقدار رو در input قرار بده\r\n    if (this.isHostInput) {\r\n      const input = this.el.nativeElement as HTMLInputElement;\r\n      input.value = c;\r\n    }\r\n\r\n    // اگر targetInput وجود داره، در اونم مقدار ست کن\r\n    if (this._targetInput instanceof HTMLInputElement) {\r\n      this._targetInput.value = c;\r\n      const event = new Event('input', { bubbles: true });\r\n      this._targetInput.dispatchEvent(event);\r\n    }\r\n\r\n    this._onChange(c);\r\n    this.change.emit(c);\r\n    this._onTouched();\r\n  }\r\n}\r\n"]}