2026-01-20 16:24:43 +08:00
|
|
|
<template>
|
|
|
|
|
<view class="page">
|
|
|
|
|
<view class="body">
|
|
|
|
|
<scroll-view scroll-y class="scroll">
|
|
|
|
|
<view class="form-wrap">
|
2026-01-23 17:10:23 +08:00
|
|
|
<form-template v-if="items.length" ref="formRef" :items="items" :form="forms" :rule="rules" :filterRule="filterRule" @change="onChange" />
|
2026-01-20 16:24:43 +08:00
|
|
|
</view>
|
2026-01-23 17:10:23 +08:00
|
|
|
<view class="scroll-spacer" />
|
2026-01-20 16:24:43 +08:00
|
|
|
</scroll-view>
|
|
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
<!-- #ifdef MP-WEIXIN -->
|
|
|
|
|
<cover-view class="footer">
|
|
|
|
|
<cover-view class="btn plain" @tap="prev">上一步</cover-view>
|
|
|
|
|
<cover-view class="btn primary" @tap="save">保存</cover-view>
|
|
|
|
|
</cover-view>
|
|
|
|
|
<!-- #endif -->
|
|
|
|
|
|
|
|
|
|
<!-- #ifndef MP-WEIXIN -->
|
2026-01-20 16:24:43 +08:00
|
|
|
<view class="footer">
|
2026-01-23 17:10:23 +08:00
|
|
|
<button class="btn plain" @click="prev">上一步</button>
|
|
|
|
|
<button class="btn primary" @click="save">保存</button>
|
2026-01-20 16:24:43 +08:00
|
|
|
</view>
|
2026-01-23 17:10:23 +08:00
|
|
|
<!-- #endif -->
|
2026-01-20 16:24:43 +08:00
|
|
|
</view>
|
|
|
|
|
</view>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup>
|
2026-01-23 17:10:23 +08:00
|
|
|
import { computed, reactive, ref } from 'vue';
|
2026-01-20 16:24:43 +08:00
|
|
|
import { onLoad } from '@dcloudio/uni-app';
|
|
|
|
|
import FormTemplate from '@/components/form-template/index.vue';
|
2026-01-23 17:10:23 +08:00
|
|
|
import { storeToRefs } from 'pinia';
|
|
|
|
|
|
|
|
|
|
import api from '@/utils/api';
|
|
|
|
|
import useAccountStore from '@/store/account';
|
|
|
|
|
import { hideLoading, loading, toast } from '@/utils/widget';
|
2026-01-20 16:24:43 +08:00
|
|
|
|
|
|
|
|
const BASE_KEY = 'patient-create-base';
|
|
|
|
|
const INNER_KEY = 'patient-create-inner';
|
2026-01-23 17:10:23 +08:00
|
|
|
const CURRENT_TEAM_STORAGE_KEY = 'ykt_case_current_team';
|
|
|
|
|
const NEED_RELOAD_STORAGE_KEY = 'ykt_case_need_reload';
|
2026-01-20 16:24:43 +08:00
|
|
|
|
|
|
|
|
const formRef = ref(null);
|
|
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
const form = reactive({});
|
|
|
|
|
const baseForm = reactive({});
|
|
|
|
|
const forms = computed(() => ({ ...baseForm, ...form }));
|
2026-01-20 16:24:43 +08:00
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
const items = ref([]);
|
2026-01-20 16:24:43 +08:00
|
|
|
|
|
|
|
|
const rules = {};
|
2026-01-23 17:10:23 +08:00
|
|
|
const filterRule = {
|
|
|
|
|
reference(formModel) {
|
|
|
|
|
const customerSource = Array.isArray(formModel.customerSource)
|
|
|
|
|
? formModel.customerSource
|
|
|
|
|
: typeof formModel.customerSource === 'string'
|
|
|
|
|
? [formModel.customerSource]
|
|
|
|
|
: [];
|
|
|
|
|
return ['同事推荐', '客户推荐'].includes(customerSource[0]) && customerSource.length === 1;
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const accountStore = useAccountStore();
|
|
|
|
|
const { account, doctorInfo } = storeToRefs(accountStore);
|
|
|
|
|
const { getDoctorInfo } = accountStore;
|
|
|
|
|
|
|
|
|
|
function normalizeChangeValue(value) {
|
|
|
|
|
if (value && typeof value === 'object' && 'value' in value) return value.value;
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function normalizeOptions(options) {
|
|
|
|
|
if (!Array.isArray(options)) return [];
|
|
|
|
|
if (!options.length) return [];
|
|
|
|
|
if (typeof options[0] === 'string') return options.filter((i) => typeof i === 'string');
|
|
|
|
|
if (typeof options[0] === 'object') {
|
|
|
|
|
return options
|
|
|
|
|
.map((i) => {
|
|
|
|
|
const label = i?.label ?? i?.name ?? i?.text ?? i?.title ?? '';
|
|
|
|
|
const value = i?.value ?? i?.id ?? i?.key ?? label;
|
|
|
|
|
if (!label && (value === undefined || value === null || value === '')) return null;
|
|
|
|
|
return { label: String(label || value), value: String(value) };
|
|
|
|
|
})
|
|
|
|
|
.filter(Boolean);
|
|
|
|
|
}
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function normalizeTemplateItem(item) {
|
|
|
|
|
const next = { ...(item || {}) };
|
|
|
|
|
|
|
|
|
|
if (next.operateType === 'custom') next.operateType = 'formCell';
|
|
|
|
|
|
|
|
|
|
const originalType = next.type;
|
|
|
|
|
const customTypeMap = {
|
|
|
|
|
customerSource: 'select',
|
|
|
|
|
customerStage: 'select',
|
|
|
|
|
tag: 'multiSelectAndOther',
|
|
|
|
|
reference: 'input',
|
|
|
|
|
selectWwuser: 'select',
|
|
|
|
|
files: 'files',
|
|
|
|
|
corpProject: 'select',
|
|
|
|
|
diagnosis: 'textarea',
|
|
|
|
|
BMI: 'input',
|
|
|
|
|
bloodPressure: 'textarea',
|
|
|
|
|
selfMultipleDiseases: 'textarea',
|
|
|
|
|
};
|
|
|
|
|
if (originalType && customTypeMap[originalType]) {
|
|
|
|
|
next.__originType = originalType;
|
|
|
|
|
next.type = customTypeMap[originalType];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const aliasTypeMap = {
|
|
|
|
|
text: 'input',
|
|
|
|
|
string: 'input',
|
|
|
|
|
number: 'input',
|
|
|
|
|
integer: 'input',
|
|
|
|
|
int: 'input',
|
|
|
|
|
};
|
|
|
|
|
if (next.type && aliasTypeMap[next.type]) {
|
|
|
|
|
next.type = aliasTypeMap[next.type];
|
|
|
|
|
if (!next.inputType && (originalType === 'number' || originalType === 'integer' || originalType === 'int')) next.inputType = 'number';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const rawRange = next.range || next.options || next.optionList || next.values || [];
|
|
|
|
|
const range = normalizeOptions(rawRange);
|
|
|
|
|
|
|
|
|
|
if (next.type === 'select' || next.type === 'selectAndOther' || next.type === 'selectAndImage') {
|
|
|
|
|
next.range = range;
|
|
|
|
|
} else if (next.type === 'radio') {
|
|
|
|
|
if (range.length && typeof range[0] === 'object') {
|
|
|
|
|
next.type = 'select';
|
|
|
|
|
next.range = range;
|
|
|
|
|
} else {
|
|
|
|
|
next.range = Array.isArray(rawRange) ? rawRange : [];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!next.operateType) next.operateType = 'formCell';
|
|
|
|
|
next.required = Boolean(next.required);
|
|
|
|
|
|
|
|
|
|
if (next.type === 'input' && (next.wordLimit === undefined || next.wordLimit === null || next.wordLimit === '')) next.wordLimit = 20;
|
|
|
|
|
if (next.type === 'textarea' && (next.wordLimit === undefined || next.wordLimit === null || next.wordLimit === '')) next.wordLimit = 200;
|
|
|
|
|
return next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function loadInternalTemplate() {
|
|
|
|
|
const corpId = String(account.value?.corpId || doctorInfo.value?.corpId || '') || '';
|
|
|
|
|
if (!corpId) return;
|
|
|
|
|
|
|
|
|
|
loading('加载中...');
|
|
|
|
|
const res = await api('getCurrentTemplate', { corpId, templateType: 'internalTemplate' });
|
|
|
|
|
hideLoading();
|
|
|
|
|
|
|
|
|
|
if (!res?.success) {
|
|
|
|
|
toast(res?.message || '获取模板失败');
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const temp = res?.data && typeof res.data === 'object' ? res.data : res;
|
|
|
|
|
const list = Array.isArray(temp.templateList) ? temp.templateList : [];
|
|
|
|
|
items.value = list
|
|
|
|
|
.filter((i) => i && i.fieldStatus !== 'disable')
|
|
|
|
|
.filter((i) => i.operateType !== 'onlyRead')
|
|
|
|
|
.map(normalizeTemplateItem);
|
|
|
|
|
}
|
2026-01-20 16:24:43 +08:00
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
onLoad(async () => {
|
2026-01-20 16:24:43 +08:00
|
|
|
const base = uni.getStorageSync(BASE_KEY);
|
|
|
|
|
if (!base || typeof base !== 'object') {
|
|
|
|
|
uni.showToast({ title: '请先填写基础信息', icon: 'none' });
|
|
|
|
|
uni.navigateBack();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2026-01-23 17:10:23 +08:00
|
|
|
Object.assign(baseForm, base);
|
2026-01-20 16:24:43 +08:00
|
|
|
|
|
|
|
|
const cached = uni.getStorageSync(INNER_KEY);
|
|
|
|
|
if (cached && typeof cached === 'object') {
|
|
|
|
|
Object.assign(form, cached);
|
|
|
|
|
}
|
2026-01-23 17:10:23 +08:00
|
|
|
|
|
|
|
|
if (!doctorInfo.value && account.value?.openid) {
|
|
|
|
|
try {
|
|
|
|
|
await getDoctorInfo();
|
|
|
|
|
} catch (e) {
|
|
|
|
|
// ignore
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
await loadInternalTemplate();
|
2026-01-20 16:24:43 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
function onChange({ title, value }) {
|
2026-01-23 17:10:23 +08:00
|
|
|
form[title] = normalizeChangeValue(value);
|
2026-01-20 16:24:43 +08:00
|
|
|
}
|
|
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
function prev() {
|
2026-01-20 16:24:43 +08:00
|
|
|
uni.navigateBack();
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
function getUserId() {
|
|
|
|
|
const d = doctorInfo.value || {};
|
|
|
|
|
const a = account.value || {};
|
|
|
|
|
return String(d.userid || d.userId || d.corpUserId || a.userid || a.userId || '') || '';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function normalizeFormValue(value) {
|
|
|
|
|
if (value && typeof value === 'object' && 'value' in value) return value.value;
|
|
|
|
|
if (Array.isArray(value)) return value.map(normalizeFormValue);
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function normalizeFormObject(obj) {
|
|
|
|
|
if (!obj || typeof obj !== 'object') return {};
|
|
|
|
|
return Object.keys(obj).reduce((acc, key) => {
|
|
|
|
|
acc[key] = normalizeFormValue(obj[key]);
|
|
|
|
|
return acc;
|
|
|
|
|
}, {});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function buildPayload(base, inner) {
|
|
|
|
|
const team = uni.getStorageSync(CURRENT_TEAM_STORAGE_KEY) || {};
|
|
|
|
|
const corpId = String(account.value?.corpId || team?.corpId || base?.corpId || '') || '';
|
|
|
|
|
const baseForm = normalizeFormObject(base);
|
|
|
|
|
const innerForm = normalizeFormObject(inner);
|
|
|
|
|
const payload = {
|
|
|
|
|
...baseForm,
|
|
|
|
|
...innerForm,
|
|
|
|
|
corpId,
|
|
|
|
|
teamId: baseForm.teamId || (team?.teamId ? String(team.teamId) : ''),
|
|
|
|
|
creator: getUserId(),
|
|
|
|
|
addMethod: 'manual',
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// 兼容旧字段命名(历史缓存)
|
|
|
|
|
if (payload.gender && !payload.sex) payload.sex = payload.gender;
|
|
|
|
|
if (payload.idNo && !payload.idCard) payload.idCard = payload.idNo;
|
|
|
|
|
if (payload.idType && !payload.cardType) payload.cardType = payload.idType;
|
|
|
|
|
if (Array.isArray(payload.teamId)) payload.teamId = payload.teamId[0] || '';
|
|
|
|
|
|
|
|
|
|
if (payload.customerSource && typeof payload.customerSource === 'string') {
|
|
|
|
|
payload.customerSource = [payload.customerSource];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return { payload, team };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function save() {
|
2026-01-20 16:24:43 +08:00
|
|
|
if (formRef.value?.verify && !formRef.value.verify()) return;
|
|
|
|
|
|
|
|
|
|
const base = uni.getStorageSync(BASE_KEY) || {};
|
|
|
|
|
uni.setStorageSync(INNER_KEY, { ...form });
|
|
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
const { payload, team } = buildPayload(base, form);
|
|
|
|
|
if (!payload.corpId || !payload.teamId || !payload.creator) {
|
|
|
|
|
toast('缺少用户/团队信息,请先完成登录与个人信息');
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
loading('请求中...');
|
|
|
|
|
try {
|
|
|
|
|
const createTeamId = Array.isArray(payload.teamId) ? (payload.teamId[0] || '') : String(payload.teamId || '');
|
|
|
|
|
const res = await api('addCustomer', {
|
|
|
|
|
createTeamId,
|
|
|
|
|
createTeamName: team?.name || '',
|
|
|
|
|
params: payload,
|
|
|
|
|
});
|
|
|
|
|
hideLoading();
|
|
|
|
|
|
|
|
|
|
if (!res?.success) {
|
|
|
|
|
toast(res?.message || '新增失败');
|
|
|
|
|
return;
|
2026-01-20 16:24:43 +08:00
|
|
|
}
|
2026-01-23 17:10:23 +08:00
|
|
|
|
|
|
|
|
toast('新增成功');
|
|
|
|
|
uni.removeStorageSync(BASE_KEY);
|
|
|
|
|
uni.removeStorageSync(INNER_KEY);
|
|
|
|
|
uni.setStorageSync(NEED_RELOAD_STORAGE_KEY, 1);
|
|
|
|
|
uni.navigateBack({ delta: 2 });
|
|
|
|
|
} catch (e) {
|
|
|
|
|
hideLoading();
|
|
|
|
|
toast('新增失败');
|
|
|
|
|
}
|
2026-01-20 16:24:43 +08:00
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
|
.page {
|
|
|
|
|
height: 100vh;
|
|
|
|
|
background: #f6f6f6;
|
2026-01-23 17:10:23 +08:00
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
2026-01-20 16:24:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.body {
|
2026-01-23 17:10:23 +08:00
|
|
|
flex: 1;
|
2026-01-20 16:24:43 +08:00
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.scroll {
|
|
|
|
|
flex: 1;
|
2026-01-23 17:10:23 +08:00
|
|
|
min-height: 0;
|
|
|
|
|
height: 0;
|
2026-01-20 16:24:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
.form-wrap {
|
|
|
|
|
background: #fff;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.footer {
|
|
|
|
|
position: fixed;
|
|
|
|
|
left: 0;
|
|
|
|
|
right: 0;
|
|
|
|
|
bottom: 0;
|
|
|
|
|
background: #fff;
|
|
|
|
|
padding: 12px 16px calc(12px + env(safe-area-inset-bottom));
|
|
|
|
|
display: flex;
|
|
|
|
|
gap: 12px;
|
2026-01-23 17:10:23 +08:00
|
|
|
z-index: 10;
|
2026-01-20 16:24:43 +08:00
|
|
|
box-shadow: 0 -4px 12px rgba(0, 0, 0, 0.06);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-23 17:10:23 +08:00
|
|
|
.scroll-spacer {
|
|
|
|
|
height: calc(120px + env(safe-area-inset-bottom));
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-20 16:24:43 +08:00
|
|
|
.btn {
|
|
|
|
|
flex: 1;
|
|
|
|
|
height: 44px;
|
|
|
|
|
line-height: 44px;
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
font-size: 15px;
|
2026-01-23 17:10:23 +08:00
|
|
|
text-align: center;
|
2026-01-20 16:24:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.btn::after {
|
|
|
|
|
border: none;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.btn.plain {
|
|
|
|
|
background: #fff;
|
|
|
|
|
color: #666;
|
|
|
|
|
border: 1px solid #ddd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.btn.primary {
|
|
|
|
|
background: #5d8aff;
|
|
|
|
|
color: #fff;
|
|
|
|
|
}
|
|
|
|
|
</style>
|