mobi.cangol.mobile.base.CustomFragmentManager.java Source code

Java tutorial

Introduction

Here is the source code for mobi.cangol.mobile.base.CustomFragmentManager.java

Source

/*
 * Copyright (c) 2013 Cangol
 * <p>
 * Licensed under the Apache License, Version 2.0 (the "License")
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * <p>
 * http://www.apache.org/licenses/LICENSE-2.0
 * <p>
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package mobi.cangol.mobile.base;

import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentTransaction;

import java.lang.ref.WeakReference;
import java.lang.reflect.Field;

import mobi.cangol.mobile.logging.Log;

public class CustomFragmentManager {
    private static final String STATE_TAG = "CustomFragmentManager";
    private FragmentStack stack = null;
    private Object lock = new Object();
    private FragmentManager fragmentManager;
    private FragmentTransaction fragmentTransaction;
    private int containerId;
    private FragmentActivity fActivity;
    private final Runnable execPendingTransactions = new Runnable() {
        @Override
        public void run() {
            if (fragmentTransaction != null && fActivity != null) {
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
                    if (!fActivity.isFinishing() && !fActivity.isDestroyed()) {
                        try {
                            fragmentTransaction.commitAllowingStateLoss();
                            fragmentManager.executePendingTransactions();
                            fragmentTransaction = null;
                        } catch (IllegalStateException e) {
                            Log.e(STATE_TAG, "execPendingTransactions", e);
                        }
                    }
                } else {
                    if (!fActivity.isFinishing()) {
                        try {
                            fragmentTransaction.commitAllowingStateLoss();
                            fragmentManager.executePendingTransactions();
                            fragmentTransaction = null;
                        } catch (IllegalStateException e) {
                            Log.e(STATE_TAG, "execPendingTransactions", e);
                        }
                    }
                }
            }
        }
    };
    private InternalHandler handler;
    private int enterAnimation;
    private int exitAnimation;
    private int popStackEnterAnimation;
    private int popStackExitAnimation;
    private boolean firstUseAnim = false;

    private CustomFragmentManager(FragmentActivity fActivity, int containerId, FragmentManager fragmentManager) {
        this.fActivity = fActivity;
        this.fragmentManager = fragmentManager;
        this.containerId = containerId;
        this.handler = new InternalHandler(fActivity);
        this.stack = new FragmentStack();
    }

    public static CustomFragmentManager forContainer(FragmentActivity activity, int containerId,
            FragmentManager fragmentManager) {
        return new CustomFragmentManager(activity, containerId, fragmentManager);
    }

    public void destroy() {
        this.stack.clear();
        this.handler.removeCallbacks(execPendingTransactions);
    }

    protected final static class InternalHandler extends Handler {
        private final WeakReference<FragmentActivity> mActivityRef;

        public InternalHandler(FragmentActivity activity) {
            super(Looper.getMainLooper());
            mActivityRef = new WeakReference<>(activity);
        }
    }

    public int getContainerId() {
        return containerId;
    }

    public void setDefaultAnimation(int enter, int exit, int popEnter, int popExit) {
        enterAnimation = enter;
        exitAnimation = exit;
        popStackEnterAnimation = popEnter;
        popStackExitAnimation = popExit;
    }

    public void saveState(Bundle outState) {
        executePendingTransactions();
        final int stackSize = stack.size();
        String[] stackTags = new String[stackSize];

        int i = 0;
        for (String tag : stack.getTag()) {
            Log.i(STATE_TAG, "tag =" + tag);
            stackTags[i++] = tag;
        }

        outState.putStringArray(STATE_TAG, stackTags);
    }

    public void restoreState(Bundle state) {
        String[] stackTags = state.getStringArray(STATE_TAG);
        for (String tag : stackTags) {
            BaseFragment f = (BaseFragment) fragmentManager.findFragmentByTag(tag);
            stack.addFragment(f);
            stack.addTag(tag);
        }
    }

    private FragmentTransaction beginTransaction() {
        if (fragmentTransaction == null)
            fragmentTransaction = fragmentManager.beginTransaction();
        handler.removeCallbacks(execPendingTransactions);
        return fragmentTransaction;
    }

    public void replace(Class<? extends BaseFragment> clazz, String tag, Bundle args) {
        if (clazz.isAssignableFrom(BaseDialogFragment.class)) {
            throw new IllegalStateException("DialogFragment can not be attached to a container view");
        } else {
            this.replace(clazz, tag, args, null);
        }
    }

    public void replace(Class<? extends BaseFragment> clazz, String tag, Bundle args,
            CustomFragmentTransaction customFragmentTransaction) {
        if (fragmentManager.isDestroyed() || isStateSaved())
            return;
        if (clazz.isAssignableFrom(BaseDialogFragment.class))
            throw new IllegalStateException("DialogFragment can not be attached to a container view");
        BaseFragment fragment = (BaseFragment) fragmentManager.findFragmentByTag(tag);
        if (fragment == null) {
            Log.i(STATE_TAG, "fragment=null newInstance");
            fragment = (BaseFragment) Fragment.instantiate(fActivity, clazz.getName(), args);
            if (fragment.isCleanStack()) {
                Log.i(STATE_TAG, "fragment isCleanStack=true,while pop all");
                while (stack.size() > 0) {
                    synchronized (lock) {
                        stack.popFragment();
                        stack.popTag();
                    }
                    fragmentManager.popBackStack();
                }
            } else {
                Log.i(STATE_TAG, "fragment isCleanStack=false");
                if (fragment.isSingleton() && stack.containsTag(tag)) {
                    Log.i(STATE_TAG, "fragment isSingleton=true,while pop all");
                    while (!tag.equals(stack.peekTag())) {
                        synchronized (lock) {
                            stack.popFragment();
                            stack.popTag();
                        }
                        fragmentManager.popBackStack();
                    }
                    synchronized (lock) {
                        stack.popFragment();
                        stack.popTag();
                    }
                    fragmentManager.popBackStack();
                    Log.i(STATE_TAG, "fragment newInstance");
                    fragment = (BaseFragment) Fragment.instantiate(fActivity, clazz.getName(), args);
                }
            }
        } else {
            Log.i(STATE_TAG, "fragment is exist");
            if (fragment.isCleanStack()) {
                if (stack.size() == 1) {
                    if (stack.peekTag().equals(tag)) {
                        return;
                    } else {
                        synchronized (lock) {
                            stack.popFragment();
                            stack.popTag();
                        }
                        fragmentManager.popBackStack();
                        fragment = (BaseFragment) Fragment.instantiate(fActivity, clazz.getName(), args);
                    }
                } else {
                    Log.i(STATE_TAG, "fragment isCleanStack=true,while pop all");
                    while (stack.size() > 0) {
                        synchronized (lock) {
                            stack.popFragment();
                            stack.popTag();
                        }
                        fragmentManager.popBackStack();
                    }
                    fragment = (BaseFragment) Fragment.instantiate(fActivity, clazz.getName(), args);
                }
            } else {
                Log.i(STATE_TAG, "fragment isCleanStack=false");
                if (!fragment.isSingleton()) {
                    Log.i(STATE_TAG, "fragment isSingleton=false,newInstance");
                    fragment = (BaseFragment) Fragment.instantiate(fActivity, clazz.getName(), args);
                } else {
                    Log.i(STATE_TAG, "fragment isSingleton=true,while pop all");
                    while (!tag.equals(stack.peekTag())) {
                        synchronized (lock) {
                            stack.popFragment();
                            stack.popTag();
                        }
                        fragmentManager.popBackStack();
                    }
                    synchronized (lock) {
                        stack.popFragment();
                        stack.popTag();
                    }
                    fragmentManager.popBackStack();
                    fragment = (BaseFragment) Fragment.instantiate(fActivity, clazz.getName(), args);
                }
            }
        }
        if (customFragmentTransaction != null)
            customFragmentTransaction.fillTargetFragment(fragment);

        if (firstUseAnim || stack.size() > 0) {
            //???
            if (customFragmentTransaction == null
                    || !customFragmentTransaction.fillCustomAnimations(beginTransaction())) {
                if (enterAnimation > 0 && exitAnimation > 0 && popStackEnterAnimation > 0
                        && popStackExitAnimation > 0) {
                    beginTransaction().setCustomAnimations(enterAnimation, exitAnimation, popStackEnterAnimation,
                            popStackExitAnimation);
                } else if (enterAnimation > 0 && exitAnimation > 0) {
                    beginTransaction().setCustomAnimations(enterAnimation, exitAnimation);
                } else {
                    beginTransaction();
                }
            }
        } else {
            //
        }

        attachFragment(fragment, tag);
        synchronized (lock) {
            stack.addFragment(fragment);
            stack.addTag(tag);
        }
    }

    public void setFirstUseAnim(boolean firstUseAnim) {
        this.firstUseAnim = firstUseAnim;
    }

    private void attachFragment(Fragment fragment, String tag) {
        if (fragment != null) {
            if (fragment.isDetached()) {
                Log.i(STATE_TAG, "attachFragment tag=" + tag);
                beginTransaction().attach(fragment);
                if (stack.size() > 0) {
                    beginTransaction().addToBackStack(tag);
                }
            } else if (!fragment.isAdded()) {
                Log.i(STATE_TAG, "replaceFragment tag=" + tag);
                beginTransaction().replace(containerId, fragment, tag);
                if (stack.size() > 0) {
                    beginTransaction().addToBackStack(tag);
                }
            } else {
                Log.i(STATE_TAG, "fragment state illegal " + fragment);
            }
        } else {
            Log.i(STATE_TAG, "fragment is null");
        }
    }

    public BaseFragment peek() {
        return stack.peekFragment();
    }

    public int size() {
        return stack.size();
    }

    public boolean popBackStack() {
        if (fragmentManager.isDestroyed() || isStateSaved())
            return false;
        if (stack.size() > 1) {
            fragmentManager.popBackStackImmediate();
            synchronized (lock) {
                BaseFragment baseFragment = stack.popFragment();
                stack.popTag();
                if (baseFragment != null && baseFragment.getTargetFragment() != null) {
                    baseFragment.notifyResult();
                }
            }
            return true;
        }
        return false;
    }

    public boolean popBackStackAll() {
        if (fragmentManager.isDestroyed() || isStateSaved())
            return false;
        if (stack.size() > 1) {
            while (stack.size() > 1) {
                synchronized (lock) {
                    stack.popFragment();
                    stack.popTag();
                }
                fragmentManager.popBackStack();
            }
            return true;
        }
        return false;
    }

    public void commit() {
        if (fragmentManager.isDestroyed() || isStateSaved())
            return;
        if (fragmentTransaction != null && !fragmentTransaction.isEmpty()) {
            handler.removeCallbacks(execPendingTransactions);
            handler.post(execPendingTransactions);
        } else {
            Log.i(STATE_TAG, "fragmentTransaction is null or empty");
        }
    }

    public boolean executePendingTransactions() {
        if (fragmentTransaction != null && !fragmentTransaction.isEmpty()) {
            handler.removeCallbacks(execPendingTransactions);
            fragmentTransaction.commitAllowingStateLoss();
            fragmentTransaction = null;
            return fragmentManager.executePendingTransactions();
        }
        return false;
    }

    /**
     * ?onSaveInstanceState
     * Support 26.0.0-alpha1 ??isStateSaved
     * ???mStateSaved
     *
     * @return
     */
    public boolean isStateSaved() {
        Class implClass = null;
        Field field = null;
        try {
            implClass = Class.forName("android.support.v4.app.FragmentManagerImpl");
            field = implClass.getDeclaredField("mStateSaved");
            field.setAccessible(true);
            return (Boolean) field.get(implClass.cast(fragmentManager));
        } catch (Exception e) {
            Log.e("isStateSaved", "" + e.getMessage(), e);
        }
        return false;
    }
}