Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to calculate transform translate(x, y) compensation for element rotation?

I'm making a profile picture crop editor which allows for dragging, scaling and rotating an image within an area.

The dragging of the image is done by capturing the mousedown and mousemove event of the area and calculating the cursors starting and stopping x/y coordinates inside the area to get the distance which the cursor has traveled. This value is then added to or subtracted from (depending on direction) the image's current inline style transform translate(x, y) values.

var dragArea = document.getElementById('drag-area');
var photoImg = document.getElementById('photo');
var cropCircle = document.getElementById('crop-circle');
var cloneContainer = document.getElementById('clone-container');
var resetAll = document.getElementById('reset-all');
var scaleSlider = document.getElementById('scale-slider');
var scaleInput = document.getElementById('scale-input');
var scaleReset = document.getElementById('scale-reset');
var rotateSlider = document.getElementById('rotate-slider');
var rotateInput = document.getElementById('rotate-input');
var rotateReset = document.getElementById('rotate-reset');
var area = {}, photo = {
  translate: {
    x: 0, y: 0
  }, 
  transformOrigin: {
    x: 0, y: 0
  }
};

photoImg.src = photoSrc();
photoImg.style.top = cropCircle.offsetTop+'px';
photoImg.style.left = cropCircle.offsetLeft+'px';
photoImg.style.transform = 'scale(1) rotate(0deg) translate(0px, 0px)';
photoImg.style.transformOrigin = '0px 0px';
photoImg.onload = function() {
  if (this.naturalWidth < this.naturalHeight) {
    this.width = cropCircle.clientWidth;
  } else if (this.naturalWidth > this.naturalHeight) {
    this.height = cropCircle.clientHeight;
  } else {
    this.height = cropCircle.clientHeight;
    this.width = cropCircle.clientWidth;
  }
}

dragArea.onmouseenter = function() {
  this.onmousedown = function(e) {
    var transform = photoImg.style.transform;
    var photoStyle = window.getComputedStyle(photoImg);
    var photoMatrix = new DOMMatrix(photoStyle.transform);
    var transformOrigin = photoImg.style.transformOrigin.replace(/px/g, '').split(' ');
  
    photo = {
      translate: {},
      x: photoMatrix.m41,
      y: photoMatrix.m42,
      scale: Number(/scale\((-?\d+(?:\.\d*)?)\)/.exec(transform)[1]),
      rotate: Number(/rotate\((-?\d+(?:\.\d*)?)deg\)/.exec(transform)[1]),
      transformOrigin: {
        x: Number(transformOrigin[0]),
        y: Number(transformOrigin[1])
      }
    }

    area = {
      start: {
        x: e.offsetX + (e.target == cropCircle ? cropCircle.offsetLeft : 0),
        y: e.offsetY + (e.target == cropCircle ? cropCircle.offsetTop : 0)
      },
      distance: {
        x: 0,
        y: 0
      }
    };

    this.onmousemove = function(e) {
      area.end = {
        x: e.offsetX + (e.target == cropCircle ? cropCircle.offsetLeft : 0),
        y: e.offsetY + (e.target == cropCircle ? cropCircle.offsetTop : 0)
      };

      if (area.end.x > area.start.x) {
        area.distance.x = {
          type: 'positive', // right
          total: area.end.x - area.start.x
        }
      } else {
        area.distance.x = {
          type: 'negative', // left
          total: area.start.x - area.end.x
        }
      }
      if (area.end.y > area.start.y) {
        area.distance.y = {
          type: 'positive', // down
          total: area.end.y - area.start.y
        }
      } else {
        area.distance.y = {
          type: 'negative', // up
          total: area.start.y - area.end.y
        }
      }

      if (area.distance.x.type == 'positive') {
        photo.translate.x = photo.x + area.distance.x.total;
      } else {
        photo.translate.x = photo.x - area.distance.x.total;
      }
      if (area.distance.y.type == 'positive') {
        photo.translate.y = photo.y + area.distance.y.total;
      } else {
        photo.translate.y = photo.y - area.distance.y.total;
      }

      photoTransform({x: photo.translate.x, y: photo.translate.y});
    }
  }
}

