-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathOrbitCamera.cpp
133 lines (102 loc) · 2.93 KB
/
OrbitCamera.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
#include "OrbitCamera.h"
#include <GLFW/glfw3.h>
#include <iostream>
OrbitCamera::OrbitCamera(int width, int height, float fovY_rad, float ratio_wh, float near_z, float far_z)
:width(width), height(height), fovY_rad(fovY_rad), ratio_wh(ratio_wh), near_z(near_z), far_z(far_z),
isLeftMousePressed(false), isRightMousePressed(false), isFirstMouseEvent(true)
{
UP = glm::vec3(0.0f, 1.0f, 0.0f);
reset();
}
glm::mat4 OrbitCamera::GetProj() {
return glm::perspective(fovY_rad, ratio_wh, near_z, far_z);
}
glm::mat4 OrbitCamera::GetView() {
glm::vec3 pos = GetCameraPosition();
return glm::lookAt(pos, target, UP);
}
void OrbitCamera::reset() {
phi = DEFAULT_PHI;
theta = DEFAULT_THETA;
distance = DEFAULT_DISTANCE;
SetTarget();
}
void OrbitCamera::SetTarget(const glm::vec3& tar) {
target = tar;
}
glm::vec3 OrbitCamera::GetCameraPosition()
{
float t = glm::radians(theta);
float p = glm::radians(phi);
float x = distance * std::sin(t) * std::cos(p);
float y = distance * std::sin(t) * std::sin(p);
float z = distance * std::cos(t);
// 注意坐标系的变化
return glm::vec3(
target.x + y,
target.y + z,
target.z + x
);
}
void OrbitCamera::OnKeyChanged(int key, int scanCode, int action, int mode)
{
if (key == GLFW_KEY_LEFT_SHIFT)
reset();
}
void OrbitCamera::OnMousePositionChanged(double xpos, double ypos)
{
if (isFirstMouseEvent)
{
lastX = xpos;
lastY = ypos;
isFirstMouseEvent = false;
return;
}
float dx = xpos - lastX;
float dy = ypos - lastY;
if (isLeftMousePressed) {
update_phi_theta(dx / height, dy / height);
}
if (isRightMousePressed) {
update_target(dx / height, dy / height);
}
lastX = xpos;
lastY = ypos;
}
void OrbitCamera::OnMouseButtonChanged(int button, int action, int mode)
{
if (button == GLFW_MOUSE_BUTTON_LEFT) {
isLeftMousePressed = (action == GLFW_PRESS);
}
if (button == GLFW_MOUSE_BUTTON_RIGHT) {
isRightMousePressed = (action == GLFW_PRESS);
}
}
void OrbitCamera::OnScrollChanged(double x, double y)
{
// 一般好像x没啥用,滚轮上滚一个单位y=1.下滚一个单位y=-1
//std::cout << x << ' ' << y << '\n';
distance *= std::pow(0.95f, y);
}
void OrbitCamera::update_phi_theta(float x, float y)
{
// xy是横纵向位移与窗口高度之比,(这里只取高度,避免因为宽高比影响不同方向的速度)
float d_theta = y * 180.0f; // 转为角度
float d_phi = x * 360.0f;
phi -= d_phi * rotate_speed;
theta -= d_theta * rotate_speed;
//phi = Math::clamp(phi, MIN_PHI, MAX_PHI); // 不限制左右转
if (theta >= MAX_THETA) theta = MAX_THETA; // 上下转(俯仰)限制一下,避免更新“向上方向”
if (theta <= MIN_THETA) theta = MIN_THETA;
}
void OrbitCamera::update_target(float x, float y)
{
glm::vec3 cur = GetCameraPosition();
glm::vec3 forward = glm::normalize(target - cur);
glm::vec3 left = glm::cross(UP, forward);
glm::vec3 up = glm::cross(forward, left);
float factor = distance * (float)tan(fovY_rad / 2) * 2;
glm::vec3 dx = x * factor * left;
glm::vec3 dy = y * factor * up;
target += dx + dy;
}