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
= [[объект уведомления] ориентация] = UIInterfaceOrientationLandscapeLeftUIInterfaceOrientation
= [UIApplication sharedApplication].statusBarOrientation = UIDeviceOrientationLandscapeRightself.frame.size
= (ширина =480, высота =90)
Здесь важно отметить, что как интерфейс, так и ориентация устройства являются альбомными, а ширина уже изменена на альбомную (тестирование на симуляторе iPhone 6.1). Сейчас выполняю тот же тест, но используя initWithFrame
:
UIDeviceOrientation
= [[объект уведомления] ориентация] = UIDeviceOrientationLandscapeRightUIInterfaceOrientation
= [UIApplication sharedApplication].statusBarOrientation = UIInterfaceOrientationPortraitself.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
Это кадр, и вы можете быть уверены, что кадр будет правильным.