/* * Copyright (C) 2015 The Android Open Source Project * * 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 * * http://www.apache.org/licenses/LICENSE-2.0 * * 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 android.support.v4.app; import android.app.Activity; import android.content.Context; import android.content.Intent; import android.os.Handler; import android.support.annotation.NonNull; import android.support.annotation.Nullable; import android.support.v4.util.SimpleArrayMap; import android.view.LayoutInflater; import android.view.View; import java.io.FileDescriptor; import java.io.PrintWriter; /** * Integration points with the Fragment host. *

* Fragments may be hosted by any object; such as an {@link Activity}. In order to * host fragments, implement {@link FragmentHostCallback}, overriding the methods * applicable to the host. */ public abstract class FragmentHostCallback extends FragmentContainer { private final Activity mActivity; final Context mContext; private final Handler mHandler; final int mWindowAnimations; final FragmentManagerImpl mFragmentManager = new FragmentManagerImpl(); private SimpleArrayMap mAllLoaderManagers; private LoaderManagerImpl mLoaderManager; private boolean mCheckedForLoaderManager; private boolean mLoadersStarted; public FragmentHostCallback(Context context, Handler handler, int windowAnimations) { this(null /*activity*/, context, handler, windowAnimations); } FragmentHostCallback(FragmentActivity activity) { this(activity, activity /*context*/, activity.mHandler, 0 /*windowAnimations*/); } FragmentHostCallback(Activity activity, Context context, Handler handler, int windowAnimations) { mActivity = activity; mContext = context; mHandler = handler; mWindowAnimations = windowAnimations; } /** * Print internal state into the given stream. * * @param prefix Desired prefix to prepend at each line of output. * @param fd The raw file descriptor that the dump is being sent to. * @param writer The PrintWriter to which you should dump your state. This will be closed * for you after you return. * @param args additional arguments to the dump request. */ public void onDump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) { } /** * Return {@code true} if the fragment's state needs to be saved. */ public boolean onShouldSaveFragmentState(Fragment fragment) { return true; } /** * Return a {@link LayoutInflater}. * See {@link Activity#getLayoutInflater()}. */ public LayoutInflater onGetLayoutInflater() { return (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE); } /** * Return the object that's currently hosting the fragment. If a {@link Fragment} * is hosted by a {@link FragmentActivity}, the object returned here should be * the same object returned from {@link Fragment#getActivity()}. */ @Nullable public abstract E onGetHost(); /** * Invalidates the activity's options menu. * See {@link FragmentActivity#supportInvalidateOptionsMenu()} */ public void onSupportInvalidateOptionsMenu() { } /** * Starts a new {@link Activity} from the given fragment. * See {@link FragmentActivity#startActivityForResult(Intent, int)}. */ public void onStartActivityFromFragment(Fragment fragment, Intent intent, int requestCode) { if (requestCode != -1) { throw new IllegalStateException( "Starting activity with a requestCode requires a FragmentActivity host"); } mContext.startActivity(intent); } /** * Requests permissions from the given fragment. * See {@link FragmentActivity#requestPermissions(String[], int)} */ public void onRequestPermissionsFromFragment(@NonNull Fragment fragment, @NonNull String[] permissions, int requestCode) { } /** * Checks wehter to show permission rationale UI from a fragment. * See {@link FragmentActivity#shouldShowRequestPermissionRationale(String)} */ public boolean onShouldShowRequestPermissionRationale(@NonNull String permission) { return false; } /** * Return {@code true} if there are window animations. */ public boolean onHasWindowAnimations() { return true; } /** * Return the window animations. */ public int onGetWindowAnimations() { return mWindowAnimations; } @Nullable @Override public View onFindViewById(int id) { return null; } @Override public boolean onHasView() { return true; } Activity getActivity() { return mActivity; } Context getContext() { return mContext; } Handler getHandler() { return mHandler; } FragmentManagerImpl getFragmentManagerImpl() { return mFragmentManager; } LoaderManagerImpl getLoaderManagerImpl() { if (mLoaderManager != null) { return mLoaderManager; } mCheckedForLoaderManager = true; mLoaderManager = getLoaderManager("(root)", mLoadersStarted, true /*create*/); return mLoaderManager; } void inactivateFragment(String who) { //Log.v(TAG, "invalidateSupportFragment: who=" + who); if (mAllLoaderManagers != null) { LoaderManagerImpl lm = (LoaderManagerImpl) mAllLoaderManagers.get(who); if (lm != null && !lm.mRetaining) { lm.doDestroy(); mAllLoaderManagers.remove(who); } } } void onAttachFragment(Fragment fragment) { } void doLoaderStart() { if (mLoadersStarted) { return; } mLoadersStarted = true; if (mLoaderManager != null) { mLoaderManager.doStart(); } else if (!mCheckedForLoaderManager) { mLoaderManager = getLoaderManager("(root)", mLoadersStarted, false); // the returned loader manager may be a new one, so we have to start it if ((mLoaderManager != null) && (!mLoaderManager.mStarted)) { mLoaderManager.doStart(); } } mCheckedForLoaderManager = true; } // retain -- whether to stop the loader or retain it void doLoaderStop(boolean retain) { if (mLoaderManager == null) { return; } if (!mLoadersStarted) { return; } mLoadersStarted = false; if (retain) { mLoaderManager.doRetain(); } else { mLoaderManager.doStop(); } } void doLoaderRetain() { if (mLoaderManager == null) { return; } mLoaderManager.doRetain(); } void doLoaderDestroy() { if (mLoaderManager == null) { return; } mLoaderManager.doDestroy(); } void reportLoaderStart() { if (mAllLoaderManagers != null) { final int N = mAllLoaderManagers.size(); LoaderManagerImpl loaders[] = new LoaderManagerImpl[N]; for (int i=N-1; i>=0; i--) { loaders[i] = (LoaderManagerImpl) mAllLoaderManagers.valueAt(i); } for (int i=0; i(); } LoaderManagerImpl lm = (LoaderManagerImpl) mAllLoaderManagers.get(who); if (lm == null) { if (create) { lm = new LoaderManagerImpl(who, this, started); mAllLoaderManagers.put(who, lm); } } else { lm.updateHostController(this); } return lm; } SimpleArrayMap retainLoaderNonConfig() { boolean retainLoaders = false; if (mAllLoaderManagers != null) { // prune out any loader managers that were already stopped and so // have nothing useful to retain. final int N = mAllLoaderManagers.size(); LoaderManagerImpl loaders[] = new LoaderManagerImpl[N]; for (int i=N-1; i>=0; i--) { loaders[i] = (LoaderManagerImpl) mAllLoaderManagers.valueAt(i); } for (int i=0; i loaderManagers) { mAllLoaderManagers = loaderManagers; } void dumpLoaders(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) { writer.print(prefix); writer.print("mLoadersStarted="); writer.println(mLoadersStarted); if (mLoaderManager != null) { writer.print(prefix); writer.print("Loader Manager "); writer.print(Integer.toHexString(System.identityHashCode(mLoaderManager))); writer.println(":"); mLoaderManager.dump(prefix + " ", fd, writer, args); } } }