dev控件swiftplot图滚动方法_无限轮播图使用Scroller就这么简单

前言

这几天又拾起老本行,复习复习Android,才发现忘的差不多了,上午做了一个小Demo,配合Scroller做了一个轮播图,效果如下,但是不知为何,录制的GIF成这样,凑乎一下看看。

beda6d4a89f33bea64d2893995c41859.gif

原理是继承ViewGroup,然后自己摆放子View,也就是摆放在一条线上,开启一个定时器,每隔X秒通过Scroller进行滚动,当然还有处理Touch事件,在手指按下的时候定时器停止,抬起的时候定时器重新启动。

Scroller简介

我们知道View中提供了scrollTo()和scrollBy()两个方法用来滚动,也就是说任何一个控件都是可以滚动的,他两的区别在于,scrollBy()让View相对于当前的位置滚动一段距离,scrollTo()则是让View相对于初始的位置滚动一段距离。

注意这里的滚动是内容的滚动,不是自身位置。

还有一个类是Scroller,用于处理滚动效果的工具类,但是这个类并不能直接使View滚动,而是提供计算滚动的值,我们拿到这个值在调用scrollTo或scrollBy进行滚动。

Scroller的两个重要方法是startScroll和computeScrollOffset,startScroll()方法是用来初始化滚动数据的,第一个参数是滚动开始时X的坐标,第二个参数是滚动开始时Y的坐标,如果我们的控件只支持X轴滚动,那么Y永远是0,,第三个参数是横向滚动的距离,第四个参数是纵向滚动的距离,通常这个方法后面会调用invalidate()方法来刷新界面。

最后就是重写computeScroll()方法,这个方法根据注释的意思是在父级请求子级更新其mScrollX值时候调用,在这个方法下通过不断调用Scroller的computeScrollOffset()方法来判断滚动操作是否完成了,如果还没完成的话,那就继续调用scrollTo()方法,传入Scroller的curX和curY,这样就完成了内容的平滑滚动。

另外Scroller还支持Interpolator,使滚动更具有效果。

实现

package com.example.androiddemo;

import android.content.Context;
import android.os.Handler;
import android.os.Message;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Scroller;

import androidx.annotation.NonNull;

import java.util.Timer;
import java.util.TimerTask;

public class BannerView extends ViewGroup {
    private static final int INVALIDATE_MSG = 1000;

    private static final int DEFAULT_DURATION = 1000;
    private static final int DEFAULT_PERIOD = 2000;
    private static final String TAG = "TAG";

    private int mPeriod;
    private int mDuration;

    private boolean isTouch = false;

    private int mDownX;

    private Scroller mScroller;
    private Timer mTimer;
    private Handler handler = new Handler() {
        @Override
        public void handleMessage(@NonNull Message msg) {
            super.handleMessage(msg);
            switch (msg.what) {
                case INVALIDATE_MSG:
                    nextItem();
                    break;
            }
        }
    };

    public BannerView(Context context) {
        super(context);
        init();
    }

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

    public BannerView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    private void init() {
        mPeriod = DEFAULT_PERIOD;
        mDuration = DEFAULT_DURATION;

        mScroller = new Scroller(getContext());

        startTimer(0);
    }

    private void startTimer(int delay) {
        mTimer = new Timer();
        mTimer.schedule(new TimerTask() {
            @Override
            public void run() {
                if (!isTouch)
                    handler.sendEmptyMessage(INVALIDATE_MSG);
            }
        }, delay, mPeriod);
    }

    private void nextItem() {
        mScroller.startScroll(getScrollX(), 0, getMeasuredWidth(), 0, mDuration);
        invalidate();
    }

    @Override
    protected void onFinishInflate() {
        super.onFinishInflate();

    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        int left = 0;

        for (int i = 0; i             View childAt = getChildAt(i);
            childAt.layout(left, t, r + left, b);
            left += r;
        }
    }

    @Override
    public void computeScroll() {
        if (mScroller.computeScrollOffset()) {
            scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
            invalidate();
        }
        if (mScroller.isFinished() && getScrollX() == getMeasuredWidth() * (getChildCount() - 1)) {
            Log.i(TAG, "computeScroll: " + getScrollX());
            scrollTo(0, 0);
        }
    }


    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mDownX = (int) event.getX();
                mTimer.cancel();
                break;
            case MotionEvent.ACTION_UP:
                int targetIndex = (getScrollX() + getWidth() / 2) / getWidth();
                int dx = targetIndex * getWidth() - getScrollX();
                mScroller.startScroll(getScrollX(), 0, dx, 0, mDuration);
                invalidate();
                startTimer(mPeriod);
                break;
            case MotionEvent.ACTION_MOVE:
                scrollBy((int) (mDownX - event.getX()), 0);
                mDownX = (int) event.getX();
                break;
        }
        return true;
    }
}

