使用Autolayout實現(xiàn)cell自適應高度

-------推薦使用最下方的【改進】方法-------
(前排補上一個簡單的小Demo:https://github.com/3KK3/AutoFitDemo/tree/master
時間倉促炼杖,有些許不足喉酌,敬請見諒厉碟。)

方法一:

  1. 對cell的subview增加針對于contentView的約束(注意設(shè)置文字的最大寬度);

  2. 設(shè)置tableview預估高度estimateHeight

  3. 在cell拿到數(shù)據(jù)的地方cellForRowAtIndexPath,進行強制刷新布局:先根據(jù)數(shù)據(jù)和cell控件要求計算實際高度,并存入dataModel模型(做緩存),然后調(diào)用方法[self layoutIfNeeded](第一次顯示內(nèi)容的時候 無需調(diào)用setNeedsLayout),這樣heightForIndexPath會緊隨其后獲取cell的真實高度,此時存入模型的cell的高度就起了作用了

PS:

預估高度機制 啟用后 改變cell的加載順序

  • 原來的順序 : 先調(diào)用heightForRowAtIndexPath獲取真實高度,拿到高度之后再調(diào)用cellForRowAtIndexPath渲染cell內(nèi)容

  • 啟用預估高度之后:先調(diào)用estimatedHeightForRowAtIndexPath返回代理估計高度秕磷,然后再根據(jù)預估高度調(diào)用cellForRowAtIndexPath渲染cell內(nèi)容,同時計算真實高度,最后再調(diào)用heightForRowAtIndexPath拿到計算的真實高度 顯示


方法二:

  1. 對cell的subview增加針對于contentView的約束(注意設(shè)置文字的最大寬度)

-(CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {

       //只創(chuàng)建一個cell用作測量高度

  static MyCell *cell = nil;

  if (!cell) cell = [self.tableView  dequeueReusableCellWithIdentifier:@"MyCell"];

     //這里把數(shù)據(jù)設(shè)置給Cell

  cell.titleLabel.text = [_dataSource objectAtIndex:indexPath.row];

  [cell layoutIfNeeded];

  [cell updateConstraintsIfNeeded];



CGFloat height = [cell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize].height;

return height;

}


方法三:

1.在cell中設(shè)置多行l(wèi)abel的約束:

// 計算UILabel的preferredMaxLayoutWidth值,多行時必須設(shè)置這個值,否則系統(tǒng)無法決定Label的寬度
CGFloat preferredMaxWidth = [UIScreen mainScreen].bounds.size.width - (16 + 4) * 2 - 44 - 4;
// Content - 多行
_contentLabel = [UILabel new];
_contentLabel.numberOfLines = 0;
_contentLabel.preferredMaxLayoutWidth = preferredMaxWidth; // 多行時必須設(shè)置
[self.contentView addSubview:_contentLabel];
[_contentLabel mas_makeConstraints:^(MASConstraintMaker *make) {   

    make.top.equalTo(_titleLabel.mas_bottom).with.offset(4);    
    make.left.equalTo(_avatarImageView.mas_right).with.offset(4);   
    make.right.equalTo(self.contentView).with.offset(-4); 
    make.bottom.equalTo(self.contentView).with.offset(-4);}
];
  // 設(shè)置高度的Content Hugging
  [_contentLabel setContentHuggingPriority:UILayoutPriorityRequired forAxis:UILayoutConstraintAxisVertical];

2.UITableView
再看看UITableView。
用systemLayoutSizeFittingSize:獲取Cell的高度
在設(shè)定好Cell的約束以后哭靖,就可以用systemLayoutSizeFittingSize:方法獲取Cell的實際高度,它的參數(shù)可以設(shè)定為兩個系統(tǒng)常量侈离,如下:
UILayoutFittingCompressedSize: 返回合適的最小大小试幽。
UILayoutFittingExpandedSize: 返回合適的最大大小。

為了在“- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath ”方法中計算Cell的高度霍狰,我們需要一個專門用于計算高度的Cell實例抡草,可以說算是Cell的“模板”。一般來說蔗坯,這個實例可以設(shè)置成函數(shù)的static變量康震,并只在第一次使用時初始化一次。

model簡單緩存高度
為了避免每次滑動時計算高度宾濒,可以將Cell的高度緩存下來腿短。如,保存在每一行對應的數(shù)據(jù)Model(Entity)中,例如:

@interface Entity : NSObject
// Data
@property (copy, nonatomic) NSString *title;
// ...// Cache height
@property (assign, nonatomic) CGFloat cellHeight;
@end

tableView代理方法中代碼:

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
 static Case4Cell *templateCell; static dispatch_once_t onceToken; 
dispatch_once(&onceToken, ^{ templateCell = [tableView dequeueReusableCellWithIdentifier:NSStringFromClass([Case4Cell class])]; });
 // 獲取對應的數(shù)據(jù)
 Case4DataEntity *dataEntity = _data[(NSUInteger) indexPath.row];
 // 填充數(shù)據(jù) 
[templateCell setupData:dataEntity];
 // 判斷高度是否已經(jīng)計算過
 if (dataEntity.cellHeight <= 0) { 
  // 根據(jù)當前數(shù)據(jù)橘忱,計算Cell的高度赴魁,
  // ------ 注意+1 ------
  dataEntity.cellHeight = [templateCell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize].height + 1; 
}
 return dataEntity.cellHeight;
}

