美文网首页iOS学习笔记iOS程序猿UI
iOS开发 | 自定义不规则label

iOS开发 | 自定义不规则label

作者: Lol刀妹 | 来源:发表于2018-04-01 11:52 被阅读1473次

    场景

    最近app改版,以下是截取的部分UI图:

    顶部的这一部分我是把它看做UICollectionView的组头:

    其中有一个不太规则的label

    这个label顶部的两个角是圆角,底部的两个角是直角,底部还有一个小三角。

    思路

    CAShapeLayer联合UIBezierPath画一个不规则的layer作为label.layermask

    具体实现

    1.自定义一个继承自UILabelIrregularLabel

    #import "IrregularLabel.h"
    
    @interface IrregularLabel ()
    
    /** 遮罩 */
    @property (nonatomic, strong) CAShapeLayer *maskLayer;
    /** 路径 */
    @property (nonatomic, strong) UIBezierPath *borderPath;
    
    @end
    

    2.在初始化方法中进行相应初始化和设置

    - (instancetype)initWithFrame:(CGRect)frame {
        if (self = [super initWithFrame:frame]) {
            // 初始化遮罩
            self.maskLayer = [CAShapeLayer layer];
            // 设置遮罩
            [self.layer setMask:self.maskLayer];
            // 初始化路径
            self.borderPath = [UIBezierPath bezierPath];
        }
        return self;
    }
    

    3.在layoutSubviews方法中进行路径的设置

    - (void)layoutSubviews {
        [super layoutSubviews];
        
        // 遮罩层frame
        self.maskLayer.frame = self.bounds;
        
        // 设置path起点
        [self.borderPath moveToPoint:CGPointMake(0, 10)];
        // 左上角的圆角
        [self.borderPath addQuadCurveToPoint:CGPointMake(10, 0) controlPoint:CGPointMake(0, 0)];
        // 直线,到右上角
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width - 10, 0)];
        // 右上角的圆角
        [self.borderPath addQuadCurveToPoint:CGPointMake(self.bounds.size.width, 10) controlPoint:CGPointMake(self.bounds.size.width, 0)];
        // 直线,到右下角
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width, self.bounds.size.height)];
        // 底部的小三角形
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width/2+5, self.bounds.size.height)];
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width/2, self.bounds.size.height-5)];
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width/2 - 5, self.bounds.size.height)];
        // 直线,到左下角
        [self.borderPath addLineToPoint:CGPointMake(0, self.bounds.size.height)];
        // 直线,回到起点
        [self.borderPath addLineToPoint:CGPointMake(0, 10)];
        
        // 将这个path赋值给maskLayer的path
        self.maskLayer.path = self.borderPath.CGPath;
    }
    

    左上角的坐标是(0,0)
    右下角的坐标是(maxX,maxY)

    圆角是用的二次贝塞尔曲线,关于二次贝塞尔曲线,我在网上看到一张比较形象的图:


    图片来自CSDN

    最终效果

    - (void)viewDidLoad {
        [super viewDidLoad];
        // Do any additional setup after loading the view, typically from a nib.
        
        self.label = [[IrregularLabel alloc] initWithFrame:CGRectMake(90, 200, 200, 40)];
        [self.view addSubview:self.label];
        self.label.text = @"这是一个不规则label";
        self.label.textAlignment = NSTextAlignmentCenter;
        self.label.backgroundColor = [UIColor redColor];
        self.label.textColor = [UIColor whiteColor];
        self.label.font = [UIFont boldSystemFontOfSize:16];
    }
    

    效果图:

    demo

    https://github.com/CaiWanFeng/IrregularLabel

    总结


    2018年4月10日更新

    有读者说要带箭头的label:


    其实思路是一样的:

    - (void)layoutSubviews {
        [super layoutSubviews];
        
        // 遮罩层frame
        self.maskLayer.frame = self.bounds;
        
        
        // 设置path起点
        [self.borderPath moveToPoint:CGPointMake(0, 10)];
        
        // 箭头
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width/2-5, 10)];
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width/2, 0)];
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width/2+5, 10)];
        
        // 到右上角
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width, 10)];
        // 到右下角
        [self.borderPath addLineToPoint:CGPointMake(self.bounds.size.width, self.bounds.size.height)];
        // 到左下角
        [self.borderPath addLineToPoint:CGPointMake(0, self.bounds.size.height)];
        // 回到起点
        [self.borderPath addLineToPoint:CGPointMake(0, 10)];
        
        // 将这个path赋值给maskLayer的path
        self.maskLayer.path = self.borderPath.CGPath;
    }
    

    如果想要文本相对方形框居中,可以在view上放一个label。

    已更新到GitHub。

    相关文章

      网友评论

      • yask:android shape就可以搞定了
        yask:@CnPeng 可以用layer-list
        CnPeng:@黑马飞马 弱弱的问一句,用shape怎么实现?圆角和直角好实现,但凹进去的三角呢?这个怎么实现?
        Lol刀妹:我旁边的Android同事认同你的说法
      • 心语风尚:回答一下 :yum:
      • 心语风尚:很想实现一个效果 下面实现的剪头是很尖的 哪个属性可以不太尖 有点圆弧
        Lol刀妹:@心语风尚 我写了个圆尖角label,更新到github了
        心语风尚:@无夜之星辰 你demo没有剪头圆角
        Lol刀妹:label.layer.cornerRadius,如果这个不是你想要的,就按照我demo里面步骤的画。
      • 叶熙雯:大佬is大佬
        Lol刀妹:司机is司机
      • xiAo__Ju:圆角用bezier去切,三角用个图盖上去:smile:
        Lol刀妹:可以的:sunglasses:
      • TinXie:想請問 如果要 四個角都是圓角 跟 箭頭突出 該怎麼做呢!? 嘗試一下子 貌似無法成功
        TinXie:@幾分甜 不過還是有點問題的 如果把這 lab 套用在地圖的 MKAnnotationView 去 addSubview
        會有顯示上的問題 會有重複畫出不同的 線
        TinXie:@无夜之星辰 666 感謝前輩, 昨天認真的研讀了那張圖片, 理解後可以使用出, 與您相同的效果~
        Lol刀妹:文章更新了,可以看看:smile:
      • 谢谢生活:为何不写成分类呢,子类太麻烦复用也不高……
        Lol刀妹:@谢谢生活 因为只是在某一个页面需要这个控件,我觉得这里这种情况新建一个类比扩展类的方法更适合
      • W_SN:代码可控性更强一些,给力
        Lol刀妹:@SN_OS :smile:
      • 半缘魔君:我想知道 卖家推荐。换新大不同几个字怎么搞?
        Lol刀妹:@半缘魔君 具体哪种字体由你们的UI设计师来定
        半缘魔君:@无夜之星辰 就是你的例子中的文字怎么打上去的?字体不是苹方字体吧?
        Lol刀妹:@半缘魔君 前后台约定了不会返回这种不对称的文本:sweat_smile:
      • Just丶Go:吊大的麻烦求解:这种情况是让UI给个图来做好呢,还是版主这样绘图好呢?性能方面
        A天天涨不停:看情况呀。得要一个根据需求和场景来决定,得要量化才好进一步讨论,这个太抽象了。光是一个View里绘图和cell里绘图完全不一样的,cell里当日图片最好,View里就自己绘图,不过我一般都喜欢麻烦UI。但是对于上边圆角下边直角的,还是绘图好点,而且这种是cell的frame不常变,在优化上也非常的简单。
        Just丶Go:@无夜之星辰 :smiley: :pray:
        Lol刀妹:@Bonlion_Sui 我帮你@吊大的。@砖打各种不服
      • TinXie:好威猛!
        Lol刀妹:见笑见笑:smile:
      • PGOne爱吃饺子:就喜欢你的写博客的形式,结合项目来
        Lol刀妹:这样印象更深刻
      • 糊涂0:总结的很好!:joy:
        Lol刀妹:哈哈:joy:

      本文标题:iOS开发 | 自定义不规则label

      本文链接:https://www.haomeiwen.com/subject/seuicftx.html