Swift/iOS - 截屏,長(zhǎng)屏,拼圖,保存圖片到指定的相冊(cè)

注意

在iOS中訪問(wèn)相冊(cè),首先我們得獲取用戶(hù)授權(quán),在iOS10之前不用再I(mǎi)nfo.plist寫(xiě)參數(shù)。
在iOS10 之后要自定義彈出的信息

Privacy - Photo Library Usage Description
Privacy - Camera Usage Description

獲取權(quán)限

    /// 獲取權(quán)限
    ///
    /// - Parameter accessBlock:  閉包回調(diào)
    func accessCameraAuthority(accessBlock:AccessBlock!)
    {
        let authStatus = AVCaptureDevice.authorizationStatus(forMediaType: AVMediaTypeVideo)
        if authStatus == AVAuthorizationStatus.notDetermined {
            // 請(qǐng)求授權(quán)
            AVCaptureDevice.requestAccess(forMediaType: AVMediaTypeAudio, completionHandler: { finish in
                if finish {
                    if (accessBlock) != nil
                    {
                        accessBlock(true)
                    }
                }
                else
                {
                    if (accessBlock) != nil
                    {
                        accessBlock(false)
                    }
                }
            })
        }
        else if authStatus == AVAuthorizationStatus.authorized
        {
            // 授權(quán)成功
            if (accessBlock) != nil
            {
                accessBlock(true)
            }
        }
        else
        {
            // 用戶(hù)拒絕授權(quán)
            if (accessBlock) != nil
            {
                accessBlock(false)
            }
        }
    }

當(dāng)返回true,我們就可以有訪問(wèn)權(quán)限了。
在iOS中截屏的方法如下:

    /// 截屏
    ///
    /// - Parameters:
    ///   - view: 要截屏的view
    /// - Returns: 一個(gè)UIImage
    func cutImageWithView(view:UIView) -> UIImage
    {
        // 參數(shù)①:截屏區(qū)域  參數(shù)②:是否透明  參數(shù)③:清晰度
        UIGraphicsBeginImageContextWithOptions(view.frame.size, true, UIScreen.main.scale)
        view.layer.render(in: UIGraphicsGetCurrentContext()!)
        let image:UIImage = UIGraphicsGetImageFromCurrentImageContext()!
        UIGraphicsEndImageContext();
        return image;
    }

這個(gè)方法是獲取當(dāng)前view的內(nèi)容,size的大小要巧妙的設(shè)置,比如你的view超出了看到的屏幕,你使用截圖下來(lái),
就是一部分回事黑色的區(qū)域。你截取當(dāng)前能看到的,就widow就可以了。
我們?cè)谑褂煤芏郃PP的時(shí)候會(huì)截出很長(zhǎng)的圖,這個(gè)時(shí)候我們就用巧取得方式來(lái)獲取,方法如下:

     /// 截屏
    ///
    /// - Parameters:
    ///   - view: 要截屏的view
    /// - Returns: 一個(gè)UIImage
    func cutFullImageWithView(scrollView:UIScrollView) -> UIImage
    {
        // 記錄當(dāng)前的scrollView的偏移量和坐標(biāo)
        let currentContentOffSet:CGPoint = scrollView.contentOffset
        let currentFrame:CGRect = scrollView.frame;
        
        // 設(shè)置為zero和相應(yīng)的坐標(biāo)
        scrollView.contentOffset = CGPoint.zero
        scrollView.frame = CGRect.init(x: 0, y: 0, width: scrollView.frame.size.width, height: scrollView.frame.size.height)
        
        // 參數(shù)①:截屏區(qū)域  參數(shù)②:是否透明  參數(shù)③:清晰度
        UIGraphicsBeginImageContextWithOptions(scrollView.contentSize, true, UIScreen.main.scale)
        scrollView.layer.render(in: UIGraphicsGetCurrentContext()!)
        let image:UIImage = UIGraphicsGetImageFromCurrentImageContext()!
        
        // 重新設(shè)置原來(lái)的參數(shù)
        scrollView.contentOffset = currentContentOffSet
        scrollView.frame = currentFrame
        
        UIGraphicsEndImageContext();
        
        return image;
    }

