InteractiveGroup.js 2.4 KB

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