1use crate::Fixed;
2use nalgebra::{Matrix4, Point2, RealField, convert};
3use simba::scalar::SubsetOf;
4
5#[derive(Debug, Clone, Copy, PartialEq, Eq)]
9#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
10pub struct Scope<N: Copy + RealField> {
11 fov: Fixed<N>,
15 zcp: (N, N),
20 oim: bool,
24 opm: bool,
28}
29
30impl<N: Copy + RealField> Default for Scope<N> {
31 fn default() -> Self {
32 Self {
33 fov: Fixed::default(),
34 zcp: (convert(1e-1), convert(1e+3)),
35 oim: false,
36 opm: false,
37 }
38 }
39}
40
41impl<N: Copy + RealField> Scope<N> {
42 #[must_use]
44 pub const fn fov(&self) -> Fixed<N> {
45 self.fov
46 }
47 pub fn set_fov(&mut self, fov: impl Into<Fixed<N>>) {
70 self.fov = fov.into();
71 }
72 #[must_use]
77 pub fn clip_planes(&self, zat: N) -> (N, N) {
78 if self.oim {
79 let (znear, zfar) = self.zcp;
80 (zat - znear, zat + zfar)
81 } else {
82 self.zcp
83 }
84 }
85 pub const fn set_clip_planes(&mut self, znear: N, zfar: N) {
89 self.zcp = (znear, zfar);
90 }
91 #[must_use]
95 pub const fn scale(&self) -> bool {
96 self.oim
97 }
98 pub const fn set_scale(&mut self, oim: bool) {
102 self.oim = oim;
103 }
104 #[must_use]
108 pub const fn ortho(&self) -> bool {
109 self.opm
110 }
111 pub const fn set_ortho(&mut self, opm: bool) {
115 self.opm = opm;
116 }
117 #[must_use]
120 pub fn projection_and_upp(&self, zat: N, max: &Point2<N>) -> (Matrix4<N>, N) {
121 let (znear, zfar) = self.clip_planes(zat);
122 if self.opm {
123 let (max, upp) = self.fov.max_and_upp(zat, max);
124 let mat = Matrix4::new_orthographic(-max.x, max.x, -max.y, max.y, znear, zfar);
125 (mat, upp)
126 } else {
127 let fov = self.fov.to_ver(max).into_inner();
128 let (max, upp) = self.fov.max_and_upp(zat, max);
129 let mat = Matrix4::new_perspective(max.x / max.y, fov, znear, zfar);
130 (mat, upp)
131 }
132 }
133 #[must_use]
135 pub fn cast<M: Copy + RealField>(self) -> Scope<M>
136 where
137 N: SubsetOf<M>,
138 {
139 let (near, far) = self.zcp;
140 Scope {
141 fov: self.fov.cast(),
142 zcp: (near.to_superset(), far.to_superset()),
143 oim: self.oim,
144 opm: self.opm,
145 }
146 }
147}
148
149#[cfg(feature = "rkyv")]
150impl<N: Copy + RealField> rkyv::Archive for Scope<N> {
151 type Archived = Self;
152 type Resolver = ();
153
154 #[inline]
155 #[allow(unsafe_code)]
156 unsafe fn resolve(&self, _: usize, (): Self::Resolver, out: *mut Self::Archived) {
157 unsafe {
158 out.write(rkyv::to_archived!(*self as Self));
159 }
160 }
161}
162
163#[cfg(feature = "rkyv")]
164impl<Ser: rkyv::Fallible + ?Sized, N: Copy + RealField> rkyv::Serialize<Ser> for Scope<N> {
165 #[inline]
166 fn serialize(&self, _: &mut Ser) -> Result<Self::Resolver, Ser::Error> {
167 Ok(())
168 }
169}
170
171#[cfg(feature = "rkyv")]
172impl<De: rkyv::Fallible + ?Sized, N: Copy + RealField> rkyv::Deserialize<Self, De> for Scope<N> {
173 #[inline]
174 fn deserialize(&self, _: &mut De) -> Result<Self, De::Error> {
175 Ok(rkyv::from_archived!(*self))
176 }
177}