The fundamentals of building a circular draggable Slider in Jetpack Compose

agusioma

Terrence Aluda

Posted on January 5, 2024

The fundamentals of building a circular draggable Slider in Jetpack Compose

If you are reading this, you most likely came from this Instagram post or this GitHub repository. If not, it's fine; you can proceed to learn about the fundamentals of creating a circular draggable Slider. The default slider provided by Jetpack Compose is linear, as shown in the image below.

In this piece, we are going to implement a custom one that will appear as shown below.

Be sure to access the full code for easier reference.

The canvas

The canvas is needed when we want to implement custom elements that are not provided by Jetpack Compose or the built-in Android implementations. We are not going to entirely talk about the canvas here, but only its aspects that will be used, which are:

  • The drawArc method
  • The drawCircle method

The methods' names tend to give us a hint of what they do. To draw the slider, an arc will be needed, and the drawArc method will be utilized.

The thumb indicator, which is a circle, will be drawn using the drawCircle method.

Let's begin dissecting the function, starting with its skeleton implementation.



@OptIn(ExperimentalComposeUiApi::class) 
@Composable 
fun CircularSlider(modifier: Modifier=Modifier,
    padding: Float=50f,
    stroke: Float=20f,
    cap: StrokeCap=StrokeCap.Round,
    touchStroke: Float=50f,
    onChange: ((Float) -> Unit)?=null) {

    //Other code here

    LaunchedEffect(key1=angle) {

        //TODO(Add code here...)
    }

    //TODO(Add code here...)

    Canvas(modifier=modifier.onGloballyPositioned {

            //TODO(Add code here...)
        }

        .pointerInteropFilter {

            //TODO(Add code here...)

        }) {

        drawArc( 
            //TODO(Add code here...)
        ) 

        drawCircle( 
            //TODO(Add code here...)
        ))
    }
}


Enter fullscreen mode Exit fullscreen mode

The //TODO(Add code here...) shows where code snippets will be added to complete the Canvas.

LaunchedEffect

Let's begin with the LaunchedEffect block. It will be used to ensure that the thumb indicator doesn't exceed the bounds of the arc by using an if conditional block. It will also be used to set the percentage traversed by the thumb indicator in relation to the arc's length using the invoke() operator.



LaunchedEffect(key1=angle) {

    if (angle < 0.0f && angle > -90f) {
        angle=0.0f
    }

    else if (angle < 0.0f && angle < -90f) {
        angle=180.0f
    }

    else if (angle >=180f) {
        angle=180f
    }

    appliedAngle=angle 
    onChange?.invoke(angle / 180f)
}


Enter fullscreen mode Exit fullscreen mode

The figure 180 was settled on for this tutorial since we are creating a semicircle which usually covers 180°. So if the thumb's position is at 90°, then the percentage covered will be 50%. appliedAngle will be used in a moment when positioning the thumb's position indicator.

The second TODO will be for creating a gradient background for the arc. You can find the snippet in the full code.

The onGloballyPositioned and the pointerInteropFilter modifiers

onGloballyPositioned is used to position the canvas and its contents. It does so using the LayoutCoordinates' width and height. The center is calculated by halving the width and height. The radius is calculated using either the width or by subtracting half of the stroke's weight and the base padding from the height's half dimensions. The smaller value between the two will be the set radius.



.onGloballyPositioned {
    width = it.size.width
    height = it.size.height
    center = Offset(width / 2f, height / 2f)
    radius = min(width.toFloat(), height.toFloat()) / 2f - padding - stroke / 2f
}


Enter fullscreen mode Exit fullscreen mode

pointerInteropFilter is where the dragging event logic is controlled.



