ViewPager经过PageTransformer实现切换时背景色的渐变

定个小目标

咱们要实现以下的效果,ViewPager正中的page显示为绿色,其他两边的显示为蓝色。中间page像左右两边切换时会从绿色慢慢过渡到蓝色,同理两边的page向中间切换时颜色也会慢慢过渡,而不是直接跳变。 bash

PageTransformer

为了实现以上效果咱们要借助PagerTransfromer,因此有必要简单介绍下微信

/**
 * A PageTransformer is invoked whenever a visible/attached page is scrolled.
 * This offers an opportunity for the application to apply a custom transformation
 * to the page views using animation properties.
 *
 * <p>As property animation is only supported as of Android 3.0 and forward,
 * setting a PageTransformer on a ViewPager on earlier platform versions will
 * be ignored.</p>
 */
public interface PageTransformer {
    /**
     * Apply a property transformation to the given page.
     *
     * @param page Apply the transformation to this page
     * @param position Position of page relative to the current front-and-center
     *                 position of the pager. 0 is front and center. 1 is one full
     *                 page position to the right, and -1 is one page position to the left.
     */
    void transformPage(View page, float position);
}
复制代码

PageTransformer很简单,就只有一个transformPage方法。
第一个参数是咱们要变换颜色的page。app

重点看第二个参数,英语好的话直接看注释说得很明白,意思就是咱们当前要变化的页面当前位置的中心距离pager中心的距离,好比当前处于中心的page的position就是0,右边第一个page的position=1,右边第二个page的position为2,左边第一个page的position=-1,左边第二个page的position=-2,以此类推。ide

固然position的值不可能一直是整数,好比当处于中间位置的page向右边切换到右边第一个位置时,position的值会从0变化到1,1就是表示一个page的宽度,从中间位置移到右边第一个位置就是移动了一个page的宽度。若是向右移动到一半,那么position就是0.5。 ui

实现的思路

只有中间位置是绿色,其他两边的都是蓝色,因此能够肯定的是position=0,确定是绿色,position的绝对值Math.abs(position)>=1必定是蓝色,剩下的咱们就是要处理-1<position<1这些位置的渐变色,也就是中间位置到左右两边第一个page的位置。this

中间位置到左右两边第一个page的位置的position取绝对值的值范围从0-1,其实就是0%-100%,咱们能够用这个值当作颜色的变化率,0就是绿色,值越大表示越往蓝色变化,1就表示变成蓝色了。spa

另外一个难点就是如何实现颜色的渐变。咱们知道颜色由RGB表示,有三个颜色的份量分别表示红绿蓝。咱们把起始颜色和最终颜色的三个颜色份量分别取出来,分别计算三个颜色份量的差值,最后用差值乘以变化率再加上初始颜色份量就是当前的颜色份量值,把三个当前颜色份量的值拼起来就是当前的颜色值。3d

初始颜色:#00ff00
最终颜色:#0000ff
初始颜色份量:红00,绿ff(255),蓝00
最终颜色份量:红00,绿00,蓝ff(255)
颜色份量差值:红00,绿-255,蓝255code

假设咱们要计算position=0.5位置的颜色 当前红色份量=0+0.500=0=0x00
当前绿色份量=255+0.5
(-255)=127=0x7f
当前蓝色份量=0+0.5*255=127=0x7form

因此position=0.5时当前的颜色为#007f7f。

代码实现

一个很简单的功能说了这么多,就是为了让你们更好的理解,若是直接放代码看了坑能会一头雾水

设置PagerTransfromer

mViewPager.setPageTransformer(true, new ViewPager.PageTransformer() {
    @Override
    public void transformPage(@NonNull View view, float v) {
        // 取绝对值
        v = Math.abs(v);
        // 超过左右两边第一个位置的page都设置成蓝色
        if (v > 1) {
            v = 1;
        }
        String color = transColor("#00ff00", "#0000ff", v);
        view.setBackgroundColor(Color.parseColor(color));
    }
});
复制代码

计算当前颜色值

/**
 * 根据起始颜色、最终颜色和变化率计算当前颜色
 * @param start 起始颜色
 * @param end 最终颜色
 * @param rate 变化率
 * @return 当前颜色
 */
private static String transColor(String start, String end, float rate) {
    // 获取初始颜色的RGB份量
    String startR = start.substring(1, 3);
    String startG = start.substring(3, 5);
    String startB = start.substring(5, 7);
    // 获取最终颜色的颜色份量
    String endR = end.substring(1, 3);
    String endG = end.substring(3, 5);
    String endB = end.substring(5, 7);
    // 计算初始颜色的RGB份量十进制值
    int startRI = Integer.parseInt(startR, 16);
    int startGI = Integer.parseInt(startG, 16);
    int startBI = Integer.parseInt(startB, 16);
    // 计算最终颜色的RGB份量十进制值
    int endRI = Integer.parseInt(endR, 16);
    int endGI = Integer.parseInt(endG, 16);
    int endBI = Integer.parseInt(endB, 16);
    // 计算当前颜色份量
    int curRI = (int) (startRI + (endRI - startRI)*rate);
    int curGI = (int) (startGI + (endGI - startGI)*rate);
    int curBI = (int) (startBI + (endBI - startBI)*rate);
    // 转成16进制
    String curR = String.format("%02x", curRI);
    String curG = String.format("%02x", curGI);
    String curB = String.format("%02x", curBI);
    // 拼成颜色码
    return "#" + curR + curG + curB;
}
复制代码

若是喜欢个人文章,能够扫描如下二维码关注个人微信公众号,我会按期发布最新的文章。
关注个人微信,回复背景色渐变获取本文的完整代码。

相关文章
相关标签/搜索