Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

(Redo) TileControls: Add tiles control events #455

Merged
merged 10 commits into from
Jan 17, 2024
11 changes: 7 additions & 4 deletions example/src/controls/GlobeControls.js
Original file line number Diff line number Diff line change
Expand Up @@ -212,11 +212,12 @@ export class GlobeControls extends TileControls {

}

_updateZoom( delta ) {
_updateZoom() {

if ( this.getDistanceToCenter() < GLOBE_TRANSITION_THRESHOLD || delta > 0 ) {
const scale = this.zoomDelta;
if ( this.getDistanceToCenter() < GLOBE_TRANSITION_THRESHOLD || scale > 0 ) {

super._updateZoom( delta );
super._updateZoom();

} else {

Expand All @@ -227,9 +228,11 @@ export class GlobeControls extends TileControls {

// zoom out directly from the globe center
this.getVectorToCenter( _vec );
this.camera.position.addScaledVector( _vec, delta * 0.0025 );
this.camera.position.addScaledVector( _vec, scale * 0.0025 );
this.camera.updateMatrixWorld();

this.zoomDelta = 0;

}

}
Expand Down
123 changes: 91 additions & 32 deletions example/src/controls/TileControls.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import {
Vector3,
Raycaster,
Plane,
EventDispatcher,
} from 'three';
import { PivotPointMesh } from './PivotPointMesh.js';
import { PointerTracker } from './PointerTracker.js';
Expand Down Expand Up @@ -34,16 +35,23 @@ const _deltaPointer = new Vector2();
const _centerPoint = new Vector2();
const _originalCenterPoint = new Vector2();

export class TileControls {
const _changeEvent = { type: 'change' };
const _startEvent = { type: 'start' };
const _endEvent = { type: 'end' };

export class TileControls extends EventDispatcher {

constructor( scene, camera, domElement ) {

super();

this.domElement = null;
this.camera = null;
this.scene = null;

// settings
this.state = NONE;
this.pinchState = NONE;
this.cameraRadius = 5;
this.rotationSpeed = 5;
this.minAltitude = 0;
Expand All @@ -55,6 +63,7 @@ export class TileControls {

// internal state
this.pointerTracker = new PointerTracker();
this.needsUpdate = false;
this.actionHeightOffset = 0;

this.dragPointSet = false;
Expand All @@ -67,6 +76,7 @@ export class TileControls {
this.zoomPointSet = false;
this.zoomDirection = new Vector3();
this.zoomPoint = new Vector3();
this.zoomDelta = 0;

this.pivotMesh = new PivotPointMesh();
this.pivotMesh.raycast = () => {};
Expand Down Expand Up @@ -112,7 +122,6 @@ export class TileControls {
this.domElement = domElement;
domElement.style.touchAction = 'none';

let pinchAction = NONE;
let shiftClicked = false;

const contextMenuCallback = e => {
Expand Down Expand Up @@ -155,6 +164,7 @@ export class TileControls {

// init the pointer
pointerTracker.addPointer( e );
this.needsUpdate = true;

// handle cases where we need to capture the pointer or
// reset state when we have too many pointers
Expand All @@ -169,7 +179,6 @@ export class TileControls {
} else if ( pointerTracker.getPointerCount() > 2 ) {

this.resetState();
pinchAction = NONE;
return;

}
Expand All @@ -193,7 +202,7 @@ export class TileControls {
pointerTracker.isLeftClicked() && shiftClicked
) {

this.state = ROTATE;
this.setState( ROTATE );
this.rotationPoint.copy( hit.point );
this.rotationPointSet = true;

Expand All @@ -206,7 +215,7 @@ export class TileControls {
// if the clicked point is coming from below the plane then don't perform the drag
if ( raycaster.ray.direction.dot( up ) < 0 ) {

this.state = DRAG;
this.setState( DRAG );
this.dragPoint.copy( hit.point );
this.dragPointSet = true;

Expand All @@ -228,6 +237,7 @@ export class TileControls {
// whenever the pointer moves we need to re-derive the zoom direction and point
this.zoomDirectionSet = false;
this.zoomPointSet = false;
this.needsUpdate = true;

const { pointerTracker } = this;
pointerTracker.setHoverEvent( e );
Expand All @@ -239,17 +249,14 @@ export class TileControls {

if ( pointerTracker.getPointerType() === 'touch' ) {

if ( pointerTracker.getPointerCount() === 1 ) {
if ( pointerTracker.getPointerCount() === 2 ) {

if ( this.state === DRAG ) {

this._updatePosition();
// this.setState( NONE, WAITING, false );
this.setState( NONE, WAITING, false );

}

} else if ( pointerTracker.getPointerCount() === 2 ) {

// We queue this event to ensure that all pointers have been updated
if ( ! _pointerMoveQueued ) {

Expand Down Expand Up @@ -278,27 +285,26 @@ export class TileControls {

if ( Math.abs( separateDelta ) > parallelDelta ) {

this.resetState();
pinchAction = ZOOM;
this.setState( NONE, ZOOM );
this.zoomDirectionSet = false;

} else {

pinchAction = ROTATE;
this.pinchState = ROTATE;
this.setState( NONE, ROTATE );

}

}

}

if ( pinchAction === ZOOM ) {
if ( this.pinchState === ZOOM ) {

this._updateZoom( separateDelta );
this.zoomDelta += separateDelta;

} else if ( pinchAction === ROTATE ) {
} else if ( this.pinchState === ROTATE ) {

this._updateRotation();
this.pivotMesh.visible = true;

}
Expand All @@ -309,18 +315,6 @@ export class TileControls {

}

} else if ( pointerTracker.getPointerType() === 'mouse' ) {

if ( this.state === DRAG ) {

this._updatePosition();

} else if ( this.state === ROTATE ) {

this._updateRotation();

}

}

};
Expand All @@ -330,7 +324,6 @@ export class TileControls {
const { pointerTracker } = this;

pointerTracker.deletePointer( e );
pinchAction = NONE;

if (
pointerTracker.getPointerType() === 'touch' &&
Expand All @@ -342,12 +335,14 @@ export class TileControls {
}

this.resetState();
this.needsUpdate = true;

};

const wheelCallback = e => {

this._updateZoom( - e.deltaY );
this.needsUpdate = true;
this.zoomDelta -= e.deltaY;

};

Expand Down Expand Up @@ -410,7 +405,14 @@ export class TileControls {

resetState() {

if ( this.state !== NONE || this.pinchState !== NONE ) {

this.dispatchEvent( _endEvent );

}

this.state = NONE;
this.pinchState = NONE;
this.dragPointSet = false;
this.rotationPointSet = false;
this.scene.remove( this.pivotMesh );
Expand All @@ -419,15 +421,69 @@ export class TileControls {

}

setState( state = this.state, pinchState = this.pinchState, fireEvent = true ) {

if ( this.state === state && this.pinchState === pinchState ) {

return;

}

if ( this.state === NONE && this.pinchState === NONE && fireEvent ) {

this.dispatchEvent( _startEvent );

}

this.state = state;
this.pinchState = pinchState;

}

update() {

const {
camera,
cameraRadius,
dragPoint,
up,
state,
pinchState,
} = this;

// update the actions
if ( this.needsUpdate ) {

const action = state || pinchState;
const zoomDelta = this.zoomDelta;
if ( action === DRAG ) {

this._updatePosition();

}

if ( action === ROTATE ) {

this._updateRotation();

}

if ( action === ZOOM || zoomDelta !== 0 ) {

this._updateZoom();

}

if ( action !== NONE || zoomDelta !== 0 ) {

this.dispatchEvent( _changeEvent );

}

this.needsUpdate = false;

}

// reuse the "hit" information since it can be slow to perform multiple hits
const hit = this._getPointBelowCamera();
if ( this.getUpDirection ) {
Expand Down Expand Up @@ -489,7 +545,7 @@ export class TileControls {
}

// private
_updateZoom( scale ) {
_updateZoom() {

const {
zoomPoint,
Expand All @@ -502,6 +558,9 @@ export class TileControls {
domElement,
} = this;

let scale = this.zoomDelta;
this.zoomDelta = 0;

// get the latest hover / touch point
if ( ! pointerTracker.getLatestPoint( _pointer ) ) {

Expand Down
Loading