微信可以检测到用户截屏行为(Home + Power),并在稍后点击附加功能按钮时询问用户是否要发送刚才截屏的图片,这个用户体验非常好。
我注意到这个功能很久了,一直很好奇这是如何做到的。但最近实在是太忙了,没有时间整理出来,现在国庆放假了,整理一下我的想法。
我首先想到的是否能够检测到用户按下了 Home 键和 Power 键,因为截图需要两个按钮同时按下,但后面我发现这样做是不行的。因为截屏是按下(TouchDown)而非按下再松开(TouchUp),所以应用的状态不会发生任何改变,相关的代理方法和通知都不会触发,然后我试了第二种方法。
第二种是注册通知,这是最简单的方法,在 iOS 4.0 之前,系统会发送 PictureWasTakenNotification 这个通知来告知开发者发生了截屏行为,但这明显不符合我们现在 Apps 的要求。所以通过通知来检测用户截屏,也是不可行的。
在 iOS 7 之后,注册通知来检测用户截屏已经变为可行的了,在 iOS 7 Beta4 引入了新的 API ,其中就包括了截屏检测的通知 UIApplicationUserDidTakeScreenshotNotification ,这也就意味着在 iOS 7 中,我们只需要注册这个通知就可以了。非常简单。
第三种是应用启动后在后台循环检测相册内最新一张照片,看它的是否符合截屏的特征。这种方法可行,但这是个笨方法,需要用户允许你的程序访问相册才可以,并且一直在后台循环会消耗更多的系统资源。Github 上有一个开源代码做了这个功能。我使用 Instruments 检测在 iPhone 4S 、 iOS 6.1.3 的环境下 CPU 占用为 %2。ShotBlocker
实际上,还有另外一种检测用户截 屏的方法,就是使用 touchesCancelled:withEvent: 这个方法。在 iOS 7 之前,如果用户截屏,系统会自动取消屏幕上的所有 touch 事件,那么我们就可以检测这个方法的调用,然后加载本地最新图片再加以判断来实现我们的目的。但在 iOS 7 之后,截屏不再会取消屏幕的 touch 事件,所以导致了 Snapchat 和 Facebook Poke 之类的应用在 iOS 7 刚发布时依赖于系统这个行为的功能受到影响。
以上简单描述了如何检测用户截屏的行为,希望对大家有所帮助。
首先,可以添加系统自带的这个通知,之后就可以成功检测到用户的截屏事件了,并在方法中进行操作
//添加截屏时的通知,只能在截屏后才会触发
[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(userDidScreenShort:) name:UIApplicationUserDidTakeScreenshotNotification object:nil];
为了可以实现类似微信的效果,必须获取到截屏的图片,因此,我封装了一个类目,用来获取截屏的图片
下面是.h文件中开放的一个API
@interface UIImage (ScreenPic)
+ (UIImage *)imageWithScreenshot;
@end
.m中具体的实现方法
//获取截屏
+ (NSData *)dataWithScreenshotInPNGFormat
{
CGSize imageSize = CGSizeZero;
UIInterfaceOrientation orientation = [UIApplication sharedApplication].statusBarOrientation; //检测横竖屏
if (UIInterfaceOrientationIsPortrait(orientation))
imageSize = [UIScreen mainScreen].bounds.size; //竖屏
else
imageSize = CGSizeMake([UIScreen mainScreen].bounds.size.height, [UIScreen mainScreen].bounds.size.width); //横屏
//开始绘制图片
UIGraphicsBeginImageContextWithOptions(imageSize, NO, 0);
CGContextRef context = UIGraphicsGetCurrentContext();
for (UIWindow *window in [[UIApplication sharedApplication] windows])
{
CGContextSaveGState(context);
CGContextTranslateCTM(context, window.center.x, window.center.y);
CGContextConcatCTM(context, window.transform);
CGContextTranslateCTM(context, -window.bounds.size.width * window.layer.anchorPoint.x, -window.bounds.size.height * window.layer.anchorPoint.y);
//根据偏转修改坐标系,来显示图片
if (orientation == UIInterfaceOrientationLandscapeLeft)
{
CGContextRotateCTM(context, M_PI_2);
CGContextTranslateCTM(context, 0, -imageSize.width);
}
else if (orientation == UIInterfaceOrientationLandscapeRight)
{
CGContextRotateCTM(context, -M_PI_2);
CGContextTranslateCTM(context, -imageSize.height, 0);
} else if (orientation == UIInterfaceOrientationPortraitUpsideDown) {
CGContextRotateCTM(context, M_PI);
CGContextTranslateCTM(context, -imageSize.width, -imageSize.height);
}
if ([window respondsToSelector:@selector(drawViewHierarchyInRect:afterScreenUpdates:)])
{
[window drawViewHierarchyInRect:window.bounds afterScreenUpdates:YES];
}
else
{
[window.layer renderInContext:context];
}
CGContextRestoreGState(context);
}
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
return UIImagePNGRepresentation(image);
}
/**
* 返回截取到的图片
*
* @return UIImage *
*/
+ (UIImage *)imageWithScreenshot
{
NSData *imageData = [self dataWithScreenshotInPNGFormat];
return [UIImage imageWithData:imageData];
}
下面是我Demo的链接https://github.com/ExoticAHOGE/ScreenShortDemo/tree/master