我有一个应用,我使用RecycleView
CardViews
。 CardView
现在包含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);
}
});
测试某些解决方案时出现问题:
答案 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);
}
});
}