UITextView在UITableViewCell中平滑自动调整大小,在iPad上显示和隐藏键盘,但适用于iPhone

Jas*_*son 17 iphone uitableview uitextview ipad ios

我已经实现了一个自定义UITableViewCell,它包含一个UITextView,可以在用户输入时自动调整大小,类似于Contacts应用程序中的"Notes"字段.它在我的iPhone上运行正常,但是当我在iPad上测试它时,我会遇到一些非常奇怪的行为:当你到达一行时,键盘会隐藏一毫秒,然后立即显示自己.我会把它写成一个古怪的bug,但它实际上会导致一些数据丢失,因为如果你打字,它会丢失一两个字符.这是我的代码:

代码

// returns the proper height/size for the UITextView based on the string it contains.
// If no string, it assumes a space so that it will always have one line.
- (CGSize)textViewSize:(UITextView*)textView {
     float fudgeFactor = 16.0;
     CGSize tallerSize = CGSizeMake(textView.frame.size.width-fudgeFactor, kMaxFieldHeight);
     NSString *testString = @" ";
     if ([textView.text length] > 0) {
          testString = textView.text;
     }
     CGSize stringSize = [testString sizeWithFont:textView.font constrainedToSize:tallerSize lineBreakMode:UILineBreakModeWordWrap];
     return stringSize;
}

// based on the proper text view size, sets the UITextView's frame
- (void) setTextViewSize:(UITextView*)textView {
     CGSize stringSize = [self textViewSize:textView];
     if (stringSize.height != textView.frame.size.height) {
          [textView setFrame:CGRectMake(textView.frame.origin.x,
                                        textView.frame.origin.y,
                                        textView.frame.size.width,
                                        stringSize.height+10)];  // +10 to allow for the space above the text itself 
     }
}

// as per: https://stackoverflow.com/questions/3749746/uitextview-in-a-uitableviewcell-smooth-auto-resize
- (void)textViewDidChange:(UITextView *)textView {

     [self setTextViewSize:textView]; // set proper text view size
     UIView *contentView = textView.superview;
     // (1) the padding above and below the UITextView should each be 6px, so UITextView's
     // height + 12 should equal the height of the UITableViewCell
     // (2) if they are not equal, then update the height of the UITableViewCell
     if ((textView.frame.size.height + 12.0f) != contentView.frame.size.height) {
         [myTableView beginUpdates];
         [myTableView endUpdates];

         [contentView setFrame:CGRectMake(0,
                                          0,
                                          contentView.frame.size.width,
                                          (textView.frame.size.height+12.0f))];
     }
}

- (CGFloat)tableView:(UITableView  *)tableView heightForRowAtIndexPath:(NSIndexPath  *)indexPath {
     int height;
     UITextView *textView = myTextView;
     [self setTextViewSize:textView];
     height = textView.frame.size.height + 12;
     if (height < 44) { // minimum height of 44
          height = 44;
          [textView setFrame:CGRectMake(textView.frame.origin.x,
                                        textView.frame.origin.y,
                                        textView.frame.size.width,
                                        44-12)];
      }
      return (CGFloat)height;
}
Run Code Online (Sandbox Code Playgroud)

问题

所以,这就是发生的事情

  1. 此代码在我的iPhone和iPhone模拟器中正常工作.当我输入文本时,UITextView会顺利增长,UITableViewCell也会随之增长.
  2. 然而,在iPad模拟器上,它变得棘手.当你在第一行输入时它工作正常,但是当你到达一行的末尾时,键盘会消失,然后立即重新出现,这样如果用户继续输入应用程序就会错过一两个字符.
  3. 以下是一些关于我注意到的奇怪行为的补充说明,可能有助于解释它:
    • 此外,我发现删除[myTableView beginUpdates]; [myTableView endUpdates];函数textViewDidChange:(UITextView *)textView中的行使UITextView正常生长,也不显示和隐藏键盘,但不幸的是,UITableViewCell没有增长到适当的高度.
    • 更新:按照这些说明,我现在能够阻止文本的奇怪运动; 但键盘仍然隐藏和显示,这是非常奇怪的.