dragArea.onmouseleave = function() {
  this.onmousemove = function(e) {
    e.preventDefault();
  }
}

dragArea.onmouseup = function() {
  this.onmousemove = function(e) {
    e.preventDefault();
  }
}

resetAll.onclick = function() {
  scaleSlider.value = scaleReset.value;
  scaleInput.value = scaleReset.value;
  rotateSlider.value = rotateReset.value;
  rotateInput.value = rotateReset.value;
  photo = {
    translate: {
      x: 0, y: 0
    }, 
    transformOrigin: {
      x: 0, y: 0
    }
  };
  photoTransform({scale: 1, rotate: '0', x: '0', y: '0'});
}

scaleSlider.oninput = function() {
  var value = this.value;
  scaleInput.value = value;
  photoTransform({scale: value});
}
scaleInput.oninput = function() {
  var value = this.value;
  this.value = value.length ? value : scaleReset.value;
  scaleSlider.value = this.value;
  photoTransform({scale: this.value});
}
scaleInput.onkeydown = function(e) {
  if (e.keyCode == 13) this.blur();
}
scaleInput.onblur = function() {
  var value = this.value;
  this.value = value.length ? value : scaleReset.value;
  scaleSlider.value = this.value;
  photoTransform({scale: this.value});
}
scaleReset.onclick = function() {
  scaleSlider.value = this.value;
  scaleInput.value = this.value;
  photoTransform({scale: this.value});
}

rotateSlider.oninput = function() {
  var value = this.value;
  rotateInput.value = value;
  photoTransform({rotate: value});
}
rotateInput.oninput = function() {
  var value = this.value;
  this.value = value.length ? value : rotateReset.value;
  rotateSlider.value = this.value;
  photoTransform({rotate: this.value});
}
rotateInput.onkeydown = function(e) {
  if (e.keyCode == 13) this.blur();
}
rotateInput.onblur = function() {
  var value = this.value;
  this.value = value.length ? value : rotateReset.value;
  rotateSlider.value = this.value;
  photoTransform({rotate: this.value});
}
rotateReset.onclick = function() {
  rotateSlider.value = this.value;
  rotateInput.value = this.value;
  photoTransform({rotate: this.value});
}

function photoTransform(property) {
  property = property || {};
  var transform = photoImg.style.transform;
  var axisX = property.axisX || photo.transformOrigin.x || (cropCircle.getBoundingClientRect().width / 2);
  var axisY = property.axisY || photo.transformOrigin.y || (cropCircle.getBoundingClientRect().height / 2);
  var scale = property.scale || photo.scale || Number(/scale\((-?\d+(?:\.\d*)?)\)/.exec(transform)[1]);
  var rotate = property.rotate || photo.rotate || Number(/rotate\((-?\d+(?:\.\d*)?)deg\)/.exec(transform)[1]);
  var translate = /translate\((-?\d+(?:\.\d*)?)px, (-?\d+(?:\.\d*)?)px\)/.exec(transform);
  var translateX = (property.x || photo.translate.x || Number(translate[1])) / scale;
  var translateY = (property.y || photo.translate.y || Number(translate[2])) / scale;

  photoImg.style.transformOrigin = axisX+'px '+axisY+'px';
  photoImg.style.transform = 'scale('+scale+') rotate('+rotate+'deg) translate('+translateX+'px, '+translateY+'px)';
  
  photo.transformOrigin = {
    x: axisX,
    y: axisY
  }
  photo.scale = scale;
  photo.rotate = rotate;
}

