给出核心代码
package com.textview.test;
import java.util.Vector;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Paint.FontMetrics;
import android.util.AttributeSet;
import android.view.View;
import android.view.WindowManager;
import android.widget.LinearLayout;
import android.widget.TextView;
public class MyView extends TextView {
private final String namespace="http://www.nearmobile.net";
private int resourceId=0;
/* 声明Paint对象 */
private Paint mPaint = null;
/* 声明TextUtil对象 */
private TextUtil mTextUtil = null;
public static int m_iTextHeight;
private WindowManager wm=null;
private String string="";
public MyView(Context context, AttributeSet set) {
super(context,set);
resourceId=set.getAttributeResourceValue(namespace, "text", 0);
if(resourceId==0)
string=set.getAttributeValue(null,"text");
else
string=this.getResources().getString(resourceId);
wm=(WindowManager)context.getSystemService(Context.WINDOW_SERVICE);
/* 构建对象 */
m_iTextHeight=2000;
mPaint = new Paint();
mPaint.setColor(Color.RED);
mPaint.setStrokeWidth(40);
mPaint.setTextSize(20);
int m_iTextWidth=wm.getDefaultDisplay().getWidth();
FontMetrics fm = mPaint.getFontMetrics();
int m_iFontHeight = (int) Math.ceil(fm.descent - fm.top) + 4;
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+2)*m_iFontHeight+2;
//用反射机制得到 m_iTextHeight 值
/* 实例化TextUtil
mTextUtil = new TextUtil(string,5,25,wm.getDefaultDisplay().getWidth(),this.getHeight(),0x0,0xffffff,255,15);
初始化TextUtil
mTextUtil.InitText(string,5,25,wm.getDefaultDisplay().getWidth(),wm.getDefaultDisplay().getHeight(),0x0,0xffffff,255,15);*/
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
/* 设置背景颜色 */
canvas.drawColor(Color.BLACK);
mPaint.setAntiAlias(true);
char ch;
int w = 0;
int istart = 0;
int m_iFontHeight;
int m_iRealLine=0;
int x=2;
int y=60;
Vector m_String=new Vector();
int m_iTextWidth=wm.getDefaultDisplay().getWidth();
FontMetrics fm = mPaint.getFontMetrics();
m_iFontHeight = (int) Math.ceil(fm.descent - fm.top) + 4;
//m_ipageLineNum = m_iTextHeight / m_iFontHeight;
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;
System.out.println("m_iTextHeight----->"+m_iTextHeight);
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.
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 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;
}
}
效果见图
<!--StartFragment -->
- 大小: 59.2 KB
分享到:
相关推荐
自定义view解决android文本排版和换行问题自定义view解决android文本排版和换行问题自定义view解决android文本排版和换行问题自定义view解决android文本排版和换行问题自定义view解决android文本排版和换行问题...
android 自定义View — 自定义组合控件
Android自定义标签view——可自动换行,该项目从github上下载下来,经测试效果非常不错,下载后可以直接运行使用。
仿ios的开关控件,采用android自定义view的方式实现
android自定义view之组合控件、重写控件
用自定义View的方式来实现圆形的遥控器菜单.rar,太多无法一一验证是否可用,程序如果跑不起来需要自调,部分代码功能进行参考学习。
在我们从Android小学生 进阶到 Android中学生的路上,肯定需要经历 自定义View。 大神已经给出了,最精简的Demo,我这里 顺水推舟,把一些基础知识 标记在代码之中,各位可以一边看源码 一边学基础。 觉得文章有用,...
android 自定义view比较综合的例子,涉及到一些复杂的实现效果,有一些基础的人可以参考下。
android demo,自定义控件view,点击该自定义view,onclick随机生成数字
自定义view控件 canvas绘制自己的组件小demo
android项目下对图片高斯模糊毛玻璃,自定义view,模糊程度可控制,高斯模糊毛玻璃,自定义view,模糊程度可控制
android一款仪表盘控件,测试可以安装。具体如何整合到你的应用,查看我的博客文章
Android自定义组件之自动换行View
自定义view,多个自定义控件的实现,欢迎大家来交流,相互学习!
android使用自定义view和自定义button实现的小demo,可以实现button的点击变换背景等功能
compile 'moe.pine:percent_clip_view:0.1.2' } Usage You can use they layouts in XML: And, they have getters and setters: LinearLayout layout = (LinearLayout) this.findViewById(R.id.layout)...
自定义View控件 1
Android中自定义View操作Android中自定义View操作Android中自定义View操作
以上是以一个自定义滑动开关的案例来总结一下自定义View控件的编写流程
Android 自定义View 实现View排列自动换行 控制间距