Skip to content

Progress indicators express an unspecified wait time or display the length of a process.

Animation of linear progress indicator beneath "Episodes" top app bar

Contents

Design and API Documentation

Using progress indicators

Before you can use Material sliders, you need to add a dependency to the Material Components for Android library. For more information, go to the Getting started page.

Progress indicators inform users about the status of ongoing processes, such as loading an app, submitting a form, or saving updates. They communicate an app’s state and indicate available actions, such as whether users can navigate away from the current screen.

Note: When displaying progress for a sequence of processes, indicate overall progress rather than the progress of each activity.

Usage

A determinate progress indicator can be added to a layout:

xml
<!-- Linear progress indicator -->
<com.google.android.material.progressindicator.LinearProgressIndicator
    android:layout_width="match_parent"
    android:layout_height="wrap_content" />
<!-- Circular progress indicator -->
<com.google.android.material.progressindicator.CircularProgressIndicator
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" />

An indeterminate progress indicator can be added:

xml
<!-- Linear progress indicator -->
<com.google.android.material.progressindicator.LinearProgressIndicator
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:indeterminate="true" />
<!-- Circular progress indicator -->
<com.google.android.material.progressindicator.CircularProgressIndicator
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:indeterminate="true" />

Animation of determinate and indeterminate linear and circular progress indicators: red
indicators travel along pink tracks

  1. Determinate
  2. Indeterminate

Switching from indeterminate to determinate

Indeterminate progress indicators can smoothly transit to determinate progress indicators by setting the progress programmatically:

kt
int progress = getLoadingProgress()
indicator.setProgressCompat(progress, true)

Note: Once indeterminate progress indicators are switched to the determinate mode (or initialized as determinate), they can be set back to indeterminate mode via calling the setIndeterminate(true) method.

Making progress indicators accessible

Progress indicators inherit accessibility support from the ProgressBar class in the framework. Please consider setting the content descriptor for use with screen readers.

That can be done in XML via the android:contentDescription attribute or programmatically like so:

kt
progressIndicator.contentDescription = contentDescription

Showing/hiding the progress indicator

By default, the progress indicator will be shown or hidden without animations. You can change the animation behaviors via app:showAnimationBehavior (or setShowAnimationBehavior method) and app:hideAnimationBehavior (or setHideAnimationBehavior method).

The modes of behaviors are:

  • none (default) - shows/hides the view immediately when the visibility is being changed via show, hide or setVisibility method.
  • outward - for the linear type, shows the view by expanding from the baseline (or bottom edge) and hides the view by collapsing to the top edge; for the circular type, shows the view by expanding from the inner edge and hides the view by collapsing to the outer edge.
  • inward - for the linear type, shows the view by expanding from the top edge and hides the view by collapsing to the baseline (bottom edge); for the circular type, shows the view by expanding from the outer edge and hides the view by collapsing to the inner edge.
  • escape - for the linear type, escapes in the progression direction; for the circular type, no effect.

When the hide animation behavior is not none, the visibility of the view will be changed after the animation finishes. Please use setVisibilityAfterHide method to set the target visibility as Visibility.INVISIBLE (default) or Visibility.GONE.

Rounded corners

Progress indicators can have rounded corners via app:trackCornerRadius or the setTrackCornerRadius method.

Types

Material Design offers two visually distinct types of progress indicators: 1. linear 2. circular

Only one type should represent each kind of activity in an app. For example, if a refresh action displays a circular indicator on one screen, that same action shouldn’t use a linear indicator elsewhere in the app.

Composite image of linear and circular progress indicator types

Linear progress indicators

Linear progress indicators display progress by animating an indicator along the length of a fixed, visible track. The behavior of the indicator is dependent on whether the progress of a process is known.

Linear progress indicators support both determinate and indeterminate operations.

  • Determinate operations display the indicator increasing in width from 0 to 100% of the track, in sync with the process’s progress.
  • Indeterminate operations display the indicator continually growing and shrinking along the track until the process is complete.

API and source code:

The following example shows a determinate linear progress indicator.

grey track" width="600"/>

In the layout:

xml
<com.google.android.material.progressindicator.LinearProgressIndicator
    android:layout_width="match_parent"
    android:layout_height="wrap_content" />

The following example shows an indeterminate linear progress indicator.

along grey track" width="600"/>

In the layout:

xml
<com.google.android.material.progressindicator.LinearProgressIndicator
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:indeterminate="true" />

Multi-color indeterminate animation type

For linear progress indicator, there are two indeterminate animation types:

  • disjoint - animates as repeated cycles with two disjoint segments in the

travels along track 2x then switches to yellow" width="600"/>

  • contiguous - animates as repeated cycles with three adjacent segments in

    animation: red, yellow, blue indicators move sequentially and cover track" width="600"/>

Note: There is a minimum requirement of 3 indicator colors to use the contiguous animation. Otherwise, an IllegalArgumentException will be thrown.

Circular progress indicators

Circular progress indicators display progress by animating an indicator along an invisible circular track in a clockwise direction. They can be applied directly to a surface, such as a button or card.

Circular progress indicators support both determinate and indeterminate processes.

  • Determinate circular indicators fill the invisible, circular track with color, as the indicator moves from 0 to 360 degrees.
  • Indeterminate circular indicators grow and shrink in size while moving along the invisible track.

API and source code:

The following example shows a determinate circular progress indicator.

circle clockwise from the top" width="600"/>

In the layout:

xml
<com.google.android.material.progressindicator.CircularProgressIndicator
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" />

The following example shows an indeterminate circular progress indicator.

a circle clockwise from the top" width="600"/>

