Каков наилучший способ обновить sh список с помощью Firebase - PullRequest
0 голосов
/ 07 марта 2020

То, как я сейчас обновляю свои списки в своем проекте, явно не является стандартом для приложений. Как мне обновить sh список без его очистки?

Этот код пишет просто отлично, без каких-либо перерывов; однако, как только пользователь закончил редактирование, список очищается и обновляется, возвращая пользователя в начало списка. Что лучше всего делать при обновлении данных, особенно если данные редактируются другим пользователем без прерывания работы текущего пользователя.

Запись:

protected void addStep() {
String stepID = Database.push().getKey();
step newStep = new step(recipeID, stepID, "stepImage", "","");
Database.child(stepID).setValue(newStep);
getData();

}

Адаптер:

package asdasd.asd;

import android.app.Activity;
import android.support.annotation.NonNull;
import android.text.Editable;
import android.text.TextWatcher;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.TextView;

import com.google.firebase.database.DatabaseReference;
import com.google.firebase.database.FirebaseDatabase;

import java.util.List;
import java.util.Timer;
import java.util.TimerTask;

/**
 * This class is the class responsible for handling the lists of steps that the user will see during the creator phase.
 * It works by setting an array adapter which uses the fragment_steps layout displaying it on a list within the StepActivity
 * On text listeners are on each of the fields, when the user edits one of the fields, the program waits 600ms, and then uploads the
 * data to the database;
 * this refreshes the list TODO change the way the list refreshes in the instance of the creator; perhaps add a delay of 1 minuite before refresh, and or if a new step has been added
 * A timer is responsible for this delay. The setting of data is to a specific path; being the recipe -> step -> long/shortText field.
 */

public class stepList extends ArrayAdapter<step>{
    private Activity context;
    private List<step> steps;
    private DatabaseReference database;
    private Timer timer1,timer2;

    public stepList(Activity context, List<step> steps) {
        super(context, R.layout.fragment_step, steps);
        this.context = context;
        this.steps = steps;
    }

    @NonNull
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        //get database
        database = FirebaseDatabase.getInstance().getReference("steps");

        //init Layout inflater
        LayoutInflater inflater = context.getLayoutInflater();

        //step
        View listViewItem = inflater.inflate(R.layout.fragment_step,null,true);

        //step objects
        final TextView descriptionText = (TextView) listViewItem.findViewById(R.id.stepRecipeShortText);
        final TextView longDescriptionText = (TextView) listViewItem.findViewById(R.id.stepRecipeLongText);
        ImageView stepImage = listViewItem.findViewById(R.id.stepImage);
        //init step
        final step step = steps.get(position);

        //get stepID
        final String stepID = step.getStepID();

        //Set Data
        descriptionText.setText(step.getStepDescription());
        longDescriptionText.setText(step.getStepLongDescription());

        //TODO If user has uploaded an image, then use that, else then use default

        //Add listener to descriptionText so that when a user edits the fields, it is uploaded to the same step in the database
        descriptionText.addTextChangedListener(new TextWatcher() {

            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }
            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                // user is typing: reset already started timer (if existing)
                if (timer1 != null) {
                    timer1.cancel();
                }
            }
            @Override
            public void afterTextChanged(Editable s) {
                timer1 = new Timer();
                timer1.schedule(new TimerTask() {
                    @Override
                    public void run() {
                        String newDescriptionText = descriptionText.getText().toString();
                        addToDatabase(stepID,"stepDescription", newDescriptionText);
                    }
                }, 600);
            }
        });
        //Add listener to LongDescriptionText so that when a user edits the fields, it is uploaded to the same step in the database
        longDescriptionText.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }
            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                // user is typing: reset already started timer (if existing)
                if (timer2 != null) {
                    timer2.cancel();
                }
            }
            @Override
            public void afterTextChanged(Editable s) {
                timer2 = new Timer();
                timer2.schedule(new TimerTask() {
                    @Override
                    public void run() {
                        String newLongDescriptionText = longDescriptionText.getText().toString();
                        addToDatabase(stepID, "stepLongDescription", newLongDescriptionText);
                    }
                }, 600);
            }
        });
        return listViewItem;
    }
    //Add the data the user is entering to the database; there is a 600ms delay on the period between the user stopping typing and the data being updated.
    private void addToDatabase(String id, String location, String text) {
        database.child(id).child(location).setValue(text);
    }
}

Получение:

public void getData() {
        //receives all the recipes and adds them to the list
        Database.addValueEventListener(new ValueEventListener() {
            @Override
            public void onDataChange(DataSnapshot dataSnapshot) {
                //Clear the list
                listStepsList.clear();
                //Iterate through the nodes
                for(DataSnapshot stepSnapshot : dataSnapshot.getChildren()){
                    //get recipe
                    step step = stepSnapshot.getValue(step.class);
                    //add step to list, if it is apart of the same recipe.
                    if(step.getRecipeID().equals(recipeID)) {
                        listStepsList.add(step);
                    }
                }
                //create Adapter
                stepList stepAdapter = new stepList(StepActivity.this, listStepsList);
                //Attatch adapter to listview
                viewStepsList.setAdapter(stepAdapter);
                stepAdapter.notifyDataSetChanged();
            }
            @Override
            public void onCancelled(DatabaseError databaseError) {

            }
        });
    }

1 Ответ

1 голос
/ 07 марта 2020

Я предполагаю, что вы видите "большой взрыв" / fla sh всякий раз, когда происходит изменение в базе данных. Если это так, то это потому, что вы обновляете весь список, даже если был изменен только один элемент в данных.

Чтобы улучшить это, вам нужно более детально обновить адаптер для изменений. Для этого вы можете прикрепить ChildEventListener, который запускает события на один уровень ниже в вашей структуре данных. Так, скажем, новый узел добавлен в ваш список, вместо того, чтобы перестраивать весь список для добавления одного узла, вы получите один вызов onChildAdded только с одним новым узлом. Затем вы обновите listStepsList вместо его перекомпоновки и расскажете адаптеру об изменениях.

В качестве примера я рекомендую проверить FirebaseUI, как все адаптеры используют этот шаблон. Они строятся из общего FirebaseArray класса , который наблюдает за базой данных, а затем имеют классы адаптера для склеивания массива с пользовательским интерфейсом. Например, вот , как FirebaseRecyclerAdapter связывает изменения в базе данных с минимальными обновлениями для представления .

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...