javascript - 3d 位置到屏幕坐标无法正常工作三个 js [r100]

标签 javascript canvas three.js 3d 2d

我有一个非常简单的测试场景,可以将一堆 html 标签放置在场景中 3D 对象的位置上。但是当我尝试将 3D 坐标转换为屏幕位置时,我会得到奇怪的值,这些值根本不与我的 3D 对象的位置相对应。请参阅下面的代码,我用它生成场景并放置标签:

var CanvasContainer = document.getElementById("KettingContainer");

function CameraSetup(container){
    var camera = new THREE.PerspectiveCamera( 50, container.offsetWidth / container.offsetHeight, 1, 300 );
    camera.position.set( 0, 0, 100 );
    camera.lookAt( 0, 0, 0 );
    return camera;
}

function SceneSetup(){
    var scene = new THREE.Scene();
    return scene;
}

function RenderSetup(container) {
    var renderer = new THREE.WebGLRenderer({ alpha: true });
    renderer.autoClear = false;
    renderer.setClearColor( 0x000000, 0 );
    renderer.setPixelRatio( window.devicePixelRatio );
    renderer.setSize( container.offsetWidth, container.offsetHeight );
    container.appendChild( renderer.domElement );
    return renderer;
}

function OnWindowResize(container){
    camera.aspect = container.offsetWidth / container.offsetHeight;
    camera.updateProjectionMatrix();
    renderer.setSize( container.offsetWidth, container.offsetHeight );
}

var kraalLabels = document.getElementById('kraalLinkLabels').getElementsByTagName('div');

function Animate(){
    requestAnimationFrame(Animate);

    Render();
}

function Render(){
    renderer.render( scene, camera );
}

function createScreenVector(pos, camera, width, height) {
    const hw = width/2, hh = height/2;
    var pos2d = pos.clone();
    pos2d.project(camera);
    console.log(pos2d);
    pos2d.x = ( pos2d.x * hw ) + hw;
    pos2d.y = - ( pos2d.y * hh ) + hh;

    return pos2d;
}

function positionLabels(positions, camera, width, height){
    for( var i = 0; i < kraalLabels.length; i++ )
    {
        const d = createScreenVector(positions, camera, width, height);
        kraalLabels[i].style.left = d.x + "px";
        kraalLabels[i].style.top = d.y + "px";
    }
}

var camera = CameraSetup(CanvasContainer);
var scene = SceneSetup();
var renderer = RenderSetup(CanvasContainer);

console.log("fc: ", camera.position);

const geometry = new THREE.SphereGeometry( 10, 32, 32 );
const material = new THREE.MeshBasicMaterial( {color: 0xff0000} );
let sphere = new THREE.Mesh( geometry, material );
sphere.position.set(20, 20, 0.1);
scene.add( sphere );

positionLabels(new THREE.Vector3(20, 20, 0.1), camera, CanvasContainer.offsetWidth, CanvasContainer.offsetHeight);

window.addEventListener("resize", function(){ OnWindowResize(CanvasContainer); }, false);
Animate();

请参阅下面的屏幕截图,其中显示了相机位置、 Canvas 容器大小以及每个标签的屏幕位置:

fc is the position of the camera

正如您所看到的,标签的位置比 Canvas 本身大得多。而且位置都是负数...

即使我的对象只是在屏幕中心的顶部和左侧一点..

我现在完全陷入困境,我不知道我做错了什么:(

如果需要额外信息,我很乐意澄清!

最佳答案

经过几个小时的寻找解决方案后,我终于在执行 vector3.project(camera); 之前发现了问题,您必须像这样更新相机矩阵: camera.project(camera); 。 updateMatrixWorld(); 这将解决奇怪值的问题。

希望这对将来的人有帮助!

关于javascript - 3d 位置到屏幕坐标无法正常工作三个 js [r100],我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55002202/

相关文章:

javascript - 如何使 Canvas 背景透明

javascript - 在fabricjs Canvas 上实现ClipTo功能

javascript - 无法读取 null 的属性 'clientHeight' - 工作 JSfiddle 教程在其他地方不起作用?

javascript - 如何使对象在 three.js 场景中仅对一个摄像机可见

javascript - 三.JS:球形线框 Material /EdgesHelper控件

javascript - 根据条件将列表分成两个新列表的最佳方法

javascript - 获取特定div中的高亮文本

python - 如何使 tkinter Canvas 背景透明?

javascript - Express.js session 在不同网络上的不同机器上的用户之间持续存在?

数组中用于模式匹配的 JavaScript 占位符