/* 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 euclid::RigidTransform3D; use js::typedarray::{Float32, Float32Array}; use webxr_api::{ApiSpace, View}; use crate::dom::bindings::buffer_source::HeapBufferSource; use crate::dom::bindings::codegen::Bindings::XRViewBinding::{XREye, XRViewMethods}; use crate::dom::bindings::num::Finite; use crate::dom::bindings::reflector::{Reflector, reflect_dom_object}; use crate::dom::bindings::root::{Dom, DomRoot}; use crate::dom::globalscope::GlobalScope; use crate::dom::xrrigidtransform::XRRigidTransform; use crate::dom::xrsession::{BaseSpace, BaseTransform, XRSession, cast_transform}; use crate::script_runtime::{CanGc, JSContext}; #[dom_struct] pub(crate) struct XRView { reflector_: Reflector, session: Dom, eye: XREye, viewport_index: usize, #[ignore_malloc_size_of = "mozjs"] proj: HeapBufferSource, #[ignore_malloc_size_of = "defined in rust-webxr"] #[no_trace] view: View, transform: Dom, requested_viewport_scale: Cell, } impl XRView { fn new_inherited( session: &XRSession, transform: &XRRigidTransform, eye: XREye, viewport_index: usize, view: View, ) -> XRView { XRView { reflector_: Reflector::new(), session: Dom::from_ref(session), eye, viewport_index, proj: HeapBufferSource::default(), view, transform: Dom::from_ref(transform), requested_viewport_scale: Cell::new(1.0), } } pub(crate) fn new( global: &GlobalScope, session: &XRSession, view: &View, eye: XREye, viewport_index: usize, to_base: &BaseTransform, can_gc: CanGc, ) -> DomRoot { let transform: RigidTransform3D = view.transform.then(to_base); let transform = XRRigidTransform::new(global, cast_transform(transform), can_gc); reflect_dom_object( Box::new(XRView::new_inherited( session, &transform, eye, viewport_index, view.cast_unit(), )), global, can_gc, ) } pub(crate) fn session(&self) -> &XRSession { &self.session } pub(crate) fn viewport_index(&self) -> usize { self.viewport_index } } impl XRViewMethods for XRView { /// fn Eye(&self) -> XREye { self.eye } /// fn ProjectionMatrix(&self, _cx: JSContext, can_gc: CanGc) -> Float32Array { if !self.proj.is_initialized() { let cx = GlobalScope::get_cx(); // row_major since euclid uses row vectors let proj = self.view.projection.to_array(); self.proj .set_data(cx, &proj, can_gc) .expect("Failed to set projection matrix.") } self.proj .get_typed_array() .expect("Failed to get projection matrix.") } /// fn Transform(&self) -> DomRoot { DomRoot::from_ref(&self.transform) } /// fn GetRecommendedViewportScale(&self) -> Option> { // Just return 1.0 since we currently will always use full-sized viewports Finite::new(1.0) } /// fn RequestViewportScale(&self, scale: Option>) { if let Some(scale) = scale { if *scale > 0.0 { let clamped_scale = scale.clamp(0.0, 1.0); self.requested_viewport_scale.set(clamped_scale); } } } /// fn IsFirstPersonObserver(&self) -> bool { // Servo is not currently supported anywhere that supports this, so return false false } }