有些应用上面的标题的背景色默认的是全透明的,也就是以背景图片来作为背景色。然后随着ListView的滑动标题的背景色会有全透明慢慢的变为和背景图一样的颜色。以图为证:

android gradient 渐变使用 android 渐变背景_ide


上面的效果图想必大家并不陌生,我们手机上面的一些应用就有。那么这个效果是怎么实现的呢?其实这个效果就是一个自定义的ScrollView,其中要重写里面的onScrollChanged这个方法,

protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        throw new RuntimeException("Stub!");
    }

int l, int t, int oldl, int oldt分别是left、top、oldLeft、oldTop。也就是左,上,上次的左,上次的上。怎么说呢,就是屏幕的原点也就是ScrollVIew的原点为(left,top)默认的初始值为(0,0)这点应该很好懂吧。为什么要重写这个方法呢?看到onScrollChanged这个方法名我们应该就能知道这是一个监听关于ScrollVIew滑动时发生变化的方法。

android gradient 渐变使用 android 渐变背景_标题背景渐变_02


看到打印出来的日志也许你就会知道这些参数的意义了。left为什么不打印,是因为我们实现的是ScrollVIew的上下滑动而不是左右滑动,所以就没有必要打印left了。

要想实现上面的标题渐变的效果,我们需要测量出标题背景图的高度,当ScrollVIew未滑动的时候,标题的背景色是全透明的,也就是alpha值为0。当我们开始滑动的时候,我们的alpha的值=top/背景图的高度,当我们向上滑动的ScrollVIew的距离大于标题的背景图的高度时,此时的标题背景色是普通的背景色也就是我们背景图的颜色。

@Override
public void onScrollChanged(CustomScrollView scrollView, int left, int top, int oldLeft, int oldTop) {
        if (top <= 0) {   
           ll_title.setBackgroundColor(Color.argb(0,  74,204,185));
        } else if (top > 0 && top <= bgHeight) { 
            float scale = (float) top / bgHeight;
            float alpha = (255 * scale);
            ll_title.setBackgroundColor(Color.argb((int) alpha, 74,204,185));
        } else {   
            ll_title.setBackgroundColor(Color.argb(255, 74,204,185));
        }
    }

我们都知道在onCreate()里面获取控件的高度是0,这是为什么呢?

12-08 14:43:25.691 29095-29095/com.lyxrobert.titlegradient I/System.out: onCreate----->
12-08 14:43:34.791 29095-29095/com.lyxrobert.titlegradient I/System.out: onMeasure----->
12-08 14:43:34.801 29095-29095/com.lyxrobert.titlegradient I/System.out: onMeasure----->

对比一下时间你就会发现,原来onCreate的执行的要比onMeasure要早。就是控件还没有测量具体的宽高的时候onCreate就已经执行了。这也是为什么我们在onCreate方法里面获取不到控件宽高度的原因。那么我们既然在onCreate里面获取不到控件的宽高怎么办呢?所以现在需要使用getViewTreeObserver().addOnGlobalLayoutListener()来获得宽度或者高度。这是获得一个view的宽度和高度的方法之一。
OnGlobalLayoutListener 是ViewTreeObserver的内部类,当一个视图树的布局发生改变时,可以被ViewTreeObserver监听到,这是一个注册监听视图树的观察者(observer),在视图树的全局事件改变时得到通知。ViewTreeObserver不能直接实例化,而是通过getViewTreeObserver()获得。

ViewTreeObserver vto = rel_title_bg.getViewTreeObserver();
        vto.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                ll_title.getViewTreeObserver().removeGlobalOnLayoutListener(
                        this);
                bgHeight = rel_title_bg.getHeight();
            }
        });

还有就是ScrollView嵌套ListView的时候会产生ListView只显示一条的效果,这个问题怎么解决呢?有两种办法。一种是测量ListView的Item的高度,根据ListView的实际高度=Item的高度*Item的总条数+Item之间的空间得到ListView的高度。

