├── .gitignore
├── LICENSE
├── README.md
├── bower.json
├── dist
├── cropper.css
└── cropper.js
├── examples
├── demo.html
├── orientation_1.JPG
├── orientation_3.JPG
├── orientation_6.JPG
└── orientation_8.JPG
├── package.json
└── src
├── cropper.js
├── index.js
└── util.js
/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 |
5 | # Runtime data
6 | pids
7 | *.pid
8 | *.seed
9 |
10 | # Directory for instrumented libs generated by jscoverage/JSCover
11 | lib-cov
12 |
13 | # Coverage directory used by tools like istanbul
14 | coverage
15 |
16 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
17 | .grunt
18 |
19 | # node-waf configuration
20 | .lock-wscript
21 |
22 | # Compiled binary addons (http://nodejs.org/api/addons.html)
23 | build/Release
24 | # dist/*.js
25 |
26 | # Dependency directory
27 | # https://www.npmjs.org/doc/misc/npm-faq.html#should-i-check-my-node_modules-folder-into-git
28 | node_modules
29 |
30 | # IDE
31 | .idea
32 | demo/**/*.bundle.js
33 |
34 | # Prototype File
35 | prototype
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2015 饿了么前端
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
23 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # image-cropper-touch
2 | A image cropper for mobile device.
3 |
4 | # requirements
5 |
6 | blueimp-load-image is required.
7 |
8 | ```Bash
9 | npm install blueimp-load-image
10 | ```
11 |
12 | ```HTML
13 |
14 | ```
15 |
16 | # Example
17 |
18 | View example in folder examples/demo.html.
19 |
20 | # License
21 | MIT
--------------------------------------------------------------------------------
/bower.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "image-cropper-touch",
3 | "main": "dist/cropper.js",
4 | "version": "0.1.2",
5 | "authors": [
6 | "long.zhang "
7 | ],
8 | "license": "MIT",
9 | "ignore": [
10 | "demo",
11 | "test",
12 | "examples"
13 | ]
14 | }
--------------------------------------------------------------------------------
/dist/cropper.css:
--------------------------------------------------------------------------------
1 | .cropper {
2 | position: relative;
3 | overflow: hidden;
4 | }
5 |
6 | .cropper .crop-box {
7 | position: absolute;
8 | left: 50%;
9 | top: 50%;
10 | -webkit-transform: translate3d(-50%, -50%, 0);
11 | transform: translate3d(-50%, -50%, 0);
12 | box-sizing: border-box;
13 | border: 1px solid #ddd;
14 | z-index: 101;
15 | }
16 |
17 | .cropper .cover {
18 | position: absolute;
19 | background: rgba(0,0,0,0.5);
20 | width: 100%;
21 | z-index: 100;
22 | }
23 |
24 | .cropper .cover-start {
25 | top: 0;
26 | }
27 |
28 | .cropper .cover-end {
29 | bottom: 0;
30 | }
31 |
32 | .cropper-horizontal .cover {
33 | height: 100%;
34 | }
35 |
36 | .cropper-horizontal .cover-start {
37 | left: 0;
38 | }
39 |
40 | .cropper-horizontal .cover-end {
41 | right: 0;
42 | }
43 |
44 | .cropper img {
45 | max-width: none;
46 | }
--------------------------------------------------------------------------------
/dist/cropper.js:
--------------------------------------------------------------------------------
1 | (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o -1) {
53 | originalWidth = image.height;
54 | originalHeight = image.width;
55 | } else {
56 | originalWidth = image.width;
57 | originalHeight = image.height;
58 | }
59 |
60 | self.imageState.width = originalWidth;
61 | self.imageState.height = originalHeight;
62 |
63 | self.initScale();
64 |
65 | var minScale = self.scaleRange[0];
66 | var imageWidth = minScale * originalWidth;
67 | var imageHeight = minScale * originalHeight;
68 | selfImage.style.width = imageWidth + 'px';
69 | selfImage.style.height = imageHeight + 'px';
70 |
71 | var imageLeft, imageTop;
72 |
73 | var cropBoxRect = self.cropBoxRect;
74 |
75 | if (originalWidth > originalHeight) {
76 | imageLeft = (cropBoxRect.width - imageWidth) / 2 +cropBoxRect.left;
77 | imageTop = cropBoxRect.top;
78 | } else {
79 | imageLeft = cropBoxRect.left;
80 | imageTop = (cropBoxRect.height - imageHeight) / 2 + cropBoxRect.top;
81 | }
82 |
83 | self.moveImage(imageLeft, imageTop);
84 |
85 | self.imageLoading = false;
86 | });
87 | };
88 | image.src = url || src;
89 | },
90 |
91 | getFocalPoint: function(event) {
92 | var focalPoint = {
93 | left: (event.touches[0].pageX + event.touches[1].pageX) / 2,
94 | top: (event.touches[0].pageY + event.touches[1].pageY) / 2
95 | };
96 |
97 | var imageState = this.imageState;
98 | var cropBoxRect = this.cropBoxRect;
99 |
100 | focalPoint.left -= cropBoxRect.left + imageState.left;
101 | focalPoint.top -= cropBoxRect.top + imageState.top;
102 |
103 | return focalPoint;
104 | },
105 |
106 | render: function(parentNode) {
107 | var element = document.createElement('div');
108 | element.className = 'cropper';
109 |
110 | var coverStart = document.createElement('div');
111 | var coverEnd = document.createElement('div');
112 | var cropBox = document.createElement('div');
113 | var image = document.createElement('img');
114 |
115 | coverStart.className = 'cover cover-start';
116 | coverEnd.className = 'cover cover-end';
117 | cropBox.className = 'crop-box';
118 |
119 | element.appendChild(coverStart);
120 | element.appendChild(coverEnd);
121 | element.appendChild(cropBox);
122 | element.appendChild(image);
123 |
124 | this.refs = {
125 | element: element,
126 | coverStart: coverStart,
127 | coverEnd: coverEnd,
128 | cropBox: cropBox,
129 | image: image
130 | };
131 |
132 | if (parentNode) {
133 | parentNode.appendChild(element);
134 | }
135 |
136 | if (element.offsetHeight > 0) {
137 | this.resetSize();
138 | }
139 |
140 | this.bindEvents();
141 | },
142 |
143 | initScale: function () {
144 | var cropBoxRect = this.cropBoxRect;
145 | var width = this.imageState.width;
146 | var height = this.imageState.height;
147 | var scale, minScale;
148 |
149 | if (width > height) {
150 | scale = this.imageState.scale = cropBoxRect.height / height;
151 | minScale = cropBoxRect.height * 0.8 / height;
152 | } else {
153 | scale = this.imageState.scale = cropBoxRect.width / width;
154 | minScale = cropBoxRect.width * 0.8 / width;
155 | }
156 |
157 | this.scaleRange = [scale, 2];
158 | this.bounceScaleRange = [minScale, 3];
159 | },
160 |
161 | resetSize: function() {
162 | var refs = this.refs;
163 | if (!refs) return;
164 |
165 | var element = refs.element;
166 | var cropBox = refs.cropBox;
167 | var coverStart = refs.coverStart;
168 | var coverEnd = refs.coverEnd;
169 |
170 | var width = element.offsetWidth;
171 | var height = element.offsetHeight;
172 |
173 | if (width > height) {
174 | element.className = 'cropper cropper-horizontal';
175 |
176 | coverStart.style.width = coverEnd.style.width = (width - height) / 2 + 'px';
177 | coverStart.style.height = coverEnd.style.height = '';
178 | cropBox.style.width = cropBox.style.height = height + 'px';
179 | } else {
180 | element.className = 'cropper';
181 |
182 | coverStart.style.height = coverEnd.style.height = (height - width) / 2 + 'px';
183 | coverStart.style.width = coverEnd.style.width = '';
184 | cropBox.style.width = cropBox.style.height = width + 'px';
185 | }
186 |
187 | var elementRect = element.getBoundingClientRect();
188 | var cropBoxRect = cropBox.getBoundingClientRect();
189 |
190 | this.cropBoxRect = {
191 | left: cropBoxRect.left - elementRect.left,
192 | top: cropBoxRect.top - elementRect.top,
193 | width: cropBoxRect.width,
194 | height: cropBoxRect.height
195 | };
196 |
197 | this.initScale();
198 |
199 | this.checkBounce(0);
200 | },
201 |
202 | checkBounce: function (speed) {
203 | var imageState = this.imageState;
204 | var cropBoxRect = this.cropBoxRect;
205 |
206 | var imageWidth = imageState.width;
207 | var imageHeight = imageState.height;
208 | var imageScale = imageState.scale;
209 |
210 | var imageOffset = getElementTranslate(this.refs.image);
211 | var left = imageOffset.left;
212 | var top = imageOffset.top;
213 |
214 | var leftRange = [-imageWidth * imageScale + cropBoxRect.width + cropBoxRect.left, cropBoxRect.left];
215 | var topRange = [-imageHeight * imageScale + cropBoxRect.height + cropBoxRect.top, cropBoxRect.top];
216 |
217 | var overflow = false;
218 |
219 | if (left < leftRange[0]) {
220 | left = leftRange[0];
221 | overflow = true;
222 | } else if (left > leftRange[1]) {
223 | left = leftRange[1];
224 | overflow = true;
225 | }
226 |
227 | if (top < topRange[0]) {
228 | top = topRange[0];
229 | overflow = true;
230 | } else if (top > topRange[1]) {
231 | top = topRange[1];
232 | overflow = true;
233 | }
234 |
235 | if (overflow) {
236 | var self = this;
237 | translate(this.refs.image, left, top, speed === undefined ? 200 : 0, function() {
238 | self.moveImage(left, top);
239 | });
240 | }
241 | },
242 |
243 | moveImage: function(left, top) {
244 | var image = this.refs.image;
245 | translateElement(image, left, top);
246 |
247 | this.imageState.left = left;
248 | this.imageState.top = top;
249 | },
250 |
251 | onTouchStart: function(event) {
252 | this.amplitude = 0;
253 | var image = this.refs.image;
254 |
255 | var fingerCount = event.touches.length;
256 | if (fingerCount) {
257 | var touchEvent = event.touches[0];
258 |
259 | var imageOffset = getElementTranslate(image);
260 |
261 | this.dragState = {
262 | timestamp: Date.now(),
263 | startTouchLeft: touchEvent.pageX,
264 | startTouchTop: touchEvent.pageY,
265 | startLeft: imageOffset.left || 0,
266 | startTop: imageOffset.top || 0
267 | };
268 | }
269 |
270 | if (fingerCount >= 2) {
271 | var zoomState = this.zoomState = {
272 | timestamp: Date.now()
273 | };
274 |
275 | zoomState.startDistance = getDistance(event);
276 | zoomState.focalPoint = this.getFocalPoint(event);
277 | }
278 | },
279 |
280 | onTouchMove: function(event) {
281 | var fingerCount = event.touches.length;
282 |
283 | var touchEvent = event.touches[0];
284 |
285 | var cropBoxRect = this.cropBoxRect;
286 | var image = this.refs.image;
287 |
288 | var imageState = this.imageState;
289 | var imageWidth = imageState.width;
290 | var imageHeight = imageState.height;
291 |
292 | var dragState = this.dragState;
293 | var zoomState = this.zoomState;
294 |
295 | if (fingerCount === 1) {
296 | var leftRange = [ -imageWidth * imageState.scale + cropBoxRect.width, cropBoxRect.left ];
297 | var topRange = [ -imageHeight * imageState.scale + cropBoxRect.height + cropBoxRect.top, cropBoxRect.top ];
298 |
299 | var deltaX = touchEvent.pageX - (dragState.lastLeft || dragState.startTouchLeft);
300 | var deltaY = touchEvent.pageY - (dragState.lastTop || dragState.startTouchTop);
301 |
302 | var imageOffset = getElementTranslate(image);
303 |
304 | var left = imageOffset.left + deltaX;
305 | var top = imageOffset.top + deltaY;
306 |
307 | if (left < leftRange[0] || left > leftRange[1]) {
308 | left -= deltaX / 2;
309 | }
310 |
311 | if (top < topRange [0] || top > topRange[1]) {
312 | top -= deltaY / 2;
313 | }
314 |
315 | this.moveImage(left, top);
316 | } else if (fingerCount >= 2) {
317 | if (!zoomState.timestamp) {
318 | zoomState = {
319 | timestamp: Date.now()
320 | };
321 |
322 | zoomState.startDistance = getDistance(event);
323 | zoomState.focalPoint = this.getFocalPoint(event);
324 |
325 | return;
326 | }
327 |
328 | var newDistance = getDistance(event);
329 | var oldScale = imageState.scale;
330 |
331 | imageState.scale = oldScale * newDistance / (zoomState.lastDistance || zoomState.startDistance);
332 |
333 | var scaleRange = this.scaleRange;
334 | if (imageState.scale < scaleRange[0]) {
335 | imageState.scale = scaleRange[0];
336 | } else if (imageState.scale > scaleRange[1]) {
337 | imageState.scale = scaleRange[1];
338 | }
339 |
340 | this.zoomWithFocal(oldScale);
341 |
342 | zoomState.focalPoint = this.getFocalPoint(event);
343 | zoomState.lastDistance = newDistance;
344 | }
345 |
346 | dragState.lastLeft = touchEvent.pageX;
347 | dragState.lastTop = touchEvent.pageY;
348 | },
349 |
350 | onTouchEnd: function(event) {
351 | var imageState = this.imageState;
352 | var zoomState = this.zoomState;
353 | var dragState = this.dragState;
354 | var amplitude = this.amplitude;
355 | var imageWidth = imageState.width;
356 | var imageHeight = imageState.height;
357 | var cropBoxRect = this.cropBoxRect;
358 |
359 | if (event.touches.length === 0 && dragState.timestamp) {
360 | var self = this;
361 | var duration = Date.now() - dragState.timestamp;
362 |
363 | if (duration > 300) {
364 | self.checkBounce();
365 | } else {
366 | var target;
367 |
368 | var top = imageState.top;
369 | var left = imageState.left;
370 |
371 | var momentumVertical = false;
372 |
373 | var timeConstant = 160;
374 |
375 | var autoScroll = function () {
376 | var elapsed, delta;
377 |
378 | if (amplitude) {
379 | elapsed = Date.now() - timestamp;
380 | delta = -amplitude * Math.exp(-elapsed / timeConstant);
381 | if (delta > 0.5 || delta < -0.5) {
382 | if (momentumVertical) {
383 | self.moveImage(left, target + delta);
384 | } else {
385 | self.moveImage(target + delta, top);
386 | }
387 |
388 | requestAnimationFrame(autoScroll);
389 | } else {
390 | var currentLeft;
391 | var currentTop;
392 |
393 | if (momentumVertical) {
394 | currentLeft = left;
395 | currentTop = target;
396 | } else {
397 | currentLeft = target;
398 | currentTop = top;
399 | }
400 |
401 | self.moveImage(currentLeft, currentTop);
402 | self.checkBounce();
403 | }
404 | }
405 | };
406 |
407 | var velocity;
408 |
409 | var deltaX = event.changedTouches[0].pageX - dragState.startTouchLeft;
410 | var deltaY = event.changedTouches[0].pageY - dragState.startTouchTop;
411 |
412 | if (Math.abs(deltaX) > Math.abs(deltaY)) {
413 | velocity = deltaX / duration;
414 | } else {
415 | momentumVertical = true;
416 | velocity = deltaY / duration;
417 | }
418 |
419 | amplitude = 80 * velocity;
420 |
421 | var range;
422 |
423 | if (momentumVertical) {
424 | target = Math.round(imageState.top + amplitude);
425 | range = [-imageHeight * imageState.scale + cropBoxRect.height / 2 + cropBoxRect.top, cropBoxRect.top + cropBoxRect.height / 2];
426 | } else {
427 | target = Math.round(imageState.left + amplitude);
428 | range = [-imageWidth * imageState.scale + cropBoxRect.width / 2, cropBoxRect.left + cropBoxRect.width / 2];
429 | }
430 |
431 | if (target < range[0]) {
432 | target = range[0];
433 | amplitude /= 2;
434 | } else if (target > range[1]) {
435 | target = range[1];
436 | amplitude /= 2;
437 | }
438 |
439 | var timestamp = Date.now();
440 | requestAnimationFrame(autoScroll);
441 | }
442 |
443 | this.dragState = {};
444 | } else if (zoomState.timestamp) {
445 | this.checkBounce();
446 |
447 | this.zoomState = {};
448 | }
449 | },
450 |
451 | zoomWithFocal: function(oldScale) {
452 | var image = this.refs.image;
453 | var imageState = this.imageState;
454 | var imageScale = imageState.scale;
455 |
456 | image.style.width = imageState.width * imageScale + 'px';
457 | image.style.height = imageState.height * imageScale + 'px';
458 |
459 | var focalPoint = this.zoomState.focalPoint;
460 |
461 | var offsetLeft = (focalPoint.left / imageScale - focalPoint.left / oldScale) * imageScale;
462 | var offsetTop = (focalPoint.top / imageScale - focalPoint.top / oldScale) * imageScale;
463 |
464 | var imageLeft = imageState.left || 0;
465 | var imageTop = imageState.top || 0;
466 |
467 | this.moveImage(imageLeft + offsetLeft, imageTop + offsetTop);
468 | },
469 |
470 | bindEvents: function() {
471 | var cropBox = this.refs.cropBox;
472 |
473 | cropBox.addEventListener('touchstart', this.onTouchStart.bind(this));
474 |
475 | cropBox.addEventListener('touchmove', this.onTouchMove.bind(this));
476 |
477 | cropBox.addEventListener('touchend', this.onTouchEnd.bind(this));
478 | },
479 |
480 | createBase64: function (callback, width) {
481 | var imageState = this.imageState;
482 | var cropBoxRect = this.cropBoxRect;
483 | var scale = imageState.scale;
484 |
485 | var canvasSize = width;
486 |
487 | if (!canvasSize) {
488 | canvasSize = cropBoxRect.width * 2;
489 | }
490 |
491 | var imageLeft = Math.round((cropBoxRect.left - imageState.left) / scale);
492 | var imageTop = Math.round((cropBoxRect.top - imageState.top) / scale);
493 | var imageSize = Math.floor(cropBoxRect.width / scale);
494 |
495 | var orientation = this.orientation;
496 | var image = this.refs.image;
497 |
498 | var cropImage = new Image();
499 | cropImage.src = image.src;
500 |
501 | cropImage.onload = function() {
502 | var resultCanvas = loadImage.scale(cropImage, {
503 | canvas: true,
504 | left: imageLeft,
505 | top: imageTop,
506 | sourceWidth: imageSize,
507 | sourceHeight: imageSize,
508 | orientation: orientation,
509 | maxWidth: canvasSize,
510 | maxHeight: canvasSize
511 | });
512 |
513 | var dataURL = resultCanvas.toDataURL();
514 | if (typeof callback === 'function') {
515 | callback({
516 | canvasSize: canvasSize,
517 | canvas: resultCanvas,
518 | dataURL: dataURL
519 | });
520 | }
521 | };
522 | },
523 |
524 | getCroppedImage: function(callback, width) {
525 | if (!this.image) return null;
526 |
527 | this.createBase64(function(result) {
528 | var canvasSize = result.canvasSize;
529 | var canvas = result.canvas;
530 | var dataURL = result.dataURL;
531 |
532 | if (typeof callback === 'function') {
533 | callback({
534 | file: canvas.toBlob ? canvas.toBlob() : dataURItoBlob(dataURL),
535 | dataUrl: dataURL,
536 | oDataURL: result.oDataURL,
537 | size: canvasSize
538 | });
539 | }
540 | }, width);
541 | }
542 | };
543 |
544 | module.exports = Cropper;
545 | },{"./util":3}],2:[function(require,module,exports){
546 | window.Cropper = require('./cropper');
547 | },{"./cropper":1}],3:[function(require,module,exports){
548 |
549 | var once = function(el, event, fn) {
550 | var listener = function() {
551 | if (fn) {
552 | fn.apply(this, arguments);
553 | }
554 | el.removeEventListener(event, listener);
555 | };
556 | el.addEventListener(event, listener);
557 | };
558 |
559 | module.exports = {
560 | dataURItoBlob: function (dataURI) {
561 | var binaryString = atob(dataURI.split(',')[1]);
562 | var arrayBuffer = new ArrayBuffer(binaryString.length);
563 | var intArray = new Uint8Array(arrayBuffer);
564 |
565 | for (var i = 0, j = binaryString.length; i < j; i++) {
566 | intArray[i] = binaryString.charCodeAt(i);
567 | }
568 |
569 | var data = [intArray];
570 | var type = 'image/png';
571 |
572 | var result;
573 |
574 | try {
575 | result = new Blob(data, { type: type });
576 | } catch(error) {
577 | // TypeError old chrome and FF
578 | window.BlobBuilder = window.BlobBuilder ||
579 | window.WebKitBlobBuilder ||
580 | window.MozBlobBuilder ||
581 | window.MSBlobBuilder;
582 |
583 | if(error.name == 'TypeError' && window.BlobBuilder){
584 | var builder = new BlobBuilder();
585 | builder.append(arrayBuffer);
586 | result = builder.getBlob(type);
587 | }
588 | }
589 |
590 | return result;
591 | },
592 | getTouchDistance: function(event) {
593 | var finger = event.touches[0];
594 | var finger2 = event.touches[1];
595 |
596 | var c1 = Math.abs(finger.pageX - finger2.pageX);
597 | var c2 = Math.abs(finger.pageY - finger2.pageY);
598 |
599 | return Math.sqrt( c1 * c1 + c2 * c2 );
600 | },
601 | translate: function(element, left, top, speed, callback) {
602 | element.style.webkitTransform = 'translate3d(' + (left || 0) + 'px, ' + (top || 0) + 'px, 0)';
603 | if (speed) {
604 | var called = false;
605 |
606 | var realCallback = function() {
607 | if (called) return;
608 | element.style.webkitTransition = '';
609 | called = true;
610 | if (callback) {
611 | callback.apply(this, arguments);
612 | }
613 | };
614 | element.style.webkitTransition = '-webkit-transform ' + speed + 'ms cubic-bezier(0.325, 0.770, 0.000, 1.000)';
615 | once(element, 'webkitTransitionEnd', realCallback);
616 | once(element, 'transitionend', realCallback);
617 | // for android...
618 | setTimeout(realCallback, speed + 50);
619 | } else {
620 | element.style.webkitTransition = '';
621 | }
622 | },
623 | translateElement: function(element, left, top) {
624 | element.style.webkitTransform = 'translate3d(' + (left || 0) + 'px, ' + (top || 0) + 'px, 0)';
625 | },
626 | getElementTranslate: function(element) {
627 | var transform = element.style.webkitTransform;
628 | var matches = /translate3d\((.*?)\)/ig.exec(transform);
629 | if (matches) {
630 | var translates = matches[1].split(',');
631 | return {
632 | left: parseInt(translates[0], 10),
633 | top: parseInt(translates[1], 10)
634 | }
635 | }
636 | return {
637 | left: 0,
638 | top: 0
639 | }
640 | }
641 | };
642 | },{}]},{},[2])
643 | //# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/watchify/node_modules/browserify/node_modules/browser-pack/_prelude.js","src/cropper.js","src/index.js","src/util.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9hBA;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var util = require('./util');\n\nvar translateElement = util.translateElement;\nvar getElementTranslate = util.getElementTranslate;\nvar getDistance = util.getTouchDistance;\nvar translate = util.translate;\nvar dataURItoBlob = util.dataURItoBlob;\nvar URLApi = window.createObjectURL && window || window.URL && URL.revokeObjectURL && URL || window.webkitURL && webkitURL;\n\nvar Cropper = function() {\n  if (!('ontouchstart' in window)) {\n    throw new Error('this demo should run in mobile device');\n  }\n\n  this.imageState = {};\n};\n\nCropper.prototype = {\n  constructor: Cropper,\n\n  setImage: function(src, file) {\n    var self = this;\n    self.imageLoading = true;\n    self.image = src;\n\n    self.resetSize();\n\n    var url;\n    if (file) {\n      url = URLApi.createObjectURL(file);\n    }\n\n    var image = new Image();\n\n    image.onload = function() {\n      var selfImage = self.refs.image;\n\n      loadImage.parseMetaData(file, function(data) {\n        var orientation;\n        if (data.exif) {\n          orientation = data.exif[0x0112];\n        }\n\n        selfImage.src = src;\n        self.orientation = orientation;\n\n        var originalWidth, originalHeight;\n\n        self.imageState.left = self.imageState.top = 0;\n\n        if (\"5678\".indexOf(orientation) > -1) {\n          originalWidth = image.height;\n          originalHeight = image.width;\n        } else {\n          originalWidth = image.width;\n          originalHeight = image.height;\n        }\n\n        self.imageState.width = originalWidth;\n        self.imageState.height = originalHeight;\n\n        self.initScale();\n\n        var minScale = self.scaleRange[0];\n        var imageWidth = minScale * originalWidth;\n        var imageHeight = minScale * originalHeight;\n        selfImage.style.width = imageWidth + 'px';\n        selfImage.style.height = imageHeight + 'px';\n\n        var imageLeft, imageTop;\n\n        var cropBoxRect = self.cropBoxRect;\n\n        if (originalWidth > originalHeight) {\n          imageLeft = (cropBoxRect.width - imageWidth) / 2 +cropBoxRect.left;\n          imageTop = cropBoxRect.top;\n        } else {\n          imageLeft = cropBoxRect.left;\n          imageTop = (cropBoxRect.height - imageHeight) / 2 + cropBoxRect.top;\n        }\n\n        self.moveImage(imageLeft, imageTop);\n\n        self.imageLoading = false;\n      });\n    };\n    image.src = url || src;\n  },\n\n  getFocalPoint: function(event) {\n    var focalPoint = {\n      left: (event.touches[0].pageX + event.touches[1].pageX) / 2,\n      top: (event.touches[0].pageY + event.touches[1].pageY) / 2\n    };\n\n    var imageState = this.imageState;\n    var cropBoxRect = this.cropBoxRect;\n\n    focalPoint.left -= cropBoxRect.left + imageState.left;\n    focalPoint.top -= cropBoxRect.top + imageState.top;\n\n    return focalPoint;\n  },\n\n  render: function(parentNode) {\n    var element = document.createElement('div');\n    element.className = 'cropper';\n\n    var coverStart = document.createElement('div');\n    var coverEnd = document.createElement('div');\n    var cropBox = document.createElement('div');\n    var image = document.createElement('img');\n\n    coverStart.className = 'cover cover-start';\n    coverEnd.className = 'cover cover-end';\n    cropBox.className = 'crop-box';\n\n    element.appendChild(coverStart);\n    element.appendChild(coverEnd);\n    element.appendChild(cropBox);\n    element.appendChild(image);\n\n    this.refs = {\n      element: element,\n      coverStart: coverStart,\n      coverEnd: coverEnd,\n      cropBox: cropBox,\n      image: image\n    };\n\n    if (parentNode) {\n      parentNode.appendChild(element);\n    }\n\n    if (element.offsetHeight > 0) {\n      this.resetSize();\n    }\n\n    this.bindEvents();\n  },\n\n  initScale: function () {\n    var cropBoxRect = this.cropBoxRect;\n    var width = this.imageState.width;\n    var height = this.imageState.height;\n    var scale, minScale;\n\n    if (width > height) {\n      scale = this.imageState.scale = cropBoxRect.height / height;\n      minScale = cropBoxRect.height * 0.8 / height;\n    } else {\n      scale = this.imageState.scale = cropBoxRect.width / width;\n      minScale = cropBoxRect.width * 0.8 / width;\n    }\n\n    this.scaleRange = [scale, 2];\n    this.bounceScaleRange = [minScale, 3];\n  },\n\n  resetSize: function() {\n    var refs = this.refs;\n    if (!refs) return;\n\n    var element = refs.element;\n    var cropBox = refs.cropBox;\n    var coverStart = refs.coverStart;\n    var coverEnd = refs.coverEnd;\n\n    var width = element.offsetWidth;\n    var height = element.offsetHeight;\n\n    if (width > height) {\n      element.className = 'cropper cropper-horizontal';\n\n      coverStart.style.width = coverEnd.style.width = (width - height) / 2 + 'px';\n      coverStart.style.height = coverEnd.style.height = '';\n      cropBox.style.width = cropBox.style.height = height + 'px';\n    } else {\n      element.className = 'cropper';\n\n      coverStart.style.height = coverEnd.style.height = (height - width) / 2 + 'px';\n      coverStart.style.width = coverEnd.style.width = '';\n      cropBox.style.width = cropBox.style.height = width + 'px';\n    }\n\n    var elementRect = element.getBoundingClientRect();\n    var cropBoxRect = cropBox.getBoundingClientRect();\n\n    this.cropBoxRect = {\n      left: cropBoxRect.left - elementRect.left,\n      top: cropBoxRect.top - elementRect.top,\n      width: cropBoxRect.width,\n      height: cropBoxRect.height\n    };\n\n    this.initScale();\n\n    this.checkBounce(0);\n  },\n\n  checkBounce: function (speed) {\n    var imageState = this.imageState;\n    var cropBoxRect = this.cropBoxRect;\n\n    var imageWidth = imageState.width;\n    var imageHeight = imageState.height;\n    var imageScale = imageState.scale;\n\n    var imageOffset = getElementTranslate(this.refs.image);\n    var left = imageOffset.left;\n    var top = imageOffset.top;\n\n    var leftRange = [-imageWidth * imageScale + cropBoxRect.width + cropBoxRect.left, cropBoxRect.left];\n    var topRange = [-imageHeight * imageScale + cropBoxRect.height + cropBoxRect.top, cropBoxRect.top];\n\n    var overflow = false;\n\n    if (left < leftRange[0]) {\n      left = leftRange[0];\n      overflow = true;\n    } else if (left > leftRange[1]) {\n      left = leftRange[1];\n      overflow = true;\n    }\n\n    if (top < topRange[0]) {\n      top = topRange[0];\n      overflow = true;\n    } else if (top > topRange[1]) {\n      top = topRange[1];\n      overflow = true;\n    }\n\n    if (overflow) {\n      var self = this;\n      translate(this.refs.image, left, top, speed === undefined ? 200 : 0, function() {\n        self.moveImage(left, top);\n      });\n    }\n  },\n\n  moveImage: function(left, top) {\n    var image = this.refs.image;\n    translateElement(image, left, top);\n\n    this.imageState.left = left;\n    this.imageState.top = top;\n  },\n\n  onTouchStart: function(event) {\n    this.amplitude = 0;\n    var image = this.refs.image;\n\n    var fingerCount = event.touches.length;\n    if (fingerCount) {\n      var touchEvent = event.touches[0];\n\n      var imageOffset = getElementTranslate(image);\n\n      this.dragState = {\n        timestamp: Date.now(),\n        startTouchLeft: touchEvent.pageX,\n        startTouchTop: touchEvent.pageY,\n        startLeft: imageOffset.left || 0,\n        startTop: imageOffset.top || 0\n      };\n    }\n\n    if (fingerCount >= 2) {\n      var zoomState = this.zoomState = {\n        timestamp: Date.now()\n      };\n\n      zoomState.startDistance = getDistance(event);\n      zoomState.focalPoint = this.getFocalPoint(event);\n    }\n  },\n\n  onTouchMove: function(event) {\n    var fingerCount = event.touches.length;\n\n    var touchEvent = event.touches[0];\n\n    var cropBoxRect = this.cropBoxRect;\n    var image = this.refs.image;\n\n    var imageState = this.imageState;\n    var imageWidth = imageState.width;\n    var imageHeight = imageState.height;\n\n    var dragState = this.dragState;\n    var zoomState = this.zoomState;\n\n    if (fingerCount === 1) {\n      var leftRange = [ -imageWidth * imageState.scale + cropBoxRect.width, cropBoxRect.left ];\n      var topRange = [ -imageHeight * imageState.scale + cropBoxRect.height + cropBoxRect.top, cropBoxRect.top ];\n\n      var deltaX = touchEvent.pageX - (dragState.lastLeft || dragState.startTouchLeft);\n      var deltaY = touchEvent.pageY - (dragState.lastTop || dragState.startTouchTop);\n\n      var imageOffset = getElementTranslate(image);\n\n      var left = imageOffset.left + deltaX;\n      var top = imageOffset.top + deltaY;\n\n      if (left < leftRange[0] || left > leftRange[1]) {\n        left -= deltaX / 2;\n      }\n\n      if (top < topRange [0] || top > topRange[1]) {\n        top -= deltaY / 2;\n      }\n\n      this.moveImage(left, top);\n    } else if (fingerCount >= 2) {\n      if (!zoomState.timestamp) {\n        zoomState = {\n          timestamp: Date.now()\n        };\n\n        zoomState.startDistance = getDistance(event);\n        zoomState.focalPoint = this.getFocalPoint(event);\n\n        return;\n      }\n\n      var newDistance = getDistance(event);\n      var oldScale = imageState.scale;\n\n      imageState.scale = oldScale * newDistance / (zoomState.lastDistance || zoomState.startDistance);\n\n      var scaleRange = this.scaleRange;\n      if (imageState.scale < scaleRange[0]) {\n        imageState.scale = scaleRange[0];\n      } else if (imageState.scale > scaleRange[1]) {\n        imageState.scale = scaleRange[1];\n      }\n\n      this.zoomWithFocal(oldScale);\n\n      zoomState.focalPoint = this.getFocalPoint(event);\n      zoomState.lastDistance = newDistance;\n    }\n\n    dragState.lastLeft = touchEvent.pageX;\n    dragState.lastTop = touchEvent.pageY;\n  },\n\n  onTouchEnd: function(event) {\n    var imageState = this.imageState;\n    var zoomState = this.zoomState;\n    var dragState = this.dragState;\n    var amplitude = this.amplitude;\n    var imageWidth = imageState.width;\n    var imageHeight = imageState.height;\n    var cropBoxRect = this.cropBoxRect;\n\n    if (event.touches.length === 0 && dragState.timestamp) {\n      var self = this;\n      var duration = Date.now() - dragState.timestamp;\n\n      if (duration > 300) {\n        self.checkBounce();\n      } else {\n        var target;\n\n        var top = imageState.top;\n        var left = imageState.left;\n\n        var momentumVertical = false;\n\n        var timeConstant = 160;\n\n        var autoScroll = function () {\n          var elapsed, delta;\n\n          if (amplitude) {\n            elapsed = Date.now() - timestamp;\n            delta = -amplitude * Math.exp(-elapsed / timeConstant);\n            if (delta > 0.5 || delta < -0.5) {\n              if (momentumVertical) {\n                self.moveImage(left, target + delta);\n              } else {\n                self.moveImage(target + delta, top);\n              }\n\n              requestAnimationFrame(autoScroll);\n            } else {\n              var currentLeft;\n              var currentTop;\n\n              if (momentumVertical) {\n                currentLeft = left;\n                currentTop = target;\n              } else {\n                currentLeft = target;\n                currentTop = top;\n              }\n\n              self.moveImage(currentLeft, currentTop);\n              self.checkBounce();\n            }\n          }\n        };\n\n        var velocity;\n\n        var deltaX = event.changedTouches[0].pageX - dragState.startTouchLeft;\n        var deltaY = event.changedTouches[0].pageY - dragState.startTouchTop;\n\n        if (Math.abs(deltaX) > Math.abs(deltaY)) {\n          velocity = deltaX / duration;\n        } else {\n          momentumVertical = true;\n          velocity = deltaY / duration;\n        }\n\n        amplitude = 80 * velocity;\n\n        var range;\n\n        if (momentumVertical) {\n          target = Math.round(imageState.top + amplitude);\n          range = [-imageHeight * imageState.scale + cropBoxRect.height / 2 + cropBoxRect.top, cropBoxRect.top + cropBoxRect.height / 2];\n        } else {\n          target = Math.round(imageState.left + amplitude);\n          range = [-imageWidth * imageState.scale + cropBoxRect.width / 2, cropBoxRect.left + cropBoxRect.width / 2];\n        }\n\n        if (target < range[0]) {\n          target = range[0];\n          amplitude /= 2;\n        } else if (target > range[1]) {\n          target = range[1];\n          amplitude /= 2;\n        }\n\n        var timestamp = Date.now();\n        requestAnimationFrame(autoScroll);\n      }\n\n      this.dragState = {};\n    } else if (zoomState.timestamp) {\n      this.checkBounce();\n\n      this.zoomState = {};\n    }\n  },\n\n  zoomWithFocal: function(oldScale) {\n    var image = this.refs.image;\n    var imageState = this.imageState;\n    var imageScale = imageState.scale;\n\n    image.style.width = imageState.width * imageScale + 'px';\n    image.style.height = imageState.height * imageScale + 'px';\n\n    var focalPoint = this.zoomState.focalPoint;\n\n    var offsetLeft = (focalPoint.left / imageScale - focalPoint.left / oldScale) * imageScale;\n    var offsetTop = (focalPoint.top / imageScale - focalPoint.top / oldScale) * imageScale;\n\n    var imageLeft = imageState.left || 0;\n    var imageTop = imageState.top || 0;\n\n    this.moveImage(imageLeft + offsetLeft, imageTop + offsetTop);\n  },\n\n  bindEvents: function() {\n    var cropBox = this.refs.cropBox;\n\n    cropBox.addEventListener('touchstart', this.onTouchStart.bind(this));\n\n    cropBox.addEventListener('touchmove', this.onTouchMove.bind(this));\n\n    cropBox.addEventListener('touchend', this.onTouchEnd.bind(this));\n  },\n\n  createBase64: function (callback, width) {\n    var imageState = this.imageState;\n    var cropBoxRect = this.cropBoxRect;\n    var scale = imageState.scale;\n\n    var canvasSize = width;\n\n    if (!canvasSize) {\n      canvasSize = cropBoxRect.width * 2;\n    }\n\n    var imageLeft = Math.round((cropBoxRect.left - imageState.left) / scale);\n    var imageTop = Math.round((cropBoxRect.top - imageState.top) / scale);\n    var imageSize = Math.floor(cropBoxRect.width / scale);\n\n    var orientation = this.orientation;\n    var image = this.refs.image;\n\n    var cropImage = new Image();\n    cropImage.src = image.src;\n\n    cropImage.onload = function() {\n      var resultCanvas = loadImage.scale(cropImage, {\n        canvas: true,\n        left: imageLeft,\n        top: imageTop,\n        sourceWidth: imageSize,\n        sourceHeight: imageSize,\n        orientation: orientation,\n        maxWidth: canvasSize,\n        maxHeight: canvasSize\n      });\n\n      var dataURL = resultCanvas.toDataURL();\n      if (typeof callback === 'function') {\n        callback({\n          canvasSize: canvasSize,\n          canvas: resultCanvas,\n          dataURL: dataURL\n        });\n      }\n    };\n  },\n\n  getCroppedImage: function(callback, width) {\n    if (!this.image) return null;\n\n    this.createBase64(function(result) {\n      var canvasSize = result.canvasSize;\n      var canvas = result.canvas;\n      var dataURL = result.dataURL;\n\n      if (typeof callback === 'function') {\n        callback({\n          file: canvas.toBlob ? canvas.toBlob() : dataURItoBlob(dataURL),\n          dataUrl: dataURL,\n          oDataURL: result.oDataURL,\n          size: canvasSize\n        });\n      }\n    }, width);\n  }\n};\n\nmodule.exports = Cropper;","window.Cropper = require('./cropper');","\nvar once = function(el, event, fn) {\n  var listener = function() {\n    if (fn) {\n      fn.apply(this, arguments);\n    }\n    el.removeEventListener(event, listener);\n  };\n  el.addEventListener(event, listener);\n};\n\nmodule.exports = {\n  dataURItoBlob: function (dataURI) {\n    var binaryString = atob(dataURI.split(',')[1]);\n    var arrayBuffer = new ArrayBuffer(binaryString.length);\n    var intArray = new Uint8Array(arrayBuffer);\n\n    for (var i = 0, j = binaryString.length; i < j; i++) {\n      intArray[i] = binaryString.charCodeAt(i);\n    }\n\n    var data = [intArray];\n    var type = 'image/png';\n\n    var result;\n\n    try {\n      result = new Blob(data, { type: type });\n    } catch(error) {\n      // TypeError old chrome and FF\n      window.BlobBuilder = window.BlobBuilder ||\n        window.WebKitBlobBuilder ||\n        window.MozBlobBuilder ||\n        window.MSBlobBuilder;\n\n      if(error.name == 'TypeError' && window.BlobBuilder){\n        var builder = new BlobBuilder();\n        builder.append(arrayBuffer);\n        result = builder.getBlob(type);\n      }\n    }\n\n    return result;\n  },\n  getTouchDistance: function(event) {\n    var finger = event.touches[0];\n    var finger2 = event.touches[1];\n\n    var c1 = Math.abs(finger.pageX - finger2.pageX);\n    var c2 = Math.abs(finger.pageY - finger2.pageY);\n\n    return Math.sqrt( c1 * c1 + c2 * c2 );\n  },\n  translate: function(element, left, top, speed, callback) {\n    element.style.webkitTransform = 'translate3d(' + (left || 0) + 'px, ' + (top || 0) + 'px, 0)';\n    if (speed) {\n      var called = false;\n\n      var realCallback = function() {\n        if (called) return;\n        element.style.webkitTransition = '';\n        called = true;\n        if (callback) {\n          callback.apply(this, arguments);\n        }\n      };\n      element.style.webkitTransition = '-webkit-transform ' + speed + 'ms cubic-bezier(0.325, 0.770, 0.000, 1.000)';\n      once(element, 'webkitTransitionEnd', realCallback);\n      once(element, 'transitionend', realCallback);\n      // for android...\n      setTimeout(realCallback, speed + 50);\n    } else {\n      element.style.webkitTransition = '';\n    }\n  },\n  translateElement: function(element, left, top) {\n    element.style.webkitTransform = 'translate3d(' + (left || 0) + 'px, ' + (top || 0) + 'px, 0)';\n  },\n  getElementTranslate: function(element) {\n    var transform = element.style.webkitTransform;\n    var matches = /translate3d\\((.*?)\\)/ig.exec(transform);\n    if (matches) {\n      var translates = matches[1].split(',');\n      return {\n        left: parseInt(translates[0], 10),\n        top: parseInt(translates[1], 10)\n      }\n    }\n    return {\n      left: 0,\n      top: 0\n    }\n  }\n};"]}
644 |
--------------------------------------------------------------------------------
/examples/demo.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
22 |
34 |
35 |
--------------------------------------------------------------------------------
/examples/orientation_1.JPG:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ElemeFE/image-cropper-touch/40bdcbb21ea50fbbdadf2c3f1d44dda35dd48686/examples/orientation_1.JPG
--------------------------------------------------------------------------------
/examples/orientation_3.JPG:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ElemeFE/image-cropper-touch/40bdcbb21ea50fbbdadf2c3f1d44dda35dd48686/examples/orientation_3.JPG
--------------------------------------------------------------------------------
/examples/orientation_6.JPG:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ElemeFE/image-cropper-touch/40bdcbb21ea50fbbdadf2c3f1d44dda35dd48686/examples/orientation_6.JPG
--------------------------------------------------------------------------------
/examples/orientation_8.JPG:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/ElemeFE/image-cropper-touch/40bdcbb21ea50fbbdadf2c3f1d44dda35dd48686/examples/orientation_8.JPG
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "image-cropper-touch",
3 | "version": "0.1.2",
4 | "description": "A image cropper for mobile device.",
5 | "repository": {
6 | "type": "git",
7 | "url": "https://github.com/ElemeFE/image-cropper-touch.git"
8 | },
9 | "keywords": [
10 | "crop",
11 | "cropper",
12 | "image",
13 | "mobile",
14 | "touch"
15 | ],
16 | "main": "src/index.js",
17 | "author": "long.zhang",
18 | "license": "MIT",
19 | "dependencies": {
20 | "blueimp-load-image": "^1.13.1"
21 | },
22 | "devDependencies": {
23 | "browserify": "^9.0.8",
24 | "watchify": "^3.3.0"
25 | },
26 | "scripts": {
27 | "build": "browserify src/index.js -o dist/cropper.js",
28 | "watch": "watchify src/index.js -o dist/cropper.js -dv"
29 | }
30 | }
31 |
--------------------------------------------------------------------------------
/src/cropper.js:
--------------------------------------------------------------------------------
1 | var util = require('./util');
2 |
3 | var translateElement = util.translateElement;
4 | var getElementTranslate = util.getElementTranslate;
5 | var getDistance = util.getTouchDistance;
6 | var translate = util.translate;
7 | var dataURItoBlob = util.dataURItoBlob;
8 | var URLApi = window.createObjectURL && window || window.URL && URL.revokeObjectURL && URL || window.webkitURL && webkitURL;
9 |
10 | var Cropper = function() {
11 | if (!('ontouchstart' in window)) {
12 | throw new Error('this demo should run in mobile device');
13 | }
14 |
15 | this.imageState = {};
16 | };
17 |
18 | Cropper.prototype = {
19 | constructor: Cropper,
20 |
21 | setImage: function(src, file) {
22 | var self = this;
23 | self.imageLoading = true;
24 | self.image = src;
25 |
26 | self.resetSize();
27 |
28 | var url;
29 | if (file) {
30 | url = URLApi.createObjectURL(file);
31 | }
32 |
33 | var image = new Image();
34 |
35 | image.onload = function() {
36 | var selfImage = self.refs.image;
37 |
38 | loadImage.parseMetaData(file, function(data) {
39 | var orientation;
40 | if (data.exif) {
41 | orientation = data.exif[0x0112];
42 | }
43 |
44 | selfImage.src = src;
45 | self.orientation = orientation;
46 |
47 | var originalWidth, originalHeight;
48 |
49 | self.imageState.left = self.imageState.top = 0;
50 |
51 | if ("5678".indexOf(orientation) > -1) {
52 | originalWidth = image.height;
53 | originalHeight = image.width;
54 | } else {
55 | originalWidth = image.width;
56 | originalHeight = image.height;
57 | }
58 |
59 | self.imageState.width = originalWidth;
60 | self.imageState.height = originalHeight;
61 |
62 | self.initScale();
63 |
64 | var minScale = self.scaleRange[0];
65 | var imageWidth = minScale * originalWidth;
66 | var imageHeight = minScale * originalHeight;
67 | selfImage.style.width = imageWidth + 'px';
68 | selfImage.style.height = imageHeight + 'px';
69 |
70 | var imageLeft, imageTop;
71 |
72 | var cropBoxRect = self.cropBoxRect;
73 |
74 | if (originalWidth > originalHeight) {
75 | imageLeft = (cropBoxRect.width - imageWidth) / 2 +cropBoxRect.left;
76 | imageTop = cropBoxRect.top;
77 | } else {
78 | imageLeft = cropBoxRect.left;
79 | imageTop = (cropBoxRect.height - imageHeight) / 2 + cropBoxRect.top;
80 | }
81 |
82 | self.moveImage(imageLeft, imageTop);
83 |
84 | self.imageLoading = false;
85 | });
86 | };
87 | image.src = url || src;
88 | },
89 |
90 | getFocalPoint: function(event) {
91 | var focalPoint = {
92 | left: (event.touches[0].pageX + event.touches[1].pageX) / 2,
93 | top: (event.touches[0].pageY + event.touches[1].pageY) / 2
94 | };
95 |
96 | var imageState = this.imageState;
97 | var cropBoxRect = this.cropBoxRect;
98 |
99 | focalPoint.left -= cropBoxRect.left + imageState.left;
100 | focalPoint.top -= cropBoxRect.top + imageState.top;
101 |
102 | return focalPoint;
103 | },
104 |
105 | render: function(parentNode) {
106 | var element = document.createElement('div');
107 | element.className = 'cropper';
108 |
109 | var coverStart = document.createElement('div');
110 | var coverEnd = document.createElement('div');
111 | var cropBox = document.createElement('div');
112 | var image = document.createElement('img');
113 |
114 | coverStart.className = 'cover cover-start';
115 | coverEnd.className = 'cover cover-end';
116 | cropBox.className = 'crop-box';
117 |
118 | element.appendChild(coverStart);
119 | element.appendChild(coverEnd);
120 | element.appendChild(cropBox);
121 | element.appendChild(image);
122 |
123 | this.refs = {
124 | element: element,
125 | coverStart: coverStart,
126 | coverEnd: coverEnd,
127 | cropBox: cropBox,
128 | image: image
129 | };
130 |
131 | if (parentNode) {
132 | parentNode.appendChild(element);
133 | }
134 |
135 | if (element.offsetHeight > 0) {
136 | this.resetSize();
137 | }
138 |
139 | this.bindEvents();
140 | },
141 |
142 | initScale: function () {
143 | var cropBoxRect = this.cropBoxRect;
144 | var width = this.imageState.width;
145 | var height = this.imageState.height;
146 | var scale, minScale;
147 |
148 | if (width > height) {
149 | scale = this.imageState.scale = cropBoxRect.height / height;
150 | minScale = cropBoxRect.height * 0.8 / height;
151 | } else {
152 | scale = this.imageState.scale = cropBoxRect.width / width;
153 | minScale = cropBoxRect.width * 0.8 / width;
154 | }
155 |
156 | this.scaleRange = [scale, 2];
157 | this.bounceScaleRange = [minScale, 3];
158 | },
159 |
160 | resetSize: function() {
161 | var refs = this.refs;
162 | if (!refs) return;
163 |
164 | var element = refs.element;
165 | var cropBox = refs.cropBox;
166 | var coverStart = refs.coverStart;
167 | var coverEnd = refs.coverEnd;
168 |
169 | var width = element.offsetWidth;
170 | var height = element.offsetHeight;
171 |
172 | if (width > height) {
173 | element.className = 'cropper cropper-horizontal';
174 |
175 | coverStart.style.width = coverEnd.style.width = (width - height) / 2 + 'px';
176 | coverStart.style.height = coverEnd.style.height = '';
177 | cropBox.style.width = cropBox.style.height = height + 'px';
178 | } else {
179 | element.className = 'cropper';
180 |
181 | coverStart.style.height = coverEnd.style.height = (height - width) / 2 + 'px';
182 | coverStart.style.width = coverEnd.style.width = '';
183 | cropBox.style.width = cropBox.style.height = width + 'px';
184 | }
185 |
186 | var elementRect = element.getBoundingClientRect();
187 | var cropBoxRect = cropBox.getBoundingClientRect();
188 |
189 | this.cropBoxRect = {
190 | left: cropBoxRect.left - elementRect.left,
191 | top: cropBoxRect.top - elementRect.top,
192 | width: cropBoxRect.width,
193 | height: cropBoxRect.height
194 | };
195 |
196 | this.initScale();
197 |
198 | this.checkBounce(0);
199 | },
200 |
201 | checkBounce: function (speed) {
202 | var imageState = this.imageState;
203 | var cropBoxRect = this.cropBoxRect;
204 |
205 | var imageWidth = imageState.width;
206 | var imageHeight = imageState.height;
207 | var imageScale = imageState.scale;
208 |
209 | var imageOffset = getElementTranslate(this.refs.image);
210 | var left = imageOffset.left;
211 | var top = imageOffset.top;
212 |
213 | var leftRange = [-imageWidth * imageScale + cropBoxRect.width + cropBoxRect.left, cropBoxRect.left];
214 | var topRange = [-imageHeight * imageScale + cropBoxRect.height + cropBoxRect.top, cropBoxRect.top];
215 |
216 | var overflow = false;
217 |
218 | if (left < leftRange[0]) {
219 | left = leftRange[0];
220 | overflow = true;
221 | } else if (left > leftRange[1]) {
222 | left = leftRange[1];
223 | overflow = true;
224 | }
225 |
226 | if (top < topRange[0]) {
227 | top = topRange[0];
228 | overflow = true;
229 | } else if (top > topRange[1]) {
230 | top = topRange[1];
231 | overflow = true;
232 | }
233 |
234 | if (overflow) {
235 | var self = this;
236 | translate(this.refs.image, left, top, speed === undefined ? 200 : 0, function() {
237 | self.moveImage(left, top);
238 | });
239 | }
240 | },
241 |
242 | moveImage: function(left, top) {
243 | var image = this.refs.image;
244 | translateElement(image, left, top);
245 |
246 | this.imageState.left = left;
247 | this.imageState.top = top;
248 | },
249 |
250 | onTouchStart: function(event) {
251 | this.amplitude = 0;
252 | var image = this.refs.image;
253 |
254 | var fingerCount = event.touches.length;
255 | if (fingerCount) {
256 | var touchEvent = event.touches[0];
257 |
258 | var imageOffset = getElementTranslate(image);
259 |
260 | this.dragState = {
261 | timestamp: Date.now(),
262 | startTouchLeft: touchEvent.pageX,
263 | startTouchTop: touchEvent.pageY,
264 | startLeft: imageOffset.left || 0,
265 | startTop: imageOffset.top || 0
266 | };
267 | }
268 |
269 | if (fingerCount >= 2) {
270 | var zoomState = this.zoomState = {
271 | timestamp: Date.now()
272 | };
273 |
274 | zoomState.startDistance = getDistance(event);
275 | zoomState.focalPoint = this.getFocalPoint(event);
276 | }
277 | },
278 |
279 | onTouchMove: function(event) {
280 | var fingerCount = event.touches.length;
281 |
282 | var touchEvent = event.touches[0];
283 |
284 | var cropBoxRect = this.cropBoxRect;
285 | var image = this.refs.image;
286 |
287 | var imageState = this.imageState;
288 | var imageWidth = imageState.width;
289 | var imageHeight = imageState.height;
290 |
291 | var dragState = this.dragState;
292 | var zoomState = this.zoomState;
293 |
294 | if (fingerCount === 1) {
295 | var leftRange = [ -imageWidth * imageState.scale + cropBoxRect.width, cropBoxRect.left ];
296 | var topRange = [ -imageHeight * imageState.scale + cropBoxRect.height + cropBoxRect.top, cropBoxRect.top ];
297 |
298 | var deltaX = touchEvent.pageX - (dragState.lastLeft || dragState.startTouchLeft);
299 | var deltaY = touchEvent.pageY - (dragState.lastTop || dragState.startTouchTop);
300 |
301 | var imageOffset = getElementTranslate(image);
302 |
303 | var left = imageOffset.left + deltaX;
304 | var top = imageOffset.top + deltaY;
305 |
306 | if (left < leftRange[0] || left > leftRange[1]) {
307 | left -= deltaX / 2;
308 | }
309 |
310 | if (top < topRange [0] || top > topRange[1]) {
311 | top -= deltaY / 2;
312 | }
313 |
314 | this.moveImage(left, top);
315 | } else if (fingerCount >= 2) {
316 | if (!zoomState.timestamp) {
317 | zoomState = {
318 | timestamp: Date.now()
319 | };
320 |
321 | zoomState.startDistance = getDistance(event);
322 | zoomState.focalPoint = this.getFocalPoint(event);
323 |
324 | return;
325 | }
326 |
327 | var newDistance = getDistance(event);
328 | var oldScale = imageState.scale;
329 |
330 | imageState.scale = oldScale * newDistance / (zoomState.lastDistance || zoomState.startDistance);
331 |
332 | var scaleRange = this.scaleRange;
333 | if (imageState.scale < scaleRange[0]) {
334 | imageState.scale = scaleRange[0];
335 | } else if (imageState.scale > scaleRange[1]) {
336 | imageState.scale = scaleRange[1];
337 | }
338 |
339 | this.zoomWithFocal(oldScale);
340 |
341 | zoomState.focalPoint = this.getFocalPoint(event);
342 | zoomState.lastDistance = newDistance;
343 | }
344 |
345 | dragState.lastLeft = touchEvent.pageX;
346 | dragState.lastTop = touchEvent.pageY;
347 | },
348 |
349 | onTouchEnd: function(event) {
350 | var imageState = this.imageState;
351 | var zoomState = this.zoomState;
352 | var dragState = this.dragState;
353 | var amplitude = this.amplitude;
354 | var imageWidth = imageState.width;
355 | var imageHeight = imageState.height;
356 | var cropBoxRect = this.cropBoxRect;
357 |
358 | if (event.touches.length === 0 && dragState.timestamp) {
359 | var self = this;
360 | var duration = Date.now() - dragState.timestamp;
361 |
362 | if (duration > 300) {
363 | self.checkBounce();
364 | } else {
365 | var target;
366 |
367 | var top = imageState.top;
368 | var left = imageState.left;
369 |
370 | var momentumVertical = false;
371 |
372 | var timeConstant = 160;
373 |
374 | var autoScroll = function () {
375 | var elapsed, delta;
376 |
377 | if (amplitude) {
378 | elapsed = Date.now() - timestamp;
379 | delta = -amplitude * Math.exp(-elapsed / timeConstant);
380 | if (delta > 0.5 || delta < -0.5) {
381 | if (momentumVertical) {
382 | self.moveImage(left, target + delta);
383 | } else {
384 | self.moveImage(target + delta, top);
385 | }
386 |
387 | requestAnimationFrame(autoScroll);
388 | } else {
389 | var currentLeft;
390 | var currentTop;
391 |
392 | if (momentumVertical) {
393 | currentLeft = left;
394 | currentTop = target;
395 | } else {
396 | currentLeft = target;
397 | currentTop = top;
398 | }
399 |
400 | self.moveImage(currentLeft, currentTop);
401 | self.checkBounce();
402 | }
403 | }
404 | };
405 |
406 | var velocity;
407 |
408 | var deltaX = event.changedTouches[0].pageX - dragState.startTouchLeft;
409 | var deltaY = event.changedTouches[0].pageY - dragState.startTouchTop;
410 |
411 | if (Math.abs(deltaX) > Math.abs(deltaY)) {
412 | velocity = deltaX / duration;
413 | } else {
414 | momentumVertical = true;
415 | velocity = deltaY / duration;
416 | }
417 |
418 | amplitude = 80 * velocity;
419 |
420 | var range;
421 |
422 | if (momentumVertical) {
423 | target = Math.round(imageState.top + amplitude);
424 | range = [-imageHeight * imageState.scale + cropBoxRect.height / 2 + cropBoxRect.top, cropBoxRect.top + cropBoxRect.height / 2];
425 | } else {
426 | target = Math.round(imageState.left + amplitude);
427 | range = [-imageWidth * imageState.scale + cropBoxRect.width / 2, cropBoxRect.left + cropBoxRect.width / 2];
428 | }
429 |
430 | if (target < range[0]) {
431 | target = range[0];
432 | amplitude /= 2;
433 | } else if (target > range[1]) {
434 | target = range[1];
435 | amplitude /= 2;
436 | }
437 |
438 | var timestamp = Date.now();
439 | requestAnimationFrame(autoScroll);
440 | }
441 |
442 | this.dragState = {};
443 | } else if (zoomState.timestamp) {
444 | this.checkBounce();
445 |
446 | this.zoomState = {};
447 | }
448 | },
449 |
450 | zoomWithFocal: function(oldScale) {
451 | var image = this.refs.image;
452 | var imageState = this.imageState;
453 | var imageScale = imageState.scale;
454 |
455 | image.style.width = imageState.width * imageScale + 'px';
456 | image.style.height = imageState.height * imageScale + 'px';
457 |
458 | var focalPoint = this.zoomState.focalPoint;
459 |
460 | var offsetLeft = (focalPoint.left / imageScale - focalPoint.left / oldScale) * imageScale;
461 | var offsetTop = (focalPoint.top / imageScale - focalPoint.top / oldScale) * imageScale;
462 |
463 | var imageLeft = imageState.left || 0;
464 | var imageTop = imageState.top || 0;
465 |
466 | this.moveImage(imageLeft + offsetLeft, imageTop + offsetTop);
467 | },
468 |
469 | bindEvents: function() {
470 | var cropBox = this.refs.cropBox;
471 |
472 | cropBox.addEventListener('touchstart', this.onTouchStart.bind(this));
473 |
474 | cropBox.addEventListener('touchmove', this.onTouchMove.bind(this));
475 |
476 | cropBox.addEventListener('touchend', this.onTouchEnd.bind(this));
477 | },
478 |
479 | createBase64: function (callback, width) {
480 | var imageState = this.imageState;
481 | var cropBoxRect = this.cropBoxRect;
482 | var scale = imageState.scale;
483 |
484 | var canvasSize = width;
485 |
486 | if (!canvasSize) {
487 | canvasSize = cropBoxRect.width * 2;
488 | }
489 |
490 | var imageLeft = Math.round((cropBoxRect.left - imageState.left) / scale);
491 | var imageTop = Math.round((cropBoxRect.top - imageState.top) / scale);
492 | var imageSize = Math.floor(cropBoxRect.width / scale);
493 |
494 | var orientation = this.orientation;
495 | var image = this.refs.image;
496 |
497 | var cropImage = new Image();
498 | cropImage.src = image.src;
499 |
500 | cropImage.onload = function() {
501 | var resultCanvas = loadImage.scale(cropImage, {
502 | canvas: true,
503 | left: imageLeft,
504 | top: imageTop,
505 | sourceWidth: imageSize,
506 | sourceHeight: imageSize,
507 | orientation: orientation,
508 | maxWidth: canvasSize,
509 | maxHeight: canvasSize
510 | });
511 |
512 | var dataURL = resultCanvas.toDataURL();
513 | if (typeof callback === 'function') {
514 | callback({
515 | canvasSize: canvasSize,
516 | canvas: resultCanvas,
517 | dataURL: dataURL
518 | });
519 | }
520 | };
521 | },
522 |
523 | getCroppedImage: function(callback, width) {
524 | if (!this.image) return null;
525 |
526 | this.createBase64(function(result) {
527 | var canvasSize = result.canvasSize;
528 | var canvas = result.canvas;
529 | var dataURL = result.dataURL;
530 |
531 | if (typeof callback === 'function') {
532 | callback({
533 | file: canvas.toBlob ? canvas.toBlob() : dataURItoBlob(dataURL),
534 | dataUrl: dataURL,
535 | oDataURL: result.oDataURL,
536 | size: canvasSize
537 | });
538 | }
539 | }, width);
540 | }
541 | };
542 |
543 | module.exports = Cropper;
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | window.Cropper = require('./cropper');
--------------------------------------------------------------------------------
/src/util.js:
--------------------------------------------------------------------------------
1 |
2 | var once = function(el, event, fn) {
3 | var listener = function() {
4 | if (fn) {
5 | fn.apply(this, arguments);
6 | }
7 | el.removeEventListener(event, listener);
8 | };
9 | el.addEventListener(event, listener);
10 | };
11 |
12 | module.exports = {
13 | dataURItoBlob: function (dataURI) {
14 | var binaryString = atob(dataURI.split(',')[1]);
15 | var arrayBuffer = new ArrayBuffer(binaryString.length);
16 | var intArray = new Uint8Array(arrayBuffer);
17 |
18 | for (var i = 0, j = binaryString.length; i < j; i++) {
19 | intArray[i] = binaryString.charCodeAt(i);
20 | }
21 |
22 | var data = [intArray];
23 | var type = 'image/png';
24 |
25 | var result;
26 |
27 | try {
28 | result = new Blob(data, { type: type });
29 | } catch(error) {
30 | // TypeError old chrome and FF
31 | window.BlobBuilder = window.BlobBuilder ||
32 | window.WebKitBlobBuilder ||
33 | window.MozBlobBuilder ||
34 | window.MSBlobBuilder;
35 |
36 | if(error.name == 'TypeError' && window.BlobBuilder){
37 | var builder = new BlobBuilder();
38 | builder.append(arrayBuffer);
39 | result = builder.getBlob(type);
40 | }
41 | }
42 |
43 | return result;
44 | },
45 | getTouchDistance: function(event) {
46 | var finger = event.touches[0];
47 | var finger2 = event.touches[1];
48 |
49 | var c1 = Math.abs(finger.pageX - finger2.pageX);
50 | var c2 = Math.abs(finger.pageY - finger2.pageY);
51 |
52 | return Math.sqrt( c1 * c1 + c2 * c2 );
53 | },
54 | translate: function(element, left, top, speed, callback) {
55 | element.style.webkitTransform = 'translate3d(' + (left || 0) + 'px, ' + (top || 0) + 'px, 0)';
56 | if (speed) {
57 | var called = false;
58 |
59 | var realCallback = function() {
60 | if (called) return;
61 | element.style.webkitTransition = '';
62 | called = true;
63 | if (callback) {
64 | callback.apply(this, arguments);
65 | }
66 | };
67 | element.style.webkitTransition = '-webkit-transform ' + speed + 'ms cubic-bezier(0.325, 0.770, 0.000, 1.000)';
68 | once(element, 'webkitTransitionEnd', realCallback);
69 | once(element, 'transitionend', realCallback);
70 | // for android...
71 | setTimeout(realCallback, speed + 50);
72 | } else {
73 | element.style.webkitTransition = '';
74 | }
75 | },
76 | translateElement: function(element, left, top) {
77 | element.style.webkitTransform = 'translate3d(' + (left || 0) + 'px, ' + (top || 0) + 'px, 0)';
78 | },
79 | getElementTranslate: function(element) {
80 | var transform = element.style.webkitTransform;
81 | var matches = /translate3d\((.*?)\)/ig.exec(transform);
82 | if (matches) {
83 | var translates = matches[1].split(',');
84 | return {
85 | left: parseInt(translates[0], 10),
86 | top: parseInt(translates[1], 10)
87 | }
88 | }
89 | return {
90 | left: 0,
91 | top: 0
92 | }
93 | }
94 | };
--------------------------------------------------------------------------------