首页 星云 工具 资源 星选 资讯 热门工具
:

PDF转图片 完全免费 小红书视频下载 无水印 抖音视频下载 无水印 数字星空

前端黑科技:使用 JavaScript 实现网页扫码功能

编程知识
2024年08月10日 09:20

在数字化时代,二维码已经渗透到我们生活的方方面面。从移动支付到产品溯源,二维码凭借其便捷性和高效性,成为了信息传递的重要载体。而随着前端技术的不断发展,我们甚至可以使用 JavaScript 在网页端实现二维码扫描功能,为用户提供更加便捷的操作体验。

本文将带您深入了解如何使用 JavaScript 调用摄像头,结合 jsQR 库,以及如何控制闪光灯,最终实现一个功能完善的网页扫码应用。

一、 项目概述

我们将创建一个简单的网页应用,该应用能够:

  1. 调用设备摄像头,获取实时视频流。
  2. 在网页上创建一个扫描区域,用户可以将二维码放置在该区域内进行扫描。
  3. 使用 jsQR 库解码扫描区域内的二维码图像数据,获取二维码内容。
  4. 提供手动输入二维码内容的功能。
  5. 如果设备支持,还可以控制闪光灯的开关,以便在光线不足的情况下进行扫描。

二、 实现步骤

1. HTML 结构

首先,我们需要构建基本的 HTML 结构,包括:

  • <video> 标签:用于展示摄像头捕获的实时视频流。
  • <canvas> 标签:用于绘制视频帧和扫描区域,并从中获取图像数据。
  • <div> 标签:用于创建扫描区域、按钮组等 UI 元素。
<!DOCTYPE html>
<html>
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>扫一扫</title>
  <link rel="stylesheet" href="style.css">
</head>
<body>
  <video autoplay></video>
  <canvas hidden></canvas>

  <div></div>

  <div>
    <button>手动输入</button>
    <button>闪光灯</button>
  </div>

  <script src="https://cdn.jsdelivr.net/npm/jsqr@1.4.0/dist/jsQR.min.js"></script>
  <script src="script.js"></script>
</body>
</html>

2. CSS 样式

为了提升用户体验,我们需要为页面元素添加一些样式:

/* style.css */
body {
  margin: 0;
  overflow: hidden;
}

#video {
  width: 100%;
  height: auto;
  display: block;
}

#overlay {
  position: absolute;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  pointer-events: none;
}

.scan-area {
  border: 3px solid yellow;
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%);
  width: 80%;
  height: 30%;
}

/* ...其他样式 */

3. JavaScript 交互

JavaScript 代码是实现扫码功能的核心部分,主要包括以下几个步骤:

  1. 获取摄像头权限: 使用 navigator.mediaDevices.getUserMedia() 方法请求访问用户的摄像头。

  2. 播放视频流: 将获取到的视频流赋值给 <video> 标签的 srcObject 属性,并调用 video.play() 方法开始播放。

  3. 创建扫描循环: 使用 requestAnimationFrame() 方法创建一个循环,不断地从视频流中获取帧图像,并进行二维码解码。

  4. 绘制视频帧: 在每一帧中,使用 canvas.drawImage() 方法将视频帧绘制到 <canvas> 元素上。

  5. 获取扫描区域图像数据: 使用 canvas.getImageData() 方法获取扫描区域的图像数据。

  6. 解码二维码: 使用 jsQR 库的 jsQR() 方法解码图像数据,如果解码成功,则获取二维码内容。

  7. 处理扫描结果: 对解码后的二维码内容进行处理,例如跳转到链接、显示信息等。

  8. 实现其他功能: 实现手动输入二维码内容和控制闪光灯等功能。

// script.js
const video = document.getElementById('video');
const overlay = document.getElementById('overlay');
const manualInputBtn = document.getElementById('manualInputBtn');
const flashBtn = document.getElementById('flashBtn');
const scanArea = document.querySelector('.scan-area');

let stream;
let scanning = false;
let flashEnabled = false;

// 获取摄像头权限并开始播放视频流
navigator.mediaDevices.getUserMedia({ video: { facingMode: "environment" } })
  .then(s => {
    stream = s;
    video.srcObject = stream;
    video.play();

    // 开始扫描
    scanning = true;
    requestAnimationFrame(scan);
  })
  .catch(err => {
    console.error("无法访问摄像头:", err);
  });

// 扫描二维码
function scan() {
  if (scanning) {
    const canvas = overlay.getContext('2d');
    const videoWidth = video.videoWidth;
    const videoHeight = video.videoHeight;

    // 设置画布大小
    overlay.width = videoWidth;
    overlay.height = videoHeight;

    // 将视频帧绘制到画布上
    canvas.drawImage(video, 0, 0, videoWidth, videoHeight);

    // ...获取扫描区域图像数据

    // 使用 jsQR 库解码二维码
    const code = jsQR(imageData.data, imageData.width, imageData.height);

    // 如果成功解码,则停止扫描并处理结果
    if (code) {
      scanning = false;
      handleScanResult(code.data);
    } else {
      requestAnimationFrame(scan);
    }
  }
}

