VVK
VVK

Reputation: 445

THREE.JS Mouse interaction with shader

I have simple point cloud shader, which renders points as circles on screen.

enter image description here

vertexShader:

  uniform mat4 modelViewMatrix;
  uniform mat4 projectionMatrix;
  uniform vec3 cameraPosition;
  uniform sampler2D texture;
  uniform vec2 mouse;
  uniform vec2 resolution;

  attribute vec3 position;
  attribute float radius;
  attribute vec3 color;

  varying vec3 vColor;

  void main() {

    vColor = color;

    vec3 pos = position;
    vec4 projected = projectionMatrix * modelViewMatrix * vec4(pos, 1.0);
    gl_Position = projected;
    gl_PointSize = radius;

  }

fraqShader:

precision mediump float;

  varying vec3 vColor;
  uniform sampler2D texture;

  uniform float useColor;
  uniform vec2 mouse;
  uniform vec2 resolution;

  void main() {

    float mx = mouse.x / resolution.x;
    float my = mouse.y / resolution.y;

    float d = sqrt((gl_PointCoord.x - mx)*(gl_PointCoord.x - mx) + (gl_PointCoord.y - mx)*(gl_PointCoord.y - mx));

    if (useColor == 1.) {
      gl_FragColor = vec4(vColor, 1.0);
    } else {
      gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);
    }

    gl_FragColor = gl_FragColor * texture2D(texture, gl_PointCoord);

    if(d < 0.1) { gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0); }

}

The question is: Is possible to make a mouse interaction with these circles, I mean if the distance from current mouse position is less than point radius then the color would be different?

I have tried to set mouse.x and mouse.y to event_.clientX and event_.clinetY, then pass it to the shader trying to calculate distance:

float mx = mouse.x / resolution.x;
float my = mouse.y / resolution.y;

float d = sqrt((gl_PointCoord.x - mx)*(gl_PointCoord.x - mx) + (gl_PointCoord.y - mx)*(gl_PointCoord.y - mx));

But it doesn't work. Is there any solutions?/

Upvotes: 1

Views: 2537

Answers (2)

VVK
VVK

Reputation: 445

