feat(health+miniprogram): 预约/报告/随访/资讯/家庭管理 — Chunk 4-6
后端: - 添加 articles 表迁移 + Entity + Service + Handler - 健康数据趋势 API (get_mini_trend) 注册路由 - article CRUD (list/get) + DTO 前端 (11个新页面 + 5个服务): - 预约挂号: 列表/创建向导/详情页 - 报告管理: 列表/详情页 - 随访管理: 任务列表/记录详情页 - 资讯文章: 文章详情页 - 个人中心: 就诊人管理/新增/我的报告/我的随访/用药提醒/设置 - 更新 app.config.ts 注册全部路由 - 更新 profile/article 页面为真实功能
This commit is contained in:
289
apps/miniprogram/src/pages/appointment/create/index.scss
Normal file
289
apps/miniprogram/src/pages/appointment/create/index.scss
Normal file
@@ -0,0 +1,289 @@
|
||||
@import '../../../styles/variables.scss';
|
||||
|
||||
.create-page {
|
||||
min-height: 100vh;
|
||||
background: $bg;
|
||||
padding-bottom: 140px;
|
||||
}
|
||||
|
||||
/* 步骤指示器 */
|
||||
.step-bar {
|
||||
display: flex;
|
||||
justify-content: space-around;
|
||||
padding: 32px 24px 0;
|
||||
}
|
||||
|
||||
.step-item {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
opacity: 0.4;
|
||||
transition: opacity 0.3s;
|
||||
}
|
||||
|
||||
.step-item.step-active {
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
.step-item.step-done {
|
||||
opacity: 0.8;
|
||||
}
|
||||
|
||||
.step-dot {
|
||||
width: 56px;
|
||||
height: 56px;
|
||||
border-radius: 50%;
|
||||
background: $bd;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
transition: background 0.3s;
|
||||
}
|
||||
|
||||
.step-active .step-dot {
|
||||
background: $pri;
|
||||
}
|
||||
|
||||
.step-done .step-dot {
|
||||
background: $acc;
|
||||
}
|
||||
|
||||
.step-num {
|
||||
font-size: 28px;
|
||||
color: white;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.step-check {
|
||||
font-size: 28px;
|
||||
color: white;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.step-label {
|
||||
font-size: 22px;
|
||||
color: $tx2;
|
||||
}
|
||||
|
||||
.step-active .step-label {
|
||||
color: $pri;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
/* 步骤连接线 */
|
||||
.step-line-wrapper {
|
||||
padding: 12px 80px 0;
|
||||
}
|
||||
|
||||
.step-line {
|
||||
height: 4px;
|
||||
background: $bd;
|
||||
border-radius: 2px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.step-line-fill {
|
||||
height: 100%;
|
||||
background: linear-gradient(90deg, $acc, $pri);
|
||||
border-radius: 2px;
|
||||
transition: width 0.3s ease;
|
||||
}
|
||||
|
||||
/* 步骤内容 */
|
||||
.step-content {
|
||||
padding: 32px 24px;
|
||||
}
|
||||
|
||||
.step-title {
|
||||
font-size: 32px;
|
||||
font-weight: bold;
|
||||
color: $tx;
|
||||
margin-bottom: 28px;
|
||||
display: block;
|
||||
}
|
||||
|
||||
/* 选择器卡片 */
|
||||
.picker-card {
|
||||
background: $card;
|
||||
border-radius: $r;
|
||||
padding: 24px 28px;
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);
|
||||
}
|
||||
|
||||
.picker-value {
|
||||
font-size: 28px;
|
||||
color: $tx;
|
||||
}
|
||||
|
||||
.picker-value.placeholder {
|
||||
color: $tx3;
|
||||
}
|
||||
|
||||
.picker-arrow {
|
||||
font-size: 24px;
|
||||
color: $tx3;
|
||||
}
|
||||
|
||||
/* 医生列表 */
|
||||
.doctor-list {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 16px;
|
||||
}
|
||||
|
||||
.doctor-card {
|
||||
background: $card;
|
||||
border-radius: $r;
|
||||
padding: 24px 28px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 20px;
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);
|
||||
border: 2px solid transparent;
|
||||
transition: border-color 0.2s;
|
||||
}
|
||||
|
||||
.doctor-card.doctor-selected {
|
||||
border-color: $pri;
|
||||
background: $pri-surface;
|
||||
}
|
||||
|
||||
.doctor-avatar {
|
||||
width: 80px;
|
||||
height: 80px;
|
||||
border-radius: 50%;
|
||||
background: $pri-l;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.doctor-avatar-text {
|
||||
font-size: 32px;
|
||||
color: $pri;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.doctor-detail {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.doctor-name {
|
||||
font-size: 30px;
|
||||
font-weight: bold;
|
||||
color: $tx;
|
||||
}
|
||||
|
||||
.doctor-title {
|
||||
font-size: 24px;
|
||||
color: $tx2;
|
||||
}
|
||||
|
||||
.doctor-specialty {
|
||||
font-size: 22px;
|
||||
color: $pri;
|
||||
}
|
||||
|
||||
.doctor-check {
|
||||
font-size: 32px;
|
||||
color: $pri;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
/* 表单 */
|
||||
.form-group {
|
||||
margin-bottom: 28px;
|
||||
}
|
||||
|
||||
.form-label {
|
||||
font-size: 26px;
|
||||
color: $tx2;
|
||||
margin-bottom: 12px;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.form-static {
|
||||
background: $card;
|
||||
border-radius: $r-sm;
|
||||
padding: 24px 28px;
|
||||
}
|
||||
|
||||
.form-static-text {
|
||||
font-size: 28px;
|
||||
color: $tx;
|
||||
}
|
||||
|
||||
.form-input {
|
||||
background: $card;
|
||||
border-radius: $r-sm;
|
||||
padding: 24px 28px;
|
||||
font-size: 28px;
|
||||
color: $tx;
|
||||
width: 100%;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
/* 空状态 */
|
||||
.empty-state {
|
||||
padding: 80px 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.empty-text {
|
||||
font-size: 28px;
|
||||
color: $tx3;
|
||||
}
|
||||
|
||||
/* 底部操作栏 */
|
||||
.bottom-bar {
|
||||
position: fixed;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
display: flex;
|
||||
gap: 16px;
|
||||
padding: 20px 24px;
|
||||
padding-bottom: constant(safe-area-inset-bottom);
|
||||
padding-bottom: env(safe-area-inset-bottom);
|
||||
background: $card;
|
||||
box-shadow: 0 -2px 8px rgba(0, 0, 0, 0.06);
|
||||
}
|
||||
|
||||
.btn {
|
||||
flex: 1;
|
||||
padding: 24px 0;
|
||||
border-radius: $r-sm;
|
||||
text-align: center;
|
||||
transition: opacity 0.2s;
|
||||
}
|
||||
|
||||
.btn-prev {
|
||||
background: $bd-l;
|
||||
}
|
||||
|
||||
.btn-next,
|
||||
.btn-submit {
|
||||
background: linear-gradient(135deg, $pri 0%, $pri-d 100%);
|
||||
}
|
||||
|
||||
.btn-disabled {
|
||||
opacity: 0.5;
|
||||
}
|
||||
|
||||
.btn-text {
|
||||
font-size: 30px;
|
||||
font-weight: bold;
|
||||
color: $tx2;
|
||||
}
|
||||
|
||||
.btn-text-white {
|
||||
color: white;
|
||||
}
|
||||
268
apps/miniprogram/src/pages/appointment/create/index.tsx
Normal file
268
apps/miniprogram/src/pages/appointment/create/index.tsx
Normal file
@@ -0,0 +1,268 @@
|
||||
import React, { useState, useCallback } from 'react';
|
||||
import { View, Text, Picker, Input } from '@tarojs/components';
|
||||
import Taro from '@tarojs/taro';
|
||||
import { listDoctors, createAppointment } from '../../../services/appointment';
|
||||
import { useAuthStore } from '../../../stores/auth';
|
||||
import './index.scss';
|
||||
|
||||
const DEPARTMENTS = ['内科', '外科', '妇科', '儿科', '体检中心'];
|
||||
|
||||
interface DoctorItem {
|
||||
id: string;
|
||||
name: string;
|
||||
title?: string;
|
||||
department?: string;
|
||||
specialty?: string;
|
||||
}
|
||||
|
||||
export default function AppointmentCreate() {
|
||||
const [currentStep, setCurrentStep] = useState(0);
|
||||
const [department, setDepartment] = useState('');
|
||||
const [deptPickerIndex, setDeptPickerIndex] = useState(0);
|
||||
const [doctors, setDoctors] = useState<DoctorItem[]>([]);
|
||||
const [selectedDoctor, setSelectedDoctor] = useState<DoctorItem | null>(null);
|
||||
const [appointmentDate, setAppointmentDate] = useState('');
|
||||
const [timeSlot, setTimeSlot] = useState('');
|
||||
const [reason, setReason] = useState('');
|
||||
const [loading, setLoading] = useState(false);
|
||||
|
||||
const currentPatient = useAuthStore((s) => s.currentPatient);
|
||||
|
||||
// Step 1: 选择科室后加载医生列表
|
||||
const onDepartmentChange = useCallback(async (e: any) => {
|
||||
const idx = e.detail.value;
|
||||
const dept = DEPARTMENTS[idx];
|
||||
setDeptPickerIndex(idx);
|
||||
setDepartment(dept);
|
||||
setSelectedDoctor(null);
|
||||
|
||||
try {
|
||||
const res = await listDoctors(dept);
|
||||
setDoctors(res.data || []);
|
||||
} catch {
|
||||
Taro.showToast({ title: '加载医生失败', icon: 'none' });
|
||||
}
|
||||
}, []);
|
||||
|
||||
// Step 2: 选择医生
|
||||
const onSelectDoctor = useCallback((doctor: DoctorItem) => {
|
||||
setSelectedDoctor(doctor);
|
||||
}, []);
|
||||
|
||||
// Step 3: 日期变更
|
||||
const onDateChange = useCallback((e: any) => {
|
||||
setAppointmentDate(e.detail.value);
|
||||
}, []);
|
||||
|
||||
// Step 3: 时段变更
|
||||
const onTimeSlotChange = useCallback((e: any) => {
|
||||
setTimeSlot(e.detail.value);
|
||||
}, []);
|
||||
|
||||
// Step 3: 备注变更
|
||||
const onReasonChange = useCallback((e: any) => {
|
||||
setReason(e.detail.value);
|
||||
}, []);
|
||||
|
||||
// 提交预约
|
||||
const handleSubmit = useCallback(async () => {
|
||||
if (!selectedDoctor) {
|
||||
Taro.showToast({ title: '请选择医生', icon: 'none' });
|
||||
return;
|
||||
}
|
||||
if (!appointmentDate) {
|
||||
Taro.showToast({ title: '请选择日期', icon: 'none' });
|
||||
return;
|
||||
}
|
||||
if (!timeSlot.trim()) {
|
||||
Taro.showToast({ title: '请输入时段', icon: 'none' });
|
||||
return;
|
||||
}
|
||||
if (!currentPatient) {
|
||||
Taro.showToast({ title: '请先选择就诊人', icon: 'none' });
|
||||
return;
|
||||
}
|
||||
|
||||
setLoading(true);
|
||||
try {
|
||||
await createAppointment({
|
||||
patient_id: currentPatient.id,
|
||||
doctor_id: selectedDoctor.id,
|
||||
schedule_id: '',
|
||||
appointment_date: appointmentDate,
|
||||
time_slot: timeSlot.trim(),
|
||||
reason: reason.trim() || undefined,
|
||||
});
|
||||
Taro.showToast({ title: '预约成功', icon: 'success' });
|
||||
setTimeout(() => {
|
||||
Taro.navigateBack();
|
||||
}, 1500);
|
||||
} catch {
|
||||
Taro.showToast({ title: '预约失败', icon: 'none' });
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
}, [selectedDoctor, appointmentDate, timeSlot, reason, currentPatient]);
|
||||
|
||||
// 步骤切换
|
||||
const goNext = () => {
|
||||
if (currentStep === 0 && !department) {
|
||||
Taro.showToast({ title: '请先选择科室', icon: 'none' });
|
||||
return;
|
||||
}
|
||||
if (currentStep === 1 && !selectedDoctor) {
|
||||
Taro.showToast({ title: '请选择医生', icon: 'none' });
|
||||
return;
|
||||
}
|
||||
setCurrentStep(Math.min(currentStep + 1, 2));
|
||||
};
|
||||
|
||||
const goPrev = () => {
|
||||
setCurrentStep(Math.max(currentStep - 1, 0));
|
||||
};
|
||||
|
||||
const stepLabels = ['选择科室', '选择医生', '选择日期与时段'];
|
||||
|
||||
return (
|
||||
<View className='create-page'>
|
||||
{/* 步骤指示器 */}
|
||||
<View className='step-bar'>
|
||||
{stepLabels.map((label, idx) => (
|
||||
<View
|
||||
className={`step-item ${idx <= currentStep ? 'step-active' : ''} ${idx < currentStep ? 'step-done' : ''}`}
|
||||
key={label}
|
||||
>
|
||||
<View className='step-dot'>
|
||||
{idx < currentStep ? (
|
||||
<Text className='step-check'>✓</Text>
|
||||
) : (
|
||||
<Text className='step-num'>{idx + 1}</Text>
|
||||
)}
|
||||
</View>
|
||||
<Text className='step-label'>{label}</Text>
|
||||
</View>
|
||||
))}
|
||||
</View>
|
||||
|
||||
{/* 步骤连接线 */}
|
||||
<View className='step-line-wrapper'>
|
||||
<View className='step-line'>
|
||||
<View className='step-line-fill' style={{ width: `${(currentStep / 2) * 100}%` }} />
|
||||
</View>
|
||||
</View>
|
||||
|
||||
{/* Step 1: 选择科室 */}
|
||||
{currentStep === 0 && (
|
||||
<View className='step-content'>
|
||||
<Text className='step-title'>请选择就诊科室</Text>
|
||||
<Picker mode='selector' range={DEPARTMENTS} value={deptPickerIndex} onChange={onDepartmentChange}>
|
||||
<View className='picker-card'>
|
||||
<Text className={`picker-value ${department ? '' : 'placeholder'}`}>
|
||||
{department || '点击选择科室'}
|
||||
</Text>
|
||||
<Text className='picker-arrow'>▾</Text>
|
||||
</View>
|
||||
</Picker>
|
||||
</View>
|
||||
)}
|
||||
|
||||
{/* Step 2: 选择医生 */}
|
||||
{currentStep === 1 && (
|
||||
<View className='step-content'>
|
||||
<Text className='step-title'>{department} - 请选择医生</Text>
|
||||
{doctors.length === 0 ? (
|
||||
<View className='empty-state'>
|
||||
<Text className='empty-text'>暂无可选医生</Text>
|
||||
</View>
|
||||
) : (
|
||||
<View className='doctor-list'>
|
||||
{doctors.map((doc) => (
|
||||
<View
|
||||
className={`doctor-card ${selectedDoctor?.id === doc.id ? 'doctor-selected' : ''}`}
|
||||
key={doc.id}
|
||||
onClick={() => onSelectDoctor(doc)}
|
||||
>
|
||||
<View className='doctor-avatar'>
|
||||
<Text className='doctor-avatar-text'>{doc.name.charAt(0)}</Text>
|
||||
</View>
|
||||
<View className='doctor-detail'>
|
||||
<Text className='doctor-name'>{doc.name}</Text>
|
||||
<Text className='doctor-title'>{doc.title || '医生'}</Text>
|
||||
{doc.specialty && <Text className='doctor-specialty'>{doc.specialty}</Text>}
|
||||
</View>
|
||||
{selectedDoctor?.id === doc.id && (
|
||||
<Text className='doctor-check'>✓</Text>
|
||||
)}
|
||||
</View>
|
||||
))}
|
||||
</View>
|
||||
)}
|
||||
</View>
|
||||
)}
|
||||
|
||||
{/* Step 3: 选择日期和时段 */}
|
||||
{currentStep === 2 && (
|
||||
<View className='step-content'>
|
||||
<Text className='step-title'>选择就诊时间</Text>
|
||||
|
||||
<View className='form-group'>
|
||||
<Text className='form-label'>医生</Text>
|
||||
<View className='form-static'>
|
||||
<Text className='form-static-text'>{selectedDoctor?.name} - {department}</Text>
|
||||
</View>
|
||||
</View>
|
||||
|
||||
<View className='form-group'>
|
||||
<Text className='form-label'>就诊日期</Text>
|
||||
<Picker mode='date' value={appointmentDate} onChange={onDateChange}>
|
||||
<View className='picker-card'>
|
||||
<Text className={`picker-value ${appointmentDate ? '' : 'placeholder'}`}>
|
||||
{appointmentDate || '点击选择日期'}
|
||||
</Text>
|
||||
<Text className='picker-arrow'>▾</Text>
|
||||
</View>
|
||||
</Picker>
|
||||
</View>
|
||||
|
||||
<View className='form-group'>
|
||||
<Text className='form-label'>就诊时段</Text>
|
||||
<Input
|
||||
className='form-input'
|
||||
placeholder='例如: 09:00-10:00'
|
||||
value={timeSlot}
|
||||
onInput={onTimeSlotChange}
|
||||
/>
|
||||
</View>
|
||||
|
||||
<View className='form-group'>
|
||||
<Text className='form-label'>备注(选填)</Text>
|
||||
<Input
|
||||
className='form-input'
|
||||
placeholder='请简要描述症状或就诊目的'
|
||||
value={reason}
|
||||
onInput={onReasonChange}
|
||||
/>
|
||||
</View>
|
||||
</View>
|
||||
)}
|
||||
|
||||
{/* 底部操作栏 */}
|
||||
<View className='bottom-bar'>
|
||||
{currentStep > 0 && (
|
||||
<View className='btn btn-prev' onClick={goPrev}>
|
||||
<Text className='btn-text'>上一步</Text>
|
||||
</View>
|
||||
)}
|
||||
{currentStep < 2 ? (
|
||||
<View className='btn btn-next' onClick={goNext}>
|
||||
<Text className='btn-text btn-text-white'>下一步</Text>
|
||||
</View>
|
||||
) : (
|
||||
<View className={`btn btn-submit ${loading ? 'btn-disabled' : ''}`} onClick={loading ? undefined : handleSubmit}>
|
||||
<Text className='btn-text btn-text-white'>{loading ? '提交中...' : '确认预约'}</Text>
|
||||
</View>
|
||||
)}
|
||||
</View>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
216
apps/miniprogram/src/pages/appointment/detail/index.scss
Normal file
216
apps/miniprogram/src/pages/appointment/detail/index.scss
Normal file
@@ -0,0 +1,216 @@
|
||||
@import '../../../styles/variables.scss';
|
||||
|
||||
.detail-page {
|
||||
min-height: 100vh;
|
||||
background: $bg;
|
||||
padding-bottom: 140px;
|
||||
}
|
||||
|
||||
/* 顶部导航 */
|
||||
.detail-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
padding: 32px;
|
||||
padding-top: 48px;
|
||||
background: linear-gradient(135deg, $pri 0%, $pri-d 100%);
|
||||
}
|
||||
|
||||
.back-btn {
|
||||
padding: 8px 0;
|
||||
}
|
||||
|
||||
.back-text {
|
||||
font-size: 28px;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.header-title {
|
||||
font-size: 34px;
|
||||
font-weight: bold;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.header-placeholder {
|
||||
width: 60px;
|
||||
}
|
||||
|
||||
/* 状态卡片 */
|
||||
.status-card {
|
||||
background: $card;
|
||||
border-radius: $r-lg;
|
||||
padding: 40px 32px;
|
||||
margin: -20px 24px 24px;
|
||||
box-shadow: 0 4px 16px rgba(0, 0, 0, 0.08);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
}
|
||||
|
||||
.status-badge {
|
||||
padding: 8px 32px;
|
||||
border-radius: 24px;
|
||||
margin-bottom: 12px;
|
||||
|
||||
.status-badge-text {
|
||||
font-size: 28px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
&.tag-pending {
|
||||
background: $wrn-l;
|
||||
.status-badge-text { color: $wrn; }
|
||||
}
|
||||
|
||||
&.tag-confirmed {
|
||||
background: $acc-l;
|
||||
.status-badge-text { color: $acc; }
|
||||
}
|
||||
|
||||
&.tag-cancelled {
|
||||
background: $bd-l;
|
||||
.status-badge-text { color: $tx3; }
|
||||
}
|
||||
|
||||
&.tag-completed {
|
||||
background: $pri-l;
|
||||
.status-badge-text { color: $pri; }
|
||||
}
|
||||
}
|
||||
|
||||
.status-doctor {
|
||||
font-size: 36px;
|
||||
font-weight: bold;
|
||||
color: $tx;
|
||||
}
|
||||
|
||||
.status-dept {
|
||||
font-size: 26px;
|
||||
color: $tx2;
|
||||
}
|
||||
|
||||
/* 详情信息 */
|
||||
.info-section {
|
||||
margin: 0 24px 24px;
|
||||
background: $card;
|
||||
border-radius: $r;
|
||||
padding: 28px;
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);
|
||||
}
|
||||
|
||||
.section-title {
|
||||
font-size: 30px;
|
||||
font-weight: bold;
|
||||
color: $tx;
|
||||
margin-bottom: 24px;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 16px 0;
|
||||
border-bottom: 1px solid $bd-l;
|
||||
}
|
||||
|
||||
.info-item:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.info-label {
|
||||
font-size: 26px;
|
||||
color: $tx2;
|
||||
}
|
||||
|
||||
.info-value {
|
||||
font-size: 26px;
|
||||
color: $tx;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.info-id {
|
||||
font-size: 22px;
|
||||
color: $tx3;
|
||||
word-break: break-all;
|
||||
max-width: 400px;
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
/* 温馨提示 */
|
||||
.tips-card {
|
||||
margin: 0 24px 24px;
|
||||
background: $wrn-l;
|
||||
border-radius: $r;
|
||||
padding: 24px 28px;
|
||||
}
|
||||
|
||||
.tips-title {
|
||||
font-size: 26px;
|
||||
font-weight: bold;
|
||||
color: $wrn;
|
||||
margin-bottom: 12px;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.tips-text {
|
||||
font-size: 24px;
|
||||
color: $tx2;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
/* 底部操作 */
|
||||
.bottom-bar {
|
||||
position: fixed;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
padding: 20px 24px;
|
||||
padding-bottom: constant(safe-area-inset-bottom);
|
||||
padding-bottom: env(safe-area-inset-bottom);
|
||||
background: $card;
|
||||
box-shadow: 0 -2px 8px rgba(0, 0, 0, 0.06);
|
||||
}
|
||||
|
||||
.cancel-btn {
|
||||
background: $dan-l;
|
||||
border-radius: $r-sm;
|
||||
padding: 24px 0;
|
||||
text-align: center;
|
||||
transition: opacity 0.2s;
|
||||
}
|
||||
|
||||
.cancel-disabled {
|
||||
opacity: 0.5;
|
||||
}
|
||||
|
||||
.cancel-text {
|
||||
font-size: 30px;
|
||||
font-weight: bold;
|
||||
color: $dan;
|
||||
}
|
||||
|
||||
/* 空状态 */
|
||||
.empty-state {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
padding: 120px 0;
|
||||
}
|
||||
|
||||
.empty-icon {
|
||||
font-size: 80px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.empty-text {
|
||||
font-size: 30px;
|
||||
color: $tx2;
|
||||
margin-bottom: 12px;
|
||||
}
|
||||
|
||||
.empty-hint {
|
||||
font-size: 24px;
|
||||
color: $tx3;
|
||||
}
|
||||
158
apps/miniprogram/src/pages/appointment/detail/index.tsx
Normal file
158
apps/miniprogram/src/pages/appointment/detail/index.tsx
Normal file
@@ -0,0 +1,158 @@
|
||||
import React, { useState } from 'react';
|
||||
import { View, Text } from '@tarojs/components';
|
||||
import Taro, { useRouter } from '@tarojs/taro';
|
||||
import { cancelAppointment } from '../../../services/appointment';
|
||||
import type { Appointment } from '../../../services/appointment';
|
||||
import './index.scss';
|
||||
|
||||
const STATUS_MAP: Record<string, { label: string; className: string }> = {
|
||||
pending: { label: '待确认', className: 'tag-pending' },
|
||||
confirmed: { label: '已确认', className: 'tag-confirmed' },
|
||||
cancelled: { label: '已取消', className: 'tag-cancelled' },
|
||||
completed: { label: '已完成', className: 'tag-completed' },
|
||||
};
|
||||
|
||||
export default function AppointmentDetail() {
|
||||
const router = useRouter();
|
||||
const id = router.params.id || '';
|
||||
const [cancelling, setCancelling] = useState(false);
|
||||
|
||||
// 从页面参数或全局缓存获取预约数据
|
||||
const encodedData = router.params.data || '';
|
||||
let appointment: Appointment | null = null;
|
||||
try {
|
||||
if (encodedData) {
|
||||
appointment = JSON.parse(decodeURIComponent(encodedData));
|
||||
}
|
||||
} catch {
|
||||
// 解析失败则尝试从 Storage 获取
|
||||
const cached = Taro.getStorageSync('appointment_detail_cache');
|
||||
if (cached && cached.id === id) {
|
||||
appointment = cached;
|
||||
}
|
||||
}
|
||||
|
||||
// 如果没有传数据,尝试从缓存获取
|
||||
if (!appointment) {
|
||||
const cached = Taro.getStorageSync('appointment_detail_cache');
|
||||
if (cached && cached.id === id) {
|
||||
appointment = cached;
|
||||
}
|
||||
}
|
||||
|
||||
const status = appointment ? (STATUS_MAP[appointment.status] || { label: appointment.status, className: 'tag-pending' }) : { label: '未知', className: 'tag-pending' };
|
||||
const canCancel = appointment && (appointment.status === 'pending' || appointment.status === 'confirmed');
|
||||
|
||||
const handleCancel = async () => {
|
||||
if (!appointment || cancelling) return;
|
||||
|
||||
const res = await Taro.showModal({
|
||||
title: '确认取消',
|
||||
content: '确定要取消此预约吗?取消后需重新预约。',
|
||||
});
|
||||
if (!res.confirm) return;
|
||||
|
||||
setCancelling(true);
|
||||
try {
|
||||
await cancelAppointment(appointment.id, appointment.version);
|
||||
Taro.showToast({ title: '已取消预约', icon: 'success' });
|
||||
setTimeout(() => {
|
||||
Taro.navigateBack();
|
||||
}, 1500);
|
||||
} catch {
|
||||
Taro.showToast({ title: '取消失败', icon: 'none' });
|
||||
} finally {
|
||||
setCancelling(false);
|
||||
}
|
||||
};
|
||||
|
||||
const goBack = () => {
|
||||
Taro.navigateBack();
|
||||
};
|
||||
|
||||
if (!appointment) {
|
||||
return (
|
||||
<View className='detail-page'>
|
||||
<View className='detail-header'>
|
||||
<View className='back-btn' onClick={goBack}>
|
||||
<Text className='back-text'>返回</Text>
|
||||
</View>
|
||||
<Text className='header-title'>预约详情</Text>
|
||||
<View className='header-placeholder' />
|
||||
</View>
|
||||
<View className='empty-state'>
|
||||
<Text className='empty-icon'>📋</Text>
|
||||
<Text className='empty-text'>未找到预约信息</Text>
|
||||
<Text className='empty-hint'>请从预约列表进入</Text>
|
||||
</View>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<View className='detail-page'>
|
||||
{/* 顶部导航 */}
|
||||
<View className='detail-header'>
|
||||
<View className='back-btn' onClick={goBack}>
|
||||
<Text className='back-text'>返回</Text>
|
||||
</View>
|
||||
<Text className='header-title'>预约详情</Text>
|
||||
<View className='header-placeholder' />
|
||||
</View>
|
||||
|
||||
{/* 状态卡片 */}
|
||||
<View className='status-card'>
|
||||
<View className={`status-badge ${status.className}`}>
|
||||
<Text className='status-badge-text'>{status.label}</Text>
|
||||
</View>
|
||||
<Text className='status-doctor'>{appointment.doctor_name}</Text>
|
||||
<Text className='status-dept'>{appointment.department}</Text>
|
||||
</View>
|
||||
|
||||
{/* 详情信息 */}
|
||||
<View className='info-section'>
|
||||
<Text className='section-title'>预约信息</Text>
|
||||
|
||||
<View className='info-item'>
|
||||
<Text className='info-label'>就诊人</Text>
|
||||
<Text className='info-value'>{appointment.patient_name}</Text>
|
||||
</View>
|
||||
|
||||
<View className='info-item'>
|
||||
<Text className='info-label'>就诊日期</Text>
|
||||
<Text className='info-value'>{appointment.appointment_date}</Text>
|
||||
</View>
|
||||
|
||||
<View className='info-item'>
|
||||
<Text className='info-label'>就诊时段</Text>
|
||||
<Text className='info-value'>{appointment.time_slot}</Text>
|
||||
</View>
|
||||
|
||||
<View className='info-item'>
|
||||
<Text className='info-label'>预约单号</Text>
|
||||
<Text className='info-value info-id'>{appointment.id}</Text>
|
||||
</View>
|
||||
</View>
|
||||
|
||||
{/* 温馨提示 */}
|
||||
{(appointment.status === 'pending' || appointment.status === 'confirmed') && (
|
||||
<View className='tips-card'>
|
||||
<Text className='tips-title'>温馨提示</Text>
|
||||
<Text className='tips-text'>请按预约时间提前15分钟到达,携带有效身份证件和医保卡。</Text>
|
||||
</View>
|
||||
)}
|
||||
|
||||
{/* 底部操作 */}
|
||||
{canCancel && (
|
||||
<View className='bottom-bar'>
|
||||
<View
|
||||
className={`cancel-btn ${cancelling ? 'cancel-disabled' : ''}`}
|
||||
onClick={cancelling ? undefined : handleCancel}
|
||||
>
|
||||
<Text className='cancel-text'>{cancelling ? '处理中...' : '取消预约'}</Text>
|
||||
</View>
|
||||
</View>
|
||||
)}
|
||||
</View>
|
||||
);
|
||||
}
|
||||
@@ -1 +1,176 @@
|
||||
@import '../../styles/variables.scss';
|
||||
|
||||
.appointment-page {
|
||||
min-height: 100vh;
|
||||
background: $bg;
|
||||
padding-bottom: 140px;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
background: linear-gradient(135deg, $pri 0%, $pri-d 100%);
|
||||
padding: 32px;
|
||||
padding-top: 48px;
|
||||
}
|
||||
|
||||
.page-title {
|
||||
font-size: 36px;
|
||||
font-weight: bold;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.appointment-list {
|
||||
padding: 0 24px;
|
||||
margin-top: -16px;
|
||||
}
|
||||
|
||||
.appointment-card {
|
||||
background: $card;
|
||||
border-radius: $r;
|
||||
padding: 28px;
|
||||
margin-bottom: 20px;
|
||||
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);
|
||||
}
|
||||
|
||||
.card-top {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.doctor-info {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
}
|
||||
|
||||
.doctor-name {
|
||||
font-size: 32px;
|
||||
font-weight: bold;
|
||||
color: $tx;
|
||||
}
|
||||
|
||||
.department {
|
||||
font-size: 24px;
|
||||
color: $pri;
|
||||
background: $pri-l;
|
||||
padding: 4px 16px;
|
||||
border-radius: 20px;
|
||||
}
|
||||
|
||||
.status-tag {
|
||||
padding: 6px 20px;
|
||||
border-radius: 20px;
|
||||
|
||||
.status-tag-text {
|
||||
font-size: 22px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
&.tag-pending {
|
||||
background: $wrn-l;
|
||||
|
||||
.status-tag-text {
|
||||
color: $wrn;
|
||||
}
|
||||
}
|
||||
|
||||
&.tag-confirmed {
|
||||
background: $acc-l;
|
||||
|
||||
.status-tag-text {
|
||||
color: $acc;
|
||||
}
|
||||
}
|
||||
|
||||
&.tag-cancelled {
|
||||
background: $bd-l;
|
||||
|
||||
.status-tag-text {
|
||||
color: $tx3;
|
||||
}
|
||||
}
|
||||
|
||||
&.tag-completed {
|
||||
background: $pri-l;
|
||||
|
||||
.status-tag-text {
|
||||
color: $pri;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.card-bottom {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 12px;
|
||||
padding-top: 20px;
|
||||
border-top: 1px solid $bd-l;
|
||||
}
|
||||
|
||||
.info-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.info-icon {
|
||||
font-size: 26px;
|
||||
}
|
||||
|
||||
.info-text {
|
||||
font-size: 26px;
|
||||
color: $tx2;
|
||||
}
|
||||
|
||||
.empty-state {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 120px 0 80px;
|
||||
}
|
||||
|
||||
.empty-icon {
|
||||
font-size: 80px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.empty-text {
|
||||
font-size: 30px;
|
||||
color: $tx2;
|
||||
margin-bottom: 12px;
|
||||
}
|
||||
|
||||
.empty-hint {
|
||||
font-size: 24px;
|
||||
color: $tx3;
|
||||
}
|
||||
|
||||
.loading-tip {
|
||||
text-align: center;
|
||||
padding: 24px 0;
|
||||
}
|
||||
|
||||
.loading-text {
|
||||
font-size: 24px;
|
||||
color: $tx3;
|
||||
}
|
||||
|
||||
.fab-btn {
|
||||
position: fixed;
|
||||
bottom: 60px;
|
||||
left: 50%;
|
||||
transform: translateX(-50%);
|
||||
background: linear-gradient(135deg, $pri 0%, $pri-d 100%);
|
||||
padding: 24px 64px;
|
||||
border-radius: 48px;
|
||||
box-shadow: 0 8px 24px rgba($pri, 0.4);
|
||||
z-index: 100;
|
||||
}
|
||||
|
||||
.fab-text {
|
||||
font-size: 30px;
|
||||
color: white;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
@@ -1,12 +1,141 @@
|
||||
import React, { useState, useCallback } from 'react';
|
||||
import { View, Text } from '@tarojs/components';
|
||||
import '../health/index.scss';
|
||||
import Taro, { useDidShow, useReachBottom, usePullDownRefresh } from '@tarojs/taro';
|
||||
import { listAppointments, cancelAppointment } from '../../services/appointment';
|
||||
import type { Appointment } from '../../services/appointment';
|
||||
import './index.scss';
|
||||
|
||||
const STATUS_MAP: Record<string, { label: string; className: string }> = {
|
||||
pending: { label: '待确认', className: 'tag-pending' },
|
||||
confirmed: { label: '已确认', className: 'tag-confirmed' },
|
||||
cancelled: { label: '已取消', className: 'tag-cancelled' },
|
||||
completed: { label: '已完成', className: 'tag-completed' },
|
||||
};
|
||||
|
||||
export default function AppointmentList() {
|
||||
const [appointments, setAppointments] = useState<Appointment[]>([]);
|
||||
const [page, setPage] = useState(1);
|
||||
const [total, setTotal] = useState(0);
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [refreshing, setRefreshing] = useState(false);
|
||||
|
||||
const fetchData = useCallback(async (pageNum: number, isRefresh = false) => {
|
||||
if (loading) return;
|
||||
setLoading(true);
|
||||
try {
|
||||
const res = await listAppointments(pageNum);
|
||||
const list = res.data || [];
|
||||
if (isRefresh) {
|
||||
setAppointments(list);
|
||||
} else {
|
||||
setAppointments((prev) => [...prev, ...list]);
|
||||
}
|
||||
setTotal(res.total);
|
||||
setPage(pageNum);
|
||||
} catch {
|
||||
Taro.showToast({ title: '加载失败', icon: 'none' });
|
||||
} finally {
|
||||
setLoading(false);
|
||||
setRefreshing(false);
|
||||
}
|
||||
}, [loading]);
|
||||
|
||||
useDidShow(() => {
|
||||
fetchData(1, true);
|
||||
});
|
||||
|
||||
usePullDownRefresh(() => {
|
||||
setRefreshing(true);
|
||||
fetchData(1, true).finally(() => {
|
||||
Taro.stopPullDownRefresh();
|
||||
});
|
||||
});
|
||||
|
||||
useReachBottom(() => {
|
||||
if (!loading && appointments.length < total) {
|
||||
fetchData(page + 1);
|
||||
}
|
||||
});
|
||||
|
||||
const goCreate = () => {
|
||||
Taro.navigateTo({ url: '/pages/appointment/create/index' });
|
||||
};
|
||||
|
||||
const goDetail = (id: string) => {
|
||||
const item = appointments.find((a) => a.id === id);
|
||||
if (item) {
|
||||
Taro.setStorageSync('appointment_detail_cache', item);
|
||||
}
|
||||
Taro.navigateTo({ url: `/pages/appointment/detail/index?id=${id}` });
|
||||
};
|
||||
|
||||
const getStatusTag = (status: string) => {
|
||||
return STATUS_MAP[status] || { label: status, className: 'tag-pending' };
|
||||
};
|
||||
|
||||
export default function Appointment() {
|
||||
return (
|
||||
<View className='placeholder-page'>
|
||||
<Text className='placeholder-icon'>📅</Text>
|
||||
<Text className='placeholder-title'>预约挂号</Text>
|
||||
<Text className='placeholder-desc'>选择科室、医生、时段</Text>
|
||||
<View className='appointment-page'>
|
||||
{/* 页面标题 */}
|
||||
<View className='page-header'>
|
||||
<Text className='page-title'>预约挂号</Text>
|
||||
</View>
|
||||
|
||||
{/* 预约列表 */}
|
||||
{appointments.length === 0 && !loading ? (
|
||||
<View className='empty-state'>
|
||||
<Text className='empty-icon'>📋</Text>
|
||||
<Text className='empty-text'>暂无预约记录</Text>
|
||||
<Text className='empty-hint'>点击下方按钮新建预约</Text>
|
||||
</View>
|
||||
) : (
|
||||
<View className='appointment-list'>
|
||||
{appointments.map((item) => {
|
||||
const tag = getStatusTag(item.status);
|
||||
return (
|
||||
<View
|
||||
className='appointment-card'
|
||||
key={item.id}
|
||||
onClick={() => goDetail(item.id)}
|
||||
>
|
||||
<View className='card-top'>
|
||||
<View className='doctor-info'>
|
||||
<Text className='doctor-name'>{item.doctor_name}</Text>
|
||||
<Text className='department'>{item.department}</Text>
|
||||
</View>
|
||||
<View className={`status-tag ${tag.className}`}>
|
||||
<Text className='status-tag-text'>{tag.label}</Text>
|
||||
</View>
|
||||
</View>
|
||||
<View className='card-bottom'>
|
||||
<View className='info-row'>
|
||||
<Text className='info-icon'>📅</Text>
|
||||
<Text className='info-text'>{item.appointment_date}</Text>
|
||||
</View>
|
||||
<View className='info-row'>
|
||||
<Text className='info-icon'>🕐</Text>
|
||||
<Text className='info-text'>{item.time_slot}</Text>
|
||||
</View>
|
||||
</View>
|
||||
</View>
|
||||
);
|
||||
})}
|
||||
{loading && (
|
||||
<View className='loading-tip'>
|
||||
<Text className='loading-text'>加载中...</Text>
|
||||
</View>
|
||||
)}
|
||||
{!loading && appointments.length >= total && total > 0 && (
|
||||
<View className='loading-tip'>
|
||||
<Text className='loading-text'>没有更多了</Text>
|
||||
</View>
|
||||
)}
|
||||
</View>
|
||||
)}
|
||||
|
||||
{/* 底部悬浮按钮 */}
|
||||
<View className='fab-btn' onClick={goCreate}>
|
||||
<Text className='fab-text'>+ 新建预约</Text>
|
||||
</View>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user