• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.support.v4.app;
18 
19 import android.app.Activity;
20 import android.content.Context;
21 import android.content.Intent;
22 import android.content.IntentSender;
23 import android.content.res.Configuration;
24 import android.content.res.Resources;
25 import android.os.Bundle;
26 import android.os.Handler;
27 import android.os.Message;
28 import android.os.Parcelable;
29 import android.support.annotation.CallSuper;
30 import android.support.annotation.NonNull;
31 import android.support.annotation.Nullable;
32 import android.support.v4.media.session.MediaControllerCompat;
33 import android.support.v4.util.SimpleArrayMap;
34 import android.support.v4.util.SparseArrayCompat;
35 import android.util.AttributeSet;
36 import android.util.Log;
37 import android.view.KeyEvent;
38 import android.view.LayoutInflater;
39 import android.view.Menu;
40 import android.view.MenuItem;
41 import android.view.View;
42 import android.view.ViewGroup;
43 import android.view.Window;
44 
45 import java.io.FileDescriptor;
46 import java.io.PrintWriter;
47 
48 /**
49  * Base class for activities that want to use the support-based
50  * {@link android.support.v4.app.Fragment} and
51  * {@link android.support.v4.content.Loader} APIs.
52  *
53  * <p>When using this class as opposed to new platform's built-in fragment
54  * and loader support, you must use the {@link #getSupportFragmentManager()}
55  * and {@link #getSupportLoaderManager()} methods respectively to access
56  * those features.
57  *
58  * <p>Known limitations:</p>
59  * <ul>
60  * <li> <p>When using the <code>&lt;fragment></code> tag, this implementation can not
61  * use the parent view's ID as the new fragment's ID.  You must explicitly
62  * specify an ID (or tag) in the <code>&lt;fragment></code>.</p>
63  * <li> <p>Prior to Honeycomb (3.0), an activity's state was saved before pausing.
64  * Fragments are a significant amount of new state, and dynamic enough that one
65  * often wants them to change between pausing and stopping.  These classes
66  * throw an exception if you try to change the fragment state after it has been
67  * saved, to avoid accidental loss of UI state.  However this is too restrictive
68  * prior to Honeycomb, where the state is saved before pausing.  To address this,
69  * when running on platforms prior to Honeycomb an exception will not be thrown
70  * if you change fragments between the state save and the activity being stopped.
71  * This means that in some cases if the activity is restored from its last saved
72  * state, this may be a snapshot slightly before what the user last saw.</p>
73  * </ul>
74  */
75 public class FragmentActivity extends BaseFragmentActivityJB implements
76         ActivityCompat.OnRequestPermissionsResultCallback,
77         ActivityCompatApi23.RequestPermissionsRequestCodeValidator {
78     private static final String TAG = "FragmentActivity";
79 
80     static final String FRAGMENTS_TAG = "android:support:fragments";
81     static final String NEXT_CANDIDATE_REQUEST_INDEX_TAG = "android:support:next_request_index";
82     static final String ALLOCATED_REQUEST_INDICIES_TAG = "android:support:request_indicies";
83     static final String REQUEST_FRAGMENT_WHO_TAG = "android:support:request_fragment_who";
84     static final int MAX_NUM_PENDING_FRAGMENT_ACTIVITY_RESULTS = 0xffff - 1;
85 
86     // This is the SDK API version of Honeycomb (3.0).
87     private static final int HONEYCOMB = 11;
88 
89     static final int MSG_REALLY_STOPPED = 1;
90     static final int MSG_RESUME_PENDING = 2;
91 
92     final Handler mHandler = new Handler() {
93         @Override
94         public void handleMessage(Message msg) {
95             switch (msg.what) {
96                 case MSG_REALLY_STOPPED:
97                     if (mStopped) {
98                         doReallyStop(false);
99                     }
100                     break;
101                 case MSG_RESUME_PENDING:
102                     onResumeFragments();
103                     mFragments.execPendingActions();
104                     break;
105                 default:
106                     super.handleMessage(msg);
107             }
108         }
109 
110     };
111     final FragmentController mFragments = FragmentController.createController(new HostCallbacks());
112 
113     boolean mCreated;
114     boolean mResumed;
115     boolean mStopped;
116     boolean mReallyStopped;
117     boolean mRetaining;
118 
119     boolean mOptionsMenuInvalidated;
120     boolean mRequestedPermissionsFromFragment;
121 
122     // A hint for the next candidate request index. Request indicies are ints between 0 and 2^16-1
123     // which are encoded into the upper 16 bits of the requestCode for
124     // Fragment.startActivityForResult(...) calls. This allows us to dispatch onActivityResult(...)
125     // to the appropriate Fragment. Request indicies are allocated by allocateRequestIndex(...).
126     int mNextCandidateRequestIndex;
127     // A map from request index to Fragment "who" (i.e. a Fragment's unique identifier). Used to
128     // keep track of the originating Fragment for Fragment.startActivityForResult(...) calls, so we
129     // can dispatch the onActivityResult(...) to the appropriate Fragment. Will only contain entries
130     // for startActivityForResult calls where a result has not yet been delivered.
131     SparseArrayCompat<String> mPendingFragmentActivityResults;
132 
133     static final class NonConfigurationInstances {
134         Object custom;
135         FragmentManagerNonConfig fragments;
136         SimpleArrayMap<String, LoaderManager> loaders;
137     }
138 
139     MediaControllerCompat mMediaController;
140 
141     // ------------------------------------------------------------------------
142     // HOOKS INTO ACTIVITY
143     // ------------------------------------------------------------------------
144 
145     /**
146      * Dispatch incoming result to the correct fragment.
147      */
148     @Override
onActivityResult(int requestCode, int resultCode, Intent data)149     protected void onActivityResult(int requestCode, int resultCode, Intent data) {
150         mFragments.noteStateNotSaved();
151         int requestIndex = requestCode>>16;
152         if (requestIndex != 0) {
153             requestIndex--;
154 
155             String who = mPendingFragmentActivityResults.get(requestIndex);
156             mPendingFragmentActivityResults.remove(requestIndex);
157             if (who == null) {
158                 Log.w(TAG, "Activity result delivered for unknown Fragment.");
159                 return;
160             }
161             Fragment targetFragment = mFragments.findFragmentByWho(who);
162             if (targetFragment == null) {
163                 Log.w(TAG, "Activity result no fragment exists for who: " + who);
164             } else {
165                 targetFragment.onActivityResult(requestCode & 0xffff, resultCode, data);
166             }
167             return;
168         }
169 
170         super.onActivityResult(requestCode, resultCode, data);
171     }
172 
173     /**
174      * Take care of popping the fragment back stack or finishing the activity
175      * as appropriate.
176      */
onBackPressed()177     public void onBackPressed() {
178         if (!mFragments.getSupportFragmentManager().popBackStackImmediate()) {
179             onBackPressedNotHandled();
180         }
181     }
182 
183     /**
184      * Sets a {@link MediaControllerCompat} for later retrieval via
185      * {@link #getSupportMediaController()}.
186      *
187      * <p>On API 21 and later, this controller will be tied to the window of the activity and
188      * media key and volume events which are received while the Activity is in the foreground
189      * will be forwarded to the controller and used to invoke transport controls or adjust the
190      * volume. Prior to API 21, the global handling of media key and volume events through an
191      * active {@link android.support.v4.media.session.MediaSessionCompat} and media button receiver
192      * will still be respected.</p>
193      *
194      * @param mediaController The controller for the session which should receive
195      *     media keys and volume changes on API 21 and later.
196      * @see #setMediaController(android.media.session.MediaController)
197      */
setSupportMediaController(MediaControllerCompat mediaController)198     final public void setSupportMediaController(MediaControllerCompat mediaController) {
199         mMediaController = mediaController;
200         if (android.os.Build.VERSION.SDK_INT >= 21) {
201             ActivityCompat21.setMediaController(this, mediaController.getMediaController());
202         }
203     }
204 
205     /**
206      * Retrieves the current {@link MediaControllerCompat} for sending media key and volume events.
207      *
208      * @return The controller which should receive events.
209      * @see #setSupportMediaController(android.support.v4.media.session.MediaController)
210      * @see #getMediaController()
211      */
getSupportMediaController()212     final public MediaControllerCompat getSupportMediaController() {
213         return mMediaController;
214     }
215 
216     /**
217      * Reverses the Activity Scene entry Transition and triggers the calling Activity
218      * to reverse its exit Transition. When the exit Transition completes,
219      * {@link #finish()} is called. If no entry Transition was used, finish() is called
220      * immediately and the Activity exit Transition is run.
221      *
222      * <p>On Android 4.4 or lower, this method only finishes the Activity with no
223      * special exit transition.</p>
224      */
supportFinishAfterTransition()225     public void supportFinishAfterTransition() {
226         ActivityCompat.finishAfterTransition(this);
227     }
228 
229     /**
230      * When {@link android.app.ActivityOptions#makeSceneTransitionAnimation(Activity,
231      * android.view.View, String)} was used to start an Activity, <var>callback</var>
232      * will be called to handle shared elements on the <i>launched</i> Activity. This requires
233      * {@link Window#FEATURE_CONTENT_TRANSITIONS}.
234      *
235      * @param callback Used to manipulate shared element transitions on the launched Activity.
236      */
setEnterSharedElementCallback(SharedElementCallback callback)237     public void setEnterSharedElementCallback(SharedElementCallback callback) {
238         ActivityCompat.setEnterSharedElementCallback(this, callback);
239     }
240 
241     /**
242      * When {@link android.app.ActivityOptions#makeSceneTransitionAnimation(Activity,
243      * android.view.View, String)} was used to start an Activity, <var>listener</var>
244      * will be called to handle shared elements on the <i>launching</i> Activity. Most
245      * calls will only come when returning from the started Activity.
246      * This requires {@link Window#FEATURE_CONTENT_TRANSITIONS}.
247      *
248      * @param listener Used to manipulate shared element transitions on the launching Activity.
249      */
setExitSharedElementCallback(SharedElementCallback listener)250     public void setExitSharedElementCallback(SharedElementCallback listener) {
251         ActivityCompat.setExitSharedElementCallback(this, listener);
252     }
253 
254     /**
255      * Support library version of {@link android.app.Activity#postponeEnterTransition()} that works
256      * only on API 21 and later.
257      */
supportPostponeEnterTransition()258     public void supportPostponeEnterTransition() {
259         ActivityCompat.postponeEnterTransition(this);
260     }
261 
262     /**
263      * Support library version of {@link android.app.Activity#startPostponedEnterTransition()}
264      * that only works with API 21 and later.
265      */
supportStartPostponedEnterTransition()266     public void supportStartPostponedEnterTransition() {
267         ActivityCompat.startPostponedEnterTransition(this);
268     }
269 
270     /**
271      * {@inheritDoc}
272      *
273      * <p><strong>Note:</strong> If you override this method you must call
274      * <code>super.onMultiWindowModeChanged</code> to correctly dispatch the event
275      * to support fragments attached to this activity.</p>
276      *
277      * @param isInMultiWindowMode True if the activity is in multi-window mode.
278      */
279     @CallSuper
onMultiWindowModeChanged(boolean isInMultiWindowMode)280     public void onMultiWindowModeChanged(boolean isInMultiWindowMode) {
281         mFragments.dispatchMultiWindowModeChanged(isInMultiWindowMode);
282     }
283 
284     /**
285      * {@inheritDoc}
286      *
287      * <p><strong>Note:</strong> If you override this method you must call
288      * <code>super.onPictureInPictureModeChanged</code> to correctly dispatch the event
289      * to support fragments attached to this activity.</p>
290      *
291      * @param isInPictureInPictureMode True if the activity is in picture-in-picture mode.
292      */
293     @CallSuper
onPictureInPictureModeChanged(boolean isInPictureInPictureMode)294     public void onPictureInPictureModeChanged(boolean isInPictureInPictureMode) {
295         mFragments.dispatchPictureInPictureModeChanged(isInPictureInPictureMode);
296     }
297 
298     /**
299      * Dispatch configuration change to all fragments.
300      */
301     @Override
onConfigurationChanged(Configuration newConfig)302     public void onConfigurationChanged(Configuration newConfig) {
303         super.onConfigurationChanged(newConfig);
304         mFragments.dispatchConfigurationChanged(newConfig);
305     }
306 
307     /**
308      * Perform initialization of all fragments and loaders.
309      */
310     @SuppressWarnings("deprecation")
311     @Override
onCreate(@ullable Bundle savedInstanceState)312     protected void onCreate(@Nullable Bundle savedInstanceState) {
313         mFragments.attachHost(null /*parent*/);
314 
315         super.onCreate(savedInstanceState);
316 
317         NonConfigurationInstances nc =
318                 (NonConfigurationInstances) getLastNonConfigurationInstance();
319         if (nc != null) {
320             mFragments.restoreLoaderNonConfig(nc.loaders);
321         }
322         if (savedInstanceState != null) {
323             Parcelable p = savedInstanceState.getParcelable(FRAGMENTS_TAG);
324             mFragments.restoreAllState(p, nc != null ? nc.fragments : null);
325 
326             // Check if there are any pending onActivityResult calls to descendent Fragments.
327             if (savedInstanceState.containsKey(NEXT_CANDIDATE_REQUEST_INDEX_TAG)) {
328                 mNextCandidateRequestIndex =
329                         savedInstanceState.getInt(NEXT_CANDIDATE_REQUEST_INDEX_TAG);
330                 int[] requestCodes = savedInstanceState.getIntArray(ALLOCATED_REQUEST_INDICIES_TAG);
331                 String[] fragmentWhos = savedInstanceState.getStringArray(REQUEST_FRAGMENT_WHO_TAG);
332                 if (requestCodes == null || fragmentWhos == null ||
333                             requestCodes.length != fragmentWhos.length) {
334                     Log.w(TAG, "Invalid requestCode mapping in savedInstanceState.");
335                 } else {
336                     mPendingFragmentActivityResults = new SparseArrayCompat<>(requestCodes.length);
337                     for (int i = 0; i < requestCodes.length; i++) {
338                         mPendingFragmentActivityResults.put(requestCodes[i], fragmentWhos[i]);
339                     }
340                 }
341             }
342         }
343 
344         if (mPendingFragmentActivityResults == null) {
345             mPendingFragmentActivityResults = new SparseArrayCompat<>();
346             mNextCandidateRequestIndex = 0;
347         }
348 
349         mFragments.dispatchCreate();
350     }
351 
352     /**
353      * Dispatch to Fragment.onCreateOptionsMenu().
354      */
355     @Override
onCreatePanelMenu(int featureId, Menu menu)356     public boolean onCreatePanelMenu(int featureId, Menu menu) {
357         if (featureId == Window.FEATURE_OPTIONS_PANEL) {
358             boolean show = super.onCreatePanelMenu(featureId, menu);
359             show |= mFragments.dispatchCreateOptionsMenu(menu, getMenuInflater());
360             if (android.os.Build.VERSION.SDK_INT >= HONEYCOMB) {
361                 return show;
362             }
363             // Prior to Honeycomb, the framework can't invalidate the options
364             // menu, so we must always say we have one in case the app later
365             // invalidates it and needs to have it shown.
366             return true;
367         }
368         return super.onCreatePanelMenu(featureId, menu);
369     }
370 
371     @Override
dispatchFragmentsOnCreateView(View parent, String name, Context context, AttributeSet attrs)372     final View dispatchFragmentsOnCreateView(View parent, String name, Context context,
373             AttributeSet attrs) {
374         return mFragments.onCreateView(parent, name, context, attrs);
375     }
376 
377     /**
378      * Destroy all fragments and loaders.
379      */
380     @Override
onDestroy()381     protected void onDestroy() {
382         super.onDestroy();
383 
384         doReallyStop(false);
385 
386         mFragments.dispatchDestroy();
387         mFragments.doLoaderDestroy();
388     }
389 
390     /**
391      * Take care of calling onBackPressed() for pre-Eclair platforms.
392      */
393     @Override
onKeyDown(int keyCode, KeyEvent event)394     public boolean onKeyDown(int keyCode, KeyEvent event) {
395         if (android.os.Build.VERSION.SDK_INT < 5 /* ECLAIR */
396                 && keyCode == KeyEvent.KEYCODE_BACK
397                 && event.getRepeatCount() == 0) {
398             // Take care of calling this method on earlier versions of
399             // the platform where it doesn't exist.
400             onBackPressed();
401             return true;
402         }
403 
404         return super.onKeyDown(keyCode, event);
405     }
406 
407     /**
408      * Dispatch onLowMemory() to all fragments.
409      */
410     @Override
onLowMemory()411     public void onLowMemory() {
412         super.onLowMemory();
413         mFragments.dispatchLowMemory();
414     }
415 
416     /**
417      * Dispatch context and options menu to fragments.
418      */
419     @Override
onMenuItemSelected(int featureId, MenuItem item)420     public boolean onMenuItemSelected(int featureId, MenuItem item) {
421         if (super.onMenuItemSelected(featureId, item)) {
422             return true;
423         }
424 
425         switch (featureId) {
426             case Window.FEATURE_OPTIONS_PANEL:
427                 return mFragments.dispatchOptionsItemSelected(item);
428 
429             case Window.FEATURE_CONTEXT_MENU:
430                 return mFragments.dispatchContextItemSelected(item);
431 
432             default:
433                 return false;
434         }
435     }
436 
437     /**
438      * Call onOptionsMenuClosed() on fragments.
439      */
440     @Override
onPanelClosed(int featureId, Menu menu)441     public void onPanelClosed(int featureId, Menu menu) {
442         switch (featureId) {
443             case Window.FEATURE_OPTIONS_PANEL:
444                 mFragments.dispatchOptionsMenuClosed(menu);
445                 break;
446         }
447         super.onPanelClosed(featureId, menu);
448     }
449 
450     /**
451      * Dispatch onPause() to fragments.
452      */
453     @Override
onPause()454     protected void onPause() {
455         super.onPause();
456         mResumed = false;
457         if (mHandler.hasMessages(MSG_RESUME_PENDING)) {
458             mHandler.removeMessages(MSG_RESUME_PENDING);
459             onResumeFragments();
460         }
461         mFragments.dispatchPause();
462     }
463 
464     /**
465      * Handle onNewIntent() to inform the fragment manager that the
466      * state is not saved.  If you are handling new intents and may be
467      * making changes to the fragment state, you want to be sure to call
468      * through to the super-class here first.  Otherwise, if your state
469      * is saved but the activity is not stopped, you could get an
470      * onNewIntent() call which happens before onResume() and trying to
471      * perform fragment operations at that point will throw IllegalStateException
472      * because the fragment manager thinks the state is still saved.
473      */
474     @Override
onNewIntent(Intent intent)475     protected void onNewIntent(Intent intent) {
476         super.onNewIntent(intent);
477         mFragments.noteStateNotSaved();
478     }
479 
480     /**
481      * Hook in to note that fragment state is no longer saved.
482      */
onStateNotSaved()483     public void onStateNotSaved() {
484         mFragments.noteStateNotSaved();
485     }
486 
487     /**
488      * Dispatch onResume() to fragments.  Note that for better inter-operation
489      * with older versions of the platform, at the point of this call the
490      * fragments attached to the activity are <em>not</em> resumed.  This means
491      * that in some cases the previous state may still be saved, not allowing
492      * fragment transactions that modify the state.  To correctly interact
493      * with fragments in their proper state, you should instead override
494      * {@link #onResumeFragments()}.
495      */
496     @Override
onResume()497     protected void onResume() {
498         super.onResume();
499         mHandler.sendEmptyMessage(MSG_RESUME_PENDING);
500         mResumed = true;
501         mFragments.execPendingActions();
502     }
503 
504     /**
505      * Dispatch onResume() to fragments.
506      */
507     @Override
onPostResume()508     protected void onPostResume() {
509         super.onPostResume();
510         mHandler.removeMessages(MSG_RESUME_PENDING);
511         onResumeFragments();
512         mFragments.execPendingActions();
513     }
514 
515     /**
516      * This is the fragment-orientated version of {@link #onResume()} that you
517      * can override to perform operations in the Activity at the same point
518      * where its fragments are resumed.  Be sure to always call through to
519      * the super-class.
520      */
onResumeFragments()521     protected void onResumeFragments() {
522         mFragments.dispatchResume();
523     }
524 
525     /**
526      * Dispatch onPrepareOptionsMenu() to fragments.
527      */
528     @Override
onPreparePanel(int featureId, View view, Menu menu)529     public boolean onPreparePanel(int featureId, View view, Menu menu) {
530         if (featureId == Window.FEATURE_OPTIONS_PANEL && menu != null) {
531             if (mOptionsMenuInvalidated) {
532                 mOptionsMenuInvalidated = false;
533                 menu.clear();
534                 onCreatePanelMenu(featureId, menu);
535             }
536             boolean goforit = onPrepareOptionsPanel(view, menu);
537             goforit |= mFragments.dispatchPrepareOptionsMenu(menu);
538             return goforit;
539         }
540         return super.onPreparePanel(featureId, view, menu);
541     }
542 
543     /**
544      * @hide
545      */
onPrepareOptionsPanel(View view, Menu menu)546     protected boolean onPrepareOptionsPanel(View view, Menu menu) {
547         return super.onPreparePanel(Window.FEATURE_OPTIONS_PANEL, view, menu);
548     }
549 
550     /**
551      * Retain all appropriate fragment and loader state.  You can NOT
552      * override this yourself!  Use {@link #onRetainCustomNonConfigurationInstance()}
553      * if you want to retain your own state.
554      */
555     @Override
onRetainNonConfigurationInstance()556     public final Object onRetainNonConfigurationInstance() {
557         if (mStopped) {
558             doReallyStop(true);
559         }
560 
561         Object custom = onRetainCustomNonConfigurationInstance();
562 
563         FragmentManagerNonConfig fragments = mFragments.retainNestedNonConfig();
564         SimpleArrayMap<String, LoaderManager> loaders = mFragments.retainLoaderNonConfig();
565 
566         if (fragments == null && loaders == null && custom == null) {
567             return null;
568         }
569 
570         NonConfigurationInstances nci = new NonConfigurationInstances();
571         nci.custom = custom;
572         nci.fragments = fragments;
573         nci.loaders = loaders;
574         return nci;
575     }
576 
577     /**
578      * Save all appropriate fragment state.
579      */
580     @Override
onSaveInstanceState(Bundle outState)581     protected void onSaveInstanceState(Bundle outState) {
582         super.onSaveInstanceState(outState);
583         Parcelable p = mFragments.saveAllState();
584         if (p != null) {
585             outState.putParcelable(FRAGMENTS_TAG, p);
586         }
587         if (mPendingFragmentActivityResults.size() > 0) {
588             outState.putInt(NEXT_CANDIDATE_REQUEST_INDEX_TAG, mNextCandidateRequestIndex);
589 
590             int[] requestCodes = new int[mPendingFragmentActivityResults.size()];
591             String[] fragmentWhos = new String[mPendingFragmentActivityResults.size()];
592             for (int i = 0; i < mPendingFragmentActivityResults.size(); i++) {
593                 requestCodes[i] = mPendingFragmentActivityResults.keyAt(i);
594                 fragmentWhos[i] = mPendingFragmentActivityResults.valueAt(i);
595             }
596             outState.putIntArray(ALLOCATED_REQUEST_INDICIES_TAG, requestCodes);
597             outState.putStringArray(REQUEST_FRAGMENT_WHO_TAG, fragmentWhos);
598         }
599     }
600 
601     /**
602      * Dispatch onStart() to all fragments.  Ensure any created loaders are
603      * now started.
604      */
605     @Override
onStart()606     protected void onStart() {
607         super.onStart();
608 
609         mStopped = false;
610         mReallyStopped = false;
611         mHandler.removeMessages(MSG_REALLY_STOPPED);
612 
613         if (!mCreated) {
614             mCreated = true;
615             mFragments.dispatchActivityCreated();
616         }
617 
618         mFragments.noteStateNotSaved();
619         mFragments.execPendingActions();
620 
621         mFragments.doLoaderStart();
622 
623         // NOTE: HC onStart goes here.
624 
625         mFragments.dispatchStart();
626         mFragments.reportLoaderStart();
627     }
628 
629     /**
630      * Dispatch onStop() to all fragments.  Ensure all loaders are stopped.
631      */
632     @Override
onStop()633     protected void onStop() {
634         super.onStop();
635 
636         mStopped = true;
637         mHandler.sendEmptyMessage(MSG_REALLY_STOPPED);
638 
639         mFragments.dispatchStop();
640     }
641 
642     // ------------------------------------------------------------------------
643     // NEW METHODS
644     // ------------------------------------------------------------------------
645 
646     /**
647      * Use this instead of {@link #onRetainNonConfigurationInstance()}.
648      * Retrieve later with {@link #getLastCustomNonConfigurationInstance()}.
649      */
onRetainCustomNonConfigurationInstance()650     public Object onRetainCustomNonConfigurationInstance() {
651         return null;
652     }
653 
654     /**
655      * Return the value previously returned from
656      * {@link #onRetainCustomNonConfigurationInstance()}.
657      */
658     @SuppressWarnings("deprecation")
getLastCustomNonConfigurationInstance()659     public Object getLastCustomNonConfigurationInstance() {
660         NonConfigurationInstances nc = (NonConfigurationInstances)
661                 getLastNonConfigurationInstance();
662         return nc != null ? nc.custom : null;
663     }
664 
665     /**
666      * Support library version of {@link Activity#invalidateOptionsMenu}.
667      *
668      * <p>Invalidate the activity's options menu. This will cause relevant presentations
669      * of the menu to fully update via calls to onCreateOptionsMenu and
670      * onPrepareOptionsMenu the next time the menu is requested.
671      */
supportInvalidateOptionsMenu()672     public void supportInvalidateOptionsMenu() {
673         if (android.os.Build.VERSION.SDK_INT >= HONEYCOMB) {
674             // If we are running on HC or greater, we can use the framework
675             // API to invalidate the options menu.
676             ActivityCompatHoneycomb.invalidateOptionsMenu(this);
677             return;
678         }
679 
680         // Whoops, older platform...  we'll use a hack, to manually rebuild
681         // the options menu the next time it is prepared.
682         mOptionsMenuInvalidated = true;
683     }
684 
685     /**
686      * Print the Activity's state into the given stream.  This gets invoked if
687      * you run "adb shell dumpsys activity <activity_component_name>".
688      *
689      * @param prefix Desired prefix to prepend at each line of output.
690      * @param fd The raw file descriptor that the dump is being sent to.
691      * @param writer The PrintWriter to which you should dump your state.  This will be
692      * closed for you after you return.
693      * @param args additional arguments to the dump request.
694      */
dump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args)695     public void dump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) {
696         if (android.os.Build.VERSION.SDK_INT >= HONEYCOMB) {
697             // XXX This can only work if we can call the super-class impl. :/
698             //ActivityCompatHoneycomb.dump(this, prefix, fd, writer, args);
699         }
700         writer.print(prefix); writer.print("Local FragmentActivity ");
701                 writer.print(Integer.toHexString(System.identityHashCode(this)));
702                 writer.println(" State:");
703         String innerPrefix = prefix + "  ";
704         writer.print(innerPrefix); writer.print("mCreated=");
705                 writer.print(mCreated); writer.print("mResumed=");
706                 writer.print(mResumed); writer.print(" mStopped=");
707                 writer.print(mStopped); writer.print(" mReallyStopped=");
708                 writer.println(mReallyStopped);
709         mFragments.dumpLoaders(innerPrefix, fd, writer, args);
710         mFragments.getSupportFragmentManager().dump(prefix, fd, writer, args);
711         writer.print(prefix); writer.println("View Hierarchy:");
712         dumpViewHierarchy(prefix + "  ", writer, getWindow().getDecorView());
713     }
714 
viewToString(View view)715     private static String viewToString(View view) {
716         StringBuilder out = new StringBuilder(128);
717         out.append(view.getClass().getName());
718         out.append('{');
719         out.append(Integer.toHexString(System.identityHashCode(view)));
720         out.append(' ');
721         switch (view.getVisibility()) {
722             case View.VISIBLE: out.append('V'); break;
723             case View.INVISIBLE: out.append('I'); break;
724             case View.GONE: out.append('G'); break;
725             default: out.append('.'); break;
726         }
727         out.append(view.isFocusable() ? 'F' : '.');
728         out.append(view.isEnabled() ? 'E' : '.');
729         out.append(view.willNotDraw() ? '.' : 'D');
730         out.append(view.isHorizontalScrollBarEnabled()? 'H' : '.');
731         out.append(view.isVerticalScrollBarEnabled() ? 'V' : '.');
732         out.append(view.isClickable() ? 'C' : '.');
733         out.append(view.isLongClickable() ? 'L' : '.');
734         out.append(' ');
735         out.append(view.isFocused() ? 'F' : '.');
736         out.append(view.isSelected() ? 'S' : '.');
737         out.append(view.isPressed() ? 'P' : '.');
738         out.append(' ');
739         out.append(view.getLeft());
740         out.append(',');
741         out.append(view.getTop());
742         out.append('-');
743         out.append(view.getRight());
744         out.append(',');
745         out.append(view.getBottom());
746         final int id = view.getId();
747         if (id != View.NO_ID) {
748             out.append(" #");
749             out.append(Integer.toHexString(id));
750             final Resources r = view.getResources();
751             if (id != 0 && r != null) {
752                 try {
753                     String pkgname;
754                     switch (id&0xff000000) {
755                         case 0x7f000000:
756                             pkgname="app";
757                             break;
758                         case 0x01000000:
759                             pkgname="android";
760                             break;
761                         default:
762                             pkgname = r.getResourcePackageName(id);
763                             break;
764                     }
765                     String typename = r.getResourceTypeName(id);
766                     String entryname = r.getResourceEntryName(id);
767                     out.append(" ");
768                     out.append(pkgname);
769                     out.append(":");
770                     out.append(typename);
771                     out.append("/");
772                     out.append(entryname);
773                 } catch (Resources.NotFoundException e) {
774                 }
775             }
776         }
777         out.append("}");
778         return out.toString();
779     }
780 
dumpViewHierarchy(String prefix, PrintWriter writer, View view)781     private void dumpViewHierarchy(String prefix, PrintWriter writer, View view) {
782         writer.print(prefix);
783         if (view == null) {
784             writer.println("null");
785             return;
786         }
787         writer.println(viewToString(view));
788         if (!(view instanceof ViewGroup)) {
789             return;
790         }
791         ViewGroup grp = (ViewGroup)view;
792         final int N = grp.getChildCount();
793         if (N <= 0) {
794             return;
795         }
796         prefix = prefix + "  ";
797         for (int i=0; i<N; i++) {
798             dumpViewHierarchy(prefix, writer, grp.getChildAt(i));
799         }
800     }
801 
doReallyStop(boolean retaining)802     void doReallyStop(boolean retaining) {
803         if (!mReallyStopped) {
804             mReallyStopped = true;
805             mRetaining = retaining;
806             mHandler.removeMessages(MSG_REALLY_STOPPED);
807             onReallyStop();
808         } else if (retaining) {
809             // We're already really stopped, but we've been asked to retain.
810             // Our fragments are taken care of but we need to mark the loaders for retention.
811             // In order to do this correctly we need to restart the loaders first before
812             // handing them off to the next activity.
813             mFragments.doLoaderStart();
814             mFragments.doLoaderStop(true);
815         }
816     }
817 
818     /**
819      * Pre-HC, we didn't have a way to determine whether an activity was
820      * being stopped for a config change or not until we saw
821      * onRetainNonConfigurationInstance() called after onStop().  However
822      * we need to know this, to know whether to retain fragments.  This will
823      * tell us what we need to know.
824      */
onReallyStop()825     void onReallyStop() {
826         mFragments.doLoaderStop(mRetaining);
827 
828         mFragments.dispatchReallyStop();
829     }
830 
831     // ------------------------------------------------------------------------
832     // FRAGMENT SUPPORT
833     // ------------------------------------------------------------------------
834 
835     /**
836      * Called when a fragment is attached to the activity.
837      *
838      * <p>This is called after the attached fragment's <code>onAttach</code> and before
839      * the attached fragment's <code>onCreate</code> if the fragment has not yet had a previous
840      * call to <code>onCreate</code>.</p>
841      */
842     @SuppressWarnings("unused")
onAttachFragment(Fragment fragment)843     public void onAttachFragment(Fragment fragment) {
844     }
845 
846     /**
847      * Return the FragmentManager for interacting with fragments associated
848      * with this activity.
849      */
getSupportFragmentManager()850     public FragmentManager getSupportFragmentManager() {
851         return mFragments.getSupportFragmentManager();
852     }
853 
getSupportLoaderManager()854     public LoaderManager getSupportLoaderManager() {
855         return mFragments.getSupportLoaderManager();
856     }
857 
858     /**
859      * Modifies the standard behavior to allow results to be delivered to fragments.
860      * This imposes a restriction that requestCode be <= 0xffff.
861      */
862     @Override
startActivityForResult(Intent intent, int requestCode)863     public void startActivityForResult(Intent intent, int requestCode) {
864         // If this was started from a Fragment we've already checked the upper 16 bits were not in
865         // use, and then repurposed them for the Fragment's index.
866         if (!mStartedActivityFromFragment) {
867             if (requestCode != -1) {
868                 checkForValidRequestCode(requestCode);
869             }
870         }
871         super.startActivityForResult(intent, requestCode);
872     }
873 
874     @Override
validateRequestPermissionsRequestCode(int requestCode)875     public final void validateRequestPermissionsRequestCode(int requestCode) {
876         // We use 16 bits of the request code to encode the fragment id when
877         // requesting permissions from a fragment. Hence, requestPermissions()
878         // should validate the code against that but we cannot override it as
879         // we can not then call super and also the ActivityCompat would call
880         // back to this override. To handle this we use dependency inversion
881         // where we are the validator of request codes when requesting
882         // permissions in ActivityCompat.
883         if (!mRequestedPermissionsFromFragment
884                 && requestCode != -1) {
885             checkForValidRequestCode(requestCode);
886         }
887     }
888 
889     /**
890      * Callback for the result from requesting permissions. This method
891      * is invoked for every call on {@link #requestPermissions(String[], int)}.
892      * <p>
893      * <strong>Note:</strong> It is possible that the permissions request interaction
894      * with the user is interrupted. In this case you will receive empty permissions
895      * and results arrays which should be treated as a cancellation.
896      * </p>
897      *
898      * @param requestCode The request code passed in {@link #requestPermissions(String[], int)}.
899      * @param permissions The requested permissions. Never null.
900      * @param grantResults The grant results for the corresponding permissions
901      *     which is either {@link android.content.pm.PackageManager#PERMISSION_GRANTED}
902      *     or {@link android.content.pm.PackageManager#PERMISSION_DENIED}. Never null.
903      *
904      * @see #requestPermissions(String[], int)
905      */
onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults)906     public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,
907             @NonNull int[] grantResults) {
908         int index = (requestCode >> 16) & 0xffff;
909         if (index != 0) {
910             index--;
911 
912             String who = mPendingFragmentActivityResults.get(index);
913             mPendingFragmentActivityResults.remove(index);
914             if (who == null) {
915                 Log.w(TAG, "Activity result delivered for unknown Fragment.");
916                 return;
917             }
918             Fragment frag = mFragments.findFragmentByWho(who);
919             if (frag == null) {
920                 Log.w(TAG, "Activity result no fragment exists for who: " + who);
921             } else {
922                 frag.onRequestPermissionsResult(requestCode & 0xffff, permissions, grantResults);
923             }
924         }
925     }
926 
927     /**
928      * Called by Fragment.startActivityForResult() to implement its behavior.
929      */
startActivityFromFragment(Fragment fragment, Intent intent, int requestCode)930     public void startActivityFromFragment(Fragment fragment, Intent intent,
931             int requestCode) {
932         startActivityFromFragment(fragment, intent, requestCode, null);
933     }
934 
935     /**
936      * Called by Fragment.startActivityForResult() to implement its behavior.
937      */
startActivityFromFragment(Fragment fragment, Intent intent, int requestCode, @Nullable Bundle options)938     public void startActivityFromFragment(Fragment fragment, Intent intent,
939             int requestCode, @Nullable Bundle options) {
940         mStartedActivityFromFragment = true;
941         try {
942             if (requestCode == -1) {
943                 ActivityCompat.startActivityForResult(this, intent, -1, options);
944                 return;
945             }
946             checkForValidRequestCode(requestCode);
947             int requestIndex = allocateRequestIndex(fragment);
948             ActivityCompat.startActivityForResult(
949                     this, intent, ((requestIndex + 1) << 16) + (requestCode & 0xffff), options);
950         } finally {
951             mStartedActivityFromFragment = false;
952         }
953     }
954 
955     /**
956      * Called by Fragment.startIntentSenderForResult() to implement its behavior.
957      */
startIntentSenderFromFragment(Fragment fragment, IntentSender intent, int requestCode, @Nullable Intent fillInIntent, int flagsMask, int flagsValues, int extraFlags, Bundle options)958     public void startIntentSenderFromFragment(Fragment fragment, IntentSender intent,
959             int requestCode, @Nullable Intent fillInIntent, int flagsMask, int flagsValues,
960             int extraFlags, Bundle options) throws IntentSender.SendIntentException {
961         mStartedIntentSenderFromFragment = true;
962         try {
963             if (requestCode == -1) {
964                 ActivityCompat.startIntentSenderForResult(this, intent, requestCode, fillInIntent,
965                         flagsMask, flagsValues, extraFlags, options);
966                 return;
967             }
968             checkForValidRequestCode(requestCode);
969             int requestIndex = allocateRequestIndex(fragment);
970             ActivityCompat.startIntentSenderForResult(this, intent,
971                     ((requestIndex + 1) << 16) + (requestCode & 0xffff), fillInIntent,
972                     flagsMask, flagsValues, extraFlags, options);
973         } finally {
974             mStartedIntentSenderFromFragment = false;
975         }
976     }
977 
978     // Allocates the next available startActivityForResult request index.
allocateRequestIndex(Fragment fragment)979     private int allocateRequestIndex(Fragment fragment) {
980         // Sanity check that we havn't exhaused the request index space.
981         if (mPendingFragmentActivityResults.size() >= MAX_NUM_PENDING_FRAGMENT_ACTIVITY_RESULTS) {
982             throw new IllegalStateException("Too many pending Fragment activity results.");
983         }
984 
985         // Find an unallocated request index in the mPendingFragmentActivityResults map.
986         while (mPendingFragmentActivityResults.indexOfKey(mNextCandidateRequestIndex) >= 0) {
987             mNextCandidateRequestIndex =
988                     (mNextCandidateRequestIndex + 1) % MAX_NUM_PENDING_FRAGMENT_ACTIVITY_RESULTS;
989         }
990 
991         int requestIndex = mNextCandidateRequestIndex;
992         mPendingFragmentActivityResults.put(requestIndex, fragment.mWho);
993         mNextCandidateRequestIndex =
994                 (mNextCandidateRequestIndex + 1) % MAX_NUM_PENDING_FRAGMENT_ACTIVITY_RESULTS;
995 
996         return requestIndex;
997     }
998 
999     /**
1000      * Called by Fragment.requestPermissions() to implement its behavior.
1001      */
requestPermissionsFromFragment(Fragment fragment, String[] permissions, int requestCode)1002     private void requestPermissionsFromFragment(Fragment fragment, String[] permissions,
1003             int requestCode) {
1004         if (requestCode == -1) {
1005             ActivityCompat.requestPermissions(this, permissions, requestCode);
1006             return;
1007         }
1008         checkForValidRequestCode(requestCode);
1009         try {
1010             mRequestedPermissionsFromFragment = true;
1011             int requestIndex = allocateRequestIndex(fragment);
1012             ActivityCompat.requestPermissions(this, permissions,
1013                     ((requestIndex + 1) << 16) + (requestCode & 0xffff));
1014         } finally {
1015             mRequestedPermissionsFromFragment = false;
1016         }
1017     }
1018 
1019     class HostCallbacks extends FragmentHostCallback<FragmentActivity> {
HostCallbacks()1020         public HostCallbacks() {
1021             super(FragmentActivity.this /*fragmentActivity*/);
1022         }
1023 
1024         @Override
onDump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args)1025         public void onDump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) {
1026             FragmentActivity.this.dump(prefix, fd, writer, args);
1027         }
1028 
1029         @Override
onShouldSaveFragmentState(Fragment fragment)1030         public boolean onShouldSaveFragmentState(Fragment fragment) {
1031             return !isFinishing();
1032         }
1033 
1034         @Override
onGetLayoutInflater()1035         public LayoutInflater onGetLayoutInflater() {
1036             return FragmentActivity.this.getLayoutInflater().cloneInContext(FragmentActivity.this);
1037         }
1038 
1039         @Override
onGetHost()1040         public FragmentActivity onGetHost() {
1041             return FragmentActivity.this;
1042         }
1043 
1044         @Override
onSupportInvalidateOptionsMenu()1045         public void onSupportInvalidateOptionsMenu() {
1046             FragmentActivity.this.supportInvalidateOptionsMenu();
1047         }
1048 
1049         @Override
onStartActivityFromFragment(Fragment fragment, Intent intent, int requestCode)1050         public void onStartActivityFromFragment(Fragment fragment, Intent intent, int requestCode) {
1051             FragmentActivity.this.startActivityFromFragment(fragment, intent, requestCode);
1052         }
1053 
1054         @Override
onStartActivityFromFragment( Fragment fragment, Intent intent, int requestCode, @Nullable Bundle options)1055         public void onStartActivityFromFragment(
1056                 Fragment fragment, Intent intent, int requestCode, @Nullable Bundle options) {
1057             FragmentActivity.this.startActivityFromFragment(fragment, intent, requestCode, options);
1058         }
1059 
1060         @Override
onStartIntentSenderFromFragment(Fragment fragment, IntentSender intent, int requestCode, @Nullable Intent fillInIntent, int flagsMask, int flagsValues, int extraFlags, Bundle options)1061         public void onStartIntentSenderFromFragment(Fragment fragment, IntentSender intent,
1062                 int requestCode, @Nullable Intent fillInIntent, int flagsMask, int flagsValues,
1063                 int extraFlags, Bundle options) throws IntentSender.SendIntentException {
1064             FragmentActivity.this.startIntentSenderFromFragment(fragment, intent, requestCode,
1065                     fillInIntent, flagsMask, flagsValues, extraFlags, options);
1066         }
1067 
1068         @Override
onRequestPermissionsFromFragment(@onNull Fragment fragment, @NonNull String[] permissions, int requestCode)1069         public void onRequestPermissionsFromFragment(@NonNull Fragment fragment,
1070                 @NonNull String[] permissions, int requestCode) {
1071             FragmentActivity.this.requestPermissionsFromFragment(fragment, permissions,
1072                     requestCode);
1073         }
1074 
1075         @Override
onShouldShowRequestPermissionRationale(@onNull String permission)1076         public boolean onShouldShowRequestPermissionRationale(@NonNull String permission) {
1077             return ActivityCompat.shouldShowRequestPermissionRationale(
1078                     FragmentActivity.this, permission);
1079         }
1080 
1081         @Override
onHasWindowAnimations()1082         public boolean onHasWindowAnimations() {
1083             return getWindow() != null;
1084         }
1085 
1086         @Override
onGetWindowAnimations()1087         public int onGetWindowAnimations() {
1088             final Window w = getWindow();
1089             return (w == null) ? 0 : w.getAttributes().windowAnimations;
1090         }
1091 
1092         @Override
onAttachFragment(Fragment fragment)1093         public void onAttachFragment(Fragment fragment) {
1094             FragmentActivity.this.onAttachFragment(fragment);
1095         }
1096 
1097         @Nullable
1098         @Override
onFindViewById(int id)1099         public View onFindViewById(int id) {
1100             return FragmentActivity.this.findViewById(id);
1101         }
1102 
1103         @Override
onHasView()1104         public boolean onHasView() {
1105             final Window w = getWindow();
1106             return (w != null && w.peekDecorView() != null);
1107         }
1108     }
1109 }
1110