前言

二维码以前也写过好几篇,今天姑且是对以前知识的总结,顺便复习一下,圣人说过:温故知新

还是和以前一样先看下效果,再来对着代码讲解。

看效果图:


之前呢,也写过用安卓实现二维码生成彩色的二维码和带logo的二维码,然后这一篇呢也是写二维码,主要也是看见很多的非常漂亮的二维码,这里呢主要模仿qq的二维码,并且也高仿实现了长按发送给朋友和保存到图库的功能,觉得不错呢就请多支持下,哪里不好呢也可以说出来。好了我们一步一步来。

第一步:简单二维码实现

先来个最简单的二维码:

看下简单代码实现:

 /*** 根据指定内容生成自定义宽高的二维码图片** @param content*            需要生成二维码的内容* @param width*            二维码宽度* @param height*            二维码高度* @throws WriterException*             生成二维码异常*/public static Bitmap makeQRImage(String content, int width, int height)throws WriterException {Hashtable<EncodeHintType, String> hints = new Hashtable<EncodeHintType, String>();hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");// 图像数据转换,使用了矩阵转换BitMatrix bitMatrix = new QRCodeWriter().encode(content,BarcodeFormat.QR_CODE, width, height, hints);int[] pixels = new int[width * height];// 按照二维码的算法,逐个生成二维码的图片,两个for循环是图片横列扫描的结果for (int y = 0; y < height; y++) {for (int x = 0; x < width; x++) {if (bitMatrix.get(x, y))//范围内为黑色的pixels[y * width + x] = 0xff000000;else//其他的地方为白色pixels[y * width + x] = 0xffffffff;}}// 生成二维码图片的格式,使用ARGB_8888Bitmap bitmap = Bitmap.createBitmap(width, height,Bitmap.Config.ARGB_8888);//设置像素矩阵的范围bitmap.setPixels(pixels, 0, width, 0, 0, width, height);return bitmap;}

第二步:简单二维码加logo

接下来给二维码加logo:(看图)

 /*** 根据指定内容生成自定义宽高的二维码图片** param logoBm*            logo图标* param content*            需要生成二维码的内容* param width*            二维码宽度* param height*            二维码高度* throws WriterException*             生成二维码异常*/public static Bitmap makeQRImage(Bitmap logoBmp, String content,int QR_WIDTH, int QR_HEIGHT) throws WriterException {try {// 图像数据转换,使用了矩阵转换Hashtable<EncodeHintType, Object> hints = new Hashtable<EncodeHintType, Object>();hints.put(EncodeHintType.CHARACTER_SET, "utf-8");hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);// 容错率hints.put(EncodeHintType.MARGIN, 2); // default is 4hints.put(EncodeHintType.MAX_SIZE, 350);hints.put(EncodeHintType.MIN_SIZE, 100);BitMatrix bitMatrix = new QRCodeWriter().encode(content,BarcodeFormat.QR_CODE, QR_WIDTH, QR_HEIGHT, hints);int[] pixels = new int[QR_WIDTH * QR_HEIGHT];for (int y = 0; y < QR_HEIGHT; y++) {// 下面这里按照二维码的算法,逐个生成二维码的图片,//两个for循环是图片横列扫描的结果for (int x = 0; x < QR_WIDTH; x++) {if (bitMatrix.get(x, y))pixels[y * QR_WIDTH + x] = 0xff000000;elsepixels[y * QR_WIDTH + x] = 0xffffffff;}}// ------------------添加图片部分------------------//Bitmap bitmap = Bitmap.createBitmap(QR_WIDTH, QR_HEIGHT,Bitmap.Config.ARGB_8888);// 设置像素点bitmap.setPixels(pixels, 0, QR_WIDTH, 0, 0, QR_WIDTH, QR_HEIGHT);// 获取图片宽高int logoWidth = logoBmp.getWidth();int logoHeight = logoBmp.getHeight();if (QR_WIDTH == 0 || QR_HEIGHT == 0) {return null;}if (logoWidth == 0 || logoHeight == 0) {return bitmap;}// 图片绘制在二维码中央,合成二维码图片// logo大小为二维码整体大小的1/2float scaleFactor = QR_WIDTH * 1.0f / 2 / logoWidth;try {Canvas canvas = new Canvas(bitmap);canvas.drawBitmap(bitmap, 0, 0, null);canvas.scale(scaleFactor, scaleFactor, QR_WIDTH / 2,QR_HEIGHT / 2);canvas.drawBitmap(logoBmp, (QR_WIDTH - logoWidth) / 2,(QR_HEIGHT - logoHeight) /2, null);canvas.save(Canvas.ALL_SAVE_FLAG);canvas.restore();return bitmap;} catch (Exception e) {bitmap = null;e.getStackTrace();}} catch (WriterException e) {e.printStackTrace();}return null;}

第三步:实现带logo的彩色二维码

接下来我们把黑白矩阵变为彩色矩阵:
就把

 if (bitMatrix.get(x, y))pixels[y * width + x] = 0xff000000;elsepixels[y * width + x] = 0xffffffff;

替换为:(这里的颜色随便设置,效果随便改)

 if (x < QR_WIDTH / 2 && y < QR_HEIGHT / 2) {pixels[y * QR_WIDTH + x] = 0xFF0094FF;// 蓝色Integer.toHexString(new Random().nextInt());} else if (x < QR_WIDTH / 2 && y > QR_HEIGHT / 2) {pixels[y * QR_WIDTH + x] = 0xFFFED545;// 黄色} else if (x > QR_WIDTH / 2 && y > QR_HEIGHT / 2) {pixels[y * QR_WIDTH + x] = 0xFF5ACF00;// 绿色} else {pixels[y * QR_WIDTH + x] = 0xFF000000;// 黑色}} else {pixels[y * QR_WIDTH + x] = 0xffffffff;// 白色}

改后的效果:

第四步:给二维码加背景

接下来我们来给二维码图片加背景:

/*** 给二维码图片加背景**/public static Bitmap addBackground(Bitmap foreground,Bitmap background){int bgWidth = background.getWidth();int bgHeight = background.getHeight();int fgWidth = foreground.getWidth();int fgHeight = foreground.getHeight();Bitmap newmap = Bitmap.createBitmap(bgWidth, bgHeight, Bitmap.Config.ARGB_8888);Canvas canvas = new Canvas(newmap);canvas.drawBitmap(background, 0, 0, null);canvas.drawBitmap(foreground, (bgWidth - fgWidth) / 2,(bgHeight - fgHeight) *3 / 5+70, null);canvas.save(Canvas.ALL_SAVE_FLAG);canvas.restore();return newmap;}

这样效果就变为:

第五步:给二维码加水印

然后二维码的个性化制作就最后一步了:加水印,位置随便放

 /*** 在图片右下角添加水印** @param srcBMP*            原图* @param markBMP*            水印图片* @return 合成水印后的图片*/public static Bitmap composeWatermark(Bitmap srcBMP, Bitmap markBMP) {if (srcBMP == null) {return null;}// 创建一个新的和SRC长度宽度一样的位图Bitmap newb = Bitmap.createBitmap(srcBMP.getWidth(),srcBMP.getHeight(), Bitmap.Config.ARGB_8888);Canvas cv = new Canvas(newb);// 在 0,0坐标开始画入原图cv.drawBitmap(srcBMP, 0, 0, null);// 在原图的右下角画入水印cv.drawBitmap(markBMP, srcBMP.getWidth() - markBMP.getWidth()*4/5,srcBMP.getHeight()*2/7 , null);// 保存cv.save(Canvas.ALL_SAVE_FLAG);// 存储cv.restore();return newb;}


这里贴下实现二维码个性化的完整代码类:

package com.ry.personalizedcode.uitls;import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.QRCodeWriter;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;import java.util.Hashtable;
import java.util.Random;/*** Created  on 2016/2/24.* 生成二维码的工具类*/
public class MakeQRCodeUtil {/*** 根据指定内容生成自定义宽高的二维码图片** param logoBm*            logo图标* param content*            需要生成二维码的内容* param width*            二维码宽度* param height*            二维码高度* throws WriterException*             生成二维码异常*/public static Bitmap makeQRImage(Bitmap logoBmp, String content,int QR_WIDTH, int QR_HEIGHT) throws WriterException {try {// 需要引入core包// 图像数据转换,使用了矩阵转换Hashtable<EncodeHintType, Object> hints = new Hashtable<EncodeHintType, Object>();hints.put(EncodeHintType.CHARACTER_SET, "utf-8");hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);// 容错率hints.put(EncodeHintType.MARGIN, 2); // default is 4hints.put(EncodeHintType.MAX_SIZE, 350);hints.put(EncodeHintType.MIN_SIZE, 100);BitMatrix bitMatrix = new QRCodeWriter().encode(content,BarcodeFormat.QR_CODE, QR_WIDTH, QR_HEIGHT, hints);int[] pixels = new int[QR_WIDTH * QR_HEIGHT];for (int y = 0; y < QR_HEIGHT; y++) {// 下面这里按照二维码的算法,逐个生成二维码的图片,//两个for循环是图片横列扫描的结果for (int x = 0; x < QR_WIDTH; x++) {if (bitMatrix.get(x, y)) {if (x < QR_WIDTH / 2 && y < QR_HEIGHT / 2) {pixels[y * QR_WIDTH + x] = 0xFF0094FF;// 蓝色Integer.toHexString(new Random().nextInt());} else if (x < QR_WIDTH / 2 && y > QR_HEIGHT / 2) {pixels[y * QR_WIDTH + x] = 0xFFFED545;// 黄色} else if (x > QR_WIDTH / 2 && y > QR_HEIGHT / 2) {pixels[y * QR_WIDTH + x] = 0xFF5ACF00;// 绿色} else {pixels[y * QR_WIDTH + x] = 0xFF000000;// 黑色}} else {pixels[y * QR_WIDTH + x] = 0xffffffff;// 白色}}}// ------------------添加图片部分------------------//Bitmap bitmap = Bitmap.createBitmap(QR_WIDTH, QR_HEIGHT,Bitmap.Config.ARGB_8888);// 设置像素点bitmap.setPixels(pixels, 0, QR_WIDTH, 0, 0, QR_WIDTH, QR_HEIGHT);// 获取图片宽高int logoWidth = logoBmp.getWidth();int logoHeight = logoBmp.getHeight();if (QR_WIDTH == 0 || QR_HEIGHT == 0) {return null;}if (logoWidth == 0 || logoHeight == 0) {return bitmap;}// 图片绘制在二维码中央,合成二维码图片// logo大小为二维码整体大小的1/2float scaleFactor = QR_WIDTH * 1.0f / 2 / logoWidth;try {Canvas canvas = new Canvas(bitmap);canvas.drawBitmap(bitmap, 0, 0, null);canvas.scale(scaleFactor, scaleFactor, QR_WIDTH / 2,QR_HEIGHT / 2);canvas.drawBitmap(logoBmp, (QR_WIDTH - logoWidth) / 2,(QR_HEIGHT - logoHeight) /2, null);canvas.save(Canvas.ALL_SAVE_FLAG);canvas.restore();return bitmap;} catch (Exception e) {bitmap = null;e.getStackTrace();}} catch (WriterException e) {e.printStackTrace();}return null;}/*** 获取十六进制的颜色代码.例如  "#6E36B4" , For HTML ,* @return String*/public static String getRandColorCode(){String r,g,b;Random random = new Random();r = Integer.toHexString(random.nextInt(256)).toUpperCase();g = Integer.toHexString(random.nextInt(256)).toUpperCase();b = Integer.toHexString(random.nextInt(256)).toUpperCase();r = r.length()==1 ? "0" + r : r ;g = g.length()==1 ? "0" + g : g ;b = b.length()==1 ? "0" + b : b ;return r+g+b;}/*** 根据指定内容生成自定义宽高的二维码图片** @param content*            需要生成二维码的内容* @param width*            二维码宽度* @param height*            二维码高度* @throws WriterException*             生成二维码异常*/public static Bitmap makeQRImage(String content, int width, int height)throws WriterException {Hashtable<EncodeHintType, String> hints = new Hashtable<EncodeHintType, String>();hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");// 图像数据转换,使用了矩阵转换BitMatrix bitMatrix = new QRCodeWriter().encode(content,BarcodeFormat.QR_CODE, width, height, hints);int[] pixels = new int[width * height];// 按照二维码的算法,逐个生成二维码的图片,两个for循环是图片横列扫描的结果for (int y = 0; y < height; y++) {for (int x = 0; x < width; x++) {if (bitMatrix.get(x, y))pixels[y * width + x] = 0xff000000;elsepixels[y * width + x] = 0xffffffff;}}// 生成二维码图片的格式,使用ARGB_8888Bitmap bitmap = Bitmap.createBitmap(width, height,Bitmap.Config.ARGB_8888);bitmap.setPixels(pixels, 0, width, 0, 0, width, height);return bitmap;}/*** 从资源文件中获取图片** @param context*            上下文* @param drawableId*            资源文件id* @return*/public static Bitmap gainBitmap(Context context, int drawableId) {Bitmap bmp = BitmapFactory.decodeResource(context.getResources(),drawableId);return bmp;}/*** 在图片右下角添加水印** @param srcBMP*            原图* @param markBMP*            水印图片* @return 合成水印后的图片*/public static Bitmap composeWatermark(Bitmap srcBMP, Bitmap markBMP) {if (srcBMP == null) {return null;}// 创建一个新的和SRC长度宽度一样的位图Bitmap newb = Bitmap.createBitmap(srcBMP.getWidth(),srcBMP.getHeight(), Bitmap.Config.ARGB_8888);Canvas cv = new Canvas(newb);// 在 0,0坐标开始画入原图cv.drawBitmap(srcBMP, 0, 0, null);// 在原图的右下角画入水印cv.drawBitmap(markBMP, srcBMP.getWidth() - markBMP.getWidth()*4/5,srcBMP.getHeight()*2/7 , null);// 保存cv.save(Canvas.ALL_SAVE_FLAG);// 存储cv.restore();return newb;}/*** 给二维码图片加背景**/public static Bitmap addBackground(Bitmap foreground,Bitmap background){int bgWidth = background.getWidth();int bgHeight = background.getHeight();int fgWidth = foreground.getWidth();int fgHeight = foreground.getHeight();Bitmap newmap = Bitmap.createBitmap(bgWidth, bgHeight, Bitmap.Config.ARGB_8888);Canvas canvas = new Canvas(newmap);canvas.drawBitmap(background, 0, 0, null);canvas.drawBitmap(foreground, (bgWidth - fgWidth) / 2,(bgHeight - fgHeight) *3 / 5+70, null);canvas.save(Canvas.ALL_SAVE_FLAG);canvas.restore();return newmap;}
}

第六步:给二维码实现长按功能

最后为了模拟下qq的查看二维码名片功能,还加了一个长按弹出actionSheet的功能。
看效果:

具体的 安卓版actionSheet的实现,前面博客有介绍需要的请移步。
这里我们先来实现发送给好友功能:(这里就不做第三方的发送)

 private void sendToFriends() {Intent intent=new Intent(Intent.ACTION_SEND);Uri imageUri=  Uri.parse(Environment.getExternalStorageDirectory()+"/code/qrcode.jpg");intent.setType("image/*");intent.putExtra(Intent.EXTRA_STREAM, imageUri);intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);startActivity(Intent.createChooser(intent, getTitle()));}

发送给朋友效果图:


然后就是要实现保存到本地图库的功能:

    /*** 先保存到本地再广播到图库* */public static void saveImageToGallery(Context context, Bitmap bmp) {// 首先保存图片File appDir = new File(Environment.getExternalStorageDirectory(),"code");if (!appDir.exists()) {appDir.mkdir();}String fileName = "qrcode.jpg";file = new File(appDir, fileName);try {FileOutputStream fos = new FileOutputStream(file);bmp.compress(CompressFormat.JPEG, 100, fos);fos.flush();fos.close();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}// 其次把文件插入到系统图库try {MediaStore.Images.Media.insertImage(context.getContentResolver(),   file.getAbsolutePath(), fileName, null);// 最后通知图库更新context.sendBroadcast(new Intent(           Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://"+ file)));} catch (FileNotFoundException e) {e.printStackTrace();}}

总结:

这篇说白了就是使用了大量的Canvas和bitmap的处理,然后篇幅也是有点长,看起来也是有点累。要看完整的代码请自己下载PersonalizedCode.rar。下一篇安卓webView实现长按二维码的自动识别功能

安卓的个性化彩色二维码的完美实现相关推荐

  1. 【基础入门详解】程序员的二维码也要玩出不同色彩,用Python生成动态彩色二维码

    一.前言 常见的二维码为QR CodeCode,QR全称是Quick Response,是一个近几年来移动设备上超流行的一种编码方式.它的结构如下: 主要有以下特点: 符号规格从版本1(21×21)到 ...

  2. 关于二维码分块上色(彩色二维码)的算法研究

    原文:关于二维码分块上色(彩色二维码)的算法研究 众所周知,二维码通常是黑白的,而且是由若干个长方形或正方形小块平铺而成.但从人们的审美角度来看,常见的黑白二维码不免让人审美疲劳.本文试着从分块上色的 ...

  3. Python制作彩色二维码

    文章目录 1.安装myqr 2.生成二维码 3.动态二维码 4.彩色二维码 1.安装myqr 先在 pycharm 安装 myqr.或者,Python3 必装,然后命令行pip install myq ...

  4. [Python]使用QRCode生成彩色二维码

    一. 安装QRCode模块 QRCode模块依赖于Pillow模块,在安装QRCode模块前需要先安装Pillow模块.可使用如下命令安装: pip install Pillow QRCode模块安装 ...

  5. 生成彩色二维码(渐变色、插入图片和文字)

    花了一天时间,研究生成彩色的二维码程序. 下面是阶段性成果:渐变颜色.插入文字和log图片. 使二维码看起来更协调.更醒目.突出强调文字和logo,而不仅仅是一个信息的黑盒. 基于zxing的c#版本 ...

  6. Android彩色二维码带logo

    最近看到彩色二维码挺有意思,感觉我们常见的黑色二维码有些单调. 看了一下大牛们的博客,这里在大牛的基础上更改了一下.做了一个有颜色渐变的二维码. 在这里记录一下自己的学习成果.我这里用的是zxing- ...

  7. C# 生成彩色二维码

    /// <summary>/// 生成彩色二维码/// </summary>/// <param name="data">二维码内容</p ...

  8. 【彩色二维码】处理生成彩色二维码图片

    调用草料二维码的API接口 https://cli.im/api/qrcode/code?text=***&mhid=*** mhid:草料二维码的模版样式id,这里选择的是[vUbEWVm7 ...

  9. Android之自定义生成彩色二维码

    先导个zxing.jar包 下面是xml布局 activity_main.xml <RelativeLayout xmlns:android="http://schemas.andro ...

  10. PHP QRCODE 彩色二维码

    彩色二维码PHP代码下载 重写了下PHPQRCODE,精简了部分代码,合并PNG GIF JPEG的输出. /*参数说明: 调用方式: QRcode::IMGout($text, $outfile = ...

最新文章

  1. Go 学习笔记(80)— Go 标准库 container/list(单链表、双链表)
  2. Bootstrap开启模态框后对数据处理(标记模态框的开启与关闭状态)
  3. manually set focus for tag library
  4. centos6.2 64安装mysql_centos6.2 64bit安装mysql+php
  5. 2021河南高考成绩查询郸城一高,郸城一高在河南有多牛 2021年高考成绩
  6. 数据结构与算---重点复习知识
  7. 关闭vue中的eslint校验
  8. 让Office无处不在——Office Web App初体验
  9. 数值分析(5)-分段低次插值和样条插值
  10. 万能免费信息采集软件-免费网站信息内容数据采集软件
  11. 职业规划-三大职业生涯阶段
  12. 淼淼Kruskal算法
  13. 推荐一款非常好用的鼠标手势软件StrokeIt
  14. java编程实现行列式计算应用_Java实现行列式计算
  15. WebAssembly 实践:如何写代码
  16. 怎么删除网络共享多余的计算机,win10系统删除网络共享中多余的计算机的图文教程...
  17. OGG/OGV文件格式解析
  18. css 怎么让图片一直旋转
  19. PCB各种表面工艺差异说明
  20. 费雪分离定理的证明与评价

热门文章

  1. Rust之crate
  2. pycharm破解补丁激活
  3. 分类器交叉验证java_使用交叉验证和F1分数选择SVM参数
  4. 前端js中文转拼音(例:张三转为ZhangSan)
  5. CATIA 安装Service Pack 时出错 检查完整性失败
  6. 机房环境监控报警系统
  7. 华为改变策略,出击超低端手机市场
  8. 错误推测法设计测试用例
  9. 2022陈箫箫胡程灿信息论课程作业 MIMO信道(multi-input multi-output,MIMO)简介
  10. 结构化思维在产品工作中的应用