TrackballControls.js 13.1 KB
Newer Older
1
/**
2
 * @author Eberhard Graether / http://egraether.com/
M
Mark Lundin 已提交
3
 * @author Mark Lundin 	/ http://mark-lundin.com
4 5
 * @author Simone Manini / http://daron1337.github.io
 * @author Luca Antiga 	/ http://lantiga.github.io
6 7
 */

8
THREE.TrackballControls = function ( object, domElement ) {
9

M
Mr.doob 已提交
10
	var _this = this;
M
Mr.doob 已提交
11
	var STATE = { NONE: -1, ROTATE: 0, ZOOM: 1, PAN: 2, TOUCH_ROTATE: 3, TOUCH_ZOOM_PAN: 4 };
12

13
	this.object = object;
14
	this.domElement = ( domElement !== undefined ) ? domElement : document;
15

16
	// API
17

18 19
	this.enabled = true;

20
	this.screen = { left: 0, top: 0, width: 0, height: 0 };
21

22 23 24
	this.rotateSpeed = 1.0;
	this.zoomSpeed = 1.2;
	this.panSpeed = 0.3;
25

26
	this.noRotate = false;
27 28
	this.noZoom = false;
	this.noPan = false;
29

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

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

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

38
	// internals
39

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

42 43
	var EPS = 0.000001;

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

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

49 50
	_eye = new THREE.Vector3(),

51 52 53 54 55 56
	_movePrev = new THREE.Vector2(),
	_moveCurr = new THREE.Vector2(),

	_lastAxis = new THREE.Vector3(),
	_lastAngle = 0,

57 58 59
	_zoomStart = new THREE.Vector2(),
	_zoomEnd = new THREE.Vector2(),

60 61 62
	_touchZoomDistanceStart = 0,
	_touchZoomDistanceEnd = 0,

63 64
	_panStart = new THREE.Vector2(),
	_panEnd = new THREE.Vector2();
65

W
WestLangley 已提交
66 67 68 69 70 71
	// for reset

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

M
Mr.doob 已提交
72 73 74
	// events

	var changeEvent = { type: 'change' };
G
gero3 已提交
75 76
	var startEvent = { type: 'start' };
	var endEvent = { type: 'end' };
M
Mr.doob 已提交
77

78

79
	// methods
80

81 82
	this.handleResize = function () {

83
		if ( this.domElement === document ) {
84

85 86 87 88 89 90 91
			this.screen.left = 0;
			this.screen.top = 0;
			this.screen.width = window.innerWidth;
			this.screen.height = window.innerHeight;

		} else {

92
			var box = this.domElement.getBoundingClientRect();
93
			// adjustments come from similar code in the jquery offset() function
94 95 96 97 98
			var d = this.domElement.ownerDocument.documentElement;
			this.screen.left = box.left + window.pageXOffset - d.clientLeft;
			this.screen.top = box.top + window.pageYOffset - d.clientTop;
			this.screen.width = box.width;
			this.screen.height = box.height;
99 100

		}
101

102 103
	};

104
	this.handleEvent = function ( event ) {
105

106
		if ( typeof this[ event.type ] == 'function' ) {
107

108
			this[ event.type ]( event );
109

110
		}
111

112
	};
113

M
Mr.doob 已提交
114
	var getMouseOnScreen = ( function () {
115

116
		var vector = new THREE.Vector2();
117

118
		return function getMouseOnScreen( pageX, pageY ) {
M
Mr.doob 已提交
119 120 121 122 123 124 125

			vector.set(
				( pageX - _this.screen.left ) / _this.screen.width,
				( pageY - _this.screen.top ) / _this.screen.height
			);

			return vector;
126

M
Mr.doob 已提交
127
		};
128

M
Mr.doob 已提交
129
	}() );
130

131 132 133 134
	var getMouseOnCircle = ( function () {

		var vector = new THREE.Vector2();

135
		return function getMouseOnCircle( pageX, pageY ) {
136 137 138

			vector.set(
				( ( pageX - _this.screen.width * 0.5 - _this.screen.left ) / ( _this.screen.width * 0.5 ) ),
139
				( ( _this.screen.height + 2 * ( _this.screen.top - pageY ) ) / _this.screen.width ) // screen.width intentional
140 141 142 143 144 145 146
			);

			return vector;
		};

	}() );

G
gero3 已提交
147
	this.rotateCamera = (function() {
148

149
		var axis = new THREE.Vector3(),
150 151 152 153 154 155
			quaternion = new THREE.Quaternion(),
			eyeDirection = new THREE.Vector3(),
			objectUpDirection = new THREE.Vector3(),
			objectSidewaysDirection = new THREE.Vector3(),
			moveDirection = new THREE.Vector3(),
			angle;
156

157
		return function rotateCamera() {
158

159 160
			moveDirection.set( _moveCurr.x - _movePrev.x, _moveCurr.y - _movePrev.y, 0 );
			angle = moveDirection.length();
161

162
			if ( angle ) {
163

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

166 167 168
				eyeDirection.copy( _eye ).normalize();
				objectUpDirection.copy( _this.object.up ).normalize();
				objectSidewaysDirection.crossVectors( objectUpDirection, eyeDirection ).normalize();
169

170 171
				objectUpDirection.setLength( _moveCurr.y - _movePrev.y );
				objectSidewaysDirection.setLength( _moveCurr.x - _movePrev.x );
172

173
				moveDirection.copy( objectUpDirection.add( objectSidewaysDirection ) );
174

175
				axis.crossVectors( moveDirection, _eye ).normalize();
176

177 178
				angle *= _this.rotateSpeed;
				quaternion.setFromAxisAngle( axis, angle );
179

180 181
				_eye.applyQuaternion( quaternion );
				_this.object.up.applyQuaternion( quaternion );
182

183 184
				_lastAxis.copy( axis );
				_lastAngle = angle;
185

186
			} else if ( !_this.staticMoving && _lastAngle ) {
187

188 189 190 191 192
				_lastAngle *= Math.sqrt( 1.0 - _this.dynamicDampingFactor );
				_eye.copy( _this.object.position ).sub( _this.target );
				quaternion.setFromAxisAngle( _lastAxis, _lastAngle );
				_eye.applyQuaternion( quaternion );
				_this.object.up.applyQuaternion( quaternion );
193

194
			}
195

196 197
			_movePrev.copy( _moveCurr );

198
		};
199

200
	}());
201

202

M
Mr.doob 已提交
203
	this.zoomCamera = function () {
204

205 206
		var factor;

M
Mr.doob 已提交
207
		if ( _state === STATE.TOUCH_ZOOM_PAN ) {
208

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

213
		} else {
214

215
			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

G
gero3 已提交
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 panCamera() {
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 ) );
M
michaelg 已提交
279
				_zoomStart.copy( _zoomEnd );
280 281 282

			}

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

285
				_this.object.position.addVectors( _this.target, _eye.setLength( _this.minDistance ) );
M
michaelg 已提交
286
				_zoomStart.copy( _zoomEnd );
287 288

			}
