自定义 UITableViewCell 元素不只出现在 iPad
Custom UITableViewCell elements not appearing ONLY on iPad
我 运行 在 运行 我的应用程序 iPad 上遇到以下问题。我的自定义第一行 UITableViewCell 的 UI 元素没有出现。我正在手动分配和定位这些元素,因此问题与使用错误的 xib 无关。我已经在 iPhone 6 vs an iPad 2 上上传了 运行 完全相同代码的结果。请注意 iPad 上的第一行单元格如何为空,但正确地具有两个 UITextField
元素和 iPhone 上的 UIButton
6
经过一些实验,我现在知道是什么导致了这个问题,但我不知道为什么。这是我对 tableView:cellForRowAtIndexPath:
的实现
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
static NSString *summonerCellIdentifier = @"summonerCellIdentifier";
static NSString *footerCellIdentifier = @"footerCellIdentifier";
static NSString *headerCellIdentifier = @"headerCellIdentifier";
UITableViewCell *cell;
if (indexPath.row == 0) {
// header view 1
cell = [tableView dequeueReusableCellWithIdentifier:headerCellIdentifier];
if (cell == nil) {
cell = [UITableViewCell new];
}
cell.backgroundColor = [UIColor LHBeigeLight];
cell.selectionStyle = UITableViewCellSelectionStyleNone;
self.summonerName = [[UITextField alloc] initWithFrame:CGRectMake(0,30,200,40)];
[self.summonerName setCenter:CGPointMake(self.view.center.x, self.summonerName.center.y)];
[self.summonerName setBackgroundColor:[UIColor whiteColor]];
[self.summonerName setLeftViewMode:UITextFieldViewModeAlways];
[self.summonerName setLeftView:[[UIView alloc] initWithFrame:CGRectMake(0, 0, 10, 10)]];
[self.summonerName setPlaceholder:@"Summoner Name"];
[self.summonerName setKeyboardType:UIKeyboardTypeEmailAddress];
[self.summonerName setAutocapitalizationType:UITextAutocapitalizationTypeNone];
[self.summonerName setAutocorrectionType:UITextAutocorrectionTypeNo];
// [self.summonerName setAutoresizingMask:(UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin)];
[cell.contentView addSubview:self.summonerName];
self.summonerRegion = [[UITextField alloc] initWithFrame:CGRectMake(0,80,200,40)];
[self.summonerRegion setCenter:CGPointMake(self.view.center.x, self.summonerRegion.center.y)];
[self.summonerRegion setBackgroundColor:[UIColor whiteColor]];
[self.summonerRegion setLeftViewMode:UITextFieldViewModeAlways];
[self.summonerRegion setLeftView:[[UIView alloc] initWithFrame:CGRectMake(0, 0, 10, 10)]];
[self.summonerRegion setPlaceholder:@"Summoner Region"];
[self.summonerRegion setKeyboardType:UIKeyboardTypeEmailAddress];
[self.summonerRegion setAutocapitalizationType:UITextAutocapitalizationTypeNone];
[self.summonerRegion setAutocorrectionType:UITextAutocorrectionTypeNo];
// [self.summonerRegion setAutoresizingMask:(UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin)];
[cell.contentView addSubview:self.summonerRegion];
self.searchButton = [[UIButton alloc]initWithFrame:CGRectMake(0,130,200,40)];
[self.searchButton setCenter:CGPointMake(self.view.center.x, self.searchButton.center.y)];
[self.searchButton setTitle:@"Search" forState:UIControlStateNormal];
[self.searchButton.titleLabel setTextColor:[UIColor whiteColor]];
[self.searchButton setBackgroundImage: [AppDelegate imageFromColor:[UIColor LHGoldColor]] forState:UIControlStateNormal];
[self.searchButton setBackgroundImage: [AppDelegate imageFromColor:[UIColor LHGoldLightColor]] forState:UIControlStateHighlighted];
[self.searchButton setBackgroundImage: [AppDelegate imageFromColor:[UIColor LHGoldLightColor]] forState:UIControlStateSelected];
[self.searchButton setContentEdgeInsets:UIEdgeInsetsMake(5, 15, 5, 15)];
[self.searchButton addTarget:self action:@selector(search:) forControlEvents:UIControlEventTouchUpInside];
// [self.searchButton setAutoresizingMask:(UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin)];
[cell.contentView addSubview:self.searchButton];
_pickerView = [[UIPickerView alloc] init];
[self.summonerName setDelegate:self];
[self.summonerRegion setInputView:_pickerView];
[self.summonerRegion setDelegate:self];
// Initialize Data
_pickerData = @[@"NA", @"EUW", @"EUNE", @"BR", @"KR", @"LAN", @"LAS", @"OCE", @"RU", @"TR"];
// Connect data
_pickerView.dataSource = self;
_pickerView.delegate = self;
_pickerView.showsSelectionIndicator = YES;
[self.summonerRegion setText:[_pickerData objectAtIndex:0]];
} else if (indexPath.row == (recentSearchesArray.count+1)) {
// footer view 1
cell = [tableView dequeueReusableCellWithIdentifier:footerCellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:footerCellIdentifier];
}
cell.frame = CGRectMake(0, 0, self.recentSearchesTable.frame.size.width, 140);
cell.backgroundColor = [UIColor LHRedColor];
cell.textLabel.textColor = [UIColor LHBeigeLight];
cell.textLabel.text = @"Follow Us On Twitter";
cell.detailTextLabel.textColor = [UIColor LHBeigeLight];
cell.detailTextLabel.text = @"for app updates and events!";
cell.selectionStyle = UITableViewCellSelectionStyleNone;
} else if (indexPath.row == (recentSearchesArray.count+2)) {
// footer view 2
cell = [tableView dequeueReusableCellWithIdentifier:footerCellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:footerCellIdentifier];
}
cell.frame = CGRectMake(0, 0, self.recentSearchesTable.frame.size.width, 140);
cell.textLabel.textColor = [UIColor LHBeigeLight];
cell.backgroundColor = [UIColor LHGreenColor];
cell.textLabel.text = @"Battlefy Tournaments";
cell.detailTextLabel.textColor = [UIColor LHBeigeLight];
cell.detailTextLabel.text = @"Check out our community tournaments!";
cell.selectionStyle = UITableViewCellSelectionStyleNone;
} else {
// regular cell
Summoner *summoner = [recentSearchesArray objectAtIndex:indexPath.row-1];
cell = [tableView dequeueReusableCellWithIdentifier:summonerCellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:summonerCellIdentifier];
}
cell.textLabel.text = summoner.summonerName;
cell.detailTextLabel.text = [summoner.summonerRegion uppercaseString];
cell.backgroundColor = [UIColor whiteColor];
cell.selectionStyle = UITableViewCellSelectionStyleDefault;
}
return cell;
}
我在第一个单元格中注释掉了破坏我的自定义视图的三行,并取消了格式设置。无论出于何种原因,如果我设置左右灵活的自动调整大小掩码,单元格元素就会在 iPad 版本上消失。我正在使用这些 UIViewAutoresizing
掩码来处理旋转(通过使元素在第一个单元格中水平居中)。
有人知道为什么会这样吗?如果我设置这个 0 索引的方式 UITableViewCell
是不好的做法,或者如果我的代码有其他问题,请告诉我!
编辑: 更多信息。这是 tableView:heightForRowAtIndexPath:
的样子:
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
if (indexPath.row == 0) {
return 200;
} else if (indexPath.row >= recentSearchesArray.count + 1) {
return 130;
} else {
return 44;
}
}
这些是 po [self.view recursiveDescription]
对于 iPhone 6:
的输出
(lldb) po [self.view recursiveDescription]
<UIView: 0x7ff3e156d290; frame = (0 0; 320 568); autoresize = W+H; layer = <CALayer: 0x7ff3e156d360>>
| <UITableView: 0x7ff3e1858e00; frame = (0 0; 320 568); clipsToBounds = YES; autoresize = RM+BM; gestureRecognizers = <NSArray: 0x7ff3e156ad60>; layer = <CALayer: 0x7ff3e1568610>; contentOffset: {0, 0}; contentSize: {320, 0}>
| | <UITableViewWrapperView: 0x7ff3e156bbf0; frame = (0 0; 320 568); gestureRecognizers = <NSArray: 0x7ff3e156c660>; layer = <CALayer: 0x7ff3e156c160>; contentOffset: {0, 0}; contentSize: {320, 568}>
| | <UIImageView: 0x7ff3e15698b0; frame = (0 565.5; 320 2.5); alpha = 0; opaque = NO; autoresize = TM; userInteractionEnabled = NO; layer = <CALayer: 0x7ff3e15294a0>>
| | <UIImageView: 0x7ff3e156a5d0; frame = (317.5 561; 2.5 7); alpha = 0; opaque = NO; autoresize = LM; userInteractionEnabled = NO; layer = <CALayer: 0x7ff3e156a5b0>>
和 iPad 2:
(lldb) po [self.view recursiveDescription]
<UIView: 0x7869a230; frame = (0 0; 320 568); autoresize = W+H; layer = <CALayer: 0x786a2b20>>
| <UITableView: 0x78907600; frame = (0 0; 320 568); clipsToBounds = YES; autoresize = RM+BM; gestureRecognizers = <NSArray: 0x78690b00>; layer = <CALayer: 0x786a62c0>; contentOffset: {0, 0}; contentSize: {320, 0}>
| | <UITableViewWrapperView: 0x786a9470; frame = (0 0; 320 568); gestureRecognizers = <NSArray: 0x786a8290>; layer = <CALayer: 0x7866ab40>; contentOffset: {0, 0}; contentSize: {320, 568}>
| | <UIImageView: 0x786a9160; frame = (0 565; 320 3); alpha = 0; opaque = NO; autoresize = TM; userInteractionEnabled = NO; layer = <CALayer: 0x7866a750>>
| | <UIImageView: 0x78679fe0; frame = (317 561; 3 7); alpha = 0; opaque = NO; autoresize = LM; userInteractionEnabled = NO; layer = <CALayer: 0x78671490>>
编辑 2: 为最佳实践添加了出队代码
编辑 3: 根据 Brian 的建议切换到更好的单元格初始化函数 [UITableViewCell new];
并将子视图添加到内容视图而不是单元格本身。仍然没有运气。
编辑 4 已修复: 最后我不得不切换到使用自定义 UITableViewCell
子类并直接在 .xib 中执行所有约束以解决此问题.我不确定根本原因是什么,但如果您仍然想重现它,您可能只需将测试应用程序中的每个单元格都设为我为 0 索引创建的单元格即可。
Edit 5 FIXED X 2: 我用上面的代码找到了原始问题的解决方案。你可以在这里查看
我回去再看问题,终于找到了真凶。我试图将子视图在单元格内水平居中,所以调用了这样的东西:
[self.summonerName setCenter:CGPointMake(self.view.center.x, self.summonerName.center.y)];
出于某种原因 self.view.center.x
在 iPad 上评估了一些完全出乎意料的事情,并将内容视图定位在屏幕外。我真正应该做的(以及解决上述代码问题的方法)是基于 cell 的 中心而不是 superview 的 [=22] 设置子视图中心=] 像这样居中:
[self.summonerName setCenter:CGPointMake(cell.center.x, self.summonerName.center.y)];
为了向所有好奇的人进行演示,我创建了一个绝对准系统示例项目来说明这个问题,您可以在此处查看:
https://github.com/oseparovic/ipaduitableviewbug
我 运行 在 运行 我的应用程序 iPad 上遇到以下问题。我的自定义第一行 UITableViewCell 的 UI 元素没有出现。我正在手动分配和定位这些元素,因此问题与使用错误的 xib 无关。我已经在 iPhone 6 vs an iPad 2 上上传了 运行 完全相同代码的结果。请注意 iPad 上的第一行单元格如何为空,但正确地具有两个 UITextField
元素和 iPhone 上的 UIButton
6
经过一些实验,我现在知道是什么导致了这个问题,但我不知道为什么。这是我对 tableView:cellForRowAtIndexPath:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
static NSString *summonerCellIdentifier = @"summonerCellIdentifier";
static NSString *footerCellIdentifier = @"footerCellIdentifier";
static NSString *headerCellIdentifier = @"headerCellIdentifier";
UITableViewCell *cell;
if (indexPath.row == 0) {
// header view 1
cell = [tableView dequeueReusableCellWithIdentifier:headerCellIdentifier];
if (cell == nil) {
cell = [UITableViewCell new];
}
cell.backgroundColor = [UIColor LHBeigeLight];
cell.selectionStyle = UITableViewCellSelectionStyleNone;
self.summonerName = [[UITextField alloc] initWithFrame:CGRectMake(0,30,200,40)];
[self.summonerName setCenter:CGPointMake(self.view.center.x, self.summonerName.center.y)];
[self.summonerName setBackgroundColor:[UIColor whiteColor]];
[self.summonerName setLeftViewMode:UITextFieldViewModeAlways];
[self.summonerName setLeftView:[[UIView alloc] initWithFrame:CGRectMake(0, 0, 10, 10)]];
[self.summonerName setPlaceholder:@"Summoner Name"];
[self.summonerName setKeyboardType:UIKeyboardTypeEmailAddress];
[self.summonerName setAutocapitalizationType:UITextAutocapitalizationTypeNone];
[self.summonerName setAutocorrectionType:UITextAutocorrectionTypeNo];
// [self.summonerName setAutoresizingMask:(UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin)];
[cell.contentView addSubview:self.summonerName];
self.summonerRegion = [[UITextField alloc] initWithFrame:CGRectMake(0,80,200,40)];
[self.summonerRegion setCenter:CGPointMake(self.view.center.x, self.summonerRegion.center.y)];
[self.summonerRegion setBackgroundColor:[UIColor whiteColor]];
[self.summonerRegion setLeftViewMode:UITextFieldViewModeAlways];
[self.summonerRegion setLeftView:[[UIView alloc] initWithFrame:CGRectMake(0, 0, 10, 10)]];
[self.summonerRegion setPlaceholder:@"Summoner Region"];
[self.summonerRegion setKeyboardType:UIKeyboardTypeEmailAddress];
[self.summonerRegion setAutocapitalizationType:UITextAutocapitalizationTypeNone];
[self.summonerRegion setAutocorrectionType:UITextAutocorrectionTypeNo];
// [self.summonerRegion setAutoresizingMask:(UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin)];
[cell.contentView addSubview:self.summonerRegion];
self.searchButton = [[UIButton alloc]initWithFrame:CGRectMake(0,130,200,40)];
[self.searchButton setCenter:CGPointMake(self.view.center.x, self.searchButton.center.y)];
[self.searchButton setTitle:@"Search" forState:UIControlStateNormal];
[self.searchButton.titleLabel setTextColor:[UIColor whiteColor]];
[self.searchButton setBackgroundImage: [AppDelegate imageFromColor:[UIColor LHGoldColor]] forState:UIControlStateNormal];
[self.searchButton setBackgroundImage: [AppDelegate imageFromColor:[UIColor LHGoldLightColor]] forState:UIControlStateHighlighted];
[self.searchButton setBackgroundImage: [AppDelegate imageFromColor:[UIColor LHGoldLightColor]] forState:UIControlStateSelected];
[self.searchButton setContentEdgeInsets:UIEdgeInsetsMake(5, 15, 5, 15)];
[self.searchButton addTarget:self action:@selector(search:) forControlEvents:UIControlEventTouchUpInside];
// [self.searchButton setAutoresizingMask:(UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin)];
[cell.contentView addSubview:self.searchButton];
_pickerView = [[UIPickerView alloc] init];
[self.summonerName setDelegate:self];
[self.summonerRegion setInputView:_pickerView];
[self.summonerRegion setDelegate:self];
// Initialize Data
_pickerData = @[@"NA", @"EUW", @"EUNE", @"BR", @"KR", @"LAN", @"LAS", @"OCE", @"RU", @"TR"];
// Connect data
_pickerView.dataSource = self;
_pickerView.delegate = self;
_pickerView.showsSelectionIndicator = YES;
[self.summonerRegion setText:[_pickerData objectAtIndex:0]];
} else if (indexPath.row == (recentSearchesArray.count+1)) {
// footer view 1
cell = [tableView dequeueReusableCellWithIdentifier:footerCellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:footerCellIdentifier];
}
cell.frame = CGRectMake(0, 0, self.recentSearchesTable.frame.size.width, 140);
cell.backgroundColor = [UIColor LHRedColor];
cell.textLabel.textColor = [UIColor LHBeigeLight];
cell.textLabel.text = @"Follow Us On Twitter";
cell.detailTextLabel.textColor = [UIColor LHBeigeLight];
cell.detailTextLabel.text = @"for app updates and events!";
cell.selectionStyle = UITableViewCellSelectionStyleNone;
} else if (indexPath.row == (recentSearchesArray.count+2)) {
// footer view 2
cell = [tableView dequeueReusableCellWithIdentifier:footerCellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:footerCellIdentifier];
}
cell.frame = CGRectMake(0, 0, self.recentSearchesTable.frame.size.width, 140);
cell.textLabel.textColor = [UIColor LHBeigeLight];
cell.backgroundColor = [UIColor LHGreenColor];
cell.textLabel.text = @"Battlefy Tournaments";
cell.detailTextLabel.textColor = [UIColor LHBeigeLight];
cell.detailTextLabel.text = @"Check out our community tournaments!";
cell.selectionStyle = UITableViewCellSelectionStyleNone;
} else {
// regular cell
Summoner *summoner = [recentSearchesArray objectAtIndex:indexPath.row-1];
cell = [tableView dequeueReusableCellWithIdentifier:summonerCellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:summonerCellIdentifier];
}
cell.textLabel.text = summoner.summonerName;
cell.detailTextLabel.text = [summoner.summonerRegion uppercaseString];
cell.backgroundColor = [UIColor whiteColor];
cell.selectionStyle = UITableViewCellSelectionStyleDefault;
}
return cell;
}
我在第一个单元格中注释掉了破坏我的自定义视图的三行,并取消了格式设置。无论出于何种原因,如果我设置左右灵活的自动调整大小掩码,单元格元素就会在 iPad 版本上消失。我正在使用这些 UIViewAutoresizing
掩码来处理旋转(通过使元素在第一个单元格中水平居中)。
有人知道为什么会这样吗?如果我设置这个 0 索引的方式 UITableViewCell
是不好的做法,或者如果我的代码有其他问题,请告诉我!
编辑: 更多信息。这是 tableView:heightForRowAtIndexPath:
的样子:
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
if (indexPath.row == 0) {
return 200;
} else if (indexPath.row >= recentSearchesArray.count + 1) {
return 130;
} else {
return 44;
}
}
这些是 po [self.view recursiveDescription]
对于 iPhone 6:
(lldb) po [self.view recursiveDescription]
<UIView: 0x7ff3e156d290; frame = (0 0; 320 568); autoresize = W+H; layer = <CALayer: 0x7ff3e156d360>>
| <UITableView: 0x7ff3e1858e00; frame = (0 0; 320 568); clipsToBounds = YES; autoresize = RM+BM; gestureRecognizers = <NSArray: 0x7ff3e156ad60>; layer = <CALayer: 0x7ff3e1568610>; contentOffset: {0, 0}; contentSize: {320, 0}>
| | <UITableViewWrapperView: 0x7ff3e156bbf0; frame = (0 0; 320 568); gestureRecognizers = <NSArray: 0x7ff3e156c660>; layer = <CALayer: 0x7ff3e156c160>; contentOffset: {0, 0}; contentSize: {320, 568}>
| | <UIImageView: 0x7ff3e15698b0; frame = (0 565.5; 320 2.5); alpha = 0; opaque = NO; autoresize = TM; userInteractionEnabled = NO; layer = <CALayer: 0x7ff3e15294a0>>
| | <UIImageView: 0x7ff3e156a5d0; frame = (317.5 561; 2.5 7); alpha = 0; opaque = NO; autoresize = LM; userInteractionEnabled = NO; layer = <CALayer: 0x7ff3e156a5b0>>
和 iPad 2:
(lldb) po [self.view recursiveDescription]
<UIView: 0x7869a230; frame = (0 0; 320 568); autoresize = W+H; layer = <CALayer: 0x786a2b20>>
| <UITableView: 0x78907600; frame = (0 0; 320 568); clipsToBounds = YES; autoresize = RM+BM; gestureRecognizers = <NSArray: 0x78690b00>; layer = <CALayer: 0x786a62c0>; contentOffset: {0, 0}; contentSize: {320, 0}>
| | <UITableViewWrapperView: 0x786a9470; frame = (0 0; 320 568); gestureRecognizers = <NSArray: 0x786a8290>; layer = <CALayer: 0x7866ab40>; contentOffset: {0, 0}; contentSize: {320, 568}>
| | <UIImageView: 0x786a9160; frame = (0 565; 320 3); alpha = 0; opaque = NO; autoresize = TM; userInteractionEnabled = NO; layer = <CALayer: 0x7866a750>>
| | <UIImageView: 0x78679fe0; frame = (317 561; 3 7); alpha = 0; opaque = NO; autoresize = LM; userInteractionEnabled = NO; layer = <CALayer: 0x78671490>>
编辑 2: 为最佳实践添加了出队代码
编辑 3: 根据 Brian 的建议切换到更好的单元格初始化函数 [UITableViewCell new];
并将子视图添加到内容视图而不是单元格本身。仍然没有运气。
编辑 4 已修复: 最后我不得不切换到使用自定义 UITableViewCell
子类并直接在 .xib 中执行所有约束以解决此问题.我不确定根本原因是什么,但如果您仍然想重现它,您可能只需将测试应用程序中的每个单元格都设为我为 0 索引创建的单元格即可。
Edit 5 FIXED X 2: 我用上面的代码找到了原始问题的解决方案。你可以在这里查看
我回去再看问题,终于找到了真凶。我试图将子视图在单元格内水平居中,所以调用了这样的东西:
[self.summonerName setCenter:CGPointMake(self.view.center.x, self.summonerName.center.y)];
出于某种原因 self.view.center.x
在 iPad 上评估了一些完全出乎意料的事情,并将内容视图定位在屏幕外。我真正应该做的(以及解决上述代码问题的方法)是基于 cell 的 中心而不是 superview 的 [=22] 设置子视图中心=] 像这样居中:
[self.summonerName setCenter:CGPointMake(cell.center.x, self.summonerName.center.y)];
为了向所有好奇的人进行演示,我创建了一个绝对准系统示例项目来说明这个问题,您可以在此处查看: https://github.com/oseparovic/ipaduitableviewbug