Learn how to execute a SQL query to return features
A feature layer
In this tutorial, you’ll write code to perform SQL queries that return a subset of features
Prerequisites
Before starting this tutorial:
-
You need an ArcGIS Location Platform or ArcGIS Online account.
-
Your system meets the system requirements.
Develop or Download
You have two options for completing this tutorial:
Option 1: Develop the code
To start the tutorial, complete the Display a map tutorial. This creates a map to display the Santa Monica Mountains in California using the topographic basemap from the ArcGIS Basemap Styles service
Open an Xcode project
- Open the
.xcodeprojproject you created by completing the Display a map tutorial. - Continue with the following instructions to execute a SQL query to return features from a feature layer based on spatial and attribute criteria.
Update the map
-
In Xcode, in the Project Navigator, click ContentView.swift.
-
In the editor, modify the
mapvariable’s initial viewpoint.ContentView.swiftlet map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)return map}() -
Create a private extension named
Modelwith a class of typeObservableObject. Move the map variable there.ContentView.swiftprivate extension ContentView {class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)return map}()}} -
Create a
ServiceFeatureTablefrom the parcels service URL, then use it to make aFeatureLayer. Then, add the parcels feature layer to the map as an operational layer.ContentView.swiftclass Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()} -
To access the model’s map in the
ContentViewstruct, create a private variable of typeModelwith a@StateObjectproperty wrapper and modify theMapViewto display the model’s map.ContentView.swiftstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?var body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}}}
Make an enumeration for predefined where clauses
To allow the user to query the parcels feature layer, create an enum that contains a set of predefined where clauses.
-
In the
ContentViewextension, make anenumtype calledWhereClausethat isCaseIterableandEquatable. Add a switch statement and a case option for each where clause.ContentView.swiftprivate extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()}}
Create a function to query the parcels layer and select the result
Create a new function that queries the parcels FeatureTable with the chosen where clause and current extent. Clear any currently selected features and create a new QueryParameters object. Set the selectedWhereClause property to the option selected by the user and geoViewExtent to the current visible area. Asynchronously, query the feature table using the QueryParameters and select the features returned in the FeatureQueryResult.
-
Create a private variable of type
Envelopewith the@Stateproperty wrapper namedgeoViewExtent. This variable keeps track of the current, visible extent.ContentView.swift18 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?var body: some View {MapView(map: model.map)}}58 collapsed linesprivate extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}} -
To update the
geoViewExtentvariable when the visible area is changed, add theonVisibleAreaChanged(perform:)task modifer to the map view. When the visible area has changed, updategeoViewExtentto the new area’sextent.ContentView.swift18 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?var body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}}}58 collapsed linesprivate extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}} -
Create a new method called
queryFeatureLayer(selectedWhereClause:extent:)in theModel. Specify that it accepts parameters of typeWhereClauseandEnvelope.ContentView.swift54 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?var body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}}}private extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}2 collapsed lines}
Add UI for selecting a predefined where clause
To allow the user to choose a where clause that queries the parcels dataset, add a Picker that presents the list of predefined where clauses.
-
In the
ContentView, create a@Statevariable namedselectedWhereClauseof typeWhereClause. Assign.governmentas the default enum. The@StatevariableselectedWhereClauseallows the model’squeryFeatureLayer(selectedWhereClause:extent:)method to track which where clause the user has selected.ContentView.swift18 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?@State private var selectedWhereClause: WhereClause = .governmentvar body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}}}58 collapsed linesprivate extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}} -
In the body, add a
Toolbarand aToolbarItemGroupwith.bottomBarplacement to theMapView.ContentView.swift26 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?@State private var selectedWhereClause: WhereClause = .governmentvar body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}.toolbar {ToolbarItemGroup(placement: .bottomBar) {}}}60 collapsed lines}private extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}} -
Add a
Pickerto the toolbar item group and label it, “Choose an SQLwhereclause”. Set the selection to$selectedWhereClause. Iterating through.allCasesof theWhereClauseto populate the Picker with all theWhereClauseenums.ContentView.swift26 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?@State private var selectedWhereClause: WhereClause = .governmentvar body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}.toolbar {ToolbarItemGroup(placement: .bottomBar) {Spacer()Picker("Choose an SQL `where` clause", selection: $selectedWhereClause) {ForEach(WhereClause.allCases, id: \.self) { clause inText(clause.expression)}}Spacer()}}}60 collapsed lines}private extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}} -
Lastly, add a
.taskmodifier that calls the model’squeryFeatureLayer(selectedWhereClause:extent:)function using theselectedWhereClauseproperty.ContentView.swift26 collapsed lines// Copyright 2024 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISstruct ContentView: View {@StateObject private var model = Model()@State private var geoViewExtent: Envelope?@State private var selectedWhereClause: WhereClause = .governmentvar body: some View {MapView(map: model.map).onVisibleAreaChanged { newVisibleArea ingeoViewExtent = newVisibleArea.extent}.toolbar {ToolbarItemGroup(placement: .bottomBar) {Spacer()Picker("Choose an SQL `where` clause", selection: $selectedWhereClause) {ForEach(WhereClause.allCases, id: \.self) { clause inText(clause.expression)}.task(id: selectedWhereClause) {guard let geoViewExtent = geoViewExtent else { return }await model.queryFeatureLayer(with: selectedWhereClause, in: geoViewExtent)}}Spacer()}}}60 collapsed lines}private extension ContentView {enum WhereClause: CaseIterable, Equatable {case government, residential, farm, taxRate10853, taxRate10860, landValueGreaterThan, landValueLessThanvar expression: String {switch self {case .government: return "UseType = 'Government'"case .residential: return "UseType = 'Residential'"case .farm: return "UseType = 'Irrigated Farm'"case .taxRate10853: return "TaxRateArea = 10853"case .taxRate10860: return "TaxRateArea = 10860"case .landValueGreaterThan: return "Roll_LandValue > 1000000"case .landValueLessThan: return "Roll_LandValue < 1000000"}}}class Model: ObservableObject {let map: Map = {let map = Map(basemapStyle: .arcGISTopographic)map.initialViewpoint = Viewpoint(latitude: 34.03000, longitude: -118.80500, scale: 6e4)let parcelsURL = URL(string: "https://services3.arcgis.com/GVgbJbqm8hXASVYi/arcgis/rest/services/LA_County_Parcels/FeatureServer/0")!let featureTable = ServiceFeatureTable(url: parcelsURL)let featureLayer = FeatureLayer(featureTable: featureTable)map.addOperationalLayer(featureLayer)return map}()func queryFeatureLayer(with selectedWhereClause: WhereClause, in extent: Envelope) async {guardlet parcelsFeatureLayer = map.operationalLayers.first as? FeatureLayer,let featureTable = parcelsFeatureLayer.featureTableelse { return }parcelsFeatureLayer.clearSelection()let parameters = QueryParameters()parameters.whereClause = selectedWhereClause.expressionparameters.geometry = extentdo {let result = try await featureTable.queryFeatures(using: parameters)let features = result.features()parcelsFeatureLayer.selectFeatures(features)} catch {print(error)}}}}
Run the solution
Press Command + R to run the app.
If you are using the Xcode simulator your system must meet these minimum requirements: macOS 14 (Sonoma), Xcode 16, iOS 18. If you are using a physical device, then refer to the system requirements.
The app loads with the map centered on the Santa Monica Mountains in California with the parcels feature layer displayed. Choose an attribute expression and to display parcels that meet the selected criteria.
Alternatively, you can download the tutorial solution, as follows.
Option 2: Download the solution
-
Click the
Download solutionlink under Solution and unzip the file to a location on your machine. -
Open the
.xcodeprojfile in Xcode.
Since the downloaded solution does not contain authentication credentials, you must first set up authentication to create credentials, and then add the developer credentials to the solution.
Set up authentication
To access the secure ArcGIS location services
You can implement API key authentication or user authentication in this tutorial. Compare the differences below:
API key authentication
- Users are not required to sign in.
- Requires creating an API key credential
API key credentials are an item that contains the parameters used to create and manage long-lived access tokens for API key authentication. They are a type of developer credential. with the correct privileges. - API keys
An API key is a long-lived access token created using API key credentials. They are valid for up to one year and are typically embedded directly into client applications. are long-lived access tokens. - Service usage is billed to the API key owner/developer.
- Simplest authentication method to implement.
- Recommended approach for new ArcGIS developers.
Learn more in API key authentication.
User authentication
- Users are required to sign in with an ArcGIS account
An ArcGIS account is an identity with a user type and set of privileges that can access specific ArcGIS products, tools, APIs, services, and resources. The main account types that can be used for development are an ArcGIS Location Platform account, ArcGIS Online account, and ArcGIS Enterprise account. ArcGIS Location Platform and ArcGIS Online accounts are also associated with a subscription. . - User accounts must have privilege
Privileges are a set of permissions assigned to ArcGIS accounts, developer credentials, and applications that grant access to secure resources and functionality in ArcGIS. to access the ArcGIS servicesA service, also known as an ArcGIS service, is software that supports an ArcGIS REST API and provides geospatial functionality or data. A service can be hosted by Esri or in ArcGIS Enterprise. used in application. - Requires creating OAuth credentials
OAuth credentials are an item that contains parameters required to implement user authentication or app authentication, including a .client_id,client_secret, and redirect URIs. They are a type of developer credential. - Application uses a redirect URL and client ID.
- Service usage is billed to the organization of the user signed into the application.
Learn more in User authentication.
To complete this tutorial, click on the tab in the switcher below for your authentication type of choice, either API key authentication or User authentication.
Create a new API key access token
-
Complete the Create an API key tutorial and create an API key with the following privilege(s)
Privileges are a set of permissions assigned to ArcGIS accounts, developer credentials, and applications that grant access to secure resources and functionality in ArcGIS. :- Privileges
- Location services > Basemaps
- Privileges
-
Copy and paste the API key access token into a safe location. It will be used in a later step.
Create new OAuth credentials to access the secure resources used in this tutorial.
-
Complete the Create OAuth credentials for user authentication tutorial to obtain a Client ID and Redirect URL.
A
Client IDuniquely identifies your app on the authenticating server. If the server cannot find an app with the provided Client ID, it will not proceed with authentication.The
Redirect URL(also referred to as a callback url) is used to identify a response from the authenticating server when the system returns control back to your app after an OAuth login. Since it does not necessarily represent a valid endpoint that a user could navigate to, the redirect URL can use a custom scheme, such asmy-app://auth. It is important to make sure the redirect URL used in your app’s code matches a redirect URL configured on the authenticating server. -
Copy and paste the Client ID and Redirect URL into a safe location. They will be used in a later step.
All users that access this application need account privileges
Set developer credentials in the solution
To allow your app users to access ArcGIS location services
Pass your API Key access token to the ArcGISEnvironment.
-
In the Project Navigator, click MainApp.swift.
-
Set the
AuthenticationModeto.apiKey.MainApp.swift// Change the `AuthenticationMode` to `.apiKey` if your application uses API key authentication.private var authenticationMode: AuthenticationMode { .apiKey } -
Set the
apiKeyproperty with your API key access token.MainApp.swift31 collapsed lines// Copyright 2022 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISimport ArcGISToolkit@mainstruct MainApp: App {// The authentication mode.private enum AuthenticationMode {case apiKeycase user}// Change the `AuthenticationMode` to `.apiKey` if your application uses API key authentication.private var authenticationMode: AuthenticationMode { .apiKey }// Please enter an API key access token if your application uses API key authentication.private let apiKey = APIKey("<#YOUR-ACCESS-TOKEN#>")43 collapsed lines// Setup an `Authenticator` with OAuth configuration if your application uses OAuth credentials.@ObservedObject var authenticator = Authenticator(oAuthUserConfigurations: [OAuthUserConfiguration(// Please enter OAuth credentials for user authentication.portalURL: URL(string: "<#YOUR-PORTAL-URL#>")!,clientID: "<#YOUR-CLIENT-ID#>",redirectURL: URL(string: "<#YOUR-REDIRECT-URL#>")!)])func setAuthentication() {switch authenticationMode {case .apiKey:ArcGISEnvironment.apiKey = apiKeycase .user:ArcGISEnvironment.authenticationManager.arcGISAuthenticationChallengeHandler = authenticator}}init() {setAuthentication()}var body: some SwiftUI.Scene {WindowGroup {ContentView().authenticator(authenticator).ignoresSafeArea()}}}
Best Practice: The access token is stored directly in the code as a convenience for this tutorial. Do not store credentials directly in source code in a production environment.
Use the Authenticator toolkit component to manage your OAuth credentialsclient_id, client_secret, and redirect URIs. They are a type of developer credential. ArcGISEnvironment.
-
In the Project Navigator, click MainApp.swift.
-
Set the
AuthenticationModeto.user.MainApp.swift// Change the `AuthenticationMode` to `.user` if your application uses OAuth credentials.private var authenticationMode: AuthenticationMode { .user } -
Set your
portalURL,clientIDandredirectURLvalues.MainApp.swift36 collapsed lines// Copyright 2022 Esri//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at//// https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.import SwiftUIimport ArcGISimport ArcGISToolkit@mainstruct MainApp: App {// The authentication mode.private enum AuthenticationMode {case apiKeycase user}// Change the `AuthenticationMode` to `.user` if your application uses OAuth credentials.private var authenticationMode: AuthenticationMode { .apiKey }// Please enter an API key access token if your application uses API key authentication.private let apiKey = APIKey("<#YOUR-ACCESS-TOKEN#>")// Setup an `Authenticator` with OAuth configuration if your application uses OAuth credentials.@ObservedObject var authenticator = Authenticator(oAuthUserConfigurations: [OAuthUserConfiguration(// Please enter OAuth credentials for user authentication.portalURL: URL(string: "<#YOUR-PORTAL-URL#>")!,clientID: "<#YOUR-CLIENT-ID#>",redirectURL: URL(string: "<#YOUR-REDIRECT-URL#>")!)])28 collapsed linesfunc setAuthentication() {switch authenticationMode {case .apiKey:ArcGISEnvironment.apiKey = apiKeycase .user:ArcGISEnvironment.authenticationManager.arcGISAuthenticationChallengeHandler = authenticator}}init() {setAuthentication()}var body: some SwiftUI.Scene {WindowGroup {ContentView().authenticator(authenticator).ignoresSafeArea()}}}
Best Practice: The OAuth credentials are stored directly in the code as a convenience for this tutorial. Do not store credentials directly in source code in a production environment.
Run the solution
Press Command + R to run the app.
If you are using the Xcode simulator your system must meet these minimum requirements: macOS 14 (Sonoma), Xcode 16, iOS 18. If you are using a physical device, then refer to the system requirements.
The app loads with the map centered on the Santa Monica Mountains in California with the parcels feature layer displayed. Choose an attribute expression and to display parcels that meet the selected criteria.
What’s next?
Learn how to use additional API features, ArcGIS location services, and ArcGIS tools in these tutorials: