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.hardware.display; 18 19 import android.annotation.NonNull; 20 import android.annotation.Nullable; 21 import android.content.Context; 22 import android.media.projection.MediaProjection; 23 import android.os.Handler; 24 import android.util.SparseArray; 25 import android.view.Display; 26 import android.view.Surface; 27 28 import java.util.ArrayList; 29 30 /** 31 * Manages the properties of attached displays. 32 * <p> 33 * Get an instance of this class by calling 34 * {@link android.content.Context#getSystemService(java.lang.String) 35 * Context.getSystemService()} with the argument 36 * {@link android.content.Context#DISPLAY_SERVICE}. 37 * </p> 38 */ 39 public final class DisplayManager { 40 private static final String TAG = "DisplayManager"; 41 private static final boolean DEBUG = false; 42 43 private final Context mContext; 44 private final DisplayManagerGlobal mGlobal; 45 46 private final Object mLock = new Object(); 47 private final SparseArray<Display> mDisplays = new SparseArray<Display>(); 48 49 private final ArrayList<Display> mTempDisplays = new ArrayList<Display>(); 50 51 /** 52 * Broadcast receiver that indicates when the Wifi display status changes. 53 * <p> 54 * The status is provided as a {@link WifiDisplayStatus} object in the 55 * {@link #EXTRA_WIFI_DISPLAY_STATUS} extra. 56 * </p><p> 57 * This broadcast is only sent to registered receivers and can only be sent by the system. 58 * </p> 59 * @hide 60 */ 61 public static final String ACTION_WIFI_DISPLAY_STATUS_CHANGED = 62 "android.hardware.display.action.WIFI_DISPLAY_STATUS_CHANGED"; 63 64 /** 65 * Contains a {@link WifiDisplayStatus} object. 66 * @hide 67 */ 68 public static final String EXTRA_WIFI_DISPLAY_STATUS = 69 "android.hardware.display.extra.WIFI_DISPLAY_STATUS"; 70 71 /** 72 * Display category: Presentation displays. 73 * <p> 74 * This category can be used to identify secondary displays that are suitable for 75 * use as presentation displays such as HDMI or Wireless displays. Applications 76 * may automatically project their content to presentation displays to provide 77 * richer second screen experiences. 78 * </p> 79 * 80 * @see android.app.Presentation 81 * @see Display#FLAG_PRESENTATION 82 * @see #getDisplays(String) 83 */ 84 public static final String DISPLAY_CATEGORY_PRESENTATION = 85 "android.hardware.display.category.PRESENTATION"; 86 87 /** 88 * Virtual display flag: Create a public display. 89 * 90 * <h3>Public virtual displays</h3> 91 * <p> 92 * When this flag is set, the virtual display is public. 93 * </p><p> 94 * A public virtual display behaves just like most any other display that is connected 95 * to the system such as an HDMI or Wireless display. Applications can open 96 * windows on the display and the system may mirror the contents of other displays 97 * onto it. 98 * </p><p> 99 * Creating a public virtual display that isn't restricted to own-content only implicitly 100 * creates an auto-mirroring display. See {@link #VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR} for 101 * restrictions on who is allowed to create an auto-mirroring display. 102 * </p> 103 * 104 * <h3>Private virtual displays</h3> 105 * <p> 106 * When this flag is not set, the virtual display is private as defined by the 107 * {@link Display#FLAG_PRIVATE} display flag. 108 * </p> 109 * 110 * <p> 111 * A private virtual display belongs to the application that created it. 112 * Only the a owner of a private virtual display is allowed to place windows upon it. 113 * The private virtual display also does not participate in display mirroring: it will 114 * neither receive mirrored content from another display nor allow its own content to 115 * be mirrored elsewhere. More precisely, the only processes that are allowed to 116 * enumerate or interact with the private display are those that have the same UID as the 117 * application that originally created the private virtual display. 118 * </p> 119 * 120 * @see #createVirtualDisplay 121 * @see #VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY 122 * @see #VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR 123 */ 124 public static final int VIRTUAL_DISPLAY_FLAG_PUBLIC = 1 << 0; 125 126 /** 127 * Virtual display flag: Create a presentation display. 128 * 129 * <h3>Presentation virtual displays</h3> 130 * <p> 131 * When this flag is set, the virtual display is registered as a presentation 132 * display in the {@link #DISPLAY_CATEGORY_PRESENTATION presentation display category}. 133 * Applications may automatically project their content to presentation displays 134 * to provide richer second screen experiences. 135 * </p> 136 * 137 * <h3>Non-presentation virtual displays</h3> 138 * <p> 139 * When this flag is not set, the virtual display is not registered as a presentation 140 * display. Applications can still project their content on the display but they 141 * will typically not do so automatically. This option is appropriate for 142 * more special-purpose displays. 143 * </p> 144 * 145 * @see android.app.Presentation 146 * @see #createVirtualDisplay 147 * @see #DISPLAY_CATEGORY_PRESENTATION 148 * @see Display#FLAG_PRESENTATION 149 */ 150 public static final int VIRTUAL_DISPLAY_FLAG_PRESENTATION = 1 << 1; 151 152 /** 153 * Virtual display flag: Create a secure display. 154 * 155 * <h3>Secure virtual displays</h3> 156 * <p> 157 * When this flag is set, the virtual display is considered secure as defined 158 * by the {@link Display#FLAG_SECURE} display flag. The caller promises to take 159 * reasonable measures, such as over-the-air encryption, to prevent the contents 160 * of the display from being intercepted or recorded on a persistent medium. 161 * </p><p> 162 * Creating a secure virtual display requires the 163 * {@link android.Manifest.permission#CAPTURE_SECURE_VIDEO_OUTPUT} permission. 164 * This permission is reserved for use by system components and is not available to 165 * third-party applications. 166 * </p> 167 * 168 * <h3>Non-secure virtual displays</h3> 169 * <p> 170 * When this flag is not set, the virtual display is considered unsecure. 171 * The content of secure windows will be blanked if shown on this display. 172 * </p> 173 * 174 * @see Display#FLAG_SECURE 175 * @see #createVirtualDisplay 176 */ 177 public static final int VIRTUAL_DISPLAY_FLAG_SECURE = 1 << 2; 178 179 /** 180 * Virtual display flag: Only show this display's own content; do not mirror 181 * the content of another display. 182 * 183 * <p> 184 * This flag is used in conjunction with {@link #VIRTUAL_DISPLAY_FLAG_PUBLIC}. 185 * Ordinarily public virtual displays will automatically mirror the content of the 186 * default display if they have no windows of their own. When this flag is 187 * specified, the virtual display will only ever show its own content and 188 * will be blanked instead if it has no windows. 189 * </p> 190 * 191 * <p> 192 * This flag is mutually exclusive with {@link #VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR}. If both 193 * flags are specified then the own-content only behavior will be applied. 194 * </p> 195 * 196 * <p> 197 * This behavior of this flag is implied whenever neither {@link #VIRTUAL_DISPLAY_FLAG_PUBLIC} 198 * nor {@link #VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR} have been set. This flag is only required to 199 * override the default behavior when creating a public display. 200 * </p> 201 * 202 * @see #createVirtualDisplay 203 */ 204 public static final int VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY = 1 << 3; 205 206 207 /** 208 * Virtual display flag: Allows content to be mirrored on private displays when no content is 209 * being shown. 210 * 211 * <p> 212 * This flag is mutually exclusive with {@link #VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY}. 213 * If both flags are specified then the own-content only behavior will be applied. 214 * </p> 215 * 216 * <p> 217 * The behavior of this flag is implied whenever {@link #VIRTUAL_DISPLAY_FLAG_PUBLIC} is set 218 * and {@link #VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY} has not been set. This flag is only 219 * required to override the default behavior when creating a private display. 220 * </p> 221 * 222 * <p> 223 * Creating an auto-mirroing virtual display requires the 224 * {@link android.Manifest.permission#CAPTURE_VIDEO_OUTPUT} 225 * or {@link android.Manifest.permission#CAPTURE_SECURE_VIDEO_OUTPUT} permission. 226 * These permissions are reserved for use by system components and are not available to 227 * third-party applications. 228 * 229 * Alternatively, an appropriate {@link MediaProjection} may be used to create an 230 * auto-mirroring virtual display. 231 * </p> 232 * 233 * @see #createVirtualDisplay 234 */ 235 public static final int VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR = 1 << 4; 236 237 /** @hide */ DisplayManager(Context context)238 public DisplayManager(Context context) { 239 mContext = context; 240 mGlobal = DisplayManagerGlobal.getInstance(); 241 } 242 243 /** 244 * Gets information about a logical display. 245 * 246 * The display metrics may be adjusted to provide compatibility 247 * for legacy applications. 248 * 249 * @param displayId The logical display id. 250 * @return The display object, or null if there is no valid display with the given id. 251 */ getDisplay(int displayId)252 public Display getDisplay(int displayId) { 253 synchronized (mLock) { 254 return getOrCreateDisplayLocked(displayId, false /*assumeValid*/); 255 } 256 } 257 258 /** 259 * Gets all currently valid logical displays. 260 * 261 * @return An array containing all displays. 262 */ getDisplays()263 public Display[] getDisplays() { 264 return getDisplays(null); 265 } 266 267 /** 268 * Gets all currently valid logical displays of the specified category. 269 * <p> 270 * When there are multiple displays in a category the returned displays are sorted 271 * of preference. For example, if the requested category is 272 * {@link #DISPLAY_CATEGORY_PRESENTATION} and there are multiple presentation displays 273 * then the displays are sorted so that the first display in the returned array 274 * is the most preferred presentation display. The application may simply 275 * use the first display or allow the user to choose. 276 * </p> 277 * 278 * @param category The requested display category or null to return all displays. 279 * @return An array containing all displays sorted by order of preference. 280 * 281 * @see #DISPLAY_CATEGORY_PRESENTATION 282 */ getDisplays(String category)283 public Display[] getDisplays(String category) { 284 final int[] displayIds = mGlobal.getDisplayIds(); 285 synchronized (mLock) { 286 try { 287 if (category == null) { 288 addAllDisplaysLocked(mTempDisplays, displayIds); 289 } else if (category.equals(DISPLAY_CATEGORY_PRESENTATION)) { 290 addPresentationDisplaysLocked(mTempDisplays, displayIds, Display.TYPE_WIFI); 291 addPresentationDisplaysLocked(mTempDisplays, displayIds, Display.TYPE_HDMI); 292 addPresentationDisplaysLocked(mTempDisplays, displayIds, Display.TYPE_OVERLAY); 293 addPresentationDisplaysLocked(mTempDisplays, displayIds, Display.TYPE_VIRTUAL); 294 } 295 return mTempDisplays.toArray(new Display[mTempDisplays.size()]); 296 } finally { 297 mTempDisplays.clear(); 298 } 299 } 300 } 301 addAllDisplaysLocked(ArrayList<Display> displays, int[] displayIds)302 private void addAllDisplaysLocked(ArrayList<Display> displays, int[] displayIds) { 303 for (int i = 0; i < displayIds.length; i++) { 304 Display display = getOrCreateDisplayLocked(displayIds[i], true /*assumeValid*/); 305 if (display != null) { 306 displays.add(display); 307 } 308 } 309 } 310 addPresentationDisplaysLocked( ArrayList<Display> displays, int[] displayIds, int matchType)311 private void addPresentationDisplaysLocked( 312 ArrayList<Display> displays, int[] displayIds, int matchType) { 313 for (int i = 0; i < displayIds.length; i++) { 314 Display display = getOrCreateDisplayLocked(displayIds[i], true /*assumeValid*/); 315 if (display != null 316 && (display.getFlags() & Display.FLAG_PRESENTATION) != 0 317 && display.getType() == matchType) { 318 displays.add(display); 319 } 320 } 321 } 322 getOrCreateDisplayLocked(int displayId, boolean assumeValid)323 private Display getOrCreateDisplayLocked(int displayId, boolean assumeValid) { 324 Display display = mDisplays.get(displayId); 325 if (display == null) { 326 display = mGlobal.getCompatibleDisplay(displayId, 327 mContext.getDisplayAdjustments(displayId)); 328 if (display != null) { 329 mDisplays.put(displayId, display); 330 } 331 } else if (!assumeValid && !display.isValid()) { 332 display = null; 333 } 334 return display; 335 } 336 337 /** 338 * Registers an display listener to receive notifications about when 339 * displays are added, removed or changed. 340 * 341 * @param listener The listener to register. 342 * @param handler The handler on which the listener should be invoked, or null 343 * if the listener should be invoked on the calling thread's looper. 344 * 345 * @see #unregisterDisplayListener 346 */ registerDisplayListener(DisplayListener listener, Handler handler)347 public void registerDisplayListener(DisplayListener listener, Handler handler) { 348 mGlobal.registerDisplayListener(listener, handler); 349 } 350 351 /** 352 * Unregisters a display listener. 353 * 354 * @param listener The listener to unregister. 355 * 356 * @see #registerDisplayListener 357 */ unregisterDisplayListener(DisplayListener listener)358 public void unregisterDisplayListener(DisplayListener listener) { 359 mGlobal.unregisterDisplayListener(listener); 360 } 361 362 /** 363 * Starts scanning for available Wifi displays. 364 * The results are sent as a {@link #ACTION_WIFI_DISPLAY_STATUS_CHANGED} broadcast. 365 * <p> 366 * Calls to this method nest and must be matched by an equal number of calls to 367 * {@link #stopWifiDisplayScan()}. 368 * </p><p> 369 * Requires {@link android.Manifest.permission#CONFIGURE_WIFI_DISPLAY}. 370 * </p> 371 * 372 * @hide 373 */ startWifiDisplayScan()374 public void startWifiDisplayScan() { 375 mGlobal.startWifiDisplayScan(); 376 } 377 378 /** 379 * Stops scanning for available Wifi displays. 380 * <p> 381 * Requires {@link android.Manifest.permission#CONFIGURE_WIFI_DISPLAY}. 382 * </p> 383 * 384 * @hide 385 */ stopWifiDisplayScan()386 public void stopWifiDisplayScan() { 387 mGlobal.stopWifiDisplayScan(); 388 } 389 390 /** 391 * Connects to a Wifi display. 392 * The results are sent as a {@link #ACTION_WIFI_DISPLAY_STATUS_CHANGED} broadcast. 393 * <p> 394 * Automatically remembers the display after a successful connection, if not 395 * already remembered. 396 * </p><p> 397 * Requires {@link android.Manifest.permission#CONFIGURE_WIFI_DISPLAY}. 398 * </p> 399 * 400 * @param deviceAddress The MAC address of the device to which we should connect. 401 * @hide 402 */ connectWifiDisplay(String deviceAddress)403 public void connectWifiDisplay(String deviceAddress) { 404 mGlobal.connectWifiDisplay(deviceAddress); 405 } 406 407 /** @hide */ pauseWifiDisplay()408 public void pauseWifiDisplay() { 409 mGlobal.pauseWifiDisplay(); 410 } 411 412 /** @hide */ resumeWifiDisplay()413 public void resumeWifiDisplay() { 414 mGlobal.resumeWifiDisplay(); 415 } 416 417 /** 418 * Disconnects from the current Wifi display. 419 * The results are sent as a {@link #ACTION_WIFI_DISPLAY_STATUS_CHANGED} broadcast. 420 * @hide 421 */ disconnectWifiDisplay()422 public void disconnectWifiDisplay() { 423 mGlobal.disconnectWifiDisplay(); 424 } 425 426 /** 427 * Renames a Wifi display. 428 * <p> 429 * The display must already be remembered for this call to succeed. In other words, 430 * we must already have successfully connected to the display at least once and then 431 * not forgotten it. 432 * </p><p> 433 * Requires {@link android.Manifest.permission#CONFIGURE_WIFI_DISPLAY}. 434 * </p> 435 * 436 * @param deviceAddress The MAC address of the device to rename. 437 * @param alias The alias name by which to remember the device, or null 438 * or empty if no alias should be used. 439 * @hide 440 */ renameWifiDisplay(String deviceAddress, String alias)441 public void renameWifiDisplay(String deviceAddress, String alias) { 442 mGlobal.renameWifiDisplay(deviceAddress, alias); 443 } 444 445 /** 446 * Forgets a previously remembered Wifi display. 447 * <p> 448 * Automatically disconnects from the display if currently connected to it. 449 * </p><p> 450 * Requires {@link android.Manifest.permission#CONFIGURE_WIFI_DISPLAY}. 451 * </p> 452 * 453 * @param deviceAddress The MAC address of the device to forget. 454 * @hide 455 */ forgetWifiDisplay(String deviceAddress)456 public void forgetWifiDisplay(String deviceAddress) { 457 mGlobal.forgetWifiDisplay(deviceAddress); 458 } 459 460 /** 461 * Gets the current Wifi display status. 462 * Watch for changes in the status by registering a broadcast receiver for 463 * {@link #ACTION_WIFI_DISPLAY_STATUS_CHANGED}. 464 * 465 * @return The current Wifi display status. 466 * @hide 467 */ getWifiDisplayStatus()468 public WifiDisplayStatus getWifiDisplayStatus() { 469 return mGlobal.getWifiDisplayStatus(); 470 } 471 472 /** 473 * Creates a virtual display. 474 * 475 * @see #createVirtualDisplay(String, int, int, int, Surface, int, VirtualDisplay.Callback) 476 */ createVirtualDisplay(@onNull String name, int width, int height, int densityDpi, @Nullable Surface surface, int flags)477 public VirtualDisplay createVirtualDisplay(@NonNull String name, 478 int width, int height, int densityDpi, @Nullable Surface surface, int flags) { 479 return createVirtualDisplay(name, width, height, densityDpi, surface, flags, null, null); 480 } 481 482 /** 483 * Creates a virtual display. 484 * <p> 485 * The content of a virtual display is rendered to a {@link Surface} provided 486 * by the application. 487 * </p><p> 488 * The virtual display should be {@link VirtualDisplay#release released} 489 * when no longer needed. Because a virtual display renders to a surface 490 * provided by the application, it will be released automatically when the 491 * process terminates and all remaining windows on it will be forcibly removed. 492 * </p><p> 493 * The behavior of the virtual display depends on the flags that are provided 494 * to this method. By default, virtual displays are created to be private, 495 * non-presentation and unsecure. Permissions may be required to use certain flags. 496 * </p><p> 497 * As of {@link android.os.Build.VERSION_CODES#KITKAT_WATCH}, the surface may 498 * be attached or detached dynamically using {@link VirtualDisplay#setSurface}. 499 * Previously, the surface had to be non-null when {@link #createVirtualDisplay} 500 * was called and could not be changed for the lifetime of the display. 501 * </p><p> 502 * Detaching the surface that backs a virtual display has a similar effect to 503 * turning off the screen. 504 * </p> 505 * 506 * @param name The name of the virtual display, must be non-empty. 507 * @param width The width of the virtual display in pixels, must be greater than 0. 508 * @param height The height of the virtual display in pixels, must be greater than 0. 509 * @param densityDpi The density of the virtual display in dpi, must be greater than 0. 510 * @param surface The surface to which the content of the virtual display should 511 * be rendered, or null if there is none initially. 512 * @param flags A combination of virtual display flags: 513 * {@link #VIRTUAL_DISPLAY_FLAG_PUBLIC}, {@link #VIRTUAL_DISPLAY_FLAG_PRESENTATION}, 514 * {@link #VIRTUAL_DISPLAY_FLAG_SECURE}, {@link #VIRTUAL_DISPLAY_FLAG_OWN_CONTENT_ONLY}, 515 * or {@link #VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR}. 516 * @param callback Callback to call when the state of the {@link VirtualDisplay} changes 517 * @param handler The handler on which the listener should be invoked, or null 518 * if the listener should be invoked on the calling thread's looper. 519 * @return The newly created virtual display, or null if the application could 520 * not create the virtual display. 521 * 522 * @throws SecurityException if the caller does not have permission to create 523 * a virtual display with the specified flags. 524 */ createVirtualDisplay(@onNull String name, int width, int height, int densityDpi, @Nullable Surface surface, int flags, @Nullable VirtualDisplay.Callback callback, @Nullable Handler handler)525 public VirtualDisplay createVirtualDisplay(@NonNull String name, 526 int width, int height, int densityDpi, @Nullable Surface surface, int flags, 527 @Nullable VirtualDisplay.Callback callback, @Nullable Handler handler) { 528 return createVirtualDisplay(null, 529 name, width, height, densityDpi, surface, flags, callback, handler); 530 } 531 532 /** @hide */ createVirtualDisplay(@ullable MediaProjection projection, @NonNull String name, int width, int height, int densityDpi, @Nullable Surface surface, int flags, @Nullable VirtualDisplay.Callback callback, @Nullable Handler handler)533 public VirtualDisplay createVirtualDisplay(@Nullable MediaProjection projection, 534 @NonNull String name, int width, int height, int densityDpi, @Nullable Surface surface, 535 int flags, @Nullable VirtualDisplay.Callback callback, @Nullable Handler handler) { 536 return mGlobal.createVirtualDisplay(mContext, projection, 537 name, width, height, densityDpi, surface, flags, callback, handler); 538 } 539 540 /** 541 * Listens for changes in available display devices. 542 */ 543 public interface DisplayListener { 544 /** 545 * Called whenever a logical display has been added to the system. 546 * Use {@link DisplayManager#getDisplay} to get more information about 547 * the display. 548 * 549 * @param displayId The id of the logical display that was added. 550 */ onDisplayAdded(int displayId)551 void onDisplayAdded(int displayId); 552 553 /** 554 * Called whenever a logical display has been removed from the system. 555 * 556 * @param displayId The id of the logical display that was removed. 557 */ onDisplayRemoved(int displayId)558 void onDisplayRemoved(int displayId); 559 560 /** 561 * Called whenever the properties of a logical display have changed. 562 * 563 * @param displayId The id of the logical display that changed. 564 */ onDisplayChanged(int displayId)565 void onDisplayChanged(int displayId); 566 } 567 } 568