1 // Copyright (c) 2012 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.
5 #include "ppapi/tests/test_browser_font.h"
9 #include "ppapi/tests/test_utils.h"
10 #include "ppapi/tests/testing_instance.h"
11 #include "ppapi/cpp/image_data.h"
12 #include "ppapi/cpp/trusted/browser_font_trusted.h"
14 REGISTER_TEST_CASE(BrowserFont
);
16 bool TestBrowserFont::Init() {
20 void TestBrowserFont::RunTests(const std::string
& filter
) {
21 RUN_TEST(FontFamilies
, filter
);
22 RUN_TEST(Measure
, filter
);
23 RUN_TEST(MeasureRTL
, filter
);
24 RUN_TEST(CharPos
, filter
);
25 // This test is disabled. It doesn't currently pass. See the
26 // CharacterOffsetForPixel API.
27 //RUN_TEST(CharPosRTL, filter);
28 RUN_TEST(Draw
, filter
);
31 // Just tests that GetFontFamilies is hooked up & returns something.
32 std::string
TestBrowserFont::TestFontFamilies() {
33 // This function is only supported out-of-process.
34 const PPB_Testing_Private
* testing_interface
= GetTestingInterface();
35 if (testing_interface
&& !testing_interface
->IsOutOfProcess())
38 pp::Var families
= pp::BrowserFont_Trusted::GetFontFamilies(instance_
);
40 ASSERT_TRUE(families
.is_string());
41 ASSERT_TRUE(!families
.AsString().empty());
45 // Tests that measuring text behaves reasonably. We aren't sure if the browser
46 // will be doing kerning or something for the particular default font, so we
47 // just make a string that we're pretty sure should be more than twice as long
48 // as another one, and verify that condition.
49 std::string
TestBrowserFont::TestMeasure() {
50 pp::BrowserFontDescription desc
;
51 pp::BrowserFont_Trusted
font(instance_
, desc
);
53 int32_t length1
= font
.MeasureText(pp::BrowserFontTextRun("WWW"));
54 ASSERT_TRUE(length1
> 0);
55 int32_t length2
= font
.MeasureText(pp::BrowserFontTextRun("WWWWWWWW"));
57 ASSERT_TRUE(length2
>= length1
* 2);
61 std::string
TestBrowserFont::TestMeasureRTL() {
62 pp::BrowserFontDescription desc
;
63 pp::BrowserFont_Trusted
font(instance_
, desc
);
65 // Mixed string, two chars of LTR, two of RTL, then two of LTR.
66 // Note this is in UTF-8 so has more than 6 bytes.
67 std::string
mixed("AB\xd7\x94\xd7\x97ZZ");
68 const int kNumChars
= 6;
69 pp::BrowserFontTextRun
run(mixed
);
71 // Note that since this is UTF-8, the two RTL chars are two bytes each.
72 int32_t len
[kNumChars
];
73 len
[0] = font
.PixelOffsetForCharacter(run
, 0);
74 len
[1] = font
.PixelOffsetForCharacter(run
, 1);
75 len
[2] = font
.PixelOffsetForCharacter(run
, 2);
76 len
[3] = font
.PixelOffsetForCharacter(run
, 3);
77 len
[4] = font
.PixelOffsetForCharacter(run
, 4);
78 len
[5] = font
.PixelOffsetForCharacter(run
, 5);
80 // First three chars should be increasing.
81 ASSERT_TRUE(len
[0] >= 0);
82 ASSERT_TRUE(len
[1] > len
[0]);
83 ASSERT_TRUE(len
[3] > len
[1]);
84 ASSERT_TRUE(len
[2] > len
[3]);
85 ASSERT_TRUE(len
[4] > len
[2]);
86 ASSERT_TRUE(len
[5] > len
[4]);
88 // Test the same sequence with force LTR. The offsets should appear in
90 pp::BrowserFontTextRun
forced_run(mixed
, false, true);
91 len
[0] = font
.PixelOffsetForCharacter(forced_run
, 0);
92 len
[1] = font
.PixelOffsetForCharacter(forced_run
, 1);
93 len
[2] = font
.PixelOffsetForCharacter(forced_run
, 2);
94 len
[3] = font
.PixelOffsetForCharacter(forced_run
, 3);
95 len
[4] = font
.PixelOffsetForCharacter(forced_run
, 4);
96 len
[5] = font
.PixelOffsetForCharacter(forced_run
, 5);
97 for (int i
= 1; i
< kNumChars
; i
++)
98 ASSERT_TRUE(len
[i
] > len
[i
- 1]);
103 // Tests that the character/pixel offset functions correctly round-trip.
104 std::string
TestBrowserFont::TestCharPos() {
105 pp::BrowserFontDescription desc
;
106 pp::BrowserFont_Trusted
font(instance_
, desc
);
108 pp::BrowserFontTextRun
run("Hello, world");
109 uint32_t original_char
= 3;
110 uint32_t pixel_offset
= font
.PixelOffsetForCharacter(run
, original_char
);
111 ASSERT_TRUE(pixel_offset
> 0);
113 uint32_t computed_char
= font
.CharacterOffsetForPixel(
114 run
, static_cast<int32_t>(pixel_offset
));
115 ASSERT_TRUE(computed_char
== original_char
);
120 // Tests that we can get character positions in a mixed LTR/RTL run.
121 std::string
TestBrowserFont::TestCharPosRTL() {
122 pp::BrowserFontDescription desc
;
123 pp::BrowserFont_Trusted
font(instance_
, desc
);
125 // Mixed string, two chars of LTR, two of RTL, than two of LTR.
126 // Note this is in UTF-8 so has more than 6 bytes.
127 std::string
mixed("AB\xd7\x94\xd7\x97ZZ");
129 pp::BrowserFontTextRun
run(mixed
);
130 static const int kNumChars
= 6;
131 int expected_char_sequence
[kNumChars
] = { 0, 1, 3, 2, 4, 5 };
133 // Check that the characters appear in the order we expect.
134 int pixel_width
= font
.MeasureText(pp::BrowserFontTextRun(mixed
));
135 int last_sequence
= 0; // Index into expected_char_sequence.
136 for (int x
= 0; x
< pixel_width
; x
++) {
137 int cur_char
= font
.CharacterOffsetForPixel(run
, x
);
138 if (cur_char
!= expected_char_sequence
[last_sequence
]) {
139 // This pixel has a different character. It should be the next one in
140 // the sequence for it to be correct.
142 ASSERT_TRUE(last_sequence
< kNumChars
);
143 ASSERT_TRUE(cur_char
== expected_char_sequence
[last_sequence
]);
147 // Try the same string with force LTR. The characters should all appear in
149 pp::BrowserFontTextRun
forced_run(mixed
, false, true);
150 int last_forced_char
= 0; // Char index into the forced sequence.
151 for (int x
= 0; x
< pixel_width
; x
++) {
152 int cur_char
= font
.CharacterOffsetForPixel(forced_run
, x
);
153 if (cur_char
!= last_forced_char
) {
155 ASSERT_TRUE(cur_char
== last_forced_char
);
162 // Tests that drawing some text produces "some" output.
163 std::string
TestBrowserFont::TestDraw() {
164 pp::BrowserFontDescription desc
;
165 desc
.set_family(PP_BROWSERFONT_TRUSTED_FAMILY_MONOSPACE
);
167 pp::BrowserFont_Trusted
font(instance_
, desc
);
169 const pp::Size
kSize(30, 10);
170 pp::ImageData
image(instance_
,
171 PP_IMAGEDATAFORMAT_BGRA_PREMUL
, // 0xAARRGGBB
173 false); // init_to_zero
174 ASSERT_FALSE(image
.is_null());
176 // Draw black text on white canvas.
177 memset(image
.data(), 0xFF, 4 * kSize
.GetArea());
178 font
.DrawSimpleText(&image
,
180 pp::Point(0, 10), // Baseline position.
181 0xFF000000, // Black text.
182 true); // image_data_is_opaque.
184 // Expect that at least a few pixels are non-white (text).
185 // Due to blending, there may be rounding errors and
186 // checking for exact black may not be correct.
187 // Also expect that all pixels are opaque.
188 const uint32_t kRGBMask
= 0x00FFFFFF;
189 const uint32_t kAlphaMask
= 0xFF000000;
190 int text_pixels
= 0, opaque_pixels
= 0;
191 const uint32_t* pixels
= static_cast<const uint32_t*>(image
.data());
192 for (int i
= 0; i
< kSize
.GetArea(); ++i
) {
193 if ((pixels
[i
] & kRGBMask
) != kRGBMask
)
195 if ((pixels
[i
] & kAlphaMask
) == kAlphaMask
)
198 ASSERT_GT(text_pixels
, 0);
199 ASSERT_EQ(opaque_pixels
, kSize
.GetArea());