Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Fragment - should I reuse view in onCreateView and how should I do that?

Actually, I always reused my view in my fragments like the following:

private View mView = null;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
{
    if (mView == null)
        mView = inflater.inflate(R.layout.view);
    return mView;
}

That worked, with viewpager and so on. Now I started using my fragments in simple activities as well and if, and only if, I add the fragment to the backstack, this will fail because of java.lang.IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first.

So my questions are:

  • Is it ok, if I check the the views parent, remove it and add it to the new parent?
  • Or should I always recreate the view and never reuse it? If yes, why?
  • Are there other points, where reusing the view will fail?
like image 260
prom85 Avatar asked Sep 05 '13 13:09

prom85


People also ask

How can fragments be reused?

replace(): Replace an existing fragment that was added to a container. So this removes the Fragment from the stack. Use FragmentTransaction. addToBackStack() to add the Fragment on top of another(s) (the stack).

Is it possible to reuse a fragment in multiple screens?

Yes you can, but you have to add more logic to your fragments and add some interfaces for each activity.

Which function should you use to inflate the view of a fragment?

onCreateView() is called by Android once the Fragment should inflate a view. onViewCreated() is called after onCreateView() and ensures that the fragment's root view is non-null .

What is onCreateView fragment?

onCreateView(LayoutInflater, ViewGroup, Bundle) creates and returns the view hierarchy associated with the fragment. onActivityCreated(Bundle) tells the fragment that its activity has completed its own Activity.


1 Answers

Maybe this can help to understand the behavior. If you check out FragmentManagerImpl.java you will find the following:

First we create a view by calling onCreateView() (line 845) and then we wrap created view with another view, which becomes a parent of our view (lines 848-849). This means our view does not become a child of real container, but it's a child of a wrapper view now. The problem with reuse happens, when view gets removed from the container (line 998). FragmentManager removes wrapper view from the container, but our real view stays added to the parent wrapper view. This is what causes the issue you experience.

Thus, if you remove the view from its parent, it can work. Even knowing this, I would not recommend reusing views in the fragment also because views can live a bit longer than fragments, because they can be used in "disappearing" animations even after the fragment is been destroyed. If you try to remove such a view from its parent at that time, then the animation might be broken.

Another argument to not cache the view is that Android doesn't support view recycling in fragments by design. Remember ListAdapter allowing to reuse the views? Android takes care for caching and proper reusing those views. This is not the case with fragment though.

like image 113
sergej shafarenka Avatar answered Sep 22 '22 16:09

sergej shafarenka