Skip to content

Commit 2b476b3

Browse files
pekingmepaulfthomas
authored andcommittedJul 17, 2023
[TopAppBar] Updated the handling of liftOnScrollColor not override android:background.
PiperOrigin-RevId: 547883467
1 parent 1d3b8e1 commit 2b476b3

File tree

3 files changed

+63
-19
lines changed

3 files changed

+63
-19
lines changed
 

‎lib/java/com/google/android/material/appbar/AppBarLayout.java

+46-14
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
import android.graphics.drawable.ColorDrawable;
3737
import android.graphics.drawable.ColorStateListDrawable;
3838
import android.graphics.drawable.Drawable;
39+
import android.graphics.drawable.LayerDrawable;
3940
import android.os.Build;
4041
import android.os.Build.VERSION;
4142
import android.os.Build.VERSION_CODES;
@@ -74,6 +75,7 @@
7475
import androidx.core.view.accessibility.AccessibilityViewCommand;
7576
import androidx.customview.view.AbsSavedState;
7677
import com.google.android.material.animation.AnimationUtils;
78+
import com.google.android.material.animation.ArgbEvaluatorCompat;
7779
import com.google.android.material.appbar.AppBarLayout.BaseBehavior.SavedState;
7880
import com.google.android.material.internal.ThemeEnforcement;
7981
import com.google.android.material.motion.MotionUtils;
@@ -271,7 +273,6 @@ public AppBarLayout(@NonNull Context context, @Nullable AttributeSet attrs, int
271273
} else {
272274
initializeLiftOnScrollWithElevation(context, materialShapeDrawable);
273275
}
274-
ViewCompat.setBackground(this, materialShapeDrawable);
275276
}
276277

277278
liftOnScrollColorDuration = MotionUtils.resolveThemeDuration(context,
@@ -337,19 +338,34 @@ private ColorStateList getBackgroundCSL() {
337338
}
338339

339340
private void initializeLiftOnScrollWithColor(MaterialShapeDrawable background) {
340-
background.setAlpha(lifted ? 255 : 0);
341-
background.setFillColor(liftOnScrollColor);
342-
liftOnScrollColorUpdateListener = valueAnimator -> {
343-
float alpha = (float) valueAnimator.getAnimatedValue();
344-
background.setAlpha((int) alpha);
341+
MaterialShapeDrawable liftBackground = new MaterialShapeDrawable();
342+
liftBackground.setFillColor(liftOnScrollColor);
343+
liftBackground.setAlpha(lifted ? 255 : 0);
344+
background.setAlpha(lifted ? 0 : 255);
345+
346+
liftOnScrollColorUpdateListener =
347+
valueAnimator -> {
348+
float liftAlpha = (float) valueAnimator.getAnimatedValue();
349+
background.setAlpha((int) (255f - liftAlpha));
350+
liftBackground.setAlpha((int) liftAlpha);
351+
352+
if (!liftOnScrollListeners.isEmpty()) {
353+
int mixedColor =
354+
ArgbEvaluatorCompat.getInstance()
355+
.evaluate(
356+
liftAlpha / 255f,
357+
background.getResolvedTintColor(),
358+
liftBackground.getResolvedTintColor());
359+
for (LiftOnScrollListener liftOnScrollListener : liftOnScrollListeners) {
360+
if (background.getFillColor() != null) {
361+
liftOnScrollListener.onUpdate(0, mixedColor);
362+
}
363+
}
364+
}
365+
};
345366

346-
for (LiftOnScrollListener liftOnScrollListener : liftOnScrollListeners) {
347-
if (background.getFillColor() != null) {
348-
liftOnScrollListener.onUpdate(
349-
0, background.getFillColor().withAlpha((int) alpha).getDefaultColor());
350-
}
351-
}
352-
};
367+
LayerDrawable layerBackground = new LayerDrawable(new Drawable[] {background, liftBackground});
368+
ViewCompat.setBackground(this, layerBackground);
353369
}
354370

355371
private void initializeLiftOnScrollWithElevation(
@@ -365,6 +381,7 @@ private void initializeLiftOnScrollWithElevation(
365381
liftOnScrollListener.onUpdate(elevation, background.getResolvedTintColor());
366382
}
367383
};
384+
ViewCompat.setBackground(this, background);
368385
}
369386

