如何在ScrollView嵌套另一个ScrollView

2025-01-24 22:52:41
推荐回答(2个)
回答1:

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);

}

}

回答2:

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()的方法添加列表。