首页 > 技术文章 > ListView中含有EditText时候--要命的焦点问题迎刃而解

lingzhao 2016-06-17 14:58 原文

最近做项目的时候遇到了一个问题,就是在ListView的item上面含有一个EditText,要求是这样:
 1当点击item的时候,item可以点击;
 2当点击EditText的时候EditText也可以正常弹出软件盘;
当时真是把我给搞疯了,item可以点击的时候,EditText不能正常使用了,EditText好了的时候,item又不可点击了,当时那个火,心中一万句草泥马都出来了。言归正传,先看一下解决的效果:

这里写图片描述 
当一个ListView的item中含有抢占焦点的控件的时候,我们通常在item的布局中加上这样一句: 
android:descendantFocusability=”blocksDescendants” 
防止抢占焦点,大多数情况下都能够解决问题,但是如果是EditText的时候,就很坑了,你会发现软件盘都不能够弹出,更别说输入文字了,要解决这个问题,就是让点击EditText的时候,item最外层不拦截点击事件就ok了,点解item的时候,在拦截点击事件也同样ok了。看getView中关键代码片段:

mHolder.edtInput.setOnTouchListener(new View.OnTouchListener() {

            @Override
            public boolean onTouch(View v, MotionEvent event) {
                ((ViewGroup) v.getParent())
                        .setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS);
                return false;
            }
        });

 

这个是EditText的时候的处理 
setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS); 
这个很关键,而且是在OnTouchListener中执行的,因为在Android中View的触摸事件传递中,View的OnTouchListener调用是早于其他事件的,而且返回值false,代表目前事件没有被消耗,继续向下传递,不影响后续事件的处理(这里不影响EditText其他事件的处理)

convertView.setOnTouchListener(new View.OnTouchListener() {

            @Override
            public boolean onTouch(View v, MotionEvent event) {
                ((ViewGroup) v)
                        .setDescendantFocusability(ViewGroup.FOCUS_BLOCK_DESCENDANTS);
                Toast.makeText(mContext, "点击了item", Toast.LENGTH_LONG).show();
                return false;
            }
        });

 

 

这个是convertView的处理,同样也是一行关键代码 
.setDescendantFocusability(ViewGroup.FOCUS_BLOCK_DESCENDANTS); 
下面附上适配器中的代码和布局代码,需要的同学自己copy一下看看吧: 
item布局代码:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:descendantFocusability="blocksDescendants"
    android:orientation="horizontal" >

    <TextView
        android:id="@+id/tvGroupTitle"
        android:layout_width="0dip"
        android:layout_height="50dip"
        android:layout_weight="1"
        android:gravity="center" />

    <EditText
        android:id="@+id/edtGroupContent"
        android:layout_width="100dip"
        android:layout_height="wrap_content" />

</LinearLayout>

 

BaseAdapter的代码:


public class NormalAdapter extends ArrayAdapter<String> {
    private List<String> mDatas;
    private Context mContext;

    public NormalAdapter(Context context, List<String> objects) {
        super(context, R.layout.activity_list, objects);
        mDatas = objects;
        mContext = context;

    }

    @Override
    public View getView(final int position, View convertView,
            final ViewGroup parent) {
        Log.e("tag", parent.toString());
        Holder mHolder = null;
        if (convertView == null) {
            convertView = LayoutInflater.from(mContext).inflate(
                    R.layout.item_group, null);
            mHolder = new Holder();
            mHolder.tvTitle = (TextView) convertView
                    .findViewById(R.id.tvGroupTitle);
            mHolder.edtInput = (EditText) convertView
                    .findViewById(R.id.edtGroupContent);
            convertView.setTag(mHolder);
        } else {
            mHolder = (Holder) convertView.getTag();
        }
        mHolder.tvTitle.setText(mDatas.get(position));
        mHolder.edtInput.setOnTouchListener(new View.OnTouchListener() {

            @Override
            public boolean onTouch(View v, MotionEvent event) {
                ((ViewGroup) v.getParent())
                        .setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS);
                return false;
            }
        });
        convertView.setOnTouchListener(new View.OnTouchListener() {

            @Override
            public boolean onTouch(View v, MotionEvent event) {
                ((ViewGroup) v)
                        .setDescendantFocusability(ViewGroup.FOCUS_BLOCK_DESCENDANTS);
                Toast.makeText(mContext, "点击了item", Toast.LENGTH_LONG).show();
                return true;
            }
        });
        return convertView;
    }

    private static final class Holder {
        TextView tvTitle;
        EditText edtInput;
    }

}

 

看看代码是不是解决问题了,可以的话,就给个赞吧!哈哈-^–^

推荐阅读