angular-color-picker-fixed
Version:
Radial Color Picker - Angular
161 lines • 21.4 kB
JavaScript
import { __decorate } from "tslib";
import { coerceBooleanProperty } from '@angular/cdk/coercion';
import { Platform } from '@angular/cdk/platform';
import { AfterViewInit, Directive, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, OnInit, Output, Renderer2, SimpleChanges } from '@angular/core';
import { fromEvent, merge } from 'rxjs';
import { filter, takeUntil, tap } from 'rxjs/operators';
import { calculateQuadrant, determineCSSRotationAngle } from '../helpers/helpers';
var RotatableDirective = /** @class */ (function () {
function RotatableDirective(el, renderer, platform) {
this.el = el;
this.renderer = renderer;
this.platform = platform;
this.dragging = false;
this.rotateStart = new EventEmitter();
this.rotating = new EventEmitter();
this.rotateStop = new EventEmitter();
if (this.platform.IOS || this.platform.ANDROID) {
this.mouseDownEv = 'touchstart';
this.mouseUpEv = 'touchend';
this.mouseMoveEv = 'touchmove';
this.cancelEv = 'touchcancel';
}
else {
this.mouseDownEv = 'mousedown';
this.mouseUpEv = 'mouseup';
this.mouseMoveEv = 'mousemove';
this.cancelEv = 'mouseout';
}
}
Object.defineProperty(RotatableDirective.prototype, "isDisabled", {
get: function () {
return this.disable ? coerceBooleanProperty(this.disable) : false;
},
enumerable: true,
configurable: true
});
RotatableDirective.prototype.ngOnInit = function () {
};
RotatableDirective.prototype.ngOnChanges = function (changes) {
if (changes.angle && changes.angle.currentValue) {
// console.log(changes.angle.currentValue);
var angle = changes.angle.currentValue + 90;
this.renderer.setStyle(this.el.nativeElement, 'transform', "rotate(" + angle + "deg)");
}
};
RotatableDirective.prototype.ngAfterViewInit = function () {
var _this = this;
// console.log(this.isDisabled);
requestAnimationFrame(this.initialRender.bind(this));
this.rect = this.el.nativeElement.getBoundingClientRect();
this.mouseUp$ = fromEvent(this.el.nativeElement, this.mouseUpEv, { passive: true });
this.mouseOut$ = fromEvent(this.el.nativeElement, this.cancelEv, { passive: true });
this.mouseDownSub = fromEvent(this.el.nativeElement, this.mouseDownEv, { passive: true })
.pipe(filter(function (val) {
return _this.active && !_this.isDisabled;
}))
.subscribe(function (downEvent) {
_this.dragging = true;
_this.rect = _this.el.nativeElement.getBoundingClientRect();
// console.log('mouse down', downEvent, this.rect);
_this.point = _this.createPoint(downEvent);
_this.rotateStart.emit(_this.point);
_this.applyRotation();
_this.mouseMoveSub = fromEvent(_this.el.nativeElement, _this.mouseMoveEv).pipe(takeUntil(merge(_this.mouseOut$, _this.mouseUp$).pipe(tap(function (upEvent) {
_this.rect = _this.el.nativeElement.getBoundingClientRect();
// console.log('mouse up', upEvent, this.rect);
_this.dragging = false;
_this.mouseMoveSub.unsubscribe();
_this.rotateStop.emit(_this.point);
})))).subscribe(function (moveEvent) {
_this.rect = _this.el.nativeElement.getBoundingClientRect();
// console.log('mouse move', moveEvent, this.rect);
_this.point = _this.createPoint(moveEvent);
// console.log(this.point);
_this.applyRotation();
});
});
};
RotatableDirective.prototype.initialRender = function () {
var angle = this.angle + 90;
this.renderer.setStyle(this.el.nativeElement, 'transform', "rotate(" + angle + "deg)");
};
RotatableDirective.prototype.rotationRender = function () {
// console.log(this.rotation);
this.renderer.setStyle(this.el.nativeElement, 'transform', "rotate(" + this.rotation + "deg)");
};
RotatableDirective.prototype.ngOnDestroy = function () {
if (this.mouseDownSub) {
this.mouseDownSub.unsubscribe();
}
if (this.mouseMoveSub) {
this.mouseMoveSub.unsubscribe();
}
if (this.mouseUpSub) {
this.mouseUpSub.unsubscribe();
}
// console.log('directive destroy');
};
RotatableDirective.prototype.applyRotation = function () {
var quadrant = calculateQuadrant(this.point);
var rotation = determineCSSRotationAngle(this.point, quadrant);
// console.log(rotation);
this.rotating.emit(rotation);
this.rotation = rotation;
requestAnimationFrame(this.rotationRender.bind(this));
};
RotatableDirective.prototype.createPoint = function (mouseEvent) {
var point;
if (mouseEvent.targetTouches) {
point = {
x: this._normalizeX(mouseEvent.targetTouches[0].clientX),
y: this._normalizeY(mouseEvent.targetTouches[0].clientY)
};
}
else {
point = {
x: this._normalizeX(mouseEvent.clientX),
y: this._normalizeY(mouseEvent.clientY)
};
}
// console.log('point', point);
return point;
};
RotatableDirective.prototype._normalizeX = function (coordX) {
return coordX - this.rect.left - this.rect.width / 2;
};
RotatableDirective.prototype._normalizeY = function (coordY) {
return ((coordY - this.rect.top) * -1) + this.rect.height / 2;
};
RotatableDirective.ctorParameters = function () { return [
{ type: ElementRef },
{ type: Renderer2 },
{ type: Platform }
]; };
__decorate([
Input()
], RotatableDirective.prototype, "angle", void 0);
__decorate([
Input()
], RotatableDirective.prototype, "disable", void 0);
__decorate([
Input()
], RotatableDirective.prototype, "active", void 0);
__decorate([
Output()
], RotatableDirective.prototype, "rotateStart", void 0);
__decorate([
Output()
], RotatableDirective.prototype, "rotating", void 0);
__decorate([
Output()
], RotatableDirective.prototype, "rotateStop", void 0);
RotatableDirective = __decorate([
Directive({
selector: '[rcpRotatable]'
})
], RotatableDirective);
return RotatableDirective;
}());
export { RotatableDirective };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rotatable.directive.js","sourceRoot":"ng://@radial-color-picker/angular-color-picker/","sources":["lib/directives/rotatable.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { Platform } from '@angular/cdk/platform';\nimport {\n  AfterViewInit,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  Renderer2,\n  SimpleChanges\n} from '@angular/core';\nimport { fromEvent, Observable, Subscription, merge } from 'rxjs';\nimport { filter, takeUntil, tap } from 'rxjs/operators';\nimport { calculateQuadrant, determineCSSRotationAngle } from '../helpers/helpers';\n\n@Directive({\n  selector: '[rcpRotatable]'\n})\nexport class RotatableDirective implements OnInit, OnChanges, OnDestroy, AfterViewInit {\n  public rotation: any;\n  public dragging = false;\n  public mouseDownSub: Subscription;\n  public mouseMoveSub: Subscription;\n  public mouseUpSub: Subscription;\n  public rect: any;\n\n  public cancelEv: string;\n  public mouseDownEv: string;\n  public mouseMoveEv: string;\n  public mouseUpEv: string;\n\n  @Input() angle: number;\n  @Input() disable: boolean;\n  @Input() active: boolean;\n\n  get isDisabled() {\n    return this.disable ? coerceBooleanProperty(this.disable) : false;\n  }\n\n  @Output() public rotateStart = new EventEmitter<any>();\n  @Output() public rotating = new EventEmitter<any>();\n  @Output() public rotateStop = new EventEmitter<any>();\n\n  private point: any;\n  private mouseUp$: Observable<any>;\n  private mouseOut$: Observable<any>;\n\n  constructor(\n    private el: ElementRef,\n    private renderer: Renderer2,\n    private platform: Platform\n  ) {\n    if (this.platform.IOS || this.platform.ANDROID) {\n      this.mouseDownEv = 'touchstart';\n      this.mouseUpEv = 'touchend';\n      this.mouseMoveEv = 'touchmove';\n      this.cancelEv = 'touchcancel';\n    } else {\n      this.mouseDownEv = 'mousedown';\n      this.mouseUpEv = 'mouseup';\n      this.mouseMoveEv = 'mousemove';\n      this.cancelEv = 'mouseout';\n    }\n\n  }\n\n  ngOnInit() {\n\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.angle && changes.angle.currentValue) {\n      // console.log(changes.angle.currentValue);\n      const angle = changes.angle.currentValue + 90;\n      this.renderer.setStyle(this.el.nativeElement, 'transform', `rotate(${angle}deg)`);\n    }\n  }\n\n  ngAfterViewInit() {\n    // console.log(this.isDisabled);\n    requestAnimationFrame(this.initialRender.bind(this));\n    this.rect = this.el.nativeElement.getBoundingClientRect();\n    this.mouseUp$ = fromEvent(this.el.nativeElement, this.mouseUpEv, { passive: true });\n    this.mouseOut$ = fromEvent(this.el.nativeElement, this.cancelEv, { passive: true });\n\n    this.mouseDownSub = fromEvent(this.el.nativeElement, this.mouseDownEv, { passive: true })\n      .pipe(\n        filter((val) => {\n          return this.active && !this.isDisabled;\n        })\n      )\n      .subscribe((downEvent) => {\n        this.dragging = true;\n        this.rect = this.el.nativeElement.getBoundingClientRect();\n        // console.log('mouse down', downEvent, this.rect);\n        this.point = this.createPoint(downEvent);\n        this.rotateStart.emit(this.point);\n        this.applyRotation();\n\n\n        this.mouseMoveSub = fromEvent(this.el.nativeElement, this.mouseMoveEv).pipe(\n          takeUntil(merge(this.mouseOut$, this.mouseUp$).pipe(\n            tap((upEvent) => {\n              this.rect = this.el.nativeElement.getBoundingClientRect();\n              // console.log('mouse up', upEvent, this.rect);\n              this.dragging = false;\n              this.mouseMoveSub.unsubscribe();\n              this.rotateStop.emit(this.point);\n            })\n          ))\n        ).subscribe((moveEvent: MouseEvent) => {\n          this.rect = this.el.nativeElement.getBoundingClientRect();\n          // console.log('mouse move', moveEvent, this.rect);\n\n          this.point = this.createPoint(moveEvent);\n          // console.log(this.point);\n          this.applyRotation();\n        });\n      });\n  }\n\n  public initialRender() {\n    const angle = this.angle + 90;\n    this.renderer.setStyle(this.el.nativeElement, 'transform', `rotate(${angle}deg)`);\n  }\n\n  public rotationRender() {\n    // console.log(this.rotation);\n    this.renderer.setStyle(this.el.nativeElement, 'transform', `rotate(${this.rotation}deg)`);\n  }\n\n  ngOnDestroy() {\n    if (this.mouseDownSub) {\n      this.mouseDownSub.unsubscribe();\n    }\n    if (this.mouseMoveSub) {\n      this.mouseMoveSub.unsubscribe();\n    }\n    if (this.mouseUpSub) {\n      this.mouseUpSub.unsubscribe();\n    }\n    // console.log('directive destroy');\n  }\n\n\n\n  private applyRotation() {\n    const quadrant = calculateQuadrant(this.point);\n    const rotation = determineCSSRotationAngle(this.point, quadrant);\n    // console.log(rotation);\n    this.rotating.emit(rotation);\n    this.rotation = rotation;\n    requestAnimationFrame(this.rotationRender.bind(this));\n\n  }\n\n  private createPoint(mouseEvent) {\n    let point;\n    if (mouseEvent.targetTouches) {\n      point = {\n        x: this._normalizeX(mouseEvent.targetTouches[0].clientX),\n        y: this._normalizeY(mouseEvent.targetTouches[0].clientY)\n      };\n    } else {\n      point = {\n        x: this._normalizeX(mouseEvent.clientX),\n        y: this._normalizeY(mouseEvent.clientY)\n      };\n    }\n    // console.log('point', point);\n    return point;\n  }\n\n  private _normalizeX(coordX) {\n    return coordX - this.rect.left - this.rect.width / 2;\n  }\n\n  private _normalizeY(coordY) {\n    return ((coordY - this.rect.top) * -1) + this.rect.height / 2;\n  }\n\n}\n"]}