Изменение Размера UITextView

у меня есть UITextView добавлено на my UIView. Добавленный textview не редактируется, он просто отображает некоторые данные. Данные, отображаемые в textview, являются динамическими. То есть количество строк не фиксировано. Она может варьироваться. Поэтому, если количество строк увеличивается, размер textview также должен быть увеличен. Я понятия не имею, как это делать. Пожалуйста, дайте мне несколько идей.

обновление:

вот что я делаю:

UIView *baseView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 300, 200)];
baseView.backgroundColor = [UIColor grayColor];
[window addSubview:baseView];

UITextView *textView = [[UITextView alloc] initWithFrame:CGRectMake(5, 30, 100, 30)];
textView.autoresizingMask = UIViewAutoresizingFlexibleHeight;
textView.text = @"asdf askjalskjalksjlakjslkasj";
[textView sizeToFit];
[baseView addSubview:textView];

10 ответов


можно использовать setFrame: или sizeToFit.

обновление:

я использую sizeToFit С UILabel, и он работает просто отлично, но UITextView является наследником UIScrollView, так что я могу понять, почему sizeToFit не дает желаемого результата.

вы все еще можете вычислить высоту текста и использовать setFrame, но вы можете воспользоваться UITextView'полосы прокрутки, если текст слишком длинный.

вот как вы получаете текст высота:

#define MAX_HEIGHT 2000

NSString *foo = @"Lorem ipsum dolor sit amet.";
CGSize size = [foo sizeWithFont:[UIFont systemFontOfSize:14]
              constrainedToSize:CGSizeMake(100, MAX_HEIGHT)
                  lineBreakMode:UILineBreakModeWordWrap];

и тогда вы можете использовать это с вашим UITextView:

[textView setFont:[UIFont systemFontOfSize:14]];
[textView setFrame:CGRectMake(5, 30, 100, size.height + 10)];

или вы можете сделать расчет высоты сначала и избежать setFrame строку:

UITextView *textView = [[UITextView alloc] initWithFrame:CGRectMake(5, 30, 100, size.height + 10)];

есть ответ, размещенный на как определить размер UITextView для его содержимого?

CGRect frame = _textView.frame;
frame.size.height = _textView.contentSize.height;
_textView.frame = frame;

или лучше (с учетом contentInset благодаря комментарию kpower)

CGRect frame = _textView.frame;
UIEdgeInsets inset = textView.contentInset;
frame.size.height = _textView.contentSize.height + inset.top + inset.bottom;
_textView.frame = frame;

Примечание: Если вы собираетесь ссылаться на свойство объекта много раз(например, frame или contentInset), лучше назначить его локальной переменной, чтобы вы не вызывали дополнительные вызовы методов (_textView.frame / [_textView frame] - это вызовы методов). Если вы вызываете этот код лот (100000s раз), то это будет заметно медленнее(десяток или около того вызовов метода незначительны).

однако... если вы хотите сделать это в одной строке без дополнительных переменных, это будет

        _textView.frame = CGRectMake(_textView.frame.origin.x, _textView.frame.origin.y, _textView.frame.size.width, _textView.contentSize.height + _textView.contentInset.top + _textView.contentInset.bottom);

за счет 5 дополнительных вызовов метода.


textView.contentSize.height на textViewDidChange можно только изменить размер после текст на самом деле растет. Для лучшего визуального результата лучше заранее изменить размер. Через несколько часов я понял, как сделать его таким же, как в Instagram (у него лучший алгоритм среди всех BTW)

инициализировать следующим образом:

    // Input
    _inputBackgroundView = [[UIImageView alloc] initWithFrame:CGRectMake(0.0f, size.height - _InputBarHeight, size.width, _InputBarHeight)];
    _inputBackgroundView.autoresizingMask = UIViewAutoresizingNone;
   _inputBackgroundView.contentMode = UIViewContentModeScaleToFill;
    _inputBackgroundView.userInteractionEnabled = YES;
    [self addSubview:_inputBackgroundView];
   [_inputBackgroundView release];

   [_inputBackgroundView setImage:[[UIImage imageNamed:@"Footer_BG.png"] stretchableImageWithLeftCapWidth:80 topCapHeight:25]];

    // Text field
    _textField = [[UITextView alloc] initWithFrame:CGRectMake(70.0f, 0, 185, 0)];
   _textField.backgroundColor = [UIColor clearColor];
    _textField.delegate = self;
   _textField.contentInset = UIEdgeInsetsMake(-4, -2, -4, 0);
   _textField.showsVerticalScrollIndicator = NO;
   _textField.showsHorizontalScrollIndicator = NO;
    _textField.font = [UIFont systemFontOfSize:15.0f];
    [_inputBackgroundView addSubview:_textField];
   [_textField release];

   [self adjustTextInputHeightForText:@""];

заполните методы делегата UITextView:

- (void) textViewDidBeginEditing:(UITextView*)textView {

   [self adjustTextInputHeightForText:_textField.text];
}

- (void) textViewDidEndEditing:(UITextView*)textView {

   [self adjustTextInputHeightForText:_textField.text];
}

- (BOOL) textView:(UITextView*)textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString*)text {

   if ([text isEqualToString:@"\n"])
   {
      [self performSelector:@selector(inputComplete:) withObject:nil afterDelay:.1];
      return NO;
   }
   else if (text.length > 0)
   {
      [self adjustTextInputHeightForText:[NSString stringWithFormat:@"%@%@", _textField.text, text]];
   }
   return YES;
}

- (void) textViewDidChange:(UITextView*)textView {

   [self adjustTextInputHeightForText:_textField.text];
}

и фокус в том...

- (void) adjustTextInputHeightForText:(NSString*)text {

   int h1 = [text sizeWithFont:_textField.font].height;
   int h2 = [text sizeWithFont:_textField.font constrainedToSize:CGSizeMake(_textField.frame.size.width - 16, 170.0f) lineBreakMode:UILineBreakModeWordWrap].height;

   [UIView animateWithDuration:.1f animations:^
   {
      if (h2 == h1)
      {
         _inputBackgroundView.frame = CGRectMake(0.0f, self.frame.size.height - _InputBarHeight, self.frame.size.width, _InputBarHeight);
      }
      else
      {
         CGSize size = CGSizeMake(_textField.frame.size.width, h2 + 24);
         _inputBackgroundView.frame = CGRectMake(0.0f, self.frame.size.height - size.height, self.frame.size.width, size.height);
      }
      CGRect r = _textField.frame;
      r.origin.y = 12;
      r.size.height = _inputBackgroundView.frame.size.height - 18;
      _textField.frame = r;

   } completion:^(BOOL finished)
   {
      //
   }];
}

sizeToFit работает

Если вы вызываете sizeToFit после установки текста при первом изменении его размера. Таким образом, после первого раза, когда вы установите его, последующие вызовы для установки текста не приведут к изменению размера. Даже если вы вызываете sizeToFit.

однако, вы можете заставить его изменить такой:

  1. Набор текста.
  2. измените высоту кадра textView на CGFLOAT_MAX.
  3. вызовите sizeToFit.

это отлично работает для меня:

    #define MAX_HEIGHT 2000     
    CGSize size = [text sizeWithFont:[UIFont systemFontOfSize:14]
          constrainedToSize:CGSizeMake(100, MAX_HEIGHT)
              lineBreakMode:UILineBreakModeWordWrap];

    [textview setFont:[UIFont systemFontOfSize:14]];
    [textview setFrame:CGRectMake(45, 6, 100, size.height + 10)];
    textview.text = text;

сделать следующее:

_textView.text = someText;
[_textView sizeToFit];
_textView.frame.height = _textView.contentSize.height;

решение аналогичной проблемы я только что создал легкий подкласс UITextView на основе автоматической компоновки, который автоматически растет и сжимается в зависимости от размера пользовательского ввода и может быть ограничен максимальной и минимальной высотой - все без одной строки кода.

