Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(520)

Side by Side Diff: Source/core/layout/svg/SVGTextQuery.cpp

Issue 1083073002: Add a "logical query mode" to SVGTextQuery (Closed) Base URL: svn://svn.chromium.org/blink/trunk
Patch Set: Missing TEs. Created 5 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 /* 1 /*
2 * Copyright (C) Research In Motion Limited 2010-2012. All rights reserved. 2 * Copyright (C) Research In Motion Limited 2010-2012. All rights reserved.
3 * 3 *
4 * This library is free software; you can redistribute it and/or 4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public 5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either 6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version. 7 * version 2 of the License, or (at your option) any later version.
8 * 8 *
9 * This library is distributed in the hope that it will be useful, 9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
(...skipping 19 matching lines...) Expand all
30 #include "platform/FloatConversion.h" 30 #include "platform/FloatConversion.h"
31 #include "wtf/MathExtras.h" 31 #include "wtf/MathExtras.h"
32 #include "wtf/Vector.h" 32 #include "wtf/Vector.h"
33 33
34 namespace blink { 34 namespace blink {
35 35
36 // Base structure for callback user data 36 // Base structure for callback user data
37 struct QueryData { 37 struct QueryData {
38 QueryData() 38 QueryData()
39 : isVerticalText(false) 39 : isVerticalText(false)
40 , processedCharacters(0) 40 , currentOffset(0)
41 , textLayoutObject(0) 41 , textLayoutObject(0)
42 , textBox(0) 42 , textBox(0)
43 { 43 {
44 } 44 }
45 45
46 bool isVerticalText; 46 bool isVerticalText;
47 unsigned processedCharacters; 47 unsigned currentOffset;
48 LayoutSVGInlineText* textLayoutObject; 48 LayoutSVGInlineText* textLayoutObject;
49 const SVGInlineTextBox* textBox; 49 const SVGInlineTextBox* textBox;
50 }; 50 };
51 51
52 static inline InlineFlowBox* flowBoxForLayoutObject(LayoutObject* layoutObject) 52 static inline InlineFlowBox* flowBoxForLayoutObject(LayoutObject* layoutObject)
53 { 53 {
54 if (!layoutObject) 54 if (!layoutObject)
55 return 0; 55 return 0;
56 56
57 if (layoutObject->isLayoutBlock()) { 57 if (layoutObject->isLayoutBlock()) {
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after
94 continue; 94 continue;
95 } 95 }
96 96
97 if (child->isSVGInlineTextBox()) 97 if (child->isSVGInlineTextBox())
98 textBoxes.append(toSVGInlineTextBox(child)); 98 textBoxes.append(toSVGInlineTextBox(child));
99 } 99 }
100 } 100 }
101 101
102 typedef bool ProcessTextFragmentCallback(QueryData*, const SVGTextFragment&); 102 typedef bool ProcessTextFragmentCallback(QueryData*, const SVGTextFragment&);
103 103
104 static bool executeQuery(LayoutObject* queryRoot, QueryData* queryData, ProcessT extFragmentCallback fragmentCallback) 104 static bool queryTextBox(QueryData* queryData, const SVGInlineTextBox* textBox, ProcessTextFragmentCallback fragmentCallback)
105 {
106 queryData->textBox = textBox;
107 queryData->textLayoutObject = &toLayoutSVGInlineText(textBox->layoutObject() );
108
109 queryData->isVerticalText = textBox->layoutObject().style()->svgStyle().isVe rticalWritingMode();
110 const Vector<SVGTextFragment>& fragments = textBox->textFragments();
111
112 // Loop over all text fragments in this text box, firing a callback for each .
113 unsigned fragmentCount = fragments.size();
pdr. 2015/04/16 02:34:58 Nit: range-based for loop would be slightly easier
fs 2015/04/16 14:22:21 Changed this loop and the one in spatialQuery. The
114 for (unsigned i = 0; i < fragmentCount; ++i) {
115 if (fragmentCallback(queryData, fragments.at(i)))
116 return true;
117 }
118 return false;
119 }
120
121 static void spatialQuery(LayoutObject* queryRoot, QueryData* queryData, ProcessT extFragmentCallback fragmentCallback)
pdr. 2015/04/16 02:34:58 Nit: Could you add a one-line comment here and abo
fs 2015/04/16 14:22:21 Done. (2-lines though.)
105 { 122 {
106 Vector<SVGInlineTextBox*> textBoxes; 123 Vector<SVGInlineTextBox*> textBoxes;
107 collectTextBoxesInFlowBox(flowBoxForLayoutObject(queryRoot), textBoxes); 124 collectTextBoxesInFlowBox(flowBoxForLayoutObject(queryRoot), textBoxes);
108 125
109 unsigned processedCharacters = 0;
110 unsigned textBoxCount = textBoxes.size(); 126 unsigned textBoxCount = textBoxes.size();
111 127
112 // Loop over all text boxes 128 // Loop over all text boxes
113 for (unsigned textBoxPosition = 0; textBoxPosition < textBoxCount; ++textBox Position) { 129 for (unsigned textBoxPosition = 0; textBoxPosition < textBoxCount; ++textBox Position) {
114 queryData->textBox = textBoxes[textBoxPosition]; 130 if (queryTextBox(queryData, textBoxes[textBoxPosition], fragmentCallback ))
115 queryData->textLayoutObject = &toLayoutSVGInlineText(queryData->textBox- >layoutObject()); 131 return;
116 ASSERT(queryData->textLayoutObject->style()); 132 }
133 }
117 134
118 queryData->isVerticalText = queryData->textLayoutObject->style()->svgSty le().isVerticalWritingMode(); 135 static void logicalQuery(LayoutObject* queryRoot, QueryData* queryData, ProcessT extFragmentCallback fragmentCallback)
119 const Vector<SVGTextFragment>& fragments = queryData->textBox->textFragm ents(); 136 {
137 if (!queryRoot)
138 return;
120 139
121 // Loop over all text fragments in this text box, firing a callback for each. 140 // Walk the layout tree in pre-order, starting at the specified root, and
122 unsigned fragmentCount = fragments.size(); 141 // run the query for each text node.
123 for (unsigned i = 0; i < fragmentCount; ++i) { 142 for (LayoutObject* layoutObject = queryRoot->slowFirstChild(); layoutObject; layoutObject = layoutObject->nextInPreOrder(queryRoot)) {
124 const SVGTextFragment& fragment = fragments.at(i); 143 if (!layoutObject->isSVGInlineText())
125 if (fragmentCallback(queryData, fragment)) 144 continue;
126 return true;
127 145
128 processedCharacters += fragment.length; 146 LayoutSVGInlineText& textLayoutObject = toLayoutSVGInlineText(*layoutObj ect);
147 ASSERT(textLayoutObject.style());
148
149 // TODO(fs): Allow filtering the search earlier, since we should be
150 // able to trivially reject (prune) at least some of the queries.
151
152 for (InlineTextBox* textBox = textLayoutObject.firstTextBox(); textBox; textBox = textBox->nextTextBox()) {
pdr. 2015/04/16 02:34:58 Does this need to be updated for each box? for (..
fs 2015/04/16 14:22:21 Ouch, yes it should (or rather "needs to") of cour
153 if (queryTextBox(queryData, toSVGInlineTextBox(textBox), fragmentCal lback))
154 return;
129 } 155 }
130 156 queryData->currentOffset += textLayoutObject.textLength();
131 queryData->processedCharacters = processedCharacters;
132 } 157 }
133 return false;
134 } 158 }
135 159
136 static void modifyStartEndPositionsRespectingLigatures(const QueryData* queryDat a, const SVGTextFragment& fragment, int& startPosition, int& endPosition) 160 static void modifyStartEndPositionsRespectingLigatures(const QueryData* queryDat a, const SVGTextFragment& fragment, int& startPosition, int& endPosition)
137 { 161 {
138 const Vector<SVGTextMetrics>& textMetricsValues = queryData->textLayoutObjec t->layoutAttributes()->textMetricsValues(); 162 const Vector<SVGTextMetrics>& textMetricsValues = queryData->textLayoutObjec t->layoutAttributes()->textMetricsValues();
139 163
140 unsigned textMetricsOffset = fragment.metricsListOffset; 164 unsigned textMetricsOffset = fragment.metricsListOffset;
141 int fragmentOffset = 0; 165 int fragmentOffset = 0;
142 int fragmentEnd = static_cast<int>(fragment.length); 166 int fragmentEnd = static_cast<int>(fragment.length);
143 167
(...skipping 16 matching lines...) Expand all
160 if (fragmentOffset >= endPosition) 184 if (fragmentOffset >= endPosition)
161 break; 185 break;
162 textMetricsOffset++; 186 textMetricsOffset++;
163 } 187 }
164 188
165 endPosition = fragmentOffset; 189 endPosition = fragmentOffset;
166 } 190 }
167 191
168 static bool mapStartEndPositionsIntoFragmentCoordinates(const QueryData* queryDa ta, const SVGTextFragment& fragment, int& startPosition, int& endPosition) 192 static bool mapStartEndPositionsIntoFragmentCoordinates(const QueryData* queryDa ta, const SVGTextFragment& fragment, int& startPosition, int& endPosition)
169 { 193 {
194 unsigned boxStart = queryData->currentOffset + queryData->textBox->start();
195
170 // Make <startPosition, endPosition> offsets relative to the current text bo x. 196 // Make <startPosition, endPosition> offsets relative to the current text bo x.
171 startPosition -= queryData->processedCharacters; 197 startPosition -= boxStart;
172 endPosition -= queryData->processedCharacters; 198 endPosition -= boxStart;
173 199
174 // Reuse the same logic used for text selection & painting, to map our 200 // Reuse the same logic used for text selection & painting, to map our
175 // query start/length into start/endPositions of the current text fragment. 201 // query start/length into start/endPositions of the current text fragment.
176 if (!queryData->textBox->mapStartEndPositionsIntoFragmentCoordinates(fragmen t, startPosition, endPosition)) 202 if (!queryData->textBox->mapStartEndPositionsIntoFragmentCoordinates(fragmen t, startPosition, endPosition))
177 return false; 203 return false;
178 204
179 modifyStartEndPositionsRespectingLigatures(queryData, fragment, startPositio n, endPosition); 205 modifyStartEndPositionsRespectingLigatures(queryData, fragment, startPositio n, endPosition);
180 ASSERT(startPosition < endPosition); 206 ASSERT(startPosition < endPosition);
181 return true; 207 return true;
182 } 208 }
183 209
184 // numberOfCharacters() implementation 210 // numberOfCharacters() implementation
185 static bool numberOfCharactersCallback(QueryData*, const SVGTextFragment&) 211 static bool numberOfCharactersCallback(QueryData*, const SVGTextFragment&)
186 { 212 {
187 // no-op 213 // no-op
188 return false; 214 return false;
189 } 215 }
190 216
191 unsigned SVGTextQuery::numberOfCharacters() const 217 unsigned SVGTextQuery::numberOfCharacters() const
192 { 218 {
193 QueryData data; 219 QueryData data;
194 executeQuery(m_queryRootLayoutObject, &data, numberOfCharactersCallback); 220 logicalQuery(m_queryRootLayoutObject, &data, numberOfCharactersCallback);
195 return data.processedCharacters; 221 return data.currentOffset;
196 } 222 }
197 223
198 // textLength() implementation 224 // textLength() implementation
199 struct TextLengthData : QueryData { 225 struct TextLengthData : QueryData {
200 TextLengthData() 226 TextLengthData()
201 : textLength(0) 227 : textLength(0)
202 { 228 {
203 } 229 }
204 230
205 float textLength; 231 float textLength;
206 }; 232 };
207 233
208 static bool textLengthCallback(QueryData* queryData, const SVGTextFragment& frag ment) 234 static bool textLengthCallback(QueryData* queryData, const SVGTextFragment& frag ment)
209 { 235 {
210 TextLengthData* data = static_cast<TextLengthData*>(queryData); 236 TextLengthData* data = static_cast<TextLengthData*>(queryData);
211 data->textLength += queryData->isVerticalText ? fragment.height : fragment.w idth; 237 data->textLength += queryData->isVerticalText ? fragment.height : fragment.w idth;
212 return false; 238 return false;
213 } 239 }
214 240
215 float SVGTextQuery::textLength() const 241 float SVGTextQuery::textLength() const
216 { 242 {
217 TextLengthData data; 243 TextLengthData data;
218 executeQuery(m_queryRootLayoutObject, &data, textLengthCallback); 244 logicalQuery(m_queryRootLayoutObject, &data, textLengthCallback);
219 return data.textLength; 245 return data.textLength;
220 } 246 }
221 247
222 // subStringLength() implementation 248 // subStringLength() implementation
223 struct SubStringLengthData : QueryData { 249 struct SubStringLengthData : QueryData {
224 SubStringLengthData(unsigned queryStartPosition, unsigned queryLength) 250 SubStringLengthData(unsigned queryStartPosition, unsigned queryLength)
225 : startPosition(queryStartPosition) 251 : startPosition(queryStartPosition)
226 , length(queryLength) 252 , length(queryLength)
227 , subStringLength(0) 253 , subStringLength(0)
228 { 254 {
(...skipping 15 matching lines...) Expand all
244 return false; 270 return false;
245 271
246 SVGTextMetrics metrics = SVGTextMetrics::measureCharacterRange(queryData->te xtLayoutObject, fragment.characterOffset + startPosition, endPosition - startPos ition, queryData->textBox->direction()); 272 SVGTextMetrics metrics = SVGTextMetrics::measureCharacterRange(queryData->te xtLayoutObject, fragment.characterOffset + startPosition, endPosition - startPos ition, queryData->textBox->direction());
247 data->subStringLength += queryData->isVerticalText ? metrics.height() : metr ics.width(); 273 data->subStringLength += queryData->isVerticalText ? metrics.height() : metr ics.width();
248 return false; 274 return false;
249 } 275 }
250 276
251 float SVGTextQuery::subStringLength(unsigned startPosition, unsigned length) con st 277 float SVGTextQuery::subStringLength(unsigned startPosition, unsigned length) con st
252 { 278 {
253 SubStringLengthData data(startPosition, length); 279 SubStringLengthData data(startPosition, length);
254 executeQuery(m_queryRootLayoutObject, &data, subStringLengthCallback); 280 logicalQuery(m_queryRootLayoutObject, &data, subStringLengthCallback);
255 return data.subStringLength; 281 return data.subStringLength;
256 } 282 }
257 283
258 // startPositionOfCharacter() implementation 284 // startPositionOfCharacter() implementation
259 struct StartPositionOfCharacterData : QueryData { 285 struct StartPositionOfCharacterData : QueryData {
260 StartPositionOfCharacterData(unsigned queryPosition) 286 StartPositionOfCharacterData(unsigned queryPosition)
261 : position(queryPosition) 287 : position(queryPosition)
262 { 288 {
263 } 289 }
264 290
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after
311 if (!mapStartEndPositionsIntoFragmentCoordinates(queryData, fragment, startP osition, endPosition)) 337 if (!mapStartEndPositionsIntoFragmentCoordinates(queryData, fragment, startP osition, endPosition))
312 return false; 338 return false;
313 339
314 data->startPosition = calculateGlyphPosition(queryData, fragment, startPosit ion); 340 data->startPosition = calculateGlyphPosition(queryData, fragment, startPosit ion);
315 return true; 341 return true;
316 } 342 }
317 343
318 FloatPoint SVGTextQuery::startPositionOfCharacter(unsigned position) const 344 FloatPoint SVGTextQuery::startPositionOfCharacter(unsigned position) const
319 { 345 {
320 StartPositionOfCharacterData data(position); 346 StartPositionOfCharacterData data(position);
321 executeQuery(m_queryRootLayoutObject, &data, startPositionOfCharacterCallbac k); 347 logicalQuery(m_queryRootLayoutObject, &data, startPositionOfCharacterCallbac k);
322 return data.startPosition; 348 return data.startPosition;
323 } 349 }
324 350
325 // endPositionOfCharacter() implementation 351 // endPositionOfCharacter() implementation
326 struct EndPositionOfCharacterData : QueryData { 352 struct EndPositionOfCharacterData : QueryData {
327 EndPositionOfCharacterData(unsigned queryPosition) 353 EndPositionOfCharacterData(unsigned queryPosition)
328 : position(queryPosition) 354 : position(queryPosition)
329 { 355 {
330 } 356 }
331 357
(...skipping 13 matching lines...) Expand all
345 // TODO(fs): mapStartEndPositionsIntoFragmentCoordinates(...) above applies 371 // TODO(fs): mapStartEndPositionsIntoFragmentCoordinates(...) above applies
346 // some heuristics for ligatures, so why not just use endPosition here? 372 // some heuristics for ligatures, so why not just use endPosition here?
347 // (rather than startPosition+1) 373 // (rather than startPosition+1)
348 data->endPosition = calculateGlyphPosition(queryData, fragment, startPositio n + 1); 374 data->endPosition = calculateGlyphPosition(queryData, fragment, startPositio n + 1);
349 return true; 375 return true;
350 } 376 }
351 377
352 FloatPoint SVGTextQuery::endPositionOfCharacter(unsigned position) const 378 FloatPoint SVGTextQuery::endPositionOfCharacter(unsigned position) const
353 { 379 {
354 EndPositionOfCharacterData data(position); 380 EndPositionOfCharacterData data(position);
355 executeQuery(m_queryRootLayoutObject, &data, endPositionOfCharacterCallback) ; 381 logicalQuery(m_queryRootLayoutObject, &data, endPositionOfCharacterCallback) ;
356 return data.endPosition; 382 return data.endPosition;
357 } 383 }
358 384
359 // rotationOfCharacter() implementation 385 // rotationOfCharacter() implementation
360 struct RotationOfCharacterData : QueryData { 386 struct RotationOfCharacterData : QueryData {
361 RotationOfCharacterData(unsigned queryPosition) 387 RotationOfCharacterData(unsigned queryPosition)
362 : position(queryPosition) 388 : position(queryPosition)
363 , rotation(0) 389 , rotation(0)
364 { 390 {
365 } 391 }
(...skipping 19 matching lines...) Expand all
385 fragmentTransform.scale(1 / fragmentTransform.xScale(), 1 / fragmentTran sform.yScale()); 411 fragmentTransform.scale(1 / fragmentTransform.xScale(), 1 / fragmentTran sform.yScale());
386 data->rotation = narrowPrecisionToFloat(rad2deg(atan2(fragmentTransform. b(), fragmentTransform.a()))); 412 data->rotation = narrowPrecisionToFloat(rad2deg(atan2(fragmentTransform. b(), fragmentTransform.a())));
387 } 413 }
388 414
389 return true; 415 return true;
390 } 416 }
391 417
392 float SVGTextQuery::rotationOfCharacter(unsigned position) const 418 float SVGTextQuery::rotationOfCharacter(unsigned position) const
393 { 419 {
394 RotationOfCharacterData data(position); 420 RotationOfCharacterData data(position);
395 executeQuery(m_queryRootLayoutObject, &data, rotationOfCharacterCallback); 421 logicalQuery(m_queryRootLayoutObject, &data, rotationOfCharacterCallback);
396 return data.rotation; 422 return data.rotation;
397 } 423 }
398 424
399 // extentOfCharacter() implementation 425 // extentOfCharacter() implementation
400 struct ExtentOfCharacterData : QueryData { 426 struct ExtentOfCharacterData : QueryData {
401 ExtentOfCharacterData(unsigned queryPosition) 427 ExtentOfCharacterData(unsigned queryPosition)
402 : position(queryPosition) 428 : position(queryPosition)
403 { 429 {
404 } 430 }
405 431
(...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after
477 if (!mapStartEndPositionsIntoFragmentCoordinates(queryData, fragment, startP osition, endPosition)) 503 if (!mapStartEndPositionsIntoFragmentCoordinates(queryData, fragment, startP osition, endPosition))
478 return false; 504 return false;
479 505
480 calculateGlyphBoundaries(queryData, fragment, startPosition, data->extent); 506 calculateGlyphBoundaries(queryData, fragment, startPosition, data->extent);
481 return true; 507 return true;
482 } 508 }
483 509
484 FloatRect SVGTextQuery::extentOfCharacter(unsigned position) const 510 FloatRect SVGTextQuery::extentOfCharacter(unsigned position) const
485 { 511 {
486 ExtentOfCharacterData data(position); 512 ExtentOfCharacterData data(position);
487 executeQuery(m_queryRootLayoutObject, &data, extentOfCharacterCallback); 513 logicalQuery(m_queryRootLayoutObject, &data, extentOfCharacterCallback);
488 return data.extent; 514 return data.extent;
489 } 515 }
490 516
491 // characterNumberAtPosition() implementation 517 // characterNumberAtPosition() implementation
492 struct CharacterNumberAtPositionData : QueryData { 518 struct CharacterNumberAtPositionData : QueryData {
493 CharacterNumberAtPositionData(const FloatPoint& queryPosition) 519 CharacterNumberAtPositionData(const FloatPoint& queryPosition)
494 : position(queryPosition) 520 : position(queryPosition)
521 , hitLayoutObject(nullptr)
522 , offsetInTextNode(0)
495 { 523 {
496 } 524 }
497 525
526 int characterNumberWithin(const LayoutObject* queryRoot) const;
527
498 FloatPoint position; 528 FloatPoint position;
529 LayoutObject* hitLayoutObject;
530 int offsetInTextNode;
pdr. 2015/04/16 02:34:58 (not for this patch) Should we be using unsigned l
fs 2015/04/16 14:22:21 I don't see how this particular value would overfl
499 }; 531 };
500 532
533 int CharacterNumberAtPositionData::characterNumberWithin(const LayoutObject* que ryRoot) const
534 {
535 if (!hitLayoutObject)
pdr. 2015/04/16 02:34:58 Can you add a comment here stating that this is pe
fs 2015/04/16 14:22:21 Done.
536 return -1;
537 ASSERT(queryRoot);
538 int characterNumber = offsetInTextNode;
539
540 // Accumulate the lengths of all the text nodes preceding the target layout
541 // object within the queried root, to get the complete character number.
542 for (const LayoutObject* layoutObject = hitLayoutObject->previousInPreOrder( queryRoot);
543 layoutObject; layoutObject = layoutObject->previousInPreOrder(queryRoot) ) {
544 if (!layoutObject->isSVGInlineText())
545 continue;
546 characterNumber += toLayoutSVGInlineText(layoutObject)->textLength();
547 }
548 return characterNumber;
549 }
550
501 static bool characterNumberAtPositionCallback(QueryData* queryData, const SVGTex tFragment& fragment) 551 static bool characterNumberAtPositionCallback(QueryData* queryData, const SVGTex tFragment& fragment)
502 { 552 {
503 CharacterNumberAtPositionData* data = static_cast<CharacterNumberAtPositionD ata*>(queryData); 553 CharacterNumberAtPositionData* data = static_cast<CharacterNumberAtPositionD ata*>(queryData);
504 554
505 // Test the query point against the bounds of the entire fragment first. 555 // Test the query point against the bounds of the entire fragment first.
506 FloatRect fragmentExtents = calculateFragmentBoundaries(*queryData->textLayo utObject, fragment); 556 FloatRect fragmentExtents = calculateFragmentBoundaries(*queryData->textLayo utObject, fragment);
507 if (!fragmentExtents.contains(data->position)) 557 if (!fragmentExtents.contains(data->position))
508 return false; 558 return false;
509 559
510 // Iterate through the glyphs in this fragment, and check if their extents 560 // Iterate through the glyphs in this fragment, and check if their extents
511 // contain the query point. 561 // contain the query point.
512 FloatRect extent; 562 FloatRect extent;
513 const Vector<SVGTextMetrics>& textMetrics = queryData->textLayoutObject->lay outAttributes()->textMetricsValues(); 563 const Vector<SVGTextMetrics>& textMetrics = queryData->textLayoutObject->lay outAttributes()->textMetricsValues();
514 unsigned textMetricsOffset = fragment.metricsListOffset; 564 unsigned textMetricsOffset = fragment.metricsListOffset;
515 unsigned fragmentOffset = 0; 565 unsigned fragmentOffset = 0;
516 while (fragmentOffset < fragment.length) { 566 while (fragmentOffset < fragment.length) {
517 calculateGlyphBoundaries(queryData, fragment, fragmentOffset, extent); 567 calculateGlyphBoundaries(queryData, fragment, fragmentOffset, extent);
518 if (extent.contains(data->position)) { 568 if (extent.contains(data->position)) {
519 // Compute the character offset of the glyph within the text box 569 // Compute the character offset of the glyph within the text node.
520 // and add to processedCharacters. 570 data->offsetInTextNode = fragment.characterOffset + fragmentOffset;
521 unsigned characterOffset = fragment.characterOffset + fragmentOffset ; 571 data->hitLayoutObject = data->textLayoutObject;
522 data->processedCharacters += characterOffset - data->textBox->start( );
523 return true; 572 return true;
524 } 573 }
525 fragmentOffset += textMetrics[textMetricsOffset].length(); 574 fragmentOffset += textMetrics[textMetricsOffset].length();
526 textMetricsOffset++; 575 textMetricsOffset++;
527 } 576 }
528 return false; 577 return false;
529 } 578 }
530 579
531 int SVGTextQuery::characterNumberAtPosition(const FloatPoint& position) const 580 int SVGTextQuery::characterNumberAtPosition(const FloatPoint& position) const
532 { 581 {
533 CharacterNumberAtPositionData data(position); 582 CharacterNumberAtPositionData data(position);
534 if (!executeQuery(m_queryRootLayoutObject, &data, characterNumberAtPositionC allback)) 583 spatialQuery(m_queryRootLayoutObject, &data, characterNumberAtPositionCallba ck);
535 return -1; 584 return data.characterNumberWithin(m_queryRootLayoutObject);
536
537 return data.processedCharacters;
538 } 585 }
539 586
540 } 587 }
OLDNEW
« LayoutTests/svg/text/bidi-getcharnumatpos.html ('K') | « LayoutTests/svg/text/bidi-text-query.svg ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698