File size: 1,363 Bytes
01488bc
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
import { useThree } from "@react-three/fiber";
import { useCallback, useEffect, useRef } from "react";

import { Vector2 } from "three";

type SplatStack = {
  mouseX: number;
  mouseY: number;
  velocityX: number;
  velocityY: number;
};

export const usePointer = ({ force }: { force: number }) => {
  const size = useThree((three) => three.size);

  const splatStack: SplatStack[] = useRef([]).current;
  const lastMouse = useRef<Vector2>(new Vector2());
  const hasMoved = useRef<boolean>(false);

  const onPointerMove = useCallback(
    (event: { x: number; y: number }) => {
      const deltaX = event.x - lastMouse.current.x;
      const deltaY = event.y - lastMouse.current.y;

      if (!hasMoved.current) {
        hasMoved.current = true;
        lastMouse.current.set(event.x, event.y);
        return;
      }

      lastMouse.current.set(event.x, event.y);

      const splatInfo = {
        mouseX: event.x / size.width,
        mouseY: 1.0 - event.y / size.height,
        velocityX: deltaX * force,
        velocityY: -deltaY * force,
      };

      splatStack.push(splatInfo);
    },
    [force, size.height, size.width, splatStack],
  );

  useEffect(() => {
    addEventListener("pointermove", onPointerMove);
    return () => {
      removeEventListener("pointermove", onPointerMove);
    };
  }, [onPointerMove]);

  return splatStack;
};