iOS之處理不等高TableViewCell的幾種方法

来源:http://www.cnblogs.com/rglmuselily/archive/2016/04/20/5411992.html
-Advertisement-
Play Games

課題一:如何計算Cell高度 方案一:直接法(面向對象) 直接法,就是把數據佈局到Cell上,然後拿到Cell最底部控制項的MaxY值。 第一步:創建Cell並正確設置約束,使文字區域高度能夠根據文字內容多少自動調整 添加好約束 第二步:再給這個Cell添加點別的東東,就叫這個東東BottomCub了 ...


課題一:如何計算Cell高度

方案一:直接法(面向對象)

直接法,就是把數據佈局到Cell上,然後拿到Cell最底部控制項的MaxY值。

第一步:創建Cell並正確設置約束,使文字區域高度能夠根據文字內容多少自動調整

QQ截圖20160329114325.png

添加好約束

第二步:再給這個Cell添加點別的東東,就叫這個東東BottomCub了。為Cub添加好約束。

QQ截圖20160329114358.png

隨便添加點什麼

第三步:為這個Cell寫一個返回Cell高度 - 也就是BottomCub最大Y值的方法

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 #import "TestCell.h"   @interface TestCell () @property (strong, nonatomic) IBOutlet UILabel *longLabel; @property (strong, nonatomic) IBOutlet UIView *bottomCub; @end   @implementation TestCell   //  Cell的構造方法 + (instancetype)creatWithTitle :(NSString *)title inTableView :(UITableView *)tableView {     TestCell *cell = [tableView dequeueReusableCellWithIdentifier:NSStringFromClass(self)];     if (!cell) {         cell = [[NSBundle mainBundle] loadNibNamed:NSStringFromClass(self) owner:nil options:kNilOptions].lastObject;     }     cell.longLabel.text = title;     return cell; }   /**  *  拿到bottomCub的最大Y值並返回  */ - (CGFloat)cellHeight {     //  強制佈局之前,需要先手動設置下cell的真實寬度,以便於準確計算     CGRect rect = self.frame;     rect.size.width = [[UIScreen mainScreen] bounds].size.width;     self.frame = rect;     [self layoutIfNeeded];    //  一定要強制佈局下,否則拿到的高度不准確     return CGRectGetMaxY(self.bottomCub.frame); }   @end

第四步:在代理方法中設置Cell高度

*註意:計算Cell高度的過程,一定不要放在heightForRow代理方法中!這一點在後文中將會有所提及。

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 #import "AskCellViewController.h" #import "TestCell.h"   @interface AskCellViewController () @property (strong, nonatomic) UITableView *tableView;   /** 測試數據 - Cell中文字內容數組*/ @property(copy,nonatomic) NSArray *testTitleArray;   @end   @implementation AskCellViewController   - (void)viewDidLoad {     [super viewDidLoad];     [self.view addSubview:self.tableView];       self.tableView.frame = self.view.bounds;     self.tableView.delegate = self;     self.tableView.dataSource = self;     self.tableView.tableFooterView = [[UIView alloc] init]; }     - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {     TestCell *cell = [TestCell creatWithTitle:self.testTitleArray[indexPath.row] inTableView:tableView];     cell.selectionStyle = UITableViewCellSelectionStyleNone;     return cell; } - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { //  *註意:計算Cell高度的過程,一定不要放在此代理方法中!這一點在後文中將會有所提及,此處僅為演示方便     CGFloat cellHeight = [[TestCell creatWithTitle:self.testTitleArray[indexPath.row] inTableView:tableView] cellHeight];     NSLog(@"%f",cellHeight);     return cellHeight; } - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {     return self.testTitleArray.count; } #pragma mark - Lazy - (UITableView *)tableView {     if (!_tableView) {         _tableView = [[UITableView alloc] init];     }     return _tableView; }     - (NSArray *)testTitleArray {     return @[@"我是第一個Cell",@"我是第二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二個Cell",@"我是第三個Cell"]; }   @end

效果

QQ截圖20160329115045.png

動態設定Cell高度結果

方案二:自己算(面向過程)

通常情況下,Cell之所以不等高,是因為Cell內部文字區域的高度會根據文字數量動態變化,圖片區域的高度會根據圖片數量而自動變化。也就是說,只要知道文字區域的高度、圖片區域的高度,就可以硬生生計算出Cell的高度了。

註意註意註意:如果產品有可能會要求調整行距,切不可用此方法計算!

替代可選方案:

1 CGSize size = [cell.content sizeThatFits:CGSizeMake(cell.content.frame.size.width, MAXFLOAT)];

