UNPKG

mc-image-editor

Version:

An image editor library for magic-cut app (http://www.magic-cut.in/)

271 lines 40.7 kB
import { Component, Input, Output, ElementRef, HostListener, ViewChild, EventEmitter, ChangeDetectionStrategy, } from '@angular/core'; import * as _ from 'lodash'; import { ImageEditorService } from '../../services/image-editor.service'; export class ImageCropperComponent { constructor(editor, el) { this.editor = editor; this.el = el; this.imagePosition = { x: 0, y: 0 }; this.originalImageDimensions = { width: 0, height: 0 }; this.imageDimensions = { width: 0, height: 0 }; this.zoomConfig = { value: 1, max: 4 }; this.configChange = new EventEmitter(); } set image(value) { this.initImageCrop(value); } set zoom(value) { this.editable.ready.subscribe(() => { this.setZoom(value); }); } get zoom() { return this.zoomConfig.value; } set top(value) { this.editable.ready.subscribe(() => { this.imagePosition.y = 0; this.moveDelta = { y: 0, x: this.imagePosition.x }; this.move({ clientY: parseInt(value, 10), clientX: this.imagePosition.x, }); delete this.moveDelta; }); } get top() { return this.imagePosition.y; } set left(value) { this.editable.ready.subscribe(() => { this.imagePosition.x = 0; this.moveDelta = { x: 0, y: this.imagePosition.y }; this.move({ clientX: parseInt(value, 10), clientY: this.imagePosition.y, }); delete this.moveDelta; }); } get left() { return this.imagePosition.x; } set maxZoom(value) { this.zoomConfig.max = parseFloat(value); } get maxZoom() { return this.zoomConfig.max; } triggerMove() { window.addEventListener('mousemove', this.eventListeners.mousemove, false); window.addEventListener('mouseup', this.eventListeners.mouseup, false); return false; } initImageCrop(image) { this.editable = this.editor.edit(image); this.editable.ready.subscribe(() => { const previewDimensions = this.getComputedDimensions(this.el.nativeElement); const fittingDimensions = this.fitArea({ width: this.cropWidth, height: this.cropHeight }, { width: previewDimensions.width, height: previewDimensions.height }); const { vborder, hborder } = this.addTransparentBorder(previewDimensions, fittingDimensions); this.borders = { vborder, hborder }; const fillingDimensions = this.fillArea({ width: this.editable.image.width, height: this.editable.image.height, }, { width: fittingDimensions.width, height: fittingDimensions.height }); this.editable.canvas.width = this.cropWidth; this.editable.canvas.height = this.cropHeight; this.previewImage.nativeElement.src = this.editable.image.src; const img = this.previewImage.nativeElement; img.style.left = hborder + (fittingDimensions.width - fillingDimensions.width) / 2 + 'px'; img.style.top = vborder + (fittingDimensions.height - fillingDimensions.height) / 2 + 'px'; img.style.width = fillingDimensions.width + 'px'; img.style.height = fillingDimensions.height + 'px'; this.originalImageDimensions = fillingDimensions; this.imageDimensions.width = fillingDimensions.width; this.imageDimensions.height = fillingDimensions.height; this.cropDimensions = fittingDimensions; this.imagePosition = { x: hborder + (fittingDimensions.width - fillingDimensions.width) / 2, y: vborder + (fittingDimensions.height - fillingDimensions.height) / 2, }; }); this.eventListeners = { mousemove: this.move.bind(this), mouseup: this.stopMove.bind(this), }; } move(e) { if (!this.moveDelta) { this.moveDelta = { x: e.clientX, y: e.clientY }; } this.imagePosition.x += e.clientX - this.moveDelta.x; this.imagePosition.y += e.clientY - this.moveDelta.y; this.moveDelta.x = e.clientX; this.moveDelta.y = e.clientY; const correctedPosition = this.getCorrectedPosition(this.imagePosition); this.previewImage.nativeElement.style.left = correctedPosition.x + 'px'; this.imagePosition.x = correctedPosition.x; this.previewImage.nativeElement.style.top = correctedPosition.y + 'px'; this.imagePosition.y = correctedPosition.y; return false; } setZoom(value) { let parsedValue = parseFloat(value); if (parsedValue <= 0) { parsedValue = 1; } if (parsedValue > this.zoomConfig.max) { parsedValue = this.zoomConfig.max; } this.zoomConfig.value = parsedValue; const newDimensions = { width: this.originalImageDimensions.width * this.zoomConfig.value, height: this.originalImageDimensions.height * this.zoomConfig.value, }; const hdiff = newDimensions.width - this.imageDimensions.width; const vdiff = newDimensions.height - this.imageDimensions.height; this.imageDimensions.width = newDimensions.width; this.imageDimensions.height = newDimensions.height; const img = this.previewImage.nativeElement; img.style.width = newDimensions.width + 'px'; img.style.height = newDimensions.height + 'px'; const correctedPosition = this.getCorrectedPosition({ x: this.imagePosition.x - hdiff / 2, y: this.imagePosition.y - vdiff / 2, }); img.style.left = correctedPosition.x + 'px'; img.style.top = correctedPosition.y + 'px'; this.imagePosition.x = correctedPosition.x; this.imagePosition.y = correctedPosition.y; img.className = 'zooming'; let finishTransition; finishTransition = () => { img.className = ''; img.removeEventListener('transitionend', finishTransition); this.configChange.emit({ left: this.imagePosition.x, top: this.imagePosition.y, zoom: this.zoom, }); }; img.addEventListener('transitionend', finishTransition); } getBlob() { this.crop(); return this.editable.getBlob(); } getDataURL() { this.crop(); return this.editable.getDataURL(); } crop() { const proportions = this.imageDimensions.width / this.editable.image.width; const sx = (this.borders.hborder - this.imagePosition.x) / proportions; const sy = (this.borders.vborder - this.imagePosition.y) / proportions; const sw = this.cropDimensions.width / proportions; const sh = sw / (this.cropWidth / this.cropHeight); this.editable.apply('crop', sx, sy, sw, sh, 0, 0, this.cropWidth, this.cropHeight); } getCorrectedPosition(pos) { const corrected = { x: 0, y: 0 }; const hdiff = this.cropDimensions.width - this.imageDimensions.width; const vdiff = this.cropDimensions.height - this.imageDimensions.height; if (pos.x < this.borders.hborder) { if (pos.x < this.borders.hborder + hdiff) { corrected.x = this.borders.hborder + hdiff; } else { corrected.x = pos.x; } } else { corrected.x = this.borders.hborder; } if (pos.y < this.borders.vborder) { if (pos.y < this.borders.vborder + vdiff) { corrected.y = this.borders.vborder + vdiff; } else { corrected.y = pos.y; } } else { corrected.y = this.borders.vborder; } return corrected; } stopMove() { delete this.moveDelta; window.removeEventListener('mousemove', this.eventListeners.mousemove); window.removeEventListener('mouseup', this.eventListeners.mouseup); this.configChange.emit({ left: this.imagePosition.x, top: this.imagePosition.y, zoom: this.zoom, }); return false; } fitArea(object, area) { const ph = object.height / area.height; const pw = object.width / area.width; const scale = ph > pw ? ph : pw; return { width: object.width / scale, height: object.height / scale }; } fillArea(object, area) { const ph = object.height / area.height; const pw = object.width / area.width; const scale = ph > pw ? pw : ph; return { width: object.width / scale, height: object.height / scale }; } addTransparentBorder(previewDimensions, fittingDimensions) { const vborder = (previewDimensions.height - fittingDimensions.height) / 2; const hborder = (previewDimensions.width - fittingDimensions.width) / 2; const borderElement = _(this.el.nativeElement.childNodes).find((c) => !!c.className && c.className.indexOf('border') >= 0); const cs = window.getComputedStyle(borderElement); if (borderElement) { borderElement.style.borderBottomWidth = vborder + 'px'; borderElement.style.borderTopWidth = vborder + 'px'; borderElement.style.borderLeftWidth = hborder + 'px'; borderElement.style.borderRightWidth = hborder + 'px'; } return { vborder, hborder }; } getComputedDimensions(element) { const cs = window.getComputedStyle(element); return { width: parseInt(cs.width, 10), height: parseInt(cs.height, 10), }; } } ImageCropperComponent.decorators = [ { type: Component, args: [{ selector: 'mc-image-cropper', template: "<img #previewImage />\n<div class=\"border\"></div>\n", changeDetection: ChangeDetectionStrategy.OnPush, styles: [".border{box-sizing:border-box;height:100%;left:0;position:absolute;top:0;width:100%}:host{cursor:move;display:block;overflow:hidden;position:relative}img{position:absolute}img.zooming{transition:width .3s,height .3s,top .3s,left .3s}/deep/ .border{border:0 solid rgba(0,0,0,.3)}"] },] } ]; ImageCropperComponent.ctorParameters = () => [ { type: ImageEditorService }, { type: ElementRef } ]; ImageCropperComponent.propDecorators = { configChange: [{ type: Output }], image: [{ type: Input, args: ['src',] }], cropWidth: [{ type: Input }], cropHeight: [{ type: Input }], zoom: [{ type: Input }], top: [{ type: Input }], left: [{ type: Input }], maxZoom: [{ type: Input }], previewImage: [{ type: ViewChild, args: ['previewImage',] }], triggerMove: [{ type: HostListener, args: ['mousedown',] }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"image-cropper.component.js","sourceRoot":"","sources":["../../../../../../projects/mc-image-editor/src/lib/components/image-cropper/image-cropper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  ElementRef,\n  HostListener,\n  ViewChild,\n  EventEmitter,\n  ChangeDetectionStrategy,\n} from '@angular/core';\nimport * as _ from 'lodash';\n\nimport { ImageEditorService } from '../../services/image-editor.service';\nimport { EditableImageService } from '../../services/editable-image.service';\n\n@Component({\n  selector: 'mc-image-cropper',\n  templateUrl: './image-cropper.component.html',\n  styleUrls: ['./image-cropper.component.css'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ImageCropperComponent {\n  private borders: { vborder: number; hborder: number };\n  private moveDelta: { x: number; y: number };\n  private imagePosition: { x: number; y: number } = { x: 0, y: 0 };\n  private originalImageDimensions: Dimension = { width: 0, height: 0 };\n  private imageDimensions: Dimension = { width: 0, height: 0 };\n  private cropDimensions: Dimension;\n  private eventListeners: { mousemove: EventListener; mouseup: EventListener };\n\n  editable: EditableImageService;\n  zoomConfig = { value: 1, max: 4 };\n  img: Blob;\n\n  @Output() configChange = new EventEmitter();\n  @Input('src') set image(value: Blob) {\n    this.initImageCrop(value);\n  }\n\n  @Input() cropWidth: number;\n  @Input() cropHeight: number;\n\n  @Input() set zoom(value: any) {\n    this.editable.ready.subscribe(() => {\n      this.setZoom(value);\n    });\n  }\n  get zoom() {\n    return this.zoomConfig.value;\n  }\n\n  @Input() set top(value: any) {\n    this.editable.ready.subscribe(() => {\n      this.imagePosition.y = 0;\n      this.moveDelta = { y: 0, x: this.imagePosition.x };\n      this.move({\n        clientY: parseInt(value, 10),\n        clientX: this.imagePosition.x,\n      } as MouseEvent);\n      delete this.moveDelta;\n    });\n  }\n  get top() {\n    return this.imagePosition.y;\n  }\n\n  @Input() set left(value: any) {\n    this.editable.ready.subscribe(() => {\n      this.imagePosition.x = 0;\n      this.moveDelta = { x: 0, y: this.imagePosition.y };\n      this.move({\n        clientX: parseInt(value, 10),\n        clientY: this.imagePosition.y,\n      } as MouseEvent);\n      delete this.moveDelta;\n    });\n  }\n  get left() {\n    return this.imagePosition.x;\n  }\n\n  @Input() set maxZoom(value: any) {\n    this.zoomConfig.max = parseFloat(value);\n  }\n  get maxZoom() {\n    return this.zoomConfig.max;\n  }\n\n  @ViewChild('previewImage') previewImage: ElementRef;\n\n  @HostListener('mousedown') public triggerMove() {\n    window.addEventListener('mousemove', this.eventListeners.mousemove, false);\n    window.addEventListener('mouseup', this.eventListeners.mouseup, false);\n    return false;\n  }\n\n  constructor(private editor: ImageEditorService, private el: ElementRef) {}\n\n  initImageCrop(image: Blob) {\n    this.editable = this.editor.edit(image);\n    this.editable.ready.subscribe(() => {\n      const previewDimensions = this.getComputedDimensions(\n        this.el.nativeElement\n      );\n      const fittingDimensions = this.fitArea(\n        { width: this.cropWidth, height: this.cropHeight },\n        { width: previewDimensions.width, height: previewDimensions.height }\n      );\n\n      const { vborder, hborder } = this.addTransparentBorder(\n        previewDimensions,\n        fittingDimensions\n      );\n      this.borders = { vborder, hborder };\n\n      const fillingDimensions = this.fillArea(\n        {\n          width: this.editable.image.width,\n          height: this.editable.image.height,\n        },\n        { width: fittingDimensions.width, height: fittingDimensions.height }\n      );\n\n      this.editable.canvas.width = this.cropWidth;\n      this.editable.canvas.height = this.cropHeight;\n\n      this.previewImage.nativeElement.src = this.editable.image.src;\n      const img = this.previewImage.nativeElement;\n\n      img.style.left =\n        hborder +\n        (fittingDimensions.width - fillingDimensions.width) / 2 +\n        'px';\n      img.style.top =\n        vborder +\n        (fittingDimensions.height - fillingDimensions.height) / 2 +\n        'px';\n      img.style.width = fillingDimensions.width + 'px';\n      img.style.height = fillingDimensions.height + 'px';\n\n      this.originalImageDimensions = fillingDimensions;\n\n      this.imageDimensions.width = fillingDimensions.width;\n      this.imageDimensions.height = fillingDimensions.height;\n\n      this.cropDimensions = fittingDimensions;\n\n      this.imagePosition = {\n        x: hborder + (fittingDimensions.width - fillingDimensions.width) / 2,\n        y: vborder + (fittingDimensions.height - fillingDimensions.height) / 2,\n      };\n    });\n\n    this.eventListeners = {\n      mousemove: this.move.bind(this),\n      mouseup: this.stopMove.bind(this),\n    };\n  }\n\n  public move(e: MouseEvent) {\n    if (!this.moveDelta) {\n      this.moveDelta = { x: e.clientX, y: e.clientY };\n    }\n\n    this.imagePosition.x += e.clientX - this.moveDelta.x;\n    this.imagePosition.y += e.clientY - this.moveDelta.y;\n\n    this.moveDelta.x = e.clientX;\n    this.moveDelta.y = e.clientY;\n\n    const correctedPosition = this.getCorrectedPosition(this.imagePosition);\n\n    this.previewImage.nativeElement.style.left = correctedPosition.x + 'px';\n    this.imagePosition.x = correctedPosition.x;\n\n    this.previewImage.nativeElement.style.top = correctedPosition.y + 'px';\n    this.imagePosition.y = correctedPosition.y;\n\n    return false;\n  }\n\n  setZoom(value: string) {\n    let parsedValue = parseFloat(value);\n    if (parsedValue <= 0) {\n      parsedValue = 1;\n    }\n    if (parsedValue > this.zoomConfig.max) {\n      parsedValue = this.zoomConfig.max;\n    }\n\n    this.zoomConfig.value = parsedValue;\n\n    const newDimensions = {\n      width: this.originalImageDimensions.width * this.zoomConfig.value,\n      height: this.originalImageDimensions.height * this.zoomConfig.value,\n    };\n\n    const hdiff = newDimensions.width - this.imageDimensions.width;\n    const vdiff = newDimensions.height - this.imageDimensions.height;\n\n    this.imageDimensions.width = newDimensions.width;\n    this.imageDimensions.height = newDimensions.height;\n\n    const img = this.previewImage.nativeElement;\n\n    img.style.width = newDimensions.width + 'px';\n    img.style.height = newDimensions.height + 'px';\n\n    const correctedPosition = this.getCorrectedPosition({\n      x: this.imagePosition.x - hdiff / 2,\n      y: this.imagePosition.y - vdiff / 2,\n    });\n\n    img.style.left = correctedPosition.x + 'px';\n    img.style.top = correctedPosition.y + 'px';\n\n    this.imagePosition.x = correctedPosition.x;\n    this.imagePosition.y = correctedPosition.y;\n\n    img.className = 'zooming';\n\n    let finishTransition: EventListener;\n    finishTransition = () => {\n      img.className = '';\n      img.removeEventListener('transitionend', finishTransition);\n      this.configChange.emit({\n        left: this.imagePosition.x,\n        top: this.imagePosition.y,\n        zoom: this.zoom,\n      });\n    };\n\n    img.addEventListener('transitionend', finishTransition);\n  }\n\n  getBlob() {\n    this.crop();\n    return this.editable.getBlob();\n  }\n\n  getDataURL() {\n    this.crop();\n    return this.editable.getDataURL();\n  }\n\n  protected crop() {\n    const proportions = this.imageDimensions.width / this.editable.image.width;\n\n    const sx = (this.borders.hborder - this.imagePosition.x) / proportions;\n    const sy = (this.borders.vborder - this.imagePosition.y) / proportions;\n    const sw = this.cropDimensions.width / proportions;\n    const sh = sw / (this.cropWidth / this.cropHeight);\n\n    this.editable.apply(\n      'crop',\n      sx,\n      sy,\n      sw,\n      sh,\n      0,\n      0,\n      this.cropWidth,\n      this.cropHeight\n    );\n  }\n\n  protected getCorrectedPosition(pos: {\n    x: number;\n    y: number;\n  }): { x: number; y: number } {\n    const corrected = { x: 0, y: 0 };\n    const hdiff = this.cropDimensions.width - this.imageDimensions.width;\n    const vdiff = this.cropDimensions.height - this.imageDimensions.height;\n\n    if (pos.x < this.borders.hborder) {\n      if (pos.x < this.borders.hborder + hdiff) {\n        corrected.x = this.borders.hborder + hdiff;\n      } else {\n        corrected.x = pos.x;\n      }\n    } else {\n      corrected.x = this.borders.hborder;\n    }\n\n    if (pos.y < this.borders.vborder) {\n      if (pos.y < this.borders.vborder + vdiff) {\n        corrected.y = this.borders.vborder + vdiff;\n      } else {\n        corrected.y = pos.y;\n      }\n    } else {\n      corrected.y = this.borders.vborder;\n    }\n\n    return corrected;\n  }\n\n  protected stopMove() {\n    delete this.moveDelta;\n\n    window.removeEventListener('mousemove', this.eventListeners.mousemove);\n    window.removeEventListener('mouseup', this.eventListeners.mouseup);\n\n    this.configChange.emit({\n      left: this.imagePosition.x,\n      top: this.imagePosition.y,\n      zoom: this.zoom,\n    });\n    return false;\n  }\n\n  protected fitArea(object: Dimension, area: Dimension) {\n    const ph = object.height / area.height;\n    const pw = object.width / area.width;\n    const scale = ph > pw ? ph : pw;\n    return { width: object.width / scale, height: object.height / scale };\n  }\n\n  protected fillArea(object: Dimension, area: Dimension) {\n    const ph = object.height / area.height;\n    const pw = object.width / area.width;\n    const scale = ph > pw ? pw : ph;\n    return { width: object.width / scale, height: object.height / scale };\n  }\n\n  protected addTransparentBorder(\n    previewDimensions: Dimension,\n    fittingDimensions: Dimension\n  ) {\n    const vborder = (previewDimensions.height - fittingDimensions.height) / 2;\n    const hborder = (previewDimensions.width - fittingDimensions.width) / 2;\n\n    const borderElement = _(this.el.nativeElement.childNodes).find(\n      (c: HTMLElement) => !!c.className && c.className.indexOf('border') >= 0\n    );\n\n    const cs = window.getComputedStyle(borderElement as Element);\n\n    if (borderElement) {\n      borderElement.style.borderBottomWidth = vborder + 'px';\n      borderElement.style.borderTopWidth = vborder + 'px';\n\n      borderElement.style.borderLeftWidth = hborder + 'px';\n      borderElement.style.borderRightWidth = hborder + 'px';\n    }\n    return { vborder, hborder };\n  }\n\n  protected getComputedDimensions(element: HTMLElement) {\n    const cs = window.getComputedStyle(element);\n    return {\n      width: parseInt(cs.width as string, 10),\n      height: parseInt(cs.height as string, 10),\n    };\n  }\n}\n\nexport type Dimension = { width: number; height: number };\n"]}