@junte/ui
Version:
Quality Angular UI components kit
380 lines • 40.4 kB
JavaScript
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"]}