iOS 截圖的那些事兒

同時(shí)按下 Home 鍵和電源鍵向楼,咔嚓一聲田炭,就得到了一張手機(jī)的截圖师抄,這操作想必 iPhone 用戶再熟悉不過(guò)了。我們作為研發(fā)人員教硫,面對(duì)的是一個(gè)個(gè)的 View叨吮,那么該怎么用代碼對(duì) View 進(jìn)行截圖呢?
這篇文章主要討論的是如何在包括 UIWebView 和 WKWebView 的網(wǎng)頁(yè)中進(jìn)行長(zhǎng)截圖瞬矩,對(duì)應(yīng)的示例代碼在這兒:https://github.com/VernonVan/PPSnapshotKit茶鉴。

UIWebView 截圖

對(duì) UIWebView 截圖比較簡(jiǎn)單,renderInContext 這個(gè)方法相信大家都不會(huì)陌生景用,這個(gè)方法是 CALayer 的一個(gè)實(shí)例方法涵叮,可以用來(lái)對(duì)大部分 View 進(jìn)行截圖。我們知道伞插,UIWebView 承載內(nèi)容的其實(shí)是作為其子 View 的 UIScrollView割粮,所以對(duì) UIWebView 截圖應(yīng)該對(duì)其 scrollView 進(jìn)行截圖。具體的截圖方法如下:

- (void)snapshotForScrollView:(UIScrollView *)scrollView
{
    // 1. 記錄當(dāng)前 scrollView 的偏移和位置
    CGPoint currentOffset = scrollView.contentOffset;
    CGRect currentFrame = scrollView.frame;

    scrollView.contentOffset = CGPointZero;
    // 2. 將 scrollView 展開為其實(shí)際內(nèi)容的大小
    scrollView.frame = CGRectMake(0, 0, scrollView.contentSize.width, scrollView.contentSize.height);

    // 3. 第三個(gè)參數(shù)設(shè)置為 0 表示設(shè)置為屏幕的默認(rèn)縮放因子
    UIGraphicsBeginImageContextWithOptions(scrollView.contentSize, YES, 0);
    [scrollView.layer renderInContext:UIGraphicsGetCurrentContext()];
    UIImage *snapshotImage = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    
    // 4. 重新設(shè)置 scrollView 的偏移和位置媚污,還原現(xiàn)場(chǎng)
    scrollView.contentOffset = currentOffset;
    scrollView.frame = currentFrame;
}

WKWebView 截圖

雖然 WKWebView 里也有 scrollView舀瓢,但是直接對(duì)這個(gè) scrollView 截圖得到的是一片空白的,具體原因不明耗美。一番 Google 之后可以看到好些人提到 drawViewHierarchyInRect 方法京髓, 可以看到這個(gè)方法是 iOS 7.0 開始引入的。官方文檔中描述為:

Renders a snapshot of the complete view hierarchy as visible onscreen into the current context.

注意其中的 visible onscreen商架,也就是將屏幕中可見部分渲染到上下文中堰怨,這也解釋了為什么對(duì) WKWebView 中的 scrollView 展開為實(shí)際內(nèi)容大小,再調(diào)用 drawViewHierarchyInRect 方法總是得到一張不完整的截圖(只有屏幕可見區(qū)域被正確截到甸私,其他區(qū)域?yàn)榭瞻祝?br> 不過(guò)诚些,這樣倒是給我們提供了一個(gè)思路,可以將 WKWebView 按屏幕高度裁成 n 頁(yè)皇型,然后將 WKWebView 一頁(yè)一頁(yè)的往上推诬烹,每推一頁(yè)就調(diào)用一次 drawViewHierarchyInRect 將當(dāng)前屏幕的截圖渲染到上下文中,最后調(diào)用 UIGraphicsGetImageFromCurrentImageContext 從上下文中獲取的圖片即為完整截圖弃鸦。

核心代碼如下(代碼為演示用途绞吁,完整代碼請(qǐng)從這里查看):

- (void)snapshotForWKWebView:(WKWebView *)webView
{
    // 1
    UIView *snapshotView = [webView snapshotViewAfterScreenUpdates:YES];
    [webView.superview addSubview:snapshotView];

    // 2
    CGPoint currentOffset = webView.scrollView.contentOffset;
    ...

    // 3
    UIView *containerView = [[UIView alloc] initWithFrame:webView.bounds];
    [webView removeFromSuperview];
    [containerView addSubview:webView];

    
    // 4
    CGSize totalSize = webView.scrollView.contentSize;
    NSInteger page = ceil(totalSize.height / containerView.bounds.size.height);

    webView.scrollView.contentOffset = CGPointZero;
    webView.frame = CGRectMake(0, 0, containerView.bounds.size.width, webView.scrollView.contentSize.height);

    UIGraphicsBeginImageContextWithOptions(totalSize, YES, UIScreen.mainScreen.scale);
    [self drawContentPage:0 maxIndex:page completion:^{
        UIImage *snapshotImage = UIGraphicsGetImageFromCurrentImageContext();
        UIGraphicsEndImageContext();

        // 8
        [webView removeFromSuperview];
        ...
    }];
}

- (void)drawContentPage(NSInteger)index maxIndex:(NSInteger)maxIndex completion:(dispatch_block_t)completion
{
    // 5
    CGRect splitFrame = CGRectMake(0, index * CGRectGetHeight(containerView.bounds), containerView.bounds.size.width, containerView.frame.size.height);
    CGRect myFrame = webView.frame;
    myFrame.origin.y = -(index * containerView.frame.size.height);
    webView.frame = myFrame;

    // 6
    [targetView drawViewHierarchyInRect:splitFrame afterScreenUpdates:YES];

    // 7
    if (index < maxIndex) {
        [self drawContentPage:index + 1 maxIndex:maxIndex completion:completion];
    } else {
        completion();
    }
}

