UIInterfaceOrientation еще не обновляется, когда UIDeviceOrientationDidChangeNotification обнаружен в UIView

Я работаю над библиотекой с открытым исходным кодом, чтобы обеспечить представление ввода текста в стиле iMessage - MessageComposerView (имеет отношение к вопросу). Вид будет прилипать к клавиатуре как inputAccessoryView и расти с текстом, но не исчезнет, ​​когда клавиатура делает.

Недавно я столкнулся с сводящей с ума проблемой при повороте этого пользовательского UIView, который появляется, только если представление было создано с помощью initWithFrame, В основном, в тот момент, когда уведомление UIDeviceOrientationDidChangeNotification было обнаружено, если представление было создано с помощью initWithFrame, UIInterfaceOrientation и рамка еще не были обновлены. Вот оба способа создания экземпляров.

loadNibNamed:

self.messageComposerView = [[NSBundle mainBundle] loadNibNamed:@"MessageComposerView" owner:nil options:nil][0];
self.messageComposerView.frame = CGRectMake(0,
                                            self.view.frame.size.height - self.messageComposerView.frame.size.height,
                                            self.messageComposerView.frame.size.width,
                                            self.messageComposerView.frame.size.height);

initWithFrame:

self.messageComposerView = [[MessageComposerView alloc] initWithFrame:CGRectMake(0, self.view.frame.size.height-54, 320, 54)];

Когда инициация через loadNibNamed и повернут к ландшафту, получив UIDeviceOrientationDidChangeNotification уведомление:

  • UIDeviceOrientation = [[объект уведомления] ориентация] = UIInterfaceOrientationLandscapeLeft
  • UIInterfaceOrientation = [UIApplication sharedApplication].statusBarOrientation = UIDeviceOrientationLandscapeRight
  • self.frame.size = (ширина =480, высота =90)

Здесь важно отметить, что как интерфейс, так и ориентация устройства являются альбомными, а ширина уже изменена на альбомную (тестирование на симуляторе iPhone 6.1). Сейчас выполняю тот же тест, но используя initWithFrame:

  • UIDeviceOrientation = [[объект уведомления] ориентация] = UIDeviceOrientationLandscapeRight
  • UIInterfaceOrientation = [UIApplication sharedApplication].statusBarOrientation = UIInterfaceOrientationPortrait
  • self.frame.size = (ширина =320, высота =54)

На этот раз обратите внимание, что ориентация интерфейса все еще портретная и что ширина рамки НЕ изменилась на ширину ландшафта. Если я установлю точку останова в setFrame:(CGRect)frame Метод Я могу видеть, что кадр установлен на альбомную рамку ПОСЛЕ UIDeviceOrientationDidChangeNotification было перехвачено и обработано.

Оба метода init имеют практически одинаковую настройку:

- (id)initWithFrame:(CGRect)frame {
    self = [super initWithFrame:frame];
    if (self) {
        // Initialization code
        self.frame = frame;
        self.sendButton = [[UIButton alloc] initWithFrame:CGRectZero];
        self.messageTextView = [[UITextView alloc] initWithFrame:CGRectZero];
        [self setup];
        [self addSubview:self.sendButton];
        [self addSubview:self.messageTextView];

    }
    return self;
}

- (void)awakeFromNib {
    [super awakeFromNib];
    [self setup];
}

С setup Делаем необходимые твики вида:

- (void)setup {
    self.backgroundColor = [UIColor lightGrayColor];
    self.autoresizesSubviews = YES;
    self.autoresizingMask = UIViewAutoresizingFlexibleTopMargin | UIViewAutoresizingFlexibleBottomMargin | UIViewAutoresizingFlexibleWidth;
    self.userInteractionEnabled = YES;
    self.multipleTouchEnabled = NO;

    CGRect sendButtonFrame = self.bounds;
    sendButtonFrame.size.width = 50;
    sendButtonFrame.size.height = 34;
    sendButtonFrame.origin.x = self.frame.size.width - kComposerBackgroundRightPadding - sendButtonFrame.size.width;
    sendButtonFrame.origin.y = kComposerBackgroundRightPadding;
    self.sendButton.frame = sendButtonFrame;
    self.sendButton.autoresizingMask = UIViewAutoresizingFlexibleBottomMargin | UIViewAutoresizingFlexibleLeftMargin;
    self.sendButton.layer.cornerRadius = 5;
    [self.sendButton setTitleColor:[UIColor whiteColor] forState:UIControlStateNormal];
    [self.sendButton setTitleColor:[UIColor colorWithRed:210/255.0 green:210/255.0 blue:210/255.0 alpha:1.0] forState:UIControlStateHighlighted];
    [self.sendButton setTitleColor:[UIColor grayColor] forState:UIControlStateSelected];
    [self.sendButton setBackgroundColor:[UIColor orangeColor]];
    [self.sendButton setTitle:@"Send" forState:UIControlStateNormal];
    self.sendButton.titleLabel.font = [UIFont boldSystemFontOfSize:14];


    CGRect messageTextViewFrame = self.bounds;
    messageTextViewFrame.origin.x = kComposerBackgroundLeftPadding;
    messageTextViewFrame.origin.y = kComposerBackgroundTopPadding;
    messageTextViewFrame.size.width = self.frame.size.width - kComposerBackgroundLeftPadding - kComposerTextViewButtonBetweenPadding - sendButtonFrame.size.width - kComposerBackgroundRightPadding;
    messageTextViewFrame.size.height = 34;
    self.messageTextView.frame = messageTextViewFrame;
    self.messageTextView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleBottomMargin;
    self.messageTextView.showsHorizontalScrollIndicator = NO;
    self.messageTextView.layer.cornerRadius = 5;
    self.messageTextView.font = [UIFont systemFontOfSize:14];
    self.messageTextView.delegate = self;

    [self addNotifications];
    [self resizeTextViewForText:@"" animated:NO];
}

Итак, мой вопрос, почему мой пользовательский init UIView через initWithFrame по-прежнему имеют портретную рамку и ориентацию интерфейса, а также время получения UIDeviceOrientationDidChangeNotification. Я использую это уведомление, чтобы выполнить настройку кадра, и мне нужно, чтобы ширина уже была обновлена ​​до ширины ландшафта.

Я надеюсь, что в программной настройке отсутствует какое-то свойство авторотации, которое отсутствует в XIB, но просто не может его найти. Я прошел через дюжину UIDeviceOrientationDidChangeNotification Stackru сообщения, не находя решения.

1 ответ

Решение

MessageComposerView выглядит аккуратно, спасибо, что поделились.

Вместо того, чтобы слушать UIDeviceOrientationDidChangeNotification, вы должны реализовать layoutSubviews, Он вызывается в ответ на ротацию и все, что понадобится системе UIView изменить положение своих подпредставлений ( /questions/26645672/kogda-vyizyivaetsya-layoutsubviews/26645684#26645684). Там вы можете расположить свои подпредставления относительно UIViewЭто кадр, и вы можете быть уверены, что кадр будет правильным.

Другие вопросы по тегам