0

我正在尝试通过旋转点并检查它们是否在相机前面(具有球形物体的半径)来执行视锥体剔除。远近的顶部和底部平面工作正常,但是当摄像机角度 camRot.y 不接近 0 时,左右并没有像应有的那样剔除。

当我旋转平面以匹配视锥体时,问题就出现了,旋转的平面最终围绕错误的轴旋转。(当我向下看时,左右平面的法线向量与近平面的方向相同)。

    (objectOffsetFromCamera.z*-cos(camRot.x)*cos(camRot.y) +
    objectOffsetFromCamera.x * sin(camRot.x)*cos(camRot.y) +
    objectOffsetFromCamera.y *-sin(camRot.y) - (DRAW_DISTANCE + radius) < 0 && //FARPLANE DISTANCE

    objectOffsetFromCamera.z*-cos(camRot.x)*cos(camRot.y) +
    objectOffsetFromCamera.x* sin(camRot.x)*cos(camRot.y) +
    objectOffsetFromCamera.y*-sin(camRot.y) + radius >= 0 && //NEAR PLANE

    objectOffsetFromCamera.z*-cos(camRot.x + (FOV_X / 2)) *cos(camRot.y) + 
    objectOffsetFromCamera.x* sin(camRot.x + (FOV_X / 2)) *cos(camRot.y) +
    objectOffsetFromCamera.y*-sin(camRot.y) + radius >= 0 && //RIGHT PLANE, bugged for none zero camRot.y values.

    objectOffsetFromCamera.z*-cos(camRot.x - (FOV_X / 2)) * cos(camRot.y) +
    objectOffsetFromCamera.x* sin(camRot.x - (FOV_X / 2)) * cos(camRot.y) +
    objectOffsetFromCamera.y*-sin(camRot.y) + radius >= 0 && //LEFT PLANE, bugged for none zero camRot.y values.

    objectOffsetFromCamera.z*-cos(camRot.x)*cos(camRot.y + (FOV_Y / 2)) +
    objectOffsetFromCamera.x* sin(camRot.x)*cos(camRot.y + (FOV_Y / 2)) +
    objectOffsetFromCamera.y*-sin(camRot.y + (FOV_Y / 2)) + radius >= 0 && //TOP PLANE

    objectOffsetFromCamera.z*-cos(camRot.x)*cos(camRot.y - (FOV_Y / 2)) +
    objectOffsetFromCamera.x* sin(camRot.x)*cos(camRot.y - (FOV_Y  / 2)) +
    objectOffsetFromCamera.y*-sin(camRot.y - (FOV_Y  / 2)) + radius >= 0) //BOTTOM PLANE

我不确定这个问题是否有一个简单的解决方案,但由于它接近工作,我希望我不必像我发现的一些教程那样重做。

需要做的是确保在向上或向下看时,额外的(FOV_X / 2)旋转与相机正确旋转,但我无法理解如何。

此外,所有 cos 和 sin 计算都是每帧完成的,只是将它们移到代码中以显示完成了什么。

如果我的帖子/代码难以理解,我很抱歉。

我现在可以正常工作了,至少看起来可以正常工作。我当前的代码(几乎)。FOV_X 和 FOV_Y 是常数,取决于 fov 和纵横比。

    void Camera::calcFrustumPlanes()
    {
frustumPlanes.pfar.x = sin(rot.x)*cos(rot.y);
frustumPlanes.pfar.y = -sin(rot.y);
frustumPlanes.pfar.z = -cos(rot.x)*cos(rot.y);

frustumPlanes.pnear.x = sin(rot.x)*cos(rot.y);
frustumPlanes.pnear.y = -sin(rot.y);
frustumPlanes.pnear.z = -cos(rot.x)*cos(rot.y);

frustumPlanes.pleft.x = sin(rot.x + cos(rot.y)*FOV_X*cos(rot.y) + cos(rot.x)*sin(sin(rot.y)*FOV_X)*sin(rot.y);
frustumPlanes.pleft.y = -sin(rot.y)*cos(sin(rot.y)*FOV_X);
frustumPlanes.pleft.z = -cos(rot.x + cos(rot.y)*FOV_X)*cos(rot.y) + sin(rot.x)*sin(sin(rot.y)*FOV_X)*sin(rot.y);

frustumPlanes.pright.x = sin(rot.x - cos(rot.y)*FOV_X)*cos(rot.y) + cos(rot.x)*sin(-sin(rot.y)*FOV_X)*sin(rot.y);
frustumPlanes.pright.y = -sin(rot.y)*cos(-sin(rot.y)*FOV_X);
frustumPlanes.pright.z = -cos(rot.x - cos(rot.y)*FOV_X)*cos(rot.y) +  sin(rot.x)*sin(-sin(rot.y)*FOV_X)*sin(rot.y);

frustumPlanes.ptop.x = sin(rot.x)*cos(rot.y + FOV_Y);
frustumPlanes.ptop.y = -sin(rot.y + FOV_Y);
frustumPlanes.ptop.z = -cos(rot.x)*cos(rot.y + FOV_Y);

frustumPlanes.pbottom.x = sin(rot.x)*cos(rot.y - FOV_Y);
frustumPlanes.pbottom.y = -sin(rot.y - FOV_Y);
frustumPlanes.pbottom.z = -cos(rot.x)*cos(rot.y - FOV_Y);
    }

    bool Camera::isWithinFrustum(glm::ivec3 objectPos, float radius)
glm::vec3 camOffset = objectPos - pos;
return (glm::dot(camOffset, frustumPlanes.pfar) < (DRAW_DISTANCE + radius ) &&
        glm::dot(camOffset, frustumPlanes.pnear)   >= -radius &&
        glm::dot(camOffset, frustumPlanes.pleft)   >= -radius &&
        glm::dot(camOffset, frustumPlanes.pright)  >= -radius  &&
        glm::dot(camOffset, frustumPlanes.ptop)    >= -radius  &&
        glm::dot(camOffset, frustumPlanes.pbottom) >= -radius);
4

0 回答 0