人员管理增加修改

This commit is contained in:
2026-06-02 09:15:38 +08:00
parent 92165c914a
commit 05a72ac7dd
6 changed files with 212 additions and 87 deletions
+1 -1
View File
@@ -3,7 +3,7 @@
"appid" : "__UNI__5A0A7D6",
"description" : "",
"versionName" : "1.2.5",
"versionCode" : 114,
"versionCode" : 116,
"transformPx" : false,
/* 5+App */
"app-plus" : {
+6 -3
View File
@@ -603,17 +603,19 @@ export default {
<style scoped>
.page{
/* padding: 10px; */
height: 100%;
height: 100vh;
display: flex;
flex-direction: column;
overflow: hidden;
}
.content{
flex: 1;
display: flex;
width: 100%;
height: 100%;
flex-direction: column;
padding: 10px 20px;
overflow: hidden;
min-height: 0;
}
.block {
height: 120px;
@@ -713,7 +715,8 @@ export default {
.column{
display: flex;
flex-direction: column;
height: 100%;
flex: 1;
min-height: 0;
margin: 16px 0;
}
+2 -2
View File
@@ -379,10 +379,10 @@ export default {
}
},
temp() {
this.RS485.sendDataString(cmd.getTemp())
this.RS485.sendDataString('020300000002C438')
},
pressure() {
this.RS485.sendDataString(cmd.getPressure())
this.RS485.sendDataString('01030001000295CB')
}
}
+120 -4
View File
@@ -16,8 +16,8 @@
<th>工号</th>
<th>卡号</th>
<th>卡号2</th>
<th>操作</th>
</tr>
</thead>
<tbody class="ant-table-tbody">
<tr v-for="(item, index) in list" :key="index">
@@ -27,6 +27,12 @@
<td>{{ item.number }}</td>
<td>{{ item.ic }}</td>
<td>{{ item.ic2 }}</td>
<td>
<view class="action-btns">
<view class="ant-btn edit-btn" @click="edit(item)">编辑</view>
<!-- <view class="ant-btn delete-btn" @click="remove(item)">删除</view> -->
</view>
</td>
</tr>
</tbody>
</table>
@@ -102,7 +108,7 @@
<uni-row>
<uni-col>
<view class="form-btn">
<view class="ant-btn" @click="submit">保存</view>
<view class="ant-btn" @click="submit">{{ isEdit ? '更新' : '保存' }}</view>
<view class="ant-btn" style="margin-left: 20px;" @click="close">取消</view>
</view>
</uni-col>
@@ -134,6 +140,8 @@ export default {
],
page: 1,
totalPage: 0,
isEdit: false,
editId: null,
form: {
name: '',
ic: '',
@@ -149,6 +157,8 @@ export default {
methods: {
add() {
this.isEdit = false
this.editId = null
this.$refs.popup.open()
this.form = {
name: '',
@@ -167,8 +177,53 @@ export default {
},
close() {
this.$refs.popup.close()
this.isEdit = false
this.editId = null
uni.$off('ic')
},
edit(item) {
this.isEdit = true
this.editId = item.id
this.form = {
name: item.name,
ic: item.ic,
ic2: item.ic2 || '',
number: item.number || '',
type: item.type || ''
}
this.$refs.popup.open()
uni.$on('ic', (val) => {
if (!this.form.ic) {
this.form.ic = val
} else if (!this.form.ic2) {
this.form.ic2 = val
}
})
},
async remove(item) {
uni.showModal({
title: '确认删除',
content: `确定要删除 ${item.name} 吗?`,
success: async (res) => {
if (res.confirm) {
try {
await db.deleteInformationType('user', { id: item.id })
uni.showToast({
title: '删除成功',
icon: 'none'
})
this.getList()
} catch (error) {
console.log(error)
uni.showToast({
title: '删除失败',
icon: 'none'
})
}
}
}
})
},
async submit() {
if (this.form.name == '' || this.form.ic == '') {
return false
@@ -179,17 +234,33 @@ export default {
await db.addUserTable()
}
try {
if (this.isEdit) {
// 编辑模式
await db.updateSQL(tableName, {
name: this.form.name,
ic: this.form.ic,
ic2: this.form.ic2,
number: this.form.number,
type: this.form.type
}, 'id', this.editId)
uni.showToast({
title: '更新成功',
icon: 'none'
})
} else {
// 新增模式
await db.addTabItem(tableName, { ...this.form })
this.$refs.popup.close()
uni.showToast({
title: '保存成功',
icon: 'none'
})
}
this.$refs.popup.close()
this.getList()
} catch (error) {
console.log(error)
uni.showToast({
title: '保存失败',
title: this.isEdit ? '更新失败' : '保存失败',
icon: 'none'
})
}
@@ -295,4 +366,49 @@ export default {
.form-btn{
margin-top: 30px;
}
.action-btns{
display: flex;
gap: 10px;
justify-content: center;
}
.edit-btn{
/* background: #1890ff;
border-color: #1890ff; */
width: 60px;
height: 36px;
line-height: 32px;
font-size: 14px;
}
.delete-btn{
/* background: #ff4d4f;
border-color: #ff4d4f; */
width: 60px;
height: 36px;
line-height: 32px;
font-size: 14px;
}
.edit-btn:hover{
/* background: #40a9ff;
border-color: #40a9ff; */
}
.delete-btn:hover{
/* background: #ff7875;
border-color: #ff7875; */
}
/* 表格内容自动换行 */
.ant-table {
width: 100%;
border-collapse: collapse;
table-layout: fixed;
}
.ant-table th,
.ant-table td {
padding: 10px 8px;
text-align: center;
word-wrap: break-word;
word-break: break-all;
overflow-wrap: break-word;
max-width: 0;
}
</style>
+35 -64
View File
@@ -55,7 +55,6 @@ export default {
RS232: undefined,
timer: null, // 定时器读取485接口数据
taskTimer: null,
leftDoorCycleTimer: null, // 左门循环定时器
last485DataTime: null, // 上次收到485数据的时间戳
heartbeatTimer: null, // 心跳检测定时器
reconnectCount: 0, // 重连次数计数器
@@ -119,9 +118,9 @@ export default {
uni.$on('closeDoor', async (data) => {
try {
cmd.LeftDoor(false)
await delay(100)
cmd.RightDoor(false)
await cmd.LeftDoor(false)
await delay(200)
await cmd.RightDoor(false)
this.addLog('门', '触摸屏关门')
// 触发关门事件
this.closeDoorEvent()
@@ -163,11 +162,6 @@ export default {
clearInterval(this.taskTimer)
this.taskTimer = null
}
// 清理左门循环定时器
if (this.leftDoorCycleTimer) {
clearInterval(this.leftDoorCycleTimer)
this.leftDoorCycleTimer = null
}
},
methods: {
init() {
@@ -519,10 +513,13 @@ export default {
this.timer = null
}
// 4. 重置时间戳
// 4. 清空RS485队列中积压的失效指令
cmd.clearRS485Queue()
// 5. 重置时间戳
this.last485DataTime = null
// 5. 重新启动RS485通信
// 6. 重新启动RS485通信
this.startRS485()
} catch (error) {
@@ -531,7 +528,23 @@ export default {
// 失败后延迟重试
await delay(5000)
if (this.reconnectCount < this.maxReconnect) {
// 还有重连机会,继续重连
this.resetRS485()
} else {
// 重连次数耗尽,自动重启应用(彻底恢复native层状态)
this.addLog('485通信', `重连${this.maxReconnect}次均失败,即将自动重启...`)
uni.showToast({
title: '通信异常,3秒后重启',
icon: 'none',
duration: 3000
})
setTimeout(() => {
// #ifdef APP-PLUS
plus.runtime.restart()
// #endif
}, 3000)
}
}
},
@@ -592,7 +605,7 @@ export default {
if (data.action == 'door' && data.status == 'closed') {
try {
await cmd.LeftDoor(false)
await delay(100)
await delay(200)
await cmd.RightDoor(false)
// 触发关门事件
await this.closeDoorEvent()
@@ -678,50 +691,7 @@ export default {
}
}
},
// handle232HexData(hex) {
// // 处理232接口上报的数据
// let data = cmd.parse232dData(hex)
// if (data == {}) return
// if (data.action == 'leave' && data.type == 'internal') {
// // 更新内镜数据
// let key = data.address - 1
// uni.$emit('notice', { title: data.address, content: '内镜取出'})
// this.updateScope(key, 'leave', '')
// // 添加日志数据
// this.addLog('位置'+key, '内镜取出')
// }
// if (data.action == 'enter' && data.type == 'internal') {
// // 验证内镜信息
// this.checkEndoCard(data.number).then(res => {
// if (res) {
// uni.$emit('notice', { title: data.number, content: '内镜存入'})
// // 更新内镜数据
// let key = data.address - 1
// this.updateScope(key, 'enter', data.number)
// // 添加日志数据
// this.addLog(data.number, '内镜存入')
// }
// })
// uni.$emit('ic', data.number)
// }
// if (data.action == 'enter' && data.type == 'person') {
// // 验证人员信息
// this.checkPersonCard(data.number).then(res => {
// if (res) {
// // 开门事件
// cmd.Door(true)
// // this.openDoorEvent()
// uni.$emit('notice', { title: data.number, content: '刷卡开门'})
// // 添加日志数据
// this.addLog(data.number, '刷卡开门')
// }
// })
// uni.$emit('ic', data.number)
// }
// if (data.action == 'door') {
// cmd.Door(false)
// }
// },
// 密码框提交
inputModalSubmit(val) {
let password = this.$store.state.base.screenPw
@@ -803,20 +773,21 @@ export default {
// 开门事件
async openDoorEvent() {
// 关闭真空泵和消毒,打开照明,打开门锁,打开风机
cmd.Vacuum(false)
await delay(300)
cmd.Light(true)
await delay(300)
await delay(500)
cmd.Vacuum(false)
await delay(500)
cmd.Wind(true)
},
// 关门事件
async closeDoorEvent() {
// 关闭照明,打开真空泵、消毒
cmd.Light(false) // 照明关闭指令
await delay(300)
cmd.Vacuum(true) //真空泵开启指令
await delay(300)
cmd.Wind(true)
await delay(500)
await cmd.Light(false) // 照明关闭指令
await delay(500)
await cmd.Vacuum(true) //真空泵开启指令
await delay(500)
await cmd.Wind(true)
},
// 验证人员卡是否可以开门
async checkPersonCard(ic) {
+44 -9
View File
@@ -38,7 +38,7 @@ const enqueueTask = (task) => {
// 规则1: 去重 - 已存在相同cmd的任务则丢弃
const isDuplicate = rs485Queue.some(t => t.cmd === task.cmd)
if (isDuplicate) {
console.warn('[RS485队列] 丢弃重复任务:', task.cmd)
// console.warn('[RS485队列] 丢弃重复任务:', task.cmd)
return false
}
@@ -54,16 +54,16 @@ const enqueueTask = (task) => {
}
}
if (lastLowIndex !== -1) {
console.warn('[RS485队列] 高优挤掉低优:', rs485Queue[lastLowIndex].cmd)
// console.warn('[RS485队列] 高优挤掉低优:', rs485Queue[lastLowIndex].cmd)
rs485Queue.splice(lastLowIndex, 1)
} else {
// 没有低优先级可挤,踢掉队首最老的
console.warn('[RS485队列] 满,踢掉队首:', rs485Queue[0].cmd)
// console.warn('[RS485队列] 满,踢掉队首:', rs485Queue[0].cmd)
rs485Queue.shift()
}
} else {
// 低优先级:直接踢掉队首最老的
console.warn('[RS485队列] 满,踢掉队首:', rs485Queue[0].cmd)
// console.warn('[RS485队列] 满,踢掉队首:', rs485Queue[0].cmd)
rs485Queue.shift()
}
}
@@ -80,10 +80,41 @@ const enqueueTask = (task) => {
rs485Queue.push(task)
}
console.log(`[RS485队列] 入队成功, 长度:${rs485Queue.length}, cmd:${task.cmd}`)
// console.log(`[RS485队列] 入队成功, 长度:${rs485Queue.length}, cmd:${task.cmd}`)
return true
}
// ========== RS485 发送超时保护 ==========
const RS485_SEND_TIMEOUT = 2000 // 发送超时2秒
/**
* 带超时的RS485发送封装
* 防止底层串口驱动卡死导致Promise永久pending,造成队列死锁
*/
const sendWithTimeout = (rs485Instance, cmd) => {
return new Promise((resolve, reject) => {
const timer = setTimeout(() => {
console.warn(`[RS485] 发送超时(${RS485_SEND_TIMEOUT}ms), cmd: ${cmd}`)
reject(new Error('RS485发送超时'))
}, RS485_SEND_TIMEOUT)
const result = rs485Instance.sendDataString(cmd)
if (result && typeof result.then === 'function') {
result.then(res => {
clearTimeout(timer)
resolve(res)
}).catch(err => {
clearTimeout(timer)
reject(err)
})
} else {
clearTimeout(timer)
resolve(result)
}
})
}
// ========== RS485 发送超时保护 END ==========
const processRS485Queue = async () => {
if (isRS485Sending || rs485Queue.length === 0) {
return
@@ -95,10 +126,8 @@ const processRS485Queue = async () => {
if (!RS485) {
throw new Error('RS485 not initialized')
}
const result = RS485.sendDataString(task.cmd)
if (result && typeof result.then === 'function') {
await result
}
// 使用带超时的发送封装,防止Promise永久pending导致队列死锁
await sendWithTimeout(RS485, task.cmd)
// 发送后延迟,确保总线空闲再发下一条
await new Promise(r => setTimeout(r, RS485_SEND_DELAY))
task.resolve()
@@ -132,6 +161,7 @@ const clearRS485Queue = () => {
const task = rs485Queue.shift()
task.reject(new Error('RS485队列已清空(重连中)'))
}
isRS485Sending = false // 强制释放发送锁,防止队列死锁
}
/**
@@ -223,6 +253,11 @@ export default {
clearRS485Queue,
getRS485QueueSize,
// 重置RS485队列锁(供外部重连时调用)
resetRS485Lock() {
clearRS485Queue() // 内部已包含清队列 + 释放锁
},
// 解析门卡数据,返回卡号
parse232dData: (hexString) => {
// IC卡 20 01 00 08 04 00 00 00 0e 26 fe ab 8f 03