如何在RecyclerView中正确更新数据,以查看从另一个活动插入或更新数据后的更改



我遇到了一个问题,这是我试图update数据RecyclerView中加载PHPMySQL查询的信息。要做到这一点,通过单击项在RecyclerView中我打开一个新的Activity,在这个位置RecyclerView项目的数据并更新它。在我的DB中,我看到更新,但是当我返回时对于Activity,它没有生成。的改变。我明白,为了这个工作,我应该使用adapter.notifyItemChanged (position);我不明白是什么在我Activity如何正确应用它。这是我的Adapter:

public class RecyclerViewListaCartuchos extends RecyclerView.Adapter<RecyclerViewListaCartuchos.ViewHolder> {
ArrayList<Cartuchos> mValues;
Context mContext;
public RecyclerViewListaCartuchos(Context context, ArrayList<Cartuchos> values) {
mValues = values;
mContext = context;
}
@NonNull
@Override
public RecyclerViewListaCartuchos.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from(mContext).inflate(R.layout.vista_diseno_cartucho, parent, false);
return new RecyclerViewListaCartuchos.ViewHolder(view);
}
@Override
public void onBindViewHolder(RecyclerViewListaCartuchos.ViewHolder Vholder, int position) {
Vholder.setData(mValues.get(position));
}
@Override
public int getItemCount() {
return mValues.size();
}
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public CardView cardView;
public TextView idCartucho;
public TextView modeloColor;
public TextView tv_fecha_mod_usuario_mod;
public ImageView iv_cartucho_img;
public View layout;
Cartuchos item;
public ViewHolder(View v) {
super(v);
layout = v;
v.setOnClickListener(this);
cardView = v.findViewById(R.id.cardView);
idCartucho = v.findViewById(R.id.idCartucho);
modeloColor = v.findViewById(R.id.tv_modelo_color);
tv_fecha_mod_usuario_mod = v.findViewById(R.id.tv_fecha_mod_usuario_mod);
iv_cartucho_img = v.findViewById(R.id.iv_cartucho_img);
switch (mContext.getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK) {
case Configuration.UI_MODE_NIGHT_YES:
case Configuration.UI_MODE_NIGHT_NO:
cardView.setBackgroundColor(mContext.getResources().getColor(R.color.white));
idCartucho.setTextColor(mContext.getResources().getColor(R.color.black));
modeloColor.setTextColor(mContext.getResources().getColor(R.color.black));
tv_fecha_mod_usuario_mod.setTextColor(mContext.getResources().getColor(R.color.black));
break;
}
}
public void setData(Cartuchos item) {
this.item = item;
String dato = item.getModelo() + " " + item.getColor();
String idC = String.valueOf(item.getIdCartucho());
String fec_us = "Última actualización: " + item.getFechaModificacion();
idCartucho.setText(idC);
modeloColor.setText(dato);
tv_fecha_mod_usuario_mod.setText(fec_us);
if(item.getModelo().contains("73")){
iv_cartucho_img.setImageResource(R.drawable.sietetresn);
}else if(item.getModelo().contains("90")){
iv_cartucho_img.setImageResource(R.drawable.nueveceron);
}
}
@Override
public void onClick(View view) {
int itemPosition = getLayoutPosition();
String idC = String.valueOf(idCartucho.getText());
String cantidad = String.valueOf(item.getCantidad());
//String mensaje = itemPosition + " / " + idC + "/Canti:" + cantidad;
Intent abrirEditar = new Intent(mContext, Editar.class);
abrirEditar.putExtra("idcartucho", String.valueOf(item.getIdCartucho()));
abrirEditar.putExtra("cantidad", String.valueOf(item.getCantidad()));
abrirEditar.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
mContext.startActivity(abrirEditar);
//Toast.makeText(mContext, mensaje, Toast.LENGTH_SHORT).show();
}
}
}

This myactivity:

