如何在CoordinatorLayout上添加边缘阴影

前言

使用RecyclerView和ScrollView等系统提供的滑动控件,需要边缘淡化,只需要配置requiresFadingEdge和fadingEdgeLength,指定淡化的方向和长度即可,很遗憾CoordinatorLayout并不支持这些属性。

requiresFadingEdge属性用于在视图的边缘添加淡出效果,当内容超出边界时,可以通过滚动视图来显示淡出效果。然而,CoordinatorLayout是一个布局容器,它的主要功能是协调子视图之间的交互和动作,并管理它们的位置和行为。它并不直接处理滚动,而是将滚动事件传递给子视图处理。

CoordinatorLayout的设计目的是协调子视图之间的交互和动作,并提供灵活的布局和交互性。

一、源码分析

跟踪 android:requiresFadingEdge 初始化设定的位置,是在最基本的 View 控件内。当 View 绘制时满足特定条件就会用内部的 ScrollabilityCache 绘制滚动条、边缘淡化等滑动用的元素。

抛开 ScrollabilityCache ,看看 View 的 draw(Canvas canvas) 方法,源代码里写得很清楚:

ini 复制代码
public void draw(Canvas canvas) {
    final int privateFlags = mPrivateFlags;
    mPrivateFlags = (privateFlags & ~PFLAG_DIRTY_MASK) | PFLAG_DRAWN;

    /*     
    * Draw traversal performs several drawing steps which must be executed     
    * in the appropriate order:     
    *     
    *      1. Draw the background     
    *      2. If necessary, save the canvas' layers to prepare for fading     
    *      3. Draw view's content     
    *      4. Draw children     
    *      5. If necessary, draw the fading edges and restore layers     
    *      6. Draw decorations (scrollbars for instance)     
    *      7. If necessary, draw the default focus highlight     
    */
    .....
    if (verticalEdges) {
        topFadeStrength = Math.max(0.0f, Math.min(1.0f, getTopFadingEdgeStrength()));
        drawTop = topFadeStrength * fadeHeight > 1.0f;
        bottomFadeStrength = Math.max(0.0f, Math.min(1.0f, getBottomFadingEdgeStrength()));
        drawBottom = bottomFadeStrength * fadeHeight > 1.0f;
    }
    ....
}

如果需要绘制边缘淡化,会多增加第2和第5步的区别

二、自定义ScollerView

既然CoordinatorLayout不支持边缘淡化,是否可以使用ScollerView把它包裹起来,然后动态控制边缘淡化显示和隐藏即可。

基于上面的源码分析,我们可以知道边缘淡化是由drawTop、drawBottom..几个变量控制,它是通过getTopFadingEdgeStrength()...这几个方法判断绘制长度。那么我们只需要在自定义控件中重写对应的方法就能使边缘淡化,实际使用中还需要动态改变并刷新 View 来精确控制效果。

kotlin 复制代码
class MaskScrollView @JvmOverloads constructor(    
    context: Context, attrs: AttributeSet? = null
) : ScrollView(context, attrs) {    

    var isShowEdge:Boolean = false        
        set(value) {            
            if (field != value) {  
                 field = value
                // 避免getTopFadingEdgeStrength()不调用                
                invalidate()            
            }        
        }    
          
    override fun onInterceptTouchEvent(ev: MotionEvent?): Boolean {
         return false    
    }    
    
    override fun getTopFadingEdgeStrength(): Float { 
           return if (isShowEdge) 1.0f else 0f   
    }
}
  • MaskScrollView只作为边缘淡化显示的载体,因此不需要处理任何事件onInterceptTouchEvent直接返回false
  • getTopFadingEdgeStrength()只会在View的onDraw()方法调用,当isShowEdge改变时确保View的刷新,增加invalidate()
  • 如果不需要底部的边缘淡化可以直接在getBottomFadingEdgeStrength()返回0f

而监听CoordinatorLayout的滑动可以通过AppBarLayout滑动监听来实现

ini 复制代码
binding.appBarLayout.addOnOffsetChangedListener(
    AppBarLayout.OnOffsetChangedListener { _, verticalOffset ->
        binding.maskScrollView.isShowEdge = verticalOffset < 0    
    }
)

参考:juejin.cn/post/684490...

相关推荐
莞凰8 小时前
昇腾CANN的“灵脉根基“:Runtime仓库探秘
android·人工智能·transformer
NiceCloud喜云9 小时前
Claude Files API 深入:从上传、复用到配额管理的工程化指南
android·java·数据库·人工智能·python·json·飞书
ujainu9 小时前
CANN pto-isa:虚拟指令集如何连接编译与执行
android·ascend
赏金术士10 小时前
第六章:UI组件与Material3主题
android·ui·kotlin·compose
TechMerger11 小时前
Android 17 重磅重构!服役 20 年的 MessageQueue 迎来无锁改造,卡顿大幅优化!
android·性能优化
yuhuofei202114 小时前
【Python入门】Python中字符串相关拓展
android·java·python
dalancon14 小时前
Android Input Spy Window
android
dalancon16 小时前
InputDispatcher派发事件,查找目标窗口
android
我命由我1234516 小时前
Android Framework P3 - MediaServer 进程、认识 ServiceManager 进程
android·c语言·开发语言·c++·visualstudio·visual studio·android runtime
天才少年曾牛17 小时前
Android14 新增系统服务后,应用调用出现 “hidden api” 警告的原因与解决方案
android·frameworks