Android如何实现URL转换成二维码

二维码已经成为我们日常生活中的一个不可获取的产物,火车票上,景区门票,超市付款等等都会有二维码的身影。

本文将实现由URL转换成二维码的过程。

先看一下示例图

从示例图中我们可以清晰地看到,URL被转换成了二维码。

下面跟随我来一起实现这个功能。

导入Google提供的开源库

compile 'com.google.zxing:core:3.3.0'

来讲解一下核心的部分:二维码转换

①生成二维码Bitmap

public static boolean createQRImage(String content, int widthPix, int heightPix, Bitmap logoBm, String filePath) {
    try {
      if (content == null || "".equals(content)) {
        return false;
      }

      //配置参数
      Map<EncodeHintType, Object> hints = new HashMap<>();
      hints.put(EncodeHintType.CHARACTER_SET, "utf-8");
      //容错级别
      hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);
      //设置空白边距的宽度
      hints.put(EncodeHintType.MARGIN, 2); //default is 4

      // 图像数据转换,使用了矩阵转换
      BitMatrix bitMatrix = new QRCodeWriter().encode(content, BarcodeFormat.QR_CODE, widthPix, heightPix, hints);
      int[] pixels = new int[widthPix * heightPix];
      // 下面这里按照二维码的算法,逐个生成二维码的图片,
      // 两个for循环是图片横列扫描的结果
      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保存到文件中再进行读取。直接返回的bitmap是没有任何压缩的,内存消耗巨大!
      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;
  }

③创建二维码文件存储目录

private static 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();
  }

④创建数据库工具类来存储临时数据

public class SPUtil {

  private static final String CONFIG = "config";

  /**
   * 获取SharedPreferences实例对象
   *
   * @param fileName
   */
  private static SharedPreferences getSharedPreference(String fileName) {
    return QRCodeApplication.getInstance().getSharedPreferences(fileName, Context.MODE_PRIVATE);
  }

  /**
   * 保存一个String类型的值!
   */
  public static void putString(String key, String value) {
    SharedPreferences.Editor editor = getSharedPreference(CONFIG).edit();
    editor.putString(key, value).apply();
  }

  /**
   * 获取String的value
   */
  public static String getString(String key, String defValue) {
    SharedPreferences sharedPreference = getSharedPreference(CONFIG);
    return sharedPreference.getString(key, defValue);
  }

}

⑤展示二维码

public static void showThreadImage(final Activity mContext, final String text, final ImageView imageView, final int centerPhoto) {
    String preContent = SPUtil.getString("share_code_content", "");
    if (text.equals(preContent)) {
      String preFilePath = SPUtil.getString("share_code_filePath", "");
      imageView.setImageBitmap(BitmapFactory.decodeFile(preFilePath));

    } else {
      SPUtil.putString("share_code_content", text);
      final String filePath = getFileRoot(mContext) + File.separator + "qr_" + System.currentTimeMillis() + ".jpg";
      SPUtil.putString("share_code_filePath", filePath);

      //二维码图片较大时,生成图片、保存文件的时间可能较长,因此放在新线程中
      new Thread(new Runnable() {
        @Override
        public void run() {
          boolean success = QRCodeUtil.createQRImage(text, 800, 800, BitmapFactory.decodeResource(mContext.getResources(), centerPhoto),
              filePath);

          if (success) {
            mContext.runOnUiThread(new Runnable() {
              @Override
              public void run() {
                imageView.setImageBitmap(BitmapFactory.decodeFile(filePath));
              }
            });
          }
        }
      }).start();
    }
  }

构造一个输入页面的类,使用Bundle通过<key,value>传值(后期会改为MVVM-DataBinding形式)

public class ContentActivity extends AppCompatActivity implements View.OnClickListener {

  private EditText etUrl;

