Android ViewPager小圆点指示器

一个很常用的功能,一个ViewPager会自动滚动,并且有一排小圆点黑和白来指示当前的滚动进度

首先写一个ViewPager的适配器,这里这个适配器为了方便里面的元素全都是ImageView

import android.content.Context;
import android.os.Handler;
import android.support.v4.view.PagerAdapter;
import android.support.v4.view.ViewPager;
import android.view.View;
import android.view.ViewGroup;
import android.view.animation.Interpolator;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.Scroller;

/**
 * Created by Administrator on 2016/2/24.
 */
public class HomeHomeBannerAdapter extends PagerAdapter{
    private Context context;
    private ImageView[] eventImageViews;
    private String[] eventUrls;
    int destWidth,destHeight;

    public HomeHomeBannerAdapter(Context context, String[] eventUrls,int destWidth,int destHeight) {
        this.context = context;
        this.eventUrls = eventUrls;
        this.destHeight = destHeight;
        this.destWidth = destWidth;
        initImageViews();
    }

    /**
     * 初始化viewPager里的几张图
     */
    private void initImageViews(){
        if(eventUrls==null)return;
        eventImageViews = new ImageView[eventUrls.length];
        for (int i=0;i<eventUrls.length;i++) {
            eventImageViews[i] = new ImageView(context);
            eventImageViews[i].setLayoutParams(new LinearLayout.LayoutParams(destWidth, destHeight));
            eventImageViews[i].setPadding(0, 0, 0, 0);
            eventImageViews[i].setScaleType(ImageView.ScaleType.FIT_XY);
            JImageUtils.loadImageFromServerByUrl(context,eventImageViews[i],eventUrls[i]);
        }//显示图片
    }

    @Override
    public void destroyItem(ViewGroup container, int position, Object object) {
        if (eventImageViews != null && eventImageViews.length > position && position >= 0)
            container.removeView(eventImageViews[position]);

    }

    @Override
    public Object instantiateItem(ViewGroup container, int position) {
        container.addView(eventImageViews[position], 0);
        return eventImageViews[position];
    }

    @Override
    public int getCount() {return eventUrls==null?0:eventUrls.length;}

    @Override
    public boolean isViewFromObject(View arg0, Object arg1) {
        return arg0 == arg1;
    }
//ViewPager监听器集成到内部类去
    static public class EventViewPagerChangeListener implements ViewPager.OnPageChangeListener {
        LinearLayout llGuideGroup;
        int oldEventPosition;
        int currentItem;

        public EventViewPagerChangeListener(LinearLayout llGuideGroup){
            this.llGuideGroup = llGuideGroup;
        }

        @Override
        public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {}

        /**
         * 控制小点的白色还是黑色的显示
         * @param position
         */
        public void onPageSelected(int position) {
            llGuideGroup.getChildAt(oldEventPosition).setBackgroundResource(R.drawable.dot_normal);//黑色点
            llGuideGroup.getChildAt(position).setBackgroundResource(R.drawable.dot_focused);//白色点
            oldEventPosition = position;
            currentItem = position;
        }

        @Override
        public void onPageScrollStateChanged(int state) {

        }
    }

    /**
     * 控制viewpager自动滑动的定时任务
     */
    public static class ScrollTask implements Runnable {
        EventViewPagerChangeListener listener;
        ViewPager vpEvent;
        int eventSize;
        Handler handler;
        public ScrollTask(EventViewPagerChangeListener listener,final ViewPager vpEvent, int eventSize){
            this.listener = listener;
            this.vpEvent = vpEvent;
            this.eventSize = eventSize;
            handler = new Handler();
        }
        public void run() {
            if(listener==null||vpEvent==null||eventSize==0)return;
            listener.currentItem = (listener.currentItem + 1) % eventSize;
            Log.i("Alex","currentItem是"+listener.currentItem);
            handler.post(new Runnable() {
                @Override
                public void run() {
                    vpEvent.setCurrentItem(listener.currentItem); // 通过Handler切换图片
                }
            });
        }
    }

    public static class FixedSpeedScroller extends Scroller {
        private static final int mDuration = 400;
        private int eventCount;

        public FixedSpeedScroller(Context context, Interpolator interpolator,int eventCount) {
            super(context, interpolator);
            this.eventCount = eventCount;
        }

        @Override
        public void startScroll(int startX, int startY, int dx, int dy, int duration) {
            if (duration % 100 == 0 && duration > 0) {
                //"现在是自动划"
                if (duration / 100 == eventCount) super.startScroll(startX, startY, dx, dy, 1);//如果是最后一张
                else super.startScroll(startX, startY, dx, dy, mDuration);

            } else {
                // "现在是手动划"
                super.startScroll(startX, startY, dx, dy, 80);
            }
        }
    }
}

在上面适配器的内部类有一个监听器,这个监听器里有一个成员LinearLayout llGuideGroup,这个线性布局里面装有几个小点,下面是这个布局的定义:

<RelativeLayout
        android:id="@+id/rlEvents"
        android:layout_width="match_parent"
        android:layout_height="200dp"
        android:visibility="gone">

        <android.support.v4.view.ViewPager
            android:id="@+id/vpEvent"
            android:layout_width="match_parent"
            android:layout_height="200dp"
            android:background="@color/black666666" />

        <LinearLayout
            android:id="@+id/llGuideGroup"
            android:layout_width="match_parent"
            android:layout_height="10dp"
            android:layout_alignParentBottom="true"
            android:layout_marginBottom="10dp"
            android:layout_centerHorizontal="true"
            android:gravity="center"
            android:orientation="horizontal" >

        </LinearLayout>
</RelativeLayout>

设置小圆点的数量并初始化监听器

/**
     * 向一个线性布局里添加小圆点,具体的控制逻辑在listener里
     * @param llGuideGroup
     * @param count 要添加多少个小圆点
     */
    public EventViewPagerChangeListener addViewPagerDots(LinearLayout llGuideGroup,ViewPager vpEvents,int count){
        if(llGuideGroup==null||vpEvents==null||count<1)return null;
        LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(15, 15);
        lp.leftMargin = 5;
        lp.rightMargin = 5;
        for(int i=0;i<count;i++){
            ImageView imageView = new ImageView(llGuideGroup.getContext());
            imageView.setLayoutParams(lp);
            imageView.setBackgroundResource(i==0?R.drawable.dot_focused:R.drawable.dot_normal);
            llGuideGroup.addView(imageView);
        }
        //控制小圆点显示的监听器
        EventViewPagerChangeListener listener = new EventViewPagerChangeListener(llGuideGroup);
        vpEvents.addOnPageChangeListener(listener);
        return listener;
    }

通过多线程实现定时切换页面

HomeHomeBannerAdapter.EventViewPagerChangeListener listener = producer.addViewPagerDots(holder.llGuideGroup,holder.viewPager,eventUrls.length);//添加用于指示的小圆点
                    // 当Activity显示出来后,每3秒钟切换一次图片显示
                    ScheduledExecutorService scheduledExecutorService = Executors.newSingleThreadScheduledExecutor();
                    scheduledExecutorService.scheduleAtFixedRate(new HomeHomeBannerAdapter.ScrollTask(listener,holder.viewPager,eventUrls.length), 3, 3, TimeUnit.SECONDS);

设置自定义滚动器降低viewPager的切换速度

