UITableView+FDTemplateLayoutCell計算行高一些缺陷的解決方法

前言

UITableView動態行高一直是iOS的一個經典問題,在沒有AutoLayout的時代,只能自己計算frame,然后返回給代理,非常痛苦。到了AutoLayout的時代,布局就變得簡單多了,甚至于通過系統提供的API都能自動計算出行高。

UITableView+FDTemplateLayoutCell就是sunnyxx大大的一個自動計算行高的框架。只要布局正確,通過它可以自動計算并緩存行高,非常方便。不過在使用上發現一些問題,也嘗試去解決了。

過程

需求是這樣的,一個類似微博的頁面,像這樣:

WechatIMG61.jpeg

這應該是比較經典的布局,內容和圖片都是不確定的,行高要根據實際數據計算。九宮格實現方式有很多,我這里是通過UICollectionView去實現的。這樣的一個好處就是UICollectionView的高度可以通過它的collectionViewLayout對象獲取,啥都不用算。不過會有一個問題,UICollectionView繼承自UIScrollView,它的高度沒法按照內容來全顯示。所以即使布局正確,通過AutoLayout來計算行高也是不包括UICollectionView的,這個問題同樣反映在一些UIView控件上。

這就十分蛋疼了,難道還要回到手算frame的時代?當然不是,是我還寫啥博客。

我說下解決的幾個方法。

方法一(不推薦):手動設置collectionView的高度,可以通過代碼或者xib來設置,我這里是xib。

image.png

像這樣手動指定collectionView的高度,然后賦值數據源的時候更新collectionView高度約束就可以了,讓它的高度等于它的contentSize.height,這樣就能全部顯示了,其它UIView控件也能這么解決。但是這樣在計算行高的時候會拋出非常多異常,都是約束的問題。我不是很清楚這是什么原因,按理說計算再后,賦值在前,應該不會這樣。而且顯示會出一些問題,計算的行高會不正確,有些許誤差。

方法二(推薦):既然不能通過這種方式,那就繞個彎吧。去掉高度約束,計算出來的高就不包含collectionView的高。然后再手動加上collectionView的高返回給代理不就行了。不過看下UITableView+FDTemplateLayoutCell的拓展方法:

- (CGFloat)fd_heightForCellWithIdentifier:(NSString *)identifier 
                            configuration:(void (^)(id cell))configuration;

只有一個設置cell數據源的block,正常情況下我們只需要把cell換成我們自己的類,然后賦值模型就行,緩存之類的框架會自動處理好。雖然我們可以獲取到緩存高度之后再加上collectionView的高,但是這樣還叫啥緩存,緩存就是不需要計算,直接取到就能用,那怎么辦呢?
雖然可以通過Method Swizzling黑魔法交換方法實現,但是這并不是最優方法,往往是一些莫名其妙的bug的源泉,作為開發者應該盡量避免這種方式。所以最后我選擇了通過分類的方式。思路是在框架計算完高度之后通過block返回,我們自行處理行高,加加減減,然后返回高度讓框架緩存。
具體代碼:
我們參考下框架這個方法的實現

- (CGFloat)fd_heightForCellWithIdentifier:(NSString *)identifier 
                         cacheByIndexPath:(NSIndexPath *)indexPath 
                            configuration:(void (^)(id cell))configuration {
    if (!identifier || !indexPath) {
        return 0;
    }
    
    //命中緩存
    if ([self.fd_indexPathHeightCache existsHeightAtIndexPath:indexPath]) {
        [self fd_debugLog:[NSString stringWithFormat:@"hit cache by index path[%@:%@] - %@", @(indexPath.section), @(indexPath.row), @([self.fd_indexPathHeightCache heightForIndexPath:indexPath])]];
        return [self.fd_indexPathHeightCache heightForIndexPath:indexPath];
    }
    
    //計算行高
    CGFloat height = [self fd_heightForCellWithIdentifier:identifier configuration:configuration];
    [self.fd_indexPathHeightCache cacheHeight:height byIndexPath:indexPath];
    [self fd_debugLog:[NSString stringWithFormat: @"cached by index path[%@:%@] - %@", @(indexPath.section), @(indexPath.row), @(height)]];
    
    return height;
}