// 处理扫描结果
function handleScanResult(data) {
  alert("扫描结果:" + data);

  // 这里可以根据扫描结果进行相应的操作,例如跳转到链接或显示信息
}

// 手动输入按钮点击事件
manualInputBtn.addEventListener('click', function() {
  // ...
});

// 闪光灯按钮点击事件
flashBtn.addEventListener('click', function() {
  // ...
});

三、完整代码

<!DOCTYPE html>
<html>
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>扫一扫</title>
  <style>
    body {
      margin: 0;
      overflow: hidden;
    }

    #video {
      width: 100%;
      height: auto;
      display: block;
    }

    #overlay {
      position: absolute;
      top: 0;
      left: 0;
      width: 100%;
      height: 100%;
      pointer-events: none;
    }

    .scan-area {
      border: 3px solid yellow;
      position: absolute;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
      width: 80%;
      height: 30%;
    }

    .btn-group {
      position: absolute;
      bottom: 20px;
      left: 50%;
      transform: translateX(-50%);
      display: flex;
    }

    button {
      background-color: rgba(0, 0, 0, 0.5);
      color: white;
      border: none;
      padding: 10px 20px;
      margin: 0 10px;
      border-radius: 5px;
      font-size: 16px;
      cursor: pointer;
    }
  </style>
</head>
<body>

<video autoplay></video>
<canvas hidden></canvas>

<div></div>

<div>
  <button>手动输入</button>
  <button>闪光灯</button>
</div>

<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/jsqr@1.4.0/dist/jsQR.min.js"></script>
<script>
  const video = document.getElementById('video');
  const overlay = document.getElementById('overlay');
  const manualInputBtn = document.getElementById('manualInputBtn');
  const flashBtn = document.getElementById('flashBtn');
  const scanArea = document.querySelector('.scan-area');

  let stream;
  let scanning = false;
  let flashEnabled = false;

  // 获取摄像头权限并开始播放视频流
  navigator.mediaDevices.getUserMedia({ video: { facingMode: "environment" } })
    .then(function(s) {
      stream = s;
      video.srcObject = stream;
      video.play();

      // 开始扫描
      requestAnimationFrame(scan);
    })
    .catch(function(err) {
      console.error("无法访问摄像头:", err);
    });

  // 扫描二维码
  function scan() {
    if (scanning) {
      const canvas = overlay.getContext('2d');
      const videoWidth = video.videoWidth;
      const videoHeight = video.videoHeight;

      // 设置画布大小
      overlay.width = videoWidth;
      overlay.height = videoHeight;

      // 将视频帧绘制到画布上
      canvas.drawImage(video, 0, 0, videoWidth, videoHeight);

      // 获取扫描区域的坐标和尺寸
      const scanAreaRect = scanArea.getBoundingClientRect();
      const scanAreaX = scanAreaRect.left;
      const scanAreaY = scanAreaRect.top;
      const scanAreaWidth = scanAreaRect.width;
      const scanAreaHeight = scanAreaRect.height;

      // 获取扫描区域的图像数据
      const imageData = canvas.getImageData(scanAreaX, scanAreaY, scanAreaWidth, scanAreaHeight);

      // 使用 jsQR 库解码二维码
      const code = jsQR(imageData.data, imageData.width, imageData.height);

      // 如果成功解码,则停止扫描并处理结果
      if (code) {
        scanning = false;
        handleScanResult(code.data);
      } else {
        requestAnimationFrame(scan);
      }
    }
  }

  // 处理扫描结果
  function handleScanResult(data) {
    alert("扫描结果:" + data);

    // 这里可以根据扫描结果进行相应的操作,例如跳转到链接或显示信息
  }

  // 手动输入按钮点击事件
  manualInputBtn.addEventListener('click', function() {
    const input = prompt("请输入二维码内容:");
    if (input) {
      handleScanResult(input);
    }
  });

  // 闪光灯按钮点击事件
  flashBtn.addEventListener('click', function() {
    if ('torch' in navigator.mediaDevices.getUserMedia({ video: true })) {
      flashEnabled = !flashEnabled;
      stream.getVideoTracks()[0].applyConstraints({
        advanced: [{ torch: flashEnabled }]
      });

      // 更新按钮文本
      flashBtn.textContent = flashEnabled ? '关闭闪光灯' : '闪光灯';
    } else {
      alert('您的设备不支持闪光灯功能。');
    }
  });

  // 开始扫描
  scanning = true;
</script>

</body>
</html> 

四、 总结

通过以上步骤,我们成功地使用 JavaScript 在网页端实现了二维码扫描功能。该功能可以广泛应用于各种场景,例如:

  • 移动支付: 用户可以使用手机扫描网页上的二维码完成支付。
  • 产品溯源: 用户可以扫描产品上的二维码,查看产品信息、生产日期、物流信息等。
  • 活动签到: 用户可以使用手机扫描二维码完成活动签到。

