iOS學習之 UITableVIew cell 創建方式(XIB, storyboard)

UITableView

  • 數據源的常用方法

  • 此方法設置 cell 的組數
    - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;

  • 每一組有多少行數據
    - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section;

  • 每一行顯示什么內容,返回 cell
    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;

  • 設置每一組的頭部信息
    - (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section;

  • 設置每一組的尾部信息
    - (NSString *)tableView:(UITableView *)tableView titleForFooterInSection:(NSInteger)section

  • tabelView常見屬性

    // 設置每一行cell的高度
    self.tableView.rowHeight = 100;
    
    // 設置每一組頭部的高度
    self.tableView.sectionHeaderHeight = 50;
    // 設置每一組尾部的高度
    self.tableView.sectionFooterHeight = 50;
    // 設置分割線顏色
    self.tableView.separatorColor = [UIColor redColor];
    // 設置分割線樣式
    self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone;
    // 設置表頭控件
    self.tableView.tableHeaderView = [[UISwitch alloc] init];
    // 設置表尾控件
    self.tableView.tableFooterView = [UIButton buttonWithType:UIButtonTypeContactAdd];
    // 設置右邊索引文字的顏色
    self.tableView.sectionIndexColor = [UIColor redColor];
    // 設置右邊索引文字的背景色
    self.tableView.sectionIndexBackgroundColor = [UIColor blackColor];
    
    
  • cell

  • UITableView的每一行都是一個UITableViewCell,通過dataSource的tableView:cellForRowAtIndexPath:方法來初始化每一行

  • UITableViewCell內部有個默認的子視圖:contentView,contentView是UITableViewCell所顯示內容的父視圖,可顯示一些輔助指示視圖

  • 輔助指示視圖的作用是顯示一個表示動作的圖標,可以通過設置UITableViewCell的accessoryType來顯示,默認是UITableViewCellAccessoryNone(不顯示輔助指示視圖)

  • 可以通過cell的accessoryView屬性來自定義輔助指示視圖(比如往右邊放一個開關)

  • 傳統的寫法

    /**
    *  每當有一個cell要進入視野范圍內,就會調用一次
    */
    - (UITableViewCell *)tableView:(UITableView *)tableView cell ForRowAtIndexPath:(NSIndexPath *)indexPath
    {
    static NSString *ID = @"wine";
    
    // 1.先去緩存池中查找可循環利用的cell
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
    
    // 2.如果緩存池中沒有可循環利用的cell
    if (!cell) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:ID];
    }
    
    // 3.設置數據
    cell.textLabel.text = [NSString stringWithFormat:@"%zd行的數據", indexPath.row];
    
    return cell;
    

}
```

  • 新的寫法(注冊cell)

    NSString *ID = @"wine";
    
    - (void)viewDidLoad {
    [super viewDidLoad];
    
    // 注冊某個重用標識 對應的 Cell類型
    [self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:ID];
    }
    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    

{
// 1.先去緩存池中查找可循環利用的cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];

// 2.設置數據
cell.textLabel.text = [NSString stringWithFormat:@"%zd行的數據", indexPath.row];

return cell;

}
```

  • 自定義 cell

  • 等高 cell 代碼

    1. 新建一個繼承自UITableViewCell的子類
    2. 重寫新建子類的-initWithStyle:reuseIdentifier:方法在這個方法中添加所有需要顯示的子控件。給子控件做一些初始化設置(設置字體、文字顏色等)
    3. 重寫-layoutSubviews方法一定要調用[superlayoutSubviews]在這個方法中計算和設置所有子控件的frame
    4. 提供模型屬性,設置模型的數據,在 set 方法中
    5. 注冊 cell。。。下面方法是從 xib 中 zuce
      [self.tableView registerNib:[UINib             
                   nibWithNibName:NSStringFromClass([MTTgCell class]) 
                           bundle:nil] 
                           forCellReuseIdentifier:ID];
      
  • 不等高 cell

    1. 在模型內部提供可設置 cell高度 的的 frame屬性,
        @interface XMGStatus : NSObject
        @property (nonatomic, assign) CGRect iconFrame;
        /** cell的高度 */
        @property (nonatomic, assign) CGFloat cellHeight;
        @end
    
    1. model.m中重寫模型cellHeight屬性的get方法

      - (CGFloat)cellHeight
      {
          if (_cellHeight == 0) {
              // ... 計算所有子控件的frame、cell的高度
          }
          return _cellHeight;
      }
      
    2. 在控制器中實現返回 cell 高度的代理方法,heightforrow

          // 返回每一行cell的具體高度
          - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
          {
          SBStatus *status = self.statuses[indexPath.row];
          return status.cellHeight;
          }
      
    3. 給 cell 模型傳值

    4. 新建一個繼承自UITableViewCell的子類,比如SBStatusCell,并且提供一個模型屬性

    5. 在SBStatusCell.m中重寫模型屬性的set方法

  • 文字尺寸的計算

    • 計算文字所占據的尺寸
        NSDictionary *nameAttrs = @{NSFontAttributeName : [UIFont systemFontOfSize:17]};
        CGSize nameSize = [self.name sizeWithAttributes:nameAttrs];
    
    • 計算一段文字的尺寸
        CGSize textMaxSize = CGSizeMake(textW, MAXFLOAT);
        NSDictionary *textAttrs = @{NSFontAttributeName : [UIFont systemFontOfSize:14]};
        CGFloat textH = [self.text boundingRectWithSize:textMaxSize
                                                options:NSStringDrawingUsesLineFragmentOrigin 
                                             attributes:textAttrs 
                                                context:nil].size.height;
    
    
    • 計算屏幕的寬
      • [UIScreen mainScreen].bounds.size.width
  • 連線修改約束
    • 定義一個高度約束且連線到 storyboard
      • @property (nonatomic, weak) IBOutlet NSLayoutConstraint *picHeight;
    • 然后在模型設置控件的 set 方法中修改
      • self.picHeight.constant = 100;
  • cellheight 不等高的估算和自動計算
    • 程序一開始就會吧 tabelViewl 的所有cell高度算出來,確定 contengSize 的大小,從而計算出滾動條的長度。所以這樣的性能不好,所以蘋果推出了估算高度
    • 告訴tabelview的 cell的估算高度,就可以使heightforrow方法的調用次數減少
    • 設置估算的方法
      1. self.tabelView.estimateRowHeight = 10
      2. - (CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath 通過代理方法
    • 處理不等高 cell 自動計算(iOS8開始才支持)
      1. 告訴tableView所有cell的真實高度是自動計算(根據設置的約束來計算)
        self.tableView.rowHeight = UITableViewAutomaticDimension;
      2. 告訴tableView所有cell的估算高度. self.tableView.estimatedRowHeight = 44;
      3. 這種需要設置約束連線,然后在 cell 的 setModel 方法中進行判斷
          // 設置配圖數據
          if (status.picture) { // 有配圖
              self.pictureHeight.constant = 100;
              self.pictureBottom.constant = 10;
              self.pictureImageView.image = [UIImage imageNamed:status.picture];
          } else { // 沒有配圖
              // 設置圖片高度為0
              self.pictureHeight.constant = 0;
              // 設置圖片底部間距為0
              self.pictureBottom.constant = 0;
      
    }
    ```
    • iOS 8之前

      1. 如果cell內部有自動換行的label,需要設置preferredMaxLayoutWidth屬性
          - (void)awakeFromNib
      {
          // 手動設置文字的最大寬度(目的是:讓label知道自己文字的最大寬度,進而能夠計算出自己的frame)
          self.text_label.preferredMaxLayoutWidth = [UIScreen mainScreen].bounds.size.width - 20;
      }
      
      
      1. 設置tableView的cell估算高度
      // 告訴tableView所有cell的估算高度(設置了估算高度,就可以減少tableView:heightForRowAtIndexPath:方法的調用次數)
      self.tableView.estimatedRowHeight = 200;
      
      1. 在代理方法中計算cell的高度
          XMGStatusCell *cell;
          - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
          {
              // 創建一個cell(cell的作用:根據模型數據布局所有的子控件,進而計算出cell的高度)
              if (!cell) {
                  cell = [tableView dequeueReusableCellWithIdentifier:ID];
              }
          
              // 設置模型數據
              cell.status = self.statuses[indexPath.row];
          
              return cell.height;
          }
          
          ```
      4. 在 cell.m中設置 height 高度
      
        - (CGFloat)height
        {
            // 強制布局cell內部的所有子控件(label根據文字多少計算出自己最真實的尺寸)
            [self layoutIfNeeded];
        
            // 計算cell的高度
            if (self.status.picture) {
                return CGRectGetMaxY(self.pictureImageView.frame) + 10;
            } else {
                return CGRectGetMaxY(self.text_label.frame) + 10;
            }
        }
      
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,460評論 6 538
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,067評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,467評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,468評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,184評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,582評論 1 325
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,616評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,794評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,343評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,096評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,291評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,863評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,513評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,941評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,190評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,026評論 3 396
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,253評論 2 375

推薦閱讀更多精彩內容