(註於:2016.1.28)

  • 第一步:硬生生的將每個Cell的高度算出來,並保存在一個數組中

  • 第二步:heightForRow方法中返回相應的CellHeight

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 #import "CalculatorViewController.h" #import "TestCell.h"   @interface CalculatorViewController ()   @property (strong, nonatomic) UITableView *tableView; /** 測試數據 - Cell中文字內容數組*/ @property(copy,nonatomic) NSArray *testTitleArray; /** 用來存Cell高度的數組*/ @property(copy,nonatomic) NSArray *cellHeightArray;   @end   @implementation CalculatorViewController   - (void)viewDidLoad {     [super viewDidLoad];     [self.view addSubview:self.tableView];       self.tableView.frame = self.view.bounds;     self.tableView.delegate = self;     self.tableView.dataSource = self;     self.tableView.tableFooterView = [[UIView alloc] init]; }     - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {     TestCell *cell = [TestCell creatWithTitle:self.testTitleArray[indexPath.row] inTableView:tableView];     cell.selectionStyle = UITableViewCellSelectionStyleNone;     return cell; } - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {     CGFloat cellHeight = [self.cellHeightArray[indexPath.row] floatValue];     return cellHeight; } - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {     return self.testTitleArray.count; } #pragma mark - Lazy - (UITableView *)tableView {     if (!_tableView) {         _tableView = [[UITableView alloc] init];     }     return _tableView; } - (NSArray *)testTitleArray {     return @[@"我是第一個Cell",@"我是第二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二二個Cell",@"我是第三個Cell"]; } - (NSArray *)cellHeightArray {     NSMutableArray *cellHeightTMPArray = [@[] mutableCopy];     //  開始硬生生的計算每一個Cell高度     for (NSString *string in self.testTitleArray) {         CGFloat cellHeight = 0;         //  一個Cell由兩部分組成 - 高度自動調整的Label & bottomCub         //  bottomCub高度是確定的 - 120,Label和bottomCub之間的間距是確定的 - 8         static CGFloat bottomCubHeight = 120;         static CGFloat bottomMargin = 8;         //  計算Label的高度 - 其實就是計算Lable中的String的總高度             //  1. 拿到將要放入Lable的String         NSString *stringForLabel = string;         //  2. 根據文字內容、字體(固定值)、文字區域最大寬度計算String總高度         static CGFloat fontSize = 17;         CGFloat labelHeight = [stringForLabel sizeWithFont:[UIFont systemFontOfSize:fontSize] constrainedToSize:CGSizeMake(self.tableView.frame.size.width, CGFLOAT_MAX)].height;             //  3. 拿到了總高度,放入數組         cellHeight = labelHeight + bottomMargin + bottomCubHeight;         [cellHeightTMPArray addObject:@(cellHeight)];     }     return cellHeightTMPArray; } @end

效果

  • 就不給效果圖了哦,和上一張是一樣一樣的~

方案三:利用iOS 8新特性

其實,iOS8已經提供了直接通過XIB讓Cell高度自適應的方法了,只要簡單拖拖線,根本木有必要計算Cell高度,就可以搞定不等高Cell

第一步:設置tableView的估算Cell高度&rowHeight值為自動計算模式