TestAdp adp = new TestAdp(this,list);
        lv_title.setAdapter(adp);
        adp.notifyDataSetChanged();
        int totalHeight = 0;
        int size = 0;
        size = adp.getCount();
        for (int i = 0; i < size; i++) {
            View listItem = adp
                    .getView(i, null, lv_title);
            listItem.measure(0, 0);
            totalHeight += listItem.getMeasuredHeight();
        }

        ViewGroup.LayoutParams params = lv_title
                .getLayoutParams();
        params.height = totalHeight
                + (lv_title.getDividerHeight() * (lv_title
                .getCount()));
        lv_title.setLayoutParams(params);

另外一种就是自定义一个ListView,然后重写里面的onMeasure方法

@Override
    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
                MeasureSpec.AT_MOST);
        super.onMeasure(widthMeasureSpec, expandSpec);
    }

至于为什么要设置为Integer.MAX_VALUE >> 2,为什么要右移两位,因为Integer.MAX_VALUE 的前两位是mode,向右移两位得到的就是高度或宽度值了。MeasureSpec.AT_MOST当模式为MeasureSpec.AT_MOST时则意味着控件大小一般随着控件的子空间或内容进行变化,此时控件尺寸只要不超过父控件允许的最大尺寸即可,我们经常使用的wrap_content就是这种模式。

还有一点就是,如果ScrollView里面嵌套ListView很容易产生ListView因获取焦点而导致打开页面的时候ListView的第一条显示在屏幕的原点,也就是说ScrollView的原点默认的不是屏幕原点,也就是下图的情况

android gradient 渐变使用 android 渐变背景_android_03


这是为什么呢?CustomScrollView 继承ScrollView继承FrameLayout 继承ViewGroup,而CustomListView 继承ListView继承AbsListView继承AdapterView继承ViewGroup。我们都知道子控件抢占父控件的焦点,而我们自定义的ListView又包含在自定义的ScrollView里面,这也就说明了这个问题。如何解决呢?很简单,就是让子空间默认不获取焦点即可。也就是设置为setFocusable(false);

不多说了看代码

MainActivity.class

package com.lyxrobert.titlegradient;

import android.app.Activity;
import android.graphics.Color;
import android.os.Bundle;
import android.view.ViewTreeObserver;
import android.widget.LinearLayout;
import android.widget.RelativeLayout;
import java.util.ArrayList;
import java.util.List;

public class MainActivity extends Activity implements CustomScrollView.ScrollChangedListener{
    private CustomScrollView tsv_title;
    private CustomListView lv_title;
    private LinearLayout ll_title;
    private RelativeLayout rel_title_bg;
    private int bgHeight;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initView();
        initEvent();
       initData();
    }
    private void initView() {
        tsv_title = (CustomScrollView) findViewById(R.id.tsv_title);
        lv_title = (CustomListView) findViewById(R.id.lv_title);
        lv_title.setFocusable(false);
        ll_title = (LinearLayout) findViewById(R.id.ll_title);
        rel_title_bg = (RelativeLayout) findViewById(R.id.rel_title_bg);
    }
    private void initEvent() {
        ViewTreeObserver vto = rel_title_bg.getViewTreeObserver();
        vto.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                ll_title.getViewTreeObserver().removeGlobalOnLayoutListener(
                        this);
                bgHeight = rel_title_bg.getHeight();
            }
        });
        tsv_title.setOnScrollChanged(this);
    }
    private void initData() {

        List<String> list = new ArrayList<>();
        for (int i =0;i<30;i++){
            list.add;
        }
        TestAdp adp = new TestAdp(this,list);
        lv_title.setAdapter(adp);
        adp.notifyDataSetChanged();
    }

    @Override
    public void onScrollChanged(CustomScrollView scrollView, int left, int top, int oldLeft, int oldTop) {
        if (top <= 0) {  
            ll_title.setBackgroundColor(Color.argb(0,  74,204,185));
        } else if (top > 0 && top <= bgHeight) {
            float scale = (float) top / bgHeight;
            float alpha = (255 * scale);
            ll_title.setBackgroundColor(Color.argb((int) alpha, 74,204,185));
        } else {    
            ll_title.setBackgroundColor(Color.argb(255, 74,204,185));
        }
    }
}

