我最近在这里实现了弧球指南:https ://en.wikibooks.org/wiki/OpenGL_Programming/Modern_OpenGL_Tutorial_Arcball在我的一个项目中,我在缩放和转换对象时遇到了一些问题。
问题:当涉及缩放和平移时,旋转会产生不正确的输出,例如向内弯曲或以特定角度放大对象的某些部分。但是,当我将缩放和平移直接应用于对象的顶点时,旋转会起作用。也就是说,在工作状态下,我对这个对象的模型矩阵是一个单位矩阵。其他任何事情,这都失败了。所以我知道事实上这是这些应用的顺序。我只是不确定什么顺序。以 TRS 顺序应用平移和缩放以及旋转也没有帮助并产生错误的结果。
编辑:修复了平移和缩放问题,但对象不会相对于更改的中心旋转。它相对于旧中心旋转。
相关代码:
对象渲染代码的一部分:
if (R.mouseMoved) {
glm::vec3 va = get_arcball_vector(R.click_mx, R.click_my);
glm::vec3 vb = get_arcball_vector(R.cur_mx, R.cur_my);
float ang = glm::degrees(acos(min(1.0f, glm::dot(va, vb)))) * MMOVE_FACTOR * R.deltaTime;
glm::vec3 axis_in_camera_coord = glm::cross(va, vb);
glm::mat3 camera2object = glm::inverse(glm::mat3(R.view) * glm::mat3(model));
glm::vec3 rotaxis = camera2object * axis_in_camera_coord;
model = glm::rotate(model, ang, rotaxis);
}
// assuming this is what I needed to do here for view matrix, TRT, this doesn't work
glm::mat4 mv = /*glm::translate(glm::mat4(1.0f), glm::vec3(0, 0, -WINDOW_X * .5)) * */R.view * model/* * glm::translate(glm::mat4(1.0f), -center)*/;
glm::mat4 pvm = Renderer::proj * mv;
pvm = glm::translate(pvm, translation);
pvm = glm::scale(pvm, scale);
pvm 矩阵在之后立即发送到着色器并渲染对象。
向量计算函数:
glm::vec3 get_arcball_vector(int x, int y) {
glm::vec3 P = glm::vec3(1.0*x / WINDOW_X * 2 - 1.0,
1.0*y / WINDOW_Y * 2 - 1.0,
0);
P.y = -P.y;
float OP_squared = P.x * P.x + P.y * P.y;
if (OP_squared <= 1 * 1)
P.z = sqrt(1 * 1 - OP_squared); // Pythagoras
else
P = glm::normalize(P); // nearest point
return P;
}
MMOVE_FACTOR 为 0.05,R.deltaTime 用于调节渲染时间和鼠标移动,使其感觉更平滑。有问题的对象可以进行任意缩放和平移,以使其出现在场景的不同位置。无论我尝试什么,它都不会产生我想要的结果。例如,我尝试了以下方法:
model = glm::translate(model, -center);
model = glm::rotate(model, glm::degrees(ang), rotaxis);
model = glm::translate(model, center);
但这也不能提供好的结果,并且旋转仍然不正确。
编辑2:根据httpdigest的建议,这是最新的代码:
glm::mat4 tr = glm::translate(glm::mat4(1.0f), -R.camera.getPos());
glm::mat4 trc = glm::translate(glm::mat4(1.0f), -center);
glm::mat4 tmc = glm::translate(glm::mat4(1.0f), center);
glm::mat4 rx = glm::rotate(glm::mat4(1.0f), -(float)R.camera.getPitch(), glm::vec3(1.0, 0, 0));
glm::mat4 ry = glm::rotate(glm::mat4(1.0f), (float)R.camera.getYaw(), glm::vec3(0, 1.0, 0));
glm::mat4 pvm = Renderer::proj * tr * rx * ry * trc * tmc;
pvm = glm::translate(pvm, translation);
pvm = glm::scale(pvm, scale);
中心是一个 glm::vec3 ,其顶点总和除以顶点数。
上述问题:它不适用于围绕 X 轴的平移。当我旋转它足够我认为它位于与平移值相同的距离时,它会在 Y 轴上正确旋转。另一个问题是它的行为不像轨迹球,但我想这是处理旋转的方式所预期的。不过我可能是错的。