标签:
原文地址:http://android.xsoftlab.net/training/custom-views/custom-drawing.html#draw
自定义View最重要的部分就是它的样子了。自定义View的绘制根据应用的需要或者简单亦或者复杂。这节课的内容涵盖了大多数通用的知识点。
绘制自定义View很重要的一个步骤就是重写它的onDraw()方法。该方法含有一个Canvas对象作为参数,用来使View绘制它本身的内容。Canvas类定义了用于绘制文本,线条,位图以及许多其它基础的物理图形的方法。你可以在onDraw()方法中使用这些方法来创建属于你自己的UI效果。
在开始任何绘制之前,必须先创建一个Paint对象。后面的部分会讨论Paint的相关知识。
android.graphics框架将绘制分为了两块区域:
举个例子,Canvas提供了用于绘制线条的方法,而Paint则定义了线条的颜色。Canvas拥有绘制矩形的方法,而Paint则定义了是否要使颜色填充这个矩形。简而言之,Canvas定义了你绘制在屏幕上的形状,而Paint则定义了这些形状的颜色、风格以及字体等等。
所以,在开始绘制任何事物之前,你需要先创建一个或多个Paint对象。示例PieChart将这些工作放在了一个名为init的方法中,它由构造方法调用:
private void init() {
mTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mTextPaint.setColor(mTextColor);
if (mTextHeight == 0) {
mTextHeight = mTextPaint.getTextSize();
} else {
mTextPaint.setTextSize(mTextHeight);
}
mPiePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mPiePaint.setStyle(Paint.Style.FILL);
mPiePaint.setTextSize(mTextHeight);
mShadowPaint = new Paint(0);
mShadowPaint.setColor(0xff101010);
mShadowPaint.setMaskFilter(new BlurMaskFilter(8, BlurMaskFilter.Blur.NORMAL));
...
提前创建对象是一项非常重要的优化手段。View会很频繁的绘制,创建绘制对象的开销十分高昂。在onDraw()方法中创建绘制对象会明显的降低应用的性能,并可能会导致UI出现停顿。
为了可以正确的绘制View,首先需要知道View的尺寸。复杂一点的View经常需要执行多次布局计算。你永远不要假定View的尺寸,就算是只有一款应用使用了你的View。因为APP需要处理不同的屏幕尺寸,不同的屏幕密度,以及在垂直模式及水平模式下不同的高宽比。
尽管View拥有很多测量尺寸的方法,但是绝大多数是不需要重写的。如果View不需要特别控制它的尺寸,你只需要重写一个方法:onSizeChanged().
onSizeChanged()方法会在首次分配尺寸的时候调用,如果尺寸再次变更时则会再次调用。我们在onSizeChanged()方法中计算View的位置、尺寸以及其它任何与尺寸相关的值,而不是在每次绘制的时候重新计算它们。在示例PieChart中,onSizeChanged()方法内部计算了饼图的矩形边框以及文本和其它可视元素的相对位置。
当View被分配了一个尺寸时,布局管理器会假设该尺寸包含了View所有的内边距。所以在计算View的尺寸时要将View的内边距计算在内。下面的代码段展示了PieChart.onSizeChanged()是如何做的:
// Account for padding
float xpad = (float)(getPaddingLeft() + getPaddingRight());
float ypad = (float)(getPaddingTop() + getPaddingBottom());
// Account for the label
if (mShowText) xpad += mTextWidth;
float ww = (float)w - xpad;
float hh = (float)h - ypad;
// Figure out how big we can make the pie.
float diameter = Math.min(ww, hh);
如果你想更精细的控制View布局的参数,请实现onMeasure()方法。这个方法的两个参数都为View.MeasureSpec。它们用于告诉View的父布局希望View是多大,这个View的尺寸是最大值还是只是一个建议值等等。随着优化,这些值被存放在一个被包装后的整型值内,你必须使用View.MeasureSpec的一个静态方法来提取存储在这个整型值内的信息。
下面是onMeasure()方法的一个示例。在这个实现中,PieCart尝试将自己的区域扩大到内部标签的大小。
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// Try for a width based on our minimum
int minw = getPaddingLeft() + getPaddingRight() + getSuggestedMinimumWidth();
int w = resolveSizeAndState(minw, widthMeasureSpec, 1);
// Whatever the width ends up being, ask for a height that would let the pie
// get as big as it can
int minh = MeasureSpec.getSize(w) - (int)mTextWidth + getPaddingBottom() + getPaddingTop();
int h = resolveSizeAndState(MeasureSpec.getSize(w) - (int)mTextWidth, heightMeasureSpec, 0);
setMeasuredDimension(w, h);
}
这里有三个非常重要的点需要关注:
完成对象创建代码与尺寸测量代码之后,接下来就可以实现onDraw()方法了。每个View的onDraw()方法都不相同,但是它们还是有一些相同特点存在的:
举个例子,下面的代码用来绘制PieChart。它混合使用了文本、线条以及形状。
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
// Draw the shadow
canvas.drawOval(
mShadowBounds,
mShadowPaint
);
// Draw the label text
canvas.drawText(mData.get(mCurrentItem).mLabel, mTextX, mTextY, mTextPaint);
// Draw the pie slices
for (int i = 0; i < mData.size(); ++i) {
Item it = mData.get(i);
mPiePaint.setShader(it.mShader);
canvas.drawArc(mBounds,
360 - it.mEndAngle,
it.mEndAngle - it.mStartAngle,
true, mPiePaint);
}
// Draw the pointer
canvas.drawLine(mTextX, mPointerY, mPointerX, mPointerY, mTextPaint);
canvas.drawCircle(mPointerX, mPointerY, mPointerSize, mTextPaint);
}
Android官方开发文档Training系列课程中文版:创建自定义View之View的绘制
标签:
原文地址:http://blog.csdn.net/sahadev_/article/details/51874371