36
36
import android .graphics .drawable .ColorDrawable ;
37
37
import android .graphics .drawable .ColorStateListDrawable ;
38
38
import android .graphics .drawable .Drawable ;
39
+ import android .graphics .drawable .LayerDrawable ;
39
40
import android .os .Build ;
40
41
import android .os .Build .VERSION ;
41
42
import android .os .Build .VERSION_CODES ;
74
75
import androidx .core .view .accessibility .AccessibilityViewCommand ;
75
76
import androidx .customview .view .AbsSavedState ;
76
77
import com .google .android .material .animation .AnimationUtils ;
78
+ import com .google .android .material .animation .ArgbEvaluatorCompat ;
77
79
import com .google .android .material .appbar .AppBarLayout .BaseBehavior .SavedState ;
78
80
import com .google .android .material .internal .ThemeEnforcement ;
79
81
import com .google .android .material .motion .MotionUtils ;
@@ -271,7 +273,6 @@ public AppBarLayout(@NonNull Context context, @Nullable AttributeSet attrs, int
271
273
} else {
272
274
initializeLiftOnScrollWithElevation (context , materialShapeDrawable );
273
275
}
274
- ViewCompat .setBackground (this , materialShapeDrawable );
275
276
}
276
277
277
278
liftOnScrollColorDuration = MotionUtils .resolveThemeDuration (context ,
@@ -337,19 +338,34 @@ private ColorStateList getBackgroundCSL() {
337
338
}
338
339
339
340
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
+ };
345
366
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 );
353
369
}
354
370
355
371
private void initializeLiftOnScrollWithElevation (
@@ -365,6 +381,7 @@ private void initializeLiftOnScrollWithElevation(
365
381
liftOnScrollListener .onUpdate (elevation , background .getResolvedTintColor ());
366
382
}
367
383
};
384
+ ViewCompat .setBackground (this , background );
368
385
}
369
386
370
387
/**
@@ -1003,7 +1020,7 @@ boolean setLiftedState(boolean lifted, boolean force) {
1003
1020
if (force && this .lifted != lifted ) {
1004
1021
this .lifted = lifted ;
1005
1022
refreshDrawableState ();
1006
- if (liftOnScroll && getBackground () instanceof MaterialShapeDrawable ) {
1023
+ if (liftOnScroll && isLiftOnScrollCompatibleBackground () ) {
1007
1024
if (liftOnScrollColor != null ) {
1008
1025
startLiftOnScrollColorAnimation (
1009
1026
lifted ? 0 : 255 , lifted ? 255 : 0 );
@@ -1017,6 +1034,21 @@ boolean setLiftedState(boolean lifted, boolean force) {
1017
1034
return false ;
1018
1035
}
1019
1036
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
+
1020
1052
private void startLiftOnScrollColorAnimation (
1021
1053
float fromValue , float toValue ) {
1022
1054
if (liftOnScrollColorAnimator != null ) {
0 commit comments