Android Open Source - Android-SplitViewController Split View Controller






From Project

Back to project page Android-SplitViewController.

License

The source code is released under:

MIT License

If you think the Android project Android-SplitViewController listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.

Java Source Code

/*
 * Copyright (c) 2014 Bradley David Bergeron
 *//from   www  .  ja  v  a 2s. c o  m
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */

package com.bradbergeron.splitviewcontroller;

import android.app.Fragment;
import android.app.FragmentManager;
import android.app.FragmentTransaction;
import android.os.Bundle;

import java.util.ArrayList;

public abstract class SplitViewController extends Fragment implements SplitViewNavigationListener {
    private final FragmentManager.OnBackStackChangedListener mBackStackListener =
            new FragmentManager.OnBackStackChangedListener() {
                @Override
                public void onBackStackChanged () {
                    configureChildFragments();
                }
            };

    private final ArrayList<OnDetailViewChangedListener> mDetailViewChangedListeners =
            new ArrayList<OnDetailViewChangedListener>();

    private SplitViewMasterFragment mMasterFragment;
    private SplitViewDetailFragment mDetailFragment;

    private boolean mNotifyDetailViewListeners = true;


    // ================================================================================
    // Fragment Lifecycle
    // ================================================================================

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

        setRetainInstance(true);
    }

    @Override
    public void onStart () {
        super.onStart();

        configureChildFragments();

        getFragmentManager().addOnBackStackChangedListener(mBackStackListener);
    }

    @Override
    public void onStop () {
        final FragmentManager fragmentManager = getFragmentManager();
        fragmentManager.removeOnBackStackChangedListener(mBackStackListener);

        if (mMasterFragment != null || mDetailFragment != null) {
            final FragmentTransaction transaction = fragmentManager.beginTransaction();

            if (mMasterFragment != null && !mMasterFragment.isDetached()) {
                transaction.detach(mMasterFragment);
            }

            if (mDetailFragment != null && !mDetailFragment.isDetached()) {
                transaction.detach(mDetailFragment);
            }

            transaction.commitAllowingStateLoss();
        }

        super.onStop();
    }


    // ================================================================================
    // Child Fragment Management
    // ================================================================================

    private void configureChildFragments () {
        if (mMasterFragment == null) {
            throw new IllegalStateException("Master view Fragment could not be found.");
        }

        mMasterFragment.setController(this);

        final FragmentManager fragmentManager = getFragmentManager();

        final SplitViewDetailFragment detailFragment = (SplitViewDetailFragment) fragmentManager
                .findFragmentById(getDetailFragmentContainerId());

        mDetailFragment =
                (detailFragment != null && detailFragment.isRemoving()) ? null : detailFragment;

        if (mDetailFragment != null) {
            mDetailFragment.setController(this);
        }

        final FragmentTransaction transaction = fragmentManager.beginTransaction();

        if (isSplitViewLayout()) {
            transaction.attach(mMasterFragment);

            if (mDetailFragment != null) {
                transaction.attach(mDetailFragment);
            }
        } else if (mDetailFragment != null) {
            transaction.attach(mDetailFragment);
        } else {
            transaction.attach(mMasterFragment);
        }

        transaction.commit();

        notifyDetailViewChangedListeners();
    }


    // ================================================================================
    // Master Fragment
    // ================================================================================

    public abstract int getMasterFragmentContainerId ();

    public void setMasterFragment (final SplitViewMasterFragment masterFragment) {
        masterFragment.setController(this);

        final FragmentTransaction transaction = getFragmentManager().beginTransaction();
        transaction.replace(getMasterFragmentContainerId(), masterFragment,
                            masterFragment.getClass().getSimpleName());
        transaction.commit();

        mMasterFragment = masterFragment;
    }


    // ================================================================================
    // Detail Fragment
    // ================================================================================

    public abstract int getDetailFragmentContainerId ();

    public void setDetailFragment (final SplitViewDetailFragment detailFragment) {
        setDetailFragment(detailFragment, getFragmentManager().beginTransaction());
    }

    public void setDetailFragment (final SplitViewDetailFragment detailFragment,
                                   final FragmentTransaction transaction) {
        final FragmentManager fragmentManager = getFragmentManager();

        if (detailFragment == null) {
            fragmentManager.popBackStackImmediate(null, FragmentManager.POP_BACK_STACK_INCLUSIVE);

            return;
        }

        detailFragment.setController(this);

        if (fragmentManager.getBackStackEntryCount() > 0) {
            mNotifyDetailViewListeners = false;
            fragmentManager.popBackStackImmediate(null, FragmentManager.POP_BACK_STACK_INCLUSIVE);
            mNotifyDetailViewListeners = true;
        }

        transaction.replace(getDetailFragmentContainerId(), detailFragment);
        transaction.addToBackStack(null);

        if (!isSplitViewLayout() && !mMasterFragment.isDetached()) {
            transaction.detach(mMasterFragment);
        }

        transaction.commit();

        mDetailFragment = detailFragment;
    }

    public void setDetailViewTitle (final CharSequence title) { }

    public void setDetailViewSubtitle (final CharSequence subtitle) { }

    private void notifyDetailViewChangedListeners () {
        if (mNotifyDetailViewListeners) {
            for (final OnDetailViewChangedListener listener : mDetailViewChangedListeners) {
                listener.onDetailViewChanged(mDetailFragment);
            }
        }
    }


    // ================================================================================
    // Properties
    // ================================================================================

    public abstract boolean isSplitViewLayout ();


    // ================================================================================
    // OnDetailViewChangedListener
    // ================================================================================

    public void addOnDetailViewChangedListener (final OnDetailViewChangedListener listener) {
        if (!mDetailViewChangedListeners.contains(listener)) {
            mDetailViewChangedListeners.add(listener);
        }
    }

    public void removeOnDetailViewChangedListener (final OnDetailViewChangedListener listener) {
        mDetailViewChangedListeners.remove(listener);
    }

    public interface OnDetailViewChangedListener {
        public void onDetailViewChanged (final SplitViewDetailFragment detailViewFragment);
    }
}




Java Source Code List

com.bradbergeron.splitviewcontroller.SplitViewAbsFragment.java
com.bradbergeron.splitviewcontroller.SplitViewController.java
com.bradbergeron.splitviewcontroller.SplitViewDetailFragment.java
com.bradbergeron.splitviewcontroller.SplitViewMasterFragment.java
com.bradbergeron.splitviewcontroller.SplitViewNavigationListener.java
com.bradbergeron.splitviewcontrollerdemo.activities.AboutActivity.java
com.bradbergeron.splitviewcontrollerdemo.activities.MainActivity.java
com.bradbergeron.splitviewcontrollerdemo.fragments.DetailFragment.java
com.bradbergeron.splitviewcontrollerdemo.fragments.DrawerFragment.java
com.bradbergeron.splitviewcontrollerdemo.fragments.ListFragment.java
com.bradbergeron.splitviewcontrollerdemo.fragments.MoreDetailsFragment.java
com.bradbergeron.splitviewcontrollerdemo.fragments.SplitViewFragment.java
com.bradbergeron.splitviewcontrollerdemo.fragments.WebViewFragment.java