/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ use std::cell::Cell; use dom_struct::dom_struct; use js::gc::CustomAutoRooterGuard; use js::typedarray::Float32Array; use webxr_api::{Frame, LayerId, SubImages}; use crate::dom::bindings::codegen::Bindings::XRFrameBinding::XRFrameMethods; use crate::dom::bindings::error::Error; use crate::dom::bindings::inheritance::Castable; use crate::dom::bindings::num::Finite; use crate::dom::bindings::reflector::{DomGlobal, Reflector, reflect_dom_object}; use crate::dom::bindings::root::{Dom, DomRoot}; use crate::dom::window::Window; use crate::dom::xrhittestresult::XRHitTestResult; use crate::dom::xrhittestsource::XRHitTestSource; use crate::dom::xrjointpose::XRJointPose; use crate::dom::xrjointspace::XRJointSpace; use crate::dom::xrpose::XRPose; use crate::dom::xrreferencespace::XRReferenceSpace; use crate::dom::xrsession::{ApiPose, XRSession}; use crate::dom::xrspace::XRSpace; use crate::dom::xrviewerpose::XRViewerPose; use crate::script_runtime::CanGc; #[dom_struct] pub(crate) struct XRFrame { reflector_: Reflector, session: Dom, #[ignore_malloc_size_of = "defined in webxr_api"] #[no_trace] data: Frame, active: Cell, animation_frame: Cell, } impl XRFrame { fn new_inherited(session: &XRSession, data: Frame) -> XRFrame { XRFrame { reflector_: Reflector::new(), session: Dom::from_ref(session), data, active: Cell::new(false), animation_frame: Cell::new(false), } } pub(crate) fn new( window: &Window, session: &XRSession, data: Frame, can_gc: CanGc, ) -> DomRoot { reflect_dom_object( Box::new(XRFrame::new_inherited(session, data)), window, can_gc, ) } /// pub(crate) fn set_active(&self, active: bool) { self.active.set(active); } /// pub(crate) fn set_animation_frame(&self, animation_frame: bool) { self.animation_frame.set(animation_frame); } pub(crate) fn get_pose(&self, space: &XRSpace) -> Option { space.get_pose(&self.data) } pub(crate) fn get_sub_images(&self, layer_id: LayerId) -> Option<&SubImages> { self.data .sub_images .iter() .find(|sub_images| sub_images.layer_id == layer_id) } } impl XRFrameMethods for XRFrame { /// fn Session(&self) -> DomRoot { DomRoot::from_ref(&self.session) } /// fn PredictedDisplayTime(&self) -> Finite { // TODO: If inline, return the same value // as the timestamp passed to XRFrameRequestCallback Finite::new(self.data.predicted_display_time) .expect("Failed to create predictedDisplayTime") } /// fn GetViewerPose( &self, reference: &XRReferenceSpace, can_gc: CanGc, ) -> Result>, Error> { if self.session != reference.upcast::().session() { return Err(Error::InvalidState); } if !self.active.get() || !self.animation_frame.get() { return Err(Error::InvalidState); } let to_base = if let Some(to_base) = reference.get_base_transform(&self.data) { to_base } else { return Ok(None); }; let viewer_pose = if let Some(pose) = self.data.pose.as_ref() { pose } else { return Ok(None); }; Ok(Some(XRViewerPose::new( self.global().as_window(), &self.session, to_base, viewer_pose, can_gc, ))) } /// fn GetPose( &self, space: &XRSpace, base_space: &XRSpace, can_gc: CanGc, ) -> Result>, Error> { if self.session != space.session() || self.session != base_space.session() { return Err(Error::InvalidState); } if !self.active.get() { return Err(Error::InvalidState); } let space = if let Some(space) = self.get_pose(space) { space } else { return Ok(None); }; let base_space = if let Some(r) = self.get_pose(base_space) { r } else { return Ok(None); }; let pose = space.then(&base_space.inverse()); Ok(Some(XRPose::new(self.global().as_window(), pose, can_gc))) } /// fn GetJointPose( &self, space: &XRJointSpace, base_space: &XRSpace, can_gc: CanGc, ) -> Result>, Error> { if self.session != space.upcast::().session() || self.session != base_space.session() { return Err(Error::InvalidState); } if !self.active.get() { return Err(Error::InvalidState); } let joint_frame = if let Some(frame) = space.frame(&self.data) { frame } else { return Ok(None); }; let base_space = if let Some(r) = self.get_pose(base_space) { r } else { return Ok(None); }; let pose = joint_frame.pose.then(&base_space.inverse()); Ok(Some(XRJointPose::new( self.global().as_window(), pose.cast_unit(), Some(joint_frame.radius), can_gc, ))) } /// fn GetHitTestResults(&self, source: &XRHitTestSource) -> Vec> { self.data .hit_test_results .iter() .filter(|r| r.id == source.id()) .map(|r| XRHitTestResult::new(self.global().as_window(), *r, self, CanGc::note())) .collect() } #[allow(unsafe_code)] /// fn FillJointRadii( &self, joint_spaces: Vec>, mut radii: CustomAutoRooterGuard, ) -> Result { if !self.active.get() { return Err(Error::InvalidState); } for joint_space in &joint_spaces { if self.session != joint_space.upcast::().session() { return Err(Error::InvalidState); } } if joint_spaces.len() > radii.len() { return Err(Error::Type( "Length of radii does not match length of joint spaces".to_string(), )); } let mut radii_vec = radii.to_vec(); let mut all_valid = true; radii_vec.iter_mut().enumerate().for_each(|(i, radius)| { if let Some(joint_frame) = joint_spaces .get(i) .and_then(|joint_space| joint_space.frame(&self.data)) { *radius = joint_frame.radius; } else { all_valid = false; } }); if !all_valid { radii_vec.fill(f32::NAN); } radii.update(&radii_vec); Ok(all_valid) } #[allow(unsafe_code)] /// fn FillPoses( &self, spaces: Vec>, base_space: &XRSpace, mut transforms: CustomAutoRooterGuard, ) -> Result { if !self.active.get() { return Err(Error::InvalidState); } for space in &spaces { if self.session != space.session() { return Err(Error::InvalidState); } } if self.session != base_space.session() { return Err(Error::InvalidState); } if spaces.len() * 16 > transforms.len() { return Err(Error::Type( "Transforms array length does not match 16 * spaces length".to_string(), )); } let mut transforms_vec = transforms.to_vec(); let mut all_valid = true; spaces.iter().enumerate().for_each(|(i, space)| { let Some(joint_pose) = self.get_pose(space) else { all_valid = false; return; }; let Some(base_pose) = self.get_pose(base_space) else { all_valid = false; return; }; let pose = joint_pose.then(&base_pose.inverse()); let elements = pose.to_transform(); let elements_arr = elements.to_array(); transforms_vec[i * 16..(i + 1) * 16].copy_from_slice(&elements_arr); }); if !all_valid { transforms_vec.fill(f32::NAN); } transforms.update(&transforms_vec); Ok(all_valid) } }