• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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.annotation.IntDef;
20 import android.annotation.IntRange;
21 import android.compat.annotation.UnsupportedAppUsage;
22 import android.graphics.Canvas;
23 import android.graphics.Paint;
24 import android.graphics.Path;
25 import android.graphics.Rect;
26 import android.graphics.text.LineBreaker;
27 import android.text.method.TextKeyListener;
28 import android.text.style.AlignmentSpan;
29 import android.text.style.LeadingMarginSpan;
30 import android.text.style.LeadingMarginSpan.LeadingMarginSpan2;
31 import android.text.style.LineBackgroundSpan;
32 import android.text.style.ParagraphStyle;
33 import android.text.style.ReplacementSpan;
34 import android.text.style.TabStopSpan;
35 
36 import com.android.internal.annotations.VisibleForTesting;
37 import com.android.internal.util.ArrayUtils;
38 import com.android.internal.util.GrowingArrayUtils;
39 
40 import java.lang.annotation.Retention;
41 import java.lang.annotation.RetentionPolicy;
42 import java.util.Arrays;
43 
44 /**
45  * A base class that manages text layout in visual elements on
46  * the screen.
47  * <p>For text that will be edited, use a {@link DynamicLayout},
48  * which will be updated as the text changes.
49  * For text that will not change, use a {@link StaticLayout}.
50  */
51 public abstract class Layout {
52     /** @hide */
53     @IntDef(prefix = { "BREAK_STRATEGY_" }, value = {
54             LineBreaker.BREAK_STRATEGY_SIMPLE,
55             LineBreaker.BREAK_STRATEGY_HIGH_QUALITY,
56             LineBreaker.BREAK_STRATEGY_BALANCED
57     })
58     @Retention(RetentionPolicy.SOURCE)
59     public @interface BreakStrategy {}
60 
61     /**
62      * Value for break strategy indicating simple line breaking. Automatic hyphens are not added
63      * (though soft hyphens are respected), and modifying text generally doesn't affect the layout
64      * before it (which yields a more consistent user experience when editing), but layout may not
65      * be the highest quality.
66      */
67     public static final int BREAK_STRATEGY_SIMPLE = LineBreaker.BREAK_STRATEGY_SIMPLE;
68 
69     /**
70      * Value for break strategy indicating high quality line breaking, including automatic
71      * hyphenation and doing whole-paragraph optimization of line breaks.
72      */
73     public static final int BREAK_STRATEGY_HIGH_QUALITY = LineBreaker.BREAK_STRATEGY_HIGH_QUALITY;
74 
75     /**
76      * Value for break strategy indicating balanced line breaking. The breaks are chosen to
77      * make all lines as close to the same length as possible, including automatic hyphenation.
78      */
79     public static final int BREAK_STRATEGY_BALANCED = LineBreaker.BREAK_STRATEGY_BALANCED;
80 
81     /** @hide */
82     @IntDef(prefix = { "HYPHENATION_FREQUENCY_" }, value = {
83             HYPHENATION_FREQUENCY_NORMAL,
84             HYPHENATION_FREQUENCY_FULL,
85             HYPHENATION_FREQUENCY_NONE
86     })
87     @Retention(RetentionPolicy.SOURCE)
88     public @interface HyphenationFrequency {}
89 
90     /**
91      * Value for hyphenation frequency indicating no automatic hyphenation. Useful
92      * for backward compatibility, and for cases where the automatic hyphenation algorithm results
93      * in incorrect hyphenation. Mid-word breaks may still happen when a word is wider than the
94      * layout and there is otherwise no valid break. Soft hyphens are ignored and will not be used
95      * as suggestions for potential line breaks.
96      */
97     public static final int HYPHENATION_FREQUENCY_NONE = LineBreaker.HYPHENATION_FREQUENCY_NONE;
98 
99     /**
100      * Value for hyphenation frequency indicating a light amount of automatic hyphenation, which
101      * is a conservative default. Useful for informal cases, such as short sentences or chat
102      * messages.
103      */
104     public static final int HYPHENATION_FREQUENCY_NORMAL = LineBreaker.HYPHENATION_FREQUENCY_NORMAL;
105 
106     /**
107      * Value for hyphenation frequency indicating the full amount of automatic hyphenation, typical
108      * in typography. Useful for running text and where it's important to put the maximum amount of
109      * text in a screen with limited space.
110      */
111     public static final int HYPHENATION_FREQUENCY_FULL = LineBreaker.HYPHENATION_FREQUENCY_FULL;
112 
113     private static final ParagraphStyle[] NO_PARA_SPANS =
114         ArrayUtils.emptyArray(ParagraphStyle.class);
115 
116     /** @hide */
117     @IntDef(prefix = { "JUSTIFICATION_MODE_" }, value = {
118             LineBreaker.JUSTIFICATION_MODE_NONE,
119             LineBreaker.JUSTIFICATION_MODE_INTER_WORD
120     })
121     @Retention(RetentionPolicy.SOURCE)
122     public @interface JustificationMode {}
123 
124     /**
125      * Value for justification mode indicating no justification.
126      */
127     public static final int JUSTIFICATION_MODE_NONE = LineBreaker.JUSTIFICATION_MODE_NONE;
128 
129     /**
130      * Value for justification mode indicating the text is justified by stretching word spacing.
131      */
132     public static final int JUSTIFICATION_MODE_INTER_WORD =
133             LineBreaker.JUSTIFICATION_MODE_INTER_WORD;
134 
135     /*
136      * Line spacing multiplier for default line spacing.
137      */
138     public static final float DEFAULT_LINESPACING_MULTIPLIER = 1.0f;
139 
140     /*
141      * Line spacing addition for default line spacing.
142      */
143     public static final float DEFAULT_LINESPACING_ADDITION = 0.0f;
144 
145     /**
146      * Return how wide a layout must be in order to display the specified text with one line per
147      * paragraph.
148      *
149      * <p>As of O, Uses
150      * {@link TextDirectionHeuristics#FIRSTSTRONG_LTR} as the default text direction heuristics. In
151      * the earlier versions uses {@link TextDirectionHeuristics#LTR} as the default.</p>
152      */
getDesiredWidth(CharSequence source, TextPaint paint)153     public static float getDesiredWidth(CharSequence source,
154                                         TextPaint paint) {
155         return getDesiredWidth(source, 0, source.length(), paint);
156     }
157 
158     /**
159      * Return how wide a layout must be in order to display the specified text slice with one
160      * line per paragraph.
161      *
162      * <p>As of O, Uses
163      * {@link TextDirectionHeuristics#FIRSTSTRONG_LTR} as the default text direction heuristics. In
164      * the earlier versions uses {@link TextDirectionHeuristics#LTR} as the default.</p>
165      */
getDesiredWidth(CharSequence source, int start, int end, TextPaint paint)166     public static float getDesiredWidth(CharSequence source, int start, int end, TextPaint paint) {
167         return getDesiredWidth(source, start, end, paint, TextDirectionHeuristics.FIRSTSTRONG_LTR);
168     }
169 
170     /**
171      * Return how wide a layout must be in order to display the
172      * specified text slice with one line per paragraph.
173      *
174      * @hide
175      */
getDesiredWidth(CharSequence source, int start, int end, TextPaint paint, TextDirectionHeuristic textDir)176     public static float getDesiredWidth(CharSequence source, int start, int end, TextPaint paint,
177             TextDirectionHeuristic textDir) {
178         return getDesiredWidthWithLimit(source, start, end, paint, textDir, Float.MAX_VALUE);
179     }
180     /**
181      * Return how wide a layout must be in order to display the
182      * specified text slice with one line per paragraph.
183      *
184      * If the measured width exceeds given limit, returns limit value instead.
185      * @hide
186      */
getDesiredWidthWithLimit(CharSequence source, int start, int end, TextPaint paint, TextDirectionHeuristic textDir, float upperLimit)187     public static float getDesiredWidthWithLimit(CharSequence source, int start, int end,
188             TextPaint paint, TextDirectionHeuristic textDir, float upperLimit) {
189         float need = 0;
190 
191         int next;
192         for (int i = start; i <= end; i = next) {
193             next = TextUtils.indexOf(source, '\n', i, end);
194 
195             if (next < 0)
196                 next = end;
197 
198             // note, omits trailing paragraph char
199             float w = measurePara(paint, source, i, next, textDir);
200             if (w > upperLimit) {
201                 return upperLimit;
202             }
203 
204             if (w > need)
205                 need = w;
206 
207             next++;
208         }
209 
210         return need;
211     }
212 
213     /**
214      * Subclasses of Layout use this constructor to set the display text,
215      * width, and other standard properties.
216      * @param text the text to render
217      * @param paint the default paint for the layout.  Styles can override
218      * various attributes of the paint.
219      * @param width the wrapping width for the text.
220      * @param align whether to left, right, or center the text.  Styles can
221      * override the alignment.
222      * @param spacingMult factor by which to scale the font size to get the
223      * default line spacing
224      * @param spacingAdd amount to add to the default line spacing
225      */
Layout(CharSequence text, TextPaint paint, int width, Alignment align, float spacingMult, float spacingAdd)226     protected Layout(CharSequence text, TextPaint paint,
227                      int width, Alignment align,
228                      float spacingMult, float spacingAdd) {
229         this(text, paint, width, align, TextDirectionHeuristics.FIRSTSTRONG_LTR,
230                 spacingMult, spacingAdd);
231     }
232 
233     /**
234      * Subclasses of Layout use this constructor to set the display text,
235      * width, and other standard properties.
236      * @param text the text to render
237      * @param paint the default paint for the layout.  Styles can override
238      * various attributes of the paint.
239      * @param width the wrapping width for the text.
240      * @param align whether to left, right, or center the text.  Styles can
241      * override the alignment.
242      * @param spacingMult factor by which to scale the font size to get the
243      * default line spacing
244      * @param spacingAdd amount to add to the default line spacing
245      *
246      * @hide
247      */
Layout(CharSequence text, TextPaint paint, int width, Alignment align, TextDirectionHeuristic textDir, float spacingMult, float spacingAdd)248     protected Layout(CharSequence text, TextPaint paint,
249                      int width, Alignment align, TextDirectionHeuristic textDir,
250                      float spacingMult, float spacingAdd) {
251 
252         if (width < 0)
253             throw new IllegalArgumentException("Layout: " + width + " < 0");
254 
255         // Ensure paint doesn't have baselineShift set.
256         // While normally we don't modify the paint the user passed in,
257         // we were already doing this in Styled.drawUniformRun with both
258         // baselineShift and bgColor.  We probably should reevaluate bgColor.
259         if (paint != null) {
260             paint.bgColor = 0;
261             paint.baselineShift = 0;
262         }
263 
264         mText = text;
265         mPaint = paint;
266         mWidth = width;
267         mAlignment = align;
268         mSpacingMult = spacingMult;
269         mSpacingAdd = spacingAdd;
270         mSpannedText = text instanceof Spanned;
271         mTextDir = textDir;
272     }
273 
274     /** @hide */
setJustificationMode(@ustificationMode int justificationMode)275     protected void setJustificationMode(@JustificationMode int justificationMode) {
276         mJustificationMode = justificationMode;
277     }
278 
279     /**
280      * Replace constructor properties of this Layout with new ones.  Be careful.
281      */
replaceWith(CharSequence text, TextPaint paint, int width, Alignment align, float spacingmult, float spacingadd)282     /* package */ void replaceWith(CharSequence text, TextPaint paint,
283                               int width, Alignment align,
284                               float spacingmult, float spacingadd) {
285         if (width < 0) {
286             throw new IllegalArgumentException("Layout: " + width + " < 0");
287         }
288 
289         mText = text;
290         mPaint = paint;
291         mWidth = width;
292         mAlignment = align;
293         mSpacingMult = spacingmult;
294         mSpacingAdd = spacingadd;
295         mSpannedText = text instanceof Spanned;
296     }
297 
298     /**
299      * Draw this Layout on the specified Canvas.
300      */
draw(Canvas c)301     public void draw(Canvas c) {
302         draw(c, null, null, 0);
303     }
304 
305     /**
306      * Draw this Layout on the specified canvas, with the highlight path drawn
307      * between the background and the text.
308      *
309      * @param canvas the canvas
310      * @param highlight the path of the highlight or cursor; can be null
311      * @param highlightPaint the paint for the highlight
312      * @param cursorOffsetVertical the amount to temporarily translate the
313      *        canvas while rendering the highlight
314      */
draw(Canvas canvas, Path highlight, Paint highlightPaint, int cursorOffsetVertical)315     public void draw(Canvas canvas, Path highlight, Paint highlightPaint,
316             int cursorOffsetVertical) {
317         final long lineRange = getLineRangeForDraw(canvas);
318         int firstLine = TextUtils.unpackRangeStartFromLong(lineRange);
319         int lastLine = TextUtils.unpackRangeEndFromLong(lineRange);
320         if (lastLine < 0) return;
321 
322         drawBackground(canvas, highlight, highlightPaint, cursorOffsetVertical,
323                 firstLine, lastLine);
324         drawText(canvas, firstLine, lastLine);
325     }
326 
isJustificationRequired(int lineNum)327     private boolean isJustificationRequired(int lineNum) {
328         if (mJustificationMode == JUSTIFICATION_MODE_NONE) return false;
329         final int lineEnd = getLineEnd(lineNum);
330         return lineEnd < mText.length() && mText.charAt(lineEnd - 1) != '\n';
331     }
332 
getJustifyWidth(int lineNum)333     private float getJustifyWidth(int lineNum) {
334         Alignment paraAlign = mAlignment;
335 
336         int left = 0;
337         int right = mWidth;
338 
339         final int dir = getParagraphDirection(lineNum);
340 
341         ParagraphStyle[] spans = NO_PARA_SPANS;
342         if (mSpannedText) {
343             Spanned sp = (Spanned) mText;
344             final int start = getLineStart(lineNum);
345 
346             final boolean isFirstParaLine = (start == 0 || mText.charAt(start - 1) == '\n');
347 
348             if (isFirstParaLine) {
349                 final int spanEnd = sp.nextSpanTransition(start, mText.length(),
350                         ParagraphStyle.class);
351                 spans = getParagraphSpans(sp, start, spanEnd, ParagraphStyle.class);
352 
353                 for (int n = spans.length - 1; n >= 0; n--) {
354                     if (spans[n] instanceof AlignmentSpan) {
355                         paraAlign = ((AlignmentSpan) spans[n]).getAlignment();
356                         break;
357                     }
358                 }
359             }
360 
361             final int length = spans.length;
362             boolean useFirstLineMargin = isFirstParaLine;
363             for (int n = 0; n < length; n++) {
364                 if (spans[n] instanceof LeadingMarginSpan2) {
365                     int count = ((LeadingMarginSpan2) spans[n]).getLeadingMarginLineCount();
366                     int startLine = getLineForOffset(sp.getSpanStart(spans[n]));
367                     if (lineNum < startLine + count) {
368                         useFirstLineMargin = true;
369                         break;
370                     }
371                 }
372             }
373             for (int n = 0; n < length; n++) {
374                 if (spans[n] instanceof LeadingMarginSpan) {
375                     LeadingMarginSpan margin = (LeadingMarginSpan) spans[n];
376                     if (dir == DIR_RIGHT_TO_LEFT) {
377                         right -= margin.getLeadingMargin(useFirstLineMargin);
378                     } else {
379                         left += margin.getLeadingMargin(useFirstLineMargin);
380                     }
381                 }
382             }
383         }
384 
385         final Alignment align;
386         if (paraAlign == Alignment.ALIGN_LEFT) {
387             align = (dir == DIR_LEFT_TO_RIGHT) ?  Alignment.ALIGN_NORMAL : Alignment.ALIGN_OPPOSITE;
388         } else if (paraAlign == Alignment.ALIGN_RIGHT) {
389             align = (dir == DIR_LEFT_TO_RIGHT) ?  Alignment.ALIGN_OPPOSITE : Alignment.ALIGN_NORMAL;
390         } else {
391             align = paraAlign;
392         }
393 
394         final int indentWidth;
395         if (align == Alignment.ALIGN_NORMAL) {
396             if (dir == DIR_LEFT_TO_RIGHT) {
397                 indentWidth = getIndentAdjust(lineNum, Alignment.ALIGN_LEFT);
398             } else {
399                 indentWidth = -getIndentAdjust(lineNum, Alignment.ALIGN_RIGHT);
400             }
401         } else if (align == Alignment.ALIGN_OPPOSITE) {
402             if (dir == DIR_LEFT_TO_RIGHT) {
403                 indentWidth = -getIndentAdjust(lineNum, Alignment.ALIGN_RIGHT);
404             } else {
405                 indentWidth = getIndentAdjust(lineNum, Alignment.ALIGN_LEFT);
406             }
407         } else { // Alignment.ALIGN_CENTER
408             indentWidth = getIndentAdjust(lineNum, Alignment.ALIGN_CENTER);
409         }
410 
411         return right - left - indentWidth;
412     }
413 
414     /**
415      * @hide
416      */
417     @UnsupportedAppUsage
drawText(Canvas canvas, int firstLine, int lastLine)418     public void drawText(Canvas canvas, int firstLine, int lastLine) {
419         int previousLineBottom = getLineTop(firstLine);
420         int previousLineEnd = getLineStart(firstLine);
421         ParagraphStyle[] spans = NO_PARA_SPANS;
422         int spanEnd = 0;
423         final TextPaint paint = mWorkPaint;
424         paint.set(mPaint);
425         CharSequence buf = mText;
426 
427         Alignment paraAlign = mAlignment;
428         TabStops tabStops = null;
429         boolean tabStopsIsInitialized = false;
430 
431         TextLine tl = TextLine.obtain();
432 
433         // Draw the lines, one at a time.
434         // The baseline is the top of the following line minus the current line's descent.
435         for (int lineNum = firstLine; lineNum <= lastLine; lineNum++) {
436             int start = previousLineEnd;
437             previousLineEnd = getLineStart(lineNum + 1);
438             final boolean justify = isJustificationRequired(lineNum);
439             int end = getLineVisibleEnd(lineNum, start, previousLineEnd);
440             paint.setStartHyphenEdit(getStartHyphenEdit(lineNum));
441             paint.setEndHyphenEdit(getEndHyphenEdit(lineNum));
442 
443             int ltop = previousLineBottom;
444             int lbottom = getLineTop(lineNum + 1);
445             previousLineBottom = lbottom;
446             int lbaseline = lbottom - getLineDescent(lineNum);
447 
448             int dir = getParagraphDirection(lineNum);
449             int left = 0;
450             int right = mWidth;
451 
452             if (mSpannedText) {
453                 Spanned sp = (Spanned) buf;
454                 int textLength = buf.length();
455                 boolean isFirstParaLine = (start == 0 || buf.charAt(start - 1) == '\n');
456 
457                 // New batch of paragraph styles, collect into spans array.
458                 // Compute the alignment, last alignment style wins.
459                 // Reset tabStops, we'll rebuild if we encounter a line with
460                 // tabs.
461                 // We expect paragraph spans to be relatively infrequent, use
462                 // spanEnd so that we can check less frequently.  Since
463                 // paragraph styles ought to apply to entire paragraphs, we can
464                 // just collect the ones present at the start of the paragraph.
465                 // If spanEnd is before the end of the paragraph, that's not
466                 // our problem.
467                 if (start >= spanEnd && (lineNum == firstLine || isFirstParaLine)) {
468                     spanEnd = sp.nextSpanTransition(start, textLength,
469                                                     ParagraphStyle.class);
470                     spans = getParagraphSpans(sp, start, spanEnd, ParagraphStyle.class);
471 
472                     paraAlign = mAlignment;
473                     for (int n = spans.length - 1; n >= 0; n--) {
474                         if (spans[n] instanceof AlignmentSpan) {
475                             paraAlign = ((AlignmentSpan) spans[n]).getAlignment();
476                             break;
477                         }
478                     }
479 
480                     tabStopsIsInitialized = false;
481                 }
482 
483                 // Draw all leading margin spans.  Adjust left or right according
484                 // to the paragraph direction of the line.
485                 final int length = spans.length;
486                 boolean useFirstLineMargin = isFirstParaLine;
487                 for (int n = 0; n < length; n++) {
488                     if (spans[n] instanceof LeadingMarginSpan2) {
489                         int count = ((LeadingMarginSpan2) spans[n]).getLeadingMarginLineCount();
490                         int startLine = getLineForOffset(sp.getSpanStart(spans[n]));
491                         // if there is more than one LeadingMarginSpan2, use
492                         // the count that is greatest
493                         if (lineNum < startLine + count) {
494                             useFirstLineMargin = true;
495                             break;
496                         }
497                     }
498                 }
499                 for (int n = 0; n < length; n++) {
500                     if (spans[n] instanceof LeadingMarginSpan) {
501                         LeadingMarginSpan margin = (LeadingMarginSpan) spans[n];
502                         if (dir == DIR_RIGHT_TO_LEFT) {
503                             margin.drawLeadingMargin(canvas, paint, right, dir, ltop,
504                                                      lbaseline, lbottom, buf,
505                                                      start, end, isFirstParaLine, this);
506                             right -= margin.getLeadingMargin(useFirstLineMargin);
507                         } else {
508                             margin.drawLeadingMargin(canvas, paint, left, dir, ltop,
509                                                      lbaseline, lbottom, buf,
510                                                      start, end, isFirstParaLine, this);
511                             left += margin.getLeadingMargin(useFirstLineMargin);
512                         }
513                     }
514                 }
515             }
516 
517             boolean hasTab = getLineContainsTab(lineNum);
518             // Can't tell if we have tabs for sure, currently
519             if (hasTab && !tabStopsIsInitialized) {
520                 if (tabStops == null) {
521                     tabStops = new TabStops(TAB_INCREMENT, spans);
522                 } else {
523                     tabStops.reset(TAB_INCREMENT, spans);
524                 }
525                 tabStopsIsInitialized = true;
526             }
527 
528             // Determine whether the line aligns to normal, opposite, or center.
529             Alignment align = paraAlign;
530             if (align == Alignment.ALIGN_LEFT) {
531                 align = (dir == DIR_LEFT_TO_RIGHT) ?
532                     Alignment.ALIGN_NORMAL : Alignment.ALIGN_OPPOSITE;
533             } else if (align == Alignment.ALIGN_RIGHT) {
534                 align = (dir == DIR_LEFT_TO_RIGHT) ?
535                     Alignment.ALIGN_OPPOSITE : Alignment.ALIGN_NORMAL;
536             }
537 
538             int x;
539             final int indentWidth;
540             if (align == Alignment.ALIGN_NORMAL) {
541                 if (dir == DIR_LEFT_TO_RIGHT) {
542                     indentWidth = getIndentAdjust(lineNum, Alignment.ALIGN_LEFT);
543                     x = left + indentWidth;
544                 } else {
545                     indentWidth = -getIndentAdjust(lineNum, Alignment.ALIGN_RIGHT);
546                     x = right - indentWidth;
547                 }
548             } else {
549                 int max = (int)getLineExtent(lineNum, tabStops, false);
550                 if (align == Alignment.ALIGN_OPPOSITE) {
551                     if (dir == DIR_LEFT_TO_RIGHT) {
552                         indentWidth = -getIndentAdjust(lineNum, Alignment.ALIGN_RIGHT);
553                         x = right - max - indentWidth;
554                     } else {
555                         indentWidth = getIndentAdjust(lineNum, Alignment.ALIGN_LEFT);
556                         x = left - max + indentWidth;
557                     }
558                 } else { // Alignment.ALIGN_CENTER
559                     indentWidth = getIndentAdjust(lineNum, Alignment.ALIGN_CENTER);
560                     max = max & ~1;
561                     x = ((right + left - max) >> 1) + indentWidth;
562                 }
563             }
564 
565             Directions directions = getLineDirections(lineNum);
566             if (directions == DIRS_ALL_LEFT_TO_RIGHT && !mSpannedText && !hasTab && !justify) {
567                 // XXX: assumes there's nothing additional to be done
568                 canvas.drawText(buf, start, end, x, lbaseline, paint);
569             } else {
570                 tl.set(paint, buf, start, end, dir, directions, hasTab, tabStops,
571                         getEllipsisStart(lineNum),
572                         getEllipsisStart(lineNum) + getEllipsisCount(lineNum));
573                 if (justify) {
574                     tl.justify(right - left - indentWidth);
575                 }
576                 tl.draw(canvas, x, ltop, lbaseline, lbottom);
577             }
578         }
579 
580         TextLine.recycle(tl);
581     }
582 
583     /**
584      * @hide
585      */
586     @UnsupportedAppUsage
drawBackground(Canvas canvas, Path highlight, Paint highlightPaint, int cursorOffsetVertical, int firstLine, int lastLine)587     public void drawBackground(Canvas canvas, Path highlight, Paint highlightPaint,
588             int cursorOffsetVertical, int firstLine, int lastLine) {
589         // First, draw LineBackgroundSpans.
590         // LineBackgroundSpans know nothing about the alignment, margins, or
591         // direction of the layout or line.  XXX: Should they?
592         // They are evaluated at each line.
593         if (mSpannedText) {
594             if (mLineBackgroundSpans == null) {
595                 mLineBackgroundSpans = new SpanSet<LineBackgroundSpan>(LineBackgroundSpan.class);
596             }
597 
598             Spanned buffer = (Spanned) mText;
599             int textLength = buffer.length();
600             mLineBackgroundSpans.init(buffer, 0, textLength);
601 
602             if (mLineBackgroundSpans.numberOfSpans > 0) {
603                 int previousLineBottom = getLineTop(firstLine);
604                 int previousLineEnd = getLineStart(firstLine);
605                 ParagraphStyle[] spans = NO_PARA_SPANS;
606                 int spansLength = 0;
607                 TextPaint paint = mPaint;
608                 int spanEnd = 0;
609                 final int width = mWidth;
610                 for (int i = firstLine; i <= lastLine; i++) {
611                     int start = previousLineEnd;
612                     int end = getLineStart(i + 1);
613                     previousLineEnd = end;
614 
615                     int ltop = previousLineBottom;
616                     int lbottom = getLineTop(i + 1);
617                     previousLineBottom = lbottom;
618                     int lbaseline = lbottom - getLineDescent(i);
619 
620                     if (end >= spanEnd) {
621                         // These should be infrequent, so we'll use this so that
622                         // we don't have to check as often.
623                         spanEnd = mLineBackgroundSpans.getNextTransition(start, textLength);
624                         // All LineBackgroundSpans on a line contribute to its background.
625                         spansLength = 0;
626                         // Duplication of the logic of getParagraphSpans
627                         if (start != end || start == 0) {
628                             // Equivalent to a getSpans(start, end), but filling the 'spans' local
629                             // array instead to reduce memory allocation
630                             for (int j = 0; j < mLineBackgroundSpans.numberOfSpans; j++) {
631                                 // equal test is valid since both intervals are not empty by
632                                 // construction
633                                 if (mLineBackgroundSpans.spanStarts[j] >= end ||
634                                         mLineBackgroundSpans.spanEnds[j] <= start) continue;
635                                 spans = GrowingArrayUtils.append(
636                                         spans, spansLength, mLineBackgroundSpans.spans[j]);
637                                 spansLength++;
638                             }
639                         }
640                     }
641 
642                     for (int n = 0; n < spansLength; n++) {
643                         LineBackgroundSpan lineBackgroundSpan = (LineBackgroundSpan) spans[n];
644                         lineBackgroundSpan.drawBackground(canvas, paint, 0, width,
645                                 ltop, lbaseline, lbottom,
646                                 buffer, start, end, i);
647                     }
648                 }
649             }
650             mLineBackgroundSpans.recycle();
651         }
652 
653         // There can be a highlight even without spans if we are drawing
654         // a non-spanned transformation of a spanned editing buffer.
655         if (highlight != null) {
656             if (cursorOffsetVertical != 0) canvas.translate(0, cursorOffsetVertical);
657             canvas.drawPath(highlight, highlightPaint);
658             if (cursorOffsetVertical != 0) canvas.translate(0, -cursorOffsetVertical);
659         }
660     }
661 
662     /**
663      * @param canvas
664      * @return The range of lines that need to be drawn, possibly empty.
665      * @hide
666      */
667     @UnsupportedAppUsage
getLineRangeForDraw(Canvas canvas)668     public long getLineRangeForDraw(Canvas canvas) {
669         int dtop, dbottom;
670 
671         synchronized (sTempRect) {
672             if (!canvas.getClipBounds(sTempRect)) {
673                 // Negative range end used as a special flag
674                 return TextUtils.packRangeInLong(0, -1);
675             }
676 
677             dtop = sTempRect.top;
678             dbottom = sTempRect.bottom;
679         }
680 
681         final int top = Math.max(dtop, 0);
682         final int bottom = Math.min(getLineTop(getLineCount()), dbottom);
683 
684         if (top >= bottom) return TextUtils.packRangeInLong(0, -1);
685         return TextUtils.packRangeInLong(getLineForVertical(top), getLineForVertical(bottom));
686     }
687 
688     /**
689      * Return the start position of the line, given the left and right bounds
690      * of the margins.
691      *
692      * @param line the line index
693      * @param left the left bounds (0, or leading margin if ltr para)
694      * @param right the right bounds (width, minus leading margin if rtl para)
695      * @return the start position of the line (to right of line if rtl para)
696      */
getLineStartPos(int line, int left, int right)697     private int getLineStartPos(int line, int left, int right) {
698         // Adjust the point at which to start rendering depending on the
699         // alignment of the paragraph.
700         Alignment align = getParagraphAlignment(line);
701         int dir = getParagraphDirection(line);
702 
703         if (align == Alignment.ALIGN_LEFT) {
704             align = (dir == DIR_LEFT_TO_RIGHT) ? Alignment.ALIGN_NORMAL : Alignment.ALIGN_OPPOSITE;
705         } else if (align == Alignment.ALIGN_RIGHT) {
706             align = (dir == DIR_LEFT_TO_RIGHT) ? Alignment.ALIGN_OPPOSITE : Alignment.ALIGN_NORMAL;
707         }
708 
709         int x;
710         if (align == Alignment.ALIGN_NORMAL) {
711             if (dir == DIR_LEFT_TO_RIGHT) {
712                 x = left + getIndentAdjust(line, Alignment.ALIGN_LEFT);
713             } else {
714                 x = right + getIndentAdjust(line, Alignment.ALIGN_RIGHT);
715             }
716         } else {
717             TabStops tabStops = null;
718             if (mSpannedText && getLineContainsTab(line)) {
719                 Spanned spanned = (Spanned) mText;
720                 int start = getLineStart(line);
721                 int spanEnd = spanned.nextSpanTransition(start, spanned.length(),
722                         TabStopSpan.class);
723                 TabStopSpan[] tabSpans = getParagraphSpans(spanned, start, spanEnd,
724                         TabStopSpan.class);
725                 if (tabSpans.length > 0) {
726                     tabStops = new TabStops(TAB_INCREMENT, tabSpans);
727                 }
728             }
729             int max = (int)getLineExtent(line, tabStops, false);
730             if (align == Alignment.ALIGN_OPPOSITE) {
731                 if (dir == DIR_LEFT_TO_RIGHT) {
732                     x = right - max + getIndentAdjust(line, Alignment.ALIGN_RIGHT);
733                 } else {
734                     // max is negative here
735                     x = left - max + getIndentAdjust(line, Alignment.ALIGN_LEFT);
736                 }
737             } else { // Alignment.ALIGN_CENTER
738                 max = max & ~1;
739                 x = (left + right - max) >> 1 + getIndentAdjust(line, Alignment.ALIGN_CENTER);
740             }
741         }
742         return x;
743     }
744 
745     /**
746      * Return the text that is displayed by this Layout.
747      */
getText()748     public final CharSequence getText() {
749         return mText;
750     }
751 
752     /**
753      * Return the base Paint properties for this layout.
754      * Do NOT change the paint, which may result in funny
755      * drawing for this layout.
756      */
getPaint()757     public final TextPaint getPaint() {
758         return mPaint;
759     }
760 
761     /**
762      * Return the width of this layout.
763      */
getWidth()764     public final int getWidth() {
765         return mWidth;
766     }
767 
768     /**
769      * Return the width to which this Layout is ellipsizing, or
770      * {@link #getWidth} if it is not doing anything special.
771      */
getEllipsizedWidth()772     public int getEllipsizedWidth() {
773         return mWidth;
774     }
775 
776     /**
777      * Increase the width of this layout to the specified width.
778      * Be careful to use this only when you know it is appropriate&mdash;
779      * it does not cause the text to reflow to use the full new width.
780      */
increaseWidthTo(int wid)781     public final void increaseWidthTo(int wid) {
782         if (wid < mWidth) {
783             throw new RuntimeException("attempted to reduce Layout width");
784         }
785 
786         mWidth = wid;
787     }
788 
789     /**
790      * Return the total height of this layout.
791      */
getHeight()792     public int getHeight() {
793         return getLineTop(getLineCount());
794     }
795 
796     /**
797      * Return the total height of this layout.
798      *
799      * @param cap if true and max lines is set, returns the height of the layout at the max lines.
800      *
801      * @hide
802      */
getHeight(boolean cap)803     public int getHeight(boolean cap) {
804         return getHeight();
805     }
806 
807     /**
808      * Return the base alignment of this layout.
809      */
getAlignment()810     public final Alignment getAlignment() {
811         return mAlignment;
812     }
813 
814     /**
815      * Return what the text height is multiplied by to get the line height.
816      */
getSpacingMultiplier()817     public final float getSpacingMultiplier() {
818         return mSpacingMult;
819     }
820 
821     /**
822      * Return the number of units of leading that are added to each line.
823      */
getSpacingAdd()824     public final float getSpacingAdd() {
825         return mSpacingAdd;
826     }
827 
828     /**
829      * Return the heuristic used to determine paragraph text direction.
830      * @hide
831      */
getTextDirectionHeuristic()832     public final TextDirectionHeuristic getTextDirectionHeuristic() {
833         return mTextDir;
834     }
835 
836     /**
837      * Return the number of lines of text in this layout.
838      */
getLineCount()839     public abstract int getLineCount();
840 
841     /**
842      * Return the baseline for the specified line (0&hellip;getLineCount() - 1)
843      * If bounds is not null, return the top, left, right, bottom extents
844      * of the specified line in it.
845      * @param line which line to examine (0..getLineCount() - 1)
846      * @param bounds Optional. If not null, it returns the extent of the line
847      * @return the Y-coordinate of the baseline
848      */
getLineBounds(int line, Rect bounds)849     public int getLineBounds(int line, Rect bounds) {
850         if (bounds != null) {
851             bounds.left = 0;     // ???
852             bounds.top = getLineTop(line);
853             bounds.right = mWidth;   // ???
854             bounds.bottom = getLineTop(line + 1);
855         }
856         return getLineBaseline(line);
857     }
858 
859     /**
860      * Return the vertical position of the top of the specified line
861      * (0&hellip;getLineCount()).
862      * If the specified line is equal to the line count, returns the
863      * bottom of the last line.
864      */
getLineTop(int line)865     public abstract int getLineTop(int line);
866 
867     /**
868      * Return the descent of the specified line(0&hellip;getLineCount() - 1).
869      */
getLineDescent(int line)870     public abstract int getLineDescent(int line);
871 
872     /**
873      * Return the text offset of the beginning of the specified line (
874      * 0&hellip;getLineCount()). If the specified line is equal to the line
875      * count, returns the length of the text.
876      */
getLineStart(int line)877     public abstract int getLineStart(int line);
878 
879     /**
880      * Returns the primary directionality of the paragraph containing the
881      * specified line, either 1 for left-to-right lines, or -1 for right-to-left
882      * lines (see {@link #DIR_LEFT_TO_RIGHT}, {@link #DIR_RIGHT_TO_LEFT}).
883      */
getParagraphDirection(int line)884     public abstract int getParagraphDirection(int line);
885 
886     /**
887      * Returns whether the specified line contains one or more
888      * characters that need to be handled specially, like tabs.
889      */
getLineContainsTab(int line)890     public abstract boolean getLineContainsTab(int line);
891 
892     /**
893      * Returns the directional run information for the specified line.
894      * The array alternates counts of characters in left-to-right
895      * and right-to-left segments of the line.
896      *
897      * <p>NOTE: this is inadequate to support bidirectional text, and will change.
898      */
getLineDirections(int line)899     public abstract Directions getLineDirections(int line);
900 
901     /**
902      * Returns the (negative) number of extra pixels of ascent padding in the
903      * top line of the Layout.
904      */
getTopPadding()905     public abstract int getTopPadding();
906 
907     /**
908      * Returns the number of extra pixels of descent padding in the
909      * bottom line of the Layout.
910      */
getBottomPadding()911     public abstract int getBottomPadding();
912 
913     /**
914      * Returns the start hyphen edit for a line.
915      *
916      * @hide
917      */
getStartHyphenEdit(int line)918     public @Paint.StartHyphenEdit int getStartHyphenEdit(int line) {
919         return Paint.START_HYPHEN_EDIT_NO_EDIT;
920     }
921 
922     /**
923      * Returns the end hyphen edit for a line.
924      *
925      * @hide
926      */
getEndHyphenEdit(int line)927     public @Paint.EndHyphenEdit int getEndHyphenEdit(int line) {
928         return Paint.END_HYPHEN_EDIT_NO_EDIT;
929     }
930 
931     /**
932      * Returns the left indent for a line.
933      *
934      * @hide
935      */
getIndentAdjust(int line, Alignment alignment)936     public int getIndentAdjust(int line, Alignment alignment) {
937         return 0;
938     }
939 
940     /**
941      * Returns true if the character at offset and the preceding character
942      * are at different run levels (and thus there's a split caret).
943      * @param offset the offset
944      * @return true if at a level boundary
945      * @hide
946      */
947     @UnsupportedAppUsage
isLevelBoundary(int offset)948     public boolean isLevelBoundary(int offset) {
949         int line = getLineForOffset(offset);
950         Directions dirs = getLineDirections(line);
951         if (dirs == DIRS_ALL_LEFT_TO_RIGHT || dirs == DIRS_ALL_RIGHT_TO_LEFT) {
952             return false;
953         }
954 
955         int[] runs = dirs.mDirections;
956         int lineStart = getLineStart(line);
957         int lineEnd = getLineEnd(line);
958         if (offset == lineStart || offset == lineEnd) {
959             int paraLevel = getParagraphDirection(line) == 1 ? 0 : 1;
960             int runIndex = offset == lineStart ? 0 : runs.length - 2;
961             return ((runs[runIndex + 1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK) != paraLevel;
962         }
963 
964         offset -= lineStart;
965         for (int i = 0; i < runs.length; i += 2) {
966             if (offset == runs[i]) {
967                 return true;
968             }
969         }
970         return false;
971     }
972 
973     /**
974      * Returns true if the character at offset is right to left (RTL).
975      * @param offset the offset
976      * @return true if the character is RTL, false if it is LTR
977      */
isRtlCharAt(int offset)978     public boolean isRtlCharAt(int offset) {
979         int line = getLineForOffset(offset);
980         Directions dirs = getLineDirections(line);
981         if (dirs == DIRS_ALL_LEFT_TO_RIGHT) {
982             return false;
983         }
984         if (dirs == DIRS_ALL_RIGHT_TO_LEFT) {
985             return  true;
986         }
987         int[] runs = dirs.mDirections;
988         int lineStart = getLineStart(line);
989         for (int i = 0; i < runs.length; i += 2) {
990             int start = lineStart + runs[i];
991             int limit = start + (runs[i+1] & RUN_LENGTH_MASK);
992             if (offset >= start && offset < limit) {
993                 int level = (runs[i+1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK;
994                 return ((level & 1) != 0);
995             }
996         }
997         // Should happen only if the offset is "out of bounds"
998         return false;
999     }
1000 
1001     /**
1002      * Returns the range of the run that the character at offset belongs to.
1003      * @param offset the offset
1004      * @return The range of the run
1005      * @hide
1006      */
getRunRange(int offset)1007     public long getRunRange(int offset) {
1008         int line = getLineForOffset(offset);
1009         Directions dirs = getLineDirections(line);
1010         if (dirs == DIRS_ALL_LEFT_TO_RIGHT || dirs == DIRS_ALL_RIGHT_TO_LEFT) {
1011             return TextUtils.packRangeInLong(0, getLineEnd(line));
1012         }
1013         int[] runs = dirs.mDirections;
1014         int lineStart = getLineStart(line);
1015         for (int i = 0; i < runs.length; i += 2) {
1016             int start = lineStart + runs[i];
1017             int limit = start + (runs[i+1] & RUN_LENGTH_MASK);
1018             if (offset >= start && offset < limit) {
1019                 return TextUtils.packRangeInLong(start, limit);
1020             }
1021         }
1022         // Should happen only if the offset is "out of bounds"
1023         return TextUtils.packRangeInLong(0, getLineEnd(line));
1024     }
1025 
1026     /**
1027      * Checks if the trailing BiDi level should be used for an offset
1028      *
1029      * This method is useful when the offset is at the BiDi level transition point and determine
1030      * which run need to be used. For example, let's think about following input: (L* denotes
1031      * Left-to-Right characters, R* denotes Right-to-Left characters.)
1032      * Input (Logical Order): L1 L2 L3 R1 R2 R3 L4 L5 L6
1033      * Input (Display Order): L1 L2 L3 R3 R2 R1 L4 L5 L6
1034      *
1035      * Then, think about selecting the range (3, 6). The offset=3 and offset=6 are ambiguous here
1036      * since they are at the BiDi transition point.  In Android, the offset is considered to be
1037      * associated with the trailing run if the BiDi level of the trailing run is higher than of the
1038      * previous run.  In this case, the BiDi level of the input text is as follows:
1039      *
1040      * Input (Logical Order): L1 L2 L3 R1 R2 R3 L4 L5 L6
1041      *              BiDi Run: [ Run 0 ][ Run 1 ][ Run 2 ]
1042      *            BiDi Level:  0  0  0  1  1  1  0  0  0
1043      *
1044      * Thus, offset = 3 is part of Run 1 and this method returns true for offset = 3, since the BiDi
1045      * level of Run 1 is higher than the level of Run 0.  Similarly, the offset = 6 is a part of Run
1046      * 1 and this method returns false for the offset = 6 since the BiDi level of Run 1 is higher
1047      * than the level of Run 2.
1048      *
1049      * @returns true if offset is at the BiDi level transition point and trailing BiDi level is
1050      *          higher than previous BiDi level. See above for the detail.
1051      * @hide
1052      */
1053     @VisibleForTesting
primaryIsTrailingPrevious(int offset)1054     public boolean primaryIsTrailingPrevious(int offset) {
1055         int line = getLineForOffset(offset);
1056         int lineStart = getLineStart(line);
1057         int lineEnd = getLineEnd(line);
1058         int[] runs = getLineDirections(line).mDirections;
1059 
1060         int levelAt = -1;
1061         for (int i = 0; i < runs.length; i += 2) {
1062             int start = lineStart + runs[i];
1063             int limit = start + (runs[i+1] & RUN_LENGTH_MASK);
1064             if (limit > lineEnd) {
1065                 limit = lineEnd;
1066             }
1067             if (offset >= start && offset < limit) {
1068                 if (offset > start) {
1069                     // Previous character is at same level, so don't use trailing.
1070                     return false;
1071                 }
1072                 levelAt = (runs[i+1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK;
1073                 break;
1074             }
1075         }
1076         if (levelAt == -1) {
1077             // Offset was limit of line.
1078             levelAt = getParagraphDirection(line) == 1 ? 0 : 1;
1079         }
1080 
1081         // At level boundary, check previous level.
1082         int levelBefore = -1;
1083         if (offset == lineStart) {
1084             levelBefore = getParagraphDirection(line) == 1 ? 0 : 1;
1085         } else {
1086             offset -= 1;
1087             for (int i = 0; i < runs.length; i += 2) {
1088                 int start = lineStart + runs[i];
1089                 int limit = start + (runs[i+1] & RUN_LENGTH_MASK);
1090                 if (limit > lineEnd) {
1091                     limit = lineEnd;
1092                 }
1093                 if (offset >= start && offset < limit) {
1094                     levelBefore = (runs[i+1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK;
1095                     break;
1096                 }
1097             }
1098         }
1099 
1100         return levelBefore < levelAt;
1101     }
1102 
1103     /**
1104      * Computes in linear time the results of calling
1105      * #primaryIsTrailingPrevious for all offsets on a line.
1106      * @param line The line giving the offsets we compute the information for
1107      * @return The array of results, indexed from 0, where 0 corresponds to the line start offset
1108      * @hide
1109      */
1110     @VisibleForTesting
primaryIsTrailingPreviousAllLineOffsets(int line)1111     public boolean[] primaryIsTrailingPreviousAllLineOffsets(int line) {
1112         int lineStart = getLineStart(line);
1113         int lineEnd = getLineEnd(line);
1114         int[] runs = getLineDirections(line).mDirections;
1115 
1116         boolean[] trailing = new boolean[lineEnd - lineStart + 1];
1117 
1118         byte[] level = new byte[lineEnd - lineStart + 1];
1119         for (int i = 0; i < runs.length; i += 2) {
1120             int start = lineStart + runs[i];
1121             int limit = start + (runs[i + 1] & RUN_LENGTH_MASK);
1122             if (limit > lineEnd) {
1123                 limit = lineEnd;
1124             }
1125             if (limit == start) {
1126                 continue;
1127             }
1128             level[limit - lineStart - 1] =
1129                     (byte) ((runs[i + 1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK);
1130         }
1131 
1132         for (int i = 0; i < runs.length; i += 2) {
1133             int start = lineStart + runs[i];
1134             byte currentLevel = (byte) ((runs[i + 1] >>> RUN_LEVEL_SHIFT) & RUN_LEVEL_MASK);
1135             trailing[start - lineStart] = currentLevel > (start == lineStart
1136                     ? (getParagraphDirection(line) == 1 ? 0 : 1)
1137                     : level[start - lineStart - 1]);
1138         }
1139 
1140         return trailing;
1141     }
1142 
1143     /**
1144      * Get the primary horizontal position for the specified text offset.
1145      * This is the location where a new character would be inserted in
1146      * the paragraph's primary direction.
1147      */
getPrimaryHorizontal(int offset)1148     public float getPrimaryHorizontal(int offset) {
1149         return getPrimaryHorizontal(offset, false /* not clamped */);
1150     }
1151 
1152     /**
1153      * Get the primary horizontal position for the specified text offset, but
1154      * optionally clamp it so that it doesn't exceed the width of the layout.
1155      * @hide
1156      */
1157     @UnsupportedAppUsage
getPrimaryHorizontal(int offset, boolean clamped)1158     public float getPrimaryHorizontal(int offset, boolean clamped) {
1159         boolean trailing = primaryIsTrailingPrevious(offset);
1160         return getHorizontal(offset, trailing, clamped);
1161     }
1162 
1163     /**
1164      * Get the secondary horizontal position for the specified text offset.
1165      * This is the location where a new character would be inserted in
1166      * the direction other than the paragraph's primary direction.
1167      */
getSecondaryHorizontal(int offset)1168     public float getSecondaryHorizontal(int offset) {
1169         return getSecondaryHorizontal(offset, false /* not clamped */);
1170     }
1171 
1172     /**
1173      * Get the secondary horizontal position for the specified text offset, but
1174      * optionally clamp it so that it doesn't exceed the width of the layout.
1175      * @hide
1176      */
1177     @UnsupportedAppUsage
getSecondaryHorizontal(int offset, boolean clamped)1178     public float getSecondaryHorizontal(int offset, boolean clamped) {
1179         boolean trailing = primaryIsTrailingPrevious(offset);
1180         return getHorizontal(offset, !trailing, clamped);
1181     }
1182 
getHorizontal(int offset, boolean primary)1183     private float getHorizontal(int offset, boolean primary) {
1184         return primary ? getPrimaryHorizontal(offset) : getSecondaryHorizontal(offset);
1185     }
1186 
getHorizontal(int offset, boolean trailing, boolean clamped)1187     private float getHorizontal(int offset, boolean trailing, boolean clamped) {
1188         int line = getLineForOffset(offset);
1189 
1190         return getHorizontal(offset, trailing, line, clamped);
1191     }
1192 
getHorizontal(int offset, boolean trailing, int line, boolean clamped)1193     private float getHorizontal(int offset, boolean trailing, int line, boolean clamped) {
1194         int start = getLineStart(line);
1195         int end = getLineEnd(line);
1196         int dir = getParagraphDirection(line);
1197         boolean hasTab = getLineContainsTab(line);
1198         Directions directions = getLineDirections(line);
1199 
1200         TabStops tabStops = null;
1201         if (hasTab && mText instanceof Spanned) {
1202             // Just checking this line should be good enough, tabs should be
1203             // consistent across all lines in a paragraph.
1204             TabStopSpan[] tabs = getParagraphSpans((Spanned) mText, start, end, TabStopSpan.class);
1205             if (tabs.length > 0) {
1206                 tabStops = new TabStops(TAB_INCREMENT, tabs); // XXX should reuse
1207             }
1208         }
1209 
1210         TextLine tl = TextLine.obtain();
1211         tl.set(mPaint, mText, start, end, dir, directions, hasTab, tabStops,
1212                 getEllipsisStart(line), getEllipsisStart(line) + getEllipsisCount(line));
1213         float wid = tl.measure(offset - start, trailing, null);
1214         TextLine.recycle(tl);
1215 
1216         if (clamped && wid > mWidth) {
1217             wid = mWidth;
1218         }
1219         int left = getParagraphLeft(line);
1220         int right = getParagraphRight(line);
1221 
1222         return getLineStartPos(line, left, right) + wid;
1223     }
1224 
1225     /**
1226      * Computes in linear time the results of calling #getHorizontal for all offsets on a line.
1227      *
1228      * @param line The line giving the offsets we compute information for
1229      * @param clamped Whether to clamp the results to the width of the layout
1230      * @param primary Whether the results should be the primary or the secondary horizontal
1231      * @return The array of results, indexed from 0, where 0 corresponds to the line start offset
1232      */
getLineHorizontals(int line, boolean clamped, boolean primary)1233     private float[] getLineHorizontals(int line, boolean clamped, boolean primary) {
1234         int start = getLineStart(line);
1235         int end = getLineEnd(line);
1236         int dir = getParagraphDirection(line);
1237         boolean hasTab = getLineContainsTab(line);
1238         Directions directions = getLineDirections(line);
1239 
1240         TabStops tabStops = null;
1241         if (hasTab && mText instanceof Spanned) {
1242             // Just checking this line should be good enough, tabs should be
1243             // consistent across all lines in a paragraph.
1244             TabStopSpan[] tabs = getParagraphSpans((Spanned) mText, start, end, TabStopSpan.class);
1245             if (tabs.length > 0) {
1246                 tabStops = new TabStops(TAB_INCREMENT, tabs); // XXX should reuse
1247             }
1248         }
1249 
1250         TextLine tl = TextLine.obtain();
1251         tl.set(mPaint, mText, start, end, dir, directions, hasTab, tabStops,
1252                 getEllipsisStart(line), getEllipsisStart(line) + getEllipsisCount(line));
1253         boolean[] trailings = primaryIsTrailingPreviousAllLineOffsets(line);
1254         if (!primary) {
1255             for (int offset = 0; offset < trailings.length; ++offset) {
1256                 trailings[offset] = !trailings[offset];
1257             }
1258         }
1259         float[] wid = tl.measureAllOffsets(trailings, null);
1260         TextLine.recycle(tl);
1261 
1262         if (clamped) {
1263             for (int offset = 0; offset < wid.length; ++offset) {
1264                 if (wid[offset] > mWidth) {
1265                     wid[offset] = mWidth;
1266                 }
1267             }
1268         }
1269         int left = getParagraphLeft(line);
1270         int right = getParagraphRight(line);
1271 
1272         int lineStartPos = getLineStartPos(line, left, right);
1273         float[] horizontal = new float[end - start + 1];
1274         for (int offset = 0; offset < horizontal.length; ++offset) {
1275             horizontal[offset] = lineStartPos + wid[offset];
1276         }
1277         return horizontal;
1278     }
1279 
1280     /**
1281      * Get the leftmost position that should be exposed for horizontal
1282      * scrolling on the specified line.
1283      */
getLineLeft(int line)1284     public float getLineLeft(int line) {
1285         final int dir = getParagraphDirection(line);
1286         Alignment align = getParagraphAlignment(line);
1287         // Before Q, StaticLayout.Builder.setAlignment didn't check whether the input alignment
1288         // is null. And when it is null, the old behavior is the same as ALIGN_CENTER.
1289         // To keep consistency, we convert a null alignment to ALIGN_CENTER.
1290         if (align == null) {
1291             align = Alignment.ALIGN_CENTER;
1292         }
1293 
1294         // First convert combinations of alignment and direction settings to
1295         // three basic cases: ALIGN_LEFT, ALIGN_RIGHT and ALIGN_CENTER.
1296         // For unexpected cases, it will fallback to ALIGN_LEFT.
1297         final Alignment resultAlign;
1298         switch(align) {
1299             case ALIGN_NORMAL:
1300                 resultAlign =
1301                         dir == DIR_RIGHT_TO_LEFT ? Alignment.ALIGN_RIGHT : Alignment.ALIGN_LEFT;
1302                 break;
1303             case ALIGN_OPPOSITE:
1304                 resultAlign =
1305                         dir == DIR_RIGHT_TO_LEFT ? Alignment.ALIGN_LEFT : Alignment.ALIGN_RIGHT;
1306                 break;
1307             case ALIGN_CENTER:
1308                 resultAlign = Alignment.ALIGN_CENTER;
1309                 break;
1310             case ALIGN_RIGHT:
1311                 resultAlign = Alignment.ALIGN_RIGHT;
1312                 break;
1313             default: /* align == Alignment.ALIGN_LEFT */
1314                 resultAlign = Alignment.ALIGN_LEFT;
1315         }
1316 
1317         // Here we must use getLineMax() to do the computation, because it maybe overridden by
1318         // derived class. And also note that line max equals the width of the text in that line
1319         // plus the leading margin.
1320         switch (resultAlign) {
1321             case ALIGN_CENTER:
1322                 final int left = getParagraphLeft(line);
1323                 final float max = getLineMax(line);
1324                 // This computation only works when mWidth equals leadingMargin plus
1325                 // the width of text in this line. If this condition doesn't meet anymore,
1326                 // please change here too.
1327                 return (float) Math.floor(left + (mWidth - max) / 2);
1328             case ALIGN_RIGHT:
1329                 return mWidth - getLineMax(line);
1330             default: /* resultAlign == Alignment.ALIGN_LEFT */
1331                 return 0;
1332         }
1333     }
1334 
1335     /**
1336      * Get the rightmost position that should be exposed for horizontal
1337      * scrolling on the specified line.
1338      */
getLineRight(int line)1339     public float getLineRight(int line) {
1340         final int dir = getParagraphDirection(line);
1341         Alignment align = getParagraphAlignment(line);
1342         // Before Q, StaticLayout.Builder.setAlignment didn't check whether the input alignment
1343         // is null. And when it is null, the old behavior is the same as ALIGN_CENTER.
1344         // To keep consistency, we convert a null alignment to ALIGN_CENTER.
1345         if (align == null) {
1346             align = Alignment.ALIGN_CENTER;
1347         }
1348 
1349         final Alignment resultAlign;
1350         switch(align) {
1351             case ALIGN_NORMAL:
1352                 resultAlign =
1353                         dir == DIR_RIGHT_TO_LEFT ? Alignment.ALIGN_RIGHT : Alignment.ALIGN_LEFT;
1354                 break;
1355             case ALIGN_OPPOSITE:
1356                 resultAlign =
1357                         dir == DIR_RIGHT_TO_LEFT ? Alignment.ALIGN_LEFT : Alignment.ALIGN_RIGHT;
1358                 break;
1359             case ALIGN_CENTER:
1360                 resultAlign = Alignment.ALIGN_CENTER;
1361                 break;
1362             case ALIGN_RIGHT:
1363                 resultAlign = Alignment.ALIGN_RIGHT;
1364                 break;
1365             default: /* align == Alignment.ALIGN_LEFT */
1366                 resultAlign = Alignment.ALIGN_LEFT;
1367         }
1368 
1369         switch (resultAlign) {
1370             case ALIGN_CENTER:
1371                 final int right = getParagraphRight(line);
1372                 final float max = getLineMax(line);
1373                 // This computation only works when mWidth equals leadingMargin plus width of the
1374                 // text in this line. If this condition doesn't meet anymore, please change here.
1375                 return (float) Math.ceil(right - (mWidth - max) / 2);
1376             case ALIGN_RIGHT:
1377                 return mWidth;
1378             default: /* resultAlign == Alignment.ALIGN_LEFT */
1379                 return getLineMax(line);
1380         }
1381     }
1382 
1383     /**
1384      * Gets the unsigned horizontal extent of the specified line, including
1385      * leading margin indent, but excluding trailing whitespace.
1386      */
getLineMax(int line)1387     public float getLineMax(int line) {
1388         float margin = getParagraphLeadingMargin(line);
1389         float signedExtent = getLineExtent(line, false);
1390         return margin + (signedExtent >= 0 ? signedExtent : -signedExtent);
1391     }
1392 
1393     /**
1394      * Gets the unsigned horizontal extent of the specified line, including
1395      * leading margin indent and trailing whitespace.
1396      */
getLineWidth(int line)1397     public float getLineWidth(int line) {
1398         float margin = getParagraphLeadingMargin(line);
1399         float signedExtent = getLineExtent(line, true);
1400         return margin + (signedExtent >= 0 ? signedExtent : -signedExtent);
1401     }
1402 
1403     /**
1404      * Like {@link #getLineExtent(int,TabStops,boolean)} but determines the
1405      * tab stops instead of using the ones passed in.
1406      * @param line the index of the line
1407      * @param full whether to include trailing whitespace
1408      * @return the extent of the line
1409      */
getLineExtent(int line, boolean full)1410     private float getLineExtent(int line, boolean full) {
1411         final int start = getLineStart(line);
1412         final int end = full ? getLineEnd(line) : getLineVisibleEnd(line);
1413 
1414         final boolean hasTabs = getLineContainsTab(line);
1415         TabStops tabStops = null;
1416         if (hasTabs && mText instanceof Spanned) {
1417             // Just checking this line should be good enough, tabs should be
1418             // consistent across all lines in a paragraph.
1419             TabStopSpan[] tabs = getParagraphSpans((Spanned) mText, start, end, TabStopSpan.class);
1420             if (tabs.length > 0) {
1421                 tabStops = new TabStops(TAB_INCREMENT, tabs); // XXX should reuse
1422             }
1423         }
1424         final Directions directions = getLineDirections(line);
1425         // Returned directions can actually be null
1426         if (directions == null) {
1427             return 0f;
1428         }
1429         final int dir = getParagraphDirection(line);
1430 
1431         final TextLine tl = TextLine.obtain();
1432         final TextPaint paint = mWorkPaint;
1433         paint.set(mPaint);
1434         paint.setStartHyphenEdit(getStartHyphenEdit(line));
1435         paint.setEndHyphenEdit(getEndHyphenEdit(line));
1436         tl.set(paint, mText, start, end, dir, directions, hasTabs, tabStops,
1437                 getEllipsisStart(line), getEllipsisStart(line) + getEllipsisCount(line));
1438         if (isJustificationRequired(line)) {
1439             tl.justify(getJustifyWidth(line));
1440         }
1441         final float width = tl.metrics(null);
1442         TextLine.recycle(tl);
1443         return width;
1444     }
1445 
1446     /**
1447      * Returns the signed horizontal extent of the specified line, excluding
1448      * leading margin.  If full is false, excludes trailing whitespace.
1449      * @param line the index of the line
1450      * @param tabStops the tab stops, can be null if we know they're not used.
1451      * @param full whether to include trailing whitespace
1452      * @return the extent of the text on this line
1453      */
getLineExtent(int line, TabStops tabStops, boolean full)1454     private float getLineExtent(int line, TabStops tabStops, boolean full) {
1455         final int start = getLineStart(line);
1456         final int end = full ? getLineEnd(line) : getLineVisibleEnd(line);
1457         final boolean hasTabs = getLineContainsTab(line);
1458         final Directions directions = getLineDirections(line);
1459         final int dir = getParagraphDirection(line);
1460 
1461         final TextLine tl = TextLine.obtain();
1462         final TextPaint paint = mWorkPaint;
1463         paint.set(mPaint);
1464         paint.setStartHyphenEdit(getStartHyphenEdit(line));
1465         paint.setEndHyphenEdit(getEndHyphenEdit(line));
1466         tl.set(paint, mText, start, end, dir, directions, hasTabs, tabStops,
1467                 getEllipsisStart(line), getEllipsisStart(line) + getEllipsisCount(line));
1468         if (isJustificationRequired(line)) {
1469             tl.justify(getJustifyWidth(line));
1470         }
1471         final float width = tl.metrics(null);
1472         TextLine.recycle(tl);
1473         return width;
1474     }
1475 
1476     /**
1477      * Get the line number corresponding to the specified vertical position.
1478      * If you ask for a position above 0, you get 0; if you ask for a position
1479      * below the bottom of the text, you get the last line.
1480      */
1481     // FIXME: It may be faster to do a linear search for layouts without many lines.
getLineForVertical(int vertical)1482     public int getLineForVertical(int vertical) {
1483         int high = getLineCount(), low = -1, guess;
1484 
1485         while (high - low > 1) {
1486             guess = (high + low) / 2;
1487 
1488             if (getLineTop(guess) > vertical)
1489                 high = guess;
1490             else
1491                 low = guess;
1492         }
1493 
1494         if (low < 0)
1495             return 0;
1496         else
1497             return low;
1498     }
1499 
1500     /**
1501      * Get the line number on which the specified text offset appears.
1502      * If you ask for a position before 0, you get 0; if you ask for a position
1503      * beyond the end of the text, you get the last line.
1504      */
getLineForOffset(int offset)1505     public int getLineForOffset(int offset) {
1506         int high = getLineCount(), low = -1, guess;
1507 
1508         while (high - low > 1) {
1509             guess = (high + low) / 2;
1510 
1511             if (getLineStart(guess) > offset)
1512                 high = guess;
1513             else
1514                 low = guess;
1515         }
1516 
1517         if (low < 0) {
1518             return 0;
1519         } else {
1520             return low;
1521         }
1522     }
1523 
1524     /**
1525      * Get the character offset on the specified line whose position is
1526      * closest to the specified horizontal position.
1527      */
getOffsetForHorizontal(int line, float horiz)1528     public int getOffsetForHorizontal(int line, float horiz) {
1529         return getOffsetForHorizontal(line, horiz, true);
1530     }
1531 
1532     /**
1533      * Get the character offset on the specified line whose position is
1534      * closest to the specified horizontal position.
1535      *
1536      * @param line the line used to find the closest offset
1537      * @param horiz the horizontal position used to find the closest offset
1538      * @param primary whether to use the primary position or secondary position to find the offset
1539      *
1540      * @hide
1541      */
getOffsetForHorizontal(int line, float horiz, boolean primary)1542     public int getOffsetForHorizontal(int line, float horiz, boolean primary) {
1543         // TODO: use Paint.getOffsetForAdvance to avoid binary search
1544         final int lineEndOffset = getLineEnd(line);
1545         final int lineStartOffset = getLineStart(line);
1546 
1547         Directions dirs = getLineDirections(line);
1548 
1549         TextLine tl = TextLine.obtain();
1550         // XXX: we don't care about tabs as we just use TextLine#getOffsetToLeftRightOf here.
1551         tl.set(mPaint, mText, lineStartOffset, lineEndOffset, getParagraphDirection(line), dirs,
1552                 false, null,
1553                 getEllipsisStart(line), getEllipsisStart(line) + getEllipsisCount(line));
1554         final HorizontalMeasurementProvider horizontal =
1555                 new HorizontalMeasurementProvider(line, primary);
1556 
1557         final int max;
1558         if (line == getLineCount() - 1) {
1559             max = lineEndOffset;
1560         } else {
1561             max = tl.getOffsetToLeftRightOf(lineEndOffset - lineStartOffset,
1562                     !isRtlCharAt(lineEndOffset - 1)) + lineStartOffset;
1563         }
1564         int best = lineStartOffset;
1565         float bestdist = Math.abs(horizontal.get(lineStartOffset) - horiz);
1566 
1567         for (int i = 0; i < dirs.mDirections.length; i += 2) {
1568             int here = lineStartOffset + dirs.mDirections[i];
1569             int there = here + (dirs.mDirections[i+1] & RUN_LENGTH_MASK);
1570             boolean isRtl = (dirs.mDirections[i+1] & RUN_RTL_FLAG) != 0;
1571             int swap = isRtl ? -1 : 1;
1572 
1573             if (there > max)
1574                 there = max;
1575             int high = there - 1 + 1, low = here + 1 - 1, guess;
1576 
1577             while (high - low > 1) {
1578                 guess = (high + low) / 2;
1579                 int adguess = getOffsetAtStartOf(guess);
1580 
1581                 if (horizontal.get(adguess) * swap >= horiz * swap) {
1582                     high = guess;
1583                 } else {
1584                     low = guess;
1585                 }
1586             }
1587 
1588             if (low < here + 1)
1589                 low = here + 1;
1590 
1591             if (low < there) {
1592                 int aft = tl.getOffsetToLeftRightOf(low - lineStartOffset, isRtl) + lineStartOffset;
1593                 low = tl.getOffsetToLeftRightOf(aft - lineStartOffset, !isRtl) + lineStartOffset;
1594                 if (low >= here && low < there) {
1595                     float dist = Math.abs(horizontal.get(low) - horiz);
1596                     if (aft < there) {
1597                         float other = Math.abs(horizontal.get(aft) - horiz);
1598 
1599                         if (other < dist) {
1600                             dist = other;
1601                             low = aft;
1602                         }
1603                     }
1604 
1605                     if (dist < bestdist) {
1606                         bestdist = dist;
1607                         best = low;
1608                     }
1609                 }
1610             }
1611 
1612             float dist = Math.abs(horizontal.get(here) - horiz);
1613 
1614             if (dist < bestdist) {
1615                 bestdist = dist;
1616                 best = here;
1617             }
1618         }
1619 
1620         float dist = Math.abs(horizontal.get(max) - horiz);
1621 
1622         if (dist <= bestdist) {
1623             best = max;
1624         }
1625 
1626         TextLine.recycle(tl);
1627         return best;
1628     }
1629 
1630     /**
1631      * Responds to #getHorizontal queries, by selecting the better strategy between:
1632      * - calling #getHorizontal explicitly for each query
1633      * - precomputing all #getHorizontal measurements, and responding to any query in constant time
1634      * The first strategy is used for LTR-only text, while the second is used for all other cases.
1635      * The class is currently only used in #getOffsetForHorizontal, so reuse with care in other
1636      * contexts.
1637      */
1638     private class HorizontalMeasurementProvider {
1639         private final int mLine;
1640         private final boolean mPrimary;
1641 
1642         private float[] mHorizontals;
1643         private int mLineStartOffset;
1644 
HorizontalMeasurementProvider(final int line, final boolean primary)1645         HorizontalMeasurementProvider(final int line, final boolean primary) {
1646             mLine = line;
1647             mPrimary = primary;
1648             init();
1649         }
1650 
init()1651         private void init() {
1652             final Directions dirs = getLineDirections(mLine);
1653             if (dirs == DIRS_ALL_LEFT_TO_RIGHT) {
1654                 return;
1655             }
1656 
1657             mHorizontals = getLineHorizontals(mLine, false, mPrimary);
1658             mLineStartOffset = getLineStart(mLine);
1659         }
1660 
get(final int offset)1661         float get(final int offset) {
1662             final int index = offset - mLineStartOffset;
1663             if (mHorizontals == null || index < 0 || index >= mHorizontals.length) {
1664                 return getHorizontal(offset, mPrimary);
1665             } else {
1666                 return mHorizontals[index];
1667             }
1668         }
1669     }
1670 
1671     /**
1672      * Return the text offset after the last character on the specified line.
1673      */
getLineEnd(int line)1674     public final int getLineEnd(int line) {
1675         return getLineStart(line + 1);
1676     }
1677 
1678     /**
1679      * Return the text offset after the last visible character (so whitespace
1680      * is not counted) on the specified line.
1681      */
getLineVisibleEnd(int line)1682     public int getLineVisibleEnd(int line) {
1683         return getLineVisibleEnd(line, getLineStart(line), getLineStart(line+1));
1684     }
1685 
getLineVisibleEnd(int line, int start, int end)1686     private int getLineVisibleEnd(int line, int start, int end) {
1687         CharSequence text = mText;
1688         char ch;
1689         if (line == getLineCount() - 1) {
1690             return end;
1691         }
1692 
1693         for (; end > start; end--) {
1694             ch = text.charAt(end - 1);
1695 
1696             if (ch == '\n') {
1697                 return end - 1;
1698             }
1699 
1700             if (!TextLine.isLineEndSpace(ch)) {
1701                 break;
1702             }
1703 
1704         }
1705 
1706         return end;
1707     }
1708 
1709     /**
1710      * Return the vertical position of the bottom of the specified line.
1711      */
getLineBottom(int line)1712     public final int getLineBottom(int line) {
1713         return getLineTop(line + 1);
1714     }
1715 
1716     /**
1717      * Return the vertical position of the bottom of the specified line without the line spacing
1718      * added.
1719      *
1720      * @hide
1721      */
getLineBottomWithoutSpacing(int line)1722     public final int getLineBottomWithoutSpacing(int line) {
1723         return getLineTop(line + 1) - getLineExtra(line);
1724     }
1725 
1726     /**
1727      * Return the vertical position of the baseline of the specified line.
1728      */
getLineBaseline(int line)1729     public final int getLineBaseline(int line) {
1730         // getLineTop(line+1) == getLineBottom(line)
1731         return getLineTop(line+1) - getLineDescent(line);
1732     }
1733 
1734     /**
1735      * Get the ascent of the text on the specified line.
1736      * The return value is negative to match the Paint.ascent() convention.
1737      */
getLineAscent(int line)1738     public final int getLineAscent(int line) {
1739         // getLineTop(line+1) - getLineDescent(line) == getLineBaseLine(line)
1740         return getLineTop(line) - (getLineTop(line+1) - getLineDescent(line));
1741     }
1742 
1743     /**
1744      * Return the extra space added as a result of line spacing attributes
1745      * {@link #getSpacingAdd()} and {@link #getSpacingMultiplier()}. Default value is {@code zero}.
1746      *
1747      * @param line the index of the line, the value should be equal or greater than {@code zero}
1748      * @hide
1749      */
getLineExtra(@ntRangefrom = 0) int line)1750     public int getLineExtra(@IntRange(from = 0) int line) {
1751         return 0;
1752     }
1753 
getOffsetToLeftOf(int offset)1754     public int getOffsetToLeftOf(int offset) {
1755         return getOffsetToLeftRightOf(offset, true);
1756     }
1757 
getOffsetToRightOf(int offset)1758     public int getOffsetToRightOf(int offset) {
1759         return getOffsetToLeftRightOf(offset, false);
1760     }
1761 
getOffsetToLeftRightOf(int caret, boolean toLeft)1762     private int getOffsetToLeftRightOf(int caret, boolean toLeft) {
1763         int line = getLineForOffset(caret);
1764         int lineStart = getLineStart(line);
1765         int lineEnd = getLineEnd(line);
1766         int lineDir = getParagraphDirection(line);
1767 
1768         boolean lineChanged = false;
1769         boolean advance = toLeft == (lineDir == DIR_RIGHT_TO_LEFT);
1770         // if walking off line, look at the line we're headed to
1771         if (advance) {
1772             if (caret == lineEnd) {
1773                 if (line < getLineCount() - 1) {
1774                     lineChanged = true;
1775                     ++line;
1776                 } else {
1777                     return caret; // at very end, don't move
1778                 }
1779             }
1780         } else {
1781             if (caret == lineStart) {
1782                 if (line > 0) {
1783                     lineChanged = true;
1784                     --line;
1785                 } else {
1786                     return caret; // at very start, don't move
1787                 }
1788             }
1789         }
1790 
1791         if (lineChanged) {
1792             lineStart = getLineStart(line);
1793             lineEnd = getLineEnd(line);
1794             int newDir = getParagraphDirection(line);
1795             if (newDir != lineDir) {
1796                 // unusual case.  we want to walk onto the line, but it runs
1797                 // in a different direction than this one, so we fake movement
1798                 // in the opposite direction.
1799                 toLeft = !toLeft;
1800                 lineDir = newDir;
1801             }
1802         }
1803 
1804         Directions directions = getLineDirections(line);
1805 
1806         TextLine tl = TextLine.obtain();
1807         // XXX: we don't care about tabs
1808         tl.set(mPaint, mText, lineStart, lineEnd, lineDir, directions, false, null,
1809                 getEllipsisStart(line), getEllipsisStart(line) + getEllipsisCount(line));
1810         caret = lineStart + tl.getOffsetToLeftRightOf(caret - lineStart, toLeft);
1811         TextLine.recycle(tl);
1812         return caret;
1813     }
1814 
getOffsetAtStartOf(int offset)1815     private int getOffsetAtStartOf(int offset) {
1816         // XXX this probably should skip local reorderings and
1817         // zero-width characters, look at callers
1818         if (offset == 0)
1819             return 0;
1820 
1821         CharSequence text = mText;
1822         char c = text.charAt(offset);
1823 
1824         if (c >= '\uDC00' && c <= '\uDFFF') {
1825             char c1 = text.charAt(offset - 1);
1826 
1827             if (c1 >= '\uD800' && c1 <= '\uDBFF')
1828                 offset -= 1;
1829         }
1830 
1831         if (mSpannedText) {
1832             ReplacementSpan[] spans = ((Spanned) text).getSpans(offset, offset,
1833                                                        ReplacementSpan.class);
1834 
1835             for (int i = 0; i < spans.length; i++) {
1836                 int start = ((Spanned) text).getSpanStart(spans[i]);
1837                 int end = ((Spanned) text).getSpanEnd(spans[i]);
1838 
1839                 if (start < offset && end > offset)
1840                     offset = start;
1841             }
1842         }
1843 
1844         return offset;
1845     }
1846 
1847     /**
1848      * Determine whether we should clamp cursor position. Currently it's
1849      * only robust for left-aligned displays.
1850      * @hide
1851      */
1852     @UnsupportedAppUsage
shouldClampCursor(int line)1853     public boolean shouldClampCursor(int line) {
1854         // Only clamp cursor position in left-aligned displays.
1855         switch (getParagraphAlignment(line)) {
1856             case ALIGN_LEFT:
1857                 return true;
1858             case ALIGN_NORMAL:
1859                 return getParagraphDirection(line) > 0;
1860             default:
1861                 return false;
1862         }
1863 
1864     }
1865 
1866     /**
1867      * Fills in the specified Path with a representation of a cursor
1868      * at the specified offset.  This will often be a vertical line
1869      * but can be multiple discontinuous lines in text with multiple
1870      * directionalities.
1871      */
getCursorPath(final int point, final Path dest, final CharSequence editingBuffer)1872     public void getCursorPath(final int point, final Path dest, final CharSequence editingBuffer) {
1873         dest.reset();
1874 
1875         int line = getLineForOffset(point);
1876         int top = getLineTop(line);
1877         int bottom = getLineBottomWithoutSpacing(line);
1878 
1879         boolean clamped = shouldClampCursor(line);
1880         float h1 = getPrimaryHorizontal(point, clamped) - 0.5f;
1881 
1882         int caps = TextKeyListener.getMetaState(editingBuffer, TextKeyListener.META_SHIFT_ON) |
1883                    TextKeyListener.getMetaState(editingBuffer, TextKeyListener.META_SELECTING);
1884         int fn = TextKeyListener.getMetaState(editingBuffer, TextKeyListener.META_ALT_ON);
1885         int dist = 0;
1886 
1887         if (caps != 0 || fn != 0) {
1888             dist = (bottom - top) >> 2;
1889 
1890             if (fn != 0)
1891                 top += dist;
1892             if (caps != 0)
1893                 bottom -= dist;
1894         }
1895 
1896         if (h1 < 0.5f)
1897             h1 = 0.5f;
1898 
1899         dest.moveTo(h1, top);
1900         dest.lineTo(h1, bottom);
1901 
1902         if (caps == 2) {
1903             dest.moveTo(h1, bottom);
1904             dest.lineTo(h1 - dist, bottom + dist);
1905             dest.lineTo(h1, bottom);
1906             dest.lineTo(h1 + dist, bottom + dist);
1907         } else if (caps == 1) {
1908             dest.moveTo(h1, bottom);
1909             dest.lineTo(h1 - dist, bottom + dist);
1910 
1911             dest.moveTo(h1 - dist, bottom + dist - 0.5f);
1912             dest.lineTo(h1 + dist, bottom + dist - 0.5f);
1913 
1914             dest.moveTo(h1 + dist, bottom + dist);
1915             dest.lineTo(h1, bottom);
1916         }
1917 
1918         if (fn == 2) {
1919             dest.moveTo(h1, top);
1920             dest.lineTo(h1 - dist, top - dist);
1921             dest.lineTo(h1, top);
1922             dest.lineTo(h1 + dist, top - dist);
1923         } else if (fn == 1) {
1924             dest.moveTo(h1, top);
1925             dest.lineTo(h1 - dist, top - dist);
1926 
1927             dest.moveTo(h1 - dist, top - dist + 0.5f);
1928             dest.lineTo(h1 + dist, top - dist + 0.5f);
1929 
1930             dest.moveTo(h1 + dist, top - dist);
1931             dest.lineTo(h1, top);
1932         }
1933     }
1934 
addSelection(int line, int start, int end, int top, int bottom, SelectionRectangleConsumer consumer)1935     private void addSelection(int line, int start, int end,
1936             int top, int bottom, SelectionRectangleConsumer consumer) {
1937         int linestart = getLineStart(line);
1938         int lineend = getLineEnd(line);
1939         Directions dirs = getLineDirections(line);
1940 
1941         if (lineend > linestart && mText.charAt(lineend - 1) == '\n') {
1942             lineend--;
1943         }
1944 
1945         for (int i = 0; i < dirs.mDirections.length; i += 2) {
1946             int here = linestart + dirs.mDirections[i];
1947             int there = here + (dirs.mDirections[i + 1] & RUN_LENGTH_MASK);
1948 
1949             if (there > lineend) {
1950                 there = lineend;
1951             }
1952 
1953             if (start <= there && end >= here) {
1954                 int st = Math.max(start, here);
1955                 int en = Math.min(end, there);
1956 
1957                 if (st != en) {
1958                     float h1 = getHorizontal(st, false, line, false /* not clamped */);
1959                     float h2 = getHorizontal(en, true, line, false /* not clamped */);
1960 
1961                     float left = Math.min(h1, h2);
1962                     float right = Math.max(h1, h2);
1963 
1964                     final @TextSelectionLayout int layout =
1965                             ((dirs.mDirections[i + 1] & RUN_RTL_FLAG) != 0)
1966                                     ? TEXT_SELECTION_LAYOUT_RIGHT_TO_LEFT
1967                                     : TEXT_SELECTION_LAYOUT_LEFT_TO_RIGHT;
1968 
1969                     consumer.accept(left, top, right, bottom, layout);
1970                 }
1971             }
1972         }
1973     }
1974 
1975     /**
1976      * Fills in the specified Path with a representation of a highlight
1977      * between the specified offsets.  This will often be a rectangle
1978      * or a potentially discontinuous set of rectangles.  If the start
1979      * and end are the same, the returned path is empty.
1980      */
getSelectionPath(int start, int end, Path dest)1981     public void getSelectionPath(int start, int end, Path dest) {
1982         dest.reset();
1983         getSelection(start, end, (left, top, right, bottom, textSelectionLayout) ->
1984                 dest.addRect(left, top, right, bottom, Path.Direction.CW));
1985     }
1986 
1987     /**
1988      * Calculates the rectangles which should be highlighted to indicate a selection between start
1989      * and end and feeds them into the given {@link SelectionRectangleConsumer}.
1990      *
1991      * @param start    the starting index of the selection
1992      * @param end      the ending index of the selection
1993      * @param consumer the {@link SelectionRectangleConsumer} which will receive the generated
1994      *                 rectangles. It will be called every time a rectangle is generated.
1995      * @hide
1996      * @see #getSelectionPath(int, int, Path)
1997      */
getSelection(int start, int end, final SelectionRectangleConsumer consumer)1998     public final void getSelection(int start, int end, final SelectionRectangleConsumer consumer) {
1999         if (start == end) {
2000             return;
2001         }
2002 
2003         if (end < start) {
2004             int temp = end;
2005             end = start;
2006             start = temp;
2007         }
2008 
2009         final int startline = getLineForOffset(start);
2010         final int endline = getLineForOffset(end);
2011 
2012         int top = getLineTop(startline);
2013         int bottom = getLineBottomWithoutSpacing(endline);
2014 
2015         if (startline == endline) {
2016             addSelection(startline, start, end, top, bottom, consumer);
2017         } else {
2018             final float width = mWidth;
2019 
2020             addSelection(startline, start, getLineEnd(startline),
2021                     top, getLineBottom(startline), consumer);
2022 
2023             if (getParagraphDirection(startline) == DIR_RIGHT_TO_LEFT) {
2024                 consumer.accept(getLineLeft(startline), top, 0, getLineBottom(startline),
2025                         TEXT_SELECTION_LAYOUT_RIGHT_TO_LEFT);
2026             } else {
2027                 consumer.accept(getLineRight(startline), top, width, getLineBottom(startline),
2028                         TEXT_SELECTION_LAYOUT_LEFT_TO_RIGHT);
2029             }
2030 
2031             for (int i = startline + 1; i < endline; i++) {
2032                 top = getLineTop(i);
2033                 bottom = getLineBottom(i);
2034                 if (getParagraphDirection(i) == DIR_RIGHT_TO_LEFT) {
2035                     consumer.accept(0, top, width, bottom, TEXT_SELECTION_LAYOUT_RIGHT_TO_LEFT);
2036                 } else {
2037                     consumer.accept(0, top, width, bottom, TEXT_SELECTION_LAYOUT_LEFT_TO_RIGHT);
2038                 }
2039             }
2040 
2041             top = getLineTop(endline);
2042             bottom = getLineBottomWithoutSpacing(endline);
2043 
2044             addSelection(endline, getLineStart(endline), end, top, bottom, consumer);
2045 
2046             if (getParagraphDirection(endline) == DIR_RIGHT_TO_LEFT) {
2047                 consumer.accept(width, top, getLineRight(endline), bottom,
2048                         TEXT_SELECTION_LAYOUT_RIGHT_TO_LEFT);
2049             } else {
2050                 consumer.accept(0, top, getLineLeft(endline), bottom,
2051                         TEXT_SELECTION_LAYOUT_LEFT_TO_RIGHT);
2052             }
2053         }
2054     }
2055 
2056     /**
2057      * Get the alignment of the specified paragraph, taking into account
2058      * markup attached to it.
2059      */
getParagraphAlignment(int line)2060     public final Alignment getParagraphAlignment(int line) {
2061         Alignment align = mAlignment;
2062 
2063         if (mSpannedText) {
2064             Spanned sp = (Spanned) mText;
2065             AlignmentSpan[] spans = getParagraphSpans(sp, getLineStart(line),
2066                                                 getLineEnd(line),
2067                                                 AlignmentSpan.class);
2068 
2069             int spanLength = spans.length;
2070             if (spanLength > 0) {
2071                 align = spans[spanLength-1].getAlignment();
2072             }
2073         }
2074 
2075         return align;
2076     }
2077 
2078     /**
2079      * Get the left edge of the specified paragraph, inset by left margins.
2080      */
getParagraphLeft(int line)2081     public final int getParagraphLeft(int line) {
2082         int left = 0;
2083         int dir = getParagraphDirection(line);
2084         if (dir == DIR_RIGHT_TO_LEFT || !mSpannedText) {
2085             return left; // leading margin has no impact, or no styles
2086         }
2087         return getParagraphLeadingMargin(line);
2088     }
2089 
2090     /**
2091      * Get the right edge of the specified paragraph, inset by right margins.
2092      */
getParagraphRight(int line)2093     public final int getParagraphRight(int line) {
2094         int right = mWidth;
2095         int dir = getParagraphDirection(line);
2096         if (dir == DIR_LEFT_TO_RIGHT || !mSpannedText) {
2097             return right; // leading margin has no impact, or no styles
2098         }
2099         return right - getParagraphLeadingMargin(line);
2100     }
2101 
2102     /**
2103      * Returns the effective leading margin (unsigned) for this line,
2104      * taking into account LeadingMarginSpan and LeadingMarginSpan2.
2105      * @param line the line index
2106      * @return the leading margin of this line
2107      */
getParagraphLeadingMargin(int line)2108     private int getParagraphLeadingMargin(int line) {
2109         if (!mSpannedText) {
2110             return 0;
2111         }
2112         Spanned spanned = (Spanned) mText;
2113 
2114         int lineStart = getLineStart(line);
2115         int lineEnd = getLineEnd(line);
2116         int spanEnd = spanned.nextSpanTransition(lineStart, lineEnd,
2117                 LeadingMarginSpan.class);
2118         LeadingMarginSpan[] spans = getParagraphSpans(spanned, lineStart, spanEnd,
2119                                                 LeadingMarginSpan.class);
2120         if (spans.length == 0) {
2121             return 0; // no leading margin span;
2122         }
2123 
2124         int margin = 0;
2125 
2126         boolean useFirstLineMargin = lineStart == 0 || spanned.charAt(lineStart - 1) == '\n';
2127         for (int i = 0; i < spans.length; i++) {
2128             if (spans[i] instanceof LeadingMarginSpan2) {
2129                 int spStart = spanned.getSpanStart(spans[i]);
2130                 int spanLine = getLineForOffset(spStart);
2131                 int count = ((LeadingMarginSpan2) spans[i]).getLeadingMarginLineCount();
2132                 // if there is more than one LeadingMarginSpan2, use the count that is greatest
2133                 useFirstLineMargin |= line < spanLine + count;
2134             }
2135         }
2136         for (int i = 0; i < spans.length; i++) {
2137             LeadingMarginSpan span = spans[i];
2138             margin += span.getLeadingMargin(useFirstLineMargin);
2139         }
2140 
2141         return margin;
2142     }
2143 
2144     private static float measurePara(TextPaint paint, CharSequence text, int start, int end,
2145             TextDirectionHeuristic textDir) {
2146         MeasuredParagraph mt = null;
2147         TextLine tl = TextLine.obtain();
2148         try {
2149             mt = MeasuredParagraph.buildForBidi(text, start, end, textDir, mt);
2150             final char[] chars = mt.getChars();
2151             final int len = chars.length;
2152             final Directions directions = mt.getDirections(0, len);
2153             final int dir = mt.getParagraphDir();
2154             boolean hasTabs = false;
2155             TabStops tabStops = null;
2156             // leading margins should be taken into account when measuring a paragraph
2157             int margin = 0;
2158             if (text instanceof Spanned) {
2159                 Spanned spanned = (Spanned) text;
2160                 LeadingMarginSpan[] spans = getParagraphSpans(spanned, start, end,
2161                         LeadingMarginSpan.class);
2162                 for (LeadingMarginSpan lms : spans) {
2163                     margin += lms.getLeadingMargin(true);
2164                 }
2165             }
2166             for (int i = 0; i < len; ++i) {
2167                 if (chars[i] == '\t') {
2168                     hasTabs = true;
2169                     if (text instanceof Spanned) {
2170                         Spanned spanned = (Spanned) text;
2171                         int spanEnd = spanned.nextSpanTransition(start, end,
2172                                 TabStopSpan.class);
2173                         TabStopSpan[] spans = getParagraphSpans(spanned, start, spanEnd,
2174                                 TabStopSpan.class);
2175                         if (spans.length > 0) {
2176                             tabStops = new TabStops(TAB_INCREMENT, spans);
2177                         }
2178                     }
2179                     break;
2180                 }
2181             }
2182             tl.set(paint, text, start, end, dir, directions, hasTabs, tabStops,
2183                     0 /* ellipsisStart */, 0 /* ellipsisEnd */);
2184             return margin + Math.abs(tl.metrics(null));
2185         } finally {
2186             TextLine.recycle(tl);
2187             if (mt != null) {
2188                 mt.recycle();
2189             }
2190         }
2191     }
2192 
2193     /**
2194      * @hide
2195      */
2196     @VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
2197     public static class TabStops {
2198         private float[] mStops;
2199         private int mNumStops;
2200         private float mIncrement;
2201 
2202         public TabStops(float increment, Object[] spans) {
2203             reset(increment, spans);
2204         }
2205 
2206         void reset(float increment, Object[] spans) {
2207             this.mIncrement = increment;
2208 
2209             int ns = 0;
2210             if (spans != null) {
2211                 float[] stops = this.mStops;
2212                 for (Object o : spans) {
2213                     if (o instanceof TabStopSpan) {
2214                         if (stops == null) {
2215                             stops = new float[10];
2216                         } else if (ns == stops.length) {
2217                             float[] nstops = new float[ns * 2];
2218                             for (int i = 0; i < ns; ++i) {
2219                                 nstops[i] = stops[i];
2220                             }
2221                             stops = nstops;
2222                         }
2223                         stops[ns++] = ((TabStopSpan) o).getTabStop();
2224                     }
2225                 }
2226                 if (ns > 1) {
2227                     Arrays.sort(stops, 0, ns);
2228                 }
2229                 if (stops != this.mStops) {
2230                     this.mStops = stops;
2231                 }
2232             }
2233             this.mNumStops = ns;
2234         }
2235 
2236         float nextTab(float h) {
2237             int ns = this.mNumStops;
2238             if (ns > 0) {
2239                 float[] stops = this.mStops;
2240                 for (int i = 0; i < ns; ++i) {
2241                     float stop = stops[i];
2242                     if (stop > h) {
2243                         return stop;
2244                     }
2245                 }
2246             }
2247             return nextDefaultStop(h, mIncrement);
2248         }
2249 
2250         /**
2251          * Returns the position of next tab stop.
2252          */
2253         public static float nextDefaultStop(float h, float inc) {
2254             return ((int) ((h + inc) / inc)) * inc;
2255         }
2256     }
2257 
2258     /**
2259      * Returns the position of the next tab stop after h on the line.
2260      *
2261      * @param text the text
2262      * @param start start of the line
2263      * @param end limit of the line
2264      * @param h the current horizontal offset
2265      * @param tabs the tabs, can be null.  If it is null, any tabs in effect
2266      * on the line will be used.  If there are no tabs, a default offset
2267      * will be used to compute the tab stop.
2268      * @return the offset of the next tab stop.
2269      */
2270     /* package */ static float nextTab(CharSequence text, int start, int end,
2271                                        float h, Object[] tabs) {
2272         float nh = Float.MAX_VALUE;
2273         boolean alltabs = false;
2274 
2275         if (text instanceof Spanned) {
2276             if (tabs == null) {
2277                 tabs = getParagraphSpans((Spanned) text, start, end, TabStopSpan.class);
2278                 alltabs = true;
2279             }
2280 
2281             for (int i = 0; i < tabs.length; i++) {
2282                 if (!alltabs) {
2283                     if (!(tabs[i] instanceof TabStopSpan))
2284                         continue;
2285                 }
2286 
2287                 int where = ((TabStopSpan) tabs[i]).getTabStop();
2288 
2289                 if (where < nh && where > h)
2290                     nh = where;
2291             }
2292 
2293             if (nh != Float.MAX_VALUE)
2294                 return nh;
2295         }
2296 
2297         return ((int) ((h + TAB_INCREMENT) / TAB_INCREMENT)) * TAB_INCREMENT;
2298     }
2299 
2300     protected final boolean isSpanned() {
2301         return mSpannedText;
2302     }
2303 
2304     /**
2305      * Returns the same as <code>text.getSpans()</code>, except where
2306      * <code>start</code> and <code>end</code> are the same and are not
2307      * at the very beginning of the text, in which case an empty array
2308      * is returned instead.
2309      * <p>
2310      * This is needed because of the special case that <code>getSpans()</code>
2311      * on an empty range returns the spans adjacent to that range, which is
2312      * primarily for the sake of <code>TextWatchers</code> so they will get
2313      * notifications when text goes from empty to non-empty.  But it also
2314      * has the unfortunate side effect that if the text ends with an empty
2315      * paragraph, that paragraph accidentally picks up the styles of the
2316      * preceding paragraph (even though those styles will not be picked up
2317      * by new text that is inserted into the empty paragraph).
2318      * <p>
2319      * The reason it just checks whether <code>start</code> and <code>end</code>
2320      * is the same is that the only time a line can contain 0 characters
2321      * is if it is the final paragraph of the Layout; otherwise any line will
2322      * contain at least one printing or newline character.  The reason for the
2323      * additional check if <code>start</code> is greater than 0 is that
2324      * if the empty paragraph is the entire content of the buffer, paragraph
2325      * styles that are already applied to the buffer will apply to text that
2326      * is inserted into it.
2327      */
2328     /* package */static <T> T[] getParagraphSpans(Spanned text, int start, int end, Class<T> type) {
2329         if (start == end && start > 0) {
2330             return ArrayUtils.emptyArray(type);
2331         }
2332 
2333         if(text instanceof SpannableStringBuilder) {
2334             return ((SpannableStringBuilder) text).getSpans(start, end, type, false);
2335         } else {
2336             return text.getSpans(start, end, type);
2337         }
2338     }
2339 
2340     private void ellipsize(int start, int end, int line,
2341                            char[] dest, int destoff, TextUtils.TruncateAt method) {
2342         final int ellipsisCount = getEllipsisCount(line);
2343         if (ellipsisCount == 0) {
2344             return;
2345         }
2346         final int ellipsisStart = getEllipsisStart(line);
2347         final int lineStart = getLineStart(line);
2348 
2349         final String ellipsisString = TextUtils.getEllipsisString(method);
2350         final int ellipsisStringLen = ellipsisString.length();
2351         // Use the ellipsis string only if there are that at least as many characters to replace.
2352         final boolean useEllipsisString = ellipsisCount >= ellipsisStringLen;
2353         final int min = Math.max(0, start - ellipsisStart - lineStart);
2354         final int max = Math.min(ellipsisCount, end - ellipsisStart - lineStart);
2355 
2356         for (int i = min; i < max; i++) {
2357             final char c;
2358             if (useEllipsisString && i < ellipsisStringLen) {
2359                 c = ellipsisString.charAt(i);
2360             } else {
2361                 c = TextUtils.ELLIPSIS_FILLER;
2362             }
2363 
2364             final int a = i + ellipsisStart + lineStart;
2365             dest[destoff + a - start] = c;
2366         }
2367     }
2368 
2369     /**
2370      * Stores information about bidirectional (left-to-right or right-to-left)
2371      * text within the layout of a line.
2372      */
2373     public static class Directions {
2374         /**
2375          * Directions represents directional runs within a line of text. Runs are pairs of ints
2376          * listed in visual order, starting from the leading margin.  The first int of each pair is
2377          * the offset from the first character of the line to the start of the run.  The second int
2378          * represents both the length and level of the run. The length is in the lower bits,
2379          * accessed by masking with RUN_LENGTH_MASK.  The level is in the higher bits, accessed by
2380          * shifting by RUN_LEVEL_SHIFT and masking by RUN_LEVEL_MASK. To simply test for an RTL
2381          * direction, test the bit using RUN_RTL_FLAG, if set then the direction is rtl.
2382          * @hide
2383          */
2384         @VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
2385         public int[] mDirections;
2386 
2387         /**
2388          * @hide
2389          */
2390         @VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
Directions(int[] dirs)2391         public Directions(int[] dirs) {
2392             mDirections = dirs;
2393         }
2394 
2395         /**
2396          * Returns number of BiDi runs.
2397          *
2398          * @hide
2399          */
getRunCount()2400         public @IntRange(from = 0) int getRunCount() {
2401             return mDirections.length / 2;
2402         }
2403 
2404         /**
2405          * Returns the start offset of the BiDi run.
2406          *
2407          * @param runIndex the index of the BiDi run
2408          * @return the start offset of the BiDi run.
2409          * @hide
2410          */
getRunStart(@ntRangefrom = 0) int runIndex)2411         public @IntRange(from = 0) int getRunStart(@IntRange(from = 0) int runIndex) {
2412             return mDirections[runIndex * 2];
2413         }
2414 
2415         /**
2416          * Returns the length of the BiDi run.
2417          *
2418          * Note that this method may return too large number due to reducing the number of object
2419          * allocations. The too large number means the remaining part is assigned to this run. The
2420          * caller must clamp the returned value.
2421          *
2422          * @param runIndex the index of the BiDi run
2423          * @return the length of the BiDi run.
2424          * @hide
2425          */
getRunLength(@ntRangefrom = 0) int runIndex)2426         public @IntRange(from = 0) int getRunLength(@IntRange(from = 0) int runIndex) {
2427             return mDirections[runIndex * 2 + 1] & RUN_LENGTH_MASK;
2428         }
2429 
2430         /**
2431          * Returns true if the BiDi run is RTL.
2432          *
2433          * @param runIndex the index of the BiDi run
2434          * @return true if the BiDi run is RTL.
2435          * @hide
2436          */
isRunRtl(int runIndex)2437         public boolean isRunRtl(int runIndex) {
2438             return (mDirections[runIndex * 2 + 1] & RUN_RTL_FLAG) != 0;
2439         }
2440     }
2441 
2442     /**
2443      * Return the offset of the first character to be ellipsized away,
2444      * relative to the start of the line.  (So 0 if the beginning of the
2445      * line is ellipsized, not getLineStart().)
2446      */
2447     public abstract int getEllipsisStart(int line);
2448 
2449     /**
2450      * Returns the number of characters to be ellipsized away, or 0 if
2451      * no ellipsis is to take place.
2452      */
2453     public abstract int getEllipsisCount(int line);
2454 
2455     /* package */ static class Ellipsizer implements CharSequence, GetChars {
2456         /* package */ CharSequence mText;
2457         /* package */ Layout mLayout;
2458         /* package */ int mWidth;
2459         /* package */ TextUtils.TruncateAt mMethod;
2460 
Ellipsizer(CharSequence s)2461         public Ellipsizer(CharSequence s) {
2462             mText = s;
2463         }
2464 
charAt(int off)2465         public char charAt(int off) {
2466             char[] buf = TextUtils.obtain(1);
2467             getChars(off, off + 1, buf, 0);
2468             char ret = buf[0];
2469 
2470             TextUtils.recycle(buf);
2471             return ret;
2472         }
2473 
getChars(int start, int end, char[] dest, int destoff)2474         public void getChars(int start, int end, char[] dest, int destoff) {
2475             int line1 = mLayout.getLineForOffset(start);
2476             int line2 = mLayout.getLineForOffset(end);
2477 
2478             TextUtils.getChars(mText, start, end, dest, destoff);
2479 
2480             for (int i = line1; i <= line2; i++) {
2481                 mLayout.ellipsize(start, end, i, dest, destoff, mMethod);
2482             }
2483         }
2484 
length()2485         public int length() {
2486             return mText.length();
2487         }
2488 
subSequence(int start, int end)2489         public CharSequence subSequence(int start, int end) {
2490             char[] s = new char[end - start];
2491             getChars(start, end, s, 0);
2492             return new String(s);
2493         }
2494 
2495         @Override
toString()2496         public String toString() {
2497             char[] s = new char[length()];
2498             getChars(0, length(), s, 0);
2499             return new String(s);
2500         }
2501 
2502     }
2503 
2504     /* package */ static class SpannedEllipsizer extends Ellipsizer implements Spanned {
2505         private Spanned mSpanned;
2506 
SpannedEllipsizer(CharSequence display)2507         public SpannedEllipsizer(CharSequence display) {
2508             super(display);
2509             mSpanned = (Spanned) display;
2510         }
2511 
getSpans(int start, int end, Class<T> type)2512         public <T> T[] getSpans(int start, int end, Class<T> type) {
2513             return mSpanned.getSpans(start, end, type);
2514         }
2515 
getSpanStart(Object tag)2516         public int getSpanStart(Object tag) {
2517             return mSpanned.getSpanStart(tag);
2518         }
2519 
getSpanEnd(Object tag)2520         public int getSpanEnd(Object tag) {
2521             return mSpanned.getSpanEnd(tag);
2522         }
2523 
getSpanFlags(Object tag)2524         public int getSpanFlags(Object tag) {
2525             return mSpanned.getSpanFlags(tag);
2526         }
2527 
2528         @SuppressWarnings("rawtypes")
nextSpanTransition(int start, int limit, Class type)2529         public int nextSpanTransition(int start, int limit, Class type) {
2530             return mSpanned.nextSpanTransition(start, limit, type);
2531         }
2532 
2533         @Override
subSequence(int start, int end)2534         public CharSequence subSequence(int start, int end) {
2535             char[] s = new char[end - start];
2536             getChars(start, end, s, 0);
2537 
2538             SpannableString ss = new SpannableString(new String(s));
2539             TextUtils.copySpansFrom(mSpanned, start, end, Object.class, ss, 0);
2540             return ss;
2541         }
2542     }
2543 
2544     private CharSequence mText;
2545     @UnsupportedAppUsage
2546     private TextPaint mPaint;
2547     private TextPaint mWorkPaint = new TextPaint();
2548     private int mWidth;
2549     private Alignment mAlignment = Alignment.ALIGN_NORMAL;
2550     private float mSpacingMult;
2551     private float mSpacingAdd;
2552     private static final Rect sTempRect = new Rect();
2553     private boolean mSpannedText;
2554     private TextDirectionHeuristic mTextDir;
2555     private SpanSet<LineBackgroundSpan> mLineBackgroundSpans;
2556     private int mJustificationMode;
2557 
2558     /** @hide */
2559     @IntDef(prefix = { "DIR_" }, value = {
2560             DIR_LEFT_TO_RIGHT,
2561             DIR_RIGHT_TO_LEFT
2562     })
2563     @Retention(RetentionPolicy.SOURCE)
2564     public @interface Direction {}
2565 
2566     public static final int DIR_LEFT_TO_RIGHT = 1;
2567     public static final int DIR_RIGHT_TO_LEFT = -1;
2568 
2569     /* package */ static final int DIR_REQUEST_LTR = 1;
2570     /* package */ static final int DIR_REQUEST_RTL = -1;
2571     @UnsupportedAppUsage
2572     /* package */ static final int DIR_REQUEST_DEFAULT_LTR = 2;
2573     /* package */ static final int DIR_REQUEST_DEFAULT_RTL = -2;
2574 
2575     /* package */ static final int RUN_LENGTH_MASK = 0x03ffffff;
2576     /* package */ static final int RUN_LEVEL_SHIFT = 26;
2577     /* package */ static final int RUN_LEVEL_MASK = 0x3f;
2578     /* package */ static final int RUN_RTL_FLAG = 1 << RUN_LEVEL_SHIFT;
2579 
2580     public enum Alignment {
2581         ALIGN_NORMAL,
2582         ALIGN_OPPOSITE,
2583         ALIGN_CENTER,
2584         /** @hide */
2585         @UnsupportedAppUsage
2586         ALIGN_LEFT,
2587         /** @hide */
2588         @UnsupportedAppUsage
2589         ALIGN_RIGHT,
2590     }
2591 
2592     private static final float TAB_INCREMENT = 20;
2593 
2594     /** @hide */
2595     @VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
2596     @UnsupportedAppUsage
2597     public static final Directions DIRS_ALL_LEFT_TO_RIGHT =
2598         new Directions(new int[] { 0, RUN_LENGTH_MASK });
2599 
2600     /** @hide */
2601     @VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
2602     @UnsupportedAppUsage
2603     public static final Directions DIRS_ALL_RIGHT_TO_LEFT =
2604         new Directions(new int[] { 0, RUN_LENGTH_MASK | RUN_RTL_FLAG });
2605 
2606     /** @hide */
2607     @Retention(RetentionPolicy.SOURCE)
2608     @IntDef(prefix = { "TEXT_SELECTION_LAYOUT_" }, value = {
2609             TEXT_SELECTION_LAYOUT_RIGHT_TO_LEFT,
2610             TEXT_SELECTION_LAYOUT_LEFT_TO_RIGHT
2611     })
2612     public @interface TextSelectionLayout {}
2613 
2614     /** @hide */
2615     public static final int TEXT_SELECTION_LAYOUT_RIGHT_TO_LEFT = 0;
2616     /** @hide */
2617     public static final int TEXT_SELECTION_LAYOUT_LEFT_TO_RIGHT = 1;
2618 
2619     /** @hide */
2620     @FunctionalInterface
2621     public interface SelectionRectangleConsumer {
2622         /**
2623          * Performs this operation on the given rectangle.
2624          *
2625          * @param left   the left edge of the rectangle
2626          * @param top    the top edge of the rectangle
2627          * @param right  the right edge of the rectangle
2628          * @param bottom the bottom edge of the rectangle
2629          * @param textSelectionLayout the layout (RTL or LTR) of the text covered by this
2630          *                            selection rectangle
2631          */
2632         void accept(float left, float top, float right, float bottom,
2633                 @TextSelectionLayout int textSelectionLayout);
2634     }
2635 
2636 }
2637