.pointerInteropFilter {
    val x=it.x 
    val y=it.y 
    val offset=Offset(x, y) 

    when (it.action) {

        MotionEvent.ACTION_DOWN -> {
            val d=distance(offset, center) 
            val a=angle(center, offset) 

            if (d >= radius - touchStroke / 2f && d <= radius + touchStroke / 2f) {
                nearTheThumbIndicator=true angle=a
            }

            else {
                nearTheThumbIndicator=false
            }
        }

        MotionEvent.ACTION_MOVE -> {
            if (nearTheThumbIndicator) {
                angle=angle(center, offset)
            }
        }

        MotionEvent.ACTION_UP -> {
            nearTheThumbIndicator=false
        }

        else ->return@pointerInteropFilter false
    }

    return@pointerInteropFilter true
}


Enter fullscreen mode Exit fullscreen mode

The events of interest here are ACTION_DOWN, ACTION_MOVE, and ACTION_UP. ACTION_DOWN is fired when the user's finger first touches the device's screen. ACTION_MOVE when the finger moves on the screen from the point of first contact. ACTION_UP is fired when the finger is lifted off the screen. A when statement is used to help decide which action will be fired.

For all three events, the finger's touch position coordinates(x and y) are captured for use in determining the offset.



val x = it.x
val y = it.y
val offset = Offset(x, y)


Enter fullscreen mode Exit fullscreen mode

The offset will be used to calculate the distance(d) from the thumb and the thumb's indicator. It will also be used in determining the angle(a) where the thumb is on the arc.



val d = distance(offset, center)
val a = angle(center, offset)


Enter fullscreen mode Exit fullscreen mode

You may be wondering why we need the distance. The distance is crucial because we don't want the thumb indicator to move when the user's finger is too far from it, but only when it is near the thumb position indicator. To effect that, the if block under the ACTION_DOWN only allows the thumb indicator to be dragged when:

  • Is equal or at a slightly greater distance than radius minus half the size of the touch stroke(d >= radius - touchStroke / 2f)
  • Is lesser than or equal to the arc's radius length plus half the size of the touch stroke(d <= radius + touchStroke / 2f)

When these conditions are met, the event is registered as near the thumb indicator(nearTheThumbIndicator = true).

If an ACTION_MOVE is registered and the finger is at the required proximity, then the thumb indicator's angle on the arc is updated using a helper method called angle(). The helper method's implementation is shown below:



fun angle(center: Offset, offset: Offset): Float {
    val rad = atan2(center.y - offset.y, center.x - offset.x)
    val deg = Math.toDegrees(rad.toDouble())
    return deg.toFloat()
}


Enter fullscreen mode Exit fullscreen mode

It calculates the degree using the atan2 mathematical function. Since it returns the values in radians, it is converted to degrees.

The proximity distance is also calculated using a helper function whose implementation is shown below.



fun distance(first: Offset, second: Offset): Float {
    return sqrt((first.x - second.x).square() + (first.y - second.y).square())
}


Enter fullscreen mode Exit fullscreen mode

It uses the Euclidean distance formula to compute the distance.

Drawing the arc and circle

The arc starts from -180° to 180° since we are moving in a clockwise direction, and that's how the canvas calibrates the angles.



drawArc(
    brush = gradient,
    startAngle = -180f,
    sweepAngle = 180f,
    topLeft = center - Offset(radius, radius),
    size = Size(radius * 2, radius * 2),
    useCenter = false,
    style = Stroke(
        width = stroke,
        cap = cap
    )
)


Enter fullscreen mode Exit fullscreen mode

For the circle, its center is set using sin and cos. The mathematics behind that can be accessed exclusively from this resource.



drawCircle(
    color = Color.White,
    radius = 30f,
    center = center + Offset(
        radius * cos((-180 + appliedAngle) * PI / 180f).toFloat(),
        radius * sin((-180 + appliedAngle) * PI / 180f).toFloat()
    )
)


Enter fullscreen mode Exit fullscreen mode

And that's it. I hope you got some insights on drawing a custom circular slider. Let me hear your thoughts in the comment section below.

💖 💪 🙅 🚩
agusioma
Terrence Aluda

Posted on January 5, 2024

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related

What was your win this week?
weeklyretro What was your win this week?

November 29, 2024

Where GitOps Meets ClickOps
devops Where GitOps Meets ClickOps

November 29, 2024