InteractiveGroup.js 3.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. ( function () {
  2. const _pointer = new THREE.Vector2();
  3. const _event = {
  4. type: '',
  5. data: _pointer
  6. };
  7. class InteractiveGroup extends THREE.Group {
  8. constructor( renderer, camera ) {
  9. super();
  10. const scope = this;
  11. const raycaster = new THREE.Raycaster();
  12. const tempMatrix = new THREE.Matrix4();
  13. // Pointer Events
  14. const element = renderer.domElement;
  15. function onPointerEvent( event ) {
  16. event.stopPropagation();
  17. const rect = renderer.domElement.getBoundingClientRect();
  18. _pointer.x = ( event.clientX - rect.left ) / rect.width * 2 - 1;
  19. _pointer.y = - ( event.clientY - rect.top ) / rect.height * 2 + 1;
  20. raycaster.setFromCamera( _pointer, camera );
  21. const intersects = raycaster.intersectObjects( scope.children, false );
  22. if ( intersects.length > 0 ) {
  23. const intersection = intersects[ 0 ];
  24. const object = intersection.object;
  25. const uv = intersection.uv;
  26. _event.type = event.type;
  27. _event.data.set( uv.x, 1 - uv.y );
  28. object.dispatchEvent( _event );
  29. }
  30. }
  31. element.addEventListener( 'pointerdown', onPointerEvent );
  32. element.addEventListener( 'pointerup', onPointerEvent );
  33. element.addEventListener( 'pointermove', onPointerEvent );
  34. element.addEventListener( 'mousedown', onPointerEvent );
  35. element.addEventListener( 'mouseup', onPointerEvent );
  36. element.addEventListener( 'mousemove', onPointerEvent );
  37. element.addEventListener( 'click', onPointerEvent );
  38. // WebXR Controller Events
  39. // TODO: Dispatch pointerevents too
  40. const events = {
  41. 'move': 'mousemove',
  42. 'select': 'click',
  43. 'selectstart': 'mousedown',
  44. 'selectend': 'mouseup'
  45. };
  46. function onXRControllerEvent( event ) {
  47. const controller = event.target;
  48. tempMatrix.identity().extractRotation( controller.matrixWorld );
  49. raycaster.ray.origin.setFromMatrixPosition( controller.matrixWorld );
  50. raycaster.ray.direction.set( 0, 0, - 1 ).applyMatrix4( tempMatrix );
  51. const intersections = raycaster.intersectObjects( scope.children, false );
  52. if ( intersections.length > 0 ) {
  53. const intersection = intersections[ 0 ];
  54. const object = intersection.object;
  55. const uv = intersection.uv;
  56. _event.type = events[ event.type ];
  57. _event.data.set( uv.x, 1 - uv.y );
  58. object.dispatchEvent( _event );
  59. }
  60. }
  61. const controller1 = renderer.xr.getController( 0 );
  62. controller1.addEventListener( 'move', onXRControllerEvent );
  63. controller1.addEventListener( 'select', onXRControllerEvent );
  64. controller1.addEventListener( 'selectstart', onXRControllerEvent );
  65. controller1.addEventListener( 'selectend', onXRControllerEvent );
  66. const controller2 = renderer.xr.getController( 1 );
  67. controller2.addEventListener( 'move', onXRControllerEvent );
  68. controller2.addEventListener( 'select', onXRControllerEvent );
  69. controller2.addEventListener( 'selectstart', onXRControllerEvent );
  70. controller2.addEventListener( 'selectend', onXRControllerEvent );
  71. }
  72. }
  73. THREE.InteractiveGroup = InteractiveGroup;
  74. } )();