Android自定义控件-仿魅族Banner 自动轮播 无限轮播

自定义ViewPager实现魅族Banner效果

这个效果是在看过鸿洋老师的公众号文章后自己实现的效果。也推荐大家关注鸿洋公众号,每天都能学到新的知识,非常牛逼。


先贴出魅族应用的Banner看看效果

这是魅族应用里面的Banner效果图,看着很绚丽

再看看传统的Banner效果

传统Banner(素材是到别的博客里复制过来的,希望别介意!)

是不是一眼就能看出区别了呢,当然,这种效果喜不喜欢是因人而异的,感觉还是挺不错的。那么仿照这样的BannerView自己实现,看下最终效果图:

最终效果图

自己实现的效果图
由于录制的原因看起来有点卡 ,真实效果肯定不会这样的。

实现思路

当我第一次看到的时候我也感觉这种效果不是那么简单实现的,可看完鸿洋老师的文章之后,真的是,当然也是因为鸿洋老师说的详细,自己还有很长的路要走^_^

魅族应用的这种BannerView与传统的Banner区别就在他能在一个页面显示多个页面的数据,关键就是要实现这样的效果.

ViewGroup中定义了一个clipChildren属性默认值为true,意思就是说子View的大小只能在父View规定的大小范围之内,多出的部分就会被裁剪。

理解了这个就简单了,只要将ViewPager的父控件的clipChildren的值为false就可以实现魅族BannerView的效果了:

<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layerType="hardware"
    android:clipChildren="false"
    >

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="match_parent"
        android:layout_height="180dp"
        android:layout_marginLeft="30dp"
        android:layout_marginRight="30dp"
       />
</RelativeLayout>

我们观察到显示前后页面的图像大小要比当前页要小一点,这又是怎么实现的呢?

这里就需要用到ViewPager.PageTransformer接口:

这个接口是用于ViewPager页面的切换动画,他的功能很强大,他只有一个方法void transformPage(View page, float position);记录了ViewPager滑动过程中各个View的位置变化。
position的变化区间为:[-Infinity,1)、[-1,1]、(1,Infinity);

  1. [-Infinity,1):当前View在屏幕之外。
  2. [-1,1]:这个区间是我们操作当前View的区间
  3. (1,Infinity):已经在屏幕之外

    这里写图片描述
    MIN_SCALF就是前后页面的显示的尺寸。

在测试过程中还发现了个问题,就是刚进入页面的时候会出现当前View的下一个页面不缩松比例的情况,设OffscreenPageLimit(缓存的页面数)为2以上就可以了。
setOffscreenPageLimit(int limit) :设置ViewPager的前后缓存页面数。意思就是说你当前处在ViewPager的哪个页面,他会自动把之后的limit个页面的数据都缓存下来。也是为了增强用户体验。
官方设置limit的最小值为1。如果说不想用这个缓存功能,也就是实现Viewpager 的懒加载、这可能就要修改她的源码,而且还必须是要低版本的v4包才可以修改。

到这里就基本完成了功能了,加上了自动轮播,无限轮播。以下为全部代码:


    private Handler handler = new Handler() {
        @Override
        public void handleMessage (Message msg) {
            super.handleMessage(msg);
            mViewPager.setCurrentItem(mViewPager.getCurrentItem() + 1);
        }
    };

    @Override
    protected void onCreate (Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mViewPager = (ViewPager) findViewById(R.id.viewpager);

        initData();
        initListener();
        startAutoPlay();
    }

    /**
     * 自动播放轮播图
     */
    private void startAutoPlay () {
       Timer timer = new Timer();
        timer.schedule(new TimerTask() {
            @Override
            public void run () {
               handler.sendEmptyMessage(0);
            }
        }, 1000, 1000 * 5);
    }

    /**
     * 初始化事件监听
     */
    private void initListener () {
        mViewPager.setPageTransformer(true, new ViewPager.PageTransformer() {
            @Override
            public void transformPage (View page, float position) {
                if (position < -1) {
                    page.setScaleY(MIN_SCALF);
                } else if (position <= 1) {
                    float scale = Math.max(MIN_SCALF, 1 - Math.abs(position));
                    page.setScaleY(scale);
                } else {
                    page.setScaleY(MIN_SCALF);
                }
            }
        }, View.LAYER_TYPE_NONE);
    }

    /**
     * 初始化图片信息
     */
    private void initImage () {
        imgs[0] = BitmapFactory.decodeResource(getResources(), R.drawable.home01);
        imgs[1] = BitmapFactory.decodeResource(getResources(), R.drawable.home02);
        imgs[2] = BitmapFactory.decodeResource(getResources(), R.drawable.home03);
        imgs[3] = BitmapFactory.decodeResource(getResources(), R.drawable.home04);
        imgs[4] = BitmapFactory.decodeResource(getResources(), R.drawable.home05);
        imgs[5] = BitmapFactory.decodeResource(getResources(), R.drawable.home06);
        imgs[6] = BitmapFactory.decodeResource(getResources(), R.drawable.home07);
        imgs[7] = BitmapFactory.decodeResource(getResources(), R.drawable.home08);
    }

    /**
     * 设置数据
     */
    private void initData () {
        initImage();
        datas = new ArrayList<>();
        //Bitmap bit = BitmapFactory.dec
        for (int i = 0; i < imgs.length; i++) {
            ImageView img = new ImageView(this);
            img.setImageBitmap(imgs[i]);
            img.setScaleType(ImageView.ScaleType.FIT_XY);
            datas.add(img);
        }
        mViewPager.setAdapter(new MyPagerAdapter(datas));
        //设置ViewPager的前后缓存页面数
        mViewPager.setOffscreenPageLimit(2);
        //设置起始位置
     mViewPager.setCurrentItem(1000 - 1000 % datas.size());
    }
}
public class MyPagerAdapter extends PagerAdapter {
    private List<ImageView> mDatas;

