TrackballControls.js 12.3 KB
Newer Older
1
/**
2
 * @author Eberhard Graether / http://egraether.com/
M
Mark Lundin 已提交
3
 * @author Mark Lundin 	/ http://mark-lundin.com
4 5
 */

6
THREE.TrackballControls = function ( object, domElement ) {
7

M
Mr.doob 已提交
8
	var _this = this;
9
	var STATE = { NONE: -1, ROTATE: 0, ZOOM: 1, PAN: 2, TOUCH_ROTATE: 3, TOUCH_ZOOM: 4, TOUCH_PAN: 5 };
10

11
	this.object = object;
12
	this.domElement = ( domElement !== undefined ) ? domElement : document;
13

14
	// API
15

16 17
	this.enabled = true;

18
	this.screen = { left: 0, top: 0, width: 0, height: 0 };
19

20 21 22
	this.rotateSpeed = 1.0;
	this.zoomSpeed = 1.2;
	this.panSpeed = 0.3;
23

24
	this.noRotate = false;
25 26
	this.noZoom = false;
	this.noPan = false;
27
	this.noRoll = false;
28

29 30
	this.staticMoving = false;
	this.dynamicDampingFactor = 0.2;
31

32 33
	this.minDistance = 0;
	this.maxDistance = Infinity;
34

35
	this.keys = [ 65 /*A*/, 83 /*S*/, 68 /*D*/ ];
36

37
	// internals
38

M
Mr.doob 已提交
39 40
	this.target = new THREE.Vector3();

41 42
	var EPS = 0.000001;

M
Mr.doob 已提交
43
	var lastPosition = new THREE.Vector3();
44

M
Mr.doob 已提交
45 46
	var _state = STATE.NONE,
	_prevState = STATE.NONE,
47

48 49
	_eye = new THREE.Vector3(),

50 51 52 53 54 55
	_rotateStart = new THREE.Vector3(),
	_rotateEnd = new THREE.Vector3(),

	_zoomStart = new THREE.Vector2(),
	_zoomEnd = new THREE.Vector2(),

56 57 58
	_touchZoomDistanceStart = 0,
	_touchZoomDistanceEnd = 0,

59 60
	_panStart = new THREE.Vector2(),
	_panEnd = new THREE.Vector2();
61

W
WestLangley 已提交
62 63 64 65 66 67
	// for reset

	this.target0 = this.target.clone();
	this.position0 = this.object.position.clone();
	this.up0 = this.object.up.clone();

M
Mr.doob 已提交
68 69 70
	// events

	var changeEvent = { type: 'change' };
71 72
	var startEvent = { type: 'start'};
	var endEvent = { type: 'end'};
M
Mr.doob 已提交
73

74

75
	// methods
76

77 78
	this.handleResize = function () {

79
		if ( this.domElement === document ) {
80

81 82 83 84 85 86 87 88
			this.screen.left = 0;
			this.screen.top = 0;
			this.screen.width = window.innerWidth;
			this.screen.height = window.innerHeight;

		} else {

			this.screen = this.domElement.getBoundingClientRect();
89 90 91 92
			// adjustments come from similar code in the jquery offset() function
			var d = this.domElement.ownerDocument.documentElement
			this.screen.left += window.pageXOffset - d.clientLeft
			this.screen.top += window.pageYOffset - d.clientTop
93 94

		}
95

96 97
	};

98
	this.handleEvent = function ( event ) {
99

100
		if ( typeof this[ event.type ] == 'function' ) {
101

102
			this[ event.type ]( event );
103

104
		}
105

106
	};
107

M
Mark Lundin 已提交
108
	this.getMouseOnScreen = function ( pageX, pageY, vector ) {
109

M
Mark Lundin 已提交
110
		return vector.set(
111 112
			( pageX - _this.screen.left ) / _this.screen.width,
			( pageY - _this.screen.top ) / _this.screen.height
113
		);
114

115
	};
116

117
	this.getMouseProjectionOnBall = (function(){
118

M
Mark Lundin 已提交
119 120
		var objectUp = new THREE.Vector3(),
		    mouseOnBall = new THREE.Vector3();
121

122

123
		return function ( pageX, pageY, projection ) {
124

M
Mark Lundin 已提交
125
			mouseOnBall.set(
126 127
				( pageX - _this.screen.width * 0.5 - _this.screen.left ) / (_this.screen.width*.5),
				( _this.screen.height * 0.5 + _this.screen.top - pageY ) / (_this.screen.height*.5),
128 129
				0.0
			);
130

131
			var length = mouseOnBall.length();
132

133
			if ( _this.noRoll ) {
134

135
				if ( length < Math.SQRT1_2 ) {
136

137
					mouseOnBall.z = Math.sqrt( 1.0 - length*length );
138

139 140 141 142 143 144 145
				} else {

					mouseOnBall.z = .5 / length;
					
				}

			} else if ( length > 1.0 ) {
146

147
				mouseOnBall.normalize();
148

149 150 151 152 153 154 155 156 157 158 159 160 161
			} else {

				mouseOnBall.z = Math.sqrt( 1.0 - length * length );

			}

			_eye.copy( _this.object.position ).sub( _this.target );

			projection.copy( _this.object.up ).setLength( mouseOnBall.y )
			projection.add( objectUp.copy( _this.object.up ).cross( _eye ).setLength( mouseOnBall.x ) );
			projection.add( _eye.setLength( mouseOnBall.z ) );

			return projection;
162
		}
163

164
	}());
165

166
	this.rotateCamera = (function(){
167

168 169
		var axis = new THREE.Vector3(),
			quaternion = new THREE.Quaternion();
170

171

172
		return function () {
173

174
			var angle = Math.acos( _rotateStart.dot( _rotateEnd ) / _rotateStart.length() / _rotateEnd.length() );
175

176
			if ( angle ) {
177

178
				axis.crossVectors( _rotateStart, _rotateEnd ).normalize();
179

180
				angle *= _this.rotateSpeed;
181

182
				quaternion.setFromAxisAngle( axis, -angle );
183

184 185
				_eye.applyQuaternion( quaternion );
				_this.object.up.applyQuaternion( quaternion );
186

187
				_rotateEnd.applyQuaternion( quaternion );
188

189
				if ( _this.staticMoving ) {
190

191
					_rotateStart.copy( _rotateEnd );
192

193
				} else {
194

195 196
					quaternion.setFromAxisAngle( axis, angle * ( _this.dynamicDampingFactor - 1.0 ) );
					_rotateStart.applyQuaternion( quaternion );
197

198
				}
199

200
			}
201
		}
202

203
	}());
204

M
Mr.doob 已提交
205
	this.zoomCamera = function () {
206

207
		if ( _state === STATE.TOUCH_ZOOM ) {
208

209 210
			var factor = _touchZoomDistanceStart / _touchZoomDistanceEnd;
			_touchZoomDistanceStart = _touchZoomDistanceEnd;
211
			_eye.multiplyScalar( factor );
212

213
		} else {
214

215
			var factor = 1.0 + ( _zoomEnd.y - _zoomStart.y ) * _this.zoomSpeed;
216

217
			if ( factor !== 1.0 && factor > 0.0 ) {
218

219 220 221 222 223 224 225 226 227 228 229
				_eye.multiplyScalar( factor );

				if ( _this.staticMoving ) {

					_zoomStart.copy( _zoomEnd );

				} else {

					_zoomStart.y += ( _zoomEnd.y - _zoomStart.y ) * this.dynamicDampingFactor;

				}
230 231

			}
232 233 234

		}

235
	};
236

237
	this.panCamera = (function(){
238

239 240 241
		var mouseChange = new THREE.Vector2(),
			objectUp = new THREE.Vector3(),
			pan = new THREE.Vector3();
242

243
		return function () {
244

245
			mouseChange.copy( _panEnd ).sub( _panStart );
246

247
			if ( mouseChange.lengthSq() ) {
248

249
				mouseChange.multiplyScalar( _eye.length() * _this.panSpeed );
250

251 252
				pan.copy( _eye ).cross( _this.object.up ).setLength( mouseChange.x );
				pan.add( objectUp.copy( _this.object.up ).setLength( mouseChange.y ) );
253

254 255
				_this.object.position.add( pan );
				_this.target.add( pan );
256

257
				if ( _this.staticMoving ) {
258

259
					_panStart.copy( _panEnd );
260

261 262 263
				} else {

					_panStart.add( mouseChange.subVectors( _panEnd, _panStart ).multiplyScalar( _this.dynamicDampingFactor ) );
264

265 266 267
				}

			}
268
		}
269

270
	}());
271

M
Mr.doob 已提交
272
	this.checkDistances = function () {
273

274
		if ( !_this.noZoom || !_this.noPan ) {
275

276
			if ( _eye.lengthSq() > _this.maxDistance * _this.maxDistance ) {
277

278
				_this.object.position.addVectors( _this.target, _eye.setLength( _this.maxDistance ) );
279 280 281

			}

282
			if ( _eye.lengthSq() < _this.minDistance * _this.minDistance ) {
283

284
				_this.object.position.addVectors( _this.target, _eye.setLength( _this.minDistance ) );
285 286

			}
287 288 289

		}

290 291
	};

M
Mr.doob 已提交
292
	this.update = function () {
M
Mr.doob 已提交
293

294
		_eye.subVectors( _this.object.position, _this.target );
M
Mr.doob 已提交
295 296 297 298 299 300

		if ( !_this.noRotate ) {

			_this.rotateCamera();

		}
301

M
Mr.doob 已提交
302 303 304 305 306 307 308 309 310 311 312 313
		if ( !_this.noZoom ) {

			_this.zoomCamera();

		}

		if ( !_this.noPan ) {

			_this.panCamera();

		}

314
		_this.object.position.addVectors( _this.target, _eye );
M
Mr.doob 已提交
315 316 317 318 319

		_this.checkDistances();

		_this.object.lookAt( _this.target );

320
		if ( lastPosition.distanceToSquared( _this.object.position ) > EPS ) {
321

M
Mr.doob 已提交
322
			_this.dispatchEvent( changeEvent );
M
Mr.doob 已提交
323 324 325 326 327 328 329

			lastPosition.copy( _this.object.position );

		}

	};

W
WestLangley 已提交
330 331 332 333 334 335 336 337 338 339 340 341 342
	this.reset = function () {

		_state = STATE.NONE;
		_prevState = STATE.NONE;

		_this.target.copy( _this.target0 );
		_this.object.position.copy( _this.position0 );
		_this.object.up.copy( _this.up0 );

		_eye.subVectors( _this.object.position, _this.target );

		_this.object.lookAt( _this.target );

343
		_this.dispatchEvent( changeEvent );
W
WestLangley 已提交
344 345 346 347 348

		lastPosition.copy( _this.object.position );

	};

349
	// listeners
350

351
	function keydown( event ) {
352

353
		if ( _this.enabled === false ) return;
354

M
Mr.doob 已提交
355
		window.removeEventListener( 'keydown', keydown );
356

M
Mr.doob 已提交
357
		_prevState = _state;
358

359
		if ( _state !== STATE.NONE ) {
360

361
			return;
362

363
		} else if ( event.keyCode === _this.keys[ STATE.ROTATE ] && !_this.noRotate ) {
364

365
			_state = STATE.ROTATE;
366

367
		} else if ( event.keyCode === _this.keys[ STATE.ZOOM ] && !_this.noZoom ) {
368

369
			_state = STATE.ZOOM;
370

371
		} else if ( event.keyCode === _this.keys[ STATE.PAN ] && !_this.noPan ) {
372

373
			_state = STATE.PAN;
374 375

		}
376

M
Mr.doob 已提交
377
	}
378

379
	function keyup( event ) {
380

381
		if ( _this.enabled === false ) return;
382

M
Mr.doob 已提交
383
		_state = _prevState;
384

M
Mr.doob 已提交
385
		window.addEventListener( 'keydown', keydown, false );
386

M
Mr.doob 已提交
387
	}
388

389
	function mousedown( event ) {
390

391
		if ( _this.enabled === false ) return;
392

393 394
		event.preventDefault();
		event.stopPropagation();
395

396
		if ( _state === STATE.NONE ) {
397

398
			_state = event.button;
399

M
Mr.doob 已提交
400
		}
401

M
Mr.doob 已提交
402
		if ( _state === STATE.ROTATE && !_this.noRotate ) {
403

M
Mark Lundin 已提交
404
			_this.getMouseProjectionOnBall( event.pageX, event.pageY, _rotateStart );
G
gero3 已提交
405
			_rotateEnd.copy(_rotateStart)
406

M
Mr.doob 已提交
407
		} else if ( _state === STATE.ZOOM && !_this.noZoom ) {
408

M
Mark Lundin 已提交
409
			_this.getMouseOnScreen( event.pageX, event.pageY, _zoomStart );
G
gero3 已提交
410
			_zoomEnd.copy(_zoomStart);
411

M
Mr.doob 已提交
412
		} else if ( _state === STATE.PAN && !_this.noPan ) {
413

M
Mark Lundin 已提交
414
			_this.getMouseOnScreen( event.pageX, event.pageY, _panStart );
G
gero3 已提交
415
			_panEnd.copy(_panStart)
416

417
		}
M
Mr.doob 已提交
418

M
Mr.doob 已提交
419 420
		document.addEventListener( 'mousemove', mousemove, false );
		document.addEventListener( 'mouseup', mouseup, false );
421 422
		_this.dispatchEvent( startEvent );

423

M
Mr.doob 已提交
424
	}
425

426
	function mousemove( event ) {
427

428 429 430 431
		if ( _this.enabled === false ) return;

		event.preventDefault();
		event.stopPropagation();
432

M
Mr.doob 已提交
433
		if ( _state === STATE.ROTATE && !_this.noRotate ) {
434

M
Mark Lundin 已提交
435
			_this.getMouseProjectionOnBall( event.pageX, event.pageY, _rotateEnd );
436

437
		} else if ( _state === STATE.ZOOM && !_this.noZoom ) {
438

M
Mark Lundin 已提交
439
			_this.getMouseOnScreen( event.pageX, event.pageY, _zoomEnd );
440

441
		} else if ( _state === STATE.PAN && !_this.noPan ) {
442

M
Mark Lundin 已提交
443
			_this.getMouseOnScreen( event.pageX, event.pageY, _panEnd );
444

445
		}
446

M
Mr.doob 已提交
447
	}
448

449
	function mouseup( event ) {
450

451
		if ( _this.enabled === false ) return;
452

453 454
		event.preventDefault();
		event.stopPropagation();
455

456
		_state = STATE.NONE;
457

M
Mr.doob 已提交
458 459
		document.removeEventListener( 'mousemove', mousemove );
		document.removeEventListener( 'mouseup', mouseup );
460
		_this.dispatchEvent( endEvent );
461

M
Mr.doob 已提交
462
	}
463

464 465
	function mousewheel( event ) {

466
		if ( _this.enabled === false ) return;
467 468 469 470 471 472

		event.preventDefault();
		event.stopPropagation();

		var delta = 0;

473 474 475 476 477 478 479 480
		if ( event.wheelDelta ) { // WebKit / Opera / Explorer 9

			delta = event.wheelDelta / 40;

		} else if ( event.detail ) { // Firefox

			delta = - event.detail / 3;

481 482
		}

483
		_zoomStart.y += delta * 0.01;
484 485
		_this.dispatchEvent( startEvent );
		_this.dispatchEvent( endEvent );
486 487 488

	}

M
Mr.doob 已提交
489 490
	function touchstart( event ) {

491
		if ( _this.enabled === false ) return;
M
Mr.doob 已提交
492 493 494 495

		switch ( event.touches.length ) {

			case 1:
496
				_state = STATE.TOUCH_ROTATE;
497
				_rotateEnd.copy( _this.getMouseProjectionOnBall( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY, _rotateStart ));
M
Mr.doob 已提交
498
				break;
499

M
Mr.doob 已提交
500
			case 2:
501 502 503 504
				_state = STATE.TOUCH_ZOOM;
				var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
				var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;
				_touchZoomDistanceEnd = _touchZoomDistanceStart = Math.sqrt( dx * dx + dy * dy );
M
Mr.doob 已提交
505
				break;
506

M
Mr.doob 已提交
507
			case 3:
508
				_state = STATE.TOUCH_PAN;
509
				_panEnd.copy( _this.getMouseOnScreen( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY, _panStart ));
M
Mr.doob 已提交
510 511
				break;

512 513 514
			default:
				_state = STATE.NONE;

M
Mr.doob 已提交
515
		}
516 517
		_this.dispatchEvent( startEvent );

M
Mr.doob 已提交
518 519 520 521 522

	}

	function touchmove( event ) {

523
		if ( _this.enabled === false ) return;
M
Mr.doob 已提交
524 525

		event.preventDefault();
526
		event.stopPropagation();
M
Mr.doob 已提交
527 528 529 530

		switch ( event.touches.length ) {

			case 1:
M
Mark Lundin 已提交
531
				_this.getMouseProjectionOnBall( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY, _rotateEnd );
M
Mr.doob 已提交
532
				break;
533

M
Mr.doob 已提交
534
			case 2:
535 536 537
				var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
				var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;
				_touchZoomDistanceEnd = Math.sqrt( dx * dx + dy * dy )
M
Mr.doob 已提交
538
				break;
539

M
Mr.doob 已提交
540
			case 3:
M
Mark Lundin 已提交
541
				_this.getMouseOnScreen( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY, _panEnd );
M
Mr.doob 已提交
542 543
				break;

544 545 546 547 548 549 550 551 552 553 554 555 556 557
			default:
				_state = STATE.NONE;

		}

	}

	function touchend( event ) {

		if ( _this.enabled === false ) return;

		switch ( event.touches.length ) {

			case 1:
558
				_rotateStart.copy( _this.getMouseProjectionOnBall( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY, _rotateEnd ));
559 560 561 562 563 564 565
				break;

			case 2:
				_touchZoomDistanceStart = _touchZoomDistanceEnd = 0;
				break;

			case 3:
566
				_panStart.copy( _this.getMouseOnScreen( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY, _panEnd ));
567 568
				break;

M
Mr.doob 已提交
569 570
		}

571
		_state = STATE.NONE;
572
		_this.dispatchEvent( endEvent );
573

M
Mr.doob 已提交
574 575
	}

576
	this.domElement.addEventListener( 'contextmenu', function ( event ) { event.preventDefault(); }, false );
577

578
	this.domElement.addEventListener( 'mousedown', mousedown, false );
M
Mr.doob 已提交
579

580
	this.domElement.addEventListener( 'mousewheel', mousewheel, false );
M
Mr.doob 已提交
581 582 583
	this.domElement.addEventListener( 'DOMMouseScroll', mousewheel, false ); // firefox

	this.domElement.addEventListener( 'touchstart', touchstart, false );
584
	this.domElement.addEventListener( 'touchend', touchend, false );
M
Mr.doob 已提交
585
	this.domElement.addEventListener( 'touchmove', touchmove, false );
M
Mr.doob 已提交
586

587 588
	window.addEventListener( 'keydown', keydown, false );
	window.addEventListener( 'keyup', keyup, false );
589

590 591
	this.handleResize();

592 593 594
	// force an update at start
	this.update();

595
};
596

M
Mr.doob 已提交
597
THREE.TrackballControls.prototype = Object.create( THREE.EventDispatcher.prototype );