扩展ViewFlow避免和ViewPager滑动冲突,同时支持无限循环,并完美和CircleFlowIndicator结合
首先,为了避免滑动冲突,我们要继承ViewFlow,重写onInterceptTouchEvent
public class MyViewFlow extends ViewFlow {
private ViewPager mPager;
public MyViewFlow(Context context, AttributeSet attrs) {
super(context, attrs);
}
public void setViewPager(ViewPager viewPager) {
mPager = viewPager;
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (mPager != null)
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
mPager.requestDisallowInterceptTouchEvent(true);
break;
case MotionEvent.ACTION_UP:
mPager.requestDisallowInterceptTouchEvent(false);
break;
case MotionEvent.ACTION_CANCEL:
mPager.requestDisallowInterceptTouchEvent(false);
break;
case MotionEvent.ACTION_MOVE:
mPager.requestDisallowInterceptTouchEvent(true);
break;
}
return super.onInterceptTouchEvent(ev);
}
}
调用setViewPager指定viewPager后,滑动便不再冲突
接下来,我们实现无限循环滚动
public class AdapterBanner extends BaseAdapter {
private LayoutInflater mInflater;
private static final int[] ids = { R.drawable.banner, R.drawable.banner, R.drawable.banner, R.drawable.banner};
public AdapterBanner(Context context) {
mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
@Override
public int getCount() {
//return ids.length;
return Integer.MAX_VALUE;//返回很大的值使得getView中的position不断增大来实现循环
}
@Override
public Object getItem(int position) {
return position;
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = mInflater.inflate(R.layout.item_fr_acmain_nearby_banner, null);
}
convertView.findViewById(R.id.imgView).setBackgroundResource(ids[position%ids.length]);
return convertView;
}
}
关键代码在第13行和31行的红色部分。不过这样实现之后,和CircleFlowIndicator结合使用,会发现程序ANR(无响应),原因是CircleFlowIndicator会调用ViewFlow.getViewCount()来绘制圆点,显然这个数是Integer.MAX_VALUE,圆点数太多了导致ANR,我们需要继续扩展MyViewFlow的getViewCount()。如下。
private int mCount;
public void setCount(int count){
mCount=count;
}
@Override
public int getViewsCount() {
return mCount;
}
需要调用MyViewFlow.setCount(int count)指定真实的数目,这样做了之后,发现程序不会ANR了。CircleFlowIndicator绘制的圆点数目也正常。但是会发现,无论怎么滑动,圆点状态都不变化,分析ViewFlow的源码,发现要重写一下onScrollChanged,我们继续扩展MyViewFlow,代码如下。
private int mLastIndex;
@Override
public void setAdapter(Adapter adapter, int initialPosition){
super.setAdapter(adapter,initialPosition);
mLastIndex = initialPosition;
} @Override
protected void onScrollChanged(int h, int v, int oldh, int oldv) {
//super.onScrollChanged(h, v, oldh, oldv);
if (mIndicator != null) {
/*
* The actual horizontal scroll origin does typically not match the
* perceived one. Therefore, we need to calculate the perceived
* horizontal scroll origin here, since we use a view buffer.
*/
int hPerceived = h + (mCurrentAdapterIndex%mCount - mCurrentBufferIndex)
* getChildWidth(); if(mLastIndex%mCount==mCount-1 && mCurrentAdapterIndex>mLastIndex) {
oldh=0;
hPerceived = 0;
}
if(mLastIndex%mCount==0&&mCurrentAdapterIndex<mLastIndex)
hPerceived=h+(mCount-1-mCurrentBufferIndex)*getChildWidth(); LogUtil.e(Config.MYTAG,"mCurrentAdapterIndex="+mCurrentAdapterIndex);
LogUtil.e(Config.MYTAG,"mLastIndex="+mLastIndex);
LogUtil.e(Config.MYTAG, "hPerceived=" + hPerceived); mIndicator.onScrolled(hPerceived, v, oldh, oldv); mLastIndex=mCurrentAdapterIndex;
}
}
当然,有几个变量mIndicator,mCurrentAdapterIndex,mCurrentBufferIndex是沿用父类ViewFlow的,如果找不到,需要将父类ViewFlow中这些变量改为public即可(虽然从oop角度,这种做法很脏,但很省事,大半夜码文字很累,我很懒)。
至此,大功告成!
整体MyViewFlow代码如下:
package common.control.viewflow; import android.content.Context;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.Adapter; import com.xxx.android.main.Config; import common.util.LogUtil; public class MyViewFlow extends ViewFlow {
private ViewPager mPager; public MyViewFlow(Context context, AttributeSet attrs) {
super(context, attrs);
} public void setViewPager(ViewPager viewPager) {
mPager = viewPager;
} @Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (mPager != null)
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
mPager.requestDisallowInterceptTouchEvent(true);
break;
case MotionEvent.ACTION_UP:
mPager.requestDisallowInterceptTouchEvent(false);
break;
case MotionEvent.ACTION_CANCEL:
mPager.requestDisallowInterceptTouchEvent(false);
break;
case MotionEvent.ACTION_MOVE:
mPager.requestDisallowInterceptTouchEvent(true);
break;
}
return super.onInterceptTouchEvent(ev);
}
private int mCount;
public void setCount(int count){
mCount=count;
}
@Override
public int getViewsCount() {
return mCount;
}
private int mLastIndex;
@Override
public void setAdapter(Adapter adapter, int initialPosition){
super.setAdapter(adapter,initialPosition);
mLastIndex = initialPosition;
} @Override
protected void onScrollChanged(int h, int v, int oldh, int oldv) {
//super.onScrollChanged(h, v, oldh, oldv);
if (mIndicator != null) {
/*
* The actual horizontal scroll origin does typically not match the
* perceived one. Therefore, we need to calculate the perceived
* horizontal scroll origin here, since we use a view buffer.
*/
int hPerceived = h + (mCurrentAdapterIndex%mCount - mCurrentBufferIndex)
* getChildWidth(); if(mLastIndex%mCount==mCount-1 && mCurrentAdapterIndex>mLastIndex) {
oldh=0;
hPerceived = 0;
}
if(mLastIndex%mCount==0&&mCurrentAdapterIndex<mLastIndex)
hPerceived=h+(mCount-1-mCurrentBufferIndex)*getChildWidth(); LogUtil.e(Config.MYTAG,"mCurrentAdapterIndex="+mCurrentAdapterIndex);
LogUtil.e(Config.MYTAG,"mLastIndex="+mLastIndex);
LogUtil.e(Config.MYTAG, "hPerceived=" + hPerceived); mIndicator.onScrolled(hPerceived, v, oldh, oldv); mLastIndex=mCurrentAdapterIndex;
}
}
}
自动播放就不用讲了,网上很多。同学们如果有收获,记得回赞。
扩展ViewFlow避免和ViewPager滑动冲突,同时支持无限循环,并完美和CircleFlowIndicator结合的更多相关文章
- (转)ViewPager,ScrollView 嵌套ViewPager滑动冲突解决
ViewPager,ScrollView 嵌套ViewPager滑动冲突解决 本篇主要讲解一下几个问题 粗略地介绍一下View的事件分发机制 解决事件滑动冲突的思路及方法 ScrollView 里面嵌 ...
- PullToRefreshListView中嵌套ViewPager滑动冲突的解决
PullToRefreshListView中嵌套ViewPager滑动冲突的解决 最近恰好遇到PullToRefreshListView中需要嵌套ViewPager的情况,ViewPager 作为头部 ...
- Android ScrollView与ViewPager滑动冲突
前段时间做项目碰到在ScrollView里添加ViewPager,但是发现ViewPager的左右滑动和ScrollView的滑动冲突了,解决这个问题的方法是重写ScrollView. 代码: pub ...
- ViewPager,实现真正的无限循环(定时+手动)
利用定时器,实现循环轮播,很简单:只需在定时器的消息里加如下代码即可: int count = adapter.getCount(); if (count > 1) { // 多于1个,才循环 ...
- 封装一个ViewPager真正的实现图片无限循环滚动带导航点
效果图: 大家在写项目的过程中常常会碰到须要实现Viewpager里面载入几张图片来循环自己主动轮播的效果,假设不封装一下的话代码分散在activity里面会显得非常乱.并且也不利于我们下次复用,所以 ...
- Android ViewPager嵌套ViewPager滑动冲突处理方法
dispatchTouchEvent方法用于事件的分发,Android中所有的事件都必须经过这个方法的分发, 然后决定是自身消费当前事件还是继续往下分发给子控件处理.返回true表示不继续分发,事件没 ...
- scrollview和viewpager滑动冲突
import android.content.Context; import android.util.AttributeSet; import android.view.MotionEvent; i ...
- 使用NestedScrollView+ViewPager+RecyclerView+SmartRefreshLayout打造酷炫下拉视差效果并解决各种滑动冲突
使用NestedScrollView+ViewPager+RecyclerView+SmartRefreshLayout打造酷炫下拉视差效果并解决各种冲突 如果你还在为处理滑动冲突而发愁,那么你需要静 ...
- Android 使用NestedScrollView+ViewPager+RecyclerView+SmartRefreshLayout打造酷炫下拉视差效果并解决各种滑动冲突
如果你还在为处理滑动冲突而发愁,那么你需要静下心来看看这边文章,如果你能彻底理解这篇文章中使用的技术,那么,一切滑动冲突的问题解决起来就轻而易举了: 先扔一个最终实现的效果图 先分析下效果图中实现的功 ...
随机推荐
- AT91-PWM应用
步骤1: make menuconfig配置内核, 开启PWM输出功能. Device Drivers ---> Misc devices ---> <*>Atmel AT3 ...
- jQuery学习实用记录
//jQuery加载 $(document).ready(function(){ //开始编写函数 }); //点击后 上下滑动隐藏效果 $("#id").click(functi ...
- 修改ubuntu DNS的步骤(图文)
有时候连接上VPN服务器后,还是打不开某些网站,这时候,需要对DNS进行更改,一般是修改成为谷歌提供的免费DNS:8.8.8.8 8.8.4.4,在windows下更改比较简单(点击查看“连接上VP ...
- 用Backbone.js创建一个联系人管理系统(二)
欢迎大家回来继续这一教程,第一部分我们学习了model,collection和view在Backbone中的 基本用法,还有怎么样用主视图去绑定collection去渲染出每个Contact. 这部分 ...
- 水果项目第3集-asp.net web api开发入门
app后台开发,可以用asp.net webservice技术. 也有一种重量级一点的叫WCF,也可以用来做app后台开发. 现在可以用asp.net web api来开发app后台. Asp.net ...
- TreeView树形控件递归绑定数据库里的数据
TreeView树形控件递归绑定数据库里的数据. 第一种:性能不好 第一步:数据库中查出来的表,字段名分别为UNAME(显示名称),DID(关联数据),UTYPE(类型) 第二步:前台代码 <% ...
- [keepalved]主从上同时出现VIP,无法消失情况
双主架构中,keepalived日志出现: more /var/log/messageOct 9 03:16:22 mysql-dzg-60-148 Keepalived_vrrp[8526]: VR ...
- 快来玩“Gift大转盘”百分百赚好礼
现在开始到今年的最后一天,你天天都可以来转100%中奖的“ Gift大转盘 ”.代金券.产品折扣.精美纪念礼,没有多余规则.全部网友都可参加,转到就是你赚到,赶快转起来吧! >>活动主页& ...
- 【转】Android 当打开“开发者模式”中的“不保留活动”后,程序应当怎么保持正常运行
当打开这个设置以后,程序的Activity会自动销毁,每次返回的时候就会不断重oncreate,此时伴随的问题多多. 参考文档:http://www.bubuko.com/infodetail-960 ...
- php函数间的参数传递(值传递/引用传递)
php:函数间的参数传递 1.值传递 代码如下: <?php function exam($var1){ $var1++: echo "In Exam:" . $var1 . ...