InteractiveGroup.js 2.9 KB

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