8af81584e19f198d597b4af1125a9431bb452914
[WebKit-https.git] / Tools / TestWebKitAPI / Tests / WebCore / ComplexTextController.cpp
1 /*
2  * Copyright (C) 2016 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27
28 #include <JavaScriptCore/InitializeThreading.h>
29 #include <WebCore/ComplexTextController.h>
30 #include <WebCore/FontCascade.h>
31 #include <wtf/MainThread.h>
32 #include <wtf/RunLoop.h>
33
34 using namespace WebCore;
35
36 namespace TestWebKitAPI {
37
38 class ComplexTextControllerTest : public testing::Test {
39 public:
40     virtual void SetUp()
41     {
42         JSC::initializeThreading();
43         RunLoop::initializeMainRunLoop();
44     }
45 };
46
47 TEST_F(ComplexTextControllerTest, InitialAdvanceWithLeftRunInRTL)
48 {
49     FontCascadeDescription description;
50     description.setOneFamily("Times");
51     description.setComputedSize(80);
52     FontCascade font(description);
53     font.update();
54     auto spaceWidth = font.primaryFont().spaceWidth();
55
56     Vector<FloatSize> advances = { FloatSize(), FloatSize(21.640625, 0.0), FloatSize(42.3046875, 0.0), FloatSize(55.8984375, 0.0), FloatSize(22.34375, 0.0) };
57     Vector<FloatPoint> origins = { FloatPoint(-15.15625, 18.046875), FloatPoint(), FloatPoint(), FloatPoint(), FloatPoint() };
58
59     FloatSize initialAdvance = FloatSize(-15.15625, 18.046875);
60
61     UChar characters[] = { 0x644, 0x637, 0x641, 0x627, 0x64b, 0x20 };
62     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
63     TextRun textRun(StringView(characters, charactersLength));
64     auto run1 = ComplexTextController::ComplexTextRun::create({ FloatSize(21.875, 0) }, { FloatPoint() }, { 5 }, { 5 }, FloatSize(), font.primaryFont(), characters, 0, charactersLength, 5, 6, false);
65     auto run2 = ComplexTextController::ComplexTextRun::create(advances, origins, { 193, 377, 447, 431, 458 }, { 4, 3, 2, 1, 0 }, initialAdvance, font.primaryFont(), characters, 0, charactersLength, 0, 5, false);
66     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
67     runs.append(WTFMove(run1));
68     runs.append(WTFMove(run2));
69     ComplexTextController controller(font, textRun, runs);
70
71     float totalWidth = 0;
72     for (size_t i = 1; i < advances.size(); ++i)
73         totalWidth += advances[i].width();
74     EXPECT_NEAR(controller.totalWidth(), spaceWidth + totalWidth, 0.0001);
75     GlyphBuffer glyphBuffer;
76     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
77     controller.advance(0, &glyphBuffer);
78     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
79     controller.advance(1, &glyphBuffer);
80     EXPECT_NEAR(controller.runWidthSoFar(), advances[4].width(), 0.0001);
81     controller.advance(6, &glyphBuffer);
82     EXPECT_NEAR(controller.runWidthSoFar(), spaceWidth + totalWidth, 0.0001);
83     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), 0, 0.0001);
84     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), 0, 0.0001);
85     EXPECT_EQ(glyphBuffer.size(), 6U);
86     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), advances[4].width(), 0.0001);
87     EXPECT_NEAR(glyphBuffer.advanceAt(1).width(), advances[3].width(), 0.0001);
88     EXPECT_NEAR(glyphBuffer.advanceAt(2).width(), advances[2].width(), 0.0001);
89     EXPECT_NEAR(glyphBuffer.advanceAt(3).width(), advances[1].width(), 0.0001);
90     EXPECT_NEAR(glyphBuffer.advanceAt(4).width(), -initialAdvance.width(), 0.0001);
91     EXPECT_NEAR(glyphBuffer.advanceAt(5).width(), spaceWidth + initialAdvance.width(), 0.0001);
92 }
93
94 TEST_F(ComplexTextControllerTest, InitialAdvanceInRTL)
95 {
96     FontCascadeDescription description;
97     description.setOneFamily("Times");
98     description.setComputedSize(80);
99     FontCascade font(description);
100     font.update();
101
102     Vector<FloatSize> advances = { FloatSize(), FloatSize(21.640625, 0.0), FloatSize(42.3046875, 0.0), FloatSize(55.8984375, 0.0), FloatSize(22.34375, 0.0) };
103     Vector<FloatPoint> origins = { FloatPoint(-15.15625, 18.046875), FloatPoint(), FloatPoint(), FloatPoint(), FloatPoint() };
104
105     FloatSize initialAdvance = FloatSize(-15.15625, 18.046875);
106
107     UChar characters[] = { 0x644, 0x637, 0x641, 0x627, 0x64b };
108     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
109     TextRun textRun(StringView(characters, charactersLength));
110     auto run = ComplexTextController::ComplexTextRun::create(advances, origins, { 193, 377, 447, 431, 458 }, { 4, 3, 2, 1, 0 }, initialAdvance, font.primaryFont(), characters, 0, charactersLength, 0, 5, false);
111     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
112     runs.append(WTFMove(run));
113     ComplexTextController controller(font, textRun, runs);
114
115     float totalWidth = 0;
116     for (size_t i = 1; i < advances.size(); ++i)
117         totalWidth += advances[i].width();
118     EXPECT_NEAR(controller.totalWidth(), totalWidth, 0.0001);
119     GlyphBuffer glyphBuffer;
120     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
121     controller.advance(0, &glyphBuffer);
122     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
123     controller.advance(1, &glyphBuffer);
124     EXPECT_NEAR(controller.runWidthSoFar(), advances[4].width(), 0.0001);
125     controller.advance(5, &glyphBuffer);
126     EXPECT_NEAR(controller.runWidthSoFar(), totalWidth, 0.0001);
127     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), initialAdvance.width(), 0.0001);
128     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), initialAdvance.height(), 0.0001);
129     EXPECT_EQ(glyphBuffer.size(), 5U);
130     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), advances[4].width(), 0.0001);
131     EXPECT_NEAR(glyphBuffer.advanceAt(1).width(), advances[3].width(), 0.0001);
132     EXPECT_NEAR(glyphBuffer.advanceAt(2).width(), advances[2].width(), 0.0001);
133     EXPECT_NEAR(glyphBuffer.advanceAt(3).width(), advances[1].width(), 0.0001);
134     EXPECT_NEAR(glyphBuffer.advanceAt(4).width(), -initialAdvance.width(), 0.0001);
135     EXPECT_NEAR(glyphBuffer.advanceAt(4).height(), initialAdvance.height(), 0.0001);
136 }
137
138 TEST_F(ComplexTextControllerTest, InitialAdvanceWithLeftRunInLTR)
139 {
140     FontCascadeDescription description;
141     description.setOneFamily("LucidaGrande");
142     description.setComputedSize(80);
143     FontCascade font(description);
144     font.update();
145     auto spaceWidth = font.primaryFont().spaceWidth();
146
147     Vector<FloatSize> advances = { FloatSize(76.347656, 0.000000), FloatSize(0.000000, 0.000000) };
148     Vector<FloatPoint> origins = { FloatPoint(), FloatPoint(-23.281250, -8.398438) };
149
150     FloatSize initialAdvance = FloatSize(28.144531, 0);
151
152     UChar characters[] = { 0x20, 0x61, 0x20e3 };
153     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
154     TextRun textRun(StringView(characters, charactersLength));
155     auto run1 = ComplexTextController::ComplexTextRun::create({ FloatSize(spaceWidth, 0) }, { FloatPoint() }, { 5 }, { 0 }, FloatSize(), font.primaryFont(), characters, 0, charactersLength, 0, 1, true);
156     auto run2 = ComplexTextController::ComplexTextRun::create(advances, origins, { 68, 1471 }, { 1, 2 }, initialAdvance, font.primaryFont(), characters, 0, charactersLength, 1, 3, true);
157     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
158     runs.append(WTFMove(run1));
159     runs.append(WTFMove(run2));
160     ComplexTextController controller(font, textRun, runs);
161
162     EXPECT_NEAR(controller.totalWidth(), spaceWidth + 76.347656 + initialAdvance.width(), 0.0001);
163     GlyphBuffer glyphBuffer;
164     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
165     controller.advance(0, &glyphBuffer);
166     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
167     controller.advance(1, &glyphBuffer);
168     EXPECT_NEAR(controller.runWidthSoFar(), spaceWidth, 0.0001);
169     controller.advance(2, &glyphBuffer);
170     EXPECT_NEAR(controller.runWidthSoFar(), spaceWidth + advances[0].width() + initialAdvance.width(), 0.0001);
171     controller.advance(3, &glyphBuffer);
172     EXPECT_NEAR(controller.runWidthSoFar(), spaceWidth + 76.347656 + initialAdvance.width(), 0.0001);
173     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), 0, 0.0001);
174     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), 0, 0.0001);
175     EXPECT_EQ(glyphBuffer.size(), 3U);
176     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), spaceWidth + initialAdvance.width(), 0.0001);
177     EXPECT_NEAR(glyphBuffer.advanceAt(1).width(), 53.066406, 0.0001);
178     EXPECT_NEAR(glyphBuffer.advanceAt(2).width(), 23.281250, 0.0001);
179 }
180
181 TEST_F(ComplexTextControllerTest, InitialAdvanceInLTR)
182 {
183     FontCascadeDescription description;
184     description.setOneFamily("LucidaGrande");
185     description.setComputedSize(80);
186     FontCascade font(description);
187     font.update();
188
189     Vector<FloatSize> advances = { FloatSize(76.347656, 0.000000), FloatSize(0.000000, 0.000000) };
190     Vector<FloatPoint> origins = { FloatPoint(), FloatPoint(-23.281250, -8.398438) };
191
192     FloatSize initialAdvance = FloatSize(28.144531, 0);
193
194     UChar characters[] = { 0x61, 0x20e3 };
195     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
196     TextRun textRun(StringView(characters, charactersLength));
197     auto run = ComplexTextController::ComplexTextRun::create(advances, origins, { 68, 1471 }, { 0, 1 }, initialAdvance, font.primaryFont(), characters, 0, charactersLength, 0, 2, true);
198     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
199     runs.append(WTFMove(run));
200     ComplexTextController controller(font, textRun, runs);
201
202     EXPECT_NEAR(controller.totalWidth(), 76.347656 + initialAdvance.width(), 0.0001);
203     GlyphBuffer glyphBuffer;
204     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
205     controller.advance(0, &glyphBuffer);
206     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
207     controller.advance(1, &glyphBuffer);
208     EXPECT_NEAR(controller.runWidthSoFar(), advances[0].width() + initialAdvance.width(), 0.0001);
209     controller.advance(2, &glyphBuffer);
210     EXPECT_NEAR(controller.runWidthSoFar(), 76.347656 + initialAdvance.width(), 0.0001);
211     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), initialAdvance.width(), 0.0001);
212     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), initialAdvance.height(), 0.0001);
213     EXPECT_EQ(glyphBuffer.size(), 2U);
214     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), 53.066406, 0.0001);
215     EXPECT_NEAR(glyphBuffer.advanceAt(1).width(), 23.281250, 0.0001);
216 }
217
218 TEST_F(ComplexTextControllerTest, InitialAdvanceInRTLNoOrigins)
219 {
220     FontCascadeDescription description;
221     description.setOneFamily("Times");
222     description.setComputedSize(48);
223     FontCascade font(description);
224     font.update();
225
226     FloatSize initialAdvance = FloatSize(4.33996383363472, 12.368896925859);
227
228     UChar characters[] = { 0x633, 0x20, 0x627, 0x650 };
229     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
230     TextRun textRun(StringView(characters, charactersLength));
231     auto run1 = ComplexTextController::ComplexTextRun::create({ FloatSize(-4.33996383363472, -12.368896925859), FloatSize(14.0397830018083, 0) }, { }, { 884, 240 }, { 3, 2 }, initialAdvance, font.primaryFont(), characters, 0, charactersLength, 2, 4, false);
232     auto run2 = ComplexTextController::ComplexTextRun::create({ FloatSize(12.0, 0) }, { }, { 3 }, { 1 }, FloatSize(), font.primaryFont(), characters, 0, charactersLength, 1, 2, false);
233     auto run3 = ComplexTextController::ComplexTextRun::create({ FloatSize(43.8119349005425, 0) }, { }, { 276 }, { 0 }, FloatSize(), font.primaryFont(), characters, 0, charactersLength, 0, 1, false);
234     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
235     runs.append(WTFMove(run1));
236     runs.append(WTFMove(run2));
237     runs.append(WTFMove(run3));
238     ComplexTextController controller(font, textRun, runs);
239
240     float totalWidth = 14.0397830018083 + 12.0 + 43.8119349005425;
241     EXPECT_NEAR(controller.totalWidth(), totalWidth, 0.0001);
242     GlyphBuffer glyphBuffer;
243     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
244     controller.advance(0, &glyphBuffer);
245     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
246     controller.advance(1, &glyphBuffer);
247     EXPECT_NEAR(controller.runWidthSoFar(), 43.8119349005425, 0.0001);
248     controller.advance(2, &glyphBuffer);
249     EXPECT_NEAR(controller.runWidthSoFar(), 43.8119349005425 + 12.0, 0.0001);
250     controller.advance(3, &glyphBuffer);
251     EXPECT_NEAR(controller.runWidthSoFar(), totalWidth, 0.0001);
252     controller.advance(4, &glyphBuffer);
253     EXPECT_NEAR(controller.runWidthSoFar(), totalWidth, 0.0001);
254     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), initialAdvance.width(), 0.0001);
255     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), initialAdvance.height(), 0.0001);
256     EXPECT_EQ(glyphBuffer.size(), 4U);
257     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), 43.8119349005425, 0.0001);
258     EXPECT_NEAR(glyphBuffer.advanceAt(1).width(), 12.0, 0.0001);
259     EXPECT_NEAR(glyphBuffer.advanceAt(2).width(), 14.0397830018083, 0.0001);
260     EXPECT_NEAR(glyphBuffer.advanceAt(3).width(), -4.33996383363472, 0.0001);
261     EXPECT_NEAR(glyphBuffer.advanceAt(3).height(), 12.368896925859, 0.0001);
262 }
263
264 TEST_F(ComplexTextControllerTest, LeadingExpansion)
265 {
266     FontCascadeDescription description;
267     description.setOneFamily("Times");
268     description.setComputedSize(48);
269     FontCascade font(description);
270     font.update();
271
272     UChar characters[] = { 'a' };
273     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
274     TextRun textRun(StringView(characters, charactersLength), 0, 100, ForceLeadingExpansion);
275     auto run = ComplexTextController::ComplexTextRun::create({ FloatSize(24, 0) }, { }, { 16 }, { 0 }, FloatSize(), font.primaryFont(), characters, 0, charactersLength, 0, 1, true);
276     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
277     runs.append(WTFMove(run));
278     ComplexTextController controller(font, textRun, runs);
279
280     float totalWidth = 100 + 24;
281     EXPECT_NEAR(controller.totalWidth(), totalWidth, 0.0001);
282     GlyphBuffer glyphBuffer;
283     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
284     controller.advance(0, &glyphBuffer);
285     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
286     controller.advance(1, &glyphBuffer);
287     EXPECT_NEAR(controller.runWidthSoFar(), totalWidth, 0.0001);
288     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), 100, 0.0001);
289     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), 0, 0.0001);
290     EXPECT_EQ(glyphBuffer.size(), 1U);
291     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), 24, 0.0001);
292 }
293
294 TEST_F(ComplexTextControllerTest, VerticalAdvances)
295 {
296     FontCascadeDescription description;
297     description.setOneFamily("Times");
298     description.setComputedSize(48);
299     FontCascade font(description);
300     font.update();
301
302     UChar characters[] = { 'a', 'b', 'c', 'd' };
303     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
304     TextRun textRun(StringView(characters, charactersLength));
305     auto run1 = ComplexTextController::ComplexTextRun::create({ FloatSize(0, 1), FloatSize(0, 2) }, { FloatPoint(0, 4), FloatPoint(0, 8) }, { 16, 17 }, { 0, 1 }, FloatSize(0, 16), font.primaryFont(), characters, 0, charactersLength, 0, 2, true);
306     auto run2 = ComplexTextController::ComplexTextRun::create({ FloatSize(0, 32), FloatSize(0, 64) }, { FloatPoint(0, 128), FloatPoint(0, 256) }, { 18, 19 }, { 2, 3 }, FloatSize(0, 512), font.primaryFont(), characters, 0, charactersLength, 2, 4, true);
307     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
308     runs.append(WTFMove(run1));
309     runs.append(WTFMove(run2));
310     ComplexTextController controller(font, textRun, runs);
311
312     EXPECT_NEAR(controller.totalWidth(), 0, 0.0001);
313     GlyphBuffer glyphBuffer;
314     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
315     controller.advance(0, &glyphBuffer);
316     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
317     controller.advance(1, &glyphBuffer);
318     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
319     controller.advance(2, &glyphBuffer);
320     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
321     controller.advance(3, &glyphBuffer);
322     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
323     controller.advance(4, &glyphBuffer);
324     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
325     EXPECT_NEAR(glyphBuffer.initialAdvance().width(), 0, 0.0001);
326     EXPECT_NEAR(glyphBuffer.initialAdvance().height(), 16, 0.0001);
327     EXPECT_EQ(glyphBuffer.size(), 4U);
328     EXPECT_NEAR(glyphBuffer.advanceAt(0).width(), 0, 0.0001);
329     EXPECT_NEAR(glyphBuffer.advanceAt(0).height(), 4 - 1 -8, 0.0001);
330     EXPECT_NEAR(glyphBuffer.advanceAt(1).width(), 0, 0.0001);
331     EXPECT_NEAR(glyphBuffer.advanceAt(1).height(), 8 - 2 - 512, 0.0001);
332     EXPECT_NEAR(glyphBuffer.advanceAt(2).width(), 0, 0.0001);
333     EXPECT_NEAR(glyphBuffer.advanceAt(2).height(), 128 - 32 - 256, 0.0001);
334     EXPECT_NEAR(glyphBuffer.advanceAt(3).width(), 0, 0.0001);
335     EXPECT_NEAR(glyphBuffer.advanceAt(3).height(), 256 - 64, 0.0001);
336 }
337
338 TEST_F(ComplexTextControllerTest, TotalWidthWithJustification)
339 {
340     FontCascadeDescription description;
341     description.setOneFamily("Times");
342     description.setComputedSize(80);
343     FontCascade font(description);
344     font.update();
345
346     Vector<FloatSize> advances = { FloatSize(1, 0), FloatSize(2, 0), FloatSize(4, 0), FloatSize(8, 0), FloatSize(16, 0) };
347     Vector<FloatPoint> origins = { FloatPoint(), FloatPoint(), FloatPoint(), FloatPoint(), FloatPoint() };
348
349     FloatSize initialAdvance = FloatSize();
350
351     UChar characters[] = { 0x644, ' ', 0x644, ' ', 0x644 };
352     size_t charactersLength = WTF_ARRAY_LENGTH(characters);
353     TextRun textRun(StringView(characters, charactersLength), 0, 14, DefaultExpansion, RTL);
354     auto run = ComplexTextController::ComplexTextRun::create(advances, origins, { 5, 6, 7, 8, 9 }, { 4, 3, 2, 1, 0 }, initialAdvance, font.primaryFont(), characters, 0, charactersLength, 0, 5, false);
355     Vector<Ref<ComplexTextController::ComplexTextRun>> runs;
356     runs.append(WTFMove(run));
357     ComplexTextController controller(font, textRun, runs);
358
359     EXPECT_NEAR(controller.totalWidth(), 1 + 20 + 7 + 4 + 20 + 7 + 16, 0.0001);
360     GlyphBuffer glyphBuffer;
361     EXPECT_NEAR(controller.runWidthSoFar(), 0, 0.0001);
362     controller.advance(5, &glyphBuffer);
363     EXPECT_EQ(glyphBuffer.size(), 5U);
364     EXPECT_NEAR(glyphBuffer.advanceAt(0).width() + glyphBuffer.advanceAt(1).width() + glyphBuffer.advanceAt(2).width() + glyphBuffer.advanceAt(3).width() + glyphBuffer.advanceAt(4).width(), controller.totalWidth(), 0.0001);
365 }
366
367 }