    public MyPagerAdapter (List<ImageView> datas) {
        this.mDatas = datas;
    }

    @Override
    public int getCount () {
        if (mDatas != null)
//            return mDatas.size();
            return Integer.MAX_VALUE;
        return 0;
    }

    @Override
    public boolean isViewFromObject (View view, Object object) {
        return view == object;
    }

    @Override
    public void destroyItem (ViewGroup container, int position, Object object) {
        Log.i("MainActivity", "destroyItem: " + position);
        container.removeView((View) object);
    }

    @Override
    public Object instantiateItem (ViewGroup container, int position) {
        Log.i("MainActivity", "instantiateItem: " + position);
        position = position%mDatas.size();
        ImageView img = mDatas.get(position);
        container.addView(img);
        return img;
    }
}

第一次写博客,写的不好请不要见怪^-^

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
现在的APP Banner大多数千篇一律,前几天看到魅族手机上所有魅族自家APP上的Banner效果不错,于是就想着来仿着做一个类似的效果。因此就有了这个库。但是为了使用方便,这个库不仅仅只有仿魅族效果BannerView 来使用,还可以当作普通的BannerView 来使用,还可以当作一个ViewPager 来使用。使用很方便,具体使用方法和API 请看后面的示例。 ---  左图为魅族APP上的Banner效果,右图是高仿效果。MZBannerView 有以下功能:1 . 仿魅族BannerView 效果。2 . 当普通Banner 使用3 . 当普通ViewPager 使用。4 . 当普通ViewPager使用(有魅族Banner效果)5 . 仿某视频网站Banner效果。Demo APKgif图片有点模糊,可以扫描下方二维码下载APK体验相关博客ViewPager系列之 仿魅族应用的广告BannerView更新日志v1.1.1 : 增加按住Banner 停止轮播,松开开始自动轮播的功能v1.1.0 : fix 在从网上获取数据后,banner 显示 造成 ANR 的bug(如果在onCreate()中设置资源显示则没问题)v1.1.2 : fix 更改数据之后,调用setPages重新刷新数据会crush的bugv2.0.0 :1,add: 添加仿魅族Banner效果,中间Page覆盖两边。 -- 2,fix 部分bug: 添加OnPageChangeListener 回调 pisition 不对的bug.DependencyAdd it in your root build.gradle at the end of repositories:allprojects {      repositories {           ...           maven { url 'https://jitpack.io' }      } }Step 2. Add the dependencydependencies {          compile 'com.github.pinguo-zhouwei:MZBannerView:v2.0.0' }自定义属性属性名属性意义取值open_mz_mode是否开启魅族模式true 为魅族Banner效果,false 则普通Banner效果canLoop是否轮播true 轮播,false 则为普通ViewPagerindicatorPaddingLeft设置指示器距离左侧的距离单位为 dp 的值indicatorPaddingRight设置指示器距离右侧的距离单位为 dp 的值indicatorAlign设置指示器的位置有三个取值:left 左边,center 剧中显示,right 右侧显示middle_page_cover设置中间Page是否覆盖(真正的魅族Banner效果)true 覆盖,false 无覆盖效果使用方法1 . xml 布局文件2 . activity中代码:mMZBanner = (MZBannerViewview.findViewById(R.id.banner);              // 设置数据         mMZBanner.setPages(list, new MZHolderCreator() {             @Override             public BannerViewHolder createViewHolder() {                 return new BannerViewHolder();             }         });  public static class BannerViewHolder implements MZViewHolder {         private ImageView mImageView;         @Override         public View createView(Context cont
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值