var circularPoint = "";

        var scene = new THREE.Scene();
        var camera = new THREE.PerspectiveCamera(45, window.innerWidth / window.innerHeight, 1, 1000);
        camera.position.set(0, 0, 10);
        var renderer = new THREE.WebGLRenderer();
        renderer.setSize(window.innerWidth, window.innerHeight);
        renderer.setClearColor(0xEEEEEE, 1.0);
        document.body.appendChild(renderer.domElement);

        var controls = new THREE.OrbitControls(camera, renderer.domElement);

        var N = 2560;
        
        var pallete = ["#FF6138", "#FFFF9D", "#BEEB9F", "#79BD8F", "#00A388"];
        
        var verts = [], colors = [], rad = [], size = [], id = [], enabled = [];
        
        for (let i = 0; i < N; i++) {
            
            verts.push(getXYZ().multiplyScalar(5)); 
            size.push(0.25 + Math.random() * 1.25);
            rad.push(size[size.length - 1] * 1.0E-1 );
            colors.push.apply(colors, randomRGB());
            enabled.push(1);
            
            var indx = new THREE.Color().setHex((i + 1));
            id.push(indx.r, indx.g, indx.b);

        }

        var geometry = new THREE.BufferGeometry().setFromPoints(verts);
        geometry.addAttribute("color", new THREE.BufferAttribute(new Float32Array(colors), 3));
        geometry.addAttribute("id", new THREE.BufferAttribute(new Float32Array(id), 3));
        geometry.addAttribute("size", new THREE.BufferAttribute(new Float32Array(size), 1));
        geometry.addAttribute("rad", new THREE.BufferAttribute(new Float32Array(rad), 1));
        geometry.addAttribute("enabled", new THREE.BufferAttribute(new Float32Array(enabled), 1));

        var material = new THREE.ShaderMaterial({

            uniforms: {

                texture: {
                    value: new THREE.TextureLoader().load(circularPoint)
                },
                ori: {
                    value: new THREE.Vector3()
                },
                dir: {
                    value: new THREE.Vector3()
                },
                scale: {
                    value: window.innerHeight / 2
                }
            },
            vertexShader: document.getElementById('vertexShader').textContent,
            fragmentShader: document.getElementById('fragmentShader').textContent,
            alphaTest: 0.9


        })

        var last_id = 0;
        
        material.extensions.fragDepth = true;
        material.extensions.drawBuffers = true;

        var points = new THREE.Points(geometry, material);
        scene.add(points);

        var raycaster = new THREE.Raycaster();
        var mouse = new THREE.Vector2();
        var inverseMatrix = new THREE.Matrix4();
        var ray = new THREE.Ray();

        pickingScene = new THREE.Scene();
        pickingTexture = new THREE.WebGLRenderTarget(window.innerWidth, window.innerHeight);
        pickingTexture.texture.minFilter = THREE.LinearFilter;
        pickingScene.add(points.clone());

        renderer.domElement.addEventListener("mousemove", onMouseMove, false);

        function onMouseMove(event) {
            
            mouse.x = (event.clientX / window.innerWidth) * 2 - 1;
            mouse.y = -(event.clientY / window.innerHeight) * 2 + 1;

            raycaster.setFromCamera(mouse, camera);

            inverseMatrix.getInverse(points.matrixWorld);
            ray.copy(raycaster.ray).applyMatrix4(inverseMatrix);

            material.uniforms.ori.value = ray.origin;
            material.uniforms.dir.value = ray.direction;
            
            renderer.render(pickingScene, camera, pickingTexture);
            var pixelBuffer = new Uint8Array(4);
            renderer.readRenderTargetPixels(
            pickingTexture, event.clientX, pickingTexture.height - event.clientY,
            1, 1, pixelBuffer);
   
             
            var id = (pixelBuffer[0] << 16) | (pixelBuffer[1] << 8) | (pixelBuffer[2]);

            if(id < N){
                
                last_id = id;
                console.log("highlighted node: " + id);
                
                for(var i = 0; i < N; i++){ if(i != (id)) { enabled[i] = 0.0; } }
                points.geometry.attributes.enabled.needsUpdate = true;
            
                
            }else if(id != last_id){
                
                for(var i = 0; i < N; i++){  enabled[i] = 1.0; }
                points.geometry.attributes.enabled.needsUpdate = true;
 
            }
            
        }

        renderer.setAnimationLoop(() => { renderer.render(scene, camera) });
        
        function getXYZ(){
            
            var n = 1E1;
            var rho = Math.random();
            var theta = Math.random() * Math.PI * 2;
            var phi = Math.random() * Math.PI * 2;
            var x = rho * Math.cos(phi) * Math.sin(theta);
            var y = rho * Math.sin(phi) * Math.sin(theta);
            var z = rho * Math.cos(theta);

            return new THREE.Vector3(x, y, z);
            
        }
        
        function randomRGB() {

            var i = Math.floor(Math.random() * 5);
            var hex = pallete[i];
            
            var result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
            return result ? [

            parseInt(result[1], 16) / 255,
            parseInt(result[2], 16) / 255,
            parseInt(result[3], 16) / 255

            ] : null;

        }
body {
        overflow: hidden;
        margin: 0;
        }
<script src="https://cdn.jsdelivr.net/npm/[email protected]/build/three.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/[email protected]/examples/js/controls/OrbitControls.js"></script>


<script type='x-shader/x-vertex' id='vertexShader'>

    uniform vec3 ori;
    uniform vec3 dir;
    attribute float rad;
    attribute float size;
    attribute vec3 color;
    uniform float scale;
    attribute float enabled;
    attribute vec3 id;
    
    varying vec3 vColor;
    
    vec3 closestPointToPoint() {
    
      vec3 target = position - ori;
      float distance = dot(target, dir);
      return dir * distance + ori;
      
    }
    
    void main() {
    
        vColor = color;
        
        if (length(position - closestPointToPoint()) < rad) if(enabled == 1.) { vColor = id; }
        
        vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
        gl_PointSize = size * ( scale / -mvPosition.z );
        gl_Position = projectionMatrix * mvPosition;
        
    }


</script>

<script type='x-shader/x-fragment' id='fragmentShader'>

    varying vec3 vColor;
    uniform sampler2D texture;
    
    void main() { 
    
        gl_FragColor = vec4(vColor, 1.);
        gl_FragColor = gl_FragColor * texture2D(texture, gl_PointCoord);
        if (gl_FragColor.a < 0.1) discard;

    }


</script>

That's the result I was looking for, but any tweaks are welcome. Thanks to prisoner849, it's mainly his code.

Upvotes: 0

Rabbid76
Rabbid76

Reputation: 211176

gl_FragCoord.xy contains the window coordinates of the fragment. The lower left is (0,0) and the upper right is the width and height of the viewport in pixels.

Probably you have to flip the y coordinate of the mouse coordinates, because at screen coordinates the upper left is (0, 0) and the bottom right is the width and height of the window:

vec2 mc = vec2(mouse.x, u_resolution.y - mouse.y);
float d = length((mc - gl_FragCoord.xy) / u_resolutuon.xy);

Upvotes: 2

Related Questions