Мои элементы RecyclerView путают положение, когда я нажимаю?
У меня проблемы с выбором элементов в RecyclerView. Когда я нажимаю, элемент отображается с наложением пурпурного цвета, но при переходе к другому экрану он отображается для другого элемента, а при прокрутке назад наложение старого элемента исчезает. Я не знаю, что происходит:(Как это исправить???
RecyclerView_Adapter
package basic.zhenyuan0502.nguyenlt.armstore;
import android.annotation.TargetApi;
import android.app.Activity;
import android.content.Context;
import android.graphics.Color;
import android.os.Build;
import android.support.percent.PercentLayoutHelper;
import android.support.percent.PercentRelativeLayout;
import android.support.v17.leanback.widget.BaseCardView;
import android.support.v7.widget.CardView;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.StaggeredGridLayoutManager;
import android.util.SparseBooleanArray;
import android.view.Gravity;
import android.view.KeyEvent;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewTreeObserver;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
import android.widget.Toast;
import java.util.ArrayList;
/**
* Created by Heisenberg on 17/04/2016.
*/
public class RecyclerView_Adapter<VH extends RecyclerView_Adapter.ViewHolder> extends RecyclerView.Adapter<VH> {
ArrayList<Item> arrayList = new ArrayList<Item>();
private Context context;
private int focusedItem = 0;
public RecyclerView_Adapter(Context context, ArrayList<Item> arrayList) {
this.context = context;
this.arrayList = arrayList;
}
@Override
public int getItemCount() {
return arrayList.size();
}
@Override
public void onAttachedToRecyclerView(final RecyclerView recyclerView) {
super.onAttachedToRecyclerView(recyclerView);
recyclerView.setOnKeyListener(new View.OnKeyListener() {
@Override
public boolean onKey(View v, int keyCode, KeyEvent event) {
RecyclerView.LayoutManager lm = recyclerView.getLayoutManager();
// Return false if scrolled to the bounds and allow focus to move off the list
if (event.getAction() == KeyEvent.ACTION_DOWN) {
if (keyCode == KeyEvent.KEYCODE_DPAD_DOWN) {
return tryMoveSelection(lm, 1);
} else if (keyCode == KeyEvent.KEYCODE_DPAD_UP) {
return tryMoveSelection(lm, -1);
}
}
return false;
}
});
}
private boolean tryMoveSelection(RecyclerView.LayoutManager lm, int direction) {
int tryFocusItem = focusedItem + direction;
if (tryFocusItem >= 0 && tryFocusItem < getItemCount()) {
notifyItemChanged(focusedItem);
focusedItem = tryFocusItem;
notifyItemChanged(focusedItem);
lm.scrollToPosition(focusedItem);
return true;
}
return false;
}
@Override
public void onBindViewHolder(final VH vHolder, int position) {
final Item item = arrayList.get(position);
vHolder.ivItem.setImageResource(item.getIvItem());
vHolder.tvPrice.setText(item.getTvPrice());
vHolder.tvName.setText(item.getTvName());
vHolder.setItemClickListener(new ItemClickListener() {
@Override
public void onItemClick(View v, int position) {
vHolder.itemView.setSelected(focusedItem == position);
//notifyItemChanged(focusedItem);
focusedItem = position;
vHolder.selected_overlay.setVisibility(View.VISIBLE);
// notifyItemChanged(focusedItem);
}
});
}
@Override
public VH onCreateViewHolder(ViewGroup viewGroup, int viewType) {
LayoutInflater inflater = LayoutInflater.from(viewGroup.getContext());
ViewGroup mainGroup = (ViewGroup) inflater.inflate(
R.layout.themeblue_item_recyclerview, viewGroup, false);
ViewHolder viewHolder = new ViewHolder(mainGroup);
return (VH) viewHolder;
}
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public ItemClickListener itemClickListener;
public TextView tvPrice, tvName;
public ImageView ivItem;
public LinearLayout llItem;
public CardView cvItem;
public View selected_overlay;
public ViewHolder(View convertView) {
super(convertView);
ivItem = (ImageView) convertView.findViewById(R.id.ivItem);
tvPrice = (TextView) convertView.findViewById(R.id.tvPrice);
tvName = (TextView) convertView.findViewById(R.id.tvName);
llItem = (LinearLayout) convertView.findViewById(R.id.llItem);
cvItem = (CardView) convertView.findViewById(R.id.cvItem);
selected_overlay = (View) convertView.findViewById(R.id.selected_overlay);
itemView.setOnClickListener(this);
}
@Override
public void onClick(View v) {
this.itemClickListener.onItemClick(v, getLayoutPosition());
}
public void setItemClickListener(ItemClickListener itemClickListener) {
this.itemClickListener = itemClickListener;
}
}
}
themeblue_item_recyclerview
<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:card_view="http://schemas.android.com/apk/res-auto"
android:id="@+id/cvItem"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:orientation="vertical"
android:foreground="?android:attr/selectableItemBackground"
card_view:cardBackgroundColor="@android:color/transparent">
<ImageView
android:id="@+id/ivItem"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:scaleType="fitXY" />
<LinearLayout
android:id="@+id/llItem"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="bottom|center_vertical"
android:background="#50000000"
android:orientation="vertical">
<TextView
android:id="@+id/tvName"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:gravity="center"
android:text="Name"
android:textColor="#FFFFFF"
android:textSize="12sp"
android:textStyle="bold" />
<TextView
android:id="@+id/tvPrice"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:gravity="center"
android:text="Price"
android:textColor="#FFFF26"
android:textSize="12sp"
android:textStyle="bold" />
</LinearLayout>
<View
android:id="@+id/selected_overlay"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#88673AB7"
android:visibility="invisible" />
</android.support.v7.widget.CardView>
OrderActivity: вы просто концентрируетесь на runOnUiThread и lvDetailMiddle
package basic.zhenyuan0502.nguyenlt.armstore;
import android.content.Intent;
import android.os.AsyncTask;
import android.os.Bundle;
import android.support.percent.PercentLayoutHelper;
import android.support.percent.PercentRelativeLayout;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.view.ActionMode;
import android.support.v7.widget.DefaultItemAnimator;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.StaggeredGridLayoutManager;
import android.view.GestureDetector;
import android.view.Menu;
import android.view.MenuItem;
import android.view.MotionEvent;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;
import android.widget.AdapterView;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageButton;
import android.widget.Toast;
import com.sileria.android.Kit;
import com.sileria.android.view.HorzListView;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.URL;
import java.net.URLConnection;
import java.util.ArrayList;
public class OrderActivity extends AppCompatActivity {
private static final int SWIPE_MIN_DISTANCE = 120;
private static final int SWIPE_MAX_OFF_PATH = 250;
private static final int SWIPE_THRESHOLD_VELOCITY = 200;
private String IP = "192.168.1.5";
private String PORT_SSL = "8080";
private final String LOGIN_URL = "http://" + IP + ":" + PORT_SSL + "/armstore/json.php";
// private ArrayList<TwoItem> arrayList = new ArrayList<>();
private ArrayList<Item> arrayList = new ArrayList<>();
private GestureDetector gestureDetector;
private View.OnTouchListener gestureListener;
RecyclerView lvDetailMiddle;
RecyclerView_Adapter adapter;
private static String docNoiDung_Tu_URL(String theUrl) {
StringBuilder content = new StringBuilder();
try {
URL url = new URL(theUrl);
URLConnection urlConnection = url.openConnection();
wrap the urlconnection in a bufferedreader
BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(urlConnection.getInputStream()));
String line;
while ((line = bufferedReader.readLine()) != null) {
content.append(line + "\n");
}
bufferedReader.close();
} catch (Exception e) {
e.printStackTrace();
}
return content.toString();
}
@Override
protected void onCreate(Bundle savedInstanceState) {
requestWindowFeature(Window.FEATURE_NO_TITLE);
Kit.init(this);
super.onCreate(savedInstanceState);
//Set full screen open
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
getSupportActionBar().hide(); //Use to hide ActionBar with theme AppCompat
//Set full screen close
setContentView(R.layout.activity_order_oneline_item);
//ListView TopBar
final HorzListView lvTopBar = (HorzListView) findViewById(R.id.lvTopBar);
String[] arrTxt = {"Món 1", "Món 2", "Món 3", "Món 4",
"Món 5", "Món 6", "Món 7", "Món 8", "Món 9", "Món 10"};
ArrayAdapter<String> adapterStringlvTopBar = new ArrayAdapter<>(this,
R.layout.themeblue_menu_horzlistview, arrTxt);
lvTopBar.setAdapter(adapterStringlvTopBar);
//ListView BottomBar
final HorzListView lvBottomBar = (HorzListView) findViewById(R.id.lvBottomBar);
ArrayAdapter<String> adapterStringlvBottomBar = new ArrayAdapter<>(this,
R.layout.themeblue_menu_horzlistview, arrTxt);
lvBottomBar.setAdapter(adapterStringlvBottomBar);
lvTopBar.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
lvBottomBar.getSelectedView().setSelected(false);
Toast.makeText(getApplicationContext(), "Chọn " + position, Toast.LENGTH_SHORT).show();
}
});
lvBottomBar.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
lvTopBar.getSelectedView().setSelected(false);
Toast.makeText(getApplicationContext(), "Chọn " + position, Toast.LENGTH_SHORT).show();
}
});
//EditText Search
final EditText etSearch = (EditText) findViewById(R.id.etSearch);
Button btnDelete = (Button) findViewById(R.id.btnDelete);
btnDelete.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
etSearch.setText("");
}
});
runOnUiThread(new Runnable() {
@Override
public void run() {
new readJSON().execute(LOGIN_URL);
}
});
lvDetailMiddle = (RecyclerView)
findViewById(R.id.lvMiddle);
lvDetailMiddle.setHasFixedSize(true);
adapter = new RecyclerView_Adapter(OrderActivity.this, arrayList);
adapter.notifyDataSetChanged();
lvDetailMiddle.setAdapter(adapter);
lvDetailMiddle.setItemAnimator(new DefaultItemAnimator());
lvDetailMiddle
.setLayoutManager(new StaggeredGridLayoutManager(2, StaggeredGridLayoutManager.HORIZONTAL));
ImageButton ibtnDetail = (ImageButton) findViewById(R.id.ibtnDetail);
ibtnDetail.setOnClickListener(new View.OnClickListener()
{
@Override
public void onClick(View v) {
Intent intent = new Intent(getApplication(), ProcessActivity.class);
startActivity(intent);
}
}
);
gestureDetector = new GestureDetector(this, new MyGestureDetector());
gestureListener = new View.OnTouchListener() {
public boolean onTouch(View v, MotionEvent event) {
return gestureDetector.onTouchEvent(event);
}
};
Kit.destroy();
}
@Override
public void onBackPressed() {
OrderActivity.this.finish();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
//noinspection SimplifiableIfStatement
if (id == R.id.action_settings) {
return true;
}
return super.onOptionsItemSelected(item);
}
class readJSON extends AsyncTask<String, Integer, String> {
@Override
protected String doInBackground(String... params) {
return docNoiDung_Tu_URL(params[0]);
}
@Override
protected void onPostExecute(String s) {
super.onPostExecute(s);
//ArrayList<String> arrProduct = new ArrayList<String>();
try {
JSONArray mang = new JSONArray(s);
for (int i = 0 ; i < mang.length(); ++i) {
JSONObject product = mang.getJSONObject(i);
arrayList.add(new Item(product.getString("name"),product.getString("price"),R.drawable.themeblue_logo_store_small));
}
} catch (JSONException e) {
e.printStackTrace();
}
}
}
class MyGestureDetector extends GestureDetector.SimpleOnGestureListener {
@Override
public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
try {
if (Math.abs(e1.getY() - e2.getY()) > SWIPE_MAX_OFF_PATH)
return false;
// right to left swipe
if (e1.getX() - e2.getX() > SWIPE_MIN_DISTANCE && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
Toast.makeText(OrderActivity.this, "Đang cuộn <-", Toast.LENGTH_SHORT).show();
} else if (e2.getX() - e1.getX() > SWIPE_MIN_DISTANCE && Math.abs(velocityX) > SWIPE_THRESHOLD_VELOCITY) {
Toast.makeText(OrderActivity.this, "Đang cuộn ->", Toast.LENGTH_SHORT).show();
}
} catch (Exception e) {
// nothing
}
return false;
}
@Override
public boolean onDown(MotionEvent e) {
return true;
}
}
}
2 ответа
Вместо того, чтобы использовать getLayoutPosition()
попробуйте использовать getAdapterPosition()
в вашем методе onClick.
Попробуй добавить vHolder.itemView.setSelected(focusedItem == position);
внутри onBindViewHolder
вашей RecyclerViewAdapter
...
@Override
public void onBindViewHolder(final VH vHolder, int position) {
...
vHolder.itemView.setSelected(focusedItem == position); // add it here for hightlight or unhightlight your RecycleViewCell when it recreate
vHolder.setItemClickListener(new ItemClickListener() {
@Override
public void onItemClick(View v, int position) {
vHolder.itemView.setSelected(focusedItem == position);
...
focusedItem = position;
...
}
});
}