iOS防截屏-完全解決

幾行代碼解決問題汽摹,有2個方案往下看
說到iOS防截屏大部分人肯定說無法實現(xiàn)李丰,但是我在這樣回復了老板后,老板給我答復是:愛奇藝逼泣,實現(xiàn)了E棵凇!@(付費視頻)

我們也是開發(fā)者嗜憔,別人能實現(xiàn)了我們也能,上效果圖
飛書20211110-135317(1).gif

1.查閱大量資料不難發(fā)現(xiàn)氏仗,愛奇藝之所以防截屏吉捶,是因為有版權的視頻都已轉化成加密流的視頻,這樣你在截圖的時候皆尔,就能達到防截屏的效果
2.接下來我們就是要做的把需要防止用戶截屏的區(qū)域做成加密流的視頻就可以了呐舔。

首先我把吧圖片生成一幀幀的加密流視頻

+ (CVPixelBufferRef) pixelBufferFromCGImage: (CGImageRef) image {
    
    CGSize size = CGSizeMake(CGImageGetWidth(image), CGImageGetHeight(image));
    
    NSDictionary *options = [NSDictionary dictionaryWithObjectsAndKeys:
                             [NSNumber numberWithBool:YES], kCVPixelBufferCGImageCompatibilityKey,
                             [NSNumber numberWithBool:YES], kCVPixelBufferCGBitmapContextCompatibilityKey,
                             nil];
    CVPixelBufferRef pxbuffer = NULL;
    
    CVReturn status = CVPixelBufferCreate(kCFAllocatorDefault,
                                          size.width,
                                          size.height,
                                          kCVPixelFormatType_32ARGB,
                                          (__bridge CFDictionaryRef) options,
                                          &pxbuffer);
    if (status != kCVReturnSuccess){
        NSLog(@"Failed to create pixel buffer");
    }
    
    CVPixelBufferLockBaseAddress(pxbuffer, 0);
    void *pxdata = CVPixelBufferGetBaseAddress(pxbuffer);
    
    
    CGColorSpaceRef rgbColorSpace = CGColorSpaceCreateDeviceRGB();
    CGContextRef context = CGBitmapContextCreate(pxdata, size.width,
                                                 size.height, 8, 4*size.width, rgbColorSpace,
                                                 kCGImageAlphaPremultipliedFirst);
    //kCGImageAlphaNoneSkipFirst);
    CGContextConcatCTM(context, CGAffineTransformMakeRotation(0));
    CGContextDrawImage(context, CGRectMake(0, 0, CGImageGetWidth(image),
                                           CGImageGetHeight(image)), image);
    CGColorSpaceRelease(rgbColorSpace);
    CGContextRelease(context);
    
    CVPixelBufferUnlockBaseAddress(pxbuffer, 0);
    
    return pxbuffer;
}

接著,我將它制作成視頻

+ (void)imageToMP4:(UIImage *)img completion:(void (^)(NSData *))handler {
    NSError *error = nil;
    
    NSFileManager *fileMgr = [NSFileManager defaultManager];
    NSString *documentsDirectory = [NSHomeDirectory()
                                    stringByAppendingPathComponent:@"tmp"];
    NSString *videoOutputPath = [documentsDirectory stringByAppendingPathComponent:@"test_output.mp4"];
    if ([fileMgr removeItemAtPath:videoOutputPath error:&error] != YES)
        NSLog(@"Unable to delete file: %@", [error localizedDescription]);
    
    CGSize imageSize = CGSizeMake(img.size.width, img.size.height);
    NSUInteger fps = 5;
    NSArray *imageArray = @[img];
    
    NSLog(@"videoOutputPath========%@", videoOutputPath);

    NSLog(@"Start building video from defined frames.");
    
    AVAssetWriter *videoWriter = [[AVAssetWriter alloc] initWithURL:
                                  [NSURL fileURLWithPath:videoOutputPath] fileType:AVFileTypeQuickTimeMovie
                                                              error:&error];
    /// !!!需要設置faststart
    videoWriter.shouldOptimizeForNetworkUse = YES;
    NSParameterAssert(videoWriter);
    
    NSDictionary *videoSettings = [NSDictionary dictionaryWithObjectsAndKeys:
                                   AVVideoCodecTypeH264, AVVideoCodecKey,
                                   [NSNumber numberWithInt:imageSize.width], AVVideoWidthKey,
                                   [NSNumber numberWithInt:imageSize.height], AVVideoHeightKey,
                                   nil];
        
    AVAssetWriterInput* videoWriterInput = [AVAssetWriterInput
                                            assetWriterInputWithMediaType:AVMediaTypeVideo
                                            outputSettings:videoSettings];
    
    
    AVAssetWriterInputPixelBufferAdaptor *adaptor = [AVAssetWriterInputPixelBufferAdaptor
                                                     assetWriterInputPixelBufferAdaptorWithAssetWriterInput:videoWriterInput
                                                     sourcePixelBufferAttributes:nil];
    
    NSParameterAssert(videoWriterInput);
    NSParameterAssert([videoWriter canAddInput:videoWriterInput]);
    videoWriterInput.expectsMediaDataInRealTime = YES;
    [videoWriter addInput:videoWriterInput];
    
    //Start a session:
    [videoWriter startWriting];
    [videoWriter startSessionAtSourceTime:kCMTimeZero];
    
    CVPixelBufferRef buffer = NULL;
    
    //convert uiimage to CGImage.
    int frameCount = 0;
    double numberOfSecondsPerFrame = 1;
    double frameDuration = fps * numberOfSecondsPerFrame;
    
    //for(VideoFrame * frm in imageArray)
    NSLog(@"**************************************************");
    for(UIImage * img in imageArray)
    {
        //UIImage * img = frm._imageFrame;
        buffer = [self pixelBufferFromCGImage:[img CGImage]];
        
        BOOL append_ok = NO;
        int j = 0;
        while (!append_ok && j < 30) {
            if (adaptor.assetWriterInput.readyForMoreMediaData)  {
                //print out status:
                NSLog(@"Processing video frame (%d,%lu)",frameCount,(unsigned long)[imageArray count]);
                
                CMTime frameTime = CMTimeMake(frameCount*frameDuration,(int32_t) fps);
                append_ok = [adaptor appendPixelBuffer:buffer withPresentationTime:frameTime];
                if(!append_ok){
                    NSError *error = videoWriter.error;
                    if(error!=nil) {
                        NSLog(@"Unresolved error %@,%@.", error, [error userInfo]);
                    }
                }
            }
            else {
                printf("adaptor not ready %d, %d\n", frameCount, j);
                [NSThread sleepForTimeInterval:0.1];
            }
            j++;
        }
        if (!append_ok) {
            printf("error appending image %d times %d\n, with error.", frameCount, j);
        }
        frameCount++;
    }
    NSLog(@"**************************************************");
    
    //Finish the session:
    [videoWriterInput markAsFinished];
    [videoWriter finishWritingWithCompletionHandler:^{
        if (handler) {
            NSData *data = [NSData dataWithContentsOfFile:videoOutputPath];
            handler(data);
        }
    }];
}