289 290 291

		}

292 293
	};

M
Mr.doob 已提交
294
	this.update = function () {
M
Mr.doob 已提交
295

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

		if ( !_this.noRotate ) {

			_this.rotateCamera();

		}
303

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

			_this.zoomCamera();

		}

		if ( !_this.noPan ) {

			_this.panCamera();

		}

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

		_this.checkDistances();

		_this.object.lookAt( _this.target );

322
		if ( lastPosition.distanceToSquared( _this.object.position ) > EPS ) {
323

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

			lastPosition.copy( _this.object.position );

		}

	};

W
WestLangley 已提交
332 333 334 335 336 337 338 339 340 341 342 343 344
	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 );

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

		lastPosition.copy( _this.object.position );

	};

351
	// listeners
352

353
	function keydown( event ) {
354

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

M
Mr.doob 已提交
357
		window.removeEventListener( 'keydown', keydown );
358

M
Mr.doob 已提交
359
		_prevState = _state;
360

361
		if ( _state !== STATE.NONE ) {
362

363
			return;
364

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

367
			_state = STATE.ROTATE;
368

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

371
			_state = STATE.ZOOM;
372

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

375
			_state = STATE.PAN;
376 377

		}
378

M
Mr.doob 已提交
379
	}
380

381
	function keyup( event ) {
382

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

M
Mr.doob 已提交
385
		_state = _prevState;
386

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

M
Mr.doob 已提交
389
	}