自定义ScrollView

package com.lyxrobert.titlegradient;

import android.content.Context;
import android.util.AttributeSet;
import android.widget.ScrollView;

public class CustomScrollView extends ScrollView {

    public interface ScrollChangedListener {

        void onScrollChanged(CustomScrollView scrollView, int left, int top, int oldLeft, int oldTop);
    }

    private ScrollChangedListener scrollChangedListener = null;

    public CustomScrollView(Context context) {
        super(context);
    }

    public CustomScrollView(Context context, AttributeSet attrs,
                            int defStyle) {
        super(context, attrs, defStyle);
    }

    public CustomScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public void setOnScrollChanged(ScrollChangedListener scrollChangedListener) {
        this.scrollChangedListener = scrollChangedListener;
    }

    @Override
    protected void onScrollChanged(int left, int top, int oldLeft, int oldTop) {
        super.onScrollChanged(left, top, oldLeft, oldTop);
        if (scrollChangedListener != null) {
            scrollChangedListener.onScrollChanged(this, left, top, oldLeft, oldTop);
        }
    }

}

自定义ListView

package com.lyxrobert.titlegradient;

import android.content.Context;
import android.util.AttributeSet;
import android.widget.ListView;

public class CustomListView extends ListView {

    public CustomListView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomListView(Context context) {
        super(context);
    }

    public CustomListView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    @Override
    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
                MeasureSpec.AT_MOST);
        super.onMeasure(widthMeasureSpec, expandSpec);
    }
}

Adapter类

package com.lyxrobert.titlegradient;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

import java.util.List;

/**
 * Created by ytx on 2016/12/8.
 */
public class TestAdp extends BaseAdapter {
    private Context mContext;
    private List<String> mList;

    public TestAdp(Context context, List<String> list) {
        this.mContext = context;
        this.mList = list;
    }

    @Override
    public int getCount() {
        return mList!=null?mList.size():0;
    }

    @Override
    public Object getItem(int position) {
        return mList.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup viewGroup) {
        ViewHolder viewHolder;
        if (convertView ==null){
            convertView = LayoutInflater.from(mContext).inflate(R.layout.item_test,null);
            viewHolder = new ViewHolder();
            viewHolder.tv_test = (TextView) convertView.findViewById(R.id.tv_test);
            convertView.setTag(viewHolder);
        }else {
            viewHolder = (ViewHolder) convertView.getTag();
        }
        viewHolder.tv_test.setText(mList.get(position));
        return convertView;
    }
class ViewHolder{
    TextView tv_test;
}
}

布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"

    >
    <com.lyxrobert.titlegradient.CustomScrollView
        android:id="@+id/tsv_title"
        android:scrollbars="none"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="vertical">
        <RelativeLayout
            android:id="@+id/rel_title_bg"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:background="@drawable/icon_title_bg">
        </RelativeLayout>
            <com.lyxrobert.titlegradient.CustomListView
                android:id="@+id/lv_title"
                android:layout_width="match_parent"
                android:layout_height="match_parent">
            </com.lyxrobert.titlegradient.CustomListView>
        </LinearLayout>
    </com.lyxrobert.titlegradient.CustomScrollView>
    <LinearLayout
        android:id="@+id/ll_title"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">
        <TextView
            android:text="我是测试标题"
            android:textSize="20dp"
            android:padding="10dp"
            android:textColor="#FFF"
            android:gravity="center"
            android:layout_width="match_parent"
            android:layout_height="wrap_content" />
    </LinearLayout>
</RelativeLayout>