2

我有一个UITableView4 UILabel's:标题、正文、作者和日期,他看起来像这样:

细胞55

我想要完成的是,当用户单击单元格本身时,应将另一个标签添加到单元格中,“正文”标签和单元格应根据此标签大小展开。

像这样的东西:

单元格大小

我怎样才能做到这一点?我搜索了stackoverflow,尝试了一些代码片段,但仍然没有找到正确的解决方案。

谢谢!

编辑 1:14.11.12 14:52

我设法用当前文本改变了 UILabel 的大小:

- (CGRect )resizeLabelByFontSize:(UILabel *)customCellLabel withMaxHeightSize:(CGFloat )maxHeight
{
    CGSize maximumLabelSize = CGSizeMake(239, maxHeight);

    CGSize expectedLabelSize = [customCellLabel.text sizeWithFont:customCellLabel.font constrainedToSize:maximumLabelSize lineBreakMode:customCellLabel.lineBreakMode];

    //adjust the label the the new height.
    CGRect newFrame = customCellLabel.frame;
    newFrame.size.height = expectedLabelSize.height;

    return newFrame;
}

但是如何根据新的 UILabel 的大小来改变单元格的大小呢?

4

6 回答 6

1

通过查看有问题的图像

这是刚刚创建动态框架的方法,UILabel看看这个通过获取高度和宽度,UIlabel您可以计算整个高度并可以设置行高UITableView.

- (void)setLabeltextWithVerticalAlignTop:(NSString *)theText
{
CGSize labelSize;
// here  labelSize is hard-wired but could use constants to populate the size

labelSize = CGSizeMake(210, 129);//this is just for example
//now create the Size from textString SO that We  could assign this size to the Label.

 CGSize theStringSize = [theText sizeWithFont:lblTitle.font  constrainedToSize:labelSize lineBreakMode:lblTitle.lineBreakMode];
 lblTitle.frame = CGRectMake(lblTitle.frame.origin.x, lblTitle.frame.origin.y, theStringSize.width, theStringSize.height);
 lblTitle.text = theText;

}

调用上述方法要设置描述标签的高度和宽度,您需要传递要在该描述标签上显示的文本。当您获得该标签的高度时,现在您可以在此基础上调整 TableView 行的高度。

编辑:上面的代码只是为 UILabel 创建动态框架

你应该看看这就是你要找的东西......!!!.here你也可以找到一个示例代码。

编辑:当你编辑你的问题时,它只是你需要将其转换为可运行代码的逻辑。

在您的代码中使用以下方法为每一行调用,并在其中进行一些计算。

 - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{
CGFloat rowHeight=0.0;
//here it seems cell have 4 subview added on it.
//so if you could calculate the totla hieht of them.

//so what you really need to do.you just use hieght calculative Method for getting hieght of each of three UILabel
//you need to modify  `setLabeltextWithVerticalAlignTop` method .
rowHeight=   [self setLabeltextWithVerticalAlignTop:@"pass the correspondingText"];// suppose it returns some hieght for FisrtLabel.

//suppoose here you get the 20.0 height here

rowHeight= rowHeight+[self setLabeltextWithVerticalAlignTop:@"pass the correspondingText"];
 

// 假设它为 secondUIlabel 返回一些高度。

//suppoose here you get the 40.0 height here

rowHeight=  rowHeight+ [self setLabeltextWithVerticalAlignTop:@"pass the correspondingText"];

 // suppose it returns some hieght for ThirdUIlabel.
// suppoose here you get the 15.0 height here
//here you have totla height you just need to add some gapping floating value for all of three UIlabel.so that the could not overlap like as.

 rowHeight= rowHeight+20.0;

  //now you can return that total height
  return rowHeight;
 }

注意:这只是您需要将其转换为可运行代码的逻辑。我相信这会有所帮助。

我希望它可以帮助你。

于 2012-11-14T11:57:39.317 回答
0

实现以下方法

– (void) tableView:(UITableView*)tableView didSelectRowAtIndexPath:(NSIndexPath*)indexPath
{
    UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];

    // cast cell, add label, expand labels etc

    [tableView reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
}

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{
    return [indexPath isEqualTo:[tableView indexPathForSelectedRow]] ? /* expanded height */ : 80 /* normal height */;
}

如果您希望该行即使在选择另一行后仍保持选中状态,则将自定义 BOOL 属性添加到您的自定义单元格,例如expanded,并使用它来确定高度。

