Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Identify layer features Sample #140

Merged
merged 18 commits into from
Nov 9, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
18 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions identify-layer-features/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
/build
33 changes: 33 additions & 0 deletions identify-layer-features/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
# Identify layer features

Identify features in all layers in a map.

![Image of identify layers](identify-layer-features.png)

## Use case

"Identify layers" operation allows users to tap on a map, returning features at that location across multiple layers. Because some layer types have sublayers, the sample recursively counts results for sublayers within each layer.

## How to use the sample

Tap to identify features. A bottom text banner will show all layers with features under the tapped location, as well as the number of features.

## How it works

1. The tapped position is passed to `MapView.identifyLayers(...)` method.
2. For each `IdentifyLayerResult` in the results, features are counted.
* Note: there is one identify result per layer with matching features; if the feature count is 0, that means a sublayer contains the matching features.

## Relevant API

* IdentifyLayerResult
* IdentifyLayerResult.sublayerResults
* LayerContent

## Additional information

The GeoView supports two methods of identify: `identifyLayer`, which identifies features within a specific layer and `identifyLayers`, which identifies features for all layers in the current view.

## Tags

identify, recursion, recursive, sublayers
32 changes: 32 additions & 0 deletions identify-layer-features/README.metadata.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
{
"category": "Search and Query",
"description": "Identify features in all layers in a map.",
"formal_name": "IdentifyLayerFeatures",
"ignore": false,
"images": [
"identify-layer-features.png"
],
"keywords": [
"identify",
"recursion",
"recursive",
"sublayers",
"IdentifyLayerResult",
"IdentifyLayerResult.sublayerResults",
"LayerContent"
],
"language": "kotlin",
"redirect_from": "",
"relevant_apis": [
"IdentifyLayerResult",
"IdentifyLayerResult.sublayerResults",
"LayerContent"
],
"snippets": [
"src/main/java/com/esri/arcgismaps/sample/identifylayerfeatures/MainActivity.kt",
"src/main/java/com/esri/arcgismaps/sample/identifylayerfeatures/components/ComposeMapView.kt",
"src/main/java/com/esri/arcgismaps/sample/identifylayerfeatures/components/MapViewModel.kt",
"src/main/java/com/esri/arcgismaps/sample/identifylayerfeatures/screens/MainScreen.kt"
],
"title": "Identify layer features"
}
48 changes: 48 additions & 0 deletions identify-layer-features/build.gradle
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
apply plugin: 'com.android.application'
apply plugin: 'org.jetbrains.kotlin.android'

android {
compileSdkVersion rootProject.ext.compileSdkVersion

defaultConfig {
applicationId "com.esri.arcgismaps.sample.identifylayerfeatures"
minSdkVersion rootProject.ext.minSdkVersion
targetSdkVersion rootProject.ext.targetSdkVersion
versionCode rootProject.ext.versionCode
versionName rootProject.ext.versionName
buildConfigField("String", "API_KEY", API_KEY)
}

buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}

buildFeatures {
compose = true
buildConfig = true
}
composeOptions {
kotlinCompilerExtensionVersion = "$kotlinCompilerExt"
}

namespace 'com.esri.arcgismaps.sample.identifylayerfeatures'
}