代碼注意項(xiàng)如下(對(duì)應(yīng)代碼注釋中的序號(hào)):

  1. 為了截圖時(shí)對(duì) frame 進(jìn)行操作不會(huì)出現(xiàn)閃屏等現(xiàn)象,我們需要蓋一個(gè)“假”的 webView 到現(xiàn)在的位置上唬格,并將真正的 webView “摘下來(lái)”家破。調(diào)用 snapshotViewAfterScreenUpdates 即可得到這樣一個(gè)“假”的 webView
  2. 保存真正的 webView 的偏移购岗、位置等信息汰聋,以便截圖完成之后“還原現(xiàn)場(chǎng)”
  3. 用一個(gè)新的視圖承載“真正的” webView喊积,這個(gè)視圖也是繪圖所用到的上下文
  4. 將 webView 按照實(shí)際內(nèi)容高度和屏幕高度分成 page 頁(yè)
  5. 得到每一頁(yè)的實(shí)際位置,并將 webView 往上推到該位置
  6. 調(diào)用 drawViewHierarchyInRect 將當(dāng)前位置的 webView 渲染到上下文中
  7. 如果還未到達(dá)最后一頁(yè)乾吻,則遞歸調(diào)用 drawViewHierarchyInRect 方法進(jìn)行渲染;如果已經(jīng)渲染完了全部頁(yè)绎签,則回調(diào)通知截圖完成
  8. 調(diào)用 UIGraphicsGetImageFromCurrentImageContext 方法從當(dāng)前上下文中獲取到完整截圖枯饿,將第 2 步中保存的信息重新賦予到 webView 上,“還原現(xiàn)場(chǎng)”

注意:我們的截圖方法中有對(duì) webView 的 frame 進(jìn)行操作诡必,如果其他地方如果有對(duì) frame 進(jìn)行操作的話奢方,是會(huì)影響我們截圖的。所以在截圖時(shí)應(yīng)該禁用掉其他地方對(duì) frame 的改變爸舒,就像這樣:

- (void)layoutWebView
{
    if (!_isCapturing) {
        self.wkWebView.frame = [self frameForWebView];
    }
}

結(jié)語(yǔ)

當(dāng)前 WKWebView 的使用越來(lái)越廣泛了蟋字,我隨意查看了內(nèi)存占用:打開同樣一個(gè)網(wǎng)頁(yè),UIWebView 直接占用了 160 MB 內(nèi)存碳抄,而 WKWebView 只占用了 40 MB 內(nèi)存愉老,差距是相當(dāng)明顯的。如果我們的業(yè)務(wù)中用到了 WKWebView 且有截圖需求的話剖效,那么還是得老老實(shí)實(shí)完成的嫉入。
最后,本文對(duì)應(yīng)的代碼在這兒:https://github.com/VernonVan/PPSnapshotKit

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末璧尸,一起剝皮案震驚了整個(gè)濱河市咒林,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌爷光,老刑警劉巖垫竞,帶你破解...
    沈念sama閱讀 221,273評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡欢瞪,警方通過(guò)查閱死者的電腦和手機(jī)活烙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,349評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)遣鼓,“玉大人啸盏,你說(shuō)我怎么就攤上這事∑锼睿” “怎么了回懦?”我有些...
    開封第一講書人閱讀 167,709評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)次企。 經(jīng)常有香客問(wèn)我怯晕,道長(zhǎng),這世上最難降的妖魔是什么缸棵? 我笑而不...
    開封第一講書人閱讀 59,520評(píng)論 1 296
  • 正文 為了忘掉前任舟茶,我火速辦了婚禮,結(jié)果婚禮上蛉谜,老公的妹妹穿的比我還像新娘稚晚。我一直安慰自己,他們只是感情好型诚,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,515評(píng)論 6 397
  • 文/花漫 我一把揭開白布客燕。 她就那樣靜靜地躺著,像睡著了一般狰贯。 火紅的嫁衣襯著肌膚如雪也搓。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,158評(píng)論 1 308
  • 那天涵紊,我揣著相機(jī)與錄音,去河邊找鬼摸柄。 笑死,一個(gè)胖子當(dāng)著我的面吹牛驱负,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播跃脊,決...
    沈念sama閱讀 40,755評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼酪术!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起橡疼,我...
    開封第一講書人閱讀 39,660評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎任斋,沒想到半個(gè)月后继阻,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,203評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡抹缕,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,287評(píng)論 3 340
  • 正文 我和宋清朗相戀三年卓研,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片奏赘。...
    茶點(diǎn)故事閱讀 40,427評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡太惠,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出凿渊,到底是詐尸還是另有隱情,我是刑警寧澤搪锣,帶...
    沈念sama閱讀 36,122評(píng)論 5 349
  • 正文 年R本政府宣布彩掐,位于F島的核電站,受9級(jí)特大地震影響堵幽,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜谐檀,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,801評(píng)論 3 333
  • 文/蒙蒙 一桐猬、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦音五、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,272評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至诗充,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間蝴蜓,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,393評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工格仲, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人凯肋。 一個(gè)月前我還...
    沈念sama閱讀 48,808評(píng)論 3 376
  • 正文 我出身青樓造烁,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親惭蟋。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,440評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容