Android自定义View使用canvas实现轮播图效果

1.功能分析
1.1 左右滑动切换图片,并且实现循环切换。
1.2 自动切换图片
1.3 导航圆点跟随轮播变更
1.4 点击图片,实现监听反馈
1.5 图片需要适配屏幕,按定义宽高显示

2.代码实现
2.1 实现原理
每次加载显示需要3张图片,并且偏移至左中右三个位置,不断地重绘view,修改偏移值,达到切换图片效果。
2.2 代码实现
创建自定义View类CarouselFigure,在onMeasure方法中,获取容器view的宽度,这里使用默认显示图片宽度与view容器宽度比值,作为适配比,然后确定容器view显示高度。

@Override

protected void onMeasure(intwidthMeasureSpec,intheightMeasureSpec) { 
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);//获取view宽高
        mWidth = MeasureSpec.getSize(widthMeasureSpec);
        mHeight = MeasureSpec.getSize(heightMeasureSpec);//默认showindex对应图片适配后宽高,做轮播图整体宽高
        if(imgList!=null&& imgList.size()>0){
               Bitmap bitmap = imgList.get(showIndex);floatscale = (float)mWidth /         bitmap.getWidth();
               mHeight = (int) (scale * bitmap.getHeight());
          }
        setMeasuredDimension(mWidth,mHeight);
}

手指左右滑动图片时候,获取横向手指偏移量,偏移量具有方向,向左为负,向右为正。向左偏移,左中右图片动态偏移量分别为,-mWidth+offset ,offset,mWidth+offset。showIndex表示显示图片imgList中索引号,pre_show_index为左图索引号,next_show_index为右图索引号。通过变更偏移和索引号,切换显示图片。

 @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        //判断偏移之后在临界值
        float min_x = mWidth-Math.abs(offset);
        //小于临界值时,改变显示图片索引值
        if(min_x<MIN_WIDTH_OFFSET_VALUE){
            //左移动
            if(offset<0 ){
                showIndex ++;
            }
            if(offset>0 ){
                showIndex--;
            }
            //回归
            reSetValue();
        }
        //初始化showIndex
        initShowIndex();
        //构画左中右三图
        handleImgWnH(pre_show_index);
        handleImgWnH(showIndex);
        handleImgWnH(next_show_index);
        canvas.drawBitmap(imgList.get(pre_show_index),-mWidth+offset,0,mPaint);
        canvas.drawBitmap(imgList.get(showIndex),offset,0,mPaint);
        canvas.drawBitmap(imgList.get(next_show_index),mWidth+offset,0,mPaint);
        //构画导航圆点
        drawCycle(canvas);
    }

一开始触摸图片,记录当前触摸坐标,标记为初始坐标startPoint,移动时获取实时坐标,计算手势滑动方向与水平方向夹角正切值,判断是否是水平滑动,如果是横向水平滑动,则记录滑动偏移量,调用invalidate进行重绘

@Override
    public boolean onTouchEvent(MotionEvent event) {
        //正在刷新禁止触摸
        if(isRefreshing){return true;}
        //正在自动轮播时,不能触发
        if(isAutoSliding||isGestureSliding){return true;}
        int action = event.getAction();
        switch (action){
            case MotionEvent.ACTION_DOWN:
                //关闭自动轮播
                isAllowAutoSlid = false;
                startPoint.set(event.getX(),event.getY());//记录初始值
                return true;
            case MotionEvent.ACTION_MOVE:
                float min_x = Math.abs(startPoint.x - event.getX());
                float min_y = Math.abs(startPoint.y - event.getY());
                //获取正切值
                float angle_tan_value = min_y / min_x;
                if(angle_tan_value<MIN_ANGLE_TAN_VALUE){ //横向滑动
                    min_x = event.getX() - startPoint.x ;
                    offset = old_offset + min_x;
                    nowPoint.set(event.getX(),event.getY());//记录当前坐标
                    invalidate();
                }
                return true;
            case MotionEvent.ACTION_UP:
                old_offset = offset;//记录上一次偏移量
                //滑动手势产生图片切换效果
                post(slidImgRunnable);
                //重新开启自动轮播
                isAllowAutoSlid = true;
                if(offset==0) {
                    onTopImageClickListeners.onClick(showIndex);
                }
                return true;
        }
        return false;
    }

