androider
androider

Reputation: 465

Replace ListFragment with another Fragment

I'm learning Fragments and trying to implemen my own task, also following these tutorials: developer.android

Here I'm dynamically adding ListFragment to container. Then after clicking on any item in ListFragment a new EditFragment is supposed to replace the current fragment in container. But I'm getting an IllegalStateException after tapping on a list item. Is this because I'm using the same TransactionManager for both fragments?

Here's my code:

MyListFragment:

public class MyListFragment extends ListFragment {

    Communicator communicator;
    private MenuItem addItem;
    private MenuItem clearAll;
    private String name;

    ArrayList list1 = new ArrayList();
    String[] list2 = {"Sunday", "Monday",
            "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"};
    ArrayAdapter<String> adapter;
    int randomNum;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        communicator = (Communicator) getActivity();
        adapter = new ArrayAdapter<String>(getActivity(), android.R.layout.simple_list_item_1, list1);
        setListAdapter(adapter);
        setHasOptionsMenu(true);
        return super.onCreateView(inflater, container, savedInstanceState);
    }

    @Override
    public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
        inflater.inflate(R.menu.list_fragment, menu);

        addItem = menu.findItem(R.id.add_string);
        clearAll = menu.findItem(R.id.clear);

        addItem.setOnMenuItemClickListener(new MenuItem.OnMenuItemClickListener() {
            @Override
            public boolean onMenuItemClick(MenuItem item) {
                Random rand = new Random();
                randomNum = rand.nextInt(list2.length);
                list1.add(list2[randomNum]);
                adapter.notifyDataSetChanged();
                return true;
            }
        });

        clearAll.setOnMenuItemClickListener(new MenuItem.OnMenuItemClickListener() {
            @Override
            public boolean onMenuItemClick(MenuItem item) {
                list1.clear();
                adapter.notifyDataSetChanged();
                return true;
            }
        });
    }

    @Override
    public void onListItemClick(ListView l, View v, int position, long id) {
        communicator.onMyListItemClick(position);
    }

    public interface Communicator{
        public void onMyListItemClick(int position);
    }
}

MainActivity:

public class MainActivity extends AppCompatActivity implements MyListFragment.Communicator{

    FragmentManager fragmentManager;
    FragmentTransaction fragmentTransaction;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        fragmentManager = getFragmentManager();
        fragmentTransaction = fragmentManager.beginTransaction();
        fragmentTransaction.add(R.id.fragContainer, new MyListFragment());
        fragmentTransaction.commit();
    }

    @Override
    public void onMyListItemClick(int position) {
        EditFragment editFragment = new EditFragment();

        Bundle args = new Bundle();
        args.putInt("position", position);
        editFragment.setArguments(args);
        fragmentTransaction.replace(R.id.fragContainer, editFragment);

        fragmentTransaction.addToBackStack(null);
        fragmentTransaction.commit();
    }
}

Upvotes: 1

Views: 641

Answers (1)

robinj
robinj

Reputation: 931

The problem isn't that you're using the same FragmentManager, but rather that you are trying to reuse a FragmentTransaction after calling commit().

You need to create a fresh transaction in onMyListItemClick(), like this:

public void onMyListItemClick(int position) {
    MainActivityFragment editFragment = new MainActivityFragment();

    Bundle args = new Bundle();
    args.putInt("position", position);
    editFragment.setArguments(args);

    FragmentTransaction fragmentTransaction = getFragmentManager().beginTransaction();
    fragmentTransaction.replace(R.id.fragContainer, editFragment);

    fragmentTransaction.addToBackStack(null);
    fragmentTransaction.commit();
}

Upvotes: 2

Related Questions