Android 动态加载二维码视图生成快照

本文介绍了一种实现动态加载视图并生成二维码快照的方法。重点解决了动态视图生成快照时获取不到bitmap的问题,并提供了具体实现代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.需求背景

需要实现一个动态加载但不显示出来的视图,且该视图上有个动态生成的二维码,最后用其去生成一张快照(也就是图片)。

(常见这种情况是来源于“图片分享”的功能需求,与普通图片分享不同在于,该快照图片是动态加载不显示的。)

 

2.需求功能拆解

  • 动态二维码的实现
  • 动态视图生成快照的实现

 

 

3.踩坑点提要

  • 获取不到动态视图的bitmap
  • 无法获取最新动态视图的bitmap

 

 

 

4.开发实现

动态加载的视图的布局文件代码:

 

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/qrcodeContentLl"
    android:background="#F0E68C"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:layout_marginTop="100dp"
        android:text="二维码快照"
        android:textSize="18sp"
        android:textStyle="italic" />

    <ImageView
        android:id="@+id/qrcodeIv"
        android:layout_width="100dp"
        android:layout_height="100dp"
        android:layout_gravity="center"
        android:layout_marginTop="@dimen/activity_vertical_margin"
        android:scaleType="fitCenter" />

    <!--<TextView-->
        <!--android:layout_width="wrap_content"-->
        <!--android:layout_height="wrap_content"-->
        <!--android:layout_marginTop="800dp"-->
        <!--android:text="ahahds"-->
        <!--android:layout_gravity="center"/>-->
</LinearLayout>


大概样式如下:

 

(上面的线框是用来显示动态生成的二维码图片的)

 

a.动态二维码的实现

关于这块内容,网上有太多例子了,其实也不用详解。主要是利用Zxing提供的jar包来进行处理。需要看这块的详细代码可以去文章最后提供的GitHub地址查看,在此只提供下该jar包的资源下载(项目中若只涉及生成二维码模块,那么只要core核心jar包即可):点击下载>> core-3.3.0.jar 

 

b.动态视图生成快照的实现

 

  private void inflateAndShowCaptureView() {
        if (hideView == null) {
            hideView = LayoutInflater.from(this).inflate(R.layout.layout_quick_capture, null);
            qrcodeIv = (ImageView) hideView.findViewById(R.id.qrcodeIv);
            hideView.setDrawingCacheEnabled(true);//设置控件允许绘制缓存
            hideView.measure(View.MeasureSpec.makeMeasureSpec(mainLayoutLl.getWidth(), View.MeasureSpec.EXACTLY),
                    View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));
            hideView.layout(0, 0, hideView.getMeasuredWidth(), hideView.getMeasuredHeight());
        } else {
            hideView.destroyDrawingCache();//要得到新的视图,就得销毁之前的缓存
        }

        showCaptureView();
    }

    private void showCaptureView() {
        String content = contentEt.getText().toString().trim();
        if (content == null || content.length() == 0) {
            return;
        }
        if (qrcodeIv.getWidth() == 0) {
            return;
        }
        Bitmap qrcodeBitmap = ZXingUtils.createQRImage(content, qrcodeIv.getWidth(), qrcodeIv.getHeight());
        qrcodeIv.setImageBitmap(qrcodeBitmap);//先将生成的二维码显示在加载的视图上

        Bitmap bitmap = hideView.getDrawingCache();  // 获取视图的绘制缓存(快照)
        if (bitmap != null) {
            showIv.setImageBitmap(bitmap);
        }

    }

 

 

1.首先获取到视图的bitmap是通过getDrawingCache()得到的。

  • 若视图是在界面上直接显示出来的——>那么使用该方法直接获取bitmap是没有问题的;
  • 若视图是动态加载且不显示出来,那么此时获取bitmap是null。

 

此处的解决办法就是手动给该视图布局:

 

            hideView.measure(View.MeasureSpec.makeMeasureSpec(mainLayoutLl.getWidth(), View.MeasureSpec.EXACTLY),
                    View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED));
            hideView.layout(0, 0, hideView.getMeasuredWidth(), hideView.getMeasuredHeight());

有关于以上两种方法的使用,可以参考这个文档的解释:点击查看>>

 

当然,篇幅有点长,以下做点简单解释:

View.MeasureSpec.makeMeasureSpec(int size , int mode)中有两个参数,size和mode,第一组MeasureSpec中我将size设置为了当前显示页面的布局的宽度(也就是屏幕宽度),然后mode设置为EXACTLY——>所表示的意义是:给hideView中的子View指定了精确的宽度大小为当前屏幕的宽度。

mode有三种,EXACTLY,AT_MOST,UNSPECIFIED。在上面代码中,将高度的size指定为0,mode指定为 UNSPECIFIED 则表示——>整个动态加载的视图高度指定为:依据于最后子View确认的高度。

若将第一组MeasureSpec的相关参数也改为size = 0, mode = UNSPECIFIED,则两组图对比显示如下:

可以看到,动态生成的快照的宽度也变成了显示二维码的ImageView的宽度了。

 

扩展:如何在宽高均为size = 0 && mode= UNSPECIFIED 的情况下获取整个屏幕大小的视图呢?

——>用几个隐藏的组件埋在视图的四个边界,啊哈哈哈哈哈!

 

2.通过destroyDrawingCache()来删除之前的缓存。

 

最后,GitHub地址>>

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ganshenml

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值