Android TextView自動換行文字排版參差不齊的原因
1、半形字元與全形字元混亂所致:這種情況一般就是漢字與數字、英文字母混用
解決方法一:
將textview中的字元全形化。即將所有的數字、字母及標點全部轉為全形字元,使它們與漢字同佔兩個位元組,這樣就可以避免由於佔位導致的排版混亂問題了。 半形轉為全形的程式碼如下,只需呼叫即可。
- public static String ToDBC(String input) {
- char[] c = input.toCharArray();
- for (int i = 0; i< c.length; i++) {
- if (c[i] == 12288) {
- c[i] = (char) 32;
- continue;
- }if (c[i]> 65280&& c[i]< 65375)
- c[i] = (char) (c[i] - 65248);
- }
- return new String(c);
- }
解決方法二:
去除特殊字元或將所有中文標號替換為英文標號。利用正規表示式將所有特殊字元過濾,或利用replaceAll()將中文標號替換為英文標號。則轉化之後,則可解決排版混亂問題。
- // 替換、過濾特殊字元
- public static String StringFilter(String str) throws PatternSyntaxException{
- str=str.replaceAll("【","[").replaceAll("】","]").replaceAll("!","!");//替換中文標號
- String regEx="[『』]"; // 清除掉特殊字元
- Pattern p = Pattern.compile(regEx);
- Matcher m = p.matcher(str);
- return m.replaceAll("").trim();
- }
2、TextView在顯示中文的時候標點符號不能顯示在一行的行首和行尾,如果一個標點符號剛好在一行的行尾,該標點符號就會連同前一個字元跳到下一行顯示。
解決方法:在標點符號後加一個空格。
3、一個英文單詞不能被顯示在兩行中( TextView在顯示英文時,標點符號是可以放在行尾的,但英文單詞也不能分開 )。
4、如果要兩行對其的顯示效果:有兩種方法
方法一:
修改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;
- }
去掉就可以了。去掉後標點符號可以顯示在行首和行尾,英文單詞也可以被分開在兩行中顯示。
方法二:
自定義View顯示文字
網上就有達人採用自定義View來解決這個問題,我做了實驗並總結了一下:
自定義View的步驟:
1)繼承View類或其子類,例子繼承了TextView類;
2)寫建構函式,通過XML獲取屬性(這一步中可以自定義屬性,見例程);
3)重寫父類的某些函式,一般都是以on開頭的函式,例子中重寫了onDraw()和onMeasure()函式;
=========================CYTextView.java=============================
- public class CYTextView extends TextView {
- public static int m_iTextHeight; //文字的高度
- public static int m_iTextWidth;//文字的寬度
- private Paint mPaint = null;
- private String string="";
- private float LineSpace = 0;//行間距
- public CYTextView(Context context, AttributeSet set)
- {
- super(context,set);
- TypedArray typedArray = context.obtainStyledAttributes(set, R.styleable.CYTextView);
- int width = typedArray.getInt(R.styleable. CY TextView_textwidth, 320);
- float textsize = typedArray.getDimension(R.styleable. CY TextView_textSize, 24);
- int textcolor = typedArray.getColor(R.styleable. CY TextView_textColor, -1442840576);
- float linespace = typedArray.getDimension(R.styleable. CY TextView_lineSpacingExtra, 15);
- int typeface = typedArray.getColor(R.styleable. CY TextView_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);
- this.setLayoutParams(new LinearLayout.LayoutParams(measuredWidth,measuredHeight));
- 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)(//注:此函式目前只有在UI執行緒中呼叫才可以把文字畫出來,在其它執行緒中<p> //無法畫文字,找了好久找不到原因,求高手解答)
- {
- string = text;
- // requestLayout();
- // invalidate();
- }
- }</p>
=======================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。";
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- this.setContentView(R.layout.main);
- mCYTextView = (CYTextView)findViewById(R.id.mv);
- mCYTextView.SetText(text);
- }
- }
相關文章
- TextView 自動換行,每行排滿的自定義TextViewTextView
- Android撩妹特效系列!仿instagram文字自動排版功能實現!Android特效
- Android TextView 在指定位置自動省略字元AndroidTextView字元
- Android TextView 富文字之 android.text.style.xxxSpanAndroidTextView
- css實現強制不換行/自動換行/強制換行CSS
- TextView、TextView的子類之EditText、EditText的子類之自動完成文字框(AutoCompleteTextView)的功能與用法TextView
- 如何讓文字居右對齊,換行後又居左對齊
- css中實現強制不換行/自動換行/強制換行CSS
- 教你如何實現 Android TextView 文字輪播效果AndroidTextView
- 短視訊商城系統,Android TextView自動調整字型大小AndroidTextView
- android炫酷的textviewAndroidTextView
- SVG 文字排版SVG
- Android之TextView設定drawableRight等圖片文字間隔AndroidTextView
- Android開發筆記——TextView文字設定不同顏色Android筆記TextView
- Android入門教程 | TextView簡介(寬高、文字、間距)AndroidTextView
- Android Textview 一行居中 兩行居左AndroidTextView
- Android更換APP字型—TextView各種字型樣式AndroidAPPTextView
- 織夢多行文字自定義欄位 支援自動換行
- pycharm怎麼自動排版PyCharm
- 【Android初級】使用TypeFace設定TextView的文字字型(附原始碼)AndroidTextView原始碼
- css li元素中的文字超出隱藏不換行效果CSS
- 短視訊系統原始碼,動態輸入文字到達行上限自動換行原始碼
- WPS/Word中公式與文字不對齊的問題公式
- 【微信小程式】微信小程式 文字過長,自動換行的問題微信小程式
- Android 設定TextView滑動滾動條和滑動效果AndroidTextView
- Day-3 文字排版
- Android元件詳解—TextViewAndroid元件TextView
- 短影片app原始碼,Android TextView文字,刪除線以及下劃線APP原始碼AndroidTextView
- canvas 如何自動去換行Canvas
- 【轉】css樣式自動換行(強制換行)CSS
- 002.08 文字輸出排版 PySimpleGUIGUI
- Android 高亮關鍵字TextViewAndroidTextView
- Android:TextView maxWidth maxLines maxLength maxEmsAndroidTextView
- Android中TextView及其子類AndroidTextView
- 輸入自動補齊
- CSS換行和不換行CSS
- 【CSS】段落文字實現兩端對齊,不滿一行則不需要CSS
- CMU 領域自適應最新進展:對齊資料分佈、誤差還不夠,標註函式也需對齊!函式
- excel如何設定自動換行?Excel