這里是保證他是scrllView,來(lái)切換相應(yīng)的offset和frame,來(lái)截取屏幕。有一點(diǎn)要注意,比如你的是tableView,下面的數(shù)據(jù)中的圖片沒(méi)有加載出來(lái),也是肯定不會(huì)顯示的。
接下來(lái)就是把截圖保存到相冊(cè),下面是保存到系統(tǒng)相冊(cè):

    func writeImageToAlbum(image:UIImage)
    {
        UIImageWriteToSavedPhotosAlbum(image, self, #selector(image(image:didFinishSavingWithError:contextInfo:)), nil)
    }
    
    func image(image: UIImage, didFinishSavingWithError error: NSError?, contextInfo:UnsafeRawPointer)
    {
        if let e = error as NSError?
        {
            print(e)
        }
        else
        {
            UIAlertController.init(title: nil,
                                   message: "保存成功!",
                                   preferredStyle: UIAlertControllerStyle.alert).show(self, sender: nil);
        }
    }

寫(xiě)入到指定的相冊(cè),這里看到的是先寫(xiě)入相冊(cè),再移動(dòng)到指定的相冊(cè)

       // 看有沒(méi)有指定的相冊(cè)
        let library = ALAssetsLibrary.init()
        let albumGroups = NSMutableArray()
        
        library.enumerateGroups(withTypes: ALAssetsGroupType(ALAssetsGroupAll), using:
            {
                oneGroup,pointer in
                albumGroups.add(oneGroup! as ALAssetsGroup)
        },
                                failureBlock:
            {
                error in
        })
        
        var haveCustomAlbum = false
        for albumGroup in albumGroups
        {
            let groupName:String = (albumGroup as! ALAssetsGroup).value(forProperty: ALAssetsGroupPropertyName) as! String
            print(groupName)
            if groupName == "指定相冊(cè)"
            {
                haveCustomAlbum = true
            }
        }

如果有指定相冊(cè)直接寫(xiě)入沒(méi)有就去創(chuàng)建

// 這里是創(chuàng)建相冊(cè)
if !haveCustomAlbum
        {
            library.addAssetsGroupAlbum(withName: "指定相冊(cè)",
                                        resultBlock:
                {alAsetGroup in
                    if alAsetGroup != nil
                    {
                        print("創(chuàng)建相冊(cè)成功");
                    }
            },
                                        failureBlock:
                {error in
                    print("創(chuàng)建相冊(cè)失敗");
            })
        }

下面是寫(xiě)入到指定的相冊(cè),下面的image,group和library都是上面獲取到的

func writeImageToCustomAlbumFinish(image:UIImage,library:ALAssetsLibrary,alAsetGroup:ALAssetsGroup)
    {
        // 首先寫(xiě)入到寫(xiě)到相冊(cè) 這里也可以用PHPhotoLibrary來(lái)做
        library.writeImage(toSavedPhotosAlbum: image.cgImage, orientation:.up, completionBlock:
            { url,error in
                if (error != nil)
                {
                    // 取到asset
                    library.asset(for: url, resultBlock:
                        {
                            aAlAset in
                            if aAlAset != nil
                            {
                                // 添加到group
                                alAsetGroup.add(aAlAset)
                            }
                    },
                                  failureBlock:
                        {
                            error in
                    })
                }
        })
    }

拼圖

func combinTwoImage(image1:UIImage,image2:UIImage) -> UIImage
    {
        let width = max(image1.size.width, image2.size.width)
        let height = image1.size.height + image2.size.height
        let offScreenSize = CGSize.init(width: width, height: height)
        
        UIGraphicsBeginImageContext(offScreenSize);
        
        let rect = CGRect.init(x:0, y:0, width:width, height:image1.size.height)
        image1.draw(in: rect)
        
        let rect2 = CGRect.init(x:0, y:image1.size.height, width:width, height:image2.size.height)
        image1.draw(in: rect2)
        
        let image:UIImage = UIGraphicsGetImageFromCurrentImageContext()!
        
        UIGraphicsEndImageContext();
        
        return image;
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,351評(píng)論 25 708
  • 因?yàn)橐Y(jié)局swift3.0中引用snapKit的問(wèn)題,看到一篇介紹Xcode8,swift3變化的文章,覺(jué)得很詳細(xì)...
    uniapp閱讀 4,877評(píng)論 0 12
  • ¥開(kāi)啟¥ 【iAPP實(shí)現(xiàn)進(jìn)入界面執(zhí)行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開(kāi)一個(gè)線程,因...
    小菜c閱讀 7,389評(píng)論 0 17
  • 一次偶然的相遇,彼此的聯(lián)絡(luò),心動(dòng)的感覺(jué),終于在那年的雙十一之后,牽上了她的手,就這樣一同大學(xué)畢業(yè),一同上研究生,一...
    角落的沉思者閱讀 222評(píng)論 4 1
  • Carmen康康閱讀 252評(píng)論 0 0

友情鏈接更多精彩內(nèi)容