script/dom/webxr/
xrframe.rs1use std::cell::Cell;
6
7use dom_struct::dom_struct;
8use js::gc::CustomAutoRooterGuard;
9use js::typedarray::Float32Array;
10use webxr_api::{Frame, LayerId, SubImages};
11
12use crate::dom::bindings::codegen::Bindings::XRFrameBinding::XRFrameMethods;
13use crate::dom::bindings::error::Error;
14use crate::dom::bindings::inheritance::Castable;
15use crate::dom::bindings::num::Finite;
16use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object};
17use crate::dom::bindings::root::{Dom, DomRoot};
18use crate::dom::window::Window;
19use crate::dom::xrhittestresult::XRHitTestResult;
20use crate::dom::xrhittestsource::XRHitTestSource;
21use crate::dom::xrjointpose::XRJointPose;
22use crate::dom::xrjointspace::XRJointSpace;
23use crate::dom::xrpose::XRPose;
24use crate::dom::xrreferencespace::XRReferenceSpace;
25use crate::dom::xrsession::{ApiPose, XRSession};
26use crate::dom::xrspace::XRSpace;
27use crate::dom::xrviewerpose::XRViewerPose;
28use crate::script_runtime::CanGc;
29
30#[dom_struct]
31pub(crate) struct XRFrame {
32 reflector_: Reflector,
33 session: Dom<XRSession>,
34 #[no_trace]
35 data: Frame,
36 active: Cell<bool>,
37 animation_frame: Cell<bool>,
38}
39
40impl XRFrame {
41 fn new_inherited(session: &XRSession, data: Frame) -> XRFrame {
42 XRFrame {
43 reflector_: Reflector::new(),
44 session: Dom::from_ref(session),
45 data,
46 active: Cell::new(false),
47 animation_frame: Cell::new(false),
48 }
49 }
50
51 pub(crate) fn new(
52 window: &Window,
53 session: &XRSession,
54 data: Frame,
55 can_gc: CanGc,
56 ) -> DomRoot<XRFrame> {
57 reflect_dom_object(
58 Box::new(XRFrame::new_inherited(session, data)),
59 window,
60 can_gc,
61 )
62 }
63
64 pub(crate) fn set_active(&self, active: bool) {
66 self.active.set(active);
67 }
68
69 pub(crate) fn set_animation_frame(&self, animation_frame: bool) {
71 self.animation_frame.set(animation_frame);
72 }
73
74 pub(crate) fn get_pose(&self, space: &XRSpace) -> Option<ApiPose> {
75 space.get_pose(&self.data)
76 }
77
78 pub(crate) fn get_sub_images(&self, layer_id: LayerId) -> Option<&SubImages> {
79 self.data
80 .sub_images
81 .iter()
82 .find(|sub_images| sub_images.layer_id == layer_id)
83 }
84}
85
86impl XRFrameMethods<crate::DomTypeHolder> for XRFrame {
87 fn Session(&self) -> DomRoot<XRSession> {
89 DomRoot::from_ref(&self.session)
90 }
91
92 fn PredictedDisplayTime(&self) -> Finite<f64> {
94 Finite::new(self.data.predicted_display_time)
97 .expect("Failed to create predictedDisplayTime")
98 }
99
100 fn GetViewerPose(
102 &self,
103 reference: &XRReferenceSpace,
104 can_gc: CanGc,
105 ) -> Result<Option<DomRoot<XRViewerPose>>, Error> {
106 if self.session != reference.upcast::<XRSpace>().session() {
107 return Err(Error::InvalidState(None));
108 }
109
110 if !self.active.get() || !self.animation_frame.get() {
111 return Err(Error::InvalidState(None));
112 }
113
114 let to_base = if let Some(to_base) = reference.get_base_transform(&self.data) {
115 to_base
116 } else {
117 return Ok(None);
118 };
119 let viewer_pose = if let Some(pose) = self.data.pose.as_ref() {
120 pose
121 } else {
122 return Ok(None);
123 };
124 Ok(Some(XRViewerPose::new(
125 self.global().as_window(),
126 &self.session,
127 to_base,
128 viewer_pose,
129 can_gc,
130 )))
131 }
132
133 fn GetPose(
135 &self,
136 space: &XRSpace,
137 base_space: &XRSpace,
138 can_gc: CanGc,
139 ) -> Result<Option<DomRoot<XRPose>>, Error> {
140 if self.session != space.session() || self.session != base_space.session() {
141 return Err(Error::InvalidState(None));
142 }
143 if !self.active.get() {
144 return Err(Error::InvalidState(None));
145 }
146 let space = if let Some(space) = self.get_pose(space) {
147 space
148 } else {
149 return Ok(None);
150 };
151 let base_space = if let Some(r) = self.get_pose(base_space) {
152 r
153 } else {
154 return Ok(None);
155 };
156 let pose = space.then(&base_space.inverse());
157 Ok(Some(XRPose::new(self.global().as_window(), pose, can_gc)))
158 }
159
160 fn GetJointPose(
162 &self,
163 space: &XRJointSpace,
164 base_space: &XRSpace,
165 can_gc: CanGc,
166 ) -> Result<Option<DomRoot<XRJointPose>>, Error> {
167 if self.session != space.upcast::<XRSpace>().session() ||
168 self.session != base_space.session()
169 {
170 return Err(Error::InvalidState(None));
171 }
172 if !self.active.get() {
173 return Err(Error::InvalidState(None));
174 }
175 let joint_frame = if let Some(frame) = space.frame(&self.data) {
176 frame
177 } else {
178 return Ok(None);
179 };
180 let base_space = if let Some(r) = self.get_pose(base_space) {
181 r
182 } else {
183 return Ok(None);
184 };
185 let pose = joint_frame.pose.then(&base_space.inverse());
186 Ok(Some(XRJointPose::new(
187 self.global().as_window(),
188 pose.cast_unit(),
189 Some(joint_frame.radius),
190 can_gc,
191 )))
192 }
193
194 fn GetHitTestResults(&self, source: &XRHitTestSource) -> Vec<DomRoot<XRHitTestResult>> {
196 self.data
197 .hit_test_results
198 .iter()
199 .filter(|r| r.id == source.id())
200 .map(|r| {
201 XRHitTestResult::new(
202 self.global().as_window(),
203 *r,
204 self,
205 CanGc::deprecated_note(),
206 )
207 })
208 .collect()
209 }
210
211 fn FillJointRadii(
213 &self,
214 joint_spaces: Vec<DomRoot<XRJointSpace>>,
215 mut radii: CustomAutoRooterGuard<Float32Array>,
216 ) -> Result<bool, Error> {
217 if !self.active.get() {
218 return Err(Error::InvalidState(None));
219 }
220
221 for joint_space in &joint_spaces {
222 if self.session != joint_space.upcast::<XRSpace>().session() {
223 return Err(Error::InvalidState(None));
224 }
225 }
226
227 if joint_spaces.len() > radii.len() {
228 return Err(Error::Type(
229 c"Length of radii does not match length of joint spaces".to_owned(),
230 ));
231 }
232
233 let mut radii_vec = radii.to_vec();
234 let mut all_valid = true;
235 radii_vec.iter_mut().enumerate().for_each(|(i, radius)| {
236 if let Some(joint_frame) = joint_spaces
237 .get(i)
238 .and_then(|joint_space| joint_space.frame(&self.data))
239 {
240 *radius = joint_frame.radius;
241 } else {
242 all_valid = false;
243 }
244 });
245
246 if !all_valid {
247 radii_vec.fill(f32::NAN);
248 }
249
250 radii.update(&radii_vec);
251
252 Ok(all_valid)
253 }
254
255 fn FillPoses(
257 &self,
258 spaces: Vec<DomRoot<XRSpace>>,
259 base_space: &XRSpace,
260 mut transforms: CustomAutoRooterGuard<Float32Array>,
261 ) -> Result<bool, Error> {
262 if !self.active.get() {
263 return Err(Error::InvalidState(None));
264 }
265
266 for space in &spaces {
267 if self.session != space.session() {
268 return Err(Error::InvalidState(None));
269 }
270 }
271
272 if self.session != base_space.session() {
273 return Err(Error::InvalidState(None));
274 }
275
276 if spaces.len() * 16 > transforms.len() {
277 return Err(Error::Type(
278 c"Transforms array length does not match 16 * spaces length".to_owned(),
279 ));
280 }
281
282 let mut transforms_vec = transforms.to_vec();
283 let mut all_valid = true;
284 spaces.iter().enumerate().for_each(|(i, space)| {
285 let Some(joint_pose) = self.get_pose(space) else {
286 all_valid = false;
287 return;
288 };
289 let Some(base_pose) = self.get_pose(base_space) else {
290 all_valid = false;
291 return;
292 };
293 let pose = joint_pose.then(&base_pose.inverse());
294 let elements = pose.to_transform();
295 let elements_arr = elements.to_array();
296 transforms_vec[i * 16..(i + 1) * 16].copy_from_slice(&elements_arr);
297 });
298
299 if !all_valid {
300 transforms_vec.fill(f32::NAN);
301 }
302
303 transforms.update(&transforms_vec);
304
305 Ok(all_valid)
306 }
307}