function photoSrc() {
  return '';
}
body {
  background-color: #eff1f3;
}
#profile-picture {
  width: 370px;
  height: 330px;
  margin: auto;
}
#profile-picture * {
  user-select: none;
}
#drag-area {
  width: 100%;
  height: 100%;
  cursor: move;
  cursor: grab;
  display: block;
  overflow: hidden;
  position: relative;
  background-color: #000;
  background-repeat: repeat;
  background-image: url('');
}
#drag-area:active {
  cursor: grabbing;
}
#clone-container {
  width: 0px;
  height: 0px;
  display: block;
  overflow: hidden;
  position: absolute;
}
#photo, #photo-clone {
  display: block;
  min-width: 230px;
  min-height: 230px;
  position: absolute;
  pointer-events: none;
}
img[src=''] {
  visibility: hidden;
}
#crop-circle {
  width: 230px;
  height: 230px;
  margin: 50px auto;
  overflow: hidden;
  position: relative;
  border-radius: 50%;
  box-shadow: 0 0 0 2px #fff, 0 0 0 100vw rgba(0,0,0,0.5);
}
#circle-thirds {
  top: 0;
  width: 100%;
  height: 100%;
  overflow: hidden;
  position: absolute;
  pointer-events: none;
  border-radius: 100%;
}
#circle-thirds * {
  z-index: 1;
  position: absolute;
  background-color: rgba(226,226,226,0.5);
}
#circle-thirds .top-horizontal {
  width: 100%;
  height: 1px;
  top: 33.33333%;
}
#circle-thirds .bottom-horizontal {
  width: 100%;
  height: 1px;
  top: 66.66666%;
}
#circle-thirds .left-vertical {
  height: 100%;
  width: 1px;
  left: 33.33333%;
}
#circle-thirds .right-vertical {
  height: 100%;
  width: 1px;
  left: 66.66666%;
}
.photo-options {
  width: 100%;
  display: block;
  position: relative;
  padding-top: 15px;
}
.option-buttons {
  width: 100%;
  display: flex;
  position: relative;
  padding-bottom: 10px;
  justify-content: space-between;
}
.option-buttons button {
  width: 100%;
}
.option-buttons button + button {
  margin-left: 10px;
}
.photo-options fieldset {
  margin: 0px;
}
.photo-options fieldset + fieldset {
  margin-top: 10px;
}
.option-slider {
  display: flex;
  position: relative;
}
.option-slider input[type=range] {
  width: 50%;
  flex-shrink: 0;
}
.option-slider input[type=number] {
  width: 20%;
  margin: 0 10px;
}
.option-slider button {
  width: 30%;
}
<div id="profile-picture">
  <div id="drag-area">
    <div id="clone-container"></div>
    <img id="photo" src="">
    <div id="crop-circle">
      <div id="circle-thirds">
        <span class="top-horizontal"></span>
        <span class="bottom-horizontal"></span>
        <span class="left-vertical"></span>
        <span class="right-vertical"></span>
      </div>
    </div>
  </div>
  <div class="photo-options">
    <div class="option-buttons">
      <button id="reset-all">Reset everything</button>
    </div>
    <fieldset>
      <legend>Scale</legend>
        <div class="option-slider">
        <input type="range" id="scale-slider" min="1" max="3" step="0.01" value="1">
        <input type="number" id="scale-input" min="1" max="3" step="0.01" value="1">
        <button id="scale-reset" value="1">Reset</button>
      </div>
    </fieldset>
    <fieldset>
      <legend>Rotate</legend>
      <div class="option-slider">
        <input type="range" id="rotate-slider" min="-180" max="180" step="1" value="0">
        <input type="number" id="rotate-input" min="-180" max="180" step="1" value="0">
        <button id="rotate-reset" value="0">Reset</button>
      </div>
    </fieldset>
  </div>
</div>

The problem is that when the image is rotated its translate(x y) values no longer corresponds relative to the area's x/y coordinates.

I've found a couple of examples on how to calculate the x/y coordinates of the four corners of a rotated square or rectangle using the radians of the rotated angle and cos and sin. But because geometry isn't my strong suit I don't know how these calculations would apply to the translate(x, y) values of the image.

