In Firebase recycler adapter onBindViewHolder and

2019-08-23 05:33发布

问题:

I am writing an android application where I needs to show a list of data using recycler view . I am using firebase realtime database with read write rules true , but the problem is onBindViewHolder and onCreateViewHolder are never called as the log says .

Modal class : ConnectedListItem.java

public class ConnectedListItem {

String name;
Double lat , lng;

public ConnectedListItem() {
}

public ConnectedListItem(String name, Double lat, Double lng) {
    this.name = name;
    this.lat = lat;
    this.lng = lng;
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}

public Double getLat() {
    return lat;
}

public void setLat(Double lat) {
    this.lat = lat;
}

public Double getLng() {
    return lng;
}

public void setLng(Double lng) {
    this.lng = lng;
}
}

ConnectionActivity.xml :

<LinearLayout 
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"
android:orientation="vertical"
tools:context=".ConnectionActivity">

<include
    android:id="@+id/connection_toolbar"
    layout="@layout/toolbar_layout" />

<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_marginStart="20dp"
    android:layout_marginBottom="15dp"
    android:fontFamily="@font/open_sans"
    android:text="Connection"
    android:textAllCaps="false"
    android:textSize="35sp"
    android:textStyle="bold" />

<RelativeLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_weight="1">

    <TextView
        android:id="@+id/emptyTextView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:text="No Connection"
        android:visibility="invisible" />


    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/connected_list_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

 </RelativeLayout>

 <Button
    android:id="@+id/connection_child_add_btn"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="15dp"
    android:background="@drawable/login_screen_button_background"
    android:text="Add Child"
    android:textColor="@color/white" />

 </LinearLayout>

Single user Layout : connected_single_layout.xml

<LinearLayout 
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="10dp"
android:background="@drawable/login_sign_up_edit_text_background"
android:orientation="vertical">

<TextView
    android:id="@+id/connectedName"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_marginStart="20dp"
    android:layout_marginEnd="20dp"
    android:text="Name" />

<TextView
    android:id="@+id/connectedLastLocation"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_marginStart="20dp"
    android:layout_marginTop="10dp"
    android:layout_marginEnd="20dp"
    android:text="Last Location" />

<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="16dp"
    android:orientation="horizontal"
    android:weightSum="2">

    <Button
        android:id="@+id/removeBtn"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_marginEnd="6dp"
        android:layout_weight="1"
        android:text="Remove" />

    <Button
        android:id="@+id/trackB only oncetn"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_marginStart="6dp"
        android:layout_weight="1"
        android:text="Track" />

</LinearLayout>

</LinearLayout>

ConnecttionActivity.java :

public class ConnectionActivity extends AppCompatActivity {

private String TAG = "ConnectionActivity";

private RecyclerView mChildConnectedListRecyclerView;
private androidx.appcompat.widget.Toolbar mToolbar;

private TextView mEmptyTextView;

private DatabaseReference mConnectedUserDatabase , mUserDatabase;
private FirebaseAuth mAuth;
private String mCurrentUser;

private Button mAddChildBtn;

private Geocoder geocoder;

FirebaseRecyclerAdapter<ConnectedListItem , ConnectedViewHolder> adapter;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_connection);

    Log.d(TAG, "onCreate: called");

    mAuth = FirebaseAuth.getInstance();

    setUpToolbar();

    if(mAuth.getCurrentUser() != null){
        mCurrentUser = mAuth.getCurrentUser().getUid();
        Log.d("CurrentUser", mCurrentUser);
    }

    mAddChildBtn = (Button) findViewById(R.id.connection_child_add_btn);
    mConnectedUserDatabase = FirebaseDatabase.getInstance().getReference().child("Connected").child(mCurrentUser);

    Log.d(TAG, "onCreate: " + mCurrentUser);

    mUserDatabase = FirebaseDatabase.getInstance().getReference().child("Users");

    mChildConnectedListRecyclerView = (RecyclerView) findViewById(R.id.connected_list_view);
    mChildConnectedListRecyclerView.setHasFixedSize(true);
    mChildConnectedListRecyclerView.setLayoutManager(new LinearLayoutManager(this));


    mEmptyTextView = (TextView) findViewById(R.id.emptyTextView);

    mAddChildBtn.setOnClickListener(v -> {

        // Some code here

    });

}

 @Override
 protected void onStart() {
    super.onStart();

    Log.d(TAG, "onStart: called");

    FirebaseRecyclerOptions options = new FirebaseRecyclerOptions.Builder<ConnectedListItem>()
                                                .setQuery(mConnectedUserDatabase , ConnectedListItem.class)
                                                .build();

    Log.d(TAG, "onStart: Firebase option done");

    adapter = new FirebaseRecyclerAdapter<ConnectedListItem, ConnectedViewHolder>(options) {
                @Override
                protected void onBindViewHolder(@NonNull ConnectedViewHolder connectedViewHolder, int i, @NonNull ConnectedListItem connectedListItem) {


                    Log.d(TAG, "onBindViewHolder: called" + connectedListItem.getName());

                     connectedViewHolder.mChildName.setText(connectedListItem.getName());

                    Double mLat = connectedListItem.getLat();
                    Double mLng = connectedListItem.getLng();

                    // geocoding starts here
                    String errorMessage = "";

                    geocoder = new Geocoder(getApplicationContext(), Locale.getDefault());

                    List<Address> addresses = null;

                    try {
                        addresses = geocoder.getFromLocation(
                                    mLat,
                                    mLng,
                                // In this sample, get just a single address.
                                1);
                    } catch (IOException e) {
                        errorMessage = getString(R.string.service_not_available);
                        Log.e(TAG, errorMessage, e);
                    } catch (IllegalArgumentException illegalArgumentException) {
                        // Catch invalid latitude or longitude values.
                        errorMessage = getString(R.string.invalid_lat_long_used);
                        Log.e(TAG, errorMessage + ". " +
                                "Latitude = " + mLat +
                                ", Longitude = " +
                                mLng, illegalArgumentException);
                    }

                    // Handle case where no address was found.
                    if (addresses == null || addresses.size() == 0) {
                        if (errorMessage.isEmpty()) {
                            errorMessage = getString(R.string.no_address_found);
                            Log.e(TAG, errorMessage);
                        }

                    } else {
                    Address address = addresses.get(0);
                        ArrayList<String> addressFragments = new ArrayList<String>();

                        // Fetch the address lines using getAddressLine,
                        // join them, and send them to the thread.
                        for (int j = 0; j <= address.getMaxAddressLineIndex(); j++) {
                            addressFragments.add(address.getAddressLine(j));
                        }
                        // Log.i(TAG, getString(R.string.address_found));

                        connectedViewHolder.mChildAddress.setText(TextUtils.join(System.getProperty("line.separator"),
                                addressFragments));

                    }

                }

                @NonNull
                @Override
                public ConnectedViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {

                    Log.d(TAG, "onCreateViewHolder: called");

                    View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.connected_single_layout , parent, false);

                    ConnectedViewHolder connectedViewHolder = new ConnectedViewHolder(view);


                    return connectedViewHolder;
                }

        @Override
        public int getItemCount() {
            return super.getItemCount();
        }
    };

    mChildConnectedListRecyclerView.setAdapter(adapter);

    Log.d(TAG, "onStart: Adapter setuped");
    adapter.startListening();
    Log.d(TAG, "onStart: Start listening called");

}

