Android 动态加载二维码视图生成快照的示例
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包即可):点击下载>>
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地址>>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。