android画廊无限轮播,ViewPager无限循环实现画廊式banner

android画廊无限轮播,ViewPager无限循环实现画廊式banner先看一下效果两边显示上一个和下一个item部分布局,可以自动滚动实现:布局主要属性:android:clipChildren=”false”//允许子布局超出父布局显示xml代码:Adapter没什么特别的只要getCount()返回一个很大的值如:returnInteger.MAX_VALUE;适配器代码:packagecom.guzhc.module_demo;importandr…

大家好,又见面了,我是你们的朋友全栈君。

先看一下效果

02137c01d3494d157819fc344857c68a.png

两边显示上一个和下一个item部分布局,可以自动滚动

实现:

布局

主要属性:android:clipChildren=”false” //允许子布局超出父布局显示

xml代码:

Adapter

没什么特别的只要 getCount()返回一个很大的值如:return Integer.MAX_VALUE;

适配器代码:

package com.guzhc.module_demo;

import android.content.Context;

import android.view.View;

import android.view.ViewGroup;

import android.view.ViewParent;

import android.widget.ImageView;

import android.widget.TextView;

import androidx.annotation.NonNull;

import androidx.viewpager.widget.PagerAdapter;

import com.bumptech.glide.Glide;

import java.util.ArrayList;

import java.util.List;

/**

* @author : GuZhC

* @date : 2019/6/1 9:46

* @description : 顶部banner viewPager 适配器

*/

public class ShortViewoViewPagerAdapter extends PagerAdapter {

private final Context context;

private final ListmData;

private PagerClick pagerClick;

public ShortViewoViewPagerAdapter(ListmData, Context context) {

this.mData = mData;

this.context = context;

}

@Override

public int getCount() {

return Integer.MAX_VALUE;

}

@Override

public void destroyItem(ViewGroup container, int position,

Object object) {

//Warning:不要在这里调用removeView

}

@Override

public boolean isViewFromObject(@NonNull View view, @NonNull Object object) {

return view == object;

}

@Override

public Object instantiateItem(ViewGroup container, int position) {

position %= mData.size();

final String data = mData.get(position);

View view = View.inflate(container.getContext(),R.layout.demo_item_short_video_viewpager, null);

ImageView img = view.findViewById(R.id.demo_img_short_video_item);

TextView tvContent = view.findViewById(R.id.demo_tv_short_video_item);

tvContent.setText(data);

Glide.with(context)

.load(“https://timgsa.baidu.com/timg?image&quality=80&size=b9999_10000&sec=1559367407524&di=2a7d6a1a12707287e908d61922a637c2&imgtype=0&src=http%3A%2F%2Fpic72.nipic.com%2Ffile%2F20150715%2F9448607_192612583000_2.jpg”)

.into(img);

//对ViewPager页号求模取出View列表中要显示的项

if (position < 0) {

position = mData.size() + position;

}

//如果View已经在之前添加到了一个父组件,则必须先remove,否则会抛出IllegalStateException。

ViewParent vp = view.getParent();

if (vp != null) {

ViewGroup parent = (ViewGroup) vp;

parent.removeView(view);

}

container.addView(view);

return view;

}

}

调用

注意:两边的item滑动没有效果吗,需要将容器的触摸事件反馈给ViewPager

//设置banner

shortViewoViewPagerAdapter = new ShortViewoViewPagerAdapter(mData, getContext());

View bannerView = LayoutInflater.from(getContext()).inflate(R.layout.demo_layout_short_video_banner, null);

viewPagerBanner = bannerView.findViewById(R.id.demo_vp_short_video_top);

LinearLayout mViewPagerContainer = bannerView.findViewById(R.id.demo_ll_short_video_vp_root);

viewPagerBanner.setAdapter(shortViewoViewPagerAdapter);

viewPagerBanner.addOnPageChangeListener(this);

//将容器的触摸事件反馈给ViewPager

mViewPagerContainer.setOnTouchListener(new View.OnTouchListener() {

@Override

public boolean onTouch(View v, MotionEvent event) {

return viewPagerBanner.dispatchTouchEvent(event);

}

});

无限循环:方式很多,这里用的Handler实现

设置viewPager.addOnPageChangeListener(this)实现接口重写方法:实现滑动时候暂停自动滑动,停止的时候开启

重写方法代码:

@Override

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

handler.sendMessage(Message.obtain(handler, ImageHandler.MSG_PAGE_CHANGED, position, 0));

}

@Override

public void onPageSelected(int position) {

}

@Override

public void onPageScrollStateChanged(int state) {

switch (state) {

//滑动中

case ViewPager.SCROLL_STATE_DRAGGING:

handler.sendEmptyMessage(ImageHandler.MSG_KEEP_SILENT);

break;

//未滑动

case ViewPager.SCROLL_STATE_IDLE:

handler.sendEmptyMessageDelayed(ImageHandler.MSG_UPDATE_IMAGE, ImageHandler.MSG_DELAY);

break;

default:

break;

}

}

Handler代码:

