| /* |
| * Copyright (C) 2006 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package android.text; |
| |
| import static com.android.text.flags.Flags.FLAG_FIX_LINE_HEIGHT_FOR_LOCALE; |
| import static com.android.text.flags.Flags.FLAG_USE_BOUNDS_FOR_WIDTH; |
| |
| import android.annotation.FlaggedApi; |
| import android.annotation.FloatRange; |
| import android.annotation.IntRange; |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.annotation.SuppressLint; |
| import android.compat.annotation.UnsupportedAppUsage; |
| import android.graphics.Paint; |
| import android.graphics.RectF; |
| import android.graphics.text.LineBreakConfig; |
| import android.graphics.text.LineBreaker; |
| import android.os.Build; |
| import android.os.Trace; |
| import android.text.style.LeadingMarginSpan; |
| import android.text.style.LeadingMarginSpan.LeadingMarginSpan2; |
| import android.text.style.LineHeightSpan; |
| import android.text.style.TabStopSpan; |
| import android.util.Log; |
| import android.util.Pools.SynchronizedPool; |
| |
| import com.android.internal.util.ArrayUtils; |
| import com.android.internal.util.GrowingArrayUtils; |
| |
| import java.util.Arrays; |
| |
| /** |
| * StaticLayout is a Layout for text that will not be edited after it |
| * is laid out. Use {@link DynamicLayout} for text that may change. |
| * <p>This is used by widgets to control text layout. You should not need |
| * to use this class directly unless you are implementing your own widget |
| * or custom display object, or would be tempted to call |
| * {@link android.graphics.Canvas#drawText(java.lang.CharSequence, int, int, |
| * float, float, android.graphics.Paint) |
| * Canvas.drawText()} directly.</p> |
| */ |
| public class StaticLayout extends Layout { |
| /* |
| * The break iteration is done in native code. The protocol for using the native code is as |
| * follows. |
| * |
| * First, call nInit to setup native line breaker object. Then, for each paragraph, do the |
| * following: |
| * |
| * - Create MeasuredParagraph by MeasuredParagraph.buildForStaticLayout which measures in |
| * native. |
| * - Run LineBreaker.computeLineBreaks() to obtain line breaks for the paragraph. |
| * |
| * After all paragraphs, call finish() to release expensive buffers. |
| */ |
| |
| static final String TAG = "StaticLayout"; |
| |
| /** |
| * Builder for static layouts. The builder is the preferred pattern for constructing |
| * StaticLayout objects and should be preferred over the constructors, particularly to access |
| * newer features. To build a static layout, first call {@link #obtain} with the required |
| * arguments (text, paint, and width), then call setters for optional parameters, and finally |
| * {@link #build} to build the StaticLayout object. Parameters not explicitly set will get |
| * default values. |
| */ |
| public final static class Builder { |
| private Builder() {} |
| |
| /** |
| * Obtain a builder for constructing StaticLayout objects. |
| * |
| * @param source The text to be laid out, optionally with spans |
| * @param start The index of the start of the text |
| * @param end The index + 1 of the end of the text |
| * @param paint The base paint used for layout |
| * @param width The width in pixels |
| * @return a builder object used for constructing the StaticLayout |
| */ |
| @NonNull |
| public static Builder obtain(@NonNull CharSequence source, @IntRange(from = 0) int start, |
| @IntRange(from = 0) int end, @NonNull TextPaint paint, |
| @IntRange(from = 0) int width) { |
| Builder b = sPool.acquire(); |
| if (b == null) { |
| b = new Builder(); |
| } |
| |
| // set default initial values |
| b.mText = source; |
| b.mStart = start; |
| b.mEnd = end; |
| b.mPaint = paint; |
| b.mWidth = width; |
| b.mAlignment = Alignment.ALIGN_NORMAL; |
| b.mTextDir = TextDirectionHeuristics.FIRSTSTRONG_LTR; |
| b.mSpacingMult = DEFAULT_LINESPACING_MULTIPLIER; |
| b.mSpacingAdd = DEFAULT_LINESPACING_ADDITION; |
| b.mIncludePad = true; |
| b.mFallbackLineSpacing = false; |
| b.mEllipsizedWidth = width; |
| b.mEllipsize = null; |
| b.mMaxLines = Integer.MAX_VALUE; |
| b.mBreakStrategy = Layout.BREAK_STRATEGY_SIMPLE; |
| b.mHyphenationFrequency = Layout.HYPHENATION_FREQUENCY_NONE; |
| b.mJustificationMode = Layout.JUSTIFICATION_MODE_NONE; |
| b.mLineBreakConfig = LineBreakConfig.NONE; |
| b.mMinimumFontMetrics = null; |
| return b; |
| } |
| |
| /** |
| * This method should be called after the layout is finished getting constructed and the |
| * builder needs to be cleaned up and returned to the pool. |
| */ |
| private static void recycle(@NonNull Builder b) { |
| b.mPaint = null; |
| b.mText = null; |
| b.mLeftIndents = null; |
| b.mRightIndents = null; |
| b.mMinimumFontMetrics = null; |
| sPool.release(b); |
| } |
| |
| // release any expensive state |
| /* package */ void finish() { |
| mText = null; |
| mPaint = null; |
| mLeftIndents = null; |
| mRightIndents = null; |
| mMinimumFontMetrics = null; |
| } |
| |
| public Builder setText(CharSequence source) { |
| return setText(source, 0, source.length()); |
| } |
| |
| /** |
| * Set the text. Only useful when re-using the builder, which is done for |
| * the internal implementation of {@link DynamicLayout} but not as part |
| * of normal {@link StaticLayout} usage. |
| * |
| * @param source The text to be laid out, optionally with spans |
| * @param start The index of the start of the text |
| * @param end The index + 1 of the end of the text |
| * @return this builder, useful for chaining |
| * |
| * @hide |
| */ |
| @NonNull |
| public Builder setText(@NonNull CharSequence source, int start, int end) { |
| mText = source; |
| mStart = start; |
| mEnd = end; |
| return this; |
| } |
| |
| /** |
| * Set the paint. Internal for reuse cases only. |
| * |
| * @param paint The base paint used for layout |
| * @return this builder, useful for chaining |
| * |
| * @hide |
| */ |
| @NonNull |
| public Builder setPaint(@NonNull TextPaint paint) { |
| mPaint = paint; |
| return this; |
| } |
| |
| /** |
| * Set the width. Internal for reuse cases only. |
| * |
| * @param width The width in pixels |
| * @return this builder, useful for chaining |
| * |
| * @hide |
| */ |
| @NonNull |
| public Builder setWidth(@IntRange(from = 0) int width) { |
| mWidth = width; |
| if (mEllipsize == null) { |
| mEllipsizedWidth = width; |
| } |
| return this; |
| } |
| |
| /** |
| * Set the alignment. The default is {@link Layout.Alignment#ALIGN_NORMAL}. |
| * |
| * @param alignment Alignment for the resulting {@link StaticLayout} |
| * @return this builder, useful for chaining |
| */ |
| @NonNull |
| public Builder setAlignment(@NonNull Alignment alignment) { |
| mAlignment = alignment; |
| return this; |
| } |
| |
| /** |
| * Set the text direction heuristic. The text direction heuristic is used to |
| * resolve text direction per-paragraph based on the input text. The default is |
| * {@link TextDirectionHeuristics#FIRSTSTRONG_LTR}. |
| * |
| * @param textDir text direction heuristic for resolving bidi behavior. |
| * @return this builder, useful for chaining |
| */ |
| @NonNull |
| public Builder setTextDirection(@NonNull TextDirectionHeuristic textDir) { |
| mTextDir = textDir; |
| return this; |
| } |
| |
| /** |
| * Set line spacing parameters. Each line will have its line spacing multiplied by |
| * {@code spacingMult} and then increased by {@code spacingAdd}. The default is 0.0 for |
| * {@code spacingAdd} and 1.0 for {@code spacingMult}. |
| * |
| * @param spacingAdd the amount of line spacing addition |
| * @param spacingMult the line spacing multiplier |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setLineSpacing |
| */ |
| @NonNull |
| public Builder setLineSpacing(float spacingAdd, @FloatRange(from = 0.0) float spacingMult) { |
| mSpacingAdd = spacingAdd; |
| mSpacingMult = spacingMult; |
| return this; |
| } |
| |
| /** |
| * Set whether to include extra space beyond font ascent and descent (which is |
| * needed to avoid clipping in some languages, such as Arabic and Kannada). The |
| * default is {@code true}. |
| * |
| * @param includePad whether to include padding |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setIncludeFontPadding |
| */ |
| @NonNull |
| public Builder setIncludePad(boolean includePad) { |
| mIncludePad = includePad; |
| return this; |
| } |
| |
| /** |
| * Set whether to respect the ascent and descent of the fallback fonts that are used in |
| * displaying the text (which is needed to avoid text from consecutive lines running into |
| * each other). If set, fallback fonts that end up getting used can increase the ascent |
| * and descent of the lines that they are used on. |
| * |
| * <p>For backward compatibility reasons, the default is {@code false}, but setting this to |
| * true is strongly recommended. It is required to be true if text could be in languages |
| * like Burmese or Tibetan where text is typically much taller or deeper than Latin text. |
| * |
| * @param useLineSpacingFromFallbacks whether to expand linespacing based on fallback fonts |
| * @return this builder, useful for chaining |
| */ |
| @NonNull |
| public Builder setUseLineSpacingFromFallbacks(boolean useLineSpacingFromFallbacks) { |
| mFallbackLineSpacing = useLineSpacingFromFallbacks; |
| return this; |
| } |
| |
| /** |
| * Set the width as used for ellipsizing purposes, if it differs from the |
| * normal layout width. The default is the {@code width} |
| * passed to {@link #obtain}. |
| * |
| * @param ellipsizedWidth width used for ellipsizing, in pixels |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setEllipsize |
| */ |
| @NonNull |
| public Builder setEllipsizedWidth(@IntRange(from = 0) int ellipsizedWidth) { |
| mEllipsizedWidth = ellipsizedWidth; |
| return this; |
| } |
| |
| /** |
| * Set ellipsizing on the layout. Causes words that are longer than the view |
| * is wide, or exceeding the number of lines (see #setMaxLines) in the case |
| * of {@link android.text.TextUtils.TruncateAt#END} or |
| * {@link android.text.TextUtils.TruncateAt#MARQUEE}, to be ellipsized instead |
| * of broken. The default is {@code null}, indicating no ellipsis is to be applied. |
| * |
| * @param ellipsize type of ellipsis behavior |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setEllipsize |
| */ |
| @NonNull |
| public Builder setEllipsize(@Nullable TextUtils.TruncateAt ellipsize) { |
| mEllipsize = ellipsize; |
| return this; |
| } |
| |
| /** |
| * Set maximum number of lines. This is particularly useful in the case of |
| * ellipsizing, where it changes the layout of the last line. The default is |
| * unlimited. |
| * |
| * @param maxLines maximum number of lines in the layout |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setMaxLines |
| */ |
| @NonNull |
| public Builder setMaxLines(@IntRange(from = 0) int maxLines) { |
| mMaxLines = maxLines; |
| return this; |
| } |
| |
| /** |
| * Set break strategy, useful for selecting high quality or balanced paragraph |
| * layout options. The default is {@link Layout#BREAK_STRATEGY_SIMPLE}. |
| * <p/> |
| * Enabling hyphenation with either using {@link Layout#HYPHENATION_FREQUENCY_NORMAL} or |
| * {@link Layout#HYPHENATION_FREQUENCY_FULL} while line breaking is set to one of |
| * {@link Layout#BREAK_STRATEGY_BALANCED}, {@link Layout#BREAK_STRATEGY_HIGH_QUALITY} |
| * improves the structure of text layout however has performance impact and requires more |
| * time to do the text layout. |
| * |
| * @param breakStrategy break strategy for paragraph layout |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setBreakStrategy |
| * @see #setHyphenationFrequency(int) |
| */ |
| @NonNull |
| public Builder setBreakStrategy(@BreakStrategy int breakStrategy) { |
| mBreakStrategy = breakStrategy; |
| return this; |
| } |
| |
| /** |
| * Set hyphenation frequency, to control the amount of automatic hyphenation used. The |
| * possible values are defined in {@link Layout}, by constants named with the pattern |
| * {@code HYPHENATION_FREQUENCY_*}. The default is |
| * {@link Layout#HYPHENATION_FREQUENCY_NONE}. |
| * <p/> |
| * Enabling hyphenation with either using {@link Layout#HYPHENATION_FREQUENCY_NORMAL} or |
| * {@link Layout#HYPHENATION_FREQUENCY_FULL} while line breaking is set to one of |
| * {@link Layout#BREAK_STRATEGY_BALANCED}, {@link Layout#BREAK_STRATEGY_HIGH_QUALITY} |
| * improves the structure of text layout however has performance impact and requires more |
| * time to do the text layout. |
| * |
| * @param hyphenationFrequency hyphenation frequency for the paragraph |
| * @return this builder, useful for chaining |
| * @see android.widget.TextView#setHyphenationFrequency |
| * @see #setBreakStrategy(int) |
| */ |
| @NonNull |
| public Builder setHyphenationFrequency(@HyphenationFrequency int hyphenationFrequency) { |
| mHyphenationFrequency = hyphenationFrequency; |
| return this; |
| } |
| |
| /** |
| * Set indents. Arguments are arrays holding an indent amount, one per line, measured in |
| * pixels. For lines past the last element in the array, the last element repeats. |
| * |
| * @param leftIndents array of indent values for left margin, in pixels |
| * @param rightIndents array of indent values for right margin, in pixels |
| * @return this builder, useful for chaining |
| */ |
| @NonNull |
| public Builder setIndents(@Nullable int[] leftIndents, @Nullable int[] rightIndents) { |
| mLeftIndents = leftIndents; |
| mRightIndents = rightIndents; |
| return this; |
| } |
| |
| /** |
| * Set paragraph justification mode. The default value is |
| * {@link Layout#JUSTIFICATION_MODE_NONE}. If the last line is too short for justification, |
| * the last line will be displayed with the alignment set by {@link #setAlignment}. |
| * When Justification mode is JUSTIFICATION_MODE_INTER_WORD, wordSpacing on the given |
| * {@link Paint} will be ignored. This behavior also affects Spans which change the |
| * wordSpacing. |
| * |
| * @param justificationMode justification mode for the paragraph. |
| * @return this builder, useful for chaining. |
| * @see Paint#setWordSpacing(float) |
| */ |
| @NonNull |
| public Builder setJustificationMode(@JustificationMode int justificationMode) { |
| mJustificationMode = justificationMode; |
| return this; |
| } |
| |
| /** |
| * Sets whether the line spacing should be applied for the last line. Default value is |
| * {@code false}. |
| * |
| * @hide |
| */ |
| @NonNull |
| /* package */ Builder setAddLastLineLineSpacing(boolean value) { |
| mAddLastLineLineSpacing = value; |
| return this; |
| } |
| |
| /** |
| * Set the line break configuration. The line break will be passed to native used for |
| * calculating the text wrapping. The default value of the line break style is |
| * {@link LineBreakConfig#LINE_BREAK_STYLE_NONE} |
| * |
| * @param lineBreakConfig the line break configuration for text wrapping. |
| * @return this builder, useful for chaining. |
| * @see android.widget.TextView#setLineBreakStyle |
| * @see android.widget.TextView#setLineBreakWordStyle |
| */ |
| @NonNull |
| public Builder setLineBreakConfig(@NonNull LineBreakConfig lineBreakConfig) { |
| mLineBreakConfig = lineBreakConfig; |
| return this; |
| } |
| |
| /** |
| * Set true for using width of bounding box as a source of automatic line breaking and |
| * drawing. |
| * |
| * If this value is false, the Layout determines the drawing offset and automatic line |
| * breaking based on total advances. By setting true, use all joined glyph's bounding boxes |
| * as a source of text width. |
| * |
| * If the font has glyphs that have negative bearing X or its xMax is greater than advance, |
| * the glyph clipping can happen because the drawing area may be bigger. By setting this to |
| * true, the Layout will reserve more spaces for drawing. |
| * |
| * @param useBoundsForWidth True for using bounding box, false for advances. |
| * @return this builder instance |
| * @see Layout#getUseBoundsForWidth() |
| * @see Layout.Builder#setUseBoundsForWidth(boolean) |
| */ |
| @NonNull |
| @FlaggedApi(FLAG_USE_BOUNDS_FOR_WIDTH) |
| public Builder setUseBoundsForWidth(boolean useBoundsForWidth) { |
| mUseBoundsForWidth = useBoundsForWidth; |
| return this; |
| } |
| |
| /** |
| * Set true for shifting the drawing x offset for showing overhang at the start position. |
| * |
| * This flag is ignored if the {@link #getUseBoundsForWidth()} is false. |
| * |
| * If this value is false, the Layout draws text from the zero even if there is a glyph |
| * stroke in a region where the x coordinate is negative. |
| * |
| * If this value is true, the Layout draws text with shifting the x coordinate of the |
| * drawing bounding box. |
| * |
| * This value is false by default. |
| * |
| * @param shiftDrawingOffsetForStartOverhang true for shifting the drawing offset for |
| * showing the stroke that is in the region where |
| * the x coordinate is negative. |
| * @see #setUseBoundsForWidth(boolean) |
| * @see #getUseBoundsForWidth() |
| */ |
| @NonNull |
| // The corresponding getter is getShiftDrawingOffsetForStartOverhang() |
| @SuppressLint("MissingGetterMatchingBuilder") |
| @FlaggedApi(FLAG_USE_BOUNDS_FOR_WIDTH) |
| public Builder setShiftDrawingOffsetForStartOverhang( |
| boolean shiftDrawingOffsetForStartOverhang) { |
| mShiftDrawingOffsetForStartOverhang = shiftDrawingOffsetForStartOverhang; |
| return this; |
| } |
| |
| /** |
| * Internal API that tells underlying line breaker that calculating bounding boxes even if |
| * the line break is performed with advances. This is useful for DynamicLayout internal |
| * implementation because it uses bounding box as well as advances. |
| * @hide |
| */ |
| public Builder setCalculateBounds(boolean value) { |
| mCalculateBounds = value; |
| return this; |
| } |
| |
| /** |
| * Set the minimum font metrics used for line spacing. |
| * |
| * <p> |
| * {@code null} is the default value. If {@code null} is set or left as default, the |
| * font metrics obtained by {@link Paint#getFontMetricsForLocale(Paint.FontMetrics)} is |
| * used. |
| * |
| * <p> |
| * The minimum meaning here is the minimum value of line spacing: maximum value of |
| * {@link Paint#ascent()}, minimum value of {@link Paint#descent()}. |
| * |
| * <p> |
| * By setting this value, each line will have minimum line spacing regardless of the text |
| * rendered. For example, usually Japanese script has larger vertical metrics than Latin |
| * script. By setting the metrics obtained by |
| * {@link Paint#getFontMetricsForLocale(Paint.FontMetrics)} for Japanese or leave it |
| * {@code null} if the Paint's locale is Japanese, the line spacing for Japanese is reserved |
| * if the text is an English text. If the vertical metrics of the text is larger than |
| * Japanese, for example Burmese, the bigger font metrics is used. |
| * |
| * @param minimumFontMetrics A minimum font metrics. Passing {@code null} for using the |
| * value obtained by |
| * {@link Paint#getFontMetricsForLocale(Paint.FontMetrics)} |
| * @see android.widget.TextView#setMinimumFontMetrics(Paint.FontMetrics) |
| * @see android.widget.TextView#getMinimumFontMetrics() |
| * @see Layout#getMinimumFontMetrics() |
| * @see Layout.Builder#setMinimumFontMetrics(Paint.FontMetrics) |
| * @see DynamicLayout.Builder#setMinimumFontMetrics(Paint.FontMetrics) |
| */ |
| @NonNull |
| @FlaggedApi(FLAG_FIX_LINE_HEIGHT_FOR_LOCALE) |
| public Builder setMinimumFontMetrics(@Nullable Paint.FontMetrics minimumFontMetrics) { |
| mMinimumFontMetrics = minimumFontMetrics; |
| return this; |
| } |
| |
| /** |
| * Build the {@link StaticLayout} after options have been set. |
| * |
| * <p>Note: the builder object must not be reused in any way after calling this |
| * method. Setting parameters after calling this method, or calling it a second |
| * time on the same builder object, will likely lead to unexpected results. |
| * |
| * @return the newly constructed {@link StaticLayout} object |
| */ |
| @NonNull |
| public StaticLayout build() { |
| StaticLayout result = new StaticLayout(this, mIncludePad, mEllipsize != null |
| ? COLUMNS_ELLIPSIZE : COLUMNS_NORMAL); |
| Builder.recycle(this); |
| return result; |
| } |
| |
| /** |
| * DO NOT USE THIS METHOD OTHER THAN DynamicLayout. |
| * |
| * This class generates a very weird StaticLayout only for getting a result of line break. |
| * Since DynamicLayout keeps StaticLayout reference in the static context for object |
| * recycling but keeping text reference in static context will end up with leaking Context |
| * due to TextWatcher via TextView. |
| * |
| * So, this is a dirty work around that creating StaticLayout without passing text reference |
| * to the super constructor, but calculating the text layout by calling generate function |
| * directly. |
| */ |
| /* package */ @NonNull StaticLayout buildPartialStaticLayoutForDynamicLayout( |
| boolean trackpadding, StaticLayout recycle) { |
| if (recycle == null) { |
| recycle = new StaticLayout(); |
| } |
| Trace.beginSection("Generating StaticLayout For DynamicLayout"); |
| try { |
| recycle.generate(this, mIncludePad, trackpadding); |
| } finally { |
| Trace.endSection(); |
| } |
| return recycle; |
| } |
| |
| private CharSequence mText; |
| private int mStart; |
| private int mEnd; |
| private TextPaint mPaint; |
| private int mWidth; |
| private Alignment mAlignment; |
| private TextDirectionHeuristic mTextDir; |
| private float mSpacingMult; |
| private float mSpacingAdd; |
| private boolean mIncludePad; |
| private boolean mFallbackLineSpacing; |
| private int mEllipsizedWidth; |
| private TextUtils.TruncateAt mEllipsize; |
| private int mMaxLines; |
| private int mBreakStrategy; |
| private int mHyphenationFrequency; |
| @Nullable private int[] mLeftIndents; |
| @Nullable private int[] mRightIndents; |
| private int mJustificationMode; |
| private boolean mAddLastLineLineSpacing; |
| private LineBreakConfig mLineBreakConfig = LineBreakConfig.NONE; |
| private boolean mUseBoundsForWidth; |
| private boolean mShiftDrawingOffsetForStartOverhang; |
| private boolean mCalculateBounds; |
| @Nullable private Paint.FontMetrics mMinimumFontMetrics; |
| |
| private final Paint.FontMetricsInt mFontMetricsInt = new Paint.FontMetricsInt(); |
| |
| private static final SynchronizedPool<Builder> sPool = new SynchronizedPool<>(3); |
| } |
| |
| /** |
| * DO NOT USE THIS CONSTRUCTOR OTHER THAN FOR DYNAMIC LAYOUT. |
| * See Builder#buildPartialStaticLayoutForDynamicLayout for the reason of this constructor. |
| */ |
| private StaticLayout() { |
| super( |
| null, // text |
| null, // paint |
| 0, // width |
| null, // alignment |
| null, // textDir |
| 1, // spacing multiplier |
| 0, // spacing amount |
| false, // include font padding |
| false, // fallback line spacing |
| 0, // ellipsized width |
| null, // ellipsize |
| 1, // maxLines |
| BREAK_STRATEGY_SIMPLE, |
| HYPHENATION_FREQUENCY_NONE, |
| null, // leftIndents |
| null, // rightIndents |
| JUSTIFICATION_MODE_NONE, |
| null, // lineBreakConfig, |
| false, // useBoundsForWidth |
| false, // shiftDrawingOffsetForStartOverhang |
| null // minimumFontMetrics |
| ); |
| |
| mColumns = COLUMNS_ELLIPSIZE; |
| mLineDirections = ArrayUtils.newUnpaddedArray(Directions.class, 2); |
| mLines = ArrayUtils.newUnpaddedIntArray(2 * mColumns); |
| } |
| |
| /** |
| * @deprecated Use {@link Builder} instead. |
| */ |
| @Deprecated |
| public StaticLayout(CharSequence source, TextPaint paint, |
| int width, |
| Alignment align, float spacingmult, float spacingadd, |
| boolean includepad) { |
| this(source, 0, source.length(), paint, width, align, |
| spacingmult, spacingadd, includepad); |
| } |
| |
| /** |
| * @deprecated Use {@link Builder} instead. |
| */ |
| @Deprecated |
| public StaticLayout(CharSequence source, int bufstart, int bufend, |
| TextPaint paint, int outerwidth, |
| Alignment align, |
| float spacingmult, float spacingadd, |
| boolean includepad) { |
| this(source, bufstart, bufend, paint, outerwidth, align, |
| spacingmult, spacingadd, includepad, null, 0); |
| } |
| |
| /** |
| * @deprecated Use {@link Builder} instead. |
| */ |
| @Deprecated |
| public StaticLayout(CharSequence source, int bufstart, int bufend, |
| TextPaint paint, int outerwidth, |
| Alignment align, |
| float spacingmult, float spacingadd, |
| boolean includepad, |
| TextUtils.TruncateAt ellipsize, int ellipsizedWidth) { |
| this(source, bufstart, bufend, paint, outerwidth, align, |
| TextDirectionHeuristics.FIRSTSTRONG_LTR, |
| spacingmult, spacingadd, includepad, ellipsize, ellipsizedWidth, Integer.MAX_VALUE); |
| } |
| |
| /** |
| * @hide |
| * @deprecated Use {@link Builder} instead. |
| */ |
| @Deprecated |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 117521430) |
| public StaticLayout(CharSequence source, int bufstart, int bufend, |
| TextPaint paint, int outerwidth, |
| Alignment align, TextDirectionHeuristic textDir, |
| float spacingmult, float spacingadd, |
| boolean includepad, |
| TextUtils.TruncateAt ellipsize, int ellipsizedWidth, int maxLines) { |
| this(Builder.obtain(source, bufstart, bufend, paint, outerwidth) |
| .setAlignment(align) |
| .setTextDirection(textDir) |
| .setLineSpacing(spacingadd, spacingmult) |
| .setIncludePad(includepad) |
| .setEllipsize(ellipsize) |
| .setEllipsizedWidth(ellipsizedWidth) |
| .setMaxLines(maxLines), includepad, |
| ellipsize != null ? COLUMNS_ELLIPSIZE : COLUMNS_NORMAL); |
| } |
| |
| private StaticLayout(Builder b, boolean trackPadding, int columnSize) { |
| super((b.mEllipsize == null) ? b.mText : (b.mText instanceof Spanned) |
| ? new SpannedEllipsizer(b.mText) : new Ellipsizer(b.mText), |
| b.mPaint, b.mWidth, b.mAlignment, b.mTextDir, b.mSpacingMult, b.mSpacingAdd, |
| b.mIncludePad, b.mFallbackLineSpacing, b.mEllipsizedWidth, b.mEllipsize, |
| b.mMaxLines, b.mBreakStrategy, b.mHyphenationFrequency, b.mLeftIndents, |
| b.mRightIndents, b.mJustificationMode, b.mLineBreakConfig, b.mUseBoundsForWidth, |
| b.mShiftDrawingOffsetForStartOverhang, b.mMinimumFontMetrics); |
| |
| mColumns = columnSize; |
| if (b.mEllipsize != null) { |
| Ellipsizer e = (Ellipsizer) getText(); |
| |
| e.mLayout = this; |
| e.mWidth = b.mEllipsizedWidth; |
| e.mMethod = b.mEllipsize; |
| } |
| |
| mLineDirections = ArrayUtils.newUnpaddedArray(Directions.class, 2); |
| mLines = ArrayUtils.newUnpaddedIntArray(2 * mColumns); |
| mMaximumVisibleLineCount = b.mMaxLines; |
| |
| mLeftIndents = b.mLeftIndents; |
| mRightIndents = b.mRightIndents; |
| |
| Trace.beginSection("Constructing StaticLayout"); |
| try { |
| generate(b, b.mIncludePad, trackPadding); |
| } finally { |
| Trace.endSection(); |
| } |
| } |
| |
| private static int getBaseHyphenationFrequency(int frequency) { |
| switch (frequency) { |
| case Layout.HYPHENATION_FREQUENCY_FULL: |
| case Layout.HYPHENATION_FREQUENCY_FULL_FAST: |
| return LineBreaker.HYPHENATION_FREQUENCY_FULL; |
| case Layout.HYPHENATION_FREQUENCY_NORMAL: |
| case Layout.HYPHENATION_FREQUENCY_NORMAL_FAST: |
| return LineBreaker.HYPHENATION_FREQUENCY_NORMAL; |
| case Layout.HYPHENATION_FREQUENCY_NONE: |
| default: |
| return LineBreaker.HYPHENATION_FREQUENCY_NONE; |
| } |
| } |
| |
| /* package */ void generate(Builder b, boolean includepad, boolean trackpad) { |
| final CharSequence source = b.mText; |
| final int bufStart = b.mStart; |
| final int bufEnd = b.mEnd; |
| TextPaint paint = b.mPaint; |
| int outerWidth = b.mWidth; |
| TextDirectionHeuristic textDir = b.mTextDir; |
| float spacingmult = b.mSpacingMult; |
| float spacingadd = b.mSpacingAdd; |
| float ellipsizedWidth = b.mEllipsizedWidth; |
| TextUtils.TruncateAt ellipsize = b.mEllipsize; |
| final boolean addLastLineSpacing = b.mAddLastLineLineSpacing; |
| |
| int lineBreakCapacity = 0; |
| int[] breaks = null; |
| float[] lineWidths = null; |
| float[] ascents = null; |
| float[] descents = null; |
| boolean[] hasTabs = null; |
| int[] hyphenEdits = null; |
| |
| mLineCount = 0; |
| mEllipsized = false; |
| mMaxLineHeight = mMaximumVisibleLineCount < 1 ? 0 : DEFAULT_MAX_LINE_HEIGHT; |
| mDrawingBounds = null; |
| boolean isFallbackLineSpacing = b.mFallbackLineSpacing; |
| |
| int v = 0; |
| boolean needMultiply = (spacingmult != 1 || spacingadd != 0); |
| |
| Paint.FontMetricsInt fm = b.mFontMetricsInt; |
| int[] chooseHtv = null; |
| |
| final int[] indents; |
| if (mLeftIndents != null || mRightIndents != null) { |
| final int leftLen = mLeftIndents == null ? 0 : mLeftIndents.length; |
| final int rightLen = mRightIndents == null ? 0 : mRightIndents.length; |
| final int indentsLen = Math.max(leftLen, rightLen); |
| indents = new int[indentsLen]; |
| for (int i = 0; i < leftLen; i++) { |
| indents[i] = mLeftIndents[i]; |
| } |
| for (int i = 0; i < rightLen; i++) { |
| indents[i] += mRightIndents[i]; |
| } |
| } else { |
| indents = null; |
| } |
| |
| int defaultTop; |
| final int defaultAscent; |
| final int defaultDescent; |
| int defaultBottom; |
| if (ClientFlags.fixLineHeightForLocale() && b.mMinimumFontMetrics != null) { |
| defaultTop = (int) Math.floor(b.mMinimumFontMetrics.top); |
| defaultAscent = Math.round(b.mMinimumFontMetrics.ascent); |
| defaultDescent = Math.round(b.mMinimumFontMetrics.descent); |
| defaultBottom = (int) Math.ceil(b.mMinimumFontMetrics.bottom); |
| |
| // Because the font metrics is provided by public APIs, adjust the top/bottom with |
| // ascent/descent: top must be smaller than ascent, bottom must be larger than descent. |
| defaultTop = Math.min(defaultTop, defaultAscent); |
| defaultBottom = Math.max(defaultBottom, defaultDescent); |
| } else { |
| defaultTop = 0; |
| defaultAscent = 0; |
| defaultDescent = 0; |
| defaultBottom = 0; |
| } |
| |
| final LineBreaker lineBreaker = new LineBreaker.Builder() |
| .setBreakStrategy(b.mBreakStrategy) |
| .setHyphenationFrequency(getBaseHyphenationFrequency(b.mHyphenationFrequency)) |
| // TODO: Support more justification mode, e.g. letter spacing, stretching. |
| .setJustificationMode(b.mJustificationMode) |
| .setIndents(indents) |
| .setUseBoundsForWidth(b.mUseBoundsForWidth) |
| .build(); |
| |
| LineBreaker.ParagraphConstraints constraints = |
| new LineBreaker.ParagraphConstraints(); |
| |
| PrecomputedText.ParagraphInfo[] paragraphInfo = null; |
| final Spanned spanned = (source instanceof Spanned) ? (Spanned) source : null; |
| if (source instanceof PrecomputedText) { |
| PrecomputedText precomputed = (PrecomputedText) source; |
| final @PrecomputedText.Params.CheckResultUsableResult int checkResult = |
| precomputed.checkResultUsable(bufStart, bufEnd, textDir, paint, |
| b.mBreakStrategy, b.mHyphenationFrequency, b.mLineBreakConfig); |
| switch (checkResult) { |
| case PrecomputedText.Params.UNUSABLE: |
| break; |
| case PrecomputedText.Params.NEED_RECOMPUTE: |
| final PrecomputedText.Params newParams = |
| new PrecomputedText.Params.Builder(paint) |
| .setBreakStrategy(b.mBreakStrategy) |
| .setHyphenationFrequency(b.mHyphenationFrequency) |
| .setTextDirection(textDir) |
| .setLineBreakConfig(b.mLineBreakConfig) |
| .build(); |
| precomputed = PrecomputedText.create(precomputed, newParams); |
| paragraphInfo = precomputed.getParagraphInfo(); |
| break; |
| case PrecomputedText.Params.USABLE: |
| // Some parameters are different from the ones when measured text is created. |
| paragraphInfo = precomputed.getParagraphInfo(); |
| break; |
| } |
| } |
| |
| if (paragraphInfo == null) { |
| final PrecomputedText.Params param = new PrecomputedText.Params(paint, |
| b.mLineBreakConfig, textDir, b.mBreakStrategy, b.mHyphenationFrequency); |
| paragraphInfo = PrecomputedText.createMeasuredParagraphs(source, param, bufStart, |
| bufEnd, false /* computeLayout */, b.mCalculateBounds); |
| } |
| |
| for (int paraIndex = 0; paraIndex < paragraphInfo.length; paraIndex++) { |
| final int paraStart = paraIndex == 0 |
| ? bufStart : paragraphInfo[paraIndex - 1].paragraphEnd; |
| final int paraEnd = paragraphInfo[paraIndex].paragraphEnd; |
| |
| int firstWidthLineCount = 1; |
| int firstWidth = outerWidth; |
| int restWidth = outerWidth; |
| |
| LineHeightSpan[] chooseHt = null; |
| if (spanned != null) { |
| LeadingMarginSpan[] sp = getParagraphSpans(spanned, paraStart, paraEnd, |
| LeadingMarginSpan.class); |
| for (int i = 0; i < sp.length; i++) { |
| LeadingMarginSpan lms = sp[i]; |
| firstWidth -= sp[i].getLeadingMargin(true); |
| restWidth -= sp[i].getLeadingMargin(false); |
| |
| // LeadingMarginSpan2 is odd. The count affects all |
| // leading margin spans, not just this particular one |
| if (lms instanceof LeadingMarginSpan2) { |
| LeadingMarginSpan2 lms2 = (LeadingMarginSpan2) lms; |
| firstWidthLineCount = Math.max(firstWidthLineCount, |
| lms2.getLeadingMarginLineCount()); |
| } |
| } |
| |
| chooseHt = getParagraphSpans(spanned, paraStart, paraEnd, LineHeightSpan.class); |
| |
| if (chooseHt.length == 0) { |
| chooseHt = null; // So that out() would not assume it has any contents |
| } else { |
| if (chooseHtv == null || chooseHtv.length < chooseHt.length) { |
| chooseHtv = ArrayUtils.newUnpaddedIntArray(chooseHt.length); |
| } |
| |
| for (int i = 0; i < chooseHt.length; i++) { |
| int o = spanned.getSpanStart(chooseHt[i]); |
| |
| if (o < paraStart) { |
| // starts in this layout, before the |
| // current paragraph |
| |
| chooseHtv[i] = getLineTop(getLineForOffset(o)); |
| } else { |
| // starts in this paragraph |
| |
| chooseHtv[i] = v; |
| } |
| } |
| } |
| } |
| // tab stop locations |
| float[] variableTabStops = null; |
| if (spanned != null) { |
| TabStopSpan[] spans = getParagraphSpans(spanned, paraStart, |
| paraEnd, TabStopSpan.class); |
| if (spans.length > 0) { |
| float[] stops = new float[spans.length]; |
| for (int i = 0; i < spans.length; i++) { |
| stops[i] = (float) spans[i].getTabStop(); |
| } |
| Arrays.sort(stops, 0, stops.length); |
| variableTabStops = stops; |
| } |
| } |
| |
| final MeasuredParagraph measuredPara = paragraphInfo[paraIndex].measured; |
| final char[] chs = measuredPara.getChars(); |
| final int[] spanEndCache = measuredPara.getSpanEndCache().getRawArray(); |
| final int[] fmCache = measuredPara.getFontMetrics().getRawArray(); |
| |
| constraints.setWidth(restWidth); |
| constraints.setIndent(firstWidth, firstWidthLineCount); |
| constraints.setTabStops(variableTabStops, TAB_INCREMENT); |
| |
| LineBreaker.Result res = lineBreaker.computeLineBreaks( |
| measuredPara.getMeasuredText(), constraints, mLineCount); |
| int breakCount = res.getLineCount(); |
| if (lineBreakCapacity < breakCount) { |
| lineBreakCapacity = breakCount; |
| breaks = new int[lineBreakCapacity]; |
| lineWidths = new float[lineBreakCapacity]; |
| ascents = new float[lineBreakCapacity]; |
| descents = new float[lineBreakCapacity]; |
| hasTabs = new boolean[lineBreakCapacity]; |
| hyphenEdits = new int[lineBreakCapacity]; |
| } |
| |
| for (int i = 0; i < breakCount; ++i) { |
| breaks[i] = res.getLineBreakOffset(i); |
| lineWidths[i] = res.getLineWidth(i); |
| ascents[i] = res.getLineAscent(i); |
| descents[i] = res.getLineDescent(i); |
| hasTabs[i] = res.hasLineTab(i); |
| hyphenEdits[i] = |
| packHyphenEdit(res.getStartLineHyphenEdit(i), res.getEndLineHyphenEdit(i)); |
| } |
| |
| final int remainingLineCount = mMaximumVisibleLineCount - mLineCount; |
| final boolean ellipsisMayBeApplied = ellipsize != null |
| && (ellipsize == TextUtils.TruncateAt.END |
| || (mMaximumVisibleLineCount == 1 |
| && ellipsize != TextUtils.TruncateAt.MARQUEE)); |
| if (0 < remainingLineCount && remainingLineCount < breakCount |
| && ellipsisMayBeApplied) { |
| // Calculate width |
| float width = 0; |
| boolean hasTab = false; // XXX May need to also have starting hyphen edit |
| for (int i = remainingLineCount - 1; i < breakCount; i++) { |
| if (i == breakCount - 1) { |
| width += lineWidths[i]; |
| } else { |
| for (int j = (i == 0 ? 0 : breaks[i - 1]); j < breaks[i]; j++) { |
| width += measuredPara.getCharWidthAt(j); |
| } |
| } |
| hasTab |= hasTabs[i]; |
| } |
| // Treat the last line and overflowed lines as a single line. |
| breaks[remainingLineCount - 1] = breaks[breakCount - 1]; |
| lineWidths[remainingLineCount - 1] = width; |
| hasTabs[remainingLineCount - 1] = hasTab; |
| |
| breakCount = remainingLineCount; |
| } |
| |
| // here is the offset of the starting character of the line we are currently |
| // measuring |
| int here = paraStart; |
| |
| int fmTop = defaultTop; |
| int fmBottom = defaultBottom; |
| int fmAscent = defaultAscent; |
| int fmDescent = defaultDescent; |
| int fmCacheIndex = 0; |
| int spanEndCacheIndex = 0; |
| int breakIndex = 0; |
| for (int spanStart = paraStart, spanEnd; spanStart < paraEnd; spanStart = spanEnd) { |
| // retrieve end of span |
| spanEnd = spanEndCache[spanEndCacheIndex++]; |
| |
| // retrieve cached metrics, order matches above |
| fm.top = fmCache[fmCacheIndex * 4 + 0]; |
| fm.bottom = fmCache[fmCacheIndex * 4 + 1]; |
| fm.ascent = fmCache[fmCacheIndex * 4 + 2]; |
| fm.descent = fmCache[fmCacheIndex * 4 + 3]; |
| fmCacheIndex++; |
| |
| if (fm.top < fmTop) { |
| fmTop = fm.top; |
| } |
| if (fm.ascent < fmAscent) { |
| fmAscent = fm.ascent; |
| } |
| if (fm.descent > fmDescent) { |
| fmDescent = fm.descent; |
| } |
| if (fm.bottom > fmBottom) { |
| fmBottom = fm.bottom; |
| } |
| |
| // skip breaks ending before current span range |
| while (breakIndex < breakCount && paraStart + breaks[breakIndex] < spanStart) { |
| breakIndex++; |
| } |
| |
| while (breakIndex < breakCount && paraStart + breaks[breakIndex] <= spanEnd) { |
| int endPos = paraStart + breaks[breakIndex]; |
| |
| boolean moreChars = (endPos < bufEnd); |
| |
| final int ascent = isFallbackLineSpacing |
| ? Math.min(fmAscent, Math.round(ascents[breakIndex])) |
| : fmAscent; |
| final int descent = isFallbackLineSpacing |
| ? Math.max(fmDescent, Math.round(descents[breakIndex])) |
| : fmDescent; |
| |
| // The fallback ascent/descent may be larger than top/bottom of the default font |
| // metrics. Adjust top/bottom with ascent/descent for avoiding unexpected |
| // clipping. |
| if (isFallbackLineSpacing) { |
| if (ascent < fmTop) { |
| fmTop = ascent; |
| } |
| if (descent > fmBottom) { |
| fmBottom = descent; |
| } |
| } |
| |
| v = out(source, here, endPos, |
| ascent, descent, fmTop, fmBottom, |
| v, spacingmult, spacingadd, chooseHt, chooseHtv, fm, |
| hasTabs[breakIndex], hyphenEdits[breakIndex], needMultiply, |
| measuredPara, bufEnd, includepad, trackpad, addLastLineSpacing, chs, |
| paraStart, ellipsize, ellipsizedWidth, lineWidths[breakIndex], |
| paint, moreChars); |
| |
| if (endPos < spanEnd) { |
| // preserve metrics for current span |
| fmTop = Math.min(defaultTop, fm.top); |
| fmBottom = Math.max(defaultBottom, fm.bottom); |
| fmAscent = Math.min(defaultAscent, fm.ascent); |
| fmDescent = Math.max(defaultDescent, fm.descent); |
| } else { |
| fmTop = fmBottom = fmAscent = fmDescent = 0; |
| } |
| |
| here = endPos; |
| breakIndex++; |
| |
| if (mLineCount >= mMaximumVisibleLineCount && mEllipsized) { |
| return; |
| } |
| } |
| } |
| |
| if (paraEnd == bufEnd) { |
| break; |
| } |
| } |
| |
| if ((bufEnd == bufStart || source.charAt(bufEnd - 1) == CHAR_NEW_LINE) |
| && mLineCount < mMaximumVisibleLineCount) { |
| final MeasuredParagraph measuredPara = |
| MeasuredParagraph.buildForBidi(source, bufEnd, bufEnd, textDir, null); |
| if (defaultAscent != 0 && defaultDescent != 0) { |
| fm.top = defaultTop; |
| fm.ascent = defaultAscent; |
| fm.descent = defaultDescent; |
| fm.bottom = defaultBottom; |
| } else { |
| paint.getFontMetricsInt(fm); |
| } |
| |
| v = out(source, |
| bufEnd, bufEnd, fm.ascent, fm.descent, |
| fm.top, fm.bottom, |
| v, |
| spacingmult, spacingadd, null, |
| null, fm, false, 0, |
| needMultiply, measuredPara, bufEnd, |
| includepad, trackpad, addLastLineSpacing, null, |
| bufStart, ellipsize, |
| ellipsizedWidth, 0, paint, false); |
| } |
| } |
| |
| private int out(final CharSequence text, final int start, final int end, int above, int below, |
| int top, int bottom, int v, final float spacingmult, final float spacingadd, |
| final LineHeightSpan[] chooseHt, final int[] chooseHtv, final Paint.FontMetricsInt fm, |
| final boolean hasTab, final int hyphenEdit, final boolean needMultiply, |
| @NonNull final MeasuredParagraph measured, |
| final int bufEnd, final boolean includePad, final boolean trackPad, |
| final boolean addLastLineLineSpacing, final char[] chs, |
| final int widthStart, final TextUtils.TruncateAt ellipsize, final float ellipsisWidth, |
| final float textWidth, final TextPaint paint, final boolean moreChars) { |
| final int j = mLineCount; |
| final int off = j * mColumns; |
| final int want = off + mColumns + TOP; |
| int[] lines = mLines; |
| final int dir = measured.getParagraphDir(); |
| |
| if (want >= lines.length) { |
| final int[] grow = ArrayUtils.newUnpaddedIntArray(GrowingArrayUtils.growSize(want)); |
| System.arraycopy(lines, 0, grow, 0, lines.length); |
| mLines = grow; |
| lines = grow; |
| } |
| |
| if (j >= mLineDirections.length) { |
| final Directions[] grow = ArrayUtils.newUnpaddedArray(Directions.class, |
| GrowingArrayUtils.growSize(j)); |
| System.arraycopy(mLineDirections, 0, grow, 0, mLineDirections.length); |
| mLineDirections = grow; |
| } |
| |
| if (chooseHt != null) { |
| fm.ascent = above; |
| fm.descent = below; |
| fm.top = top; |
| fm.bottom = bottom; |
| |
| for (int i = 0; i < chooseHt.length; i++) { |
| if (chooseHt[i] instanceof LineHeightSpan.WithDensity) { |
| ((LineHeightSpan.WithDensity) chooseHt[i]) |
| .chooseHeight(text, start, end, chooseHtv[i], v, fm, paint); |
| } else { |
| chooseHt[i].chooseHeight(text, start, end, chooseHtv[i], v, fm); |
| } |
| } |
| |
| above = fm.ascent; |
| below = fm.descent; |
| top = fm.top; |
| bottom = fm.bottom; |
| } |
| |
| boolean firstLine = (j == 0); |
| boolean currentLineIsTheLastVisibleOne = (j + 1 == mMaximumVisibleLineCount); |
| |
| if (ellipsize != null) { |
| // If there is only one line, then do any type of ellipsis except when it is MARQUEE |
| // if there are multiple lines, just allow END ellipsis on the last line |
| boolean forceEllipsis = moreChars && (mLineCount + 1 == mMaximumVisibleLineCount); |
| |
| boolean doEllipsis = |
| (((mMaximumVisibleLineCount == 1 && moreChars) || (firstLine && !moreChars)) && |
| ellipsize != TextUtils.TruncateAt.MARQUEE) || |
| (!firstLine && (currentLineIsTheLastVisibleOne || !moreChars) && |
| ellipsize == TextUtils.TruncateAt.END); |
| if (doEllipsis) { |
| calculateEllipsis(start, end, measured, widthStart, |
| ellipsisWidth, ellipsize, j, |
| textWidth, paint, forceEllipsis); |
| } else { |
| mLines[mColumns * j + ELLIPSIS_START] = 0; |
| mLines[mColumns * j + ELLIPSIS_COUNT] = 0; |
| } |
| } |
| |
| final boolean lastLine; |
| if (mEllipsized) { |
| lastLine = true; |
| } else { |
| final boolean lastCharIsNewLine = widthStart != bufEnd && bufEnd > 0 |
| && text.charAt(bufEnd - 1) == CHAR_NEW_LINE; |
| if (end == bufEnd && !lastCharIsNewLine) { |
| lastLine = true; |
| } else if (start == bufEnd && lastCharIsNewLine) { |
| lastLine = true; |
| } else { |
| lastLine = false; |
| } |
| } |
| |
| if (firstLine) { |
| if (trackPad) { |
| mTopPadding = top - above; |
| } |
| |
| if (includePad) { |
| above = top; |
| } |
| } |
| |
| int extra; |
| |
| if (lastLine) { |
| if (trackPad) { |
| mBottomPadding = bottom - below; |
| } |
| |
| if (includePad) { |
| below = bottom; |
| } |
| } |
| |
| if (needMultiply && (addLastLineLineSpacing || !lastLine)) { |
| double ex = (below - above) * (spacingmult - 1) + spacingadd; |
| if (ex >= 0) { |
| extra = (int)(ex + EXTRA_ROUNDING); |
| } else { |
| extra = -(int)(-ex + EXTRA_ROUNDING); |
| } |
| } else { |
| extra = 0; |
| } |
| |
| lines[off + START] = start; |
| lines[off + TOP] = v; |
| lines[off + DESCENT] = below + extra; |
| lines[off + EXTRA] = extra; |
| |
| // special case for non-ellipsized last visible line when maxLines is set |
| // store the height as if it was ellipsized |
| if (!mEllipsized && currentLineIsTheLastVisibleOne) { |
| // below calculation as if it was the last line |
| int maxLineBelow = includePad ? bottom : below; |
| // similar to the calculation of v below, without the extra. |
| mMaxLineHeight = v + (maxLineBelow - above); |
| } |
| |
| v += (below - above) + extra; |
| lines[off + mColumns + START] = end; |
| lines[off + mColumns + TOP] = v; |
| |
| // TODO: could move TAB to share same column as HYPHEN, simplifying this code and gaining |
| // one bit for start field |
| lines[off + TAB] |= hasTab ? TAB_MASK : 0; |
| if (mEllipsized) { |
| if (ellipsize == TextUtils.TruncateAt.START) { |
| lines[off + HYPHEN] = packHyphenEdit(Paint.START_HYPHEN_EDIT_NO_EDIT, |
| unpackEndHyphenEdit(hyphenEdit)); |
| } else if (ellipsize == TextUtils.TruncateAt.END) { |
| lines[off + HYPHEN] = packHyphenEdit(unpackStartHyphenEdit(hyphenEdit), |
| Paint.END_HYPHEN_EDIT_NO_EDIT); |
| } else { // Middle and marquee ellipsize should show text at the start/end edge. |
| lines[off + HYPHEN] = packHyphenEdit( |
| Paint.START_HYPHEN_EDIT_NO_EDIT, Paint.END_HYPHEN_EDIT_NO_EDIT); |
| } |
| } else { |
| lines[off + HYPHEN] = hyphenEdit; |
| } |
| |
| lines[off + DIR] |= dir << DIR_SHIFT; |
| mLineDirections[j] = measured.getDirections(start - widthStart, end - widthStart); |
| |
| mLineCount++; |
| return v; |
| } |
| |
| private void calculateEllipsis(int lineStart, int lineEnd, |
| MeasuredParagraph measured, int widthStart, |
| float avail, TextUtils.TruncateAt where, |
| int line, float textWidth, TextPaint paint, |
| boolean forceEllipsis) { |
| avail -= getTotalInsets(line); |
| if (textWidth <= avail && !forceEllipsis) { |
| // Everything fits! |
| mLines[mColumns * line + ELLIPSIS_START] = 0; |
| mLines[mColumns * line + ELLIPSIS_COUNT] = 0; |
| return; |
| } |
| |
| float ellipsisWidth = paint.measureText(TextUtils.getEllipsisString(where)); |
| int ellipsisStart = 0; |
| int ellipsisCount = 0; |
| int len = lineEnd - lineStart; |
| |
| // We only support start ellipsis on a single line |
| if (where == TextUtils.TruncateAt.START) { |
| if (mMaximumVisibleLineCount == 1) { |
| float sum = 0; |
| int i; |
| |
| for (i = len; i > 0; i--) { |
| float w = measured.getCharWidthAt(i - 1 + lineStart - widthStart); |
| if (w + sum + ellipsisWidth > avail) { |
| while (i < len |
| && measured.getCharWidthAt(i + lineStart - widthStart) == 0.0f) { |
| i++; |
| } |
| break; |
| } |
| |
| sum += w; |
| } |
| |
| ellipsisStart = 0; |
| ellipsisCount = i; |
| } else { |
| if (Log.isLoggable(TAG, Log.WARN)) { |
| Log.w(TAG, "Start Ellipsis only supported with one line"); |
| } |
| } |
| } else if (where == TextUtils.TruncateAt.END || where == TextUtils.TruncateAt.MARQUEE || |
| where == TextUtils.TruncateAt.END_SMALL) { |
| float sum = 0; |
| int i; |
| |
| for (i = 0; i < len; i++) { |
| float w = measured.getCharWidthAt(i + lineStart - widthStart); |
| |
| if (w + sum + ellipsisWidth > avail) { |
| break; |
| } |
| |
| sum += w; |
| } |
| |
| ellipsisStart = i; |
| ellipsisCount = len - i; |
| if (forceEllipsis && ellipsisCount == 0 && len > 0) { |
| ellipsisStart = len - 1; |
| ellipsisCount = 1; |
| } |
| } else { |
| // where = TextUtils.TruncateAt.MIDDLE We only support middle ellipsis on a single line |
| if (mMaximumVisibleLineCount == 1) { |
| float lsum = 0, rsum = 0; |
| int left = 0, right = len; |
| |
| float ravail = (avail - ellipsisWidth) / 2; |
| for (right = len; right > 0; right--) { |
| float w = measured.getCharWidthAt(right - 1 + lineStart - widthStart); |
| |
| if (w + rsum > ravail) { |
| while (right < len |
| && measured.getCharWidthAt(right + lineStart - widthStart) |
| == 0.0f) { |
| right++; |
| } |
| break; |
| } |
| rsum += w; |
| } |
| |
| float lavail = avail - ellipsisWidth - rsum; |
| for (left = 0; left < right; left++) { |
| float w = measured.getCharWidthAt(left + lineStart - widthStart); |
| |
| if (w + lsum > lavail) { |
| break; |
| } |
| |
| lsum += w; |
| } |
| |
| ellipsisStart = left; |
| ellipsisCount = right - left; |
| } else { |
| if (Log.isLoggable(TAG, Log.WARN)) { |
| Log.w(TAG, "Middle Ellipsis only supported with one line"); |
| } |
| } |
| } |
| mEllipsized = true; |
| mLines[mColumns * line + ELLIPSIS_START] = ellipsisStart; |
| mLines[mColumns * line + ELLIPSIS_COUNT] = ellipsisCount; |
| } |
| |
| private float getTotalInsets(int line) { |
| int totalIndent = 0; |
| if (mLeftIndents != null) { |
| totalIndent = mLeftIndents[Math.min(line, mLeftIndents.length - 1)]; |
| } |
| if (mRightIndents != null) { |
| totalIndent += mRightIndents[Math.min(line, mRightIndents.length - 1)]; |
| } |
| return totalIndent; |
| } |
| |
| // Override the base class so we can directly access our members, |
| // rather than relying on member functions. |
| // The logic mirrors that of Layout.getLineForVertical |
| // FIXME: It may be faster to do a linear search for layouts without many lines. |
| @Override |
| public int getLineForVertical(int vertical) { |
| int high = mLineCount; |
| int low = -1; |
| int guess; |
| int[] lines = mLines; |
| while (high - low > 1) { |
| guess = (high + low) >> 1; |
| if (lines[mColumns * guess + TOP] > vertical){ |
| high = guess; |
| } else { |
| low = guess; |
| } |
| } |
| if (low < 0) { |
| return 0; |
| } else { |
| return low; |
| } |
| } |
| |
| @Override |
| public int getLineCount() { |
| return mLineCount; |
| } |
| |
| @Override |
| public int getLineTop(int line) { |
| return mLines[mColumns * line + TOP]; |
| } |
| |
| /** |
| * @hide |
| */ |
| @Override |
| public int getLineExtra(int line) { |
| return mLines[mColumns * line + EXTRA]; |
| } |
| |
| @Override |
| public int getLineDescent(int line) { |
| return mLines[mColumns * line + DESCENT]; |
| } |
| |
| @Override |
| public int getLineStart(int line) { |
| return mLines[mColumns * line + START] & START_MASK; |
| } |
| |
| @Override |
| public int getParagraphDirection(int line) { |
| return mLines[mColumns * line + DIR] >> DIR_SHIFT; |
| } |
| |
| @Override |
| public boolean getLineContainsTab(int line) { |
| return (mLines[mColumns * line + TAB] & TAB_MASK) != 0; |
| } |
| |
| @Override |
| public final Directions getLineDirections(int line) { |
| if (line > getLineCount()) { |
| throw new ArrayIndexOutOfBoundsException(); |
| } |
| return mLineDirections[line]; |
| } |
| |
| @Override |
| public int getTopPadding() { |
| return mTopPadding; |
| } |
| |
| @Override |
| public int getBottomPadding() { |
| return mBottomPadding; |
| } |
| |
| // To store into single int field, pack the pair of start and end hyphen edit. |
| static int packHyphenEdit( |
| @Paint.StartHyphenEdit int start, @Paint.EndHyphenEdit int end) { |
| return start << START_HYPHEN_BITS_SHIFT | end; |
| } |
| |
| static int unpackStartHyphenEdit(int packedHyphenEdit) { |
| return (packedHyphenEdit & START_HYPHEN_MASK) >> START_HYPHEN_BITS_SHIFT; |
| } |
| |
| static int unpackEndHyphenEdit(int packedHyphenEdit) { |
| return packedHyphenEdit & END_HYPHEN_MASK; |
| } |
| |
| /** |
| * Returns the start hyphen edit value for this line. |
| * |
| * @param lineNumber a line number |
| * @return A start hyphen edit value. |
| * @hide |
| */ |
| @Override |
| public @Paint.StartHyphenEdit int getStartHyphenEdit(int lineNumber) { |
| return unpackStartHyphenEdit(mLines[mColumns * lineNumber + HYPHEN] & HYPHEN_MASK); |
| } |
| |
| /** |
| * Returns the packed hyphen edit value for this line. |
| * |
| * @param lineNumber a line number |
| * @return An end hyphen edit value. |
| * @hide |
| */ |
| @Override |
| public @Paint.EndHyphenEdit int getEndHyphenEdit(int lineNumber) { |
| return unpackEndHyphenEdit(mLines[mColumns * lineNumber + HYPHEN] & HYPHEN_MASK); |
| } |
| |
| /** |
| * @hide |
| */ |
| @Override |
| public int getIndentAdjust(int line, Alignment align) { |
| if (align == Alignment.ALIGN_LEFT) { |
| if (mLeftIndents == null) { |
| return 0; |
| } else { |
| return mLeftIndents[Math.min(line, mLeftIndents.length - 1)]; |
| } |
| } else if (align == Alignment.ALIGN_RIGHT) { |
| if (mRightIndents == null) { |
| return 0; |
| } else { |
| return -mRightIndents[Math.min(line, mRightIndents.length - 1)]; |
| } |
| } else if (align == Alignment.ALIGN_CENTER) { |
| int left = 0; |
| if (mLeftIndents != null) { |
| left = mLeftIndents[Math.min(line, mLeftIndents.length - 1)]; |
| } |
| int right = 0; |
| if (mRightIndents != null) { |
| right = mRightIndents[Math.min(line, mRightIndents.length - 1)]; |
| } |
| return (left - right) >> 1; |
| } else { |
| throw new AssertionError("unhandled alignment " + align); |
| } |
| } |
| |
| @Override |
| public int getEllipsisCount(int line) { |
| if (mColumns < COLUMNS_ELLIPSIZE) { |
| return 0; |
| } |
| |
| return mLines[mColumns * line + ELLIPSIS_COUNT]; |
| } |
| |
| @Override |
| public int getEllipsisStart(int line) { |
| if (mColumns < COLUMNS_ELLIPSIZE) { |
| return 0; |
| } |
| |
| return mLines[mColumns * line + ELLIPSIS_START]; |
| } |
| |
| @Override |
| @NonNull |
| public RectF computeDrawingBoundingBox() { |
| // Cache the drawing bounds result because it does not change after created. |
| if (mDrawingBounds == null) { |
| mDrawingBounds = super.computeDrawingBoundingBox(); |
| } |
| return mDrawingBounds; |
| } |
| |
| /** |
| * Return the total height of this layout. |
| * |
| * @param cap if true and max lines is set, returns the height of the layout at the max lines. |
| * |
| * @hide |
| */ |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) |
| public int getHeight(boolean cap) { |
| if (cap && mLineCount > mMaximumVisibleLineCount && mMaxLineHeight == -1 |
| && Log.isLoggable(TAG, Log.WARN)) { |
| Log.w(TAG, "maxLineHeight should not be -1. " |
| + " maxLines:" + mMaximumVisibleLineCount |
| + " lineCount:" + mLineCount); |
| } |
| |
| return cap && mLineCount > mMaximumVisibleLineCount && mMaxLineHeight != -1 |
| ? mMaxLineHeight : super.getHeight(); |
| } |
| |
| @UnsupportedAppUsage |
| private int mLineCount; |
| private int mTopPadding, mBottomPadding; |
| @UnsupportedAppUsage |
| private int mColumns; |
| private RectF mDrawingBounds = null; // lazy calculation. |
| |
| /** |
| * Keeps track if ellipsize is applied to the text. |
| */ |
| private boolean mEllipsized; |
| |
| /** |
| * If maxLines is set, ellipsize is not set, and the actual line count of text is greater than |
| * or equal to maxLine, this variable holds the ideal visual height of the maxLine'th line |
| * starting from the top of the layout. If maxLines is not set its value will be -1. |
| * |
| * The value is the same as getLineTop(maxLines) for ellipsized version where structurally no |
| * more than maxLines is contained. |
| */ |
| private int mMaxLineHeight = DEFAULT_MAX_LINE_HEIGHT; |
| |
| private static final int COLUMNS_NORMAL = 5; |
| private static final int COLUMNS_ELLIPSIZE = 7; |
| private static final int START = 0; |
| private static final int DIR = START; |
| private static final int TAB = START; |
| private static final int TOP = 1; |
| private static final int DESCENT = 2; |
| private static final int EXTRA = 3; |
| private static final int HYPHEN = 4; |
| @UnsupportedAppUsage |
| private static final int ELLIPSIS_START = 5; |
| private static final int ELLIPSIS_COUNT = 6; |
| |
| @UnsupportedAppUsage |
| private int[] mLines; |
| @UnsupportedAppUsage |
| private Directions[] mLineDirections; |
| @UnsupportedAppUsage |
| private int mMaximumVisibleLineCount = Integer.MAX_VALUE; |
| |
| private static final int START_MASK = 0x1FFFFFFF; |
| private static final int DIR_SHIFT = 30; |
| private static final int TAB_MASK = 0x20000000; |
| private static final int HYPHEN_MASK = 0xFF; |
| private static final int START_HYPHEN_BITS_SHIFT = 3; |
| private static final int START_HYPHEN_MASK = 0x18; // 0b11000 |
| private static final int END_HYPHEN_MASK = 0x7; // 0b00111 |
| |
| private static final float TAB_INCREMENT = 20; // same as Layout, but that's private |
| |
| private static final char CHAR_NEW_LINE = '\n'; |
| |
| private static final double EXTRA_ROUNDING = 0.5; |
| |
| private static final int DEFAULT_MAX_LINE_HEIGHT = -1; |
| |
| // Unused, here because of gray list private API accesses. |
| /*package*/ static class LineBreaks { |
| private static final int INITIAL_SIZE = 16; |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) |
| public int[] breaks = new int[INITIAL_SIZE]; |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) |
| public float[] widths = new float[INITIAL_SIZE]; |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) |
| public float[] ascents = new float[INITIAL_SIZE]; |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) |
| public float[] descents = new float[INITIAL_SIZE]; |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) |
| public int[] flags = new int[INITIAL_SIZE]; // hasTab |
| // breaks, widths, and flags should all have the same length |
| } |
| |
| @Nullable private int[] mLeftIndents; |
| @Nullable private int[] mRightIndents; |
| } |