public class MainActivity extends AppCompatActivity {
RecyclerView recyclerView;
ArrayList<Cartuchos> arrayList;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
arrayList = new ArrayList<>();
recyclerView = findViewById(R.id.rvInicio);
recyclerView.setLayoutManager(new LinearLayoutManager(MainActivity.this));
recyclerView.setHasFixedSize(true);
listadoCartuchos();
}
public void listadoCartuchos() {
StringRequest stringRequest = new StringRequest(Request.Method.POST, "url",
response -> {
try {
JSONArray jsonArray = new JSONArray(response);
for (int i = 0; i < jsonArray.length(); i++) {
JSONObject jsonObject1 = jsonArray.getJSONObject(i);
int id = jsonObject1.getInt("dat");
String modelo = jsonObject1.getString("da");
String color = jsonObject1.getString("d");
String fec = jsonObject1.getString("data");
int cantidad = jsonObject1.getInt("cantidad");
arrayList.add(new Cartuchos(id, modelo, color, fec, cantidad));
}
RecyclerViewListaCartuchos adaptador = new RecyclerViewListaCartuchos(getApplicationContext(), arrayList);
recyclerView.setAdapter(adaptador);
} catch (JSONException e) {
e.printStackTrace();
}
}, Throwable::printStackTrace);
RequestQueue requestQueue = Volley.newRequestQueue(MainActivity.this);
requestQueue.add(stringRequest);
}
}

Editar活动:

public class Editar extends AppCompatActivity {
private EditText etEditarCantidad;
private final String mensajeVacio = "No puede estar vacío ni el valor debe ser 0 (cero)";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_editar);
String idC = getIntent().getStringExtra("idcartucho");
int id = Integer.parseInt(idC);
String cantidad = getIntent().getStringExtra("cantidad");
etEditarCantidad = findViewById(R.id.etEditarCantidad);
etEditarCantidad.setText(cantidad);
Button btnEditar = findViewById(R.id.btnEditar);
btnEditar.setOnClickListener(v -> {
if(etEditarCantidad.getText().toString().length() == 0
|| etEditarCantidad.getText().toString().equals("0")){
Toast.makeText(Editar.this, mensajeVacio, Toast.LENGTH_SHORT).show();
}else{
try{
editarCantidad(id);
}catch (Exception e){
Log.e("Error", "Editar error: " + e.getMessage());
}
}
});
}
private void editarCantidad(int id){
if(TextUtils.isEmpty(etEditarCantidad.getText().toString())){
etEditarCantidad.setError("No puede estar vacío.");
etEditarCantidad.requestFocus();
return;
}
String url = "url";
final int cant = Integer.parseInt(etEditarCantidad.getText().toString());
StringRequest stringRequest = new StringRequest(Request.Method.POST, url, response -> {
try {
if (response.contains("Error")) {
Toast.makeText(Editar.this, "Lo sentimos, ha ocurrido un error.", Toast.LENGTH_SHORT).show();
} else {
Intent data = getIntent();
//put necessary data if there's any and send it back
setResult(1, data);
Toast.makeText(Editar.this, "Modificado con éxito.", Toast.LENGTH_SHORT).show();
Editar.this.finish();
}
} catch (Exception e) {
e.printStackTrace();
}
}, error -> Log.e("onErrorEdit", "onErrorResponse: " + error.getMessage())){
@Override
protected Map<String, String> getParams(){
Map<String, String> parametros = new HashMap<>();
parametros.put("cantidad", String.valueOf(cant));
parametros.put("id", String.valueOf(id));
return parametros;
}
};
RequestQueue requestQueue = Volley.newRequestQueue(Editar.this);
requestQueue.add(stringRequest);
}
}

你应该使用startActivityForResult来启动这个其他活动,当其他活动关闭时,onActivityResult在你的主活动被触发,所以当这种情况发生时,你知道其他活动已经完成了它的工作,你应该从你的DB更新项目。

