从RecycleView中添加项目重点关注EditText

时间:2016-07-14 13:39:11

标签: android android-edittext focus android-recyclerview

我有一个应用,我使用RecycleView CardViewsCardView现在包含EditText,当我向CardView添加新的RecycleView时,EditText应该聚焦,键盘应该出现。

我怎样才能实现这一目标?我试图在onBindViewHolder中添加一些代码:

public void onBindViewHolder(TodoViewHolder holder, final int position) {
    ...
    if(holder.tvDescription.requestFocus()) {
        window.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE);
    }
    ...
}

或者在创建ViewHolder时,但它没有奏效。

public class TodoViewHolder extends RecyclerView.ViewHolder {
    protected CheckBox cbDone;
    protected EditText tvDescription;
    protected FloatingActionButton btnDelete;

    public TodoViewHolder(View itemView) {
        super(itemView);

        cbDone = (CheckBox)itemView.findViewById(R.id.cbDone);
        tvDescription = (EditText) itemView.findViewById(R.id.tvDescription);
        btnDelete = (FloatingActionButton) itemView.findViewById(R.id.btnDelete);

        if(tvDescription.requestFocus()) {
            window.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE);
        }
    }
}

以下是我的AdapterCode解决方案:

public abstract class ArrayAdapter<T, VH extends RecyclerView.ViewHolder>
        extends RecyclerView.Adapter<VH> {

    private static final String TAG = "CustomArrayAdapter";

    private List<T> mObjects;

    public ArrayAdapter(final List<T> objects) {
        mObjects = objects;
    }

    /**
     * Adds the specified object at the end of the array.
     *
     * @param object The object to add at the end of the array.
     */
    public void add(final T object) {
        mObjects.add(object);
        notifyItemInserted(getItemCount() - 1);
    }

    /**
     * Remove all elements from the list.
     */
    public void clear() {
        final int size = getItemCount();
        mObjects.clear();
        notifyItemRangeRemoved(0, size);
    }

    @Override
    public int getItemCount() {
        return mObjects.size();
    }

    public T getItem(final int position) {
        return mObjects.get(position);
    }

    public long getItemId(final int position) {
        return position;
    }

    public List<T> getItems() {
        return mObjects;
    }

    /**
     * Returns the position of the specified item in the array.
     *
     * @param item The item to retrieve the position of.
     * @return The position of the specified item.
     */
    public int getPosition(final T item) {
        return mObjects.indexOf(item);
    }

    /**
     * Inserts the specified object at the specified index in the array.
     *
     * @param object The object to insert into the array.
     * @param index  The index at which the object must be inserted.
     */
    public void insert(final T object, int index) {
        mObjects.add(index, object);
        notifyItemInserted(index);

    }

    /**
     * Removes the specified object from the array.
     *
     * @param object The object to remove.
     */
    public void remove(T object) {
        final int position = getPosition(object);
        remove(position);
    }

    public void remove(int position) {
        if (position < 0 || position >= mObjects.size()) {
            Log.e(TAG, "remove: index=" + position);
        } else {
            mObjects.remove(position);
            notifyItemRemoved(position);
        }
    }

    /**
     * Sorts the content of this adapter using the specified comparator.
     *
     * @param comparator The comparator used to sort the objects contained in this adapter.
     */
    public void sort(Comparator<? super T> comparator) {
        Collections.sort(mObjects, comparator);
        notifyItemRangeChanged(0, getItemCount());
    }
}

已实施的Adapter

public class RecyclerViewAdapter extends ArrayAdapter<Todo, RecyclerViewAdapter.TodoViewHolder> {

    private static final String TAG = "RecyclerViewAdapter";
    private Todo selectedItem;
    private final Window window;

    public RecyclerViewAdapter(List<Todo> todos, Window window) {
        super(todos);
        this.window = window;
    }

    public Todo getSelectedItem() {
        return selectedItem;
    }

    public class TodoViewHolder extends RecyclerView.ViewHolder implements View.OnCreateContextMenuListener {
        protected CheckBox cbDone;
        protected EditText tvDescription;
        protected FloatingActionButton btnDelete;

        public TodoViewHolder(View itemView) {
            super(itemView);

            cbDone = (CheckBox)itemView.findViewById(R.id.cbDone);
            tvDescription = (EditText) itemView.findViewById(R.id.tvDescription);
            btnDelete = (FloatingActionButton) itemView.findViewById(R.id.btnDelete);

            itemView.setOnCreateContextMenuListener(this);
        }

        @Override
        public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) {
            menu.setHeaderTitle("Send to:");
            menu.add(0, v.getId(), 0, "all");

            Log.d(TAG, "view id: " + v.getId());
        }
    }

    @Override
    public void add(Todo object) {
        object.shouldBeFocused = true;
        super.add(object);
    }

    @Override
    public TodoViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) {
        View view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.todo_layout, viewGroup, false);
        return new TodoViewHolder(view);
    }

    @Override
    public void onBindViewHolder(final TodoViewHolder holder, final int position) {
        final Todo todo = getItem(holder.getAdapterPosition());
        holder.cbDone.setChecked(todo.isChecked);
        holder.tvDescription.setText(todo.description);

        holder.tvDescription.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
                // Do nothing
            }

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

            @Override
            public void afterTextChanged(Editable s) {
                todo.description = s.toString();
            }
        });

        holder.cbDone.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                Log.i(TAG, "onCheckedChanged called: isDone=" + isChecked);
                todo.isChecked = isChecked;
            }
        });

        holder.btnDelete.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Log.i(TAG, "onClick called: remove todo.");
                remove(todo);
            }
        });

        View.OnLongClickListener onClickListener = new View.OnLongClickListener() {
            @Override
            public boolean onLongClick(View v) {
                selectedItem = todo;
                return false;
            }
        };

        holder.cbDone.setOnLongClickListener(onClickListener);
        holder.tvDescription.setOnLongClickListener(onClickListener);
        holder.btnDelete.setOnLongClickListener(onClickListener);

        if (todo.shouldBeFocused) {
            holder.tvDescription.post(new Runnable() {
                @Override
                public void run() {
                    if (holder.tvDescription.requestFocus()) {
                        window.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE);
                        InputMethodManager inputMethodManager = (InputMethodManager) holder.tvDescription.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
                        inputMethodManager.showSoftInput(holder.tvDescription, InputMethodManager.SHOW_IMPLICIT);
                    }
                }
            });
            todo.shouldBeFocused = false;
        }
    }
}

