iOS如何实现一个简单的画板

写在前面

最近公司要做一个画板类App,所以研究了一下iOS画板的实现。通过查阅资料,我发现大家做画板很多是使用UIView的drawRect来绘制线条。但是经过一些大牛的验证(本人也已经验证过),使用UIView的drawRect方法时,如果view的尺寸过大,会直接导致内存暴涨,具体原因大家可以看看这篇博客:http://blog.csdn.net/jijiji000111/article/details/50480405

实现思路

既然drawRect方法不好用,那么我们可以使用CAShapeLayer和UIBezierPath来实现一个简单的画板程序。
基本实现思路:新建一个画板类继承自UIView,在这个画板类的touchesBegan方法中获取起始点作为路径CGPath的起始点;在touchesMoved方法中将后续的触摸点连成线,保存到CGPath中,然后赋值给layer,画本逻辑基本就实现了代码如下。下面就具体功能实现介绍一下。

@interface ZYSDrawboardView ()

/// current layer
@property (nonatomic, strong) CAShapeLayer *currentDrawLayer;

/// current path(start from touchBegin)
@property (nonatomic, strong) UIBezierPath *currentDrawPath;

/// save all layers
@property (nonatomic, strong) NSMutableArray *drawLayerArray;

@end

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    UITouch *touch = touches.anyObject;
    CGPoint startPoint = [touch locationInView:self];
    
    self.currentDrawLayer = [self makeDrawLayer:self.isEraserEnabled];
    self.currentDrawPath = [self makeDrawPath];
    [self.currentDrawPath moveToPoint:startPoint];
}

- (void)touchesMoved:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    UITouch *touch = touches.anyObject;
    CGPoint currentPoint = [touch locationInView:self];
    
    [self.currentDrawPath addLineToPoint:currentPoint];
    self.currentDrawLayer.path = self.currentDrawPath.CGPath;
}
画板实现的思考

最开始,我只要做一个能画线条的画板就行了,于是我只使用了一个layer和一个path来实现,这样确实可行(此时画板上的线条都属于一个layer的一个path,只是path不连续而已),但是后续在添加橡皮擦的功能时,我还是把这个简单的逻辑放弃了。
因为我要加橡皮擦时如果还是使用一个layer,只是改变一下layer的线条颜色再继续画,这样这个layer以前的path颜色都会改变,所以这样行不通。于是我马上新加了一个eraserLayer来绘制,这样画板里就有两个layer在画了,只是橡皮擦layer的层级在画笔layer之上,这样橡皮擦的path才会覆盖画笔layer的path,从而达到“擦除”的效果。
但是擦完之后,问题又来了。橡皮擦擦过的这片区域已经被橡皮擦layer所占用,而且橡皮擦的所处层级在画笔之上,这样画笔就不能继续在这块区域上画了。该怎么办呢?
想来想去,我还是想出了一个不是很完美的办法。具体实现思路如下。

橡皮擦功能的实现

既然一个画笔layer不能在橡皮擦之上继续画,那么我就在每次调用touchesBegan时都新建一个layer(层级在之前的layer之上)和path来绘制图形,这样,一个画板上的一个线条其实就是一个layer和对应的一个path(橡皮擦layer其实也是一个特殊的layer),而且后面的layer总可以覆盖前面的layer,这样橡皮擦的问题自然就解决了。你可能会想,这样做得新建多少个layer对象和path对象,会不会造成内存的问题呢?关于这个问题,我确实验证过,内存增长的并不多。
这个方法确实不是很好,确实有可以优化的地方,比如,我只需要在调用了橡皮擦之后才新建一个啦画笔layer,并不需要在每次调用touchesBegan时都新建一个layer,不过这个想法我还没具体实现,如果大家有更好的思路,欢迎指教~

Demo

我做了demo放在了GitHub上,欢迎大家参考~
GitHub

参考:http://www.jianshu.com/p/4e16611969fd
http://blog.csdn.net/jijiji000111/article/details/50480405

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容