@junte/ui
Version:
Quality Angular UI components kit
344 lines • 37.3 kB
JavaScript
var ImageCropperComponent_1;
import { __decorate, __metadata } from "tslib";
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, HostBinding, HostListener, Input, OnInit, Output, ViewChild } from '@angular/core';
import { ControlValueAccessor, FormBuilder, NG_VALUE_ACCESSOR } from '@angular/forms';
import { DomSanitizer, SafeStyle, SafeUrl } from '@angular/platform-browser';
import { NGXLogger } from 'ngx-logger';
import { PropertyApi } from '../../core/decorators/api';
import { Shape } from '../../core/enums/shape';
import { UI } from '../../core/enums/ui';
import { I18N_PROVIDERS } from '../../core/i18n/providers';
const CROPPER_SIZE = 200;
const DEFAULT_SCALE = 1;
const DEFAULT_MIN = 0.01;
const DEFAULT_MAX = 5;
const DEFAULT_STEP = 0.01;
export var MoveTypes;
(function (MoveTypes) {
MoveTypes["Move"] = "move";
MoveTypes["Pinch"] = "pinch";
})(MoveTypes || (MoveTypes = {}));
export class MoveStart {
constructor() {
this.active = false;
this.type = null;
this.left = 0;
this.top = 0;
this.clientX = 0;
this.clientY = 0;
}
}
export class ImagePosition {
constructor() {
this.left = 0;
this.top = 0;
this.scale = DEFAULT_SCALE;
this.width = 0;
this.height = 0;
}
}
export class CropperPosition {
constructor() {
this.width = CROPPER_SIZE;
this.height = CROPPER_SIZE;
}
}
let ImageCropperComponent = ImageCropperComponent_1 = class ImageCropperComponent {
constructor(logger, cd, fb, sanitizer) {
this.logger = logger;
this.cd = cd;
this.fb = fb;
this.sanitizer = sanitizer;
this.ui = UI;
this.host = 'jnt-image-cropper-host';
this.moveStart = new MoveStart();
this.sizeRetries = 0;
this._min = DEFAULT_MIN;
this._max = DEFAULT_MAX;
this._step = DEFAULT_STEP;
this.marginLeft = '0px';
this.moveTypes = MoveTypes;
this.imagePosition = new ImagePosition();
this.form = this.fb.group({
zoom: [this.imagePosition.scale]
});
this.area = new CropperPosition();
this.disabled = false;
this._shape = Shape.circle;
this.loaded = new EventEmitter();
this.failed = new EventEmitter();
this.onChange = () => this.logger.error('value accessor is not registered');
this.onTouched = () => this.logger.error('value accessor is not registered');
this.registerOnChange = fn => this.onChange = fn;
this.registerOnTouched = fn => this.onTouched = fn;
this.onBlur = () => this.onTouched();
}
set min(min) {
this._min = min || DEFAULT_MIN;
}
get min() {
return this._min;
}
set max(max) {
this._max = max || DEFAULT_MAX;
}
get max() {
return this._max;
}
set step(step) {
this._step = step || DEFAULT_STEP;
}
get step() {
return this._step;
}
set url(url) {
this._url = url;
if (!!url) {
this.moveStart = new MoveStart();
this.imagePosition = new ImagePosition();
}
}
get url() {
return this._url;
}
set shape(shape) {
this._shape = shape || Shape.circle;
}
ngOnInit() {
this.form.valueChanges.subscribe(({ zoom }) => this.zoom(zoom));
}
inView() {
if (this.image.nativeElement.currentSrc.includes('image/svg')) {
this.disabled = true;
}
this.loaded.emit();
this.sizeRetries = 0;
setTimeout(() => this.checkImageMaxSizeRecursively());
}
checkImageMaxSizeRecursively() {
if (this.sizeRetries > 40) {
this.failed.emit();
}
else if (!!this.image && !!this.image.nativeElement && this.image.nativeElement.offsetWidth > 0) {
const image = this.image.nativeElement;
const wrapper = this.wrapper.nativeElement;
this.imagePosition.width = image.offsetWidth;
this.imagePosition.height = image.offsetHeight;
this.imagePosition.top = (wrapper.offsetHeight - image.offsetHeight) / 2;
this.imagePosition.left = (wrapper.offsetWidth - image.offsetWidth) / 2;
let scale = Math.trunc(wrapper.offsetWidth / image.offsetWidth * 100) / 100;
scale = Math.min(scale, Math.trunc(wrapper.offsetHeight / image.offsetHeight * 100) / 100, this.max);
this.zoom(scale);
this.cd.detectChanges();
}
else {
this.sizeRetries++;
setTimeout(() => this.checkImageMaxSizeRecursively(), 50);
}
}
startMove(event, moveType) {
if (!!this.moveStart && this.moveStart.active && this.moveStart.type === MoveTypes.Pinch) {
return;
}
if (!!event.preventDefault) {
event.preventDefault();
}
this.moveStart = Object.assign({ active: true, type: moveType, clientX: this.getClientX(event), clientY: this.getClientY(event) }, this.imagePosition);
}
moveImg(event) {
if (this.moveStart.active) {
if (event.stopPropagation) {
event.stopPropagation();
}
if (event.preventDefault) {
event.preventDefault();
}
if (this.moveStart.type === MoveTypes.Move) {
this.move(event);
}
this.cd.detectChanges();
}
}
moveStop() {
if (this.moveStart.active) {
this.moveStart.active = false;
this.crop();
}
}
crop() {
const image = this.image.nativeElement;
const cropper = this.cropper.nativeElement;
const scale = this.imagePosition.scale;
this.onChange({
left: Math.round(cropper.offsetLeft - (image.offsetLeft - ((image.width * scale - image.width) / 2))),
top: Math.round(cropper.offsetTop - (image.offsetTop - ((image.height * scale - image.height) / 2))),
scale
});
}
move(event) {
const diffX = this.getClientX(event) - this.moveStart.clientX;
const diffY = this.getClientY(event) - this.moveStart.clientY;
this.imagePosition.left = this.moveStart.left + diffX;
this.imagePosition.top = this.moveStart.top + diffY;
}
;
getClientX(event) {
return (event.touches && event.touches[0] ? event.touches[0].clientX : event.clientX) || 0;
}
getClientY(event) {
return (event.touches && event.touches[0] ? event.touches[0].clientY : event.clientY) || 0;
}
zoom(scale = DEFAULT_SCALE) {
this.imagePosition.scale = scale;
this.imagePosition.width = this.imagePosition.width * scale;
this.imagePosition.height = this.imagePosition.height * scale;
this.transformStyle = this.sanitizer
.bypassSecurityTrustStyle(`scaleX(${scale || DEFAULT_SCALE})scaleY(${scale || DEFAULT_SCALE})`);
this.cd.detectChanges();
this.crop();
}
writeValue(data) {
if (!!data) {
this.imagePosition = Object.assign(Object.assign({}, this.imagePosition), data);
}
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
}
};
ImageCropperComponent.ctorParameters = () => [
{ type: NGXLogger },
{ type: ChangeDetectorRef },
{ type: FormBuilder },
{ type: DomSanitizer }
];
__decorate([
HostBinding('attr.host'),
__metadata("design:type", Object)
], ImageCropperComponent.prototype, "host", void 0);
__decorate([
ViewChild('wrapper', { static: true }),
__metadata("design:type", ElementRef)
], ImageCropperComponent.prototype, "wrapper", void 0);
__decorate([
ViewChild('image'),
__metadata("design:type", ElementRef)
], ImageCropperComponent.prototype, "image", void 0);
__decorate([
ViewChild('cropper'),
__metadata("design:type", ElementRef)
], ImageCropperComponent.prototype, "cropper", void 0);
__decorate([
PropertyApi({
description: 'Size of crop area',
type: 'CropperPosition',
default: '{width: 200, height: 200}'
}),
Input(),
__metadata("design:type", Object)
], ImageCropperComponent.prototype, "area", void 0);
__decorate([
PropertyApi({
description: 'Min of cropping',
type: 'number',
default: '0.01'
}),
Input(),
__metadata("design:type", Number),
__metadata("design:paramtypes", [Number])
], ImageCropperComponent.prototype, "min", null);
__decorate([
PropertyApi({
description: 'Max of cropping',
type: 'number',
default: '5'
}),
Input(),
__metadata("design:type", Number),
__metadata("design:paramtypes", [Number])
], ImageCropperComponent.prototype, "max", null);
__decorate([
PropertyApi({
description: 'Step of cropping',
type: 'number',
default: '0.01'
}),
Input(),
__metadata("design:type", Number),
__metadata("design:paramtypes", [Number])
], ImageCropperComponent.prototype, "step", null);
__decorate([
PropertyApi({
description: 'Url of image',
type: 'string',
default: null
}),
Input(),
__metadata("design:type", Object),
__metadata("design:paramtypes", [Object])
], ImageCropperComponent.prototype, "url", null);
__decorate([
HostBinding('attr.data-disabled'),
__metadata("design:type", Object)
], ImageCropperComponent.prototype, "disabled", void 0);
__decorate([
HostBinding('attr.data-shape'),
__metadata("design:type", String)
], ImageCropperComponent.prototype, "_shape", void 0);
__decorate([
PropertyApi({
description: 'Avatar shape',
path: 'ui.shape',
default: Shape.circle,
options: [Shape.circle, Shape.square]
}),
Input(),
__metadata("design:type", String),
__metadata("design:paramtypes", [String])
], ImageCropperComponent.prototype, "shape", null);
__decorate([
Output(),
__metadata("design:type", Object)
], ImageCropperComponent.prototype, "loaded", void 0);
__decorate([
Output(),
__metadata("design:type", Object)
], ImageCropperComponent.prototype, "failed", void 0);
__decorate([
HostListener('blur'),
__metadata("design:type", Object)
], ImageCropperComponent.prototype, "onBlur", void 0);
__decorate([
HostListener('document:mousemove', ['$event']),
HostListener('document:touchmove', ['$event']),
__metadata("design:type", Function),
__metadata("design:paramtypes", [Object]),
__metadata("design:returntype", void 0)
], ImageCropperComponent.prototype, "moveImg", null);
__decorate([
HostListener('document:mouseup'),
HostListener('document:touchend'),
__metadata("design:type", Function),
__metadata("design:paramtypes", []),
__metadata("design:returntype", void 0)
], ImageCropperComponent.prototype, "moveStop", null);
ImageCropperComponent = ImageCropperComponent_1 = __decorate([
Component({
selector: 'jnt-image-cropper',
template: "<label child-of=\"jnt-image-cropper-host\" *ngIf=\"!imagePosition.width && !imagePosition.height\">\n <jnt-icon child-of=\"jnt-image-cropper-host\" [icon]=\"ui.icons.download\"></jnt-icon>\n</label>\n\n<div child-of=\"jnt-image-cropper-host\" #wrapper data-wrapper>\n <img child-of=\"jnt-image-cropper-host\" #image\n *ngIf=\"!!url\"\n [src]=\"url\"\n (load)=\"inView()\"\n [style.top.px]=\"imagePosition.top\"\n [style.left.px]=\"imagePosition.left\"\n [style.transform]=\"transformStyle\"\n (mousedown)=\"startMove($event, moveTypes.Move)\"\n (touchstart)=\"startMove($event, moveTypes.Move)\" />\n\n <div child-of=\"jnt-image-cropper-host\" #cropper\n data-cropper *ngIf=\"!!imagePosition.width && !!imagePosition.height\"\n [style.top]=\"'calc(50% - ' + area.height / 2 + 'px)'\"\n [style.left]=\"'calc(50% - ' + area.width / 2 + 'px)'\"\n [style.width.px]=\"area.width\"\n [style.height.px]=\"area.height\"\n tabindex=\"0\">\n </div>\n</div>\n\n<form child-of=\"jnt-image-cropper-host\" [formGroup]=\"form\" data-zoom [attr.data-disabled]=\"!imagePosition.width || !imagePosition.height || disabled\">\n <jnt-slider child-of=\"jnt-image-cropper-host\" formControlName=\"zoom\"\n [min]=\"min\"\n [max]=\"max\"\n [step]=\"step\">\n </jnt-slider>\n</form>",
changeDetection: ChangeDetectionStrategy.OnPush,
providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => ImageCropperComponent_1),
multi: true
}, ...I18N_PROVIDERS
]
}),
__metadata("design:paramtypes", [NGXLogger,
ChangeDetectorRef,
FormBuilder,
DomSanitizer])
], ImageCropperComponent);
export { ImageCropperComponent };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-cropper.component.js","sourceRoot":"ng://@junte/ui/","sources":["lib/forms/image-cropper/image-cropper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  forwardRef,\n  HostBinding,\n  HostListener,\n  Input, OnInit,\n  Output,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, FormBuilder, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { DomSanitizer, SafeStyle, SafeUrl } from '@angular/platform-browser';\nimport { NGXLogger } from 'ngx-logger';\nimport { PropertyApi } from '../../core/decorators/api';\nimport { Shape } from '../../core/enums/shape';\nimport { UI } from '../../core/enums/ui';\nimport { I18N_PROVIDERS } from '../../core/i18n/providers';\n\nconst CROPPER_SIZE = 200;\nconst DEFAULT_SCALE = 1;\nconst DEFAULT_MIN = 0.01;\nconst DEFAULT_MAX = 5;\nconst DEFAULT_STEP = 0.01;\n\nexport enum MoveTypes {\n  Move = 'move',\n  Pinch = 'pinch'\n}\n\nexport class MoveStart {\n  active: boolean = false;\n  type: MoveTypes | null = null;\n  left: number = 0;\n  top: number = 0;\n  clientX: number = 0;\n  clientY: number = 0;\n}\n\nexport class ImagePosition {\n  left: number = 0;\n  top: number = 0;\n  scale: number = DEFAULT_SCALE;\n  width: number = 0;\n  height: number = 0;\n}\n\nexport class CropperPosition {\n  width: number = CROPPER_SIZE;\n  height: number = CROPPER_SIZE;\n}\n\nexport type ImageCroppedData = {\n  left: number,\n  top: number,\n  scale: number\n}\n\n@Component({\n  selector: 'jnt-image-cropper',\n  templateUrl: './image-cropper.encapsulated.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => ImageCropperComponent),\n      multi: true\n    }, ...I18N_PROVIDERS\n  ]\n})\nexport class ImageCropperComponent implements OnInit, ControlValueAccessor {\n\n  ui = UI;\n\n  @HostBinding('attr.host')\n  readonly host = 'jnt-image-cropper-host';\n\n  private moveStart = new MoveStart();\n  private sizeRetries = 0;\n  private _url: SafeUrl | string;\n  private _min = DEFAULT_MIN;\n  private _max = DEFAULT_MAX;\n  private _step = DEFAULT_STEP;\n\n  transformStyle: SafeStyle | string;\n  marginLeft: SafeStyle | string = '0px';\n  moveTypes = MoveTypes;\n  imagePosition = new ImagePosition();\n\n  form = this.fb.group({\n    zoom: [this.imagePosition.scale]\n  });\n\n  @ViewChild('wrapper', {static: true}) wrapper: ElementRef;\n  @ViewChild('image') image: ElementRef;\n  @ViewChild('cropper') cropper: ElementRef;\n\n  @PropertyApi({\n    description: 'Size of crop area',\n    type: 'CropperPosition',\n    default: '{width: 200, height: 200}'\n  })\n  @Input() area = new CropperPosition();\n\n  @PropertyApi({\n    description: 'Min of cropping',\n    type: 'number',\n    default: '0.01'\n  })\n  @Input() set min(min: number) {\n    this._min = min || DEFAULT_MIN;\n  }\n\n  get min() {\n    return this._min;\n  }\n\n  @PropertyApi({\n    description: 'Max of cropping',\n    type: 'number',\n    default: '5'\n  })\n  @Input() set max(max: number) {\n    this._max = max || DEFAULT_MAX\n  }\n\n  get max() {\n    return this._max;\n  }\n\n  @PropertyApi({\n    description: 'Step of cropping',\n    type: 'number',\n    default: '0.01'\n  })\n  @Input() set step(step: number) {\n    this._step = step || DEFAULT_STEP;\n  }\n\n  get step() {\n    return this._step;\n  }\n\n  @PropertyApi({\n    description: 'Url of image',\n    type: 'string',\n    default: null\n  })\n  @Input() set url(url: SafeUrl | string) {\n    this._url = url;\n    if (!!url) {\n      this.moveStart = new MoveStart();\n      this.imagePosition = new ImagePosition();\n    }\n  }\n\n  get url() {\n    return this._url;\n  }\n\n  @HostBinding('attr.data-disabled')\n  disabled = false;\n\n  @HostBinding('attr.data-shape')\n  _shape: Shape = Shape.circle;\n\n  @PropertyApi({\n    description: 'Avatar shape',\n    path: 'ui.shape',\n    default: Shape.circle,\n    options: [Shape.circle, Shape.square]\n  })\n  @Input() set shape(shape: Shape) {\n    this._shape = shape || Shape.circle;\n  }\n\n  @Output() loaded = new EventEmitter<void>();\n  @Output() failed = new EventEmitter<void>();\n\n  onChange: (date: ImageCroppedData) => void = () => this.logger.error('value accessor is not registered');\n  onTouched: () => void = () => this.logger.error('value accessor is not registered');\n  registerOnChange = fn => this.onChange = fn;\n  registerOnTouched = fn => this.onTouched = fn;\n  @HostListener('blur') onBlur = () => this.onTouched();\n\n  constructor(private logger: NGXLogger,\n              private cd: ChangeDetectorRef,\n              private fb: FormBuilder,\n              public sanitizer: DomSanitizer) {\n  }\n\n  ngOnInit() {\n    this.form.valueChanges.subscribe(({zoom}) => this.zoom(zoom))\n  }\n\n  inView(): void {\n    if (this.image.nativeElement.currentSrc.includes('image/svg')) {\n      this.disabled = true;\n    }\n    this.loaded.emit();\n    this.sizeRetries = 0;\n    setTimeout(() => this.checkImageMaxSizeRecursively());\n  }\n\n  private checkImageMaxSizeRecursively(): void {\n    if (this.sizeRetries > 40) {\n      this.failed.emit();\n    } else if (!!this.image && !!this.image.nativeElement && this.image.nativeElement.offsetWidth > 0) {\n      const image = this.image.nativeElement;\n      const wrapper = this.wrapper.nativeElement;\n      this.imagePosition.width = image.offsetWidth;\n      this.imagePosition.height = image.offsetHeight;\n      this.imagePosition.top = (wrapper.offsetHeight - image.offsetHeight) / 2;\n      this.imagePosition.left = (wrapper.offsetWidth - image.offsetWidth) / 2;\n      let scale = Math.trunc(wrapper.offsetWidth / image.offsetWidth * 100) / 100;\n      scale = Math.min(scale, Math.trunc(wrapper.offsetHeight / image.offsetHeight * 100) / 100, this.max);\n      this.zoom(scale);\n      this.cd.detectChanges();\n    } else {\n      this.sizeRetries++;\n      setTimeout(() => this.checkImageMaxSizeRecursively(), 50);\n    }\n  }\n\n  startMove(event: any, moveType: MoveTypes): void {\n    if (!!this.moveStart && this.moveStart.active && this.moveStart.type === MoveTypes.Pinch) {\n      return;\n    }\n    if (!!event.preventDefault) {\n      event.preventDefault();\n    }\n    this.moveStart = {\n      active: true,\n      type: moveType,\n      clientX: this.getClientX(event),\n      clientY: this.getClientY(event),\n      ...this.imagePosition\n    };\n  }\n\n  @HostListener('document:mousemove', ['$event'])\n  @HostListener('document:touchmove', ['$event'])\n  moveImg(event: any): void {\n    if (this.moveStart.active) {\n      if (event.stopPropagation) {\n        event.stopPropagation();\n      }\n      if (event.preventDefault) {\n        event.preventDefault();\n      }\n      if (this.moveStart.type === MoveTypes.Move) {\n        this.move(event);\n      }\n      this.cd.detectChanges();\n    }\n  }\n\n  @HostListener('document:mouseup')\n  @HostListener('document:touchend')\n  moveStop(): void {\n    if (this.moveStart.active) {\n      this.moveStart.active = false;\n      this.crop();\n    }\n  }\n\n  private crop() {\n    const image = this.image.nativeElement;\n    const cropper = this.cropper.nativeElement;\n    const scale = this.imagePosition.scale;\n    this.onChange({\n      left: Math.round(cropper.offsetLeft - (image.offsetLeft - ((image.width * scale - image.width) / 2))),\n      top: Math.round(cropper.offsetTop - (image.offsetTop - ((image.height * scale - image.height) / 2))),\n      scale\n    });\n  }\n\n  private move(event: MouseEvent) {\n    const diffX = this.getClientX(event) - this.moveStart.clientX;\n    const diffY = this.getClientY(event) - this.moveStart.clientY;\n\n    this.imagePosition.left = this.moveStart.left + diffX;\n    this.imagePosition.top = this.moveStart.top + diffY;\n  };\n\n  private getClientX(event: any): number {\n    return (event.touches && event.touches[0] ? event.touches[0].clientX : event.clientX) || 0;\n  }\n\n  private getClientY(event: any): number {\n    return (event.touches && event.touches[0] ? event.touches[0].clientY : event.clientY) || 0;\n  }\n\n  zoom(scale = DEFAULT_SCALE) {\n    this.imagePosition.scale = scale;\n    this.imagePosition.width = this.imagePosition.width * scale;\n    this.imagePosition.height = this.imagePosition.height * scale;\n    this.transformStyle = this.sanitizer\n      .bypassSecurityTrustStyle(`scaleX(${scale || DEFAULT_SCALE})scaleY(${scale || DEFAULT_SCALE})`);\n    this.cd.detectChanges();\n    this.crop();\n  }\n\n  writeValue(data: ImageCroppedData): void {\n    if (!!data) {\n      this.imagePosition = {...this.imagePosition, ...data};\n    }\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.disabled = isDisabled;\n  }\n}\n"]}