1 /* 2 * Copyright (C) 2006 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.text; 18 19 import android.emoji.EmojiFactory; 20 import android.graphics.Canvas; 21 import android.graphics.Paint; 22 import android.graphics.Path; 23 import android.graphics.Rect; 24 import android.text.method.TextKeyListener; 25 import android.text.style.AlignmentSpan; 26 import android.text.style.LeadingMarginSpan; 27 import android.text.style.LeadingMarginSpan.LeadingMarginSpan2; 28 import android.text.style.LineBackgroundSpan; 29 import android.text.style.ParagraphStyle; 30 import android.text.style.ReplacementSpan; 31 import android.text.style.TabStopSpan; 32 33 import com.android.internal.util.ArrayUtils; 34 import com.android.internal.util.GrowingArrayUtils; 35 36 import java.util.Arrays; 37 38 /** 39 * A base class that manages text layout in visual elements on 40 * the screen. 41 * <p>For text that will be edited, use a {@link DynamicLayout}, 42 * which will be updated as the text changes. 43 * For text that will not change, use a {@link StaticLayout}. 44 */ 45 public abstract class Layout { 46 private static final ParagraphStyle[] NO_PARA_SPANS = 47 ArrayUtils.emptyArray(ParagraphStyle.class); 48 49 /* package */ static final EmojiFactory EMOJI_FACTORY = EmojiFactory.newAvailableInstance(); 50 /* package */ static final int MIN_EMOJI, MAX_EMOJI; 51 52 static { 53 if (EMOJI_FACTORY != null) { 54 MIN_EMOJI = EMOJI_FACTORY.getMinimumAndroidPua(); 55 MAX_EMOJI = EMOJI_FACTORY.getMaximumAndroidPua(); 56 } else { 57 MIN_EMOJI = -1; 58 MAX_EMOJI = -1; 59 } 60 } 61 62 /** 63 * Return how wide a layout must be in order to display the 64 * specified text with one line per paragraph. 65 */ getDesiredWidth(CharSequence source, TextPaint paint)66 public static float getDesiredWidth(CharSequence source, 67 TextPaint paint) { 68 return getDesiredWidth(source, 0, source.length(), paint); 69 } 70 71 /** 72 * Return how wide a layout must be in order to display the 73 * specified text slice with one line per paragraph. 74 */ getDesiredWidth(CharSequence source, int start, int end, TextPaint paint)75 public static float getDesiredWidth(CharSequence source, 76 int start, int end, 77 TextPaint paint) { 78 float need = 0; 79 80 int next; 81 for (int i = start; i <= end; i = next) { 82 next = TextUtils.indexOf(source, '\n', i, end); 83 84 if (next < 0) 85 next = end; 86 87 // note, omits trailing paragraph char 88 float w = measurePara(paint, source, i, next); 89 90 if (w > need) 91 need = w; 92 93 next++; 94 } 95 96 return need; 97 } 98 99 /** 100 * Subclasses of Layout use this constructor to set the display text, 101 * width, and other standard properties. 102 * @param text the text to render 103 * @param paint the default paint for the layout. Styles can override 104 * various attributes of the paint. 105 * @param width the wrapping width for the text. 106 * @param align whether to left, right, or center the text. Styles can 107 * override the alignment. 108 * @param spacingMult factor by which to scale the font size to get the 109 * default line spacing 110 * @param spacingAdd amount to add to the default line spacing 111 */ Layout(CharSequence text, TextPaint paint, int width, Alignment align, float spacingMult, float spacingAdd)112 protected Layout(CharSequence text, TextPaint paint, 113 int width, Alignment align, 114 float spacingMult, float spacingAdd) { 115 this(text, paint, width, align, TextDirectionHeuristics.FIRSTSTRONG_LTR, 116 spacingMult, spacingAdd); 117 } 118 119 /** 120 * Subclasses of Layout use this constructor to set the display text, 121 * width, and other standard properties. 122 * @param text the text to render 123 * @param paint the default paint for the layout. Styles can override 124 * various attributes of the paint. 125 * @param width the wrapping width for the text. 126 * @param align whether to left, right, or center the text. Styles can 127 * override the alignment. 128 * @param spacingMult factor by which to scale the font size to get the 129 * default line spacing 130 * @param spacingAdd amount to add to the default line spacing 131 * 132 * @hide 133 */ Layout(CharSequence text, TextPaint paint, int width, Alignment align, TextDirectionHeuristic textDir, float spacingMult, float spacingAdd)134 protected Layout(CharSequence text, TextPaint paint, 135 int width, Alignment align, TextDirectionHeuristic textDir, 136 float spacingMult, float spacingAdd) { 137 138 if (width < 0) 139 throw new IllegalArgumentException("Layout: " + width + " < 0"); 140 141 // Ensure paint doesn't have baselineShift set. 142 // While normally we don't modify the paint the user passed in, 143 // we were already doing this in Styled.drawUniformRun with both 144 // baselineShift and bgColor. We probably should reevaluate bgColor. 145 if (paint != null) { 146 paint.bgColor = 0; 147 paint.baselineShift = 0; 148 } 149 150 mText = text; 151 mPaint = paint; 152 mWorkPaint = new TextPaint(); 153 mWidth = width; 154 mAlignment = align; 155 mSpacingMult = spacingMult; 156 mSpacingAdd = spacingAdd; 157 mSpannedText = text instanceof Spanned; 158 mTextDir = textDir; 159 } 160 161 /** 162 * Replace constructor properties of this Layout with new ones. Be careful. 163 */ replaceWith(CharSequence text, TextPaint paint, int width, Alignment align, float spacingmult, float spacingadd)164 /* package */ void replaceWith(CharSequence text, TextPaint paint, 165 int width, Alignment align, 166 float spacingmult, float spacingadd) { 167 if (width < 0) { 168 throw new IllegalArgumentException("Layout: " + width + " < 0"); 169 } 170 171 mText = text; 172 mPaint = paint; 173 mWidth = width; 174 mAlignment = align; 175 mSpacingMult = spacingmult; 176 mSpacingAdd = spacingadd; 177 mSpannedText = text instanceof Spanned; 178 } 179 180 /** 181 * Draw this Layout on the specified Canvas. 182 */ draw(Canvas c)183 public void draw(Canvas c) { 184 draw(c, null, null, 0); 185 } 186 187 /** 188 * Draw this Layout on the specified canvas, with the highlight path drawn 189 * between the background and the text. 190 * 191 * @param canvas the canvas 192 * @param highlight the path of the highlight or cursor; can be null 193 * @param highlightPaint the paint for the highlight 194 * @param cursorOffsetVertical the amount to temporarily translate the 195 * canvas while rendering the highlight 196 */ draw(Canvas canvas, Path highlight, Paint highlightPaint, int cursorOffsetVertical)197 public void draw(Canvas canvas, Path highlight, Paint highlightPaint, 198 int cursorOffsetVertical) { 199 final long lineRange = getLineRangeForDraw(canvas); 200 int firstLine = TextUtils.unpackRangeStartFromLong(lineRange); 201 int lastLine = TextUtils.unpackRangeEndFromLong(lineRange); 202 if (lastLine < 0) return; 203 204 drawBackground(canvas, highlight, highlightPaint, cursorOffsetVertical, 205 firstLine, lastLine); 206 drawText(canvas, firstLine, lastLine); 207 } 208 209 /** 210 * @hide 211 */ drawText(Canvas canvas, int firstLine, int lastLine)212 public void drawText(Canvas canvas, int firstLine, int lastLine) { 213 int previousLineBottom = getLineTop(firstLine); 214 int previousLineEnd = getLineStart(firstLine); 215 ParagraphStyle[] spans = NO_PARA_SPANS; 216 int spanEnd = 0; 217 TextPaint paint = mPaint; 218 CharSequence buf = mText; 219 220 Alignment paraAlign = mAlignment; 221 TabStops tabStops = null; 222 boolean tabStopsIsInitialized = false; 223 224 TextLine tl = TextLine.obtain(); 225 226 // Draw the lines, one at a time. 227 // The baseline is the top of the following line minus the current line's descent. 228 for (int i = firstLine; i <= lastLine; i++) { 229 int start = previousLineEnd; 230 previousLineEnd = getLineStart(i + 1); 231 int end = getLineVisibleEnd(i, start, previousLineEnd); 232 233 int ltop = previousLineBottom; 234 int lbottom = getLineTop(i+1); 235 previousLineBottom = lbottom; 236 int lbaseline = lbottom - getLineDescent(i); 237 238 int dir = getParagraphDirection(i); 239 int left = 0; 240 int right = mWidth; 241 242 if (mSpannedText) { 243 Spanned sp = (Spanned) buf; 244 int textLength = buf.length(); 245 boolean isFirstParaLine = (start == 0 || buf.charAt(start - 1) == '\n'); 246 247 // New batch of paragraph styles, collect into spans array. 248 // Compute the alignment, last alignment style wins. 249 // Reset tabStops, we'll rebuild if we encounter a line with 250 // tabs. 251 // We expect paragraph spans to be relatively infrequent, use 252 // spanEnd so that we can check less frequently. Since 253 // paragraph styles ought to apply to entire paragraphs, we can 254 // just collect the ones present at the start of the paragraph. 255 // If spanEnd is before the end of the paragraph, that's not 256 // our problem. 257 if (start >= spanEnd && (i == firstLine || isFirstParaLine)) { 258 spanEnd = sp.nextSpanTransition(start, textLength, 259 ParagraphStyle.class); 260 spans = getParagraphSpans(sp, start, spanEnd, ParagraphStyle.class); 261 262 paraAlign = mAlignment; 263 for (int n = spans.length - 1; n >= 0; n--) { 264 if (spans[n] instanceof AlignmentSpan) { 265 paraAlign = ((AlignmentSpan) spans[n]).getAlignment(); 266 break; 267 } 268 } 269 270 tabStopsIsInitialized = false; 271 } 272 273 // Draw all leading margin spans. Adjust left or right according 274 // to the paragraph direction of the line. 275 final int length = spans.length; 276 boolean useFirstLineMargin = isFirstParaLine; 277 for (int n = 0; n < length; n++) { 278 if (spans[n] instanceof LeadingMarginSpan2) { 279 int count = ((LeadingMarginSpan2) spans[n]).getLeadingMarginLineCount(); 280 int startLine = getLineForOffset(sp.getSpanStart(spans[n])); 281 // if there is more than one LeadingMarginSpan2, use 282 // the count that is greatest 283 if (i < startLine + count) { 284 useFirstLineMargin = true; 285 break; 286 } 287 } 288 } 289 for (int n = 0; n < length; n++) { 290 if (spans[n] instanceof LeadingMarginSpan) { 291 LeadingMarginSpan margin = (LeadingMarginSpan) spans[n]; 292 if (dir == DIR_RIGHT_TO_LEFT) { 293 margin.drawLeadingMargin(canvas, paint, right, dir, ltop, 294 lbaseline, lbottom, buf, 295 start, end, isFirstParaLine, this); 296 right -= margin.getLeadingMargin(useFirstLineMargin); 297 } else { 298 margin.drawLeadingMargin(canvas, paint, left, dir, ltop, 299 lbaseline, lbottom, buf, 300 start, end, isFirstParaLine, this); 301 left += margin.getLeadingMargin(useFirstLineMargin); 302 } 303 } 304 } 305 } 306 307 boolean hasTabOrEmoji = getLineContainsTab(i); 308 // Can't tell if we have tabs for sure, currently 309 if (hasTabOrEmoji && !tabStopsIsInitialized) { 310 if (tabStops == null) { 311 tabStops = new TabStops(TAB_INCREMENT, spans); 312 } else { 313 tabStops.reset(TAB_INCREMENT, spans); 314 } 315 tabStopsIsInitialized = true; 316 } 317 318 // Determine whether the line aligns to normal, opposite, or center. 319 Alignment align = paraAlign; 320 if (align == Alignment.ALIGN_LEFT) { 321 align = (dir == DIR_LEFT_TO_RIGHT) ? 322 Alignment.ALIGN_NORMAL : Alignment.ALIGN_OPPOSITE; 323 } else if (align == Alignment.ALIGN_RIGHT) { 324 align = (dir == DIR_LEFT_TO_RIGHT) ? 325 Alignment.ALIGN_OPPOSITE : Alignment.ALIGN_NORMAL; 326 } 327 328 int x; 329 if (align == Alignment.ALIGN_NORMAL) { 330 if (dir == DIR_LEFT_TO_RIGHT) { 331 x = left; 332 } else { 333 x = right; 334 } 335 } else { 336 int max = (int)getLineExtent(i, tabStops, false); 337 if (align == Alignment.ALIGN_OPPOSITE) { 338 if (dir == DIR_LEFT_TO_RIGHT) { 339 x = right - max; 340 } else { 341 x = left - max; 342 } 343 } else { // Alignment.ALIGN_CENTER 344 max = max & ~1; 345 x = (right + left - max) >> 1; 346 } 347 } 348 349 Directions directions = getLineDirections(i); 350 if (directions == DIRS_ALL_LEFT_TO_RIGHT && !mSpannedText && !hasTabOrEmoji) { 351 // XXX: assumes there's nothing additional to be done 352 canvas.drawText(buf, start, end, x, lbaseline, paint); 353 } else { 354 tl.set(paint, buf, start, end, dir, directions, hasTabOrEmoji, tabStops); 355 tl.draw(canvas, x, ltop, lbaseline, lbottom); 356 } 357 } 358 359 TextLine.recycle(tl); 360 } 361 362 /** 363 * @hide 364 */ drawBackground(Canvas canvas, Path highlight, Paint highlightPaint, int cursorOffsetVertical, int firstLine, int lastLine)365 public void drawBackground(Canvas canvas, Path highlight, Paint highlightPaint, 366 int cursorOffsetVertical, int firstLine, int lastLine) { 367 // First, draw LineBackgroundSpans. 368 // LineBackgroundSpans know nothing about the alignment, margins, or 369 // direction of the layout or line. XXX: Should they? 370 // They are evaluated at each line. 371 if (mSpannedText) { 372 if (mLineBackgroundSpans == null) { 373 mLineBackgroundSpans = new SpanSet<LineBackgroundSpan>(LineBackgroundSpan.class); 374 } 375 376 Spanned buffer = (Spanned) mText; 377 int textLength = buffer.length(); 378 mLineBackgroundSpans.init(buffer, 0, textLength); 379 380 if (mLineBackgroundSpans.numberOfSpans > 0) { 381 int previousLineBottom = getLineTop(firstLine); 382 int previousLineEnd = getLineStart(firstLine); 383 ParagraphStyle[] spans = NO_PARA_SPANS; 384 int spansLength = 0; 385 TextPaint paint = mPaint; 386 int spanEnd = 0; 387 final int width = mWidth; 388 for (int i = firstLine; i <= lastLine; i++) { 389 int start = previousLineEnd; 390 int end = getLineStart(i + 1); 391 previousLineEnd = end; 392 393 int ltop = previousLineBottom; 394 int lbottom = getLineTop(i + 1); 395 previousLineBottom = lbottom; 396 int lbaseline = lbottom - getLineDescent(i); 397 398 if (start >= spanEnd) { 399 // These should be infrequent, so we'll use this so that 400 // we don't have to check as often. 401 spanEnd = mLineBackgroundSpans.getNextTransition(start, textLength); 402 // All LineBackgroundSpans on a line contribute to its background. 403 spansLength = 0; 404 // Duplication of the logic of getParagraphSpans 405 if (start != end || start == 0) { 406 // Equivalent to a getSpans(start, end), but filling the 'spans' local 407 // array instead to reduce memory allocation 408 for (int j = 0; j < mLineBackgroundSpans.numberOfSpans; j++) { 409 // equal test is valid since both intervals are not empty by 410 // construction 411 if (mLineBackgroundSpans.spanStarts[j] >= end || 412 mLineBackgroundSpans.spanEnds[j] <= start) continue; 413 spans = GrowingArrayUtils.append( 414 spans, spansLength, mLineBackgroundSpans.spans[j]); 415 spansLength++; 416 } 417 } 418 } 419 420 for (int n = 0; n < spansLength; n++) { 421 LineBackgroundSpan lineBackgroundSpan = (LineBackgroundSpan) spans[n]; 422 lineBackgroundSpan.drawBackground(canvas, paint, 0, width, 423 ltop, lbaseline, lbottom, 424 buffer, start, end, i); 425 } 426 } 427 } 428 mLineBackgroundSpans.recycle(); 429 } 430 431 // There can be a highlight even without spans if we are drawing 432 // a non-spanned transformation of a spanned editing buffer. 433 if (highlight != null) { 434 if (cursorOffsetVertical != 0) canvas.translate(0, cursorOffsetVertical); 435 canvas.drawPath(highlight, highlightPaint); 436 if (cursorOffsetVertical != 0) canvas.translate(0, -cursorOffsetVertical); 437 } 438 } 439 440 /** 441 * @param canvas 442 * @return The range of lines that need to be drawn, possibly empty. 443 * @hide 444 */ getLineRangeForDraw(Canvas canvas)445 public long getLineRangeForDraw(Canvas canvas) { 446 int dtop, dbottom; 447 448 synchronized (sTempRect) { 449 if (!canvas.getClipBounds(sTempRect)) { 450 // Negative range end used as a special flag 451 return TextUtils.packRangeInLong(0, -1); 452 } 453 454 dtop = sTempRect.top; 455 dbottom = sTempRect.bottom; 456 } 457 458 final int top = Math.max(dtop, 0); 459 final int bottom = Math.min(getLineTop(getLineCount()), dbottom); 460 461 if (top >= bottom) return TextUtils.packRangeInLong(0, -1); 462 return TextUtils.packRangeInLong(getLineForVertical(top), getLineForVertical(bottom)); 463 } 464 465 /** 466 * Return the start position of the line, given the left and right bounds 467 * of the margins. 468 * 469 * @param line the line index 470 * @param left the left bounds (0, or leading margin if ltr para) 471 * @param right the right bounds (width, minus leading margin if rtl para) 472 * @return the start position of the line (to right of line if rtl para) 473 */ getLineStartPos(int line, int left, int right)474 private int getLineStartPos(int line, int left, int right) { 475 // Adjust the point at which to start rendering depending on the 476 // alignment of the paragraph. 477 Alignment align = getParagraphAlignment(line); 478 int dir = getParagraphDirection(line); 479 480 if (align == Alignment.ALIGN_LEFT) { 481 align = (dir == DIR_LEFT_TO_RIGHT) ? Alignment.ALIGN_NORMAL : Alignment.ALIGN_OPPOSITE; 482 } else if (align == Alignment.ALIGN_RIGHT) { 483 align = (dir == DIR_LEFT_TO_RIGHT) ? Alignment.ALIGN_OPPOSITE : Alignment.ALIGN_NORMAL; 484 } 485 486 int x; 487 if (align == Alignment.ALIGN_NORMAL) { 488 if (dir == DIR_LEFT_TO_RIGHT) { 489 x = left; 490 } else { 491 x = right; 492 } 493 } else { 494 TabStops tabStops = null; 495 if (mSpannedText && getLineContainsTab(line)) { 496 Spanned spanned = (Spanned) mText; 497 int start = getLineStart(line); 498 int spanEnd = spanned.nextSpanTransition(start, spanned.length(), 499 TabStopSpan.class); 500 TabStopSpan[] tabSpans = getParagraphSpans(spanned, start, spanEnd, 501 TabStopSpan.class); 502 if (tabSpans.length > 0) { 503 tabStops = new TabStops(TAB_INCREMENT, tabSpans); 504 } 505 } 506 int max = (int)getLineExtent(line, tabStops, false); 507 if (align == Alignment.ALIGN_OPPOSITE) { 508 if (dir == DIR_LEFT_TO_RIGHT) { 509 x = right - max; 510 } else { 511 // max is negative here 512 x = left - max; 513 } 514 } else { // Alignment.ALIGN_CENTER 515 max = max & ~1; 516 x = (left + right - max) >> 1; 517 } 518 } 519 return x; 520 } 521 522 /** 523 * Return the text that is displayed by this Layout. 524 */ getText()525 public final CharSequence getText() { 526 return mText; 527 } 528 529 /** 530 * Return the base Paint properties for this layout. 531 * Do NOT change the paint, which may result in funny 532 * drawing for this layout. 533 */ getPaint()534 public final TextPaint getPaint() { 535 return mPaint; 536 } 537 538 /** 539 * Return the width of this layout. 540 */ getWidth()541 public final int getWidth() { 542 return mWidth; 543 } 544 545 /** 546 * Return the width to which this Layout is ellipsizing, or 547 * {@link #getWidth} if it is not doing anything special. 548 */ getEllipsizedWidth()549 public int getEllipsizedWidth() { 550 return mWidth; 551 } 552 553 /** 554 * Increase the width of this layout to the specified width. 555 * Be careful to use this only when you know it is appropriate— 556 * it does not cause the text to reflow to use the full new width. 557 */ increaseWidthTo(int wid)558 public final void increaseWidthTo(int wid) { 559 if (wid < mWidth) { 560 throw new RuntimeException("attempted to reduce Layout width"); 561 } 562 563 mWidth = wid; 564 } 565 566 /** 567 * Return the total height of this layout. 568 */ getHeight()569 public int getHeight() { 570 return getLineTop(getLineCount()); 571 } 572 573 /** 574 * Return the base alignment of this layout. 575 */ getAlignment()576 public final Alignment getAlignment() { 577 return mAlignment; 578 } 579 580 /** 581 * Return what the text height is multiplied by to get the line height. 582 */ getSpacingMultiplier()583 public final float getSpacingMultiplier() { 584 return mSpacingMult; 585 } 586 587 /** 588 * Return the number of units of leading that are added to each line. 589 */ getSpacingAdd()590 public final float getSpacingAdd() { 591 return mSpacingAdd; 592 } 593 594 /** 595 * Return the heuristic used to determine paragraph text direction. 596 * @hide 597 */ getTextDirectionHeuristic()598 public final TextDirectionHeuristic getTextDirectionHeuristic() { 599 return mTextDir; 600 } 601 602 /** 603 * Return the number of lines of text in this layout. 604 */ getLineCount()605 public abstract int getLineCount(); 606 607 /** 608 * Return the baseline for the specified line (0…getLineCount() - 1) 609 * If bounds is not null, return the top, left, right, bottom extents 610 * of the specified line in it. 611 * @param line which line to examine (0..getLineCount() - 1) 612 * @param bounds Optional. If not null, it returns the extent of the line 613 * @return the Y-coordinate of the baseline 614 */ getLineBounds(int line, Rect bounds)615 public int getLineBounds(int line, Rect bounds) { 616 if (bounds != null) { 617 bounds.left = 0; // ??? 618 bounds.top = getLineTop(line); 619 bounds.right = mWidth; // ??? 620 bounds.bottom = getLineTop(line + 1); 621 } 622 return getLineBaseline(line); 623 } 624 625 /** 626 * Return the vertical position of the top of the specified line 627 * (0…getLineCount()). 628 * If the specified line is equal to the line count, returns the 629 * bottom of the last line. 630 */ getLineTop(int line)631 public abstract int getLineTop(int line); 632 633 /** 634 * Return the descent of the specified line(0…getLineCount() - 1). 635 */ getLineDescent(int line)636 public abstract int getLineDescent(int line); 637 638 /** 639 * Return the text offset of the beginning of the specified line ( 640 * 0…getLineCount()). If the specified line is equal to the line 641 * count, returns the length of the text. 642 */ getLineStart(int line)643 public abstract int getLineStart(int line); 644 645 /** 646 * Returns the primary directionality of the paragraph containing the 647 * specified line, either 1 for left-to-right lines, or -1 for right-to-left 648 * lines (see {@link #DIR_LEFT_TO_RIGHT}, {@link #DIR_RIGHT_TO_LEFT}). 649 */ getParagraphDirection(int line)650 public abstract int getParagraphDirection(int line); 651 652 /** 653 * Returns whether the specified line contains one or more 654 * characters that need to be handled specially, like tabs 655 * or emoji. 656 */ getLineContainsTab(int line)657 public abstract boolean getLineContainsTab(int line); 658 659 /** 660 * Returns the directional run information for the specified line. 661 * The array alternates counts of characters in left-to-right 662 * and right-to-left segments of the line. 663 * 664 * <p>NOTE: this is inadequate to support bidirectional text, and will change. 665 */ getLineDirections(int line)666 public abstract Directions getLineDirections(int line); 667 668 /** 669 * Returns the (negative) number of extra pixels of ascent padding in the 670 * top line of the Layout. 671 */ getTopPadding()672 public abstract int getTopPadding(); 673 674 /** 675 * Returns the number of extra pixels of descent padding in the 676 * bottom line of the Layout. 677 */ getBottomPadding()678 public abstract int getBottomPadding(); 679 680 681 /** 682 * Returns true if the character at offset and the preceding character 683 * are at different run levels (and thus there's a split caret). 684 * @param offset the offset 685 * @return true if at a level boundary 686 * @hide 687 */ isLevelBoundary(int offset)688 public boolean isLevelBoundary(int offset) { 689 int line = getLineForOffset(offset); 690 Directions dirs = getLineDirections(line); 691 if (dirs == DIRS_ALL_LEFT_TO_RIGHT || dirs == DIRS_ALL_RIGHT_TO_LEFT) { 692 return false; 693 } 694 695 int[] runs = dirs.mDirections; 696 int lineStart = getLineStart(line); 697 int lineEnd = getLineEnd(line); 698 if (offset == lineStart || offset == lineEnd) { 699 int paraLevel = getParagraphDirection(line) == 1 ? 0 : 1; 700 int runIndex = offset == lineStart ? 0 : runs.length - 2; 701 return ((runs[runIndex + 1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK) != paraLevel; 702 } 703 704 offset -= lineStart; 705 for (int i = 0; i < runs.length; i += 2) { 706 if (offset == runs[i]) { 707 return true; 708 } 709 } 710 return false; 711 } 712 713 /** 714 * Returns true if the character at offset is right to left (RTL). 715 * @param offset the offset 716 * @return true if the character is RTL, false if it is LTR 717 */ isRtlCharAt(int offset)718 public boolean isRtlCharAt(int offset) { 719 int line = getLineForOffset(offset); 720 Directions dirs = getLineDirections(line); 721 if (dirs == DIRS_ALL_LEFT_TO_RIGHT) { 722 return false; 723 } 724 if (dirs == DIRS_ALL_RIGHT_TO_LEFT) { 725 return true; 726 } 727 int[] runs = dirs.mDirections; 728 int lineStart = getLineStart(line); 729 for (int i = 0; i < runs.length; i += 2) { 730 int start = lineStart + runs[i]; 731 int limit = start + (runs[i+1] & RUN_LENGTH_MASK); 732 if (offset >= start && offset < limit) { 733 int level = (runs[i+1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK; 734 return ((level & 1) != 0); 735 } 736 } 737 // Should happen only if the offset is "out of bounds" 738 return false; 739 } 740 primaryIsTrailingPrevious(int offset)741 private boolean primaryIsTrailingPrevious(int offset) { 742 int line = getLineForOffset(offset); 743 int lineStart = getLineStart(line); 744 int lineEnd = getLineEnd(line); 745 int[] runs = getLineDirections(line).mDirections; 746 747 int levelAt = -1; 748 for (int i = 0; i < runs.length; i += 2) { 749 int start = lineStart + runs[i]; 750 int limit = start + (runs[i+1] & RUN_LENGTH_MASK); 751 if (limit > lineEnd) { 752 limit = lineEnd; 753 } 754 if (offset >= start && offset < limit) { 755 if (offset > start) { 756 // Previous character is at same level, so don't use trailing. 757 return false; 758 } 759 levelAt = (runs[i+1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK; 760 break; 761 } 762 } 763 if (levelAt == -1) { 764 // Offset was limit of line. 765 levelAt = getParagraphDirection(line) == 1 ? 0 : 1; 766 } 767 768 // At level boundary, check previous level. 769 int levelBefore = -1; 770 if (offset == lineStart) { 771 levelBefore = getParagraphDirection(line) == 1 ? 0 : 1; 772 } else { 773 offset -= 1; 774 for (int i = 0; i < runs.length; i += 2) { 775 int start = lineStart + runs[i]; 776 int limit = start + (runs[i+1] & RUN_LENGTH_MASK); 777 if (limit > lineEnd) { 778 limit = lineEnd; 779 } 780 if (offset >= start && offset < limit) { 781 levelBefore = (runs[i+1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK; 782 break; 783 } 784 } 785 } 786 787 return levelBefore < levelAt; 788 } 789 790 /** 791 * Get the primary horizontal position for the specified text offset. 792 * This is the location where a new character would be inserted in 793 * the paragraph's primary direction. 794 */ getPrimaryHorizontal(int offset)795 public float getPrimaryHorizontal(int offset) { 796 return getPrimaryHorizontal(offset, false /* not clamped */); 797 } 798 799 /** 800 * Get the primary horizontal position for the specified text offset, but 801 * optionally clamp it so that it doesn't exceed the width of the layout. 802 * @hide 803 */ getPrimaryHorizontal(int offset, boolean clamped)804 public float getPrimaryHorizontal(int offset, boolean clamped) { 805 boolean trailing = primaryIsTrailingPrevious(offset); 806 return getHorizontal(offset, trailing, clamped); 807 } 808 809 /** 810 * Get the secondary horizontal position for the specified text offset. 811 * This is the location where a new character would be inserted in 812 * the direction other than the paragraph's primary direction. 813 */ getSecondaryHorizontal(int offset)814 public float getSecondaryHorizontal(int offset) { 815 return getSecondaryHorizontal(offset, false /* not clamped */); 816 } 817 818 /** 819 * Get the secondary horizontal position for the specified text offset, but 820 * optionally clamp it so that it doesn't exceed the width of the layout. 821 * @hide 822 */ getSecondaryHorizontal(int offset, boolean clamped)823 public float getSecondaryHorizontal(int offset, boolean clamped) { 824 boolean trailing = primaryIsTrailingPrevious(offset); 825 return getHorizontal(offset, !trailing, clamped); 826 } 827 getHorizontal(int offset, boolean trailing, boolean clamped)828 private float getHorizontal(int offset, boolean trailing, boolean clamped) { 829 int line = getLineForOffset(offset); 830 831 return getHorizontal(offset, trailing, line, clamped); 832 } 833 getHorizontal(int offset, boolean trailing, int line, boolean clamped)834 private float getHorizontal(int offset, boolean trailing, int line, boolean clamped) { 835 int start = getLineStart(line); 836 int end = getLineEnd(line); 837 int dir = getParagraphDirection(line); 838 boolean hasTabOrEmoji = getLineContainsTab(line); 839 Directions directions = getLineDirections(line); 840 841 TabStops tabStops = null; 842 if (hasTabOrEmoji && mText instanceof Spanned) { 843 // Just checking this line should be good enough, tabs should be 844 // consistent across all lines in a paragraph. 845 TabStopSpan[] tabs = getParagraphSpans((Spanned) mText, start, end, TabStopSpan.class); 846 if (tabs.length > 0) { 847 tabStops = new TabStops(TAB_INCREMENT, tabs); // XXX should reuse 848 } 849 } 850 851 TextLine tl = TextLine.obtain(); 852 tl.set(mPaint, mText, start, end, dir, directions, hasTabOrEmoji, tabStops); 853 float wid = tl.measure(offset - start, trailing, null); 854 TextLine.recycle(tl); 855 856 if (clamped && wid > mWidth) { 857 wid = mWidth; 858 } 859 int left = getParagraphLeft(line); 860 int right = getParagraphRight(line); 861 862 return getLineStartPos(line, left, right) + wid; 863 } 864 865 /** 866 * Get the leftmost position that should be exposed for horizontal 867 * scrolling on the specified line. 868 */ getLineLeft(int line)869 public float getLineLeft(int line) { 870 int dir = getParagraphDirection(line); 871 Alignment align = getParagraphAlignment(line); 872 873 if (align == Alignment.ALIGN_LEFT) { 874 return 0; 875 } else if (align == Alignment.ALIGN_NORMAL) { 876 if (dir == DIR_RIGHT_TO_LEFT) 877 return getParagraphRight(line) - getLineMax(line); 878 else 879 return 0; 880 } else if (align == Alignment.ALIGN_RIGHT) { 881 return mWidth - getLineMax(line); 882 } else if (align == Alignment.ALIGN_OPPOSITE) { 883 if (dir == DIR_RIGHT_TO_LEFT) 884 return 0; 885 else 886 return mWidth - getLineMax(line); 887 } else { /* align == Alignment.ALIGN_CENTER */ 888 int left = getParagraphLeft(line); 889 int right = getParagraphRight(line); 890 int max = ((int) getLineMax(line)) & ~1; 891 892 return left + ((right - left) - max) / 2; 893 } 894 } 895 896 /** 897 * Get the rightmost position that should be exposed for horizontal 898 * scrolling on the specified line. 899 */ getLineRight(int line)900 public float getLineRight(int line) { 901 int dir = getParagraphDirection(line); 902 Alignment align = getParagraphAlignment(line); 903 904 if (align == Alignment.ALIGN_LEFT) { 905 return getParagraphLeft(line) + getLineMax(line); 906 } else if (align == Alignment.ALIGN_NORMAL) { 907 if (dir == DIR_RIGHT_TO_LEFT) 908 return mWidth; 909 else 910 return getParagraphLeft(line) + getLineMax(line); 911 } else if (align == Alignment.ALIGN_RIGHT) { 912 return mWidth; 913 } else if (align == Alignment.ALIGN_OPPOSITE) { 914 if (dir == DIR_RIGHT_TO_LEFT) 915 return getLineMax(line); 916 else 917 return mWidth; 918 } else { /* align == Alignment.ALIGN_CENTER */ 919 int left = getParagraphLeft(line); 920 int right = getParagraphRight(line); 921 int max = ((int) getLineMax(line)) & ~1; 922 923 return right - ((right - left) - max) / 2; 924 } 925 } 926 927 /** 928 * Gets the unsigned horizontal extent of the specified line, including 929 * leading margin indent, but excluding trailing whitespace. 930 */ getLineMax(int line)931 public float getLineMax(int line) { 932 float margin = getParagraphLeadingMargin(line); 933 float signedExtent = getLineExtent(line, false); 934 return margin + (signedExtent >= 0 ? signedExtent : -signedExtent); 935 } 936 937 /** 938 * Gets the unsigned horizontal extent of the specified line, including 939 * leading margin indent and trailing whitespace. 940 */ getLineWidth(int line)941 public float getLineWidth(int line) { 942 float margin = getParagraphLeadingMargin(line); 943 float signedExtent = getLineExtent(line, true); 944 return margin + (signedExtent >= 0 ? signedExtent : -signedExtent); 945 } 946 947 /** 948 * Like {@link #getLineExtent(int,TabStops,boolean)} but determines the 949 * tab stops instead of using the ones passed in. 950 * @param line the index of the line 951 * @param full whether to include trailing whitespace 952 * @return the extent of the line 953 */ getLineExtent(int line, boolean full)954 private float getLineExtent(int line, boolean full) { 955 int start = getLineStart(line); 956 int end = full ? getLineEnd(line) : getLineVisibleEnd(line); 957 958 boolean hasTabsOrEmoji = getLineContainsTab(line); 959 TabStops tabStops = null; 960 if (hasTabsOrEmoji && mText instanceof Spanned) { 961 // Just checking this line should be good enough, tabs should be 962 // consistent across all lines in a paragraph. 963 TabStopSpan[] tabs = getParagraphSpans((Spanned) mText, start, end, TabStopSpan.class); 964 if (tabs.length > 0) { 965 tabStops = new TabStops(TAB_INCREMENT, tabs); // XXX should reuse 966 } 967 } 968 Directions directions = getLineDirections(line); 969 // Returned directions can actually be null 970 if (directions == null) { 971 return 0f; 972 } 973 int dir = getParagraphDirection(line); 974 975 TextLine tl = TextLine.obtain(); 976 tl.set(mPaint, mText, start, end, dir, directions, hasTabsOrEmoji, tabStops); 977 float width = tl.metrics(null); 978 TextLine.recycle(tl); 979 return width; 980 } 981 982 /** 983 * Returns the signed horizontal extent of the specified line, excluding 984 * leading margin. If full is false, excludes trailing whitespace. 985 * @param line the index of the line 986 * @param tabStops the tab stops, can be null if we know they're not used. 987 * @param full whether to include trailing whitespace 988 * @return the extent of the text on this line 989 */ getLineExtent(int line, TabStops tabStops, boolean full)990 private float getLineExtent(int line, TabStops tabStops, boolean full) { 991 int start = getLineStart(line); 992 int end = full ? getLineEnd(line) : getLineVisibleEnd(line); 993 boolean hasTabsOrEmoji = getLineContainsTab(line); 994 Directions directions = getLineDirections(line); 995 int dir = getParagraphDirection(line); 996 997 TextLine tl = TextLine.obtain(); 998 tl.set(mPaint, mText, start, end, dir, directions, hasTabsOrEmoji, tabStops); 999 float width = tl.metrics(null); 1000 TextLine.recycle(tl); 1001 return width; 1002 } 1003 1004 /** 1005 * Get the line number corresponding to the specified vertical position. 1006 * If you ask for a position above 0, you get 0; if you ask for a position 1007 * below the bottom of the text, you get the last line. 1008 */ 1009 // FIXME: It may be faster to do a linear search for layouts without many lines. getLineForVertical(int vertical)1010 public int getLineForVertical(int vertical) { 1011 int high = getLineCount(), low = -1, guess; 1012 1013 while (high - low > 1) { 1014 guess = (high + low) / 2; 1015 1016 if (getLineTop(guess) > vertical) 1017 high = guess; 1018 else 1019 low = guess; 1020 } 1021 1022 if (low < 0) 1023 return 0; 1024 else 1025 return low; 1026 } 1027 1028 /** 1029 * Get the line number on which the specified text offset appears. 1030 * If you ask for a position before 0, you get 0; if you ask for a position 1031 * beyond the end of the text, you get the last line. 1032 */ getLineForOffset(int offset)1033 public int getLineForOffset(int offset) { 1034 int high = getLineCount(), low = -1, guess; 1035 1036 while (high - low > 1) { 1037 guess = (high + low) / 2; 1038 1039 if (getLineStart(guess) > offset) 1040 high = guess; 1041 else 1042 low = guess; 1043 } 1044 1045 if (low < 0) 1046 return 0; 1047 else 1048 return low; 1049 } 1050 1051 /** 1052 * Get the character offset on the specified line whose position is 1053 * closest to the specified horizontal position. 1054 */ getOffsetForHorizontal(int line, float horiz)1055 public int getOffsetForHorizontal(int line, float horiz) { 1056 int max = getLineEnd(line) - 1; 1057 int min = getLineStart(line); 1058 Directions dirs = getLineDirections(line); 1059 1060 if (line == getLineCount() - 1) 1061 max++; 1062 1063 int best = min; 1064 float bestdist = Math.abs(getPrimaryHorizontal(best) - horiz); 1065 1066 for (int i = 0; i < dirs.mDirections.length; i += 2) { 1067 int here = min + dirs.mDirections[i]; 1068 int there = here + (dirs.mDirections[i+1] & RUN_LENGTH_MASK); 1069 int swap = (dirs.mDirections[i+1] & RUN_RTL_FLAG) != 0 ? -1 : 1; 1070 1071 if (there > max) 1072 there = max; 1073 int high = there - 1 + 1, low = here + 1 - 1, guess; 1074 1075 while (high - low > 1) { 1076 guess = (high + low) / 2; 1077 int adguess = getOffsetAtStartOf(guess); 1078 1079 if (getPrimaryHorizontal(adguess) * swap >= horiz * swap) 1080 high = guess; 1081 else 1082 low = guess; 1083 } 1084 1085 if (low < here + 1) 1086 low = here + 1; 1087 1088 if (low < there) { 1089 low = getOffsetAtStartOf(low); 1090 1091 float dist = Math.abs(getPrimaryHorizontal(low) - horiz); 1092 1093 int aft = TextUtils.getOffsetAfter(mText, low); 1094 if (aft < there) { 1095 float other = Math.abs(getPrimaryHorizontal(aft) - horiz); 1096 1097 if (other < dist) { 1098 dist = other; 1099 low = aft; 1100 } 1101 } 1102 1103 if (dist < bestdist) { 1104 bestdist = dist; 1105 best = low; 1106 } 1107 } 1108 1109 float dist = Math.abs(getPrimaryHorizontal(here) - horiz); 1110 1111 if (dist < bestdist) { 1112 bestdist = dist; 1113 best = here; 1114 } 1115 } 1116 1117 float dist = Math.abs(getPrimaryHorizontal(max) - horiz); 1118 1119 if (dist <= bestdist) { 1120 bestdist = dist; 1121 best = max; 1122 } 1123 1124 return best; 1125 } 1126 1127 /** 1128 * Return the text offset after the last character on the specified line. 1129 */ getLineEnd(int line)1130 public final int getLineEnd(int line) { 1131 return getLineStart(line + 1); 1132 } 1133 1134 /** 1135 * Return the text offset after the last visible character (so whitespace 1136 * is not counted) on the specified line. 1137 */ getLineVisibleEnd(int line)1138 public int getLineVisibleEnd(int line) { 1139 return getLineVisibleEnd(line, getLineStart(line), getLineStart(line+1)); 1140 } 1141 getLineVisibleEnd(int line, int start, int end)1142 private int getLineVisibleEnd(int line, int start, int end) { 1143 CharSequence text = mText; 1144 char ch; 1145 if (line == getLineCount() - 1) { 1146 return end; 1147 } 1148 1149 for (; end > start; end--) { 1150 ch = text.charAt(end - 1); 1151 1152 if (ch == '\n') { 1153 return end - 1; 1154 } 1155 1156 if (ch != ' ' && ch != '\t') { 1157 break; 1158 } 1159 1160 } 1161 1162 return end; 1163 } 1164 1165 /** 1166 * Return the vertical position of the bottom of the specified line. 1167 */ getLineBottom(int line)1168 public final int getLineBottom(int line) { 1169 return getLineTop(line + 1); 1170 } 1171 1172 /** 1173 * Return the vertical position of the baseline of the specified line. 1174 */ getLineBaseline(int line)1175 public final int getLineBaseline(int line) { 1176 // getLineTop(line+1) == getLineTop(line) 1177 return getLineTop(line+1) - getLineDescent(line); 1178 } 1179 1180 /** 1181 * Get the ascent of the text on the specified line. 1182 * The return value is negative to match the Paint.ascent() convention. 1183 */ getLineAscent(int line)1184 public final int getLineAscent(int line) { 1185 // getLineTop(line+1) - getLineDescent(line) == getLineBaseLine(line) 1186 return getLineTop(line) - (getLineTop(line+1) - getLineDescent(line)); 1187 } 1188 getOffsetToLeftOf(int offset)1189 public int getOffsetToLeftOf(int offset) { 1190 return getOffsetToLeftRightOf(offset, true); 1191 } 1192 getOffsetToRightOf(int offset)1193 public int getOffsetToRightOf(int offset) { 1194 return getOffsetToLeftRightOf(offset, false); 1195 } 1196 getOffsetToLeftRightOf(int caret, boolean toLeft)1197 private int getOffsetToLeftRightOf(int caret, boolean toLeft) { 1198 int line = getLineForOffset(caret); 1199 int lineStart = getLineStart(line); 1200 int lineEnd = getLineEnd(line); 1201 int lineDir = getParagraphDirection(line); 1202 1203 boolean lineChanged = false; 1204 boolean advance = toLeft == (lineDir == DIR_RIGHT_TO_LEFT); 1205 // if walking off line, look at the line we're headed to 1206 if (advance) { 1207 if (caret == lineEnd) { 1208 if (line < getLineCount() - 1) { 1209 lineChanged = true; 1210 ++line; 1211 } else { 1212 return caret; // at very end, don't move 1213 } 1214 } 1215 } else { 1216 if (caret == lineStart) { 1217 if (line > 0) { 1218 lineChanged = true; 1219 --line; 1220 } else { 1221 return caret; // at very start, don't move 1222 } 1223 } 1224 } 1225 1226 if (lineChanged) { 1227 lineStart = getLineStart(line); 1228 lineEnd = getLineEnd(line); 1229 int newDir = getParagraphDirection(line); 1230 if (newDir != lineDir) { 1231 // unusual case. we want to walk onto the line, but it runs 1232 // in a different direction than this one, so we fake movement 1233 // in the opposite direction. 1234 toLeft = !toLeft; 1235 lineDir = newDir; 1236 } 1237 } 1238 1239 Directions directions = getLineDirections(line); 1240 1241 TextLine tl = TextLine.obtain(); 1242 // XXX: we don't care about tabs 1243 tl.set(mPaint, mText, lineStart, lineEnd, lineDir, directions, false, null); 1244 caret = lineStart + tl.getOffsetToLeftRightOf(caret - lineStart, toLeft); 1245 tl = TextLine.recycle(tl); 1246 return caret; 1247 } 1248 getOffsetAtStartOf(int offset)1249 private int getOffsetAtStartOf(int offset) { 1250 // XXX this probably should skip local reorderings and 1251 // zero-width characters, look at callers 1252 if (offset == 0) 1253 return 0; 1254 1255 CharSequence text = mText; 1256 char c = text.charAt(offset); 1257 1258 if (c >= '\uDC00' && c <= '\uDFFF') { 1259 char c1 = text.charAt(offset - 1); 1260 1261 if (c1 >= '\uD800' && c1 <= '\uDBFF') 1262 offset -= 1; 1263 } 1264 1265 if (mSpannedText) { 1266 ReplacementSpan[] spans = ((Spanned) text).getSpans(offset, offset, 1267 ReplacementSpan.class); 1268 1269 for (int i = 0; i < spans.length; i++) { 1270 int start = ((Spanned) text).getSpanStart(spans[i]); 1271 int end = ((Spanned) text).getSpanEnd(spans[i]); 1272 1273 if (start < offset && end > offset) 1274 offset = start; 1275 } 1276 } 1277 1278 return offset; 1279 } 1280 1281 /** 1282 * Determine whether we should clamp cursor position. Currently it's 1283 * only robust for left-aligned displays. 1284 * @hide 1285 */ shouldClampCursor(int line)1286 public boolean shouldClampCursor(int line) { 1287 // Only clamp cursor position in left-aligned displays. 1288 switch (getParagraphAlignment(line)) { 1289 case ALIGN_LEFT: 1290 return true; 1291 case ALIGN_NORMAL: 1292 return getParagraphDirection(line) > 0; 1293 default: 1294 return false; 1295 } 1296 1297 } 1298 /** 1299 * Fills in the specified Path with a representation of a cursor 1300 * at the specified offset. This will often be a vertical line 1301 * but can be multiple discontinuous lines in text with multiple 1302 * directionalities. 1303 */ getCursorPath(int point, Path dest, CharSequence editingBuffer)1304 public void getCursorPath(int point, Path dest, 1305 CharSequence editingBuffer) { 1306 dest.reset(); 1307 1308 int line = getLineForOffset(point); 1309 int top = getLineTop(line); 1310 int bottom = getLineTop(line+1); 1311 1312 boolean clamped = shouldClampCursor(line); 1313 float h1 = getPrimaryHorizontal(point, clamped) - 0.5f; 1314 float h2 = isLevelBoundary(point) ? getSecondaryHorizontal(point, clamped) - 0.5f : h1; 1315 1316 int caps = TextKeyListener.getMetaState(editingBuffer, TextKeyListener.META_SHIFT_ON) | 1317 TextKeyListener.getMetaState(editingBuffer, TextKeyListener.META_SELECTING); 1318 int fn = TextKeyListener.getMetaState(editingBuffer, TextKeyListener.META_ALT_ON); 1319 int dist = 0; 1320 1321 if (caps != 0 || fn != 0) { 1322 dist = (bottom - top) >> 2; 1323 1324 if (fn != 0) 1325 top += dist; 1326 if (caps != 0) 1327 bottom -= dist; 1328 } 1329 1330 if (h1 < 0.5f) 1331 h1 = 0.5f; 1332 if (h2 < 0.5f) 1333 h2 = 0.5f; 1334 1335 if (Float.compare(h1, h2) == 0) { 1336 dest.moveTo(h1, top); 1337 dest.lineTo(h1, bottom); 1338 } else { 1339 dest.moveTo(h1, top); 1340 dest.lineTo(h1, (top + bottom) >> 1); 1341 1342 dest.moveTo(h2, (top + bottom) >> 1); 1343 dest.lineTo(h2, bottom); 1344 } 1345 1346 if (caps == 2) { 1347 dest.moveTo(h2, bottom); 1348 dest.lineTo(h2 - dist, bottom + dist); 1349 dest.lineTo(h2, bottom); 1350 dest.lineTo(h2 + dist, bottom + dist); 1351 } else if (caps == 1) { 1352 dest.moveTo(h2, bottom); 1353 dest.lineTo(h2 - dist, bottom + dist); 1354 1355 dest.moveTo(h2 - dist, bottom + dist - 0.5f); 1356 dest.lineTo(h2 + dist, bottom + dist - 0.5f); 1357 1358 dest.moveTo(h2 + dist, bottom + dist); 1359 dest.lineTo(h2, bottom); 1360 } 1361 1362 if (fn == 2) { 1363 dest.moveTo(h1, top); 1364 dest.lineTo(h1 - dist, top - dist); 1365 dest.lineTo(h1, top); 1366 dest.lineTo(h1 + dist, top - dist); 1367 } else if (fn == 1) { 1368 dest.moveTo(h1, top); 1369 dest.lineTo(h1 - dist, top - dist); 1370 1371 dest.moveTo(h1 - dist, top - dist + 0.5f); 1372 dest.lineTo(h1 + dist, top - dist + 0.5f); 1373 1374 dest.moveTo(h1 + dist, top - dist); 1375 dest.lineTo(h1, top); 1376 } 1377 } 1378 addSelection(int line, int start, int end, int top, int bottom, Path dest)1379 private void addSelection(int line, int start, int end, 1380 int top, int bottom, Path dest) { 1381 int linestart = getLineStart(line); 1382 int lineend = getLineEnd(line); 1383 Directions dirs = getLineDirections(line); 1384 1385 if (lineend > linestart && mText.charAt(lineend - 1) == '\n') 1386 lineend--; 1387 1388 for (int i = 0; i < dirs.mDirections.length; i += 2) { 1389 int here = linestart + dirs.mDirections[i]; 1390 int there = here + (dirs.mDirections[i+1] & RUN_LENGTH_MASK); 1391 1392 if (there > lineend) 1393 there = lineend; 1394 1395 if (start <= there && end >= here) { 1396 int st = Math.max(start, here); 1397 int en = Math.min(end, there); 1398 1399 if (st != en) { 1400 float h1 = getHorizontal(st, false, line, false /* not clamped */); 1401 float h2 = getHorizontal(en, true, line, false /* not clamped */); 1402 1403 float left = Math.min(h1, h2); 1404 float right = Math.max(h1, h2); 1405 1406 dest.addRect(left, top, right, bottom, Path.Direction.CW); 1407 } 1408 } 1409 } 1410 } 1411 1412 /** 1413 * Fills in the specified Path with a representation of a highlight 1414 * between the specified offsets. This will often be a rectangle 1415 * or a potentially discontinuous set of rectangles. If the start 1416 * and end are the same, the returned path is empty. 1417 */ getSelectionPath(int start, int end, Path dest)1418 public void getSelectionPath(int start, int end, Path dest) { 1419 dest.reset(); 1420 1421 if (start == end) 1422 return; 1423 1424 if (end < start) { 1425 int temp = end; 1426 end = start; 1427 start = temp; 1428 } 1429 1430 int startline = getLineForOffset(start); 1431 int endline = getLineForOffset(end); 1432 1433 int top = getLineTop(startline); 1434 int bottom = getLineBottom(endline); 1435 1436 if (startline == endline) { 1437 addSelection(startline, start, end, top, bottom, dest); 1438 } else { 1439 final float width = mWidth; 1440 1441 addSelection(startline, start, getLineEnd(startline), 1442 top, getLineBottom(startline), dest); 1443 1444 if (getParagraphDirection(startline) == DIR_RIGHT_TO_LEFT) 1445 dest.addRect(getLineLeft(startline), top, 1446 0, getLineBottom(startline), Path.Direction.CW); 1447 else 1448 dest.addRect(getLineRight(startline), top, 1449 width, getLineBottom(startline), Path.Direction.CW); 1450 1451 for (int i = startline + 1; i < endline; i++) { 1452 top = getLineTop(i); 1453 bottom = getLineBottom(i); 1454 dest.addRect(0, top, width, bottom, Path.Direction.CW); 1455 } 1456 1457 top = getLineTop(endline); 1458 bottom = getLineBottom(endline); 1459 1460 addSelection(endline, getLineStart(endline), end, 1461 top, bottom, dest); 1462 1463 if (getParagraphDirection(endline) == DIR_RIGHT_TO_LEFT) 1464 dest.addRect(width, top, getLineRight(endline), bottom, Path.Direction.CW); 1465 else 1466 dest.addRect(0, top, getLineLeft(endline), bottom, Path.Direction.CW); 1467 } 1468 } 1469 1470 /** 1471 * Get the alignment of the specified paragraph, taking into account 1472 * markup attached to it. 1473 */ getParagraphAlignment(int line)1474 public final Alignment getParagraphAlignment(int line) { 1475 Alignment align = mAlignment; 1476 1477 if (mSpannedText) { 1478 Spanned sp = (Spanned) mText; 1479 AlignmentSpan[] spans = getParagraphSpans(sp, getLineStart(line), 1480 getLineEnd(line), 1481 AlignmentSpan.class); 1482 1483 int spanLength = spans.length; 1484 if (spanLength > 0) { 1485 align = spans[spanLength-1].getAlignment(); 1486 } 1487 } 1488 1489 return align; 1490 } 1491 1492 /** 1493 * Get the left edge of the specified paragraph, inset by left margins. 1494 */ getParagraphLeft(int line)1495 public final int getParagraphLeft(int line) { 1496 int left = 0; 1497 int dir = getParagraphDirection(line); 1498 if (dir == DIR_RIGHT_TO_LEFT || !mSpannedText) { 1499 return left; // leading margin has no impact, or no styles 1500 } 1501 return getParagraphLeadingMargin(line); 1502 } 1503 1504 /** 1505 * Get the right edge of the specified paragraph, inset by right margins. 1506 */ getParagraphRight(int line)1507 public final int getParagraphRight(int line) { 1508 int right = mWidth; 1509 int dir = getParagraphDirection(line); 1510 if (dir == DIR_LEFT_TO_RIGHT || !mSpannedText) { 1511 return right; // leading margin has no impact, or no styles 1512 } 1513 return right - getParagraphLeadingMargin(line); 1514 } 1515 1516 /** 1517 * Returns the effective leading margin (unsigned) for this line, 1518 * taking into account LeadingMarginSpan and LeadingMarginSpan2. 1519 * @param line the line index 1520 * @return the leading margin of this line 1521 */ getParagraphLeadingMargin(int line)1522 private int getParagraphLeadingMargin(int line) { 1523 if (!mSpannedText) { 1524 return 0; 1525 } 1526 Spanned spanned = (Spanned) mText; 1527 1528 int lineStart = getLineStart(line); 1529 int lineEnd = getLineEnd(line); 1530 int spanEnd = spanned.nextSpanTransition(lineStart, lineEnd, 1531 LeadingMarginSpan.class); 1532 LeadingMarginSpan[] spans = getParagraphSpans(spanned, lineStart, spanEnd, 1533 LeadingMarginSpan.class); 1534 if (spans.length == 0) { 1535 return 0; // no leading margin span; 1536 } 1537 1538 int margin = 0; 1539 1540 boolean isFirstParaLine = lineStart == 0 || 1541 spanned.charAt(lineStart - 1) == '\n'; 1542 1543 boolean useFirstLineMargin = isFirstParaLine; 1544 for (int i = 0; i < spans.length; i++) { 1545 if (spans[i] instanceof LeadingMarginSpan2) { 1546 int spStart = spanned.getSpanStart(spans[i]); 1547 int spanLine = getLineForOffset(spStart); 1548 int count = ((LeadingMarginSpan2) spans[i]).getLeadingMarginLineCount(); 1549 // if there is more than one LeadingMarginSpan2, use the count that is greatest 1550 useFirstLineMargin |= line < spanLine + count; 1551 } 1552 } 1553 for (int i = 0; i < spans.length; i++) { 1554 LeadingMarginSpan span = spans[i]; 1555 margin += span.getLeadingMargin(useFirstLineMargin); 1556 } 1557 1558 return margin; 1559 } 1560 1561 /* package */ 1562 static float measurePara(TextPaint paint, CharSequence text, int start, int end) { 1563 1564 MeasuredText mt = MeasuredText.obtain(); 1565 TextLine tl = TextLine.obtain(); 1566 try { 1567 mt.setPara(text, start, end, TextDirectionHeuristics.LTR); 1568 Directions directions; 1569 int dir; 1570 if (mt.mEasy) { 1571 directions = DIRS_ALL_LEFT_TO_RIGHT; 1572 dir = Layout.DIR_LEFT_TO_RIGHT; 1573 } else { 1574 directions = AndroidBidi.directions(mt.mDir, mt.mLevels, 1575 0, mt.mChars, 0, mt.mLen); 1576 dir = mt.mDir; 1577 } 1578 char[] chars = mt.mChars; 1579 int len = mt.mLen; 1580 boolean hasTabs = false; 1581 TabStops tabStops = null; 1582 // leading margins should be taken into account when measuring a paragraph 1583 int margin = 0; 1584 if (text instanceof Spanned) { 1585 Spanned spanned = (Spanned) text; 1586 LeadingMarginSpan[] spans = getParagraphSpans(spanned, start, end, 1587 LeadingMarginSpan.class); 1588 for (LeadingMarginSpan lms : spans) { 1589 margin += lms.getLeadingMargin(true); 1590 } 1591 } 1592 for (int i = 0; i < len; ++i) { 1593 if (chars[i] == '\t') { 1594 hasTabs = true; 1595 if (text instanceof Spanned) { 1596 Spanned spanned = (Spanned) text; 1597 int spanEnd = spanned.nextSpanTransition(start, end, 1598 TabStopSpan.class); 1599 TabStopSpan[] spans = getParagraphSpans(spanned, start, spanEnd, 1600 TabStopSpan.class); 1601 if (spans.length > 0) { 1602 tabStops = new TabStops(TAB_INCREMENT, spans); 1603 } 1604 } 1605 break; 1606 } 1607 } 1608 tl.set(paint, text, start, end, dir, directions, hasTabs, tabStops); 1609 return margin + tl.metrics(null); 1610 } finally { 1611 TextLine.recycle(tl); 1612 MeasuredText.recycle(mt); 1613 } 1614 } 1615 1616 /** 1617 * @hide 1618 */ 1619 /* package */ static class TabStops { 1620 private int[] mStops; 1621 private int mNumStops; 1622 private int mIncrement; 1623 1624 TabStops(int increment, Object[] spans) { 1625 reset(increment, spans); 1626 } 1627 1628 void reset(int increment, Object[] spans) { 1629 this.mIncrement = increment; 1630 1631 int ns = 0; 1632 if (spans != null) { 1633 int[] stops = this.mStops; 1634 for (Object o : spans) { 1635 if (o instanceof TabStopSpan) { 1636 if (stops == null) { 1637 stops = new int[10]; 1638 } else if (ns == stops.length) { 1639 int[] nstops = new int[ns * 2]; 1640 for (int i = 0; i < ns; ++i) { 1641 nstops[i] = stops[i]; 1642 } 1643 stops = nstops; 1644 } 1645 stops[ns++] = ((TabStopSpan) o).getTabStop(); 1646 } 1647 } 1648 if (ns > 1) { 1649 Arrays.sort(stops, 0, ns); 1650 } 1651 if (stops != this.mStops) { 1652 this.mStops = stops; 1653 } 1654 } 1655 this.mNumStops = ns; 1656 } 1657 1658 float nextTab(float h) { 1659 int ns = this.mNumStops; 1660 if (ns > 0) { 1661 int[] stops = this.mStops; 1662 for (int i = 0; i < ns; ++i) { 1663 int stop = stops[i]; 1664 if (stop > h) { 1665 return stop; 1666 } 1667 } 1668 } 1669 return nextDefaultStop(h, mIncrement); 1670 } 1671 1672 public static float nextDefaultStop(float h, int inc) { 1673 return ((int) ((h + inc) / inc)) * inc; 1674 } 1675 } 1676 1677 /** 1678 * Returns the position of the next tab stop after h on the line. 1679 * 1680 * @param text the text 1681 * @param start start of the line 1682 * @param end limit of the line 1683 * @param h the current horizontal offset 1684 * @param tabs the tabs, can be null. If it is null, any tabs in effect 1685 * on the line will be used. If there are no tabs, a default offset 1686 * will be used to compute the tab stop. 1687 * @return the offset of the next tab stop. 1688 */ 1689 /* package */ static float nextTab(CharSequence text, int start, int end, 1690 float h, Object[] tabs) { 1691 float nh = Float.MAX_VALUE; 1692 boolean alltabs = false; 1693 1694 if (text instanceof Spanned) { 1695 if (tabs == null) { 1696 tabs = getParagraphSpans((Spanned) text, start, end, TabStopSpan.class); 1697 alltabs = true; 1698 } 1699 1700 for (int i = 0; i < tabs.length; i++) { 1701 if (!alltabs) { 1702 if (!(tabs[i] instanceof TabStopSpan)) 1703 continue; 1704 } 1705 1706 int where = ((TabStopSpan) tabs[i]).getTabStop(); 1707 1708 if (where < nh && where > h) 1709 nh = where; 1710 } 1711 1712 if (nh != Float.MAX_VALUE) 1713 return nh; 1714 } 1715 1716 return ((int) ((h + TAB_INCREMENT) / TAB_INCREMENT)) * TAB_INCREMENT; 1717 } 1718 1719 protected final boolean isSpanned() { 1720 return mSpannedText; 1721 } 1722 1723 /** 1724 * Returns the same as <code>text.getSpans()</code>, except where 1725 * <code>start</code> and <code>end</code> are the same and are not 1726 * at the very beginning of the text, in which case an empty array 1727 * is returned instead. 1728 * <p> 1729 * This is needed because of the special case that <code>getSpans()</code> 1730 * on an empty range returns the spans adjacent to that range, which is 1731 * primarily for the sake of <code>TextWatchers</code> so they will get 1732 * notifications when text goes from empty to non-empty. But it also 1733 * has the unfortunate side effect that if the text ends with an empty 1734 * paragraph, that paragraph accidentally picks up the styles of the 1735 * preceding paragraph (even though those styles will not be picked up 1736 * by new text that is inserted into the empty paragraph). 1737 * <p> 1738 * The reason it just checks whether <code>start</code> and <code>end</code> 1739 * is the same is that the only time a line can contain 0 characters 1740 * is if it is the final paragraph of the Layout; otherwise any line will 1741 * contain at least one printing or newline character. The reason for the 1742 * additional check if <code>start</code> is greater than 0 is that 1743 * if the empty paragraph is the entire content of the buffer, paragraph 1744 * styles that are already applied to the buffer will apply to text that 1745 * is inserted into it. 1746 */ 1747 /* package */static <T> T[] getParagraphSpans(Spanned text, int start, int end, Class<T> type) { 1748 if (start == end && start > 0) { 1749 return ArrayUtils.emptyArray(type); 1750 } 1751 1752 return text.getSpans(start, end, type); 1753 } 1754 1755 private char getEllipsisChar(TextUtils.TruncateAt method) { 1756 return (method == TextUtils.TruncateAt.END_SMALL) ? 1757 TextUtils.ELLIPSIS_TWO_DOTS[0] : 1758 TextUtils.ELLIPSIS_NORMAL[0]; 1759 } 1760 1761 private void ellipsize(int start, int end, int line, 1762 char[] dest, int destoff, TextUtils.TruncateAt method) { 1763 int ellipsisCount = getEllipsisCount(line); 1764 1765 if (ellipsisCount == 0) { 1766 return; 1767 } 1768 1769 int ellipsisStart = getEllipsisStart(line); 1770 int linestart = getLineStart(line); 1771 1772 for (int i = ellipsisStart; i < ellipsisStart + ellipsisCount; i++) { 1773 char c; 1774 1775 if (i == ellipsisStart) { 1776 c = getEllipsisChar(method); // ellipsis 1777 } else { 1778 c = '\uFEFF'; // 0-width space 1779 } 1780 1781 int a = i + linestart; 1782 1783 if (a >= start && a < end) { 1784 dest[destoff + a - start] = c; 1785 } 1786 } 1787 } 1788 1789 /** 1790 * Stores information about bidirectional (left-to-right or right-to-left) 1791 * text within the layout of a line. 1792 */ 1793 public static class Directions { 1794 // Directions represents directional runs within a line of text. 1795 // Runs are pairs of ints listed in visual order, starting from the 1796 // leading margin. The first int of each pair is the offset from 1797 // the first character of the line to the start of the run. The 1798 // second int represents both the length and level of the run. 1799 // The length is in the lower bits, accessed by masking with 1800 // DIR_LENGTH_MASK. The level is in the higher bits, accessed 1801 // by shifting by DIR_LEVEL_SHIFT and masking by DIR_LEVEL_MASK. 1802 // To simply test for an RTL direction, test the bit using 1803 // DIR_RTL_FLAG, if set then the direction is rtl. 1804 1805 /* package */ int[] mDirections; 1806 /* package */ Directions(int[] dirs) { 1807 mDirections = dirs; 1808 } 1809 } 1810 1811 /** 1812 * Return the offset of the first character to be ellipsized away, 1813 * relative to the start of the line. (So 0 if the beginning of the 1814 * line is ellipsized, not getLineStart().) 1815 */ 1816 public abstract int getEllipsisStart(int line); 1817 1818 /** 1819 * Returns the number of characters to be ellipsized away, or 0 if 1820 * no ellipsis is to take place. 1821 */ 1822 public abstract int getEllipsisCount(int line); 1823 1824 /* package */ static class Ellipsizer implements CharSequence, GetChars { 1825 /* package */ CharSequence mText; 1826 /* package */ Layout mLayout; 1827 /* package */ int mWidth; 1828 /* package */ TextUtils.TruncateAt mMethod; 1829 1830 public Ellipsizer(CharSequence s) { 1831 mText = s; 1832 } 1833 1834 public char charAt(int off) { 1835 char[] buf = TextUtils.obtain(1); 1836 getChars(off, off + 1, buf, 0); 1837 char ret = buf[0]; 1838 1839 TextUtils.recycle(buf); 1840 return ret; 1841 } 1842 1843 public void getChars(int start, int end, char[] dest, int destoff) { 1844 int line1 = mLayout.getLineForOffset(start); 1845 int line2 = mLayout.getLineForOffset(end); 1846 1847 TextUtils.getChars(mText, start, end, dest, destoff); 1848 1849 for (int i = line1; i <= line2; i++) { 1850 mLayout.ellipsize(start, end, i, dest, destoff, mMethod); 1851 } 1852 } 1853 1854 public int length() { 1855 return mText.length(); 1856 } 1857 1858 public CharSequence subSequence(int start, int end) { 1859 char[] s = new char[end - start]; 1860 getChars(start, end, s, 0); 1861 return new String(s); 1862 } 1863 1864 @Override 1865 public String toString() { 1866 char[] s = new char[length()]; 1867 getChars(0, length(), s, 0); 1868 return new String(s); 1869 } 1870 1871 } 1872 1873 /* package */ static class SpannedEllipsizer extends Ellipsizer implements Spanned { 1874 private Spanned mSpanned; 1875 1876 public SpannedEllipsizer(CharSequence display) { 1877 super(display); 1878 mSpanned = (Spanned) display; 1879 } 1880 1881 public <T> T[] getSpans(int start, int end, Class<T> type) { 1882 return mSpanned.getSpans(start, end, type); 1883 } 1884 1885 public int getSpanStart(Object tag) { 1886 return mSpanned.getSpanStart(tag); 1887 } 1888 1889 public int getSpanEnd(Object tag) { 1890 return mSpanned.getSpanEnd(tag); 1891 } 1892 1893 public int getSpanFlags(Object tag) { 1894 return mSpanned.getSpanFlags(tag); 1895 } 1896 1897 @SuppressWarnings("rawtypes") 1898 public int nextSpanTransition(int start, int limit, Class type) { 1899 return mSpanned.nextSpanTransition(start, limit, type); 1900 } 1901 1902 @Override 1903 public CharSequence subSequence(int start, int end) { 1904 char[] s = new char[end - start]; 1905 getChars(start, end, s, 0); 1906 1907 SpannableString ss = new SpannableString(new String(s)); 1908 TextUtils.copySpansFrom(mSpanned, start, end, Object.class, ss, 0); 1909 return ss; 1910 } 1911 } 1912 1913 private CharSequence mText; 1914 private TextPaint mPaint; 1915 /* package */ TextPaint mWorkPaint; 1916 private int mWidth; 1917 private Alignment mAlignment = Alignment.ALIGN_NORMAL; 1918 private float mSpacingMult; 1919 private float mSpacingAdd; 1920 private static final Rect sTempRect = new Rect(); 1921 private boolean mSpannedText; 1922 private TextDirectionHeuristic mTextDir; 1923 private SpanSet<LineBackgroundSpan> mLineBackgroundSpans; 1924 1925 public static final int DIR_LEFT_TO_RIGHT = 1; 1926 public static final int DIR_RIGHT_TO_LEFT = -1; 1927 1928 /* package */ static final int DIR_REQUEST_LTR = 1; 1929 /* package */ static final int DIR_REQUEST_RTL = -1; 1930 /* package */ static final int DIR_REQUEST_DEFAULT_LTR = 2; 1931 /* package */ static final int DIR_REQUEST_DEFAULT_RTL = -2; 1932 1933 /* package */ static final int RUN_LENGTH_MASK = 0x03ffffff; 1934 /* package */ static final int RUN_LEVEL_SHIFT = 26; 1935 /* package */ static final int RUN_LEVEL_MASK = 0x3f; 1936 /* package */ static final int RUN_RTL_FLAG = 1 << RUN_LEVEL_SHIFT; 1937 1938 public enum Alignment { 1939 ALIGN_NORMAL, 1940 ALIGN_OPPOSITE, 1941 ALIGN_CENTER, 1942 /** @hide */ 1943 ALIGN_LEFT, 1944 /** @hide */ 1945 ALIGN_RIGHT, 1946 } 1947 1948 private static final int TAB_INCREMENT = 20; 1949 1950 /* package */ static final Directions DIRS_ALL_LEFT_TO_RIGHT = 1951 new Directions(new int[] { 0, RUN_LENGTH_MASK }); 1952 /* package */ static final Directions DIRS_ALL_RIGHT_TO_LEFT = 1953 new Directions(new int[] { 0, RUN_LENGTH_MASK | RUN_RTL_FLAG }); 1954 1955 } 1956