@Override
protected void onDestroy() {
    super.onDestroy();

    adapter.stopListening();

    Log.d(TAG, "onDestroy: Called Stop");
}

@Override
protected void onStop() {
    super.onStop();

    adapter.stopListening();
}

@Override
protected void onResume() {
    super.onResume();

    adapter.startListening();

    Log.d(TAG, "onResume: called resume");
}

public static class ConnectedViewHolder extends RecyclerView.ViewHolder {

    TextView mChildName , mChildAddress;

    Button mTrackBtn , mRemoveBtn;

    private ConnectedViewHolder(@NonNull View itemView) {
        super(itemView);

        mChildAddress = itemView.findViewById(R.id.connectedLastLocation);
        mChildName = itemView.findViewById(R.id.connectedName);

        mTrackBtn = (Button) itemView.findViewById(R.id.trackBtn);
        mRemoveBtn = (Button) itemView.findViewById(R.id.removeBtn);
    }

}
}

App-level build.gradle :

//Firebase dependencies here
implementation 'com.google.firebase:firebase-core:16.0.9'
implementation 'com.google.firebase:firebase-auth:17.0.0'
implementation 'com.google.firebase:firebase-database:17.0.0'
implementation 'com.firebaseui:firebase-ui-database:5.0.0'

Logcat :

ConnectionActivity: onCreate: called
ConnectionActivity: onCreate: 8zQPcYnUHnQBtRetg4vkylXJI1O2
ConnectionActivity: onStart: called
ConnectionActivity: onStart: Firebase option done
ConnectionActivity: onStart: Adapter setuped
ConnectionActivity: onStart: Start listening called
ConnectionActivity: onResume: called resume
Logging event (FE): screen_view(_vs), 
           Bundle[{firebase_event_origin(_o)=auto, 
           firebase_previous_class(_pc)=ParentHomeActivity, 
           firebase_previous_id(_pi)=4706309106774484658, 
           firebase_screen_class(_sc)=ConnectionActivity, 
           firebase_screen_id(_si)=4706309106774484659}]
Logging event (FE): user_engagement(_e), 
           Bundle[{firebase_event_origin(_o)=auto, 
                   engagement_time_msec(_et)=8849, 
           firebase_screen_class(_sc)=ConnectionActivity, 
           firebase_screen_id(_si)=4706309106774484659}]
Logging event (FE): screen_view(_vs), 
           Bundle[{firebase_event_origin(_o)=auto, 
           firebase_previous_class(_pc)=ConnectionActivity, 
           firebase_previous_id(_pi)=4706309106774484659, 
           firebase_screen_class(_sc)=ParentHomeActivity, 
           firebase_screen_id(_si)=4706309106774484658}]
ConnectionActivity: onDestroy: Called Stop

I searched other questions on stackOverflow , but none of them worked for me . Please help

回答1:

I finally solved it by removing this line mChildConnectedListRecyclerView.setHasFixedSize(true);