  private Button btnConvert;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_content);

    initView();
  }

  private void initView() {
    etUrl = (EditText) findViewById(R.id.et_url);
    btnConvert = (Button) findViewById(R.id.btn_convert);

    btnConvert.setOnClickListener(this);

  }

  @Override
  public void onClick(View v) {
    switch (v.getId()) {
      case R.id.btn_convert:
        String str_url = "https://" + etUrl.getText().toString();
        Bundle bundle = new Bundle();
        bundle.putString("url", str_url);
        // 当输入框为空时,提示用户
        if (str_url.equals("https://")) {
          Toast.makeText(getApplicationContext(), "输入框不能为空", Toast.LENGTH_SHORT).show();
        } else {
          Intent intent = new Intent(ContentActivity.this, MainActivity.class);
          intent.putExtras(bundle);
          startActivity(intent);
        }
        break;
      default:
        break;
    }
  }
}

将二维码图片展示在页面上(后期会改为MVVM-DataBinding形式)

public class MainActivity extends AppCompatActivity {

  private ImageView iv;

//  private String url = "http://weibo.com/cnwutianhao";

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    String str_url = getIntent().getExtras().getString("url");

    iv = (ImageView) findViewById(R.id.iv_qrcode);

    QRCodeUtil.showThreadImage(this, str_url, iv, R.mipmap.ic_launcher);
  }
}

布局文件

①输入页面(后期会改为DataBinding形式)

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical"
  android:padding="10dp">

  <EditText
    android:id="@+id/et_url"
    android:layout_width="match_parent"
    android:layout_height="50dp"
    android:layout_marginTop="100dp"
    android:hint="请输入网址"
    android:inputType="textUri" />

  <Button
    android:id="@+id/btn_convert"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_alignParentBottom="true"
    android:layout_centerHorizontal="true"
    android:layout_marginBottom="20dp"
    android:text="转换成二维码" />

</RelativeLayout>

②二维码展示页面

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  tools:context="com.tnnowu.android.qrcode.MainActivity">

  <ImageView
    android:id="@+id/iv_qrcode"
    android:layout_width="220dp"
    android:layout_height="220dp"
    android:layout_centerInParent="true"
    android:layout_marginTop="40dp"
    android:background="#FFFFFF" />

</RelativeLayout>

源代码已上传至Github,https://github.com/cnwutianhao/QRCode

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 基于Android实现个性彩色好看的二维码

    我编码的风格,先给大家展示下效果图,亲们感觉效果还不错,很满意的话,请继续往下阅读. 之前呢,也写过用安卓实现二维码生成彩色的二维码和带logo的二维码,也知道可以使用QRCode和ZXing两种方式,然后这一篇呢也是写二维码使用BarcodeFormat.QR_CODE,主要也是看见很多的非常漂亮的二维码,这里呢主要模仿qq的二维码,并且也高仿实现了长按发送给朋友和保存到图库的功能,觉得不错呢就请多支持下,哪里不好呢也可以说出来.好了我们一步一步来. 第一步:简单二维码实现 先来个最简单的二维

  • Android平台生成二维码并实现扫描 & 识别功能

    1.二维码的前世今生 "二维条码/二维码(2-dimensional bar code)是用某种特定的几何图形按一定规律在平面(二维方向上)分布的黑白相间的图形记录数据符号信息的:在代码编制上巧妙地利用构成计算机内部逻辑基础的"0"."1"比特流的概念,使用若干个与二进制相对应的几何形体来表示文字数值信息,通过图象输入设备或光电扫描设备自动识读以实现信息自动处理:它具有条码技术的一些共性:每种码制有其特定的字符集:每个字符占有一定的宽度:具有一定的校验功能

  • Android开发框架之自定义ZXing二维码扫描界面并解决取景框拉伸问题

    先给大家展示下效果图: 扫描内容是下面这张,二维码是用zxing库生成的 由于改了好几个类,还是去年的事都忘得差不多了,所以只能上这个类的代码了,主要就是改了这个CaptureActivity.java package com.zxing.activity; import java.io.IOException; import java.util.Vector; import android.app.Activity; import android.content.Intent; import

  • Android实现二维码扫描和生成的简单方法

    这里简单介绍一下ZXing库.ZXing是一个开放源码的,用Java实现的多种格式的1D/2D条码图像处理库,它包含了联系到其他语言的端口.Zxing可以实现使用手机的内置的摄像头完成条形码的扫描及解码.该项目可实现的条形码编码和解码.目前支持以下格式:UPC-A,UPC-E.EAN-8,EAN-13.39码.93码.ZXing是个很经典的条码/二维码识别的开源类库,以前在功能机上,就有开发者使用J2ME运用ZXing了,不过要支持JSR-234规范(自动对焦)的手机才能发挥其威力. ZXing

  • Android基于google Zxing实现二维码的生成

    最近项目用到了二维码的生成与识别,之前没有接触这块,然后就上网搜了搜,发现有好多这方面的资源,特别是google Zxing对二维码的封装,实现的已经不错了,可以直接拿过来引用,下载了他们的源码后,只做了少少的改动,就是在Demo中增加了长按识别的功能,网上虽然也有长按识别的Demo,但好多下载下来却无法运行,然后总结了一下,加在了下面的Demo中.   下面来介绍这个Demo的主类 public class BarCodeTestActivity extends Activity { priv

  • Android基于google Zxing实现各类二维码扫描效果

    随着微信的到来,二维码越来越火爆,随处能看到二维码,比如商城里面,肯德基,餐厅等等,对于二维码扫描我们使用的是google的开源框架Zxing,我们可以去http://code.google.com/p/zxing/下载源码和Jar包,之前我项目中的二维码扫描功能只实现了扫描功能,其UI真的是其丑无比,一个好的应用软件,其UI界面也要被大众所接纳,不然人家就不会用你的软件啦,所以说应用软件功能和界面一样都很重要,例如微信,相信微信UI被很多应用软件所模仿,我也仿照微信扫描二维码效果进行模仿,虽然

  • iOS和Android用同一个二维码实现跳转下载链接的方法

    前言 最近一个项目需要iOS和安卓使用一个二维码,让扫描的机器自己识别操作系统实现跳转到相应的下载链接.比如iPhone用微信进行扫描就让他跳转appStore的下载页面,安卓机器使用微信扫描就直接跳浏览器下载.但是这二维码还有一个需求就是,用户已经下载了这个app,当用户打开app进入到注册页面时,再次扫描这个二维码时,自动填写邀请码进行注册.那么该如何实现,细节就不说了,直接上代码. 使用js实现,其实代码非常简单. 使用时直接拷贝代码,改掉相应的链接就好. PS:该链接在微信环境打开时还是

  • Android编程实现二维码的生成与解析

    本文实例讲述了Android编程实现二维码的生成与解析.分享给大家供大家参考,具体如下: 直接上代码,代码上面有具体的解析,并且提供jar供下载:二维码Jar包.rar. 根据文本生成对应的二维码: // 生成QR图 private void createImage() { try { // 需要引入core包 QRCodeWriter writer = new QRCodeWriter(); String text = qr_text.getText().toString(); Log.i(T

  • Android上使用ZXing识别条形码与二维码的方法

    目前有越来越多的手机具备自动对焦的拍摄功能,这也意味着这些手机可以具备条码扫描的功能.手机具备条码扫描的功能,可以优化购物流程,快速存储电子名片(二维码)等. 本文所述实例就使用了ZXing 1.6实现条码/二维码识别.ZXing是个很经典的条码/二维码识别的开源类库,早在很久以前,就有开发者在J2ME上使用ZXing了,只不过需要支持JSR-234规范(自动对焦)的手机才能发挥其威力,而目前已经有不少Android手机具备自动对焦的功能. 本文代码运行的结果如下,使用91手机助手截图时,无法截

  • Android 二维码 生成和识别二维码 附源码下载

    今天讲一下目前移动领域很常用的技术--二维码.现在大街小巷.各大网站都有二维码的踪迹,不管是IOS.Android.WP都有相关支持的软件.之前我就想了解二维码是如何工作,最近因为工作需要使用相关技术,所以做了初步了解.今天主要是讲解如何使用ZXing库,生成和识别二维码.这篇文章实用性为主,理论性不会讲解太多,有兴趣可以自己查看源码. 1.ZXing库介绍 这里简单介绍一下ZXing库.ZXing是一个开放源码的,用Java实现的多种格式的1D/2D条码图像处理库,它包含了联系到其他语言的端口

随机推荐