項目中使用View截圖分享,RecyclerView截圖分享

項目中用到了許多截圖分享到第三方的地方,遇到了很多坑,類似:截圖失敗、獲取不到緩存圖片、截取RecyclerView長圖中ImageView同步加載、長圖獲取緩存出現OOM等,本次就遇到的所有問題進行匯總,并提交Demo到github。

簡單的截圖

從網上搜索到的結果,大部分都是 setDrawingCacheEnable(true) ; 然后從緩存中獲取到數據 view.getDrawingCache(), 代碼如下:

Bitmap cacheBitmap = null;
cacheView.setEnabled(false);
// 首先獲取一個View,
View view = getUiView().mActivity.getWindow().getDecorView();
// 設置可以從緩存中獲取到數據
view.setDrawingCacheEnabled(true);
// 獲取到bitmap
cacheBitmap = view.getDrawingCache();
view.setDrawingCacheEnabled(false);
但是官網中對該使用方法標識已經廢棄了:

隨著API 11中硬件加速渲染的引入,視圖繪制緩存基本上已過時。通過硬件加速,中間緩存層在很大程度上是不必要的,并且很容易導致性能的凈損失由于創建和更新圖層的成本;所以官網建議我們使用PixelCopy,不過PixelCopy需要傳入SurfaceView或者Window,我們可以看看setDrawingCacheEnabled(true) 之后獲取getDrawingCache()的源碼:

/**
 * private, internal implementation of buildDrawingCache, used to enable tracing
 */
private void buildDrawingCacheImpl(boolean autoScale) {
   
//    ...... 省略部分代碼
  // 以下判斷當前創建的Bitmap大小是否比系統最大值還大,如果超過則拋出異常
    final int drawingCacheBackgroundColor = mDrawingCacheBackgroundColor;
    final boolean opaque = drawingCacheBackgroundColor != 0 || isOpaque();
    final boolean use32BitCache = attachInfo != null && attachInfo.mUse32BitDrawingCache;
    final long projectedBitmapSize = width * height * (opaque && !use32BitCache ? 2 : 4);
    final long drawingCacheSize =
            ViewConfiguration.get(mContext).getScaledMaximumDrawingCacheSize();
    if (width <= 0 || height <= 0 || projectedBitmapSize > drawingCacheSize) {
        if (width > 0 && height > 0) {
            Log.w(VIEW_LOG_TAG, getClass().getSimpleName() + " not displayed because it is"
                    + " too large to fit into a software layer (or drawing cache), needs "
                    + projectedBitmapSize + " bytes, only "
                    + drawingCacheSize + " available");
        }
        destroyDrawingCache();
        mCachingFailed = true;
        return;
    }
//    ...... 省略部分代碼
       // 清除內存數據
       if (bitmap != null) bitmap.recycle();
       try {
            // 創建bitmap
           bitmap = Bitmap.createBitmap(mResources.getDisplayMetrics(),
                   width, height, quality);
           bitmap.setDensity(getResources().getDisplayMetrics().densityDpi);
           
        } catch(OutOfMemoryError e) {
            //    ...... 省略部分代碼
        }    
   }

// 創建Canvas
   Canvas canvas;
   if (attachInfo != null) {
       canvas = attachInfo.mCanvas;
       if (canvas == null) {
           canvas = new Canvas();
       }
      // 將bitmap設置到畫布中
       canvas.setBitmap(bitmap);
       // Temporarily clobber the cached Canvas in case one of our children
       // is also using a drawing cache. Without this, the children would
       // steal the canvas by attaching their own bitmap to it and bad, bad
       // thing would happen (invisible views, corrupted drawings, etc.)
       attachInfo.mCanvas = null;
   } else {
       // This case should hopefully never or seldom happen
       canvas = new Canvas(bitmap);
   }
//    ...... 省略部分代碼

      // 畫數據
       draw(canvas);
}

通過以上代碼,所以在stackflow中有人總結出:直接通過View創建一個Bitmap,然后設置給Canvas,再通過View的draw方法傳入Canvas:

public void setBitmap(View view) {    
    Bitmap bitmap = Bitmap.createBitmap(view.getWidth(), view.getHeight(), Bitmap.Config.RGB_565);
    Canvas canvas = new Canvas(bitmap);
    view.draw(canvas);
}

當然,上面的兩種方法,都是基于View已經被測量出來,顯示在界面上,只需要拿到數據即可,大多數情況下,我們需要使用LayoutInflater來獲取一個View,這時候我們是獲取不到View的寬高的,所以需要我們手動去measure和layout;

從layout中加載的View截圖