Here is a pen of the how the image cropper currently works: https://codepen.io/ClubAce/pen/maNJNZ

The image is supposed to be dragged along the x/y axis of the area regardless of the image's rotation.

I really hope someone can help me figure out how the script can be modified to produce the desired dragging behaviour.

Thx.

like image 488
Ace Avatar asked Jan 21 '19 11:01

Ace


1 Answers

You simply need to change the order from this:

 photoImg.style.transform = 'scale('+scale+') rotate('+rotate+'deg)  translate('+translateX+'px, '+translateY+'px) ';

to this

photoImg.style.transform = 'scale('+scale+') translate('+translateX+'px, '+translateY+'px) rotate('+rotate+'deg) ';

Order is very important when using Transform

Full code:

var dragArea = document.getElementById('drag-area');
var photoImg = document.getElementById('photo');
var cropCircle = document.getElementById('crop-circle');
var cloneContainer = document.getElementById('clone-container');
var resetAll = document.getElementById('reset-all');
var scaleSlider = document.getElementById('scale-slider');
var scaleInput = document.getElementById('scale-input');
var scaleReset = document.getElementById('scale-reset');
var rotateSlider = document.getElementById('rotate-slider');
var rotateInput = document.getElementById('rotate-input');
var rotateReset = document.getElementById('rotate-reset');
var area = {}, photo = {
  translate: {
    x: 0, y: 0
  }, 
  transformOrigin: {
    x: 0, y: 0
  }
};

photoImg.src = photoSrc();
photoImg.style.top = cropCircle.offsetTop+'px';
photoImg.style.left = cropCircle.offsetLeft+'px';
photoImg.style.transform = 'scale(1) rotate(0deg) translate(0px, 0px)';
photoImg.style.transformOrigin = '0px 0px';
photoImg.onload = function() {
  if (this.naturalWidth < this.naturalHeight) {
    this.width = cropCircle.clientWidth;
  } else if (this.naturalWidth > this.naturalHeight) {
    this.height = cropCircle.clientHeight;
  } else {
    this.height = cropCircle.clientHeight;
    this.width = cropCircle.clientWidth;
  }
}

dragArea.onmouseenter = function() {
  this.onmousedown = function(e) {
    var transform = photoImg.style.transform;
    var photoStyle = window.getComputedStyle(photoImg);
    var photoMatrix = new DOMMatrix(photoStyle.transform);
    var transformOrigin = photoImg.style.transformOrigin.replace(/px/g, '').split(' ');
  
    photo = {
      translate: {},
      x: photoMatrix.m41,
      y: photoMatrix.m42,
      scale: Number(/scale\((-?\d+(?:\.\d*)?)\)/.exec(transform)[1]),
      rotate: Number(/rotate\((-?\d+(?:\.\d*)?)deg\)/.exec(transform)[1]),
      transformOrigin: {
        x: Number(transformOrigin[0]),
        y: Number(transformOrigin[1])
      }
    }

    area = {
      start: {
        x: e.offsetX + (e.target == cropCircle ? cropCircle.offsetLeft : 0),
        y: e.offsetY + (e.target == cropCircle ? cropCircle.offsetTop : 0)
      },
      distance: {
        x: 0,
        y: 0
      }
    };

    this.onmousemove = function(e) {
      area.end = {
        x: e.offsetX + (e.target == cropCircle ? cropCircle.offsetLeft : 0),
        y: e.offsetY + (e.target == cropCircle ? cropCircle.offsetTop : 0)
      };

      if (area.end.x > area.start.x) {
        area.distance.x = {
          type: 'positive', // right
          total: area.end.x - area.start.x
        }
      } else {
        area.distance.x = {
          type: 'negative', // left
          total: area.start.x - area.end.x
        }
      }
      if (area.end.y > area.start.y) {
        area.distance.y = {
          type: 'positive', // down
          total: area.end.y - area.start.y
        }
      } else {
        area.distance.y = {
          type: 'negative', // up
          total: area.start.y - area.end.y
        }
      }

      if (area.distance.x.type == 'positive') {
        photo.translate.x = photo.x + area.distance.x.total;
      } else {
        photo.translate.x = photo.x - area.distance.x.total;
      }
      if (area.distance.y.type == 'positive') {
        photo.translate.y = photo.y + area.distance.y.total;
      } else {
        photo.translate.y = photo.y - area.distance.y.total;
      }

      photoTransform({x: photo.translate.x, y: photo.translate.y});
    }
  }
}

