阿里云服务器免费领卷啦。

捡代码论坛-最全的游戏源码下载技术网站!

 找回密码
 立 即 注 册

QQ登录

只需一步,快速开始

搜索
关于源码区的附件失效或欺骗帖, 处理办法
查看: 2105|回复: 0

Github项目解析(十)-->几行代码快速集成二维码扫描库

[复制链接]

4208

主题

210

回帖

12万

积分

管理员

管理员

Rank: 9Rank: 9Rank: 9

积分
126189
QQ
发表于 2016-8-24 22:05:20 | 显示全部楼层 |阅读模式

上一篇文章中我们讲解了activity切换动画相关的知识点,这里的切换动画指的是是activity跳转时的动画效果。在上一篇文章中我们总结了有五种方式实现activity切换时实现动画效果。并且依次的介绍一下每种实现activity切换动画效果的实现方式,以及效果,具体更多关于Activity切换动画的知识点可参考我的:Github项目解析(九)–>实现Activity跳转动画的五种方式

而本文将讲解一下我最近写的一个快速集成二维码扫描库,这里需要说明的是其核心的实现扫描的功能,是通过调用ZXing库实现的。内部App中使用到了二维码扫描功能,但是网上找了一些关于二维码扫描的例子,只是我在集成的时候发现通过Android studio集成zxing二维码库不是特别方便,由于我就有了将其制作成标准库的想法,也就有了本文即快速集成二维码扫描库。

本文的项目地址是在:android-zxingLibrary,欢迎star和follow。

使用说明

  • 可打开默认二维码扫描页面
  • 支持对图片Bitmap的扫描功能
  • 支持对UI的定制化操作
  • 支持对条形码的扫描功能
  • 支持生成二维码操作
  • 支持控制闪光灯开关

使用方式:

  • 集成默认的二维码扫描页面

在具体介绍该扫描库之前我们先看一下其具体的使用方式,看看是不是几行代码就可以集成二维码扫描的功能。

  • 在module的build.gradle中执行compile操作
compile 'cn.yipianfengye.android:zxing-library:1.8'
  • 1
  • 1
  • 在代码中执行打开扫描二维码界面操作
/**         * 打开默认二维码扫描界面         */        button1.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                Intent intent = new Intent(MainActivity.this, CaptureActivity.class);                startActivityForResult(intent, REQUEST_CODE);            }        });
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

这里的REQUEST_CODE是我们定义的int型常量。

  • 在Activity的onActivityResult方法中接收扫描结果
/**         * 处理二维码扫描结果         */        if (requestCode == REQUEST_CODE) {            //处理扫描结果(在界面上显示)            if (null != data) {                Bundle bundle = data.getExtras();                if (bundle == null) {                    return;                }                if (bundle.getInt(CodeUtils.RESULT_TYPE) == CodeUtils.RESULT_SUCCESS) {                    String result = bundle.getString(CodeUtils.RESULT_STRING);                    Toast.makeText(this, "解析结果:" + result, Toast.LENGTH_LONG).show();                } else if (bundle.getInt(CodeUtils.RESULT_TYPE) == CodeUtils.RESULT_FAILED) {                    Toast.makeText(MainActivity.this, "解析二维码失败", Toast.LENGTH_LONG).show();                }            }        }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

怎么样是不是很简单?下面我们可以来看一下具体的执行效果:

执行效果:

但是这样的话是不是太简单了,如果我想选择图片解析呢?别急,对二维码图片的解析也是支持的

  • 集成对二维码图片的解析功能
  • 调用系统API打开图库

Intent intent = new Intent(Intent.ACTION_GET_CONTENT);intent.addCategory(Intent.CATEGORY_OPENABLE);intent.setType("image/*");startActivityForResult(intent, REQUEST_IMAGE);
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4
  • 在Activity的onActivityResult方法中获取用户选中的图片并调用二维码图片解析API
if (requestCode == REQUEST_IMAGE) {            if (data != null) {                Uri uri = data.getData();                ContentResolver cr = getContentResolver();                try {                    Bitmap mBitmap = MediaStore.Images.Media.getBitmap(cr, uri);//显得到bitmap图片                    CodeUtils.analyzeBitmap(mBitmap, new CodeUtils.AnalyzeCallback() {                        @Override                        public void onAnalyzeSuccess(Bitmap mBitmap, String result) {                            Toast.makeText(MainActivity.this, "解析结果:" + result, Toast.LENGTH_LONG).show();                        }                        @Override                        public void onAnalyzeFailed() {                            Toast.makeText(MainActivity.this, "解析二维码失败", Toast.LENGTH_LONG).show();                        }                    });                    if (mBitmap != null) {                        mBitmap.recycle();                    }                } catch (Exception e) {                    e.printStackTrace();                }            }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26

执行效果

有了默认的二维码扫描界面,也有了对二维码图片的解析,可能有的同学会说如果我想定制化显示UI怎么办呢?没关系也支持滴。

  • 定制化显示扫描UI

由于我们的扫描组件是通过Fragment实现的,所以能够很轻松的实现扫描UI的定制化。

  • 在新的Activity中定义Layout布局文件
<?xml version="1.0" encoding="utf-8"?><FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:id="@+id/activity_second"    android:layout_width="match_parent"    android:layout_height="match_parent">    <Button        android:id="@+id/second_button1"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:text="取消"        android:layout_marginTop="20dp"        android:layout_marginLeft="20dp"        android:layout_marginRight="20dp"        android:layout_marginBottom="10dp"        android:layout_gravity="bottom|center_horizontal"        />    <FrameLayout        android:id="@+id/fl_my_container"        android:layout_width="match_parent"        android:layout_height="match_parent"        ></FrameLayout></FrameLayout>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25

启动id为fl_my_container的FrameLayout就是我们需要替换的扫描组件,也就是说我们会将我们定义的扫描Fragment替换到id为fl_my_container的FrameLayout的位置。而上面的button是我们添加的一个额外的控件,在这里你可以添加任意的控件,各种UI效果等。具体可以看下面在Activity的初始化过程。

  • 在Activity中执行Fragment的初始化操作
@Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_second);        /**         * 执行扫面Fragment的初始化操作         */        CaptureFragment captureFragment = new CaptureFragment();        // 为二维码扫描界面设置定制化界面        CodeUtils.setFragmentArgs(captureFragment, R.layout.my_camera);        captureFragment.setAnalyzeCallback(analyzeCallback);        /**         * 替换我们的扫描控件         */ getSupportFragmentManager().beginTransaction().replace(R.id.fl_my_container, captureFragment).commit();    }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

其中analyzeCallback是我们定义的扫描回调函数,其具体的定义:

/**     * 二维码解析回调函数     */    CodeUtils.AnalyzeCallback analyzeCallback = new CodeUtils.AnalyzeCallback() {        @Override        public void onAnalyzeSuccess(Bitmap mBitmap, String result) {            Intent resultIntent = new Intent();            Bundle bundle = new Bundle();            bundle.putInt(CodeUtils.RESULT_TYPE, CodeUtils.RESULT_SUCCESS);            bundle.putString(CodeUtils.RESULT_STRING, result);            resultIntent.putExtras(bundle);            SecondActivity.this.setResult(RESULT_OK, resultIntent);            SecondActivity.this.finish();        }        @Override        public void onAnalyzeFailed() {            Intent resultIntent = new Intent();            Bundle bundle = new Bundle();            bundle.putInt(CodeUtils.RESULT_TYPE, CodeUtils.RESULT_FAILED);            bundle.putString(CodeUtils.RESULT_STRING, "");            resultIntent.putExtras(bundle);            SecondActivity.this.setResult(RESULT_OK, resultIntent);            SecondActivity.this.finish();        }    };
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26

仔细看的话,你会发现我们调用了CondeUtils.setFragmentArgs方法,该方法主要用于修改扫描界面扫描框与透明框相对位置的,与若不调用的话,其会显示默认的组件效果,而如果调用该方法的话,可以修改扫描框与透明框的相对位置等UI效果,我们可以看一下my_camera布局文件的实现。

<?xml version="1.0" encoding="utf-8"?><FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:app="http://schemas.android.com/apk/res-auto"    android:layout_width="fill_parent"    android:layout_height="fill_parent" >    <SurfaceView        android:id="@+id/preview_view"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        />    <com.uuzuche.lib_zxing.view.ViewfinderView        android:id="@+id/viewfinder_view"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        app:inner_width="200dp"        app:inner_height="200dp"        app:inner_margintop="150dp"        app:inner_corner_color="@color/scan_corner_color"        app:inner_corner_length="30dp"        app:inner_corner_width="5dp"        app:inner_scan_bitmap="@drawable/scan_image"        app:inner_scan_speed="10"        /></FrameLayout>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

上面我们自定义的扫描控件的布局文件,下面我们看一下默认的扫描控件的布局文件:

<?xml version="1.0" encoding="utf-8"?><FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:app="http://schemas.android.com/apk/res-auto"    android:layout_width="fill_parent"    android:layout_height="fill_parent" >    <SurfaceView        android:id="@+id/preview_view"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        />    <com.uuzuche.lib_zxing.view.ViewfinderView        android:id="@+id/viewfinder_view"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        /></FrameLayout>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

可以发现其主要的区别就是在自定义的扫描控件中多了几个自定义的扫描框属性:

<declare-styleable name="innerrect">        <attr name="inner_width" format="dimension"/><!-- 控制扫描框的宽度 -->        <attr name="inner_height" format="dimension"/><!-- 控制扫描框的高度 -->        <attr name="inner_margintop" format="dimension" /><!-- 控制扫描框距离顶部的距离 -->        <attr name="inner_corner_color" format="color" /><!-- 控制扫描框四角的颜色 -->        <attr name="inner_corner_length" format="dimension" /><!-- 控制扫描框四角的长度 -->        <attr name="inner_corner_width" format="dimension" /><!-- 控制扫描框四角的宽度 -->        <attr name="inner_scan_bitmap" format="reference" /><!-- 控制扫描图 -->        <attr name="inner_scan_speed" format="integer" /><!-- 控制扫描速度 -->    </declare-styleable>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

通过以上几个属性我们就可以定制化的显示我们的扫描UI了,比如定制化微信扫描UI:

执行效果

当然了如果以上的以上,你还是对定制化UI方面不太满意,可以直接下载我的项目,然后引入lib-zxing module作为你的module,直接修改其代码。

  • 生成二维码图片
  • 生成带Logo的二维码图片:

/**         * 生成二维码图片         */        button.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                String textContent = editText.getText().toString();                if (TextUtils.isEmpty(textContent)) {                    Toast.makeText(ThreeActivity.this, "您的输入为空!", Toast.LENGTH_SHORT).show();                    return;                }                editText.setText("");                mBitmap = CodeUtils.createImage(textContent, 400, 400, BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher));                imageView.setImageBitmap(mBitmap);            }        });
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 生成不带logo的二维码图片
/**         * 生成不带logo的二维码图片         */        button1.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                String textContent = editText.getText().toString();                if (TextUtils.isEmpty(textContent)) {                    Toast.makeText(ThreeActivity.this, "您的输入为空!", Toast.LENGTH_SHORT).show();                    return;                }                editText.setText("");                mBitmap = CodeUtils.createImage(textContent, 400, 400, null);                imageView.setImageBitmap(mBitmap);            }        });
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 执行效果

  • 支持控制闪光灯
/** * 打开闪光灯 */CodeUtils.isLightEnable(true);/** * 关闭闪光灯 */ CodeUtils.isLightEnable(false);
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

创建二维码库的流程:

  • 创建一个android studio项目,并创建zxingLibrary库;

  • 在zxingLibrary库中实现二维码的扫描操作;

在实现过程中有以下几个注意点:

(1)由于二维码扫描界面是一个Activity,所以需要在AndroidManifest.xml中定义,而我为了减少对项目的依赖,选择了在zxingLibrary中的AndroidManifest.xml中定义该Activity。

<application android:allowBackup="true"        >        <activity            android:configChanges="orientation|keyboardHidden"            android:name="com.uuzuche.lib_zxing.activity.CaptureActivity"            android:screenOrientation="portrait"            android:windowSoftInputMode="stateAlwaysHidden"            android:label="扫描二维码"           android:theme="@style/Theme.AppCompat.NoActionBar"            ></activity>    </application>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

(2)由于扫描操作需要使用摄像头等权限,也在zxingLibrary中的AndroidManifest.xml中申请了部分权限:

<uses-permission android:name="android.permission.CAMERA" />    <uses-permission android:name="android.permission.FLASHLIGHT" />    <uses-feature android:name="android.hardware.camera" />    <uses-feature android:name="android.hardware.camera.autofocus" />    <uses-permission android:name="android.permission.VIBRATE" />    <uses-permission android:name="android.permission.WAKE_LOCK" />    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />    <uses-permission android:name="android.permission.INTERNET" />
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

(3)原来的项目中二维码扫面的部分是作为主项目实现的,部分代码使用了switch(id)的操作,而这样的代码在library中有问题,具体可参考:在Android library中不能使用switch-case语句访问资源ID的原因分析及解决方案,所以我做了修改:

修改之前:

@Override  public void handleMessage(Message message) {    switch (message.what) {      case R.id.decode:        decode((byte[]) message.obj, message.arg1, message.arg2);        break;      case R.id.quit:        Looper.myLooper().quit();        break;    }  }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

修改之后:

@Override  public void handleMessage(Message message) {    if (message.what == R.id.decode) {      decode((byte[]) message.obj, message.arg1, message.arg2);    } else if (message.what == R.id.quit) {      Looper.myLooper().quit();    }  }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 将zxingLibrary库上传至Jcenter中

为了使用compile,所以讲zxingLibrary库上传至了jCenter中,具体的上传过程可参考我的:Github项目解析(二)–>将Android项目发布至JCenter代码库

有同学反应扫描过程中,二维码图片有拉伸的现象,最新的compile已经改正,具体原因可参考:Zxing图片拉伸解决 Android 二维码扫描

总结:

以上就是我实现的这个快速继承二维码扫描库的过程。主要优点就是可以通过android studio快速的继承到我们的项目中,有兴趣的同学可以到github上看一下具体实现。项目地址:android-zxingLibrary


另外对github项目,开源项目解析感兴趣的同学可以参考我的:
github项目解析(一)–>上传Android项目至github
github项目解析(二)–>将Android项目发布至JCenter代码库
github项目解析(三)–>Android内存泄露监测之leakcanary
github项目解析(四)–>动态更改TextView的字体大小
github项目解析(五)–>Android日志框架
github项目解析(六)–>自定义实现ButterKnife框架
github项目解析(七)–>防止按钮重复点击
Github项目解析(八)–>Activity启动过程中获取组件宽高的五种方式
Github项目解析(九)–>实现Activity跳转动画的五种方式



捡代码论坛-最全的游戏源码下载技术网站! - 论坛版权郑重声明:
1、本主题所有言论和图片纯属会员个人意见,与本论坛立场无关
2、本站所有主题由该帖子作者发表,该帖子作者与捡代码论坛-最全的游戏源码下载技术网站!享有帖子相关版权
3、捡代码论坛版权,详细了解请点击。
4、本站所有内容均由互联网收集整理、网友上传,并且以计算机技术研究交流为目的,仅供大家参考、学习,不存在任何商业目的与商业用途。
5、若您需要商业运营或用于其他商业活动,请您购买正版授权并合法使用。 我们不承担任何技术及版权问题,且不对任何资源负法律责任。
6、如无法链接失效或侵犯版权,请给我们来信:jiandaima@foxmail.com

回复

使用道具 举报

*滑块验证:
您需要登录后才可以回帖 登录 | 立 即 注 册

本版积分规则

技术支持
在线咨询
QQ咨询
3351529868

QQ|手机版|小黑屋|捡代码论坛-专业源码分享下载 ( 陕ICP备15015195号-1|网站地图

GMT+8, 2024-4-27 14:46

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表