iOS筆記之UIButton的UIEdgeInsetsMake使用詳解

1、UIEdgeInsetsMake介紹

首先簡單介紹一下UIEdgeInsetsMake
引用:UIEdgeInsetsMake使用詳解的圖片和講解,再加上自己的理解與實現

先看定義

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實際就是一個結構體UIEdgeInsetsMake(CGFloat top , CGFloat left , CGFloat bottom , CGFloat right ),要設置的就是四個邊距,(warning:left以右為正方向,right以左為正方向,這個不一定對、不一定對、不一定對(說三遍),但是可以幫助理解下面進行偏移時+ -距離,top向下為正和bottom向上為正)。

先看一張圖:


圖UIEdgeInsets

圖中,藍色標識為可變區域, 綠色標識為不變區域。UIEdgeInsets結構體的屬性topbottom為一對,用來指定縱向可變區域(黑色虛線矩形),leftright為一對,用來指定橫向可變區域(白色虛線矩形)。當UIButton/UIImageView的size大于UIImage的size時,會調整圖片中可變區域大小以鋪滿整個控件,具體調整規則如下:
(1)控件寬度大于圖片寬度,拉伸白色虛線矩形
(2)控件高度大于圖片高度,拉伸黑色虛線矩形
(3)控件寬度小于圖片寬度時,橫向整體縮小(可變區與不變區比例不變)
(4)控件高度小于圖片高度時,縱向整體縮小(可變區與不變區比例不變)
說明:這四句總結我不知道你們有沒有理解,反正我是沒有理解。


上面都是引用別人寫的東西,也該說說自己的內容了。

UIEdgeInsetsMake(CGFloat top , CGFloat left , CGFloat bottom , CGFloat right )

UIEdgeInsetsMake是一個用來描述內容物在包裹容器里面的內邊距的一個結構體描述,用UIButton來舉例,button里面的Image是內容物,button是包裹容器,可以用UIEdgeInsetsMake結構體來描述Image在button里面的內邊距。
拿圖“UIEdgeInsets”來說就是設置UIEdgeInsetsMake以后,Image的位置就是黑色虛線和白色虛線重合的中間區域。
這種理解有一個前提就是用來描述單獨內容物,比如在UIButton里面你只設置了Image或者只設置了TitleLabel的時候,當同時設置了Image和TitleLabel的時候,會有一些差別,下面再說。

2、UIEdgeInsetsMake在UIButton上的應用

回到主題《UIButton的UIEdgeInsetsMake的使用》
這里主要是講UIEdgeInsetsMake在UIButton上的應用,相信很多人都遇到了那種需求,就是原本button圖片在左,title在右,很多時候確實也是這樣,但是有時候需求可能是“圖片在上,title在下”或者是“圖片在右,title在左”,如下圖所示:

原始按鈕
修改后的按鈕

這時候原來系統自帶按鈕就需要我們使用UIEdgeInsetsMake來進行設置了。上面說到內容物是單一的和多個的是不一樣,不一樣在哪?

對于設置了image和title的button,系統會在設置以后自動設置一個合適的ImageEdgeInsets和TitleEdgeInsets

問:啥叫合適的?
答:你new一個button顯示一下就會發現,他邊距確實蠻合適的,雖然有時候title擠在一起顯示不全,但是起碼邊距看上去蠻順眼,所以我也不知道啥叫合適的,我只知道new一個button,設置了image和title以后,就會有一個默認的ImageEdgeInsets和TitleEdgeInsets。

下面我們在設置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));

會發現兩者的邊距,會發現兩者打印出來都是(0,0,0,0)和(0,0,0,0)。

打印結果

WTF???你是來逗我的,這邊距明顯都不為0啊,所以我猜想,這個其實并不是image和titleLabel相對于button的邊距,這是在原有基礎邊距基礎上進行的一個偏移。

上面我們說到,在給button設置了image和title以后,系統會自動設置一個合適(多合適,看原始按鈕那個圖就知道,確實蠻合適的這邊距)的ImageEdgeInsets和TitleEdgeInsets,所以我們此時打印before其實可能打印的并不是真實的imageEdgeInsets和titleEdgeInsets,而是我們設置的EdgeInsets,因為我們沒有設置,所以都打印(0,0,0,0),后來我們設置以后,打印出了{0, 110.5, 0, -110.5}和{0, -23, 0, 23}
現在就要說說設置的這兩行代碼:

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。
還記得我們一開始設定的正負方向嗎,(warning:left以右為正方向,right以左為正方向,但是可以幫助理解下面進行偏移時+ -距離,top向下為正和bottom向上為正,這個可以參照使用masonry進行約束設置判定方向)。

我們使用button.titleLabel.intrinsicContentSize.width計算titleLabel的寬度(tips:使用button.titleLabel.bounds.size.width的在iOS8以上會得到寬度為0的結果,造成錯誤的結果),然后因為image的left和right都要向右偏移一個titleLabel的寬度,而left以右為正方向,right以左為正方向,所以是設置的是UIEdgeInsetsMake(0, button.titleLabel.intrinsicContentSize.width, 0, -button.titleLabel.intrinsicContentSize.width);
同理,titleLabel需要向左移一個image的寬度,按照我們預想的正負方向進行設置:button.titleEdgeInsets = UIEdgeInsetsMake(0, -button.currentImage.size.width, 0, button.currentImage.size.width);
這里需要注意使用button.currentImage.size.width計算得到的效果要好于button.imageView計算出來的width


總結模式:

  • 圖片在左,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,用下面的這種設置
        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. 根據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;
}

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,119評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,382評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,038評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,853評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,616評論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,112評論 1 323
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,192評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,355評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,869評論 1 334
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,727評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,928評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,467評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,165評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,570評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,813評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,585評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,892評論 2 372

推薦閱讀更多精彩內容