有没有人知道如何让键盘不断显示,而不是隐藏和显示当你到达iPad的终点时?

PS:我对使用ThreeTwenty不感兴趣.

Den*_*cik 15

你应该返回NO:

 -(BOOL) textViewShouldEndEditing:(UITextView *)textView
Run Code Online (Sandbox Code Playgroud)

如果您想随时显示键盘.您应该通过向此委托函数返回YES来处理应该隐藏键盘的情况.

编辑:

我调了一下,当[tableView endUpdates]调用时,它基本上做了三件事:

  1. 禁用tableView上的用户交互
  2. 更新单元格更改
  3. tableView上启用用户交互

SDK(平台)之间的区别在于[UIView setUserInteractionEnabled]方法.由于UITableView不会覆盖setUserInteractionEnabled方法,因此从super(UIView)调用它.

iPhone 调用setUserInteractionEnabled时,查找私有字段_shouldResignFirstResponderWithInteractionDisabled,默认返回NO,因此不会重新签名第一个响应者(UITextView)

但是在iPad上没有这样的检查AFAIK,所以它在步骤1重新调整UITextView,并设置焦点并使其成为第3步的第一响应者

基本上,textViewShouldEndEditing允许您保持焦点,根据SDK文档,它是您唯一的选择ATM.

当要求文本视图重新签署第一响应者状态时,将调用此方法.当用户尝试将编辑焦点更改为另一个控件时,可能会发生这种情况.但是,在焦点实际更改之前,文本视图会调用此方法以使您的代理人有机会决定是否应该这样做.


Flo*_*lke 11

我对iPad应用程序有同样的问题,并提出了另一种解决方案,而没有计算文本本身的高度.

首先在IB中创建一个自定义UITableViewCell,并在单元格的contentView中放置一个UITextField.将文本视图的scrollEnabled设置为NO并将autoresizingMask设置为flexibleWidth和flexibleHeight非常重要.


在ViewController中实现文本视图的委托方法-textViewDidChanged:如下所示,其中textHeight是类型为CGFloat的实例变量,-tableViewNeedsToUpdateHeight是我们将在下一步中定义的自定义方法.

- (void)textViewDidChange:(UITextView *)textView
{
    CGFloat newTextHeight = [textView contentSize].height;

    if (newTextHeight != textHeight)
    {
        textHeight = newTextHeight;
        [self tableViewNeedsToUpdateHeight];
    }
}
Run Code Online (Sandbox Code Playgroud)

方法-tableViewNeedsToUpdateHeight调用表视图的beginUpdatesendUpdates,因此表视图本身将调用-tableView:heightForRowAtIndexPath: delegate方法.

- (void)tableViewNeedsToUpdateHeight
{
    BOOL animationsEnabled = [UIView areAnimationsEnabled];
    [UIView setAnimationsEnabled:NO];
    [table beginUpdates];
    [table endUpdates];
    [UIView setAnimationsEnabled:animationsEnabled];
}
Run Code Online (Sandbox Code Playgroud)

在表视图的-tableView:heightForRowAtIndexPath: delegate方法中,我们需要根据textHeight计算文本视图单元格的新高度.

首先,我们需要将文本视图单元格高度调整为最大可用高度(在减去表视图中所有其他单元格的高度之后).然后我们检查textHeight是否大于计算的高度.

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{   
    CGFloat heightForRow = 44.0;

    if ([indexPath row] == kRowWithTextViewEmbedded)
    {
        CGFloat tableViewHeight = [tableView bounds].size.height;
        heightForRow = tableViewHeight - ((kYourTableViewsNumberOfRows - 1) * heightForRow);

        if (heightForRow < textHeight) 
        {
            heightForRow = textHeight;
        }
    }

    return heightForRow;
}
Run Code Online (Sandbox Code Playgroud)

为了获得更好的用户体验,请将表视图的内容插入设置为底部,例如50.0.

我已经在iPad上使用iOS 4.2.1进行了测试,并按预期工作.

弗洛里安