仅在必要时在特定位置添加换行符

时间:2012-12-05 15:43:43

标签: java android android-layout android-xml

在我的Android版面中,我有一个TextView,它使用屏幕可用宽度的一半。在运行时,我将文本设置为(长)电子邮件地址。例如:

googleandroiddeveloper@gmail.com

如果文本不适合一行,Android会插入换行符,这是所需的行为。但是,换行符的位置在第一个不适合该行的字符之前。结果可能是这样的:

googleandroiddeveloper@gmai
l.com

我认为,这有点难看,特别是在电子邮件地址中。我希望换行符出现在@字符之前:

googleandroiddeveloper
@gmail.com

当然,我可以在\n中添加strings.xml。但是,电子邮件地址在每种情况下都会使用两行,即使它适合一行。

我已经认为我已经找到了在电子邮件地址中添加ZERO WIDTH SPACE(\u200B)的解决方案。

<string name="email">googleandroiddeveloper\u200B@gmail.com</string>

但除了标准空间之外,Android不会将特殊空格字符检测为可破坏空间,因此此时不会添加换行符。

当我在应用程序的多个位置处理大量电子邮件地址时,我正在寻找一种解决方案,在@字符之前添加一个易碎且不可见的空间,因此Android会包装电子邮件地址不适合一行。

2 个答案:

答案 0 :(得分:6)

@ Luksprog的解决方案非常好,在许多情况下解决了这个问题。但是,我在几个点修改了类,使其更好。这些是修改:

  • 我使用onSizeChanged代替onMeasure来检查和操作文本,因为onMeasureLinearLayout一起使用时layout_weight存在问题。
  • 我使用getPaddingLeft()getPaddingRight()
  • 考虑了文字的水平填充
  • 在衡量afterAt时,我将position替换为position + 1,否则生成的电子邮件地址包含两个@

代码:

public class EmailTextView extends TextView {

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

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        // the width the text can use, that is the total width of the view minus
        // the padding
        int availableWidth = w - getPaddingLeft() - getPaddingRight();
        String text = getText().toString();
        if (text.contains("\n@")) {
            // the text already contains a line break before @
            return;
        }
        // get the position of @ in the string
        int position = -1;
        for (int i = 0; i < text.length(); i++) {
            if (text.charAt(i) == '@') {
                position = i;
                break;
            }
        }
        if (position > 0) {
            final Paint pnt = getPaint();
            // measure width before the @ and after the @
            String beforeAt = text.subSequence(0, position).toString();
            String afterAt = text.subSequence(position + 1, text.length())
                    .toString();
            final float beforeAtSize = pnt.measureText(beforeAt);
            final float afterAtSize = pnt.measureText(afterAt);
            final float atSize = pnt.measureText("@");
            if (beforeAtSize > availableWidth) {
                // the text before the @ is bigger than the width
                // so Android will break it
                return;
            } else {
                if ((beforeAtSize + afterAtSize + atSize) <= availableWidth) {
                // the entire text is smaller than the available width
                    return;
                } else {
                    // insert the line break before the @
                    setText(beforeAt + "\n@" + afterAt);
                }
            }
        }
    }
}

以下是EmailTextView与默认TextView相比较的屏幕截图:

EmailTextView

对于所有电子邮件地址,它按预期工作。最后一个地址没有改变,因为@之前的文本已经太宽了,所以系统之前打破了它,因此电子邮件地址无论如何都搞砸了,所以不需要包含另一个换行符

答案 1 :(得分:2)

您可以查看下面的自定义TextView类(尽管可能效率不高),在某些情况下应该插入(假设在极少数测试中)所需的换行符:

public static class NewLineText extends TextView {

    private static final String CHALLANGE_TEXT = "\n@";

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

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        String text = getText().toString();
        if (text.contains(CHALLANGE_TEXT)) {
            return;
        }
        int position = -1;
        for (int i = 0; i < text.length(); i++) {
            if (text.charAt(i) == '@') {
                position = i;
                break;
            }
        }
        if (position > 0) {
            final Paint pnt = getPaint();
            String beforeAt = text.subSequence(0, position).toString();
            String afterAt = text.subSequence(position, text.length())
                    .toString();
            final float beforeAtSize = pnt.measureText(beforeAt);
            final float afterAtSize = pnt.measureText(afterAt);
            final float atSize = pnt.measureText("@");
            if (beforeAtSize > getMeasuredWidth()) {
                // return ?! the text before the @ is bigger than the width
                // so Android will break it
                return;
            } else {
                if ((beforeAtSize + afterAtSize + atSize) <= getMeasuredWidth()) {
                    return;
                } else {
                    setText(beforeAt + CHALLANGE_TEXT + afterAt);
                }
            }
        }
    }
}