如何在ScrollView嵌套另一个ScrollView

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何在ScrollView嵌套另一个ScrollView相关的知识,希望对你有一定的参考价值。

android开发中ScrollView嵌套ScrollView是android最常用的功能,每一位学习安卓培训的都应该对它们的原理有深入的了解,我专门整理出几个实例, 为大家解析一下android培训中ScrollView嵌套ScrollView的原理。

大家好,众所周知,android 里两个相同方向的ScrollView是不能嵌套的,那要是有这样的需求怎么办?(这个需求一般都是不懂android的人提出来的)

难道就真的不能嵌套吗? 当然可以,只要你再写一个ScrollView,在里面做点脚,它就支持嵌套了。

目前做的这个只支持两个ScrollView嵌套,两个以上还有待改进,能套两个就已经能满足很多需求了,呵呵,另外现在只做了纵向scrollview的支持,横向的还没来的急做哦。

package com.sun.shine.study.innerscrollview.view;

import android.content.Context;

import android.util.AttributeSet;

import android.view.MotionEvent;

import android.view.View;

import android.widget.ScrollView;

public class InnerScrollView extends ScrollView

/**

*/

public ScrollView parentScrollView;

public InnerScrollView(Context context, AttributeSet attrs)

super(context, attrs);



private int lastScrollDelta = 0;

public void resume()

overScrollBy(0, -lastScrollDelta, 0, getScrollY(), 0, getScrollRange(), 0, 0, true);

lastScrollDelta = 0;



int mTop = 10;

/**

* 将targetView滚到最顶端

*/

public void scrollTo(View targetView)

int oldScrollY = getScrollY();

int top = targetView.getTop() - mTop;

int delatY = top - oldScrollY;

lastScrollDelta = delatY;

overScrollBy(0, delatY, 0, getScrollY(), 0, getScrollRange(), 0, 0, true);



private int getScrollRange()

int scrollRange = 0;

if (getChildCount() > 0)

View child = getChildAt(0);

scrollRange = Math.max(0, child.getHeight() - (getHeight()));



return scrollRange;



int currentY;

@Override

public boolean onInterceptTouchEvent(MotionEvent ev)

if (parentScrollView == null)

return super.onInterceptTouchEvent(ev);

else

if (ev.getAction() == MotionEvent.ACTION_DOWN)

// 将父scrollview的滚动事件拦截

currentY = (int)ev.getY();

setParentScrollAble(false);

return super.onInterceptTouchEvent(ev);

else if (ev.getAction() == MotionEvent.ACTION_UP)

// 把滚动事件恢复给父Scrollview

setParentScrollAble(true);

else if (ev.getAction() == MotionEvent.ACTION_MOVE)





return super.onInterceptTouchEvent(ev);



@Override

public boolean onTouchEvent(MotionEvent ev)

View child = getChildAt(0);

if (parentScrollView != null)

if (ev.getAction() == MotionEvent.ACTION_MOVE)

int height = child.getMeasuredHeight();

height = height - getMeasuredHeight();

// System.out.println("height=" + height);

int scrollY = getScrollY();

// System.out.println("scrollY" + scrollY);

int y = (int)ev.getY();

// 手指向下滑动

if (currentY < y)

if (scrollY <= 0)

// 如果向下滑动到头,就把滚动交给父Scrollview

setParentScrollAble(true);

return false;

else

setParentScrollAble(false);



else if (currentY > y)

if (scrollY >= height)

// 如果向上滑动到头,就把滚动交给父Scrollview

setParentScrollAble(true);

return false;

else

setParentScrollAble(false);





currentY = y;





return super.onTouchEvent(ev);



/**

* 是否把滚动事件交给父scrollview

*

* @param flag

*/

private void setParentScrollAble(boolean flag)

parentScrollView.requestDisallowInterceptTouchEvent(!flag);



参考技术A 1、最简单的布局:只有一个ListView
如果整个页面只有一个ListView的话,那么由于ListView本身带有滚动效果,所以当加载的数据超过页面显示的范围时,可以通过上下滑动来查看所有的item。
因此这种情况下,不需要添加ScrollView。
2、其它布局A+ListView
这种情况下,如果布局A定义在ListView的前面,那么当布局A所占的比例较大,或者ListView加载的数据较多时,都会导致ListView显示不完全。同样,由于ListView自身可以滚动,因此仍然可以通过上下滚动来查看ListView的所有item。
如图所示:

3、其它布局B+ListView
这种情况下,假设布局B定义在ListView的后面,那么就会出现两种情况:
(1)、ListView加载的数据不多,可以完全显示ListView的每一项,那么如果后面还有足够剩余的空间的话,布局B能正常显示;
(2)、ListView加载的数据加多,那么就会导致留给布局B的空间不足,或者根本就没有,布局B将会显示不完全或者完全不显示。
而不管怎么,ListView本身的内容,都可以通过滑动来查看。
对于第三种情况,如果不重新考虑布局的话,那么就需要添加ScrollView,用于查看剩余页面内容。

二、ScrollView和ListView的冲突问题
1、给ListView指定一个高度
例如设置android:layout_height="240dip",那么可以解决,可能会影响美观。

2、外面再添加个ScrollView
这种情况下,会出现问题。

3、给ScrollView设置属性:android:fillViewport="true"
测试的时候发现,如果ListView加载的数据不多的话,确实可以解决,但是当ListView加载的数据较多的时候,仍旧无法显示完全,并且这个时候ListView自身也无法滚动了。

三、解决办法有两种
1、在计算listview总高度并设置
ListView listView = (ListView) findViewById(id);
YourAdapter adapter = new MyAdapter("初始化你的适配器");
listView.setAdapter(adapter);
setListViewHeightBasedOnChildren(listView);(在setAdapter后调用自定义的方法)

复制代码代码如下:

/**
* @param listView
*/
private void setListViewHeightBasedOnChildren(ListView listView)

ListAdapter listAdapter = listView.getAdapter();
if (listAdapter == null)
return;


int totalHeight = 0;
for (int i = 0; i < listAdapter.getCount(); i++)
View listItem = listAdapter.getView(i, null, listView);
listItem.measure(0, 0);
totalHeight += listItem.getMeasuredHeight();


ViewGroup.LayoutParams params = listView.getLayoutParams();
params.height = totalHeight
+ (listView.getDividerHeight() * (listAdapter.getCount() - 1));
listView.setLayoutParams(params);


使用该方法需要注意:子ListView的每个Item必须是LinearLayout,不能是其他的,因为其他的Layout(如RelativeLayout)没有重写onMeasure(),所以会在onMeasure()时抛出异常。
2、 自定义ListView,重载onMeasure()方法,设置全部显示

复制代码代码如下:

package com.meiya.ui;

import android.widget.ListView;

/**
*
* @Description: scrollview中内嵌listview的简单实现
*
* @File: ScrollViewWithListView.java
*
* @Paceage com.meiya.ui
*
*
* @Date 下午03:02:38
*
* @Version
*/
public class ScrollViewWithListView extends ListView

public ScrollViewWithListView(android.content.Context context,
android.util.AttributeSet attrs)
super(context, attrs);


/**
* Integer.MAX_VALUE >> 2,如果不设置,系统默认设置是显示两条
*/
public void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, expandSpec);





以上可以解决scrollView内嵌listView,但是有一个问题是第一次进入界面时动态加载listview的items后页面会跳转到listview的第一个子项,这很蛋疼,
无奈又不知道怎么解决,就先用

复制代码代码如下:

scrollView.post(new Runnable()
//让scrollview跳转到顶部,必须放在runnable()方法中
@Override
public void run()
scrollView.scrollTo(0, 0);

);

这个方法过度下,希望有知道的朋友还给点解决方案
3、使用scrollView +LinearLayout用addView()的方法添加列表。

以上是关于如何在ScrollView嵌套另一个ScrollView的主要内容,如果未能解决你的问题,请参考以下文章

如何确定垂直 ScrollView 的方向?

Android 解决ScrollView嵌套RecyclerView导致滑动不流畅的问题

Android实战技巧:如何在ScrollView中嵌套ListView

SCrollView嵌套GridView,ScrollView如何实现上啦加载下拉刷新???

如何在ScrollView中嵌套ListView

滚动其子uiscrollview时如何防止滚动uiscrollview?