1籍琳、UIEdgeInsetsMake介紹
首先簡單介紹一下UIEdgeInsetsMake
引用:UIEdgeInsetsMake使用詳解的圖片和講解哩盲,再加上自己的理解與實(shí)現(xiàn)
先看定義
typedef struct UIEdgeInsets {
CGFloat top, left, bottom, right; // specify amount to inset (positive) for each of the edges. values can be negative to 'outset'
} UIEdgeInsets;
UIEdgeInsets實(shí)際就是一個結(jié)構(gòu)體UIEdgeInsetsMake(CGFloat top , CGFloat left , CGFloat bottom , CGFloat right )艰管,要設(shè)置的就是四個邊距滓侍,(warning:left以右為正方向,right以左為正方向牲芋,這個不一定對撩笆、不一定對捺球、不一定對(說三遍),但是可以幫助理解下面進(jìn)行偏移時+ -距離夕冲,top向下為正和bottom向上為正)氮兵。
先看一張圖:
圖中,藍(lán)色標(biāo)識為可變區(qū)域歹鱼, 綠色標(biāo)識為不變區(qū)域泣栈。UIEdgeInsets結(jié)構(gòu)體的屬性top與bottom為一對,用來指定縱向可變區(qū)域(黑色虛線矩形)弥姻,left與right為一對南片,用來指定橫向可變區(qū)域(白色虛線矩形)。當(dāng)UIButton/UIImageView的size大于UIImage的size時庭敦,會調(diào)整圖片中可變區(qū)域大小以鋪滿整個控件,具體調(diào)整規(guī)則如下:
(1)控件寬度大于圖片寬度疼进,拉伸白色虛線矩形
(2)控件高度大于圖片高度,拉伸黑色虛線矩形
(3)控件寬度小于圖片寬度時秧廉,橫向整體縮小(可變區(qū)與不變區(qū)比例不變)
(4)控件高度小于圖片高度時伞广,縱向整體縮小(可變區(qū)與不變區(qū)比例不變)
說明:這四句總結(jié)我不知道你們有沒有理解,反正我是沒有理解疼电。
上面都是引用別人寫的東西嚼锄,也該說說自己的內(nèi)容了。
UIEdgeInsetsMake(CGFloat top , CGFloat left , CGFloat bottom , CGFloat right )
UIEdgeInsetsMake是一個用來描述內(nèi)容物在包裹容器里面的內(nèi)邊距的一個結(jié)構(gòu)體描述澜沟,用UIButton來舉例灾票,button里面的Image是內(nèi)容物,button是包裹容器茫虽,可以用UIEdgeInsetsMake結(jié)構(gòu)體來描述Image在button里面的內(nèi)邊距刊苍。
拿圖“UIEdgeInsets”來說就是設(shè)置UIEdgeInsetsMake以后,Image的位置就是黑色虛線和白色虛線重合的中間區(qū)域濒析。
這種理解有一個前提就是用來描述單獨(dú)內(nèi)容物正什,比如在UIButton里面你只設(shè)置了Image或者只設(shè)置了TitleLabel的時候,當(dāng)同時設(shè)置了Image和TitleLabel的時候号杏,會有一些差別婴氮,下面再說。
2盾致、UIEdgeInsetsMake在UIButton上的應(yīng)用
回到主題《UIButton的UIEdgeInsetsMake的使用》
這里主要是講UIEdgeInsetsMake在UIButton上的應(yīng)用主经,相信很多人都遇到了那種需求,就是原本button圖片在左庭惜,title在右罩驻,很多時候確實(shí)也是這樣,但是有時候需求可能是“圖片在上护赊,title在下”或者是“圖片在右,title在左”,如下圖所示:
這時候原來系統(tǒng)自帶按鈕就需要我們使用UIEdgeInsetsMake來進(jìn)行設(shè)置了猿棉。上面說到內(nèi)容物是單一的和多個的是不一樣,不一樣在哪抽高?
對于設(shè)置了image和title的button,系統(tǒng)會在設(shè)置以后自動設(shè)置一個合適的ImageEdgeInsets和TitleEdgeInsets
問:啥叫合適的透绩?
答:你new一個button顯示一下就會發(fā)現(xiàn)翘骂,他邊距確實(shí)蠻合適的,雖然有時候title擠在一起顯示不全帚豪,但是起碼邊距看上去蠻順眼雏胃,所以我也不知道啥叫合適的,我只知道new一個button志鞍,設(shè)置了image和title以后瞭亮,就會有一個默認(rèn)的ImageEdgeInsets和TitleEdgeInsets。
下面我們在設(shè)置image和title之后固棚,打印一下兩者的邊距:
//生成button
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
[button setTitle:@"這是一個按鈕" forState:UIControlStateNormal];
[button setImage:[UIImage imageNamed:@"image"] forState:UIControlStateNormal];
button.frame = CGRectMake(100, 100, 160, 40);
button.backgroundColor = [UIColor redColor];
button.titleLabel.backgroundColor = [UIColor purpleColor];
[self.view addSubview:button];
//打印before
NSLog(@"before-->%@",NSStringFromUIEdgeInsets(button.imageEdgeInsets));
NSLog(@"before-->%@",NSStringFromUIEdgeInsets(button.titleEdgeInsets));
//修改邊距
button.imageEdgeInsets = UIEdgeInsetsMake(0, button.titleLabel.intrinsicContentSize.width, 0, -button.titleLabel.intrinsicContentSize.width);
button.titleEdgeInsets = UIEdgeInsetsMake(0, -button.currentImage.size.width, 0, button.currentImage.size.width);
//打印after
NSLog(@"after-->%@",NSStringFromUIEdgeInsets(button.imageEdgeInsets));
NSLog(@"after-->%@",NSStringFromUIEdgeInsets(button.titleEdgeInsets));
會發(fā)現(xiàn)兩者的邊距统翩,會發(fā)現(xiàn)兩者打印出來都是(0,0此洲,0厂汗,0)和(0,0呜师,0娶桦,0)。
WTF汁汗?衷畦??你是來逗我的知牌,這邊距明顯都不為0啊祈争,所以我猜想,這個其實(shí)并不是image和titleLabel相對于button的邊距角寸,這是在原有基礎(chǔ)邊距基礎(chǔ)上進(jìn)行的一個偏移菩混。
上面我們說到,在給button設(shè)置了image和title以后扁藕,系統(tǒng)會自動設(shè)置一個合適(多合適沮峡,看原始按鈕那個圖就知道,確實(shí)蠻合適的這邊距)的ImageEdgeInsets和TitleEdgeInsets亿柑,所以我們此時打印before其實(shí)可能打印的并不是真實(shí)的imageEdgeInsets和titleEdgeInsets邢疙,而是我們設(shè)置的EdgeInsets,因為我們沒有設(shè)置,所以都打用刂ⅰ(0,0式矫,0乡摹,0),后來我們設(shè)置以后采转,打印出了{(lán)0, 110.5, 0, -110.5}和{0, -23, 0, 23}聪廉。
現(xiàn)在就要說說設(shè)置的這兩行代碼:
button.imageEdgeInsets = UIEdgeInsetsMake(0, button.titleLabel.intrinsicContentSize.width, 0, -button.titleLabel.intrinsicContentSize.width);
button.titleEdgeInsets = UIEdgeInsetsMake(0, -button.currentImage.size.width, 0, button.currentImage.size.width);
一個更新image的邊距,一個更新title的邊距故慈。
首先我們說更新image的邊距板熊,我們需要讓image右移一個titleLabel的寬度的距離,所以我們需要image的左右邊距都右移一個titleLabel察绷。
還記得我們一開始設(shè)定的正負(fù)方向嗎干签,(warning:left以右為正方向,right以左為正方向拆撼,但是可以幫助理解下面進(jìn)行偏移時+ -距離容劳,top向下為正和bottom向上為正,這個可以參照使用masonry進(jìn)行約束設(shè)置判定方向)闸度。
我們使用button.titleLabel.intrinsicContentSize.width計算titleLabel的寬度(tips:使用button.titleLabel.bounds.size.width的在iOS8以上會得到寬度為0的結(jié)果竭贩,造成錯誤的結(jié)果),然后因為image的left和right都要向右偏移一個titleLabel的寬度莺禁,而left以右為正方向留量,right以左為正方向,所以是設(shè)置的是UIEdgeInsetsMake(0, button.titleLabel.intrinsicContentSize.width, 0, -button.titleLabel.intrinsicContentSize.width);
同理哟冬,titleLabel需要向左移一個image的寬度楼熄,按照我們預(yù)想的正負(fù)方向進(jìn)行設(shè)置:button.titleEdgeInsets = UIEdgeInsetsMake(0, -button.currentImage.size.width, 0, button.currentImage.size.width);
這里需要注意使用button.currentImage.size.width
計算得到的效果要好于button.imageView計算出來的width
總結(jié)模式:
- 圖片在左,title在右
- 圖片在上浩峡,title在下
- 圖片在右孝赫,title在左
- 圖片在下,title在上
貼出代碼红符,就不一一講解了:
- (void)layoutButtonWithEdgeInsetsStyle:(MKButtonEdgeInsetsStyle)style
imageTitleSpace:(CGFloat)space {
// 1. 得到imageView和titleLabel的寬青柄、高
// CGFloat imageWith = self.imageView.frame.size.width;
// CGFloat imageHeight = self.imageView.frame.size.height;
CGFloat imageWith = self.currentImage.size.width;
CGFloat imageHeight = self.currentImage.size.height;
CGFloat labelWidth = 0.0;
CGFloat labelHeight = 0.0;
if ([UIDevice currentDevice].systemVersion.floatValue >= 8.0) {
// 由于iOS8中titleLabel的size為0,用下面的這種設(shè)置
labelWidth = self.titleLabel.intrinsicContentSize.width;
labelHeight = self.titleLabel.intrinsicContentSize.height;
} else {
labelWidth = self.titleLabel.frame.size.width;
labelHeight = self.titleLabel.frame.size.height;
}
// 2. 聲明全局的imageEdgeInsets和labelEdgeInsets
UIEdgeInsets imageEdgeInsets = UIEdgeInsetsZero;
UIEdgeInsets labelEdgeInsets = UIEdgeInsetsZero;
// 3. 根據(jù)style和space得到imageEdgeInsets和labelEdgeInsets的值
switch (style) {
case MKButtonEdgeInsetsStyleTop: {
imageEdgeInsets = UIEdgeInsetsMake(-labelHeight-space, 0, 0, -labelWidth);
labelEdgeInsets = UIEdgeInsetsMake(0, -imageWith, -imageHeight-space, 0);
}
break;
case MKButtonEdgeInsetsStyleLeft: {
imageEdgeInsets = UIEdgeInsetsMake(0, -space, 0, space);
labelEdgeInsets = UIEdgeInsetsMake(0, space, 0, -space);
}
break;
case MKButtonEdgeInsetsStyleBottom: {
imageEdgeInsets = UIEdgeInsetsMake(0, 0, -labelHeight-space, -labelWidth);
labelEdgeInsets = UIEdgeInsetsMake(-imageHeight-space, -imageWith, 0, 0);
}
break;
case MKButtonEdgeInsetsStyleRight: {
imageEdgeInsets = UIEdgeInsetsMake(0, labelWidth+space, 0, -labelWidth-space);
labelEdgeInsets = UIEdgeInsetsMake(0, -imageWith-space, 0, imageWith+space);
}
break;
default:
break;
}
// 4. 賦值
self.titleEdgeInsets = labelEdgeInsets;
self.imageEdgeInsets = imageEdgeInsets;
}