Augmented Reality With ARKit: Feature Points and Horizontal Plane Detection
Often, when using augmented reality, you want to place your virtual object on a flat surface such as a table, a desk, or even the ground. In order to do this accurately, it’s important that you are able to detect these objects before you begin. After your plane is detected, the object will use a series of dots to anchor your virtual object to it, and the object will remain there, even as you move your device around.
This tutorial will focus on showing horizontal planes and feature points in ARKit. You’ll need some basic knowledge of Swift, Xcode, and ARKit.
Getting Started
Xcode Version
Before we begin, make sure you have the latest version of Xcode installed on your Mac. This is very important because ARKit will only be available on Xcode 11 or newer. You can check your version by opening Xcode and going to Xcode > About Xcode in the upper toolbar.
If your version of Xcode is older than Xcode 11, you can go to the Mac App Store and update it for free. If you don’t already have Xcode, you can also download and install it for free.
Sample Project
New Project
After you’ve made sure you have the right version of Xcode, you’ll need to make a new Xcode project.
Go ahead and open Xcode and click Create a new Xcode project.
You may be used to making a Single View Application, but for this tutorial, you will need to choose an Augmented Reality App. Then click Next.
Gaming Frameworks
You can name your project anything you like, but I will be naming mine Plane Detection. You will also notice that there is an option at the bottom where you can select from SceneKit, SpriteKit, and Metal.
These are all Apple’s gaming frameworks, and for the purpose of the tutorial, we will be using SceneKit because we’ll be using 3D objects.
Go ahead and select SceneKit if it isn’t already selected. Your screen should look something like this:
Preparing to Test
Connecting an iPhone
Since the Xcode Simulator doesn’t have a camera, you’ll need to plug in your iPhone. Unfortunately, if you don’t have an iPhone, you’ll need to borrow one to be able to follow along with this tutorial (and for any other camera-related apps). If you already have an iPhone connected to Xcode, you can skip ahead to the next step.
A nifty new feature in Xcode 9 is that you can wirelessly debug your app on a device, so let’s take the time to set that up now:
In the top menu bar, choose Window > Devices and Simulators. In the window that appears, make sure that Devices is selected at the top.
Now, plug in your device using a lightning cable. This should make your device appear in the left pane of the Devices and Simulators window. Simply click your device, and check the Connect via Network box.
You will now be able to wirelessly debug on this iPhone for all future apps.
Complete Setup
Now your setup is complete. You should have a working ARKit app, and you can test it on the iPhone that you just connected. In the upper left of Xcode, next to the Run and Stop buttons, select your device from the simulator dropdown. I’ve selected Vardhan’s iPhone, but you need to select your specific device.
Now you’re done creating your starter project, and you should see a virtual spaceship appear in your world once you click Run. Here’s what it should look like:
Great! You’ve now got yourself a working ARKit app. Your goal for this app is to be able to detect a horizontal plane and visualize it with feature points (virtual dots which are placed on scenes in ARKit).
Starter Code
You now have a solid understanding of what feature points and horizontal planes are, and you’re now ready to start programming them into an app. If you want more background on feature points and horizontal planes, I would recommend reading Apple’s Documentation.
Preparing Your Project
If you open your ViewController.swift file, you’ll notice that Apple has some starter code already set up for you (which renders a spaceship). Since a lot of the code is boilerplate code, we’ll keep it, but as for the spaceship, let’s get it out of the way by removing the code which handles its placement. In your ViewController.swift file, within your viewDidLoad()
method, remove the following lines of code:
This code configures debug options using an array of options. You can enable other features by adding them to this array, but for now, we just need to display our feature points. Here’s what it should look like when you run your app:
As you can see, all the detected feature points are visible. If you don’t see them, try moving your iPhone around, and you should see yellow dots appear. Part of the reason why they don’t appear immediately is that ARKit is still detecting the scene.
Configuration
Now, we’re ready for plane detection. If you take a look at your viewWillAppear()
method, you'll see the following two lines of code. Let's take a moment to learn what they mean as they will be relevant for plane detection:
Here, an instance of the ARWorldTrackingConfiguration
class is made. This class is responsible for device positioning. After that, we run the configuration on our current sceneView
session. If you're interested in learning more about this, you can visit Apple's Documentation about it. For now, though, you're ready to continue to the next step.
Now, let’s enable plane detection on our ARWorldTrackingConfiguration
. Below the line which creates the configuration, you'll need to insert the following line:
Excellent! Now, horizontal plane detection is enabled, but we can’t see it because it’s happening under the hood, meaning that the iPhone knows where the plane is, but we can’t see what it thinks.
Plane Detection
Checking for Anchors
Before we can start visualizing our detected planes, we’ll need to find out when and where the planes are being detected. This can be easily done with a delegate method which is provided to us by Apple.
Start off by declaring a method for the ARSCNViewDelegate
, and as you'll see, the ViewController
class already conforms to this delegate in the starter code. Paste the following delegate method into your ViewController
class:
This is called when new ARAnchors
are added with a corresponding node, which, in this case, would be our horizontal planes.
Plane Geometry
For most apps, though, you would want the user to be able to tell where your app thinks it spots horizontal planes, so we’ll be learning how to actually display these planes to the user using the node and the anchor provided in the delegate method we called earlier.
Our first question is: is the ARAnchor
really a plane, or is it something we don't want? We can check this using type-casting and optional binding. Put this line of code into your delegate method:
You may have noticed that the anchor
parameter has the type ARAnchor
. If we want to know whether this anchor is a plane, we can attempt to type-cast it as an ARPlaneAnchor
. If that succeeds, we know that a plane has been detected.
Inside the optional binding statement, let’s add code to create an SCNPlane
with the dimensions of the anchor we received from the delegate method. Write out the following two lines of code:
The first line here creates a two-dimensional SCNPlane
which takes two parameters in its constructor: a width and a height. You'll see that we pass in anchor.extent.x
and anchor.extent.z
, which, as their names suggest, represent the width and the height of the detected planes. You'll notice that we've omitted the y-value of the anchor, and that's because the planes we're detecting are two-dimensional, and the x and z axes run across planes that we're identifying.
The next line of code applies a translucent red hue to these detected planes, and you can see through these planes because the alpha value is set to 0.5. Of course, you can use any colors or appearances you wish, as long as you’ll be able to see them.
Creating the Node
We’re not done yet; we still need to create the node to add to our view. Though our plane is a two-dimensional object, we still need to represent it in 3D with an x, y, and z coordinate. These coordinates will be relative to the parent node of the detected plane anchor and not the sceneView
. Enter the following lines of code:
1. let planeNode = SCNNode(geometry: plane)
2. planeNode.position
= SCNVector3(CGFloat(anchor.center.x), CGFloat(anchor.center.y), CGFloat(anchor.center.z))
3. planeNode.eulerAngles.x
= -.pi
/ 2
4. node.addChildNode(planeNode)
We’re first creating a SCNNode
with the plane geometry we created in the previous step. This is being done by passing in the geometry as a parameter of SCNNode
's constructor.
Next, we’re positioning the plane node directly in the center of the plane anchor using a SCNVector3
representation of x, y, and z coordinates in the three-dimensional plane.
Also, in the next line, you’ll see that the x
value of our eulerAngles
is being set to -π/2
. The property eulerAngles
represents the pitch, yaw, and roll of the SCNNode
. According to Apple's documentation, the x value represents the pitch (or rotation about the x-axis), and we want ours to be flat against the plane (not sticking upright).
Last, the newly created node is added as a child node of the detected plane node to make it visible to the user at the same location.
Conclusion
Great job! You should now be able to see horizontal planes (and, of course, the feature points) when you run your application on your iPhone. If you don’t, make sure you move your phone around slowly for it to scan the surface, and try increasing the alpha of the color we set earlier. Here’s what mine looks like:
You now know how to detect flat surfaces such as tables or desks and display them to the user as horizontal planes. In addition to this, you’ve been able to see how ARKit views the world with feature points.