390

391
	function mousedown( event ) {
392

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

395 396
		event.preventDefault();
		event.stopPropagation();
397

398
		if ( _state === STATE.NONE ) {
399

400
			_state = event.button;
401

M
Mr.doob 已提交
402
		}
403

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

406 407
			_moveCurr.copy( getMouseOnCircle( event.pageX, event.pageY ) );
			_movePrev.copy(_moveCurr);
408

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

M
Mr.doob 已提交
411
			_zoomStart.copy( getMouseOnScreen( event.pageX, event.pageY ) );
G
gero3 已提交
412
			_zoomEnd.copy(_zoomStart);
413

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

M
Mr.doob 已提交
416
			_panStart.copy( getMouseOnScreen( event.pageX, event.pageY ) );
417
			_panEnd.copy(_panStart);
418

419
		}
M
Mr.doob 已提交
420

M
Mr.doob 已提交
421 422
		document.addEventListener( 'mousemove', mousemove, false );
		document.addEventListener( 'mouseup', mouseup, false );
423

M
Mr.doob 已提交
424
		_this.dispatchEvent( startEvent );
425

M
Mr.doob 已提交
426
	}
427

428
	function mousemove( event ) {
429

430 431 432 433
		if ( _this.enabled === false ) return;

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

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

437 438
			_movePrev.copy(_moveCurr);
			_moveCurr.copy( getMouseOnCircle( event.pageX, event.pageY ) );
439

440
		} else if ( _state === STATE.ZOOM && !_this.noZoom ) {
441

M
Mr.doob 已提交
442
			_zoomEnd.copy( getMouseOnScreen( event.pageX, event.pageY ) );
443

444
		} else if ( _state === STATE.PAN && !_this.noPan ) {
445

M
Mr.doob 已提交
446
			_panEnd.copy( getMouseOnScreen( event.pageX, event.pageY ) );
447

448
		}
449

M
Mr.doob 已提交
450
	}
451

452
	function mouseup( event ) {
453

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

456 457
		event.preventDefault();
		event.stopPropagation();
458

459
		_state = STATE.NONE;
460

M
Mr.doob 已提交
461 462
		document.removeEventListener( 'mousemove', mousemove );
		document.removeEventListener( 'mouseup', mouseup );
463
		_this.dispatchEvent( endEvent );
464

M
Mr.doob 已提交
465
	}
466

467 468
	function mousewheel( event ) {

469
		if ( _this.enabled === false ) return;
470 471 472 473 474 475

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

		var delta = 0;

476 477 478 479 480 481 482 483
		if ( event.wheelDelta ) { // WebKit / Opera / Explorer 9

			delta = event.wheelDelta / 40;

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

			delta = - event.detail / 3;

484 485
		}

486
		_zoomStart.y += delta * 0.01;
487 488
		_this.dispatchEvent( startEvent );
		_this.dispatchEvent( endEvent );
489 490 491

	}

