zoukankan      html  css  js  c++  java
  • ThreeJS中的点击与交互——Raycaster的用法

    基础概念

    坐标系

    我们的手机屏幕是二维的,但是我们展示物体的世界是三维的,当我们在构建一个物体的时候我们是以一个三维世界既是世界坐标来构建,而转化为屏幕坐标展示在我们眼前,则需要经历多道矩阵变化,中间webGL替我们操作了许多事情。

     世界坐标系:在webGL中,世界坐标系是以屏幕中心为原点(0, 0, 0),且是始终不变的。你面对屏幕,你的右边是x正轴,上面是y正轴,屏幕指向你的为z正轴。长度单位这样来定:窗口范围按此单位恰好是(-1,-1,-1)到(1,1,1)。

    • 屏幕坐标系:

    webGL的重要功能之一就是将三维的世界坐标经过变换、投影等计算,最终算出它在显示设备上对应的位置,这个位置就称为设备坐标。在屏幕、打印机等设备上的坐标是二维坐标。

    • 视点坐标系:

    是以视点(照相机)为原点,以视线的方向为Z+轴正方向的坐标系中的方向。webGL会将世界坐标先变换到视点坐标,然后进行裁剪,只有在视线范围(视见体)之内的场景才会进入下一阶段的计算。

    Raycaster

    Raycaster threeJs官方文档

    这个类设计用于鼠标去获取在3D世界被鼠标选中的一些物体

    Raycaster( origin, direction, near, far ) 
    
    origin — 射线的起点向量。
    direction — 射线的方向向量,应该归一标准化。
    near — 所有返回的结果应该比 near 远。Near不能为负,默认值为0。
    far — 所有返回的结果应该比 far 近。Far 不能小于 near,默认值为无穷大。

    找到点击物体的大致思路

     

     
    鼠标在屏幕上点击的时候,得到二维坐标p(x, y),再加上深度坐标的范围(0, 1), 就可以形成两个三位坐标A(x1, y1, 0), B(x2, y, 1), 由于它们的Z轴坐标是0和1,则转变到投影坐标系的话,一定分别是前剪切平面上的点和后剪切平面上的点,也就是说,在投影坐标系中,A点一定在能看见的所有模型的最前面,B点一定在能看见的所有的模型的最后边,将AB点连成线,AB线穿过的物体就是被点击的物体。而 Three.js提供一个射线类Raycasting来拾取场景里面的物体。更方便的使用鼠标来操作3D场景。(不过在实际代码中我们组成射线的两个点是摄像机所在视点与屏幕上点击的点连接而成的射线)

    来一个Raycasting的官方实例

    代码实现

    function onDocumentMouseDown(e) {
        e.preventDefault();
        
        //将鼠标点击位置的屏幕坐标转成threejs中的标准坐标,具体解释见代码释义
        mouse.x = (e.clientX / window.innerWidth) * 2 - 1;
        mouse.y = -(e.clientY / window.innerHeight) * 2 + 1;
        //新建一个三维单位向量 假设z方向就是0.5
        //根据照相机,把这个向量转换到视点坐标系
          var vector = new THREE.Vector3(mouse.x, mouse.y,0.5).unproject(camera);
    
        //在视点坐标系中形成射线,射线的起点向量是照相机, 射线的方向向量是照相机到点击的点,这个向量应该归一标准化。
        var raycaster = new THREE.Raycaster(camera.position, vector.sub(camera.position).normalize());
    
        //射线和模型求交,选中一系列直线
        var intersects = raycaster.intersectObjects(objects);
        console.log('imtersrcts=' + intersects)
    
        if (intersects.length > 0) {
            //选中第一个射线相交的物体
            SELECTED = intersects[0].object;
            var intersected = intersects[0].object;
            console.log(intersects[0].object)
        }
    
    
    }

    代码释义

     

     //得到
     mouse.x = (e.clientX / window.innerWidth) * 2 - 1;
     mouse.y = -(e.clientY / window.innerHeight) * 2 + 1;
      
     推导过程:
     设A点为点击点(x1,y1),x1=e.clintX, y1=e.clientY
     设A点在世界坐标中的坐标值为B(x2,y2);
     
     由于A点的坐标值的原点是以屏幕左上角为(0,0);
     我们可以计算可得以屏幕中心为原点的B'值
     x2' = x1 - innerWidth/2
     y2' = innerHeight/2 - y1
     又由于在世界坐标的范围是[-1,1],要得到正确的B值我们必须要将坐标标准化
     x2 = (x1 -innerWidth/2)/(innerwidth/2) = (x1/innerWidth)*2-1
     同理得 y2 = -(y1/innerHeight)*2 +1

    参考资料

    Three.js中的拾取
    OpenGL中各种坐标系的理解
    threejs对象拾取
    前端填坑指南

  • 相关阅读:
    [技巧] 使用Word2010直接编辑、发布博客→博客园cnblogs
    POJ 1201 Intervals【差分约束】
    HDU 2896 病毒侵袭【AC自动机】
    opengl中的gluOrtho2D【转】
    【转】x86和x64的含义和区别
    POJ 1704 Georgia and Bob【Nim博弈】
    POJ 1947 Rebuilding Roads【树状DP】
    POJ 3207/ POJ 3678 【2SAT】
    POJ 1067 取石子游戏【威佐夫博奕】
    apache+webdav的安装配置
  • 原文地址:https://www.cnblogs.com/smedas/p/12445201.html
Copyright © 2011-2022 走看看