Mouse / Canvas X, Y to Three.js World X, Y, Z
Asked Answered
W

11

88

I've searched around for an example that matches my use case but cannot find one. I'm trying to convert screen mouse co-ordinates into 3D world co-ordinates taking into account the camera.

Solutions I've found all do ray intersection to achieve object picking.

What I am trying to do is position the center of a Three.js object at the co-ordinates that the mouse is currently "over".

My camera is at x:0, y:0, z:500 (although it will move during the simulation) and all my objects are at z = 0 with varying x and y values so I need to know the world X, Y based on assuming a z = 0 for the object that will follow the mouse position.

This question looks like a similar issue but doesn't have a solution: Getting coordinates of the mouse in relation to 3D space in THREE.js

Given the mouse position on screen with a range of "top-left = 0, 0 | bottom-right = window.innerWidth, window.innerHeight", can anyone provide a solution to move a Three.js object to the mouse co-ordinates along z = 0?

Weksler answered 24/10, 2012 at 18:12 Comment(2)
Hey Rob fancy running into you here :)Intracardiac
Hi could you post a little jsfiddle for this case?Silverstein
I
162

You do not need to have any objects in your scene to do this.

You already know the camera position.

Using vector.unproject( camera ) you can get a ray pointing in the direction you want.

You just need to extend that ray, from the camera position, until the z-coordinate of the tip of the ray is zero.

You can do that like so:

var vec = new THREE.Vector3(); // create once and reuse
var pos = new THREE.Vector3(); // create once and reuse

vec.set(
  ( event.clientX / window.innerWidth ) * 2 - 1,
  - ( event.clientY / window.innerHeight ) * 2 + 1,
  0.5,
);
    
vec.unproject( camera );
    
vec.sub( camera.position ).normalize();
    
var distance = - camera.position.z / vec.z;
    
pos.copy( camera.position ).add( vec.multiplyScalar( distance ) );

The variable pos is the position of the point in 3D space, "under the mouse", and in the plane z=0.


EDIT: If you need the point "under the mouse" and in the plane z = targetZ, replace the distance computation with:

var distance = ( targetZ - camera.position.z ) / vec.z;

three.js r.98

