ショートバージョン:onResume()
私は2つの他のフラグメントを表示するためのViewPager
を維持フラグメントを持っている、のは彼らFragmentOne
とFragmentTwo
を呼びましょう。アプリを起動すると、はオフスクリーンになり、ビューを左にスワイプすると表示されるようになります。FragmentOne
が表示されます。
通常、アプリケーションが開始されるとすぐに、両方のフラグメントに対してすぐにonStart()
とonResume()
が呼び出されます。
私が持っている問題はFragmentOne
は、カスタムLoader
を開始したとき、それは完全に見えるようになるまで、onResume()
がFragmentTwo
に呼び出されませんです。
質問:
これは私のコードやAndroidサポートライブラリのバグに問題ですか? (問題はライブラリのリビジョン12では発生しませんでした。リビジョン13から始まりました)
リビジョン13とリビジョン18のバグなら、回避策がありますか?
私のカスタムLoader
に何か問題がありますか?
ロングバージョン:私は、問題を示すサンプルアプリケーションを構築している
。私はコードを最小限に抑えようとしましたが、それでもまだたくさんありますので、私と一緒にご負担ください。
MainActivity
にはMainFragment
がロードされ、ViewPager
が作成されています。私のアプリケーションでは、ViewPagerはアクティビティではなくフラグメントによって維持されることが重要です。
MainFragment
は、順番に断片FragmentOne
とFragmentTwo
を作成FragmentPagerAdapter
を作成します。
のは面白いビットから始めましょうは、二つの断片:
FragmentOne
は、コンテンツをロードするためのカスタムLoader
を使用していますListFragment
です:
public class FragmentOne extends ListFragment implements LoaderCallbacks<List<String>> {
private ArrayAdapter<String> adapter;
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
adapter = new ArrayAdapter<String>(getActivity(), android.R.layout.simple_list_item_1);
setListAdapter(adapter);
setEmptyText("Empty");
}
@Override
public void onResume() {
super.onResume();
// initializing the loader seems to cause the problem!
getLoaderManager().initLoader(0, null, this);
}
@Override
public Loader<List<String>> onCreateLoader(int id, Bundle args) {
return new MyLoader(getActivity());
}
@Override
public void onLoadFinished(Loader<List<String>> loader, List<String> data) {
adapter.clear();
adapter.addAll(data);
}
@Override
public void onLoaderReset(Loader<List<String>> loader) {
adapter.clear();
}
public static class MyLoader extends AsyncTaskLoader<List<String>> {
public MyLoader(Context context) {
super(context);
}
@Override
protected void onStartLoading() {
forceLoad();
}
@Override
public List<String> loadInBackground() {
return Arrays.asList("- - - - - - - - - - - - - - - - - - - foo",
"- - - - - - - - - - - - - - - - - - - bar",
"- - - - - - - - - - - - - - - - - - - baz");
}
}
}
それは問題を引き起こすように見えることLoader
ということです。 initLoader行をコメントアウトすると、フラグメントのライフサイクルが再び期待通りに動作します。
FragmentTwo
はonResume()
が呼び出されているか否かに基づいて、その内容に変更:
public class FragmentTwo extends Fragment {
private TextView text;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
text = new TextView(container.getContext());
text.setText("onCreateView() called");
return text;
}
@Override
public void onResume() {
super.onResume();
Log.i("Fragment2", "onResume() called");
text.setText("onResume() called");
}
}
をそしてここでのコードの退屈残りです。
MainActivity
:
public class MainActivity extends FragmentActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Fragment fragment = new MainFragment();
getSupportFragmentManager().beginTransaction().add(R.id.container, fragment).commit();
}
}
レイアウトactivity_main
:
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent" />
MainFragment
:
public class MainFragment extends Fragment {
private ViewPager viewPager;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View layout = inflater.inflate(R.layout.frag_master, container, false);
viewPager = (ViewPager) layout.findViewById(R.id.view_pager);
return layout;
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
viewPager.setAdapter(new MyPagerAdapter(getChildFragmentManager()));
}
private static final class MyPagerAdapter extends FragmentPagerAdapter {
public MyPagerAdapter(FragmentManager fragmentManager) {
super(fragmentManager);
}
@Override
public int getCount() {
return 2;
}
@Override
public Fragment getItem(int position) {
if (position == 0)
return new FragmentOne();
else
return new FragmentTwo();
}
}
}
レイアウトfrag_master
:
<android.support.v4.view.ViewPager xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/view_pager"
android:layout_width="match_parent"
android:layout_height="match_parent" />
あなたがそうであるように、私はまったく同じ問題を抱えている、あなたのソリューションは動作しますが、パフォーマンスで:
–なぜそれをハンドラに投稿しますか? – Xenione
@ Xenioneこれは単なる回避策ですが、安全に使用できます。ここに私の考えがあります。最初のフラグメントでinitLoader()を呼び出すと、他のフラグメントでonResume()を呼び出すことができなくなります(ほとんどの場合、ライブラリのバグが原因です)。そのような呼び出しがない場合、すべてのonResume()メソッドが単一の実行フレームで呼び出されます。論理的には、フレームワークがすべてのフラグメントでonResume()メソッドを呼び出すようにするには、initLoader()を呼び出さないでください。論理的には、onResume()の直後にローダーを初期化するために、次の実行フレームでそれを行う必要があります。これは、handler.post()が多く役立つところです。 –