0

UITableView をグループ化しました。

これが私のヘッダービューメソッドです-注意してください。要素をリロードするのではなく、一度だけロードして次回ロードするようにしました。すでにサブビューがある場合はheaderViewを返しますリロードさせたくない):

-(UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section{
static NSString *headerReuseIdentifier = @"myHeader";

UITableViewHeaderFooterView *headerView = [tableView dequeueReusableHeaderFooterViewWithIdentifier:headerReuseIdentifier];
if (headerView == nil) {
    headerView = [[UITableViewHeaderFooterView alloc] initWithReuseIdentifier:headerReuseIdentifier];//[tableView dequeueReusableHeaderFooterViewWithIdentifier:headerReuseIdentifier];
    CGRect frame=CGRectMake(0, 0, tableView.bounds.size.width, 44);
    headerView.frame=frame;
}

if (headerView.contentView.subviews.count!=0){
    return headerView;//do not reload 
}
    UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(5, 0, tableView.frame.size.width-85, 44)];
    [label setFont:[UIFont boldSystemFontOfSize:16]];
    NSString *name=@"not found";
    [label setText:name];

    headerView.backgroundColor=[UIColor clearColor];
    [headerView.contentView addSubview:label];
    //... etc elements

return headerView;
}

セルにも同様のロジックを実装しました。つまり、サブビューが既にセルに追加されている場合、要素をリロードしません。そこでもどのように機能するかを確認するだけです。

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
static NSString *CellIdentifier = @"Cell";

UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) {
    cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
}

if (cell.contentView.subviews.count!=0) {
    return cell;//do not reload 
}

//adding elements here...
return cell;
}

================================================== ===================

ここでは、1回目の様子を示しています.Step:1とStep:2を正しい昇順で示していますここに画像の説明を入力

[myTableView reloadData] が呼び出されると、テーブルは次のようになります。ここに画像の説明を入力

注: headerView ビューは逆です! ただし、セルは常に正しい順序で表示されます。

[myTableView reloadData] が再度呼び出されると、headerViews が適切な順序で表示されます。

reloadData の呼び出しを回避することでこれを防ぐことができますが、headerView ではなくセルにデータをリロードする必要があります。

この headerViews のフリッピングを経験した人はいますか? どのように修正しましたか?

Xcode 8.3.2、iPad iOS9.0 シミュレーターで動作

この間違ったヘッドビューの並べ替えをテストできる完全なコード:

.h ファイル

@interface TableTestViewController : UIViewController
@property(strong,nonatomic)IBOutlet UITableView *myTable;
@property(strong,nonatomic)NSMutableArray *steps;
@end

.m ファイル

#import "TableTestViewController.h"

@interface TableTestViewController ()

@end

@implementation TableTestViewController
@synthesize steps,myTable;

- (void)viewDidLoad {
[super viewDidLoad];
steps=[[NSMutableArray alloc]initWithObjects:@"Step 1",@"Step 2", nil];
}

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
return steps.count;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
return 1;
}

-(CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section{
return 44.0;
}

-(UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section{
static NSString *headerReuseIdentifier = @"myHeader";

UITableViewHeaderFooterView *headerView = [tableView dequeueReusableHeaderFooterViewWithIdentifier:headerReuseIdentifier];
if (headerView == nil) {
    headerView = [[UITableViewHeaderFooterView alloc] initWithReuseIdentifier:headerReuseIdentifier];//[tableView dequeueReusableHeaderFooterViewWithIdentifier:headerReuseIdentifier];
    CGRect frame=CGRectMake(0, 0, tableView.bounds.size.width, 44);
    headerView.frame=frame;
}

if (headerView.contentView.subviews.count!=0) {
    return headerView;//do not reload when running
}

while (headerView.contentView.subviews.count) {
    id subview=[headerView.contentView.subviews objectAtIndex:0];
    [subview removeFromSuperview];
}

headerView.backgroundColor=[UIColor clearColor];

UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(5, 0, tableView.frame.size.width-85, 44)];
[label setFont:[UIFont boldSystemFontOfSize:16]];
[label setText:[steps objectAtIndex:section]];


[headerView.contentView addSubview:label];

    int btnSz=tableView.bounds.size.width/6;
    UIButton *startStepBtn = [[UIButton alloc]initWithFrame:CGRectMake(btnSz*5+5 ,5, 30, 30)];
    [startStepBtn setTitle:@"" forState:UIControlStateNormal];
    startStepBtn.tag=section+1000;
    startStepBtn.enabled=NO;
    [startStepBtn addTarget:self action:@selector(startStep:) forControlEvents:(UIControlEvents)UIControlEventTouchUpInside];

    [headerView.contentView addSubview:startStepBtn];


return headerView;
}

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{
int i=1;
return 60*ceil((double)i/5)+10;
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
static NSString *CellIdentifier = @"Cell";

UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) {
    cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
}

if ([cell.contentView.subviews count]!=0) {
    return cell;//do not reload when running
}

//clear 1st
while( [cell.contentView.subviews count] ){
    id subview = [cell.contentView.subviews objectAtIndex:0];
    [subview removeFromSuperview];
}

int btnSz=tableView.bounds.size.width/6;

UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(5, 0, tableView.frame.size.width-85, 44)];
[label setFont:[UIFont boldSystemFontOfSize:16]];
[label setText:[steps objectAtIndex:indexPath.section]];
[cell.contentView addSubview:label];

CGRect frame;

frame = CGRectMake(btnSz*5+5 ,5, 30, 30);
    UIButton *delStepBtn = [[UIButton alloc]initWithFrame:frame];
    [delStepBtn setTitle:@"❌" forState:UIControlStateNormal];
    [delStepBtn addTarget:self action:@selector(deleteStep:) forControlEvents:(UIControlEvents)UIControlEventTouchUpInside];
    [cell.contentView addSubview:delStepBtn];

    frame = CGRectMake(btnSz*5+5 ,5+30, 30, 30);
    UIButton *editStepBtn = [[UIButton alloc]initWithFrame:frame];
    [editStepBtn setTitle:@"" forState:UIControlStateNormal];
    [editStepBtn addTarget:self action:@selector(editStep:) forControlEvents:(UIControlEvents)UIControlEventTouchUpInside];
    [cell.contentView addSubview:editStepBtn];

return cell;
}

-(void)editStep:(id)sender{
[myTable reloadData];
}

@end

実装すると、ヘッダーが昇順で表示され、ハンマー ボタンをタッチするとテーブルがリロードされ、ヘッダーが降順になり、もう一度タッチすると昇順に戻ります。

ただし、セルの内容は常に昇順のままです。

4

2 に答える 2

0

あなたがしているそのチェックは必要ありません:

if (headerView.contentView.subviews.count!=0){
    return headerView;//do not reload 
}

ただし、UILabel毎回インスタンス化するのは良い習慣ではありません。すべてのインスタンス化はif (headerView == nil)if-case内で行うUILabel必要があり、それはこのヘッダーまたはセルのプロパティである必要があります。

text方法よりも、ヘッダー/セルをデキューしたときにプロパティを変更するだけです。

于 2017-06-13T16:41:12.710 に答える