M
Mr.doob 已提交
492 493
	function touchstart( event ) {

494
		if ( _this.enabled === false ) return;
M
Mr.doob 已提交
495 496 497 498

		switch ( event.touches.length ) {

			case 1:
499
				_state = STATE.TOUCH_ROTATE;
500 501
				_moveCurr.copy( getMouseOnCircle( event.touches[ 0 ].pageX, event.touches[ 0 ].pageY ) );
				_movePrev.copy(_moveCurr);
M
Mr.doob 已提交
502
				break;
503

M
Mr.doob 已提交
504
			case 2:
M
Mr.doob 已提交
505
				_state = STATE.TOUCH_ZOOM_PAN;
506 507 508
				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 已提交
509 510 511

				var x = ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX ) / 2;
				var y = ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY ) / 2;
512
				_panStart.copy( getMouseOnScreen( x, y ) );
M
Mr.doob 已提交
513
				_panEnd.copy( _panStart );
M
Mr.doob 已提交
514 515
				break;

516 517 518
			default:
				_state = STATE.NONE;

M
Mr.doob 已提交
519
		}
520 521
		_this.dispatchEvent( startEvent );

M
Mr.doob 已提交
522 523 524 525 526

	}

	function touchmove( event ) {

527
		if ( _this.enabled === false ) return;
M
Mr.doob 已提交
528 529

		event.preventDefault();
530
		event.stopPropagation();
M
Mr.doob 已提交
531 532 533 534

		switch ( event.touches.length ) {

			case 1:
535 536
				_movePrev.copy(_moveCurr);
				_moveCurr.copy( getMouseOnCircle(  event.touches[ 0 ].pageX, event.touches[ 0 ].pageY ) );
M
Mr.doob 已提交
537
				break;
538

M
Mr.doob 已提交
539
			case 2:
540 541
				var dx = event.touches[ 0 ].pageX - event.touches[ 1 ].pageX;
				var dy = event.touches[ 0 ].pageY - event.touches[ 1 ].pageY;
542
				_touchZoomDistanceEnd = Math.sqrt( dx * dx + dy * dy );
M
Mr.doob 已提交
543 544 545 546

				var x = ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX ) / 2;
				var y = ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY ) / 2;
				_panEnd.copy( getMouseOnScreen( x, y ) );
M
Mr.doob 已提交
547 548
				break;

549 550 551 552 553 554 555 556 557 558 559 560 561 562
			default:
				_state = STATE.NONE;

		}

	}

	function touchend( event ) {

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

		switch ( event.touches.length ) {

			case 1:
563 564
				_movePrev.copy(_moveCurr);
				_moveCurr.copy( getMouseOnCircle(  event.touches[ 0 ].pageX, event.touches[ 0 ].pageY ) );
565 566 567 568
				break;

			case 2:
				_touchZoomDistanceStart = _touchZoomDistanceEnd = 0;
M
Mr.doob 已提交
569

570
				var x = ( event.touches[ 0 ].pageX + event.touches[ 1 ].pageX ) / 2;
M
Mr.doob 已提交
571 572 573
				var y = ( event.touches[ 0 ].pageY + event.touches[ 1 ].pageY ) / 2;
				_panEnd.copy( getMouseOnScreen( x, y ) );
				_panStart.copy( _panEnd );
574 575
				break;

M
Mr.doob 已提交
576 577
		}

578
		_state = STATE.NONE;
579
		_this.dispatchEvent( endEvent );
580

M
Mr.doob 已提交
581 582
	}

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

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

587
	this.domElement.addEventListener( 'mousewheel', mousewheel, false );
M
Mr.doob 已提交
588 589 590
	this.domElement.addEventListener( 'DOMMouseScroll', mousewheel, false ); // firefox

	this.domElement.addEventListener( 'touchstart', touchstart, false );
591
	this.domElement.addEventListener( 'touchend', touchend, false );
M
Mr.doob 已提交
592
	this.domElement.addEventListener( 'touchmove', touchmove, false );
M
Mr.doob 已提交
593

594 595
	window.addEventListener( 'keydown', keydown, false );
	window.addEventListener( 'keyup', keyup, false );
596

597 598
	this.handleResize();

599 600 601
	// force an update at start
	this.update();

602
};
603

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