背景
最近项目需求自定义了一个View,用来绘制签名。
本来想着在用户绘制之前,在View上设置一个自定义的文本,为了保证灵活性,要在Activity中调用自定义View的setHintText方法绘制文本。
而且,如果已经签过名的,要在Activity启动时调用drawImg回显上次的签名图片。
问题:
当在Activity的onCreate/onResume/onAttachedToWindow等方法调用上述2个方法,均不能正确绘制。
甚至在drawImg时,会因为width或height小于等于零而报错崩溃
public void drawImg(Bitmap bitmap) {
Matrix matrix = new Matrix();
//bitmapWidth 与bitmapHeight 为自定义View 的宽高
matrix.setScale((float) bitmapWidth / bitmap.getWidth(), (float) bitmapHeight / bitmap.getHeight());
cachebBitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, false);
cacheCanvas = new Canvas(cachebBitmap);
postInvalidate();
}
原因
我们来看看Activity的生命周期与View的绘制过程混合起来是怎样的过程
PS:
1.View 列里,绿色框中,因为未执行onMeasure,所以View的准确大小无法获取,调用getMeasuredWidth/getMeasuredHeight,返回的结果是0;
2.View列里,橙色框为能正确获取View的大小的方法
3.onWindowFocusChanged true/false 即是否hasFoucus
4.M1LD M2LD,即onMeasure 1次或2次,onLayout 一次,onDraw 一次。
5.onVisibilityChanged 4,为不可见INVISIBLE, 0为可见 VISIBLE
6.虚线框的,可能并不会被调用!
如上图所示,在Activity的生命周期中,并没有合适的时机能够正确给自定义View设置要绘制的内容。过早(onAttachedToWindow及之前)的设置,因为自定义View的大小还未测量,不能绘制。又不能再onPause
或者onStop
中设置。
所以,我们考虑在Activity的onAttachedToWindow
或之前的任意生命周期函数里将要绘制的内容交给自定义View,如果还未测量大小就不立即绘制,让自定义View 自己选择合适的时机进行绘制。
在自定义View的方法中,只有onSizeChanged
里已经完成View大小的测量且只调用了一次,所以我们在这里进行一点自己的设置。
//判断是否已经执行了onSizeChanged;
private boolean sizeAlreadyChanged = false;
private String mHint;
public void setHintText(String hint) {
mHint = hint;
if (sizeAlreadyChanged) {
drawText(hint);//绘制文本
}
}
...
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
//已经执行了onSizeChanged 设置为true
sizeAlreadyChanged=true;
if (mHint != null)
drawText(mHint);//绘制文本
}