Using Maps and Location Data in Your SwiftUI (+Realm) App
Rate this tutorial
Embedding Apple Maps and location functionality in SwiftUI apps used to be a bit of a pain. It required writing your own SwiftUI wrapper around UIKit code—see these examples from the O-FISH app:
If you only need to support iOS14 and later, then you can forget most of that messy code 😊. If you need to support iOS13—sorry, you need to go the O-FISH route!
This article shows you how to embed Apple Maps into your app views using Mapkit's Map view. We'll then look at how you can fetch the user's current location—with their permission, of course!
Finally, we'll see how to store the location data in Realm in a format that lets MongoDB Atlas Device Sync it to MongoDB Atlas. Once in Atlas, you can add a geospatial index and use MongoDB Charts to plot the data on a map—we'll look at that too.
Most of the code snippets have been extracted from the RChat app. That app is a good place to see maps and location data in action. Building a Mobile Chat App Using Realm – The New and Easier Way is a good place to learn more about the RChat app—including how to enable MongoDB Atlas Device Sync.
- Realm-Cocoa 10.8.0+ (may work with some 10.7.X versions)
- iOS 14.5+ (Mapkit was introduced in iOS 14.0 and so most features should work with earlier iOS 14.X versions)
To begin, let's create a simple view that displays a map, the coordinates of the center of that map, and the zoom level:

With Mapkit and SwiftUI, this only takes a few lines of code:
Note that
showsUserLocation
won't work unless the user has already given the app permission to use their location—we'll get to that.region
is initialized to a starting location, but it's updated by the Map
view as the user scrolls and zooms in and out.Pins can be added to a map in the form of "annotations." Let's start with a single pin:

Annotations are provided as an array of structs where each instance must contain the coordinates of the pin. The struct must also conform to the Identifiable protocol:
We can now create an array of
MyAnnotationItem
structs:We then pass
annotationItems
to the MapView
and indicate that we want a MapMarker
at the contained coordinates:That gives us the result we wanted.
What if we want multiple pins? Not a problem. Just add more
MyAnnotationItem
instances to the array.All of the pins will be the same default color. But, what if we want different colored pins? It's simple to extend our code to produce this:

Firstly, we need to extend
MyAnnotationItem
to include an optional color
and a tint
that returns color
if it's been defined and "red" if not:In our sample data, we can now choose to provide a color for each annotation:
The
MapView
can then use the tint
:If you get bored of pins, you can use
MapAnnotation
to use any view you like for your annotations:This is the result:

You could also include the name of the system image to use with each annotation.
Apple is pretty vocal about respecting the privacy of their users, and so it shouldn't be a shock that your app will have to request permission before being able to access a user's location.
The first step is to add a key-value pair to your Xcode project to indicate that the app may request permission to access the user's location, and what text should be displayed in the alert. You can add the pair to the "Info.plist" file:

Once that setting has been added, the user should see an alert the first time that the app attempts to access their current location:

While Mapkit has made maps simple and native in SwiftUI, the same can't be said for location data.
You need to create a SwiftUI wrapper for Apple's Core Location functionality. There's not a lot of value in explaining this boilerplate code—just copy this code from RChat's LocationHelper.swift file, and paste it into your app:
Once added, you can access the user's location with this simple call:
Realm doesn't have a native type for a geographic location, and so it's up to us how we choose to store it in a Realm Object. That is, unless we want to synchronize the data to MongoDB Atlas using Device Sync, and go on to use MongoDB's geospatial functionality.
To make the best use of the location data in Atlas, we need to add a geospatial index to the field (which we’ll see how to do soon.) That means storing the location in a supported format. Not all options will work with Atlas Device Sync (e.g., it's not guaranteed that attributes will appear in the same order in your Realm Object and the synced Atlas document). The most robust approach is to use an array where the first element is longitude and the second is latitude:
The RChat app gives users the option to include their location in a chat message—this means that we need to include the location in the ChatMessage Object:
The
location
array that's passed to that initializer is formed like this:The easiest way to create your backend MongoDB Atlas Application Services schema is to enable Development Mode—that way, the schema is automatically generated from your Swift Realm Objects.
This is a document that's been created from a synced Realm
ChatMessage
object:
Now that you have location data stored in Atlas, it would be nice to be able to work with it—e.g., running geospatial queries. To enable this, you need to add a geospatial index to the
location
field.From the Atlas UI, select the "Indexes" tab for your collection and click "CREATE INDEX":

You should then configure a
2dsphere
index:
Most chat messages won't include the user's location and so I set the
sparse
option for efficiency.Note that you'll get an error message if your ChatMessage collection contains any documents where the value in the location attribute isn't in a valid geospatial format.
Atlas will then build the index. This will be very quick, unless you already have a huge number of documents containing the location field. Once complete, you can move onto the next section.
MongoDB Charts is a simple way to visualize MongoDB data. You can access it through the same UI as Application Services and Atlas. Just click on the "Charts" button:

The first step is to click the "Add Data Source" button:

Select your Atlas cluster:

Select the
RChat.ChatMessage
collection:
Click “Finish.” You’ll be taken to the default Dashboards view, which is empty for now. Click "Add Dashboard":

In your new dashboard, click "ADD CHART":

Configure your chart as shown here by:
- Setting the chart type to "Geospatial" and the sub-type to "Scatter."
- Dragging the "location" attribute to the coordinates box.
- Dragging the "author" field to the "Color" box.

Once you've created your chart, you can embed it in web apps, etc. That's beyond the scope of this article, but check out the MongoDB Charts docs if you're interested.
SwiftUI makes it easy to embed Apple Maps in your SwiftUI apps. As with most Apple frameworks, there are extra maps features available if you break out from SwiftUI, but I'd suggest that the simplicity of working with SwiftUI is enough incentive for you to avoid that unless you have a compelling reason.
Accessing location information from within SwiftUI still feels a bit of a hack, but in reality, you cut and paste the helper code once, and then you're good to go.
By storing the location as a
[longitude, latitude]
array (List
) in your Realm database, it's simple to sync it with MongoDB Atlas. Once in Atlas, you have the full power of MongoDB's geospatial functionality to work your location data.If you have questions, please head to our developer community website where the MongoDB engineers and the MongoDB community will help you build your next big idea with MongoDB.