自定义形状按钮的实现

自定义形状按钮的实现

对于大多数iOS开发来说,这个是很少遇到的,毕竟一个不规则的按钮再移动端不常见,但是免不了会遇到某些特殊的需求,比如一个饼状图,比如一个扇形图,比如中国地图中的每个省.....
我们很可能在某一次开发中需要实现不规则形状的按钮,那么我们怎么去实现呢?
其实这只是一个很小很小的知识点,却有很多人不会去考虑,所以就遇到了这样的情况

偶然有一天,朋友给了我一套产品原型,他说叫我帮他写几个按钮,很简单的按钮,我当时感觉有坑,但是也没有想太多,想着本来这段时间有点闲,就帮下忙,
于是他发来原型图


这里写图片描述

哟,就是很简单的4+1个按钮蛮
先写4个正方形的九宫格式的按钮,再在中心加一个圆形按钮就好了呀.
于是接下来他又发来产品设计图


这里写图片描述

果然,有了设计之后,瞬间好看了好多,
但是想象中也不复杂呀.
先用一个view当底图,然后5个按钮加在这个view上,然后view来个半径,不就结束了吗?
但是事实真的是这么简单吗?

显然不是,一个很明显的结果就是,你会发现你使用了圆形半径的不显示的部分仍然能够响应按钮的点击.
这个是什么鬼,其实很简单的原理,就是虽然我们设置了半径,让圆形外的图层不显示,但是并不是代表其不存在,其仍然是可以点击的,这样在某些情况下就不符合按钮的设计标准了,因为我们想要的是所见即所得

那么所见即所得便成了我要实现的目标.
还得从按钮上下手,不要想得这么简单,虽然这个案例是一个很简单的比较偏正常形状的一个按钮,但是我需要将其想象成一个很复杂的不规则的图形来处理

1.因为是自定义形状的按钮,那么说白了,它还是一个按钮,我们没有必要自定义复杂的控件,继承UIButton即可

2.构思如何实现特殊形状
其实这块很容易想到,实现一个圆形按钮,就是对layer进行修改形状而已

如果你有阅读过iOS核心动画高级技巧的话,你很容易发现我们想要的东西


这里写图片描述

这里写图片描述

是的,这里我们能看到几个关键点,一个原来的背景+一个mask遮罩 = 一个自定义形状的图形

引用到我们想要的按钮上,我们就会发现同样适用,
那么我们必须要在适用图片来制作mask吗?
很显然,我们在开发中不可能这么去做,那么我们如何自定义一个形状呢?


这里写图片描述

这里写图片描述

这里写图片描述

于是我们能够很方便的使用UIBezierPath来绘制成我们想要的形状,然后使用CAShapeLayer来根据形状创建图层的路径
创建完成之后,我们使用这样的图层来当做按钮的mask即可

所以对于我们来说比较有用的就是一个贝塞尔曲线的定制
于是很简单的一个不规则按钮的类即可实现

#import <UIKit/UIKit.h>

@interface IrregularButton : UIButton

@property(nonatomic, strong)UIBezierPath *maskPath;

@end


#import "IrregularButton.h"

@implementation IrregularButton

-(void)setMaskPath:(UIBezierPath *)maskPath{
    _maskPath = maskPath;
    CAShapeLayer *layer = [[CAShapeLayer alloc]init];
    layer.path = maskPath.CGPath;
    self.layer.mask = layer;
}

@end

当我们需要定制一个特殊按钮的时候,我们只要继承该类 ,或者直接使用该类
,并给maskPath赋予一个我们想要的路径即可

于是上面的四个半圆按钮甚至全圆形的按钮我们就可以轻松实现

#import "IrregularButton.h"

typedef NS_ENUM(NSUInteger, QuarterCircleType) {
    QuarterCircleTypeTopNone,
    QuarterCircleTypeTopLeft,
    QuarterCircleTypeTopRight,
    QuarterCircleTypeBottomLeft,
    QuarterCircleTypeBottomRight,
    QuarterCircleTypeAllCircle,
};

@interface QuarterCircleButton : IrregularButton

@property(nonatomic, assign)QuarterCircleType circleType;

@end

#import "QuarterCircleButton.h"

@implementation QuarterCircleButton

-(void)setFrame:(CGRect)frame{
    [super setFrame:frame];
    
    self.circleType = self.circleType;
}