1 2 3 4 5 6 - (void)viewDidLoad {     [super viewDidLoad];           self.tableView.estimatedRowHeight = 100;  //  隨便設個不那麼離譜的值     self.tableView.rowHeight = UITableViewAutomaticDimension; }

第二步:為Cell中最下麵的View設置約束 - 除了要定高、定寬、左上角粘著Label外,還要設置bottom距contentView的bottom間距為固定值,如0

QQ截圖20160329114819.png

bottomCub約束的添加方式

第三步:一定要註意 - 不能實現heightForRow代理方法!!!不能實現heightForRow代理方法!!!不能實現heightForRow代理方法!!!重要的事情說三遍...

 

QQ截圖20160329114956.png

iOS8新特性實現Cell高度的自適應

效果:一樣杠杠滴~

課題二:在哪計算Cell高度

方案一:在heightForRow代理方法中計算

  • 示例代碼:見課題一方案一

  • 說明:在這裡進行計算是非常糟糕的選擇,因為系統調用heightForRow方法非常頻繁 感興趣的小伙伴可以列印測試下...在這裡進行計算,意味著系統每調用一次heightForRow方法,就會執行一次高度計算...好可怕有木有?

方案二:在請求到數據後馬上計算

  • 示例代碼:見課題一方案二

  • 說明:在這裡進行計算相對於方案一來說進步了很多,在這裡計算是不錯的選擇哦!

方案三:在cellForRow代理方法中算

  • 說明:其實,要隆重介紹的是方案三~

  • 思路:

  1.既然想知道Cell的高度,那麼一定是Cell自己最懂自己有多高啦(面向對象的思維)。

  2.那麼,在哪裡能拿到Cell和Cell的高度呢? - 當然是CellForRow代理方法中啦!

  3.但是,在CellForRow中拿到Cell高度後,如何傳遞給heightForRow代理方法呢? - 可以將Cell高度保存在一個數組中,或者保存在Cell對應的Model中~

  4.但是,我們知道系統對tableView代理方法的調用順序,是先調取heightForRow再調取cellForRow的呀,這意味著,我們在cellForRow方法中拿到cell高度之前,就需要設置heightForRow...怎麼辦?

    a.解決方案:實現estimatedHeightForRow代理方法!

    b.實現這個代理方法後,系統會先調取cellForRow,再調取heightForRow,而且實現這個代理方法之後,腰不酸了,腿不疼了,一口氣上五樓也不費勁了~

  • 示例代碼:可以參考下我之前的文章哦!傳送門 - iOS項目實例:QQ聊天界面UI搭建

  • 註意:如果實現了estimatedHeightForRow代理方法,可能會造成tableView的ContentSize值不正確哦!所以,該方法請選擇使用...

結論

  • 處理不等高TableViewCell,優先使用iOS8新特性(課題一方案三)

  • 不能使用iOS8新特性的情況下,優先選擇課題一方案一+課題二方案三組合

  • 不能用上面兩種,優先選擇使用課題一方案一+課題二方案二組合~

補充

tableView的contentSize電腦制

實驗方案

自定義一個tableView的子類,重寫setContentSize方法,在該方法中列印contentSizeHeight,觀察總結。

分兩組:第一組實現estimatedHeightForRow方法,第二組不實現。

  • 第一組:section = 1 rowNumber = 5 | 估算行高10 | 實際行高100

  • 第二組:section = 1 rowNumber = 5 | 未實現估算行高 | 實際行高100

實驗結果

第一組

36.png

實驗一 - 1組5行|估算行高10|實際行高100|計算contentSize.gif

第二組

QQ截圖20160329115440.png

實驗二 - 1組5行|不估算|實際行高100|計算contentSize.gif

實驗結論

  • viewDidAppear方法中拿到的contentSize才是準確的contentSize

  • contentSize至少會設置3次,如果估算行高與實際行高不符,會再次設置contentSize

未解之謎

  • 通過列印我們可以看到,獲取cell之前,詭異地對heighForRow方法遍歷了三次...為什麼是三次?

  • 為什麼最少設置3次contentSize,不能只設置1次嗎?

  • 能否把返回Cell的方法和heightForRow方法合併成一個代理方法?


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • npm install <name>安裝nodejs的依賴包 例如npm install express 就會預設安裝express的最新版本,也可以通過在後面加版本號的方式安裝指定版本,如npm install [email protected] npm install <name> -g 將包安裝到全局 ...
  • 這個圖片是我用xmind編輯的。歡迎大家來糾正或添加! 再來說說我對網站開發的理解與碰到的問題。 網站的前端有幾個部分我覺得很重要: 1.優雅的文字提示 2.合理的頁面設計 3.適當的腳本特效 一進入這個網站,就讓人有一種非常舒適的感覺。並且在舒適的基礎上有奪人眼球的特效,再有貼心的提示,我相信肯定 ...
  • do_Http組件就是實現http/https協議的組件. 首先要說明一下,Do平臺只是一個移動端開發的平臺,不涉及任何後臺服務端的技術,你可以使用你自己任何熟悉的語言和技術來實現服務端,do_Http組件就是用來和服務端通信。 http是一個最基礎的應用層的通信協議,開發者應該很熟悉,但是在QQ群 ...
  • 2015 was the year of React with tons of new releases and developer conferences dedicated to the topic all over the world. For a detailed list of the m ...
  • 作者:裡脊串 授權本站轉載。 啟動圖(LaunchImage)的管理其實在iOS開始中算比較簡單的了,尤其是Xcode引入了xcassets之後,完全是傻瓜式的操作。但是有的時候我們還是需要在Launch Image上做文章。 LaunchImage在APP初始化完之後會立即消失並顯示APP的界面 ...
  • - (UIImage *)stretchableImageWithLeftCapWidth:(NSInteger)leftCapWidth topCapHeight:(NSInteger)topCapHeight 這個UIImage類 實例方法的作用是 圖片實例 被拉伸時 1. 橫向拉伸時 從左邊數 ...
  • 時間就是金錢。編碼效率的提升意味著更多的收入。可是當我們的開發技巧已經到達一定高度時,如何讓開發效率更上一層樓呢?答案就是使用開發工具!在這篇文章中,我會向你介紹一些幫助我提升編碼速度和工作效率的工具。 Xcode插件 幾乎所有開發者都知道Alcatraz是一個開源的包管理工具,可以讓我們更輕鬆地管 ...
  • 當點擊workingCopy時錯誤如下 google了一下,有是有解決的辦法,可是這些都是直接使用sqlite時產生的問題。 sqlite錯誤 The database disk image is malformed database disk image is malformed 可解決 修複SQ ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...