它是这样的:

  1. main activity调用startActivityForResult (Intent Intent, int requestCode)来启动第二个activity,该Intent包含第二个activity要使用的数据和一个请求代码,该请求代码是一个整数,表示您正在打开哪个activity(我们将在稍后使用)。
  2. 第二个活动做所需的工作提供的意图从主活动,就在它调用完成之前,它应该调用setResult(int resultCode, intent data)
    其中结果代码:一个整数,表示它所做的工作是成功还是失败(它可以是你指定的任何整数,例如1代表成功,0代表失败)
    data是一个意图,从你的第二个活动携带一些数据,为了我们的情况在这里只是发送相同的意图,你在第二个活动收到的第一个,因为它携带必要的数据。
  3. 因为你开始了你的第二个活动startActivityForResult,当它完成时,onActivityResult (int requestCode, int resultCode, Intent data)在主活动中调用requestCode,你应该使用if语句来确保它的requestCode,你开始第二个活动(在那里,如果你有第三个活动,做别的事情,你会携带正确的操作流程,当它返回),和数据应该表明你正在做的工作。
  4. 既然你有所需的数据,你可以在这里自由地再次查询你的数据库,以正确更新所提到的项目,然后调用adapter.notifyItemChanged

你的主要活动应该看起来像:

public class MainActivity extends AppCompatActivity {
private final int MY_SECOND_ACTIVITY_REQUEST_CODE =2 , MY_THIRD_ACTIVITY_REQUEST_CODE =3 ;
private final int RESULT_SUCCESS =1 , RESULT_FALIURE =0;
... 
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
if (requestCode == MY_SECOND_ACTIVITY_REQUEST_CODE){
if (resultCode ==RESULT_SUCCESS)
SecondActivityFinishedUpdateAdapterData(data);
else somethingWrongHappenedInSecondActivity();
}else if (requestCode == MY_THIRD_ACTIVITY_REQUEST_CODE){
//do something else here in case if you have third activity
}else  super.onActivityResult(requestCode, resultCode, data);
}

public void startMyActivity(Intent data)
{
startActivityForResult(data), MY_SECOND_ACTIVITY_REQUEST_CODE );
}
}

你的视图持有人应该携带一个引用到你的主要活动能够调用startActivityForResult(我不确定这是正确的方式,但目前我不能想到任何其他的方式),所以它可能看起来像这样:

public class RecyclerViewListaCartuchos extends RecyclerView.Adapter<RecyclerViewListaCartuchos.ViewHolder> {
ArrayList<Cartuchos> mValues;
Context mContext;
MainActivity myMainActivity
public RecyclerViewListaCartuchos(MainActivity mainActivity, Context context, ArrayList<Cartuchos> values) {
mValues = values;
mContext = context;
this.myMainActivity = mainActivity ;
}
...
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
public CardView cardView;
public TextView idCartucho;
public TextView modeloColor;
public TextView tv_fecha_mod_usuario_mod;
public ImageView iv_cartucho_img;
public View layout;
Cartuchos item;
MainActivity myMainActivity ;
public ViewHolder(View v, MainActivity mainActivity) {
super(v);
this.myMainActivity = mainActivity ;
...
}
@Override
public void onClick(View view) {
int itemPosition = getLayoutPosition();
String idC = String.valueOf(idCartucho.getText());
String cantidad = String.valueOf(item.getCantidad());
//String mensaje = itemPosition + " / " + idC + "/Canti:" + cantidad;
Intent abrirEditar = new Intent(mContext, Editar.class);
abrirEditar.putExtra("idcartucho", String.valueOf(item.getIdCartucho()));
abrirEditar.putExtra("cantidad", String.valueOf(item.getCantidad()));
abrirEditar.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
myMainActivity.startMyActivity(abrirEditar);
//mContext.startActivity(abrirEditar); //commented this
//Toast.makeText(mContext, mensaje, Toast.LENGTH_SHORT).show();
}
}
}

在你的第二个活动结束之前,你应该像这样调用seresult:

Intent data = getIntent();
//put necessary data if there's any and send it back
setResult(1,data);
finish()

where intentdata应该包含必要的数据,以便在您返回时能够更新适配器(如果有的话)。举个例子,你可以用holder。gettabsoulteadapterposition在你的viewholder onclick中获取被点击的位置,并将其存储在intent中,稍后将被发送到主activity onActivityResult。