從View的加載過程我們得知,View要繪制到界面上,需要經歷onMeasure()、onLayout()、onDraw(),因為從layout中加載的View,沒有執行這幾個過程,所以無法得到View的緩存圖,必須得要我們手動去執行這幾個方法:

public void getViewFromLayout() {
    // 獲取到View
    View view = LayoutInflater.from(getUiView().mActivity).inflate(R.layout.view_layout, null, false);
    // 調用measure() 方法,測量View的寬高
    view.measure(View.MeasureSpec.makeMeasureSpec(DensityUtil.getWidth(), View.MeasureSpec.EXACTLY),
                    View.MeasureSpec.makeMeasureSpec(DensityUtil.getHeight(), View.MeasureSpec.EXACTLY));
    //   調用layout() 方法,確定View及其子View的位置
    view.layout(0, 0, view.getMeasuredWidth(), view.getMeasuredHeight());
    Bitmap bitmap = Bitmap.createBitmap(view.getWidth(), view.getHeight(), Bitmap.Config.RGB_565);
    Canvas canvas = new Canvas(bitmap);
    // 繪制出View
    view.draw(canvas);
}

上面的代碼比第一種平常的截圖就多了兩個方法,measure和layout,measure方法需要傳入View的寬高,一般設置一個確定的值即可。

RecyclerView截圖

網上有很多例子和教程來教大家怎么截取RecyclerView的圖片的,但是現在基本上RecyclerView都是有圖片的,所以涉及到異步去加載圖片;然而,當使用Stack Overflow中代碼時,截取到的獲取還只是你設置到圖片上的加載中的圖片,我找了許久,才知道應該用同步的加載圖片方法,當所有的圖片加載完之后獲取到截圖;

當然,這些同步加載圖片的方法,因為是耗時線程,需要將代碼放到異步線程中,然后處理完之后切換到同步線程中,我這里使用了Rxjava2中的defer方法:

// 分享  截圖
Observable
        .defer(new Callable<ObservableSource<Bitmap>>() {
            @Override
            public ObservableSource<Bitmap> call() throws Exception {
                return new ObservableSource<Bitmap>() {
                    @Override
                    public void subscribe(Observer<? super Bitmap> observer) {
                        // MVP模式,從p層獲取到RecyclerView的截圖Bitmap數據
                        observer.onNext(mPresenter.getRecyclerViewScreenSpot(mRecyclerView));
                    }
                };
            }
        })
        .compose(applyAsySchedulers())
        .as(bindLifecycle())
        .subscribe(new Consumer<Bitmap>() {
            @Override
            public void accept(Bitmap bitmap) throws Exception {
                // 拿到結果之后,處理成自己想要的圖片,
                mPresenter.getShareSpecimenFile(shareSpecimenBean, bitmap);
            }
        });

defer方法傳入了一個Callable,當前執行的代碼在異步線程中,compse方法注冊在異步中,訂閱在主線程中,所以就起到了切換線程的作用;拿到Bitmap之后,因為我這里需要將Bitmap嵌套到另外一個View,分享另外一個從LayoutInflater中加載的View,所以得在主線程中執行:

getRecyclerViewScrrentSpot(mRecyclerView):
@Override
public Bitmap getRecyclerViewScreenSpot(RecyclerView recyclerView) {
    //獲取設置的adapter
    LoadMoreWrapper adapter = (LoadMoreWrapper) recyclerView.getAdapter();
    //創建保存截圖的bitmap
    Bitmap bigBitmap = null;
    if (adapter != null) {
        //獲取item的數量
        int size = adapter.getItemCount();
        //recycler的完整高度 用于創建bitmap時使用
        int height = 0;
        //獲取最大可用內存
        final int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
        // 使用1/8的緩存
        final int cacheSize = maxMemory / 8;
        //把每個item的繪圖緩存存儲在LruCache中
        LruCache<String, Bitmap> bitmapCache = new LruCache<>(cacheSize);
        for (int i = 0; i < size; i++) {
            //手動調用創建和綁定ViewHolder方法,
            RecyclerView.ViewHolder holder = 
                          adapter.createViewHolder(recyclerView,  adapter.getItemViewType(i));
            adapter.onBindViewImageSync(holder, i);
            //測量
            holder.itemView.measure(
                    View.MeasureSpec.makeMeasureSpec(recyclerView.getWidth() / 3, 
                                  View.MeasureSpec.EXACTLY), 
                    View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));
            //布局
            holder.itemView.layout(0, 0, 
                                      holder.itemView.getMeasuredWidth(),
                                      holder.itemView.getMeasuredHeight());
            //開啟繪圖緩存
            holder.itemView.setDrawingCacheEnabled(true);
            holder.itemView.buildDrawingCache();
            Bitmap drawingCache = holder.itemView.getDrawingCache();
            if (drawingCache != null) {
                bitmapCache.put(String.valueOf(i), drawingCache);
            }
            //獲取itemView的實際高度并累加
            if ((i + 1) % 3 == 0) {
                height += holder.itemView.getMeasuredHeight() + 
                              DensityUtil.dp2px(holder.itemView.getContext(), 20);
            }
            if ((i + 1) % 3 != 0 && (i + 1) == size) {
                height += holder.itemView.getMeasuredHeight() + 
                              DensityUtil.dp2px(holder.itemView.getContext(), 20);
            }
        }
        //根據計算出的recyclerView高度創建bitmap
        bigBitmap = Bitmap.createBitmap(recyclerView.getMeasuredWidth(), 
                                        height, Bitmap.Config.ARGB_8888);
        //創建一個canvas畫板
        Canvas canvas = new Canvas(bigBitmap);
        //當前bitmap的高度
        int top = 0;
        int left = 0;
        //畫筆
        Paint paint = new Paint();
        for (int i = 0; i < size; i++) {
            Bitmap bitmap = bitmapCache.get(String.valueOf(i));
            canvas.drawBitmap(bitmap, left, top, paint);
            if ((i + 1) % 3 == 0) {
                left = 0;
                top += bitmap.getHeight() + DensityUtil.dp2px(getUiView(), 20);
            } else {
                left += bitmapCache.get(String.valueOf(i)).getWidth();
            }
        }
    }
    return bigBitmap;
}

