UITableView佈局
UITableView內置了兩種樣式:UITableViewStylePlain,UITableViewStyleGrouped優化
<UITableViewDataSource,UITableViewDelegate>裏的方法:orm
tableView處理步驟對象
#pragma mark 1.有多少組繼承
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView索引
#pragma mark 2.第section組頭部控件有多高事件
- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section圖片
#pragma mark 3.第section組有多少行ip
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section字符串
#pragma mark 4.indexPath這行的cell有多高
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
#pragma mark 5.indexPath這行的cell長什麼樣子
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
#pragma mark 6.第section組頭部顯示什麼控件
- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
//每當有一個cell進入視野屏幕就會調用,因此在這個方法內部就須要優化。
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
if(cell==nil){
//在這裏面作建立的工做。循環優化。防止刷新cell進入屏幕的時候重複的建立
}
}
//當調用reloadData的時候,會從新刷新調用數據源內全部方法,其餘事情都不會作呀
[self reloadData]
//這個方法只有在一開始有多少條數據纔會算多少個高度,這個方法只會調用一次,可是每次reloadData的時候也會調用
//並且會一次性算出全部cell的高度,好比有100條數據,一次性調用100次
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
-(NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView //右側索引
-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath //行點擊事件
NSIndexPath *path = [self.tableView indexPathForSelectedRow]; //得到被選中的indexPath能夠獲得section,row
[self.tableView reloadRowsAtIndexPaths:[self.tableView indexPathsForSelectedRows] withRowAnimation:UITableViewRowAnimationNone]; //刷新table指定行的數據
[self.tableView reloadData]; //刷新table全部行的數據
UITableView經常使用屬性:
UITableView *tableView = [[UITableView alloc]initWithFrame:CGRectMake(0, 0, 320, 460) style:UITableViewStylePlain]; // 初始化表格
分隔線屬性
tableView.separatorStyle = UITableViewCellSeparatorStyleSingleLine; //UITableViewCellSeparatorStyleNone;
[self.tableView setSeparatorStyle:UITableViewCellSeparatorStyleNone]; //取消分隔線
tableView.separatorColor = [UIColor lightGrayColor];
條目多選
tableView.allowsMultipleSelection = YES;
// 設置標題行高
[_tableView setSectionHeaderHeight:kHeaderHeight];
[_tableView setSectionFooterHeight:0];
// 設置表格行高
[_tableView setRowHeight:50];
//設置背景色
self.tableView.backgroundView 優先級高,若是要設置backgroundColor的時候要先把view設置爲nil
self.tableView.backgroundColor
//在tableView的頭部或者尾部添加view,footerView寬度是不用設置的
xxxView.bounds = CGRectMake(0,0,0,height);
self.tableView.tableFooterView =xxxView;
self.tableView.tableHeaderView =xxxView;
UIButton *bt = (UIButton*)[self.contentView viewWithTag:i+100];
增長tableview滾動區域
self.tableView.contentInset = UIEdgeInsetsMake(0, 0, xx, 0);
UITableViewCell
//建立UITableViewCell
UITableViewCell *cell = [[UITableViewCell alloc]
initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:nil];
[cell.textLabel setBackgroundColor:[UIColor clearColor]];// 清空標籤背景顏色
cell.backgroundView =xx; //設置背景圖片
cell.backgroundVColor =xx;
cell.selectedBackgroundView = selectedBgView; //設置選中時的背景顏色
cell.accessoryView = xxxView; //設置右邊視圖
[cell setAccessoryType:UITableViewCellAccessoryNone]; //設置右側箭頭
[self setSelectionStyle:UITableViewCellSelectionStyleNone]; //選中樣式
cell.selectionStyle = UITableViewCellSelectionStyleBlue;
//設置cell的高度
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
contentView下默認有3個子視圖,其中的2個是UILabel,經過textLabel和detailTextLabel屬性訪問,第3個是UIImageView,經過imageView屬性訪問.
UITableViewCellStyleDefault, UITableViewCellStyleValue1, UITableViewCellStyleValue2, UITableViewCellStyleSubtitle
#pragma mark - 從新調整UITalbleViewCell中的控件佈局
- (void)layoutSubviews{
[super layoutSubviews];
…
}
cell 裏面還有一個contentView
UITableViewCell表格優化
UITableViewCell對象的重用原理:
重用原理:當滾動列表時,部分UITableViewCell會移出窗口,UITableView會將窗口外的UITableViewCell放入一個對象池中,等待重用。當UITableView要求dataSource返回UITableViewCell時,dataSource會先查看這個對象池,若是池中有未使用的UITableViewCell,dataSource會用新的數據配置這個UITableViewCell,而後返回給UITableView,從新顯示到窗口中,從而避免建立新對象
還有一個很是重要的問題:有時候須要自定義UITableViewCell(用一個子類繼承UITableViewCell),並且每一行用的不必定是同一種UITableViewCell(如短信聊天佈局),因此一個UITableView可能擁有不一樣類型的UITableViewCell,對象池中也會有不少不一樣類型的UITableViewCell,時可能會獲得錯誤類型的UITableViewCell那麼UITableView在重用UITableViewCell。解決方案:UITableViewCell有個NSString *reuseIdentifier屬性,能夠在初始化UITableViewCell的時候傳入一個特定的字符串標識來設置reuseIdentifier(通常用UITableViewCell的類名)。當UITableView要求dataSource返回UITableViewCell時,先經過一個字符串標識到對象池中查找對應類型的UITableViewCell對象,若是有,就重用,若是沒有,就傳入這個字符串標識來初始化一個UITableViewCell對象
/**
單元格優化
1. 標示符統一,使用static的目的能夠保證表格標示符永遠只有一個
2. 首先在緩衝池中找名爲"myCell"的單元格對象
3. 若是沒有找到,實例化一個新的cell
**/
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
static NSString *cellIdentifier = @"myCell";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];
//使用這種方法不用判斷下面的cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier forIndexPath:indexPath];
if (cell == nil) {
cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];
}
return cell;
}
表格的編輯模式
刪除、插入
- (void)setEditing:(BOOL)editing animated:(BOOL)animated; 開啓表格編輯狀態
- (UITableViewCellEditingStyle)tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath{
返回表格編輯編輯樣式。不實現默認都是刪除
return editingStyle : UITableViewCellEditingStyleDelete, UITableViewCellEditingStyleInsert
}
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath{
//根據editingStyle處理是刪除仍是添加操做
完成刪除、插入操做刷新表格
- (void)insertRowsAtIndexPaths:(NSArray *)indexPaths withRowAnimation:(UITableViewRowAnimation)animation;
-(void)deleteRowsAtIndexPaths:(NSArray *)indexPaths withRowAnimation:(UITableViewRowAnimation)animation;
}
移動
- (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath
sourceIndexPath 移動的行
destinationIndexPath 目標的行
自定義表格行UITableViewCell
storyboard方式建立:
直接拖到UITableView裏面設置UITableViewCell
注意:
1.經過XIB或者Storyboard自定義單元格時,在xib和Storyboard裏面須要指定單元格的可重用標示符Identifier
2.注意表格的優化中的差異
在Storyboard中二者等效
xxCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
xxCell *cell1 = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];
在xib文件中有差異:
第一種狀況,只能在iOS 6以上使用,若是在viewDidLoad註冊了nib文件,而且指定了「單元格」的可重用標示符,那麼
dequeueReusableCellWithIdentifier:
dequeueReusableCellWithIdentifier:forIndexPath:
方法是等效的。若是在viewDidLoad中註冊了nib文件,表格緩衝池中的管理,有系統接管!
第二種狀況,是在iOS 4以上都可以使用,若是沒有在viewDidLoad註冊nib文件,那麼,只能使用
dequeueReusableCellWithIdentifier:而且須要判斷cell沒有被實例化,並作相應的處理
在代碼建立中差異:
用代碼建立cell中的處理和nib同樣,註冊了cell就有系統接管而且能夠用帶forIndexPath的方法,沒有註冊就要本身去實例化cell,不能用帶forIndexPath的方法
[tableView registerClass:XxxCell class] forCellReuseIdentifier:@"xxCell"];
xib方式建立:
//註冊Identifier
- (void)viewDidLoad{
[super viewDidLoad];
/**
注意:如下幾句註冊XIB的代碼,必定要在viewDidLoad中!
註冊XIB文件,得到根視圖,而且轉換成TableView,爲tableView註冊xib
Identifier名要在xib文件中定義,而且保持一致
**/
UINib *nib = [UINib nibWithNibName:@"BookCell" bundle:[NSBundle mainBundle]];
UITableView *tableView = (UITableView *)self.view;
[tableView registerNib:nib forCellReuseIdentifier:@"bookCell"];
}
//沒有註冊Identifier只能使用下面方法
static NSString *CellIdentifier = @"bookCell";
BookCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) {
cell = [[BookCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
NSBundle *bundle = [NSBundle mainBundle];
NSArray *array = [bundle loadNibNamed:@"BookCell" owner:nil options:nil];
cell = [array lastObject];
}
代碼方式建立:
創建UITableViewCell的類,繼承UITableViewCell
往cell裏面加入view的時候注意點:
//新建的組件放入contentView中
[self.contentView addSubview:xxView];
//設置圖片拉伸屬性stretch
UIImage *normalImage = [UIImage imageNamed:@"xx.png"];
normalImage = [normalImage stretchableImageWithLeftCapWidth:
normalImage.size.width / 2 topCapHeight:normalImage.size.height / 2];
//在tableView裏面viewDiDLoad裏面要註冊cell類
[tableView registerClass:XxxCell class] forCellReuseIdentifier:@"xxCell"];
自定義表格中Header
//自定義表格在這個方法中定義
-(UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section
-(NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section