核价器
Some checks failed
CI / build (18.x) (push) Failing after 18m45s
CI / build (20.x) (push) Failing after 28m56s
CI / deploy-preview (push) Has been skipped
CI / lint (push) Failing after 16m45s
CI / test (push) Successful in 39m19s
CI / security (push) Successful in 26m10s

This commit is contained in:
97694731 2026-06-26 12:01:34 +08:00
parent 50cdfb167b
commit 104780d195
6 changed files with 475 additions and 28 deletions

1
.gitignore vendored
View File

@ -1,3 +1,2 @@
dist/
vite.config.js
text.txt

View File

@ -1,71 +1,71 @@
{
"hash": "a0d5344e",
"browserHash": "906d96e7",
"hash": "780fd6b7",
"browserHash": "11a50aed",
"optimized": {
"@element-plus/icons-vue": {
"src": "../../@element-plus/icons-vue/dist/index.js",
"file": "@element-plus_icons-vue.js",
"fileHash": "b0bff00e",
"fileHash": "e73a6617",
"needsInterop": false
},
"axios": {
"src": "../../axios/index.js",
"file": "axios.js",
"fileHash": "3661d772",
"fileHash": "9ada0434",
"needsInterop": false
},
"crypto-js": {
"src": "../../crypto-js/index.js",
"file": "crypto-js.js",
"fileHash": "42456c05",
"fileHash": "35b4399e",
"needsInterop": true
},
"dayjs": {
"src": "../../dayjs/dayjs.min.js",
"file": "dayjs.js",
"fileHash": "00d3ce60",
"fileHash": "3c458ca9",
"needsInterop": true
},
"element-plus": {
"src": "../../element-plus/es/index.mjs",
"file": "element-plus.js",
"fileHash": "8289c005",
"fileHash": "65c76065",
"needsInterop": false
},
"element-plus/es/locale/lang/zh-cn": {
"src": "../../element-plus/es/locale/lang/zh-cn.mjs",
"file": "element-plus_es_locale_lang_zh-cn.js",
"fileHash": "2b74d0fe",
"fileHash": "05b5b909",
"needsInterop": false
},
"jsbarcode": {
"src": "../../jsbarcode/bin/JsBarcode.js",
"file": "jsbarcode.js",
"fileHash": "5774375a",
"fileHash": "d60441c9",
"needsInterop": true
},
"json-bigint": {
"src": "../../json-bigint/index.js",
"file": "json-bigint.js",
"fileHash": "a76f358e",
"fileHash": "4eb91e5f",
"needsInterop": true
},
"pinia": {
"src": "../../pinia/dist/pinia.mjs",
"file": "pinia.js",
"fileHash": "3c37150a",
"fileHash": "461b0d9b",
"needsInterop": false
},
"vue": {
"src": "../../vue/dist/vue.runtime.esm-bundler.js",
"file": "vue.js",
"fileHash": "c01889f7",
"fileHash": "aa9868f1",
"needsInterop": false
},
"vue-router": {
"src": "../../vue-router/dist/vue-router.mjs",
"file": "vue-router.js",
"fileHash": "ced1274b",
"fileHash": "010be937",
"needsInterop": false
}
},

View File

@ -250,6 +250,34 @@ export class ServiceManager {
return false;
}
}
/**
* 注册服务 exe 路径注册到启动器
* @param {Object} serviceInfo - { id, name, exe_path, args }
*/
async registerService(serviceInfo) {
this.log('info', `注册服务: ${serviceInfo.id || serviceInfo.exe_path}`);
this.log('debug', `POST /api/services`);
try {
const response = await fetch(`${this.baseUrl}/api/services`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(serviceInfo)
});
this.log('debug', `响应状态: ${response.status}`);
const result = await response.json();
this.log('debug', `响应数据: ${JSON.stringify(result)}`);
if (result.code === 0) {
this.log('success', `注册成功: ${result.msg}`);
return result.data || true;
}
this.log('error', `注册失败: ${result.msg}`);
return false;
}
catch (error) {
this.log('error', `注册请求失败: ${error.message}`);
return false;
}
}
// ============================================
// 自动刷新
// ============================================

View File

