UIButton自定义图片文字位置总结

UIButton的默认显示样式是图片居左,文字居右,如下图1:


图1.png

但是有时候我们根据需求需要调整按钮中图片和标题的位置,最常见的就是图片居上,文字居下显示的按钮,如下图2所示:


图2.png

实现这种居中显示的图上文下的按钮方式有很多,比如最简单的方法:自定义View,上面放一个UIButton一个UILabel子控件。

这里我们介绍其他方法来实现UIButton图片和标题的位置调整:

  • 一种是通过类别(Category)调整位置;
  • 一种是通过继承重写系统方法调整位置。

通过类别(Category)调整位置

通过类别的方法修改图片标题的位置主要是用了UIButton的两个位置属性:titleEdgeInsetsimageEdgeInsets。通过修改按钮中imagetitleedgeInsets来改变相对位置。

@property(nonatomic)          UIEdgeInsets titleEdgeInsets;                // default is UIEdgeInsetsZero
@property(nonatomic)          UIEdgeInsets imageEdgeInsets;                // default is UIEdgeInsetsZero

UIEdgeInsets有四个参数:top, left, bottom, right,分别代表上左下右的偏移量。

top : 为正数的时候,是往下偏移,为负数的时候往上偏移;
left : 为正数的时候往右偏移,为负数的时候往左偏移;
bottom : 为正数的时候往上偏移,为负数的时候往下偏移;
right :为正数的时候往左偏移,为负数的时候往右偏移;

修改按钮中image和title位置的关键就是正确的设置这两个属性值的参数,网上很多介绍的方法里写的都是固定参数,如果每次使用都要去计算的话那就太麻烦了,这里我们通过获取image和title的size来设定偏移量,实现方法的通用。这也是创建一个UIButton的category的原因,方便以后的复用。

下面贴上category的代码,其中计算titleEdgeInsetsimageEdgeInsets的代码是最核心的地方,具体的计算思路可以从代码里看出来,就不文字介绍了。这里的计算思路是最关键的地方,只要知道怎么计算,就可以使用UIButton实现其他图文效果,而不止是图上文下的居中效果。

#import <UIKit/UIKit.h>

@interface UIButton (ZXCenterButton)
/**
 图片居上,文字居下,图文居中按钮

 @param spaceBetween image和label之间的间距,若设为nil则距离为0
 @param fontSize 标题字体大小,若设为nil则为默认字体大小17
 */
-(instancetype)initCenterBtnWithFrame:(CGRect)frame
                         spaceBetween:(NSNumber*)spaceBetween
                           imageNamed:(NSString *)imageName
                                title:(NSString *)title
                             fontSize:(NSNumber*)fontSize;

@end
#import "UIButton+ZXCenterButton.h"

@implementation UIButton (ZXCenterButton)

-(instancetype)initCenterBtnWithFrame:(CGRect)frame spaceBetween:(NSNumber*)spaceBetween imageNamed:(NSString *)imageName title:(NSString *)title fontSize:(NSNumber*)fontSize{

    if (self = [super initWithFrame:frame]) {
        CGFloat font = fontSize == nil ? 17 : [fontSize floatValue];
        [self setImage:[UIImage imageNamed:imageName] forState:UIControlStateNormal];
        [self setTitle:title forState:UIControlStateNormal];
        self.titleLabel.font = [UIFont systemFontOfSize:font];

        //  图片的宽高
        CGFloat imageW = self.imageView.frame.size.width;
        CGFloat imageH = self.imageView.frame.size.height;
       
        //  计算标题的size
        CGSize titleSize = [self string:title sizeWithFont:[UIFont systemFontOfSize:font]];
        
        // 图片+标题的总宽高
        CGFloat totalW = imageW + titleSize.width;
        CGFloat totalH = imageH + titleSize.height + spaceBetween.floatValue;
        // 设置按钮图片偏移
        [self setImageEdgeInsets:UIEdgeInsetsMake(-(totalH/2 - imageH/2), totalW/2 - imageW/2, (totalH/2 - imageH/2), -(totalW/2 - imageW/2))];
        // 设置按钮标题偏移
        [self setTitleEdgeInsets:UIEdgeInsetsMake((totalH/2 - titleSize.height/2), -(totalW/2 - titleSize.width/2), -(totalH/2 - titleSize.height/2), totalW/2 - titleSize.width/2)];
    }
    return self;
}

//  根据文字的长度、字体  来计算size
- (CGSize)string:(NSString *)string sizeWithFont:(UIFont *)font{
    NSMutableDictionary *attrs = [NSMutableDictionary dictionary];
    attrs[NSFontAttributeName] = font;
    CGSize maxSize = CGSizeMake(MAXFLOAT, MAXFLOAT);
    return [string boundingRectWithSize:maxSize options:NSStringDrawingUsesLineFragmentOrigin attributes:attrs context:nil].size;
}

@end

这种方法使用的时候注意Button的宽度设置和title的string长度,如果按钮宽度太小,标题的string太长,title无法全部显示时,label的宽度计算会出现偏差造成布局不居中。所以使用的时候要确保title能完全显示


通过继承调整位置

通过继承的方法调整image和label的位置其实也有两种实现方法:

  • 一种是在- (void)layoutSubviews里面重写self.titleLabel.frameself.imageView.frame的rect值;
  • 另一种就是本文要介绍的,重写系统方法改变title和image的布局。

通过继承需要重写的两个系统方法是:

 Objective-C
- (CGRect)titleRectForContentRect:(CGRect)contentRect;  
- (CGRect)imageRectForContentRect:(CGRect)contentRect;

具体实现方法直接见代码:

- (CGRect)titleRectForContentRect:(CGRect)contentRect
{
    CGRect titleRect = CGRectMake(0, CGRectGetHeight(self.bounds)/3 * 2, CGRectGetWidth(self.bounds),CGRectGetHeight(self.bounds)/3);
    return titleRect;
}

- (CGRect)imageRectForContentRect:(CGRect)contentRect
{
    CGFloat width = CGRectGetWidth(self.bounds);
    //  图片占按钮高度的2/3
    CGRect imageRect = CGRectMake(0, 0, width, CGRectGetHeight(self.bounds)/3 * 2);

    return imageRect;
}

使用这种方法的时候记得按钮的标题需要设置为居中,此方法来自我朋友的一篇文章,详情请移步,谢谢。


总结

解决问题的方法有各种各样,重要的是能解决问题,每个人的偏好都不一样。本文介绍的两种方法,第一种通过类别修改位置需要对UIEdgeInsets属性理解透彻而且需要有很好的图形位置想象能力,否则自己很容易搞晕,但是类别的好处是以后直接拿来用就可以,不用再创建类复制粘贴了;第二种计算就相对来说比较简单了,但是需要创建类来继承,复制粘贴代码。大家选择适合自己的就好。

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

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,022评论 4 62
  • 关于UIButton大家都很熟悉,系统默认的样式,是image在左,title在右的,如下图所示: 但是,很多情况...
    流火绯瞳阅读 25,520评论 16 63
  • 据说这个软件还蛮好写的,总要试试看。 然而他们说得markdown又是怎么回事?是在这个页面里面写吗?然而似乎我不...
    三生无忌阅读 153评论 0 0
  • 希望你带我吃饭 是因为希望我吃的健康更强壮 希望你给我发消息是因为想我 想知道我的一切 希望你没有联系我的时候忽略...
    小清楼阅读 130评论 0 0
  • 我的CSDN博客同步发布:Android OpenGL入门 转载请注明出处:【huachao1001的简书:htt...
    huachao1001阅读 14,074评论 3 67