很多项目都会有这样一个功能,就是点击状态栏后让ScrollView或者子类能够自动滚到顶部。
现在我们就来实现这个功能。
实现思路
我们可以在状态栏上面加一个UIWindow,点击UIWindow时判断是不是ScrollView或者子类,如果是的就调用- (void)setContentOffset:(CGPoint)contentOffset animated:(BOOL)animated方法让其滚到顶部。
首先
我们先定制一个类来实现这个功能。我们可以继承自NSObject
向外界暴露一个接口用来显示我们的UIWindow。代码如下
//为了让window不被提前释放
static UIWindow *window_;
//每个类调用一次initialize。这个调用的时间发生在你的类接收到消息之前,但是在它的超类接收到initialize之后。
//1。实现某些单独运行的工作。2.希望实现仅仅运行一次的方法
+ (void)initialize
{
//创建window
window_ = [[UIWindow alloc] init];
window_.frame = CGRectMake(0, 0, [UIScreen mainScreen].bounds.size.width, 20);
//设置window的优先级为最高显示
window_.windowLevel = UIWindowLevelAlert;
//添加手势
[window_ addGestureRecognizer:[[UITapGestureRecognizer alloc]
initWithTarget:self action:@selector(windowClick)]];
}
//实现类方法
+ (void)show
{
//默认是yes隐藏
window_.hidden = NO;
}
//窗口的点击
//前面是类方法,这里必须使用类方法
+ (void)windowClick
{
UIWindow *window = [UIApplication sharedApplication].keyWindow;
//判断是不是ScrollView或者子类
[self searchScrollViewInView:window];
}
判断是不是ScrollView或者子类
+ (void)searchScrollViewInView:(UIView *)superview
{
for (UIScrollView *subview in superview.subviews) {
// 如果是scrollview, 滚动最顶部
// isShowingOnKeyWindow是对UIView添加的类别方法
if ([subview isKindOfClass:[UIScrollView class]] && subview.isShowingOnKeyWindow) {
CGPoint offset = subview.contentOffset;
// 之前设置过内边距所以要到负的y值才会到顶部(没有设置可以忽略)
offset.y = - subview.contentInset.top;
[subview setContentOffset:offset animated:YES];
}
// 继续查找子控件(递归的思想)
[self searchScrollViewInView:subview];
}
}
UIView的类别方法
- (BOOL)isShowingOnKeyWindow
{
// 主窗口
UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow;
// 以主窗口左上角为坐标原点, 计算self的矩形框
CGRect newFrame = [keyWindow convertRect:self.frame fromView:self.superview];
CGRect winBounds = keyWindow.bounds;
// 主窗口的bounds 和 self的矩形框 是否有重叠
BOOL intersects = CGRectIntersectsRect(newFrame, winBounds);
return !self.isHidden && self.alpha > 0.01 && self.window == keyWindow && intersects;
}
//在appdelegate中调用show方法
- (void)applicationDidBecomeActive:(UIApplication *)application {
// 添加一个window, 点击这个window, 可以让屏幕上的scrollView滚到最顶部
[AZTopWindow show];
}
//顺便说一下状态栏的控制。
//在iOS7之前是UIApplication控制
[UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleDefault;
在iOS7之后交给ViewController管理了
- (UIStatusBarStyle)preferredStatusBarStyle{
return UIStatusBarStyleLightContent;
}
如果iOS7之后还需要通过UIApplication来管理,需要在plist文件中将 controller-based status bar appearance键设为NO。
网友评论