上面的代碼是模板代碼,網上搜基本上都是這樣的寫法;目的是為了解決OOM的問題,所以這里模擬了Adapter加載數據的情況,讓它在后臺執行加載數據,需要注意的是,在measure和layout的時候,需要根據你布局的實際情況來測量和布局,我這里的是GridLayoutManager, 所以每次測量的時候都是一行三列,實際邏輯還是得跟你的界面來;

在獲取到RecyclerView的holder之后,調用了adapter.onBindViewImageSync(holder, i);這是我自定義的一個同步獲取方法,目的是同步加載ImageView的圖片,其他的數據都是和onBindView()重載的方法一樣,我這里使用的是Glide加載圖片,4.x的標本和3.x標本不同,基本上使用還是差不多的:

try {
    File file = Glide.with(this)
            .load(sampleListBean.getImgUrl())
            .downloadOnly(100, 100)
            .get(5, TimeUnit.SECONDS);
    Bitmap bitmap = BitmapFactory.decodeFile(file.getAbsolutePath());
    ivPlant.setImageBitmap(bitmap);
    L.d("圖片加載成功! ==== " + position + "   " + bitmap.toString());
} catch (InterruptedException e) {
    e.printStackTrace();
} catch (ExecutionException e) {
    e.printStackTrace();
} catch (TimeoutException e) {
    e.printStackTrace();
}

當然,這里有耗時操作,在截圖的時候,需要彈出提示框加載;

截圖完之后,因為項目需求美觀,分享出去的圖片并不是一張赤裸裸的長圖,需要制作一張精美的圖片,但是當RecyclerView得數據很多時,我使用了getDrawingCache() 時,獲取到的數據為空,報錯信息:

 Log.w(VIEW_LOG_TAG, getClass().getSimpleName() + " not displayed because it is"
                    + " too large to fit into a software layer (or drawing cache), needs "
                    + projectedBitmapSize + " bytes, only "
                    + drawingCacheSize + " available");

這是getDrawingCache()方法中判斷當前創建緩存圖片大小和系統最大圖片大小,超過最大時報錯,所以,我修改成Canvas之后便可以了;隨后將制作的圖片保存到本地:

File file = FileUtils.getInstance().getOwnCacheDirectory(getUiView(), Constants.PLANT_CACHE_DIR);
File takePhotoFile = new File(file, "specimenShare.png");
FileOutputStream fout = null;
try {
    fout = new FileOutputStream(takePhotoFile);
} catch (FileNotFoundException e) {
    e.printStackTrace();
}
bitmap.compress(Bitmap.CompressFormat.PNG, 100, fout);

通過保存后的文件,便可以分享到QQ、微信了!

Github代碼

總結

  • getDrawingCache() 方法已經過時,谷歌推薦使用PixelCopy中的API使用,目前我沒有試過,有使用過的朋友方便提交代碼?
  • 可以使用Canvas來解決OOM的問題
  • 異步線程中同步加載圖片方法
  • RecyclerView截圖需要判斷當前LayoutManager
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,606評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,582評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,540評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,028評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,801評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,223評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,294評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,442評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,976評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,800評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,996評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,543評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,233評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,926評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,702評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,991評論 2 374