ykt-wxapp/pages/case/search.vue

326 lines
7.9 KiB
Vue
Raw Permalink Normal View History

2026-01-20 16:24:43 +08:00
<template>
<view class="search-container">
<!-- Search Header -->
<view class="search-header">
<view class="search-bar">
<uni-icons type="search" size="18" color="#999" class="search-icon"></uni-icons>
<input
class="search-input"
placeholder="搜索患者名称/手机号/院内ID号"
v-model="searchQuery"
confirm-type="search"
focus
@input="handleSearch"
/>
<uni-icons v-if="searchQuery" type="clear" size="18" color="#ccc" @click="clearSearch" class="clear-icon"></uni-icons>
</view>
<text class="cancel-btn" @click="goBack">取消</text>
</view>
<!-- Search Results -->
<scroll-view v-if="searchQuery" scroll-y class="search-results">
<view v-if="searching && searchResults.length === 0" class="empty-state">
<text class="empty-text">搜索中...</text>
</view>
<view v-else-if="searchResults.length === 0" class="empty-state">
2026-01-20 16:24:43 +08:00
<text class="empty-text">暂无搜索结果</text>
</view>
<view v-else>
<view v-for="(patient, index) in searchResults" :key="index" class="patient-card" @click="goDetail(patient)">
2026-01-20 16:24:43 +08:00
<!-- Row 1 -->
<view class="card-row-top">
<view class="patient-info">
<text class="patient-name">{{ patient.name }}</text>
<text class="patient-meta">{{ patient.gender }}/{{ patient.age }}</text>
</view>
<view class="patient-tags">
<view v-for="(tag, tIndex) in patient.tags" :key="tIndex" class="tag">
{{ tag }}
</view>
</view>
</view>
<!-- Row 2 -->
<view class="card-row-bottom">
<text v-if="patient.record" class="record-text">
{{ patient.record.type }} / {{ patient.record.date }} / {{ patient.record.diagnosis }}
</text>
<text v-else class="no-record">暂无病历记录</text>
</view>
</view>
</view>
</scroll-view>
<!-- History or Suggestions (when no search) -->
<view v-else class="search-tips">
<text class="tips-text">输入患者名称手机号或院内ID号进行搜索</text>
</view>
</view>
</template>
<script setup>
import { ref, computed } from 'vue';
import { storeToRefs } from 'pinia';
import api from '@/utils/api';
import useAccountStore from '@/store/account';
import useDebounce from '@/utils/useDebounce';
import { toast } from '@/utils/widget';
2026-01-20 16:24:43 +08:00
// State
const searchQuery = ref('');
const searchResultsRaw = ref([]);
const searching = ref(false);
2026-01-20 16:24:43 +08:00
const CURRENT_TEAM_STORAGE_KEY = 'ykt_case_current_team';
const DETAIL_STORAGE_KEY = 'ykt_case_archive_detail';
const accountStore = useAccountStore();
const { account, doctorInfo } = storeToRefs(accountStore);
function asArray(value) {
return Array.isArray(value) ? value : [];
}
function getUserId() {
const d = doctorInfo.value || {};
const a = account.value || {};
return String(d.userid || d.userId || d.corpUserId || a.userid || a.userId || '') || '';
}
function getTeamContext() {
const cached = uni.getStorageSync(CURRENT_TEAM_STORAGE_KEY) || {};
const corpId = String(cached.corpId || account.value?.corpId || '') || '';
const teamId = String(cached.teamId || '') || '';
return { corpId, teamId };
}
function formatPatient(raw) {
const rawTags = asArray(raw?.tags).filter((i) => typeof i === 'string');
const rawTagNames = asArray(raw?.tagNames).filter((i) => typeof i === 'string');
const tagIds = asArray(raw?.tagIds).map(String).filter(Boolean);
const mobiles = asArray(raw?.mobiles).map(String).filter(Boolean);
const mobile = raw?.mobile ? String(raw.mobile) : (mobiles[0] || '');
return {
...raw,
_id: raw?._id || raw?.id || '',
name: raw?.name || raw?.customerName || '',
gender: raw?.sex || raw?.gender || '',
age: raw?.age ?? '',
tags: rawTags.length ? rawTags : (rawTagNames.length ? rawTagNames : tagIds),
mobiles,
mobile,
record: null,
createTime: raw?.createTime || '',
creator: raw?.creatorName || raw?.creator || '',
hospitalId: raw?.customerNumber || raw?.hospitalId || '',
};
}
2026-01-20 16:24:43 +08:00
// Computed
const searchResults = computed(() => {
return searchResultsRaw.value || [];
2026-01-20 16:24:43 +08:00
});
// Methods
const doSearch = useDebounce(async () => {
const q = String(searchQuery.value || '').trim();
if (!q) {
searchResultsRaw.value = [];
return;
}
const userId = getUserId();
const { corpId, teamId } = getTeamContext();
if (!corpId || !teamId || !userId) {
toast('缺少用户/团队信息,请先完成登录与个人信息');
return;
}
searching.value = true;
const res = await api('searchCorpCustomerWithFollowTime', {
corpId,
userId,
teamId,
name: q,
page: 1,
pageSize: 50,
});
searching.value = false;
if (!res?.success) {
toast(res?.message || '搜索失败');
return;
}
const payload =
res && typeof res === 'object'
? res.data && typeof res.data === 'object' && !Array.isArray(res.data)
? res.data
: res
: {};
const list = Array.isArray(payload.list) ? payload.list : Array.isArray(payload.data) ? payload.data : [];
searchResultsRaw.value = list.map(formatPatient);
}, 600);
const handleSearch = () => doSearch();
2026-01-20 16:24:43 +08:00
const clearSearch = () => {
searchQuery.value = '';
searchResultsRaw.value = [];
2026-01-20 16:24:43 +08:00
};
const goBack = () => {
uni.navigateBack();
};
const goDetail = (patient) => {
const id = patient?._id || patient?.id || patient?.mobile || patient?.phone || '';
uni.setStorageSync(DETAIL_STORAGE_KEY, {
_id: id,
name: patient.name,
sex: patient.gender,
age: patient.age,
mobile: patient.mobile,
mobiles: Array.isArray(patient.mobiles) ? patient.mobiles : (patient.mobile ? [patient.mobile] : []),
createTime: patient.createTime,
creator: patient.creator,
});
uni.navigateTo({ url: `/pages/case/archive-detail?id=${encodeURIComponent(String(id))}` });
};
2026-01-20 16:24:43 +08:00
</script>
<style lang="scss" scoped>
.search-container {
display: flex;
flex-direction: column;
height: 100vh;
background-color: #f7f8fa;
}
.search-header {
display: flex;
align-items: center;
padding: 10px 15px;
background-color: #fff;
border-bottom: 1px solid #f0f0f0;
.search-bar {
flex: 1;
height: 36px;
background-color: #f5f5f5;
border-radius: 18px;
display: flex;
align-items: center;
padding: 0 12px;
margin-right: 10px;
.search-icon {
margin-right: 8px;
}
.search-input {
flex: 1;
font-size: 14px;
color: #333;
}
.clear-icon {
margin-left: 8px;
}
}
.cancel-btn {
font-size: 14px;
color: #5d8aff;
}
}
.search-results {
flex: 1;
.empty-state {
padding: 80px 20px;
text-align: center;
.empty-text {
font-size: 14px;
color: #999;
}
}
}
.search-tips {
flex: 1;
padding: 20px;
.tips-text {
font-size: 14px;
color: #999;
}
}
.patient-card {
background-color: #fff;
padding: 15px;
margin-bottom: 1px;
border-bottom: 1px solid #f0f0f0;
.card-row-top {
display: flex;
align-items: center;
margin-bottom: 8px;
flex-wrap: wrap;
.patient-info {
display: flex;
align-items: flex-end;
margin-right: 10px;
.patient-name {
font-size: 18px;
font-weight: bold;
color: #333;
margin-right: 8px;
}
.patient-meta {
font-size: 12px;
color: #999;
margin-bottom: 2px;
}
}
.patient-tags {
display: flex;
gap: 5px;
.tag {
font-size: 10px;
color: #5d8aff;
border: 1px solid #5d8aff;
padding: 0 4px;
border-radius: 8px;
height: 16px;
line-height: 14px;
}
}
}
.card-row-bottom {
font-size: 14px;
.record-text {
color: #666;
}
.no-record {
color: #bdc3c7;
}
}
}
</style>