java - 如何在 RecyclerView 中正确更新数据以在插入或更新另一个活动的数据后查看更改
问题描述
我遇到了一个问题,我正在尝试从加载来自查询中的信息的update
数据。为此,通过单击I open a new中的一个项目,在此位置该项目的数据并更新它。在我的更新中,我看到了更新,但是当我返回列表时,它并没有做出改变。我知道要使它起作用,我应该使用; 我不明白的是如何在我的. 这是我的:RecyclerView
MySQL
PHP
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();
}
}
}
这是我的activity
:
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);
}
}
编辑活动:
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会被触发,所以当这种情况发生时,您知道其他活动已经完成了它的工作,您应该从您的数据库中更新项目。
它是这样的:
- 主要活动调用startActivityForResult (Intent intent, int requestCode)来启动第二个活动,其意图包含要在第二个活动中使用的数据和一个请求代码,它是一个整数,表示您正在打开哪个活动(我们稍后将使用)。
- 第二个活动完成所需的工作,提供来自主要活动的意图,就在它调用完成之前,它应该调用setResult(int resultCode, Intent data)
其中结果代码:一个整数,表示它所做的工作是否导致成功或失败(它可以是您分配的任何整数,例如 1 表示成功,0 表示失败)
data:是一个从您的第二个活动中携带一些数据的意图,为了我们这里的情况,只需将您在第二个活动中收到的相同意图发送到第一个,因为它带有必要的数据。 - 由于您使用startActivityForResult开始了您的第二个活动,因此当它完成时,在主活动中使用 requestCode 调用onActivityResult (int requestCode, int resultCode, Intent data)您应该使用 if 语句来确保它是您启动的 requestCode第二个活动(如果你有第三个活动做其他事情,你会在它返回时携带正确的操作流程),以及应该指示你正在处理哪个项目的数据。
- 由于您拥有所需的数据,因此您可以在这里再次查询您的数据库以正确更新提到的项目,然后调用
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();
}
}
}
在完成之前的第二个活动中,您应该像这样调用 setResult :
Intent data = getIntent();
//put necessary data if there's any and send it back
setResult(1,data);
finish()
其中意图数据应包含必要的数据,以便在您返回时能够更新适配器(如果有)。例如,您可以在视图持有者 onclick 中使用holder.getAbsoulteAdapterPosition来获取被点击的位置并将其存储在意图中,以便稍后将其恢复到主要活动 onActivityResult。
推荐阅读
- sql - 同一查询中的嵌套存储过程 VS Exec 过程
- c++ - 此代码被转储并显示分段失败。我无法修复它
- flutter - 为什么在android中安装后应用程序覆盖?
- android - Flutter:我们检测到您的应用在您的 1 个或多个 app bundle 或 APK 的清单文件中包含 requestLegacyExternalStorage 标志
- carousel - Laravel 8中的轮播不工作只滑动一项
- javascript - 模块构建失败(来自 ./node_modules/babel-loader/lib/index.js):错误:插件/预设文件不允许导出对象,只有函数
- deployment - 如何将我网站的源代码部署到托管服务提供商
- git - 我们如何 `git stash apply` 在 repo 的所有状态之间移动?
- flutter - 类'未来
' 没有实例方法 '[]'。扑 - liquibase - 具有西里尔符号的相同文件上的不同 Liquibase 校验和