ScrollView中只能放一个控件,一般都放LinearLayout,orientation属性值为vertical。在LinearLayout中放需要呈现的内容。ListView也在其中,ListView的高度设为适应自身内容(wrap_content)。但是为啥在scrollview中嵌套listview会出现只显示第一条listitem的高度呢,原因是:scrollview的ontach方法的滚动事件消费处理,ListView控件的高度设定问题

从谷歌那里找到的ScrollView嵌套ListView只显示一行的解决办法相信很多人都遇到过,然后大部分都是用这位博主的办法解决的吧

刚开始我也是用这个办法解决的,首先感谢这位哥的大私奉献,贴上地址

http://blog.csdn.net/p106786860/article/details/10461015

2、解决的核心代码

  1. public void setListViewHeightBasedOnChildren(ListView listView) {
  2. // 获取ListView对应的Adapter
  3. ListAdapter listAdapter = listView.getAdapter();
  4. if (listAdapter == null) {
  5. return;
  6. }
  7. int totalHeight = 0;
  8. for (int i = 0, len = listAdapter.getCount(); i < len; i++) {
  9. // listAdapter.getCount()返回数据项的数目
  10. View listItem = listAdapter.getView(i, null, listView);
  11. // 计算子项View 的宽高
  12. listItem.measure(0, 0);
  13. // 统计所有子项的总高度
  14. totalHeight += listItem.getMeasuredHeight();
  15. }
  16. ViewGroup.LayoutParams params = listView.getLayoutParams();
  17. params.height = totalHeight+ (listView.getDividerHeight() * (listAdapter.getCount() - 1));
  18. // listView.getDividerHeight()获取子项间分隔符占用的高度
  19. // params.height最后得到整个ListView完整显示需要的高度
  20. listView.setLayoutParams(params);
  21. }

这个代码让控件去计算Listview自己的高度然后设置这个Listview的高度

但是这个代码里面有一个问题,就是这个当你的ListView里面有多行的TextView的话,ListView的高度就会计算错误,它只算到了一行TextView的高度,

这个问题在so上的概述为以下:

http://stackoverflow.com/questions/14386584/getmeasuredheight-of-textview-with-wrapped-text

3、终极解决办法

这个问题头疼了一阵后,查找了一下,应该重写一个TextView的onMeasure方法比较好解决

代码有

  1. @Override
  2. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  3. super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  4. Layout layout = getLayout();
  5. if (layout != null) {
  6. int height = (int)FloatMath.ceil(getMaxLineHeight(this.getText().toString()))
  7. + getCompoundPaddingTop() + getCompoundPaddingBottom();
  8. int width = getMeasuredWidth();
  9. setMeasuredDimension(width, height);
  10. }
  11. }
  12. private float getMaxLineHeight(String str) {
  13. float height = 0.0f;
  14. float screenW = ((Activity)context).getWindowManager().getDefaultDisplay().getWidth();
  15. float paddingLeft = ((LinearLayout)this.getParent()).getPaddingLeft();
  16. float paddingReft = ((LinearLayout)this.getParent()).getPaddingRight();
  17. //这里具体this.getPaint()要注意使用,要看你的TextView在什么位置,这个是拿TextView父控件的Padding的,为了更准确的算出换行
  18. int line = (int) Math.ceil( (this.getPaint().measureText(str)/(screenW-paddingLeft-paddingReft))); height = (this.getPaint().getFontMetrics().descent-this.getPaint().getFontMetrics().ascent)*line; return height;}

上面的代码完成更能为,在ListView开始测量时,测量到TextView时,就调用我们的onMeasure方法,我们就可以测量字体的总宽度除与去掉边距的屏幕的大小,就可以算出文字要几行来显示,然后测量字体的高度*行数可以得到字体的总高度,然后在加上上下边距就是TextView真正的高度,然后setMeasuredDimension进去就可以计算出正确的值出来。

完整大代码我贴出来

public class MyListView2 extends LinearLayout {

private BaseAdapter adapter;

private MyOnItemClickListener onItemClickListener;

boolean footerViewAttached = false;

private View footerview;





@Override

protected void onLayout(boolean changed, int l, int t, int r, int b) {

// TODO Auto-generated method stub

super.onLayout(changed, l, t, r, b);

}





@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,

MeasureSpec.AT_MOST);

super.onMeasure(widthMeasureSpec, expandSpec);

}









public void notifyChange() {

int count = getChildCount();

if (footerViewAttached) {

count--;

}

LayoutParams params = new LayoutParams(LayoutParams.FILL_PARENT,

LayoutParams.WRAP_CONTENT);

for (int i = count; i < adapter.getCount(); i++) {

final int index = i;

final LinearLayout layout = new LinearLayout(getContext());

layout.setLayoutParams(params);

layout.setOrientation(VERTICAL);

View v = adapter.getView(i, null, null);

v.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

if (onItemClickListener != null) {

onItemClickListener.onItemClick(MyListView2.this,

layout, index, adapter.getItem(index));

}

}

});

ImageView imageView = new ImageView(getContext());

imageView.setLayoutParams(params);

layout.addView(v);

layout.addView(imageView);

addView(layout, index);

}

}





public MyListView2(Context context) {

super(context);

initAttr(null);

}





public MyListView2(Context context, AttributeSet attrs) {

super(context, attrs);

initAttr(attrs);

}





public void initAttr(AttributeSet attrs) {

setOrientation(VERTICAL);

}





public void initFooterView(final View footerView) {

this.footerview = footerView;

}





public void setFooterViewListener(OnClickListener onClickListener) {

this.footerview.setOnClickListener(onClickListener);

}





public BaseAdapter getAdapter() {

return adapter;

}





public void setAdapter(BaseAdapter adpater) {

this.adapter = adpater;

removeAllViews();

if (footerViewAttached)

addView(footerview);

notifyChange();

}





