ngx-input-color
Version:
Angular color input component and color picker (with HSL, HSV, RGB, CMYK, HEX, alpha, eye-dropper, etc)
248 lines • 31.5 kB
JavaScript
import { Directive, forwardRef, Input, ElementRef, HostListener, Output, EventEmitter, Inject, } from '@angular/core';
import { NG_VALUE_ACCESSOR, NG_VALIDATORS, } from '@angular/forms';
import { ColorInspector } from '../models/ColorInspector.enum';
import { NgxInputColorComponent } from '../lib/ngx-input-color/ngx-input-color.component';
import { NgxColor } from '../utils/color-helper';
import { DOCUMENT } from '@angular/common';
import * as i0 from "@angular/core";
export class NgxInputColorDirective {
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.setInputBackgroundColor = true;
this.defaultInspector = ColorInspector.Picker;
this.simpleMode = false;
this.outputType = 'HEX';
this.theme = 'auto';
this.boundInputHandler = (e) => {
this.writeValue(e.target.value);
};
this.change = new EventEmitter();
this.isHostInput = false;
this.inValid = false;
this.isDisabled = false;
this._onChange = (value) => { };
this._onTouched = () => { };
this._onValidateChange = () => { };
}
ngAfterViewInit() {
setTimeout(() => {
if (this._targetInput && this._targetInput.tagName.toLowerCase() === 'input') {
this.writeValue(this._targetInput.value);
}
});
}
ngOnDestroy() {
this.destroyColorPicker();
}
onClick(ev) {
ev.stopPropagation();
ev.preventDefault();
this.toggleColorPicker();
}
writeValue(value) {
try {
this.color = value ? new NgxColor(value) : undefined;
const colorStr = this.color?.toHexString() ?? '';
// اگر دایرکتیو روی input باشه (ControlValueAccessor)
if (this.isHostInput) {
const input = this.el.nativeElement;
input.value = colorStr;
}
// اگر input خارجی مشخص شده
if (this._targetInput instanceof HTMLInputElement) {
this._targetInput.value = colorStr;
}
if (this.setInputBackgroundColor && colorStr) {
this.renderer.setStyle(this.el.nativeElement, 'backgroundColor', colorStr);
}
this.inValid = false;
this._onValidateChange();
}
catch (e) {
this.color = new NgxColor('#000'); // مقدار پیشفرض
this.inValid = true;
}
}
registerOnChange(fn) {
this._onChange = fn;
}
registerOnTouched(fn) {
this._onTouched = fn;
}
setDisabledState(disabled) {
this.isDisabled = disabled;
}
registerOnValidatorChange(fn) {
this._onValidateChange = fn;
}
validate(control) {
if ((this.color && this.color.isValid === false) || this.inValid === true) {
return { invalid: true };
}
return null;
}
toggleColorPicker() {
if (this.colorPickerComponentRef) {
this.destroyColorPicker();
return;
}
this.colorPickerComponentRef = this.viewContainerRef.createComponent(NgxInputColorComponent);
const instance = this.colorPickerComponentRef.instance;
instance.defaultInspector = this.defaultInspector;
instance.simpleMode = this.simpleMode;
instance.outputType = this.outputType;
instance.setTheme = this.theme;
if (this.color?.isValid)
instance.initColor(this.color);
instance.change.subscribe((c) => {
this.emitChange(c);
});
this.backdrop = this.renderer.createElement('div');
if (this.backdrop) {
this.backdrop.className = 'ngx-input-color-backdrop';
this.backdrop.style.cssText = `
background: #5e5e5e1e;
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
overflow: auto;
transition: all 300ms;
z-index: 9998;
`;
this.backdrop.onclick = () => this.destroyColorPicker();
}
this.colorPickerEl = this.colorPickerComponentRef.hostView.rootNodes[0];
this.renderer.appendChild(this.backdrop, this.colorPickerEl);
this.renderer.appendChild(this._doc.body, this.backdrop);
this.setPosition();
}
setPosition() {
// setTimeout(() => {
if (!this.colorPickerEl)
return;
const hostRect = this.el.nativeElement.getBoundingClientRect();
const pickerEl = this.colorPickerEl;
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`);
// });
}
destroyColorPicker() {
this.colorPickerComponentRef?.destroy();
this.colorPickerComponentRef = undefined;
if (this.backdrop) {
this.renderer.removeChild(this._doc.body, this.backdrop);
this.backdrop = undefined;
}
this.colorPickerEl = undefined;
}
async emitChange(c) {
if (this.setInputBackgroundColor) {
this.renderer.setStyle(this.el.nativeElement, 'backgroundColor', 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: NgxInputColorDirective, 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: NgxInputColorDirective, selector: "[ngxInputColor]", inputs: { setInputBackgroundColor: "setInputBackgroundColor", defaultInspector: "defaultInspector", simpleMode: "simpleMode", outputType: "outputType", theme: "theme", ngxInputColor: "ngxInputColor" }, outputs: { change: "change" }, host: { listeners: { "click": "onClick($event)", "window:resize": "setPosition()" } }, providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => NgxInputColorDirective),
multi: true,
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => NgxInputColorDirective),
multi: true,
},
], ngImport: i0 }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NgxInputColorDirective, decorators: [{
type: Directive,
args: [{
selector: '[ngxInputColor]',
providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => NgxInputColorDirective),
multi: true,
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => NgxInputColorDirective),
multi: true,
},
],
}]
}], ctorParameters: () => [{ type: Document, decorators: [{
type: Inject,
args: [DOCUMENT]
}] }, { type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ViewContainerRef }], propDecorators: { setInputBackgroundColor: [{
type: Input
}], defaultInspector: [{
type: Input
}], simpleMode: [{
type: Input
}], outputType: [{
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']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-input-color.directive.js","sourceRoot":"","sources":["../../../../projects/ngx-input-color/src/directives/ngx-input-color.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Directive,\r\n  forwardRef,\r\n  OnDestroy,\r\n  Input,\r\n  ComponentRef,\r\n  ElementRef,\r\n  Renderer2,\r\n  ViewContainerRef,\r\n  HostListener,\r\n  AfterViewInit,\r\n  Output,\r\n  EventEmitter,\r\n  Inject,\r\n} from '@angular/core';\r\nimport {\r\n  NG_VALUE_ACCESSOR,\r\n  NG_VALIDATORS,\r\n  ControlValueAccessor,\r\n  Validator,\r\n  AbstractControl,\r\n  ValidationErrors,\r\n} from '@angular/forms';\r\nimport { ColorInspector } from '../models/ColorInspector.enum';\r\nimport { NgxInputColorComponent } from '../lib/ngx-input-color/ngx-input-color.component';\r\nimport { NgxColor, OutputType } from '../utils/color-helper';\r\nimport { DOCUMENT } from '@angular/common';\r\n\r\n@Directive({\r\n  selector: '[ngxInputColor]',\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => NgxInputColorDirective),\r\n      multi: true,\r\n    },\r\n    {\r\n      provide: NG_VALIDATORS,\r\n      useExisting: forwardRef(() => NgxInputColorDirective),\r\n      multi: true,\r\n    },\r\n  ],\r\n})\r\nexport class NgxInputColorDirective implements AfterViewInit, OnDestroy, ControlValueAccessor, Validator {\r\n  @Input() setInputBackgroundColor = true;\r\n  @Input() defaultInspector: ColorInspector = ColorInspector.Picker;\r\n  @Input() simpleMode = false;\r\n  @Input() outputType: OutputType = 'HEX';\r\n  @Input() theme: 'light' | 'dark' | 'auto' = 'auto';\r\n  private boundInputHandler = (e: Event) => {\r\n    this.writeValue((e.target as HTMLInputElement).value);\r\n  };\r\n  private _targetInput?: HTMLInputElement;\r\n\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  private color?: NgxColor;\r\n  private colorPickerComponentRef?: ComponentRef<NgxInputColorComponent>;\r\n  private backdrop?: HTMLDivElement;\r\n  private colorPickerEl?: HTMLElement;\r\n  private isHostInput = false;\r\n  inValid: boolean = false;\r\n  isDisabled = false;\r\n  _onChange = (value: string) => {};\r\n  _onTouched = () => {};\r\n  _onValidateChange = () => {};\r\n\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  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  ngOnDestroy(): void {\r\n    this.destroyColorPicker();\r\n  }\r\n\r\n  @HostListener('click', ['$event'])\r\n  onClick(ev: Event) {\r\n    ev.stopPropagation();\r\n    ev.preventDefault();\r\n    this.toggleColorPicker();\r\n  }\r\n\r\n  writeValue(value: any): void {\r\n    try {\r\n      this.color = value ? new NgxColor(value) : undefined;\r\n\r\n      const colorStr = this.color?.toHexString() ?? '';\r\n\r\n      // اگر دایرکتیو روی input باشه (ControlValueAccessor)\r\n      if (this.isHostInput) {\r\n        const input = this.el.nativeElement as HTMLInputElement;\r\n        input.value = colorStr;\r\n      }\r\n\r\n      // اگر input خارجی مشخص شده\r\n      if (this._targetInput instanceof HTMLInputElement) {\r\n        this._targetInput.value = colorStr;\r\n      }\r\n\r\n      if (this.setInputBackgroundColor && colorStr) {\r\n        this.renderer.setStyle(this.el.nativeElement, 'backgroundColor', colorStr);\r\n      }\r\n\r\n      this.inValid = false;\r\n      this._onValidateChange();\r\n    } catch (e) {\r\n      this.color = new NgxColor('#000'); // مقدار پیش‌فرض\r\n      this.inValid = true;\r\n    }\r\n  }\r\n\r\n  registerOnChange(fn: any): void {\r\n    this._onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: any): void {\r\n    this._onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(disabled: boolean): void {\r\n    this.isDisabled = disabled;\r\n  }\r\n\r\n  registerOnValidatorChange(fn: () => void): void {\r\n    this._onValidateChange = fn;\r\n  }\r\n\r\n  validate(control: AbstractControl): ValidationErrors | null {\r\n    if ((this.color && this.color.isValid === false) || this.inValid === true) {\r\n      return { invalid: true };\r\n    }\r\n    return null;\r\n  }\r\n\r\n  private toggleColorPicker() {\r\n    if (this.colorPickerComponentRef) {\r\n      this.destroyColorPicker();\r\n      return;\r\n    }\r\n\r\n    this.colorPickerComponentRef = this.viewContainerRef.createComponent(NgxInputColorComponent);\r\n    const instance = this.colorPickerComponentRef.instance;\r\n\r\n    instance.defaultInspector = this.defaultInspector;\r\n    instance.simpleMode = this.simpleMode;\r\n    instance.outputType = this.outputType;\r\n    instance.setTheme = this.theme;\r\n\r\n    if (this.color?.isValid) instance.initColor(this.color);\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.className = 'ngx-input-color-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: 9998;\r\n        `;\r\n      this.backdrop.onclick = () => this.destroyColorPicker();\r\n    }\r\n    this.colorPickerEl = (this.colorPickerComponentRef.hostView as any).rootNodes[0] as HTMLElement;\r\n    this.renderer.appendChild(this.backdrop, this.colorPickerEl);\r\n    this.renderer.appendChild(this._doc.body, this.backdrop);\r\n    this.setPosition();\r\n  }\r\n\r\n  @HostListener('window:resize')\r\n  setPosition() {\r\n    // setTimeout(() => {\r\n    if (!this.colorPickerEl) return;\r\n\r\n    const hostRect = this.el.nativeElement.getBoundingClientRect();\r\n    const pickerEl = this.colorPickerEl;\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    const pickerRect = pickerEl.getBoundingClientRect();\r\n\r\n    let left = hostRect.left + hostRect.width / 2 - pickerRect.width / 2;\r\n    let top = hostRect.bottom;\r\n\r\n    if (left + pickerRect.width > window.innerWidth) left = window.innerWidth - pickerRect.width - 8;\r\n    if (left < 8) left = 8;\r\n    if (top + pickerRect.height > window.innerHeight) top = hostRect.top - pickerRect.height;\r\n    if (top < 8) top = 8;\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  private destroyColorPicker() {\r\n    this.colorPickerComponentRef?.destroy();\r\n    this.colorPickerComponentRef = undefined;\r\n\r\n    if (this.backdrop) {\r\n      this.renderer.removeChild(this._doc.body, this.backdrop);\r\n      this.backdrop = undefined;\r\n    }\r\n\r\n    this.colorPickerEl = undefined;\r\n  }\r\n\r\n  private async emitChange(c: string) {\r\n    if (this.setInputBackgroundColor) {\r\n      this.renderer.setStyle(this.el.nativeElement, 'backgroundColor', 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"]}