Todo

public class Todo implements Serializable {

    // The creationDate is not used at the moment
    protected Date creationDate;
    protected String description;
    protected boolean isChecked;
    protected boolean shouldBeFocused;

    public Todo(String description) {
        this.description = description;
        this.creationDate = new Date();
    }

    public Date getCreationDate() {
        return creationDate;
    }

    public String getDescription() { return description; }

    @Override
    public String toString() {
        return creationDate + ": " + description + " state[" + isChecked + "]";
    }
}

MainActivity添加方法:

FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        adapter.add(new Todo(""));
        int count = adapter.getItemCount();
        recyclerView.smoothScrollToPosition(count - 1);
    }
});

测试某些解决方案时出现问题:

Correct behavior at the beginning Still correct behavior Here you can see, that some entries are duplicated in the list

2 个答案:

答案 0 :(得分:7)

问题是因为您过早致电requestFocus(),导致您的观看次数尚未显示在屏幕上。另外,当您添加新元素时,您应该添加一些标记 - 如果您请求关注此视图,则应该关注RecyclerView中的所有先前视图。假设您在CardView的末尾添加了新的RecyclerView,那么Adapter的添加方法应如下所示:

public void addToEnd(Model item) {
    item.shouldBeFocused = true;
    dataset.add(item);
    notifyItemInserted(dataset.size() - 1);
}

然后在你的onBindViewHolder()做类似的事情:

Model item = dataset.get(position);
...
if (item.shouldBeFocused) {
    holder.tvDescription.post(new Runnable() {
        @Override
        public void run() {
            if (holder.tvDescription.requestFocus()) {
                window.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_VISIBLE);
                InputMethodManager inputMethodManager = (InputMethodManager) holder.tvDescription.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
                inputMethodManager.showSoftInput(holder.tvDescription, InputMethodManager.SHOW_IMPLICIT);
            }
        }
    });
    item.shouldBeFocused = false;
}
...

此外,您可能需要滚动到RecyclerView的最后位置,以便为新添加的元素调用onBindViewHolder()。您可以通过setStackFromEnd = true行执行此操作。

<强>更新

您的问题是,您在TextWatcher方法中添加onBindViewHolder,首先是非常昂贵的操作,其次是您将输入的文本保存到{{ 1}}引用,这就是为什么final之后会给出不适当的结果的原因。

因此,请尝试创建自定义RecyclerView,以便将当前项目的位置保持在TextWatcher 优先

Adapter

然后将其添加到private static class PositionTextWatcher implements TextWatcher { private int position; public void updatePosition(int position) { this.position = position; } @Override public void beforeTextChanged(CharSequence charSequence, int i, int i2, int i3) { // no op } @Override public void onTextChanged(CharSequence charSequence, int i, int i2, int i3) { final Todo todo = getItem(position); todo.description = charSequence.toString(); } @Override public void afterTextChanged(Editable editable) { // no op } } 构建器中的EditText,同时调用ViewHolder

onCreateViewHolder

终于,而不是每次在@Override public TodoViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) { View view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.todo_layout, viewGroup, false); return new TodoViewHolder(view, new PositionTextWatcher()); } public class TodoViewHolder extends RecyclerView.ViewHolder implements View.OnCreateContextMenuListener { protected CheckBox cbDone; protected EditText tvDescription; protected FloatingActionButton btnDelete; protected PositionTextWatcher positionTextWatcher; public TodoViewHolder(View itemView, PositionTextWatcher positionTextWatcher) { super(itemView); cbDone = (CheckBox)itemView.findViewById(R.id.cbDone); tvDescription = (EditText) itemView.findViewById(R.id.tvDescription); btnDelete = (FloatingActionButton) itemView.findViewById(R.id.btnDelete); this.positionTextWatcher = positionTextWatcher; tvDescription.addTextChangedListener(this.positionTextWatcher); itemView.setOnCreateContextMenuListener(this); } @Override public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) { menu.setHeaderTitle("Send to:"); menu.add(0, v.getId(), 0, "all"); Log.d(TAG, "view id: " + v.getId()); } } 添加新的TextWatcher,只需更新自定义onBindViewHolder()中的位置:

TextWatcher

这应该像魅力一样!从this perfect answer获得了该解决方案,因此请查看更多背景信息。

答案 1 :(得分:0)

试试这样:

@Override
public void onBindViewHolder(final ViewHolder holder, int position) {

    ...

    // If this ViewHolder needs keyboard focus, execute below code.
    holder.itemView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
        @Override
        public void onGlobalLayout() {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
                holder.itemView.getViewTreeObserver().removeOnGlobalLayoutListener(this);
            } else {
                holder.itemView.getViewTreeObserver().removeGlobalOnLayoutListener(this);
            }

            EditText editText = holder.tvDescription;
            editText.requestFocus();
            InputMethodManager imm = (InputMethodManager) editText.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
            imm.showSoftInput(editText, 0);
        }
    });
}