改進:

iOS 8的新特性
iOS 8大大簡化了Cell的高度計算,只要:
1.設(shè)置好Cell中控件的約束;

  1. 然后 UITabelView初始化時候:
// UITabelView初始化時候
tableView.rowHeight = UITableViewAutomaticDimension;
tableView.estimatedRowHeight = 80;

3.在UITableview 代理方法:

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { 
  // 只用返回這個钝诚! 
  return UITableViewAutomaticDimension;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末颖御,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子凝颇,更是在濱河造成了極大的恐慌潘拱,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,539評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件拧略,死亡現(xiàn)場離奇詭異芦岂,居然都是意外死亡,警方通過查閱死者的電腦和手機垫蛆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評論 3 396
  • 文/潘曉璐 我一進店門禽最,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人袱饭,你說我怎么就攤上這事川无。” “怎么了宁赤?”我有些...
    開封第一講書人閱讀 165,871評論 0 356
  • 文/不壞的土叔 我叫張陵舀透,是天一觀的道長栓票。 經(jīng)常有香客問我决左,道長,這世上最難降的妖魔是什么走贪? 我笑而不...
    開封第一講書人閱讀 58,963評論 1 295
  • 正文 為了忘掉前任佛猛,我火速辦了婚禮,結(jié)果婚禮上坠狡,老公的妹妹穿的比我還像新娘继找。我一直安慰自己,他們只是感情好逃沿,可當我...
    茶點故事閱讀 67,984評論 6 393
  • 文/花漫 我一把揭開白布婴渡。 她就那樣靜靜地躺著馋记,像睡著了一般贵涵。 火紅的嫁衣襯著肌膚如雪罗捎。 梳的紋絲不亂的頭發(fā)上歼狼,一...
    開封第一講書人閱讀 51,763評論 1 307
  • 那天籽慢,我揣著相機與錄音怠堪,去河邊找鬼弛饭。 笑死密幔,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的臼予。 我是一名探鬼主播鸣戴,決...
    沈念sama閱讀 40,468評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼粘拾!你這毒婦竟也來了窄锅?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤缰雇,失蹤者是張志新(化名)和其女友劉穎酬滤,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體寓涨,經(jīng)...
    沈念sama閱讀 45,850評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡盯串,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,002評論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了戒良。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片体捏。...
    茶點故事閱讀 40,144評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖糯崎,靈堂內(nèi)的尸體忽然破棺而出几缭,到底是詐尸還是另有隱情,我是刑警寧澤沃呢,帶...
    沈念sama閱讀 35,823評論 5 346
  • 正文 年R本政府宣布年栓,位于F島的核電站,受9級特大地震影響薄霜,放射性物質(zhì)發(fā)生泄漏某抓。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,483評論 3 331
  • 文/蒙蒙 一惰瓜、第九天 我趴在偏房一處隱蔽的房頂上張望否副。 院中可真熱鬧,春花似錦崎坊、人聲如沸备禀。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽曲尸。三九已至,卻和暖如春男翰,著一層夾襖步出監(jiān)牢的瞬間另患,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評論 1 272
  • 我被黑心中介騙來泰國打工奏篙, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留柴淘,地道東北人迫淹。 一個月前我還...
    沈念sama閱讀 48,415評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像为严,于是被迫代替她去往敵國和親敛熬。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,092評論 2 355

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