mirror of
https://github.com/servo/servo.git
synced 2025-08-05 05:30:08 +01:00
Auto merge of #23164 - Manishearth:getpose, r=asajeffrey
Add XRFrame.getPose() I think I've figured out the model of poses, waiting on Nell for confirmation. Basically, `getViewerPose(p)` is equivalent to `getPose(source=viewerSpace, relative_to=p)` The eye-level space, for example, is stationary and stuck to the origin. The position-disabled and identity spaces somewhat counterintuitively follow you around (and appear to be stationary from `getViewerPose()` but not `getPose()`. The incorrect mental model kinda "works" when looking at only `getViewerPose()`, but we need to figure it out for `getPose()`. Todo (may add to this PR, but probably not) - implement `XRSession.viewerSpace` - implement position-disabled - implement floor-level (hard to test without a 6dof device) r? @asajeffrey <!-- Reviewable:start --> --- This change is [<img src="https://reviewable.io/review_button.svg" height="34" align="absmiddle" alt="Reviewable"/>](https://reviewable.io/reviews/servo/servo/23164) <!-- Reviewable:end -->
This commit is contained in:
commit
967efc7fbc
7 changed files with 127 additions and 41 deletions
|
@ -8,6 +8,7 @@
|
||||||
interface XRFrame {
|
interface XRFrame {
|
||||||
readonly attribute XRSession session;
|
readonly attribute XRSession session;
|
||||||
|
|
||||||
XRViewerPose? getViewerPose(XRReferenceSpace referenceSpace);
|
[Throws] XRViewerPose? getViewerPose(XRReferenceSpace referenceSpace);
|
||||||
|
[Throws] XRPose? getPose(XRSpace space, XRSpace relativeTo);
|
||||||
// XRInputPose? getInputPose(XRInputSource inputSource, optional XRReferenceSpace referenceSpace);
|
// XRInputPose? getInputPose(XRInputSource inputSource, optional XRReferenceSpace referenceSpace);
|
||||||
};
|
};
|
||||||
|
|
|
@ -20,6 +20,7 @@ interface XRSession : EventTarget {
|
||||||
readonly attribute XREnvironmentBlendMode environmentBlendMode;
|
readonly attribute XREnvironmentBlendMode environmentBlendMode;
|
||||||
|
|
||||||
readonly attribute XRRenderState renderState;
|
readonly attribute XRRenderState renderState;
|
||||||
|
readonly attribute XRSpace viewerSpace;
|
||||||
|
|
||||||
// // Methods
|
// // Methods
|
||||||
Promise<XRReferenceSpace> requestReferenceSpace(XRReferenceSpaceOptions options);
|
Promise<XRReferenceSpace> requestReferenceSpace(XRReferenceSpaceOptions options);
|
||||||
|
|
|
@ -4,11 +4,15 @@
|
||||||
|
|
||||||
use crate::dom::bindings::codegen::Bindings::XRFrameBinding;
|
use crate::dom::bindings::codegen::Bindings::XRFrameBinding;
|
||||||
use crate::dom::bindings::codegen::Bindings::XRFrameBinding::XRFrameMethods;
|
use crate::dom::bindings::codegen::Bindings::XRFrameBinding::XRFrameMethods;
|
||||||
|
use crate::dom::bindings::error::Error;
|
||||||
|
use crate::dom::bindings::inheritance::Castable;
|
||||||
use crate::dom::bindings::reflector::{reflect_dom_object, DomObject, Reflector};
|
use crate::dom::bindings::reflector::{reflect_dom_object, DomObject, Reflector};
|
||||||
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::xrpose::XRPose;
|
||||||
use crate::dom::xrreferencespace::XRReferenceSpace;
|
use crate::dom::xrreferencespace::XRReferenceSpace;
|
||||||
use crate::dom::xrsession::XRSession;
|
use crate::dom::xrsession::XRSession;
|
||||||
|
use crate::dom::xrspace::XRSpace;
|
||||||
use crate::dom::xrviewerpose::XRViewerPose;
|
use crate::dom::xrviewerpose::XRViewerPose;
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use webvr_traits::WebVRFrameData;
|
use webvr_traits::WebVRFrameData;
|
||||||
|
@ -50,13 +54,34 @@ impl XRFrameMethods for XRFrame {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// https://immersive-web.github.io/webxr/#dom-xrframe-getviewerpose
|
/// https://immersive-web.github.io/webxr/#dom-xrframe-getviewerpose
|
||||||
fn GetViewerPose(&self, reference: &XRReferenceSpace) -> Option<DomRoot<XRViewerPose>> {
|
fn GetViewerPose(
|
||||||
|
&self,
|
||||||
|
reference: &XRReferenceSpace,
|
||||||
|
) -> Result<Option<DomRoot<XRViewerPose>>, Error> {
|
||||||
|
if self.session != reference.upcast::<XRSpace>().session() {
|
||||||
|
return Err(Error::InvalidState);
|
||||||
|
}
|
||||||
let pose = reference.get_viewer_pose(&self.data);
|
let pose = reference.get_viewer_pose(&self.data);
|
||||||
Some(XRViewerPose::new(
|
Ok(Some(XRViewerPose::new(
|
||||||
&self.global(),
|
&self.global(),
|
||||||
&self.session,
|
&self.session,
|
||||||
pose,
|
pose,
|
||||||
&self.data,
|
&self.data,
|
||||||
))
|
)))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// https://immersive-web.github.io/webxr/#dom-xrframe-getpose
|
||||||
|
fn GetPose(
|
||||||
|
&self,
|
||||||
|
space: &XRSpace,
|
||||||
|
relative_to: &XRSpace,
|
||||||
|
) -> Result<Option<DomRoot<XRPose>>, Error> {
|
||||||
|
if self.session != space.session() || self.session != relative_to.session() {
|
||||||
|
return Err(Error::InvalidState);
|
||||||
|
}
|
||||||
|
let space = space.get_pose(&self.data);
|
||||||
|
let relative_to = relative_to.get_pose(&self.data);
|
||||||
|
let pose = relative_to.inverse().pre_mul(&space);
|
||||||
|
Ok(Some(XRPose::new(&self.global(), pose)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,9 +54,12 @@ impl XRReferenceSpaceMethods for XRReferenceSpace {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl XRReferenceSpace {
|
impl XRReferenceSpace {
|
||||||
/// Gets viewer pose represented by this space
|
/// Gets pose of the viewer with respect to this space
|
||||||
|
///
|
||||||
|
/// This is equivalent to `get_pose(self).inverse() * get_pose(viewerSpace)`, however
|
||||||
|
/// we specialize it to be efficient
|
||||||
pub fn get_viewer_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
pub fn get_viewer_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
let pose = self.get_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
|
||||||
let offset = self.transform.get().transform();
|
let offset = self.transform.get().transform();
|
||||||
|
@ -64,16 +67,44 @@ impl XRReferenceSpace {
|
||||||
inverse.pre_mul(&pose)
|
inverse.pre_mul(&pose)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Gets pose represented by 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_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
pub fn get_unoffset_viewer_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
if let Some(stationary) = self.downcast::<XRStationaryReferenceSpace>() {
|
if let Some(stationary) = self.downcast::<XRStationaryReferenceSpace>() {
|
||||||
stationary.get_pose(base_pose)
|
stationary.get_unoffset_viewer_pose(base_pose)
|
||||||
} else {
|
} else {
|
||||||
// non-subclassed XRReferenceSpaces exist, obtained via the "identity"
|
// non-subclassed XRReferenceSpaces exist, obtained via the "identity"
|
||||||
// type. The pose does not depend on the base pose.
|
// type. These poses are equivalent to the viewer pose and follow the headset
|
||||||
|
// around, so the viewer is always at an identity transform with respect to them
|
||||||
RigidTransform3D::identity()
|
RigidTransform3D::identity()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Gets pose represented by this space
|
||||||
|
///
|
||||||
|
/// The reference origin used is common between all
|
||||||
|
/// get_pose calls for spaces from the same device, so this can be used to compare
|
||||||
|
/// with other spaces
|
||||||
|
pub fn get_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
|
let pose = self.get_unoffset_pose(base_pose);
|
||||||
|
|
||||||
|
// This may change, see https://github.com/immersive-web/webxr/issues/567
|
||||||
|
let offset = self.transform.get().transform();
|
||||||
|
offset.post_mul(&pose)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Gets pose represented by this space
|
||||||
|
///
|
||||||
|
/// Does not apply originOffset, use get_viewer_pose instead if you need it
|
||||||
|
pub fn get_unoffset_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
|
if let Some(stationary) = self.downcast::<XRStationaryReferenceSpace>() {
|
||||||
|
stationary.get_unoffset_pose(base_pose)
|
||||||
|
} else {
|
||||||
|
// non-subclassed XRReferenceSpaces exist, obtained via the "identity"
|
||||||
|
// type. These are equivalent to the viewer pose and follow the headset
|
||||||
|
// around
|
||||||
|
XRSpace::viewer_pose_from_frame_data(base_pose)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,6 +21,7 @@ use crate::dom::vrdisplay::VRDisplay;
|
||||||
use crate::dom::xrlayer::XRLayer;
|
use crate::dom::xrlayer::XRLayer;
|
||||||
use crate::dom::xrreferencespace::XRReferenceSpace;
|
use crate::dom::xrreferencespace::XRReferenceSpace;
|
||||||
use crate::dom::xrrenderstate::XRRenderState;
|
use crate::dom::xrrenderstate::XRRenderState;
|
||||||
|
use crate::dom::xrspace::XRSpace;
|
||||||
use crate::dom::xrstationaryreferencespace::XRStationaryReferenceSpace;
|
use crate::dom::xrstationaryreferencespace::XRStationaryReferenceSpace;
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
|
@ -82,6 +83,11 @@ impl XRSessionMethods for XRSession {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://immersive-web.github.io/webxr/#dom-xrsession-viewerspace
|
||||||
|
fn ViewerSpace(&self) -> DomRoot<XRSpace> {
|
||||||
|
XRSpace::new_viewerspace(&self.global(), &self)
|
||||||
|
}
|
||||||
|
|
||||||
/// https://immersive-web.github.io/webxr/#dom-xrsession-requestanimationframe
|
/// https://immersive-web.github.io/webxr/#dom-xrsession-requestanimationframe
|
||||||
fn UpdateRenderState(&self, init: &XRRenderStateInit) -> Rc<Promise> {
|
fn UpdateRenderState(&self, init: &XRRenderStateInit) -> Rc<Promise> {
|
||||||
let p = Promise::new(&self.global());
|
let p = Promise::new(&self.global());
|
||||||
|
|
|
@ -11,13 +11,14 @@ use crate::dom::globalscope::GlobalScope;
|
||||||
use crate::dom::xrreferencespace::XRReferenceSpace;
|
use crate::dom::xrreferencespace::XRReferenceSpace;
|
||||||
use crate::dom::xrsession::XRSession;
|
use crate::dom::xrsession::XRSession;
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use euclid::RigidTransform3D;
|
use euclid::{RigidTransform3D, Rotation3D, Vector3D};
|
||||||
use webvr_traits::WebVRFrameData;
|
use webvr_traits::WebVRFrameData;
|
||||||
|
|
||||||
#[dom_struct]
|
#[dom_struct]
|
||||||
pub struct XRSpace {
|
pub struct XRSpace {
|
||||||
eventtarget: EventTarget,
|
eventtarget: EventTarget,
|
||||||
session: Dom<XRSession>,
|
session: Dom<XRSession>,
|
||||||
|
is_viewerspace: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl XRSpace {
|
impl XRSpace {
|
||||||
|
@ -25,13 +26,21 @@ impl XRSpace {
|
||||||
XRSpace {
|
XRSpace {
|
||||||
eventtarget: EventTarget::new_inherited(),
|
eventtarget: EventTarget::new_inherited(),
|
||||||
session: Dom::from_ref(session),
|
session: Dom::from_ref(session),
|
||||||
|
is_viewerspace: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unused)]
|
fn new_viewerspace_inner(session: &XRSession) -> XRSpace {
|
||||||
pub fn new(global: &GlobalScope, session: &XRSession) -> DomRoot<XRSpace> {
|
XRSpace {
|
||||||
|
eventtarget: EventTarget::new_inherited(),
|
||||||
|
session: Dom::from_ref(session),
|
||||||
|
is_viewerspace: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new_viewerspace(global: &GlobalScope, session: &XRSession) -> DomRoot<XRSpace> {
|
||||||
reflect_dom_object(
|
reflect_dom_object(
|
||||||
Box::new(XRSpace::new_inherited(session)),
|
Box::new(XRSpace::new_viewerspace_inner(session)),
|
||||||
global,
|
global,
|
||||||
XRSpaceBinding::Wrap,
|
XRSpaceBinding::Wrap,
|
||||||
)
|
)
|
||||||
|
@ -39,25 +48,35 @@ impl XRSpace {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl XRSpace {
|
impl XRSpace {
|
||||||
/// Gets viewer pose represented by this space
|
/// Gets pose represented by this space
|
||||||
#[allow(unused)]
|
///
|
||||||
pub fn get_viewer_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
/// The reference origin used is common between all
|
||||||
|
/// get_pose calls for spaces from the same device, so this can be used to compare
|
||||||
|
/// with other spaces
|
||||||
|
pub fn get_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
if let Some(reference) = self.downcast::<XRReferenceSpace>() {
|
if let Some(reference) = self.downcast::<XRReferenceSpace>() {
|
||||||
reference.get_viewer_pose(base_pose)
|
reference.get_pose(base_pose)
|
||||||
|
} else if self.is_viewerspace {
|
||||||
|
XRSpace::viewer_pose_from_frame_data(base_pose)
|
||||||
} else {
|
} else {
|
||||||
unreachable!()
|
unreachable!()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Gets pose represented by this space
|
pub fn viewer_pose_from_frame_data(data: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
///
|
let pos = data.pose.position.unwrap_or([0., 0., 0.]);
|
||||||
/// Does not apply originOffset, use get_viewer_pose instead if you need it
|
let translation = Vector3D::new(pos[0] as f64, pos[1] as f64, pos[2] as f64);
|
||||||
#[allow(unused)]
|
let orient = data.pose.orientation.unwrap_or([0., 0., 0., 0.]);
|
||||||
pub fn get_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
let rotation = Rotation3D::quaternion(
|
||||||
if let Some(reference) = self.downcast::<XRReferenceSpace>() {
|
orient[0] as f64,
|
||||||
reference.get_pose(base_pose)
|
orient[1] as f64,
|
||||||
} else {
|
orient[2] as f64,
|
||||||
unreachable!()
|
orient[3] as f64,
|
||||||
}
|
);
|
||||||
|
RigidTransform3D::new(rotation, translation)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn session(&self) -> &XRSession {
|
||||||
|
&self.session
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,8 +10,9 @@ use crate::dom::globalscope::GlobalScope;
|
||||||
use crate::dom::xrreferencespace::XRReferenceSpace;
|
use crate::dom::xrreferencespace::XRReferenceSpace;
|
||||||
use crate::dom::xrrigidtransform::XRRigidTransform;
|
use crate::dom::xrrigidtransform::XRRigidTransform;
|
||||||
use crate::dom::xrsession::XRSession;
|
use crate::dom::xrsession::XRSession;
|
||||||
|
use crate::dom::xrspace::XRSpace;
|
||||||
use dom_struct::dom_struct;
|
use dom_struct::dom_struct;
|
||||||
use euclid::{RigidTransform3D, Rotation3D, Vector3D};
|
use euclid::RigidTransform3D;
|
||||||
use webvr_traits::WebVRFrameData;
|
use webvr_traits::WebVRFrameData;
|
||||||
|
|
||||||
#[dom_struct]
|
#[dom_struct]
|
||||||
|
@ -50,20 +51,22 @@ impl XRStationaryReferenceSpace {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl XRStationaryReferenceSpace {
|
impl XRStationaryReferenceSpace {
|
||||||
|
/// Gets pose of the viewer with respect to this space
|
||||||
|
///
|
||||||
|
/// Does not apply originOffset, use get_viewer_pose on XRReferenceSpace instead
|
||||||
|
pub fn get_unoffset_viewer_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
|
// XXXManishearth add floor-level transform for floor-level and disable position in position-disabled
|
||||||
|
XRSpace::viewer_pose_from_frame_data(base_pose)
|
||||||
|
}
|
||||||
|
|
||||||
/// Gets pose represented by this space
|
/// Gets pose represented by this space
|
||||||
///
|
///
|
||||||
/// Does not apply originOffset, use get_viewer_pose instead
|
/// Does not apply originOffset, use get_pose on XRReferenceSpace instead
|
||||||
pub fn get_pose(&self, base_pose: &WebVRFrameData) -> RigidTransform3D<f64> {
|
pub fn get_unoffset_pose(&self, _: &WebVRFrameData) -> RigidTransform3D<f64> {
|
||||||
// XXXManishearth add floor-level transform for floor-level and disable position in position-disabled
|
// XXXManishearth add floor-level transform for floor-level and disable position in position-disabled
|
||||||
let pos = base_pose.pose.position.unwrap_or([0., 0., 0.]);
|
|
||||||
let translation = Vector3D::new(pos[0] as f64, pos[1] as f64, pos[2] as f64);
|
// The eye-level pose is basically whatever the headset pose was at t=0, which
|
||||||
let orient = base_pose.pose.orientation.unwrap_or([0., 0., 0., 0.]);
|
// for most devices is (0, 0, 0)
|
||||||
let rotation = Rotation3D::quaternion(
|
RigidTransform3D::identity()
|
||||||
orient[0] as f64,
|
|
||||||
orient[1] as f64,
|
|
||||||
orient[2] as f64,
|
|
||||||
orient[3] as f64,
|
|
||||||
);
|
|
||||||
RigidTransform3D::new(rotation, translation)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue