Android自定義原生分享界面以及View轉(zhuǎn)Bitmap的問(wèn)題

前言

前幾天設(shè)計(jì)師說(shuō)要把APP里的一個(gè)界面轉(zhuǎn)成圖片分享出去,本來(lái)想隨意寫(xiě)個(gè)Dialog算了,不過(guò)看到知乎里面的分享界面覺(jué)得還不錯(cuò),反正無(wú)聊自己也寫(xiě)成這樣算了,現(xiàn)在記錄下。

效果圖:
Screenshot_20180120-114048.jpg

1、界面

據(jù)觀察知乎的分享界面就是用BottomSheetDialog或者BottomSheetDialogFragment寫(xiě)的,所以這里我就用BottomSheetDialogFragment,具體界面就是

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/share_layout"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="@dimen/dip_fifty_six"
        android:layout_marginStart="@dimen/dip_sixteen"
        android:gravity="center_vertical"
        android:text="@string/share_out_to"
        android:textColor="@color/text_black_474c59"
        android:textSize="@dimen/sp_sixteen"/>

    <android.support.v7.widget.RecyclerView
        android:id="@+id/share_recycler_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

</LinearLayout>

RecyclerView的LayoutManager就用GridLayoutManager,設(shè)置一行三個(gè)填充。

2、數(shù)據(jù)

現(xiàn)在來(lái)獲取手機(jī)里具有分享功能的APP,在我們的手機(jī)中安裝了的應(yīng)用支持ACTION_SEND的Activity都會(huì)被列入可選列表。

由于我們需要顯示的內(nèi)容為APP的Icon、Label就可以了,所以通過(guò)PackageManager來(lái)獲取ResolveInfo列表就行。具體代碼為:

List<ResolveInfo> resolveList = packageManager.queryIntentActivities(intent, 0);

由于公司項(xiàng)目已經(jīng)集成了微信SDK,而且由原生分享出去的圖片在微信里并不會(huì)顯示來(lái)自哪個(gè)APP,所以在獲取到resolveList后我需要對(duì)獲取到的列表進(jìn)行篩選處理,去掉分享到微信的原生分享Activity數(shù)據(jù),這里直接通過(guò)微信的包名鑒別,去掉即可。然后再單獨(dú)在ResolveInfo集合的前面插入自己的數(shù)據(jù),這樣還能確保分享到朋友和分享到朋友圈始終在顯示的最前面兩個(gè)。

    /**
     * 得到支持分享的應(yīng)用
     *
     * @return 返回支持分享的app集合
     */
    public List<ShareItem> scanShreaApp() {
        mShareIntent = new Intent(Intent.ACTION_SEND);
        mShareIntent.setType("image/*");
        PackageManager packageManager = mContext.getPackageManager();
        mResolveInfos.clear();
        for (ResolveInfo resolveInfo : packageManager.queryIntentActivities(mShareIntent, 0)) {
            if (!resolveInfo.activityInfo.packageName.contains("com.tencent.mm")) {
                mResolveInfos.add(resolveInfo);
            }
        }
        ArrayList<ShareItem> shareItems = new ArrayList<>();
        for (ResolveInfo resolveInfo : mResolveInfos) {
            ShareItem shareItem = new ShareItem(resolveInfo.loadLabel(packageManager),
                    resolveInfo.loadIcon(packageManager));
            shareItems.add(shareItem);
        }
        return shareItems;
    }

然后在使用時(shí)將分享給朋友和分享到朋友圈加進(jìn)去:

shareItems.add(new ShareItem("發(fā)送給朋友",
                ContextCompat.getDrawable(getContext(), R.drawable.share_icon_wechat)));
shareItems.add(new ShareItem("發(fā)送到朋友圈",
                ContextCompat.getDrawable(getContext(), R.drawable.share_icon_moments)));
shareItems.addAll(mShare.scanShreaApp());
mShareItemAdapter.setShareItems(shareItems);

做點(diǎn)擊事件時(shí)單獨(dú)區(qū)分調(diào)用微信分享還是原生分享

