android - 在 recyclerview 上的方向更改期间再次从互联网加载数据

标签 android

有 RecyclerView 并且在方向更改期间我希望数据不会再次从 Internet 加载。你是怎样做的?

这是代码

    package picodiploma.dicoding.database.picodiploma.dicoding.database.tv;


import android.content.Intent;
import android.os.Bundle;
import android.os.Parcelable;
import android.provider.Settings;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ProgressBar;

import java.util.ArrayList;
import java.util.List;

import picodiploma.dicoding.database.ApiClient;
import picodiploma.dicoding.database.ApiInterface;
import picodiploma.dicoding.database.R;
import retrofit2.Call;
import retrofit2.Callback;

import static android.view.View.VISIBLE;


/**
 * A simple {@link Fragment} subclass.
 */
public class TvShow extends Fragment  {

    private static RecyclerView recyclerView;
    private static final String API_KEY = "2e08750083b7e21e96e915011d3f8e2d";
    private static final String TAG = TvShow.class.getSimpleName();
    private static ProgressBar progressBar;
    private static List<ResultsItem> resultsTv = new ArrayList<>();
    private static TvShowRecyclerAdapter recyclerAdapter;


    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

    }

    public TvShow() {
        // Required empty public constructor
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
        final View view = inflater.inflate(R.layout.fragment_tv_show, container, false);

        recyclerView = view.findViewById(R.id.recycler_tv);
        progressBar = view.findViewById(R.id.progress_bar);

        progressBar.setVisibility(VISIBLE);


        RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(getContext());
        recyclerView.setLayoutManager(layoutManager);

        getData();

        if (savedInstanceState == null) {
            getData();
        }
        else  {
            resultsTv = savedInstanceState.getParcelableArrayList("tv");
            recyclerAdapter.refill(resultsTv);
        }


        return view;
    }

    @Override
    public void onSaveInstanceState(@NonNull Bundle outState) {
        outState.putParcelableArrayList("tv", (ArrayList<? extends Parcelable>) resultsTv);
        super.onSaveInstanceState(outState);
    }

    @Override
    public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
        // TODO Auto-generated method stub
        super.onCreateOptionsMenu(menu, inflater);
        inflater.inflate(R.menu.menu, menu);
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        if (item.getItemId() == R.id.action_change_settings){
            Intent mIntent = new Intent(Settings.ACTION_LOCALE_SETTINGS);
            startActivity(mIntent);
        }
        return super.onOptionsItemSelected(item);
    }

    public void getData() {
        ApiInterface apiInterface = ApiClient.getList().create(ApiInterface.class);

        Call<Response> responseCall = apiInterface.getTvList(API_KEY);
        responseCall.enqueue(new Callback<Response>() {
            @Override
            public void onResponse(Call<Response> call, retrofit2.Response<Response> response) {
                resultsTv = response.body().getResults();
                recyclerAdapter = new TvShowRecyclerAdapter(resultsTv, getContext());
                recyclerView.setAdapter(recyclerAdapter);
                recyclerView.setVisibility(VISIBLE);
                progressBar.setVisibility(View.GONE);
            }

            @Override
            public void onFailure(Call<Response> call, Throwable t) {
                progressBar.setVisibility(View.GONE);
                Log.d(TAG, t.toString());
            }
        });
    }


}

现在,方向改变后,RecyclerView 再次从互联网加载数据。我想让它使用以前保存在 fragment 上的数据。顺便说一句, fragment 也连接到 MainActivity 上的 FragmentPagerAdapter。感谢您的回答。

最佳答案

处理此类情况(处理方向更改)而无需从网络重新获取数据的推荐方法是使用来自 google 的 Android 架构组件的 ViewModel。

你的 ViewModel 可以像下面这样

public class YourViewModel extends ViewModel {

private MutableLiveData<List<ResultsItem>> resultsItem;

public LiveData<List< ResultsItem >> getData() {
    if (resultsItem == null) {
        resultsItem = new MutableLiveData<List<resultsItem>>();
        loadData();
    }
    return resultsItem;
}

private void loadData() {
    // Request your data from network
}
}

然后在您的 fragment

YourViewModel model = ViewModelProviders.of(this).get(YourViewModel.class);
    model.getData().observe(this, resultsItem -> {
        // here you can update UI - passing the list of data
    });

关于android - 在 recyclerview 上的方向更改期间再次从互联网加载数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56838586/

相关文章:

android - react-native-sound 不适用于 iOS 的生产构建,但适用于模拟器和生产 android

android - 在运行时从位图列表中选择可绘制对象的简洁方法?

java - 错误:(144) Attribute "srcCompat" has already been defined

java - 用户的屏幕通知

android - 什么时候在 Android 中使用 HTML5?什么时候不用?

android - 在同一设备上运行的单独版本?

android - 显示原生广告时出现 java.util.concurrent.RejectedExecutionException

android - 陀螺为什么叫陀螺?

java - 从 Android 设备上的 libGDX 引擎上的游戏发送电子邮件?

Android:按钮不显示在布局中