添加新中新二代证读取器支持

This commit is contained in:
yuchenglong
2025-11-20 10:27:20 +08:00
parent 8aa5f7802f
commit 2406d600ef
27 changed files with 356 additions and 67 deletions

170
electron/idcard-worker.js Normal file
View File

@@ -0,0 +1,170 @@
const { parentPort } = require("worker_threads");
const koffi = require("koffi");
const path = require("path");
const iconv = require("iconv-lite");
// 定义结构体
const IDCardData = koffi.struct("IDCardData", {
Name: koffi.array("char", 32),
Sex: koffi.array("char", 6),
Nation: koffi.array("char", 20),
Born: koffi.array("char", 18),
Address: koffi.array("char", 72),
IDCardNo: koffi.array("char", 38),
GrantDept: koffi.array("char", 32),
UserLifeBegin: koffi.array("char", 18),
UserLifeEnd: koffi.array("char", 18),
reserved: koffi.array("char", 38),
PhotoFileName: koffi.array("char", 255),
});
let lib = null;
let running = false;
// 解码 GBK 字符串
function decodeGBK(buffer) {
// 找到第一个 null 字节 (0x00) 的位置
let nullIndex = -1;
for (let i = 0; i < buffer.length; i++) {
if (buffer[i] === 0) {
nullIndex = i;
break;
}
}
// 如果找到了 null截取前面的部分否则使用整个 buffer
const validBuffer = nullIndex !== -1 ? buffer.slice(0, nullIndex) : buffer;
return iconv.decode(validBuffer, "gbk").trim();
}
// 加载 DLL
function loadDll() {
if (lib) return true;
try {
const fs = require("fs");
const dllNames = ["Syn_IDCardRead.dll", "SynIDCardRead.dll"];
// 确定 DLL 目录:优先检查生产环境 resources/xzx否则使用开发环境 resources/xzx
let dllDir = path.join(process.cwd(), "resources", "xzx");
if (process.resourcesPath) {
const prodDir = path.join(process.resourcesPath, "xzx");
if (fs.existsSync(prodDir)) {
dllDir = prodDir;
}
}
let dllPath = null;
for (const name of dllNames) {
const p = path.join(dllDir, name);
if (fs.existsSync(p)) {
dllPath = p;
break;
}
}
if (!dllPath) {
throw new Error(`DLL not found in ${dllDir}`);
}
lib = koffi.load(dllPath);
return {
Syn_FindUSBReader: lib.func("int Syn_FindUSBReader()"),
Syn_USBOpenPort: lib.func("int Syn_USBOpenPort(int)"),
Syn_USBClosePort: lib.func("int Syn_USBClosePort(int)"),
Syn_USBStartFindIDCard: lib.func(
"int Syn_USBStartFindIDCard(int, uint8*, int)"
),
Syn_USBSelectIDCard: lib.func(
"int Syn_USBSelectIDCard(int, uint8*, int)"
),
Syn_ReadMsg: lib.func("int Syn_ReadMsg(int, int, _Out_ IDCardData*)"),
};
} catch (err) {
parentPort.postMessage({
type: "error",
payload: `Failed to load DLL: ${err.message}`,
});
return null;
}
}
async function startListen() {
if (running) return;
running = true;
const api = loadDll();
if (!api) {
running = false;
return;
}
try {
let port = api.Syn_FindUSBReader();
if (port <= 0) port = 1001;
const openRes = api.Syn_USBOpenPort(port);
if (openRes !== 0) {
parentPort.postMessage({
type: "error",
payload: `Open port ${port} failed: ${openRes}`,
});
running = false;
return;
}
const iin = Buffer.alloc(4);
const sn = Buffer.alloc(8);
// IDCardData 结构体大小计算: 32+6+20+18+72+38+32+18+18+38+255 = 547 字节
// koffi 会自动处理结构体内存分配
while (running) {
// 寻卡
api.Syn_USBStartFindIDCard(port, iin, 0);
// 选卡
api.Syn_USBSelectIDCard(port, sn, 0);
// 读卡
const data = {}; // koffi 输出对象
const ret = api.Syn_ReadMsg(port, 0, data);
if (ret === 0 && data) {
const payload = {
name: decodeGBK(Buffer.from(data.Name)),
sex: decodeGBK(Buffer.from(data.Sex)),
nation: decodeGBK(Buffer.from(data.Nation)),
born: decodeGBK(Buffer.from(data.Born)),
address: decodeGBK(Buffer.from(data.Address)),
id_card_no: decodeGBK(Buffer.from(data.IDCardNo)),
grant_dept: decodeGBK(Buffer.from(data.GrantDept)),
life_begin: decodeGBK(Buffer.from(data.UserLifeBegin)),
life_end: decodeGBK(Buffer.from(data.UserLifeEnd)),
photo_path: decodeGBK(Buffer.from(data.PhotoFileName)),
};
parentPort.postMessage({ type: "data", payload });
// 读到卡后暂停一下,避免重复读取太快
await new Promise((r) => setTimeout(r, 500));
} else {
// 没读到卡,稍微等待
await new Promise((r) => setTimeout(r, 150));
}
}
api.Syn_USBClosePort(port);
} catch (err) {
parentPort.postMessage({
type: "error",
payload: `Worker error: ${err.message}`,
});
running = false;
}
}
parentPort.on("message", (msg) => {
if (msg === "start") {
startListen();
} else if (msg === "stop") {
running = false;
}
});

View File

@@ -1,42 +1,95 @@
const { app, BrowserWindow } = require("electron");
const path = require("path");
const { app, BrowserWindow, ipcMain } = require('electron');
const path = require('path');
const { Worker } = require('worker_threads');
let idCardWorker = null;
let mainWindow = null;
function createIdCardWorker() {
if (idCardWorker) return;
idCardWorker = new Worker(path.join(__dirname, 'idcard-worker.js'));
idCardWorker.on('message', (msg) => {
if (!mainWindow) return;
if (msg.type === 'data') {
mainWindow.webContents.send('idcard-data', { payload: msg.payload });
} else if (msg.type === 'error') {
mainWindow.webContents.send('idcard-error', { payload: msg.payload });
}
});
idCardWorker.on('error', (err) => {
console.error('Worker error:', err);
});
idCardWorker.on('exit', (code) => {
if (code !== 0) console.error(`Worker stopped with exit code ${code}`);
idCardWorker = null;
});
}
function createWindow() {
const win = new BrowserWindow({
width: 1080,
height: 1920,
webPreferences: {
preload: path.join(__dirname, "preload.js"),
preload: path.join(__dirname, 'preload.js'),
nodeIntegration: false,
contextIsolation: true,
},
});
mainWindow = win;
// 开发环境下加载 Vite 开发服务器地址
// 生产环境下加载打包后的 index.html
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>¼<EFBFBD><EFBFBD><EFBFBD> Vite <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ַ
// <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>¼<EFBFBD><EFBFBD>ش<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> index.html
const isDev = !app.isPackaged;
if (isDev) {
win.loadURL("http://localhost:5173");
// 打开开发者工具
win.loadURL('http://localhost:5173');
// <EFBFBD>򿪿<EFBFBD><EFBFBD><EFBFBD><EFBFBD>߹<EFBFBD><EFBFBD><EFBFBD>
win.webContents.openDevTools();
} else {
win.loadFile(path.join(__dirname, "../dist/index.html"));
win.loadFile(path.join(__dirname, '../dist/index.html'));
}
}
app.whenReady().then(() => {
createWindow();
createIdCardWorker();
app.on("activate", () => {
// IPC <20><><EFBFBD><EFBFBD>
ipcMain.handle('start_idcard_listen', () => {
if (idCardWorker) {
idCardWorker.postMessage('start');
} else {
createIdCardWorker();
idCardWorker.postMessage('start');
}
return 'started';
});
ipcMain.handle('stop_idcard_listen', () => {
if (idCardWorker) {
idCardWorker.postMessage('stop');
}
return 'stopped';
});
app.on('activate', () => {
if (BrowserWindow.getAllWindows().length === 0) {
createWindow();
}
});
});
app.on("window-all-closed", () => {
if (process.platform !== "darwin") {
app.on('window-all-closed', () => {
if (idCardWorker) {
idCardWorker.terminate();
}
if (process.platform !== 'darwin') {
app.quit();
}
});

View File

@@ -1,6 +1,12 @@
const { contextBridge, ipcRenderer } = require("electron");
const { contextBridge, ipcRenderer } = require('electron');
contextBridge.exposeInMainWorld("electronAPI", {
// 在这里暴露安全的 API 给渲染进程
// example: sendMessage: (message) => ipcRenderer.send('message', message)
contextBridge.exposeInMainWorld('electronAPI', {
startIdCardListen: () => ipcRenderer.invoke('start_idcard_listen'),
stopIdCardListen: () => ipcRenderer.invoke('stop_idcard_listen'),
onIdCardData: (callback) => ipcRenderer.on('idcard-data', (event, value) => callback(value)),
onIdCardError: (callback) => ipcRenderer.on('idcard-error', (event, value) => callback(value)),
removeIdCardListeners: () => {
ipcRenderer.removeAllListeners('idcard-data');
ipcRenderer.removeAllListeners('idcard-error');
}
});