前端实现扫一扫,扫描二维码(VUE,H5);jsQR,zxing两种方式

文章介绍

        这里使用了jsQR和zxing两种方式,分别在普通的H5和vue中使用,文章附上完整demo和一些注意事项

注意事项

         这里H5也好,vue也好,如果想要部署到服务器上,需要用https协议,否则无法使用。本地启动项目的时候,很多人在vue中无法实现该效果,那是因为需要我们从localhost路径打开,否则也无法使用摄像头

vue 或 H5,使用jsQR

使用jsQR这个JS库

这里附上这个库的地址:https://s3.gendome.net/activity/js/jsQR.js

先把上面这个JS文件下载下来,比如我这里命名为jsQR.js,使用我这样的写法

H5中

<!DOCTYPE html>
<html lang="en">
 
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Document</title>
  <script src="./jsQR.js"></script>
</head>
 
<body>
  <video style="display: none;" id="video"></video>
  <canvas style="width: 100%; height: 100%;" id="canvas"></canvas>
  <canvas style="display: none;" id="2d"></canvas>
</body>
<script type="text/javascript">
  var video = document.createElement("video");
  var canvasElement = document.getElementById("canvas");
  var canvas = canvasElement.getContext("2d");
  // 尝试打开手机上安装后置摄像头
  navigator.mediaDevices.getUserMedia({
    video: { facingMode: "environment" }
  }).then(function (stream) {
    video.srcObject = stream;
    // 阻止IOS视频全屏
    video.setAttribute("playsinline", true);
    video.play();
    requestAnimationFrame(tick);
  });
 
  function tick() {
    if (video.readyState === video.HAVE_ENOUGH_DATA) {
      canvasElement.hidden = false;
 
      canvasElement.height = video.videoHeight;
      canvasElement.width = video.videoWidth;
      canvas.drawImage(video, 0, 0, canvasElement.width, canvasElement.height);
      var imageData = canvas.getImageData(0, 0, canvasElement.width, canvasElement.height);
 
      // QR码解析
      var code = jsQR(
        imageData.data,   // 图像数据
        imageData.width,  // 宽度
        imageData.height, // 高度
        {
          inversionAttempts: "dontInvert",
        }
      );
 
      if (code) {
        console.log(code.data);
      }
    }
    requestAnimationFrame(tick);
  }
</script>
 
</html>

vue中

<template>
  <div>
    <video style="display: none" id="video"></video>
    <canvas style="width: 100vw; margin-top: 13vw" id="canvas"></canvas>
    <canvas style="display: none" id="2d"></canvas> 
  </div>
</template>
<script setup>
import { ref, onMounted} from "vue";
import "@/utils/jsQR.js"; // 添加关闭摄像头的函数
 
const stopMediaTracks = () => {
  if (streams) {
    streams.getTracks().forEach((track) => track.stop());
  }
  video.srcObject = null;
};
const streams = ref(null); // 初始化 stream 变量
 
 
onMounted(() => {
  var video = document.createElement("video");
  var canvasElement = document.getElementById("canvas");
  var canvas = canvasElement.getContext("2d");
  console.log(navigator.mediaDevices);
 
  // 尝试打开手机上安装后置摄像头
  navigator.mediaDevices
    .getUserMedia({
      video: { facingMode: "environment" },
    })
    .then(function (stream) {
      streams.value = stream;
      video.srcObject = stream;
      // 阻止IOS视频全屏
      video.setAttribute("playsinline", true);
      video.play();
      requestAnimationFrame(tick);
    });
 
  function tick() {
    if (video.readyState === video.HAVE_ENOUGH_DATA) {
      canvasElement.hidden = false;
 
      canvasElement.height = video.videoHeight;
      canvasElement.width = video.videoWidth;
      canvas.drawImage(video, 0, 0, canvasElement.width, canvasElement.height);
      var imageData = canvas.getImageData(
        0,
        0,
        canvasElement.width,
        canvasElement.height
      );
 
      // QR码解析
      var code = jsQR(
        imageData.data, // 图像数据
        imageData.width, // 宽度
        imageData.height, // 高度
        {
          inversionAttempts: "dontInvert",
        }
      );
 
      if (code) {
        console.log(code.data);
        alert(code.data);
      }
    }
    requestAnimationFrame(tick);
  } 
});
</script>

Vue.js结合HTML5的WebRTC技术可以实现H5页面调用原生摄像头进行码功能。通常的做法是利用浏览器提供的`navigator.mediaDevices.getUserMedia()` API获取用户的摄像头权限,并通过`getUserMedia()`请求访问设备。然后,你可以使用第三方库如jsqrcode-reader-api或者 ZXing JavaScript 库来识别扫描到的条形码。 以下是个简单的步骤概述: 1. 引入必要的库和模块: ```html <script src="https://cdn.jsdelivr.net/npm/@vue/web-component-wrapper"></script> <script src="https://cdnjs.cloudflare.com/ajax/libs/jsqrcode-reader/1.0.6/jsqrcode.min.js"></script> ``` 2.Vue组件中使用ref绑定相机元素: ```html <template> <video ref="scanner" id="scanner" style="display:none;"></video> </template> ``` 3. 使用`@mediaStream`事件处理视频流,并启动码功能: ```javascript <script> export default { mounted() { navigator.mediaDevices.getUserMedia({ video: true }) .then(stream => { this.$refs.scanner.srcObject = stream; this.$refs.scanner.play(); // 初始化二维码识别实例 const scanner = new JsQR(this.$refs.scanner); scanner.callback = data => { console.log('Scanned:', data.result); // 打印扫描结果 }; }); } } </script> ``` 4. 用户可以选择停止摄像头或者关闭码功能: ```javascript methods: { stopScanner() { this.$refs.scanner.pause(); // 停止摄像头 if (this.$refs.scanner.srcObject) { this.$refs.scanner.srcObject.getTracks().forEach(track => track.stop()); // 关闭视频源 } }, } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值