AddForm.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552
  1. <template>
  2. <el-drawer v-model="dialogVisible" :title="title" size="90%" :before-close="handleClosed">
  3. <div class="assessment-form">
  4. <div class="form-header">
  5. <el-form :model="dataForm" :rules="rules" ref="formRef" label-width="80px">
  6. <el-row :gutter="20">
  7. <el-col :span="6">
  8. <el-form-item label="姓名" prop="elderName">
  9. <SelectElder
  10. v-model="dataForm.elderId"
  11. ref="selectElderRef" :tId="dataForm.tenantId" @elder="elderUp" :disabled="isDetail"
  12. />
  13. </el-form-item>
  14. </el-col>
  15. <el-col :span="6">
  16. <el-form-item label="年龄">
  17. <el-input v-model="dataForm.elderAge" disabled />
  18. </el-form-item>
  19. </el-col>
  20. <el-col :span="6">
  21. <el-form-item label="性别">
  22. <el-input v-model="dataForm.elderSex" disabled />
  23. </el-form-item>
  24. </el-col>
  25. <el-col :span="6">
  26. <el-form-item label="房号/床号">
  27. <el-input v-model="dataForm.bedName" disabled />
  28. </el-form-item>
  29. </el-col>
  30. </el-row>
  31. </el-form>
  32. </div>
  33. <!-- 危险因素评估表 -->
  34. <div class="risk-factors-section">
  35. <div class="section-title">危险因素评估</div>
  36. <table class="assessment-table">
  37. <thead>
  38. <tr>
  39. <th class="col-factor">危险因素</th>
  40. <th class="col-score-header">是</th>
  41. <th class="col-score-header">否</th>
  42. <th class="col-result">得分</th>
  43. </tr>
  44. </thead>
  45. <tbody>
  46. <tr v-for="(item, itemIndex) in riskFactors" :key="itemIndex">
  47. <td class="col-factor">{{ item.name }}</td>
  48. <td class="col-score-header">
  49. <el-radio-group v-model="form.assessment.scores[itemIndex]" :disabled="isDetail" @change="calculateTotal">
  50. <el-radio :label="item.yesScore">{{ item.yesScore }}分</el-radio>
  51. </el-radio-group>
  52. </td>
  53. <td class="col-score-header">
  54. <el-radio-group v-model="form.assessment.scores[itemIndex]" :disabled="isDetail" @change="calculateTotal">
  55. <el-radio :label="item.noScore">{{ item.noScore }}分</el-radio>
  56. </el-radio-group>
  57. </td>
  58. <td class="col-result">{{ form.assessment.scores[itemIndex] }}</td>
  59. </tr>
  60. </tbody>
  61. </table>
  62. <!-- 风险程度判断 -->
  63. <div class="risk-judgment">
  64. <span class="judgment-title">风险程度判断:</span>
  65. <el-radio-group v-model="form.riskLevel" :disabled="isDetail">
  66. <el-radio label="none">无风险:≤1分</el-radio>
  67. <el-radio label="low">低风险:2分</el-radio>
  68. <el-radio label="medium">中风险:3分</el-radio>
  69. <el-radio label="high">高风险:≥4分</el-radio>
  70. </el-radio-group>
  71. </div>
  72. </div>
  73. <!-- 评估记录 -->
  74. <div class="assessment-records">
  75. <div class="record-section">
  76. <div class="record-header">评估记录</div>
  77. <el-row :gutter="20">
  78. <el-col :span="6">
  79. <el-form-item label="评估日期:" label-width="120px">
  80. <el-date-picker
  81. v-model="form.assessment.assessDate"
  82. type="date"
  83. placeholder="选择日期"
  84. value-format="YYYY-MM-DD"
  85. :disabled="isDetail"
  86. style="width: 100%"
  87. />
  88. </el-form-item>
  89. </el-col>
  90. <el-col :span="6">
  91. <el-form-item label="评估人签名:" label-width="120px">
  92. <el-input v-model="form.assessment.assessor" :disabled="isDetail" />
  93. </el-form-item>
  94. </el-col>
  95. <el-col :span="6">
  96. <el-form-item label="评估总得分:" label-width="120px">
  97. <span class="score-text">{{ form.assessment.totalScore }} 分</span>
  98. </el-form-item>
  99. </el-col>
  100. <el-col :span="6">
  101. <el-form-item label="风险等级结果:" label-width="120px">
  102. <el-tag :type="getRiskTagType(form.riskLevel)">
  103. {{ getRiskText(form.riskLevel) }}
  104. </el-tag>
  105. </el-form-item>
  106. </el-col>
  107. </el-row>
  108. </div>
  109. </div>
  110. <!-- 预防措施 -->
  111. <div class="prevention-section">
  112. <div class="section-title">预防措施:</div>
  113. <el-checkbox-group v-model="form.preventionMeasures" :disabled="isDetail" class="prevention-group">
  114. <el-row :gutter="20">
  115. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  116. <el-checkbox label="加勤巡视,随时发现安全隐患">加勤巡视,随时发现安全隐患</el-checkbox>
  117. </el-col>
  118. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  119. <el-checkbox label="睡觉时拉起两侧床栏如需下床请将床栏放下,切勿翻越床栏">睡觉时拉起两侧床栏如需下床请将床栏放下,切勿翻越床栏</el-checkbox>
  120. </el-col>
  121. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  122. <el-checkbox label="常检查床的牢固性,发现不稳定时立即对床进行加固">常检查床的牢固性,发现不稳定时立即对床进行加固</el-checkbox>
  123. </el-col>
  124. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  125. <el-checkbox label="发现老人睡在床边缘时,应将老人调整到床中央">发现老人睡在床边缘时,应将老人调整到床中央</el-checkbox>
  126. </el-col>
  127. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  128. <el-checkbox label="翻身时叮嘱老人动作要慢,幅度要小确保安全">翻身时叮嘱老人动作要慢,幅度要小确保安全</el-checkbox>
  129. </el-col>
  130. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  131. <el-checkbox label="将生活用品、尿壶便器等放于易拿之处">将生活用品、尿壶便器等放于易拿之处</el-checkbox>
  132. </el-col>
  133. </el-row>
  134. </el-checkbox-group>
  135. </div>
  136. </div>
  137. <template #footer>
  138. <el-button @click="handleClosed">关闭</el-button>
  139. <!-- <el-button v-if="isDetail" type="success" @click="handleExport">打印</el-button>-->
  140. <el-button style="margin-left: 22px;margin-right: 30px" v-loading="formLoading" type="primary"
  141. v-show="!isDetail" @click="submitForm">确定</el-button>
  142. </template>
  143. </el-drawer>
  144. </template>
  145. <script lang="ts" setup>
  146. import { computed, ref, reactive } from 'vue'
  147. import dayjs from 'dayjs'
  148. import {
  149. fallPreventionCreate,
  150. fallPreventionGetById,
  151. fallPreventionUpdate
  152. } from "@/api/social-work";
  153. const message = useMessage()
  154. const { t } = useI18n()
  155. const title = ref('')
  156. const dialogVisible = ref(false)
  157. const formRef = ref()
  158. const selectElderRef = ref()
  159. const isDetail = ref(false)
  160. const formLoading = ref(false)
  161. let dataForm = ref({
  162. id: undefined,
  163. elderName: '',
  164. elderSex: '',
  165. elderAge: '',
  166. bedName: '',
  167. elderId: '',
  168. tenantId: undefined
  169. })
  170. const elderUp = (e: any) => {
  171. dataForm.value.elderName = e.elderName
  172. dataForm.value.elderId = e.id
  173. dataForm.value.elderSex = e.elderSex === 1 ? '男' : '女'
  174. dataForm.value.bedName = e.bedName || ''
  175. dataForm.value.elderAge = e.elderAge
  176. }
  177. // 危险因素定义
  178. const riskFactors = [
  179. { name: '最近一年曾有不明原因的坠床或跌倒经历', yesScore: 1, noScore: 0 },
  180. { name: '意识障碍', yesScore: 1, noScore: 0 },
  181. { name: '近期有癫痫病史', yesScore: 1, noScore: 0 },
  182. { name: '视力障碍', yesScore: 1, noScore: 0 },
  183. { name: '活动障碍,肢体偏瘫', yesScore: 3, noScore: 0 },
  184. { name: '年龄大于或等于 65 岁', yesScore: 1, noScore: 0 },
  185. { name: '体能虚弱', yesScore: 3, noScore: 0 },
  186. { name: '头晕、眩晕、体位性低血压', yesScore: 2, noScore: 0 },
  187. { name: '服用影响意识或行动的药物,如催眠剂、镇静安神药物,利尿剂、抗癫痫剂、麻醉止痛剂', yesScore: 1, noScore: 0 },
  188. { name: '吸毒、酗酒史', yesScore: 1, noScore: 0 },
  189. { name: '住院中无人陪伴', yesScore: 1, noScore: 0 },
  190. { name: '睡气垫床', yesScore: 1, noScore: 0 }
  191. ]
  192. // 表单数据
  193. const form = reactive({
  194. assessment: {
  195. assessDate: '',
  196. assessor: '',
  197. scores: new Array(12).fill(0),
  198. totalScore: 0
  199. },
  200. preventionMeasures: [],
  201. riskLevel: ''
  202. })
  203. // 计算总得分
  204. const calculateTotal = () => {
  205. const scores = form.assessment.scores
  206. const total = scores.reduce((sum, score) => sum + (score || 0), 0)
  207. form.assessment.totalScore = total
  208. // 自动判断风险等级
  209. if (total <= 1) {
  210. form.riskLevel = 'none'
  211. } else if (total === 2) {
  212. form.riskLevel = 'low'
  213. } else if (total === 3) {
  214. form.riskLevel = 'medium'
  215. } else if (total >= 4) {
  216. form.riskLevel = 'high'
  217. }
  218. }
  219. // 获取风险等级文本
  220. const getRiskText = (riskLevel: string) => {
  221. switch (riskLevel) {
  222. case 'none': return '无风险'
  223. case 'low': return '低风险'
  224. case 'medium': return '中风险'
  225. case 'high': return '高风险'
  226. default: return '-'
  227. }
  228. }
  229. // 获取风险等级标签类型
  230. const getRiskTagType = (riskLevel: string) => {
  231. switch (riskLevel) {
  232. case 'none': return 'success'
  233. case 'low': return 'info'
  234. case 'medium': return 'warning'
  235. case 'high': return 'danger'
  236. default: return ''
  237. }
  238. }
  239. /** 将表单数据序列化为 JSON 对象 */
  240. const serializeFormData = () => {
  241. return {
  242. assessment: {
  243. assessDate: form.assessment.assessDate ? dayjs(form.assessment.assessDate).format('YYYY-MM-DD') : '',
  244. assessor: form.assessment.assessor || '',
  245. scores: form.assessment.scores || [],
  246. totalScore: form.assessment.totalScore || 0
  247. },
  248. preventionMeasures: form.preventionMeasures || [],
  249. riskLevel: form.riskLevel
  250. }
  251. }
  252. /** 将 JSON 对象反序列化为表单数据 */
  253. const deserializeFormData = (formData: Record<string, any>) => {
  254. if (!formData) return
  255. if (formData.assessment) {
  256. form.assessment.assessDate = formData.assessment.assessDate || ''
  257. form.assessment.assessor = formData.assessment.assessor || ''
  258. form.assessment.scores = formData.assessment.scores || new Array(12).fill(0)
  259. form.assessment.totalScore = formData.assessment.totalScore || 0
  260. }
  261. form.preventionMeasures = formData.preventionMeasures || []
  262. form.riskLevel = formData.riskLevel || ''
  263. }
  264. /** 重置表单数据 */
  265. const resetForm = () => {
  266. dataForm.value = {
  267. id: undefined,
  268. elderName: '',
  269. elderSex: '',
  270. elderAge: '',
  271. bedName: '',
  272. elderId: '',
  273. tenantId: undefined
  274. }
  275. formRef.value?.resetFields()
  276. form.assessment = {
  277. assessDate: '',
  278. assessor: '',
  279. scores: new Array(12).fill(0),
  280. totalScore: 0
  281. }
  282. form.preventionMeasures = []
  283. form.riskLevel = ''
  284. }
  285. /** 打开弹窗 */
  286. const open = async (tenantId: any, id?: any, detail: boolean = false) => {
  287. resetForm()
  288. dialogVisible.value = true
  289. dataForm.value.id = id || undefined
  290. dataForm.value.tenantId = tenantId
  291. isDetail.value = detail
  292. if (id) {
  293. title.value = "编辑-防坠床评估表"
  294. await loadData(id)
  295. } else {
  296. title.value = "新增-防坠床评估表"
  297. }
  298. }
  299. /** 加载评估数据 */
  300. const loadData = async (id: number) => {
  301. try {
  302. const res = await fallPreventionGetById(id)
  303. if (res) {
  304. dataForm.value.elderName = res.elderName || ''
  305. dataForm.value.elderId = res.elderId || ''
  306. dataForm.value.elderSex = res.elderSex || ''
  307. dataForm.value.bedName = res.bedName || ''
  308. dataForm.value.elderAge = res.elderAge || ''
  309. await selectElderRef.value.upData(res.elderName, res.elderId)
  310. if (res.assessData) {
  311. const formData = JSON.parse(res.assessData)
  312. deserializeFormData(formData)
  313. }
  314. }
  315. } catch (error) {
  316. message.error('加载评估数据失败')
  317. }
  318. }
  319. defineExpose({ open })
  320. const emit = defineEmits(['success'])
  321. /** 提交表单 */
  322. const submitForm = async () => {
  323. if (!dataForm.value.elderId) {
  324. message.error('请选择长者')
  325. return
  326. }
  327. formLoading.value = true
  328. try {
  329. const formData = serializeFormData()
  330. const payload = {
  331. id: dataForm.value.id,
  332. elderId: dataForm.value.elderId,
  333. tenantId: dataForm.value.tenantId,
  334. elderName: dataForm.value.elderName,
  335. elderSex: dataForm.value.elderSex,
  336. elderAge: dataForm.value.elderAge,
  337. bedName: dataForm.value.bedName,
  338. assessData: JSON.stringify(formData),
  339. assessScore: form.assessment.totalScore || 0,
  340. riskLevel: form.riskLevel,
  341. assessor: form.assessment.assessor,
  342. assessDate: form.assessment.assessDate ? dayjs(form.assessment.assessDate).format('YYYY-MM-DD') : ''
  343. }
  344. if (dataForm.value.id) {
  345. await fallPreventionUpdate(payload)
  346. message.success(t('common.updateSuccess'))
  347. } else {
  348. await fallPreventionCreate(payload)
  349. message.success(t('common.createSuccess'))
  350. }
  351. dialogVisible.value = false
  352. emit('success')
  353. } catch (error) {
  354. message.error('提交失败')
  355. } finally {
  356. formLoading.value = false
  357. }
  358. }
  359. /** 关闭弹窗 */
  360. const handleClosed = () => {
  361. dialogVisible.value = false
  362. }
  363. /** 导出/打印 */
  364. const handleExport = () => {
  365. message.info('打印功能开发中')
  366. }
  367. const rules = {
  368. elderName: [{ required: true, message: '请选择长者', trigger: 'change' }]
  369. }
  370. </script>
  371. <style scoped lang="scss">
  372. .assessment-form {
  373. padding: 0 20px;
  374. .form-header {
  375. margin-bottom: 20px;
  376. }
  377. .risk-factors-section {
  378. margin-bottom: 20px;
  379. .section-title {
  380. font-size: 16px;
  381. font-weight: bold;
  382. margin-bottom: 15px;
  383. color: #333;
  384. border-left: 4px solid #409eff;
  385. padding-left: 10px;
  386. }
  387. .assessment-table {
  388. width: 100%;
  389. border-collapse: collapse;
  390. border: 1px solid #e4e7ed;
  391. border-radius: 4px;
  392. overflow: hidden;
  393. th, td {
  394. padding: 12px;
  395. border: 1px solid #e4e7ed;
  396. text-align: center;
  397. }
  398. thead {
  399. background-color: #f5f7fa;
  400. font-weight: bold;
  401. }
  402. .col-factor {
  403. width: 50%;
  404. text-align: left;
  405. }
  406. .col-score-header {
  407. width: 20%;
  408. }
  409. .col-result {
  410. width: 10%;
  411. font-weight: bold;
  412. color: #409eff;
  413. }
  414. }
  415. .risk-judgment {
  416. margin-top: 15px;
  417. padding: 15px;
  418. background-color: #f5f7fa;
  419. border-radius: 4px;
  420. .judgment-title {
  421. font-weight: bold;
  422. margin-right: 15px;
  423. }
  424. }
  425. }
  426. .assessment-records {
  427. margin-top: 20px;
  428. .record-section {
  429. margin-bottom: 20px;
  430. padding: 15px;
  431. border: 1px solid #e4e7ed;
  432. border-radius: 4px;
  433. .record-header {
  434. font-size: 14px;
  435. font-weight: bold;
  436. margin-bottom: 15px;
  437. color: #409eff;
  438. border-bottom: 1px solid #e4e7ed;
  439. padding-bottom: 10px;
  440. }
  441. .score-text {
  442. font-size: 16px;
  443. font-weight: bold;
  444. color: #409eff;
  445. }
  446. }
  447. }
  448. .prevention-section {
  449. margin-top: 20px;
  450. .section-title {
  451. font-size: 16px;
  452. font-weight: bold;
  453. margin-bottom: 15px;
  454. color: #333;
  455. border-left: 4px solid #409eff;
  456. padding-left: 10px;
  457. }
  458. .prevention-group {
  459. .el-checkbox {
  460. margin-bottom: 10px;
  461. margin-right: 20px;
  462. }
  463. }
  464. }
  465. .api-params-section {
  466. margin-top: 30px;
  467. padding: 20px;
  468. background-color: #f5f7fa;
  469. border: 1px solid #e4e7ed;
  470. border-radius: 4px;
  471. .section-title {
  472. font-size: 16px;
  473. font-weight: bold;
  474. margin-bottom: 15px;
  475. color: #333;
  476. border-left: 4px solid #67c23a;
  477. padding-left: 10px;
  478. }
  479. .risk-rules {
  480. margin-top: 15px;
  481. padding: 10px;
  482. background-color: #fff;
  483. border-radius: 4px;
  484. .rules-title {
  485. font-weight: bold;
  486. margin-bottom: 10px;
  487. color: #606266;
  488. }
  489. .rule-tag {
  490. margin-right: 10px;
  491. margin-bottom: 5px;
  492. }
  493. }
  494. }
  495. }
  496. </style>