1 /* 2 * Copyright (C) 2016 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 #include "compile/Image.h" 18 19 #include "test/Test.h" 20 21 namespace aapt { 22 23 // Pixels are in RGBA_8888 packing. 24 25 #define RED "\xff\x00\x00\xff" 26 #define BLUE "\x00\x00\xff\xff" 27 #define GREEN "\xff\x00\x00\xff" 28 #define GR_70 "\xff\x00\x00\xb3" 29 #define GR_50 "\xff\x00\x00\x80" 30 #define GR_20 "\xff\x00\x00\x33" 31 #define BLACK "\x00\x00\x00\xff" 32 #define WHITE "\xff\xff\xff\xff" 33 #define TRANS "\x00\x00\x00\x00" 34 35 static uint8_t* k2x2[] = { 36 (uint8_t*)WHITE WHITE, (uint8_t*)WHITE WHITE, 37 }; 38 39 static uint8_t* kMixedNeutralColor3x3[] = { 40 (uint8_t*)WHITE BLACK TRANS, (uint8_t*)TRANS RED TRANS, 41 (uint8_t*)WHITE WHITE WHITE, 42 }; 43 44 static uint8_t* kTransparentNeutralColor3x3[] = { 45 (uint8_t*)TRANS BLACK TRANS, (uint8_t*)BLACK RED BLACK, 46 (uint8_t*)TRANS BLACK TRANS, 47 }; 48 49 static uint8_t* kSingleStretch7x6[] = { 50 (uint8_t*)WHITE WHITE BLACK BLACK BLACK WHITE WHITE, 51 (uint8_t*)WHITE RED RED RED RED RED WHITE, 52 (uint8_t*)BLACK RED RED RED RED RED WHITE, 53 (uint8_t*)BLACK RED RED RED RED RED WHITE, 54 (uint8_t*)WHITE RED RED RED RED RED WHITE, 55 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE WHITE, 56 }; 57 58 static uint8_t* kMultipleStretch10x7[] = { 59 (uint8_t*)WHITE WHITE BLACK WHITE BLACK BLACK WHITE BLACK WHITE WHITE, 60 (uint8_t*)BLACK RED BLUE RED BLUE BLUE RED BLUE RED WHITE, 61 (uint8_t*)BLACK RED BLUE RED BLUE BLUE RED BLUE RED WHITE, 62 (uint8_t*)WHITE RED BLUE RED BLUE BLUE RED BLUE RED WHITE, 63 (uint8_t*)BLACK RED BLUE RED BLUE BLUE RED BLUE RED WHITE, 64 (uint8_t*)BLACK RED BLUE RED BLUE BLUE RED BLUE RED WHITE, 65 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE, 66 }; 67 68 static uint8_t* kPadding6x5[] = { 69 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE, 70 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE, 71 (uint8_t*)WHITE WHITE WHITE WHITE WHITE BLACK, 72 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE, 73 (uint8_t*)WHITE WHITE BLACK BLACK WHITE WHITE, 74 }; 75 76 static uint8_t* kLayoutBoundsWrongEdge3x3[] = { 77 (uint8_t*)WHITE RED WHITE, (uint8_t*)RED WHITE WHITE, 78 (uint8_t*)WHITE WHITE WHITE, 79 }; 80 81 static uint8_t* kLayoutBoundsNotEdgeAligned5x5[] = { 82 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 83 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 84 (uint8_t*)WHITE WHITE WHITE WHITE RED, 85 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 86 (uint8_t*)WHITE WHITE RED WHITE WHITE, 87 }; 88 89 static uint8_t* kLayoutBounds5x5[] = { 90 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 91 (uint8_t*)WHITE WHITE WHITE WHITE RED, 92 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 93 (uint8_t*)WHITE WHITE WHITE WHITE RED, 94 (uint8_t*)WHITE RED WHITE RED WHITE, 95 }; 96 97 static uint8_t* kAsymmetricLayoutBounds5x5[] = { 98 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 99 (uint8_t*)WHITE WHITE WHITE WHITE RED, 100 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 101 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 102 (uint8_t*)WHITE RED WHITE WHITE WHITE, 103 }; 104 105 static uint8_t* kPaddingAndLayoutBounds5x5[] = { 106 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 107 (uint8_t*)WHITE WHITE WHITE WHITE RED, 108 (uint8_t*)WHITE WHITE WHITE WHITE BLACK, 109 (uint8_t*)WHITE WHITE WHITE WHITE RED, 110 (uint8_t*)WHITE RED BLACK RED WHITE, 111 }; 112 113 static uint8_t* kColorfulImage5x5[] = { 114 (uint8_t*)WHITE BLACK WHITE BLACK WHITE, 115 (uint8_t*)BLACK RED BLUE GREEN WHITE, 116 (uint8_t*)BLACK RED GREEN GREEN WHITE, 117 (uint8_t*)WHITE TRANS BLUE GREEN WHITE, 118 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 119 }; 120 121 static uint8_t* kOutlineOpaque10x10[] = { 122 (uint8_t*)WHITE BLACK BLACK BLACK BLACK BLACK BLACK BLACK BLACK WHITE, 123 (uint8_t*)WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 124 (uint8_t*)WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 125 (uint8_t*)WHITE TRANS TRANS GREEN GREEN GREEN GREEN TRANS TRANS WHITE, 126 (uint8_t*)WHITE TRANS TRANS GREEN GREEN GREEN GREEN TRANS TRANS WHITE, 127 (uint8_t*)WHITE TRANS TRANS GREEN GREEN GREEN GREEN TRANS TRANS WHITE, 128 (uint8_t*)WHITE TRANS TRANS GREEN GREEN GREEN GREEN TRANS TRANS WHITE, 129 (uint8_t*)WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 130 (uint8_t*)WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 131 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE, 132 }; 133 134 static uint8_t* kOutlineTranslucent10x10[] = { 135 (uint8_t*)WHITE BLACK BLACK BLACK BLACK BLACK BLACK BLACK BLACK WHITE, 136 (uint8_t*)WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 137 (uint8_t*)WHITE TRANS TRANS GR_20 GR_20 GR_20 GR_20 TRANS TRANS WHITE, 138 (uint8_t*)WHITE TRANS TRANS GR_50 GR_50 GR_50 GR_50 TRANS TRANS WHITE, 139 (uint8_t*)WHITE TRANS GR_20 GR_50 GR_70 GR_70 GR_50 GR_20 TRANS WHITE, 140 (uint8_t*)WHITE TRANS GR_20 GR_50 GR_70 GR_70 GR_50 GR_20 TRANS WHITE, 141 (uint8_t*)WHITE TRANS TRANS GR_50 GR_50 GR_50 GR_50 TRANS TRANS WHITE, 142 (uint8_t*)WHITE TRANS TRANS GR_20 GR_20 GR_20 GR_20 TRANS TRANS WHITE, 143 (uint8_t*)WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 144 (uint8_t*)WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE, 145 }; 146 147 static uint8_t* kOutlineOffsetTranslucent12x10[] = { 148 (uint8_t*) 149 WHITE WHITE WHITE BLACK BLACK BLACK BLACK BLACK BLACK BLACK BLACK WHITE, 150 (uint8_t*) 151 WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 152 (uint8_t*) 153 WHITE TRANS TRANS TRANS TRANS GR_20 GR_20 GR_20 GR_20 TRANS TRANS WHITE, 154 (uint8_t*) 155 WHITE TRANS TRANS TRANS TRANS GR_50 GR_50 GR_50 GR_50 TRANS TRANS WHITE, 156 (uint8_t*) 157 WHITE TRANS TRANS TRANS GR_20 GR_50 GR_70 GR_70 GR_50 GR_20 TRANS WHITE, 158 (uint8_t*) 159 WHITE TRANS TRANS TRANS GR_20 GR_50 GR_70 GR_70 GR_50 GR_20 TRANS WHITE, 160 (uint8_t*) 161 WHITE TRANS TRANS TRANS TRANS GR_50 GR_50 GR_50 GR_50 TRANS TRANS WHITE, 162 (uint8_t*) 163 WHITE TRANS TRANS TRANS TRANS GR_20 GR_20 GR_20 GR_20 TRANS TRANS WHITE, 164 (uint8_t*) 165 WHITE TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS TRANS WHITE, 166 (uint8_t*) 167 WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE WHITE, 168 }; 169 170 static uint8_t* kOutlineRadius5x5[] = { 171 (uint8_t*)WHITE BLACK BLACK BLACK WHITE, 172 (uint8_t*)BLACK TRANS GREEN TRANS WHITE, 173 (uint8_t*)BLACK GREEN GREEN GREEN WHITE, 174 (uint8_t*)BLACK TRANS GREEN TRANS WHITE, 175 (uint8_t*)WHITE WHITE WHITE WHITE WHITE, 176 }; 177 178 static uint8_t* kStretchAndPadding5x5[] = { 179 (uint8_t*)WHITE WHITE BLACK WHITE WHITE, (uint8_t*)WHITE RED RED RED WHITE, 180 (uint8_t*)BLACK RED RED RED BLACK, (uint8_t*)WHITE RED RED RED WHITE, 181 (uint8_t*)WHITE WHITE BLACK WHITE WHITE, 182 }; 183 184 TEST(NinePatchTest, Minimum3x3) { 185 std::string err; 186 EXPECT_EQ(nullptr, NinePatch::Create(k2x2, 2, 2, &err)); 187 EXPECT_FALSE(err.empty()); 188 } 189 190 TEST(NinePatchTest, MixedNeutralColors) { 191 std::string err; 192 EXPECT_EQ(nullptr, NinePatch::Create(kMixedNeutralColor3x3, 3, 3, &err)); 193 EXPECT_FALSE(err.empty()); 194 } 195 196 TEST(NinePatchTest, TransparentNeutralColor) { 197 std::string err; 198 EXPECT_NE(nullptr, 199 NinePatch::Create(kTransparentNeutralColor3x3, 3, 3, &err)); 200 } 201 202 TEST(NinePatchTest, SingleStretchRegion) { 203 std::string err; 204 std::unique_ptr<NinePatch> nine_patch = 205 NinePatch::Create(kSingleStretch7x6, 7, 6, &err); 206 ASSERT_NE(nullptr, nine_patch); 207 208 ASSERT_EQ(1u, nine_patch->horizontal_stretch_regions.size()); 209 ASSERT_EQ(1u, nine_patch->vertical_stretch_regions.size()); 210 211 EXPECT_EQ(Range(1, 4), nine_patch->horizontal_stretch_regions.front()); 212 EXPECT_EQ(Range(1, 3), nine_patch->vertical_stretch_regions.front()); 213 } 214 215 TEST(NinePatchTest, MultipleStretchRegions) { 216 std::string err; 217 std::unique_ptr<NinePatch> nine_patch = 218 NinePatch::Create(kMultipleStretch10x7, 10, 7, &err); 219 ASSERT_NE(nullptr, nine_patch); 220 221 ASSERT_EQ(3u, nine_patch->horizontal_stretch_regions.size()); 222 ASSERT_EQ(2u, nine_patch->vertical_stretch_regions.size()); 223 224 EXPECT_EQ(Range(1, 2), nine_patch->horizontal_stretch_regions[0]); 225 EXPECT_EQ(Range(3, 5), nine_patch->horizontal_stretch_regions[1]); 226 EXPECT_EQ(Range(6, 7), nine_patch->horizontal_stretch_regions[2]); 227 228 EXPECT_EQ(Range(0, 2), nine_patch->vertical_stretch_regions[0]); 229 EXPECT_EQ(Range(3, 5), nine_patch->vertical_stretch_regions[1]); 230 } 231 232 TEST(NinePatchTest, InferPaddingFromStretchRegions) { 233 std::string err; 234 std::unique_ptr<NinePatch> nine_patch = 235 NinePatch::Create(kMultipleStretch10x7, 10, 7, &err); 236 ASSERT_NE(nullptr, nine_patch); 237 EXPECT_EQ(Bounds(1, 0, 1, 0), nine_patch->padding); 238 } 239 240 TEST(NinePatchTest, Padding) { 241 std::string err; 242 std::unique_ptr<NinePatch> nine_patch = 243 NinePatch::Create(kPadding6x5, 6, 5, &err); 244 ASSERT_NE(nullptr, nine_patch); 245 EXPECT_EQ(Bounds(1, 1, 1, 1), nine_patch->padding); 246 } 247 248 TEST(NinePatchTest, LayoutBoundsAreOnWrongEdge) { 249 std::string err; 250 EXPECT_EQ(nullptr, NinePatch::Create(kLayoutBoundsWrongEdge3x3, 3, 3, &err)); 251 EXPECT_FALSE(err.empty()); 252 } 253 254 TEST(NinePatchTest, LayoutBoundsMustTouchEdges) { 255 std::string err; 256 EXPECT_EQ(nullptr, 257 NinePatch::Create(kLayoutBoundsNotEdgeAligned5x5, 5, 5, &err)); 258 EXPECT_FALSE(err.empty()); 259 } 260 261 TEST(NinePatchTest, LayoutBounds) { 262 std::string err; 263 std::unique_ptr<NinePatch> nine_patch = 264 NinePatch::Create(kLayoutBounds5x5, 5, 5, &err); 265 ASSERT_NE(nullptr, nine_patch); 266 EXPECT_EQ(Bounds(1, 1, 1, 1), nine_patch->layout_bounds); 267 268 nine_patch = NinePatch::Create(kAsymmetricLayoutBounds5x5, 5, 5, &err); 269 ASSERT_NE(nullptr, nine_patch); 270 EXPECT_EQ(Bounds(1, 1, 0, 0), nine_patch->layout_bounds); 271 } 272 273 TEST(NinePatchTest, PaddingAndLayoutBounds) { 274 std::string err; 275 std::unique_ptr<NinePatch> nine_patch = 276 NinePatch::Create(kPaddingAndLayoutBounds5x5, 5, 5, &err); 277 ASSERT_NE(nullptr, nine_patch); 278 EXPECT_EQ(Bounds(1, 1, 1, 1), nine_patch->padding); 279 EXPECT_EQ(Bounds(1, 1, 1, 1), nine_patch->layout_bounds); 280 } 281 282 TEST(NinePatchTest, RegionColorsAreCorrect) { 283 std::string err; 284 std::unique_ptr<NinePatch> nine_patch = 285 NinePatch::Create(kColorfulImage5x5, 5, 5, &err); 286 ASSERT_NE(nullptr, nine_patch); 287 288 std::vector<uint32_t> expected_colors = { 289 NinePatch::PackRGBA((uint8_t*)RED), 290 (uint32_t)android::Res_png_9patch::NO_COLOR, 291 NinePatch::PackRGBA((uint8_t*)GREEN), 292 (uint32_t)android::Res_png_9patch::TRANSPARENT_COLOR, 293 NinePatch::PackRGBA((uint8_t*)BLUE), 294 NinePatch::PackRGBA((uint8_t*)GREEN), 295 }; 296 EXPECT_EQ(expected_colors, nine_patch->region_colors); 297 } 298 299 TEST(NinePatchTest, OutlineFromOpaqueImage) { 300 std::string err; 301 std::unique_ptr<NinePatch> nine_patch = 302 NinePatch::Create(kOutlineOpaque10x10, 10, 10, &err); 303 ASSERT_NE(nullptr, nine_patch); 304 EXPECT_EQ(Bounds(2, 2, 2, 2), nine_patch->outline); 305 EXPECT_EQ(0x000000ffu, nine_patch->outline_alpha); 306 EXPECT_EQ(0.0f, nine_patch->outline_radius); 307 } 308 309 TEST(NinePatchTest, OutlineFromTranslucentImage) { 310 std::string err; 311 std::unique_ptr<NinePatch> nine_patch = 312 NinePatch::Create(kOutlineTranslucent10x10, 10, 10, &err); 313 ASSERT_NE(nullptr, nine_patch); 314 EXPECT_EQ(Bounds(3, 3, 3, 3), nine_patch->outline); 315 EXPECT_EQ(0x000000b3u, nine_patch->outline_alpha); 316 EXPECT_EQ(0.0f, nine_patch->outline_radius); 317 } 318 319 TEST(NinePatchTest, OutlineFromOffCenterImage) { 320 std::string err; 321 std::unique_ptr<NinePatch> nine_patch = 322 NinePatch::Create(kOutlineOffsetTranslucent12x10, 12, 10, &err); 323 ASSERT_NE(nullptr, nine_patch); 324 325 // TODO(adamlesinski): The old AAPT algorithm searches from the outside to the 326 // middle for each inset. If the outline is shifted, the search may not find a 327 // closer bounds. 328 // This check should be: 329 // EXPECT_EQ(Bounds(5, 3, 3, 3), ninePatch->outline); 330 // but until I know what behavior I'm breaking, I will leave it at the 331 // incorrect: 332 EXPECT_EQ(Bounds(4, 3, 3, 3), nine_patch->outline); 333 334 EXPECT_EQ(0x000000b3u, nine_patch->outline_alpha); 335 EXPECT_EQ(0.0f, nine_patch->outline_radius); 336 } 337 338 TEST(NinePatchTest, OutlineRadius) { 339 std::string err; 340 std::unique_ptr<NinePatch> nine_patch = 341 NinePatch::Create(kOutlineRadius5x5, 5, 5, &err); 342 ASSERT_NE(nullptr, nine_patch); 343 EXPECT_EQ(Bounds(0, 0, 0, 0), nine_patch->outline); 344 EXPECT_EQ(3.4142f, nine_patch->outline_radius); 345 } 346 347 ::testing::AssertionResult BigEndianOne(uint8_t* cursor) { 348 if (cursor[0] == 0 && cursor[1] == 0 && cursor[2] == 0 && cursor[3] == 1) { 349 return ::testing::AssertionSuccess(); 350 } 351 return ::testing::AssertionFailure() << "Not BigEndian 1"; 352 } 353 354 TEST(NinePatchTest, SerializePngEndianness) { 355 std::string err; 356 std::unique_ptr<NinePatch> nine_patch = 357 NinePatch::Create(kStretchAndPadding5x5, 5, 5, &err); 358 ASSERT_NE(nullptr, nine_patch); 359 360 size_t len; 361 std::unique_ptr<uint8_t[]> data = nine_patch->SerializeBase(&len); 362 ASSERT_NE(nullptr, data); 363 ASSERT_NE(0u, len); 364 365 // Skip past wasDeserialized + numXDivs + numYDivs + numColors + xDivsOffset + 366 // yDivsOffset 367 // (12 bytes) 368 uint8_t* cursor = data.get() + 12; 369 370 // Check that padding is big-endian. Expecting value 1. 371 EXPECT_TRUE(BigEndianOne(cursor)); 372 EXPECT_TRUE(BigEndianOne(cursor + 4)); 373 EXPECT_TRUE(BigEndianOne(cursor + 8)); 374 EXPECT_TRUE(BigEndianOne(cursor + 12)); 375 } 376 377 } // namespace aapt 378