Android中使用ZXing生成二維碼(支持添加Logo圖案)
ZXing是谷歌的一個(gè)開源庫,可以用來生成二維碼、掃描二維碼。本文所介紹的是第一部分。
首先上效果圖:
ZXing相關(guān)各種文件官方下載地址:https://github.com/zxing/zxing/releases
或者在這里下載(只有本項(xiàng)目所用的jar包,版本號:3.2.0):鏈接:http://pan.baidu.com/s/1pLqAR5x
1.生成二維碼的工具類
/** * 二維碼生成工具類 */ public class QRCodeUtil { /** * 生成二維碼Bitmap * * @param content 內(nèi)容 * @param widthPix 圖片寬度 * @param heightPix 圖片高度 * @param logoBm 二維碼中心的Logo圖標(biāo)(可以為null) * @param filePath 用于存儲(chǔ)二維碼圖片的文件路徑 * @return 生成二維碼及保存文件是否成功 */ public static boolean createQRImage(String content, int widthPix, int heightPix, Bitmap logoBm, String filePath) { try { if (content == null || "".equals(content)) { return false; } //配置參數(shù) Map<encodehinttype, object=""> hints = new HashMap<>(); hints.put(EncodeHintType.CHARACTER_SET, "utf-8"); //容錯(cuò)級別 hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H); //設(shè)置空白邊距的寬度 // hints.put(EncodeHintType.MARGIN, 2); //default is 4 // 圖像數(shù)據(jù)轉(zhuǎn)換,使用了矩陣轉(zhuǎn)換 BitMatrix bitMatrix = new QRCodeWriter().encode(content, BarcodeFormat.QR_CODE, widthPix, heightPix, hints); int[] pixels = new int[widthPix * heightPix]; // 下面這里按照二維碼的算法,逐個(gè)生成二維碼的圖片, // 兩個(gè)for循環(huán)是圖片橫列掃描的結(jié)果 for (int y = 0; y < heightPix; y++) { for (int x = 0; x < widthPix; x++) { if (bitMatrix.get(x, y)) { pixels[y * widthPix + x] = 0xff000000; } else { pixels[y * widthPix + x] = 0xffffffff; } } } // 生成二維碼圖片的格式,使用ARGB_8888 Bitmap bitmap = Bitmap.createBitmap(widthPix, heightPix, Bitmap.Config.ARGB_8888); bitmap.setPixels(pixels, 0, widthPix, 0, 0, widthPix, heightPix); if (logoBm != null) { bitmap = addLogo(bitmap, logoBm); } //必須使用compress方法將bitmap保存到文件中再進(jìn)行讀取。直接返回的bitmap是沒有任何壓縮的,內(nèi)存消耗巨大! return bitmap != null && bitmap.compress(Bitmap.CompressFormat.JPEG, 100, new FileOutputStream(filePath)); } catch (WriterException | IOException e) { e.printStackTrace(); } return false; } /** * 在二維碼中間添加Logo圖案 */ private static Bitmap addLogo(Bitmap src, Bitmap logo) { if (src == null) { return null; } if (logo == null) { return src; } //獲取圖片的寬高 int srcWidth = src.getWidth(); int srcHeight = src.getHeight(); int logoWidth = logo.getWidth(); int logoHeight = logo.getHeight(); if (srcWidth == 0 || srcHeight == 0) { return null; } if (logoWidth == 0 || logoHeight == 0) { return src; } //logo大小為二維碼整體大小的1/5 float scaleFactor = srcWidth * 1.0f / 5 / logoWidth; Bitmap bitmap = Bitmap.createBitmap(srcWidth, srcHeight, Bitmap.Config.ARGB_8888); try { Canvas canvas = new Canvas(bitmap); canvas.drawBitmap(src, 0, 0, null); canvas.scale(scaleFactor, scaleFactor, srcWidth / 2, srcHeight / 2); canvas.drawBitmap(logo, (srcWidth - logoWidth) / 2, (srcHeight - logoHeight) / 2, null); canvas.save(Canvas.ALL_SAVE_FLAG); canvas.restore(); } catch (Exception e) { bitmap = null; e.getStackTrace(); } return bitmap; } }</encodehinttype,>
2.在Activity中的使用:
/** * 二維碼生成 */ public class MainActivity extends ActionBarActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //內(nèi)容 final EditText contentET = (EditText) findViewById(R.id.create_qr_content); //顯示二維碼圖片 final ImageView imageView = (ImageView) findViewById(R.id.create_qr_iv); //是否添加Logo final CheckBox addLogoCB = (CheckBox) findViewById(R.id.create_qr_addLogo); Button createQrBtn = (Button) findViewById(R.id.create_qr_btn); createQrBtn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { final String filePath = getFileRoot(MainActivity.this) + File.separator + "qr_" + System.currentTimeMillis() + ".jpg"; //二維碼圖片較大時(shí),生成圖片、保存文件的時(shí)間可能較長,因此放在新線程中 new Thread(new Runnable() { @Override public void run() { boolean success = QRCodeUtil.createQRImage(contentET.getText().toString().trim(), 800, 800, addLogoCB.isChecked() ? BitmapFactory.decodeResource(getResources(), R.mipmap.qr_logo) : null, filePath); if (success) { runOnUiThread(new Runnable() { @Override public void run() { imageView.setImageBitmap(BitmapFactory.decodeFile(filePath)); } }); } } }).start(); } }); } //文件存儲(chǔ)根目錄 private String getFileRoot(Context context) { if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { File external = context.getExternalFilesDir(null); if (external != null) { return external.getAbsolutePath(); } } return context.getFilesDir().getAbsolutePath(); } }
3.本項(xiàng)目中圖片文件保存在
context.getExternalFilesDir(null)
目錄之下的。按照官方的api文檔,從KitKat開始(Android 4.4),保存文件到這個(gè)目錄下將不需要SD卡讀寫權(quán)限。但是經(jīng)過測驗(yàn)表明,在紅米Note和魅族MX3上(系統(tǒng)均為android 4.4.4),的確不需要權(quán)限;但是在本人的華為P6上(Android 4.4.2),必須聲明權(quán)限才能成功的保存文件,即必須在manifest中添加如下內(nèi)容:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"></uses-permission></uses-permission>
因此,個(gè)人猜測,所謂的不需要權(quán)限是從 Android 4.4.4開始的。
下面再給大家分享Android生成二維碼及添加logo的代碼
具體代碼如下所示:
@Override public Bitmap generateBitmap(String content, int width, int height) { QRCodeWriter qrCodeWriter = new QRCodeWriter(); Map<EncodeHintType, String> hints = new HashMap<>(); hints.put(EncodeHintType.CHARACTER_SET, "utf-8");//默認(rèn) hints.put(EncodeHintType.MARGIN, "1");//無白色邊框 try { BitMatrix encode = qrCodeWriter.encode(content, BarcodeFormat.QR_CODE, width, height, hints); int[] pixels = new int[width * height]; for (int i = 0; i < height; i++) { for (int j = 0; j < width; j++) { if (encode.get(j, i)) { pixels[i * width + j] = 0x00000000; } else { pixels[i * width + j] = 0xffffffff; } } } return Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565); } catch (WriterException e) { e.printStackTrace(); } return null; } @Override public Bitmap addLogo(Bitmap qrBitmap, Bitmap logoBitmap) { int qrBitmapWidth = qrBitmap.getWidth(); int qrBitmapHeight = qrBitmap.getHeight(); int logoBitmapWidth = logoBitmap.getWidth(); int logoBitmapHeight = logoBitmap.getHeight(); Bitmap blankBitmap = Bitmap.createBitmap(qrBitmapWidth, qrBitmapHeight, Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(blankBitmap); canvas.drawBitmap(qrBitmap, 0, 0, null); canvas.save(Canvas.ALL_SAVE_FLAG); float scaleSize = 1.0f; while ((logoBitmapWidth / scaleSize) > (qrBitmapWidth / 3.5) || (logoBitmapHeight / scaleSize) > (qrBitmapHeight / 3.5)) { scaleSize *= 2; } float sx = 1.0f / scaleSize; canvas.scale(sx, sx, qrBitmapWidth / 2, qrBitmapHeight / 2); canvas.drawBitmap(logoBitmap, (qrBitmapWidth - logoBitmapWidth) / 2, (qrBitmapHeight - logoBitmapHeight) / 2, null); canvas.restore(); return blankBitmap; }
好了,代碼到此結(jié)束了,希望對大家有所幫助!
- Android實(shí)現(xiàn)二維碼掃描和生成的簡單方法
- Android 超簡易Zxing框架 生成二維碼+掃碼功能
- Android 二維碼掃描和生成二維碼功能
- Android平臺生成二維碼并實(shí)現(xiàn)掃描 & 識別功能
- Android 二維碼 生成和識別二維碼 附源碼下載
- Android編程實(shí)現(xiàn)二維碼的生成與解析
- Android基于google Zxing實(shí)現(xiàn)二維碼的生成
- Android生成條形碼和二維碼功能
- Android 點(diǎn)擊生成二維碼功能實(shí)現(xiàn)代碼
- Android中二維碼的掃描和生成(使用zxing庫)
相關(guān)文章
Android實(shí)現(xiàn)判斷手機(jī)未接來電及處理方法
這篇文章主要介紹了Android實(shí)現(xiàn)判斷手機(jī)未接來電及處理方法,需要的朋友可以參考下2014-07-07Android通過startService實(shí)現(xiàn)文件批量下載
這篇文章主要為大家詳細(xì)介紹了Android通過startService實(shí)現(xiàn)文件批量下載的示例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2015-12-12Android使用animator實(shí)現(xiàn)fragment的3D翻轉(zhuǎn)效果
這篇文章主要為大家詳細(xì)介紹了Android使用animator實(shí)現(xiàn)fragment的3D翻轉(zhuǎn)效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12Android 物理游戲之重力系統(tǒng)開發(fā)示例代碼
介紹Android 物理游戲之重力系統(tǒng),這里提供了詳細(xì)的資料整理,并附示例代碼和實(shí)現(xiàn)效果圖,有興趣的小伙伴可以參考下2016-08-08Android自定義滑動(dòng)接聽電話控件組實(shí)例
這篇文章主要介紹了Android自定義滑動(dòng)接聽電話控件組,接聽電話可以左右滑動(dòng),感興趣的小伙伴們可以參考一下。2016-10-10