解决水平ListView在ScrollView中出现的滑动冲突
解决的问题有两个:
1)实现水平滑动的ListView。重写AdapterView,上代码:
- package com.liucanwen.horizontallistview.view;
- import java.util.LinkedList;
- import java.util.Queue;
- import android.content.Context;
- import android.database.DataSetObserver;
- import android.graphics.Rect;
- import android.util.AttributeSet;
- import android.view.GestureDetector;
- import android.view.GestureDetector.OnGestureListener;
- import android.view.View.MeasureSpec;
- import android.view.MotionEvent;
- import android.view.View;
- import android.widget.AdapterView;
- import android.widget.ListAdapter;
- import android.widget.Scroller;
- /**
- * 重写ListView,以达到水平滑动
- */
- public class HorizontalListView extends AdapterView<ListAdapter>
- {
- public boolean mAlwaysOverrideTouch = true;
- protected ListAdapter mAdapter;
- ;
- ;
- protected int mCurrentX;
- protected int mNextX;
- private int mMaxX = Integer.MAX_VALUE;
- ;
- protected Scroller mScroller;
- private GestureDetector mGesture;
- private Queue<View> mRemovedViewQueue = new LinkedList<View>();
- private OnItemSelectedListener mOnItemSelected;
- private OnItemClickListener mOnItemClicked;
- private OnItemLongClickListener mOnItemLongClicked;
- private boolean mDataChanged = false;
- public HorizontalListView(Context context, AttributeSet attrs)
- {
- super(context, attrs);
- initView();
- }
- private synchronized void initView()
- {
- ;
- ;
- ;
- ;
- ;
- mMaxX = Integer.MAX_VALUE;
- mScroller = new Scroller(getContext());
- mGesture = new GestureDetector(getContext(), mOnGesture);
- }
- @Override
- public void setOnItemSelectedListener(
- AdapterView.OnItemSelectedListener listener)
- {
- mOnItemSelected = listener;
- }
- @Override
- public void setOnItemClickListener(AdapterView.OnItemClickListener listener)
- {
- mOnItemClicked = listener;
- }
- @Override
- public void setOnItemLongClickListener(
- AdapterView.OnItemLongClickListener listener)
- {
- mOnItemLongClicked = listener;
- }
- private DataSetObserver mDataObserver = new DataSetObserver()
- {
- @Override
- public void onChanged()
- {
- synchronized (HorizontalListView.this)
- {
- mDataChanged = true;
- }
- invalidate();
- requestLayout();
- }
- @Override
- public void onInvalidated()
- {
- reset();
- invalidate();
- requestLayout();
- }
- };
- @Override
- public ListAdapter getAdapter()
- {
- return mAdapter;
- }
- @Override
- public View getSelectedView()
- {
- // TODO: implement
- return null;
- }
- @Override
- public void setAdapter(ListAdapter adapter)
- {
- if (mAdapter != null)
- {
- mAdapter.unregisterDataSetObserver(mDataObserver);
- }
- mAdapter = adapter;
- mAdapter.registerDataSetObserver(mDataObserver);
- reset();
- }
- private synchronized void reset()
- {
- initView();
- removeAllViewsInLayout();
- requestLayout();
- }
- @Override
- public void setSelection(int position)
- {
- // TODO: implement
- }
- private void addAndMeasureChild(final View child, int viewPos)
- {
- LayoutParams params = child.getLayoutParams();
- if (params == null)
- {
- params = new LayoutParams(LayoutParams.FILL_PARENT,
- LayoutParams.FILL_PARENT);
- }
- addViewInLayout(child, viewPos, params, true);
- child.measure(
- MeasureSpec.makeMeasureSpec(getWidth(), MeasureSpec.AT_MOST),
- MeasureSpec.makeMeasureSpec(getHeight(), MeasureSpec.AT_MOST));
- }
- @Override
- protected synchronized void onLayout(boolean changed, int left, int top,
- int right, int bottom)
- {
- super.onLayout(changed, left, top, right, bottom);
- if (mAdapter == null)
- {
- return;
- }
- if (mDataChanged)
- {
- int oldCurrentX = mCurrentX;
- initView();
- removeAllViewsInLayout();
- mNextX = oldCurrentX;
- mDataChanged = false;
- }
- if (mScroller.computeScrollOffset())
- {
- int scrollx = mScroller.getCurrX();
- mNextX = scrollx;
- }
- )
- {
- ;
- mScroller.forceFinished(true);
- }
- if (mNextX >= mMaxX)
- {
- mNextX = mMaxX;
- mScroller.forceFinished(true);
- }
- int dx = mCurrentX - mNextX;
- removeNonVisibleItems(dx);
- fillList(dx);
- positionItems(dx);
- mCurrentX = mNextX;
- if (!mScroller.isFinished())
- {
- post(new Runnable()
- {
- @Override
- public void run()
- {
- requestLayout();
- }
- });
- }
- }
- private void fillList(final int dx)
- {
- ;
- );
- if (child != null)
- {
- edge = child.getRight();
- }
- fillListRight(edge, dx);
- ;
- );
- if (child != null)
- {
- edge = child.getLeft();
- }
- fillListLeft(edge, dx);
- }
- private void fillListRight(int rightEdge, final int dx)
- {
- while (rightEdge + dx < getWidth()
- && mRightViewIndex < mAdapter.getCount())
- {
- View child = mAdapter.getView(mRightViewIndex,
- mRemovedViewQueue.poll(), this);
- );
- rightEdge += child.getMeasuredWidth();
- )
- {
- mMaxX = mCurrentX + rightEdge - getWidth();
- }
- )
- {
- ;
- }
- mRightViewIndex++;
- }
- }
- private void fillListLeft(int leftEdge, final int dx)
- {
- && mLeftViewIndex >= 0)
- {
- View child = mAdapter.getView(mLeftViewIndex,
- mRemovedViewQueue.poll(), this);
- );
- leftEdge -= child.getMeasuredWidth();
- mLeftViewIndex--;
- mDisplayOffset -= child.getMeasuredWidth();
- }
- }
- private void removeNonVisibleItems(final int dx)
- {
- );
- )
- {
- mDisplayOffset += child.getMeasuredWidth();
- mRemovedViewQueue.offer(child);
- removeViewInLayout(child);
- mLeftViewIndex++;
- );
- }
- );
- while (child != null && child.getLeft() + dx >= getWidth())
- {
- mRemovedViewQueue.offer(child);
- removeViewInLayout(child);
- mRightViewIndex--;
- );
- }
- }
- private void positionItems(final int dx)
- {
- )
- {
- mDisplayOffset += dx;
- int left = mDisplayOffset;
- ; i < getChildCount(); i++)
- {
- View child = getChildAt(i);
- int childWidth = child.getMeasuredWidth();
- , left + childWidth,
- child.getMeasuredHeight());
- left += childWidth + child.getPaddingRight();
- }
- }
- }
- public synchronized void scrollTo(int x)
- {
- , x - mNextX, 0);
- requestLayout();
- }
- @Override
- public boolean dispatchTouchEvent(MotionEvent ev)
- {
- boolean handled = super.dispatchTouchEvent(ev);
- handled |= mGesture.onTouchEvent(ev);
- return handled;
- }
- protected boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
- float velocityY)
- {
- synchronized (HorizontalListView.this)
- {
- , (int) -velocityX, 0, 0, mMaxX, 0, 0);
- }
- requestLayout();
- return true;
- }
- protected boolean onDown(MotionEvent e)
- {
- mScroller.forceFinished(true);
- return true;
- }
- private OnGestureListener mOnGesture = new GestureDetector.SimpleOnGestureListener()
- {
- @Override
- public boolean onDown(MotionEvent e)
- {
- return HorizontalListView.this.onDown(e);
- }
- @Override
- public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,
- float velocityY)
- {
- return HorizontalListView.this
- .onFling(e1, e2, velocityX, velocityY);
- }
- @Override
- public boolean onScroll(MotionEvent e1, MotionEvent e2,
- float distanceX, float distanceY)
- {
- synchronized (HorizontalListView.this)
- {
- mNextX += (int) distanceX;
- }
- requestLayout();
- return true;
- }
- @Override
- public boolean onSingleTapConfirmed(MotionEvent e)
- {
- ; i < getChildCount(); i++)
- {
- View child = getChildAt(i);
- if (isEventWithinView(e, child))
- {
- if (mOnItemClicked != null)
- {
- mOnItemClicked.onItemClick(HorizontalListView.this,
- + i,
- + i));
- }
- if (mOnItemSelected != null)
- {
- mOnItemSelected.onItemSelected(HorizontalListView.this,
- + i,
- + i));
- }
- break;
- }
- }
- return true;
- }
- @Override
- public void onLongPress(MotionEvent e)
- {
- int childCount = getChildCount();
- ; i < childCount; i++)
- {
- View child = getChildAt(i);
- if (isEventWithinView(e, child))
- {
- if (mOnItemLongClicked != null)
- {
- mOnItemLongClicked.onItemLongClick(
- HorizontalListView.this, child, mLeftViewIndex
- + i,
- + i));
- }
- break;
- }
- }
- }
- private boolean isEventWithinView(MotionEvent e, View child)
- {
- Rect viewRect = new Rect();
- ];
- child.getLocationOnScreen(childPosition);
- ];
- int right = left + child.getWidth();
- ];
- int bottom = top + child.getHeight();
- viewRect.set(left, top, right, bottom);
- return viewRect.contains((int) e.getRawX(), (int) e.getRawY());
- }
- };
- }
2)第一步实现了水平滑动,往往我们会把这个水平ListView放到ScrollView里面(见截图实现),而这两个控件恰好滑动会有冲突,滑动水平ListView时会有卡顿,因此重写ScrollView,以达到流畅滑动:
- package com.liucanwen.horizontallistview.view;
- import android.content.Context;
- import android.util.AttributeSet;
- import android.view.GestureDetector;
- import android.view.GestureDetector.SimpleOnGestureListener;
- import android.view.MotionEvent;
- import android.view.View;
- import android.widget.ScrollView;
- /**
- * 重写ScrollView,以解决ScrollView与水平listView滑动时冲突
- */
- public class MyScrollView extends ScrollView
- {
- private GestureDetector mGestureDetector;
- View.OnTouchListener mGestureListener;
- public MyScrollView(Context context, AttributeSet attrs)
- {
- super(context, attrs);
- mGestureDetector = new GestureDetector(new YScrollDetector());
- );
- }
- @Override
- public boolean onInterceptTouchEvent(MotionEvent ev)
- {
- return super.onInterceptTouchEvent(ev)
- && mGestureDetector.onTouchEvent(ev);
- }
- class YScrollDetector extends SimpleOnGestureListener
- {
- @Override
- public boolean onScroll(MotionEvent e1, MotionEvent e2,
- float distanceX, float distanceY)
- {
- if (Math.abs(distanceY) > Math.abs(distanceX))
- {
- return true;
- }
- return false;
- }
- }
- }
好了,大功告成!
以下系项目的源代码下载地址:
http://download.csdn.net/detail/qq15989177612/6943633
解决水平ListView在ScrollView中出现的滑动冲突的更多相关文章
- ListView在ScrollView中不显示全部的问题
在实际应用中,我们可能会遇到把ListView放到ScrollView中的情况,在这种情况下,ListView的滑动属性与ScrollView的滑动出现冲突,从而ListView只显示一项.这里提供一 ...
- PullToRefreshListView中嵌套ViewPager滑动冲突的解决
PullToRefreshListView中嵌套ViewPager滑动冲突的解决 最近恰好遇到PullToRefreshListView中需要嵌套ViewPager的情况,ViewPager 作为头部 ...
- 解决Android ListView 和 ScrollView 共存时冲突 问题 方法其一
转载请注明出处: http://www.goteny.com/articles/2013/11/8.html http://www.cnblogs.com/zjjne/p/3428480.html 当 ...
- 解决嵌套在ScrollView中的TableView滑动手势冲突问题
最近在迭代开发公司项目的时候遇到了一个问题,在可以左右切换标签视图的ScrollView中嵌套了两个TableView用于展示视图,感觉一切so easy的情况下,问题出现了,因为左右两个视图既可以实 ...
- 解决scrollView中嵌套viewPager的冲突问题
很简单,在外层ScrollView中添加android:fillViewport="true"属性,然后给viewPager添加一个固定高度
- 解决ScrollView嵌套百度地图滑动冲突
一.问题描述 scrollview中嵌套百度地图时会出现滑动冲突,地图无法滑动的情况. 二.期望结果 焦点在地图上时,只有地图移动,焦点在地图外部时,可以滑动scrollview. 三.解决方法 自定 ...
- [Android Pro] 完美解决隐藏Listview和RecyclerView去掉滚动条和滑动到边界阴影的方案
reference to : http://blog.csdn.net/ming2316780/article/details/51578621 一.首先是Listview的属性设置 设置滑动到顶部和 ...
- H5使用Swiper过程中遇到的滑动冲突
一.问题 (1)PC端可以鼠标可以拖动中间的轮子让页面上下滑动,点击左键按着也是拖不动 (2)手机端浏览H5手指不能滑动页面,导致很多页面下面的文字看不到 二.解决问题 1.下面分先说css的问题,主 ...
- Scrollview嵌套Recyclerview嵌套滑动冲突,导致滑动时会出现卡顿的现象
recyclerView.setLayoutManager(new GridLayoutManager(mContext,2){ @Override public boolean canScrollV ...
随机推荐
- Nginx的负载均衡 - 最少连接 (least_conn)
Nginx版本:1.9.1 我的博客:http://blog.csdn.net/zhangskd 算法介绍 我们知道轮询算法是把请求平均的转发给各个后端,使它们的负载大致相同. 这有个前提,就是每个请 ...
- ormlite介绍一
概述 ORMlite是类似hibernate的对象映射框架,主要面向java语言,同时,是时下最流行的android面向数据库的的编程工具. 官方网站:http://ormlite.com ...
- UE4读取scv文件 -- 数据驱动游戏性元素
官方文档链接:http://docs.unrealengine.com/latest/CHN/Gameplay/DataDriven/index.html 略懒,稍微麻烦重复的工作,总希望能找人帮忙一 ...
- Android中FrameAnimation动画的使用
Frame Animation 表示帧动画,是顺序播放事先做好的图像,跟电影类似,Android SDK提供了另外一个类AnimationDrawable来定义使用Frame Animation. 下 ...
- Dynamics CRM2011 在Visual Studio中开启Javascript的Xrm.Page智能提示
前面一篇博文:http://blog.csdn.net/vic0228/article/details/49512699 讲到了在Visual Studio中开启xml编辑的智能提示,本篇接着来讲下如 ...
- 【Unity Shader实战】卡通风格的Shader(二)
写在前面 本系列其他文章: 卡通风格的Shader(一) 好久没写博客了,一定是因为课程作业比较多,一定不是因为我懒,恩恩. 三个月以前,在一篇讲卡通风格的Shader的最后,我们说到在Surface ...
- Xcode7.2使用sqlite3数据库的方法
大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) 之前版本的Xcode是可以直接连接sqlite3的库文件的,但 ...
- 插件开发之360 DroidPlugin源码分析(三)Binder代理
转载请注明出处:http://blog.csdn.net/hejjunlin/article/details/52138483 Hook机制中Binder代理类关系图 Hook机制中Binder代理时 ...
- 在Android中使用AlarmManager
AlarmManager是Android中的一种系统级别的提醒服务,它会为我们在特定的时刻广播一个指定的Intent.而使用Intent的时候,我们还需要它执行一个动作,如startActivity, ...
- Android开发中StackOverflowError
Android开发中StackOverflowError错误实例分析 一.概述 我在一个复杂的layout嵌套较多的android界面,碰到了java.lang.StackOverflowError这 ...