ScrollView与ListView之间的滑动冲突解决方法

方法一:(摘自:http://blog.csdn.net/wulianghuan/article/details/8627958)

在ScrollView中嵌套使用ListView,ListView只会显示一行到两行的数据。起初我以为是样式的问题,一直在对XML文件的样式进行尝试性设置,但始终得不到想要的效果。后来在网上查了查,ScrollView和ListView两个View都有滚动的效果,在嵌套使用时起了冲突,一般不建议两者套用。

 

下面说说具体解决方案。方案的主要思路就是根据ListView子项重置其高度。

解决方案代码如下:

 Java代码:

  1. /**  
  2.      * 重新计算ListView的高度,解决ScrollView和ListView两个View都有滚动的效果,在嵌套使用时起冲突的问题  
  3.      * @param listView  
  4.      */  
  5.     public void setListViewHeight(ListView listView) {    
  6.             
  7.         // 获取ListView对应的Adapter    
  8.         
  9.         ListAdapter listAdapter = listView.getAdapter();    
  10.         
  11.         if (listAdapter == null) {    
  12.             return;    
  13.         }    
  14.         int totalHeight = 0;    
  15.         for (int i = 0len = listAdapter.getCount(); i < len; i++) { // listAdapter.getCount()返回数据项的数目    
  16.             View listItem = listAdapter.getView(i, null, listView);    
  17.             listItem.measure(0, 0); // 计算子项View 的宽高    
  18.             totalHeight += listItem.getMeasuredHeight(); // 统计所有子项的总高度    
  19.         }    
  20.         
  21.         ViewGroup.LayoutParams params = listView.getLayoutParams();    
  22.         params.height = totalHeight + (listView.getDividerHeight() * (listAdapter.getCount() - 1));    
  23.         listView.setLayoutParams(params);    
  24.     }    



在设置LIstView的Adapter后调用此方法便可。

 

但是要注意的是,子ListView的每个Item必须是LinearLayout,不能是其他的,因为其他的Layout(如RelativeLayout)没有重写onMeasure(),所以会在onMeasure()时抛出异常。




方法二:

使用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);
}


方法三:

自定义可适应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一样

方法三比方法二更简单、代码更少、同时保留了ListView原有的所有方法、包括notifyDataSetChanged方法、相比其他方法是最趋近于完美的方法、只是需要在Activity中设定ScrollView滚动至顶端



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 iOS 开发中,嵌套在 `UIScrollView` 中的 `UITableView` 在滑动时可能会与 `UIScrollView` 的滑动手势产生冲突,导致无法正常滑动。这个问题可以通过以下两种方式解决: 1. 禁用 `UIScrollView` 的滑动手势 可以通过设置 `UIScrollView` 的 `panGestureRecognizer` 的 `enabled` 属性为 `NO` 来禁用滑动手势,这样就不会与 `UITableView` 的滑动手势产生冲突了。 ```objc scrollView.panGestureRecognizer.enabled = NO; ``` 2. 实现 `UIGestureRecognizerDelegate` 协议的方法 在 `UIViewController` 中实现 `UIGestureRecognizerDelegate` 协议的 `gestureRecognizer:shouldRecognizeSimultaneouslyWithGestureRecognizer:` 方法,可以控制两个手势是否允许同时识别。在这个方法中,可以判断当前的手势是否为 `UIScrollView` 的滑动手势,如果是,则允许与 `UITableView` 的滑动手势同时识别,否则不允许。 ```objc - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer { if ([gestureRecognizer.view isKindOfClass:[UIScrollView class]] && [otherGestureRecognizer.view isKindOfClass:[UITableView class]]) { return YES; } return NO; } ``` 需要注意的是,在实现这个方法时,要将 `UIScrollView` 的 `delegate` 设置为当前的 `UIViewController`,否则这个方法不会被调用。 ```objc scrollView.delegate = self; ``` 以上两种方式都可以解决嵌套在 `UIScrollView` 中的 `UITableView` 滑动手势冲突的问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值