• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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.
4 
5 #include "ppapi/tests/test_browser_font.h"
6 
7 #include <string.h>
8 
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"
13 
14 REGISTER_TEST_CASE(BrowserFont);
15 
Init()16 bool TestBrowserFont::Init() {
17   return true;
18 }
19 
RunTests(const std::string & filter)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);
29 }
30 
31 // Just tests that GetFontFamilies is hooked up & returns something.
TestFontFamilies()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())
36     PASS();
37 
38   pp::Var families = pp::BrowserFont_Trusted::GetFontFamilies(instance_);
39 
40   ASSERT_TRUE(families.is_string());
41   ASSERT_TRUE(!families.AsString().empty());
42   PASS();
43 }
44 
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.
TestMeasure()49 std::string TestBrowserFont::TestMeasure() {
50   pp::BrowserFontDescription desc;
51   pp::BrowserFont_Trusted font(instance_, desc);
52 
53   int32_t length1 = font.MeasureText(pp::BrowserFontTextRun("WWW"));
54   ASSERT_TRUE(length1 > 0);
55   int32_t length2 = font.MeasureText(pp::BrowserFontTextRun("WWWWWWWW"));
56 
57   ASSERT_TRUE(length2 >= length1 * 2);
58   PASS();
59 }
60 
TestMeasureRTL()61 std::string TestBrowserFont::TestMeasureRTL() {
62   pp::BrowserFontDescription desc;
63   pp::BrowserFont_Trusted font(instance_, desc);
64 
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);
70 
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);
79 
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]);
87 
88   // Test the same sequence with force LTR. The offsets should appear in
89   // sequence.
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]);
99 
100   PASS();
101 }
102 
103 // Tests that the character/pixel offset functions correctly round-trip.
TestCharPos()104 std::string TestBrowserFont::TestCharPos() {
105   pp::BrowserFontDescription desc;
106   pp::BrowserFont_Trusted font(instance_, desc);
107 
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);
112 
113   uint32_t computed_char = font.CharacterOffsetForPixel(
114       run, static_cast<int32_t>(pixel_offset));
115   ASSERT_TRUE(computed_char == original_char);
116 
117   PASS();
118 }
119 
120 // Tests that we can get character positions in a mixed LTR/RTL run.
TestCharPosRTL()121 std::string TestBrowserFont::TestCharPosRTL() {
122   pp::BrowserFontDescription desc;
123   pp::BrowserFont_Trusted font(instance_, desc);
124 
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");
128 
129   pp::BrowserFontTextRun run(mixed);
130   static const int kNumChars = 6;
131   int expected_char_sequence[kNumChars] = { 0, 1, 3, 2, 4, 5 };
132 
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.
141       last_sequence++;
142       ASSERT_TRUE(last_sequence < kNumChars);
143       ASSERT_TRUE(cur_char == expected_char_sequence[last_sequence]);
144     }
145   }
146 
147   // Try the same string with force LTR. The characters should all appear in
148   // sequence.
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) {
154       last_forced_char++;
155       ASSERT_TRUE(cur_char == last_forced_char);
156     }
157   }
158 
159   PASS();
160 }
161 
162 // Tests that drawing some text produces "some" output.
TestDraw()163 std::string TestBrowserFont::TestDraw() {
164   pp::BrowserFontDescription desc;
165   desc.set_family(PP_BROWSERFONT_TRUSTED_FAMILY_MONOSPACE);
166   desc.set_size(10);
167   pp::BrowserFont_Trusted font(instance_, desc);
168 
169   const pp::Size kSize(30, 10);
170   pp::ImageData image(instance_,
171                       PP_IMAGEDATAFORMAT_BGRA_PREMUL,  // 0xAARRGGBB
172                       kSize,
173                       false);  // init_to_zero
174   ASSERT_FALSE(image.is_null());
175 
176   // Draw black text on white canvas.
177   memset(image.data(), 0xFF, 4 * kSize.GetArea());
178   font.DrawSimpleText(&image,
179                       "Hello",
180                       pp::Point(0, 10),  // Baseline position.
181                       0xFF000000,  // Black text.
182                       true);  // image_data_is_opaque.
183 
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)
194       ++text_pixels;
195     if ((pixels[i] & kAlphaMask) == kAlphaMask)
196       ++opaque_pixels;
197   }
198   ASSERT_GT(text_pixels, 0);
199   ASSERT_EQ(opaque_pixels, kSize.GetArea());
200   PASS();
201 }
202