最后完成慷蠕。有些小伙伴可能會發(fā)現(xiàn)圖片變形了珊拼,那就是最大的坑了,繪制圖片需要寬度是16的整數倍于是開碼:

+ (UIImage *)composite_Picture:(UIImage *)image{
    CGSize imageContSize = CGSizeMake(((int)image.size.width/16+1)*16, ((int)image.size.width/16+1)*16/(image.size.width/image.size.height));
    //開啟圖形上下文
    UIGraphicsBeginImageContext(imageContSize);
    
    [image drawInRect:CGRectMake(0, 0, imageContSize.width, imageContSize.height)];
    //獲取圖片
    UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
    //關閉上下文
    UIGraphicsEndImageContext();
    
    NSLog(@"%f%f",newImage.size.width,newImage.size.height);
    
    return newImage;
}

請真機運行流炕,進入Demo先點生成視頻澎现,在截圖。如果對您有幫助每辟,麻煩您點贊點贊剑辫,謝謝!
demo傳送門:https://gitee.com/shyzine/i-os-anti-screenshot.git 不贊不回渠欺,謝謝

方案二:
任何添加到返回View上的控件妹蔽,全部防截屏
swift代碼

static func makeSecView() -> UIView {
        let field = UITextField()
        field.isSecureTextEntry = true
        guard let view = field.subviews.first else {
            return UIView()
        }
        view.subviews.forEach { $0.removeFromSuperview() }
        view.isUserInteractionEnabled = true
        return view
    }

OC代碼

-(UIView *)getBgView{
    UITextField *bgTextField = [[UITextField alloc] init];
    [bgTextField setSecureTextEntry:true];
    
    UIView *bgView = bgTextField.subviews.firstObject;
    [bgView setUserInteractionEnabled:true];
    return bgView;
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市峻堰,隨后出現(xiàn)的幾起案子讹开,更是在濱河造成了極大的恐慌,老刑警劉巖捐名,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件旦万,死亡現(xiàn)場離奇詭異,居然都是意外死亡镶蹋,警方通過查閱死者的電腦和手機成艘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進店門赏半,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人淆两,你說我怎么就攤上這事断箫。” “怎么了秋冰?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵仲义,是天一觀的道長。 經常有香客問我剑勾,道長埃撵,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任虽另,我火速辦了婚禮暂刘,結果婚禮上,老公的妹妹穿的比我還像新娘捂刺。我一直安慰自己谣拣,他們只是感情好,可當我...
    茶點故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布族展。 她就那樣靜靜地躺著森缠,像睡著了一般。 火紅的嫁衣襯著肌膚如雪苛谷。 梳的紋絲不亂的頭發(fā)上辅鲸,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天格郁,我揣著相機與錄音腹殿,去河邊找鬼。 笑死例书,一個胖子當著我的面吹牛锣尉,可吹牛的內容都是我干的。 我是一名探鬼主播决采,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼自沧,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了树瞭?” 一聲冷哼從身側響起拇厢,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎晒喷,沒想到半個月后孝偎,有當地人在樹林里發(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡凉敲,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年衣盾,在試婚紗的時候發(fā)現(xiàn)自己被綠了寺旺。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡势决,死狀恐怖阻塑,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情果复,我是刑警寧澤陈莽,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布,位于F島的核電站虽抄,受9級特大地震影響传透,放射性物質發(fā)生泄漏。R本人自食惡果不足惜极颓,卻給世界環(huán)境...
    茶點故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一朱盐、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧菠隆,春花似錦兵琳、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至破衔,卻和暖如春清女,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背晰筛。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工嫡丙, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人读第。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓曙博,卻偏偏與公主長得像,于是被迫代替她去往敵國和親怜瞒。 傳聞我的和親對象是個殘疾皇子父泳,可洞房花燭夜當晚...
    茶點故事閱讀 42,762評論 2 345

推薦閱讀更多精彩內容