温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

在Android中使用AutoWrapTextView时出现中英文排版问题如何解决

发布时间:2020-12-05 16:15:48 来源:亿速云 阅读:303 作者:Leah 栏目:移动开发

这篇文章将为大家详细讲解有关在Android中使用AutoWrapTextView时出现中英文排版问题如何解决,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

实现

首先创建一个继承自View的AutoWrapTextView

public class AutoWrapTextView extends View {

}

来看看它的构造方法

public AutoWrapTextView(Context context, AttributeSet attrs) {
  super(context, attrs);

  init(context, attrs);
}

private void init(Context context, AttributeSet attrs) {
  initStyle(context, attrs);
  initPaint();
}

init方法里分别调用了initStyle方法和initPaint方法;

initStyle方法主要解析自定义的属性

 private void initStyle(Context context, AttributeSet attrs) {
  TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.AutoWrapTextViewStyle);
  mPaddingLeft = typedArray.getDimensionPixelSize(R.styleable.AutoWrapTextViewStyle_paddingLeft, 0);
  mPaddingRight = typedArray.getDimensionPixelSize(R.styleable.AutoWrapTextViewStyle_paddingRight, 0);
  mPaddingTop = typedArray.getDimensionPixelSize(R.styleable.AutoWrapTextViewStyle_paddingTop, 0);
  mPaddingBottom = typedArray.getDimensionPixelSize(R.styleable.AutoWrapTextViewStyle_paddingBottom, 0);

  mTextColor = typedArray.getColor(R.styleable.AutoWrapTextViewStyle_textColor, Color.BLACK);
  mTextSize = typedArray.getDimensionPixelSize(R.styleable.AutoWrapTextViewStyle_textSize, 50);
  mLineSpacingExtra = typedArray.getInteger(R.styleable.AutoWrapTextViewStyle_lineSpacingExtra, 7);

  typedArray.recycle();
 }

属性名含义都很明显不用过多解释,initPaint方法就是初始化一个文本画笔

 private void initPaint() {
  mTextPaint = new TextPaint();
  mTextPaint.setAntiAlias(true);
  mTextPaint.setTextSize(mTextSize);
  mTextPaint.setColor(mTextColor);
  mTextPaint.setTextAlign(Paint.Align.LEFT);
 }

接下来我们看看设置文本的方法setText方法

 public void setText(String text) {
  if (TextUtils.isEmpty(text)) return;

  //把文本转换成Char数组
  mTextCharArray = text.toCharArray();
  requestLayout();
 }

首先把文本转换成Char数组,然后循环数组把整个文本拆分成N行文本,下面来看看核心方法splitText方法

 private void splitText(int heightMode) {
  if (mTextCharArray == null) return;

  mSplitTextList = new ArrayList<>();
  mSingleTextWidth = getMeasuredWidth() - mPaddingLeft - mPaddingRight;
  int currentSingleTextWidth = 0;
  StringBuffer lineStringBuffer = new StringBuffer();
  for (int i = 0, length = mTextCharArray.length; i < length; i++) {
   char textChar = mTextCharArray[i];
   currentSingleTextWidth += getSingleCharWidth(textChar);
   if (currentSingleTextWidth > mSingleTextWidth) {
    mSplitTextList.add(lineStringBuffer.toString());
    lineStringBuffer = new StringBuffer();
    currentSingleTextWidth = 0;
    i--;
   } else {
    lineStringBuffer.append(textChar);
    if (i == length - 1) mSplitTextList.add(lineStringBuffer.toString());
   }
  }

  int textHeight = 0;
  mSplitTextRectArray = new Rect[mSplitTextList.size()];
  for (int m = 0, length = mSplitTextList.size(); m < length; m++) {
   String lineText = mSplitTextList.get(m);
   Rect lineTextRect = new Rect();
   mTextPaint.getTextBounds(lineText, 0, lineText.length(), lineTextRect);
   if (heightMode == MeasureSpec.AT_MOST) {
    textHeight += (lineTextRect.height() + mLineSpacingExtra);
    if (m == length - 1) {
     textHeight = textHeight + mPaddingBottom + mPaddingTop;
    }
   } else {
    if (textHeight == 0)
     textHeight = getMeasuredHeight();
   }
   mSplitTextRectArray[m] = lineTextRect;
  }

  setMeasuredDimension(getMeasuredWidth(), textHeight);
 }

首先创建一个属性名为mSplitTextList的List集合用来存放拆分的文本;

mSingleTextWidth 为单行文本显示的宽度;

currentSingleTextWidth 为当前一行累计计算的宽度;

然后开始循环Char数组,getSingleCharWidth方法就是计算单个Char的宽度;

如果currentSingleTextWidth 小于 mSingleTextWidth 就把Char添加到lineStringBuffer 当中,如果是最后一个Char就直接把lineStringBuffer添加到mSplitTextList集合当中

如果currentSingleTextWidth 大于 mSingleTextWidth,就把lineStringBuffer添加到mSplitTextList集合当中,重新给lineStringBuffer赋值,currentSingleTextWidth 归0;

循环结束以后拆分好的文本就都添加到mSplitTextList集合当中了。

拆分完成以后循环mSplitTextList集合,得到每一行文本的Rect值,绘制文本的时候会用到,然后设置View的宽高。

接下来就是绘制方法drawText

 public void drawText(Canvas canvas) {
  if (mSplitTextList == null || mSplitTextList.size() == 0) return;

  int marginTop = getTopTextMarginTop();
  for (int m = 0, length = mSplitTextList.size(); m < length; m++) {
   String lineText = mSplitTextList.get(m);
   canvas.drawText(lineText, mPaddingLeft, marginTop, mTextPaint);
   marginTop += (mSplitTextRectArray[m].height() + mLineSpacingExtra);
  }
 }

首先得到第一行文本距离顶部的高度marginTop,然后循环文本绘制每一行文本内容。

关于在Android中使用AutoWrapTextView时出现中英文排版问题如何解决就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI