初学安卓之二维码的简单实现,android开发网上购物app

  • 打开闪光灯

  • ScanQRcode完整代码

  • 实现结果

  • 最后注解

  • 参考链接

前言

=============================================================

随着互联网的发展,在网上的二维码种类和作用越来越多,因此我想着自己也实现生成二维码以及扫码。经过上网查资料发现,大多实现二维码功能的,主要都是靠集成了ZXing开源项目的功能,目前还有人在维护,足够我们学习使用。

生成二维码

================================================================

准备工作


  • 在ZXing开源库中我们可以看到它已经升级到了3.4.0,于是导入3.4.0的依赖

在这里插入图片描述

  • build.gradle中添加依赖com.google.zxing:core:3.4.0

apply plugin: ‘com.android.application’

android {

compileSdkVersion 29

buildToolsVersion “29.0.3”

defaultConfig {

applicationId “com.example.qrcodetest”

minSdkVersion 19

targetSdkVersion 29

versionCode 1

versionName “1.0”

testInstrumentationRunner “androidx.test.runner.AndroidJUnitRunner”

}

buildTypes {

release {

minifyEnabled false

proguardFiles getDefaultProguardFile(‘proguard-android-optimize.txt’), ‘proguard-rules.pro’

}

}

}

dependencies {

implementation fileTree(dir: ‘libs’, include: [’*.jar’])

implementation ‘androidx.appcompat:appcompat:1.1.0’

implementation ‘androidx.constraintlayout:constraintlayout:1.1.3’

testImplementation ‘junit:junit:4.12’

androidTestImplementation ‘androidx.test.ext:junit:1.1.1’

androidTestImplementation ‘androidx.test.espresso:espresso-core:3.2.0’

implementation ‘com.google.zxing:core:3.4.0’

}

  • activity_main.xml文件中添加两个按钮,一个是生成二维码,一个是扫描二维码,并在MainActivity中添加监听按钮点击,实现以下界面

主界面

生成简单二维码


  1. 创建一个名为GenerateQRcode的Activity,Android Studio自动生成其布局文件,布局文件改名为generate_qrcode.xml:放置一个EditText输入二维码内容、一个Button监听操作以及一个ImageView提供二维码的显示,对应的string在string.xml中添加
<?xml version="1.0" encoding="utf-8"?>

<androidx.constraintlayout.widget.ConstraintLayout xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:app=“http://schemas.android.com/apk/res-auto”

xmlns:tools=“http://schemas.android.com/tools”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

tools:context=".MainActivity">

<EditText

android:id="@+id/editText"

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_alignParentEnd=“true”

android:text="@string/firstTest"

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toTopOf=“parent” />

<Button

android:id="@+id/btn"

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:text="@string/generate_btn"

app:layout_constraintTop_toBottomOf="@+id/editText"

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintStart_toStartOf=“parent”/>

<ImageView

android:id="@+id/imageView"

android:layout_width=“256dp”

android:layout_height=“256dp”

android:layout_centerInParent=“true”

app:layout_constraintBottom_toBottomOf=“parent”

app:layout_constraintEnd_toEndOf=“parent”

app:layout_constraintStart_toStartOf=“parent”

app:layout_constraintTop_toTopOf=“parent” />

</androidx.constraintlayout.widget.ConstraintLayout>

https://www.baidu.com

生成二维码

  1. AndroidManifest.xml中注册(Android Studio会自动生成)

<application

  1. 要实现二维码的生成,主要是靠zxing提供的接口和方法,在GenerateQRcode文件中新建一个generateSimpleBitmap方法用于生成二维码,参数为二维码内容、宽和高,并将参数传入到QRCodeWriterencode方法生成BitMatrix对象,创建一个大小为二维码宽*高的数组,根据BitMatrix对象,往其中放置黑白色块,最后使用createBitmap返回一个Bitmap对象

private Bitmap generateSimpleBitmap(String content, int width, int height) {

// 字符串内容判空

if (TextUtils.isEmpty(content)) {

Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();

return null;

}

QRCodeWriter qrCodeWriter = new QRCodeWriter();

Map<EncodeHintType, String> hints = new HashMap<>();

hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);

try {

BitMatrix encode = qrCodeWriter.encode(content, BarcodeFormat.QR_CODE, width, height, hints);

int[] pixels = new int[width * height];

for (int y = 0; y < height; y++) {

for (int x = 0; x < width; x++) {

if (encode.get(x, y)) {

pixels[y * width + x] = 0x00000000;

} else {

pixel
s[y * width + x] = 0xFFFFFFFF;

}

}

}

return Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);

} catch (WriterException e) {

e.printStackTrace();

}

return null;

}

  1. GenerateQRcode文件中的onCreate引用,将结果注入布局的ImageView组件中

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.generate_qrcode);

btn = (Button)findViewById(R.id.btn);

editText = (EditText)findViewById(R.id.editText);

imageView = (ImageView)findViewById(R.id.imageView);

btn.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View view) {

String content = editText.getText().toString();

Bitmap qrBitmap = generateSimpleBitmap(content,256, 256);

imageView.setImageBitmap(qrBitmap);

}

});

}

  1. 实现结果

SimpleBitmap

修改二维码颜色


  1. 在以上基础上,在GenerateQRcode文件中新建一个generateQRCodeBitmap方法,相比generateSimpleBitmap,参数多了两个颜色,一个为原二维码黑点的颜色,另一个为原来白色的背景颜色

public Bitmap generateQRCodeBitmap(String content, int width, int height,

int color_point, int color_back) {

//字符串内容判空

if (TextUtils.isEmpty(content)) {

Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();

return null;

}

Map<EncodeHintType, String> hints = new HashMap<>();

//格式utf-8

hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);

//空白边距设置

hints.put(EncodeHintType.MARGIN, “1”);

try {

//生成BitMatrix对象encode

BitMatrix encode = new QRCodeWriter().encode(content, BarcodeFormat.QR_CODE, width, height, hints);

//生成宽*高的像素数组

int[] pixels = new int[width * height];

for (int y = 0; y < height; y++) {

for (int x = 0; x < width; x++) {

if (encode.get(x, y)) {

pixels[y * width + x] = color_point;

} else {

pixels[y * width + x] = color_back;

}

}

}

return Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);

} catch (WriterException e) {

e.printStackTrace();

return null;

}

}

  1. 将上面简单二维码的引用改为

Bitmap qrBitmap = generateQRCodeBitmap(content, 256, 256, Color.BLUE, Color.GREEN);

  1. 实现结果

QRCodeBitmap1

在二维码中心添加logo图片


  1. GenerateQRcode文件中新增一个addLogo方法,三个参数分别是原二维码、logo图片以及logo在二维码中占据的比例

private static Bitmap addLogo(Bitmap srcBitmap, Bitmap logoBitmap, float logoPercent){

if(srcBitmap == null){

return null;

}

if(logoBitmap == null){

return srcBitmap;

}

if(logoPercent < 0F || logoPercent > 1F){

logoPercent = 0.2F;

}

int srcWidth = srcBitmap.getWidth();

int srcHeight = srcBitmap.getHeight();

int logoWidth = logoBitmap.getWidth();

int logoHeight = logoBitmap.getHeight();

Bitmap bitmap = Bitmap.createBitmap(srcWidth, srcHeight, Bitmap.Config.ARGB_8888);

float scaleWidth = srcWidth * logoPercent / logoWidth;

float scaleHeight = srcHeight * logoPercent / logoHeight;

Canvas canvas = new Canvas(bitmap);

canvas.drawBitmap(srcBitmap, 0, 0, null);

canvas.scale(scaleWidth, scaleHeight, srcWidth/2, srcHeight/2);

canvas.drawBitmap(logoBitmap, srcWidth/2 - logoWidth/2, srcHeight/2 - logoHeight/2, null);

return bitmap;

}

  1. 修改刚才的generateQRCodeBitmap方法,添加两个参数,一个是logo图片,一个是所占比例,取值范围[0,1],并在其中引用addLogo方法

public Bitmap generateQRCodeBitmap(String content, int width, int height,

int color_point, int color_back,

Bitmap logoBitmap, float logoPercent) {

// 字符串内容判空

if (TextUtils.isEmpty(content)) {

Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();

return null;

}

Map<EncodeHintType, String> hints = new HashMap<>();

//格式utf-8

hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);

//空白边距设置

hints.put(EncodeHintType.MARGIN, “1”);

try {

//生成BitMatrix对象encode

BitMatrix encode = new QRCodeWriter().encode(content, BarcodeFormat.QR_CODE, width, height, hints);

//生成宽*高的像素数组

int[] pixels = new int[width * height];

for (int y = 0; y < height; y++) {

for (int x = 0; x < width; x++) {

if (encode.get(x, y)) {

pixels[y * width + x] = color_point;

} else {

pixels[y * width + x] = color_back;

}

}

}

Bitmap bitmap = Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);

if(logoBitmap != null){

return addLogo(bitmap, logoBitmap, logoPercent);

}

return bitmap;

} catch (WriterException e) {

e.printStackTrace();

return null;

}

}

  1. 从网上下载一张图片,存放在drawable文件中,然后在GenerateQRcode文件中的onCreate引用,将结果注入布局的ImageView组件中

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.generate_qrcode);

btn = (Button)findViewById(R.id.btn);

editText = (EditText)findViewById(R.id.editText);

imageView = (ImageView)findViewById(R.id.imageView);

btn.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View view) {

String content = editText.getText().toString();

Resources res = getResources();

Bitmap logoBitmap= BitmapFactory.decodeResource(res,R.drawable.cat);

//Bitmap qrBitmap = generateSimpleBitmap(content,256, 256);

Bitmap qrBitmap = generateQRCodeBitmap(content, 256, 256, Color.BLUE, Color.GREEN, logoBitmap, 0.2F);

imageView.setImageBitmap(qrBitmap);

}

});

}

  1. 实现结果

QRCodeBitmap2

GenerateQRcode完整代码


package com.example.qrcodetest;

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 android.content.res.Resources;

import android.graphics.BitmapFactory;

import android.graphics.Canvas;

import android.graphics.Color;

import android.os.Bundle;

import android.graphics.Bitmap;

import android.text.TextUtils;

import android.view.View;

import android.widget.Button;

import android.widget.EditText;

import android.widget.ImageView;

import android.widget.Toast;

import androidx.appcompat.app.AppCompatActivity;

import java.util.HashMap;

import java.util.Map;

public class GenerateQRcode extends AppCompatActivity {

private Button btn;

private ImageView imageView;

private EditText editText;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.generate_qrcode);

btn = (Button)findViewById(R.id.btn);

editText = (EditText)findViewById(R.id.editText);

imageView = (ImageView)findViewById(R.id.imageView);

btn.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View view) {

String content = editText.getText().toString();

Resources res = getResources();

Bitmap logoBitmap= BitmapFactory.decodeResource(res,R.drawable.cat);

//Bitmap qrBitmap = generateSimpleBitmap(content,256, 256);

Bitmap qrBitmap = generateQRCodeBitmap(content, 256, 256, Color.BLUE, Color.GREEN, logoBitmap, 0.2F);

imageView.setImageBitmap(qrBitmap);

}

});

}

private Bitmap generateSimpleBitmap(String content, int width, int height) {

// 字符串内容判空

if (TextUtils.isEmpty(content)) {

Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();

return null;

}

QRCodeWriter qrCodeWriter = new QRCodeWriter();

Map<EncodeHintType, String> hints = new HashMap<>();

hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);

try {

BitMatrix encode = qrCodeWriter.encode(content, BarcodeFormat.QR_CODE, width, height, hints);

int[] pixels = new int[width * height];

for (int y = 0; y < height; y++) {

for (int x = 0; x < width; x++) {

if (encode.get(x, y)) {

pixels[y * width + x] = 0x00000000;//黑色

} else {

pixels[y * width + x] = 0xFFFFFFFF;//白色

}

}

}

return Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);

} catch (WriterException e) {

e.printStackTrace();

}

return null;

}

