mirror of
https://github.com/Relintai/sfw.git
synced 2025-01-03 05:09:36 +01:00
Grab some of the camera helper methods from Pandemonium.
This commit is contained in:
parent
28a66b0e09
commit
b04effdfe3
@ -47,6 +47,8 @@ void Camera2D::make_current() {
|
|||||||
current_camera = this;
|
current_camera = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Camera2D::Camera2D() {
|
Camera2D::Camera2D() {
|
||||||
}
|
}
|
||||||
Camera2D::~Camera2D() {
|
Camera2D::~Camera2D() {
|
||||||
|
@ -27,6 +27,8 @@ public:
|
|||||||
|
|
||||||
void make_current();
|
void make_current();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//void push_transform(const Transform2D &transform);
|
//void push_transform(const Transform2D &transform);
|
||||||
//void pop_transform();
|
//void pop_transform();
|
||||||
|
|
||||||
|
@ -53,6 +53,110 @@ void Camera3D::make_current() {
|
|||||||
current_camera = this;
|
current_camera = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Vector3 Camera3D::project_ray_normal(const Point2 &p_pos) const {
|
||||||
|
Vector3 ray = project_local_ray_normal(p_pos);
|
||||||
|
return transform.basis.xform(ray).normalized();
|
||||||
|
};
|
||||||
|
|
||||||
|
Vector3 Camera3D::project_local_ray_normal(const Point2 &p_pos) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
Vector2 cpos = p_pos;
|
||||||
|
Vector3 ray;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
cm.set_perspective(70, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
Vector2 screen_he = cm.get_viewport_half_extents();
|
||||||
|
ray = Vector3(((cpos.x / viewport_size.width) * 2.0 - 1.0) * screen_he.x, ((1.0 - (cpos.y / viewport_size.height)) * 2.0 - 1.0) * screen_he.y, -znear).normalized();
|
||||||
|
|
||||||
|
return ray;
|
||||||
|
};
|
||||||
|
|
||||||
|
Vector3 Camera3D::project_ray_origin(const Point2 &p_pos) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
Vector2 cpos = p_pos;
|
||||||
|
ERR_FAIL_COND_V(viewport_size.y == 0, Vector3());
|
||||||
|
|
||||||
|
Vector2 pos = cpos / viewport_size;
|
||||||
|
float vsize, hsize;
|
||||||
|
|
||||||
|
vsize = size / viewport_size.aspect();
|
||||||
|
hsize = size;
|
||||||
|
|
||||||
|
Vector3 ray;
|
||||||
|
ray.x = pos.x * (hsize)-hsize / 2;
|
||||||
|
ray.y = (1.0 - pos.y) * (vsize)-vsize / 2;
|
||||||
|
ray.z = -znear;
|
||||||
|
ray = transform.xform(ray);
|
||||||
|
return ray;
|
||||||
|
};
|
||||||
|
|
||||||
|
bool Camera3D::is_position_behind(const Vector3 &p_pos) const {
|
||||||
|
Transform t = transform;
|
||||||
|
Vector3 eyedir = -t.basis.get_axis(2).normalized();
|
||||||
|
return eyedir.dot(p_pos - t.origin) < znear;
|
||||||
|
}
|
||||||
|
|
||||||
|
Vector<Vector3> Camera3D::get_near_plane_points() const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_perspective(70, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
|
||||||
|
Vector3 endpoints[8];
|
||||||
|
cm.get_endpoints(Transform(), endpoints);
|
||||||
|
|
||||||
|
Vector<Vector3> points;
|
||||||
|
points.push_back(Vector3());
|
||||||
|
for (int i = 0; i < 4; i++) {
|
||||||
|
points.push_back(endpoints[i + 4]);
|
||||||
|
}
|
||||||
|
return points;
|
||||||
|
}
|
||||||
|
|
||||||
|
Point2 Camera3D::unproject_position(const Vector3 &p_pos) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_perspective(70, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
|
||||||
|
Plane p(transform.xform_inv(p_pos), 1.0);
|
||||||
|
|
||||||
|
p = cm.xform(p);
|
||||||
|
p.normal /= p.d;
|
||||||
|
|
||||||
|
Point2 res;
|
||||||
|
res.x = (p.normal.x * 0.5 + 0.5) * viewport_size.x;
|
||||||
|
res.y = (-p.normal.y * 0.5 + 0.5) * viewport_size.y;
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
Vector3 Camera3D::project_position(const Point2 &p_point, float p_z_depth) const {
|
||||||
|
if (p_z_depth == 0) {
|
||||||
|
return transform.origin;
|
||||||
|
}
|
||||||
|
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_perspective(70, viewport_size.aspect(), p_z_depth, zfar, true);
|
||||||
|
|
||||||
|
Vector2 vp_he = cm.get_viewport_half_extents();
|
||||||
|
|
||||||
|
Vector2 point;
|
||||||
|
point.x = (p_point.x / viewport_size.x) * 2.0 - 1.0;
|
||||||
|
point.y = (1.0 - (p_point.y / viewport_size.y)) * 2.0 - 1.0;
|
||||||
|
point *= vp_he;
|
||||||
|
|
||||||
|
Vector3 p(point.x, point.y, -p_z_depth);
|
||||||
|
|
||||||
|
return transform.xform(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Camera3D::Camera3D() {
|
Camera3D::Camera3D() {
|
||||||
screen_aspect_ratio = 1; //p_viewport_size.width / (float)p_viewport_size.height,
|
screen_aspect_ratio = 1; //p_viewport_size.width / (float)p_viewport_size.height,
|
||||||
|
|
||||||
@ -78,6 +182,68 @@ void OrthographicCamera::bind() {
|
|||||||
Camera3D::bind();
|
Camera3D::bind();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Vector3 OrthographicCamera::project_local_ray_normal(const Point2 &p_pos) const {
|
||||||
|
return Vector3(0, 0, -1);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Point2 OrthographicCamera::unproject_position(const Vector3 &p_pos) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_orthogonal(size, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
|
||||||
|
Plane p(transform.xform_inv(p_pos), 1.0);
|
||||||
|
|
||||||
|
p = cm.xform(p);
|
||||||
|
p.normal /= p.d;
|
||||||
|
|
||||||
|
Point2 res;
|
||||||
|
res.x = (p.normal.x * 0.5 + 0.5) * viewport_size.x;
|
||||||
|
res.y = (-p.normal.y * 0.5 + 0.5) * viewport_size.y;
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
Vector3 OrthographicCamera::project_position(const Point2 &p_point, float p_z_depth) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_orthogonal(size, viewport_size.aspect(), p_z_depth, zfar, true);
|
||||||
|
|
||||||
|
Vector2 vp_he = cm.get_viewport_half_extents();
|
||||||
|
|
||||||
|
Vector2 point;
|
||||||
|
point.x = (p_point.x / viewport_size.x) * 2.0 - 1.0;
|
||||||
|
point.y = (1.0 - (p_point.y / viewport_size.y)) * 2.0 - 1.0;
|
||||||
|
point *= vp_he;
|
||||||
|
|
||||||
|
Vector3 p(point.x, point.y, -p_z_depth);
|
||||||
|
|
||||||
|
return transform.xform(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Vector<Vector3> OrthographicCamera::get_near_plane_points() const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_orthogonal(size, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
|
||||||
|
Vector3 endpoints[8];
|
||||||
|
cm.get_endpoints(Transform(), endpoints);
|
||||||
|
|
||||||
|
Vector<Vector3> points;
|
||||||
|
points.push_back(Vector3());
|
||||||
|
for (int i = 0; i < 4; i++) {
|
||||||
|
points.push_back(endpoints[i + 4]);
|
||||||
|
}
|
||||||
|
return points;
|
||||||
|
}
|
||||||
|
|
||||||
OrthographicCamera::OrthographicCamera() :
|
OrthographicCamera::OrthographicCamera() :
|
||||||
Camera3D() {
|
Camera3D() {
|
||||||
}
|
}
|
||||||
@ -95,6 +261,85 @@ void PerspectiveCamera::bind() {
|
|||||||
Camera3D::bind();
|
Camera3D::bind();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Vector3 PerspectiveCamera::project_local_ray_normal(const Point2 &p_pos) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
Vector2 cpos = p_pos;
|
||||||
|
Vector3 ray;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
cm.set_perspective(fov, viewport_size.aspect(), znear, zfar, false);
|
||||||
|
Vector2 screen_he = cm.get_viewport_half_extents();
|
||||||
|
ray = Vector3(((cpos.x / viewport_size.width) * 2.0 - 1.0) * screen_he.x, ((1.0 - (cpos.y / viewport_size.height)) * 2.0 - 1.0) * screen_he.y, -znear).normalized();
|
||||||
|
|
||||||
|
return ray;
|
||||||
|
};
|
||||||
|
|
||||||
|
Vector3 PerspectiveCamera::project_ray_origin(const Point2 &p_pos) const {
|
||||||
|
return transform.origin;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Point2 PerspectiveCamera::unproject_position(const Vector3 &p_pos) const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_perspective(fov, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
Plane p(transform.xform_inv(p_pos), 1.0);
|
||||||
|
|
||||||
|
p = cm.xform(p);
|
||||||
|
p.normal /= p.d;
|
||||||
|
|
||||||
|
Point2 res;
|
||||||
|
res.x = (p.normal.x * 0.5 + 0.5) * viewport_size.x;
|
||||||
|
res.y = (-p.normal.y * 0.5 + 0.5) * viewport_size.y;
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
Vector3 PerspectiveCamera::project_position(const Point2 &p_point, float p_z_depth) const {
|
||||||
|
if (p_z_depth == 0) {
|
||||||
|
return transform.origin;
|
||||||
|
}
|
||||||
|
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_perspective(fov, viewport_size.aspect(), p_z_depth, zfar, true);
|
||||||
|
|
||||||
|
Vector2 vp_he = cm.get_viewport_half_extents();
|
||||||
|
|
||||||
|
Vector2 point;
|
||||||
|
point.x = (p_point.x / viewport_size.x) * 2.0 - 1.0;
|
||||||
|
point.y = (1.0 - (p_point.y / viewport_size.y)) * 2.0 - 1.0;
|
||||||
|
point *= vp_he;
|
||||||
|
|
||||||
|
Vector3 p(point.x, point.y, -p_z_depth);
|
||||||
|
|
||||||
|
return transform.xform(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Vector<Vector3> PerspectiveCamera::get_near_plane_points() const {
|
||||||
|
Size2 viewport_size = RenderState::render_rect.size;
|
||||||
|
|
||||||
|
Projection cm;
|
||||||
|
|
||||||
|
cm.set_perspective(fov, viewport_size.aspect(), znear, zfar, true);
|
||||||
|
|
||||||
|
Vector3 endpoints[8];
|
||||||
|
cm.get_endpoints(Transform(), endpoints);
|
||||||
|
|
||||||
|
Vector<Vector3> points;
|
||||||
|
points.push_back(Vector3());
|
||||||
|
for (int i = 0; i < 4; i++) {
|
||||||
|
points.push_back(endpoints[i + 4]);
|
||||||
|
}
|
||||||
|
return points;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
PerspectiveCamera::PerspectiveCamera() :
|
PerspectiveCamera::PerspectiveCamera() :
|
||||||
Camera3D() {
|
Camera3D() {
|
||||||
fov = 70;
|
fov = 70;
|
||||||
|
@ -30,6 +30,16 @@ public:
|
|||||||
|
|
||||||
void make_current();
|
void make_current();
|
||||||
|
|
||||||
|
// From Pandemonium
|
||||||
|
virtual Vector3 project_ray_normal(const Point2 &p_pos) const;
|
||||||
|
virtual Vector3 project_ray_origin(const Point2 &p_pos) const;
|
||||||
|
virtual Vector3 project_local_ray_normal(const Point2 &p_pos) const;
|
||||||
|
virtual Point2 unproject_position(const Vector3 &p_pos) const;
|
||||||
|
bool is_position_behind(const Vector3 &p_pos) const;
|
||||||
|
virtual Vector3 project_position(const Point2 &p_point, float p_z_depth) const;
|
||||||
|
|
||||||
|
virtual Vector<Vector3> get_near_plane_points() const;
|
||||||
|
|
||||||
Camera3D();
|
Camera3D();
|
||||||
virtual ~Camera3D();
|
virtual ~Camera3D();
|
||||||
|
|
||||||
@ -51,6 +61,12 @@ class OrthographicCamera : public Camera3D {
|
|||||||
public:
|
public:
|
||||||
void bind();
|
void bind();
|
||||||
|
|
||||||
|
virtual Vector3 project_local_ray_normal(const Point2 &p_pos) const;
|
||||||
|
virtual Point2 unproject_position(const Vector3 &p_pos) const;
|
||||||
|
virtual Vector3 project_position(const Point2 &p_point, float p_z_depth) const;
|
||||||
|
|
||||||
|
virtual Vector<Vector3> get_near_plane_points() const;
|
||||||
|
|
||||||
OrthographicCamera();
|
OrthographicCamera();
|
||||||
~OrthographicCamera();
|
~OrthographicCamera();
|
||||||
};
|
};
|
||||||
@ -61,6 +77,13 @@ public:
|
|||||||
|
|
||||||
void bind();
|
void bind();
|
||||||
|
|
||||||
|
virtual Vector3 project_local_ray_normal(const Point2 &p_pos) const;
|
||||||
|
virtual Vector3 project_ray_origin(const Point2 &p_pos) const;
|
||||||
|
virtual Point2 unproject_position(const Vector3 &p_pos) const;
|
||||||
|
virtual Vector3 project_position(const Point2 &p_point, float p_z_depth) const;
|
||||||
|
|
||||||
|
virtual Vector<Vector3> get_near_plane_points() const;
|
||||||
|
|
||||||
PerspectiveCamera();
|
PerspectiveCamera();
|
||||||
~PerspectiveCamera();
|
~PerspectiveCamera();
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user