关于UIView及其子类重绘drawRect
转载自:
https://nezha.gitbooks.io/ios-developmentarticles/content/UIView%E7%9A%84drawRect%E9%87%8D%E7%BB%98.html
有了view的子类,只要重载一个方法drawRect
-(void)drawRect:(CGRect)aRect;
它的参数是个矩形,这个矩形就是你要重绘的区域,你可以忽略参数,它只是为了性能优化,只在固定的区域绘图.
注意!!!
永远不要去调用drawRect,因为drawRect不是让你调用的,而是系统会去调用的.
那怎么去告诉系统需要重绘呢?
你发送两个消息,setNeedsDisplay和setNeedsDisplayInRect.
-(void)setNeedsDisplay;
-(void)setNeedsDisplayInRect:(CGRect)aRect;
你们可以认为初始化的时候的设置是一个点,然后晚些系统查看所有需要重绘的东西,
然后把它们按顺序排列,因为有些东西可能会重叠,然后再非常高效的把需要的东西绘制出来.
这样做有两个好处,一是让系统依据层的情况最优化性能,
二是如果你的property有一些setter,当你设置的时候需要重绘,这种情况也被最优化了.
所以你所有的setter都会调用self的setNeedsDisplay来重绘.
如果有人用了你的view,然后调用了好几个这样的setter,只需要重绘一次.
每个setNeedsDisplay都被一起传过去,然后一次性画出来.而不是每次都重绘.
环境(Context)决定了你在哪绘图,所以你创建环境的方法决定了要在哪绘图.
关于这个环境要注意的有一点,每次调用drawRect环境都是不一样的,所以不要把它保存起来,而是每次都去获取新的,
CGContextRef context = UIGraphicsGetCurrentContext();
这就是调用的方法,这也几乎是每个drawRect的第一行.
CGContextRef 是个cookie,不知道到底是什么,它不是个对象,因为没有*
当你有了环境,就可以用它来建立轨迹了,
这里就演示一个轨迹的例子,用CGContextBeginPath(context)来传递刚刚得到的环境,
然后就可以进行移动、添加直线弧线之类的了,
下面的几行代码就创建了右边的那个轨迹,
CGContextMoveToPoint(context , 75 , 10);
CGContextAddLineToPoint(context , 160 ,150);
CGContextAddLineToPoint(context , 10 ,150);
再说一次左上角是(0 , 0),画轨迹而不是缩放,
还有CGContextClosePath(context); 它会画条线回到起始点来封闭图形,
封闭图形不是必须的,你可以画个开放的.
上面做的这些看起来像是在绘画,但是只有上面几行代码的话其实什么也没画,虽然轨迹被创建了,但是没有真的画到屏幕上,
为什么会这样呢?因为你需要描边(stroke) 或者填充(fill) 来显示轨迹,
在描边或设置填充之前,还可以设置图形属性,这里用了UIColor设置颜色.
[ [ UIColor greenColor] setFill];
[ [ UIColor redColor] setStroke];
注意在用这个UIColor的时候不需要指出它的环境,当你在使用对象的时候,事实上你只用到UIColor,UIFont,NSString.
这里你使用对象的时候不用指出它的环境,它会假设你用的是当前环境,只要在前面的CGContext表面环境就可以了.
但是这里设置了描边和填充,还是什么都没画,还需要调用CGContextDrawPath
CGContextDrawPath(context , kCGPathFillStroke);
这个参数kCGPathFillStroke是常量标志,表示描边和填充还是仅描边或仅填充.
其实还可以建立一个轨迹,保存到一个轨迹变量CGPath,使用完之后还能交给其他环境继续用,
更多功能,需要去查看文档来使用.
Graphics State
图形状态,你用的最主要的是颜色,但你可以用一些复杂的,比如线宽、图案填充等,
CGContextSetLineWidth(context , 1.0);
CGContextSetFillPattern(context , (CGPatternRef)pattern , (CGFloat [] )components ); 图形状态有一点要小心的就是子程序(subroutines),因为比如我有了一个新环境然后设置它的图形状态,
那么这时候我调用的子程序会怎么样呢?
子程序里面也有自己的图形状态,这就和我的设置冲突了,
所以对此有个机制叫push和pop环境,
UIGraphicsPopContext();
所以我就对开始的那个环境入栈,然后再做我要做的绘图,
然后在出栈回到调用子程序之前的环境,
在drawRect里面绘制图片
UIImage *image="";
[image drawAtPoint:(CGPoint) p ];//按原大小绘图
[image drawInRect:(CGRect) r ];//会缩放图片
[image drawAsPatternInRect:(CGRect) pathRect];//会重复绘图以填满指定的矩形区域
你还可以用PNG或者JPG格式的二进制数据来表示图片:
一些常见问题
问1:由UIImageView派生出的类,为什么在drawRect中,无法绘制矩形?但如果从UIView派出就可以?
答:苹果的官方文档说了, UIImageView是专门为显示图片做的控件,用了最优显示技术,是不让调用darwrect方法, 要调用这个方法,只能从uiview里重写。
问2:在ios开发中,为什么自定义UIView 重写drawRect方法之后,绘图区域之外为黑,合理的情况是,我没有绘制或者填充的区域应该是透明的才对啊,如果我希望没有绘制的地方为透明该如何做?
答:self.backgroundColor = [UIColor clearColor]; 并记得[super drawRect];