Glance simplifies handling user interaction via the `Action` classes. Glance's
`Action` classes define the actions a user can take, and you can specify the
operation performed in response to the action. You can apply an `Action` to any
component with the [`GlanceModifier.clickable`](https://developer.android.com/reference/kotlin/androidx/glance/GlanceModifier#(androidx.glance.GlanceModifier).clickable(androidx.glance.action.Action)) method.

App widgets live on a remote process, so the actions are defined at creation
time and the execution happens in the remote process. In native `RemoteViews`,
this is done via `PendingIntents`.

The following actions are described on this page:

- [Launch an activity](https://developer.android.com/develop/ui/compose/glance/user-interaction#launch-activity)
- [Launch a service](https://developer.android.com/develop/ui/compose/glance/user-interaction#launch-service)
- [Send a broadcast event](https://developer.android.com/develop/ui/compose/glance/user-interaction#send-broadcast)
- [Run callback](https://developer.android.com/develop/ui/compose/glance/user-interaction#run-actioncallback)

| **Note:** See [`ActionAppWidget.kt`](https://cs.android.com/androidx/platform/frameworks/support/+/androidx-main:glance/glance-appwidget/integration-tests/demos/src/main/java/androidx/glance/appwidget/demos/ActionAppWidget.kt) for an example of how to handle actions.

## Launch an activity

To launch an activity on user interaction, provide the
[`actionStartActivity`](https://developer.android.com/reference/kotlin/androidx/glance/appwidget/action/package-summary#actionStartActivity(android.content.Intent,%20androidx.glance.action.ActionParameters))(..) function to a `Button` or other composable via the
`GlanceModifier.clickable`(..) modifier.

Provide one of the following in `actionStartActivity`:

- The target activity class
- The [`ComponentName`](https://developer.android.com/reference/android/content/ComponentName)
- An Intent

Glance translates the Action into a `PendingIntent` with the provided target and
parameters. In the following example, the `NavigationActivity` is launched when a
user clicks the button:


```kotlin
@Composable
fun MyContent() {
    // ..
    Button(
        text = "Go Home",
        onClick = actionStartActivity<MyActivity>()
    )
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L186-L193
```

<br />

## Launch a service

Similar to launching an activity, launch a service on user interaction using one
of the [`actionStartService`](https://developer.android.com/reference/kotlin/androidx/glance/appwidget/action/package-summary#actionstartservice) methods.

Provide one of the following in `actionStartService`:

- The target activity class
- The [`ComponentName`](https://developer.android.com/reference/android/content/ComponentName)
- An intent


```kotlin
@Composable
fun MyButton() {
    // ..
    Button(
        text = "Sync",
        onClick = actionStartService<SyncService>(
            isForegroundService = true // define how the service is launched
        )
    )
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L200-L209
```

<br />

## Send a broadcast event

Send a broadcast event on user interaction using one of the
[`actionSendBroadcast`](https://developer.android.com/reference/kotlin/androidx/glance/appwidget/action/package-summary#actionstartbroadcastreceiver) methods:

Provide one of the following in `actionSendBroadcast`:

- String action
- The [`ComponentName`](https://developer.android.com/reference/android/content/ComponentName)
- An intent
- [`BroadcastReceiver`](https://developer.android.com/reference/android/content/BroadcastReceiver) class


```kotlin
@Composable
fun MyButton() {
    // ..
    Button(
        text = "Send",
        onClick = actionSendBroadcast<MyReceiver>()
    )
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L216-L223
```

<br />

## Perform custom actions

Instead of launching a specific target, Glance can use a lambda action or an
`actionRunCallback` to perform an action, such as updating the UI or state on
user interaction.

### Run lambda actions

You can use lambda functions as callbacks to the UI interactions.
| **Note:** Lambda callbacks run in the context of a `WorkManager` worker that is run in a `Service`. Apps that target Android 12 or higher can't start activities from services or broadcast receivers that act as [trampolines](https://developer.android.com/about/versions/12/behavior-changes-12#notification-trampolines). Instead of starting activities from lambdas, use the [`actionStartActivity`](https://developer.android.com/reference/kotlin/androidx/glance/appwidget/action/package-summary#actionStartActivity(android.content.Intent,%20androidx.glance.action.ActionParameters)) callback to start the activity from the `GlanceAppWidget`.

For example, pass the lambda function to the `GlanceModifier.clickable`
modifier:


```kotlin
Text(
    text = "Submit",
    modifier = GlanceModifier.clickable {
        submitData()
    }
)https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L231-L236
```

<br />

Or, pass it to the `onClick` parameter on composables that support it:


```kotlin
Button(
    text = "Submit",
    onClick = {
        submitData()
    }
)https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L243-L248
```

<br />

### Run ActionCallback

Alternatively, use the [`actionRunCallback`](https://developer.android.com/reference/kotlin/androidx/glance/appwidget/action/package-summary#actionruncallback) methods to perform an action on
user interaction. To do this, provide a custom implementation of the
[`ActionCallback`](https://developer.android.com/reference/kotlin/androidx/glance/appwidget/action/ActionCallback):


```kotlin
@Composable
private fun MyContent() {
    // ..
    Image(
        provider = ImageProvider(R.drawable.ic_hourglass_animated),
        modifier = GlanceModifier.clickable(
            onClick = actionRunCallback<RefreshAction>()
        ),
        contentDescription = "Refresh"
    )
}

class RefreshAction : ActionCallback {
    override suspend fun onAction(
        context: Context,
        glanceId: GlanceId,
        parameters: ActionParameters
    ) {
        // TODO implement
    }
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L277-L297
```

<br />

On the user click, the `suspend onAction` method of the provided
`ActionCallback` is called, executing the defined logic (i.e., requesting
refresh data).

To update the widget after the action is performed, create a new instance and
call `update`(..). For more details, see the [Manage GlanceAppWidget state](https://developer.android.com/develop/ui/compose/glance/glance-app-widget#manage-state)
section.


```kotlin
class RefreshAction : ActionCallback {
    override suspend fun onAction(
        context: Context,
        glanceId: GlanceId,
        parameters: ActionParameters
    ) {
        // do some work but offset long-term tasks (e.g a Worker)
        MyAppWidget().update(context, glanceId)
    }
}
https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L255-L265
```

<br />

| **Note:** Glance uses a custom async `BroadcastReceiver` to handle the user click and call the `onRun` method of the provided `ActionCallback`. This allows extra execution time, but certain restrictions apply. Any long or consuming tasks should be offset into, for example, a [`Worker`](https://developer.android.com/topic/libraries/architecture/workmanager).

## Provide parameters to actions

To provide additional information to an action, use the [`ActionParameters`](https://developer.android.com/reference/kotlin/androidx/glance/action/ActionParameters)
API to create a typed key-value pair. For example, to define the clicked
destination:


```kotlin
private val destinationKey = ActionParameters.Key<String>(
    NavigationActivity.KEY_DESTINATION
)

class MyAppWidget : GlanceAppWidget() {

    // ..

    @Composable
    private fun MyContent() {
        // ..
        Button(
            text = "Home",
            onClick = actionStartActivity<NavigationActivity>(
                actionParametersOf(destinationKey to "home")
            )
        )
        Button(
            text = "Work",
            onClick = actionStartActivity<NavigationActivity>(
                actionParametersOf(destinationKey to "work")
            )
        )
    }

    override suspend fun provideGlance(context: Context, id: GlanceId) {
        provideContent { MyContent() }
    }
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L303-L331
```

<br />

Underneath, the parameters are included in the intent used to launch the
activity, allowing the target Activity to retrieve it.


```kotlin
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val destination = intent.extras?.getString(KEY_DESTINATION) ?: return
        // ...
    }
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L338-L343
```

<br />

The parameters are also provided to the `ActionCallback`. Use the defined
`Parameters.Key` to retrieve the value:


```kotlin
class RefreshAction : ActionCallback {

    private val destinationKey = ActionParameters.Key<String>(
        NavigationActivity.KEY_DESTINATION
    )

    override suspend fun onAction(
        context: Context,
        glanceId: GlanceId,
        parameters: ActionParameters
    ) {
        val destination: String = parameters[destinationKey] ?: return
        // ...
    }
}https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/glance/GlanceSnippets.kt#L347-L361
```

<br />