Anchors
Anchors are references to real-world features (like surfaces, points, or objects) detected by an AR/MR system. These anchors act as reference points that allow virtual objects to be placed and maintained in the 3D world relative to real-world coordinates. In AR/MR, objects need to stay in place even as the user moves around or changes perspective. Anchors ensure that virtual content remains fixed in relation to real-world surfaces or points, allowing it to stay in the correct position, scale, and orientation.
Once you enabled the anchor feature in Babylon.js, there are two ways to add an anchor:
addAnchorPointUsingHitTestResultAsync
- it allows to add an anchor at hit-test positionaddAnchorAtPositionAndRotationAsync
- it allows to add an anchor in any position and rotation in the scene.
In this example, the first method is used to add the anchor(s). It begins with the Hit Test example, with the key difference being that we now track the hit test result. The loadAssetContainerAsync method loads the 3D model and stores it in a variable for later use. When a user perform a pinch gesture (or a click using the controller), the hit test result is passed to the addAnchorPointUsingHitTestResultAsync
method, which handles setting the anchor. The onAnchorAddedObservable
and onAnchorRemovedObservable
are then responsible for attaching the 3D model to the anchor and managing its removal or disposal.
const scene = useScene();
const xrExperience = useXrExperience();
const torusRef = useRef<Mesh>(null);
const cheetahRef = useRef<Mesh>(null);
useEffect(() => {
let hitTest: IWebXRHitResult = undefined;
const featuresManager = xrExperience.baseExperience.featuresManager;
const xtTest = featuresManager.enableFeature(WebXRHitTest, 'latest') as WebXRHitTest;
const xrAnchors = featuresManager.enableFeature(WebXRAnchorSystem, 'latest') as WebXRAnchorSystem;
loadAssetContainerAsync('/meshes/cheetah/cheetah.obj', scene).then(container => {
const cheetah = container.meshes[0] as Mesh;
cheetah.rotation.y = Tools.ToRadians(140);
cheetahRef.current = cheetah;
});
xtTest.onHitTestResultObservable.add(result => {
const torus = torusRef.current!;
if (result.length) {
torus.isVisible = true;
hitTest = result[0];
result[0].transformationMatrix.decompose(torus.scaling, torus.rotationQuaternion as Quaternion, torus.position);
} else {
torus.isVisible = false;
hitTest = undefined;
}
});
xrAnchors.onAnchorAddedObservable.add(anchor => {
const cheetah = cheetahRef.current!.clone(`cheetah-${anchor.id}`);
const transformNode = new TransformNode('cheetah-parent');
cheetah.parent = transformNode;
anchor.attachedNode = transformNode;
});
xrAnchors.onAnchorRemovedObservable.add(anchor => {
if (anchor.attachedNode) {
anchor.attachedNode.dispose();
}
});
scene.onPointerDown = () => {
if (xtTest && hitTest && xrAnchors && cheetahRef.current && xrExperience.baseExperience.state === WebXRState.IN_XR) {
xrAnchors.addAnchorPointUsingHitTestResultAsync(hitTest);
}
};
}, []);
// ...
<>
<torus ref={torusRef} name='torus' options={{ diameter: 0.15, thickness: 0.05 }} isVisible={torusRef.current?.isVisible} rotationQuaternion={new Quaternion()}>
<standardMaterial name='material' diffuseColor={Color3.FromHexString('#1e90ff')} />
</torus>
</>