于 2012-11-14T11:44:50.237 回答
0

您可以使用 tableView:didSelectRowAtIndexPath

在该方法中,您可以创建代码以取消隐藏正文标签,调整其他所有内容的相对位置。计算新的行高,然后调用 Table View 的 reloadRowsAtIndexPath: withRowAnimation: 方法。

抱歉,如果没有太多细节,但希望这能让你走上正轨。

于 2012-11-14T11:45:24.367 回答
0

好的,首先......要扩展你需要这样的东西:

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath;

现在这是问题所在:

  • 您应该计算 UITableViewCell 的大小(展开和非展开)
  • 在您实际滚动时这样做可能会很昂贵并且会给您带来糟糕的体验

我的建议:

  • 在你真正结束构建之前计算两边,UITableView因为你想要动态大小。如果您不这样做,并且所有单元格的扩展尺寸相同,您可以使用 lammmert 所说的。
于 2012-11-14T11:46:41.093 回答
0

NSIndexPath *selectedRow;

 - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
selectedRow = indexPath;
}


 - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{
 if(indexPath == selectedRow){
    //return your custom value
}
return 100;

}

我认为它看起来像这样

于 2012-11-14T12:26:50.407 回答
0

因此,为了做到这一点,使用扩展的 UITableViewCell,我创建了 2 个不同的自定义单元格,在开始时表格显示第一个单元格,当我单击单元格时,表格显示第二个单元格。就这么简单——是的!

所以我有 UIViewController 和 UITableView 实现表委托方法:

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{    
    if([self.selectedCellIndexPath isEqual:indexPath])
    {
        return [self expandedCellHeight:indexPath];
    }
    else
    {
        return kRegularCellHeight;
    }
}

-(CGFloat)expandedCellHeight:(NSIndexPath *)indexPath
{
    CGSize maxSize = CGSizeMake(303, 200);
    NSString* bodyText  = [[self.data objectAtIndex:indexPath.row] objectForKey:kForumMessagesBody];
    CGSize fitSize = [bodyText sizeWithFont:[UIFont systemFontOfSize:13] constrainedToSize:maxSize lineBreakMode:UILineBreakModeWordWrap];

    CGFloat height = 384 - 69 + fitSize.height;
    NSLog(@"expandedHeight: %f",height);
    return height;
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    // Answer cell
    if ([self.selectedCellIndexPath isEqual:indexPath])
    {
        cell = [tableView dequeueReusableCellWithIdentifier:[ForumCell expandedAnswerReuseIdentifier]];
        if (cell == nil)
        {
            cell = [ForumCell expandedAnswerCell];
        }
        self.expandedCell = cell;

    }
    else
    {
        cell = [tableView dequeueReusableCellWithIdentifier:[ForumCell reqularAnswerReuseIdentifier]];
        if (cell == nil)
        {
            cell = [ForumCell regularAnswerCell];
        }
    }

    cell.labelMedia.text = [self.data objectAtIndex:indexPath.row];

    return cell;
}

我也有自定义单元格,名为ForumCell.hand的类ForumCell.m,它有 2 个不同的 XIB 文件:ForumRegularAnswerCell.xib并且ForumExpandedAnswerCell.xib,我在里面有以下代码ForumCell.h

+ (NSString*)reqularAnswerReuseIdentifier
{
    return @"RegularAnswerCellReuseIdentifier";
}

+ (NSString*)expandedAnswerReuseIdentifier
{
    return @"ExpandedAnswerCellReuseIdentifier";
}

+ (ForumCell*)regularAnswerCell
{
    NSArray* objs = [[NSBundle mainBundle] loadNibNamed:@"ForumRegularAnswerCell" owner:self options:nil];
    ForumCell* result = [objs objectAtIndex:0];

    return result;
}

+ (ForumCell*)expandedAnswerCell
{
    NSArray* objs = [[NSBundle mainBundle] loadNibNamed:@"ForumExpandedAnswerCell" owner:self options:nil];
    ForumCell* result = [objs objectAtIndex:0];

    return result;
}

- (id)initWithCoder:(NSCoder *)decoder
{
    self = [super initWithCoder:decoder];
    if (self)
    {
        _originalCellHeight = self.frame.size.height;
        _originalLblBodyHeight = self.lblBody.frame.size.height;
    }
    return self;
}

如果您愿意,也可以使用 2 个以上的 xib。但这是基础。

享受!

于 2013-01-01T08:37:48.617 回答