-(void)setCircleType:(QuarterCircleType)circleType{
    _circleType = circleType;
    
    CGFloat width = self.frame.size.width;
    CGFloat height = self.frame.size.height;
    switch (circleType) {
        case QuarterCircleTypeTopLeft:{
            CGPoint bottomRightPoint = CGPointMake(width, height);
            CGPoint bottomLeftPoint = CGPointMake(0, height);
            
            UIBezierPath *path = [[UIBezierPath alloc]init];
            [path moveToPoint:bottomRightPoint];
            [path addLineToPoint:bottomLeftPoint];
            [path addArcWithCenter:bottomRightPoint radius:MAX(width, height) startAngle:M_PI endAngle:-M_PI_2 clockwise:YES];
            [path addLineToPoint:bottomRightPoint];
            [path closePath];
            self.maskPath = path;
        }
            break;
        case QuarterCircleTypeTopRight:{
            CGPoint bottomLeftPoint = CGPointMake(0, height);
            CGPoint topLeftPoint = CGPointMake(0, 0);
            
            UIBezierPath *path = [[UIBezierPath alloc]init];
            [path moveToPoint:bottomLeftPoint];
            [path addLineToPoint:topLeftPoint];
            [path addArcWithCenter:bottomLeftPoint radius:MAX(width, height) startAngle:-M_PI endAngle:0 clockwise:YES];
            [path addLineToPoint:bottomLeftPoint];
            [path closePath];
            self.maskPath = path;
        }
            break;
        case QuarterCircleTypeBottomLeft:{
            CGPoint topRightPoint = CGPointMake(width, 0);
            CGPoint bottomRightPoint = CGPointMake(width, height);
            
            UIBezierPath *path = [[UIBezierPath alloc]init];
            [path moveToPoint:topRightPoint];
            [path addLineToPoint:bottomRightPoint];
            [path addArcWithCenter:topRightPoint radius:MAX(width, height) startAngle:M_PI_2 endAngle:M_PI clockwise:YES];
            [path addLineToPoint:topRightPoint];
            [path closePath];
            self.maskPath = path;
        }
            break;
        case QuarterCircleTypeBottomRight:{
            CGPoint topLeftPoint = CGPointMake(0, 0);
            CGPoint topRightPoint = CGPointMake(width, 0);
            
            UIBezierPath *path = [[UIBezierPath alloc]init];
            [path moveToPoint:topLeftPoint];
            [path addLineToPoint:topRightPoint];
            [path addArcWithCenter:topLeftPoint radius:MAX(width, height) startAngle:0 endAngle:M_PI_2 clockwise:YES];
            [path addLineToPoint:topLeftPoint];
            [path closePath];
            self.maskPath = path;
        }
            break;
        case QuarterCircleTypeAllCircle:{
            CGPoint centerPoint = CGPointMake(width/2, height/2);
            
            UIBezierPath *path = [[UIBezierPath alloc]init];
            [path addArcWithCenter:centerPoint radius:MAX(width/2, height/2) startAngle:0 endAngle:M_PI * 2 clockwise:YES];
            [path closePath];
            self.maskPath = path;
        }
            break;
            
        default:
            break;
    }
}

于是,很轻松的我创建了这样的图形


这里写图片描述

但是问题仍然不可忽视的,又来了,在半圆形外面的四个区域仍然可以相应按钮点击,
解释下来也就是mask外面的区域虽然看不到,但是其仍然属于按钮的区域,仍然可以响应点击,


这里写图片描述

所以我们需要规避
那么如何规避呢?
其毕竟仍然是一个不规则的形状啊?
其实我们前期已经做好准备了....

3.规避不规则区域外的点击
我们只要简单的实现这个点击是否响应的方法即可
于是完整的不规则按钮的实现是这样的

#import <UIKit/UIKit.h>

@interface IrregularButton : UIButton

@property(nonatomic, strong)UIBezierPath *maskPath;

@end


#import "IrregularButton.h"

@implementation IrregularButton

-(void)setMaskPath:(UIBezierPath *)maskPath{
    _maskPath = maskPath;
    CAShapeLayer *layer = [[CAShapeLayer alloc]init];
    layer.path = maskPath.CGPath;
    self.layer.mask = layer;
}

-(BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event{
    BOOL res = [super pointInside:point withEvent:event];
    if (res) {
        if (!self.maskPath || [self.maskPath containsPoint:point]) {
            return YES;
        }
        return NO;
    }
    return res;
}

@end

我们只要保证点击的point在我们的mask的区域内即可(这里要保证我们的mask是一个封闭的区域),在mask区域内允许点击,区域外则不允许点击,
于是一个maskPath被我们使用了两次,第一次是设置遮罩显示不规则按钮,第二次是判断点击时候的响应区域

至此,能够适应各种自定义形状的基类按钮便完成了
当我们想要实现的时候,我们只要根据原型中的不规则形状绘制我们想要的图形来创造一个贝塞尔曲线即可(这部分因为形状不同,我们只能自己绘制)

当然我们也有第三方工具
这里写图片描述

如果你觉得绘制一个不规则形状太麻烦的话(偷懒)
paintCode也可以帮你的忙,你只要动动鼠标既可以跟类似ps一样的绘制一个不规则图形,而这个工具会自动帮你把贝塞尔曲线的代码给写好,你只要复制进去即可

好了demo奉上吧...
https://github.com/spicyShrimp/IrregularButton

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

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,945评论 4 60
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,056评论 25 707
  • 看着屏幕上的朋友还在奋战着,我不由得点下了“退出游戏”的按钮。常在网络上游荡。喜欢看周星驰的电影,乱糟糟的头发,一...
    烟风银月阅读 295评论 0 1
  • A:今天跟同事聊天,她跟我说她观察我对新员工相处的很好,会适当用肢体接触表达赞赏之情,她也学着用一用,果然新员工和...
    邓男神Sweety阅读 111评论 0 0
  • 二十三这个说大不大,说小不小的尴尬年龄,本想着毕业了,可以随着自己的想法去做想做的事,偏偏毕业回家,家里人开始催各...
    薄情y阅读 178评论 0 1