onCreateView результат запросов ожидания

Мне нужно выполнить запрос к моей БД, чтобы получить некоторые идентификаторы, а затем использовать эти идентификаторы, чтобы выполнить другой запрос к базе данных Realm и вернуть результат в адаптер, который используется для создания пользовательского интерфейса. Моя проблема заключается в том, что адаптер создается в onCreatView (основной поток), и мне нужно, чтобы результаты ожидания ожидания. Ты знаешь, как я могу это сделать? Большое спасибо:)

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

    Realm realm = Realm.getDefaultInstance();
    cardList = new ArrayList(realm.where(Card.class).findAll().sort("rarity"));
    View rootView = inflater.inflate(R.layout.inventory_fragment_layout, null);
    RecyclerView recyclerView = rootView.findViewById(R.id.inventory_recycler);
    recyclerView.setLayoutManager(new GridLayoutManager(this.getContext(), 2));
    adapter = new inventoryFragmentRecyclerAdapter(this.getContext(), getCards()); <----- getCards() is used here.
    adapter.setClickListener(this);
    recyclerView.setAdapter(adapter);
    TextView topText = rootView.findViewById(R.id.inv_topText);
    topText.setTypeface(Typeface.createFromAsset(getActivity().getAssets(), "fonts/Square.ttf"));

    return rootView;
}



.
.
.
.
.
.



public List<Card> getCards() { //It is used when adapter is created, and it need to contain all cards to display

    //Realm realm = Realm.getDefaultInstance();

    String urlToGet = "myurl";
    String user_name = settings.getString("username", null);

    OkHttpClient client = new OkHttpClient();

    RequestBody formBody = new FormBody.Builder()
            .add("user_name", user_name)
            .build();

    Request request = new Request.Builder()
            .url(urlToGet)
            .post(formBody)
            .build();

    client.newCall(request).enqueue(new Callback() {
        @Override
        public void onFailure(Call call, IOException e) {
            e.printStackTrace();
        }

        @Override
        public void onResponse(Call call, final Response response) throws IOException {
            if (!response.isSuccessful()) {
                throw new IOException("Unexpected code " + response);
            }

            responseCardInInventory = response.body().string();

            handler.post(new Runnable() {
                @Override
                public void run() {

                    List<CardInInv> cardListInInv = new ArrayList<>();

                    Gson gson = new Gson();
                    CardInInventory civ = gson.fromJson(responseCardInInventory, CardInInventory.class);
                    cardListInInv = getCards(civ);

                    //HERE I NEED TO USE REALM TO EXECUTE A QUERY USING cardListInInv


                }
            });
        }
    });

    //HERE I NEED TO RETURN THE RESULT OF THE QUERY EXECUTED BEFORE
    return cardsIninv;

}

РЕДАКТИРОВАТЬ 1: Hre мой код адаптера

public class inventoryFragmentRecyclerAdapter extends RecyclerView.Adapter<inventoryFragmentRecyclerAdapter.ViewHolder> {

private List<Card> dataCard = new ArrayList<>();
private LayoutInflater mInflater;
private ItemClickListener mClickListener;


// data is passed into the constructor
public inventoryFragmentRecyclerAdapter(Context context, List<Card> data) {
    this.mInflater = LayoutInflater.from(context);
    this.dataCard = data;
}

// inflates the cell layout from xml when needed
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View view = mInflater.inflate(R.layout.inventory_rw, parent, false);
    return new ViewHolder(view);
}

// binds the data to the in each cell
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
    holder.cardImage.setImageBitmap(BitmapFactory.decodeByteArray(dataCard.get(position).getCardImage(), 0, dataCard.get(position).getCardImage().length));
}

// total number of cells
@Override
public int getItemCount() {
    return dataCard.size();
}


// stores and recycles views as they are scrolled off screen
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
    ImageView cardImage;

    //instanzio le componenti della pagina

    ViewHolder(View itemView) {
        super(itemView);

        cardImage = itemView.findViewById(R.id.inv_rw_cardimage);
        itemView.setOnClickListener(this);

    }

    @Override
    public void onClick(View view) {
        if (mClickListener != null) mClickListener.onItemClick(view, getAdapterPosition());
    }
}

// convenience method for getting data at click position
Card getItem(int id) {
    return dataCard.get(id);
}

// allows clicks events to be caught
public void setClickListener(ItemClickListener itemClickListener) {
    this.mClickListener = itemClickListener;
}

// parent activity will implement this method to respond to click events
public interface ItemClickListener {
    void onItemClick(View view, int position);
}
}

3 ответа

Решение

Переместите запрос на адаптер и запустите его с пустым списком. Сделайте ваш запрос вызова в конце вашего конструктора. Когда вы успешно завершили свой запрос, сообщите адаптеру, что он изменился, и дайте ему обработать изменения с вашими новыми данными в конце ваших запросов, которые будут успешно завершены. Таким образом, после этого ваш адаптер будет выглядеть примерно так.

public inventoryFragmentRecyclerAdapter(Context context, List<Card> data) {
    this.mInflater = LayoutInflater.from(context);
    getCards();
}

public void getCards() { //It is used when adapter is created, and it need to contain all cards to display
//Realm realm = Realm.getDefaultInstance();

String urlToGet = "myurl";
String user_name = settings.getString("username", null);

OkHttpClient client = new OkHttpClient();

RequestBody formBody = new FormBody.Builder()
        .add("user_name", user_name)
        .build();

Request request = new Request.Builder()
        .url(urlToGet)
        .post(formBody)
        .build();

client.newCall(request).enqueue(new Callback() {
    @Override
    public void onFailure(Call call, IOException e) {
        e.printStackTrace();
    }

    @Override
    public void onResponse(Call call, final Response response) throws IOException {
        if (!response.isSuccessful()) {
            throw new IOException("Unexpected code " + response);
        }

        responseCardInInventory = response.body().string();

        handler.post(new Runnable() {
            @Override
            public void run() {

                List<CardInInv> cardListInInv = new ArrayList<>();

                Gson gson = new Gson();
                CardInInventory civ = gson.fromJson(responseCardInInventory, CardInInventory.class);
                cardListInInv = getCards(civ);

                //use realm to do all your card inventory stuff to get your resulting object
                dataCard = yourObjectListAfterUsingRealm;
                notifyDataSetChanged();
            }
        });
    }
});
}

Переместить эту линию adapter = new inventoryFragmentRecyclerAdapter(context, yourList); внутри onResponse() и передать список напрямую без getCards();

Редактировать:

инициализируйте свой адаптер, когда ваши данные будут готовы. Где? это основано на ваших потребностях.

или инициализировать его пустым списком и вызвать adapter.notifyDataSetChanged() когда что-то изменилось (добавлено или удалено из списка или инициализирован новый список)

Создайте свой адаптер с пустым списком
adapter = new inventoryFragmentRecyclerAdapter(this.getContext(), new ArrayList<Card>());

а затем в вашем адаптере создать такой метод

public void updateCardList(List<Card> cardList) { this.mCards = cardList; notifyDataSetChanged(); }

и вызвать эту функцию в
@Override public void onResponse(Call call, final Response response)

Другие вопросы по тегам