| OLD | NEW |
| (Empty) |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "cc/resources/display_item_list.h" | |
| 6 | |
| 7 #include <vector> | |
| 8 | |
| 9 #include "cc/output/filter_operation.h" | |
| 10 #include "cc/output/filter_operations.h" | |
| 11 #include "cc/resources/clip_display_item.h" | |
| 12 #include "cc/resources/drawing_display_item.h" | |
| 13 #include "cc/resources/filter_display_item.h" | |
| 14 #include "cc/resources/transform_display_item.h" | |
| 15 #include "cc/test/skia_common.h" | |
| 16 #include "skia/ext/refptr.h" | |
| 17 #include "testing/gtest/include/gtest/gtest.h" | |
| 18 #include "third_party/skia/include/core/SkBitmap.h" | |
| 19 #include "third_party/skia/include/core/SkCanvas.h" | |
| 20 #include "third_party/skia/include/core/SkColor.h" | |
| 21 #include "third_party/skia/include/core/SkPictureRecorder.h" | |
| 22 #include "third_party/skia/include/effects/SkBitmapSource.h" | |
| 23 #include "third_party/skia/include/utils/SkPictureUtils.h" | |
| 24 #include "ui/gfx/geometry/rect_conversions.h" | |
| 25 #include "ui/gfx/skia_util.h" | |
| 26 | |
| 27 namespace cc { | |
| 28 | |
| 29 TEST(DisplayItemListTest, SingleDrawingItem) { | |
| 30 gfx::Rect layer_rect(100, 100); | |
| 31 SkPictureRecorder recorder; | |
| 32 skia::RefPtr<SkCanvas> canvas; | |
| 33 skia::RefPtr<SkPicture> picture; | |
| 34 SkPaint blue_paint; | |
| 35 blue_paint.setColor(SK_ColorBLUE); | |
| 36 SkPaint red_paint; | |
| 37 red_paint.setColor(SK_ColorRED); | |
| 38 unsigned char pixels[4 * 100 * 100] = {0}; | |
| 39 const bool use_cached_picture = true; | |
| 40 scoped_refptr<DisplayItemList> list = | |
| 41 DisplayItemList::Create(layer_rect, use_cached_picture); | |
| 42 | |
| 43 gfx::PointF offset(8.f, 9.f); | |
| 44 gfx::RectF recording_rect(offset, layer_rect.size()); | |
| 45 canvas = skia::SharePtr( | |
| 46 recorder.beginRecording(gfx::RectFToSkRect(recording_rect))); | |
| 47 canvas->translate(offset.x(), offset.y()); | |
| 48 canvas->drawRectCoords(0.f, 0.f, 60.f, 60.f, red_paint); | |
| 49 canvas->drawRectCoords(50.f, 50.f, 75.f, 75.f, blue_paint); | |
| 50 picture = skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 51 auto* item = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 52 item->SetNew(picture); | |
| 53 list->ProcessAppendedItems(); | |
| 54 list->CreateAndCacheSkPicture(); | |
| 55 DrawDisplayList(pixels, layer_rect, list); | |
| 56 | |
| 57 SkBitmap expected_bitmap; | |
| 58 unsigned char expected_pixels[4 * 100 * 100] = {0}; | |
| 59 SkImageInfo info = | |
| 60 SkImageInfo::MakeN32Premul(layer_rect.width(), layer_rect.height()); | |
| 61 expected_bitmap.installPixels(info, expected_pixels, info.minRowBytes()); | |
| 62 SkCanvas expected_canvas(expected_bitmap); | |
| 63 expected_canvas.clipRect(gfx::RectToSkRect(layer_rect)); | |
| 64 expected_canvas.drawRectCoords(0.f + offset.x(), 0.f + offset.y(), | |
| 65 60.f + offset.x(), 60.f + offset.y(), | |
| 66 red_paint); | |
| 67 expected_canvas.drawRectCoords(50.f + offset.x(), 50.f + offset.y(), | |
| 68 75.f + offset.x(), 75.f + offset.y(), | |
| 69 blue_paint); | |
| 70 | |
| 71 EXPECT_EQ(0, memcmp(pixels, expected_pixels, 4 * 100 * 100)); | |
| 72 } | |
| 73 | |
| 74 TEST(DisplayItemListTest, ClipItem) { | |
| 75 gfx::Rect layer_rect(100, 100); | |
| 76 SkPictureRecorder recorder; | |
| 77 skia::RefPtr<SkCanvas> canvas; | |
| 78 skia::RefPtr<SkPicture> picture; | |
| 79 SkPaint blue_paint; | |
| 80 blue_paint.setColor(SK_ColorBLUE); | |
| 81 SkPaint red_paint; | |
| 82 red_paint.setColor(SK_ColorRED); | |
| 83 unsigned char pixels[4 * 100 * 100] = {0}; | |
| 84 const bool use_cached_picture = true; | |
| 85 scoped_refptr<DisplayItemList> list = | |
| 86 DisplayItemList::Create(layer_rect, use_cached_picture); | |
| 87 | |
| 88 gfx::PointF first_offset(8.f, 9.f); | |
| 89 gfx::RectF first_recording_rect(first_offset, layer_rect.size()); | |
| 90 canvas = skia::SharePtr( | |
| 91 recorder.beginRecording(gfx::RectFToSkRect(first_recording_rect))); | |
| 92 canvas->translate(first_offset.x(), first_offset.y()); | |
| 93 canvas->drawRectCoords(0.f, 0.f, 60.f, 60.f, red_paint); | |
| 94 picture = skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 95 auto* item1 = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 96 item1->SetNew(picture.Pass()); | |
| 97 | |
| 98 gfx::Rect clip_rect(60, 60, 10, 10); | |
| 99 auto* item2 = list->CreateAndAppendItem<ClipDisplayItem>(); | |
| 100 item2->SetNew(clip_rect, std::vector<SkRRect>()); | |
| 101 | |
| 102 gfx::PointF second_offset(2.f, 3.f); | |
| 103 gfx::RectF second_recording_rect(second_offset, layer_rect.size()); | |
| 104 canvas = skia::SharePtr( | |
| 105 recorder.beginRecording(gfx::RectFToSkRect(second_recording_rect))); | |
| 106 canvas->translate(second_offset.x(), second_offset.y()); | |
| 107 canvas->drawRectCoords(50.f, 50.f, 75.f, 75.f, blue_paint); | |
| 108 picture = skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 109 auto* item3 = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 110 item3->SetNew(picture.Pass()); | |
| 111 | |
| 112 list->CreateAndAppendItem<EndClipDisplayItem>(); | |
| 113 list->ProcessAppendedItems(); | |
| 114 list->CreateAndCacheSkPicture(); | |
| 115 | |
| 116 DrawDisplayList(pixels, layer_rect, list); | |
| 117 | |
| 118 SkBitmap expected_bitmap; | |
| 119 unsigned char expected_pixels[4 * 100 * 100] = {0}; | |
| 120 SkImageInfo info = | |
| 121 SkImageInfo::MakeN32Premul(layer_rect.width(), layer_rect.height()); | |
| 122 expected_bitmap.installPixels(info, expected_pixels, info.minRowBytes()); | |
| 123 SkCanvas expected_canvas(expected_bitmap); | |
| 124 expected_canvas.clipRect(gfx::RectToSkRect(layer_rect)); | |
| 125 expected_canvas.drawRectCoords(0.f + first_offset.x(), 0.f + first_offset.y(), | |
| 126 60.f + first_offset.x(), | |
| 127 60.f + first_offset.y(), red_paint); | |
| 128 expected_canvas.clipRect(gfx::RectToSkRect(clip_rect)); | |
| 129 expected_canvas.drawRectCoords( | |
| 130 50.f + second_offset.x(), 50.f + second_offset.y(), | |
| 131 75.f + second_offset.x(), 75.f + second_offset.y(), blue_paint); | |
| 132 | |
| 133 EXPECT_EQ(0, memcmp(pixels, expected_pixels, 4 * 100 * 100)); | |
| 134 } | |
| 135 | |
| 136 TEST(DisplayItemListTest, TransformItem) { | |
| 137 gfx::Rect layer_rect(100, 100); | |
| 138 SkPictureRecorder recorder; | |
| 139 skia::RefPtr<SkCanvas> canvas; | |
| 140 skia::RefPtr<SkPicture> picture; | |
| 141 SkPaint blue_paint; | |
| 142 blue_paint.setColor(SK_ColorBLUE); | |
| 143 SkPaint red_paint; | |
| 144 red_paint.setColor(SK_ColorRED); | |
| 145 unsigned char pixels[4 * 100 * 100] = {0}; | |
| 146 const bool use_cached_picture = true; | |
| 147 scoped_refptr<DisplayItemList> list = | |
| 148 DisplayItemList::Create(layer_rect, use_cached_picture); | |
| 149 | |
| 150 gfx::PointF first_offset(8.f, 9.f); | |
| 151 gfx::RectF first_recording_rect(first_offset, layer_rect.size()); | |
| 152 canvas = skia::SharePtr( | |
| 153 recorder.beginRecording(gfx::RectFToSkRect(first_recording_rect))); | |
| 154 canvas->translate(first_offset.x(), first_offset.y()); | |
| 155 canvas->drawRectCoords(0.f, 0.f, 60.f, 60.f, red_paint); | |
| 156 picture = skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 157 auto* item1 = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 158 item1->SetNew(picture); | |
| 159 | |
| 160 gfx::Transform transform; | |
| 161 transform.Rotate(45.0); | |
| 162 auto* item2 = list->CreateAndAppendItem<TransformDisplayItem>(); | |
| 163 item2->SetNew(transform); | |
| 164 | |
| 165 gfx::PointF second_offset(2.f, 3.f); | |
| 166 gfx::RectF second_recording_rect(second_offset, layer_rect.size()); | |
| 167 canvas = skia::SharePtr( | |
| 168 recorder.beginRecording(gfx::RectFToSkRect(second_recording_rect))); | |
| 169 canvas->translate(second_offset.x(), second_offset.y()); | |
| 170 canvas->drawRectCoords(50.f, 50.f, 75.f, 75.f, blue_paint); | |
| 171 picture = skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 172 auto* item3 = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 173 item3->SetNew(picture); | |
| 174 | |
| 175 list->CreateAndAppendItem<EndTransformDisplayItem>(); | |
| 176 list->ProcessAppendedItems(); | |
| 177 list->CreateAndCacheSkPicture(); | |
| 178 | |
| 179 DrawDisplayList(pixels, layer_rect, list); | |
| 180 | |
| 181 SkBitmap expected_bitmap; | |
| 182 unsigned char expected_pixels[4 * 100 * 100] = {0}; | |
| 183 SkImageInfo info = | |
| 184 SkImageInfo::MakeN32Premul(layer_rect.width(), layer_rect.height()); | |
| 185 expected_bitmap.installPixels(info, expected_pixels, info.minRowBytes()); | |
| 186 SkCanvas expected_canvas(expected_bitmap); | |
| 187 expected_canvas.clipRect(gfx::RectToSkRect(layer_rect)); | |
| 188 expected_canvas.drawRectCoords(0.f + first_offset.x(), 0.f + first_offset.y(), | |
| 189 60.f + first_offset.x(), | |
| 190 60.f + first_offset.y(), red_paint); | |
| 191 expected_canvas.setMatrix(transform.matrix()); | |
| 192 expected_canvas.drawRectCoords( | |
| 193 50.f + second_offset.x(), 50.f + second_offset.y(), | |
| 194 75.f + second_offset.x(), 75.f + second_offset.y(), blue_paint); | |
| 195 | |
| 196 EXPECT_EQ(0, memcmp(pixels, expected_pixels, 4 * 100 * 100)); | |
| 197 } | |
| 198 | |
| 199 TEST(DisplayItemListTest, FilterItem) { | |
| 200 gfx::Rect layer_rect(100, 100); | |
| 201 FilterOperations filters; | |
| 202 unsigned char pixels[4 * 100 * 100] = {0}; | |
| 203 const bool use_cached_picture = true; | |
| 204 scoped_refptr<DisplayItemList> list = | |
| 205 DisplayItemList::Create(layer_rect, use_cached_picture); | |
| 206 | |
| 207 SkBitmap source_bitmap; | |
| 208 source_bitmap.allocN32Pixels(50, 50); | |
| 209 SkCanvas source_canvas(source_bitmap); | |
| 210 source_canvas.clear(SkColorSetRGB(128, 128, 128)); | |
| 211 | |
| 212 // For most SkImageFilters, the |dst| bounds computed by computeFastBounds are | |
| 213 // dependent on the provided |src| bounds. This means, for example, that | |
| 214 // translating |src| results in a corresponding translation of |dst|. But this | |
| 215 // is not the case for all SkImageFilters; for some of them (e.g. | |
| 216 // SkBitmapSource), the computation of |dst| in computeFastBounds doesn't | |
| 217 // involve |src| at all. Incorrectly assuming such a relationship (e.g. by | |
| 218 // translating |dst| after it is computed by computeFastBounds, rather than | |
| 219 // translating |src| before it provided to computedFastBounds) can cause | |
| 220 // incorrect clipping of filter output. To test for this, we include an | |
| 221 // SkBitmapSource filter in |filters|. Here, |src| is |filter_bounds|, defined | |
| 222 // below. | |
| 223 skia::RefPtr<SkImageFilter> image_filter = | |
| 224 skia::AdoptRef(SkBitmapSource::Create(source_bitmap)); | |
| 225 filters.Append(FilterOperation::CreateReferenceFilter(image_filter)); | |
| 226 filters.Append(FilterOperation::CreateBrightnessFilter(0.5f)); | |
| 227 gfx::RectF filter_bounds(10.f, 10.f, 50.f, 50.f); | |
| 228 auto* item = list->CreateAndAppendItem<FilterDisplayItem>(); | |
| 229 item->SetNew(filters, filter_bounds); | |
| 230 list->CreateAndAppendItem<EndFilterDisplayItem>(); | |
| 231 list->ProcessAppendedItems(); | |
| 232 list->CreateAndCacheSkPicture(); | |
| 233 | |
| 234 DrawDisplayList(pixels, layer_rect, list); | |
| 235 | |
| 236 SkBitmap expected_bitmap; | |
| 237 unsigned char expected_pixels[4 * 100 * 100] = {0}; | |
| 238 SkPaint paint; | |
| 239 paint.setColor(SkColorSetRGB(64, 64, 64)); | |
| 240 SkImageInfo info = | |
| 241 SkImageInfo::MakeN32Premul(layer_rect.width(), layer_rect.height()); | |
| 242 expected_bitmap.installPixels(info, expected_pixels, info.minRowBytes()); | |
| 243 SkCanvas expected_canvas(expected_bitmap); | |
| 244 expected_canvas.drawRect(RectFToSkRect(filter_bounds), paint); | |
| 245 | |
| 246 EXPECT_EQ(0, memcmp(pixels, expected_pixels, 4 * 100 * 100)); | |
| 247 } | |
| 248 | |
| 249 TEST(DisplayItemListTest, CompactingItems) { | |
| 250 gfx::Rect layer_rect(100, 100); | |
| 251 SkPictureRecorder recorder; | |
| 252 skia::RefPtr<SkCanvas> canvas; | |
| 253 skia::RefPtr<SkPicture> picture; | |
| 254 SkPaint blue_paint; | |
| 255 blue_paint.setColor(SK_ColorBLUE); | |
| 256 SkPaint red_paint; | |
| 257 red_paint.setColor(SK_ColorRED); | |
| 258 unsigned char pixels[4 * 100 * 100] = {0}; | |
| 259 | |
| 260 gfx::PointF offset(8.f, 9.f); | |
| 261 gfx::RectF recording_rect(offset, layer_rect.size()); | |
| 262 | |
| 263 bool use_cached_picture = false; | |
| 264 scoped_refptr<DisplayItemList> list_without_caching = | |
| 265 DisplayItemList::Create(layer_rect, use_cached_picture); | |
| 266 | |
| 267 canvas = skia::SharePtr( | |
| 268 recorder.beginRecording(gfx::RectFToSkRect(recording_rect))); | |
| 269 canvas->translate(offset.x(), offset.y()); | |
| 270 canvas->drawRectCoords(0.f, 0.f, 60.f, 60.f, red_paint); | |
| 271 canvas->drawRectCoords(50.f, 50.f, 75.f, 75.f, blue_paint); | |
| 272 picture = skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 273 auto* item1 = list_without_caching->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 274 item1->SetNew(picture); | |
| 275 list_without_caching->ProcessAppendedItems(); | |
| 276 DrawDisplayList(pixels, layer_rect, list_without_caching); | |
| 277 | |
| 278 unsigned char expected_pixels[4 * 100 * 100] = {0}; | |
| 279 use_cached_picture = true; | |
| 280 scoped_refptr<DisplayItemList> list_with_caching = | |
| 281 DisplayItemList::Create(layer_rect, use_cached_picture); | |
| 282 auto* item2 = list_with_caching->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 283 item2->SetNew(picture); | |
| 284 list_with_caching->ProcessAppendedItems(); | |
| 285 list_with_caching->CreateAndCacheSkPicture(); | |
| 286 DrawDisplayList(expected_pixels, layer_rect, list_with_caching); | |
| 287 | |
| 288 EXPECT_EQ(0, memcmp(pixels, expected_pixels, 4 * 100 * 100)); | |
| 289 } | |
| 290 | |
| 291 TEST(DisplayItemListTest, PictureMemoryUsage) { | |
| 292 scoped_refptr<DisplayItemList> list; | |
| 293 size_t memory_usage; | |
| 294 | |
| 295 // Make an SkPicture whose size is known. | |
| 296 gfx::Rect layer_rect(100, 100); | |
| 297 SkPictureRecorder recorder; | |
| 298 SkPaint blue_paint; | |
| 299 blue_paint.setColor(SK_ColorBLUE); | |
| 300 SkCanvas* canvas = recorder.beginRecording(gfx::RectFToSkRect(layer_rect)); | |
| 301 for (int i = 0; i < 100; i++) | |
| 302 canvas->drawPaint(blue_paint); | |
| 303 skia::RefPtr<SkPicture> picture = | |
| 304 skia::AdoptRef(recorder.endRecordingAsPicture()); | |
| 305 size_t picture_size = SkPictureUtils::ApproximateBytesUsed(picture.get()); | |
| 306 ASSERT_GE(picture_size, 100 * sizeof(SkPaint)); | |
| 307 ASSERT_LE(picture_size, 200 * sizeof(SkPaint)); | |
| 308 | |
| 309 // Using a cached picture, we should get about the right size. | |
| 310 list = DisplayItemList::Create(layer_rect, true); | |
| 311 auto* item = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 312 item->SetNew(picture); | |
| 313 list->ProcessAppendedItems(); | |
| 314 list->CreateAndCacheSkPicture(); | |
| 315 memory_usage = list->PictureMemoryUsage(); | |
| 316 EXPECT_GE(memory_usage, picture_size); | |
| 317 EXPECT_LE(memory_usage, 2 * picture_size); | |
| 318 | |
| 319 // Using no cached picture, we should still get the right size. | |
| 320 list = DisplayItemList::Create(layer_rect, false); | |
| 321 item = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 322 item->SetNew(picture); | |
| 323 list->ProcessAppendedItems(); | |
| 324 memory_usage = list->PictureMemoryUsage(); | |
| 325 EXPECT_GE(memory_usage, picture_size); | |
| 326 EXPECT_LE(memory_usage, 2 * picture_size); | |
| 327 | |
| 328 // To avoid double counting, we expect zero size to be computed if both the | |
| 329 // picture and items are retained (currently this only happens due to certain | |
| 330 // categories being traced). | |
| 331 list = new DisplayItemList(layer_rect, true, true); | |
| 332 item = list->CreateAndAppendItem<DrawingDisplayItem>(); | |
| 333 item->SetNew(picture); | |
| 334 list->ProcessAppendedItems(); | |
| 335 list->CreateAndCacheSkPicture(); | |
| 336 memory_usage = list->PictureMemoryUsage(); | |
| 337 EXPECT_EQ(static_cast<size_t>(0), memory_usage); | |
| 338 } | |
| 339 | |
| 340 } // namespace cc | |
| OLD | NEW |