1 /*
2  * Copyright (C) 2012 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.app;
18 
19 import static android.view.WindowManager.LayoutParams.INVALID_WINDOW_TYPE;
20 import static android.view.WindowManager.LayoutParams.TYPE_PRESENTATION;
21 import static android.view.WindowManager.LayoutParams.TYPE_PRIVATE_PRESENTATION;
22 
23 import android.annotation.NonNull;
24 import android.compat.annotation.UnsupportedAppUsage;
25 import android.content.Context;
26 import android.content.res.Resources;
27 import android.hardware.display.DisplayManager;
28 import android.hardware.display.DisplayManager.DisplayListener;
29 import android.os.Build;
30 import android.os.Handler;
31 import android.os.Looper;
32 import android.util.TypedValue;
33 import android.view.ContextThemeWrapper;
34 import android.view.Display;
35 import android.view.Gravity;
36 import android.view.Window;
37 import android.view.WindowManager;
38 import android.view.WindowManager.LayoutParams.WindowType;
39 
40 import java.util.Objects;
41 
42 /**
43  * Base class for presentations.
44  * <p>
45  * A presentation is a special kind of dialog whose purpose is to present
46  * content on a secondary display.  A {@link Presentation} is associated with
47  * the target {@link Display} at creation time and configures its context and
48  * resource configuration according to the display's metrics.
49  * </p><p>
50  * Notably, the {@link Context} of a presentation is different from the context
51  * of its containing {@link Activity}.  It is important to inflate the layout
52  * of a presentation and load other resources using the presentation's own context
53  * to ensure that assets of the correct size and density for the target display
54  * are loaded.
55  * </p><p>
56  * A presentation is automatically canceled (see {@link Dialog#cancel()}) when
57  * the display to which it is attached is removed.  An activity should take
58  * care of pausing and resuming whatever content is playing within the presentation
59  * whenever the activity itself is paused or resumed.
60  * </p>
61  *
62  * <h3>Choosing a presentation display</h3>
63  * <p>
64  * Before showing a {@link Presentation} it's important to choose the {@link Display}
65  * on which it will appear.  Choosing a presentation display is sometimes difficult
66  * because there may be multiple displays attached.  Rather than trying to guess
67  * which display is best, an application should let the system choose a suitable
68  * presentation display.
69  * </p><p>
70  * There are two main ways to choose a {@link Display}.
71  * </p>
72  *
73  * <h4>Using the media router to choose a presentation display</h4>
74  * <p>
75  * The easiest way to choose a presentation display is to use the
76  * {@link android.media.MediaRouter MediaRouter} API.  The media router service keeps
77  * track of which audio and video routes are available on the system.
78  * The media router sends notifications whenever routes are selected or unselected
79  * or when the preferred presentation display of a route changes.
80  * So an application can simply watch for these notifications and show or dismiss
81  * a presentation on the preferred presentation display automatically.
82  * </p><p>
83  * The preferred presentation display is the display that the media router recommends
84  * that the application should use if it wants to show content on the secondary display.
85  * Sometimes there may not be a preferred presentation display in which
86  * case the application should show its content locally without using a presentation.
87  * </p><p>
88  * Here's how to use the media router to create and show a presentation on the preferred
89  * presentation display using {@link android.media.MediaRouter.RouteInfo#getPresentationDisplay()}.
90  * </p>
91  * <pre>
92  * MediaRouter mediaRouter = (MediaRouter) context.getSystemService(Context.MEDIA_ROUTER_SERVICE);
93  * MediaRouter.RouteInfo route = mediaRouter.getSelectedRoute(MediaRouter.ROUTE_TYPE_LIVE_VIDEO);
94  * if (route != null) {
95  *     Display presentationDisplay = route.getPresentationDisplay();
96  *     if (presentationDisplay != null) {
97  *         Presentation presentation = new MyPresentation(context, presentationDisplay);
98  *         presentation.show();
99  *     }
100  * }</pre>
101  * <p>
102  * The following sample code from <code>ApiDemos</code> demonstrates how to use the media
103  * router to automatically switch between showing content in the main activity and showing
104  * the content in a presentation when a presentation display is available.
105  * </p>
106  * {@sample development/samples/ApiDemos/src/com/example/android/apis/app/PresentationWithMediaRouterActivity.java
107  *      activity}
108  *
109  * <h4>Using the display manager to choose a presentation display</h4>
110  * <p>
111  * Another way to choose a presentation display is to use the {@link DisplayManager} API
112  * directly.  The display manager service provides functions to enumerate and describe all
113  * displays that are attached to the system including displays that may be used
114  * for presentations.
115  * </p><p>
116  * The display manager keeps track of all displays in the system.  However, not all
117  * displays are appropriate for showing presentations.  For example, if an activity
118  * attempted to show a presentation on the main display it might obscure its own content
119  * (it's like opening a dialog on top of your activity).  Creating a presentation on the main
120  * display will result in {@link android.view.WindowManager.InvalidDisplayException} being thrown
121  * when invoking {@link #show()}.
122  * </p><p>
123  * Here's how to identify suitable displays for showing presentations using
124  * {@link DisplayManager#getDisplays(String)} and the
125  * {@link DisplayManager#DISPLAY_CATEGORY_PRESENTATION} category.
126  * </p>
127  * <pre>
128  * DisplayManager displayManager = (DisplayManager) context.getSystemService(Context.DISPLAY_SERVICE);
129  * Display[] presentationDisplays = displayManager.getDisplays(DisplayManager.DISPLAY_CATEGORY_PRESENTATION);
130  * if (presentationDisplays.length > 0) {
131  *     // If there is more than one suitable presentation display, then we could consider
132  *     // giving the user a choice.  For this example, we simply choose the first display
133  *     // which is the one the system recommends as the preferred presentation display.
134  *     Display display = presentationDisplays[0];
135  *     Presentation presentation = new MyPresentation(context, presentationDisplay);
136  *     presentation.show();
137  * }</pre>
138  * <p>
139  * The following sample code from <code>ApiDemos</code> demonstrates how to use the display
140  * manager to enumerate displays and show content on multiple presentation displays
141  * simultaneously.
142  * </p>
143  * {@sample development/samples/ApiDemos/src/com/example/android/apis/app/PresentationActivity.java
144  *      activity}
145  *
146  * @see android.media.MediaRouter#ROUTE_TYPE_LIVE_VIDEO for information on about live
147  * video routes and how to obtain the preferred presentation display for the
148  * current media route.
149  * @see DisplayManager for information on how to enumerate displays and receive
150  * notifications when displays are added or removed.
151  */
152 public class Presentation extends Dialog {
153     private static final String TAG = "Presentation";
154 
155     private final Display mDisplay;
156     private final DisplayManager mDisplayManager;
157     private final Handler mHandler = new Handler(Objects.requireNonNull(Looper.myLooper(),
158             "Presentation must be constructed on a looper thread."));
159 
160     /**
161      * Creates a new presentation that is attached to the specified display
162      * using the default theme.
163      *
164      * @param outerContext The context of the application that is showing the presentation.
165      * The presentation will create its own context (see {@link #getContext()}) based
166      * on this context and information about the associated display.
167      * @param display The display to which the presentation should be attached.
168      */
Presentation(Context outerContext, Display display)169     public Presentation(Context outerContext, Display display) {
170         this(outerContext, display, 0);
171     }
172 
173     /**
174      * Creates a new presentation that is attached to the specified display
175      * using the optionally specified theme.
176      *
177      * @param outerContext The context of the application that is showing the presentation.
178      * The presentation will create its own context (see {@link #getContext()}) based
179      * on this context and information about the associated display.
180      * From {@link android.os.Build.VERSION_CODES#S}, the presentation will create its own window
181      * context based on this context, information about the associated display. Customizing window
182      * type by {@link Window#setType(int) #getWindow#setType(int)} causes the mismatch of the window
183      * and the created window context, which leads to
184      * {@link android.view.WindowManager.InvalidDisplayException} when invoking {@link #show()}.
185      * @param display The display to which the presentation should be attached.
186      * @param theme A style resource describing the theme to use for the window.
187      * See <a href="{@docRoot}guide/topics/resources/available-resources.html#stylesandthemes">
188      * Style and Theme Resources</a> for more information about defining and using
189      * styles.  This theme is applied on top of the current theme in
190      * <var>outerContext</var>.  If 0, the default presentation theme will be used.
191      */
Presentation(Context outerContext, Display display, int theme)192     public Presentation(Context outerContext, Display display, int theme) {
193         this(outerContext, display, theme, INVALID_WINDOW_TYPE);
194     }
195 
196     /**
197      * Creates a new presentation that is attached to the specified display
198      * using the optionally specified theme, and override the default window type for the
199      * presentation.
200      * @param outerContext The context of the application that is showing the presentation.
201      * The presentation will create its own context (see {@link #getContext()}) based
202      * on this context and information about the associated display.
203      * From {@link android.os.Build.VERSION_CODES#S}, the presentation will create its own window
204      * context based on this context, information about the associated display and the window type.
205      * If the window type is not specified, the presentation will choose the default type for the
206      * presentation.
207      * @param display The display to which the presentation should be attached.
208      * @param theme A style resource describing the theme to use for the window.
209      * See <a href="{@docRoot}guide/topics/resources/available-resources.html#stylesandthemes">
210      * Style and Theme Resources</a> for more information about defining and using
211      * styles.  This theme is applied on top of the current theme in
212      * <var>outerContext</var>.  If 0, the default presentation theme will be used.
213      * @param type Window type.
214      *
215      * @hide
216      */
Presentation(@onNull Context outerContext, @NonNull Display display, int theme, @WindowType int type)217     public Presentation(@NonNull Context outerContext, @NonNull Display display, int theme,
218             @WindowType int type) {
219         super(createPresentationContext(outerContext, display, theme, type), theme, false);
220 
221         mDisplay = display;
222         mDisplayManager = getContext().getSystemService(DisplayManager.class);
223 
224         final Window w = getWindow();
225         final WindowManager.LayoutParams attr = w.getAttributes();
226         w.setAttributes(attr);
227         w.setGravity(Gravity.FILL);
228         w.setType(getWindowType(type, display));
229         setCanceledOnTouchOutside(false);
230     }
231 
getWindowType(@indowType int type, @NonNull Display display)232     private static @WindowType int getWindowType(@WindowType int type, @NonNull Display display) {
233         if (type != INVALID_WINDOW_TYPE) {
234             return type;
235         }
236         return (display.getFlags() & Display.FLAG_PRIVATE) != 0 ? TYPE_PRIVATE_PRESENTATION
237                 : TYPE_PRESENTATION;
238     }
239 
240     /**
241      * Gets the {@link Display} that this presentation appears on.
242      *
243      * @return The display.
244      */
getDisplay()245     public Display getDisplay() {
246         return mDisplay;
247     }
248 
249     /**
250      * Gets the {@link Resources} that should be used to inflate the layout of this presentation.
251      * This resources object has been configured according to the metrics of the
252      * display that the presentation appears on.
253      *
254      * @return The presentation resources object.
255      */
getResources()256     public Resources getResources() {
257         return getContext().getResources();
258     }
259 
260     @Override
onStart()261     protected void onStart() {
262         super.onStart();
263         mDisplayManager.registerDisplayListener(mDisplayListener, mHandler);
264     }
265 
266     @Override
onStop()267     protected void onStop() {
268         mDisplayManager.unregisterDisplayListener(mDisplayListener);
269         super.onStop();
270     }
271 
272     /**
273      * Inherited from {@link Dialog#show}. Will throw
274      * {@link android.view.WindowManager.InvalidDisplayException} if the specified secondary
275      * {@link Display} can't be found or if it does not have {@link Display#FLAG_PRESENTATION} set.
276      */
277     @Override
show()278     public void show() {
279         super.show();
280     }
281 
282     /**
283      * Called by the system when the {@link Display} to which the presentation
284      * is attached has been removed.
285      *
286      * The system automatically calls {@link #cancel} to dismiss the presentation
287      * after sending this event.
288      *
289      * @see #getDisplay
290      */
onDisplayRemoved()291     public void onDisplayRemoved() {
292     }
293 
294     /**
295      * Called by the system when the properties of the {@link Display} to which
296      * the presentation is attached have changed.
297      *
298      * @see #getDisplay
299      */
onDisplayChanged()300     public void onDisplayChanged() {
301     }
302 
handleDisplayRemoved()303     private void handleDisplayRemoved() {
304         onDisplayRemoved();
305         cancel();
306     }
307 
handleDisplayChanged()308     private void handleDisplayChanged() {
309         onDisplayChanged();
310     }
311 
312     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, publicAlternatives = "{@code N/A}")
createPresentationContext(Context outerContext, Display display, int theme)313     private static Context createPresentationContext(Context outerContext, Display display,
314             int theme) {
315         return createPresentationContext(outerContext, display, theme, INVALID_WINDOW_TYPE);
316     }
317 
createPresentationContext( Context outerContext, Display display, int theme, @WindowType int type)318     private static Context createPresentationContext(
319             Context outerContext, Display display, int theme, @WindowType int type) {
320         if (outerContext == null) {
321             throw new IllegalArgumentException("outerContext must not be null");
322         }
323         if (display == null) {
324             throw new IllegalArgumentException("display must not be null");
325         }
326 
327         Context windowContext = outerContext.createDisplayContext(display)
328                 .createWindowContext(getWindowType(type, display), null /* options */);
329         if (theme == 0) {
330             TypedValue outValue = new TypedValue();
331             windowContext.getTheme().resolveAttribute(
332                     com.android.internal.R.attr.presentationTheme, outValue, true);
333             theme = outValue.resourceId;
334         }
335         return new ContextThemeWrapper(windowContext, theme);
336     }
337 
338     private final DisplayListener mDisplayListener = new DisplayListener() {
339         @Override
340         public void onDisplayAdded(int displayId) {
341         }
342 
343         @Override
344         public void onDisplayRemoved(int displayId) {
345             if (displayId == mDisplay.getDisplayId()) {
346                 handleDisplayRemoved();
347             }
348         }
349 
350         @Override
351         public void onDisplayChanged(int displayId) {
352             if (displayId == mDisplay.getDisplayId()) {
353                 handleDisplayChanged();
354             }
355         }
356     };
357 }
358