美文网首页Android开发我收藏的Android开发文章Android开发之Kotlin
TextView设置文字包含中英文时自动换行问题的终极解决方案

TextView设置文字包含中英文时自动换行问题的终极解决方案

作者: SiberianDante | 来源:发表于2018-03-01 17:45 被阅读369次

    情景,正常TextView中设置文本内容中包含中英文时会造成自动换行的问题,影响界面显示效果,如图:

    sd_string_error.png

    网上很多解决途径,甚至有多三方框架处理,但是效果并不能达到,最终是要如下代码完美解决,效果图如下:

    sd_string_right.png

    具体实现过程 以及代码

    基本思路:先测量TextView的最大可用宽度,然后替换所有的空格符并按行分割,如果小于TextView最大宽度,则不处理;如果大于TextView最大宽度,进行单个字符进行测量,超过最大宽度则加入换行符;

    public class SDAdaptiveTextView extends TextView {
        public SDAdaptiveTextView(Context context) {
            super(context);
        }
    
        public SDAdaptiveTextView(Context context, @Nullable AttributeSet attrs) {
            super(context, attrs);
        }
    
        public SDAdaptiveTextView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
        }
    
        /**
         * 使用该方法设置TextView的文本内容,改方法不能再主线程中执行
         * @param text
         */
        public void setAdaptiveText(String text) {
            this.setText(text);
            this.setText(adaptiveText(this));
        }
        
        private String adaptiveText(final TextView textView) {
            final String originalText = textView.getText().toString(); //原始文本
            final Paint tvPaint = textView.getPaint();//获取TextView的Paint
            final float tvWidth = textView.getWidth() - textView.getPaddingLeft() - textView.getPaddingRight(); //TextView的可用宽度
            //将原始文本按行拆分
            String[] originalTextLines = originalText.replaceAll("\r", "").split("\n");
            StringBuilder newTextBuilder = new StringBuilder();
            for (String originalTextLine : originalTextLines) {
                //文本内容小于TextView宽度,即不换行,不作处理
                if (tvPaint.measureText(originalTextLine) <= tvWidth) {
                    newTextBuilder.append(originalTextLine);
                } else {
                    //如果整行宽度超过控件可用宽度,则按字符测量,在超过可用宽度的前一个字符处手动换行
                    float lineWidth = 0;
                    for (int i = 0; i != originalTextLine.length(); ++i) {
                        char charAt = originalTextLine.charAt(i);
                        lineWidth += tvPaint.measureText(String.valueOf(charAt));
                        if (lineWidth <= tvWidth) {
                            newTextBuilder.append(charAt);
                        } else {
                            //单行超过TextView可用宽度,换行
                            newTextBuilder.append("\n");
                            lineWidth = 0;
                            --i;//该代码作用是将本轮循环回滚,在新的一行重新循环判断该字符
                        }
                    }
                }
            }
            return newTextBuilder.toString();
        }
    }
    
    • 使用 setAdaptiveText 方法替代 原生的 setText 方法,注意该方法不能再主线程中执行
    • 如果TextView宽度设置为WrapContent,为了测量它的准确宽度,可先使用setText()方法设值,再调用setAdaptiveText()设值
      举例使用:
            tvSdAdaptive.post(new Runnable() {
                @Override
                public void run() {
                    tvSdAdaptive.setAdaptiveText("");
                }
            });
    
            tvSdAdaptive.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
                @Override
                public void onGlobalLayout() {
                    tvSdAdaptive.setAdaptiveText("");
                    tvSdAdaptive.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                }
            });
    

    了解更多——GitHub链接

    相关文章

      网友评论

      • 蓅哖伊人为谁笑:这种方式有点耗性能,如果在列表中表现得会更加明显,还有种方式是改变text 半角全角,亲测可以呢
        SiberianDante:为什么我转化的不起作用呢:sweat:

      本文标题:TextView设置文字包含中英文时自动换行问题的终极解决方案

      本文链接:https://www.haomeiwen.com/subject/pllnxftx.html