https://github.com/MatejBalantic/MBAutoGrowingTextView


ответ дал @Гейб не работает в iOS7.1 казалось бы, после viewDidAppear. См. мои тесты ниже.

UPDATE: на самом деле, ситуация еще более сложная. Если вы назначаете textView.текст в методе resizeTheTextView в iOS7 изменение размера равно разрешению только одной строки текста. Серьезно странно.

UPDATE2: Смотрите также размер контента UITextView отличается в iOS7

UPDATE3: см. мой код в самом низу для что я использую сейчас. Похоже, он справляется.

#import "ViewController.h"

@interface ViewController ()
{
    UITextView *textView;
}
@end

@implementation ViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.

    textView = [[UITextView alloc] initWithFrame:CGRectMake(50, 50, 200, 1)];
    [self.view addSubview:textView];
    CALayer *layer = textView.layer;
    layer.borderColor = [UIColor blackColor].CGColor;
    layer.borderWidth = 1;

    textView.text = @"hello world\n\n";

    // Calling the method directly, after the view is rendered, i.e., after viewDidAppear, works on both iOS6.1 and iOS7.1
    UIButton *button = [UIButton buttonWithType:UIButtonTypeRoundedRect];
    [button setTitle:@"Change size" forState:UIControlStateNormal];
    [button addTarget:self action:@selector(resizeTheTextView) forControlEvents:UIControlEventTouchUpInside];
    [button sizeToFit];
    CGRect frame = button.frame;
    frame.origin.y = 400;
    button.frame = frame;
    [self.view addSubview:button];

    // Works on iOS6.1, but does not work on iOS7.1
    //[self resizeTheTextView];
}

- (void) viewWillAppear:(BOOL)animated
{
    // Does not work on iOS7.1, but does work on iOS6.1
    //[self resizeTheTextView];
}

- (void) viewDidAppear:(BOOL)animated
{
    // Does work on iOS6.1 and iOS7.1
    //[self resizeTheTextView];
}

- (void) resizeTheTextView
{
    NSLog(@"textView.frame.size.height: %f", textView.frame.size.height);

    NSLog(@"textView.contentSize.height: %f", textView.contentSize.height);

    // 5) From https://stackoverflow.com/questions/728704/resizing-uitextview
    CGRect frame = textView.frame;
    UIEdgeInsets inset = textView.contentInset;
    frame.size.height = textView.contentSize.height + inset.top + inset.bottom;
    textView.frame = frame;

    NSLog(@"inset.top: %f, inset.bottom: %f",  inset.top,  inset.bottom);

    NSLog(@"textView.frame.size.height: %f", textView.frame.size.height);

    NSLog(@"textView.contentSize.height: %f", textView.contentSize.height);
}

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

@end

UPDATE3:

if ([[UIDevice currentDevice] majorVersionNumber] < 7.0) {
    CGRect frame = _abstractTextView.frame;
    UIEdgeInsets inset = _abstractTextView.contentInset;
    frame.size.height = _abstractTextView.contentSize.height + inset.top + inset.bottom;
    _abstractTextView.frame = frame;
}
else {
    CGSize textViewSize = [_abstractTextView sizeThatFits:CGSizeMake(_abstractTextView.frame.size.width, FLT_MAX)];
    _abstractTextView.frameHeight = textViewSize.height;
}

после добавления UITextView к его родителю, если вы установить режим содержания на нем, похоже, автоматически изменяется размер.

Это означает, что вам не нужно разрабатывать высоту вручную и применять противопоставление высоты. Кажется, это работает!! Протестировано в iOS7 и iOS8 на iPad.

например

--
textView.contentMode = UIViewContentMode.Center;
--

если кто-нибудь может объяснить, почему это работает, это будет высоко ценится.. Я нашел его случайно, когда возился с опциями в интерфейсе строитель.


просто набор scrollEnabled до NO, или снимите флажок Прокрутка в разделе просмотра прокрутки в IB и UITextView будет self-size.