What is ARAnchor exactly?

2020-02-01 04:58发布

I'm trying to understand and use ARKit. But there is one thing that I cannot fully understand.

Apple said about ARAnchor:

A real-world position and orientation that can be used for placing objects in an AR scene.

But that's not enough. So my questions are:

  • What is ARAnchor exactly?
  • What are the differences between anchors and feature points?
  • Is ARAnchor just part of feature points?
  • And how does ARKit determines its anchors?

1条回答
Explosion°爆炸
2楼-- · 2020-02-01 05:34

Updated: January 19, 2020.

ARAnchor

Simply put, ARAnchor is an invisible null-object that can hold a 3D content (at anchor's position) in World Space. Think of ARAnchor just like it's a local axis for your 3D object. Every 3D object has a pivot point, right? So this pivot point must meet an ARAnchor.

If you do not use anchors in ARKit/RealityKit app, your virtual objects can drift away from where they were placed and this can impact your app's realism and user experience. Thus, anchors are crucial elements in AR scene.

According to ARKit documentation 2017:

ARAnchor is a real-world position and orientation that can be used for placing objects in AR Scene. Adding an anchor to the session helps ARKit to optimize world-tracking accuracy in the area around that anchor, so that virtual objects appear to stay in place relative to the real world. If a virtual object moves, remove the corresponding anchor from the old position and add one at the new position.

ARAnchor is a parent class for all other types of anchors existing in ARKit framework, hence all these subclasses inherit from ARAnchor class but cannot use it directly in your code. I should also say that ARAnchor and Feature Points have nothing in common. Feature Points are rather for debugging.

Here's an image with visual representation of plane anchor. But keep in mind: by default, you can neither see a detected plane nor its corresponding ARPlaneAnchor.

enter image description here

In ARKit you can automatically add ARAnchors to your scene using different scenarios:

  • ARPlaneAnchor

    • If horizontal or/and vertical planeDetection instance property is ON, ARKit is able to add ARPlaneAnchors to the session.
  • ARImageAnchor (conforms to ARTrackable protocol)

    • This type of anchors contains information about the position and orientation of an image detected (anchor's located in the image center) in AR world-tracking session. For activation use detectionImages instance property. In ARKit 2.0 you can totally track up to 25 images, in ARKit 3.0 – up to 100 images, respectively. But, in both cases, not more than just 4 images simultaneously.
  • ARBodyAnchor (conforms to ARTrackable protocol)

    • In the latest release of ARKit you can enable body tracking by running your session with ARBodyTrackingConfiguration(). You'll get ARBodyAnchor in a Root Joint of 3D Skeleton.
  • ARFaceAnchor (conforms to ARTrackable protocol)

    • Face Anchor stores the information about the topology and pose, as well as face's expression that you can detect with a front TrueDepth camera. When face is detected, Face Anchor will be attached slightly behind a nose, in the center of a face. In ARKit 2.0 you can track just one face, in ARKit 3.0 – up to 3 faces, simultaneously.
  • ARObjectAnchor

    • This type of anchors holds an information about 6 Degrees of Freedom (position and orientation) of a real-world 3D object detected in a world-tracking session.
  • AREnvironmentProbeAnchor

    • Probe Anchor provides environmental lighting information for a specific area of space in a world-tracking session. ARKit's Artificial Intelligence uses it to supply metallic shaders with environmental reflections.
  • ARParticipantAnchor

    • This is an indispensable anchor type for multiuser AR experiences. If you want to employ it, use true value for isCollaborationEnabled instance property and Multipeer Connectivity framework.

There are also other regular approaches to create anchors in AR session:

  • Hit-Testing method

    • Tapping on the screen, projects a point onto a hidden detected plane, placing ARAnchor on a location where imaginary ray intersects with this plane.
  • Ray-Casting method

    • Tapping on the screen, also projects a point onto a hidden detected plane. The main difference of Ray-Casting from Hit-Testing is that, when using the first one ARKit can keep refining the ray cast as it learns more about detected surfaces.
  • Feature points

    • Special yellow points that ARKit automatically generates on a high-contrast margins of real-world objects, can give you a place to put an ARAnchor on. This approach is possible thanks to a hit-testing method too.
  • ARCamera's transform

    • iPhone's camera's position and orientation (Matrix 4x4) can be easily used as a place for ARAnchor.
  • Any arbitrary world position

    • Place your ARAnchor anywhere in a scene you wish to.

And below is a code's snippet how to implement anchors inside renderer method:

func renderer(_ renderer: SCNSceneRenderer, didAdd node: SCNNode, for anchor: ARAnchor) {

    guard let planeAnchor = anchor as? ARPlaneAnchor else { return }
    let grid = Grid(anchor: planeAnchor)
    node.addChildNode(grid)
}

enter image description here

AnchorEntity

RealityKit framework was released in 2019. It has a new class named AnchorEntity. You can use AnchorEntity as the root point of an entity hierarchy, and you can add it to the anchors collection for a scene instance. This enables ARKit to place the anchor entity, along with all of its hierarchical descendants, into the real world.

According to RealityKit documentation 2019:

AnchorEntity is an anchor that tethers virtual content to a real-world object in an AR session.

Let's look at a code snippet:

func makeUIView(context: Context) -> ARView {

    let arView = ARView(frame: .zero)
    let modelAnchor = try! Experience.loadModel()
    arView.scene.anchors.append(modelAnchor)
    return arView
}

AnchorEntity stores three components:

To find out the difference between ARAnchor and AnchorEntity look at THIS POST.

Here are seven AnchorEntity's cases available in current version of RealityKit:

// Fixed position in the AR scene
AnchorEntity(.world(transform: mtx)) 

// For body tracking (Motion Capture)
AnchorEntity(.body)

// Pinned to the tracking camera
AnchorEntity(.camera)

// For face tracking (front camera)
AnchorEntity(.face)

// For image tracking config
AnchorEntity(.image(group: "Group", name: "model"))

// For object tracking config
AnchorEntity(.object(group: "Group", name: "object"))

// For horizontal plane detection
AnchorEntity(.plane(.horizontal, classification: .table, minimumBounds: [3.0, 7.0]))

Also it’s not superfluous to say that you can use subclass of ARAnchor for AnchorEntity needs:

func session(_ session: ARSession, didUpdate anchors: [ARAnchor]) {

    guard let faceAnchor = anchors.first as? ARFaceAnchor else { return }
    let anchor = AnchorEntity(anchor: faceAnchor)
    anchor.addChild(model)        
    arView.scene.anchors.append(anchor)
}
查看更多
登录 后发表回答