iOS筆記之UIButton的UIEdgeInsetsMake使用詳解

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向上為正)氮兵。

先看一張圖:


圖UIEdgeInsets

圖中,藍(lán)色標(biāo)識為可變區(qū)域歹鱼, 綠色標(biāo)識為不變區(qū)域泣栈。UIEdgeInsets結(jié)構(gòu)體的屬性topbottom為一對,用來指定縱向可變區(qū)域(黑色虛線矩形)弥姻,leftright為一對南片,用來指定橫向可變區(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)。

打印結(jié)果

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;
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末预侯,一起剝皮案震驚了整個濱河市致开,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌萎馅,老刑警劉巖双戳,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異糜芳,居然都是意外死亡飒货,警方通過查閱死者的電腦和手機(jī)魄衅,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來塘辅,“玉大人晃虫,你說我怎么就攤上這事】鄱眨” “怎么了哲银?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長呻惕。 經(jīng)常有香客問我荆责,道長,這世上最難降的妖魔是什么亚脆? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任做院,我火速辦了婚禮,結(jié)果婚禮上濒持,老公的妹妹穿的比我還像新娘山憨。我一直安慰自己,他們只是感情好弥喉,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布郁竟。 她就那樣靜靜地躺著,像睡著了一般由境。 火紅的嫁衣襯著肌膚如雪棚亩。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天虏杰,我揣著相機(jī)與錄音讥蟆,去河邊找鬼。 笑死纺阔,一個胖子當(dāng)著我的面吹牛瘸彤,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播笛钝,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼质况,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了玻靡?” 一聲冷哼從身側(cè)響起结榄,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎囤捻,沒想到半個月后臼朗,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年视哑,在試婚紗的時候發(fā)現(xiàn)自己被綠了绣否。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡挡毅,死狀恐怖蒜撮,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情慷嗜,我是刑警寧澤,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布丹壕,位于F島的核電站庆械,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏菌赖。R本人自食惡果不足惜缭乘,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望琉用。 院中可真熱鬧堕绩,春花似錦、人聲如沸邑时。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽晶丘。三九已至黍氮,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間浅浮,已是汗流浹背沫浆。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留滚秩,地道東北人专执。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像郁油,于是被迫代替她去往敵國和親本股。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評論 2 354

推薦閱讀更多精彩內(nèi)容