public Bitmap generateQRCodeBitmap(String content, int width, int height,

int color_point, int color_back,

Bitmap logoBitmap, float logoPercent) {

// 字符串内容判空

if (TextUtils.isEmpty(content)) {

Toast.makeText(getApplicationContext(),“输入为空!”,Toast.LENGTH_LONG).show();

return null;

}

Map<EncodeHintType, String> hints = new HashMap<>();

//格式utf-8

hints.put(EncodeHintType.CHARACTER_SET, “utf-8”);

//空白边距设置

hints.put(EncodeHintType.MARGIN, “1”);

try {

//生成BitMatrix对象encode

BitMatrix encode = new QRCodeWriter().encode(content, BarcodeFormat.QR_CODE, width, height, hints);

//生成宽*高的像素数组

int[] pixels = new int[width * height];

for (int y = 0; y < height; y++) {

for (int x = 0; x < width; x++) {

if (encode.get(x, y)) {

pixels[y * width + x] = color_point;

} else {

pixels[y * width + x] = color_back;

}

}

}

Bitmap bitmap = Bitmap.createBitmap(pixels, 0, width, width, height, Bitmap.Config.RGB_565);

if(logoBitmap != null){

return addLogo(bitmap, logoBitmap, logoPercent);

}

return bitmap;

} catch (WriterException e) {

e.printStackTrace();

return null;

}

}

private static Bitmap addLogo(Bitmap srcBitmap, Bitmap logoBitmap, float logoPercent){

if(srcBitmap == null){

return null;

}

if(logoBitmap == null){

return srcBitmap;

}

if(logoPercent < 0F || logoPercent > 1F){

logoPercent = 0.2F;

}

int srcWidth = srcBitmap.getWidth();

int srcHeight = srcBitmap.getHeight();

int logoWidth = logoBitmap.getWidth();

int logoHeight = logoBitmap.getHeight();

Bitmap bitmap = Bitmap.createBitmap(srcWidth, srcHeight, Bitmap.Config.ARGB_8888);

float scaleWidth = srcWidth * logoPercent / logoWidth;

float scaleHeight = srcHeight * logoPercent / logoHeight;

Canvas canvas = new Canvas(bitmap);

canvas.drawBitmap(srcBitmap, 0, 0, null);

canvas.scale(scaleWidth, scaleHeight, srcWidth/2, srcHeight/2);

canvas.drawBitmap(logoBitmap, srcWidth/2 - logoWidth/2, srcHeight/2 - logoHeight/2, null);

return bitmap;

}

}

扫描二维码

================================================================

准备工作


打开真机调试

  • 因为在虚拟机中的摄像头显示是一片黑暗,所以这里使用真机调试

  • 首先在将手机和电脑用USB线连接,USB连接方式选择传输文件

传输文件

  • 然后在手机中打开开发者选项,启动USB调试。开发者选项一般是隐藏的,可在关于手机处多次点击版本号,即可出现。

USB调试

  • 即可在Android Studio调用

真机调试

集成工作

  • build.gradle中添加依赖com.google.zxing:core:3.4.0,此步在前面已添加,因此可不用再次操作

implementation ‘com.google.zxing:core:3.4.0’

  • 打开ZXing的开源地址项目https://github.com/zxing/zxing,在该项目下方的readme中看到,ZXing已将扫描仪的一些功能封装整合到一个包中

Compinents

  • 我们可以下载其中的Android调用,但是,ZXing在封装了多个功能,而我们只需要做到一个简单扫码的功能,因此此处选择网上大神们从其中挑出并修改整合好的扫码的一些基本功能

  • 参考链接https://github.com/ahuyangdong/QrCodeDemo4,下载备用

  • 来到下载文件处,获取zxing内容,复制进自己的项目

在这里插入图片描述

在这里插入图片描述

  • 获得音频文件,直接将raw放入自己项目中的res文件下(也可自己去别处下载)

音频文件

音频文件

  • 拷贝attrs.xml、colors.xml以及ids.xml三个文件

在这里插入图片描述

  • 获得图片

获得图片

获得图片

  • 拷贝导航栏布局toolbar_scanner.xml以及扫码界面的布局activity_scanner.xml

布局文件

821.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQyMzMzMTg2,size_16,color_FFFFFF,t_70)

  • 拷贝attrs.xml、colors.xml以及ids.xml三个文件

在这里插入图片描述

  • 获得图片

获得图片

获得图片

  • 拷贝导航栏布局toolbar_scanner.xml以及扫码界面的布局activity_scanner.xml

布局文件

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
©️2022 CSDN 皮肤主题:深蓝海洋 设计师:CSDN官方博客 返回首页
评论

打赏作者

普通网友

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

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

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

打赏作者

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

抵扣说明:

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

余额充值