<?xml  version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:background="#FFFFFF"android:paddingLeft="10dp"android:paddingRight="10dp"android:paddingTop="20dp"tools:context=".MainActivity">

    <RelativeLayoutandroid:layout_width="match_parent"android:layout_height="150dp">
        <com.example.androiddemo.BannerViewandroid:layout_width="match_parent"android:layout_height="match_parent">
            <ImageViewandroid:scaleType="centerCrop"android:src="@drawable/banner1"android:layout_width="match_parent"android:layout_height="match_parent">ImageView>

            <ImageViewandroid:scaleType="centerCrop"android:src="@drawable/banner2"android:layout_width="match_parent"android:layout_height="match_parent">ImageView>
            <ImageViewandroid:scaleType="centerCrop"android:src="@drawable/banner3"android:layout_width="match_parent"android:layout_height="match_parent">ImageView>
            <ImageViewandroid:scaleType="centerCrop"android:src="@drawable/banner4"android:layout_width="match_parent"android:layout_height="match_parent">ImageView>
            <ImageViewandroid:scaleType="centerCrop"android:src="@drawable/banner1"android:layout_width="match_parent"android:layout_height="match_parent">ImageView>
        com.example.androiddemo.BannerView>

    RelativeLayout>

LinearLayout>

- END -

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/558589.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JVM——CPU缓存架构与Java 内存模型

导航一、CPU缓存架构与一致性协议1.1 CPU缓存架构1.2 缓存行与伪共享问题1.3 MESI 缓存一致性协议1.4 伪共享的解决办法二、JMM Java 内存模型2.1 JMM 简介2.2 原子性、可见性、有序性2.3 八大内存交互操作2.4 happens-before 原则一、CPU缓存架构与一致性协议 1.1 CPU缓存架构…

蓝牙连接不上车要hfp_汽车上hfp是什么意思

汽车上hfp的全称是Hands-free-Profile&#xff0c;是一款让蓝牙设备控制电话&#xff0c;实现接听、挂断、拒接、语音拨号等功能的软件。hfp车载蓝牙功能使用上十分方便&#xff0c;可以有效地降低安全事故&#xff0c;在手机和车载蓝牙连接之后&#xff0c;在打电话的状况下可…

python3中的 input函数_Python3中的input函数

