valentinilk/compose-shimmer

Shimmer effect is not clipped for rounded composables like FABs

WebTiger89 opened this issue · 6 comments

When applying shimmer modifier on extended FAB which has rounded corners, the shimmer effect is not clipped accordingly.

Any ideas how I can overcome this issue?

The issue is that any modifier we add to the ExtendedFloatingActionButton will be applied before the actual shadow and clipping is applied.
This will cause the shimmer-Modifier to draw the effect on the shadow and therefore background as well.

If you don't care about the shadow too much, you can simply clip the shimmering. The shape I'm using here is the default parameter of the FAB.

ExtendedFloatingActionButton(
    modifier = Modifier
        .clip(MaterialTheme.shapes.small.copy(CornerSize(percent = 50)))
        .shimmer(),
    ...

If you want to keep the shadow, you have to apply it yourself:

val interactionSource = remember { MutableInteractionSource() }
val elevation = FloatingActionButtonDefaults.elevation()
ExtendedFloatingActionButton(
    modifier = Modifier
        .shadow(
            elevation = elevation.elevation(interactionSource = interactionSource).value,
            shape = MaterialTheme.shapes.small.copy(CornerSize(percent = 50)),
            clip = true,
        )
        .shimmer(),
    interactionSource = interactionSource,
    ...

If you are using a shimmer that has an alpha value, the later might look a bit weird when pushing the button. That's because two shadows are now drawn within the clipping area, underneath the button.

But I think you could play around a little bit and see what fits best for your app. Try using a fixed elevation for example.

Thanks for the fast reply. Works so far. Unfortunately this part does not work elevation.elevation(interactionSource = interactionSource).value

I have checked the source code of FAB and the FloatingActionButtonElevation instance does not has any public functions. I would need elevation.shadowElevation(interactionSource = interactionSource).value but this fun is internal. Does material 3 lack support for this case here or did I miss something?

I need the delta value of elevation in the different FAB states

Did you get it working in the meantime? If not, let me suggest a different approach. I tried it with material 3 today and it really is difficult.

But we can simply draw a shimmering overlay on top of the FAB:

var isLoading by remember { mutableStateOf(true) }
LaunchedEffect(key1 = true) {
    delay(5_000)
    isLoading = false
}
Box(
    modifier = Modifier
        .height(IntrinsicSize.Min)
        .width(IntrinsicSize.Min)
) {
    ExtendedFloatingActionButton(
        onClick = { /* do something */ },
        icon = { Icon(Icons.Filled.Add, "Description") },
        text = { Text(text = "Extended FAB") },
    )
    AnimatedVisibility(
        visible = isLoading,
        exit = fadeOut(animationSpec = tween(500)),
    ) {
        Box(
            modifier = Modifier
                .fillMaxSize()
                .clip(FloatingActionButtonDefaults.extendedFabShape)
                .background(Color.LightGray)
                .shimmer()
                .background(Color.DarkGray)
        )
    }
}
shimmer_overlay.mov

EDIT: Improved the implementation

Sorry, a little bit late. Thanks for your investigation. I came up with this solution:

@Composable
private fun ScannerFab(
    modifier: Modifier = Modifier,
    shape: Shape = FloatingActionButtonDefaults.extendedFabShape,
    isScanning: Boolean,
    onAction: () -> Unit = {},
) {
    val shimmer = rememberShimmer(
        shimmerBounds = ShimmerBounds.View,
        theme = defaultShimmerTheme.copy(
            blendMode = BlendMode.Hardlight,
            rotation = 25f,
            shaderColors = listOf(
                Color.White.copy(alpha = 0.0f),
                Color.White.copy(alpha = 0.6f),
                Color.White.copy(alpha = 0.0f),
            ),
            shaderColorStops = null,
            shimmerWidth = 400.dp,
        ),
    )

    val interactionSource = remember { MutableInteractionSource() }

    var shadowElevation by remember { mutableStateOf(6.dp) }

    LaunchedEffect(interactionSource) {
        interactionSource.interactions.collect { interaction ->
            shadowElevation = when (interaction) {
                is PressInteraction.Press -> 6.dp
                is HoverInteraction.Enter -> 8.dp
                is FocusInteraction.Focus -> 6.dp
                else -> 6.dp
            }
        }
    }

    /*
     * Since the elevation shadow is rendered before the shimmer effect,
     * shimmer effect is applied on the shadow. So disable default elevation shadow and
     * render shadow manually, then apply shimmer effect afterwards. By leveraging the
     * interaction source, we can control the elevation in different states.
     */
    ExtendedFloatingActionButton(
        modifier = modifier
            .fillMaxWidth()
            .padding(horizontal = 16.dp)
            .shadow(
                elevation = shadowElevation,
                shape = shape,
                clip = true,
            )
            .then(if (isScanning) Modifier.shimmer(shimmer) else Modifier),
        onClick = onAction,
        shape = shape,
        elevation = FloatingActionButtonDefaults.elevation(0.dp, 0.dp, 0.dp, 0.dp),
        interactionSource = interactionSource,
    ) {
        val stringRes = if (isScanning) R.string.btn_searching else R.string.btn_search
        Text(stringResource(stringRes).uppercase())
    }
}