dragArea.onmouseleave = function() {
  this.onmousemove = function(e) {
    e.preventDefault();
  }
}

dragArea.onmouseup = function() {
  this.onmousemove = function(e) {
    e.preventDefault();
  }
}

resetAll.onclick = function() {
  scaleSlider.value = scaleReset.value;
  scaleInput.value = scaleReset.value;
  rotateSlider.value = rotateReset.value;
  rotateInput.value = rotateReset.value;
  photo = {
    translate: {
      x: 0, y: 0
    }, 
    transformOrigin: {
      x: 0, y: 0
    }
  };
  photoTransform({scale: 1, rotate: '0', x: '0', y: '0'});
}

scaleSlider.oninput = function() {
  var value = this.value;
  scaleInput.value = value;
  photoTransform({scale: value});
}
scaleInput.oninput = function() {
  var value = this.value;
  this.value = value.length ? value : scaleReset.value;
  scaleSlider.value = this.value;
  photoTransform({scale: this.value});
}
scaleInput.onkeydown = function(e) {
  if (e.keyCode == 13) this.blur();
}
scaleInput.onblur = function() {
  var value = this.value;
  this.value = value.length ? value : scaleReset.value;
  scaleSlider.value = this.value;
  photoTransform({scale: this.value});
}
scaleReset.onclick = function() {
  scaleSlider.value = this.value;
  scaleInput.value = this.value;
  photoTransform({scale: this.value});
}

rotateSlider.oninput = function() {
  var value = this.value;
  rotateInput.value = value;
  photoTransform({rotate: value});
}
rotateInput.oninput = function() {
  var value = this.value;
  this.value = value.length ? value : rotateReset.value;
  rotateSlider.value = this.value;
  photoTransform({rotate: this.value});
}
rotateInput.onkeydown = function(e) {
  if (e.keyCode == 13) this.blur();
}
rotateInput.onblur = function() {
  var value = this.value;
  this.value = value.length ? value : rotateReset.value;
  rotateSlider.value = this.value;
  photoTransform({rotate: this.value});
}
rotateReset.onclick = function() {
  rotateSlider.value = this.value;
  rotateInput.value = this.value;
  photoTransform({rotate: this.value});
}

function photoTransform(property) {
  property = property || {};
  var transform = photoImg.style.transform;
  var axisX = property.axisX || photo.transformOrigin.x || (cropCircle.getBoundingClientRect().width / 2);
  var axisY = property.axisY || photo.transformOrigin.y || (cropCircle.getBoundingClientRect().height / 2);
  var scale = property.scale || photo.scale || Number(/scale\((-?\d+(?:\.\d*)?)\)/.exec(transform)[1]);
  var rotate = property.rotate || photo.rotate || Number(/rotate\((-?\d+(?:\.\d*)?)deg\)/.exec(transform)[1]);
  var translate = /translate\((-?\d+(?:\.\d*)?)px, (-?\d+(?:\.\d*)?)px\)/.exec(transform);
  var translateX = (property.x || photo.translate.x || Number(translate[1])) / scale;
  var translateY = (property.y || photo.translate.y || Number(translate[2])) / scale;

  photoImg.style.transformOrigin = axisX+'px '+axisY+'px';
  photoImg.style.transform = 'scale('+scale+') translate('+translateX+'px, '+translateY+'px) rotate('+rotate+'deg) ';
  
  photo.transformOrigin = {
    x: axisX,
    y: axisY
  }
  photo.scale = scale;
  photo.rotate = rotate;
}