public class ImageHandler extends Handler {

/**

* 请求更新显示的View。

*/

protected static final int MSG_UPDATE_IMAGE = 1;

/**

* 请求暂停轮播。

*/

protected static final int MSG_KEEP_SILENT = 2;

/**

* 请求恢复轮播。

*/

protected static final int MSG_BREAK_SILENT = 3;

/**

* 记录最新的页号,当用户手动滑动时需要记录新页号,否则会使轮播的页面出错。

* 例如当前如果在第一页,本来准备播放的是第二页,而这时候用户滑动到了末页,

* 则应该播放的是第一页,如果继续按照原来的第二页播放,则逻辑上有问题。

*/

protected static final int MSG_PAGE_CHANGED = 4;

//轮播间隔时间

protected static final long MSG_DELAY = 5000;

//使用弱引用避免Handler泄露.

private WeakReferenceweakReference;

private int currentItem = 0;

protected ImageHandler(WeakReferencewk){

weakReference = wk;

}

@Override

public void handleMessage(Message msg) {

super.handleMessage(msg);

ShortVedioFragmet shortVedioFragmet = weakReference.get();

if (shortVedioFragmet==null){

//Activity已经回收,无需再处理UI了

return ;

}

//检查消息队列并移除未发送的消息,这主要是避免在复杂环境下消息出现重复等问题。

if (shortVedioFragmet.handler.hasMessages(MSG_UPDATE_IMAGE)){

shortVedioFragmet.handler.removeMessages(MSG_UPDATE_IMAGE);

}

switch (msg.what) {

case MSG_UPDATE_IMAGE:

currentItem++;

shortVedioFragmet.viewPagerBanner.setCurrentItem(currentItem);

//准备下次播放

shortVedioFragmet.handler.sendEmptyMessageDelayed(MSG_UPDATE_IMAGE, MSG_DELAY);

break;

case MSG_KEEP_SILENT:

//只要不发送消息就暂停了

break;

case MSG_BREAK_SILENT:

shortVedioFragmet.handler.sendEmptyMessageDelayed(MSG_UPDATE_IMAGE, MSG_DELAY);

break;

case MSG_PAGE_CHANGED:

//记录当前的页号,避免播放的时候页面显示不正确。

currentItem = msg.arg1;

break;

default:

break;

}

}

}

如果是viewpager嵌套Fragment的界面中使用可以通过setUserVisibleHint(boolean isVisibleToUser)方法这样开启和暂停:

setUserVisibleHint()方法会在fragment显示和隐藏的时候被调用

@Override

public void setUserVisibleHint(boolean isVisibleToUser) {

super.setUserVisibleHint(isVisibleToUser);

if(isVisibleToUser) {

//相当于OnResume(),可以做相关逻辑

//开始轮播效果

handler.sendEmptyMessageDelayed(ImageHandler.MSG_UPDATE_IMAGE, ImageHandler.MSG_DELAY);

}else {

//相当于OnPause()

//暂停轮播效果

handler.sendEmptyMessage(ImageHandler.MSG_KEEP_SILENT);

}

}

在Activity使用 可以在onResum() 和onPause()开启可暂停。

至此完成。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/129964.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • Linux的#和$区别[通俗易懂]

    Linux的#和$区别[通俗易懂]【#】代表root权限【$】代表普通用户如果更改了/etc/profile,或~/.bashrc等文档,可以用任何符号来代替它们。linux窗口下的【root@locate~】其中的【~】代

  • Codeblocks中文乱码解决方法。[通俗易懂]

    Codeblocks中文乱码解决方法。[通俗易懂]Codeblocks中文乱码解决方法。 如需安装包请后台留言!!Codeblocks中文乱码解决方法:特别提示:出现中文乱码情况才执行以下操作,未出现请勿随意修改!!!!打开Codeblocks-&gt;设置-&gt;编辑器:然后点击Encodingsettings-&gt;选择编码-&gt;选择UTF-8-…

  • C语言判断回文字符串(指针)

    C语言判断回文字符串(指针)东北大学在线编程社区problem1678题目描述:编写函数:intfun(char*p),功能是判断一个字符串是否是回文字符串(提示:回文字符串是指正读和反读都一样的字符串),要求从主函数中由键盘输入字符串,调用函数fun后,根据函数fun的返回值,主函数输出是否为回文字符串的判断。输入样例:haah输出样例:是回文串//该代码使用MicrosoftVisualStudio2019编写#define_CRT_SECURE_NO_WARNINGS#include<stdio.

  • 一、Linux下的SVN服务器搭建

    一、Linux下的SVN服务器搭建这里自己做个总结。环境:contos7,百度云服务下载svn服务器,必须是联网情况下。yum-yinstallsubversion查看下载后的信息,安装位置及详细信息。rpm-qlsubversion3.创建版本库目录,可以再chenjy目录上放置多个项目,不必为每个项目创建一个版本库。下面是我的版本库mkdir/opt/svn/svnrepos/ch…

  • scrapy ip池(scrapy多线程)

    反爬策略有很多,最常用的也就是ip池,下面让我们一起跟着小省开始ip池之旅吧直接上代码:由于我们的ip池是自己维护在数据库中的,所以会有查库这一说#!/usr/bin/envpython#-*-coding:utf-8-*-#Createbyshengjk1on2017/11/6fromscreptileimportpoolfromutilspider.dp

  • 关于AD域的介绍

    关于AD域的介绍关于AD域第一次写博客,记录一下如何搭建自己的域服务器,以及其中遇到的一些问题,感谢“我的bug我做主”的文章《C#实现AD域验证登录(一)》,为防止原文被作者删除,手动将原文复制下来,如有侵权,请及时告知。域的简单介绍为什么要使用域?假设你是公司的系统管理员,你们公司有一千台电脑。如果你要为每台电脑设置登录帐户,设置权限(比如是否允许登录帐户安装软件),那你要分别坐在这一千台电脑前工作。如…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号