mirror of
https://github.com/servo/servo.git
synced 2025-08-05 13:40:08 +01:00
Add a viewer typed reference space
This commit is contained in:
parent
57c85361be
commit
b818af794a
4 changed files with 21 additions and 23 deletions
|
@ -10,9 +10,7 @@ use crate::dom::bindings::reflector::{reflect_dom_object, DomObject};
|
||||||
use crate::dom::bindings::root::{Dom, DomRoot};
|
use crate::dom::bindings::root::{Dom, DomRoot};
|
||||||
use crate::dom::globalscope::GlobalScope;
|
use crate::dom::globalscope::GlobalScope;
|
||||||
use crate::dom::xrrigidtransform::XRRigidTransform;
|
use crate::dom::xrrigidtransform::XRRigidTransform;
|
||||||
use crate::dom::xrsession::{
|
use crate::dom::xrsession::{cast_transform, ApiPose, ApiRigidTransform, ApiViewerPose, XRSession};
|
||||||
cast_transform, cast_transform_to_pose, ApiPose, ApiRigidTransform, XRSession,
|
|
||||||
};
|
|
||||||
use crate::dom::xrspace::XRSpace;
|
use crate::dom::xrspace::XRSpace;
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use euclid::{TypedRigidTransform3D, TypedVector3D};
|
use euclid::{TypedRigidTransform3D, TypedVector3D};
|
||||||
|
@ -82,7 +80,7 @@ impl XRReferenceSpace {
|
||||||
///
|
///
|
||||||
/// This is equivalent to `get_pose(self).inverse() * get_pose(viewerSpace)` (in column vector notation),
|
/// This is equivalent to `get_pose(self).inverse() * get_pose(viewerSpace)` (in column vector notation),
|
||||||
/// however we specialize it to be efficient
|
/// however we specialize it to be efficient
|
||||||
pub fn get_viewer_pose(&self, base_pose: &Frame) -> ApiRigidTransform {
|
pub fn get_viewer_pose(&self, base_pose: &Frame) -> ApiViewerPose {
|
||||||
let pose = self.get_unoffset_viewer_pose(base_pose);
|
let pose = self.get_unoffset_viewer_pose(base_pose);
|
||||||
|
|
||||||
// This may change, see https://github.com/immersive-web/webxr/issues/567
|
// This may change, see https://github.com/immersive-web/webxr/issues/567
|
||||||
|
@ -100,8 +98,8 @@ impl XRReferenceSpace {
|
||||||
/// Gets pose of the viewer with respect to this space
|
/// Gets pose of the viewer with respect to this space
|
||||||
///
|
///
|
||||||
/// Does not apply originOffset, use get_viewer_pose instead if you need it
|
/// Does not apply originOffset, use get_viewer_pose instead if you need it
|
||||||
pub fn get_unoffset_viewer_pose(&self, base_pose: &Frame) -> ApiRigidTransform {
|
pub fn get_unoffset_viewer_pose(&self, base_pose: &Frame) -> ApiViewerPose {
|
||||||
let viewer_pose = cast_transform(base_pose.transform);
|
let viewer_pose: ApiViewerPose = cast_transform(base_pose.transform);
|
||||||
// all math is in column-vector notation
|
// all math is in column-vector notation
|
||||||
// we use the following equation to verify correctness here:
|
// we use the following equation to verify correctness here:
|
||||||
// get_viewer_pose(space) = get_pose(space).inverse() * get_pose(viewer_space)
|
// get_viewer_pose(space) = get_pose(space).inverse() * get_pose(viewer_space)
|
||||||
|
@ -164,7 +162,7 @@ impl XRReferenceSpace {
|
||||||
// the floor-level space is 2m below the eye-level space, which is (0, 0, 0)
|
// the floor-level space is 2m below the eye-level space, which is (0, 0, 0)
|
||||||
TypedVector3D::new(0., -2., 0.).into()
|
TypedVector3D::new(0., -2., 0.).into()
|
||||||
},
|
},
|
||||||
XRReferenceSpaceType::Viewer => cast_transform_to_pose(base_pose.transform),
|
XRReferenceSpaceType::Viewer => cast_transform(base_pose.transform),
|
||||||
_ => unimplemented!(),
|
_ => unimplemented!(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -276,17 +276,14 @@ impl XRSessionMethods for XRSession {
|
||||||
pub struct ApiSpace;
|
pub struct ApiSpace;
|
||||||
// The pose of an object in native-space. Should never be exposed.
|
// The pose of an object in native-space. Should never be exposed.
|
||||||
pub type ApiPose = TypedRigidTransform3D<f32, ApiSpace, webxr_api::Native>;
|
pub type ApiPose = TypedRigidTransform3D<f32, ApiSpace, webxr_api::Native>;
|
||||||
|
// The pose of the viewer in some api-space.
|
||||||
|
pub type ApiViewerPose = TypedRigidTransform3D<f32, webxr_api::Viewer, ApiSpace>;
|
||||||
// A transform between objects in some API-space
|
// A transform between objects in some API-space
|
||||||
pub type ApiRigidTransform = TypedRigidTransform3D<f32, ApiSpace, ApiSpace>;
|
pub type ApiRigidTransform = TypedRigidTransform3D<f32, ApiSpace, ApiSpace>;
|
||||||
|
|
||||||
#[allow(unsafe_code)]
|
#[allow(unsafe_code)]
|
||||||
pub fn cast_transform_to_pose<T>(
|
pub fn cast_transform<T, U, V, W>(
|
||||||
transform: TypedRigidTransform3D<f32, T, webxr_api::Native>,
|
transform: TypedRigidTransform3D<f32, T, U>,
|
||||||
) -> ApiPose {
|
) -> TypedRigidTransform3D<f32, V, W> {
|
||||||
unsafe { mem::transmute(transform) }
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(unsafe_code)]
|
|
||||||
pub fn cast_transform<T, U>(transform: TypedRigidTransform3D<f32, T, U>) -> ApiRigidTransform {
|
|
||||||
unsafe { mem::transmute(transform) }
|
unsafe { mem::transmute(transform) }
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,7 @@ use crate::dom::bindings::root::{Dom, DomRoot};
|
||||||
use crate::dom::globalscope::GlobalScope;
|
use crate::dom::globalscope::GlobalScope;
|
||||||
use crate::dom::vrframedata::create_typed_array;
|
use crate::dom::vrframedata::create_typed_array;
|
||||||
use crate::dom::xrrigidtransform::XRRigidTransform;
|
use crate::dom::xrrigidtransform::XRRigidTransform;
|
||||||
use crate::dom::xrsession::{cast_transform, ApiRigidTransform, XRSession};
|
use crate::dom::xrsession::{cast_transform, ApiViewerPose, XRSession};
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use js::jsapi::{Heap, JSContext, JSObject};
|
use js::jsapi::{Heap, JSContext, JSObject};
|
||||||
use std::ptr::NonNull;
|
use std::ptr::NonNull;
|
||||||
|
@ -45,13 +45,16 @@ impl XRView {
|
||||||
session: &XRSession,
|
session: &XRSession,
|
||||||
view: &View<V>,
|
view: &View<V>,
|
||||||
eye: XREye,
|
eye: XREye,
|
||||||
pose: &ApiRigidTransform,
|
pose: &ApiViewerPose,
|
||||||
) -> DomRoot<XRView> {
|
) -> DomRoot<XRView> {
|
||||||
// XXXManishearth compute and cache projection matrices on the Display
|
// XXXManishearth compute and cache projection matrices on the Display
|
||||||
let offset = cast_transform(view.transform);
|
|
||||||
|
|
||||||
let transform = pose.post_mul(&offset.into());
|
// this transform is the pose of the viewer in the eye space, i.e. it is the transform
|
||||||
let transform = XRRigidTransform::new(global, transform);
|
// from the viewer space to the eye space. We invert it to get the pose of the eye in the viewer space.
|
||||||
|
let offset = view.transform.inverse();
|
||||||
|
|
||||||
|
let transform = pose.pre_mul(&offset);
|
||||||
|
let transform = XRRigidTransform::new(global, cast_transform(transform));
|
||||||
|
|
||||||
let ret = reflect_dom_object(
|
let ret = reflect_dom_object(
|
||||||
Box::new(XRView::new_inherited(session, &transform, eye)),
|
Box::new(XRView::new_inherited(session, &transform, eye)),
|
||||||
|
|
|
@ -10,7 +10,7 @@ use crate::dom::bindings::root::DomRoot;
|
||||||
use crate::dom::globalscope::GlobalScope;
|
use crate::dom::globalscope::GlobalScope;
|
||||||
use crate::dom::xrpose::XRPose;
|
use crate::dom::xrpose::XRPose;
|
||||||
use crate::dom::xrrigidtransform::XRRigidTransform;
|
use crate::dom::xrrigidtransform::XRRigidTransform;
|
||||||
use crate::dom::xrsession::{ApiRigidTransform, XRSession};
|
use crate::dom::xrsession::{cast_transform, ApiViewerPose, XRSession};
|
||||||
use crate::dom::xrview::XRView;
|
use crate::dom::xrview::XRView;
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use js::conversions::ToJSValConvertible;
|
use js::conversions::ToJSValConvertible;
|
||||||
|
@ -37,7 +37,7 @@ impl XRViewerPose {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
global: &GlobalScope,
|
global: &GlobalScope,
|
||||||
session: &XRSession,
|
session: &XRSession,
|
||||||
pose: ApiRigidTransform,
|
pose: ApiViewerPose,
|
||||||
) -> DomRoot<XRViewerPose> {
|
) -> DomRoot<XRViewerPose> {
|
||||||
rooted_vec!(let mut views);
|
rooted_vec!(let mut views);
|
||||||
session.with_session(|s| match s.views() {
|
session.with_session(|s| match s.views() {
|
||||||
|
@ -49,7 +49,7 @@ impl XRViewerPose {
|
||||||
views.push(XRView::new(global, session, &right, XREye::Right, &pose));
|
views.push(XRView::new(global, session, &right, XREye::Right, &pose));
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
let transform = XRRigidTransform::new(global, pose);
|
let transform = XRRigidTransform::new(global, cast_transform(pose));
|
||||||
let pose = reflect_dom_object(
|
let pose = reflect_dom_object(
|
||||||
Box::new(XRViewerPose::new_inherited(&transform)),
|
Box::new(XRViewerPose::new_inherited(&transform)),
|
||||||
global,
|
global,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue