android 为什么TextView在多行时有结尾填充?

k4ymrczo  于 2023-01-15  发布在  Android
关注(0)|答案(4)|浏览(111)

如果TextView的layout_width="wrap_content"必须换行到第二行才能包含文本,则它将调整宽度以用完所有可用空间(考虑到边距等)。但是为什么在视图的末尾有填充?我只是告诉它 * wrap_content *,所以它应该 Package 那个内容!这看起来像个bug,这在股票Messenger应用程序的聊天界面中可见。2(图片来自我自己的应用程序。3但在9补丁中绝对没有额外的空间。4)
有什么解决办法吗?
更新:回应者/评论者没有抓住要点。也许我上传的图片是误导性的,因为它是从我的应用程序样式。问题发生在任何TextView,你可以看到,通过样式的背景,视图边界将不再紧张。我上传了一个不同的图像。以下是图像中的TextView的XML:

<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_marginEnd="20dp"
    android:layout_marginStart="20dp"
    android:background="#dddddd"
    android:text="This doesn't wrap"
    android:layout_marginTop="20dp"
    android:layout_marginBottom="20dp"
    android:layout_gravity="center_horizontal"
    />
<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_marginEnd="20dp"
    android:layout_marginStart="20dp"
    android:layout_gravity="center_horizontal"
    android:background="#dddddd"
    android:text="This wraps and look, the bounds does not fit tight against the right edge of text"
    />

uinbv5nw

uinbv5nw1#

我发现选择的答案很有帮助,尽管它没有考虑到填充。我将选择的答案与this post's answer结合起来,得出了一个可以使用填充的视图。仅供参考,另一篇文章的答案有一个缺陷,即视图有时会在末尾被几个像素截断。

public class TightTextView extends TextView {
  public TightTextView(Context context) {
    super(context);
  }

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

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

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

    int specModeW = MeasureSpec.getMode(widthMeasureSpec);
    if (specModeW != MeasureSpec.EXACTLY) {
      Layout layout = getLayout();
      if (layout != null) {
        int w = (int) Math.ceil(getMaxLineWidth(layout)) + getCompoundPaddingLeft() + getCompoundPaddingRight();
        if (w < getMeasuredWidth()) {
          super.onMeasure(MeasureSpec.makeMeasureSpec(w, MeasureSpec.AT_MOST),
          heightMeasureSpec);
        }
      }
    }
  }

  private float getMaxLineWidth(Layout layout) {
    float max_width = 0.0f;
    int lines = layout.getLineCount();
    for (int i = 0; i < lines; i++) {
      if (layout.getLineWidth(i) > max_width) {
        max_width = layout.getLineWidth(i);
      }
    }
    return max_width;
  }
}
polhcujo

polhcujo2#

一开始,看到你的帖子,我以为问题是因为标准的Android TextView在他们的基本样式中定义了一些默认的填充。如果你想删除它,你可以尝试这样做:

android:paddingEnd="0dp"

android:paddingRight="0dp"

由于您的帖子已经更新,我知道您的问题不是来自填充,而是来自换行。确实,当有几行要显示时,Android TextView会在宽度上使用整个可用空间。
正如in this post所述,没有标准的解决方案,您需要自定义您的文本视图,以固定其宽度后,填补它。
如下所示覆盖textViewonMeasure方法应该可以工作(灵感来自“sky”答案):

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

    int specModeW = MeasureSpec.getMode(widthMeasureSpec);
    if (specModeW != MeasureSpec.EXACTLY) {
        Layout layout = getLayout();
        int linesCount = layout.getLineCount();
        if (linesCount > 1) {
            float textRealMaxWidth = 0;
            for (int n = 0; n < linesCount; ++n) {
                textRealMaxWidth = Math.max(textRealMaxWidth, layout.getLineWidth(n));
            }
            int w = (int) Math.ceil(textRealMaxWidth);
            if (w < getMeasuredWidth()) {
                super.onMeasure(MeasureSpec.makeMeasureSpec(w, MeasureSpec.AT_MOST),
                        heightMeasureSpec);
            }
        }
    }
}
cbjzeqam

cbjzeqam3#

最近,我在为一个应用程序开发聊天气泡视图时遇到了类似的问题,所以我使用了公认解决方案中的想法和@hoffware的改进,并在Kotlin中重新实现了它们。

import android.content.Context
import android.util.AttributeSet
import android.view.View.MeasureSpec.*
import androidx.appcompat.widget.AppCompatTextView
import kotlin.math.ceil

class TightTextView
@JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = android.R.attr.textViewStyle
) : AppCompatTextView(context, attrs, defStyleAttr) {

    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec)

        val lineCount = layout.lineCount
        if (lineCount > 1 && getMode(widthMeasureSpec) != EXACTLY) {
            val textWidth = (0 until lineCount).maxOf(layout::getLineWidth)
            val padding = compoundPaddingLeft + compoundPaddingRight
            val w = ceil(textWidth).toInt() + padding

            if (w < measuredWidth) {
                val newWidthMeasureSpec = makeMeasureSpec(w, AT_MOST)
                super.onMeasure(newWidthMeasureSpec, heightMeasureSpec)
            }
        }
    }
}
vsmadaxz

vsmadaxz4#

这肯定是因为“against”不适合放在单词“fit”和右边缘之间。如果您想测试一下,请尝试将文本更改为一堆|每个字符之间有一个空格。

相关问题