Android向けにTextViewを自動調整


231

バックグラウンド

多くの場合、与えられた境界にTextViewのフォントを自動調整する必要があります。

問題

悲しいことに、この問題について話しているスレッドや投稿(および推奨される解決策)はたくさんありますが(例:ここここここここ)、実際にうまく機能するものはありません。

だからこそ、実際の取引が見つかるまで、それぞれをテストすることにしました。

私はそのようなtextViewからの要件は次のようになるべきだと思います:

  1. フォント、書体、スタイル、および文字のセットを使用できるようにする必要があります。

  2. 幅と高さの両方を処理する必要があります

  3. 制限のためにテキストが収まらない場合を除き、切り捨ては行いません(例:長すぎるテキスト、小さすぎる使用可能なサイズ)。ただし、必要に応じて、水平/垂直スクロールバーをリクエストすることもできます。

  4. 複数行または単一行を許可する必要があります。複数行の場合、最大行と最小行を許可します。

  5. 計算が遅くなることはありません。最適なサイズを見つけるためにループを使用していますか?少なくとも最適化し、毎回1ずつサンプリングを増分しないでください。

  6. 複数行の場合、サイズ変更またはより多くの行を使用することを許可するか、「\ n」文字を使用して自分で行を選択できるようにする必要があります。

私が試したこと

私は非常に多くのサンプル(リンクのサンプルを含めて、私が書いた)を試してみましたが、ケースを処理するためにそれらを変更することも試みたと話しましたが、実際に機能するものはありません。

TextViewが正しく自動調整されるかどうかを視覚的に確認できるサンプルプロジェクトを作成しました。

現在、私のサンプルプロジェクトでは、テキスト(英語のアルファベットと数字)とtextViewのサイズのみをランダム化し、1行のままにしておきますが、これでも、試したどのサンプルでもうまく機能しません。

ここにコードがあります(ここにもあります):

ファイル res/layout/activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
  android:layout_height="match_parent" tools:context=".MainActivity">
  <Button android:id="@+id/button1" android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_alignParentBottom="true"
    android:layout_centerHorizontal="true" android:text="Button" />
  <FrameLayout android:layout_width="match_parent"
    android:layout_height="wrap_content" android:layout_above="@+id/button1"
    android:layout_alignParentLeft="true" android:background="#ffff0000"
    android:layout_alignParentRight="true" android:id="@+id/container"
    android:layout_alignParentTop="true" />

</RelativeLayout>

ファイル src/.../MainActivity.java

public class MainActivity extends Activity
  {
  private final Random        _random            =new Random();
  private static final String ALLOWED_CHARACTERS ="qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM1234567890";

  @Override
  protected void onCreate(final Bundle savedInstanceState)
    {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    final ViewGroup container=(ViewGroup)findViewById(R.id.container);
    findViewById(R.id.button1).setOnClickListener(new OnClickListener()
      {
        @Override
        public void onClick(final View v)
          {
          container.removeAllViews();
          final int maxWidth=container.getWidth();
          final int maxHeight=container.getHeight();
          final FontFitTextView fontFitTextView=new FontFitTextView(MainActivity.this);
          final int width=_random.nextInt(maxWidth)+1;
          final int height=_random.nextInt(maxHeight)+1;
          fontFitTextView.setLayoutParams(new LayoutParams(width,height));
          fontFitTextView.setSingleLine();
          fontFitTextView.setBackgroundColor(0xff00ff00);
          final String text=getRandomText();
          fontFitTextView.setText(text);
          container.addView(fontFitTextView);
          Log.d("DEBUG","width:"+width+" height:"+height+" text:"+text);
          }
      });
    }

  private String getRandomText()
    {
    final int textLength=_random.nextInt(20)+1;
    final StringBuilder builder=new StringBuilder();
    for(int i=0;i<textLength;++i)
      builder.append(ALLOWED_CHARACTERS.charAt(_random.nextInt(ALLOWED_CHARACTERS.length())));
    return builder.toString();
    }
  }

質問

実際に機能するこの一般的な問題の解決策を誰かが知っていますか?

私が書いたものよりも機能がはるかに少ないソリューションでも、たとえば、テキストの行数が一定で、サイズに応じてフォントを調整しますが、奇妙な不具合はなく、テキストも取得します利用可能なスペースと比較して大きい/小さい。


GitHubプロジェクト

これはとても重要なTextViewなので、誰でも簡単に使用して貢献できるように、ここでライブラリを公開することにしました



@Thrakbadそれは私が言及したリンクの一つです。また、テストに合格しません。
Androidデベロッパー

申し訳ありませんが、最後の例をなんとか逃してしまいました
Thrakbad

はい、信じてください。私は多くのサンプルを試してみました。また、見つけた問題を修正するためにサンプルを変更しようとしましたが、成功しませんでした。うまくいくと思うものがあれば、テストしてみてください。このためのサンプルコードを投稿しました。
Android開発者

1
@ruleこれは既に読んだ投稿の1つであり、すべてのコードサンプルをテストしました。また、あなたは二重に投稿したと思います。
Android開発者

回答:


147

MartinHの簡単な修正のおかげで、ここでは、このコードもの世話をするandroid:drawableLeftandroid:drawableRightandroid:drawableTopおよびandroid:drawableBottomタグ。


ここでの私の答えは、境界内に収まるようにTextViewテキストを自動スケールすることを喜ばせるでしょう

テストケースを変更しました:

@Override
protected void onCreate(final Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    final ViewGroup container = (ViewGroup) findViewById(R.id.container);
    findViewById(R.id.button1).setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(final View v) {
            container.removeAllViews();
            final int maxWidth = container.getWidth();
            final int maxHeight = container.getHeight();
            final AutoResizeTextView fontFitTextView = new AutoResizeTextView(MainActivity.this);
            final int width = _random.nextInt(maxWidth) + 1;
            final int height = _random.nextInt(maxHeight) + 1;
            fontFitTextView.setLayoutParams(new FrameLayout.LayoutParams(
                    width, height));
            int maxLines = _random.nextInt(4) + 1;
            fontFitTextView.setMaxLines(maxLines);
            fontFitTextView.setTextSize(500);// max size
            fontFitTextView.enableSizeCache(false);
            fontFitTextView.setBackgroundColor(0xff00ff00);
            final String text = getRandomText();
            fontFitTextView.setText(text);
            container.addView(fontFitTextView);
            Log.d("DEBUG", "width:" + width + " height:" + height
                    + " text:" + text + " maxLines:" + maxLines);
        }
    });
}

私はアンドロイド開発者の要求に従ってここにコードを投稿しています:

最終的な効果:

ここに画像の説明を入力してください

サンプルレイアウトファイル:

<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>

そしてJavaコード:

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 mInitializedDimens;

    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;
        }
    }

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

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

    public int getMaxLines() {
        return mMaxLines;
    }

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

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

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

    @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();
    }

    @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;
        adjustTextSize();
    }

    private void adjustTextSize() {
        if (!mInitializedDimens) {
            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);
        }
        int key = getText().toString().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 the 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);
        adjustTextSize();
    }

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

警告:

ただし、Android 3.1(Honeycomb)で解決された このバグには注意してください。


はい、コードをテストしましたが、問題はないようです。ただし、API 16用のgetMaxLines()を使用しましたが、maxLinesを保存して、setMaxLinesをオーバーライドしてその値を保存することで取得できます。これを変更し、正常に機能するようになりました。また、テキストが最小サイズに対して長すぎることもあるので、私はsetEllipsizeを使用しようとしましたが、それもうまくいきました!勝者はいると思う。ここにコードを投稿して、正しいコードとしてマークしてください。
Android開発者

getMaxLines()を削除せず、独自のものを使用した。それでもAPI16からのみ機能します...誰でも使用できるように変更してください。getMaxLinesを使用する代わりに、setMaxLinesをオーバーライドしてパラメーターをフィールドに格納し、フィールドにアクセスすることをお勧めします。
Android開発者

今すぐ確認し、最大行数のサポートを追加しました。
M-WaJeEh 2013

元気そうです。「@TargetApi(Build.VERSION_CODES.JELLY_BEAN)」をonTestSize()メソッドに追加して、Lintがエラー/警告を出さないようにし、代わりにCTORで初期化する必要があります。バイナリ検索を静的に設定することも良いことです。最大のCTORで初期化を行い、他のCTORから呼び出すこともできます。しかし、それは本当に最良の答えであり、あなたはVに値する。これは、この古い質問に対する良い答えです。よくやった!
Android開発者

@ M-WaJeEhこのスレッドの新しい投稿をチェックしてください。"MartinH"と呼ばれるユーザーは、彼があなたのコードを修正したと言っています。
Android開発者

14

M-WaJeEhの回答を少し変更して、両サイドの複合ドローアブルを考慮に入れました。

getCompoundPaddingXXXX()方法は返しますpadding of the view + drawable space。例を参照してください:TextView.getCompoundPaddingLeft()

問題: これにより、テキストに使用できるTextViewスペースの幅と高さの測定が修正されます。ドローアブルのサイズを考慮しない場合、それは無視され、テキストはドローアブルと重なります。


更新されたセグメントadjustTextSize(String)

private void adjustTextSize(final String text) {
  if (!mInitialized) {
    return;
  }
  int heightLimit = getMeasuredHeight() - getCompoundPaddingBottom() - getCompoundPaddingTop();
  mWidthLimit = getMeasuredWidth() - getCompoundPaddingLeft() - getCompoundPaddingRight();

  mAvailableSpaceRect.right = mWidthLimit;
  mAvailableSpaceRect.bottom = heightLimit;

  int maxTextSplits = text.split(" ").length;
  AutoResizeTextView.super.setMaxLines(Math.min(maxTextSplits, mMaxLines));

  super.setTextSize(
      TypedValue.COMPLEX_UNIT_PX,
      binarySearch((int) mMinTextSize, (int) mMaxTextSize,
                   mSizeTester, mAvailableSpaceRect));
}

M-WaJeEhのソースを更新しました。なぜそれを新しい答えにしたのですか?私はあなたが善意を持っていることを知っていますが、それを答えとして入れるのは良いことですか?私は彼があなたの答えに気づくのを見ることができるかわかりません...
アンドロイド開発者

1
そのアップデートを提供できるように登録しました。私の評判数は1にすぎないため、私のものではない投稿にコメントすることはできません(最低15)。
MartinH 2013年

心配する必要はありません。たぶん、現時点では彼に連絡することさえできないので、私の答えをM-WajeEhに送信したいと思います:/
MartinH

自分が何をしたのか説明してください。バグを修正しましたか?バグを実証するためにスクリーンショットを表示できますか?
Android開発者

1
こんにちは@ MartinH、SOへようこそ。私はこれを注意深く調べ、あなたの名前を言及し、テスト後にあなたの投稿へのリンクを投稿することによって私の答えを更新します。ほんの数日ください。最近はどこかに行き詰まっています。
M-WaJeEh 2013年

7

さて、先週、コードを大幅に書き換えて、テストに正確に適合するようにしました。これで1:1をコピーでき、すぐに機能しsetSingleLine()ます。調整しMIN_TEXT_SIZEMAX_TEXT_SIZE極端な値を求める場合は忘れないでください。

収束アルゴリズムは次のようになります。

for (float testSize; (upperTextSize - lowerTextSize) > mThreshold;) {

    // Go to the mean value...
    testSize = (upperTextSize + lowerTextSize) / 2;

    // ... inflate the dummy TextView by setting a scaled textSize and the text...
    mTestView.setTextSize(TypedValue.COMPLEX_UNIT_SP, testSize / mScaledDensityFactor);
    mTestView.setText(text);

    // ... call measure to find the current values that the text WANTS to occupy
    mTestView.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);
    int tempHeight = mTestView.getMeasuredHeight();

    // ... decide whether those values are appropriate.
    if (tempHeight >= targetFieldHeight) {
        upperTextSize = testSize; // Font is too big, decrease upperSize
    }
    else {
        lowerTextSize = testSize; // Font is too small, increase lowerSize
    }
}

クラス全体はここにあります。

結果は非常に柔軟です。これは次のようにxmlで宣言されたのと同じように機能します。

<com.example.myProject.AutoFitText
    android:id="@+id/textView"
    android:layout_width="match_parent"
    android:layout_height="0dp"
    android:layout_weight="4"
    android:text="@string/LoremIpsum" />

...テストのようにプログラムでビルドされます。

これを今すぐ使えるといいな。setText(CharSequence text)ちなみに今すぐ電話して使えます。クラスは途方もなくまれな例外を処理し、堅実でなければなりません。アルゴリズムがまだサポートしていない唯一のものは次のとおりです。

  • setMaxLines(x)どこへの呼び出しx >= 2

しかし、私はあなたが望むならこれを構築するのを助けるために広範なコメントを追加しました!


ご注意ください:

これを1行に限定せずに通常どおりに使用すると、前述のように改行が発生する可能性があります。これはAndroidの機能あり、の障害ではありませんAutoFitText。Androidは常に、TextViewには長すぎる単語を分割しますが、実際には非常に便利です。ここに介入したい場合は、203行目から私のコメントとコードを参照してください。すでに十分な分割と認識を書いています。これから行う必要があるのは、単語を分割して、必要に応じて変更することだけです。 。

結論として、次のようにスペース文字もサポートするようにテストを書き直すことを強く検討する必要があります。

final Random _random = new Random();
final String ALLOWED_CHARACTERS = "qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM1234567890";
final int textLength = _random.nextInt(80) + 20;
final StringBuilder builder = new StringBuilder();
for (int i = 0; i < textLength; ++i) {
    if (i % 7 == 0 && i != 0) {
        builder.append(" ");
    }
    builder.append(ALLOWED_CHARACTERS.charAt(_random.nextInt(ALLOWED_CHARACTERS.length())));
}
((AutoFitText) findViewById(R.id.textViewMessage)).setText(builder.toString());

これにより、非常に美しい(より現実的な)結果が得られます。
あなたはこの問題でもあなたを始めるためにコメントを見つけるでしょう。

幸運とよろしく


いくつか問題がありますが、全体的には非常にうまくいきます。問題:(あなたが書いたように)複数行をサポートせず、それを行おうとすると単語が壊れる、完全に回避できる(または少なくとも代わりにOnPreDrawListenerを使用する)API16関数(addOnGlobalLayoutListener)を含める、使用バイナリ検索なので、まったく適合しないサイズをテストする可能性があり(そして、理由がなくてメモリを多く使用します(大きすぎると例外が発生する可能性があります)、何かが変更されたとき(テキスト自体など)の処理をサポートしません。
Android開発者

ところで、mTestView変数を使用して現在のビューで直接テストを行う必要はありません。これにより、特別なことをすべて考慮する必要がなくなります。また、MAX_TEXT_SIZEの代わりにtargetFieldHeightを使用できると思います。
Android開発者

純粋なバイナリ検索(特に2つ)の代わりに、測定されたサイズに対するターゲットの比率によって、より正確な推測を行うことができると思います。他の方法(ニュートンや覚えていないもの)もありますが、ここには当てはまらないと思います。最大サイズをテストする代わりに最小サイズから始めることができるので、私が今提案したものはより良いです。事実は、より良い推測をし続けるので最大サイズを必要としないということです。
Androidデベロッパー

私はまた、条件が単純にできた場合2進検索は、単一の行にマージすることができることを考える:(getMeasuredWidth()> = targetFieldWidth || getMeasuredHeight()> = targetFieldHeight)の場合
Androidデベロッパー

1
私が見つけた別の問題は、テキストの重力です。垂直方向に中央揃えできないと思います。
Android開発者

4

私の要件は

  • ScalableTextViewをクリックします。
  • listActivityを開き、さまざまな長さの文字列アイテムを表示します。
  • リストからテキストを選択します。
  • 別のアクティビティでテキストをScalableTextViewに戻します。

私はリンクを参照しました:AutoView TextView Text to Fit within Bounds(含むコメント)とDialogTitle.java

提供されたソリューションは素晴らしくシンプルであることがわかりましたが、テキストボックスのサイズを動的に変更しません。リストビューから選択したテキストの長さが、の既存のテキストの長さよりも 大きい場合に効果 的ですScalableTextView。で既存のテキストよりも短い長さのテキストを選択すると、テキストScalableTextViewのサイズは増加せず、小さいサイズで表示されます。

ScalableTextView.javaを変更して、テキストの長さに基づいてテキストサイズを再調整しました。これが私のScalableTextView.java

public class ScalableTextView extends TextView
{
float defaultTextSize = 0.0f;

public ScalableTextView(Context context, AttributeSet attrs, int defStyle)
{
    super(context, attrs, defStyle);
    setSingleLine();
    setEllipsize(TruncateAt.END);
    defaultTextSize = getTextSize();
}

public ScalableTextView(Context context, AttributeSet attrs)
{
    super(context, attrs);
    setSingleLine();
    setEllipsize(TruncateAt.END);
    defaultTextSize = getTextSize();
}

public ScalableTextView(Context context)
{
    super(context);
    setSingleLine();
    setEllipsize(TruncateAt.END);
    defaultTextSize = getTextSize();
}

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
{
    setTextSize(TypedValue.COMPLEX_UNIT_PX, defaultTextSize);
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);

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

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

                super.onMeasure(widthMeasureSpec, heightMeasureSpec);
                ellipsisCount = layout.getEllipsisCount(lineCount - 1);
            }
        }
    }
}
}

ハッピーコーディング...


反復ごとに1つずつ減らすのではなく、バイナリ検索を使用してそれを行うべきではありませんか?また、このコードは、TextViewを複数行にするのではなく、1行のテキストにするように強制しているようです。
Android開発者

これが私にとって有効な唯一の解決策でした。私の問題は、4.1でのみ
TExtViewが

100%親と一致しないようです。スクリーンショットを見る:s14.postimg.org/93c2xgs75/Screenshot_2.png
user25

4

この属性がどのように機能するかについて、Androidのバージョンを段階的に説明します。

1- Androidサポートライブラリ26.xxをプロジェクトグラドルファイルにインポートします。IDEにサポートライブラリがない場合は、自動的にダウンロードされます。

dependencies {
    compile 'com.android.support:support-v4:26.1.0'
    compile 'com.android.support:appcompat-v7:26.1.0'
    compile 'com.android.support:support-v13:26.1.0' }

allprojects {
    repositories {
        jcenter()
        maven {
            url "https://maven.google.com"
        }
    } }

2-レイアウトXMLファイルを開き、このようにTextViewにタグを付けてリファクタリングします。このシナリオは次のとおりです。システムでフォントサイズを大きくした場合、テキストをワードラップではなく、使用可能な幅に合わせます。

<android.support.v7.widget.AppCompatTextView
            android:id="@+id/textViewAutoSize"
            android:layout_width="match_parent"
            android:layout_height="25dp"
            android:ellipsize="none"
            android:text="Auto size text with compatible lower android versions."
            android:textSize="12sp"
            app:autoSizeMaxTextSize="14sp"
            app:autoSizeMinTextSize="4sp"
            app:autoSizeStepGranularity="0.5sp"
            app:autoSizeTextType="uniform" />

ただし、自動フィットTextViewの実装はうまく機能しません。回ではなく、フォントサイズをリサイズのそれは良いことだかのように、テキストだけで、ひどく...それも、自分の動画に示されています、次の行に折り返さことがありますyoutu.be/fjUdJ2aVqE4?t=14。「TextView」の「w」が次の行に移動する方法に注意してください...とにかく、ここで新しいリクエストを作成しました:issuetracker.google.com/issues/68787108。出演をご検討ください。
android開発者

@androiddeveloperそうです。このタグの実装が1行で機能することについては説明しませんでした。行を折り返す必要がある場合は、この属性を変更する必要があります:android:layout_height = "wrap_content"。ただし、android.support.v7.widget.AppCompatTextViewとGradleの実装の保証では、この属性が機能します。
Sinan Ergin 2017年

理解できないと思います。それを解決する他の方法がない限り、私は部分語が次の行に折り返すことを望みません。ビデオでは、フォントが小さくなっただけの可能性があることを簡単に確認できます。それが問題です。あなたはそれを修正する方法を知っていると言っていますか?ワードラップはなく、フォントサイズのみが変わるという意味ですか?
android開発者

@androiddeveloper多くのシナリオがあります。私のシナリオは:ワードラップではなく、テキストを自動修正して固定します。システムフォントサイズの増加中、テキストはtextviewでワードラップしないで表示する必要があります。ワードラップが問題にならない場合は、固定高さを設定しないでください。このプロパティはワードラップだけを使用するのではなく、固定幅のTextViewでテキストを自動調整できます。
Sinan Ergin 2017年

私のテストによると、フォントサイズが正しく変更されず、ワードラップの問題が発生する可能性があります。
android開発者

3

警告、Android 3(Honeycomb)およびAndroid 4.0(Ice Cream Sandwich)のバグ

Androidバージョン:3.1-4.04にはバグがあり、TextView内のsetTextSize()は初めて(最初の呼び出し)のみ機能します。

このバグについては、問題22493:Android 4.0のTextViewの高さのバグ問題17343:HoneyCombでテキストサイズを増減した後、ボタンの高さとテキストが元の状態に戻らないことで説明されています

回避策は、サイズを変更する前に、TextViewに割り当てられたテキストに改行文字を追加することです。

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ウィジェットに埋め込むこともできますが、修正コードを外部に残したいと思っていました。


どの行(どの行の前/後)およびどの関数に正確に表示できますか?
Android開発者、

わかりました。これですべてが修正され、これを読んだ人なら誰でも役立つと思います。現在、私はそれをチェックアウトできません。多分次回はそれを使うでしょう。
Android開発者

このコードは、ビューのコードの外部から呼び出されます。代わりにビューのコード内でこれを処理するための優れた代替方法を知っていますか?
Android開発者

あなたはtextView.setText()をオーバーライドして、このコードをTextViewクラスの中に置くことができます
Malachiasz

「getText()」が余分な文字を含むテキストを返すことも意味しませんか?
Android開発者

3

現在、この問題に対する公式の解決策があります。Android Oで導入されたTextViewの自動サイズ調整は、サポートライブラリ26で利用でき、Android 4.0まで下位互換性があります。

https://developer.android.com/preview/features/autosizing-textview.html

この情報も含まれているhttps://stackoverflow.com/a/42940171/47680が管理者によって削除された理由がわかりません。


1
はい、私も聞いたことがありますが、どれほど良いですか?それを使用したサンプルはありますか?彼らのビデオでさえ、私はバグに気づきました:1つの単語からの文字が(折り返し)の後の行に届きました:youtu.be/1N9KveJ-FU8?t=781(「W」に注意してください)
android developer

3

2018年6月以降、Androidはこの機能Android 4.0(APIレベル14)以降で 正式にサポートしています
Android 8.0(APIレベル26)以降:

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

Android 8.0(APIレベル26)より前のAndroidバージョン:

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

詳細な回答をご覧ください。


1

テキストビューを画像に変換し、境界内で画像を拡大縮小します。

ビューを画像に変換する方法の例を次に示します。Androidで表示せずにビューをビットマップに変換しますか?

問題は、テキストを選択できないことですが、うまくいくはずです。まだ試していないので、(スケーリングのため)どうなるかわかりません。


いいアイデアですが、これによりテキストがピクセル化され、textViewで使用できる機能がすべて削除されます。
Android開発者

0

以下は、カスタムフォントの機能が追加されたavalancha TextViewです。

使用法:

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

                <de.meinprospekt.androidhd.view.AutoFitText
                android:layout_width="wrap_content"
                android:layout_height="10dp"
                android:text="Small Text"
                android:textColor="#FFFFFF"
                android:textSize="100sp"
                foo:customFont="fonts/Roboto-Light.ttf" />

</FrameLayout>

追加することを忘れないでください:xmlns:foo = "http://schemas.android.com/apk/res-auto"。フォントはアセットファイアトリにある必要があります

import java.util.ArrayList;
import java.util.List;

import android.annotation.SuppressLint;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Paint;
import android.graphics.Typeface;
import android.os.Build;
import android.util.AttributeSet;
import android.util.Log;
import android.util.TypedValue;
import android.view.View;
import android.view.ViewGroup.LayoutParams;
import android.view.ViewTreeObserver;
import android.view.ViewTreeObserver.OnGlobalLayoutListener;
import android.widget.TextView;
import de.meinprospekt.androidhd.R;
import de.meinprospekt.androidhd.adapter.BrochuresHorizontalAdapter;
import de.meinprospekt.androidhd.util.LOG;

/**
 * https://stackoverflow.com/a/16174468/2075875 This class builds a new android Widget named AutoFitText which can be used instead of a TextView to
 * have the text font size in it automatically fit to match the screen width. Credits go largely to Dunni, gjpc, gregm and speedplane from
 * Stackoverflow, method has been (style-) optimized and rewritten to match android coding standards and our MBC. This version upgrades the original
 * "AutoFitTextView" to now also be adaptable to height and to accept the different TextView types (Button, TextClock etc.)
 * 
 * @author pheuschk
 * @createDate: 18.04.2013
 * 
 * combined with: https://stackoverflow.com/a/7197867/2075875
 */
@SuppressWarnings("unused")
public class AutoFitText extends TextView {

    private static final String TAG = AutoFitText.class.getSimpleName();

    /** Global min and max for text size. Remember: values are in pixels! */
    private final int MIN_TEXT_SIZE = 10;
    private final int MAX_TEXT_SIZE = 400;

    /** Flag for singleLine */
    private boolean mSingleLine = false;

    /**
     * A dummy {@link TextView} to test the text size without actually showing anything to the user
     */
    private TextView mTestView;

    /**
     * A dummy {@link Paint} to test the text size without actually showing anything to the user
     */
    private Paint mTestPaint;

    /**
     * Scaling factor for fonts. It's a method of calculating independently (!) from the actual density of the screen that is used so users have the
     * same experience on different devices. We will use DisplayMetrics in the Constructor to get the value of the factor and then calculate SP from
     * pixel values
     */
    private float mScaledDensityFactor;

    /**
     * Defines how close we want to be to the factual size of the Text-field. Lower values mean higher precision but also exponentially higher
     * computing cost (more loop runs)
     */
    private final float mThreshold = 0.5f;

    /**
     * Constructor for call without attributes --> invoke constructor with AttributeSet null
     * 
     * @param context
     */
    public AutoFitText(Context context) {
        this(context, null);
    }

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

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

    private void init(Context context, AttributeSet attrs) {
        //TextViewPlus part https://stackoverflow.com/a/7197867/2075875
        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.AutoFitText);
        String customFont = a.getString(R.styleable.AutoFitText_customFont);
        setCustomFont(context, customFont);
        a.recycle();

        // AutoFitText part
        mScaledDensityFactor = context.getResources().getDisplayMetrics().scaledDensity;
        mTestView = new TextView(context);

        mTestPaint = new Paint();
        mTestPaint.set(this.getPaint());

        this.getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {

            @Override
            public void onGlobalLayout() {
                // make an initial call to onSizeChanged to make sure that refitText is triggered
                onSizeChanged(AutoFitText.this.getWidth(), AutoFitText.this.getHeight(), 0, 0);
                // Remove the LayoutListener immediately so we don't run into an infinite loop
                //AutoFitText.this.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                removeOnGlobalLayoutListener(AutoFitText.this, this);
            }
        });
    }

    public boolean setCustomFont(Context ctx, String asset) {
        Typeface tf = null;
        try {
        tf = Typeface.createFromAsset(ctx.getAssets(), asset);  
        } catch (Exception e) {
            LOG.e(TAG, "Could not get typeface: "+e.getMessage());
            return false;
        }

        setTypeface(tf);  
        return true;
    }

    @SuppressLint("NewApi")
    public static void removeOnGlobalLayoutListener(View v, ViewTreeObserver.OnGlobalLayoutListener listener){
        if (Build.VERSION.SDK_INT < 16) {
            v.getViewTreeObserver().removeGlobalOnLayoutListener(listener);
        } else {
            v.getViewTreeObserver().removeOnGlobalLayoutListener(listener);
        }
    }

    /**
     * Main method of this widget. Resizes the font so the specified text fits in the text box assuming the text box has the specified width. This is
     * done via a dummy text view that is refit until it matches the real target width and height up to a certain threshold factor
     * 
     * @param targetFieldWidth The width that the TextView currently has and wants filled
     * @param targetFieldHeight The width that the TextView currently has and wants filled
     */
    private void refitText(String text, int targetFieldWidth, int targetFieldHeight) {

        // Variables need to be visible outside the loops for later use. Remember size is in pixels
        float lowerTextSize = MIN_TEXT_SIZE;
        float upperTextSize = MAX_TEXT_SIZE;

        // Force the text to wrap. In principle this is not necessary since the dummy TextView
        // already does this for us but in rare cases adding this line can prevent flickering
        this.setMaxWidth(targetFieldWidth);

        // Padding should not be an issue since we never define it programmatically in this app
        // but just to to be sure we cut it off here
        targetFieldWidth = targetFieldWidth - this.getPaddingLeft() - this.getPaddingRight();
        targetFieldHeight = targetFieldHeight - this.getPaddingTop() - this.getPaddingBottom();

        // Initialize the dummy with some params (that are largely ignored anyway, but this is
        // mandatory to not get a NullPointerException)
        mTestView.setLayoutParams(new LayoutParams(targetFieldWidth, targetFieldHeight));

        // maxWidth is crucial! Otherwise the text would never line wrap but blow up the width
        mTestView.setMaxWidth(targetFieldWidth);

        if (mSingleLine) {
            // the user requested a single line. This is very easy to do since we primarily need to
            // respect the width, don't have to break, don't have to measure...

            /*************************** Converging algorithm 1 ***********************************/
            for (float testSize; (upperTextSize - lowerTextSize) > mThreshold;) {

                // Go to the mean value...
                testSize = (upperTextSize + lowerTextSize) / 2;

                mTestView.setTextSize(TypedValue.COMPLEX_UNIT_SP, testSize / mScaledDensityFactor);
                mTestView.setText(text);
                mTestView.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);

                if (mTestView.getMeasuredWidth() >= targetFieldWidth) {
                    upperTextSize = testSize; // Font is too big, decrease upperSize
                } else {
                    lowerTextSize = testSize; // Font is too small, increase lowerSize
                }
            }
            /**************************************************************************************/

            // In rare cases with very little letters and width > height we have vertical overlap!
            mTestView.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);

            if (mTestView.getMeasuredHeight() > targetFieldHeight) {
                upperTextSize = lowerTextSize;
                lowerTextSize = MIN_TEXT_SIZE;

                /*************************** Converging algorithm 1.5 *****************************/
                for (float testSize; (upperTextSize - lowerTextSize) > mThreshold;) {

                    // Go to the mean value...
                    testSize = (upperTextSize + lowerTextSize) / 2;

                    mTestView.setTextSize(TypedValue.COMPLEX_UNIT_SP, testSize / mScaledDensityFactor);
                    mTestView.setText(text);
                    mTestView.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);

                    if (mTestView.getMeasuredHeight() >= targetFieldHeight) {
                        upperTextSize = testSize; // Font is too big, decrease upperSize
                    } else {
                        lowerTextSize = testSize; // Font is too small, increase lowerSize
                    }
                }
                /**********************************************************************************/
            }
        } else {

            /*********************** Converging algorithm 2 ***************************************/
            // Upper and lower size converge over time. As soon as they're close enough the loop
            // stops
            // TODO probe the algorithm for cost (ATM possibly O(n^2)) and optimize if possible
            for (float testSize; (upperTextSize - lowerTextSize) > mThreshold;) {

                // Go to the mean value...
                testSize = (upperTextSize + lowerTextSize) / 2;

                // ... inflate the dummy TextView by setting a scaled textSize and the text...
                mTestView.setTextSize(TypedValue.COMPLEX_UNIT_SP, testSize / mScaledDensityFactor);
                mTestView.setText(text);

                // ... call measure to find the current values that the text WANTS to occupy
                mTestView.measure(MeasureSpec.UNSPECIFIED, MeasureSpec.UNSPECIFIED);
                int tempHeight = mTestView.getMeasuredHeight();
                // int tempWidth = mTestView.getMeasuredWidth();

                // LOG.debug("Measured: " + tempWidth + "x" + tempHeight);
                // LOG.debug("TextSize: " + testSize / mScaledDensityFactor);

                // ... decide whether those values are appropriate.
                if (tempHeight >= targetFieldHeight) {
                    upperTextSize = testSize; // Font is too big, decrease upperSize
                } else {
                    lowerTextSize = testSize; // Font is too small, increase lowerSize
                }
            }
            /**************************************************************************************/

            // It is possible that a single word is wider than the box. The Android system would
            // wrap this for us. But if you want to decide fo yourself where exactly to break or to
            // add a hyphen or something than you're going to want to implement something like this:
            mTestPaint.setTextSize(lowerTextSize);
            List<String> words = new ArrayList<String>();

            for (String s : text.split(" ")) {
                Log.i("tag", "Word: " + s);
                words.add(s);
            }
            for (String word : words) {
                if (mTestPaint.measureText(word) >= targetFieldWidth) {
                    List<String> pieces = new ArrayList<String>();
                    // pieces = breakWord(word, mTestPaint.measureText(word), targetFieldWidth);

                    // Add code to handle the pieces here...
                }
            }
        }

        /**
         * We are now at most the value of threshold away from the actual size. To rather undershoot than overshoot use the lower value. To match
         * different screens convert to SP first. See {@link http://developer.android.com/guide/topics/resources/more-resources.html#Dimension} for
         * more details
         */
        this.setTextSize(TypedValue.COMPLEX_UNIT_SP, lowerTextSize / mScaledDensityFactor);
        return;
    }

    /**
     * This method receives a call upon a change in text content of the TextView. Unfortunately it is also called - among others - upon text size
     * change which means that we MUST NEVER CALL {@link #refitText(String)} from this method! Doing so would result in an endless loop that would
     * ultimately result in a stack overflow and termination of the application
     * 
     * So for the time being this method does absolutely nothing. If you want to notify the view of a changed text call {@link #setText(CharSequence)}
     */
    @Override
    protected void onTextChanged(CharSequence text, int start, int lengthBefore, int lengthAfter) {
        // Super implementation is also intentionally empty so for now we do absolutely nothing here
        super.onTextChanged(text, start, lengthBefore, lengthAfter);
    }

    @Override
    protected void onSizeChanged(int width, int height, int oldWidth, int oldHeight) {
        if (width != oldWidth && height != oldHeight) {
            refitText(this.getText().toString(), width, height);
        }
    }

    /**
     * This method is guaranteed to be called by {@link TextView#setText(CharSequence)} immediately. Therefore we can safely add our modifications
     * here and then have the parent class resume its work. So if text has changed you should always call {@link TextView#setText(CharSequence)} or
     * {@link TextView#setText(CharSequence, BufferType)} if you know whether the {@link BufferType} is normal, editable or spannable. Note: the
     * method will default to {@link BufferType#NORMAL} if you don't pass an argument.
     */
    @Override
    public void setText(CharSequence text, BufferType type) {

        int targetFieldWidth = this.getWidth();
        int targetFieldHeight = this.getHeight();

        if (targetFieldWidth <= 0 || targetFieldHeight <= 0 || text.equals("")) {
            // Log.v("tag", "Some values are empty, AutoFitText was not able to construct properly");
        } else {
            refitText(text.toString(), targetFieldWidth, targetFieldHeight);
        }
        super.setText(text, type);
    }

    /**
     * TODO add sensibility for {@link #setMaxLines(int)} invocations
     */
    @Override
    public void setMaxLines(int maxLines) {
        // TODO Implement support for this. This could be relatively easy. The idea would probably
        // be to manipulate the targetHeight in the refitText-method and then have the algorithm do
        // its job business as usual. Nonetheless, remember the height will have to be lowered
        // dynamically as the font size shrinks so it won't be a walk in the park still
        if (maxLines == 1) {
            this.setSingleLine(true);
        } else {
            throw new UnsupportedOperationException("MaxLines != 1 are not implemented in AutoFitText yet, use TextView instead");
        }
    }

    @Override
    public void setSingleLine(boolean singleLine) {
        // save the requested value in an instance variable to be able to decide later
        mSingleLine = singleLine;
        super.setSingleLine(singleLine);
    }
}

既知のバグ:Android 4.03では動作しません-フォントが表示されないか非常に小さい(元のアバランチャも動作しません)以下は、そのバグの回避策です:https : //stackoverflow.com/a/21851239/2075875


0

これを試して

TextWatcher changeText = new TextWatcher() {
     @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                tv3.setText(et.getText().toString());
                tv3.post(new Runnable() {           
                    @Override
                    public void run() {
                    while(tv3.getLineCount() >= 3){                     
                            tv3.setTextSize((tv3.getTextSize())-1);                     
                        }
                    }
                });
            }

            @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) {}

            @Override public void afterTextChanged(Editable s) { }
        };

テキストサイズを変更するたびに、ブロックで終了した後にのみ描画が発生するため、これは機能しないと思います(つまり、イベントキューに追加されます)。つまり、ループはテキストが変更されるたびに1回だけ機能し、行数がテキストの3行を超えないことは保証されません。
Android開発者

0

もっと簡単なものを探しているなら:

 public MyTextView extends TextView{

    public void resize(String text, float textViewWidth, float textViewHeight) {
       Paint p = new Paint();
       Rect bounds = new Rect();
       p.setTextSize(1);
       p.getTextBounds(text, 0, text.length(), bounds);
       float widthDifference = (textViewWidth)/bounds.width();
       float heightDifference = (textViewHeight);
       textSize = Math.min(widthDifference, heightDifference);
       setTextSize(TypedValue.COMPLEX_UNIT_PX, textSize);
}

非常に短いように見えますが、私が用意したテストに合格することはできますか?また、この関数が呼び出されても取得できません。
Android開発者

いいえ、それはまだ提案です。できるだけ早くテストに合格できるように改善します。
サンダー

それが提案である場合、Githubに直接投稿していただけますか?
Android開発者

0

@Malachiaszで説明されている問題のクイックフィックス

自動サイズ変更クラスでこれに対するカスタムサポートを追加することで問題を修正しました:

public void setTextCompat(final CharSequence text) {
    setTextCompat(text, BufferType.NORMAL);
}

public void setTextCompat(final CharSequence text, BufferType type) {
    // Quick fix for Android Honeycomb and Ice Cream Sandwich which sets the text only on the first call
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB_MR1 &&
        Build.VERSION.SDK_INT <= Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1) {
        super.setText(DOUBLE_BYTE_WORDJOINER + text + DOUBLE_BYTE_WORDJOINER, type);
    } else {
        super.setText(text, type);
    }
}

@Override
public CharSequence getText() {
    String originalText = super.getText().toString();
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB_MR1 &&
        Build.VERSION.SDK_INT <= Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1) {
        // We try to remove the word joiners we added using compat method - if none found - this will do nothing.
        return originalText.replaceAll(DOUBLE_BYTE_WORDJOINER, "");
    } else {
        return originalText;
    }
}

yourView.setTextCompat(newTextValue)代わりに電話するだけyourView.setText(newTextValue)


なぜsetTextをオーバーライドする代わりに、setTextCompatの新しい関数を作成するのですか?また、どの問題?
android開発者

setText()はTextViewの最後のメソッドであるため、オーバーライドすることはできません。別のオプションは、カスタムTextViewの外でこれを行うことですが、このソリューションは、TextView内で使用するためのものです。
Ionut Negru

ではない正確に。「setText」の一部はプライベートであり、一部はパブリックであり、一部はパブリックではありません。ほとんどはpublic void setText(CharSequence text、BufferType type)をオーバーライドすることで処理できるようです。目的はわかりませんが、最終的な関数があります:public final void setText(char [] text、int start、int len)。たぶんコードをより深く見てください。
android開発者

これらのsetText()のすべてのバリエーションは、実際には最後にsetText(CharSequence text)メソッドを呼び出します。同じ動作を保証したい場合は、そのメソッドをオーバーライドする必要があります。それ以外の場合は、独自のカスタムsetText()メソッドを追加するほうがはるかに優れています。
Ionut Negru 2016年

はい、しかしほとんどの場合、それで問題ありません。
Android開発者

0

LayoutParamsand MaxWidthMaxHeightをに追加してみてくださいTextView。レイアウトが強制的に親コンテナーを尊重し、オーバーフローしないようにします。

textview.setLayoutParams(new LayoutParams(LinearLayout.MATCH_PARENT,LinearLayout.WRAP_CONTENT));

int GeneralApproxWidthOfContainer = 400;
int GeneralApproxHeightOfContainer = 600;
textview.setMaxWidth(400);
textview.setMaxHeight(600);` 

あなたが何を話しているのか分かりません。これをtextViewサンプルに追加して、時々見られる小さな問題が発生しないようにすることをお勧めしますか?もしそうなら、なぜそれについてgithubウェブサイトに投稿しなかったのですか?
Android開発者

0

Android O以降、xml内のテキストのサイズを自動で変更することが可能です。

https://developer.android.com/preview/features/autosizing-textview.html

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

Android Oでは、TextViewにテキストサイズを自動的に拡大または縮小させて、TextViewの特性と境界に基づいてレイアウトを満たすように指示できます。この設定により、動的コンテンツのあるさまざまな画面でテキストサイズを簡単に最適化できます。

サポートライブラリ26.0ベータは、Android Oより前のAndroidバージョンを実行しているデバイスのTextViewの自動サイズ設定機能を完全にサポートしています。ライブラリは、Android 4.0(APIレベル14)以降をサポートしています。android.support.v4.widgetパッケージには、下位互換性のある方法で機能にアクセスするためのTextViewCompatクラスが含まれています。


悲しいことに、私のテストによると、Google IOのビデオでさえ、フォントのサイズを変更するのではなく、誤って一部の単語を折り返すなどの問題があることに気付きました。私はこれについてここで報告しました:issuetracker.google.com/issues/38468964、そしてそれが私がまだそれを使わない理由です。
Android開発者

0

Androidの公式Autosizing TextViewを試した後、AndroidのバージョンがAndroid 8.0(APIレベル26)より前android.support.v7.widget.AppCompatTextViewかどうかを確認しました。使用する必要があり、サポートライブラリのバージョンが26.0.0以上であることを確認してください。例:

<android.support.v7.widget.AppCompatTextView
    android:layout_width="130dp"
    android:layout_height="32dp"
    android:maxLines="1"
    app:autoSizeMaxTextSize="22sp"
    app:autoSizeMinTextSize="12sp"
    app:autoSizeStepGranularity="2sp"
    app:autoSizeTextType="uniform" />

更新

@ android-developerの返信によると、私はAppCompatActivityソースコードをチェックし、これらの2行を見つけましたonCreate

final AppCompatDelegate delegate = getDelegate(); delegate.installViewFactory();

そして中AppCompatDelegateImplcreateView

    if (mAppCompatViewInflater == null) {
        mAppCompatViewInflater = new AppCompatViewInflater();
    }

AppCompatViewInflaterインフレータビューを使用します。createViewは、AppCompatViewInflater「TextView」にAppCompatTextViewを使用します。

public final View createView(){
    ...
    View view = null;
    switch (name) {
        case "TextView":
            view = new AppCompatTextView(context, attrs);
            break;
        case "ImageView":
            view = new AppCompatImageView(context, attrs);
            break;
        case "Button":
            view = new AppCompatButton(context, attrs);
            break;
    ...
}

私のプロジェクトではを使用しないAppCompatActivityので<android.support.v7.widget.AppCompatTextView>、xmlで使用する必要があります。


1
ほとんどの場合、「TextView」だけを記述する場合、インフレータはAppCompatTextViewを使用するため、そうする必要はありません。
Android開発者

@androiddeveloperしかし、TextViewを使用すると、自動サイズ調整が機能しません。
weei.zh

@androiddeveloperありがとう、理由は私が使っていないことAppCompatActivityです。AppCompatActivityはAppCompatViewInflaterインフレーターレイアウトを使用します。
weei.zh

理解できません
Android開発者、
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.