dependencies {
// lib dependencies from rootProject build.gradle
implementation "androidx.core:core-ktx:$ktxAndroidCore"
implementation "androidx.lifecycle:lifecycle-runtime-ktx:$ktxLifecycle"
implementation "androidx.lifecycle:lifecycle-viewmodel-compose:$ktxLifecycle"
implementation "androidx.activity:activity-compose:$composeActivityVersion"
// Jetpack Compose Bill of Materials
implementation platform("androidx.compose:compose-bom:$composeBOM")
// Jetpack Compose dependencies
implementation "androidx.compose.ui:ui"
implementation "androidx.compose.material3:material3"
implementation "androidx.compose.ui:ui-tooling"
implementation "androidx.compose.ui:ui-tooling-preview"
implementation project(path: ':samples-lib')
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
21 changes: 21 additions & 0 deletions identify-layer-features/proguard-rules.pro
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
# Add project specific ProGuard rules here.
# You can control the set of applied configuration files using the
# proguardFiles setting in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html

# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}

# Uncomment this to preserve the line number information for
# debugging stack traces.
#-keepattributes SourceFile,LineNumberTable

# If you keep the line number information, uncomment this to
# hide the original source file name.
#-renamesourcefileattribute SourceFile
24 changes: 24 additions & 0 deletions identify-layer-features/src/main/AndroidManifest.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android">

<uses-permission android:name="android.permission.INTERNET" />

<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity
android:exported="true"
android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>

</manifest>
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
/* Copyright 2023 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
*
* http://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.
*
*/

package com.esri.arcgismaps.sample.identifylayerfeatures

import android.os.Bundle
import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Surface
import androidx.compose.runtime.Composable
import com.arcgismaps.ApiKey
import com.arcgismaps.ArcGISEnvironment
import com.esri.arcgismaps.sample.sampleslib.theme.SampleAppTheme
import com.esri.arcgismaps.sample.identifylayerfeatures.screens.MainScreen

class MainActivity : ComponentActivity() {

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
// authentication with an API key or named user is
// required to access basemaps and other location services
ArcGISEnvironment.apiKey = ApiKey.create(BuildConfig.API_KEY)

setContent {
SampleAppTheme {
IdentifyLayerFeaturesApp()
}
}
}

@Composable
private fun IdentifyLayerFeaturesApp() {
Surface(
color = MaterialTheme.colorScheme.background
) {
MainScreen(
sampleName = getString(R.string.app_name),
application = application
)
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
/* Copyright 2023 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
*
* http://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.
*
*/

package com.esri.arcgismaps.sample.identifylayerfeatures.components

import androidx.compose.runtime.Composable
import androidx.compose.runtime.DisposableEffect
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.collectAsState
import androidx.compose.runtime.getValue
import androidx.compose.ui.Modifier
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.platform.LocalLifecycleOwner
import androidx.compose.ui.viewinterop.AndroidView
import androidx.lifecycle.LifecycleOwner
import com.arcgismaps.mapping.view.MapView

/**
* Wraps the MapView in a Composable function.
*/
@Composable
fun ComposeMapView(
modifier: Modifier = Modifier,
mapViewModel: MapViewModel
) {
// get an instance of the current lifecycle owner
val lifecycleOwner = LocalLifecycleOwner.current
// collect the latest state of the MapViewState
val mapViewState by mapViewModel.mapViewState.collectAsState()
// create and add MapView to the activity lifecycle
val mapView = createMapViewInstance(lifecycleOwner)

// wrap the MapView as an AndroidView
AndroidView(
modifier = modifier,
factory = { mapView },
// recomposes the MapView on changes in the MapViewState
update = { mapView ->
mapView.apply {
map = mapViewState.arcGISMap
setViewpoint(mapViewState.viewpoint)
}
}
)

// launch coroutine functions in the composition's CoroutineContext
LaunchedEffect(Unit) {
mapView.onSingleTapConfirmed.collect {
// call identifyLayers when a tap event occurs
val identifyResult = mapView.identifyLayers(
screenCoordinate = it.screenCoordinate,
tolerance = 12.0,
returnPopupsOnly = false,
maximumResults = 10
)
mapViewModel.handleIdentifyResult(identifyResult)
}
}
}

/**
* Create the MapView instance and add it to the Activity lifecycle
*/
@Composable
fun createMapViewInstance(lifecycleOwner: LifecycleOwner): MapView {
// create the MapView
val mapView = MapView(LocalContext.current)
// add the side effects for MapView composition
DisposableEffect(lifecycleOwner) {
lifecycleOwner.lifecycle.addObserver(mapView)
onDispose {
lifecycleOwner.lifecycle.removeObserver(mapView)
}
}
return mapView
}
Loading
Loading