在 Android 有时候会用到一个 ScrollView 嵌套一个 ScrollView 的问题,虽说不推荐这样使用。但是还是可以了解一下。
下面是一个 ListView 嵌套一个 ListView 。效果如下:
在上面的效果图中,主要是在于子 ListView 的滑动。这时候就需要由事件的分发机制来解决了。子 ListView 需要滑动,那么父 ListView 就不能把事件给拦截了,需要交给子 ListView 。
基于上述思想,我们可以自定义父 ListView ,在 onInterceptTouchEvent() 方法中处理事件。这样就可以滑动子 ListView 了。
父 ListView 代码:
ParentListView.java :
package com.example.scrollevent;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.widget.ListView;
public class ParentListView extends ListView {
public ParentListView(Context context) {
super(context);
}
public ParentListView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public ParentListView(Context context, AttributeSet attrs) {
super(context, attrs);
}
/**
* 将 onInterceptTouchEvent的返回值设置为 false,取消其对触摸事件的处理,将事件分发给子view
*/
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
return false;
}
}
其中:将 onInterceptTouchEvent() 方法的返回值设置为 false,取消其对触摸事件的处理,将事件分发给子 ListView ,让其处理。
在布局文件中引用:
activity_main.xml :
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.example.scrollevent.MainActivity">
<com.example.scrollevent.ParentListView
android:id="@+id/list_main"
android:layout_width="match_parent"
android:layout_height="match_parent">
</com.example.scrollevent.ParentListView>
</RelativeLayout>
其余的布局文件如下:
act_list_main_item.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/list_main_item"
android:text="good"
android:gravity="center_vertical"
android:layout_width="wrap_content"
android:layout_height="50dp" />
</LinearLayout>
act_list_child.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ListView
android:id="@+id/list_child"
android:layout_width="match_parent"
android:layout_height="match_parent">
</ListView>
</LinearLayout>
此布局可以是 RelativeLayout 布局方式。
MainActivity.java :
package com.example.scrollevent;
import android.content.Context;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
import java.util.ArrayList;
public class MainActivity extends AppCompatActivity {
private ArrayList<String> mArrayList;
private ArrayList<String> childArrayList;
private static final int NUM = 5;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initData();
initView();
}
private void initData() {
mArrayList = new ArrayList<>();
childArrayList = new ArrayList<>();
for (int i = 0; i <= 20; i++){
mArrayList.add("数据: " + i);
childArrayList.add("子 ListView: " + i);
}
}
private void initView() {
ListView mListViewMain = (ListView)findViewById(R.id.list_main);
mListViewMain.setAdapter(new MainAdapter(this, mArrayList, childArrayList));
mListViewMain.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
if (position != NUM)
Toast.makeText(MainActivity.this, "主 ListView: " + position, Toast.LENGTH_SHORT).show();
}
});
}
private class MainAdapter extends BaseAdapter {
private LayoutInflater inflater;
private Context context;
private ArrayList<String> arrayList = new ArrayList<>();
private ArrayList<String> childList = new ArrayList<>();
public MainAdapter(Context context, ArrayList arrayList, ArrayList<String> childList){
this.context = context;
inflater = LayoutInflater.from(context);
this.arrayList = arrayList;
this.childList = childList;
}
@Override
public int getCount() {
return arrayList.size();
}
@Override
public Object getItem(int position) {
return arrayList.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
convertView = inflater.inflate(R.layout.act_list_main_item, parent, false);
TextView mainTextView = (TextView)convertView.findViewById(R.id.list_main_item);
mainTextView.setText(arrayList.get(position)+"");
if (position == NUM) {
convertView = inflater.inflate(R.layout.act_list_child, parent, false);
ListView childListView = (ListView)convertView.findViewById(R.id.list_child);
ArrayAdapter arrayAdapter = new ArrayAdapter(context,
android.R.layout.simple_list_item_1, childList);
TextView childTextView = new TextView(context);
childTextView.setText(childList.get(position) +"");
childListView.setAdapter(arrayAdapter);
childListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Toast.makeText(MainActivity.this, "子 ListView: " + position, Toast.LENGTH_SHORT).show();
}
});
}
return convertView;
}
}
}
在对 MainAdapter 的优化中,采用 if (convertView == null) 的优化方式会出现异常的情况,其位置是 mainTextView.setText(arrayList.get(position)+""); 这行代码。由于没有采用 ViewHolder 方法来优化该 MainAdapter ,所以不知道会不会出现异常。由于只是测试,所以也没有把适配器分开。