解决桌面设备二维码快速识别的工具-ClipQR

0 阅读6分钟

解决桌面设备二维码快速识别的工具-ClipQR

前言

不知道你有没有遇到过这样的场景:在电脑上看到一个二维码,想要扫码识别内容,却不得不——

  1. 把图片保存下来
  2. 通过微信/QQ发到手机上,或者找第三方网站工具去解析
  3. 再重新发回电脑上打开链接

整个流程非常繁琐,让人烦躁。为什么我们不能直接在桌面上识别二维码呢?带着这个痛点,我开发了 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 提供三种识别方式:

  1. 剪贴板识别:按下 Ctrl+C 复制二维码图片,点击「读取剪贴板」即可识别
  2. 拖拽识别:直接将图片文件拖拽到窗口中
  3. 文件选择:点击「选择文件」按钮从文件管理器选择

事实上,我更侧重于托盘菜单带来的快速解析,所以前端更像是手动化的。

总结与下载

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 国际许可协议 进行许可。