android - recylerView 未显示在应用程序中
问题描述
嘿,我是使用 Android 的新手,我用这个应用程序来制作我正在学习的教程,当我尝试在 Fragment 中工作时,教程从 MainActivity.java 读取和写入数据。
虽然没有错误,但应用程序没有显示任何内容,但 Firebase 实时数据库的 JSON 数据未按预期显示在 RecylerView 中。
这是我的文件
MainActivity.java
package com.example.plutus;
import android.os.Bundle;
import android.view.MenuItem;
import android.widget.FrameLayout;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentTransaction;
import com.google.android.material.bottomnavigation.BottomNavigationView;
public class MainActivity extends AppCompatActivity {
private TaskFragment taskFragment;
private TimeFragment timeFragment;
private HabitFragment habitFragment;
private HourFragment hourFragment;
private BottomNavigationView mMainNav;
private FrameLayout mMainFrame;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mMainNav = (BottomNavigationView) findViewById(R.id.main_nav);
mMainFrame = (FrameLayout) findViewById(R.id.main_frame);
taskFragment = new TaskFragment();
timeFragment = new TimeFragment();
habitFragment = new HabitFragment();
hourFragment = new HourFragment();
setFragment(taskFragment);
mMainNav.setOnNavigationItemSelectedListener(new BottomNavigationView.OnNavigationItemSelectedListener() {
@Override
public boolean onNavigationItemSelected(@NonNull MenuItem item) {
switch (item.getItemId()) {
case R.id.nav_task:
setFragment(taskFragment);
return true;
case R.id.nav_time:
setFragment(timeFragment);
return true;
case R.id.nav_habit:
setFragment(habitFragment);
return true;
case R.id.nav_hour:
setFragment(hourFragment);
return true;
default:
return false;
}
}
});
}
private void setFragment(Fragment fragment) {
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.main_frame, fragment);
fragmentTransaction.commit();
}
}
任务适配器.java
package com.example.plutus;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;
import com.google.firebase.database.ValueEventListener;
import java.util.ArrayList;
public class TaskAdapter extends RecyclerView.Adapter<TaskAdapter.MyViewHolder> {
ValueEventListener context;
ArrayList<MyTasks> myTasks;
public TaskAdapter(ValueEventListener c, ArrayList<MyTasks> p) {
context = c;
myTasks = p;
}
@NonNull
@Override
public MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
return new MyViewHolder(LayoutInflater.from((Context) context).inflate(R.layout.task_item, parent, false));
}
@Override
public void onBindViewHolder(@NonNull MyViewHolder holder, int position) {
holder.tasktitle.setText(myTasks.get(position).getTasktitle());
}
@Override
public int getItemCount() {
return myTasks.size();
}
class MyViewHolder extends RecyclerView.ViewHolder {
TextView tasktitle;
public MyViewHolder(@NonNull View itemView) {
super(itemView);
tasktitle = (TextView) itemView.findViewById(R.id.tasktitletext);
}
}
}
任务片段.java
package com.example.plutus;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.fragment.app.Fragment;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.google.firebase.database.DataSnapshot;
import com.google.firebase.database.DatabaseError;
import com.google.firebase.database.DatabaseReference;
import com.google.firebase.database.FirebaseDatabase;
import com.google.firebase.database.ValueEventListener;
import java.util.ArrayList;
/**
* A simple {@link Fragment} subclass.
*/
public class TaskFragment extends Fragment {
//Firebase Json integration
RecyclerView ourTasks;
ArrayList<MyTasks> list;
TaskAdapter taskAdapter;
public TaskFragment() {
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View rootView = inflater.inflate(R.layout.fragment_task, container, false);
ourTasks = rootView.findViewById(R.id.ourTasks);
ourTasks.setLayoutManager(new LinearLayoutManager(getActivity()));
list = new ArrayList<MyTasks>();
// Inflate the layout for this fragment
DatabaseReference taskDatabaseReference = FirebaseDatabase.getInstance().getReference();
DatabaseReference taskuser = taskDatabaseReference.child("Task1").child("tasktitle");
taskuser.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
for (DataSnapshot dataSnapshot1: dataSnapshot.getChildren()){
MyTasks p = dataSnapshot1.getValue(MyTasks.class);
list.add(p);
}
taskAdapter = new TaskAdapter(this, list);
ourTasks.setAdapter(taskAdapter);
taskAdapter.notifyDataSetChanged();
}
@Override
public void onCancelled(@NonNull DatabaseError databaseError) {
Toast.makeText(getContext(), "I'm fetching this", Toast.LENGTH_SHORT).show();
}
});
return rootView;
};
}
MyTasks.java
package com.example.plutus;
public class MyTasks {
String tasktitle;
public MyTasks() {
}
public MyTasks(String tasktitle) {
this.tasktitle = tasktitle;
}
public String getTasktitle() {
return tasktitle;
}
public void setTasktitle(String tasktitle) {
this.tasktitle = tasktitle;
}
}
我不明白真正的问题在哪里。
编辑:包含的 JSON 文件。
{
"Plutus":{
"Task1":{
"tasktitle":"Create an awesome app"
},
"Task2":{
"tasktitle":"Learn More Android"
},
"Task3":{
"tasktitle":"Learn Android Pentesting too"
},
"Task4":{
"tasktitle":"Oi don't forget Bug Hunting"
}
}
}
解决方案
您的代码中存在一些问题:
- 您没有
Plutus
在代码中处理 JSON 的级别。 - 您的行为好像快照中可以有多个节点,但实际上您正在加载一个非常特定的节点。所以你不需要循环
getChildren()
。 - 由于您
title
仅加载节点,因此结果不是MyTasks
.
要仅加载一项任务的标题,您可以执行以下操作:
DatabaseReference taskDatabaseReference = FirebaseDatabase.getInstance().getReference();
DatabaseReference taskuser = taskDatabaseReference.child("Plutus/Task1/tasktitle");
taskuser.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
Log.i("Firebase", dataSnapshot.getValue(String.class));
}
@Override
public void onCancelled(@NonNull DatabaseError databaseError) {
throw databaseError.toException(); // make sure to show the programming error
}
});
要加载单个任务,但要加载整个任务,它会是这样的:
DatabaseReference taskDatabaseReference = FirebaseDatabase.getInstance().getReference();
DatabaseReference taskuser = taskDatabaseReference.child("Plutus/Task1");
taskuser.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
MyTasks p = dataSnapshot.getValue(MyTasks.class);
Log.i("Firebase", p.toString());
}
@Override
public void onCancelled(@NonNull DatabaseError databaseError) {
throw databaseError.toException(); // make sure to show the programming error
}
});
并加载所有任务,并将它们添加到适配器/视图,它会是这样的:
taskAdapter = new TaskAdapter(this, list);
ourTasks.setAdapter(taskAdapter);
DatabaseReference taskDatabaseReference = FirebaseDatabase.getInstance().getReference();
DatabaseReference taskuser = taskDatabaseReference.child("Plutus");
taskuser.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
for (DataSnapshot taskSnapshot: dataSnapshot.getChildren()){
MyTasks p = taskSnapshot.getValue(MyTasks.class);
list.add(p);
}
taskAdapter.notifyDataSetChanged();
}
@Override
public void onCancelled(@NonNull DatabaseError databaseError) {
throw databaseError.toException(); // make sure to show the programming error
}
});
你会注意到我在这里把适配器从监听器中拉了出来。您不需要在每次数据更改时都重新构建一个新的适配器,而是可以只更新其中的数据,list
然后通知适配器,以便它更新 UI。
推荐阅读
- reactjs - 如何在 Modal 具有固定大小的 React 的 Modal.Body 上应用滚动
- javascript - 返回所有匹配元素的数组 Javascript/React
- javascript - Vue。如何在循环 v-for 中获取 prev 和 current 值以进行比较?
- java - E/ACameraMetadata:getConstEntry:找不到元数据标签 65578
- c# - 从网站获取 Token 并将其传递回 WPF
- php - 单击文本后如何将值放入变量中
- javascript - 有没有办法在不禁用 swiper-prev 的情况下在最后一张幻灯片上启动 Swiper 滑块?
- swift - 如何快速将麦克风的输入转换为另一种格式?
- sql - 除了使用 psql 之外,还有其他将数据导入 Postgres 的方法吗?
- azure - Azure 数据工厂数据流中“数据集”和“内联”源之间的区别?