1 /* 2 * Copyright (C) 2010 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.view; 18 19 import android.graphics.Matrix; 20 21 /** 22 * <p>A display list records a series of graphics related operations and can replay 23 * them later. Display lists are usually built by recording operations on a 24 * {@link HardwareCanvas}. Replaying the operations from a display list avoids 25 * executing application code on every frame, and is thus much more efficient.</p> 26 * 27 * <p>Display lists are used internally for all views by default, and are not 28 * typically used directly. One reason to consider using a display is a custom 29 * {@link View} implementation that needs to issue a large number of drawing commands. 30 * When the view invalidates, all the drawing commands must be reissued, even if 31 * large portions of the drawing command stream stay the same frame to frame, which 32 * can become a performance bottleneck. To solve this issue, a custom View might split 33 * its content into several display lists. A display list is updated only when its 34 * content, and only its content, needs to be updated.</p> 35 * 36 * <p>A text editor might for instance store each paragraph into its own display list. 37 * Thus when the user inserts or removes characters, only the display list of the 38 * affected paragraph needs to be recorded again.</p> 39 * 40 * <h3>Hardware acceleration</h3> 41 * <p>Display lists can only be replayed using a {@link HardwareCanvas}. They are not 42 * supported in software. Always make sure that the {@link android.graphics.Canvas} 43 * you are using to render a display list is hardware accelerated using 44 * {@link android.graphics.Canvas#isHardwareAccelerated()}.</p> 45 * 46 * <h3>Creating a display list</h3> 47 * <pre class="prettyprint"> 48 * HardwareRenderer renderer = myView.getHardwareRenderer(); 49 * if (renderer != null) { 50 * DisplayList displayList = renderer.createDisplayList(); 51 * HardwareCanvas canvas = displayList.start(width, height); 52 * try { 53 * // Draw onto the canvas 54 * // For instance: canvas.drawBitmap(...); 55 * } finally { 56 * displayList.end(); 57 * } 58 * } 59 * </pre> 60 * 61 * <h3>Rendering a display list on a View</h3> 62 * <pre class="prettyprint"> 63 * protected void onDraw(Canvas canvas) { 64 * if (canvas.isHardwareAccelerated()) { 65 * HardwareCanvas hardwareCanvas = (HardwareCanvas) canvas; 66 * hardwareCanvas.drawDisplayList(mDisplayList); 67 * } 68 * } 69 * </pre> 70 * 71 * <h3>Releasing resources</h3> 72 * <p>This step is not mandatory but recommended if you want to release resources 73 * held by a display list as soon as possible.</p> 74 * <pre class="prettyprint"> 75 * // Mark this display list invalid, it cannot be used for drawing anymore, 76 * // and release resources held by this display list 77 * displayList.clear(); 78 * </pre> 79 * 80 * <h3>Properties</h3> 81 * <p>In addition, a display list offers several properties, such as 82 * {@link #setScaleX(float)} or {@link #setLeft(int)}, that can be used to affect all 83 * the drawing commands recorded within. For instance, these properties can be used 84 * to move around a large number of images without re-issuing all the individual 85 * <code>drawBitmap()</code> calls.</p> 86 * 87 * <pre class="prettyprint"> 88 * private void createDisplayList() { 89 * HardwareRenderer renderer = getHardwareRenderer(); 90 * if (renderer != null) { 91 * mDisplayList = renderer.createDisplayList(); 92 * HardwareCanvas canvas = mDisplayList.start(width, height); 93 * try { 94 * for (Bitmap b : mBitmaps) { 95 * canvas.drawBitmap(b, 0.0f, 0.0f, null); 96 * canvas.translate(0.0f, b.getHeight()); 97 * } 98 * } finally { 99 * displayList.end(); 100 * } 101 * } 102 * } 103 * 104 * protected void onDraw(Canvas canvas) { 105 * if (canvas.isHardwareAccelerated()) { 106 * HardwareCanvas hardwareCanvas = (HardwareCanvas) canvas; 107 * hardwareCanvas.drawDisplayList(mDisplayList); 108 * } 109 * } 110 * 111 * private void moveContentBy(int x) { 112 * // This will move all the bitmaps recorded inside the display list 113 * // by x pixels to the right and redraw this view. All the commands 114 * // recorded in createDisplayList() won't be re-issued, only onDraw() 115 * // will be invoked and will execute very quickly 116 * mDisplayList.offsetLeftAndRight(x); 117 * invalidate(); 118 * } 119 * </pre> 120 * 121 * <h3>Threading</h3> 122 * <p>Display lists must be created on and manipulated from the UI thread only.</p> 123 * 124 * @hide 125 */ 126 public abstract class DisplayList { 127 private boolean mDirty; 128 129 /** 130 * Flag used when calling 131 * {@link HardwareCanvas#drawDisplayList(DisplayList, android.graphics.Rect, int)} 132 * When this flag is set, draw operations lying outside of the bounds of the 133 * display list will be culled early. It is recommeneded to always set this 134 * flag. 135 * 136 * @hide 137 */ 138 public static final int FLAG_CLIP_CHILDREN = 0x1; 139 140 // NOTE: The STATUS_* values *must* match the enum in DrawGlInfo.h 141 142 /** 143 * Indicates that the display list is done drawing. 144 * 145 * @see HardwareCanvas#drawDisplayList(DisplayList, android.graphics.Rect, int) 146 * 147 * @hide 148 */ 149 public static final int STATUS_DONE = 0x0; 150 151 /** 152 * Indicates that the display list needs another drawing pass. 153 * 154 * @see HardwareCanvas#drawDisplayList(DisplayList, android.graphics.Rect, int) 155 * 156 * @hide 157 */ 158 public static final int STATUS_DRAW = 0x1; 159 160 /** 161 * Indicates that the display list needs to re-execute its GL functors. 162 * 163 * @see HardwareCanvas#drawDisplayList(DisplayList, android.graphics.Rect, int) 164 * @see HardwareCanvas#callDrawGLFunction(int) 165 * 166 * @hide 167 */ 168 public static final int STATUS_INVOKE = 0x2; 169 170 /** 171 * Indicates that the display list performed GL drawing operations. 172 * 173 * @see HardwareCanvas#drawDisplayList(DisplayList, android.graphics.Rect, int) 174 * 175 * @hide 176 */ 177 public static final int STATUS_DREW = 0x4; 178 179 /** 180 * Starts recording the display list. All operations performed on the 181 * returned canvas are recorded and stored in this display list. 182 * 183 * Calling this method will mark the display list invalid until 184 * {@link #end()} is called. Only valid display lists can be replayed. 185 * 186 * @param width The width of the display list's viewport 187 * @param height The height of the display list's viewport 188 * 189 * @return A canvas to record drawing operations. 190 * 191 * @see #end() 192 * @see #isValid() 193 */ start(int width, int height)194 public abstract HardwareCanvas start(int width, int height); 195 196 /** 197 * Ends the recording for this display list. A display list cannot be 198 * replayed if recording is not finished. Calling this method marks 199 * the display list valid and {@link #isValid()} will return true. 200 * 201 * @see #start(int, int) 202 * @see #isValid() 203 */ end()204 public abstract void end(); 205 206 /** 207 * Clears resources held onto by this display list. After calling this method 208 * {@link #isValid()} will return false. 209 * 210 * @see #isValid() 211 * @see #reset() 212 */ clear()213 public abstract void clear(); 214 215 216 /** 217 * Reset native resources. This is called when cleaning up the state of display lists 218 * during destruction of hardware resources, to ensure that we do not hold onto 219 * obsolete resources after related resources are gone. 220 * 221 * @see #clear() 222 * 223 * @hide 224 */ reset()225 public abstract void reset(); 226 227 /** 228 * Sets the dirty flag. When a display list is dirty, {@link #clear()} should 229 * be invoked whenever possible. 230 * 231 * @see #isDirty() 232 * @see #clear() 233 * 234 * @hide 235 */ markDirty()236 public void markDirty() { 237 mDirty = true; 238 } 239 240 /** 241 * Removes the dirty flag. This method can be used to cancel a cleanup 242 * previously scheduled by setting the dirty flag. 243 * 244 * @see #isDirty() 245 * @see #clear() 246 * 247 * @hide 248 */ clearDirty()249 protected void clearDirty() { 250 mDirty = false; 251 } 252 253 /** 254 * Indicates whether the display list is dirty. 255 * 256 * @see #markDirty() 257 * @see #clear() 258 * 259 * @hide 260 */ isDirty()261 public boolean isDirty() { 262 return mDirty; 263 } 264 265 /** 266 * Returns whether the display list is currently usable. If this returns false, 267 * the display list should be re-recorded prior to replaying it. 268 * 269 * @return boolean true if the display list is able to be replayed, false otherwise. 270 */ isValid()271 public abstract boolean isValid(); 272 273 /** 274 * Return the amount of memory used by this display list. 275 * 276 * @return The size of this display list in bytes 277 * 278 * @hide 279 */ getSize()280 public abstract int getSize(); 281 282 /////////////////////////////////////////////////////////////////////////// 283 // DisplayList Property Setters 284 /////////////////////////////////////////////////////////////////////////// 285 286 /** 287 * Set the caching property on the display list, which indicates whether the display list 288 * holds a layer. Layer display lists should avoid creating an alpha layer, since alpha is 289 * handled in the drawLayer operation directly (and more efficiently). 290 * 291 * @param caching true if the display list represents a hardware layer, false otherwise. 292 * 293 * @hide 294 */ setCaching(boolean caching)295 public abstract void setCaching(boolean caching); 296 297 /** 298 * Set whether the display list should clip itself to its bounds. This property is controlled by 299 * the view's parent. 300 * 301 * @param clipToBounds true if the display list should clip to its bounds 302 */ setClipToBounds(boolean clipToBounds)303 public abstract void setClipToBounds(boolean clipToBounds); 304 305 /** 306 * Set the static matrix on the display list. The specified matrix is combined with other 307 * transforms (such as {@link #setScaleX(float)}, {@link #setRotation(float)}, etc.) 308 * 309 * @param matrix A transform matrix to apply to this display list 310 * 311 * @see #getMatrix(android.graphics.Matrix) 312 * @see #getMatrix() 313 */ setMatrix(Matrix matrix)314 public abstract void setMatrix(Matrix matrix); 315 316 /** 317 * Returns the static matrix set on this display list. 318 * 319 * @return A new {@link Matrix} instance populated with this display list's static 320 * matrix 321 * 322 * @see #getMatrix(android.graphics.Matrix) 323 * @see #setMatrix(android.graphics.Matrix) 324 */ getMatrix()325 public Matrix getMatrix() { 326 return getMatrix(new Matrix()); 327 } 328 329 /** 330 * Copies this display list's static matrix into the specified matrix. 331 * 332 * @param matrix The {@link Matrix} instance in which to copy this display 333 * list's static matrix. Cannot be null 334 * 335 * @return The <code>matrix</code> parameter, for convenience 336 * 337 * @see #getMatrix() 338 * @see #setMatrix(android.graphics.Matrix) 339 */ getMatrix(Matrix matrix)340 public abstract Matrix getMatrix(Matrix matrix); 341 342 /** 343 * Set the Animation matrix on the display list. This matrix exists if an Animation is 344 * currently playing on a View, and is set on the display list during at draw() time. When 345 * the Animation finishes, the matrix should be cleared by sending <code>null</code> 346 * for the matrix parameter. 347 * 348 * @param matrix The matrix, null indicates that the matrix should be cleared. 349 * 350 * @hide 351 */ setAnimationMatrix(Matrix matrix)352 public abstract void setAnimationMatrix(Matrix matrix); 353 354 /** 355 * Sets the translucency level for the display list. 356 * 357 * @param alpha The translucency of the display list, must be a value between 0.0f and 1.0f 358 * 359 * @see View#setAlpha(float) 360 * @see #getAlpha() 361 */ setAlpha(float alpha)362 public abstract void setAlpha(float alpha); 363 364 /** 365 * Returns the translucency level of this display list. 366 * 367 * @return A value between 0.0f and 1.0f 368 * 369 * @see #setAlpha(float) 370 */ getAlpha()371 public abstract float getAlpha(); 372 373 /** 374 * Sets whether the display list renders content which overlaps. Non-overlapping rendering 375 * can use a fast path for alpha that avoids rendering to an offscreen buffer. By default 376 * display lists consider they do not have overlapping content. 377 * 378 * @param hasOverlappingRendering False if the content is guaranteed to be non-overlapping, 379 * true otherwise. 380 * 381 * @see android.view.View#hasOverlappingRendering() 382 * @see #hasOverlappingRendering() 383 */ setHasOverlappingRendering(boolean hasOverlappingRendering)384 public abstract void setHasOverlappingRendering(boolean hasOverlappingRendering); 385 386 /** 387 * Indicates whether the content of this display list overlaps. 388 * 389 * @return True if this display list renders content which overlaps, false otherwise. 390 * 391 * @see #setHasOverlappingRendering(boolean) 392 */ hasOverlappingRendering()393 public abstract boolean hasOverlappingRendering(); 394 395 /** 396 * Sets the translation value for the display list on the X axis 397 * 398 * @param translationX The X axis translation value of the display list, in pixels 399 * 400 * @see View#setTranslationX(float) 401 * @see #getTranslationX() 402 */ setTranslationX(float translationX)403 public abstract void setTranslationX(float translationX); 404 405 /** 406 * Returns the translation value for this display list on the X axis, in pixels. 407 * 408 * @see #setTranslationX(float) 409 */ getTranslationX()410 public abstract float getTranslationX(); 411 412 /** 413 * Sets the translation value for the display list on the Y axis 414 * 415 * @param translationY The Y axis translation value of the display list, in pixels 416 * 417 * @see View#setTranslationY(float) 418 * @see #getTranslationY() 419 */ setTranslationY(float translationY)420 public abstract void setTranslationY(float translationY); 421 422 /** 423 * Returns the translation value for this display list on the Y axis, in pixels. 424 * 425 * @see #setTranslationY(float) 426 */ getTranslationY()427 public abstract float getTranslationY(); 428 429 /** 430 * Sets the rotation value for the display list around the Z axis 431 * 432 * @param rotation The rotation value of the display list, in degrees 433 * 434 * @see View#setRotation(float) 435 * @see #getRotation() 436 */ setRotation(float rotation)437 public abstract void setRotation(float rotation); 438 439 /** 440 * Returns the rotation value for this display list around the Z axis, in degrees. 441 * 442 * @see #setRotation(float) 443 */ getRotation()444 public abstract float getRotation(); 445 446 /** 447 * Sets the rotation value for the display list around the X axis 448 * 449 * @param rotationX The rotation value of the display list, in degrees 450 * 451 * @see View#setRotationX(float) 452 * @see #getRotationX() 453 */ setRotationX(float rotationX)454 public abstract void setRotationX(float rotationX); 455 456 /** 457 * Returns the rotation value for this display list around the X axis, in degrees. 458 * 459 * @see #setRotationX(float) 460 */ getRotationX()461 public abstract float getRotationX(); 462 463 /** 464 * Sets the rotation value for the display list around the Y axis 465 * 466 * @param rotationY The rotation value of the display list, in degrees 467 * 468 * @see View#setRotationY(float) 469 * @see #getRotationY() 470 */ setRotationY(float rotationY)471 public abstract void setRotationY(float rotationY); 472 473 /** 474 * Returns the rotation value for this display list around the Y axis, in degrees. 475 * 476 * @see #setRotationY(float) 477 */ getRotationY()478 public abstract float getRotationY(); 479 480 /** 481 * Sets the scale value for the display list on the X axis 482 * 483 * @param scaleX The scale value of the display list 484 * 485 * @see View#setScaleX(float) 486 * @see #getScaleX() 487 */ setScaleX(float scaleX)488 public abstract void setScaleX(float scaleX); 489 490 /** 491 * Returns the scale value for this display list on the X axis. 492 * 493 * @see #setScaleX(float) 494 */ getScaleX()495 public abstract float getScaleX(); 496 497 /** 498 * Sets the scale value for the display list on the Y axis 499 * 500 * @param scaleY The scale value of the display list 501 * 502 * @see View#setScaleY(float) 503 * @see #getScaleY() 504 */ setScaleY(float scaleY)505 public abstract void setScaleY(float scaleY); 506 507 /** 508 * Returns the scale value for this display list on the Y axis. 509 * 510 * @see #setScaleY(float) 511 */ getScaleY()512 public abstract float getScaleY(); 513 514 /** 515 * Sets all of the transform-related values of the display list 516 * 517 * @param alpha The alpha value of the display list 518 * @param translationX The translationX value of the display list 519 * @param translationY The translationY value of the display list 520 * @param rotation The rotation value of the display list 521 * @param rotationX The rotationX value of the display list 522 * @param rotationY The rotationY value of the display list 523 * @param scaleX The scaleX value of the display list 524 * @param scaleY The scaleY value of the display list 525 * 526 * @hide 527 */ setTransformationInfo(float alpha, float translationX, float translationY, float rotation, float rotationX, float rotationY, float scaleX, float scaleY)528 public abstract void setTransformationInfo(float alpha, float translationX, float translationY, 529 float rotation, float rotationX, float rotationY, float scaleX, float scaleY); 530 531 /** 532 * Sets the pivot value for the display list on the X axis 533 * 534 * @param pivotX The pivot value of the display list on the X axis, in pixels 535 * 536 * @see View#setPivotX(float) 537 * @see #getPivotX() 538 */ setPivotX(float pivotX)539 public abstract void setPivotX(float pivotX); 540 541 /** 542 * Returns the pivot value for this display list on the X axis, in pixels. 543 * 544 * @see #setPivotX(float) 545 */ getPivotX()546 public abstract float getPivotX(); 547 548 /** 549 * Sets the pivot value for the display list on the Y axis 550 * 551 * @param pivotY The pivot value of the display list on the Y axis, in pixels 552 * 553 * @see View#setPivotY(float) 554 * @see #getPivotY() 555 */ setPivotY(float pivotY)556 public abstract void setPivotY(float pivotY); 557 558 /** 559 * Returns the pivot value for this display list on the Y axis, in pixels. 560 * 561 * @see #setPivotY(float) 562 */ getPivotY()563 public abstract float getPivotY(); 564 565 /** 566 * Sets the camera distance for the display list. Refer to 567 * {@link View#setCameraDistance(float)} for more information on how to 568 * use this property. 569 * 570 * @param distance The distance in Z of the camera of the display list 571 * 572 * @see View#setCameraDistance(float) 573 * @see #getCameraDistance() 574 */ setCameraDistance(float distance)575 public abstract void setCameraDistance(float distance); 576 577 /** 578 * Returns the distance in Z of the camera of the display list. 579 * 580 * @see #setCameraDistance(float) 581 */ getCameraDistance()582 public abstract float getCameraDistance(); 583 584 /** 585 * Sets the left position for the display list. 586 * 587 * @param left The left position, in pixels, of the display list 588 * 589 * @see View#setLeft(int) 590 * @see #getLeft() 591 */ setLeft(int left)592 public abstract void setLeft(int left); 593 594 /** 595 * Returns the left position for the display list in pixels. 596 * 597 * @see #setLeft(int) 598 */ getLeft()599 public abstract float getLeft(); 600 601 /** 602 * Sets the top position for the display list. 603 * 604 * @param top The top position, in pixels, of the display list 605 * 606 * @see View#setTop(int) 607 * @see #getTop() 608 */ setTop(int top)609 public abstract void setTop(int top); 610 611 /** 612 * Returns the top position for the display list in pixels. 613 * 614 * @see #setTop(int) 615 */ getTop()616 public abstract float getTop(); 617 618 /** 619 * Sets the right position for the display list. 620 * 621 * @param right The right position, in pixels, of the display list 622 * 623 * @see View#setRight(int) 624 * @see #getRight() 625 */ setRight(int right)626 public abstract void setRight(int right); 627 628 /** 629 * Returns the right position for the display list in pixels. 630 * 631 * @see #setRight(int) 632 */ getRight()633 public abstract float getRight(); 634 635 /** 636 * Sets the bottom position for the display list. 637 * 638 * @param bottom The bottom position, in pixels, of the display list 639 * 640 * @see View#setBottom(int) 641 * @see #getBottom() 642 */ setBottom(int bottom)643 public abstract void setBottom(int bottom); 644 645 /** 646 * Returns the bottom position for the display list in pixels. 647 * 648 * @see #setBottom(int) 649 */ getBottom()650 public abstract float getBottom(); 651 652 /** 653 * Sets the left and top positions for the display list 654 * 655 * @param left The left position of the display list, in pixels 656 * @param top The top position of the display list, in pixels 657 * @param right The right position of the display list, in pixels 658 * @param bottom The bottom position of the display list, in pixels 659 * 660 * @see View#setLeft(int) 661 * @see View#setTop(int) 662 * @see View#setRight(int) 663 * @see View#setBottom(int) 664 */ setLeftTopRightBottom(int left, int top, int right, int bottom)665 public abstract void setLeftTopRightBottom(int left, int top, int right, int bottom); 666 667 /** 668 * Offsets the left and right positions for the display list 669 * 670 * @param offset The amount that the left and right positions of the display 671 * list are offset, in pixels 672 * 673 * @see View#offsetLeftAndRight(int) 674 */ offsetLeftAndRight(float offset)675 public abstract void offsetLeftAndRight(float offset); 676 677 /** 678 * Offsets the top and bottom values for the display list 679 * 680 * @param offset The amount that the top and bottom positions of the display 681 * list are offset, in pixels 682 * 683 * @see View#offsetTopAndBottom(int) 684 */ offsetTopAndBottom(float offset)685 public abstract void offsetTopAndBottom(float offset); 686 } 687