UNPKG

@junte/ui

Version:

Quality Angular UI components kit

380 lines 40.4 kB
import { __assign, __decorate, __metadata, __read, __spread } 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'; var CROPPER_SIZE = 200; var DEFAULT_SCALE = 1; var DEFAULT_MIN = 0.01; var DEFAULT_MAX = 5; var DEFAULT_STEP = 0.01; export var MoveTypes; (function (MoveTypes) { MoveTypes["Move"] = "move"; MoveTypes["Pinch"] = "pinch"; })(MoveTypes || (MoveTypes = {})); var MoveStart = /** @class */ (function () { function MoveStart() { this.active = false; this.type = null; this.left = 0; this.top = 0; this.clientX = 0; this.clientY = 0; } return MoveStart; }()); export { MoveStart }; var ImagePosition = /** @class */ (function () { function ImagePosition() { this.left = 0; this.top = 0; this.scale = DEFAULT_SCALE; this.width = 0; this.height = 0; } return ImagePosition; }()); export { ImagePosition }; var CropperPosition = /** @class */ (function () { function CropperPosition() { this.width = CROPPER_SIZE; this.height = CROPPER_SIZE; } return CropperPosition; }()); export { CropperPosition }; var ImageCropperComponent = /** @class */ (function () { function ImageCropperComponent(logger, cd, fb, sanitizer) { var _this = this; 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 = function () { return _this.logger.error('value accessor is not registered'); }; this.onTouched = function () { return _this.logger.error('value accessor is not registered'); }; this.registerOnChange = function (fn) { return _this.onChange = fn; }; this.registerOnTouched = function (fn) { return _this.onTouched = fn; }; this.onBlur = function () { return _this.onTouched(); }; } ImageCropperComponent_1 = ImageCropperComponent; Object.defineProperty(ImageCropperComponent.prototype, "min", { get: function () { return this._min; }, set: function (min) { this._min = min || DEFAULT_MIN; }, enumerable: true, configurable: true }); Object.defineProperty(ImageCropperComponent.prototype, "max", { get: function () { return this._max; }, set: function (max) { this._max = max || DEFAULT_MAX; }, enumerable: true, configurable: true }); Object.defineProperty(ImageCropperComponent.prototype, "step", { get: function () { return this._step; }, set: function (step) { this._step = step || DEFAULT_STEP; }, enumerable: true, configurable: true }); Object.defineProperty(ImageCropperComponent.prototype, "url", { get: function () { return this._url; }, set: function (url) { this._url = url; if (!!url) { this.moveStart = new MoveStart(); this.imagePosition = new ImagePosition(); } }, enumerable: true, configurable: true }); Object.defineProperty(ImageCropperComponent.prototype, "shape", { set: function (shape) { this._shape = shape || Shape.circle; }, enumerable: true, configurable: true }); ImageCropperComponent.prototype.ngOnInit = function () { var _this = this; this.form.valueChanges.subscribe(function (_a) { var zoom = _a.zoom; return _this.zoom(zoom); }); }; ImageCropperComponent.prototype.inView = function () { var _this = this; if (this.image.nativeElement.currentSrc.includes('image/svg')) { this.disabled = true; } this.loaded.emit(); this.sizeRetries = 0; setTimeout(function () { return _this.checkImageMaxSizeRecursively(); }); }; ImageCropperComponent.prototype.checkImageMaxSizeRecursively = function () { var _this = this; if (this.sizeRetries > 40) { this.failed.emit(); } else if (!!this.image && !!this.image.nativeElement && this.image.nativeElement.offsetWidth > 0) { var image = this.image.nativeElement; var 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; var 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(function () { return _this.checkImageMaxSizeRecursively(); }, 50); } }; ImageCropperComponent.prototype.startMove = function (event, moveType) { if (!!this.moveStart && this.moveStart.active && this.moveStart.type === MoveTypes.Pinch) { return; } if (!!event.preventDefault) { event.preventDefault(); } this.moveStart = __assign({ active: true, type: moveType, clientX: this.getClientX(event), clientY: this.getClientY(event) }, this.imagePosition); }; ImageCropperComponent.prototype.moveImg = function (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(); } }; ImageCropperComponent.prototype.moveStop = function () { if (this.moveStart.active) { this.moveStart.active = false; this.crop(); } }; ImageCropperComponent.prototype.crop = function () { var image = this.image.nativeElement; var cropper = this.cropper.nativeElement; var 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: scale }); }; ImageCropperComponent.prototype.move = function (event) { var diffX = this.getClientX(event) - this.moveStart.clientX; var diffY = this.getClientY(event) - this.moveStart.clientY; this.imagePosition.left = this.moveStart.left + diffX; this.imagePosition.top = this.moveStart.top + diffY; }; ; ImageCropperComponent.prototype.getClientX = function (event) { return (event.touches && event.touches[0] ? event.touches[0].clientX : event.clientX) || 0; }; ImageCropperComponent.prototype.getClientY = function (event) { return (event.touches && event.touches[0] ? event.touches[0].clientY : event.clientY) || 0; }; ImageCropperComponent.prototype.zoom = function (scale) { if (scale === void 0) { 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(); }; ImageCropperComponent.prototype.writeValue = function (data) { if (!!data) { this.imagePosition = __assign(__assign({}, this.imagePosition), data); } }; ImageCropperComponent.prototype.setDisabledState = function (isDisabled) { this.disabled = isDisabled; }; var ImageCropperComponent_1; ImageCropperComponent.ctorParameters = function () { return [ { 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: __spread([ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(function () { return ImageCropperComponent_1; }), multi: true } ], I18N_PROVIDERS) }), __metadata("design:paramtypes", [NGXLogger, ChangeDetectorRef, FormBuilder, DomSanitizer]) ], ImageCropperComponent); return 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"]}