| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552 |
- <template>
- <el-drawer v-model="dialogVisible" :title="title" size="90%" :before-close="handleClosed">
- <div class="assessment-form">
- <div class="form-header">
- <el-form :model="dataForm" :rules="rules" ref="formRef" label-width="80px">
- <el-row :gutter="20">
- <el-col :span="6">
- <el-form-item label="姓名" prop="elderName">
- <SelectElder
- v-model="dataForm.elderId"
- ref="selectElderRef" :tId="dataForm.tenantId" @elder="elderUp" :disabled="isDetail"
- />
- </el-form-item>
- </el-col>
- <el-col :span="6">
- <el-form-item label="年龄">
- <el-input v-model="dataForm.elderAge" disabled />
- </el-form-item>
- </el-col>
- <el-col :span="6">
- <el-form-item label="性别">
- <el-input v-model="dataForm.elderSex" disabled />
- </el-form-item>
- </el-col>
- <el-col :span="6">
- <el-form-item label="房号/床号">
- <el-input v-model="dataForm.bedName" disabled />
- </el-form-item>
- </el-col>
- </el-row>
- </el-form>
- </div>
- <!-- 危险因素评估表 -->
- <div class="risk-factors-section">
- <div class="section-title">危险因素评估</div>
- <table class="assessment-table">
- <thead>
- <tr>
- <th class="col-factor">危险因素</th>
- <th class="col-score-header">是</th>
- <th class="col-score-header">否</th>
- <th class="col-result">得分</th>
- </tr>
- </thead>
- <tbody>
- <tr v-for="(item, itemIndex) in riskFactors" :key="itemIndex">
- <td class="col-factor">{{ item.name }}</td>
- <td class="col-score-header">
- <el-radio-group v-model="form.assessment.scores[itemIndex]" :disabled="isDetail" @change="calculateTotal">
- <el-radio :label="item.yesScore">{{ item.yesScore }}分</el-radio>
- </el-radio-group>
- </td>
- <td class="col-score-header">
- <el-radio-group v-model="form.assessment.scores[itemIndex]" :disabled="isDetail" @change="calculateTotal">
- <el-radio :label="item.noScore">{{ item.noScore }}分</el-radio>
- </el-radio-group>
- </td>
- <td class="col-result">{{ form.assessment.scores[itemIndex] }}</td>
- </tr>
- </tbody>
- </table>
- <!-- 风险程度判断 -->
- <div class="risk-judgment">
- <span class="judgment-title">风险程度判断:</span>
- <el-radio-group v-model="form.riskLevel" :disabled="isDetail">
- <el-radio label="none">无风险:≤1分</el-radio>
- <el-radio label="low">低风险:2分</el-radio>
- <el-radio label="medium">中风险:3分</el-radio>
- <el-radio label="high">高风险:≥4分</el-radio>
- </el-radio-group>
- </div>
- </div>
- <!-- 评估记录 -->
- <div class="assessment-records">
- <div class="record-section">
- <div class="record-header">评估记录</div>
- <el-row :gutter="20">
- <el-col :span="6">
- <el-form-item label="评估日期:" label-width="120px">
- <el-date-picker
- v-model="form.assessment.assessDate"
- type="date"
- placeholder="选择日期"
- value-format="YYYY-MM-DD"
- :disabled="isDetail"
- style="width: 100%"
- />
- </el-form-item>
- </el-col>
- <el-col :span="6">
- <el-form-item label="评估人签名:" label-width="120px">
- <el-input v-model="form.assessment.assessor" :disabled="isDetail" />
- </el-form-item>
- </el-col>
- <el-col :span="6">
- <el-form-item label="评估总得分:" label-width="120px">
- <span class="score-text">{{ form.assessment.totalScore }} 分</span>
- </el-form-item>
- </el-col>
- <el-col :span="6">
- <el-form-item label="风险等级结果:" label-width="120px">
- <el-tag :type="getRiskTagType(form.riskLevel)">
- {{ getRiskText(form.riskLevel) }}
- </el-tag>
- </el-form-item>
- </el-col>
- </el-row>
- </div>
- </div>
- <!-- 预防措施 -->
- <div class="prevention-section">
- <div class="section-title">预防措施:</div>
- <el-checkbox-group v-model="form.preventionMeasures" :disabled="isDetail" class="prevention-group">
- <el-row :gutter="20">
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-checkbox label="加勤巡视,随时发现安全隐患">加勤巡视,随时发现安全隐患</el-checkbox>
- </el-col>
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-checkbox label="睡觉时拉起两侧床栏如需下床请将床栏放下,切勿翻越床栏">睡觉时拉起两侧床栏如需下床请将床栏放下,切勿翻越床栏</el-checkbox>
- </el-col>
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-checkbox label="常检查床的牢固性,发现不稳定时立即对床进行加固">常检查床的牢固性,发现不稳定时立即对床进行加固</el-checkbox>
- </el-col>
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-checkbox label="发现老人睡在床边缘时,应将老人调整到床中央">发现老人睡在床边缘时,应将老人调整到床中央</el-checkbox>
- </el-col>
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-checkbox label="翻身时叮嘱老人动作要慢,幅度要小确保安全">翻身时叮嘱老人动作要慢,幅度要小确保安全</el-checkbox>
- </el-col>
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-checkbox label="将生活用品、尿壶便器等放于易拿之处">将生活用品、尿壶便器等放于易拿之处</el-checkbox>
- </el-col>
- </el-row>
- </el-checkbox-group>
- </div>
- </div>
- <template #footer>
- <el-button @click="handleClosed">关闭</el-button>
- <!-- <el-button v-if="isDetail" type="success" @click="handleExport">打印</el-button>-->
- <el-button style="margin-left: 22px;margin-right: 30px" v-loading="formLoading" type="primary"
- v-show="!isDetail" @click="submitForm">确定</el-button>
- </template>
- </el-drawer>
- </template>
- <script lang="ts" setup>
- import { computed, ref, reactive } from 'vue'
- import dayjs from 'dayjs'
- import {
- fallPreventionCreate,
- fallPreventionGetById,
- fallPreventionUpdate
- } from "@/api/social-work";
- const message = useMessage()
- const { t } = useI18n()
- const title = ref('')
- const dialogVisible = ref(false)
- const formRef = ref()
- const selectElderRef = ref()
- const isDetail = ref(false)
- const formLoading = ref(false)
- let dataForm = ref({
- id: undefined,
- elderName: '',
- elderSex: '',
- elderAge: '',
- bedName: '',
- elderId: '',
- tenantId: undefined
- })
- const elderUp = (e: any) => {
- dataForm.value.elderName = e.elderName
- dataForm.value.elderId = e.id
- dataForm.value.elderSex = e.elderSex === 1 ? '男' : '女'
- dataForm.value.bedName = e.bedName || ''
- dataForm.value.elderAge = e.elderAge
- }
- // 危险因素定义
- const riskFactors = [
- { name: '最近一年曾有不明原因的坠床或跌倒经历', yesScore: 1, noScore: 0 },
- { name: '意识障碍', yesScore: 1, noScore: 0 },
- { name: '近期有癫痫病史', yesScore: 1, noScore: 0 },
- { name: '视力障碍', yesScore: 1, noScore: 0 },
- { name: '活动障碍,肢体偏瘫', yesScore: 3, noScore: 0 },
- { name: '年龄大于或等于 65 岁', yesScore: 1, noScore: 0 },
- { name: '体能虚弱', yesScore: 3, noScore: 0 },
- { name: '头晕、眩晕、体位性低血压', yesScore: 2, noScore: 0 },
- { name: '服用影响意识或行动的药物,如催眠剂、镇静安神药物,利尿剂、抗癫痫剂、麻醉止痛剂', yesScore: 1, noScore: 0 },
- { name: '吸毒、酗酒史', yesScore: 1, noScore: 0 },
- { name: '住院中无人陪伴', yesScore: 1, noScore: 0 },
- { name: '睡气垫床', yesScore: 1, noScore: 0 }
- ]
- // 表单数据
- const form = reactive({
- assessment: {
- assessDate: '',
- assessor: '',
- scores: new Array(12).fill(0),
- totalScore: 0
- },
- preventionMeasures: [],
- riskLevel: ''
- })
- // 计算总得分
- const calculateTotal = () => {
- const scores = form.assessment.scores
- const total = scores.reduce((sum, score) => sum + (score || 0), 0)
- form.assessment.totalScore = total
- // 自动判断风险等级
- if (total <= 1) {
- form.riskLevel = 'none'
- } else if (total === 2) {
- form.riskLevel = 'low'
- } else if (total === 3) {
- form.riskLevel = 'medium'
- } else if (total >= 4) {
- form.riskLevel = 'high'
- }
- }
- // 获取风险等级文本
- const getRiskText = (riskLevel: string) => {
- switch (riskLevel) {
- case 'none': return '无风险'
- case 'low': return '低风险'
- case 'medium': return '中风险'
- case 'high': return '高风险'
- default: return '-'
- }
- }
- // 获取风险等级标签类型
- const getRiskTagType = (riskLevel: string) => {
- switch (riskLevel) {
- case 'none': return 'success'
- case 'low': return 'info'
- case 'medium': return 'warning'
- case 'high': return 'danger'
- default: return ''
- }
- }
- /** 将表单数据序列化为 JSON 对象 */
- const serializeFormData = () => {
- return {
- assessment: {
- assessDate: form.assessment.assessDate ? dayjs(form.assessment.assessDate).format('YYYY-MM-DD') : '',
- assessor: form.assessment.assessor || '',
- scores: form.assessment.scores || [],
- totalScore: form.assessment.totalScore || 0
- },
- preventionMeasures: form.preventionMeasures || [],
- riskLevel: form.riskLevel
- }
- }
- /** 将 JSON 对象反序列化为表单数据 */
- const deserializeFormData = (formData: Record<string, any>) => {
- if (!formData) return
- if (formData.assessment) {
- form.assessment.assessDate = formData.assessment.assessDate || ''
- form.assessment.assessor = formData.assessment.assessor || ''
- form.assessment.scores = formData.assessment.scores || new Array(12).fill(0)
- form.assessment.totalScore = formData.assessment.totalScore || 0
- }
- form.preventionMeasures = formData.preventionMeasures || []
- form.riskLevel = formData.riskLevel || ''
- }
- /** 重置表单数据 */
- const resetForm = () => {
- dataForm.value = {
- id: undefined,
- elderName: '',
- elderSex: '',
- elderAge: '',
- bedName: '',
- elderId: '',
- tenantId: undefined
- }
- formRef.value?.resetFields()
- form.assessment = {
- assessDate: '',
- assessor: '',
- scores: new Array(12).fill(0),
- totalScore: 0
- }
- form.preventionMeasures = []
- form.riskLevel = ''
- }
- /** 打开弹窗 */
- const open = async (tenantId: any, id?: any, detail: boolean = false) => {
- resetForm()
- dialogVisible.value = true
- dataForm.value.id = id || undefined
- dataForm.value.tenantId = tenantId
- isDetail.value = detail
- if (id) {
- title.value = "编辑-防坠床评估表"
- await loadData(id)
- } else {
- title.value = "新增-防坠床评估表"
- }
- }
- /** 加载评估数据 */
- const loadData = async (id: number) => {
- try {
- const res = await fallPreventionGetById(id)
- if (res) {
- dataForm.value.elderName = res.elderName || ''
- dataForm.value.elderId = res.elderId || ''
- dataForm.value.elderSex = res.elderSex || ''
- dataForm.value.bedName = res.bedName || ''
- dataForm.value.elderAge = res.elderAge || ''
- await selectElderRef.value.upData(res.elderName, res.elderId)
- if (res.assessData) {
- const formData = JSON.parse(res.assessData)
- deserializeFormData(formData)
- }
- }
- } catch (error) {
- message.error('加载评估数据失败')
- }
- }
- defineExpose({ open })
- const emit = defineEmits(['success'])
- /** 提交表单 */
- const submitForm = async () => {
- if (!dataForm.value.elderId) {
- message.error('请选择长者')
- return
- }
- formLoading.value = true
- try {
- const formData = serializeFormData()
- const payload = {
- id: dataForm.value.id,
- elderId: dataForm.value.elderId,
- tenantId: dataForm.value.tenantId,
- elderName: dataForm.value.elderName,
- elderSex: dataForm.value.elderSex,
- elderAge: dataForm.value.elderAge,
- bedName: dataForm.value.bedName,
- assessData: JSON.stringify(formData),
- assessScore: form.assessment.totalScore || 0,
- riskLevel: form.riskLevel,
- assessor: form.assessment.assessor,
- assessDate: form.assessment.assessDate ? dayjs(form.assessment.assessDate).format('YYYY-MM-DD') : ''
- }
- if (dataForm.value.id) {
- await fallPreventionUpdate(payload)
- message.success(t('common.updateSuccess'))
- } else {
- await fallPreventionCreate(payload)
- message.success(t('common.createSuccess'))
- }
- dialogVisible.value = false
- emit('success')
- } catch (error) {
- message.error('提交失败')
- } finally {
- formLoading.value = false
- }
- }
- /** 关闭弹窗 */
- const handleClosed = () => {
- dialogVisible.value = false
- }
- /** 导出/打印 */
- const handleExport = () => {
- message.info('打印功能开发中')
- }
- const rules = {
- elderName: [{ required: true, message: '请选择长者', trigger: 'change' }]
- }
- </script>
- <style scoped lang="scss">
- .assessment-form {
- padding: 0 20px;
- .form-header {
- margin-bottom: 20px;
- }
- .risk-factors-section {
- margin-bottom: 20px;
- .section-title {
- font-size: 16px;
- font-weight: bold;
- margin-bottom: 15px;
- color: #333;
- border-left: 4px solid #409eff;
- padding-left: 10px;
- }
- .assessment-table {
- width: 100%;
- border-collapse: collapse;
- border: 1px solid #e4e7ed;
- border-radius: 4px;
- overflow: hidden;
- th, td {
- padding: 12px;
- border: 1px solid #e4e7ed;
- text-align: center;
- }
- thead {
- background-color: #f5f7fa;
- font-weight: bold;
- }
- .col-factor {
- width: 50%;
- text-align: left;
- }
- .col-score-header {
- width: 20%;
- }
- .col-result {
- width: 10%;
- font-weight: bold;
- color: #409eff;
- }
- }
- .risk-judgment {
- margin-top: 15px;
- padding: 15px;
- background-color: #f5f7fa;
- border-radius: 4px;
- .judgment-title {
- font-weight: bold;
- margin-right: 15px;
- }
- }
- }
- .assessment-records {
- margin-top: 20px;
- .record-section {
- margin-bottom: 20px;
- padding: 15px;
- border: 1px solid #e4e7ed;
- border-radius: 4px;
- .record-header {
- font-size: 14px;
- font-weight: bold;
- margin-bottom: 15px;
- color: #409eff;
- border-bottom: 1px solid #e4e7ed;
- padding-bottom: 10px;
- }
- .score-text {
- font-size: 16px;
- font-weight: bold;
- color: #409eff;
- }
- }
- }
- .prevention-section {
- margin-top: 20px;
- .section-title {
- font-size: 16px;
- font-weight: bold;
- margin-bottom: 15px;
- color: #333;
- border-left: 4px solid #409eff;
- padding-left: 10px;
- }
- .prevention-group {
- .el-checkbox {
- margin-bottom: 10px;
- margin-right: 20px;
- }
- }
- }
- .api-params-section {
- margin-top: 30px;
- padding: 20px;
- background-color: #f5f7fa;
- border: 1px solid #e4e7ed;
- border-radius: 4px;
- .section-title {
- font-size: 16px;
- font-weight: bold;
- margin-bottom: 15px;
- color: #333;
- border-left: 4px solid #67c23a;
- padding-left: 10px;
- }
- .risk-rules {
- margin-top: 15px;
- padding: 10px;
- background-color: #fff;
- border-radius: 4px;
- .rules-title {
- font-weight: bold;
- margin-bottom: 10px;
- color: #606266;
- }
- .rule-tag {
- margin-right: 10px;
- margin-bottom: 5px;
- }
- }
- }
- }
- </style>
|