In the layout:

xml
<com.google.android.material.progressindicator.CircularProgressIndicator
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:indeterminate="true" />

Anatomy and key properties

A progress indicator consists of a track and an indicator.

  1. Active indicator
  2. Track
  3. Stop indicator

Common attributes

The following attributes are shared between linear and circular progress indicators:

ElementAttributeRelated method(s)Default value
Track thicknessapp:trackThicknesssetTrackThicknessgetTrackThickness4dp
Indicator colorapp:indicatorColorsetIndicatorColorgetIndicatorColorcolorPrimary
Track colorapp:trackColorsetTrackColorgetTrackColorcolorPrimaryContainer (linear)@android:color/transparent (circular)
Track corner radiusapp:trackCornerRadiussetTrackCornerRadiusgetTrackCornerRadius2dp
Indicator track gap sizeapp:indicatorTrackGapSizesetIndicatorTrackGapSizegetIndicatorTrackGapSize4dp
Show animation behaviorapp:showAnimationBehaviorsetShowAnimationBehaviorgetShowAnimationBehaviornone
Hide animation behaviorapp:hideAnimationBehaviorsetHideAnimationBehaviorgetHideAnimationBehaviornone
Delay (in ms) to showapp:showDelayN/A0
Min delay (in ms) to hideapp:minHideDelayN/A0
Wavelengthapp:wavelengthsetWavelength0
Wavelength in determinate modeapp:wavelengthDeterminatesetWavelengthDeterminategetWavelenthDeterminatewavelength
Wavelength in indeterminate modeapp:wavelengthIndeterminatesetWavelengthIndeterminategetWavelengthIndeterminatewavelength
Wave amplitudeapp:waveAmplitudesetWaveAmplitudegetWaveAmplitude0
Wave speedapp:waveSpeedsetWaveSpeedgetWaveSpeed0

Linear type specific attributes

Linear type progress indicators also have the following attributes:

ElementAttributeRelated method(s)Default value
Indeterminate animation typeapp:indeterminateAnimationTypesetIndeterminateAnimationTypegetIndeterminateAnimationTypedisjoint
Indicator directionapp:indicatorDirectionLinearsetIndicatorDirectiongetIndicatorDirectionleftToRight
Track stop indicator sizeapp:trackStopIndicatorSizesetTrackStopIndicatorSizegetTrackStopIndicatorSize4dp

Circular type specific attributes

Circular type progress indicators also have the following attributes:

ElementAttributeRelated method(s)Default value
Spinner size (outer diameter)app:indicatorSizesetIndicatorSizegetIndicatorSize40dp
Insetapp:indicatorInsetsetIndicatorInsetgetIndicatorInset4dp
Indicator directionapp:indicatorDirectionCircularsetIndicatorDirectiongetIndicatorDirectionclockwise
Indeterminate animation typeapp:indeterminateAnimationTypeCircularsetIndeterminateAnimationTypegetIndeterminateAnimationTypeadvance

Styles

ElementStyle
Default linear
style
Widget.Material3.LinearProgressIndicator
Default circular
style
Widget.Material3.CircularProgressIndicator
Medium circular
style
Widget.Material3.CircularProgressIndicator.Medium
Small circular
style
Widget.Material3.CircularProgressIndicator.Small
Extra small circular
style
Widget.Material3.CircularProgressIndicator.ExtraSmall

Default linear style theme attribute: ?attr/linearProgressIndicatorStyle

Default circular style theme attribute: ?attr/circularProgressIndicatorStyle

See the full list of styles and attributes.

Non-Text Contrast update

In order to comply with the latest accessibility requirements, the LinearProgressIndicator and CircularProgressIndicator have been updated with additional attributes:

  • app:indicatorTrackGapSize: size of the gap between the indicator and the track, 4dp by default.
  • app:trackStopIndicatorSize: size of the stop at the end of the track, 4dp by default. Only applies to the linear determinate configuration.

*.Legacy styles have been added to revert to the previous behavior (not recommended):

  • Widget.Material3.LinearProgressIndicator.Legacy
  • Widget.Material3.CircularProgressIndicator.Legacy

Theming

Progress indicators support Material theming which can customize color and size.

Theming progress indicators

API and source code:

The following example shows a circular progress indicator with Material Theming.

"Circular progress indicator animation: pink circle segment circles center,
then pink circle segment fills
circle"

Implementing progress indicator theming

Use theme attributes and styles in res/values/styles.xml, which applies to all circular progress indicators and affects other components:

xml
<style name="Theme.App" parent="Theme.Material3.*">
    ...
    <item name="colorPrimary">@color/shrine_pink_100</item>
</style>

Use a default type theme attribute, styles and a theme overlay, which applies to all circular progress indicators but does not affect other components:

xml
<style name="Theme.App" parent="Theme.Material3.*">
    ...
    <item name="circularProgressIndicatorStyle">@style/Widget.App.CircularProgressIndicator</item>
</style>

<style name="Widget.App.CircularProgressIndicator" parent="Widget.Material3.CircularProgressIndicator.Legacy">
    <item name="materialThemeOverlay">@style/ThemeOverlay.App.CircularProgressIndicator</item>
    <item name="trackThickness">20dp</item>
</style>

<style name="ThemeOverlay.App.CircularProgressIndicator" parent="">
    <item name="colorPrimary">@color/shrine_pink_100</item>
</style>

Use the style in the layout, which affects only this specific circular progress indicator:

xml
<com.google.android.material.progressindicator.CircularProgressIndicator
    ...
    style="@style/Widget.App.CircularProgressIndicator"  />
has loaded