它先從緩存中尋找行高,命中之后直接返回。否則計算行高,存入緩存,然后返回。所以很簡單,我們可以直接復制它的代碼。寫一個帶編輯行高功能的方法:

typedef CGFloat(^editCellHeightAction)(id cell, CGFloat cellHeight);

- (CGFloat)jh_heightForCellWithIdentifier:(NSString *)identifier
                         cacheByIndexPath:(NSIndexPath *)indexPath
                            configuration:(void (^)(id cell))configuration
                               editAction:(editCellHeightAction)editAction {
    if (!identifier || !indexPath) {
        return 0;
    }
    // Hit cache
    if ([self.fd_indexPathHeightCache existsHeightAtIndexPath:indexPath]) {
        [self fd_debugLog:[NSString stringWithFormat:@"hit cache by index path[%@:%@] - %@", @(indexPath.section), @(indexPath.row), @([self.fd_indexPathHeightCache heightForIndexPath:indexPath])]];
        return [self.fd_indexPathHeightCache heightForIndexPath:indexPath];
    }
    
    CGFloat height = 0;
    //獲取緩存中的cell
    UITableViewCell *templateLayoutCell = [self fd_templateCellForReuseIdentifier:identifier];
    //這里插入編輯行高的代碼
    if (editAction) {
        height = editAction(templateLayoutCell, [self fd_heightForCellWithIdentifier:identifier configuration:configuration]);
    }
    else {
        height = [self fd_heightForCellWithIdentifier:identifier configuration:configuration];
    }
    
    [self.fd_indexPathHeightCache cacheHeight:height byIndexPath:indexPath];
    [self fd_debugLog:[NSString stringWithFormat: @"cached by index path[%@:%@] - %@", @(indexPath.section), @(indexPath.row), @(height)]];
    return height;
}
}

使用起來像這樣:

- (CGFloat)tableView:(UITableView *)tableView
heightForRowAtIndexPath:(NSIndexPath *)indexPath {
     return [tableView jh_heightForCellWithIdentifier:@"MineCell" cacheByIndexPath:indexPath configuration:^(HeSquareCell *cell) {
            //正常賦值數據源
            cell.model = self.model;
        } editAction:^CGFloat(MineCell *cell, CGFloat cellHeight) {
            //cellHeight是上面的block計算后回調過來的 所以直接加上額外的高度即可
            //因為緩存的關系這里只會走一次 所以可以放心寫
            return cellHeight + [cell collectionViewHeightWithModel:self.model];
        }];
}

這樣高度就能正常顯示了,而且也不會拋異常,還能享受框架帶來的便利。
UITableView+FDTemplateLayoutCell的接口設計很易于拓展,所以寫起來很簡單。還有個問題,我發現在使用這個框架的時候,如果_tableView.tableFooterView = [[UIView alloc] init];這句話寫在注冊cell之前,程序會crash,不造為啥。如果各位有更好的解決思路或者文中有錯誤的地方歡迎給我留言。

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

推薦閱讀更多精彩內容

  • 發現 關注 消息 iOS 第三方庫、插件、知名博客總結 作者大灰狼的小綿羊哥哥關注 2017.06.26 09:4...
    肇東周閱讀 12,180評論 4 61
  • 很早就入睡的習慣還是小時候在姥姥家養成的、后來回媽媽家、每晚也總會催促著早睡、媽媽總會叮囑著、晚上不要熬夜、不然第...
    陳琳琳閱讀 170評論 0 0
  • 有人說:每一個不想戀愛的人,心里都住著一個不可能的人。當時我就在默念著:“那不就是你嗎?小漁。” 坐公交的時候偶爾...
    步槍45閱讀 277評論 0 0
  • 看到網上好多教程的是在VS環境下OpenCV的安裝,而我一直都是在windows7,32位,sublime+cmd...
    Kedi閱讀 1,935評論 1 5
  • 用人生講一個故事,不求悲愴,不求歡喜,只愿故事講完那一天,眸子里的自己是淺笑的。 關于人世,向來遙遠,我愿一生是個...
    若彥閱讀 154評論 2 0