`
dowhathowtodo
  • 浏览: 798362 次
文章分类
社区版块
存档分类
最新评论

Android TextView 自动换行问题

 
阅读更多

Android的TextView在显示文字的时候有个问题就是一行还没显示满就跳到下一行,原因是:

1) TextView在显示中文的时候 标点符号不能显示在一行的行首和行尾,如果一个标点符号刚好在一行的行尾,该标点符号就会连同前一个字符跳到下一行显示;

2)一个英文单词不能被显示在两行中( TextView在显示英文时,标点符号是可以放在行尾的,但英文单词也不能分开 );

如果只是想让标点符号可以显示在行尾,有一个简单的方法就是在标点符号后加一个空格,则该标点符号就可以显示在行尾了;

如果想要两端对齐的显示效果,有两种方法:

1)修改Android源代码;将frameworks/base/core/java/android/text下的StaticLayout.java文件中的如下代码:

if (c == ' ' || c == '/t' ||
((c == '.' || c == ',' || c == ':' || c == ';') &&
(j - 1 < here || !Character.isDigit(chs[j - 1 - start])) &&
(j + 1 >= next || !Character.isDigit(chs[j + 1 - start]))) ||
((c == '/' || c == '-') &&
(j + 1 >= next || !Character.isDigit(chs[j + 1 - start]))) ||
(c >= FIRST_CJK && isIdeographic(c, true) &&
j + 1 < next && isIdeographic(chs[j + 1 - start], false))) {
okwidth = w;
ok = j + 1;

if (fittop < oktop)
oktop = fittop;
if (fitascent < okascent)
okascent = fitascent;
if (fitdescent > okdescent)
okdescent = fitdescent;
if (fitbottom > okbottom)
okbottom = fitbottom;
} 


去掉就可以了。去掉后标点符号可以显示在行首和行尾,英文单词也可以被分开在两行中显示。

2)自定义View显示文本

网上就有达人采用自定义View来解决这个问题,我做了实验并总结了一下:

自定义View的步骤:

1)继承View类或其子类,例子继承了TextView类;

2)写构造函数,通过XML获取属性(这一步中可以自定义属性,见例程);

3)重写父类的某些函数,一般都是以on开头的函数,例子中重写了onDraw()和onMeasure()函数;

=========================StartCustomTextView.java=============================

public class StartCustomTextView extends TextView {
    public  static  int m_iTextHeight; //文本的高度
    public  static  int m_iTextWidth;//文本的宽度
    
    private Paint mPaint = null;
    private String string="";
    private float LineSpace = 0;//行间距
    private int left_Margin;
    private int right_Margin;
    private int bottom_Margin;
        
    public StartCustomTextView(Context context, AttributeSet set) 
    {       
        super(context,set);  
        DisplayMetrics displayMetrics = getResources().getDisplayMetrics();
        TypedArray typedArray = context.obtainStyledAttributes(set, R.styleable.CYTextView);
        int width = displayMetrics.widthPixels;
        left_Margin = 29;
        right_Margin = 29;
        bottom_Margin = 29;
        width = width - left_Margin -right_Margin;
        float textsize = typedArray.getDimension(R.styleable.CYTextView_textSize, 34);
        int textcolor = typedArray.getColor(R.styleable.CYTextView_textColor, getResources().getColor(R.color.white));
        float linespace = typedArray.getDimension(R.styleable.CYTextView_lineSpacingExtra, 15);
        int typeface = typedArray.getColor(R.styleable.CYTextView_typeface, 0);
        
        typedArray.recycle();
        
        //设置 CY TextView的宽度和行间距www.linuxidc.com
        m_iTextWidth=width;
        LineSpace=linespace;
        
        // 构建paint对象      
        mPaint = new Paint();
        mPaint.setAntiAlias(true);
        mPaint.setColor(textcolor);
        mPaint.setTextSize(textsize);
        switch(typeface){
        case 0:
            mPaint.setTypeface(Typeface.DEFAULT);
            break;
        case 1:
            mPaint.setTypeface(Typeface.SANS_SERIF);
            break;
        case 2:
            mPaint.setTypeface(Typeface.SERIF);
            break;
        case 3:
            mPaint.setTypeface(Typeface.MONOSPACE);
            break;
        default:
            mPaint.setTypeface(Typeface.DEFAULT);    
            break;
        }
        
    }
  
    @Override
    protected void onDraw(Canvas canvas) 
    {  
       super.onDraw(canvas);       
        
        char ch;
        int w = 0;
        int istart = 0;
        int m_iFontHeight;
        int m_iRealLine=0;
        int x=2;
        int y=30;
        
        Vector   m_String=new Vector();
        
        FontMetrics fm = mPaint.getFontMetrics();        
        m_iFontHeight = (int) Math.ceil(fm.descent - fm.top) + (int)LineSpace;//计算字体高度(字体高度+行间距)

        for (int i = 0; i < string.length(); i++)
        {
            ch = string.charAt(i);
            float[] widths = new float[1];
            String srt = String.valueOf(ch);
            mPaint.getTextWidths(srt, widths);

            if (ch == '\n'){
                m_iRealLine++;
                m_String.addElement(string.substring(istart, i));
                istart = i + 1;
                w = 0;
            }else{
                w += (int) (Math.ceil(widths[0]));
                if (w > m_iTextWidth){
                    m_iRealLine++;
                    m_String.addElement(string.substring(istart, i));
                    istart = i;
                    i--;
                    w = 0;
                }else{
                    if (i == (string.length() - 1)){
                        m_iRealLine++;
                        m_String.addElement(string.substring(istart, string.length()));
                    }
                }
            }
        }
        m_iTextHeight=m_iRealLine*m_iFontHeight+2;
        canvas.setViewport(m_iTextWidth, m_iTextWidth);
        for (int i = 0, j = 0; i < m_iRealLine; i++, j++)
        {
            canvas.drawText((String)(m_String.elementAt(i)), x,  y+m_iFontHeight * j, mPaint);
        }
    }  
   
    
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) 
    {          
        int measuredHeight = measureHeight(heightMeasureSpec);          
        int measuredWidth = measureWidth(widthMeasureSpec);           
        this.setMeasuredDimension(measuredWidth, measuredHeight);
        LayoutParams layout = new LinearLayout.LayoutParams(measuredWidth,measuredHeight);
        layout.leftMargin= left_Margin;
        layout.rightMargin= right_Margin;
        layout.bottomMargin= bottom_Margin;
        this.setLayoutParams(layout);
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }  
                 
    private int measureHeight(int measureSpec) 
    {  
        int specMode = MeasureSpec.getMode(measureSpec);          
        int specSize = MeasureSpec.getSize(measureSpec);                   
        // Default size if no limits are specified.  
        initHeight();
        int result = m_iTextHeight;          
        if (specMode == MeasureSpec.AT_MOST){         
            // Calculate the ideal size of your          
            // control within this maximum size.          
            // If your control fills the available           
            // space return the outer bound.          
            result = specSize;           
        }else if (specMode == MeasureSpec.EXACTLY){           
            // If your control can fit within these bounds return that value.            
//            result = specSize;           
        }     
        return result;            
    }
    
    private void initHeight()
    {
        //设置 CY TextView的初始高度为0
        m_iTextHeight=0;
        
        //大概计算 CY TextView所需高度
        FontMetrics fm = mPaint.getFontMetrics();        
        int m_iFontHeight = (int) Math.ceil(fm.descent - fm.top) + (int)LineSpace;
        int line=0;
        int istart=0;
        
        int w=0;
        for (int i = 0; i < string.length(); i++)
        {
            char ch = string.charAt(i);
            float[] widths = new float[1];
            String srt = String.valueOf(ch);
            mPaint.getTextWidths(srt, widths);

            if (ch == '\n'){
                line++;
                istart = i + 1;
                w = 0;
            }else{
                w += (int) (Math.ceil(widths[0]));
                if (w > m_iTextWidth){
                    line++;
                    istart = i;
                    i--;
                    w = 0;
                }else{
                    if (i == (string.length() - 1)){
                        line++;
                    }
                }
            }
        }
        m_iTextHeight=(line)*m_iFontHeight+2;
    }
                 
    private int measureWidth(int measureSpec) 
    {  
        int specMode = MeasureSpec.getMode(measureSpec);           
        int specSize = MeasureSpec.getSize(measureSpec);             

        // Default size if no limits are specified.          
        int result = 500;          
        if (specMode == MeasureSpec.AT_MOST){          
            // Calculate the ideal size of your control           
            // within this maximum size.         
            // If your control fills the available space         
            // return the outer bound.         
            result = specSize;          
        }else if (specMode == MeasureSpec.EXACTLY){           
            // If your control can fit within these bounds return that value.           
            result = specSize;            
        }           
        return result;          
    }
    
    public void SetText(String text)
    {
        string = text;
       // requestLayout();
       // invalidate();
    }   
} 


=======================attrs.xml===============================

该文件是自定义的属性,放在工程的res/values下

<resources>
<attr name="textwidth" format="integer"/>
<attr name="typeface">
<enum name="normal" value="0"/>
<enum name="sans" value="1"/>
<enum name="serif" value="2"/>
<enum name="monospace" value="3"/>
</attr>

<declare-styleable name="CYTextView"> 
<attr name="textwidth" /> 
<attr name="textSize" format="dimension"/>
<attr name="textColor" format="reference|color"/>
<attr name="lineSpacingExtra" format="dimension"/>
<attr name="typeface" />
</declare-styleable>
</resources> 


=======================main.xml==========================

<?xml version="1.0" encoding="utf-8"?>
<ScrollView
xmlns:Android="http://schemas.android.com/apk/res/android"
Android:layout_width="320px"
Android:layout_height="320px"
Android:background="#ffffffff"
>
<LinearLayout 
xmlns:Android="http://schemas.android.com/apk/res/android"
Android:orientation="vertical"
Android:layout_width="fill_parent"
Android:layout_height="fill_parent">
<com.cy.CYTextView.CYTextView 
xmlns:cy="http://schemas.Android.com/apk/res/ com.cy.CYTextView "
Android:id="@+id/mv"
Android:layout_height="wrap_content"
Android:layout_width="wrap_content" 
cy :textwidth="320" 
cy :textSize="24sp"
cy :textColor="#aa000000"
cy :lineSpacingExtra="15sp"
cy :typeface="serif">
</com. cy .CYTextView.CYTextView> 
</LinearLayout>
</ScrollView>

蓝色代码即为自定义View,其中以cy命名空间开头的属性是自定义属性;

=======================Main.java=============================

public class Main extends Activity {
CYTextView mCYTextView;
String text = "Android提供了精巧和有力的组件化模型构建用户的UI部分。主要是基于布局类:View和 ViewGroup。在此基础上,android平台提供了大量的预制的View和xxxViewGroup子 类,即布局(layout)和窗口小部件(widget)。可以用它们构建自己的UI。";

/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.setContentView(R.layout.main);

mCYTextView = (CYTextView)findViewById(R.id.mv);
mCYTextView.SetText(text);
}

} 


转自:http://hi.baidu.com/java_rose/blog/item/2940a030d1ec7f3e96ddd847.html

分享到:
评论

相关推荐

    TextView解决自动换行问题

    本文将详细讲解如何在`TextView`中解决自动换行问题,并设置超过指定行数后使用省略号。 首先,我们需要了解`TextView`的一些关键属性: 1. `android:singleLine`:此属性在API 26及以下版本可用,当设置为`true`...

    Android开发TextView排版问题取消自动换行

    本文将详细讲解如何解决TextView的自动换行问题,并结合提供的`CustomTextView.java`源码和`textattr.xml`资源文件来深入理解这一技术。 首先,了解TextView的基本属性。TextView默认会根据其宽度自动进行换行,以...

    TextView异常换行

    4. **布局约束**:`TextView`周围的约束,如父布局的宽度、边距等,也可能导致换行问题。 5. **动态加载内容**:当`TextView`的内容是动态加载时,如果加载过程中的数据格式或者处理不当,也可能造成换行异常。 ...

    解决TextView异常换行

    用TextView显示的时候,会出现一个问题就是显示的文字中有括号、数字、或者英文的话他会立刻就换行,因为括号、数字等是半角,而汉字是全角,所以我们需要把半角转换为全角,使它们与汉字同占两个字节,现在已经解决...

    Android textView根据长度自动换行-IT计算机-毕业设计.zip

    以下是一些关于TextView自动换行的关键知识点: 1. **最大行数限制**:通过设置`setMaxLines(int maxLines)`,可以限制TextView显示的最大行数。一旦达到这个数量,TextView将会截断超出的内容并显示省略号。 2. *...

    TextView解决中英文换行bug和点击显示全部功能

    在Android开发中,...通过学习和理解这些代码,开发者可以更好地掌握解决TextView换行问题和动态显示全部内容的方法,提高应用的用户体验。记得在实际项目中测试各种情况,确保解决方案在不同场景下都能正常工作。

    android textview 文字排版 换行

    3. **多行显示**:默认情况下,`TextView`会根据内容自动换行。若需禁止换行,可以设置`android:breakStrategy`和`android:hyphenationFrequency`属性来调整换行策略。 ### 三、文字排版高级技巧 1. **行间距与...

    Android高级应用源码-textView根据长度自动换行.zip

    这个"Android高级应用源码-textView根据长度自动换行.zip"压缩包很可能是包含了一个示例项目,演示了如何让TextView在文本超出其宽度时自动换行,以适应不同屏幕尺寸和布局需求。 首先,我们来了解TextView的基本...

    Android-TextView换行排版混乱-允许设置SpannableString

    本文将深入探讨如何解决`TextView`自动换行导致的排版问题,并允许我们设置`SpannableString`以实现更多样化的文本效果。 首先,理解`TextView`的换行机制至关重要。`TextView`默认会根据其宽度自动进行文本换行,...

    TextView自动换行源代码

    本类实现了文字的竖直排版显示(中国古时的行文形式),虽然仍有许多特效及功能仍未实现,但基本的使用已经能满足。版权:尽管放心用吧,可以自行随意改进转载和使用,转载时请保留这段文字即可另特别感谢 老僧xp ...

    安卓Android源码——textView根据长度自动换行.zip

    这个压缩包"安卓Android源码——textView根据长度自动换行.zip"很可能包含了一个示例项目或代码片段,演示了如何在TextView中实现根据文本长度自动换行的功能。 在Android的TextView中,自动换行是默认开启的。当你...

    android 逐行显示自动换行的textview

    在Android开发中,文本视图(TextView)是用于展示文本的基本组件。然而,有时我们可能需要实现更复杂的显示效果,比如在应用中播放通知字幕时,逐行逐字显示文本,这通常用于模仿电视字幕或者游戏中的对话效果。在...

    Android自动换行标签控件LineBreakLayout

    `LineBreakLayout`是一个专门用于实现自动换行标签效果的自定义控件,它解决了在一个有限的宽度内显示多个标签,并且能根据屏幕大小自动调整布局,使标签能够整齐、美观地换行显示。在本文中,我们将深入探讨`...

    Android LinearLayout实现自动换行

    Android LinearLayout 实现自动换行 Android LinearLayout 实现自动换行是 Android 开发中常见的需求。LinearLayout 是 Android 中最常用的布局控件之一,但它有一个缺陷,就是不能自动换行。今天,我们将详细介绍...

    Android自动换行标签控件(二)

    在这个场景中,我们关注的是一个名为"Android自动换行标签控件(二)"的主题,它涉及到如何利用`ViewGroup`来实现一个可以自动换行的标签控件。这样的控件通常用于展示多条分类标签,例如新闻类别或者产品属性,当...

    Android 标签,可以自动换行的ViewGroup

    总之,"Android 标签,可以自动换行的ViewGroup"是Android UI设计中一个重要的概念,它通过自动换行的能力提高了界面的适应性和用户体验。开发者可以利用现有的开源库或自定义ViewGroup来实现这一功能,以创建更加...

    Android 实现View的排列自动换行

    总之,“Android实现View的排列自动换行”是一个涉及自定义View组、测量、布局和性能优化的综合性问题。通过理解并实现这样的功能,开发者能够更好地掌握Android UI的底层机制,从而设计出更加灵活和个性化的用户...

    解决 TextView 中文、英文、数字、符号 排版问题

    然而,当TextView中包含中文、英文、数字和符号时,可能会出现排版问题,如字符间距离不均、换行混乱等。为了解决这些问题,开发者需要对TextView的属性进行调整或使用特定的解决方案。 首先,我们要理解Android...

    Android-MultipleTextView是一个通过重写TextView实现去除原生默认内边距并扩展了一些其他功能

    5. **动态换行**:在一些情况下,开发者可能希望文本在达到特定宽度后自动换行,`MultipleTextView`可能提供了这样的功能,以适应不同屏幕尺寸和布局变化。 6. **文本测量优化**:为了提高性能,`MultipleTextView`...

    自定义textview显示文字对齐到最右侧才换行

    然而,系统默认的TextView在文本达到右边界时会自动换行,这在某些情况下可能不符合设计需求。标题和描述提到的问题,即如何实现“自定义TextView显示文字对齐到最右侧才换行”,是一个典型的布局定制问题。为了解决...

Global site tag (gtag.js) - Google Analytics