随着 Web 技术的不断发展,相信未来会有更多创新的应用场景出现。

希望本文能够帮助您了解网页扫码功能的实现原理,并激发您探索更多前端黑科技的兴趣。

From:https://www.cnblogs.com/geekcjj/p/18352006
本文地址: http://shuzixingkong.net/article/959
0评论
提交 加载更多评论
其他文章 七夕——程序员独有的表白方式
七夕节,作为中国的传统情人节,虽然历来与诗词歌赋、浪漫约会紧密相连,但在当今的数字时代,程序员们也能以他们独有的方式,用代码和技术的力量来表达爱意。本文简要介绍了七种七夕节程序员常用的表白方式,包括编写定制程序、定制网页网站、二维码情书、游戏表白、数据分析情书、自动化提醒、代码表白;无论您是程序员还
CryptoHouse:由 ClickHouse 和 Goldsky 支持的免费区块链分析服务(ClickHouse 博客)
我们很高兴地宣布 CryptoHouse,在 crypto.clickhouse.com 上可访问,这是一个由 ClickHouse 提供支持的免费区块链分析服务。 https://crypto.clickhouse.com/ 现有的公共区块链分析服务通常需要定时、异步查询,而 ClickHouse
CryptoHouse:由 ClickHouse 和 Goldsky 支持的免费区块链分析服务(ClickHouse 博客) CryptoHouse:由 ClickHouse 和 Goldsky 支持的免费区块链分析服务(ClickHouse 博客) CryptoHouse:由 ClickHouse 和 Goldsky 支持的免费区块链分析服务(ClickHouse 博客)
Spring AI 更新:支持OpenAI的结构化输出,增强对JSON响应的支持
就在昨晚,Spring AI发了个比较重要的更新。由于最近OpenAI推出了结构化输出的功能,可确保 AI 生成的响应严格遵守预定义的 JSON 模式。此功能显着提高了人工智能生成内容在现实应用中的可靠性和可用性。Spring AI 紧随其后,现在也可以对OpenAI的结构化输出完美支持了。 下图展
Spring AI 更新:支持OpenAI的结构化输出,增强对JSON响应的支持
[python] Python并行计算库Joblib使用指北
Joblib是用于高效并行计算的Python开源库,其提供了简单易用的内存映射和并行计算的工具,以将任务分发到多个工作进程中。Joblib库特别适合用于需要进行重复计算或大规模数据处理的任务。Joblib库的官方仓库见:joblib,官方文档见:joblib-doc。 Jolib库安装代码如下: p
[python] Python并行计算库Joblib使用指北 [python] Python并行计算库Joblib使用指北
FFmpeg开发笔记(四十五)使用SRT Streamer开启APP直播推流
​SRT Streamer是一个安卓手机端的开源SRT协议直播推流框架,可用于RTMP直播和SRT直播。SRT Streamer支持的视频编码包括H264、H265等等,支持的音频编码包括AAC、OPUS等等,可谓功能强大的APP直播框架。 相比之下,另一款APP直播框架RTMP Streamer支
FFmpeg开发笔记(四十五)使用SRT Streamer开启APP直播推流 FFmpeg开发笔记(四十五)使用SRT Streamer开启APP直播推流
mongo变更流使用及windows下副本集五分钟搭建
mongodb的变更流解释: 变更流(Change Streams)允许应用程序访问实时数据变更,从而避免事先手动追踪 oplog&#160;的复杂性和风险。应用程序可使用变更流来订阅针对单个集合、数据库或整个部署的所有数据变更,并立即对它们做出响应。由于变更流采用聚合框架,因此,应用程序还可对特定
mongo变更流使用及windows下副本集五分钟搭建 mongo变更流使用及windows下副本集五分钟搭建 mongo变更流使用及windows下副本集五分钟搭建
使用SiliconCloud尝试GraphRag——以《三国演义》为例(手把手教程,适合小白)
本文介绍了使用SiliconCloud尝试GraphRag——以《三国演义》为例(手把手教程,适合小白)。
使用SiliconCloud尝试GraphRag——以《三国演义》为例(手把手教程,适合小白) 使用SiliconCloud尝试GraphRag——以《三国演义》为例(手把手教程,适合小白) 使用SiliconCloud尝试GraphRag——以《三国演义》为例(手把手教程,适合小白)
关键点检测(2)——关键点检测发展历程
关键点检测是计算机视觉领域中 一项重要任务,旨在识别图像或视频中具有特定意义或信息的关键点。如人脸上的鼻子,眼睛,或是关节等。在神经网络中,用于关键点检测的卷积神经网络(Convolutional Neural Networks, CNNs)有许多出名的模型。从DeepPose开始,直到现在的先进方
关键点检测(2)——关键点检测发展历程 关键点检测(2)——关键点检测发展历程 关键点检测(2)——关键点检测发展历程