/**
     * 给ViewPager设置自定义的滚动器,降低默认的滚动速度
     * @param vpEvent
     */
    public void setViewPagerScroller(ViewPager vpEvent){
        if(vpEvent==null)return;
        Field mField;
        Scroller mScroller;
        try {
            mField = ViewPager.class.getDeclaredField("mScroller");
            mField.setAccessible(true);
            mScroller = new HomeHomeBannerAdapter.FixedSpeedScroller(vpEvent.getContext(), new AccelerateInterpolator(),vpEvent.getChildCount());
            try {
                mField.set(vpEvent, mScroller);
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        }
    }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android实现仿网易新闻的顶部导航指示器

    我们知道,页面导航器(Navigator)在几乎所有的项目中都会用到,平时大多数时候为了节省时间,都会直接在github上面拿别人的开源项目来用,最近自己在复习自定义View,就尝试封装了一下,源码参考项目PagerSlidingTabStrip 大家先来看一下效果图 基于文字的页面导航器 基于图片的页面导航器 使用方法 主要步骤分为三步 1)在xml文件里面 <com.xujun.viewpagertabindicator.TabPagerIndicator android:id="@+

  • Android自定义ViewPagerIndicator实现炫酷导航栏指示器(ViewPager+Fragment)

    ViewPagerIndicator导航栏指示器运行效果: 实现这个效果,我是看了很多大神写的博客和视频后自己敲的,欢迎指正 github地址:https://github.com/dl10210950/TabViewPagerIndicator 自定义一个ViewPagerIndicator 自定义一个Indicator继承LinearLayout,在构造方法里面设置画笔的一些属性 public ViewPagerIndicator(Context context, AttributeSet

  • Android自定义圆点指示器

    本文实例为大家分享了Android自定义圆点指示器的具体代码,供大家参考,具体内容如下 先上效果图 大概思路就是自定义View 从左至右绘制圆点 然后在ViewPager的OnPageChangeListener中设置当前页面的圆点 下面是代码 先定义属性 <resources> <attr name="selectedColor" format="color"/> <attr name="unselectedColor&qu

  • Android实现基于ViewPager的无限循环自动播放带指示器的轮播图CarouselFigureView控件

    最近用到需要无限轮播自动播放的轮播轮播图,网上感觉都有这样那样的问题,于是自己写了一个通用的控件CarouselFigureView. 特点: 1.可以轮播view可以自己定义,不一定是要是ImageView2.指示器默认显示,但是可以隐藏3.可以设置指示器的颜色.间距.大小 4.有基础的可以自己修改代码改变指示器位置,这个应该不难5.可以自己开启和关闭自动轮播,开启轮播的时候可以设置轮播时间间隔,默认3000毫秒 我们先来看看效果图: 然后来看看使用代码 xml代码 <?xml version

  • Android之IphoneTreeView带组指示器的ExpandableListView效果

    之前实现过一次这种效果的ExpandableListView:http://www.jb51.net/article/38482.htm,带效果比较挫,最近,在参考联系人源码PinnedHeaderListView,以及网上各位大侠的源码,封装了一个效果最好,而且使用最简单的IphoneTreeView,下面先看看效果图:  首先让我们看看封装得比较完善的IphoneTreeView: 复制代码 代码如下: public class IphoneTreeView extends Expandab

  • Android progressbar实现带底部指示器和文字的进度条

    本文实例为大家分享了Android实现带指示器和文字的进度条,供大家参考,具体内容如下 根据项目要求需要实现以下效果: 列出源码: public class TextProgressBar extends LinearLayout { String text; Paint mPaint; private Rect textRect; private Bitmap bitmap; private ProgressBar progressBar; int progress; int proWidth

  • Android之带group指示器的ExpandableListView(自写)

    我们都知道Android缺省的ExpandableListView的group header无法固定在界面上,当向下滚动后,不能对当前显示的那些child 指示出它们归属于哪个group,在网上搜了很多关于仿手机QQ好友分组效果的ExpandableListView,发现都不尽如意,于是乎在别人的基础上改进了一点点,其实原理还是差不多的,只是增加了往上挤出去的动画效果,而且更加简单,只不过还是没有完全到达跟QQ一样的效果,希望有高手能实现更加逼真的效果,下面我们先看看效果图:  我这里没有把Ex

  • Android开发实现的ViewPager引导页功能(动态加载指示器)详解

    本文实例讲述了Android开发实现的ViewPager引导页功能(动态加载指示器).分享给大家供大家参考,具体如下: 先看效果图咯~ 现在几乎每个App都会有引导页,是不是感觉很炫很厉害,所以就想做出来一个学习一下~让自己的App看起来更加的美观~ 现在来分析一下: 这个引导页可以分为俩部分~ 1.小红点--来提醒这是第几页了~ 2."开始体验"这个Button--可以进入主界面,但是要控制这个Button只能在最后一页出现 布局的话使用相对布局~ 那现在来看看布局吧: activi

  • Android TabLayout设置指示器宽度的方法

    anroid 5.0 Design  v7 包中引用了TabLayout 简单快速的写出属于自己的Tab切换效果 如图所示: 但是正常使用中你发现无法设置tablayout指示器的宽度.查看源码你会发现设计师将指示器的宽度设置成TabView最大的宽度.并且设计师并没有给我们暴漏出接口,这导致有时使用TabLayout无法满足一些产品设计要求,这么好的组件无法使用还需要自定义费时费力.这个时候我们可以通过反射机制拿到TabLayout中的指示器对象对它的宽度进行处理就可以满足我们的要求:具体代码

  • Android实现引导页的圆点指示器

    在App引导界面通常有引导界面提示小圆点,我们使用一个集成的类 来整体封装实现: 使用的方法: 首先在 XML布局中引入这个自定义的控件: <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent&quo

随机推荐