ScrollView中嵌套ListView滚动冲突的两种解决方案
2015-10-30 12:02:47  By: dwtedx

一、 为什么要使用ScrollView嵌套ListView

很多时间都会碰到ScrollView嵌套ListView的问题、ScrollView和ListView都是滚动结构、按理说、这两个控件在UI上的功能是一样的、但是看看下面这个设计、就是这种情况我们会使用到ScrollView嵌套ListView

scrollview和listview


这是阿里的APP首页、ScrollView中嵌套了ListView、ListView上面有固定的一些控件、但又要实现整体滚动、列表数据要嵌在固定数据中间、并且作为整体一起滚动、有了这样的设计需求、于是就有了ScrollView嵌套ListView的奇怪结构、这时你会发现ListView会显示不完全

它的高度始终有问题、因为默认情况下Android是禁止在ScrollView中放入另外的ScrollView的、它的高度是无法计算的、这里如果在Adapter里动态设置ListView的调试是可以解决问题的、但我不推荐那么干

网上的解决方法有很多种、我试过很多种方法、它们各有利弊、最后我觉得以下两种方案比较好、实际上不光是ListView、其他继承自AbsListView的类也适用、包括ExpandableListView、GridView等等、为了方便说明、以下均用ListView来代表


二、解决方案

1、使用LinearLayout取代ListView

既然ListView不能适应ScrollView、那就换一个可以适应ScrollView的控件、干嘛非要吊死在ListView这一棵树上呢?而LinearLayout是最好的选择、但如果我仍想继续使用已经定义好的Adater呢?我们可以直接循环已经写好的Adapter、把item添加到LinearLayout即可、代码如下

/**
 * 绑定布局
 */
public void bindLinearLayout() {
    int count = adapter.getCount();
    this.removeAllViews();
    for (int i = 0; i < count; i  ) {
        View v = adapter.getView(i, null, null);
        v.setOnClickListener(this.mClickListener);
        //绑定Adapter的数据到LinearLayout布局
        mLinearLayout.addView(v);
    }
   Log.v("countTAG", ""   count);
}


2、自定义可适应ScrollView的ListView

就是要用ListView怎么办?那就只好自定义一个类继承自ListView、通过重写其onMeasure方法、达到对ScrollView适配的效果、下面是继承了ListView的自定义类、可以直接使用的

import android.content.Context;
import android.util.AttributeSet;
import android.widget.ListView;
public class ListViewForScrollView extends ListView {
    public ListViewForScrollView(Context context) {
        super(context);
    }
    public ListViewForScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }
    public ListViewForScrollView(Context context, AttributeSet attrs,
        int defStyle) {
        super(context, attrs, defStyle);
    }
    @Override
    /**
     * 重写该方法、达到使ListView适应ScrollView的效果
     */
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
        MeasureSpec.AT_MOST);
        super.onMeasure(widthMeasureSpec, expandSpec);
    }
}


这个方法只要重写onMeasure方法、需要改动的地方比起方法一少不少代码、在xml布局中和Activty中使用的ListView改成这个自定义ListView就行了、这个方法有一个的小毛病、就是默认显示的首项是ListView、需要手动把ScrollView滚动至最顶端、代码如下

sv = (ScrollView) findViewById(R.id.act_solution_4_sv);
sv.smoothScrollTo(0, 0);


三、总结

方法一优点是完全解决了ScrollView嵌套ListView的问题、在Activity中手动为LinearLayout添加子项控件、不过需要注意的是、在添加前需要调用其removeAllViews的方法、否则可能会出现预想不到的事情、缺点是不能向ListView那样可以使用ViewHolder结构、在加载大量子项时会费很多时间在findViewById中、如果你的列表数据比较少的话、不妨试试这个方法、除了不能使用ViewHolder结构、使用方法几乎和ListView一样

方法2比方法1更简单、代码更少、同时保留了ListView原有的所有方法、包括notifyDataSetChanged方法、相比其他方法是最趋近于完美的方法、只是需要在Activity中设定ScrollView滚动至顶端、如果你还在犹豫不决的话就选这个方法吧、我想我以后是只会用这个方法了


若资源对你有帮助、浏览后有很大收获、不妨小额打赏我一下、你的鼓励是维持我不断写博客最大动力

想获取DD博客最新代码、你可以扫描下方的二维码、关注DD博客微信公众号(ddblogs)

或者你也可以关注我的新浪微博、了解DD博客的最新动态:DD博客官方微博(dwtedx的微博)

如对资源有任何疑问或觉得仍然有很大的改善空间、可以对该博文进行评论、希望不吝赐教

为保证及时回复、可以使用博客留言板给我留言: DD博客留言板(dwtedx的留言板)

感谢你的访问、祝你生活愉快、工作顺心、欢迎常来逛逛


快速评论


技术评论

  • 该技术还没有评论、赶快抢沙发吧...
DD记账
top
+