我正在创建一个可重用的框架,用于在iOS应用程序中显示通知。我希望将通知视图添加到应用程序中其他所有内容的顶部,有点像UIAlertView。当我初始化侦听NSNotification事件并在响应中添加视图的管理器时,我需要获得对应用程序中最顶层视图的引用。这就是我现在所拥有的:
_topView = [[[[UIApplication sharedApplication] keyWindow] subviews] lastObject];
这适用于任何iOS应用程序,还是更安全/更好的方式来获得顶视图?
通常这会给你顶视图,但不保证它对用户可见。它可以在屏幕外,具有0.0的alpha值,或者可以具有0x0的大小。
它也可能是keyWindow没有子视图,所以你应该首先测试它。这不寻常,但并非不可能。
UIWindow是UIView的子类,因此如果您想确保用户可以看到您的通知,您可以使用addSubview:
将其直接添加到keyWindow,它将立即成为最顶层的视图。我不确定这是不是你想要做的。 (根据你的问题,看起来你已经知道了。)
每当我想在其他所有内容上显示一些叠加时,我只需将它直接添加到应用程序窗口的顶部:
[[[UIApplication sharedApplication] keyWindow] addSubview:someView]
问题分为两部分:顶部窗口,顶部窗口顶视图。
所有现有答案都错过了顶部窗口部分。但[[UIApplication sharedApplication] keyWindow]
并不能保证成为最佳窗口。
windowLevel
共存应用程序,所以我们可以通过windowLevel
对所有窗口进行排序并获得最顶层的窗口。
UIWindow *topWindow = [[[UIApplication sharedApplication].windows sortedArrayUsingComparator:^NSComparisonResult(UIWindow *win1, UIWindow *win2) {
return win1.windowLevel - win2.windowLevel;
}] lastObject];
UIView *topView = [[topWindow subviews] lastObject];
实际上,您的应用程序中可能有多个UIWindow。例如,如果键盘在屏幕上,那么[[UIApplication sharedApplication] windows]
将包含至少两个窗口(您的键窗口和键盘窗口)。
因此,如果您希望您的视图显示在它们之上,那么您必须执行以下操作:
[[[[UIApplication sharedApplication] windows] lastObject] addSubview:view];
(假设lastObject包含windowLevel优先级最高的窗口)。
我坚持这个问题是标题,而不是讨论。在任何给定点上哪个视图最顶部?
@implementation UIView (Extra)
- (UIView *)findTopMostViewForPoint:(CGPoint)point
{
for(int i = self.subviews.count - 1; i >= 0; i--)
{
UIView *subview = [self.subviews objectAtIndex:i];
if(!subview.hidden && CGRectContainsPoint(subview.frame, point))
{
CGPoint pointConverted = [self convertPoint:point toView:subview];
return [subview findTopMostViewForPoint:pointConverted];
}
}
return self;
}
- (UIWindow *)topmostWindow
{
UIWindow *topWindow = [[[UIApplication sharedApplication].windows sortedArrayUsingComparator:^NSComparisonResult(UIWindow *win1, UIWindow *win2) {
return win1.windowLevel - win2.windowLevel;
}] lastObject];
return topWindow;
}
@end
可以直接与任何UIWindow一起用作接收器或任何UIView作为接收器。
如果要添加加载视图(例如活动指示器视图),请确保您具有UIWindow类的对象。如果您在显示加载视图之前显示操作表,则keyWindow
将是UIActionSheet
而不是UIWindow
。由于操作表将消失,加载视图将随之消失。或者这就是造成我问题的原因。
UIWindow *keyWindow = [[UIApplication sharedApplication] keyWindow];
if (![NSStringFromClass([keyWindow class]) isEqualToString:@"UIWindow"]) {
// find uiwindow in windows
NSArray *windows = [UIApplication sharedApplication].windows;
for (UIWindow *window in windows) {
if ([NSStringFromClass([window class]) isEqualToString:@"UIWindow"]) {
keyWindow = window;
break;
}
}
}
如果您的应用程序仅适用于纵向,这就足够了:
[[[UIApplication sharedApplication] keyWindow] addSubview:yourView]
并且您的视图不会显示在键盘和状态栏上。
如果您想通过键盘或状态栏获得最顶层视图,或者您希望最顶层视图可以使用设备正确旋转,请尝试以下框架:
https://github.com/HarrisonXi/TopmostView
它支持iOS7 / 8/9。
如果您想在屏幕上的所有内容上添加视图,请使用此代码。
[[UIApplication sharedApplication].keyWindow addSubView: yourView];
试试这个
UIWindow *window = [[[UIApplication sharedApplication] windows] lastObject];