您的位置:首页 > 移动开发 > Android开发

Android之ScrollView

2016-04-04 19:24 513 查看
1、ScrollView和HorizontalScrollView是为控件或者布局添加滚动条

2、上述两个控件只能有一个孩子,但是它并不是传统意义上的容器

3、上述两个控件可以互相嵌套

4、滚动条的位置现在的实验结果是:可以由layout_width和layout_height设定

5、ScrollView用于设置垂直滚动条,HorizontalScrollView用于设置水平滚动条:需要注意的是,有一个属性是 scrollbars 可以设置滚动条的方向:但是ScrollView设置成horizontal是和设置成none是效果同,HorizontalScrollView设置成vertical和none的效果同。

ScrollView嵌套listView时,listView会产生只能显示一行的状况,可以用如下方法解决。

解决方法

方法1:手动设置listView高度

01.public class MainActivity extends Activity {
02.    private ListView listView;
03.    @Override
04.    protected void onCreate(Bundle savedInstanceState) {
05.        super.onCreate(savedInstanceState);
06.        setContentView(R.layout.activity_main);
07.        listView = (ListView) findViewById(R.id.listView1);
08.        String[] adapterData = new String[] { "Afghanistan", "Albania",… … "Bosnia"};
09.        listView.setAdapter(new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,adapterData));
10.        setListViewHeightBasedOnChildren(listView);
11.    }
12.    public void setListViewHeightBasedOnChildren(ListView listView) {
13.        // 获取ListView对应的Adapter
14.        ListAdapter listAdapter = listView.getAdapter();
15.        if (listAdapter == null) {
16.            return;
17.        }
18.
19.        int totalHeight = 0;
20.        for (int i = 0, len = listAdapter.getCount(); i < len; i++) {
21.            // listAdapter.getCount()返回数据项的数目
22.            View listItem = listAdapter.getView(i, null, listView);
23.            // 计算子项View 的宽高
24.            listItem.measure(0, 0);
25.            // 统计所有子项的总高度
26.            totalHeight += listItem.getMeasuredHeight();
27.        }
28.
29.        ViewGroup.LayoutParams params = listView.getLayoutParams();
30.        params.height = totalHeight+ (listView.getDividerHeight() * (listAdapter.getCount() - 1));
31.        // listView.getDividerHeight()获取子项间分隔符占用的高度
32.        // params.height最后得到整个ListView完整显示需要的高度
33.        listView.setLayoutParams(params);
34.    }
35.}


方法2:使用单个ListView取代ScrollView中所有内容

就是说,把整个需要放在ScrollView中的内容,统统放在ListView中,原ListView上方的数据和下方数据,都作为现ListView的一个itemView,和原ListView中的单条数据是平级的关系。

public View getView(int position, View convertView, ViewGroup parent) {
//列表第一项
if(position == 0){
convertView = inflater.inflate(R.layout.item_solution2_top, null);
return convertView;
}
//列表最后一项
else if(position == 21){
convertView = inflater.inflate(R.layout.item_solution2_bottom, null);
return convertView;
}
//普通列表项
ViewHolder h = null;
if(convertView == null || convertView.getTag() == null){
convertView = inflater.inflate(R.layout.item_listview_data, null);
h = new ViewHolder();
h.tv = (TextView) convertView.findViewById(R.id.item_listview_data_tv);
convertView.setTag(h);
}else{
h = (ViewHolder) convertView.getTag();
}
h.tv.setText("第"+ position + "条数据");
return convertView;
}


方法3:使用LinearLayout取代ListView

既然ListView不能适应ScrollView,那就换一个可以适应ScrollView的控件,干嘛非要吊死在ListView这一棵树上呢?而LinearLayout是最好的选择。但如果我仍想继续使用已经定义好的Adater呢?我们只需要自定义一个类继承自LinearLayout,为其加上对BaseAdapter的适配。


/**
* 取代ListView的LinearLayout,使之能够成功嵌套在ScrollView中
* @author terry_龙
*/
public class LinearLayoutForListView extends LinearLayout {
private BaseAdapter adapter;
private OnClickListener onClickListener = null;
/**
* 绑定布局
*/
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.onClickListener);
addView(v, i);
}
Log.v("countTAG", "" + count);
}
public LinearLayoutForListView(Context context) {
super(context);
}


上面的代码拷贝保存为LinearLayoutForListView.class,或者直接拷贝Demo中的这个类在自己的工程里。我们只需要把原来xml布局文件中的ListView替换为这个类就行了:

<pm.nestificationbetweenscrollviewandabslistview.mywidgets.LinearLayoutForListView
android:id="@+id/act_solution_3_mylinearlayout"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:orientation="vertical" >
</pm.nestificationbetweenscrollviewandabslistview.mywidgets.LinearLayoutForListView>


在Activity中也把ListView改成LinearLayoutForListView,就能成功运行了。

mylinearlayout = (LinearLayoutForListView) findViewById(R.id.act_solution_3_mylinearlayout);
adapter = new AdapterForListView(this);
mylinearlayout.setAdapter(adapter);


方法四

自定义可适应ScrollView的ListView

这个方法和上面的方法是异曲同工,方法3是自定义了LinearLayout以取代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方法,需要改动的地方比起方法3少了不是一点半点…

在xml布局中和Activty中使用的ListView改成这个自定义ListView就行了。代码就省了吧…

这个方法和方法1有一个同样的毛病,就是默认显示的首项是ListView,需要手动把ScrollView滚动至最顶端。

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


参考链接

日积月累:ScrollView嵌套ListView只显示一行 - 郑文亮 - 博客园

四种方案解决ScrollView嵌套ListView问题 - Android开发者交流 - 安卓论坛

完成 ,效果如下

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: