OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 } |
OLD | NEW |