Я заполняю ListView
песен, используя базу данных Firebase и Firebase-UI, все зависимости инициализированы правильно, и приложение подключено к базе данных, но когда оно отображает список, он не отображает текст, а просто пустые поля, такие как:
Когда элемент добавляется в базу данных, добавляется поле, но текст в нем не отображается. Вот код:
Класс песни:
package com.example.gloriadesideri.animas;
public class Song
{
private String myName;
private String myURL;
private String myAuthor;
public Song(){
/*myName = "";
myURL = "";
myAuthor = "";*/
}
public Song(String Author, String Song, String URL) {
this.myName=Song;
this.myURL=URL;
this.myAuthor=Author;
}
public String getName()
{
return myName;
}
public String getURL()
{
return myURL;
}
public String getAuthor()
{
return myAuthor;
}
public void setName(String name)
{
this.myName=name;
}
public void setURL ( String URL)
{
this.myURL=URL;
}
public void setAuthor(String author)
{
this.myAuthor=author;
}
}
Макет песни:
<?xml version = "1.0" encoding = "utf-8"?>
<LinearLayout
xmlns:android = "http://schemas.android.com/apk/res/android"
android:layout_width = "match_parent"
android:layout_height = "wrap_content"
android:orientation = "vertical"
android:layout_margin = "20dp"
android:padding = "10dp">
<TextView
android:layout_width = "wrap_content"
android:layout_height = "wrap_content"
android:id = "@+id/songName"/>
<TextView
android:layout_width = "wrap_content"
android:layout_height = "wrap_content"
android:id = "@+id/author"/>
</LinearLayout>
Действие, которое должно иметь представление списка:
package com.example.gloriadesideri.animas;
import android.content.Intent;
import android.support.design.widget.NavigationView;
import android.support.v4.view.GravityCompat;
import android.support.v4.widget.DrawerLayout;
import android.support.v7.app.ActionBar;
import android.support.v7.app.ActionBarDrawerToggle;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.MenuItem;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.TextView;
import com.firebase.ui.database.FirebaseListAdapter;
import com.firebase.ui.database.FirebaseListOptions;
import com.google.firebase.database.ChildEventListener;
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.Query;
import com.google.firebase.database.ValueEventListener;
import java.util.ArrayList;
import java.util.List;
public class Canzoni extends AppCompatActivity implements NavigationView.OnNavigationItemSelectedListener
{
//nav view parameters
private DrawerLayout cDrawerLayout;
private ActionBarDrawerToggle cToggle;
//list view parameters
private ListView mListView;
//firebase parameters
private FirebaseListAdapter mAdapter;
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_canzoni);
//nav view code
cDrawerLayout= (DrawerLayout) findViewById(R.id.draw_viewC);
cToggle= new ActionBarDrawerToggle(this, cDrawerLayout,R.string.Open, R.string.Close);
cDrawerLayout.addDrawerListener(cToggle);
cToggle.syncState();
ActionBar actionBar = getSupportActionBar();
if (actionBar != null)
actionBar.setDisplayHomeAsUpEnabled(true);
NavigationView cNavigationView = (NavigationView) findViewById(R.id.nav_viewC);
if (cNavigationView != null)
{
cNavigationView.setNavigationItemSelectedListener(this);
}
Query mQuery= FirebaseDatabase.getInstance().getReference().child("songs");
mListView= (ListView) findViewById(R.id.canzoni_list);
FirebaseListOptions<Song> mOptions= new FirebaseListOptions.Builder<Song>()
.setLayout(R.layout.song_layout)
.setQuery(mQuery, Song.class)
.setLifecycleOwner(this)
.build();
mAdapter= new FirebaseListAdapter <Song>(mOptions){
@Override
protected void populateView(View v, Song model, int position) {
TextView songName= v.findViewById(R.id.songName);
TextView songAuthor=v.findViewById(R.id.author);
songName.setText(model.getName());
songAuthor.setText(model.getAuthor());
}
};
mListView.setAdapter(mAdapter);
}
@Override
protected void onStart() {
super.onStart();
mAdapter.startListening();
}
@Override
protected void onStop() {
super.onStop();
mAdapter.stopListening();
}
@Override
public boolean onOptionsItemSelected(MenuItem Item)
{
if (cToggle.onOptionsItemSelected(Item))
{
return true;
}
return super.onOptionsItemSelected(Item);
}
@Override
public boolean onNavigationItemSelected(MenuItem Item)
{
int id = Item.getItemId();
Intent intent;
if (id == R.id.preghiere)
{
intent= new Intent(this, Preghiere.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
this.startActivity(intent);
}
else if ( id== R.id.bans)
{
intent= new Intent(this, Bans.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
this.startActivity(intent);
}
else if (id== R.id.canzoni)
{
intent= new Intent(this, this.getClass());
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
this.startActivity(intent);
}
else if (id==R.id.calendario)
{
intent= new Intent(this, Calendario.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
this.startActivity(intent);
}
else if (id== R.id.per_riflettere)
{
intent= new Intent(this, perRiflettere.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
this.startActivity(intent);
}
else if ( id== R.id.home)
{
intent= new Intent(this, MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
this.startActivity(intent);
}
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.draw_viewC);
drawer.closeDrawer(GravityCompat.START);
return true;
}
}
макет вышеуказанной деятельности
<android.support.v4.widget.DrawerLayout xmlns:android = "http://schemas.android.com/apk/res/android"
xmlns:app = "http://schemas.android.com/apk/res-auto"
xmlns:tools = "http://schemas.android.com/tools"
android:layout_width = "match_parent"
android:layout_height = "match_parent"
tools:context = ".Canzoni"
android:id = "@+id/draw_viewC">
<ListView
android:layout_width = "match_parent"
android:layout_height = "match_parent"
android:id = "@+id/canzoni_list">
</ListView>
<android.support.design.widget.NavigationView
android:layout_width = "239dp"
android:layout_height = "match_parent"
android:layout_gravity = "start"
android:background = "@color/white"
android:fitsSystemWindows = "true"
app:headerLayout = "@layout/header"
app:itemIconTint = "@color/black"
app:itemTextColor = "@color/black"
app:menu = "@menu/drawem_menu"
tools:layout_editor_absoluteX = "0dp"
tools:layout_editor_absoluteY = "0dp"
tools:ignore = "MissingConstraints"
android:id = "@+id/nav_viewC"/>
</android.support.v4.widget.DrawerLayout>
Есть предложения, как это исправить?
Редактировать:
база данных выглядит так:
и правила письма и чтения верны
Редактировать
Я изменил частные имена в классе Song, чтобы они совпадали с базой данных.
private String Author;
private String Song;
private String Url;
он начинает работать, проблема в том, что теперь он отображает только автора
да, они читаются правильно, просто они не отображаются, и я не знаю почему. Правило базы данных справедливо как для записи, так и для чтения
Не могли бы вы добавить нулевые проверки и добавить фиктивные данные, если db возвращает null
У меня есть код для отображения данных из firebase, но он отличается от вашего кода, а также я использую recyclerview. если вы хотите предоставить мой код.
@Venky Я добавил некоторые забавные данные, например песню с одним полем в ней, она фактически отображает новое пустое поле, но не вызывает никаких ошибок или исключений, поэтому я думаю, что на самом деле не читает никаких данных в песни.
@AndroidTeam да, я хотел бы увидеть ваш код
я предоставляю свой код.
Этот код извлекает пользовательские данные из firebase и показывает данные в переработанном виде.
В макете я использовал постоянный макет, вы можете его изменить ..
display_data.xml
<?xml version = "1.0" encoding = "utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android = "http://schemas.android.com/apk/res/android"
xmlns:app = "http://schemas.android.com/apk/res-auto"
android:layout_width = "match_parent"
android:layout_height = "match_parent">
<android.support.v7.widget.RecyclerView
android:id = "@+id/rvData"
android:layout_width = "match_parent"
android:layout_height = "match_parent">
</android.support.v7.widget.RecyclerView>
<TextView
android:id = "@+id/dlTvEmpty"
android:layout_width = "wrap_content"
android:layout_height = "wrap_content"
android:text = "No Found"
android:textSize = "16dp"
android:visibility = "gone"
app:layout_constraintBottom_toBottomOf = "parent"
app:layout_constraintLeft_toLeftOf = "parent"
app:layout_constraintRight_toRightOf = "parent"
app:layout_constraintTop_toTopOf = "parent" />
</android.support.constraint.ConstraintLayout>
затем после привязки row_layout к просмотру ресайклера ..
row_layout.xml
<?xml version = "1.0" encoding = "utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android = "http://schemas.android.com/apk/res/android"
xmlns:app = "http://schemas.android.com/apk/res-auto"
android:layout_width = "match_parent"
android:layout_height = "wrap_content"
android:padding = "10dp">
<TextView
android:id = "@+id/rlTvName"
android:layout_width = "wrap_content"
android:layout_height = "wrap_content"
android:text = "Name"
android:textSize = "20dp" />
<TextView
android:id = "@+id/rlTvEmail"
android:layout_width = "wrap_content"
android:layout_height = "wrap_content"
android:text = "Email"
android:textSize = "20dp"
app:layout_constraintTop_toBottomOf = "@+id/rlTvName" />
<TextView
android:id = "@+id/rlTvPwd"
android:layout_width = "wrap_content"
android:layout_height = "wrap_content"
android:text = "Password"
android:textSize = "20dp"
app:layout_constraintTop_toBottomOf = "@+id/rlTvEmail" />
</android.support.constraint.ConstraintLayout>
затем после создания класса pojo для пользователя, который вставляет в firebase ..
User.java
public class User {
public String name;
public String email;
public String pwd;
// Default constructor required for calls to
// DataSnapshot.getValue(User.class)
public User() {
}
public User(String name, String email,String pwd) {
this.name = name;
this.email = email;
this.pwd=pwd;
}
}
затем после создания класса адаптера.
DisplayAllData.java
public class DisplayAllData extends RecyclerView.Adapter<DisplayAllData.ItemViewHolder> {
private List<User> mUserLsit = new ArrayList<>();
private Context mContext;
@Override
public ItemViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.row_layout, parent, false);
return new ItemViewHolder(view);
}
public DisplayAllData(Context mContext, List<User> mUserLsit) {
this.mContext = mContext;
this.mUserLsit = mUserLsit;
}
@Override
public void onBindViewHolder(ItemViewHolder holder, int position) {
User user = mUserLsit.get(position);
holder.mTvName.setText(user.name);
holder.mTvEmail.setText(user.email);
holder.mTvPwd.setText(user.pwd);
}
@Override
public int getItemCount() {
return mUserLsit.size();
}
public class ItemViewHolder extends RecyclerView.ViewHolder {
TextView mTvName, mTvEmail, mTvPwd;
public ItemViewHolder(View itemView) {
super(itemView);
mTvEmail = itemView.findViewById(R.id.rlTvEmail);
mTvName = itemView.findViewById(R.id.rlTvName);
mTvPwd = itemView.findViewById(R.id.rlTvPwd);
}
}
}
затем, наконец, сделать класс отображения для извлечения пользовательской записи из firebase и привязки к recyclerview.
DisplayActivity.java
public class DisplayActivity extends AppCompatActivity {
private RecyclerView mRvData;
private DisplayAllData allDataAdapter;
private DatabaseReference mDatabase;
private TextView mTvEmpty;
private FirebaseDatabase mFirebaseInstance;
private List<User> mUserList = new ArrayList<>();
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.display_data);
initView();
}
private void initView() {
mFirebaseInstance = FirebaseDatabase.getInstance();
mDatabase = mFirebaseInstance.getReference("usersDb/UserTable");
mRvData = findViewById(R.id.rvData);
mTvEmpty = findViewById(R.id.dlTvEmpty);
mRvData.setLayoutManager(new LinearLayoutManager(this));
mDatabase.addValueEventListener(new ValueEventListener() {
@Override
public void onDataChange(DataSnapshot dataSnapshot) {
mUserList.clear();
for (DataSnapshot dataSnapshot1 : dataSnapshot.getChildren()) {
User user = dataSnapshot1.getValue(User.class);
mUserList.add(user);
}
allDataAdapter = new DisplayAllData(DisplayActivity.this, mUserList);
mRvData.setAdapter(allDataAdapter);
allDataAdapter.notifyDataSetChanged();
if (mUserList.isEmpty())
mTvEmpty.setVisibility(View.VISIBLE);
else
mTvEmpty.setVisibility(View.GONE);
}
@Override
public void onCancelled(DatabaseError databaseError) {
}
});
}
}
Просто попробовал, но не лучше Я думаю, что делаю ошибки в самой базе данных или при получении ссылок
Вышеупомянутый код работает для меня, также я надеюсь, что вы добавите разрешение в Интернет в файл манифеста Android .. вы можете ссылаться на эту ссылку для лучшего представления simpleifiedcoding.net/firebase-realtime-database-crud
В вашей базе данных у вас есть свойство под названием песни под названием myAuthor
, но в классе модели у вас есть поле с именем Author
, оба должны быть одинаковыми.
Чтобы решить эту проблему, у вас есть два решения. Во-первых, нужно удалить все записи из вашей базы данных и добавить свежие данные в соответствии с классом вашей модели. Убедитесь, что свойства в вашей базе данных добавлены с использованием имени вашего поля, которое существует в вашем классе модели: myName
, myURL
, myAuthor
и НЕТAuthor
, Song
, Url
.
Если вы не находитесь в режиме тестирования, просто используйте аннотацию в своем классе модели следующим образом:
@PropertyName("Song")
private String myName;
@PropertyName("Url")
private String myURL;
@PropertyName("Author")
private String myAuthor;
Согласно вашему отредактированному вопросу, ваши поля неверны. Пожалуйста, как должен выглядеть ваш класс Song
:
public class Song {
private String author, song, url;
public Song() {}
public Song(String author, String song, String url) {
this.author = author;
this.song = song;
this.url = url;
}
public String getAuthor() { return author; }
public String getSong() { return song; }
public String getUrl() { return url; }
}
Видите поля? Называются с первой буквы в нижнем регистре.
Он начинает работать, но теперь отображает только одно поле, только автора, а не название песни
myName
- это соответствующее имя поля для названия песни?
да, они находятся в том же порядке, что и поля базы данных, и теперь у них одинаковые имена
Какое решение вы используете, первое или второе?
Я использовал ваше решение.
Какой, первый (который удаляет все данные) или второй с аннотациями?
Я изменил имена своих частных лиц, чтобы они совпадали с полями базы данных.
Хорошо, вы использовали мое первое решение. Не могли бы вы отредактировать свой вопрос, добавив снимок экрана с вашей новой базой данных и кодом, который вы используете? Пожалуйста, добавьте только соответствующий код, а не весь класс.
Это не так. Пожалуйста, посмотрите мой обновленный ответ. Удалите свои данные еще раз, добавьте свежие данные и попробуйте еще раз. А теперь работает?
После нескольких экспериментов с кодом вроде измените это
songName.setText(model.getName());
songAuthor.setText(model.getAuthor());
к этому
songName.setText(model.getURL());
songAuthor.setText(model.getAuthor());
Я узнал, что проблема на самом деле заключалась в именах, которые я дал рядовым, и в именах, которые я дал методам get в классе Song. Имена частных лиц должны совпадать с полями базы данных, а имена методов должны быть getYourPrivateName ().
вы успешно читаете данные из базы данных firebase. потому что я думаю, что в вашем коде есть ошибка при извлечении данных из базы данных firebase.