Android 中ListView 嵌套 ListView

本文探讨了在Android中如何处理ListView嵌套ListView的情况。通过自定义父ListView并在onInterceptTouchEvent()方法中返回false,允许事件传递给子ListView,实现子ListView的滑动。示例代码包括ParentListView.java、activity_main.xml、act_list_main_item.xml和act_list_child.xml的布局以及MainActivity.java的相关实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在 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 ,所以不知道会不会出现异常。由于只是测试,所以也没有把适配器分开。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值