- iOS 12.0+
- Xcode 11.3+
This sample app demonstrates a simple shared AR experience for two or more iOS 12 devices. Before exploring the code, try building and running the app to familiarize yourself with the user experience it demonstrates:
Run the app on one device. You can look around the local environment, and tap to place a virtual 3D character on real-world surfaces. (Tap again to place multiple copies of the character.)
Run the app on a second device. On both device screens, a message indicates that they have automatically joined a shared session.
Tap the Send World Map button on one device. Make sure the other device is in an area that the first device visited before sending the map, or has a similar view of the surrounding environment.
The other device displays a message indicating that it has received the map and is attempting to use it. When that process succeeds, both devices show virtual content at the same real-world positions, and tapping on either device places virtual content visible to both.
Follow the steps below to see how this app uses the
ARWorld class to save and restore ARKit’s spatial mapping state, and the MultipeerConnectivity framework to send world maps between nearby devices.
Requires Xcode 10.0, iOS 12.0 and two or more iOS devices with A9 or later processors.
Run the AR Session and Place AR Content
This app extends the basic workflow for building an ARKit app. (For details, see Tracking and Visualizing Planes.) It defines an
ARWorld with plane detection enabled, then runs that configuration in the
ARSession attached to the
ARSCNView that displays the AR experience.
UITap detects a tap on the screen, the
handle method uses ARKit hit-testing to find a 3D point on a real-world surface, then places an
ARAnchor marking that position. When ARKit calls the delegate method
renderer(_:, the app loads a 3D model for
ARSCNView to display at the anchor’s position.
Connect to Peer Devices
Multipeer class provides a simple abstraction around the MultipeerConnectivity features this app uses. After the main view controller creates a
Multipeer instance (at app launch), it starts running an
MCNearby to broadcast the device’s ability to join multipeer sessions and an
MCNearby to find other devices:
MCNearby finds another device, it calls the
browser(_: delegate method. To invite that other device to a shared session, call the browser’s
When the other device receives that invitation,
MCNearby calls the
advertiser(_: delegate method. To accept the invitation, call the provided
In a multipeer session, all participants are by definition equal peers; there is no explicit separation of devices into host (or server or master) and guest (or client or slave) roles. However, you may wish to define such roles for your own AR experience. For example, a multiplayer game design might require a server role to arbitrate gameplay. If you need to separate peers by role, you can choose a way to do so that fits the design of your app. For example:
Have the user choose whether to act as a host or guest before starting a session. The host uses
MCNearbyto broadcast availability, and guests use
MCNearbyto find a host to join.
Join a session as peers, then negotiate between peers to nominate a master. (This approach can be helpful for designs that need a master role but also allow peers to join or leave at any time.)
Capture and Send the AR World Map
ARWorld object contains a snapshot of all the spatial mapping information that ARKit uses to locate the user’s device in real-world space. Reliably sharing a map to another device requires two key steps: finding a good time to capture a map, and capturing and sending it.
ARKit provides a
world value that indicates whether it’s currently a good time to capture a world map (or if it’s better to wait until ARKit has mapped more of the local environment). This app uses that value to provide visual feedback on its Send World Map button:
When the user taps the Send World Map button, the app calls
get to capture the map from the running ARSession, then serializes it to a
Data object with
NSKeyed and sends it to other devices in the multipeer session:
Receive and Relocalize to the Shared Map
When a device receives data sent by another participant in the multipeer session, the
session(_: delegate method provides that data. To make use of it, the app uses
NSKeyed to deserialize an
ARWorld object, then creates and runs a new
ARWorld using that map as the
ARKit then attempts to relocalize to the new world map—that is, to reconcile the received spatial-mapping information with what it senses of the local environment. For best results:
Thoroughly scan the local environment on the sending device before sharing a world map.
Place the receiving device next to the sending device, so that both see the same view of the environment.
Share AR Content and User Actions
Sharing the world map also shares all existing anchors. In this app, this means that as soon as a receiving device relocalizes to the world map, it shows all the 3D characters that were placed by the sending device before it captured and sent a world map. However, recording and transmitting a world map and relocalizing to a world map are time-consuming, bandwidth-intensive operations, so you should take those steps only once, when a new device joins a session.
To create an ongoing shared AR experience, where each user’s actions affect the AR scene visible to other users, after each device relocalizes to the same world map you should share only the information needed to recreate each user action. For example, in this app the user can tap to place a virtual 3D character in the scene. That character is static, so all that is needed to place the character on another participating device is the character’s position and orientation in world space.
This app communicates virtual character positions by sharing
ARAnchor objects between peers. When one user taps in the scene, the app creates an anchor and adds it to the local
ARSession, then serializes that
Data and sends it to other devices in the multipeer session:
When other peers receive data from the multipeer session, they test for whether that data contains an archived
ARAnchor; if so, they decode it and add it to their session:
This is just one strategy for adding dynamic features to a shared AR experience—many other strategies are possible. Choose one that fits the user interaction, rendering, and networking requirements of your app. For example, a game where users throw projectiles in the AR world space might define custom data types with attributes like initial position and velocity, then use Swift’s
Codable protocols to serialize that information to a binary representation for sending over the network.