解决桌面设备二维码快速识别的工具-ClipQR
前言
不知道你有没有遇到过这样的场景:在电脑上看到一个二维码,想要扫码识别内容,却不得不——
- 把图片保存下来
- 通过微信/QQ发到手机上,或者找第三方网站工具去解析
- 再重新发回电脑上打开链接
整个流程非常繁琐,让人烦躁。为什么我们不能直接在桌面上识别二维码呢?带着这个痛点,我开发了 ClipQR——一个专门为桌面设备设计的二维码快速识别工具。
需求与灵感
日常使用中,二维码无处不在,但当二维码出现在电脑屏幕上时,我们依旧不得不依赖手机或者第三方网页工具完成识别。这明明应该是桌面应用就能搞定的事情!所以我决定自己动手,丰衣足食
应用截图
演示视频
clipqr.needhelp.icu/%E6%BC%94%E…
技术选型
为什么选择 Tauri?
在选择技术栈时,我考虑了几个方案:
| 方案 | 优点 | 缺点 |
|---|---|---|
| Electron | 生态成熟,前端开发友好 | 体积大(动辄上百MB),内存占用高 |
| 原生开发 (GTK/Qt) | 性能好,体积小 | 开发效率低,跨平台适配麻烦 |
| Tauri | 体积小(~10MB),Rust 性能好,跨平台,前端灵活 | 生态相对较新,但已经足够成熟 |
最终我选择了 Tauri 2.x,它完美符合我的需求:
编译后的应用只有不到 10MB,相比 Electron 小一个数量级,Rust 后端性能强劲,二维码解析几乎是瞬间完成,可以继续使用我熟悉的 Vue 做前端开发,这点最重要,跨平台支持,Linux/macOS/Windows 一套代码编译
二维码解析库选择
选择 rqrr,纯 Rust 无依赖,编译简单,解析速度快,对于普通二维码识别率完全够用。
整体架构选择
采用 Vue 前端 + Rust 后端 的经典 Tauri 架构:
前端负责交互展示:剪贴板读取按钮、拖拽区域、结果展示,后端负责图片处理和二维码解析:利用 rqrr 进行识别,通过 Tauri 的命令调用机制前后端通信
整体架构流程
graph TD
A[前端 Vue UI] --> B[用户操作]
B --> C{输入方式}
C -->|剪贴板图片| D[读取 RGBA 数据]
C -->|拖拽文件| E[读取图片文件]
C -->|选择文件| F[文件对话框]
D --> G[Rust 后端]
E --> G[Rust 后端]
F --> G[Rust 后端]
G --> H[转换为灰度图]
H --> I[rqrr 二维码解析]
I --> J{解析成功?}
J -->|是| K[返回结果给前端]
J -->|否| L[显示错误提示]
K --> M[展示结果给用户]
M --> N[一键复制/打开链接]
关键代码展示
前端:从剪贴板读取图片
前端通过 Tauri 的 clipboard-manager 插件读取剪贴板中的图片,然后将 RGBA 数据传给 Rust 后端解析:
export async function parseClipboardImage(): Promise<string | null> {
const { readImage } = await import("@tauri-apps/plugin-clipboard-manager");
const image = await readImage();
const { width, height } = await image.size();
const rgbaData = await image.rgba();
const result = await invoke<string | null>("decode_qr", {
rgba: Array.from(rgbaData),
width,
height,
});
return result;
}
对应的按钮组件很简单:
<template>
<div class="w-full flex flex-col items-center space-y-4">
<button @click="() => handleRead().catch((err) => console.error(err))" class="...">
读取剪贴板
</button>
<div v-if="qrResult" class="...">
<div class="...">识别结果:</div>
<div class="...">{{ qrResult }}</div>
<button @click="copyToClipboard" class="...">
{{ copied ? "已复制" : "复制" }}
</button>
</div>
</div>
</template>
<script setup lang="ts">
import { ref } from "vue";
import { parseClipboardImage, copyToClipboard as copyToClipboardUtil } from "../utils/qr";
const qrResult = ref<string | null>(null);
const copied = ref(false);
const copyToClipboard = async () => {
if (!qrResult.value) return;
await copyToClipboardUtil(qrResult.value);
copied.value = true;
setTimeout(() => {
copied.value = false;
}, 2000);
};
const handleRead = async () => {
qrResult.value = null;
try {
const result = await parseClipboardImage();
qrResult.value = result;
} catch (e) {
qrResult.value = "读取失败,剪贴板中没有图片";
}
};
</script>
后端:Rust 二维码解析逻辑
我不会Rust,
我不太会,所以AI代劳了,rust编译器的负反馈带来的正循环倒是方便ai优化迭代
Rust 后端接收 RGBA 数据,转换为灰度图后使用 rqrr 解析:
use image::{ImageBuffer, Luma};
use rqrr::PreparedImage;
#[tauri::command]
fn decode_qr(rgba: Vec<u8>, width: u32, height: u32) -> Option<String> {
// Convert RGBA to 8-bit grayscale for rqrr
let mut pixels = Vec::with_capacity((width * height) as usize);
for chunk in rgba.chunks(4) {
// RGBA -> luminance: 0.299*R + 0.587*G + 0.114*B
let r = chunk[0] as f32;
let g = chunk[1] as f32;
let b = chunk[2] as f32;
let gray = (0.299 * r + 0.587 * g + 0.114 * b) as u8;
pixels.push(gray);
}
// Create grayscale image
let img_buffer = ImageBuffer::<Luma<u8>, Vec<u8>>::from_vec(width, height, pixels)
.expect("Failed to create image buffer");
// Prepare image for QR detection
let mut prepared = PreparedImage::prepare(img_buffer);
let grids = prepared.detect_grids();
// Try to decode each detected grid
for grid in grids {
if let Ok((_, content)) = grid.decode() {
return Some(content);
}
}
None
}
对于文件直接读取的情况,我们还需要验证文件类型:
/// Check if file is an image based on extension and magic numbers
fn is_image_file(path: &str, data: &[u8]) -> bool {
// Check extension
let lower = path.to_lowercase();
let has_image_ext = lower.ends_with(".png")
|| lower.ends_with(".jpg")
|| lower.ends_with(".jpeg")
|| lower.ends_with(".gif")
|| lower.ends_with(".bmp")
|| lower.ends_with(".webp")
|| lower.ends_with(".ico");
if !has_image_ext {
return false;
}
// Check magic number
if data.len() < 4 {
return false;
}
match &data[0..4] {
[0x89, 0x50, 0x4E, 0x47] => true, // PNG
[0xFF, 0xD8, 0xFF, _] => true, // JPEG
[0x47, 0x49, 0x46, 0x38] => true, // GIF
[0x42, 0x4D, _, _] => true, // BMP
[0x52, 0x49, 0x46, 0x46] => true, // WEBP
[0x00, 0x00, 0x01, 0x00] => true, // ICO
_ => false,
}
}
#[tauri::command]
fn decode_qr_from_file(path: String) -> Option<String> {
match std::fs::read(&path) {
Ok(data) => {
if !is_image_file(&path, &data) {
return None;
}
match image::load_from_memory(&data) {
Ok(img) => {
let gray_img = img.to_luma8();
let mut prepared = rqrr::PreparedImage::prepare(gray_img);
let grids = prepared.detect_grids();
for grid in grids {
if let Ok((_, content)) = grid.decode() {
return Some(content);
}
}
None
}
Err(e) => {
eprintln!("Failed to load image from {}: {}", path, e);
None
}
}
}
Err(e) => {
eprintln!("Failed to read file {}: {}", path, e);
None
}
}
}
拖拽文件处理
Tauri 2.x 有自己的拖拽事件系统,我们需要使用 Tauri 提供的事件而不是原生 DOM 事件:
详情可以回顾我之前的文章 解决Tauri2.x拖拽事件问题
import { listen } from "@tauri-apps/api/event";
import { parseFile, processQrContent } from "./qr";
export interface DragListeners {
onResult: (result: string | null) => void;
onDragStateChange: (isDragging: boolean) => void;
}
export async function initFileDrop({
onResult,
onDragStateChange,
}: DragListeners): Promise<Array<() => void>> {
const unlistens: Array<() => void> = [];
unlistens.push(
await listen("tauri://drag-drop", async (event) => {
onDragStateChange(false);
const { paths } = event.payload as { paths: string[] };
if (paths.length === 0) return;
const filePath = paths[0];
try {
const result = await parseFile(filePath);
if (result) {
processQrContent(result);
}
onResult(result);
} catch (e) {
console.error("解析失败:", e);
onResult("解析失败: " + e);
}
}),
);
unlistens.push(
await listen("tauri://drag-enter", () => {
onDragStateChange(true);
}),
);
unlistens.push(
await listen("tauri://drag-leave", () => {
onDragStateChange(false);
}),
);
return unlistens;
}
在主组件中初始化拖拽监听:
import { initFileDrop } from "./utils/drag";
let unlistens: Array<() => void> = [];
onMounted(async () => {
unlistens = await initFileDrop({
onResult: (result) => {
if (result) {
qrResult.value = result;
} else {
qrResult.value = "未找到二维码";
}
},
onDragStateChange: (state) => {
isDragging.value = state;
},
});
});
onUnmounted(() => {
unlistens.forEach((unlisten) => unlisten());
});
使用方式
ClipQR 提供三种识别方式:
- 剪贴板识别:按下
Ctrl+C复制二维码图片,点击「读取剪贴板」即可识别 - 拖拽识别:直接将图片文件拖拽到窗口中
- 文件选择:点击「选择文件」按钮从文件管理器选择
事实上,我更侧重于托盘菜单带来的快速解析,所以前端更像是手动化的。
总结与下载
ClipQR 解决了我日常使用中的一个小痛点,让桌面二维码识别变得便捷。整个项目代码不多,但是通过 Tauri + Rust + Vue 的组合,得到了一个体积小、速度快的跨平台应用。
如果你也有同样的痛点,欢迎下载使用:
GitHub 地址: github.com/xingwxg/Cli…
宣传页 地址: ClipQR
下载可以在 Release 页面找到对应平台的安装包:github.com/xingwxg/Cli…
欢迎 Star,欢迎提 Issue!
解决桌面设备二维码快速识别的工具-ClipQR 作者:xingwangzhe
本文链接: xingwangzhe.fun/posts/clipq…
本文采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。