阅读 1006

android中foreground水波实现过程分析

hello,大家好,上一篇介绍了drawable如何显示到view上,基本上是以background属性来讲的,其实在view中用到的drawable地方还是挺多的,还不属性drawable显示到view的流程,可以看下我写的上一篇android中drawable显示到view上的过程,今天要介绍的也是跟drawable一个相关的属性foreground属性,不过该属性之前只是针对FrameLayout的,后来在23的api之后所有的view都能用该属性,因此大家知道这么回事就行了,而且在后面view源码中也会看到该属性兼容的代码,该属性一般在开发中能实现水波点击的效果,不知道大家平时用得多不多,好了,下面还是跟往常一样,通过一个简单的例子来介绍该属性的使用:

<TextView
    android:id="@+id/view"
    android:layout_width="100dp"
    android:layout_height="100dp"
    android:layout_marginTop="50dp"
    android:background="#cccccc"
    android:foreground="#ff0000"
    android:gravity="center"
    android:text="我是测试的view" />

<TextView
    android:id="@+id/view1"
    android:layout_width="100dp"
    android:layout_height="100dp"
    android:layout_marginTop="50dp"
    android:background="#cccccc"
    android:foreground="?attr/selectableItemBackground"
    android:gravity="center"
    android:text="我是测试的view" />
复制代码

simple
demo是很简单,为了演示效果,上面textview的foreground属性是一个颜色值,下面textview的foreground是获取应用的style里面的selectableItemBackground属性。第一个textview的foreground属性颜色直接把background属性覆盖掉了,而第二个textview的foreground是一个波纹效果,因此带着这些问题顺着源码看下这些问题,直接看获取view的foreground属性地方:

在此处看到该属性值在api>=23或view是frameLayout的时候调用了setForeground方法,该方法其实跟setBackground方法做的是类似的事,先是判断有没有foreground,如果有先销毁掉foreground,然后调用applyForegroundTint方法设置foreground的着色情况,最后也是触发了重新绘制view。那直接看view绘制的时候,是怎么绘制foreground的:

public void draw(Canvas canvas) {
    final int privateFlags = mPrivateFlags;
    final boolean dirtyOpaque = (privateFlags & PFLAG_DIRTY_MASK) == PFLAG_DIRTY_OPAQUE &&
            (mAttachInfo == null || !mAttachInfo.mIgnoreDirtyState);
  

    if (!dirtyOpaque) {
        drawBackground(canvas);
    }

    if (!dirtyOpaque) onDraw(canvas);

    onDrawForeground(canvas);
}
复制代码

这里我把draw方法几个关键方法给列出来了,先是绘制background,然后是onDraw,最后才是foreground,所以说在上面第一个例子中,因为最后才绘制foreground,因此显示的结果只有foreground的颜色了,下面来看看onDrawForeground方法是怎么绘制foreground的:

public void onDrawForeground(Canvas canvas) {
    onDrawScrollIndicators(canvas);
    onDrawScrollBars(canvas);

    final Drawable foreground = mForegroundInfo != null ? mForegroundInfo.mDrawable : null;
    if (foreground != null) {
        if (mForegroundInfo.mBoundsChanged) {
            mForegroundInfo.mBoundsChanged = false;
            final Rect selfBounds = mForegroundInfo.mSelfBounds;
            final Rect overlayBounds = mForegroundInfo.mOverlayBounds;

            if (mForegroundInfo.mInsidePadding) {
                selfBounds.set(0, 0, getWidth(), getHeight());
            } else {
                selfBounds.set(getPaddingLeft(), getPaddingTop(),
                        getWidth() - getPaddingRight(), getHeight() - getPaddingBottom());
            }

            final int ld = getLayoutDirection();
            //根据mForegroundInfo.mGravity得到foreground的bounds
            Gravity.apply(mForegroundInfo.mGravity, foreground.getIntrinsicWidth(),
                    foreground.getIntrinsicHeight(), selfBounds, overlayBounds, ld);
            foreground.setBounds(overlayBounds);
        }

        foreground.draw(canvas);
    }
}
复制代码

其实跟background的绘制差不多,只不过在foreground设置bounds的时候,多了一个foreground.gravity的判断,意思是foreground的权重,但是我测试过权重只有fill的情况下才起作用,其他的其中foreground.gravity都会让foreground的颜色失去作用。

写到这的时候,大家知道了事例一中为什么加了foreground属性颜色值之后,为什么设置textview的background以及text属性都看不到了吧,因为foreground是在绘制之后最后绘制的,所以被foreground的颜色给覆盖了。那第二个事例中为什么会有点击的波纹效果呢,这个就需要了解?attr/selectableItemBackground代表的是啥,这个其实是跟咱们的主题style属性相关,也就是顺着app的application的style属性可以找到该属性是什么:

直接来到21下面的Base.Theme.AppCompat.Light下面找:

此处找到了关于selectableItemBackground属性,但还是style里面的属性,不要紧,咱们继续找父style,最后在Theme.Material.Lightstyle下面找到了:

也就是说水波效果用到的资源文件是item_background_material的drawable文件,继续看下该资源文件是怎么定义的:

<ripple xmlns:android="http://schemas.android.com/apk/res/android"
    android:color="?attr/colorControlHighlight">
    <item android:id="@id/mask">
        <color android:color="@color/white" />
    </item>
</ripple>
复制代码

color颜色用的是?attr/colorControlHighlight,咱们可以看下该属性是怎么定义的,该属性也是在Theme.Material.Lightstyle下面定义的:

继续看下ripple_material_light是怎么定义的:

<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:alpha="@dimen/highlight_alpha_material_light"
          android:color="@color/foreground_material_light" />
</selector>
复制代码

此处定义了一个透明度为0.12,颜色为黑色的selector颜色值。在上一节我们知道drawable的子类是根据子类的各种标签生成不同的drawable,而水波的资源文件是ripple标签,所以从这里可以知道实质是一个RippleDrawable,关于RippleDrawable后面再讲解它们怎么绘制的。下面我们尝试下改变水波效果的颜色,按照系统自带的这个水波效果来写写,定义了一个change.xml的drawable文件:

<ripple xmlns:android="http://schemas.android.com/apk/res/android"
    android:color="@drawable/ripple_color">
    <item android:id="@android:id/mask">
        <color android:color="@android:color/white" />
    </item>
</ripple>
复制代码

可以看到这里引用了一个ripple_color的文件:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:alpha="0.5" android:color="@color/colorPrimary" />
</selector>
复制代码

用到了一个透明度为0.5,并且颜色用的是系统生成的颜色值。最后在view上引用change.xml文件:

效果大家可以录制的gif:

好了关于水波效果就说到这里,后面主要说说StateListDrawable、RippleDrawable实现效果的绘制是怎么来的,以及介绍drawable相关的api是如何使用的,以及使用drawable下面其他的不常用的drawable来实现好玩的功能。

文章分类
Android
文章标签