සීමාවන් තුළ ගැලපෙන පරිදි ස්වයංක්‍රීය පරිමාණ පෙළ දර්ශන පෙළ


871

පෙළ TextViewඑතීමේ ප්‍රමාණය නැවත ලබා ගැනීම සඳහා ප්‍රශස්ත ක්‍රමයක් මම සොයමි, එමඟින් එහි getHeight සහ getWidth සීමාවන්ට ගැලපේ. මම හුදෙක් පෙළ එතීමට ක්‍රමයක් සොයන්නේ නැත- මට එය දෙකම එතීමට වග බලා ගත යුතු අතර එය සම්පූර්ණයෙන්ම තිරය මතට ගැළපෙන තරම් කුඩාය.

ස්වයංක්‍රීය ප්‍රතිප්‍රමාණනය අවශ්‍ය වූ අවස්ථා කිහිපයක් මම ස්ටැක් ඕවර්ෆ්ලෝ හි දැක ඇත්තෙමි, නමුත් ඒවා එක්කෝ හැක් විසඳුම් සහිත විශේෂ අවස්ථා, විසඳුමක් නොමැති වීම හෝ TextViewප්‍රමාණවත් තරම් කුඩා වන තෙක් පුනරාවර්තන ලෙස නැවත ඇඳීම සම්බන්ධ වේ (එය මතකය තීව්‍ර වන අතර එය බල කරයි පෙළ බැලීමට පරිශීලකයා සෑම පුනරාවර්තනයක් සමඟම පියවරෙන් පියවර හැකිලී යයි).

නමුත් මට විශ්වාසයි එතන සිටින කවුරුහරි මම කරන දේට සම්බන්ධ නොවන හොඳ විසඳුමක් සොයාගෙන ඇත: පෙළ විග්‍රහ කොට මැනීමට, පෙළ ප්‍රමාණයට වෙනස් කිරීමට සහ සුදුසු කුඩා ප්‍රමාණයක් සොයා ගන්නා තෙක් නැවත නැවත කිරීමට බර වැඩ කිහිපයක් ලිවීම.

කුමක්ද චර්යාවන් TextViewපෙළ ඔතාගෙන භාවිතය? පෙළ ප්‍රමාණවත් තරම් කුඩා වේදැයි පුරෝකථනය කිරීමට ඒවා කෙසේ හෝ භාවිතා කළ නොහැකිද?

tl; dr : TextViewගැළපෙන, ඔතා, එහි getHeight සහ getWidth සීමාවන් තුළ ස්වයංක්‍රීයව ප්‍රතිනිර්මාණය කිරීමට හොඳම ක්‍රමයක් තිබේද?


පෙළ සීමාවෙන් ඔබ්බට යන්නේ කවදාදැයි සොයා ගැනීමට මම ස්ථිතික පිරිසැලසුමෙහි getEllipsisCount භාවිතා කිරීමට උත්සාහ කළෙමි, නමුත් එය මට වැඩක් නොවීය
Nathan Fig

ඔබ පැච් නවයක පෙළ දර්ශනයක් අඳින්නේ නැත්තේ ඇයි? එය ස්වයංක්‍රීයව එහි සීමාවන්ට ගැලපෙනු ඇත. මම ඔයාව නිවැරදි සහෝදරයා ගන්නවාද?
AZ_

පැච් පෙළ නවයක්? මට පැච් නවයක් ගැන හුරුපුරුදු නැත, එය රූප ආකෘතියක් ලෙස පෙනේ ... එය පෙළ දර්ශනයක් සමඟ භාවිතා කළ හැකි ආකාරය පිළිබඳ ලියකියවිලි සොයමි. මට තවත් තොරතුරු ලබා ගත හැකි ස්ථානය දැන ගන්න?
නේතන් රූපය

තව ටිකක් ඒ ගැන කියවන විට මට හැඟී යන්නේ පැච් නවය යනු දැනටමත් නිර්වචනය කර ඇති ආකෘතියක් ප්‍රතිනිර්මාණය කිරීම සඳහා පමණක් බවය - නමුත් මට යමක් අවශ්‍ය වන අතර එය නූලක් ගෙන එහි ප්‍රශස්ත ප්‍රමාණය හා ආකෘතිය යම් සීමාවන් තුළ සොයා ගනී.
නේතන් රූපය

විය හැකි අනුපිටපත: stackoverflow.com/a/7875656/435605
AlikElzin-kilaka

Answers:


189

2018 ජුනි මාසයේ සිට ඇන්ඩ්‍රොයිඩ් නිල වශයෙන් ඇන්ඩ්‍රොයිඩ් 4.0 (ඒපීඅයි මට්ටම 14) සහ ඊට වැඩි සඳහා මෙම විශේෂාංගයට සහය දැක්වීමට පටන් ගත්තේය .
එය පරීක්ෂා කරන්න: පෙළ දර්ශන ස්වයංක්‍රීය කිරීම

ඇන්ඩ්‍රොයිඩ් 8.0 (API මට්ටම 26) සහ ඊට වැඩි :

<?xml version="1.0" encoding="utf-8"?>
<TextView
    android:layout_width="match_parent"
    android:layout_height="200dp"
    android:autoSizeTextType="uniform"
    android:autoSizeMinTextSize="12sp"
    android:autoSizeMaxTextSize="100sp"
    android:autoSizeStepGranularity="2sp" />

ක්‍රමලේඛනය:

setAutoSizeTextTypeUniformWithConfiguration(int autoSizeMinTextSize, int autoSizeMaxTextSize, 
        int autoSizeStepGranularity, int unit)

textView.setAutoSizeTextTypeUniformWithConfiguration(
                1, 17, 1, TypedValue.COMPLEX_UNIT_DIP);


ඇන්ඩ්‍රොයිඩ් 8.0 ට පෙර ඇන්ඩ්‍රොයිඩ් අනුවාදයන් (API මට්ටම 26) :

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

  <TextView
      android:layout_width="match_parent"
      android:layout_height="200dp"
      app:autoSizeTextType="uniform"
      app:autoSizeMinTextSize="12sp"
      app:autoSizeMaxTextSize="100sp"
      app:autoSizeStepGranularity="2sp" />

</LinearLayout>

ක්‍රමලේඛනය:

TextViewCompat.setAutoSizeTextTypeUniformWithConfiguration(
TextView textView, int autoSizeMinTextSize, int autoSizeMaxTextSize, int autoSizeStepGranularity, int unit) 

TextViewCompat.setAutoSizeTextTypeUniformWithConfiguration(textView, 1, 17, 1,
TypedValue.COMPLEX_UNIT_DIP);

අවධානය: TextView හි පිරිසැලසුම_විඩ්ත් = " ගැලපීම් මාපිය " හෝ නිරපේක්ෂ ප්‍රමාණය තිබිය යුතුය !


4
අනාගත පාඨකයින්, එකතු කිරීමට මතක තබා ගන්න implementation 'com.android.support:support-compat:28.0.0'තුල app/build.gradleසඳහා app: ...වැඩ කිරීමට ගුණාංග.
ඩකා

ඔබ ලියන්නේ: "2018 ජුනි මාසයේ සිට ඇන්ඩ්‍රොයිඩ් නිල වශයෙන් ඇන්ඩ්‍රොයිඩ් 4.0 (ඒපීඅයි මට්ටම 14) සහ ඊට වැඩි සඳහා මෙම අංගයට සහය දැක්වීම ආරම්භ කළේය" ??
nnyerges

4
nnyerges ඔබට පැරණි API අනුවාදය (<26) සඳහා ආධාරක යෙදුම් සංසන්දනය භාවිතා කළ හැකිය, භාවිතා කරන්න: යෙදුම: autoSizeTextType = "ඒකාකාර"
දෙවරක් කේතය සිතන්න

මා පැකිලී ගිය දෙය නම්, ඔබ සැමවිටම ඇන්ඩ්‍රොයිඩ් සැකසුම් තුළ සැකසිය හැකි විවිධ පෙළ ප්‍රමාණවලින් මෙය පරීක්ෂා කළ යුතුය. එසේ නොමැතිනම් autoSizeMaxTextSizeඑය ඉතා අඩු විය හැකි අතර පෙළ කපා දමනු ලැබේ.
L3n95

TextView තිබිය යුතුය android:lines="1"සහ layout_width="0dp"ConstraintLayout දී
Black_Zerg

1109

ජංගම සංවර්ධකයෙකු ලෙස, ස්වයංක්‍රීය ප්‍රතිප්‍රමාණනය සඳහා සහාය වන ස්වදේශීය කිසිවක් සොයා නොගැනීම ගැන මම කනගාටු වෙමි. මගේ සෙවීම් මා වෙනුවෙන් වැඩ කළ කිසිවක් සොයා නොගත් අතර අවසානයේ මම මගේ සති අන්තයේ වැඩි කාලයක් ගත කළ අතර මගේම ස්වයංක්‍රීය ප්‍රමාණයේ පෙළ දර්ශනයක් නිර්මාණය කළෙමි. මම මෙහි කේතය පළ කරමි, එය වෙනත් කෙනෙකුට ප්‍රයෝජනවත් වනු ඇතැයි බලාපොරොත්තු වෙමි.

මෙම පන්තිය උස මැනීම සඳහා මුල් පෙළ දර්ශනයේ පෙළ තීන්ත සමඟ ස්ථිතික පිරිසැලසුමක් භාවිතා කරයි. එතැන් සිට, මම අකුරු පික්සල් 2 කින් බැස, මට ගැළපෙන ප්‍රමාණයක් ලැබෙන තුරු නැවත මතක් කරමි. අවසානයේදී, පෙළ තවමත් නොගැලපේ නම්, මම ඉලිප්සයක් එකතු කරමි. පෙළ සජීවිකරණය කිරීමට සහ නැවත භාවිතා කිරීමට මට අවශ්‍යතා තිබූ අතර මෙය මා සතුව ඇති උපාංගවල හොඳින් ක්‍රියාත්මක වන අතර මට වේගයෙන් ධාවනය වන බව පෙනේ.

/**
 *               DO WHAT YOU WANT TO PUBLIC LICENSE
 *                    Version 2, December 2004
 * 
 * Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
 * 
 * Everyone is permitted to copy and distribute verbatim or modified
 * copies of this license document, and changing it is allowed as long
 * as the name is changed.
 * 
 *            DO WHAT YOU WANT TO PUBLIC LICENSE
 *   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
 * 
 *  0. You just DO WHAT YOU WANT TO.
 */

import android.content.Context;
import android.text.Layout.Alignment;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.widget.TextView;

/**
 * Text view that auto adjusts text size to fit within the view.
 * If the text size equals the minimum text size and still does not
 * fit, append with an ellipsis.
 * 
 * @author Chase Colburn
 * @since Apr 4, 2011
 */
public class AutoResizeTextView extends TextView {

    // Minimum text size for this text view
    public static final float MIN_TEXT_SIZE = 20;

    // Interface for resize notifications
    public interface OnTextResizeListener {
        public void onTextResize(TextView textView, float oldSize, float newSize);
    }

    // Our ellipse string
    private static final String mEllipsis = "...";

    // Registered resize listener
    private OnTextResizeListener mTextResizeListener;

    // Flag for text and/or size changes to force a resize
    private boolean mNeedsResize = false;

    // Text size that is set from code. This acts as a starting point for resizing
    private float mTextSize;

    // Temporary upper bounds on the starting text size
    private float mMaxTextSize = 0;

    // Lower bounds for text size
    private float mMinTextSize = MIN_TEXT_SIZE;

    // Text view line spacing multiplier
    private float mSpacingMult = 1.0f;

    // Text view additional line spacing
    private float mSpacingAdd = 0.0f;

    // Add ellipsis to text that overflows at the smallest text size
    private boolean mAddEllipsis = true;

    // Default constructor override
    public AutoResizeTextView(Context context) {
        this(context, null);
    }

    // Default constructor when inflating from XML file
    public AutoResizeTextView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    // Default constructor override
    public AutoResizeTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mTextSize = getTextSize();
    }

    /**
     * When text changes, set the force resize flag to true and reset the text size.
     */
    @Override
    protected void onTextChanged(final CharSequence text, final int start, final int before, final int after) {
        mNeedsResize = true;
        // Since this view may be reused, it is good to reset the text size
        resetTextSize();
    }

    /**
     * If the text view size changed, set the force resize flag to true
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        if (w != oldw || h != oldh) {
            mNeedsResize = true;
        }
    }

    /**
     * Register listener to receive resize notifications
     * @param listener
     */
    public void setOnResizeListener(OnTextResizeListener listener) {
        mTextResizeListener = listener;
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(float size) {
        super.setTextSize(size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(int unit, float size) {
        super.setTextSize(unit, size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set line spacing to update our internal reference values
     */
    @Override
    public void setLineSpacing(float add, float mult) {
        super.setLineSpacing(add, mult);
        mSpacingMult = mult;
        mSpacingAdd = add;
    }

    /**
     * Set the upper text size limit and invalidate the view
     * @param maxTextSize
     */
    public void setMaxTextSize(float maxTextSize) {
        mMaxTextSize = maxTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return upper text size limit
     * @return
     */
    public float getMaxTextSize() {
        return mMaxTextSize;
    }

    /**
     * Set the lower text size limit and invalidate the view
     * @param minTextSize
     */
    public void setMinTextSize(float minTextSize) {
        mMinTextSize = minTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return lower text size limit
     * @return
     */
    public float getMinTextSize() {
        return mMinTextSize;
    }

    /**
     * Set flag to add ellipsis to text that overflows at the smallest text size
     * @param addEllipsis
     */
    public void setAddEllipsis(boolean addEllipsis) {
        mAddEllipsis = addEllipsis;
    }

    /**
     * Return flag to add ellipsis to text that overflows at the smallest text size
     * @return
     */
    public boolean getAddEllipsis() {
        return mAddEllipsis;
    }

    /**
     * Reset the text to the original size
     */
    public void resetTextSize() {
        if (mTextSize > 0) {
            super.setTextSize(TypedValue.COMPLEX_UNIT_PX, mTextSize);
            mMaxTextSize = mTextSize;
        }
    }

    /**
     * Resize text after measuring
     */
    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        if (changed || mNeedsResize) {
            int widthLimit = (right - left) - getCompoundPaddingLeft() - getCompoundPaddingRight();
            int heightLimit = (bottom - top) - getCompoundPaddingBottom() - getCompoundPaddingTop();
            resizeText(widthLimit, heightLimit);
        }
        super.onLayout(changed, left, top, right, bottom);
    }

    /**
     * Resize the text size with default width and height
     */
    public void resizeText() {

        int heightLimit = getHeight() - getPaddingBottom() - getPaddingTop();
        int widthLimit = getWidth() - getPaddingLeft() - getPaddingRight();
        resizeText(widthLimit, heightLimit);
    }

    /**
     * Resize the text size with specified width and height
     * @param width
     * @param height
     */
    public void resizeText(int width, int height) {
        CharSequence text = getText();
        // Do not resize if the view does not have dimensions or there is no text
        if (text == null || text.length() == 0 || height <= 0 || width <= 0 || mTextSize == 0) {
            return;
        }

        if (getTransformationMethod() != null) {
            text = getTransformationMethod().getTransformation(text, this);
        }

        // Get the text view's paint object
        TextPaint textPaint = getPaint();

        // Store the current text size
        float oldTextSize = textPaint.getTextSize();
        // If there is a max text size set, use the lesser of that and the default text size
        float targetTextSize = mMaxTextSize > 0 ? Math.min(mTextSize, mMaxTextSize) : mTextSize;

        // Get the required text height
        int textHeight = getTextHeight(text, textPaint, width, targetTextSize);

        // Until we either fit within our text view or we had reached our min text size, incrementally try smaller sizes
        while (textHeight > height && targetTextSize > mMinTextSize) {
            targetTextSize = Math.max(targetTextSize - 2, mMinTextSize);
            textHeight = getTextHeight(text, textPaint, width, targetTextSize);
        }

        // If we had reached our minimum text size and still don't fit, append an ellipsis
        if (mAddEllipsis && targetTextSize == mMinTextSize && textHeight > height) {
            // Draw using a static layout
            // modified: use a copy of TextPaint for measuring
            TextPaint paint = new TextPaint(textPaint);
            // Draw using a static layout
            StaticLayout layout = new StaticLayout(text, paint, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, false);
            // Check that we have a least one line of rendered text
            if (layout.getLineCount() > 0) {
                // Since the line at the specific vertical position would be cut off,
                // we must trim up to the previous line
                int lastLine = layout.getLineForVertical(height) - 1;
                // If the text would not even fit on a single line, clear it
                if (lastLine < 0) {
                    setText("");
                }
                // Otherwise, trim to the previous line and add an ellipsis
                else {
                    int start = layout.getLineStart(lastLine);
                    int end = layout.getLineEnd(lastLine);
                    float lineWidth = layout.getLineWidth(lastLine);
                    float ellipseWidth = textPaint.measureText(mEllipsis);

                    // Trim characters off until we have enough room to draw the ellipsis
                    while (width < lineWidth + ellipseWidth) {
                        lineWidth = textPaint.measureText(text.subSequence(start, --end + 1).toString());
                    }
                    setText(text.subSequence(0, end) + mEllipsis);
                }
            }
        }

        // Some devices try to auto adjust line spacing, so force default line spacing
        // and invalidate the layout as a side effect
        setTextSize(TypedValue.COMPLEX_UNIT_PX, targetTextSize);
        setLineSpacing(mSpacingAdd, mSpacingMult);

        // Notify the listener if registered
        if (mTextResizeListener != null) {
            mTextResizeListener.onTextResize(this, oldTextSize, targetTextSize);
        }

        // Reset force resize flag
        mNeedsResize = false;
    }

    // Set the text size of the text paint object and use a static layout to render text off screen before measuring
    private int getTextHeight(CharSequence source, TextPaint paint, int width, float textSize) {
        // modified: make a copy of the original TextPaint object for measuring
        // (apparently the object gets modified while measuring, see also the
        // docs for TextView.getPaint() (which states to access it read-only)
        TextPaint paintCopy = new TextPaint(paint);
        // Update the text paint object
        paintCopy.setTextSize(textSize);
        // Measure using a static layout
        StaticLayout layout = new StaticLayout(source, paintCopy, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
        return layout.getHeight();
    }

}

අවවාදයයි. ඇන්ඩ්‍රොයිඩ් 3.1 - 4.04 ට බලපාන වැදගත් ස්ථාවර දෝෂයක් ඇති අතර එමඟින් සියලුම ස්වයංක්‍රීය ප්‍රතිනිර්මාණ ටෙක්ස්ට්වීව් විජට් ක්‍රියා නොකරයි. කරුණාකර කියවන්න: https://stackoverflow.com/a/21851157/2075875


8
මෙය ඇත්ත වශයෙන්ම මම මෙතෙක් දැක ඇති හොඳම විසඳුමයි. ඔතා ගැනීමේදී ඇන්ඩ්‍රොයිඩ් වචන කැඩීමට කැමති නිසා එය මගේ අවශ්‍යතාවන්ට සම්පූර්ණයෙන්ම ගැලපෙන්නේ නැත, එබැවින් මම මගේ අභිරුචි විසඳුම සමඟ බැඳී සිටිමි- නමුත් ඔබේ පන්තිය සාමාන්‍ය භාවිතය සඳහා වඩා හොඳය.
නේතන් රූපය

249
ඇන්ඩ්‍රොයිඩ් හි ආත්මය තුළ, ඒ සියල්ල ඔබගේ ය! ඔබ ප්‍රයෝජනවත් යමක් ගොඩනඟන විට, ඔබට එය බෙදා ගැනීමට හැකි වේ යැයි බලාපොරොත්තු වෙමු :)
Chase

7
හරි, ඒ නිසා මම කේතයට යාවත්කාලීන කිරීමක් කළා. රේඛා දර්ශකයෙන් 1 ක් අඩු කිරීම හිතාමතා කළ අතර මම අදහසක් එක් කළෙමි. ඔබට ඇති දෝෂය බොහෝ විට එක් පෙළක් පවා ඉදිරිපත් කිරීමට ප්‍රමාණවත් උස නොමැති වීම විය හැකිය. මම චෙක්පතක් එකතු කර එම අවස්ථාවේ පෙළ ඉවත් කළෙමි. (මම ෂිබුයා හි වැඩ කරමි, එබැවින් මම ඕනෑම වේලාවක බියර් පානය කිරීමට සූදානම් වෙමි :)
චේස්

7
ඔබ නවතම සංස්කරණය කළ අනුවාදය උත්සාහ කළාද? මම සතියකට පමණ පෙර ජෙලි බීන් වෙනුවෙන් කුඩා වෙනසක් කළා. මෙය බොහෝ විට ගිතුබ් මත විය යුතුය.
චේස්

7
ListView අයිතම ඇතුළත මෙන්ම ඇතුළත පෙළ දර්ශනය සඳහා ක්‍රියා නොකරයි
Vladyslav Matviienko

147

යාවත්කාලීන කිරීම: මෙහි විස්තර කර ඇති පරිදි පරිපූර්ණ ස්වයංක්‍රීය පරිමාණ පෙළ දර්ශනයක අවශ්‍යතාවය පහත කේතය සපුරාලයි : ඇන්ඩ්‍රොයිඩ් සඳහා ස්වයංක්‍රීයව ගැලපෙන පෙළ දර්ශනය සහ ජයග්‍රාහකයා ලෙස සලකුණු කර ඇත.

යාවත්කාලීනය 2: එකතු කරන ලද මැක්ස්ලයින් සඳහා සහය, දැන් API මට්ටම 16 ට පෙර හොඳින් ක්‍රියාත්මක වේ.

යාවත්කාලීන 3: සඳහා සහාය android:drawableLeft, android:drawableRight, android:drawableTopහා android:drawableBottomටැග් එකතු MartinH සරල විසඳුමක් ස්තුති මෙතන .


මගේ අවශ්‍යතා ටිකක් වෙනස් විය. ප්‍රමාණය සම්පූර්ණ කිරීමට මට කාර්යක්ෂම ක්‍රමයක් අවශ්‍ය වූයේ මා පූර්ණ සංඛ්‍යාවක් සජීවිකරණය කරන නිසා TextViewතත්පර 2 කින් 0 සිට 4000 දක්වා විය හැකි අතර ඒ අනුව ප්‍රමාණය වෙනස් කිරීමට මට අවශ්‍ය විය. මගේ විසඳුම ටිකක් වෙනස් ලෙස ක්‍රියා කරයි. අවසාන ප්‍රති result ලය පෙනෙන්නේ මෙන්න:

රූප විස්තරය මෙහි ඇතුළත් කරන්න

සහ එය නිපදවූ කේතය:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:orientation="vertical"
    android:padding="16dp" >

    <com.vj.widgets.AutoResizeTextView
        android:layout_width="match_parent"
        android:layout_height="100dp"
        android:ellipsize="none"
        android:maxLines="2"
        android:text="Auto Resized Text, max 2 lines"
        android:textSize="100sp" /> <!-- maximum size -->

    <com.vj.widgets.AutoResizeTextView
        android:layout_width="match_parent"
        android:layout_height="100dp"
        android:ellipsize="none"
        android:gravity="center"
        android:maxLines="1"
        android:text="Auto Resized Text, max 1 line"
        android:textSize="100sp" /> <!-- maximum size -->

    <com.vj.widgets.AutoResizeTextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Auto Resized Text"
        android:textSize="500sp" /> <!-- maximum size -->

</LinearLayout>

අවසානයේ ජාවා කේතය:

import android.annotation.TargetApi;
import android.content.Context;
import android.content.res.Resources;
import android.graphics.RectF;
import android.os.Build;
import android.text.Layout.Alignment;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.util.SparseIntArray;
import android.util.TypedValue;
import android.widget.TextView;

public class AutoResizeTextView extends TextView {
private interface SizeTester {
    /**
     * 
     * @param suggestedSize
     *            Size of text to be tested
     * @param availableSpace
     *            available space in which text must fit
     * @return an integer < 0 if after applying {@code suggestedSize} to
     *         text, it takes less space than {@code availableSpace}, > 0
     *         otherwise
     */
    public int onTestSize(int suggestedSize, RectF availableSpace);
}

private RectF mTextRect = new RectF();

private RectF mAvailableSpaceRect;

private SparseIntArray mTextCachedSizes;

private TextPaint mPaint;

private float mMaxTextSize;

private float mSpacingMult = 1.0f;

private float mSpacingAdd = 0.0f;

private float mMinTextSize = 20;

private int mWidthLimit;

private static final int NO_LINE_LIMIT = -1;
private int mMaxLines;

private boolean mEnableSizeCache = true;
private boolean mInitiallized;

public AutoResizeTextView(Context context) {
    super(context);
    initialize();
}

public AutoResizeTextView(Context context, AttributeSet attrs) {
    super(context, attrs);
    initialize();
}

public AutoResizeTextView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    initialize();
}

private void initialize() {
    mPaint = new TextPaint(getPaint());
    mMaxTextSize = getTextSize();
    mAvailableSpaceRect = new RectF();
    mTextCachedSizes = new SparseIntArray();
    if (mMaxLines == 0) {
        // no value was assigned during construction
        mMaxLines = NO_LINE_LIMIT;
    }
    mInitiallized = true;
}

@Override
public void setText(final CharSequence text, BufferType type) {
    super.setText(text, type);
    adjustTextSize(text.toString());
}

@Override
public void setTextSize(float size) {
    mMaxTextSize = size;
    mTextCachedSizes.clear();
    adjustTextSize(getText().toString());
}

@Override
public void setMaxLines(int maxlines) {
    super.setMaxLines(maxlines);
    mMaxLines = maxlines;
    reAdjust();
}

public int getMaxLines() {
    return mMaxLines;
}

@Override
public void setSingleLine() {
    super.setSingleLine();
    mMaxLines = 1;
    reAdjust();
}

@Override
public void setSingleLine(boolean singleLine) {
    super.setSingleLine(singleLine);
    if (singleLine) {
        mMaxLines = 1;
    } else {
        mMaxLines = NO_LINE_LIMIT;
    }
    reAdjust();
}

@Override
public void setLines(int lines) {
    super.setLines(lines);
    mMaxLines = lines;
    reAdjust();
}

@Override
public void setTextSize(int unit, float size) {
    Context c = getContext();
    Resources r;

    if (c == null)
        r = Resources.getSystem();
    else
        r = c.getResources();
    mMaxTextSize = TypedValue.applyDimension(unit, size,
            r.getDisplayMetrics());
    mTextCachedSizes.clear();
    adjustTextSize(getText().toString());
}

@Override
public void setLineSpacing(float add, float mult) {
    super.setLineSpacing(add, mult);
    mSpacingMult = mult;
    mSpacingAdd = add;
}

/**
 * Set the lower text size limit and invalidate the view
 * 
 * @param minTextSize
 */
public void setMinTextSize(float minTextSize) {
    mMinTextSize = minTextSize;
    reAdjust();
}

private void reAdjust() {
    adjustTextSize(getText().toString());
}

private void adjustTextSize(String string) {
    if (!mInitiallized) {
        return;
    }
    int startSize = (int) mMinTextSize;
    int heightLimit = getMeasuredHeight() - getCompoundPaddingBottom()
        - getCompoundPaddingTop();
    mWidthLimit = getMeasuredWidth() - getCompoundPaddingLeft()
        - getCompoundPaddingRight();
    mAvailableSpaceRect.right = mWidthLimit;
    mAvailableSpaceRect.bottom = heightLimit;
    super.setTextSize(
            TypedValue.COMPLEX_UNIT_PX,
            efficientTextSizeSearch(startSize, (int) mMaxTextSize,
                    mSizeTester, mAvailableSpaceRect));
}

private final SizeTester mSizeTester = new SizeTester() {
    @TargetApi(Build.VERSION_CODES.JELLY_BEAN)
    @Override
    public int onTestSize(int suggestedSize, RectF availableSPace) {
        mPaint.setTextSize(suggestedSize);
        String text = getText().toString();
        boolean singleline = getMaxLines() == 1;
        if (singleline) {
            mTextRect.bottom = mPaint.getFontSpacing();
            mTextRect.right = mPaint.measureText(text);
        } else {
            StaticLayout layout = new StaticLayout(text, mPaint,
                    mWidthLimit, Alignment.ALIGN_NORMAL, mSpacingMult,
                    mSpacingAdd, true);
            // return early if we have more lines
            if (getMaxLines() != NO_LINE_LIMIT
                    && layout.getLineCount() > getMaxLines()) {
                return 1;
            }
            mTextRect.bottom = layout.getHeight();
            int maxWidth = -1;
            for (int i = 0; i < layout.getLineCount(); i++) {
                if (maxWidth < layout.getLineWidth(i)) {
                    maxWidth = (int) layout.getLineWidth(i);
                }
            }
            mTextRect.right = maxWidth;
        }

        mTextRect.offsetTo(0, 0);
        if (availableSPace.contains(mTextRect)) {
            // may be too small, don't worry we will find the best match
            return -1;
        } else {
            // too big
            return 1;
        }
    }
};

/**
 * Enables or disables size caching, enabling it will improve performance
 * where you are animating a value inside TextView. This stores the font
 * size against getText().length() Be careful though while enabling it as 0
 * takes more space than 1 on some fonts and so on.
 * 
 * @param enable
 *            enable font size caching
 */
public void enableSizeCache(boolean enable) {
    mEnableSizeCache = enable;
    mTextCachedSizes.clear();
    adjustTextSize(getText().toString());
}

private int efficientTextSizeSearch(int start, int end,
        SizeTester sizeTester, RectF availableSpace) {
    if (!mEnableSizeCache) {
        return binarySearch(start, end, sizeTester, availableSpace);
    }
    String text = getText().toString();
    int key = text == null ? 0 : text.length();
    int size = mTextCachedSizes.get(key);
    if (size != 0) {
        return size;
    }
    size = binarySearch(start, end, sizeTester, availableSpace);
    mTextCachedSizes.put(key, size);
    return size;
}

private static int binarySearch(int start, int end, SizeTester sizeTester,
        RectF availableSpace) {
    int lastBest = start;
    int lo = start;
    int hi = end - 1;
    int mid = 0;
    while (lo <= hi) {
        mid = (lo + hi) >>> 1;
        int midValCmp = sizeTester.onTestSize(mid, availableSpace);
        if (midValCmp < 0) {
            lastBest = lo;
            lo = mid + 1;
        } else if (midValCmp > 0) {
            hi = mid - 1;
            lastBest = hi;
        } else {
            return mid;
        }
    }
    // make sure to return last best
    // this is what should always be returned
    return lastBest;

}

@Override
protected void onTextChanged(final CharSequence text, final int start,
        final int before, final int after) {
    super.onTextChanged(text, start, before, after);
    reAdjust();
}

@Override
protected void onSizeChanged(int width, int height, int oldwidth,
        int oldheight) {
    mTextCachedSizes.clear();
    super.onSizeChanged(width, height, oldwidth, oldheight);
    if (width != oldwidth || height != oldheight) {
        reAdjust();
    }
}
}

හොඳයි, මම ඔබේ ප්‍රශ්නය පරීක්ෂා කර ඇත්තෙමි. ඔබට මෙම AutoResizeTextView අවශ්‍ය නොවේ ඔබේ ගැටලුව වෙනත් දෙයක්. මම ඔබේ ප්‍රශ්නයට අදහස් දක්වා ඇත්තෙමි. මෙම ගැටලුවට ප්‍රමාණය සමඟ කිසිදු සම්බන්ධයක් නැත.
එම්-වජේ

ArTargetApi (Build.VERSION_CODES.JELLY_BEAN) ඇති බව මම දුටුවෙමි. මෙය 2.3 හි වැඩ කළ හැකිද?
නික් ජියාන්

10
ඔබ එයට කැමති වීම ගැන සතුටුයි, ඔව් මට මෙහි බලපත්‍ර පරීක්‍ෂණයක් තිබේ. docs.google.com/document/d/…
M-WaJeEh

8
සිරස් අතට දේවල් කපා හැරෙන බැවින් මෙය අභිරුචි ටයිප්ෆේස් සමඟ ක්‍රියා නොකරන බව පෙනේ. ඔබ එය ආමන්ත්‍රණය කරන්නේ කෙසේද? අභිරුචි අකුරු සමඟ යමෙකු උත්සාහ කර තිබේද?
RealCasually

1
ObRobinHood සමහර විට StaticLayoutගරු නොකරයි width. 'ආපසු onTestSize()(ලබා ගත හැකි අවකාශය (mTextRect)) {// සමඟ අවසන් ආපසු එන රේඛා ආදේශ කරන්න. } else {if (mTextRect.bottom <availableSpace.bottom && mTextRect.right> availableSpace.right) {// හැක්: ඕ ආපසු -1; } // ඉතා විශාල ප්‍රතිලාභ 1; : '
එම්-වජේ

41

ඇත්ත වශයෙන්ම විසඳුමක් ගූගල් හි ඩයලොග් ටයිටල් පන්තියේ ඇත ... එය පිළිගත් එක තරම් effective ලදායී නොවුවද , එය බෙහෙවින් සරල වන අතර අනුවර්තනය වීමට පහසුය.

public class SingleLineTextView extends TextView {

  public SingleLineTextView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    setSingleLine();
    setEllipsize(TruncateAt.END);
  }

  public SingleLineTextView(Context context, AttributeSet attrs) {
    super(context, attrs);
    setSingleLine();
    setEllipsize(TruncateAt.END);
  }

  public SingleLineTextView(Context context) {
    super(context);
    setSingleLine();
    setEllipsize(TruncateAt.END);
  }

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);

    final Layout layout = getLayout();
    if (layout != null) {
      final int lineCount = layout.getLineCount();
      if (lineCount > 0) {
        final int ellipsisCount = layout.getEllipsisCount(lineCount - 1);
        if (ellipsisCount > 0) {

          final float textSize = getTextSize();

          // textSize is already expressed in pixels
          setTextSize(TypedValue.COMPLEX_UNIT_PX, (textSize - 1));

          super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        }
      }
    }
  }

}

2
මෙය තවත් + 1 ක් නොවන්නේ මන්දැයි සිතන්න. එය නිසැකවම ස්වභාවයෙන්ම කප්පාදුවක් නමුත් ඉතා සරල විසඳුමකි. ස්වල්පයක් පමණක් තැබූ යෙදුමක් සඳහා, අනුචලනය නොකල සරල ලයින් ටෙක්ස්ට් වීව්ස් මෙය යාමට පහසුම ක්‍රමයයි. නියමයි!
ටොම්

ඔබ ඇමතුමක් requestLayoutලබා නොගතහොත් මෙම විසඳුම ක්‍රියා නොකරයි onTextChanged. මෙම නිවැරදි කිරීම සමඟ වුවද, මෙහි ඇති විසඳුම බොහෝ අවස්ථාවන්හිදී අදාළ නොවේ, මන්ද පෙළ ප්‍රමාණය 1 කින් අඩු කිරීම සෑම අවස්ථාවකම ක්‍රියාත්මක නොවන බව පෙනේ: අපට පෙළ ප්‍රමාණය තවදුරටත් අඩු කිරීමට අවශ්‍ය විය හැකිය.
ෂි වැන්ග්

10
ErSergioCarneiro හොඳ සොයා ගැනීමක්, නමුත් පුනරාවර්තනයක් ඇති super.onMeasure(widthMeasureSpec, heightMeasureSpec);නොකරයි, නමුත් measure(widthMeasureSpec, heightMeasureSpec);. ඒ නිසා මම 'මිනුම් (widthMeasureSpec, heightMeasureSpec) "ලෙස වෙනස් කළ අතර එය ක්‍රියාත්මක වේ. මම ඇන්ඩ්‍රොයිඩ් 4.4 (නෙක්සස් 5) සහ ඇන්ඩ්‍රොයිඩ් 4.0.4 (සැම්සුන් එස්අයි-එල්ටීඊ) මත පරීක්‍ෂා කළ අතර එය ඒ දෙකම සමඟ හොඳින් ක්‍රියාත්මක විය (මම පවා භාවිතා කළෙමි අභිරුචි OTF ජපන් අකුරු). මගේ නිවැරදි කිරීම මෙතැනින් ලබා ගන්න: gist.github.com/mrleolink/0dfeef749da1b854a44b
ලියෝ

2
හොඳම පිළිතුර. ලියෝ ලින්ක් අනුවාදය භාවිතා කරමින්, එය මගේ 2.3 සහ 4.0 ඉමියුලේටර් සහ සත්‍ය උපාංග මත හොඳින් ක්‍රියා කරයි. ලැයිස්තු දර්ශනයක භාවිතා කරන්නේ නම්, පෙළ සැකසීමෙන් පසු requestLayout () අමතන්න මතක තබා ගන්න.
adbie

1
මම, @LeoLink අනුවාදය බෙදුනු එය භාවිතා කර ඇත maxLinesසහ requestLayout()මත setText@adbie කිව්වා වගේ. FixedLineTextView -> gist.github.com/Kevinrob/09742d9069e4e4e4ab66
කෙවින් රොබටෙල්

36

මම චේස්ගේ විසඳුම සමඟ ආරම්භ කළ නමුත් එය මගේ උපාංගයේ අපේක්ෂිත පරිදි ක්‍රියා කිරීමට පෙර කරුණු දෙකක් අනුවර්තනය වීමට සිදු විය (Galaxy Nexus, Android 4.1):

  1. පිරිසැලසුම මැනීම සඳහා ටෙක්ස්ට් පේන්ට් පිටපතක් භාවිතා කිරීම ටෙක්ස්ට්වීව්.ජෙට්පයින්ට් () සඳහා වන ලේඛනයේ සඳහන් වන්නේ එය කියවීමට පමණක් භාවිතා කළ යුතු බවයි, එබැවින් අපි තීන්ත වස්තුව මැනීම සඳහා භාවිතා කරන ස්ථාන දෙකෙහිම පිටපතක් සාදන ලදී:

    // 1. in resizeText()
    if (mAddEllipsis && targetTextSize == mMinTextSize && textHeight > height) {
      // Draw using a static layout
      // modified: use a copy of TextPaint for measuring
      TextPaint paint = new TextPaint(textPaint);
    
    // 2. in getTextHeight()
    private int getTextHeight(CharSequence source, TextPaint originalPaint, int width, float textSize) {
      // modified: make a copy of the original TextPaint object for measuring
      // (apparently the object gets modified while measuring, see also the
      // docs for TextView.getPaint() (which states to access it read-only)
      TextPaint paint = new TextPaint(originalPaint);
      // Update the text paint object
      paint.setTextSize(textSize);
      ...
  2. පෙළ ප්‍රමාණය සැකසීමට ඒකකයක් එක් කිරීම

    // modified: setting text size via this.setTextSize (instead of textPaint.setTextSize(targetTextSize))
    setTextSize(TypedValue.COMPLEX_UNIT_PX, targetTextSize);
    setLineSpacing(mSpacingAdd, mSpacingMult);

මෙම වෙනස් කිරීම් දෙක සමඟ විසඳුම මා වෙනුවෙන් හොඳින් ක්‍රියාත්මක වේ, ස්තූතියි චේස්! ඇන්ඩ්‍රොයිඩ් 4.x නිසා මුල් විසඳුම ක්‍රියාත්මක නොවන්නේ දැයි මම නොදනිමි. ඔබට එය ක්‍රියාකාරීව දැකීමට අවශ්‍ය නම් හෝ එය ඔබගේ උපාංගයේ සැබවින්ම ක්‍රියාත්මක වේදැයි පරීක්ෂා කිරීමට අවශ්‍ය නම්, ඔබට මගේ ෆ්ලෑෂ් කාඩ්පත් යෙදුම වන ෆ්ලෑෂ් කාඩ්පත් ටෝගෝ දෙස බැලිය හැකිය, එහිදී මම මෙම විසඳුම භාවිතා කරන්නේ ෆ්ලෑෂ් කාඩ්පතක පෙළ පරිමාණයට ය. පා text යට අත්තනෝමතික දිග තිබිය හැකි අතර, ෆ්ලෑෂ් කාඩ්පත් විවිධ ක්‍රියාකාරකම් වල, සමහර විට කුඩා, සමහර විට විශාල, සහ භූ දර්ශන + ආලේඛ්‍ය මාදිලියේ ප්‍රදර්ශනය වන අතර, විසඳුම නිසියාකාරව ක්‍රියාත්මක නොවන කිසිදු කොනක් මට හමු නොවීය ...


23

මම චේස්ගේ AutoResizeTextView පන්තියෙන් ආරම්භ කළ අතර එය සුළු වෙනසක් සිදු කළ අතර එය සිරස් සහ තිරස් අතට ගැලපේ.

තරමක් අපැහැදිලි තත්වයන් යටතේ පිරිසැලසුම් සංස්කාරකයේ (සූර්යග්‍රහණයේ) ශුන්‍ය දර්ශක ව්‍යතිරේකයක් ඇති කරන දෝෂයක් ද මම සොයා ගතිමි.

1 වෙනස් කරන්න: පෙළ සිරස් අතට හා තිරස් අතට සවි කරන්න

චේස්ගේ මුල් අනුවාදය සිරස් අතට ගැලපෙන තෙක් පෙළ ප්‍රමාණය අඩු කරයි, නමුත් පෙළ ඉලක්කයට වඩා පුළුල් වීමට ඉඩ දෙයි. මගේ නඩුවේදී, නිශ්චිත පළලකට සරිලන සේ මට පෙළ අවශ්‍ය විය.

මෙම වෙනස මඟින් පෙළ සිරස් අතට හා තිරස් අතට ගැලපෙන තෙක් ප්‍රමාණය වෙනස් කරයි.

දී resizeText(int ,int) වෙනසක්:

// Get the required text height
int textHeight = getTextHeight(text, textPaint, width, targetTextSize);

// Until we either fit within our text view or we had reached our min text size, incrementally try smaller sizes
while(textHeight > height && targetTextSize > mMinTextSize) {
    targetTextSize = Math.max(targetTextSize - 2, mMinTextSize);
    textHeight = getTextHeight(text, textPaint, width, targetTextSize);
    }

වෙත:

// Get the required text height
int textHeight = getTextHeight(text, textPaint, width, targetTextSize);
int textWidth  = getTextWidth(text, textPaint, width, targetTextSize);

// Until we either fit within our text view or we had reached our min text size, incrementally try smaller sizes
while(((textHeight >= height) || (textWidth >= width) ) && targetTextSize > mMinTextSize) {
    targetTextSize = Math.max(targetTextSize - 2, mMinTextSize);
    textHeight = getTextHeight(text, textPaint, width, targetTextSize);
    textWidth  = getTextWidth(text, textPaint, width, targetTextSize);
    }

ඉන්පසු, ගොනුව අවසානයේ, getTextWidth()පුරුද්ද එකතු කරන්න; එය තරමක් වෙනස් කර getTextHeight()ඇත. උස හා පළල යන දෙකම ලබා දෙන එක් පුරුද්දකට ඒවා ඒකාබද්ධ කිරීම වඩා කාර්යක්ෂම වනු ඇත.

// Set the text size of the text paint object and use a static layout to render text off screen before measuring
private int getTextWidth(CharSequence source, TextPaint paint, int width, float textSize) {
    // Update the text paint object
    paint.setTextSize(textSize);
    // Draw using a static layout
    StaticLayout layout = new StaticLayout(source, paint, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
    layout.draw(sTextResizeCanvas);
    return layout.getWidth();
}  




වෙනස 2: එක්ලිප්ස් ඇන්ඩ්‍රොයිඩ් පිරිසැලසුම් සංස්කාරකයේ හිස් ස්ටැක්එක්සෙප්ෂන් එකක් සවි කරන්න

තරමක් අපැහැදිලි සහ ඉතා නිවැරදි තත්වයන් යටතේ, පිරිසැලසුම් සංස්කාරකය පිරිසැලසුමේ චිත්‍රමය ප්‍රදර්ශනය කිරීමට අසමත් වනු ඇත; එය com.android.ide.eclipse.adt හි "EmpyStackException: null" ව්‍යතිරේකයක් විසි කරයි.

අවශ්‍ය කොන්දේසි:
- AutoResizeTextView විජට් එකක්
සාදන්න - එම විජට් සඳහා ශෛලියක් සාදන්න - ශෛලිය
තුළ පෙළ අයිතමය සඳහන් කරන්න; විජට් අර්ථ දැක්වීමේදී නොවේ

මෙන්:

res / layout / main.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <com.ajw.DemoCrashInADT.AutoResizeTextView
        android:id="@+id/resizingText"
        style="@style/myTextStyle" />

</LinearLayout>

res / values ​​/ myStyles.xml:

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <style name="myTextStyle" parent="@android:style/Widget.TextView">
        <item name="android:layout_height">wrap_content</item>
        <item name="android:layout_width">fill_parent</item>
        <item name="android:text">some message</item>
    </style>

</resources>

මෙම ලිපිගොනු සමඟ, සංස්කරණය කිරීමේදී චිත්‍රක පිරිසැලසුම් පටිත්ත තේරීමෙන් main.xmlපෙන්වනු ඇත:

දෝෂයක්!
EmpyStackException: ශුන්‍ය
ව්‍යතිරේක තොරතුරු කවුළුව තුළ ලොග් වී ඇත> දර්ශනය පෙන්වන්න> දෝෂ ලොගය

පිරිසැලසුමේ චිත්‍රමය දර්ශනය වෙනුවට.

දැනටමත් ඉතා දිගු කතාවක් කෙටියෙන් තබා ගැනීම සඳහා, මම මෙය පහත දැක්වෙන රේඛාවලට ලුහුබැඳ ගියෙමි (නැවත ඇතුළත resizeText):

// If there is a max text size set, use the lesser of that and the default text size
float targetTextSize = mMaxTextSize > 0 ? Math.min(mTextSize, mMaxTextSize) : mTextSize;

ගැටළුව වන්නේ නිශ්චිත කොන්දේසි යටතේ mTextSize කිසි විටෙකත් ආරම්භ නොකිරීමයි; එහි අගය 0 වේ.

ඉහත සඳහන් දෑ සමඟ targetTextSizeබිංදුවට සකසා ඇත (Math.min හි ප්‍රති result ලයක් ලෙස).

එම ශුන්‍යය තර්කය ලෙස getTextHeight()(සහ getTextWidth()) වෙත යොමු textSizeවේ. එය ලබා ගත් විට
layout.draw(sTextResizeCanvas);
අපට ව්‍යතිරේකය ලැබේ.

පරීක්ෂා කිරීමට වඩා (mTextSize == 0)ආරම්භයේදීම පරීක්ෂා කිරීම වඩා කාර්යක්ෂම resizeText()වේgetTextHeight() හා getTextWidth(); කලින් පරීක්‍ෂා කිරීමෙන් මැදිහත් වන සියලු කාර්යයන් ඉතිරි වේ.

මෙම යාවත්කාල කිරීම් සමඟ, ගොනුව (මගේ ක්‍රෑෂ්-ඩෙමෝ පරීක්ෂණ යෙදුමේ මෙන්) දැන්:

//
// from:  http://stackoverflow.com/questions/5033012/auto-scale-textview-text-to-fit-within-bounds
//
//

package com.ajw.DemoCrashInADT;

import android.content.Context;
import android.graphics.Canvas;
import android.text.Layout.Alignment;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.widget.TextView;

/**
 * Text view that auto adjusts text size to fit within the view. If the text
 * size equals the minimum text size and still does not fit, append with an
 * ellipsis.
 *
 * 2011-10-29 changes by Alan Jay Weiner
 *              * change to fit both vertically and horizontally  
 *              * test mTextSize for 0 in resizeText() to fix exception in Layout Editor
 *
 * @author Chase Colburn
 * @since Apr 4, 2011
 */
public class AutoResizeTextView extends TextView {

    // Minimum text size for this text view
    public static final float MIN_TEXT_SIZE = 20;

    // Interface for resize notifications
    public interface OnTextResizeListener {
        public void onTextResize(TextView textView, float oldSize, float newSize);
    }

    // Off screen canvas for text size rendering
    private static final Canvas sTextResizeCanvas = new Canvas();

    // Our ellipse string
    private static final String mEllipsis = "...";

    // Registered resize listener
    private OnTextResizeListener mTextResizeListener;

    // Flag for text and/or size changes to force a resize
    private boolean mNeedsResize = false;

    // Text size that is set from code. This acts as a starting point for
    // resizing
    private float mTextSize;

    // Temporary upper bounds on the starting text size
    private float mMaxTextSize = 0;

    // Lower bounds for text size
    private float mMinTextSize = MIN_TEXT_SIZE;

    // Text view line spacing multiplier
    private float mSpacingMult = 1.0f;

    // Text view additional line spacing
    private float mSpacingAdd = 0.0f;

    // Add ellipsis to text that overflows at the smallest text size
    private boolean mAddEllipsis = true;


    // Default constructor override
    public AutoResizeTextView(Context context) {
        this(context, null);
    }


    // Default constructor when inflating from XML file
    public AutoResizeTextView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }


    // Default constructor override
    public AutoResizeTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mTextSize = getTextSize();
    }


    /**
     * When text changes, set the force resize flag to true and reset the text
     * size.
     */
    @Override
    protected void onTextChanged(final CharSequence text, final int start,
            final int before, final int after) {
        mNeedsResize = true;
        // Since this view may be reused, it is good to reset the text size
        resetTextSize();
    }


    /**
     * If the text view size changed, set the force resize flag to true
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        if (w != oldw || h != oldh) {
            mNeedsResize = true;
        }
    }


    /**
     * Register listener to receive resize notifications
     *
     * @param listener
     */
    public void setOnResizeListener(OnTextResizeListener listener) {
        mTextResizeListener = listener;
    }


    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(float size) {
        super.setTextSize(size);
        mTextSize = getTextSize();
    }


    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(int unit, float size) {
        super.setTextSize(unit, size);
        mTextSize = getTextSize();
    }


    /**
     * Override the set line spacing to update our internal reference values
     */
    @Override
    public void setLineSpacing(float add, float mult) {
        super.setLineSpacing(add, mult);
        mSpacingMult = mult;
        mSpacingAdd = add;
    }


    /**
     * Set the upper text size limit and invalidate the view
     *
     * @param maxTextSize
     */
    public void setMaxTextSize(float maxTextSize) {
        mMaxTextSize = maxTextSize;
        requestLayout();
        invalidate();
    }


    /**
     * Return upper text size limit
     *
     * @return
     */
    public float getMaxTextSize() {
        return mMaxTextSize;
    }


    /**
     * Set the lower text size limit and invalidate the view
     *
     * @param minTextSize
     */
    public void setMinTextSize(float minTextSize) {
        mMinTextSize = minTextSize;
        requestLayout();
        invalidate();
    }


    /**
     * Return lower text size limit
     *
     * @return
     */
    public float getMinTextSize() {
        return mMinTextSize;
    }


    /**
     * Set flag to add ellipsis to text that overflows at the smallest text size
     *
     * @param addEllipsis
     */
    public void setAddEllipsis(boolean addEllipsis) {
        mAddEllipsis = addEllipsis;
    }


    /**
     * Return flag to add ellipsis to text that overflows at the smallest text
     * size
     *
     * @return
     */
    public boolean getAddEllipsis() {
        return mAddEllipsis;
    }


    /**
     * Reset the text to the original size
     */
    public void resetTextSize() {
        super.setTextSize(TypedValue.COMPLEX_UNIT_PX, mTextSize);
        mMaxTextSize = mTextSize;
    }


    /**
     * Resize text after measuring
     */
    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        if (changed || mNeedsResize) {
            int widthLimit = (right - left) - getCompoundPaddingLeft()
                    - getCompoundPaddingRight();
            int heightLimit = (bottom - top) - getCompoundPaddingBottom()
                    - getCompoundPaddingTop();
            resizeText(widthLimit, heightLimit);
        }
        super.onLayout(changed, left, top, right, bottom);
    }


    /**
     * Resize the text size with default width and height
     */
    public void resizeText() {
        int heightLimit = getHeight() - getPaddingBottom() - getPaddingTop();
        int widthLimit = getWidth() - getPaddingLeft() - getPaddingRight();
        resizeText(widthLimit, heightLimit);
    }


    /**
     * Resize the text size with specified width and height
     *
     * @param width
     * @param height
     */
    public void resizeText(int width, int height) {
        CharSequence text = getText();
        // Do not resize if the view does not have dimensions or there is no
        // text
        // or if mTextSize has not been initialized
        if (text == null || text.length() == 0 || height <= 0 || width <= 0
                || mTextSize == 0) {
            return;
        }

        // Get the text view's paint object
        TextPaint textPaint = getPaint();

        // Store the current text size
        float oldTextSize = textPaint.getTextSize();

        // If there is a max text size set, use the lesser of that and the
        // default text size
        float targetTextSize = mMaxTextSize > 0 ? Math.min(mTextSize, mMaxTextSize)
                : mTextSize;

        // Get the required text height
        int textHeight = getTextHeight(text, textPaint, width, targetTextSize);
        int textWidth = getTextWidth(text, textPaint, width, targetTextSize);

        // Until we either fit within our text view or we had reached our min
        // text size, incrementally try smaller sizes
        while (((textHeight > height) || (textWidth > width))
                && targetTextSize > mMinTextSize) {
            targetTextSize = Math.max(targetTextSize - 2, mMinTextSize);
            textHeight = getTextHeight(text, textPaint, width, targetTextSize);
            textWidth = getTextWidth(text, textPaint, width, targetTextSize);
        }

        // If we had reached our minimum text size and still don't fit, append
        // an ellipsis
        if (mAddEllipsis && targetTextSize == mMinTextSize && textHeight > height) {
            // Draw using a static layout
            StaticLayout layout = new StaticLayout(text, textPaint, width,
                    Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, false);
            layout.draw(sTextResizeCanvas);
            int lastLine = layout.getLineForVertical(height) - 1;
            int start = layout.getLineStart(lastLine);
            int end = layout.getLineEnd(lastLine);
            float lineWidth = layout.getLineWidth(lastLine);
            float ellipseWidth = textPaint.measureText(mEllipsis);

            // Trim characters off until we have enough room to draw the
            // ellipsis
            while (width < lineWidth + ellipseWidth) {
                lineWidth = textPaint.measureText(text.subSequence(start, --end + 1)
                        .toString());
            }
            setText(text.subSequence(0, end) + mEllipsis);

        }

        // Some devices try to auto adjust line spacing, so force default line
        // spacing
        // and invalidate the layout as a side effect
        textPaint.setTextSize(targetTextSize);
        setLineSpacing(mSpacingAdd, mSpacingMult);

        // Notify the listener if registered
        if (mTextResizeListener != null) {
            mTextResizeListener.onTextResize(this, oldTextSize, targetTextSize);
        }

        // Reset force resize flag
        mNeedsResize = false;
    }


    // Set the text size of the text paint object and use a static layout to
    // render text off screen before measuring
    private int getTextHeight(CharSequence source, TextPaint paint, int width,
            float textSize) {
        // Update the text paint object
        paint.setTextSize(textSize);
        // Draw using a static layout
        StaticLayout layout = new StaticLayout(source, paint, width,
                Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
        layout.draw(sTextResizeCanvas);
        return layout.getHeight();
    }


    // Set the text size of the text paint object and use a static layout to
    // render text off screen before measuring
    private int getTextWidth(CharSequence source, TextPaint paint, int width,
            float textSize) {
        // Update the text paint object
        paint.setTextSize(textSize);
        // Draw using a static layout
        StaticLayout layout = new StaticLayout(source, paint, width,
                Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
        layout.draw(sTextResizeCanvas);
        return layout.getWidth();
    }

}



ආරම්භක කේතය පළ කිරීම ගැන චේස්ට විශාල ස්තූතියක්. එය ක්‍රියාත්මක වන ආකාරය බැලීමට මම එය කියවීමෙන් සතුටට පත් වූ අතර එයට එකතු කිරීමට හැකිවීම ගැන මම සතුටු වෙමි.


3
BUG DETECTED: getTextWidth()ඔබ StaticLayoutඉදිකිරීම්කරු තුළ අපේක්ෂිත පළල පසු කරන බැවින් කිසිසේත් ක්‍රියා නොකරයි . එම අවස්ථාවේ දී getWidth()ක්‍රමයෙන් කුමන පළල ලබා දෙනු ඇත්දැයි අනුමාන කරන්න ?
se.solovyev

මේ වගේ පැකිලෙන මා වැනි අනාගත පුද්ගලයින් සඳහා සටහනක්: තීන්ත වස්තුවට මිනුම් ටෙක්ස්ට් ක්‍රමයක් ඇති අතර එය පා of යේ පළල ලබා ගැනීම සඳහා කැඳවිය හැකිය.
Xono

2
ලස්සනයි. මෙම කේතය මගේ KeepScore යෙදුමේ ( github.com/nolanlawson/KeepScore ) මගේ Galaxy Nexus හි Android 4.1 හි වැඩ කර ඇති අතර චේස්ගේ අනුවාදය එසේ නොවීය. විශේෂාංග ඉල්ලීම: කරුණාකර මෙම කේතය GitHub හි තබන්න, යාලුවනේ! පැච් සහ කේත සමාලෝචන සඳහා ස්ථානය ස්ටැක් ඕවර්ෆ්ලෝ නොවේ. :)
nlawson

1
ඉහත සඳහන් කළ පරිදි, layout.getWidth () එය නිර්මාණය කළ පළල නැවත ලබා දෙයි. මෙම කාර්යය කිරීමට මට කළ යුතුව තිබුණේ 4096 පළලකින් පිරිසැලසුම නිර්මාණය කිරීම, ඉන්පසු සියලු රේඛාවලින් getLineWidth () අමතා උපරිම අගය ලබා දීමයි.
එඩ්වඩ් ෆෝක්

1
22 වැඩ නොකරන කේතයක් සඳහා ඉහළ නැංවීම / ඇයි ?! පේළි කිහිපයක් Igave කර ඇති අතර අවකාශයට

22

AppcompatTextView දැන් ආධාරක පුස්තකාලයෙන් ආරම්භ වන ස්වයංක්‍රීය ප්‍රමාණයට සහය දක්වයි 26.0. ඇන්ඩ්‍රොයිඩ් ඕ හි පෙළ දර්ශනය ද එලෙසම ක්‍රියාත්මක වේ. වැඩි විස්තර මෙහි සොයාගත හැකිය. සරල නිරූපණ යෙදුමක් මෙහි සොයාගත හැකිය .

<LinearLayout
      xmlns:android="http://schemas.android.com/apk/res/android"
      xmlns:app="http://schemas.android.com/apk/res-auto"
      android:layout_width="match_parent"
      android:layout_height="wrap_content">

      <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:autoSizeTextType="uniform"
        app:autoSizeMinTextSize="12sp"
        app:autoSizeMaxTextSize="100sp"
        app:autoSizeStepGranularity="2sp"
      />

</LinearLayout>

2
මම සකස් විය android:lines="1"පෙළ නැවත ප්රමාණය කිරීම සඳහා TextView දී. මෙම ගුණාංගය නොමැතිව, පෙළ පේළි දෙකකින් ඔතා ඇත.
පණ ගැන්වේ

මෙම කාර්යය කිරීමට ආනයනය කුමක්ද? 26.0.0-බීටා -1
Psest328

1
@ Psest328, ඔබ maven {url " maven.google.com " using භාවිතා කරනවාද?
මකාටා

17

ඇන්ඩ්‍රොයිඩ් 4.x සඳහා වන ක්‍රියාමාර්ග:

මට AutoResizeTextView හමු වූ අතර එය මගේ ඇන්ඩ්‍රොයිඩ් 2.1 ඉමුලේටරය මත හොඳින් ක්‍රියා කරයි. මම ඒකට ගොඩක් කැමතියි. නමුත් අවාසනාවකට එය මගේම 4.0.4 ජංගම දුරකථනයකින් සහ 4.1 ඉමියුලේටරයකින් අසමත් විය. උත්සාහ කිරීමෙන් පසු xml හි AutoResizeTextView පන්තියේ පහත ගුණාංග එකතු කිරීමෙන් එය පහසුවෙන් විසඳා ගත හැකි බව මට පෙනී ගියේය:

android: ellipsize = "කිසිවක් නැත"

android: singleLine = "සත්‍ය"

ඉහත පේළි 2 සමඟ, දැන් AutoResizeTextView මගේ 2.1 සහ 4.1 ඉමුලේටර් සහ මගේම 4.0.4 ජංගම දුරකථනය මත හොඳින් ක්‍රියා කරයි.

මෙය ඔබට උපකාරී වේ යැයි සිතමි. :-)


2
Oonoelle විසින් යෝජනා කරන ලද විසඳුම සඳහා මම අනිවාර්යයෙන්ම නිර්දේශ කරමි. එය සෑම අවස්ථාවකම පරිපූර්ණව ක්රියා කරයි. උදා: මගේ නඩුව තනි පේළි පෙළ දර්ශනයක් නොවේ.
බොරිස් ස්ට්‍රැන්ඩ්ජෙව්

16

ඇන්ඩ්‍රොයිඩ් පැණි වද සහ අයිස්ක්‍රීම් සැන්ඩ්විච් වල අනතුරු ඇඟවීම, දෝෂය

ඇන්ඩ්‍රොයිඩ් අනුවාද: 3.1 - 4.04 හි දෝෂයක් ඇත, එය ටෙක්ස්ට්වීව් හි සෙට් ටෙක්ස්ට්සයිස් () ක්‍රියාත්මක වන්නේ පළමු වරට පමණි (1 වන ආයාචනය).

දෝෂය මෙහි විස්තර කර ඇත: http://code.google.com/p/android/issues/detail?id=22493 http://code.google.com/p/android/issues/detail?id=17343#c9

වැඩ කිරීම යනු ප්‍රමාණය වෙනස් කිරීමට පෙර පෙළ දර්ශනයට පවරා ඇති පෙළට නව රේඛා අක්ෂර එකතු කිරීමයි:

final String DOUBLE_BYTE_SPACE = "\u3000";
textView.append(DOUBLE_BYTE_SPACE);

මම එය මගේ කේතයේ පහත පරිදි භාවිතා කරමි:

final String DOUBLE_BYTE_SPACE = "\u3000";
AutoResizeTextView textView = (AutoResizeTextView) view.findViewById(R.id.aTextView);
String fixString = "";
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.HONEYCOMB_MR1
   && android.os.Build.VERSION.SDK_INT <= android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1) {  
    fixString = DOUBLE_BYTE_SPACE;
}
textView.setText(fixString + "The text" + fixString);

මම මෙම "\ u3000" අක්‍ෂරය මගේ පෙළෙහි වමේ සහ දකුණේ එකතු කර එය කේන්ද්‍රගතව තබා ගන්නෙමි. ඔබ එය වමට පෙළගස්වා ඇත්නම් දකුණට පමණක් එකතු කරන්න. ඇත්ත වශයෙන්ම එය AutoResizeTextView විජට් සමඟද කාවැද්දිය හැකිය, නමුත් මට අවශ්‍ය වූයේ පිටත කේත නිවැරදිව තබා ගැනීමටයි.


පෙළ සිරස් අතට කේන්ද්‍රගතව තබා ගැනීමට ⁠ "\ u2060" එකතු කරන ලදි.
igla

14

2017 දී ගූගල් අයිඕ සමුළුවේදී ගූගල් විසින් ටෙක්ස්ට්වීව් හි ස්වයංක්‍රීය ප්‍රමාණ දේපල හඳුන්වා දෙන ලදී

https://youtu.be/fjUdJ2aVqE4

<android.support.v7.widget.AppCompatTextView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:text="@string/my_text"
        app:autoSizeTextType="uniform"
        app:autoSizeMaxTextSize="10sp"
        app:autoSizeMinTextSize="6sp"
        app:autoSizeStepGranularity="1sp"/>

android.support.v7.widget.AppCompatTextView වැදගත්. එය යෙදුම මෙන් ක්‍රියා නොකරයි: ස්වයංක්‍රීය පෙළ ටෙක්ස්ට් ටයිප් ස්ථාවර පෙළ දර්ශනයේ ගුණාංග. සුභ පැතුම් ...
සිනාන් අර්ජින්

මට වැඩ කරයි! යෙදුම එකතු කළ යුතුය: autoSizeTextType = "ඒකාකාර" රේඛාව
avisper

13

මගේ අවශ්‍යතාවය වූයේ දර්ශන සීමාවට හොඳින් ගැලපෙන පරිදි පෙළ ප්‍රමාණය වෙනස් කිරීමයි. චේස්ගේ විසඳුම පෙළ ප්‍රමාණය පමණක් අඩු කරයි, ප්‍රමාණවත් ඉඩක් ඇත්නම් මෙයද පෙළ විශාල කරයි.

සියල්ල වේගවත් හා නිරවද්‍ය කිරීමට මම ක්‍රමවේදය තුළ දැකිය හැකි පරිදි, පුනරාවර්තනයක් වෙනුවට ද්වි-වෙන් කිරීමේ ක්‍රමයක් භාවිතා කළෙමි resizeText(). ඔබටත් අMAX_TEXT_SIZE විකල්පයක් . මම ඔනෙල්ගේ ඉඟි ද ඇතුළත් කළෙමි.

ඇන්ඩ්‍රොයිඩ් 4.4 හි පරීක්ෂා කර ඇත

/**
 *               DO WHAT YOU WANT TO PUBLIC LICENSE
 *                    Version 2, December 2004
 *
 * Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
 *
 * Everyone is permitted to copy and distribute verbatim or modified
 * copies of this license document, and changing it is allowed as long
 * as the name is changed.
 *
 *            DO WHAT YOU WANT TO PUBLIC LICENSE
 *   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
 *
 *  0. You just DO WHAT YOU WANT TO.
 */

import android.content.Context;
import android.text.Layout.Alignment;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.widget.TextView;

/**
 * Text view that auto adjusts text size to fit within the view.
 * If the text size equals the minimum text size and still does not
 * fit, append with an ellipsis.
 *
 * @author Chase Colburn
 * @since Apr 4, 2011
 */
public class AutoResizeTextView extends TextView {

    // Minimum text size for this text view
    public static final float MIN_TEXT_SIZE = 26;

    // Maximum text size for this text view
    public static final float MAX_TEXT_SIZE = 128;

    private static final int BISECTION_LOOP_WATCH_DOG = 30;

    // Interface for resize notifications
    public interface OnTextResizeListener {
        public void onTextResize(TextView textView, float oldSize, float newSize);
    }

    // Our ellipse string
    private static final String mEllipsis = "...";

    // Registered resize listener
    private OnTextResizeListener mTextResizeListener;

    // Flag for text and/or size changes to force a resize
    private boolean mNeedsResize = false;

    // Text size that is set from code. This acts as a starting point for resizing
    private float mTextSize;

    // Temporary upper bounds on the starting text size
    private float mMaxTextSize = MAX_TEXT_SIZE;

    // Lower bounds for text size
    private float mMinTextSize = MIN_TEXT_SIZE;

    // Text view line spacing multiplier
    private float mSpacingMult = 1.0f;

    // Text view additional line spacing
    private float mSpacingAdd = 0.0f;

    // Add ellipsis to text that overflows at the smallest text size
    private boolean mAddEllipsis = true;

    // Default constructor override
    public AutoResizeTextView(Context context) {
        this(context, null);
    }

    // Default constructor when inflating from XML file
    public AutoResizeTextView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    // Default constructor override
    public AutoResizeTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mTextSize = getTextSize();
    }

    /**
     * When text changes, set the force resize flag to true and reset the text size.
     */
    @Override
    protected void onTextChanged(final CharSequence text, final int start, final int before, final int after) {
        mNeedsResize = true;
        // Since this view may be reused, it is good to reset the text size
        resetTextSize();
    }

    /**
     * If the text view size changed, set the force resize flag to true
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        if (w != oldw || h != oldh) {
            mNeedsResize = true;
        }
    }

    /**
     * Register listener to receive resize notifications
     * @param listener
     */
    public void setOnResizeListener(OnTextResizeListener listener) {
        mTextResizeListener = listener;
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(float size) {
        super.setTextSize(size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(int unit, float size) {
        super.setTextSize(unit, size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set line spacing to update our internal reference values
     */
    @Override
    public void setLineSpacing(float add, float mult) {
        super.setLineSpacing(add, mult);
        mSpacingMult = mult;
        mSpacingAdd = add;
    }

    /**
     * Set the upper text size limit and invalidate the view
     * @param maxTextSize
     */
    public void setMaxTextSize(float maxTextSize) {
        mMaxTextSize = maxTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return upper text size limit
     * @return
     */
    public float getMaxTextSize() {
        return mMaxTextSize;
    }

    /**
     * Set the lower text size limit and invalidate the view
     * @param minTextSize
     */
    public void setMinTextSize(float minTextSize) {
        mMinTextSize = minTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return lower text size limit
     * @return
     */
    public float getMinTextSize() {
        return mMinTextSize;
    }

    /**
     * Set flag to add ellipsis to text that overflows at the smallest text size
     * @param addEllipsis
     */
    public void setAddEllipsis(boolean addEllipsis) {
        mAddEllipsis = addEllipsis;
    }

    /**
     * Return flag to add ellipsis to text that overflows at the smallest text size
     * @return
     */
    public boolean getAddEllipsis() {
        return mAddEllipsis;
    }

    /**
     * Reset the text to the original size
     */
    public void resetTextSize() {
        if(mTextSize > 0) {
            super.setTextSize(TypedValue.COMPLEX_UNIT_PX, mTextSize);
            //mMaxTextSize = mTextSize;
        }
    }

    /**
     * Resize text after measuring
     */
    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        if(changed || mNeedsResize) {
            int widthLimit = (right - left) - getCompoundPaddingLeft() - getCompoundPaddingRight();
            int heightLimit = (bottom - top) - getCompoundPaddingBottom() - getCompoundPaddingTop();
            resizeText(widthLimit, heightLimit);
        }
        super.onLayout(changed, left, top, right, bottom);
    }


    /**
     * Resize the text size with default width and height
     */
    public void resizeText() {
        int heightLimit = getHeight() - getPaddingBottom() - getPaddingTop();
        int widthLimit = getWidth() - getPaddingLeft() - getPaddingRight();
        resizeText(widthLimit, heightLimit);
    }

    /**
     * Resize the text size with specified width and height
     * @param width
     * @param height
     */
    public void resizeText(int width, int height) {
        CharSequence text = getText();
        // Do not resize if the view does not have dimensions or there is no text
        if(text == null || text.length() == 0 || height <= 0 || width <= 0 || mTextSize == 0) {
            return;
        }

        // Get the text view's paint object
        TextPaint textPaint = getPaint();

        // Store the current text size
        float oldTextSize = textPaint.getTextSize();

        // Bisection method: fast & precise
        float lower = mMinTextSize;
        float upper = mMaxTextSize;
        int loop_counter=1;
        float targetTextSize = (lower+upper)/2;
        int textHeight = getTextHeight(text, textPaint, width, targetTextSize);
        while(loop_counter < BISECTION_LOOP_WATCH_DOG && upper - lower > 1) {
            targetTextSize = (lower+upper)/2;
            textHeight = getTextHeight(text, textPaint, width, targetTextSize);
            if(textHeight > height)
                upper = targetTextSize;
            else
                lower = targetTextSize;
            loop_counter++;
        }

        targetTextSize = lower;
        textHeight = getTextHeight(text, textPaint, width, targetTextSize);

        // If we had reached our minimum text size and still don't fit, append an ellipsis
        if(mAddEllipsis && targetTextSize == mMinTextSize && textHeight > height) {
            // Draw using a static layout
            // modified: use a copy of TextPaint for measuring
            TextPaint paintCopy = new TextPaint(textPaint);
            paintCopy.setTextSize(targetTextSize);
            StaticLayout layout = new StaticLayout(text, paintCopy, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, false);
            // Check that we have a least one line of rendered text
            if(layout.getLineCount() > 0) {
                // Since the line at the specific vertical position would be cut off,
                // we must trim up to the previous line
                int lastLine = layout.getLineForVertical(height) - 1;
                // If the text would not even fit on a single line, clear it
                if(lastLine < 0) {
                    setText("");
                }
                // Otherwise, trim to the previous line and add an ellipsis
                else {
                    int start = layout.getLineStart(lastLine);
                    int end = layout.getLineEnd(lastLine);
                    float lineWidth = layout.getLineWidth(lastLine);
                    float ellipseWidth = paintCopy.measureText(mEllipsis);

                    // Trim characters off until we have enough room to draw the ellipsis
                    while(width < lineWidth + ellipseWidth) {
                        lineWidth = paintCopy.measureText(text.subSequence(start, --end + 1).toString());
                    }
                    setText(text.subSequence(0, end) + mEllipsis);
                }
            }
        }

        // Some devices try to auto adjust line spacing, so force default line spacing
        // and invalidate the layout as a side effect
        setTextSize(TypedValue.COMPLEX_UNIT_PX, targetTextSize);
        setLineSpacing(mSpacingAdd, mSpacingMult);

        // Notify the listener if registered
        if(mTextResizeListener != null) {
            mTextResizeListener.onTextResize(this, oldTextSize, targetTextSize);
        }

        // Reset force resize flag
        mNeedsResize = false;
    }

    // Set the text size of the text paint object and use a static layout to render text off screen before measuring
    private int getTextHeight(CharSequence source, TextPaint originalPaint, int width, float textSize) {
        // modified: make a copy of the original TextPaint object for measuring
        // (apparently the object gets modified while measuring, see also the
        // docs for TextView.getPaint() (which states to access it read-only)
        TextPaint paint = new TextPaint(originalPaint);
        // Update the text paint object
        paint.setTextSize(textSize);
        // Measure using a static layout
        StaticLayout layout = new StaticLayout(source, paint, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
        return layout.getHeight();
    }

}

මෙය ක්‍රියාත්මක වේ! මුල් කේතයෙන් සමහර ගැටලු ද විසඳන්න.
නික් ජියාන්

11

මම මෙය සදහටම සොයමින් සිටින අතර, මීට ටික කලකට පෙර මෙහි නැති වූ විසඳුමක් මට හමු වූ හෙයින්, අනාගත යොමු කිරීම සඳහා ද මම එය මෙහි ලියන්නෙමි.

සටහන: මෙම කේතය ගූගල් ඇන්ඩ්‍රොයිඩ් ලොලිපොප් ඩයලර් වෙතින් ටික වේලාවකට පෙර කෙලින්ම ලබා ගන්නා ලදි, මට මතක නැත එකල වෙනස්කම් සිදු කර ඇත්නම්. එසේම, මෙය කුමන බලපත්‍රය යටතේ දැයි මම නොදනිමි, නමුත් එය එසේ යැයි සිතීමට මට හේතු තිබේ Apache 2.0.

පංතිය ResizeTextView, ඇත්තView

public class ResizeTextView extends TextView {

private final int mOriginalTextSize;
private final int mMinTextSize;
private final static int sMinSize = 20;
public ResizeTextView(Context context, AttributeSet attrs) {
    super(context, attrs);
    mOriginalTextSize = (int) getTextSize();
    mMinTextSize = (int) sMinSize;
}
@Override
protected void onTextChanged(CharSequence text, int start, int lengthBefore, int lengthAfter) {
    super.onTextChanged(text, start, lengthBefore, lengthAfter);
    ViewUtil.resizeText(this, mOriginalTextSize, mMinTextSize);
}
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    ViewUtil.resizeText(this, mOriginalTextSize, mMinTextSize);
}

මෙම ResizeTextViewපංතියට පෙළ දර්ශනය සහ එහි සියලුම දරුවන් මා විසින් අවලංගු කරන පරිදි දීර් extend කළ හැකිය, එබැවින් සංස්කරණ ටෙක්ස්ට් ද.

ViewUtilක්රමය සමඟ පන්තියresizeText(...)

/*
* Copyright (C) 2012 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.
*/

import android.graphics.Paint;
import android.util.TypedValue;
import android.widget.TextView;

public class ViewUtil {

    private ViewUtil() {}

    public static void resizeText(TextView textView, int originalTextSize, int minTextSize) {
        final Paint paint = textView.getPaint();
        final int width = textView.getWidth();
        if (width == 0) return;
        textView.setTextSize(TypedValue.COMPLEX_UNIT_PX, originalTextSize);
        float ratio = width / paint.measureText(textView.getText().toString());
        if (ratio <= 1.0f) {
            textView.setTextSize(TypedValue.COMPLEX_UNIT_PX,
                    Math.max(minTextSize, originalTextSize * ratio));
        }
    }
}

ඔබ ඔබේ මතය මෙසේ සැකසිය යුතුය

<yourpackage.yourapp.ResizeTextView
            android:layout_width="match_parent"
            android:layout_height="64dp"
            android:gravity="center"
            android:maxLines="1"/>

එය උපකාරී වේ යැයි සිතමු!


ලබා ගැනීමේ පළල සැමවිටම 0 වේ
ආචාර්ය aNdRO

එය ස්වයංක්‍රීයව ප්‍රතිනිර්මාණය කළ යුතු බැවින් ඔබට වලංගු පළලක් ලබා දීමට අදහස් නොකෙරේ. ඔබට දර්ශනයේ සත්‍ය පළල ලබා ගැනීමට අවශ්‍ය නම් ඔබට තවමත් ViewTreeObserver අමතන්න.
මැටියෝ

1
ඔබේ විසඳුම ප්‍රශස්ත ලෙස පරිමාණය නොවේ. පෙළ දර්ශනයක් තුළ සුදු අවකාශය වැඩිය.
CoolMind

8

මෙය ඔබට උපකාරී වනු ඇතැයි මම බලාපොරොත්තු වෙමි

import android.content.Context;
import android.graphics.Rect;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.widget.TextView;

/* Based on 
 * from http://stackoverflow.com/questions/2617266/how-to-adjust-text-font-size-to-fit-textview
 */
public class FontFitTextView extends TextView {

private static float MAX_TEXT_SIZE = 20;

public FontFitTextView(Context context) {
    this(context, null);
}

public FontFitTextView(Context context, AttributeSet attrs) {
    super(context, attrs);

    float size = this.getTextSize();
    if (size > MAX_TEXT_SIZE)
        setTextSize(MAX_TEXT_SIZE);
}

private void refitText(String text, int textWidth) {
    if (textWidth > 0) {
        float availableWidth = textWidth - this.getPaddingLeft()
                - this.getPaddingRight();

        TextPaint tp = getPaint();
        Rect rect = new Rect();
        tp.getTextBounds(text, 0, text.length(), rect);
        float size = rect.width();

        if (size > availableWidth)
            setTextScaleX(availableWidth / size);
    }
}

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    int parentWidth = MeasureSpec.getSize(widthMeasureSpec);
    int parentHeight = MeasureSpec.getSize(heightMeasureSpec);
    refitText(this.getText().toString(), parentWidth);
    this.setMeasuredDimension(parentWidth, parentHeight);
}

@Override
protected void onTextChanged(final CharSequence text, final int start,
        final int before, final int after) {
    refitText(text.toString(), this.getWidth());
}

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    if (w != oldw) {
        refitText(this.getText().toString(), w);
    }
}
}

සටහන: පෙළ ප්‍රමාණය 20 ට වඩා විශාල නම් මම MAX_TEXT_SIZE භාවිතා කරමි, මන්ද මගේ දර්ශනයට විශාල අකුරු යෙදීමට ඉඩ දීමට මට අවශ්‍ය නැති නිසා, මෙය ඔබගේ කාරණය නොවේ නම්, ඔබට එය ඉවත් කළ හැකිය.


2
හ්ම්- “setTextScaleX” භාවිතය පා text ය කුඩා (නමුත් කියවිය හැකි) ආකෘතියකට ප්‍රතිප්‍රමාණනය කරනවාට වඩා තිරස් අතට තලා දැමීමට හේතු වන බව පෙනේ. එසේම, එය පෙළ එතුම හිතකාමී නොවන බව පෙනේ.
නේතන් පය

SetTextScaleX හි අතුරු ආබාධය සිත්ගන්නාසුලු ලෙස භාවිතා කිරීම. පෙළ ප්‍රමාණය වෙනස් කිරීමට වඩා එය සැලකිය යුතු ලෙස වෙනස් දැයි මට සිතේ. කේතය එක හා සමාන විය හැකි නමුත් එය ස්වදේශීය බැවින් මට ප්‍රභවය නොපෙනේ.
SMBiggs

තවත් අදහසක්, ඇයි පරීක්ෂා කරන්නේ scale > availableWidth? පවතින ප්‍රමාණය මුලින් තිබූ ප්‍රමාණයට වඩා කුඩා නම් මෙය පමණක් ප්‍රමාණය වෙනස් කරයි. දිගු කිරීම (හෝ පෙළ විශාල කිරීම) සිදු නොවේ.
SMBiggs

8

TextChangedListened එයට එකතු කර ඇති TextView භාවිතා කරන සරල විසඳුමක් මෙන්න:

expressionView = (TextView) findViewById(R.id.expressionView);
expressionView.addTextChangedListener(textAutoResizeWatcher(expressionView, 25, 55));

private TextWatcher textAutoResizeWatcher(final TextView view, final int MIN_SP, final int MAX_SP) {
    return new TextWatcher() {
        @Override
        public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {}

        @Override
        public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {}

        @Override
        public void afterTextChanged(Editable editable) {

            final int widthLimitPixels = view.getWidth() - view.getPaddingRight() - view.getPaddingLeft();
            Paint paint = new Paint();
            float fontSizeSP = pixelsToSp(view.getTextSize());
            paint.setTextSize(spToPixels(fontSizeSP));

            String viewText = view.getText().toString();

            float widthPixels = paint.measureText(viewText);

            // Increase font size if necessary.
            if (widthPixels < widthLimitPixels){
                while (widthPixels < widthLimitPixels && fontSizeSP <= MAX_SP){
                    ++fontSizeSP;
                    paint.setTextSize(spToPixels(fontSizeSP));
                    widthPixels = paint.measureText(viewText);
                }
                --fontSizeSP;
            }
            // Decrease font size if necessary.
            else {
                while (widthPixels > widthLimitPixels || fontSizeSP > MAX_SP) {
                    if (fontSizeSP < MIN_SP) {
                        fontSizeSP = MIN_SP;
                        break;
                    }
                    --fontSizeSP;
                    paint.setTextSize(spToPixels(fontSizeSP));
                    widthPixels = paint.measureText(viewText);
                }
            }

            view.setTextSize(fontSizeSP);
        }
    };
}

private float pixelsToSp(float px) {
    float scaledDensity = getResources().getDisplayMetrics().scaledDensity;
    return px/scaledDensity;
}

private float spToPixels(float sp) {
    float scaledDensity = getResources().getDisplayMetrics().scaledDensity;
    return sp * scaledDensity;
}

පරාමිතීන් ලෙස ලැබී ඇති MIN_SP සහ MAX_SP සීමාවන්ට ගරු කරමින් මෙම ප්‍රවේශය පෙළට ගැලපෙන පරිදි අකුරු ප්‍රමාණය වැඩි කිරීම හෝ අඩු කිරීම සිදු කරයි.


මට නම් මෙය හොඳම හා හොඳ පිළිතුරකි. මම අභිරුචි අකුරු භාවිතා කරමින් සිටියෙමි, එබැවින් මට තීන්ත එක් කළ යුතුය. SetTypeface (typeFace); මැජික් වගේ වැඩ කරනවා .. ස්තූතියි
අභිජිත්

7

මම මේ ගැන බ්ලොග් සටහනක් ලිව්වා.

නව ඇන්ඩ්‍රොයිඩ් වෙළඳපල යෙදුමේ භාවිතා කරන අභිරුචි සංරචක ගැන ResizableButtonකිරිල් ග්‍රූච්නිකොව්ගේ බ්ලොග් සටහන මත පදනම්ව මම සංරචකයක් නිර්මාණය කළෙමි . මම මෙහි src කේතය තැබුවෙමි .

අනෙක් අතට, මොසබුවා මගේ ලිපිය කියවා මට කීවේ ඔහු මට වඩා වේගයෙන් ක්‍රියාත්මක කිරීම විවෘත මූලාශ්‍රයක් වෙත ගෙන යන බවයි. මම හිතනවා ඔහු එය ඉක්මනින්ම නිදහස් කරයි කියලා :)


මෙය තනි පෙළ පේළියක් සඳහා ක්‍රියා කරයි- නමුත් පෙළ එතීම සඳහා ක්‍රියා නොකරයි. මට මග හැරුණු යමක් නොමැති නම්? මිනුම () සියලු පෙළ එකම රේඛාවක පවතිනු ඇතැයි යන උපකල්පනය මත පළල මැනීම පෙනේ.
නේතන් පය

ඔබගේ ක්‍රියාවට නැංවීම පෙරනිමියෙන් පෙළ කේන්ද්‍රගත නොවන බව මම දුටුවෙමි. එය වමට පාවෙයි. සිතුවිලි?
ඩොන් ෆෙල්කර්

පෙළ කේන්ද්‍රගත නොවීම සම්බන්ධයෙන් මට එකම ගැටළුවක් ඇති බව දුටුවෙමි, මම android: singleLine = "true" යන පේළිය එක් කළ අතර එය පසුව කේන්ද්‍රගත විය
clayton33

ඔබට මෙම කේත කොටස GitHub හි පළ කළ හැකිය, එය බෙහෙවින් ප්‍රයෝජනවත් වේ. මූලද්‍රව්‍යවල අකුරු ස්වයංක්‍රීයව ප්‍රතිප්‍රමාණනය කිරීමට ඔබට ඉඩ නොදීම ගැන මම ඇන්ඩ්‍රොයිඩ් පිරිසැලසුම් පද්ධතියට වෛර කරමි.
ලැචෙසර්

5

මා වෙනුවෙන් හොඳින් වැඩ කිරීමට පහත සඳහන් දෑ මට හමු විය. එය ලූපයක් නොවන අතර උස සහ පළල යන දෙකම සඳහා හේතු වේ. දර්ශනය මත setTextSize අමතන විට PX ඒකකය සඳහන් කිරීම වැදගත් බව සලකන්න.

Paint paint = adjustTextSize(getPaint(), numChars, maxWidth, maxHeight);
setTextSize(TypedValue.COMPLEX_UNIT_PX,paint.getTextSize());

මෙන්න මම භාවිතා කරන පුරුද්ද, දර්ශනයෙන් getPaint () පසුකරමින්. සත්‍ය නූලෙන් ස්වාධීනව පළල තක්සේරු කිරීමට 'පුළුල්' අක්ෂරයක් සහිත අක්ෂර 10 ක නූලක් භාවිතා කරයි.

private static final String text10="OOOOOOOOOO";
public static Paint adjustTextSize(Paint paint, int numCharacters, int widthPixels, int heightPixels) {
    float width = paint.measureText(text10)*numCharacters/text10.length();
    float newSize = (int)((widthPixels/width)*paint.getTextSize());
    paint.setTextSize(newSize);

    // remeasure with font size near our desired result
    width = paint.measureText(text10)*numCharacters/text10.length();
    newSize = (int)((widthPixels/width)*paint.getTextSize());
    paint.setTextSize(newSize);

    // Check height constraints
    FontMetricsInt metrics = paint.getFontMetricsInt();
    float textHeight = metrics.descent-metrics.ascent;
    if (textHeight > heightPixels) {
        newSize = (int)(newSize * (heightPixels/textHeight));
        paint.setTextSize(newSize);
    }

    return paint;
}

5

මගේ ක්‍රියාත්මක කිරීම ටිකක් සංකීර්ණයි, නමුත් පහත සඳහන් හොඳ දේ සමඟ පැමිණේ:

  • පවතින පළල සහ ලබා ගත හැකි උස සැලකිල්ලට ගනී
  • තනි රේඛා සහ බහු ලේබල් සමඟ ක්‍රියා කරයි
  • අවම අකුරු ප්‍රමාණයට පහර දුන් විට ඉලිප්සිස් භාවිතා කරයි
  • අභ්‍යන්තර පෙළ නිරූපණය වෙනස් වී ඇති හෙයින්, මුලින් සකසා ඇති පෙළ වෙනම විචල්‍යයකින් සිහිපත් කරයි
  • කැන්වසය සැමවිටම අවශ්‍ය ප්‍රමාණයට වඩා විශාල බව සහතික කරන අතර, එය දෙමව්පියන්ගේ පවතින සියලු උස භාවිතා කරයි
/**
 * Text view that auto adjusts text size to fit within the view. If the text
 * size equals the minimum text size and still does not fit, append with an
 * ellipsis.
 * 
 * Based on the original work from Chase Colburn
 * &lt;http://stackoverflow.com/a/5535672/305532>
 *
 * @author Thomas Keller &lt;me@thomaskeller.biz>
 */
public class AutoResizeTextView extends TextView {

    // in dip
    private static final int MIN_TEXT_SIZE = 20;

    private static final boolean SHRINK_TEXT_SIZE = true;

    private static final char ELLIPSIS = '\u2026';

    private static final float LINE_SPACING_MULTIPLIER_MULTILINE = 0.8f;

    private static final float LINE_SPACING_MULTIPLIER_SINGLELINE = 1f;

    private static final float LINE_SPACING_EXTRA = 0.0f;

    private CharSequence mOriginalText;

    // temporary upper bounds on the starting text size
    private float mMaxTextSize;

    // lower bounds for text size
    private float mMinTextSize;

    // determines whether we're currently in the process of measuring ourselves,
    // so we do not enter onMeasure recursively
    private boolean mInMeasure = false;

    // if the text size should be shrinked or if the text size should be kept
    // constant and only characters should be removed to hit the boundaries
    private boolean mShrinkTextSize;

    public AutoResizeTextView(Context context) {
        this(context, null);
        init(context, null);
    }

    public AutoResizeTextView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
        init(context, attrs);
    }

    public AutoResizeTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init(context, attrs);
    }

    private void init(Context context, AttributeSet attrs) {
        // the current text size is used as maximum text size we can apply to
        // our widget
        mMaxTextSize = getTextSize();
        if (attrs != null) {
            TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.AutoResizeTextView);
            mMinTextSize = a.getFloat(R.styleable.AutoResizeTextView_minFontSize, MIN_TEXT_SIZE);
            mShrinkTextSize = a.getBoolean(R.styleable.AutoResizeTextView_shrinkTextSize, SHRINK_TEXT_SIZE);
            a.recycle();
        }
    }

    @Override
    public void setTextSize(float size) {
        mMaxTextSize = size;
        super.setTextSize(size);
    }

    /**
     * Returns the original, unmodified text of this widget
     * 
     * @return
     */
    public CharSequence getOriginalText() {
        // text has not been resized yet
        if (mOriginalText == null) {
            return getText();
        }
        return mOriginalText;
    }

    @Override
    public void setText(CharSequence text, BufferType type) {
        if (!mInMeasure) {
            mOriginalText = text.toString();
        }
        super.setText(text, type);
    }

    @SuppressLint("DrawAllocation")
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        mInMeasure = true;
        try {
            int availableWidth = MeasureSpec.getSize(widthMeasureSpec) - getCompoundPaddingLeft()
                    - getCompoundPaddingRight();
            int availableHeight = MeasureSpec.getSize(heightMeasureSpec) - getCompoundPaddingTop()
                    - getCompoundPaddingBottom();

            // Do not resize if the view does not have dimensions or there is no
            // text
            if (mOriginalText == null || mOriginalText.length() == 0 || availableWidth <= 0) {
                return;
            }

            TextPaint textPaint = getPaint();

            // start with the recorded max text size
            float targetTextSize = mMaxTextSize;
            String originalText = mOriginalText.toString();
            String finalText = originalText;

            Rect textSize = getTextSize(originalText, textPaint, targetTextSize);
            boolean textExceedsBounds = textSize.height() > availableHeight || textSize.width() > availableWidth;
            if (mShrinkTextSize && textExceedsBounds) {
                // check whether all lines can be rendered in the available
                // width / height without violating the bounds of the parent and
                // without using a text size that is smaller than the minimum
                // text size
                float heightMultiplier = availableHeight / (float) textSize.height();
                float widthMultiplier = availableWidth / (float) textSize.width();
                float multiplier = Math.min(heightMultiplier, widthMultiplier);
                targetTextSize = Math.max(targetTextSize * multiplier, mMinTextSize);

                // measure again
                textSize = getTextSize(finalText, textPaint, targetTextSize);
            }

            // we cannot shrink the height further when we hit the available
            // height, but we can shrink the width by applying an ellipsis on
            // each line
            if (textSize.width() > availableWidth) {
                StringBuilder modifiedText = new StringBuilder();
                String lines[] = originalText.split(System.getProperty("line.separator"));
                for (int i = 0; i < lines.length; i++) {
                    modifiedText.append(resizeLine(textPaint, lines[i], availableWidth));
                    // add the separator back to all but the last processed line
                    if (i != lines.length - 1) {
                        modifiedText.append(System.getProperty("line.separator"));
                    }
                }
                finalText = modifiedText.toString();

                // measure again
                textSize = getTextSize(finalText, textPaint, targetTextSize);
            }

            textPaint.setTextSize(targetTextSize);
            boolean isMultiline = finalText.indexOf('\n') > -1;
            // do not include extra font padding (for accents, ...) for
            // multiline texts, this will prevent proper placement with
            // Gravity.CENTER_VERTICAL
            if (isMultiline) {
                setLineSpacing(LINE_SPACING_EXTRA, LINE_SPACING_MULTIPLIER_MULTILINE);
                setIncludeFontPadding(false);
            } else {
                setLineSpacing(LINE_SPACING_EXTRA, LINE_SPACING_MULTIPLIER_SINGLELINE);
                setIncludeFontPadding(true);
            }

            // according to
            // <http://code.google.com/p/android/issues/detail?id=22493>
            // we have to add a unicode character to trigger the text centering
            // in ICS. this particular character is known as "zero-width" and
            // does no harm.
            setText(finalText + "\u200B");

            int measuredWidth = textSize.width() + getCompoundPaddingLeft() + getCompoundPaddingRight();
            int measuredHeight = textSize.height() + getCompoundPaddingTop() + getCompoundPaddingBottom();

            // expand the view to the parent's height in case it is smaller or
            // to the minimum height that has been set
            // FIXME: honor the vertical measure mode (EXACTLY vs AT_MOST) here
            // somehow
            measuredHeight = Math.max(measuredHeight, MeasureSpec.getSize(heightMeasureSpec));
            setMeasuredDimension(measuredWidth, measuredHeight);
        } finally {
            mInMeasure = false;
        }
    }

    private Rect getTextSize(String text, TextPaint textPaint, float textSize) {
        textPaint.setTextSize(textSize);
        // StaticLayout depends on a given width in which it should lay out the
        // text (and optionally also split into separate lines).
        // Therefor we calculate the current text width manually and start with
        // a fake (read: maxmimum) width for the height calculation.
        // We do _not_ use layout.getLineWidth() here since this returns
        // slightly smaller numbers and therefor would lead to exceeded text box
        // drawing.
        StaticLayout layout = new StaticLayout(text, textPaint, Integer.MAX_VALUE, Alignment.ALIGN_NORMAL, 1f, 0f, true);
        int textWidth = 0;
        String lines[] = text.split(System.getProperty("line.separator"));
        for (int i = 0; i < lines.length; ++i) {
            textWidth = Math.max(textWidth, measureTextWidth(textPaint, lines[i]));
        }
        return new Rect(0, 0, textWidth, layout.getHeight());
    }

    private String resizeLine(TextPaint textPaint, String line, int availableWidth) {
        checkArgument(line != null && line.length() > 0, "expected non-empty string");
        int textWidth = measureTextWidth(textPaint, line);
        int lastDeletePos = -1;
        StringBuilder builder = new StringBuilder(line);
        while (textWidth > availableWidth && builder.length() > 0) {
            lastDeletePos = builder.length() / 2;
            builder.deleteCharAt(builder.length() / 2);
            // don't forget to measure the ellipsis character as well; it
            // doesn't matter where it is located in the line, it just has to be
            // there, since there are no (known) ligatures that use this glyph
            String textToMeasure = builder.toString() + ELLIPSIS;
            textWidth = measureTextWidth(textPaint, textToMeasure);
        }
        if (lastDeletePos > -1) {
            builder.insert(lastDeletePos, ELLIPSIS);
        }
        return builder.toString();
    }

    // there are several methods in Android to determine the text width, namely
    // getBounds() and measureText().
    // The latter works for us the best as it gives us the best / nearest
    // results without that our text canvas needs to wrap its text later on
    // again.
    private int measureTextWidth(TextPaint textPaint, String line) {
        return Math.round(textPaint.measureText(line));
    }
}

[2012-11-21 දින සංශෝධනය කරන ලදි]

  • ඉලිප්සිස් ස්ථානගත කිරීම සවි කර ඇත (එකින් එක දෝෂයකි)
  • නැවත සකස් කරන ලද පෙළ ප්‍රමාණ ගණනය කිරීම; තනි මිනුම් රේඛා දෙකක උස එකතු කිරීම සමස්තයක් ලෙස පා of යේ උස මැනීමේ ප්‍රති result ලයට තුඩු නොදුන් විට ගැටලු නිරාකරණය කිරීම සඳහා දැන් සෑම විටම රේඛා බිඳීම් ඇතුළු සම්පූර්ණ පෙළ මනිනු ලැබේ.
  • ලබා ගත හැකි කුඩාම පෙළ ප්‍රමාණය සොයා ගැනීම සඳහා ලූප වෙනුවට, පළමු මිනුමෙන් පසුව එය ගණනය කරන්න

1
StyleableTextView යනු කුමක්ද?
තුෂාර්

අභ්‍යන්තර පංතියක්, සමාවෙන්න, මම මෙය මෙතැනින් ඉවත් කරමි.
තෝමස් කෙලර්

2
දෝෂ වලින් පිරී ඇති අතර තවමත් StylableTextView
LairdPleng

3

තවමත් සොයන ඕනෑම කෙනෙකුට මා සොයාගත් වෙනත් දේ පිළිබඳ ගණනය කිරීමක් මෙන්න:

1) මෙන්න විසඳුමක් පෙළ දර්ශනය ගැලපෙන තෙක් නැවත නැවත පින්තාරු . මෙයින් අදහස් කරන්නේ ඔබේ පා text ය තැනට හැකිලී යාමයි, නමුත් එය සිදු වූ විට අවම වශයෙන් එය ගැලපේ. කේතය ක්‍රියාත්මක කිරීම සඳහා යම්කිසි වෙනස් කිරීමක් අවශ්‍ය වනු ඇත, නමුත් එය බොහෝ දුරට පවතී.

2) ඔබ වැනි අභිරුචි විසඳුමක් එකට අනවසරයෙන් ඇතුළුවී උත්සාහ කළ හැකි මෙම , හෝ dunni ගේ පන්ති මේ මම නියම ප්රමාණය සෙවීමට මෙම getPaint (). MeasureText (STR) භාවිතා කළ දේ වන, නමුත් එය මට අවශ්ය සිට ගොඩක් messier වී එය හිස් අවකාශයේ පමණක් ඔතා ...

3) ඔබට දිගටම සෙවිය හැකිය- මට ගණන් කළ හැකි ප්‍රමාණයට වඩා විකල්ප මම උත්සාහ කර ඇත්තෙමි. ස්ථිතික සැකැස්ම පිළිබඳ ටෙඩ්ගේ උපදෙස් මා වෙනුවෙන් ගෙවා නැති නමුත් සමහර විට එහි යමක් තිබේ; පෙළ තිරයෙන් ඉවතට යන්නේද යන්න තීරණය කිරීම සඳහා මම StaticLayout.getEllipsis (line) භාවිතා කිරීමට උත්සාහ කළෙමි. ඒ පිළිබඳ මගේ (දැනට පිළිතුරු නොලැබූ) ලිපිය මෙතැනින් බලන්න .


2

මට නිශ්චිත විසඳුමක් අවශ්‍ය විය. මගේ පිරිසැලසුමෙහි සංස්කරණ පා text යක් සහ පෙළ දර්ශනයක් මා සතුව ඇත. පෙළ දර්ශනය ස්ථාවර උස සහ පළල වේ. පරිශීලකයා සංස්කරණ පෙළ ටයිප් කිරීමට පටන් ගත් විට, පෙළ වහාම පෙළ දර්ශනයේ දිස්විය යුතුය. පෙළ ක්‍ෂේත්‍රයේ ඇති පෙළ ස්වයංක්‍රීයව ප්‍රතිනිර්මාණය කළ යුතුය. ඒ නිසා මම වැඩ කිරීමට චේස්ගේ විසඳුම යාවත්කාලීන කළා. එබැවින් පෙළ දර්ශනයේ පෙළ වෙනස් වූ විට, ප්‍රමාණය වෙනස් කිරීම ආරම්භ වේ. මගේ සහ චේස්ගේ ද්‍රාවණය අතර වෙනස: පරිශීලකයා සමහර අක්ෂර මකා දැමුවද, ප්‍රමාණය වෙනස් කිරීම සිදු කරයි. එය යමෙකුට උපකාර කළ හැකි යැයි මම බලාපොරොත්තු වෙමි.

public class TextFitTextView extends TextView {

// Minimum text size for this text view
public static final float MIN_TEXT_SIZE = 10;

// Maximum text size for this text view - if it is 0, then the text acts
// like match_parent
public static final float MAX_TEXT_SIZE = 0;

// Our ellipse string
private static final String mEllipsis = "...";

// Text size that is set from code. This acts as a starting point for
// resizing
private float mTextSize;

// Lower bounds for text size
private float mMinTextSize = MIN_TEXT_SIZE;

// Max bounds for text size
private float mMaxTextSize = MAX_TEXT_SIZE;

// Text view line spacing multiplier
private float mSpacingMult = 1.0f;

// Text view additional line spacing
private float mSpacingAdd = 0.0f;

// Add ellipsis to text that overflows at the smallest text size
private boolean mAddEllipsis = true;

// Add ellipsis to text that overflows at the smallest text size
private int heightLimit;
private int widthLimit;

// Default constructor override
public TextFitTextView(Context context) {
    this(context, null);
}

// Default constructor when inflating from XML file
public TextFitTextView(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
}

// Default constructor override
public TextFitTextView(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    mTextSize = getTextSize();
}

/**
 * When text changes resize the text size.
 */
@Override
protected void onTextChanged(final CharSequence text, final int start, final int before, final int after) {
    // if we are adding new chars to text
    if (before <= after && after != 1) {
        resizeText(true);
        // now we are deleting chars
    } else {
        resizeText(false);
    }
}

/**
 * Override the set text size to update our internal reference values
 */
@Override
public void setTextSize(float size) {
    super.setTextSize(size);
    mTextSize = getTextSize();
}

/**
 * Override the set text size to update our internal reference values
 */
@Override
public void setTextSize(int unit, float size) {
    super.setTextSize(unit, size);
    mTextSize = getTextSize();
}

/**
 * Override the set line spacing to update our internal reference values
 */
@Override
public void setLineSpacing(float add, float mult) {
    super.setLineSpacing(add, mult);
    mSpacingMult = mult;
    mSpacingAdd = add;
}

/**
 * Set the lower text size limit and invalidate the view
 * 
 * @param minTextSize
 */
public void setMinTextSize(float minTextSize) {
    mMinTextSize = minTextSize;
    requestLayout();
    invalidate();
}

/**
 * Return lower text size limit
 * 
 * @return
 */
public float getMinTextSize() {
    return mMinTextSize;
}

/**
 * Set flag to add ellipsis to text that overflows at the smallest text size
 * 
 * @param addEllipsis
 */
public void setAddEllipsis(boolean addEllipsis) {
    mAddEllipsis = addEllipsis;
}

/**
 * Return flag to add ellipsis to text that overflows at the smallest text
 * size
 * 
 * @return
 */
public boolean getAddEllipsis() {
    return mAddEllipsis;
}

/**
 * Get width and height limits
 */
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
    if (widthLimit == 0 && heightLimit == 0) {
        widthLimit = (right - left) - getCompoundPaddingLeft() - getCompoundPaddingRight();
        heightLimit = (bottom - top) - getCompoundPaddingBottom() - getCompoundPaddingTop();
    }
    super.onLayout(changed, left, top, right, bottom);
}

/**
 * Resize the text size with specified width and height
 * 
 * @param width
 * @param height
 */
public void resizeText(boolean increase) {
    CharSequence text = getText();
    // Do not resize if the view does not have dimensions or there is no
    // text
    if (text == null || text.length() == 0 || heightLimit <= 0 || widthLimit <= 0 || mTextSize == 0) {
        return;
    }

    // Get the text view's paint object
    TextPaint textPaint = getPaint();

    // Get the required text height
    int textHeight = getTextHeight(text, textPaint, widthLimit, mTextSize);


    // If the text length is increased 
    // Until we either fit within our text view or we had reached our min
    // text size, incrementally try smaller sizes
    if (increase) {
        while (textHeight > heightLimit && mTextSize > mMinTextSize) {
            mTextSize = Math.max(mTextSize - 2, mMinTextSize);
            textHeight = getTextHeight(text, textPaint, widthLimit, mTextSize);
        }
    } 
//      text length has been decreased
    else {
//          if max test size is set then add it to while condition
        if (mMaxTextSize != 0) {
            while (textHeight < heightLimit && mTextSize <= mMaxTextSize) {
                mTextSize = mTextSize + 2;
                textHeight = getTextHeight(text, textPaint, widthLimit, mTextSize);
            }
        } else {
            while (textHeight < heightLimit) {
                mTextSize = mTextSize + 2;
                textHeight = getTextHeight(text, textPaint, widthLimit, mTextSize);
            }
        }
        mTextSize = textHeight > heightLimit ? mTextSize - 2 : mTextSize;
    }

    // If we had reached our minimum text size and still don't fit, append
    // an ellipsis
    if (mAddEllipsis && mTextSize == mMinTextSize && textHeight > heightLimit) {
        // Draw using a static layout
        TextPaint paint = new TextPaint(textPaint);
        StaticLayout layout = new StaticLayout(text, paint, widthLimit, Alignment.ALIGN_NORMAL, mSpacingMult,
                mSpacingAdd, false);
        // Check that we have a least one line of rendered text
        if (layout.getLineCount() > 0) {
            // Since the line at the specific vertical position would be cut
            // off,
            // we must trim up to the previous line
            int lastLine = layout.getLineForVertical(heightLimit) - 1;
            // If the text would not even fit on a single line, clear it
            if (lastLine < 0) {
                setText("");
            }
            // Otherwise, trim to the previous line and add an ellipsis
            else {
                int start = layout.getLineStart(lastLine);
                int end = layout.getLineEnd(lastLine);
                float lineWidth = layout.getLineWidth(lastLine);
                float ellipseWidth = paint.measureText(mEllipsis);

                // Trim characters off until we have enough room to draw the
                // ellipsis
                while (widthLimit < lineWidth + ellipseWidth) {
                    lineWidth = paint.measureText(text.subSequence(start, --end + 1).toString());
                }
                setText(text.subSequence(0, end) + mEllipsis);
            }
        }
    }

    // Some devices try to auto adjust line spacing, so force default line
    // spacing
    // and invalidate the layout as a side effect
    setTextSize(TypedValue.COMPLEX_UNIT_PX, mTextSize);
    setLineSpacing(mSpacingAdd, mSpacingMult);

}

// Set the text size of the text paint object and use a static layout to
// render text off screen before measuring
private int getTextHeight(CharSequence source, TextPaint originalPaint, int width, float textSize) {
    // Update the text paint object
    TextPaint paint = new TextPaint(originalPaint);
    paint.setTextSize(textSize);
    // Measure using a static layout
    StaticLayout layout = new StaticLayout(source, paint, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd,
            true);
    return layout.getHeight();
}

}

2

Xamarin.Android හි කේත කරන අය සඳහා C # හි නැවත ලියා ඇති ඉහළ පිළිතුරේ මෙම අනුවාදය සැපයීම . මා වෙනුවෙන් හොඳින් වැඩ කළා.

 /**
 *               DO WHAT YOU WANT TO PUBLIC LICENSE
 *                    Version 2, December 2004
 * 
 * Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
 * 
 * Everyone is permitted to copy and distribute verbatim or modified
 * copies of this license document, and changing it is allowed as long
 * as the name is changed.
 * 
 *            DO WHAT YOU WANT TO PUBLIC LICENSE
 *   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
 * 
 *  0. You just DO WHAT YOU WANT TO.
 */


using System;
using Android.Content;
using Android.Runtime;
using Android.Text;
using Android.Util;
using Android.Widget;
using Java.Lang;

namespace App.GuestGuide.Droid.Controls
{
    public class OnTextResizeEventArgs : EventArgs
    {
        public TextView TextView { get; set; }
        public float OldSize { get; set; }
        public float NewSize { get; set; }
    }

    /// <inheritdoc />
    /// <summary>
    /// Text view that auto adjusts text size to fit within the view.
    /// If the text size equals the minimum text size and still does not
    /// fit, append with an ellipsis.
    /// </summary>
    public class AutoResizeTextView : TextView
    {
        /// <summary>
        /// Minimum text size for this text view
        /// </summary>
        public static float MIN_TEXT_SIZE = 10;

        /// <summary>
        /// Our ellipse string
        /// </summary>
        private const string Ellipsis = "...";


        private float _mMaxTextSize;

        private float _mMinTextSize = MIN_TEXT_SIZE;

        /// <summary>
        /// Register subscriber to receive resize notifications
        /// </summary>
        public event EventHandler<OnTextResizeEventArgs> OnTextResize;

        /// <summary>
        /// Flag for text and/or size changes to force a resize
        /// </summary>
        private bool _needsResize;

        /// <summary>
        /// Text size that is set from code. This acts as a starting point for resizing
        /// </summary>
        private float _textSize;

        /// <summary>
        /// Text view line spacing multiplier
        /// </summary>
        private float _spacingMult = 1.0f;

        /// <summary>
        /// Text view additional line spacing
        /// </summary>
        private float _spacingAdd;

        /// <summary>
        /// Add ellipsis to text that overflows at the smallest text size
        /// </summary>
        public bool ShouldAddEllipsis { get; set; }

        /// <inheritdoc />
        /// <summary>
        /// Override the set text size to update our internal reference values
        /// </summary>
        public override float TextSize
        {
            get => base.TextSize;
            set
            {
                base.TextSize = value;
                _textSize = TextSize;
            }
        }

        /// <summary>
        /// Temporary upper bounds on the starting text size
        /// </summary>
        public float MaxTextSize
        {
            get => _mMaxTextSize;
            // Set the upper text size limit and invalidate the view
            set
            {
                _mMaxTextSize = value;
                RequestLayout();
                Invalidate();
            }
        }

        /// <summary>
        /// Lower bounds for text size
        /// </summary>
        public float MinTextSize
        {
            get => _mMinTextSize;
            //Set the lower text size limit and invalidate the view
            set
            {
                _mMinTextSize = value;
                RequestLayout();
                Invalidate();
            }
        }

        public AutoResizeTextView(Context context) : this(context, null)
        {
        }

        public AutoResizeTextView(Context context, IAttributeSet attrs) : this(context, attrs, 0)
        {
        }

        public AutoResizeTextView(Context context, IAttributeSet attrs, int defStyleAttr) : base(context, attrs, defStyleAttr)
        {
            _textSize = TextSize;
        }

        public AutoResizeTextView(Context context, IAttributeSet attrs, int defStyleAttr, int defStyleRes) : base(context, attrs, defStyleAttr, defStyleRes)
        {
            _textSize = TextSize;
        }

        protected AutoResizeTextView(IntPtr javaReference, JniHandleOwnership transfer) : base(javaReference, transfer)
        {
            _textSize = TextSize;
        }

        /// <inheritdoc />
        /// <summary>
        /// When text changes, set the force resize flag to true and reset the text size.
        /// </summary>
        /// <param name="text"></param>
        /// <param name="start"></param>
        /// <param name="lengthBefore"></param>
        /// <param name="lengthAfter"></param>
        protected override void OnTextChanged(ICharSequence text, int start, int lengthBefore, int lengthAfter)
        {
            _needsResize = true;
            // Since this view may be reused, it is good to reset the text size
            ResetTextSize();
        }

        /// <inheritdoc />
        /// <summary>
        /// If the text view size changed, set the force resize flag to true
        /// </summary>
        /// <param name="w"></param>
        /// <param name="h"></param>
        /// <param name="oldw"></param>
        /// <param name="oldh"></param>
        protected override void OnSizeChanged(int w, int h, int oldw, int oldh)
        {
            if (w != oldw || h != oldh)
            {
                _needsResize = true;
            }
        }

        public override void SetTextSize([GeneratedEnum] ComplexUnitType unit, float size)
        {
            base.SetTextSize(unit, size);
            _textSize = TextSize;
        }

        /// <inheritdoc />
        /// <summary>
        /// Override the set line spacing to update our internal reference values
        /// </summary>
        /// <param name="add"></param>
        /// <param name="mult"></param>
        public override void SetLineSpacing(float add, float mult)
        {
            base.SetLineSpacing(add, mult);
            _spacingMult = mult;
            _spacingAdd = add;
        }

        /// <summary>
        /// Reset the text to the original size
        /// </summary>
        public void ResetTextSize()
        {
            if (_textSize > 0)
            {
                base.SetTextSize(ComplexUnitType.Px, _textSize);
                _mMaxTextSize = _textSize;
            }
        }

        /// <inheritdoc />
        /// <summary>
        /// Resize text after measuring
        /// </summary>
        /// <param name="changed"></param>
        /// <param name="left"></param>
        /// <param name="top"></param>
        /// <param name="right"></param>
        /// <param name="bottom"></param>
        protected override void OnLayout(bool changed, int left, int top, int right, int bottom)
        {
            if (changed || _needsResize)
            {
                var widthLimit = (right - left) - CompoundPaddingLeft - CompoundPaddingRight;
                var heightLimit = (bottom - top) - CompoundPaddingBottom - CompoundPaddingTop;
                ResizeText(widthLimit, heightLimit);
            }

            base.OnLayout(changed, left, top, right, bottom);
        }

        /// <summary>
        /// Resize the text size with default width and height
        /// </summary>
        public void ResizeText()
        {
            var heightLimit = Height - PaddingBottom - PaddingTop;
            var widthLimit = Width - PaddingLeft - PaddingRight;
            ResizeText(widthLimit, heightLimit);
        }

        /// <summary>
        /// Resize the text size with specified width and height
        /// </summary>
        /// <param name="width"></param>
        /// <param name="height"></param>
        public void ResizeText(int width, int height)
        {
            ICharSequence text = null;

            if (!string.IsNullOrEmpty(Text))
            {
                text = new Java.Lang.String(Text);
            }

            // Do not resize if the view does not have dimensions or there is no text
            if (text == null || text.Length() == 0 || height <= 0 || width <= 0 || _textSize == 0)
            {
                return;
            }

            if (TransformationMethod != null)
            {
                text = TransformationMethod.GetTransformationFormatted(text, this);
            }

            // Get the text view's paint object
            var textPaint = Paint;
            // Store the current text size
            var oldTextSize = textPaint.TextSize;
            // If there is a max text size set, use the lesser of that and the default text size
            var targetTextSize = _mMaxTextSize > 0 ? System.Math.Min(_textSize, _mMaxTextSize) : _textSize;

            // Get the required text height
            var textHeight = GetTextHeight(text, textPaint, width, targetTextSize);

            // Until we either fit within our text view or we had reached our min text size, incrementally try smaller sizes
            while (textHeight > height && targetTextSize > _mMinTextSize)
            {
                targetTextSize = System.Math.Max(targetTextSize - 2, _mMinTextSize);
                textHeight = GetTextHeight(text, textPaint, width, targetTextSize);
            }

            // If we had reached our minimum text size and still don't fit, append an ellipsis
            if (ShouldAddEllipsis && targetTextSize == _mMinTextSize && textHeight > height)
            {
                // Draw using a static layout
                // modified: use a copy of TextPaint for measuring
                var paint = new TextPaint(textPaint);
                // Draw using a static layout
                var layout = new StaticLayout(text, paint, width, Layout.Alignment.AlignNormal, _spacingMult, _spacingAdd, false);

                // Check that we have a least one line of rendered text
                if (layout.LineCount > 0)
                {
                    // Since the line at the specific vertical position would be cut off,
                    // we must trim up to the previous line
                    var lastLine = layout.GetLineForVertical(height) - 1;
                    // If the text would not even fit on a single line, clear it
                    if (lastLine < 0)
                    {
                        Text = string.Empty;
                    }
                    // Otherwise, trim to the previous line and add an ellipsis
                    else
                    {
                        var start = layout.GetLineStart(lastLine);
                        var end = layout.GetLineEnd(lastLine);
                        var lineWidth = layout.GetLineWidth(lastLine);
                        var ellipseWidth = textPaint.MeasureText(Ellipsis);

                        // Trim characters off until we have enough room to draw the ellipsis
                        while (width < lineWidth + ellipseWidth)
                        {
                            lineWidth = textPaint.MeasureText(text.SubSequence(start, --end + 1));
                        }

                        Text = text.SubSequence(0, end) + Ellipsis;
                    }
                }
            }

            // Some devices try to auto adjust line spacing, so force default line spacing
            // and invalidate the layout as a side effect
            SetTextSize(ComplexUnitType.Px, targetTextSize);
            SetLineSpacing(_spacingAdd, _spacingMult);

            var notifyArgs = new OnTextResizeEventArgs
            {
                TextView = this,
                NewSize = targetTextSize,
                OldSize = oldTextSize
            };

            // Notify the listener if registered
            OnTextResize?.Invoke(this, notifyArgs);

            // Reset force resize flag
            _needsResize = false;
        }

        /// <summary>
        /// Set the text size of the text paint object and use a static layout to render text off screen before measuring
        /// </summary>
        /// <param name="source"></param>
        /// <param name="paint"></param>
        /// <param name="width"></param>
        /// <param name="textSize"></param>
        /// <returns></returns>
        private int GetTextHeight(ICharSequence source, TextPaint paint, int width, float textSize)
        {
            // modified: make a copy of the original TextPaint object for measuring
            // (apparently the object gets modified while measuring, see also the
            // docs for TextView.getPaint() (which states to access it read-only)
            // Update the text paint object
            var paintCopy = new TextPaint(paint)
            {
                TextSize = textSize
            };

            // Measure using a static layout
            var layout = new StaticLayout(source, paintCopy, width, Layout.Alignment.AlignNormal, _spacingMult, _spacingAdd, true);

            return layout.Height;
        }
    }
}

මම Xamarin වෙත නවකයෙක් වන අතර VS17 හි ව්‍යාපෘතියක් නිර්මාණය කළ නොහැක. ඔයාට මට මෙතනින් උදව් කරන්න පුළුවන්ද? Thanx
ෂම්බු

@ ශම්බු, කරුණාකර මෙහි පරීක්ෂා කරන්න youtube.com/watch?v=NGvn-pGZFPA
Oleg Kosuakiv

Le ඔලෙග් කොසුකිව් ඔබට මෙය භාවිතා කරමින් සංකල්ප යෙදුමක් තිබේද? එය අත්හදා බැලීමට කුතුහලයෙන්!
envyM6

vy envyM6, ඔබ අදහස් කරන්නේ පෙළ දර්ශනය ස්වයංක්‍රීය කිරීම හෝ Xamarin Android ද?
ඔලෙග් කොසුකිව්

ඔව් සර් එය සැබවින්ම Xamarin Android
envyM6

1

මේ සඳහා ඔබට android.text.StaticLayoutපන්තිය භාවිතා කළ හැකිය . එය TextViewඅභ්‍යන්තරව භාවිතා කරයි.


4
ඔබට විස්තර කළ හැකිද? මම ස්ථිතික සැකැස්ම මත dev ලේඛනය දකිමි, එයට සිත්ගන්නාසුලු ක්‍රම කිහිපයක් ඇත, නමුත් මා ඉල්ලන දේ නිර්මාණය කිරීමට මෙය භාවිතා කරන්නේ කෙසේදැයි මට විශ්වාස නැත?
නේතන් රූපය

1

මම පහත දැක්වෙන ක්‍රමය නිර්මාණය කළෙමි (චේස්ගේ අදහස් මත පදනම්ව) ඔබට ඕනෑම කැන්වසයකට පෙළ ඇඳීමට අවශ්‍ය නම් එය ඔබට උපකාරී වනු ඇත:

private static void drawText(Canvas canvas, int xStart, int yStart,
        int xWidth, int yHeigth, String textToDisplay,
        TextPaint paintToUse, float startTextSizeInPixels,
        float stepSizeForTextSizeSteps) {

    // Text view line spacing multiplier
    float mSpacingMult = 1.0f;
    // Text view additional line spacing
    float mSpacingAdd = 0.0f;
    StaticLayout l = null;
    do {
        paintToUse.setTextSize(startTextSizeInPixels);
        startTextSizeInPixels -= stepSizeForTextSizeSteps;
        l = new StaticLayout(textToDisplay, paintToUse, xWidth,
                Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
    } while (l.getHeight() > yHeigth);

    int textCenterX = xStart + (xWidth / 2);
    int textCenterY = (yHeigth - l.getHeight()) / 2;

    canvas.save();
    canvas.translate(textCenterX, textCenterY);
    l.draw(canvas);
    canvas.restore();
}

මෙය ඕනෑම අභිරුචි දර්ශනයක ඕනෑම onDraw () ක්‍රමයකින් භාවිතා කළ හැකිය.


චිත්‍ර ඇඳීමේ ක්‍රියාවලියේදී වස්තු නිර්මාණය කිරීම නරක පුරුද්දකි
ඔෆෙක් රොන්

1

මෙන්න තවත් විසඳුමක්, පයින් ගැසීම සඳහා. එය බොහෝ විට ඉතා කාර්යක්ෂම නොවේ, නමුත් එය පෙළෙහි උස හා පළල යන දෙකටම හා සලකුණු කළ පෙළ සමඟ කටයුතු කරයි.

@Override
protected void onMeasure(final int widthMeasureSpec, final int heightMeasureSpec)
{
    if ((MeasureSpec.getMode(widthMeasureSpec) != MeasureSpec.UNSPECIFIED)
            && (MeasureSpec.getMode(heightMeasureSpec) != MeasureSpec.UNSPECIFIED)) {

        final float desiredWidth = MeasureSpec.getSize(widthMeasureSpec);
        final float desiredHeight = MeasureSpec.getSize(heightMeasureSpec);

        float textSize = getTextSize();
        float lastScale = Float.NEGATIVE_INFINITY;
        while (textSize > MINIMUM_AUTO_TEXT_SIZE_PX) {
            // Measure how big the textview would like to be with the current text size.
            super.onMeasure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);

            // Calculate how much we'd need to scale it to fit the desired size, and
            // apply that scaling to the text size as an estimate of what we need.
            final float widthScale = desiredWidth / getMeasuredWidth();
            final float heightScale = desiredHeight / getMeasuredHeight();
            final float scale = Math.min(widthScale, heightScale);

            // If we don't need to shrink the text, or we don't seem to be converging, we're done.
            if ((scale >= 1f) || (scale <= lastScale)) {
                break;
            }

            // Shrink the text size and keep trying.
            textSize = Math.max((float) Math.floor(scale * textSize), MINIMUM_AUTO_TEXT_SIZE_PX);
            setTextSize(TypedValue.COMPLEX_UNIT_PX, textSize);
            lastScale = scale;
        }
    }
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}

මම තවත් පිළිතුරු 6 ක් උත්සාහ කර ඇත, නමුත් මෙය මට හොඳම ප්‍රති .ල ලබා දෙයි. තවමත් එය පරිපූර්ණ නොවේ, මන්ද පෙළ ප්‍රමාණය සමහර අවස්ථාවල තිබිය යුතු ප්‍රමාණයට වඩා තරමක් කුඩා වන නමුත් අනෙක් පිළිතුරු වලට වඩා හොඳය.
තෝමස් ඩබ්ලිව්

1

මම ඉහත යෝජනා සමහරක් ඒකාබද්ධ කර ඉහළට හා පහළට තරාදි කර ද්විමාන ක්‍රමයක් භාවිතා කළෙමි. එය පළල තුළ ද පරිමාණය කරයි.

/**
 *               DO WHAT YOU WANT TO PUBLIC LICENSE
 *                    Version 2, December 2004
 *
 * Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
 *
 * Everyone is permitted to copy and distribute verbatim or modified
 * copies of this license document, and changing it is allowed as long
 * as the name is changed.
 *
 *            DO WHAT YOU WANT TO PUBLIC LICENSE
 *   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
 *
 *  0. You just DO WHAT YOU WANT TO.
 */

import android.content.Context;
import android.text.Layout.Alignment;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.widget.TextView;

/**
 * Text view that auto adjusts text size to fit within the view. If the text
 * size equals the minimum text size and still does not fit, append with an
 * ellipsis.
 * 
 * @author Chase Colburn
 * @since Apr 4, 2011
 */
public class AutoResizeTextView extends TextView {

    // Minimum text size for this text view
    public static final float MIN_TEXT_SIZE = 10;

    // Minimum text size for this text view
    public static final float MAX_TEXT_SIZE = 128;

    private static final int BISECTION_LOOP_WATCH_DOG = 30;

    // Interface for resize notifications
    public interface OnTextResizeListener {
        public void onTextResize(TextView textView, float oldSize, float newSize);
    }

    // Our ellipse string
    private static final String mEllipsis = "...";

    // Registered resize listener
    private OnTextResizeListener mTextResizeListener;

    // Flag for text and/or size changes to force a resize
    private boolean mNeedsResize = false;

    // Text size that is set from code. This acts as a starting point for
    // resizing
    private float mTextSize;

    // Temporary upper bounds on the starting text size
    private float mMaxTextSize = MAX_TEXT_SIZE;

    // Lower bounds for text size
    private float mMinTextSize = MIN_TEXT_SIZE;

    // Text view line spacing multiplier
    private float mSpacingMult = 1.0f;

    // Text view additional line spacing
    private float mSpacingAdd = 0.0f;

    // Add ellipsis to text that overflows at the smallest text size
    private boolean mAddEllipsis = true;

    // Default constructor override
    public AutoResizeTextView(Context context) {
        this(context, null);
    }

    // Default constructor when inflating from XML file
    public AutoResizeTextView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    // Default constructor override
    public AutoResizeTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mTextSize = getTextSize();
    }

    /**
     * When text changes, set the force resize flag to true and reset the text
     * size.
     */
    @Override
    protected void onTextChanged(final CharSequence text, final int start,
            final int before, final int after) {
        mNeedsResize = true;
        // Since this view may be reused, it is good to reset the text size
        resetTextSize();
    }

    /**
     * If the text view size changed, set the force resize flag to true
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        if (w != oldw || h != oldh) {
            mNeedsResize = true;
        }
    }

    /**
     * Register listener to receive resize notifications
     * 
     * @param listener
     */
    public void setOnResizeListener(OnTextResizeListener listener) {
        mTextResizeListener = listener;
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(float size) {
        super.setTextSize(size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(int unit, float size) {
        super.setTextSize(unit, size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set line spacing to update our internal reference values
     */
    @Override
    public void setLineSpacing(float add, float mult) {
        super.setLineSpacing(add, mult);
        mSpacingMult = mult;
        mSpacingAdd = add;
    }

    /**
     * Set the upper text size limit and invalidate the view
     * 
     * @param maxTextSize
     */
    public void setMaxTextSize(float maxTextSize) {
        mMaxTextSize = maxTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return upper text size limit
     * 
     * @return
     */
    public float getMaxTextSize() {
        return mMaxTextSize;
    }

    /**
     * Set the lower text size limit and invalidate the view
     * 
     * @param minTextSize
     */
    public void setMinTextSize(float minTextSize) {
        mMinTextSize = minTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return lower text size limit
     * 
     * @return
     */
    public float getMinTextSize() {
        return mMinTextSize;
    }

    /**
     * Set flag to add ellipsis to text that overflows at the smallest text size
     * 
     * @param addEllipsis
     */
    public void setAddEllipsis(boolean addEllipsis) {
        mAddEllipsis = addEllipsis;
    }

    /**
     * Return flag to add ellipsis to text that overflows at the smallest text
     * size
     * 
     * @return
     */
    public boolean getAddEllipsis() {
        return mAddEllipsis;
    }

    /**
     * Reset the text to the original size
     */
    public void resetTextSize() {
        if (mTextSize > 0) {
            super.setTextSize(TypedValue.COMPLEX_UNIT_PX, mTextSize);
            // mMaxTextSize = mTextSize;
        }
    }

    /**
     * Resize text after measuring
     */

    @Override
    protected void onLayout(boolean changed, int left, int top, int right,
            int bottom) {
        if (changed || mNeedsResize) {
            int widthLimit = (right - left) - getCompoundPaddingLeft()
                    - getCompoundPaddingRight();
            int heightLimit = (bottom - top) - getCompoundPaddingBottom()
                    - getCompoundPaddingTop();
            resizeText(widthLimit, heightLimit);
        }
        super.onLayout(changed, left, top, right, bottom);
    }

    /**
     * Resize the text size with default width and height
     */
    public void resizeText() {

        // Height and width with a padding as a percentage of height
        int heightLimit = getHeight() - getPaddingBottom() - getPaddingTop();
        int widthLimit = getWidth() - getPaddingLeft() - getPaddingRight();
        resizeText(widthLimit, heightLimit);
    }

    /**
     * Resize the text size with specified width and height
     * 
     * @param width
     * @param height
     */
    public void resizeText(int width, int height) {
        CharSequence text = getText();

        // Do not resize if the view does not have dimensions or there is no
        // text
        if (text == null || text.length() == 0 || height <= 0 || width <= 0
                || mTextSize == 0) {
            return;
        }

        // Get the text view's paint object
        TextPaint textPaint = getPaint();

        // Store the current text size
        float oldTextSize = textPaint.getTextSize();

        // Bisection method: fast & precise
        float lower = mMinTextSize;
        float upper = mMaxTextSize;
        int loop_counter = 1;
        float targetTextSize = (lower + upper) / 2;
        int textHeight = getTextHeight(text, textPaint, width, targetTextSize);
        int textWidth = getTextWidth(text, textPaint, width, targetTextSize);

        while (loop_counter < BISECTION_LOOP_WATCH_DOG && upper - lower > 1) {
            targetTextSize = (lower + upper) / 2;
            textHeight = getTextHeight(text, textPaint, width, targetTextSize);
            textWidth = getTextWidth(text, textPaint, width, targetTextSize);
            if (textHeight > (height) || textWidth > (width))
                upper = targetTextSize;
            else
                lower = targetTextSize;
            loop_counter++;
        }

        targetTextSize = lower;
        textHeight = getTextHeight(text, textPaint, width, targetTextSize);

        // If we had reached our minimum text size and still don't fit, append
        // an ellipsis
        if (mAddEllipsis && targetTextSize == mMinTextSize
                && textHeight > height) {
            // Draw using a static layout
            // modified: use a copy of TextPaint for measuring
            TextPaint paintCopy = new TextPaint(textPaint);
            paintCopy.setTextSize(targetTextSize);
            StaticLayout layout = new StaticLayout(text, paintCopy, width,
                    Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, false);
            // Check that we have a least one line of rendered text
            if (layout.getLineCount() > 0) {
                // Since the line at the specific vertical position would be cut
                // off,
                // we must trim up to the previous line
                int lastLine = layout.getLineForVertical(height) - 1;
                // If the text would not even fit on a single line, clear it
                if (lastLine < 0) {
                    setText("");
                }
                // Otherwise, trim to the previous line and add an ellipsis
                else {
                    int start = layout.getLineStart(lastLine);
                    int end = layout.getLineEnd(lastLine);
                    float lineWidth = layout.getLineWidth(lastLine);
                    float ellipseWidth = paintCopy.measureText(mEllipsis);

                    // Trim characters off until we have enough room to draw the
                    // ellipsis
                    while (width < lineWidth + ellipseWidth) {
                        lineWidth = paintCopy.measureText(text.subSequence(
                                start, --end + 1).toString());
                    }
                    setText(text.subSequence(0, end) + mEllipsis);
                }
            }
        }

        // Some devices try to auto adjust line spacing, so force default line
        // spacing
        // and invalidate the layout as a side effect
        setTextSize(TypedValue.COMPLEX_UNIT_PX, targetTextSize);
        setLineSpacing(mSpacingAdd, mSpacingMult);

        // Notify the listener if registered
        if (mTextResizeListener != null) {
            mTextResizeListener.onTextResize(this, oldTextSize, targetTextSize);
        }

        // Reset force resize flag
        mNeedsResize = false;
    }

    // Set the text size of the text paint object and use a static layout to
    // render text off screen before measuring
    private int getTextHeight(CharSequence source, TextPaint originalPaint,
            int width, float textSize) {
        // modified: make a copy of the original TextPaint object for measuring
        // (apparently the object gets modified while measuring, see also the
        // docs for TextView.getPaint() (which states to access it read-only)
        TextPaint paint = new TextPaint(originalPaint);
        // Update the text paint object
        paint.setTextSize(textSize);
        // Measure using a static layout
        StaticLayout layout = new StaticLayout(source, paint, width,
                Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
        return layout.getHeight();
    }

    // Set the text size of the text paint object and use a static layout to
    // render text off screen before measuring
    private int getTextWidth(CharSequence source, TextPaint originalPaint,
            int width, float textSize) {
        // Update the text paint object
        TextPaint paint = new TextPaint(originalPaint);
        // Draw using a static layout
        paint.setTextSize(textSize);

        StaticLayout layout = new StaticLayout(source, paint, width,
                Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);

        return (int) layout.getLineWidth(0);
    }
}

1

මට චේස් සහ එම්-වජේ වෙතින් කේත භාවිතා කර ඇති අතර මෙහි යම් වාසියක් හා අවාසියක් මට හමු විය

හඹා යාමේ සිට

වාසිය:

  • පේළි 1 ක් සඳහා එය පරිපූර්ණයි

අවාසිය:

  • එය අභිරුචි අකුරු සහිත පේළි 1 ට වඩා වැඩි නම් සමහර පෙළ අතුරුදහන් වනු ඇත

  • එය ඉලිප්සාකාරය සක්‍රීය කරන්නේ නම්, එය ඉලිප්සයට ඉඩක් සූදානම් කළේ නැත

  • එය අභිරුචි අකුරු (යතුරු පුවරුව) නම්, එය සහාය නොදක්වයි

M-WaJeEh වෙතින්

වාසිය:

  • එය බහු රේඛා සඳහා පරිපූර්ණයි

අවාසිය:

  • උස එතීමේ අන්තර්ගතයක් ලෙස සකසන්නේ නම්, මෙම කේතය අවම ප්‍රමාණයෙන් ආරම්භ වන අතර එය හැකි තරම් කුඩාම දක්වා අඩු වනු ඇත, සෙට්සයිස් වෙතින් නොව සීමිත පළලින් අඩු වේ

  • එය අභිරුචි අකුරු (යතුරු පුවරුව) නම්, එය සහාය නොදක්වයි


1
getTextHeight()පෙළ හෝ පෙළ ප්‍රමාණය සැකසීමේදී බිඳ වැටේ . ඇන්ඩ්‍රොයිඩ් 4.0.4 ඉමුලේටරය. java.lang.IllegalArgumentException: Layout: -40 < 0 at android.text.Layout.<init>(Layout.java:140) at android.text.StaticLayout.<init>(StaticLayout.java:104) at android.text.StaticLayout.<init>(StaticLayout.java:90) at android.text.StaticLayout.<init>(StaticLayout.java:68) at android.text.StaticLayout.<init>(StaticLayout.java:48)
TalkLittle

1

මගේ ක්‍රමය:

public void changeTextSize(int initialSize, TextView tv) {

    DisplayMetrics displayMetrics = new DisplayMetrics();
    getWindowManager().getDefaultDisplay().getMetrics(displayMetrics);
    double width = displayMetrics.widthPixels / displayMetrics.xdpi;
    double height = displayMetrics.heightPixels / displayMetrics.ydpi;

    Log.i("LOG", "The width of the tested emulator is: " + width);
    Log.i("LOG", "The height of the tested emulator is: " + height);

    double scale = Math.min(width / 2.25, height / 4.0); //See the logcat >>> width = 2.25 and heigt = 4.0
    tv.setTextSize(TypedValue.COMPLEX_UNIT_SP, (int) (initialSize * scale));

}

උදාහරණයක් වශයෙන්:

changeTextSize(16, findViewById(R.id.myTextView));
changeTextSize(12, findViewById(R.id.myEditText));

0

මෙම විසඳුම් අප වෙනුවෙන් ක්‍රියාත්මක වේ:

public class CustomFontButtonTextFit extends CustomFontButton
{
    private final float DECREMENT_FACTOR = .1f;

    public CustomFontButtonTextFit(Context context) {
        super(context);
    }

    public CustomFontButtonTextFit(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomFontButtonTextFit(Context context, AttributeSet attrs,
            int defStyle) {
        super(context, attrs, defStyle);
    }

    private synchronized void refitText(String text, int textWidth) {
        if (textWidth > 0) 
        {
            float availableWidth = textWidth - this.getPaddingLeft()
                    - this.getPaddingRight();

            TextPaint tp = getPaint();
            Rect rect = new Rect();
            tp.getTextBounds(text, 0, text.length(), rect);
            float size = rect.width();

            while(size > availableWidth)
            {
                setTextSize( getTextSize() - DECREMENT_FACTOR );
                tp = getPaint();

                tp.getTextBounds(text, 0, text.length(), rect);
                size = rect.width();
            }
        }
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) 
    {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        int parentWidth = MeasureSpec.getSize(widthMeasureSpec);
        int parentHeight = MeasureSpec.getSize(heightMeasureSpec);

        refitText(this.getText().toString(), parentWidth);

        if(parentWidth < getSuggestedMinimumWidth())
            parentWidth = getSuggestedMinimumWidth();

        if(parentHeight < getSuggestedMinimumHeight())
            parentHeight = getSuggestedMinimumHeight();

        this.setMeasuredDimension(parentWidth, parentHeight);
    }

    @Override
    protected void onTextChanged(final CharSequence text, final int start,
            final int before, final int after) 
    {
        super.onTextChanged(text, start, before, after);

        refitText(text.toString(), this.getWidth());
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh)
    {
        super.onSizeChanged(w, h, oldw, oldh);

        if (w != oldw) 
            refitText(this.getText().toString(), w);
    }
}

2
මෙය විහිදෙන CustomFontButton යනු කුමක්ද?
සංකීර්ණ

මම පන්තියේ CustomFontButtonTextFitසිට දිගු කළහොත් ButtonUI කිසි විටෙකත් නොපෙන්වයි. මම කියන්නේ මේ පන්තිය කැඩිලා කියලා ...
කවුරුහරි කොහේ හරි

0

කම්මැලි ක්‍රමලේඛකයින් සඳහා චේස් සහ ඔනෙල්ට ස්තූතියි, පෙළ දර්ශනයක් වෙනුවට බොත්තමක් මත අනුවර්තනය කරන ලද ඔවුන්ගේ අපූරු ඒකාබද්ධ කේතයේ ක්‍රියාකාරී පිටපතක් මෙහි පළ කිරීමට මට ඉඩ දෙන්න.

ඔබගේ සියලු බොත්තම් (ImageButtons නොව) AutoResizeTextButtons සමඟ ආදේශ කරන්න, එම නීරස ගැටළුවද ඔවුන් සඳහාම විසඳා ඇත.

මෙන්න කේතය. මම ආනයන ඉවත් කළා.

/**
 *            DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
 *                    Version 2, December 2004
 * 
 * Copyright (C) 2004 Sam Hocevar <sam@hocevar.net>
 * 
 * Everyone is permitted to copy and distribute verbatim or modified
 * copies of this license document, and changing it is allowed as long
 * as the name is changed.
 * 
 *            DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
 *   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
 * 
 *  0. You just DO WHAT THE FUCK YOU WANT TO.
 *  made better by onoelle
 *  adapted for button by beppi
 */

/**
 * Text Button that auto adjusts text size to fit within the view.
 * If the text size equals the minimum text size and still does not
 * fit, append with an ellipsis.
 * 
 * @author Chase Colburn
 * @since Apr 4, 2011
 */
public class AutoResizeTextButton extends Button {

    // Minimum text size for this text view
    public static final float MIN_TEXT_SIZE = 20;

    // Interface for resize notifications
    public interface OnTextResizeListener {
        public void onTextResize(Button textView, float oldSize, float newSize);
    }

    // Our ellipse string
    private static final String mEllipsis = "...";

    // Registered resize listener
    private OnTextResizeListener mTextResizeListener;

    // Flag for text and/or size changes to force a resize
    private boolean mNeedsResize = false;

    // Text size that is set from code. This acts as a starting point for resizing
    private float mTextSize;

    // Temporary upper bounds on the starting text size
    private float mMaxTextSize = 0;

    // Lower bounds for text size
    private float mMinTextSize = MIN_TEXT_SIZE;

    // Text view line spacing multiplier
    private float mSpacingMult = 1.0f;

    // Text view additional line spacing
    private float mSpacingAdd = 0.0f;

    // Add ellipsis to text that overflows at the smallest text size
    private boolean mAddEllipsis = true;

    // Default constructor override
    public AutoResizeTextButton(Context context) {
        this(context, null);
    }

    // Default constructor when inflating from XML file
    public AutoResizeTextButton(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    // Default constructor override
    public AutoResizeTextButton(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mTextSize = getTextSize();
    }

    /**
     * When text changes, set the force resize flag to true and reset the text size.
     */
    @Override
    protected void onTextChanged(final CharSequence text, final int start, final int before, final int after) {
        mNeedsResize = true;
        // Since this view may be reused, it is good to reset the text size
        resetTextSize();
    }

    /**
     * If the text view size changed, set the force resize flag to true
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        if (w != oldw || h != oldh) {
            mNeedsResize = true;
        }
    }

    /**
     * Register listener to receive resize notifications
     * @param listener
     */
    public void setOnResizeListener(OnTextResizeListener listener) {
        mTextResizeListener = listener;
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(float size) {
        super.setTextSize(size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set text size to update our internal reference values
     */
    @Override
    public void setTextSize(int unit, float size) {
        super.setTextSize(unit, size);
        mTextSize = getTextSize();
    }

    /**
     * Override the set line spacing to update our internal reference values
     */
    @Override
    public void setLineSpacing(float add, float mult) {
        super.setLineSpacing(add, mult);
        mSpacingMult = mult;
        mSpacingAdd = add;
    }

    /**
     * Set the upper text size limit and invalidate the view
     * @param maxTextSize
     */
    public void setMaxTextSize(float maxTextSize) {
        mMaxTextSize = maxTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return upper text size limit
     * @return
     */
    public float getMaxTextSize() {
        return mMaxTextSize;
    }

    /**
     * Set the lower text size limit and invalidate the view
     * @param minTextSize
     */
    public void setMinTextSize(float minTextSize) {
        mMinTextSize = minTextSize;
        requestLayout();
        invalidate();
    }

    /**
     * Return lower text size limit
     * @return
     */
    public float getMinTextSize() {
        return mMinTextSize;
    }

    /**
     * Set flag to add ellipsis to text that overflows at the smallest text size
     * @param addEllipsis
     */
    public void setAddEllipsis(boolean addEllipsis) {
        mAddEllipsis = addEllipsis;
    }

    /**
     * Return flag to add ellipsis to text that overflows at the smallest text size
     * @return
     */
    public boolean getAddEllipsis() {
        return mAddEllipsis;
    }

    /**
     * Reset the text to the original size
     */
    public void resetTextSize() {
        if(mTextSize > 0) {
            super.setTextSize(TypedValue.COMPLEX_UNIT_PX, mTextSize);
            mMaxTextSize = mTextSize;
        }
    }

    /**
     * Resize text after measuring
     */
    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        if(changed || mNeedsResize) {
            int widthLimit = (right - left) - getCompoundPaddingLeft() - getCompoundPaddingRight();
            int heightLimit = (bottom - top) - getCompoundPaddingBottom() - getCompoundPaddingTop();
            resizeText(widthLimit, heightLimit);
        }
        super.onLayout(changed, left, top, right, bottom);
    }


    /**
     * Resize the text size with default width and height
     */
    public void resizeText() {
        int heightLimit = getHeight() - getPaddingBottom() - getPaddingTop();
        int widthLimit = getWidth() - getPaddingLeft() - getPaddingRight();
        resizeText(widthLimit, heightLimit);
    }

    /**
     * Resize the text size with specified width and height
     * @param width
     * @param height
     */
    public void resizeText(int width, int height) {
        CharSequence text = getText();
        // Do not resize if the view does not have dimensions or there is no text
        if(text == null || text.length() == 0 || height <= 0 || width <= 0 || mTextSize == 0) {
            return;
        }

        // Get the text view's paint object
        TextPaint textPaint = getPaint();

        // Store the current text size
        float oldTextSize = textPaint.getTextSize();
        // If there is a max text size set, use the lesser of that and the default text size
        float targetTextSize = mMaxTextSize > 0 ? Math.min(mTextSize, mMaxTextSize) : mTextSize;

        // Get the required text height
        int textHeight = getTextHeight(text, textPaint, width, targetTextSize);

        // Until we either fit within our text view or we had reached our min text size, incrementally try smaller sizes
        while(textHeight > height && targetTextSize > mMinTextSize) {
            targetTextSize = Math.max(targetTextSize - 2, mMinTextSize);
            textHeight = getTextHeight(text, textPaint, width, targetTextSize);
        }

        // If we had reached our minimum text size and still don't fit, append an ellipsis
        if(mAddEllipsis && targetTextSize == mMinTextSize && textHeight > height) {
            // Draw using a static layout
            // modified: use a copy of TextPaint for measuring
            TextPaint paint = new TextPaint(textPaint);
            StaticLayout layout = new StaticLayout(text, paint, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, false);
            // Check that we have a least one line of rendered text
            if(layout.getLineCount() > 0) {
                // Since the line at the specific vertical position would be cut off,
                // we must trim up to the previous line
                int lastLine = layout.getLineForVertical(height) - 1;
                // If the text would not even fit on a single line, clear it
                if(lastLine < 0) {
                    setText("");
                }
                // Otherwise, trim to the previous line and add an ellipsis
                else {
                    int start = layout.getLineStart(lastLine);
                    int end = layout.getLineEnd(lastLine);
                    float lineWidth = layout.getLineWidth(lastLine);
                    float ellipseWidth = textPaint.measureText(mEllipsis);

                    // Trim characters off until we have enough room to draw the ellipsis
                    while(width < lineWidth + ellipseWidth) {
                        lineWidth = textPaint.measureText(text.subSequence(start, --end + 1).toString());
                    }
                    setText(text.subSequence(0, end) + mEllipsis);
                }
            }
        }

        // Some devices try to auto adjust line spacing, so force default line spacing
        // and invalidate the layout as a side effect
//      textPaint.setTextSize(targetTextSize);
     // modified: setting text size via this.setTextSize (instead of textPaint.setTextSize(targetTextSize))
        setTextSize(TypedValue.COMPLEX_UNIT_PX, targetTextSize);
        setLineSpacing(mSpacingAdd, mSpacingMult);

        // Notify the listener if registered
        if(mTextResizeListener != null) {
            mTextResizeListener.onTextResize(this, oldTextSize, targetTextSize);
        }

        // Reset force resize flag
        mNeedsResize = false;
    }

    // Set the text size of the text paint object and use a static layout to render text off screen before measuring
    private int getTextHeight(CharSequence source, TextPaint originalPaint, int width, float textSize) {
          // modified: make a copy of the original TextPaint object for measuring
          // (apparently the object gets modified while measuring, see also the
          // docs for TextView.getPaint() (which states to access it read-only)
        // Update the text paint object
          TextPaint paint = new TextPaint(originalPaint);
        paint.setTextSize(textSize);
        // Measure using a static layout
        StaticLayout layout = new StaticLayout(source, paint, width, Alignment.ALIGN_NORMAL, mSpacingMult, mSpacingAdd, true);
        return layout.getHeight();
    }

}

භාවිතය:

වෙන කිසිවක් වෙනස් නොකර සාමාන්‍ය බොත්තමක් වෙනුවට ඔබේ xml තුළ AutoResizeTextButton එකක් දමන්න. OnCreate () දමා (උදාහරණයක් ලෙස):

    myButton = (AutoResizeTextButton)getView().findViewById(id.myButton);
    myButton.setMinTextSize(8f);
    myButton.resizeText();

බොත්තම් පසුබිම අදෘශ්‍යමාන ය. ඔබගේ බොත්තම් දර්ශනයට xml: style = "? Android: attr / buttonBarButtonStyle"
ක්‍රිස්ටි වේල්ස්

0

මෙන්න මම ගන්නා ප්රවේශය. එය ඉතා සරල ය. එය අකුරු ප්‍රමාණයෙන් බිංදුවට අනුක්‍රමික දළ වශයෙන් භාවිතා කරන අතර සාමාන්‍යයෙන් එය පුනරාවර්තන 10 කට වඩා අඩු ගණනකින් ගණනය කළ හැකිය. පා activity ය ප්‍රදර්ශනය කිරීම සඳහා ඔබ භාවිතා කරන ඕනෑම දර්ශනයක පළල සමඟ "ක්‍රියාකාරකම් පළල" ආදේශ කරන්න. මගේ උදාහරණයේ දී, එය තිරයේ පළලට පෞද්ගලික ක්ෂේත්‍රයක් ලෙස සකසා ඇත. 198 හි ආරම්භක අකුරු සැකසෙන්නේ ක්‍රමය ව්‍යතිරේකයක් ජනනය කළහොත් පමණි (එය කිසි විටෙකත් සිදු නොවිය යුතුය):

  private float GetFontSizeForScreenWidth(String text)
  {
    float fontsize = 198;

    try
    {
      Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
      paint.setColor(Color.RED);
      Typeface typeface = Typeface.create("Helvetica", Typeface.BOLD);
      paint.setTypeface(typeface);
      paint.setTextAlign(Align.CENTER);

      int lowVal = 0;
      int highVal = 2000;
      int currentVal = highVal;

      /*
       * Successively approximate the screen size until it is 
       * within 2 pixels of the maximum screen width. Generally
       * this will get you to the closest font size within about 10
       * iterations.
       */

      do
      {
        paint.setTextSize(currentVal);
        float textWidth = paint.measureText(text);

        float diff = activityWidth - textWidth;

        if ((diff >= 0) && (diff <= 2))
        {
          fontsize = paint.getTextSize();
          return fontsize;
        }

        if (textWidth > activityWidth)
          highVal = currentVal;
        else if (textWidth < activityWidth)
          lowVal = currentVal;
        else
        {
          fontsize = paint.getTextSize();
          return fontsize;
        }

        currentVal = (highVal - lowVal) / 2 + lowVal;

      } while (true);      
    }
    catch (Exception ex)
    {
      return fontsize;
    }
  }

0

පෙළ දර්ශනය දිගු කර පහත කේතය සමඟ onDraw ඉක්මවා යන්න. එය පෙළ දර්ශන අනුපාතය තබා ඇති නමුත් අවකාශය පිරවීම සඳහා එය විශාල කරයි. අවශ්‍ය නම් දිගු කිරීම සඳහා ඔබට පහසුවෙන් කේතය වෙනස් කළ හැකිය.

  @Override
  protected void onDraw(@NonNull Canvas canvas) {
    TextPaint textPaint = getPaint();
    textPaint.setColor(getCurrentTextColor());
    textPaint.setTextAlign(Paint.Align.CENTER);
    textPaint.drawableState = getDrawableState();

    String text = getText().toString();
    float desiredWidth = getMeasuredWidth() - getPaddingLeft() - getPaddingRight() - 2;
    float desiredHeight = getMeasuredHeight() - getPaddingTop() - getPaddingBottom() - 2;
    float textSize = textPaint.getTextSize();

    for (int i = 0; i < 10; i++) {
      textPaint.getTextBounds(text, 0, text.length(), rect);
      float width = rect.width();
      float height = rect.height();

      float deltaWidth = width - desiredWidth;
      float deltaHeight = height - desiredHeight;

      boolean fitsWidth = deltaWidth <= 0;
      boolean fitsHeight = deltaHeight <= 0;

      if ((fitsWidth && Math.abs(deltaHeight) < 1.0)
          || (fitsHeight && Math.abs(deltaWidth) < 1.0)) {
        // close enough
        break;
      }

      float adjustX = desiredWidth / width;
      float adjustY = desiredHeight / height;

      textSize = textSize * (adjustY < adjustX ? adjustY : adjustX);

      // adjust text size
      textPaint.setTextSize(textSize);
    }
    float x = desiredWidth / 2f;
    float y = desiredHeight / 2f - rect.top - rect.height() / 2f;
    canvas.drawText(text, x, y, textPaint);
  }

කොහෙන්ද rect?
තෝමස් ඩබ්ලිව්

OnDraw තුළ නිර්මාණය කර ඇති වස්තූන් ගණන අවම කළ යුතු බැවින් ඔබ එය onDraw වලින් පිටත ප්‍රකාශ කර නිර්මාණය කළ යුතුය. එවිට අගය getTextBounds මඟින් සකසා ඇත.
ග්‍රෙග් බැචස්
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.