@ -25,14 +25,16 @@
</el-tooltip>
</span>
</template>
<el-input v-model="dir" placeholder="如 C:\\verifyTool" clearable @clear="dir = ''" />
<el-input v-model="dir" placeholder="如 C:\\verifyTool\\kfz-goods-pricing.exe" clearable @clear="dir = ''" />
<input ref="fileInputRef" type="file" accept=".exe" style="display: none" @change="handleFileSelected" />
<div class="save-bar" style="margin-top: 18px;">
<el-tooltip content="通过自定义协议启动本机程序" placement="top" trigger="click">
<el-button type="primary" @click="handleOpenExe" size="small">打开程序</el-button>
<el-tooltip content="选择本机程序文件" placement="top" trigger="click">
<el-button type="primary" @click="handleBrowseExe" size="small">选择地址</el-button>
</el-tooltip>
<el-tooltip content="通过自定义协议启动本机程序" placement="top" trigger="click">
<el-button type="success" @click="handleOpenExe" size="small">打开程序</el-button>
</el-tooltip>
</div>
</el-form-item>
@ -279,6 +281,7 @@ import { ref, onMounted } from 'vue'
import axios from 'axios'
import { ElMessage, ElMessageBox } from 'element-plus'
import { Delete, Plus, Setting, User, Link, VideoPlay } from '@element-plus/icons-vue'
import { ServiceManager } from '@/utils/ServiceManager'
import {
testConnection,
kongfzLogin,
@ -406,17 +409,166 @@ export default {
const bindLoading = ref(false)
const result = ref<{ success: boolean; message: string } | null>(null)
/** 通过自定义协议启动本地 kfz-goods-pricing.exe */
const handleOpenExe = () => {
/** 通过本地服务管理器启动核价器 exe */
const fileInputRef = ref(null)
const serviceManager = new ServiceManager('http://127.0.0.1:5000')
const SERVICE_ID = 'kfz-goods-pricing'
const handleBrowseExe = () => {
fileInputRef.value?.click()
}
const handleFileSelected = async () => {
const input = fileInputRef.value
if (!input?.files?.length) return
const file = input.files[0]
// file.path
let fullPath = (file as any).path || ''
// ServiceManager
if (!fullPath || !(fullPath.includes('\\') || fullPath.includes('/'))) {
try {
const services = await serviceManager.getServicesStatus()
const matched = services.find(s => s.exe_path && s.exe_path.includes(file.name))
if (matched) {
fullPath = matched.exe_path
}
} catch {
// ServiceManager
}
}
dir.value = fullPath || file.name
localStorage.setItem(STORAGE_KEY_FILE_DIR, dir.value)
//
if (fullPath && (fullPath.includes('\\') || fullPath.includes('/'))) {
const exeName = file.name.replace(/\.exe$/i, '')
try {
await serviceManager.registerService({
id: SERVICE_ID,
name: '核价器',
exe_path: fullPath,
})
ElMessage.success('已注册到启动器')
} catch {
//
}
// Vite API
try {
const resp = await fetch('/api/local/launch-exe', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ exe_path: fullPath })
})
const result = await resp.json()
if (result.code === 0) {
ElMessage.success({ message: `核价器已启动 (PID ${result.pid})`, duration: 1000, customClass: 'scan-success-message' })
} else {
// ServiceManager
await serviceManager.startService(SERVICE_ID)
}
} catch {
// Vite API ServiceManager
try {
await serviceManager.startService(SERVICE_ID)
} catch { /* 无法启动 */ }
}
}
// input 便 change
input.value = ''
}
const handleOpenExe = async () => {
if (!dir.value) {
ElMessage.warning({ message: '请先设置程序所在位置', customClass: 'scan-warning-message' })
return
}
// Vite API /
try {
// kfzgs://launcher.exe dir exe
window.location.href = `kfzgs://launch?dir=${encodeURIComponent(dir.value)}`
} catch (err: any) {
ElMessage.error({ message: `启动失败: ${err.message}`, customClass: 'scan-error-message' })
const resp = await fetch('/api/local/launch-exe', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ exe_path: dir.value })
})
const result = await resp.json()
if (result.code === 0) {
ElMessage.success({ message: `核价器已启动 (PID ${result.pid})`, duration: 1000, customClass: 'scan-success-message' })
return
}
console.warn('[启动程序] Vite 本地 API 失败:', result.msg)
} catch {
// Vite API
}
try {
//
const services = await serviceManager.getServicesStatus()
if (services.length > 0) {
let target = null
// exe_path SERVICE_ID
const isFullPath = dir.value.includes('\\') || dir.value.includes('/')
if (isFullPath) {
target = services.find(s => s.exe_path && s.exe_path.includes(dir.value))
}
if (!target) {
target = services.find(s => s.id === SERVICE_ID)
}
if (target) {
const ok = await serviceManager.startService(target.id)
if (ok) {
ElMessage.success({ message: `已启动: ${target.name || target.id}`, duration: 1000, customClass: 'scan-success-message' })
return
}
}
}
// 退
console.warn('[启动程序] 服务管理器不可用,回退到自定义协议')
// exe
const lastSep = Math.max(dir.value.lastIndexOf('\\'), dir.value.lastIndexOf('/'))
const dirPath = lastSep > 0 ? dir.value.substring(0, lastSep) : dir.value
const exeName = lastSep > 0 ? dir.value.substring(lastSep + 1) : ''
let launched = false
// 1. kfzgs://
try {
if (exeName) {
window.location.href = `kfzgs://launch?dir=${encodeURIComponent(dirPath)}&exe=${encodeURIComponent(exeName)}`
} else {
window.location.href = `kfzgs://launch?dir=${encodeURIComponent(dirPath)}`
}
launched = true
} catch { /* ignore */ }
// 2. file:// + window.open
if (!launched) {
const fileUrl = `file:///${dir.value.replace(/\\/g, '/')}`
const win = window.open(fileUrl, '_blank')
if (!win) {
// a
const a = document.createElement('a')
a.href = fileUrl
a.target = '_blank'
a.click()
}
}
ElMessage.success({ message: '启动指令已发送', duration: 1000, customClass: 'scan-success-message' })
} catch (err) {
// 退
console.warn('[启动程序] 服务管理器连接失败:', err)
try {
const fileUrl = `file:///${dir.value.replace(/\\/g, '/')}`
const win = window.open(fileUrl, '_blank')
if (!win) {
const a = document.createElement('a')
a.href = fileUrl
a.target = '_blank'
a.click()
}
ElMessage.success({ message: '启动指令已发送', duration: 1000, customClass: 'scan-success-message' })
} catch (err2) {
ElMessage.error({ message: `启动失败: ${err2.message || '未知错误'}`, customClass: 'scan-error-message' })
}
}
}
@ -791,6 +943,9 @@ export default {
return {
dir,
fileInputRef,
handleBrowseExe,
handleFileSelected,
ip,
port,
verifyIndex,

View File

@ -70,6 +70,13 @@
<el-button type="primary" @click="handlePrintBarcode('Alt+b')">打印条码</el-button>
</div>
</el-form-item>
<el-form-item label="提交快捷键">
<div class="printer-row">
<el-input :model-value="'Alt+r'" disabled style="width: 200px" />
<img v-if="barcodeImages['Alt+r']" :src="barcodeImages['Alt+r']" />
<el-button type="primary" @click="handlePrintBarcode('Alt+r')">打印条码</el-button>
</div>
</el-form-item>
</el-form>
@ -128,7 +135,7 @@ const cameraLoading = ref(false)
const cameras = ref([])
const selectedCamera = ref('')
const shortcutItems = ['Alt+c', 'Alt+a', 'Alt+x', 'Alt+b']
const shortcutItems = ['Alt+c', 'Alt+a', 'Alt+x', 'Alt+b', 'Alt+r']
const barcodeImages = reactive({})
shortcutItems.forEach(key => {
@ -191,6 +198,9 @@ const confirmPaperSize = () => {
if (paperSize.value) {
localStorage.setItem(STORAGE_KEY_PAPER_SIZE, paperSize.value)
ElMessage.success({ message: '小票纸大小已保存', duration: 1000, customClass: 'scan-success-message' })
} else {
localStorage.removeItem(STORAGE_KEY_PAPER_SIZE)
ElMessage.success({ message: '小票纸大小已清除', duration: 1000, customClass: 'scan-success-message' })
}
}
@ -206,16 +216,26 @@ const confirmExpressPrinter = () => {
const fetchCameras = async () => {
cameraLoading.value = true
let stream = null
try {
if (!navigator.mediaDevices) {
throw new Error('当前环境不支持摄像头')
}
// HTTP
stream = await navigator.mediaDevices.getUserMedia({ video: true })
const devices = await navigator.mediaDevices.enumerateDevices()
cameras.value = devices.filter((d) => d.kind === 'videoinput').map((d) => ({
deviceId: d.deviceId,
label: d.label || `摄像头 ${d.deviceId.slice(0, 8)}`
}))
} catch {
ElMessage.error({ message: '获取摄像头列表失败', customClass: 'scan-error-message' })
} catch (err) {
console.warn('[摄像头] 获取列表失败:', err)
ElMessage.error({ message: '获取摄像头列表失败: ' + (err?.message || '未知错误'), customClass: 'scan-error-message' })
cameras.value = []
} finally {
if (stream) {
stream.getTracks().forEach(track => track.stop())
}
cameraLoading.value = false
}
}

245
vite.config.js Normal file
View File

@ -0,0 +1,245 @@
const { defineConfig } = require('vite')
const vue = require('@vitejs/plugin-vue').default
const path = require('path')
const { spawn } = require('child_process')
// https://vitejs.dev/config/
module.exports = defineConfig({
css: {
preprocessorOptions: {
scss: {
silenceDeprecations: ['legacy-js-api'],
api: 'modern-compiler'
}
}
},
plugins: [
vue(),
// 孔夫子旧书网登录代理(避免浏览器跨域限制)
{
name: 'kongfz-login-proxy',
configureServer(server) {
server.middlewares.use('/kongfz-login', async (req, res) => {
if (req.method !== 'POST') {
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({ code: 405, message: '仅支持 POST 请求' }))
return
}
let body = ''
req.on('data', chunk => { body += chunk.toString() })
req.on('end', async () => {
try {
const params = new URLSearchParams(body)
const username = params.get('loginName') || ''
const password = params.get('loginPass') || ''
if (!username || !password) {
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({ code: 400, message: '请输入用户名和密码!' }))
return
}
const https = require('https')
const http = require('http')
function requestWithRedirect(options, bodyData, redirectCount = 0) {
return new Promise((resolve, reject) => {
if (redirectCount > 5) {
reject(new Error('重定向次数过多'))
return
}
const mod = options.port === 443 ? https : http
const req = mod.request(options, (response) => {
if (response.statusCode >= 300 && response.statusCode < 400 && response.headers.location) {
const redirectUrl = new URL(response.headers.location, `https://${options.hostname}`)
const redirectOpts = {
hostname: redirectUrl.hostname,
port: redirectUrl.port || (redirectUrl.protocol === 'https:' ? 443 : 80),
path: redirectUrl.pathname + redirectUrl.search,
method: 'GET',
headers: {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8'
},
timeout: 15000
}
if (response.headers['set-cookie']) {
const cookies = Array.isArray(response.headers['set-cookie'])
? response.headers['set-cookie'].join('; ')
: response.headers['set-cookie']
redirectOpts.headers['Cookie'] = cookies.split(';')[0].trim()
}
resolve(requestWithRedirect(redirectOpts, null, redirectCount + 1))
return
}
let responseData = ''
response.on('data', chunk => { responseData += chunk })
response.on('end', () => resolve({ statusCode: response.statusCode, headers: response.headers, body: responseData }))
})
req.on('error', err => reject(new Error(`登录请求失败: ${err.message}`)))
req.on('timeout', () => { req.destroy(); reject(new Error('登录请求超时')) })
if (bodyData) req.write(bodyData)
req.end()
})
}
const formData = new URLSearchParams()
formData.append('loginName', username)
formData.append('loginPass', password)
formData.append('returnUrl', 'http://user.kongfz.com/')
const formDataStr = formData.toString()
const loginResponse = await requestWithRedirect({
hostname: 'login.kongfz.com',
port: 443,
path: '/Pc/Login/account',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
'Content-Length': Buffer.byteLength(formDataStr),
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8'
},
timeout: 15000
}, formDataStr)
if (loginResponse.statusCode !== 200) {
throw new Error(`登录失败(HTTP状态码: ${loginResponse.statusCode})`)
}
const setCookie = loginResponse.headers['set-cookie'] || []
const cookieStr = Array.isArray(setCookie) ? setCookie.join('; ') : setCookie || ''
if (loginResponse.body.includes("window.location.href='https://login.kongfz.cn/Pc/Session/rsync")) {
if (!cookieStr) throw new Error('登录成功但未获取到Cookie')
if (!cookieStr.includes('PHPSESSID=')) throw new Error('登录失败: 未找到 PHPSESSID')
const token = cookieStr.split('PHPSESSID=')[1].split(';')[0]
const userInfoResponse = await requestWithRedirect({
hostname: 'user.kongfz.com',
port: 443,
path: '/User/Index/getUserInfo/',
method: 'GET',
headers: {
'Cookie': `PHPSESSID=${token}`,
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
'Accept': 'application/json, text/plain, */*',
'Accept-Language': 'zh-CN,zh;q=0.9,en;q=0.8'
},
timeout: 15000
}, null)
let displayName = username
try {
const userJson = JSON.parse(userInfoResponse.body)
if (userJson.status && userJson.data) displayName = userJson.data.nickname || username
} catch { }
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({
code: 200, message: '登录成功',
data: { token, username: displayName, nickname: displayName }
}))
} else {
try {
const errJson = JSON.parse(loginResponse.body)
if (errJson.errCode === 1001 || errJson.errCode === 1005) throw new Error('账号或密码错误')
throw new Error(errJson.errInfo || '登录失败,未知错误!')
} catch (parseErr) {
if (parseErr instanceof SyntaxError) throw new Error('登录失败,未知错误!')
throw parseErr
}
}
} catch (error) {
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({ code: 500, message: error.message }))
}
})
})
}
},
// 本地程序启动器(无需协议注册/脚本/额外服务Vite 服务器直接 spawn exe
{
name: 'local-launcher',
configureServer(server) {
server.middlewares.use('/api/local/launch-exe', (req, res) => {
res.setHeader('Access-Control-Allow-Origin', '*')
res.setHeader('Access-Control-Allow-Methods', 'POST, OPTIONS')
res.setHeader('Access-Control-Allow-Headers', 'Content-Type')
if (req.method === 'OPTIONS') {
res.statusCode = 204
return res.end()
}
if (req.method !== 'POST') {
res.setHeader('Content-Type', 'application/json')
return res.end(JSON.stringify({ code: 405, msg: '仅支持 POST' }))
}
let body = ''
req.on('data', chunk => { body += chunk.toString() })
req.on('end', () => {
try {
const { exe_path } = JSON.parse(body)
if (!exe_path) {
res.setHeader('Content-Type', 'application/json')
return res.end(JSON.stringify({ code: 400, msg: '缺少 exe_path' }))
}
const fs = require('fs')
if (!fs.existsSync(exe_path)) {
res.setHeader('Content-Type', 'application/json')
return res.end(JSON.stringify({ code: 404, msg: `文件不存在: ${exe_path}` }))
}
const exeDir = path.dirname(exe_path)
const proc = spawn(exe_path, [], {
cwd: exeDir,
detached: true,
stdio: 'ignore',
windowsHide: false
})
proc.unref()
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({ code: 0, msg: '已启动', pid: proc.pid }))
} catch (err) {
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify({ code: 500, msg: err.message }))
}
})
})
}
}
],
resolve: {
alias: {
'@': path.resolve(__dirname, './src')
}
},
define: {
__VUE_PROD_HYDRATION_MISMATCH_DETAILS__: 'false'
},
server: {
port: 5174,
host: '0.0.0.0',
historyApiFallback: true,
proxy: {
'/api': {
// target: 'http://127.0.0.1:9090',
// target: 'http://192.168.101.213:9090',
target: 'https://psi.api.buzhiyushu.cn',
// target: 'https://psi.api.buzhiyushu.cn',
changeOrigin: true
},
'/api/print': {
// target: 'http://192.168.101.127:8075',
target: 'https://print.buzhiyushu.cn',
changeOrigin: true
}
}
}
})