此时,已经实现让图片跟随手指偏移。但是图片并不会自动保持滑动惯性,使自己显示完全。 因此,需要在ACTION_UP处补充一个方法,让图片继续完成剩下偏移。此处使用线程slidImgRunnable

private Runnable slidImgRunnable = new Runnable() {
        @Override
        public void run() {
            float abs_offset = Math.abs(offset);
            //向左移动
            if(offset<0) {
                if(abs_offset<ALLOW_SLID_IMG_OFFSET){  //允许产生滑动的偏移量,否则图片回归原位置
                    offset += SLID_IMG_INTERVAL_OFFSET;
                    if(offset>0){
                        reSetValue();
                    }
                }else {
                    offset += -SLID_IMG_INTERVAL_OFFSET;
                }
            }else if(offset>0){ //向右移动
                if(abs_offset<ALLOW_SLID_IMG_OFFSET){ //允许产生滑动的偏移量,否则图片回归原位置
                    offset += -SLID_IMG_INTERVAL_OFFSET;
                    if(offset<0){
                        reSetValue();
                    }
                }else {
                    offset += SLID_IMG_INTERVAL_OFFSET;
                }
            }
            //当滑动之后 offset重置为0 结束循环
            if(abs_offset==0){
                isGestureSliding = false;
                return;
            }
            isGestureSliding = true;
            invalidate();
            postDelayed(slidImgRunnable,SLID_IMG_INTERVALS);
        }
    };

无论向左还是向右移动,offset 绝对值都在增大,并且offset区间在0~mWidth之间,所以,只要设定一个递增偏移常量,不断循环执行修改偏移量和重绘,就可以让图片自动完成偏移。
自动轮播效果,需要另启一循环线程执行。autoSlidImg决定轮播周期,autoSlidImgRunnable 完成偏移量递增,和重绘view
开启自动轮播

  /**
     * 开启自动轮播
     */
    private void autoSlidImg()  {
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    while (true){
                        Thread.sleep(4000);//每4秒轮播一次
                        if(isAllowAutoSlid){
                            post(autoSlidImgRunnable);
                        }
                    }
                }catch (Exception e ){
                }
            }
        }).start();
    }

    /**
     * 实现自动轮播效果
     */
    private Runnable autoSlidImgRunnable = new Runnable() {
        @Override
        public void run() {
            offset += -SLID_IMG_INTERVAL_OFFSET;
            if (isNextCycle) {//判断是否可以继续产生偏移,完成一次轮播后退出
                offset=0;
                isNextCycle=false;
                isAutoSliding = false;
                return;
            }else{
                isAutoSliding = true;//正在轮播滑动
            }
            invalidate();
            postDelayed(autoSlidImgRunnable, SLID_IMG_INTERVALS);
        }
    };

定义接口返回点击显示图片监听

public interface OnTopImageClickListeners{
         void onClick(int showIndex);
}

carouselFigure.setOnTopImageClickListeners(new  CarouselFigure.OnTopImageClickListeners() {
        @Override
        public void onClick(int showIndex) {
            Log.i("tag","index: "+showIndex);
        }
});

github分享地址
CSDN下载地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,464评论 25 707
  • 主要思路 1.我们需要自定义一个继承自FrameLayout的布局,利用FrameLayout布-局的特性(在同一...
    ZebraWei阅读 2,257评论 0 5
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,019评论 4 62
  • 今天是朗轩的生日,晚上邀请了小伙伴一起玩。 昨天给他买了他一直朝朝暮暮的佩琦和乔治,因为从网上拍的,到货后打开包装...
    多金阅读 173评论 0 0
  • 玉の緒のたえてみじかき命もて年月ながき恋もするかな
    松本静阅读 222评论 0 0