-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathPinchZoom.js
192 lines (171 loc) · 5.64 KB
/
PinchZoom.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
/**
* Makes the given dom node zoomable within its parent by pinch gesture
*
* @link https://github.com/cosmocode/pinchzoom
* @param {Node} obj The DOM node to pinch zoom, needs to have a scrollable parent node
* @param {[Object]} opts Optional options
* @constructor
*/
var PinchZoom = function (obj, opts) {
var prefixedTransform = 'transform';
/**
* default options
*/
var options = {
minScale: 0.5,
maxScale: 3,
debug: null
};
/**
* Store the input data and provide some functions on it
*/
var input = {
isPinching: true, // are we pinching right now?
startDistance: 0, // fingers started at that distance
x1: 0, // finger 1 X
y1: 0, // finger 1 Y
x2: 0, // finger 2 X
y2: 0, // finger 2 Y
startScale: 1.0, // initial scale
currentScale: 1.0, // current scale,
scrollTop: 0, // scroll postion
scrollLeft: 0, // scroll position
/**
* store the finger positions from the given event
*
* @param event
*/
storeFingerPosition: function (event) {
this.x1 = event.touches[0].clientX;
this.y1 = event.touches[0].clientY;
this.x2 = event.touches[1].clientX;
this.y2 = event.touches[1].clientY;
},
/**
* Calculate the distance between the fingers
*
* @returns {number}
*/
calcFingerDistance: function () {
var dx = this.x2 - this.x1;
var dy = this.y2 - this.y1;
return Math.sqrt(dx * dx + dy * dy);
},
/**
* reset the input variable to be ready for the next event
*/
reset: function () {
this.startScale = this.currentScale;
this.startDistance = 0;
this.x1 = 0;
this.y1 = 0;
this.x2 = 0;
this.y2 = 0;
this.isPinching = false;
}
};
/**
* Begin the pinch handling
*
* @param event
*/
function onTouchStart(event) {
if (event.touches.length !== 2) return;
event.preventDefault();
event.stopPropagation();
input.storeFingerPosition(event);
input.startDistance = input.calcFingerDistance();
input.isPinching = true;
input.scrollTop = obj.parentNode.scrollTop;
input.scrollLeft = obj.parentNode.scrollLeft;
}
/**
* During pinch handling
*
* @param event
*/
function onTouchMove(event) {
if (event.touches.length !== 2) return;
event.preventDefault();
event.stopPropagation();
input.storeFingerPosition(event);
}
/**
* finish pinch handling
*
* @param event
*/
function onTouchEnd(event) {
if (!input.isPinching) return;
event.preventDefault();
event.stopPropagation();
input.reset();
}
/**
* Execute rendering during animation frame
*/
function onAnimationFrame() {
requestAnimationFrame(onAnimationFrame);
render();
}
/**
* Calculates and applies the new scale
*/
function render() {
if (!input.isPinching) return;
var newDistance = input.calcFingerDistance();
/*
* Calculate the scaling from the distance the fingers made since the
* beginning of the pinch
*
* @todo the pixel-to-scalechange ratio should be calculated from the device's display density
*/
var scalePixelChange = newDistance - input.startDistance;
input.currentScale = input.startScale + scalePixelChange * 0.01;
if (input.currentScale < options.minScale) input.currentScale = options.minScale;
if (input.currentScale > options.maxScale) input.currentScale = options.maxScale;
/*
* OffsetWidth and height are not scaled and stay in the original size.
* Scaling happens from center of the element. Here we calculate how much of it
* scaled out of the original boundary
*/
var transX = ((obj.offsetWidth * input.currentScale) / 2) - (obj.offsetWidth / 2);
var transY = ((obj.offsetHeight * input.currentScale) / 2) - (obj.offsetHeight / 2);
/*
* translates are influenced by the scaling, so we remove that from the
* calculated value
*/
transX = transX / input.currentScale;
transY = transY / input.currentScale;
/*
* Apply scaling and translate (this basically moves the origin to 0,0), then
* the outer container is scrolled to make up for the translation
*/
obj.style[prefixedTransform] = 'scale(' + input.currentScale + ') translate(' + transX + 'px, ' + transY + 'px)';
if (transX > 0) obj.parentNode.scrollLeft = input.scrollLeft + transX;
if (transY > 0) obj.parentNode.scrollTop = input.scrollTop + transY;
if(options.debug) options.debug(input);
}
/*
* Main
*/
// merge options
if (opts) for (var attr in opts) {
if (opts.hasOwnProperty(attr)) {
options[attr] = opts[attr];
}
}
// vendor prefixes if needed
var vendors = ['ms', 'moz', 'webkit', 'o'];
for (var x = 0; x < vendors.length; ++x) {
if (vendors[x] + 'Transform' in document.body.style) {
prefixedTransform = vendors[x] + 'Transform';
break;
}
}
// attach listeners
obj.addEventListener('touchstart', onTouchStart);
obj.addEventListener('touchmove', onTouchMove);
obj.addEventListener('touchend', onTouchEnd);
requestAnimationFrame(onAnimationFrame);
};