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

 更新时间:2020年05月28日 08:51:56   作者:tnnowu  
这篇文章主要为大家详细介绍了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

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

相关文章

  • Android实现日夜间模式的深入理解

    Android实现日夜间模式的深入理解

    相信Android的日间/夜间模式切换相信大家在平时使用 APP 的过程中都遇到过,比如知乎、简书中就有相关的模式切换。实现日间/夜间模式切换的方案也有许多种,趁着今天有空来讲一下日间/夜间模式切换的几种实现方案,也可以做一个横向的对比来看看哪种方案最好。
    2016-09-09
  • Android 跨进程模拟按键(KeyEvent )实例详解

    Android 跨进程模拟按键(KeyEvent )实例详解

    这篇文章主要介绍了Android 跨进程模拟按键(KeyEvent )实例详解的相关资料,类似手机遥控器的需求就可以这么做,需要的朋友可以参考下
    2016-11-11
  • Android实现登录注册功能

    Android实现登录注册功能

    这篇文章主要为大家详细介绍了Android实现登录注册功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • android 仿微信demo——注册功能实现(服务端)

    android 仿微信demo——注册功能实现(服务端)

    本篇文章主要介绍了微信小程序-阅读小程序实例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧,希望能给你们提供帮助
    2021-06-06
  • Android日期选择控件使用详解

    Android日期选择控件使用详解

    这篇文章主要为大家详细介绍了Android日期选择控件的使用,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-08-08
  • Android入门教程之Fragment的具体使用详解

    Android入门教程之Fragment的具体使用详解

    Fragment是Android3.0后引入的一个新的API,他出现的初衷是为了适应大屏幕的平板电脑, 当然现在他仍然是平板APP UI设计的宠儿,而且我们普通手机开发也会加入这个Fragment, 我们可以把他看成一个小型的Activity,又称Activity片段
    2021-10-10
  • Android SQLite详解及示例代码

    Android SQLite详解及示例代码

    本文主要讲解Android SQLite,这里对数据库SQLite 做了详细的知识资料整理,并附示例代码和实现效果图,有需要的小伙伴可以参考下
    2016-08-08
  • Android 中 SwipeLayout一个展示条目底层菜单的侧滑控件源码解析

    Android 中 SwipeLayout一个展示条目底层菜单的侧滑控件源码解析

    这篇文章主要介绍了Android 中 SwipeLayout一个展示条目底层菜单的侧滑控件源码解析,需要的朋友可以参考下
    2016-12-12
  • Android bindservice失败解决方法

    Android bindservice失败解决方法

    这篇文章主要介绍了Android bindservice失败解决方法的相关资料,希望通过本文能帮助大家解决这样的问题,需要的朋友可以参考下
    2017-08-08
  • Android实现一个丝滑的自动轮播控件实例代码

    Android实现一个丝滑的自动轮播控件实例代码

    轮播图对大家来说应该再熟悉不过了,下面这篇文章主要给大家介绍了关于Android实现一个丝滑的自动轮播控件的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧
    2018-08-08

最新评论