function photoSrc() {
  return '';
}
body {
  background-color: #eff1f3;
}
#profile-picture {
  width: 370px;
  height: 330px;
  margin: auto;
}
#profile-picture * {
  user-select: none;
}
#drag-area {
  width: 100%;
  height: 100%;
  cursor: move;
  cursor: grab;
  display: block;
  overflow: hidden;
  position: relative;
  background-color: #000;
  background-repeat: repeat;
  background-image: url('');
}
#drag-area:active {
  cursor: grabbing;
}
#clone-container {
  width: 0px;
  height: 0px;
  display: block;
  overflow: hidden;
  position: absolute;
}
#photo, #photo-clone {
  display: block;
  min-width: 230px;
  min-height: 230px;
  position: absolute;
  pointer-events: none;
}
img[src=''] {
  visibility: hidden;
}
#crop-circle {
  width: 230px;
  height: 230px;
  margin: 50px auto;
  overflow: hidden;
  position: relative;
  border-radius: 50%;
  box-shadow: 0 0 0 2px #fff, 0 0 0 100vw rgba(0,0,0,0.5);
}
#circle-thirds {
  top: 0;
  width: 100%;
  height: 100%;
  overflow: hidden;
  position: absolute;
  pointer-events: none;
  border-radius: 100%;
}
#circle-thirds * {
  z-index: 1;
  position: absolute;
  background-color: rgba(226,226,226,0.5);
}
#circle-thirds .top-horizontal {
  width: 100%;
  height: 1px;
  top: 33.33333%;
}
#circle-thirds .bottom-horizontal {
  width: 100%;
  height: 1px;
  top: 66.66666%;
}
#circle-thirds .left-vertical {
  height: 100%;
  width: 1px;
  left: 33.33333%;
}
#circle-thirds .right-vertical {
  height: 100%;
  width: 1px;
  left: 66.66666%;
}
.photo-options {
  width: 100%;
  display: block;
  position: relative;
  padding-top: 15px;
}
.option-buttons {
  width: 100%;
  display: flex;
  position: relative;
  padding-bottom: 10px;
  justify-content: space-between;
}
.option-buttons button {
  width: 100%;
}
.option-buttons button + button {
  margin-left: 10px;
}
.photo-options fieldset {
  margin: 0px;
}
.photo-options fieldset + fieldset {
  margin-top: 10px;
}
.option-slider {
  display: flex;
  position: relative;
}
.option-slider input[type=range] {
  width: 50%;
  flex-shrink: 0;
}
.option-slider input[type=number] {
  width: 20%;
  margin: 0 10px;
}
.option-slider button {
  width: 30%;
}
<div id="profile-picture">
  <div id="drag-area">
    <div id="clone-container"></div>
    <img id="photo" src="">
    <div id="crop-circle">
      <div id="circle-thirds">
        <span class="top-horizontal"></span>
        <span class="bottom-horizontal"></span>
        <span class="left-vertical"></span>
        <span class="right-vertical"></span>
      </div>
    </div>
  </div>
  <div class="photo-options">
    <div class="option-buttons">
      <button id="reset-all">Reset everything</button>
    </div>
    <fieldset>
      <legend>Scale</legend>
        <div class="option-slider">
        <input type="range" id="scale-slider" min="1" max="3" step="0.01" value="1">
        <input type="number" id="scale-input" min="1" max="3" step="0.01" value="1">
        <button id="scale-reset" value="1">Reset</button>
      </div>
    </fieldset>
    <fieldset>
      <legend>Rotate</legend>
      <div class="option-slider">
        <input type="range" id="rotate-slider" min="-180" max="180" step="1" value="0">
        <input type="number" id="rotate-input" min="-180" max="180" step="1" value="0">
        <button id="rotate-reset" value="0">Reset</button>
      </div>
    </fieldset>
  </div>
</div>
like image 123
Temani Afif Avatar answered Nov 01 '22 08:11

Temani Afif