{"moduleinfo":{"card_count":[{"count_phone":1,"count":1}],"search_count":[{"count_phone":4,"count":4}]},"card":[{"des":"阿里技术人对外发布原创技术内容的最大平台&…

python转二进制字符串_python如何将二进制串(UTF-8)转换为字符串?

已知一个二进制串(UTF-8编码格式)列表(每一项为一字节8bit):b[01010000, 01111001, 01110100, 01101000, 01101111, 01101110, 11100101, 10100100, 10100111, 11100110, 10110011, 10010101]#即字符串"Python大法"如何将其转换为字符串呢?string ""for …

python中reshape_Numpy之reshape()使用详解

如下所示&#xff1a;Numpy中reshape的使用方法为:numpy.reshape(a, newshape, orderC)参数详解&#xff1a;1.a: type:array_like(伪数组&#xff0c;可以看成是对数组的扩展&#xff0c;但是不影响原始数组。)需要reshape的array2.newshape:新的数组新形状应与原形状兼容。如…

vue 手机端路由切换滑动_vue移动端router-view嵌套实现底部导航切换

路由使用 vue-router组件库使用 vue-ydui效果图&#xff1a;登录.png主界面-首页.png主界面-办事大厅.png项目结构图片.png大体流程图片.png路由代码import Vue from vueimport Router from vue-routerimport home from ../page/homeimport office from ../page/officeimport m…

bs4 乱码_Python BeautifulSoup中文乱码问题的2种解决方法

解决方法一&#xff1a;使用python的BeautifulSoup来抓取网页然后输出网页标题,但是输出的总是乱码,找了好久找到解决办法,下面分享给大家首先是代码复制代码 代码如下:from bs4 import BeautifulSoupimport urllib2url //www.jb51.net/page urllib2.urlopen(url)soup Beaut…

白噪声检测_科学家尝试用智能扬声器的白噪声来监测婴儿的呼吸运动

华盛顿大学的一支研究团队&#xff0c;刚刚介绍了他们开发的一种新型智能扬声器技术。这种设备能够借助白噪声来安抚熟睡的婴儿&#xff0c;并监测他们的呼吸和运动。具体说来是&#xff0c;通过智能扬声器发出的白噪声&#xff0c;原型设备能够将之与生命体征监测仪的数据相匹…

最大值_285期 博最大值2路,已经箭在弦上!

往期数据P-5掉码 跨度 和尾 012断路 余数和 位数86072 1 8 4 200 断1路2 5 对214对 双双双79703 0 2 3 101 断2路2 4 对215对 单单单62386 0 4 1 020 断1路2 4 错216对 双双单71903 0 8 7 110 断2路2 5 错217对 单单单64838 0 4 8 012 来3路3 4 错218对 双双双02052 0 2 2 020 …

商品领域ddd_为 Gopher 打造 DDD 系列:领域模型-资源库

前言&#xff1a; 作为领域模型中最重要的环节之一的Repository&#xff0c;其通过对外暴露接口屏蔽了内部的复杂性&#xff0c;又有其隐式写时复制的巧妙代码设计&#xff0c;完美的将DDD中的Repository的概念与代码相结合&#xff01;Repository资源库通常标识一个存储的区域…

mysql5.7主从全备恢复_Mysql5.7—运维常用备份方式(超全)

小生博客&#xff1a;http://xsboke.blog.51cto.com小生 Q Q&#xff1a;1770058260-------谢谢您的参考&#xff0c;如有疑问&#xff0c;欢迎交流一、 Mysqldump备份结合binlog日志恢复使用mysqldump进行全库备份&#xff0c;并使用binlog日志备份&#xff0c;还原时&#xf…

客户端连接mysql 自动断开_MySql连接空闲8小时自动断开的原因及连接池配置方法...

数据库连接超时时间查询非交互式超时时间&#xff0c;如 JDBC 程序show global variables like wait_timeout;交互式超时时间&#xff0c;如数据库工具show global variables like interactive_timeout;MySQL服务器默认的“wait_timeout”是28800秒即8小时&#xff0c;意味着如…

docker 运行容器_Docker之运行 Django 容器

首先此篇笔记默认你已经安装好了 Docker&#xff0c;并了解 Docker 的基础概念&#xff0c;诸如镜像、容器、以及他们之间的关系等。如果不太了解&#xff0c;等我回头了解清楚以后&#xff0c;可以再写一篇文章阐述一下。&#xff08;狗头当然&#xff0c;对于这篇文章&#x…

mysql类似的数据库_MemSQL学习笔记-类似MySQL的数据库

http://gigaom.com/cloud/ex-facebookers-launch-memsql-to-make-your-database-fly/-- 多主-从http://www.mysqlops.com/2012/02/14/diy_multi_master_replication.htmlhttp://www.cnblogs.com/liuhao/archive/2012/06/26/2563702.html前facebook员工和前微软sql server工程师…

mysql8.0与mysql7.0_MySQL 5.7 vs 8.0,哪个性能更牛?

测试mysql5.7和mysql8.0分别在读写&#xff0c;选定&#xff0c;只写模式下不同并发时的性能(tps&#xff0c;qps)最早测试使用版本为mysql5.7.22和mysql8.0.15sysbench测试前先重启mysql服务&#xff0c;并清除os的缓存(避免多次测试时命中缓存)每次进行测试都是新生成测试数据…

linux+mysql+导出备份_Linux系统MySQL备份的导入导出的具体分析

问题描述如何对 ECS Linux 系统中的 MySQL 进行备份的导入和导出。处理办法MySQL 备份的导出MySQL 备份的导入MySQL 备份的导出注意&#xff1a;如果您使用的是帮助中心的一键环境配置&#xff0c;那么 MySQL 的安装目录是 /alidata/server/mysql。如果您将 MySQL 安装到其他目…

springmvc使用requestmapping无法访问控制类_研究人员称人类使用的新烟碱类杀虫剂让蜜蜂无法入睡...

来自布里斯托尔大学的科学家进行了研究&#xff0c;显示常见的杀虫剂可以阻止蜜蜂和苍蝇睡个好觉。就像人类一样&#xff0c;许多昆虫也需要睡眠才能正常工作。然而&#xff0c;如果它们接触过新烟碱类杀虫剂&#xff0c;它们的睡眠就会受到影响&#xff0c;新烟碱类杀虫剂是一…

linux 监控mysql脚本_Linux系统MySQL主从同步监控shell脚本

操作系统&#xff1a;CentOS系统目的&#xff1a;定时监控MySQL数据库主从是否同步&#xff0c;如果不同步&#xff0c;记录故障时间&#xff0c;并执行命令使主从恢复同步状态1、创建脚本文件vi /home/crontab/check_mysql_slave.sh #编辑&#xff0c;添加下面代码#!/bin/sh…

mysql 1066解决方法_MySQL查询语法帮助:错误#1066-表格/别名不唯...

我有四个表,user,user_billingprofile,user_shippingprofile和user_address.用户&#xff1a;userId,dateCreateduser_billingprofile&#xff1a;userId,地址user_shippingprofile&#xff1a;userId,地址user_address&#xff1a;随机地址废话这是我必须一目了然地获取用户帐…

python创建新进程_Python:创建新进程

我是Python新手。我应该创建一个有多个菜单的GUI。单击特定菜单时&#xff0c;应启动一个新进程&#xff0c;并且不应挂起用户界面。但我不能做到这一点。在网上搜索之后&#xff0c;我做了一个类似的代码。在在这段代码中&#xff0c;我的目标是使“print deep”语句处于活动状…