滚动时onTextChanged更新ListView中的EditText

问题描述 投票:7回答:2

我一直在寻找答案,但解决方案似乎对我没有用。我在列表项中有一个TextView和一个EditText。我试图在用户编辑时更新EditTexts的存储值。

@Override
public View getView(int index, View convertView, ViewGroup parent) {
    ViewHolder holder = null;
    final int pos = index;
    if (convertView == null) {
        convertView = mInflater.inflate(R.layout.user_details_list_row, parent,false);
        holder = new ViewHolder();
        holder.mCaptionTextView  = (TextView)convertView.findViewById(id.user_detail_row_caption);
        holder.mDetailEditText = (EditText)convertView.findViewById(id.user_detail_row_value);
        convertView.setTag(holder);
    }else{
        holder = (ViewHolder) convertView.getTag();
    }

    holder.mDetailEditText.addTextChangedListener(new TextWatcher() {
        @Override
        public void onTextChanged(CharSequence s, int start, int before, int count) {
        }

        @Override
        public void beforeTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) {
            // TODO Auto-generated method stub

        }

        @Override
        public void afterTextChanged(Editable s) {              
            mUserDetails.set(pos, s.toString());
        }
    });     

    holder.mCaptionTextView.setText(mUserCaptions.get(index));
    holder.mDetailEditText.setText(mUserDetails.get(index),BufferType.EDITABLE);

    return convertView;
}

public static class ViewHolder{
    public TextView mCaptionTextView;
    public EditText mDetailEditText;
}

当我这样做时,滚动触发TextWatcher并更新值,用其他EditTexts之一的重复文本覆盖正确的文本。

我没有使用TextWatcher,而是尝试了这段代码:

holder.mDetailEditText.setOnFocusChangeListener(new OnFocusChangeListener() {

@Override
public void onFocusChange(View v, boolean hasFocus) {
    if (!hasFocus){
        EditText et =   (EditText)v.findViewById(id.user_detail_row_value);
        mUserDetails.set(index, et.getText().toString().trim());
    }
}
});

它还会更新错误的EditTexts。我在这里错过了什么?

编辑:还试过这个:

        final ViewHolder testHolder = holder; 
    holder.mDetailEditText.setOnFocusChangeListener(new OnFocusChangeListener() {

        @Override
        public void onFocusChange(View v, boolean hasFocus) {
            if (!hasFocus){
                EditText et =   (EditText)v.findViewById(id.user_detail_row_value);
                mUserDetails.set(testHolder.ref, et.getText().toString().trim());
            }
        }
    });

它纠正了我看到的滚动更改问题,但现在在编辑其中一个EditTexts之后,它也改变了其他一些。

android android-listview android-edittext
2个回答
2
投票

检查索引值是否与您正在处理的行位置匹配。如果使用支架图案,则该值可能不会从0变为7-8,具体取决于您的屏幕尺寸。这样的事情发生在我身上,我想我通过实施getCount方法解决了这个问题。这样,行50给出了位置50而不是7(例如)。


35
投票

问题与您没有从EditText小部件中删除以前添加的文本观察器这一事实有关。相反,您会继续将新的观察者添加到列表中。一旦您尝试编辑窗口小部件内容,所有文本观察者将按顺序获得通知,从而导致更新错误的用户详细信息。

EditText / TextView没有提供删除以前添加的文本观察者的方法,而没有explicit reference。这意味着您必须重新编写代码以保持对文本观察者的引用,并为每个getView方法执行创建/添加/删除观察者,或者扩展TextWatcher,允许在触发后更改用户详细信息以进行更新。后者在下面实现。

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    ViewHolder holder = null;
    if (convertView == null) {
        convertView = mInflater.inflate(R.layout.user_details_list_row, parent,false);
        holder = new ViewHolder();
        holder.mCaptionTextView  = (TextView)convertView.findViewById(id.user_detail_row_caption);
        holder.mWatcher = new MutableWatcher();
        holder.mDetailEditText = (EditText)convertView.findViewById(id.user_detail_row_value);
        holder.mDetailEditText.addTextChangedListener(holder.mWatcher);
        convertView.setTag(holder);
    } else{
        holder = (ViewHolder) convertView.getTag();
    }

    holder.mCaptionTextView.setText(mUserCaptions.get(position));

    holder.mWatcher.setActive(false);
    holder.mDetailEditText.setText(mUserDetails.get(position),BufferType.EDITABLE);
    holder.mWatcher.setPosition(position);
    holder.mWatcher.setActive(true);

    return convertView;
}

static class ViewHolder{
    public TextView mCaptionTextView;
    public EditText mDetailEditText;
    public MutableWatcher mWatcher;
}

class MutableWatcher implements TextWatcher {

    private int mPosition;
    private boolean mActive;

    void setPosition(int position) {
        mPosition = position;
    }

    void setActive(boolean active) {
        mActive = active;
    }

    @Override
    public void beforeTextChanged(CharSequence s, int start, int count, int after) { }

    @Override
    public void onTextChanged(CharSequence s, int start, int before, int count) { }

    @Override
    public void afterTextChanged(Editable s) {
        if (mActive) {
            mUserDetails.set(mPosition, s.toString());
        }
    }
}

希望这可以帮助。

© www.soinside.com 2019 - 2024. All rights reserved.