UNPKG

@junte/ui

Version:

Quality Angular UI components kit

344 lines 37.3 kB
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"]}