public void setOnItemClickListener(MyOnItemClickListener onClickListener) {

this.onItemClickListener = onClickListener;

}





public void noMorePages() {

if (footerview != null && footerViewAttached) {

removeView(footerview);

footerViewAttached = false;

}

}





public void mayHaveMorePages() {

if (!footerViewAttached && footerview != null) {

addView(footerview);

footerViewAttached = true;

}

}





public static interface MyOnItemClickListener {

public void onItemClick(ViewGroup parent, View view, int position,

Object o);

}

}

这个adapter就是你获取数据后设置的,也就是上面两点的综合

android scrollview嵌套listview计算高度的问题的更多相关文章

  1. Android ScrollView 嵌套ListView的替代方案

    概要:本例仅提供替代思路. 原需求:实现下图这个布局 要求:头部菜单固定,实现Viewpager.中间的按钮菜单,底部的listview一起能够上下滚动. 做法: 把Viewpager.中间的按钮菜单 ...

  2. android:ScrollView嵌套ListView的问题

    在ScrollView中嵌套使用ListView,看起来ListView只会显示一行多一点,不能滑动.ListView的高度怎么改都有问题,与预期不符合.搜索了一些解决方案,我觉得最好不要用这样的设计 ...

  3. Android ScrollView 嵌套 ListView、 ListView 嵌套ScrollView Scroll事件冲突解决办法

    本人菜鸟一名,最近工作了,开始学习Android. 最近在做项目的时候,UX给了个design,大概就是下拉刷新的ListView中嵌套了ScrollView,而且还要在ScrollView中添加动画 ...

  4. Android scrollview嵌套listview运行后最先显示出来的位置不在顶部而是中间问题

    scrollview里面嵌套了一个listview ,通过设置一个方法设置了listview的高度 现在的情况就是进到这个界面的时候看到的不是最上面 而是中间 ,该问题的解决办法为: mScrollV ...

  5. android 解决ScrollView嵌套ListView的问题,不能全屏,全屏不能显示下面控件

    在开发中遇到ScrollView嵌套ListView的问题,最开始发出不能全屏,效果是这样的: 但我想要的效果是这样的: 下面看一下布局文件: <?xml version="1.0&q ...

  6. android中ScrollView嵌套ListView或GridView显示位置问题

    Android中ScrollView中嵌套ListView或GridView时在开始进入界面时总是显示中间位置,开头的位置显示不出来.这种情况下只需要在ScrollView的父控件中添加以下两行代码即 ...

  7. 【Android - 问题解决】之ScrollView嵌套ListView时总是自动滑动到ListView顶部的问题

    最近做了一个项目,里面有一个ScrollView嵌套ListView的布局. 做出来之后发现,进入这个界面之后,总是自动滑动到ListView的顶部,而ScrollView中位于ListView上面的 ...

  8. Android——MeasureSpec学习 - 解决ScrollView嵌套ListView和GridView冲突的方法

      原文地址:http://blog.csdn.net/yuhailong626/article/details/20639217   在自定义View和ViewGroup的时候,我们经常会遇到int ...

  9. 关于ScrollView嵌套ListView问题

    Android开发之ScrollView中嵌套ListView的解决方案   原文:http://blog.csdn.net/minimicall/article/details/40983331   ...

随机推荐

  1. VMWare - Ubuntu 64 (16.04)之扩容介绍

    背景 貌似是一个老生常谈的问题哈,由于自己之前也没有弄过,今天正好有时间稍微折腾了一下. 这里就选择最简单的方式来为大家呈现. VMWare 的设置 没有什么可以过多说的,完全是图形操作.这里直接上图 ...

  2. React 关于组件(界面)更新

    在最近在学 React , 将组件的UI更新稍微整理了一下.根据业务要求,可能会出现如下的技术实现要求:1.更新自己2.更新子组件3.更新兄弟组件4.更新父组件5.父 call 子  function ...

  3. nodejs基础教程回顾01

    最近在复习nodejs,因为框架太多隔一段时间不用就会忘了,所以没办法必须时常拿出来练练,就像家里有好几辆车,要时不常的轮流开一圈.我就从最基础的开始写,怎么下载安装就不说了,首先是nodejs的三类 ...

  4. Tomcat和JDK的内存配置

    1.jvm内存管理机制: 1)堆(Heap)和非堆(Non-heap)内存 按照官方的说法:"Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配.堆是在 Ja ...

  5. Vim8.0在Debian下,normal模式的O命令出现延时

    Update 2018/4/26 问题是什么 在Debian的Vim8.0中,normal模式下,使用O创建新行,常常出现延迟情况:按下O后,可以看见O首先在当前光标位置出现,过了大约0.5-1秒,接 ...

  6. WMI远程启动软件(某个应用程序)

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.M ...

  7. python中修改字符串的几种方法

    在Python中,字符串是不可变类型,即无法直接修改字符串的某一位字符.因此改变一个字符串的元素需要新建一个新的字符串.常见的修改方法有以下4种. 方法1:将字符串转换成列表后修改值,然后用join组 ...

  8. C stat函数的用法举例(转载)

    stat函数讲解表头文件:    #include <sys/stat.h>             #include <unistd.h>定义函数:    int stat( ...

  9. Django项目实战之用户头像上传与访问

      1 将文件保存到服务器本地 upload.html <!DOCTYPE html> <html lang="en"> <head> < ...

  10. C#基础拾遗系列之二:C#7.0新增功能点

    第一部分: C#是一种通用的,类型安全的,面向对象的编程语言.有如下特点: (1)面向对象:c# 是面向对象的范例的一个丰富实现, 它包括封装.继承和多态性.C#面向对象的行为包括: 统一的类型系统 ...