Irrelative answered 26/10, 2012 at 17:35 Comment(28)
Perfect answer to the same question I had. Could mention that projector can just be instantiated and doesnt need to be set up in anyway - projector = new THREE.Projector();Rotenone
Not sure why, but after using projector.unprojectVector() function, I'm getting NaN for x,y & z. I'm on r54Rubdown
@Rubdown - You need to make a new post.Irrelative
after applying zoom, some irrelevant position giving?Jetty
This is super helpful—thanks! Having trouble figuring out where in this code I can specify the value of z for the intersecting plane (e.g., I need a pos in the plane z=100). Any ideas?Hearts
I think I figured it out. If you replace var distance = -camera.position.z / dir.z; with var distance = (targetZ - camera.position.z) / dir.z;, you can specify the z value (as targetZ).Hearts
How could this answer be used in a world where the camera is facing down the Y-Axis at an XZ-Plane?Richella
What are clientX and clientY in this context?Hutcherson
@Irrelative - Thanks for the great answer. I have only one doubt, could you please explain why is the z coordinate of vector set to 0.5 in line 6? Would that line also be different when we want to specify a value of z different from 0 (as in SCCOTTT's case)?Waggish
@Waggish The code is "unprotecting" a point from Normalized Device Coordinate (NDC) space to world space. The 0.5 value is arbitrary. Google NDC space if you do not understand the concept.Irrelative
@Irrelative - Thanks for elaborating, I want to share this link that also helped me figure things out. The idea is that everything in the threejs space can be described with coordinates xyz between -1 and 1 (NDC). For the x and y that is done by renormalizing the event.client variables, for z it is done by selecting an arbitrary value between -1 and 1 (i.e. 0.5). The value chosen has no effect on the rest of the code since we are using it to define a direction along a ray.Waggish
@Irrelative - one last thing. Is it correct to state that the unproject function is returning the point in the threejs space that projects to the 2d point with coordinates x,y given in the vector? There is a ray of points that map onto that 2d point and so we ask for the one with specified z value.Waggish
@Waggish Yes.... But be sure not to confuse an orthographic camera frustum (as discussed in your link) with NDC space. They are both boxes, but they are different concepts, and they have different units.Irrelative
Hi could you post a little jsfiddle for this case?Silverstein
THANK YOU. This is finally a full example, there are a few like this but not noob friendly. Got it working, thank you!Houk
fun fact: if you forget camera.position.set(0, 0, 5);camera.lookAt(scene.position); pos is 0,0,0. fun fact2: if you are having trouble trying to train the mouse to a mesh that has a position of say, -300, try hard coding the distance to 300.Interview
Just one side comment - var dir = ... operation is performed without cloning, so dir == vector. I would prefer to leave dir out completely if cloning was not performed, so operate purely on "vector.". Otherwise this might confuse the reader of code.Constant
@Constant Agreed. Fixed, and avoided cloning altogether.Irrelative
My three.js scene doesn't take up the full page, rather, it's within a square <div> with a bunch of surrounding margins and padding. Is there a reliable way to convert these mouse movements to scene co-ordinates that start at (0,0) at one corner of the three.js scene? Pixi.js handles this case somehow, with event.data.getLocalPosition().Conventionalism
@Conventionalism vec.x and vec.y in this answer must return -1 and +1 on the edges of the canvas; 0 in the center. See this answer for how to ensure that when the canvas is inside a div.Irrelative
Hm, if I use this solution for a flat scene with camera.z = 1000, with the targetX worked in, the pos vector is (z:0,y:-0,z:0) no matter what my event coordinates are.Allaround
Can someone explain what the code line "vec.sub( camera.position ).normalize();" is doing? Why the substruction?Siberia
I have problems to understand the substruction of the camera position from the ray direction, followed by adding the result to the camera position. Can you please add a vector diagram that explains what these operations do?Siberia
@AvnerMoshkovitz If you are unfamiliar with these concepts, I suggest you post a question on discourse.threejs.org.Irrelative
I posted a question here. ThanksSiberia
What is the meaning of "- camera.position.z / vec.z"?Midbrain
I'm not using ThreeJs in full screen, then, I used event.offsetX and event.offsetY instead of event.clientX and event.clientYChopfallen
when vec.z = 0 then we get divide by 0 error which results to NaN in var distance = - camera.position.z / vec.z; - also it will result Infinity when the vec.z is very small. so question here - how to deal with this? @IrrelativeColoquintida
M
14

This worked for me when using an orthographic camera

let vector = new THREE.Vector3();
vector.set(
  (event.clientX / window.innerWidth) * 2 - 1,
  - (event.clientY / window.innerHeight) * 2 + 1,
  0,
);
vector.unproject(camera);

WebGL three.js r.89

Michell answered 2/1, 2018 at 22:12 Comment(3)
Worked for me, for an orthographic camera. Thanks! (This other one here works too, but it's not as simple as your solution: https://mcmap.net/q/236879/-mouse-canvas-x-y-to-three-js-world-x-y-z. But that one should work for non-top-down cameras, whereas this one I'm not sure if it would.)Phaih
If using React Three Fiber, you can shorten this even more. The formulas above convert the cursor position to NDC space before unprojecting (threejs.org/docs/#api/en/math/Vector3.unproject), but RTF already calculates this for you in useThree().mouse (i.e. const { mouse } = useThree(); vector.set(mouse.x, mouse.y, 0); vector.unproject(camera);)Incivility
I found the calculations to work a little better when I used the canvas' width/height instead of the windows width/height.Remediless
H
8

In r.58 this code works for me:

var planeZ = new THREE.Plane(new THREE.Vector3(0, 0, 1), 0);
var mv = new THREE.Vector3(
    (event.clientX / window.innerWidth) * 2 - 1,
    -(event.clientY / window.innerHeight) * 2 + 1,
    0.5 );
var raycaster = projector.pickingRay(mv, camera);
var pos = raycaster.ray.intersectPlane(planeZ);
console.log("x: " + pos.x + ", y: " + pos.y);
Hurless answered 2/7, 2013 at 11:7 Comment(5)
Why 0.5? Looks as though the 0.5 can be anything because it is in the direction of the normal. I've tried it with other numbers and it doesn't seem to make any difference.Villenage
To me, this solution is the cleanest. @ChrisSeddon: The z-coordinate is immediately overwritten in the pickingRay method.Ironlike
pickingRay has been removed so this doesn't work with the most recent version (as of 29/10/2014)Sennet
It says replaced with raycaster.setFromCamera but that's not from a projector use new THREE.Raycaster();Emileeemili
This works, but I found an even simpler solution here (might only work for top-down camera, though): https://mcmap.net/q/236879/-mouse-canvas-x-y-to-three-js-world-x-y-zPhaih
Y
5

Below is an ES6 class I wrote based on WestLangley's reply, which works perfectly for me in THREE.js r77.

Note that it assumes your render viewport takes up your entire browser viewport.

class CProjectMousePosToXYPlaneHelper
{
    constructor()
    {
        this.m_vPos = new THREE.Vector3();
        this.m_vDir = new THREE.Vector3();
    }

    Compute( nMouseX, nMouseY, Camera, vOutPos )
    {
        let vPos = this.m_vPos;
        let vDir = this.m_vDir;

        vPos.set(
            -1.0 + 2.0 * nMouseX / window.innerWidth,
            -1.0 + 2.0 * nMouseY / window.innerHeight,
            0.5
        ).unproject( Camera );

        // Calculate a unit vector from the camera to the projected position
        vDir.copy( vPos ).sub( Camera.position ).normalize();

        // Project onto z=0
        let flDistance = -Camera.position.z / vDir.z;
        vOutPos.copy( Camera.position ).add( vDir.multiplyScalar( flDistance ) );
    }
}

You can use the class like this:

// Instantiate the helper and output pos once.
let Helper = new CProjectMousePosToXYPlaneHelper();
let vProjectedMousePos = new THREE.Vector3();

...

// In your event handler/tick function, do the projection.
Helper.Compute( e.clientX, e.clientY, Camera, vProjectedMousePos );

vProjectedMousePos now contains the projected mouse position on the z=0 plane.

Yusuk answered 5/6, 2016 at 19:14 Comment(0)
V
3

I had a canvas that was smaller than my full window, and needed to determine the world coordinates of a click:

// get the position of a canvas event in world coords
function getWorldCoords(e) {
  // get x,y coords into canvas where click occurred
  var rect = canvas.getBoundingClientRect(),
      x = e.clientX - rect.left,
      y = e.clientY - rect.top;
  // convert x,y to clip space; coords from top left, clockwise:
  // (-1,1), (1,1), (-1,-1), (1, -1)
  var mouse = new THREE.Vector3();
  mouse.x = ( (x / canvas.clientWidth ) * 2) - 1;
  mouse.y = (-(y / canvas.clientHeight) * 2) + 1;
  mouse.z = 0.5; // set to z position of mesh objects
  // reverse projection from 3D to screen
  mouse.unproject(camera);
  // convert from point to a direction
  mouse.sub(camera.position).normalize();
  // scale the projected ray
  var distance = -camera.position.z / mouse.z,
      scaled = mouse.multiplyScalar(distance),
      coords = camera.position.clone().add(scaled);
  return coords;
}

var canvas = renderer.domElement;
canvas.addEventListener('click', getWorldCoords);

Here's an example. Click the same region of the donut before and after sliding and you'll find the coords remain constant (check the browser console):

// three.js boilerplate
var container = document.querySelector('body'),
    w = container.clientWidth,
    h = container.clientHeight,
    scene = new THREE.Scene(),
    camera = new THREE.PerspectiveCamera(75, w/h, 0.001, 100),
    controls = new THREE.MapControls(camera, container),
    renderConfig = {antialias: true, alpha: true},
    renderer = new THREE.WebGLRenderer(renderConfig);
controls.panSpeed = 0.4;
camera.position.set(0, 0, -10);
renderer.setPixelRatio(window.devicePixelRatio);
renderer.setSize(w, h);
container.appendChild(renderer.domElement);

window.addEventListener('resize', function() {
  w = container.clientWidth;
  h = container.clientHeight;
  camera.aspect = w/h;
  camera.updateProjectionMatrix();
  renderer.setSize(w, h);
})

function render() {
  requestAnimationFrame(render);
  renderer.render(scene, camera);
  controls.update();
}

// draw some geometries
var geometry = new THREE.TorusGeometry( 10, 3, 16, 100, );
var material = new THREE.MeshNormalMaterial( { color: 0xffff00, } );
var torus = new THREE.Mesh( geometry, material, );
scene.add( torus );

// convert click coords to world space
// get the position of a canvas event in world coords
function getWorldCoords(e) {
  // get x,y coords into canvas where click occurred
  var rect = canvas.getBoundingClientRect(),
      x = e.clientX - rect.left,
      y = e.clientY - rect.top;
  // convert x,y to clip space; coords from top left, clockwise:
  // (-1,1), (1,1), (-1,-1), (1, -1)
  var mouse = new THREE.Vector3();
  mouse.x = ( (x / canvas.clientWidth ) * 2) - 1;
  mouse.y = (-(y / canvas.clientHeight) * 2) + 1;
  mouse.z = 0.0; // set to z position of mesh objects
  // reverse projection from 3D to screen
  mouse.unproject(camera);
  // convert from point to a direction
  mouse.sub(camera.position).normalize();
  // scale the projected ray
  var distance = -camera.position.z / mouse.z,
      scaled = mouse.multiplyScalar(distance),
      coords = camera.position.clone().add(scaled);
  console.log(mouse, coords.x, coords.y, coords.z);
}

var canvas = renderer.domElement;
canvas.addEventListener('click', getWorldCoords);

render();
html,
body {
  width: 100%;
  height: 100%;
  background: #000;
}
body {
  margin: 0;
  overflow: hidden;
}
canvas {
  width: 100%;
  height: 100%;
}
<script src='https://cdnjs.cloudflare.com/ajax/libs/three.js/97/three.min.js'></script>
<script src=' https://threejs.org/examples/js/controls/MapControls.js'></script>
Vagrant answered 2/6, 2019 at 15:52 Comment(2)
For me, this snippet displays a black screen that doesn't change.Interpose
Probably because the three.js api has changed. The code above would only work with three.js version 97Vagrant
J
2

to get the mouse coordinates of a 3d object use projectVector:

var width = 640, height = 480;
var widthHalf = width / 2, heightHalf = height / 2;

var projector = new THREE.Projector();
var vector = projector.projectVector( object.matrixWorld.getPosition().clone(), camera );

vector.x = ( vector.x * widthHalf ) + widthHalf;
vector.y = - ( vector.y * heightHalf ) + heightHalf;

to get the three.js 3D coordinates that relate to specific mouse coordinates, use the opposite, unprojectVector:

var elem = renderer.domElement, 
    boundingRect = elem.getBoundingClientRect(),
    x = (event.clientX - boundingRect.left) * (elem.width / boundingRect.width),
    y = (event.clientY - boundingRect.top) * (elem.height / boundingRect.height);

var vector = new THREE.Vector3( 
    ( x / WIDTH ) * 2 - 1, 
    - ( y / HEIGHT ) * 2 + 1, 
    0.5 
);

projector.unprojectVector( vector, camera );
var ray = new THREE.Ray( camera.position, vector.subSelf( camera.position ).normalize() );
var intersects = ray.intersectObjects( scene.children );

There is a great example here. However, to use project vector, there must be an object where the user clicked. intersects will be an array of all objects at the location of the mouse, regardless of their depth.

Johnette answered 24/10, 2012 at 18:18 Comment(7)
Cool, so then I assign the object's position to x: vector.x, y: vector.y, z:0?Weksler
not sure I understand, are you trying to move the object to a mouse position, or find the mouse position of an object? Are you going from mouse coords to three.js coords, or the other way around?Johnette
Actually, that doesn't look right... where is object.matrixWorld.getPosition().clone() coming from? There is no object to start with, I want to create a new one and position it where the mouse event occurred.Weksler
Just saw your last message, yes move an object to the mouse position :)Weksler
Thanks for that. It's almost there but I already found posts to find the intersection of existing objects. What I need is if the world is empty apart from the camera, how would I create a new object where the mouse was clicked, and then continue to move that object to the mouse position as it is moved.Weksler
I believe the way to do that is to not have an empty world, but rather have many transparent objects far away to capture the projected rays.Johnette
Hmm I see. I considered adding a plane to the world to do intersection checking but that would mean the plane would need to be massive, or at least large enough to fill the viewport (to capture any potential click intersect) and then move it with the camera. It seems like an inelegant solution to require objects in the scene though... there MUST be a way to do this without them. I wonder if I could follow the ray's vector until the z=0 point? Not sure how to do that though...Weksler
S
1

ThreeJS is slowly mowing away from Projector.(Un)ProjectVector and the solution with projector.pickingRay() doesn't work anymore, just finished updating my own code.. so the most recent working version should be as follow:

var rayVector = new THREE.Vector3(0, 0, 0.5);
var camera = new THREE.PerspectiveCamera(fov,this.offsetWidth/this.offsetHeight,0.1,farFrustum);
var raycaster = new THREE.Raycaster();
var scene = new THREE.Scene();

//...

function intersectObjects(x, y, planeOnly) {
  rayVector.set(((x/this.offsetWidth)*2-1), (1-(y/this.offsetHeight)*2), 1).unproject(camera);
  raycaster.set(camera.position, rayVector.sub(camera.position ).normalize());
  var intersects = raycaster.intersectObjects(scene.children);
  return intersects;
}
Sennet answered 29/10, 2014 at 15:23 Comment(0)
S
1

For those using @react-three/fiber (aka r3f and react-three-fiber), I found this discussion and it's associated code samples by Matt Rossman helpful. In particular, many examples using the methods above are for simple orthographic views, not for when OrbitControls are in play.

Discussion: https://github.com/pmndrs/react-three-fiber/discussions/857

Simple example using Matt's technique: https://codesandbox.io/s/r3f-mouse-to-world-elh73?file=/src/index.js

More generalizable example: https://codesandbox.io/s/react-three-draggable-cxu37?file=/src/App.js

Sateia answered 24/12, 2021 at 17:54 Comment(2)
How could I slow the following a bit down? so that the object is not immediately on the same position as the mouse, rather than following the mouse?Peay
btw. this should be the correct answer, since it is almost a one liner.Peay
B
1

Here's a current answer (THREE.REVISION==157) that works with rotated cameras and smaller-than-window renderers, and finds a point on the ground plane. (Substitute any other plane you like.)

const raycaster = new THREE.Raycaster()
const pt = new THREE.Vector3()
renderer.domElement.addEventListener('mousemove', evt => {
  const rect = evt.target.getBoundingClientRect()

  // Update the vector to use normalized screen coordinates [-1,1]
  pt.set(
    ((evt.clientX - rect.left) / rect.width) * 2 - 1,
    ((rect.top - evt.clientY) / rect.height) * 2 + 1,
    1
  )

  raycaster.setFromCamera(pt, camera)

  // Intersecting with the ground plane, where +Y is up
  const ground = new THREE.Plane(new THREE.Vector3(0, 1, 0), 0);
  const pointHitsPlane = raycaster.ray.intersectPlane(ground, pt);
  if (pointHitsPlane) {
    // pt is now a position in the 3D world
    // move a global test object to that location to verify
    testObject.position.copy(pt)
  }
})
Britzka answered 17/10, 2023 at 22:19 Comment(0)
W
0

Here is my take at creating an es6 class out of it. Working with Three.js r83. The method of using rayCaster comes from mrdoob here: Three.js Projector and Ray objects

    export default class RaycasterHelper
    {
      constructor (camera, scene) {
        this.camera = camera
        this.scene = scene
        this.rayCaster = new THREE.Raycaster()
        this.tapPos3D = new THREE.Vector3()
        this.getIntersectsFromTap = this.getIntersectsFromTap.bind(this)
      }
      // objects arg below needs to be an array of Three objects in the scene 
      getIntersectsFromTap (tapX, tapY, objects) {
        this.tapPos3D.set((tapX / window.innerWidth) * 2 - 1, -(tapY / 
        window.innerHeight) * 2 + 1, 0.5) // z = 0.5 important!
        this.tapPos3D.unproject(this.camera)
        this.rayCaster.set(this.camera.position, 
        this.tapPos3D.sub(this.camera.position).normalize())
        return this.rayCaster.intersectObjects(objects, false)
      }
    }

You would use it like this if you wanted to check against all your objects in the scene for hits. I made the recursive flag false above because for my uses I did not need it to be.

var helper = new RaycasterHelper(camera, scene)
var intersects = helper.getIntersectsFromTap(tapX, tapY, 
this.scene.children)
...
Waistband answered 12/4, 2017 at 16:58 Comment(0)
E
0

Although the provided answers can be useful in some scenarios, I hardly can imagine those scenarios (maybe games or animations) because they are not precise at all (guessing around target's NDC z?). You can't use those methods to unproject screen coordinates to the world ones if you know target z-plane. But for the most scenarios, you should know this plane.

For example, if you draw sphere by center (known point in model space) and radius - you need to get radius as delta of unprojected mouse coordinates - but you can't! With all due respect @WestLangley's method with targetZ doesn't work, it gives incorrect results (I can provide jsfiddle if needed). Another example - you need to set orbit controls target by mouse double click, but without "real" raycasting with scene objects (when you have nothing to pick).

The solution for me is to just create the virtual plane in target point along z-axis and use raycasting with this plane afterward. Target point can be current orbit controls target or vertex of object you need to draw step by step in existing model space etc. This works perfectly and it is simple (example in typescript):

screenToWorld(v2D: THREE.Vector2, camera: THREE.PerspectiveCamera = null, target: THREE.Vector3 = null): THREE.Vector3 {
    const self = this;

    const vNdc = self.toNdc(v2D);
    return self.ndcToWorld(vNdc, camera, target);
}

//get normalized device cartesian coordinates (NDC) with center (0, 0) and ranging from (-1, -1) to (1, 1)
toNdc(v: THREE.Vector2): THREE.Vector2 {
    const self = this;

    const canvasEl = self.renderers.WebGL.domElement;

    const bounds = canvasEl.getBoundingClientRect();        

    let x = v.x - bounds.left;      

    let y = v.y - bounds.top;       

    x = (x / bounds.width) * 2 - 1;     

    y = - (y / bounds.height) * 2 + 1;      

    return new THREE.Vector2(x, y);     
}

ndcToWorld(vNdc: THREE.Vector2, camera: THREE.PerspectiveCamera = null, target: THREE.Vector3 = null): THREE.Vector3 {
    const self = this;      

    if (!camera) {
        camera = self.camera;
    }

    if (!target) {
        target = self.getTarget();
    }

    const position = camera.position.clone();

    const origin = self.scene.position.clone();

    const v3D = target.clone();

    self.raycaster.setFromCamera(vNdc, camera);

    const normal = new THREE.Vector3(0, 0, 1);

    const distance = normal.dot(origin.sub(v3D));       

    const plane = new THREE.Plane(normal, distance);

    self.raycaster.ray.intersectPlane(plane, v3D);

    return v3D; 
}
Embower answered 15/4, 2018 at 21:44 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.