370387
/**
@@ -1003,7 +1020,7 @@ boolean setLiftedState(boolean lifted, boolean force) {
10031020
if (force && this.lifted != lifted) {
10041021
this.lifted = lifted;
10051022
refreshDrawableState();
1006-
if (liftOnScroll && getBackground() instanceof MaterialShapeDrawable) {
1023+
if (liftOnScroll && isLiftOnScrollCompatibleBackground()) {
10071024
if (liftOnScrollColor != null) {
10081025
startLiftOnScrollColorAnimation(
10091026
lifted ? 0 : 255, lifted ? 255 : 0);
@@ -1017,6 +1034,21 @@ boolean setLiftedState(boolean lifted, boolean force) {
10171034
return false;
10181035
}
10191036

1037+
private boolean isLiftOnScrollCompatibleBackground() {
1038+
if (getBackground() instanceof MaterialShapeDrawable) {
1039+
return true;
1040+
}
1041+
if (getBackground() instanceof LayerDrawable) {
1042+
LayerDrawable layerBackground = (LayerDrawable) getBackground();
1043+
if (layerBackground.getNumberOfLayers() == 2
1044+
&& layerBackground.getDrawable(0) instanceof MaterialShapeDrawable
1045+
&& layerBackground.getDrawable(1) instanceof MaterialShapeDrawable) {
1046+
return true;
1047+
}
1048+
}
1049+
return false;
1050+
}
1051+
10201052
private void startLiftOnScrollColorAnimation(
10211053
float fromValue, float toValue) {
10221054
if (liftOnScrollColorAnimator != null) {

‎testing/java/com/google/android/material/testapp/res/layout/design_appbar_toolbar_liftonscroll_color.xml

+1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
android:id="@+id/app_bar"
2727
android:layout_height="wrap_content"
2828
android:layout_width="match_parent"
29+
android:background="@color/material_blue_grey_900"
2930
app:liftOnScroll="true"
3031
app:liftOnScrollColor="@color/material_blue_grey_950">
3132

‎tests/javatests/com/google/android/material/appbar/AppBarWithToolbarTest.java

+16-5
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
import static org.junit.Assert.assertEquals;
2323

2424
import android.graphics.Rect;
25+
import android.graphics.drawable.LayerDrawable;
2526
import android.view.View;
2627
import android.view.ViewGroup;
2728
import androidx.coordinatorlayout.widget.CoordinatorLayout;
@@ -394,7 +395,19 @@ public void testLiftOnScrollColor() throws Throwable {
394395
final int appbarHeight = mAppBar.getHeight();
395396
final int longSwipeAmount = 3 * appbarHeight / 2;
396397

397-
assertEquals(0, ((MaterialShapeDrawable) mAppBar.getBackground()).getAlpha());
398+
LayerDrawable background = (LayerDrawable) mAppBar.getBackground();
399+
MaterialShapeDrawable backgroundLayer = (MaterialShapeDrawable) background.getDrawable(0);
400+
MaterialShapeDrawable liftLayer = (MaterialShapeDrawable) background.getDrawable(1);
401+
402+
assertEquals(
403+
mAppBar.getResources().getColor(R.color.material_blue_grey_900),
404+
backgroundLayer.getResolvedTintColor());
405+
assertEquals(
406+
mAppBar.getResources().getColor(R.color.material_blue_grey_950),
407+
liftLayer.getResolvedTintColor());
408+
409+
assertEquals(255, backgroundLayer.getAlpha());
410+
assertEquals(0, liftLayer.getAlpha());
398411

399412
// Perform a swipe-up gesture across the horizontal center of the screen.
400413
performVerticalSwipeUpGesture(
@@ -403,9 +416,7 @@ public void testLiftOnScrollColor() throws Throwable {
403416
originalAppbarBottom + 3 * longSwipeAmount / 2,
404417
longSwipeAmount);
405418

406-
assertEquals(255, ((MaterialShapeDrawable) mAppBar.getBackground()).getAlpha());
407-
assertEquals(
408-
mAppBar.getResources().getColor(R.color.material_blue_grey_950),
409-
((MaterialShapeDrawable) mAppBar.getBackground()).getFillColor().getDefaultColor());
419+
assertEquals(0, backgroundLayer.getAlpha());
420+
assertEquals(255, liftLayer.getAlpha());
410421
}
411422
}

0 commit comments

Comments
 (0)
Please sign in to comment.