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