mShareItemAdapter.setItmClickListener(new ShareItemAdapter.OnShareItmClickListener() {
            @Override
            public void onClick(int position) {
                mViewBitmap = generateShareImg(time, mArticleTitle, mArticleContent);
                mShareUri = BitmapUtil.saveBitmap(mViewBitmap);
                if (position == 0) {
                    shareToWX(SendMessageToWX.Req.WXSceneSession);
                } else if (position == 1) {
                    shareToWX(SendMessageToWX.Req.WXSceneTimeline);
                } else {
                    mShare.share(position - 2, mShareUri);
                }
                dismiss();
            }
        });

3、View轉(zhuǎn)Bitmap

現(xiàn)在來(lái)說(shuō)下View轉(zhuǎn)Bitmap遇到的問(wèn)題,這個(gè)東西還是可以用到很多地方的,比如之前寫(xiě)彈底部彈窗,設(shè)計(jì)師要求彈窗顯示的時(shí)候,背景是當(dāng)前界面的高斯模糊后的效果,在Android里面寫(xiě)高斯模糊可沒(méi)有iOS那么方便,人家直接就是系統(tǒng)提供的一個(gè)控件,咱們就得把當(dāng)前界面截屏然后轉(zhuǎn)為Bitmap,然后再給它高斯模糊下,再作為背景。

其實(shí)一般View轉(zhuǎn)Bitmap就兩個(gè)套路,

套路一

Bitmap shareBitmap = Bitmap.createBitmap(view.getMeasuredWidth(),
                view.getMeasuredHeight(),
                Bitmap.Config.ARGB_8888);
        Canvas c = new Canvas(shareBitmap);
        view.draw(c);
        

或者

套路二

view.setDrawingCacheEnabled(true);
Bitmap shareBitmap = Bitmap.createBitmap(view.getDrawingCache());
view.setDrawingCacheEnabled(false);

在使用中,如果view是已經(jīng)顯示在界面上了,那么直接使用時(shí)沒(méi)什么問(wèn)題的,但如果是一個(gè)未曾顯示的界面,想要轉(zhuǎn)換成Bitmap就需要先創(chuàng)建并計(jì)算大小,代碼如下:

View view = View.inflate(this, R.layout.share_out_layout, null);
view.measure(View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED),
                View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));
view.layout(0, 0, view.getMeasuredWidth(), view.getMeasuredHeight());
view.setDrawingCacheEnabled(true);
Bitmap shareBitmap = Bitmap.createBitmap(view.getDrawingCache());
view.setDrawingCacheEnabled(false);
return shareBitmap;

到這里其實(shí)已經(jīng)差不多了,但是?。?! 我在Activity中這樣使用一切正常,放在Fragment中用的時(shí)候就GG了,報(bào)錯(cuò)信息:java.lang.NullPointerException: Attempt to invoke virtual method 'int android.graphics.Bitmap.getWidth()' on a null object reference 查看Google搜索出來(lái)的結(jié)果,有人說(shuō)測(cè)量view的時(shí)候,如果你的布局中包含有 RelativeLayout )API 為17 或者 低于17 會(huì)報(bào)空指針異常。
但是我去掉RelativeLayout后還是有問(wèn)題,再把measure時(shí)的參數(shù)改成int + View.MeasureSpec.EXACTLY ,就確實(shí)不會(huì)報(bào)錯(cuò)了,但是由于view的大小被設(shè)置為固定數(shù)值,效果太差了。后面試了下套路一,一切正常!??!

所以文章到此結(jié)束.....

最后編輯于
?著作權(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,190評(píng)論 25 708
  • ¥開(kāi)啟¥ 【iAPP實(shí)現(xiàn)進(jìn)入界面執(zhí)行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開(kāi)一個(gè)線程,因...
    小菜c閱讀 7,362評(píng)論 0 17
  • 原文鏈接:https://github.com/opendigg/awesome-github-android-u...
    IM魂影閱讀 33,170評(píng)論 6 472
  • 內(nèi)容抽屜菜單ListViewWebViewSwitchButton按鈕點(diǎn)贊按鈕進(jìn)度條TabLayout圖標(biāo)下拉刷新...
    皇小弟閱讀 47,169評(píng)論 22 665
  • 今天是開(kāi)始復(fù)習(xí)生理學(xué)的第三天,今天也剛好背到第三章。說(shuō)實(shí)話,我有點(diǎn)焦躁。焦躁的原因有好幾個(gè):1 1月6號(hào)考生理,1...
    12d051326acc閱讀 299評(píng)論 0 0

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