feat(miniprogram): 老年友好版本全面重设计 — 5→4 Tab + 首页/健康/消息/我的重写
Some checks failed
CI / rust-check (push) Has been cancelled
CI / rust-test (push) Has been cancelled
CI / frontend-build (push) Has been cancelled
CI / security-audit (push) Has been cancelled

- TabBar 从 5 Tab 调整为 4 Tab(首页/健康/消息/我的)
- 首页重写为 5 区域布局:问候+进度环+体征2x2+待办+快捷操作
- 健康页重写:体征录入大输入框+趋势柱状图+BLE设备卡片
- 新建消息页:咨询对话+系统通知双 Tab
- 我的页调整:菜单高度64px+新增积分商城入口
- 设计系统更新:色彩对比度提升(WCAG AA)+触控参数+老年友好 mixin
- 新增 ProgressRing 组件(CSS conic-gradient 实现)
- 修复 diagnoses 页面 $suc-l 未定义变量
This commit is contained in:
iven
2026-04-30 22:51:05 +08:00
parent 813843e8cc
commit 50772878da
14 changed files with 1256 additions and 771 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 333 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 334 B

View File

@@ -0,0 +1,29 @@
@import '../styles/variables.scss';
.progress-ring {
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
}
.progress-ring-inner {
background: $card;
border-radius: 50%;
display: flex;
align-items: baseline;
justify-content: center;
}
.progress-ring-percent {
font-family: 'Georgia', 'Times New Roman', serif;
font-size: 22px;
font-weight: bold;
line-height: 1;
}
.progress-ring-unit {
font-size: 12px;
font-weight: 600;
line-height: 1;
}

View File

@@ -0,0 +1,40 @@
import { View, Text } from '@tarojs/components';
import './ProgressRing.scss';
interface ProgressRingProps {
percent: number;
size?: number;
strokeWidth?: number;
color?: string;
trackColor?: string;
}
export default function ProgressRing({
percent,
size = 72,
strokeWidth = 7,
color = '#C4623A',
trackColor = '#E8E2DC',
}: ProgressRingProps) {
const clamped = Math.max(0, Math.min(100, percent));
const innerSize = size - strokeWidth * 2;
return (
<View
className='progress-ring'
style={`width:${size}px;height:${size}px;background:conic-gradient(${color} ${clamped}%, ${trackColor} ${clamped}%);border-radius:50%;padding:${strokeWidth}px;`}
>
<View
className='progress-ring-inner'
style={`width:${innerSize}px;height:${innerSize}px;`}
>
<Text className='progress-ring-percent' style={`color:${color};`}>
{clamped}
</Text>
<Text className='progress-ring-unit' style={`color:${color};`}>
%
</Text>
</View>
</View>
);
}

View File

@@ -9,9 +9,6 @@
/* ─── 页头 ─── */ /* ─── 页头 ─── */
.health-header { .health-header {
display: flex;
align-items: center;
justify-content: space-between;
padding: 24px 32px 8px; padding: 24px 32px 8px;
} }
@@ -22,248 +19,272 @@
color: $tx; color: $tx;
} }
.health-add-btn { /* ─── 类型 Tab ─── */
background: $pri; .vital-tabs {
padding: 10px 28px; display: flex;
padding: 12px 24px;
gap: 12px;
}
.vital-tab {
flex: 1;
height: $tab-h;
border-radius: $r;
background: $surface-alt;
@include flex-center;
position: relative;
&:active {
opacity: 0.85;
}
&.vital-tab-active {
background: $pri;
.vital-tab-text {
color: #fff;
}
}
}
.vital-tab-text {
font-size: 26px;
font-weight: 600;
color: $tx2;
}
.vital-tab-dot {
position: absolute;
top: 10px;
right: 10px;
width: 8px;
height: 8px;
border-radius: 50%;
background: $wrn;
}
/* ─── 录入区 ─── */
.input-section {
margin: 0 24px 24px;
background: $card;
border-radius: $r;
padding: 24px;
box-shadow: $shadow-sm;
}
.input-group {
margin-bottom: 20px;
}
.input-label {
font-size: 26px;
color: $tx;
font-weight: 600;
display: block;
margin-bottom: 12px;
}
.input-field {
height: 56px;
background: $bg;
border: 2px solid $bd;
border-radius: $r-sm; border-radius: $r-sm;
padding: 0 20px;
font-family: 'Georgia', 'Times New Roman', serif;
font-size: 28px;
color: $tx;
width: 100%;
box-sizing: border-box;
}
.input-ref {
font-size: 22px;
color: $tx2;
display: block;
margin-top: 12px;
}
/* ─── 血糖时段选择 ─── */
.period-group {
display: flex;
gap: 12px;
margin-top: 16px;
}
.period-btn {
flex: 1;
height: $btn-primary-h;
border-radius: $r-sm;
background: $surface-alt;
@include flex-center;
&.period-active {
background: $pri;
.period-btn-text {
color: #fff;
}
}
&:active { &:active {
opacity: 0.85; opacity: 0.85;
} }
} }
.health-add-text { .period-btn-text {
font-size: 26px; font-size: 26px;
color: #fff;
font-weight: 600; font-weight: 600;
color: $tx2;
} }
/* ─── 快捷操作 ─── */ /* ─── 保存按钮 ─── */
.health-actions-row { .save-btn {
display: flex; @include btn-primary;
flex-wrap: wrap; margin-top: 24px;
gap: 12px; border-radius: $r;
padding: 16px 24px 24px;
} }
.action-item { .save-btn-text {
flex: 1; font-size: 28px;
min-width: 100px; font-weight: 600;
color: #fff;
}
/* ─── 趋势图 ─── */
.trend-section {
margin: 0 24px 24px;
}
.trend-empty {
background: $card; background: $card;
border-radius: $r; border-radius: $r;
padding: 20px 12px; padding: 36px;
text-align: center;
}
.trend-empty-text {
font-size: 24px;
color: $tx2;
}
.trend-chart {
background: $card;
border-radius: $r;
padding: 24px;
box-shadow: $shadow-sm;
}
.trend-bars {
display: flex;
align-items: flex-end;
height: 200px;
gap: 8px;
}
.trend-bar-col {
flex: 1;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
align-items: center; align-items: center;
gap: 8px; height: 100%;
box-shadow: $shadow-sm; justify-content: flex-end;
}
&:active { .trend-bar {
opacity: 0.7; width: 100%;
max-width: 40px;
border-radius: 6px 6px 0 0;
min-height: 16px;
&.trend-bar-normal {
background: $pri;
}
&.trend-bar-warn {
background: $wrn;
} }
} }
.action-icon { .trend-bar-label {
width: 72px; font-size: 22px;
height: 72px; color: $tx2;
border-radius: 50%; margin-top: 8px;
@include flex-center;
&.icon-primary { background: $pri-l; }
&.icon-accent { background: $acc-l; }
&.icon-warn { background: $wrn-l; }
} }
.action-char { /* ─── BLE 设备卡片 ─── */
font-family: 'Georgia', 'Times New Roman', serif; .device-section {
font-size: 32px; margin: 0 24px 16px;
font-weight: bold;
color: $pri;
.icon-accent & { color: $acc; }
.icon-warn & { color: $wrn; }
} }
.action-label { .device-card {
font-size: 24px; display: flex;
color: $tx; align-items: center;
font-weight: 500;
}
/* ─── 通用 section ─── */
.health-section {
margin: 0 24px 28px;
}
/* ─── 体征概览 ─── */
.vitals-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 16px;
}
.vital-card {
background: $card; background: $card;
border-radius: $r; border-radius: $r;
padding: 24px 20px; padding: 24px;
box-shadow: $shadow-sm; box-shadow: $shadow-sm;
transition: opacity 0.2s;
&:active { &:active {
opacity: 0.7; opacity: 0.85;
} }
} }
.vital-label { .device-icon {
width: 56px;
height: 56px;
border-radius: $r-sm;
background: $pri-l;
@include flex-center;
margin-right: 16px;
flex-shrink: 0;
}
.device-icon-text {
font-size: 26px;
font-weight: bold;
color: $pri;
}
.device-info {
flex: 1;
min-width: 0;
}
.device-name {
font-size: 28px;
font-weight: 600;
color: $tx;
display: block;
margin-bottom: 4px;
}
.device-desc {
font-size: 22px; font-size: 22px;
color: $tx2; color: $tx2;
display: block; display: block;
margin-bottom: 10px;
} }
.vital-value { .device-arrow {
@include serif-number; font-size: 32px;
font-size: 44px;
font-weight: bold;
color: $tx;
display: block;
margin-bottom: 8px;
line-height: 1.1;
}
.vital-bottom {
display: flex;
justify-content: space-between;
align-items: center;
}
.vital-unit {
font-size: 20px;
color: $tx3; color: $tx3;
flex-shrink: 0;
} }
.vital-tag { /* ─── 健康资讯入口 ─── */
@include tag($acc-l, $acc); .article-entry {
margin: 0 24px 24px;
&.tag-warn {
@include tag($wrn-l, $wrn);
}
}
.vital-ref {
font-size: 20px;
color: $tx3;
margin-top: 8px;
display: block;
}
.vital-bar-track {
height: 6px;
background: $bd-l;
border-radius: 3px;
margin-top: 12px;
overflow: hidden;
}
.vital-bar-fill {
height: 100%;
border-radius: 3px;
transition: width 0.3s ease;
&.bar-green { background: $acc; }
&.bar-orange { background: $wrn; }
&.bar-red { background: $dan; }
}
/* ─── 趋势入口 — 水平滚动卡片 ─── */
.trend-scroll {
white-space: nowrap;
width: 100%;
}
.trend-card {
display: inline-flex;
flex-direction: column;
align-items: center;
width: 200px;
background: $card; background: $card;
border-radius: $r; border-radius: $r;
padding: 24px 16px; padding: 24px;
margin-right: 16px;
box-shadow: $shadow-sm; box-shadow: $shadow-sm;
vertical-align: top;
&:active { &:active {
opacity: 0.7; opacity: 0.85;
} }
} }
.trend-card-icon { .article-entry-text {
width: 64px;
height: 64px;
border-radius: $r;
background: $pri-l;
@include flex-center;
margin-bottom: 12px;
}
.trend-card-char {
font-family: 'Georgia', 'Times New Roman', serif;
font-size: 28px;
font-weight: bold;
color: $pri;
}
.trend-card-label {
font-size: 26px; font-size: 26px;
color: $tx;
font-weight: 500;
white-space: nowrap;
}
.trend-card-arrow {
font-size: 22px;
color: $tx3;
margin-top: 8px;
}
/* ─── 最近监测 ─── */
.record-card {
background: $card;
border-radius: $r;
padding: 20px 24px;
margin-bottom: 12px;
box-shadow: $shadow-sm;
}
.record-date {
font-size: 24px;
color: $pri; color: $pri;
font-weight: 600;
display: block;
margin-bottom: 12px;
}
.record-data {
display: flex;
gap: 32px;
flex-wrap: wrap;
}
.record-item {
display: flex;
flex-direction: column;
gap: 4px;
}
.record-item-label {
font-size: 22px;
color: $tx3;
}
.record-item-value {
@include serif-number;
font-size: 26px;
color: $tx;
font-weight: 500; font-weight: 500;
} }

View File

@@ -1,235 +1,330 @@
import { useState } from 'react'; import { useState } from 'react';
import { View, Text, ScrollView } from '@tarojs/components'; import { View, Text, Input } from '@tarojs/components';
import Taro, { useDidShow } from '@tarojs/taro'; import Taro, { useDidShow } from '@tarojs/taro';
import { useHealthStore } from '../../stores/health'; import { useHealthStore } from '../../stores/health';
import { listDailyMonitoring, DailyMonitoring } from '../../services/health';
import { usePointsStore } from '../../stores/points';
import { useAuthStore } from '../../stores/auth'; import { useAuthStore } from '../../stores/auth';
import { trackEvent } from '../../services/analytics'; import { inputVitalSign, getTrend } from '../../services/health';
import Loading from '../../components/Loading'; import Loading from '../../components/Loading';
import './index.scss'; import './index.scss';
const QUICK_ACTIONS = [ type VitalType = 'blood_pressure' | 'heart_rate' | 'blood_sugar' | 'weight';
{ label: '日常上报', char: '日', bg: 'icon-primary' },
{ label: '体征录入', char: '录', bg: 'icon-accent' }, const VITAL_TABS: { key: VitalType; label: string }[] = [
{ label: '查看趋势', char: '势', bg: 'icon-warn' }, { key: 'blood_pressure', label: '血压' },
{ key: 'heart_rate', label: '心率' },
{ key: 'blood_sugar', label: '血糖' },
{ key: 'weight', label: '体重' },
]; ];
const TREND_LINKS = [ const REF_RANGES: Record<VitalType, { range: string; warn: string }> = {
{ label: '血压趋势', indicator: 'blood_pressure_systolic', char: '' }, blood_pressure: { range: '收缩压 90-140 / 舒张压 60-90 mmHg', warn: '血压偏高,确认提交?' },
{ label: '心率趋势', indicator: 'heart_rate', char: '' }, heart_rate: { range: '60-100 bpm', warn: '心率异常,确认提交?' },
{ label: '血糖趋势', indicator: 'blood_sugar_fasting', char: '' }, blood_sugar: { range: '空腹 3.9-6.1 / 餐后 <7.8 mmol/L', warn: '血糖偏高,确认提交?' },
]; weight: { range: '根据 BMI 18.5-24 计算', warn: '' },
};
function getStatusTag(status?: string) { interface TrendPoint {
if (status === 'high') return { label: '偏高', cls: 'tag-warn' }; date: string;
if (status === 'low') return { label: '偏低', cls: 'tag-warn' }; value: number;
if (status === 'normal') return { label: '正常', cls: 'tag-ok' };
return null;
}
/** 根据 status 计算 sparkline bar 的颜色 */
function getBarColor(status?: string): string {
if (status === 'normal') return 'bar-green';
if (status === 'high' || status === 'low') return 'bar-orange';
return 'bar-green';
}
/** 计算数值在参考范围中的位置百分比 (0-100) */
function getBarPercent(value: number | undefined, ref?: string): number {
if (!value || !ref) return 50;
const match = ref.match(/([\d.]+)\s*[-]\s*([\d.]+)/);
if (!match) return 50;
const low = parseFloat(match[1]);
const high = parseFloat(match[2]);
if (high <= low) return 50;
// 将值映射到 0-100 范围,参考范围占据中间 70%15%-85%
const range = high - low;
const normalized = (value - low + range * 0.3) / (range * 1.6);
return Math.max(5, Math.min(95, normalized * 100));
} }
export default function Health() { export default function Health() {
const { todaySummary, loading, refreshToday } = useHealthStore(); const { todaySummary, loading, refreshToday, getTrend: fetchTrend } = useHealthStore();
const { checkinStatus, refresh: refreshPoints } = usePointsStore();
const { currentPatient } = useAuthStore(); const { currentPatient } = useAuthStore();
const [recentRecords, setRecentRecords] = useState<DailyMonitoring[]>([]); const [activeTab, setActiveTab] = useState<VitalType>('blood_pressure');
const [systolic, setSystolic] = useState('');
const [diastolic, setDiastolic] = useState('');
const [heartRateVal, setHeartRateVal] = useState('');
const [sugarVal, setSugarVal] = useState('');
const [sugarPeriod, setSugarPeriod] = useState<'fasting' | 'postprandial'>('fasting');
const [weightVal, setWeightVal] = useState('');
const [saving, setSaving] = useState(false);
const [trendData, setTrendData] = useState<TrendPoint[]>([]);
const [trendLoading, setTrendLoading] = useState(false);
useDidShow(() => { useDidShow(() => {
refreshToday(); refreshToday();
refreshPoints(); loadTrend(activeTab);
loadRecentRecords();
}); });
const loadRecentRecords = async () => { const loadTrend = async (type: VitalType) => {
if (currentPatient) { setTrendLoading(true);
try { try {
const resp = await listDailyMonitoring(currentPatient.id, { page: 1, page_size: 3 }); const indicatorMap: Record<VitalType, string> = {
setRecentRecords(resp.data || []); blood_pressure: 'blood_pressure_systolic',
} catch { heart_rate: 'heart_rate',
// daily monitoring API 可能不可用 blood_sugar: 'blood_sugar_fasting',
weight: 'weight',
};
const points = await fetchTrend(indicatorMap[type], '7d');
setTrendData(points);
} catch {
setTrendData([]);
} finally {
setTrendLoading(false);
}
};
const handleTabChange = (tab: VitalType) => {
setActiveTab(tab);
loadTrend(tab);
};
const getWarnStatus = (type: VitalType): string | null => {
if (type === 'blood_pressure') {
const sys = parseFloat(systolic);
const dia = parseFloat(diastolic);
if (sys > 140 || dia > 90) return REF_RANGES.blood_pressure.warn;
} else if (type === 'heart_rate') {
const val = parseFloat(heartRateVal);
if (val > 100 || val < 60) return REF_RANGES.heart_rate.warn;
} else if (type === 'blood_sugar') {
const val = parseFloat(sugarVal);
if (sugarPeriod === 'fasting' && val > 6.1) return REF_RANGES.blood_sugar.warn;
if (sugarPeriod === 'postprandial' && val > 7.8) return REF_RANGES.blood_sugar.warn;
}
return null;
};
const handleSave = async () => {
const patientId = currentPatient?.id;
if (!patientId) {
Taro.showToast({ title: '请先登录', icon: 'none' });
return;
}
const warnMsg = getWarnStatus(activeTab);
if (warnMsg) {
const { confirm } = await Taro.showModal({
title: '异常提示',
content: warnMsg,
confirmText: '确认提交',
cancelText: '再看看',
});
if (!confirm) return;
}
setSaving(true);
try {
switch (activeTab) {
case 'blood_pressure': {
const sys = parseFloat(systolic);
const dia = parseFloat(diastolic);
if (!sys || !dia) { Taro.showToast({ title: '请填写完整', icon: 'none' }); return; }
await inputVitalSign(patientId, {
indicator_type: 'blood_pressure',
value: sys,
extra: { systolic: sys, diastolic: dia },
});
setSystolic('');
setDiastolic('');
break;
}
case 'heart_rate': {
const val = parseFloat(heartRateVal);
if (!val) { Taro.showToast({ title: '请填写心率', icon: 'none' }); return; }
await inputVitalSign(patientId, { indicator_type: 'heart_rate', value: val });
setHeartRateVal('');
break;
}
case 'blood_sugar': {
const val = parseFloat(sugarVal);
if (!val) { Taro.showToast({ title: '请填写血糖值', icon: 'none' }); return; }
await inputVitalSign(patientId, { indicator_type: 'blood_sugar', value: val });
setSugarVal('');
break;
}
case 'weight': {
const val = parseFloat(weightVal);
if (!val) { Taro.showToast({ title: '请填写体重', icon: 'none' }); return; }
await inputVitalSign(patientId, { indicator_type: 'weight', value: val });
setWeightVal('');
break;
}
} }
Taro.showToast({ title: '保存成功', icon: 'success' });
refreshToday(true);
loadTrend(activeTab);
} catch {
Taro.showToast({ title: '保存失败', icon: 'none' });
} finally {
setSaving(false);
} }
}; };
const goToInput = () => { const maxTrendValue = Math.max(...trendData.map((d) => d.value), 1);
Taro.navigateTo({ url: '/pages/pkg-health/input/index' }); const dayLabels = ['日', '一', '二', '三', '四', '五', '六'];
};
const goToDailyMonitoring = () => {
Taro.navigateTo({ url: '/pages/pkg-health/daily-monitoring/index' });
};
const goToTrend = (indicator: string) => {
Taro.navigateTo({ url: `/pages/pkg-health/trend/index?indicator=${indicator}` });
};
const goToMall = () => {
Taro.switchTab({ url: '/pages/mall/index' });
};
const summary = todaySummary || {};
const items = [
{ label: '血压', value: summary.blood_pressure ? `${summary.blood_pressure.systolic}/${summary.blood_pressure.diastolic}` : '--/--', unit: 'mmHg', indicator: 'blood_pressure_systolic', status: summary.blood_pressure?.status, ref: summary.blood_pressure?.reference_range, numValue: summary.blood_pressure?.systolic },
{ label: '心率', value: summary.heart_rate ? `${summary.heart_rate.value}` : '--', unit: 'bpm', indicator: 'heart_rate', status: summary.heart_rate?.status, ref: summary.heart_rate?.reference_range, numValue: summary.heart_rate?.value },
{ label: '血糖', value: summary.blood_sugar ? `${summary.blood_sugar.value}` : '--', unit: 'mmol/L', indicator: 'blood_sugar_fasting', status: summary.blood_sugar?.status, ref: summary.blood_sugar?.reference_range, numValue: summary.blood_sugar?.value },
{ label: '体重', value: summary.weight ? `${summary.weight.value}` : '--', unit: 'kg', indicator: 'weight', status: summary.weight?.status, ref: summary.weight?.reference_range, numValue: summary.weight?.value },
];
const quickActions = [
{ ...QUICK_ACTIONS[0], action: goToDailyMonitoring },
{ ...QUICK_ACTIONS[1], action: goToInput },
{ ...QUICK_ACTIONS[2], action: () => goToTrend('blood_pressure_systolic') },
];
const trendLinks = TREND_LINKS;
const formatBp = (record: DailyMonitoring) => {
const parts: string[] = [];
if (record.morning_bp_systolic && record.morning_bp_diastolic) {
parts.push(`${record.morning_bp_systolic}/${record.morning_bp_diastolic}`);
}
if (record.evening_bp_systolic && record.evening_bp_diastolic) {
parts.push(`${record.evening_bp_systolic}/${record.evening_bp_diastolic}`);
}
return parts.length > 0 ? parts.join(' ') : '--';
};
return ( return (
<View className='health-page'> <View className='health-page'>
{/* 页头 */} {/* 页头 */}
<View className='health-header'> <View className='health-header'>
<Text className='health-title'></Text> <Text className='health-title'></Text>
<View className='health-add-btn' onClick={goToInput}> </View>
<Text className='health-add-text'></Text>
{/* 类型 Tab */}
<View className='vital-tabs'>
{VITAL_TABS.map((tab) => {
const hasData = tab.key === 'blood_pressure' ? !!todaySummary?.blood_pressure
: tab.key === 'heart_rate' ? !!todaySummary?.heart_rate
: tab.key === 'blood_sugar' ? !!todaySummary?.blood_sugar
: !!todaySummary?.weight;
return (
<View
key={tab.key}
className={`vital-tab ${activeTab === tab.key ? 'vital-tab-active' : ''}`}
onClick={() => handleTabChange(tab.key)}
>
<Text className='vital-tab-text'>{tab.label}</Text>
{!hasData && <View className='vital-tab-dot' />}
</View>
);
})}
</View>
{/* 录入区 */}
<View className='input-section'>
{activeTab === 'blood_pressure' && (
<View className='input-group'>
<Text className='input-label'></Text>
<Input
className='input-field'
type='number'
placeholder='如 130'
value={systolic}
onInput={(e) => setSystolic(e.detail.value)}
/>
<Text className='input-label' style='margin-top:20px;'></Text>
<Input
className='input-field'
type='number'
placeholder='如 85'
value={diastolic}
onInput={(e) => setDiastolic(e.detail.value)}
/>
<Text className='input-ref'>{REF_RANGES.blood_pressure.range}</Text>
</View>
)}
{activeTab === 'heart_rate' && (
<View className='input-group'>
<Text className='input-label'></Text>
<Input
className='input-field'
type='digit'
placeholder='如 72'
value={heartRateVal}
onInput={(e) => setHeartRateVal(e.detail.value)}
/>
<Text className='input-ref'>{REF_RANGES.heart_rate.range}</Text>
</View>
)}
{activeTab === 'blood_sugar' && (
<View className='input-group'>
<Text className='input-label'></Text>
<Input
className='input-field'
type='digit'
placeholder='如 5.6'
value={sugarVal}
onInput={(e) => setSugarVal(e.detail.value)}
/>
<View className='period-group'>
<View
className={`period-btn ${sugarPeriod === 'fasting' ? 'period-active' : ''}`}
onClick={() => setSugarPeriod('fasting')}
>
<Text className='period-btn-text'></Text>
</View>
<View
className={`period-btn ${sugarPeriod === 'postprandial' ? 'period-active' : ''}`}
onClick={() => setSugarPeriod('postprandial')}
>
<Text className='period-btn-text'> 2h</Text>
</View>
</View>
<Text className='input-ref'>{REF_RANGES.blood_sugar.range}</Text>
</View>
)}
{activeTab === 'weight' && (
<View className='input-group'>
<Text className='input-label'> (kg)</Text>
<Input
className='input-field'
type='digit'
placeholder='如 65.5'
value={weightVal}
onInput={(e) => setWeightVal(e.detail.value)}
/>
<Text className='input-ref'>{REF_RANGES.weight.range}</Text>
</View>
)}
<View className='save-btn' onClick={handleSave}>
<Text className='save-btn-text'>{saving ? '保存中...' : '保存'}</Text>
</View> </View>
</View> </View>
{/* 快捷操作 + 打卡状态紧凑合并 */} {/* 趋势图 */}
<View className='health-actions-row'> <View className='trend-section'>
{quickActions.map((a) => ( <Text className='section-title'> 7 </Text>
<View className='action-item' key={a.label} onClick={a.action}> {trendLoading ? (
<View className={`action-icon ${a.bg}`}>
<Text className='action-char'>{a.char}</Text>
</View>
<Text className='action-label'>{a.label}</Text>
</View>
))}
{checkinStatus && (
<View
className='action-item checkin-badge'
onClick={!checkinStatus.checked_in_today ? goToMall : undefined}
>
<View className={`action-icon ${checkinStatus.checked_in_today ? 'icon-accent' : 'icon-warn'}`}>
<Text className='action-char'></Text>
</View>
<Text className='action-label'>
{checkinStatus.checked_in_today
? (checkinStatus.consecutive_days > 0 ? `已打卡${checkinStatus.consecutive_days}` : '已打卡')
: '去打卡'}
</Text>
</View>
)}
</View>
{/* 今日体征概览 */}
<View className='health-section'>
<Text className='section-title'></Text>
{loading && !todaySummary ? (
<Loading /> <Loading />
) : trendData.length === 0 ? (
<View className='trend-empty'>
<Text className='trend-empty-text'></Text>
</View>
) : ( ) : (
<View className='vitals-grid'> <View className='trend-chart'>
{items.map((item) => { <View className='trend-bars'>
const tag = getStatusTag(item.status); {trendData.map((point, i) => {
const barColor = getBarColor(item.status); const heightPct = Math.max(8, (point.value / maxTrendValue) * 100);
const barPercent = getBarPercent(item.numValue, item.ref); const isAbnormal = activeTab === 'blood_pressure' ? point.value > 140
return ( : activeTab === 'heart_rate' ? (point.value > 100 || point.value < 60)
<View className='vital-card' key={item.label} onClick={() => goToTrend(item.indicator)}> : activeTab === 'blood_sugar' ? point.value > 6.1
<Text className='vital-label'>{item.label}</Text> : false;
<Text className='vital-value'>{item.value}</Text> const dayOfWeek = new Date(point.date).getDay();
<View className='vital-bottom'> return (
<Text className='vital-unit'>{item.unit}</Text> <View className='trend-bar-col' key={i}>
{tag && <Text className={`vital-tag ${tag.cls}`}>{tag.label}</Text>} <View
className={`trend-bar ${isAbnormal ? 'trend-bar-warn' : 'trend-bar-normal'}`}
style={`height:${heightPct}%;`}
/>
<Text className='trend-bar-label'>{dayLabels[dayOfWeek]}</Text>
</View> </View>
{/* Sparkline bar */} );
{item.ref && item.numValue != null && ( })}
<View className='vital-bar-track'> </View>
<View className={`vital-bar-fill ${barColor}`} style={`width: ${barPercent}%`} />
</View>
)}
{item.ref && <Text className='vital-ref'> {item.ref}</Text>}
</View>
);
})}
</View> </View>
)} )}
</View> </View>
{/* 趋势快捷入口 — 水平滚动卡片 */} {/* BLE 设备卡片 */}
<View className='health-section'> <View className='device-section'>
<Text className='section-title'></Text> <View
<ScrollView className='trend-scroll' scrollX> className='device-card'
{trendLinks.map((t) => ( onClick={() => Taro.navigateTo({ url: '/pages/device-sync/index' })}
<View className='trend-card' key={t.label} onClick={() => goToTrend(t.indicator)}> >
<View className='trend-card-icon'> <View className='device-icon'>
<Text className='trend-card-char'>{t.char}</Text> <Text className='device-icon-text'></Text>
</View> </View>
<Text className='trend-card-label'>{t.label}</Text> <View className='device-info'>
<Text className='trend-card-arrow'> </Text> <Text className='device-name'></Text>
</View> <Text className='device-desc'></Text>
))} </View>
</ScrollView> <Text className='device-arrow'></Text>
</View>
</View> </View>
{/* 最近监测记录 */} {/* 健康资讯入口 */}
{recentRecords.length > 0 && ( <View
<View className='health-section'> className='article-entry'
<Text className='section-title'></Text> onClick={() => Taro.navigateTo({ url: '/pages/article/index' })}
{recentRecords.map((record) => ( >
<View className='record-card' key={record.id}> <Text className='article-entry-text'> </Text>
<Text className='record-date'>{record.record_date}</Text> </View>
<View className='record-data'>
<View className='record-item'>
<Text className='record-item-label'></Text>
<Text className='record-item-value'>{formatBp(record)}</Text>
</View>
{record.weight != null && (
<View className='record-item'>
<Text className='record-item-label'></Text>
<Text className='record-item-value'>{record.weight} kg</Text>
</View>
)}
{record.blood_sugar != null && (
<View className='record-item'>
<Text className='record-item-label'></Text>
<Text className='record-item-value'>{record.blood_sugar} mmol/L</Text>
</View>
)}
</View>
</View>
))}
</View>
)}
</View> </View>
); );
} }

View File

@@ -7,101 +7,160 @@
padding-bottom: calc(120px + env(safe-area-inset-bottom)); padding-bottom: calc(120px + env(safe-area-inset-bottom));
} }
/* ─── 问候区 ─── */ /* ─── 区域 1问候 + 日期 + 消息 ─── */
.greeting-section { .greeting-section {
background: linear-gradient(135deg, $pri 0%, $pri-d 100%); background: linear-gradient(135deg, $pri 0%, $pri-d 100%);
padding: 48px 32px 72px; padding: 48px 32px 60px;
color: #fff; color: #fff;
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: flex-start; align-items: center;
} }
.greeting-left { .greeting-left {
display: flex; flex: 1;
flex-direction: column;
} }
.greeting-time { .greeting-text {
font-size: 26px; font-size: 28px;
opacity: 0.85;
margin-bottom: 4px;
}
.greeting-name {
font-family: 'Georgia', 'Times New Roman', serif;
font-size: 44px;
font-weight: bold; font-weight: bold;
display: block;
margin-bottom: 6px;
} }
.greeting-date { .greeting-date {
font-size: 24px; font-size: 22px;
opacity: 0.7; opacity: 0.75;
margin-top: 8px;
} }
/* ─── 今日健康 ─── */ .greeting-msg {
.health-section { width: 56px;
height: 56px;
border-radius: 50%;
background: rgba(255, 255, 255, 0.2);
@include flex-center;
&:active {
background: rgba(255, 255, 255, 0.1);
}
}
.greeting-msg-icon {
font-size: 22px;
color: #fff;
font-weight: 600;
}
/* ─── 区域 2今日体征完成度 ─── */
.checkin-card {
background: $card; background: $card;
border-radius: $r; border-radius: $r;
box-shadow: $shadow-md; box-shadow: $shadow-md;
margin: -36px 24px 24px; margin: -28px 24px 24px;
padding: 28px; padding: 24px;
display: flex;
align-items: center;
gap: 24px;
&:active {
opacity: 0.9;
}
} }
.section-title { .checkin-left {
@include section-title; flex-shrink: 0;
} }
.health-grid { .checkin-right {
flex: 1;
min-width: 0;
}
.checkin-title {
font-size: 26px;
font-weight: 600;
color: $tx;
display: block;
margin-bottom: 12px;
}
.checkin-capsules {
display: flex;
flex-wrap: wrap;
gap: 8px;
}
.capsule {
font-size: 22px;
padding: 4px 12px;
border-radius: $r-pill;
font-weight: 500;
&.capsule-done {
background: $acc-l;
color: $acc;
}
&.capsule-pending {
background: $surface-alt;
color: $tx2;
}
}
/* ─── 区域 3今日体征 2x2 ─── */
.vitals-section {
margin: 0 24px 24px;
}
.vitals-grid {
display: grid; display: grid;
grid-template-columns: 1fr 1fr; grid-template-columns: 1fr 1fr;
gap: 16px; gap: 16px;
} }
.health-cell { .vital-card {
background: $bg; background: $card;
border-radius: $r-sm; border-radius: $r;
padding: 20px 16px; padding: 20px;
box-shadow: $shadow-sm;
text-align: center; text-align: center;
transition: opacity 0.2s;
&:active { &:active {
opacity: 0.7; opacity: 0.7;
} }
} }
.health-cell-label { .vital-label {
font-size: 22px; font-size: 24px;
color: $tx2; color: $tx2;
display: block; display: block;
margin-bottom: 8px; margin-bottom: 8px;
} }
.health-cell-value { .vital-value {
@include serif-number; @include serif-number;
font-size: 44px; font-size: 48px;
font-weight: bold; font-weight: bold;
color: $tx; color: $tx;
display: block; display: block;
line-height: 1.1; line-height: 1.1;
margin-bottom: 8px;
} }
.health-cell-bottom { .vital-bottom {
display: flex; display: flex;
justify-content: center; justify-content: center;
align-items: center; align-items: center;
gap: 8px; gap: 8px;
margin-top: 8px;
} }
.health-cell-unit { .vital-unit {
font-size: 20px; font-size: 22px;
color: $tx3; color: $tx2;
} }
.health-cell-tag { .vital-tag {
font-size: 18px; font-size: 22px;
font-weight: 500; font-weight: 500;
padding: 2px 10px; padding: 2px 10px;
border-radius: $r-sm; border-radius: $r-sm;
@@ -116,89 +175,42 @@
background: $wrn-l; background: $wrn-l;
color: $wrn; color: $wrn;
} }
}
/* ─── 快捷服务 ─── */ &.tag-empty {
.services-section { background: $surface-alt;
margin: 0 24px 24px; color: $tx2;
}
.services-row {
display: flex;
justify-content: space-between;
gap: 8px;
}
.service-btn {
display: flex;
flex-direction: column;
align-items: center;
gap: 8px;
flex: 1;
&:active {
opacity: 0.7;
} }
} }
.service-icon-wrap { /* ─── 区域 4今日待办 ─── */
width: 88px; .todo-section {
height: 88px; margin: 0 24px 24px;
border-radius: $r;
background: $pri-l;
@include flex-center;
} }
.service-icon-text { .todo-empty {
font-family: 'Georgia', 'Times New Roman', serif;
font-size: 32px;
font-weight: bold;
color: $pri;
}
.service-label {
font-size: 22px;
color: $tx2;
text-align: center;
}
/* ─── 待办事项 ─── */
.upcoming-section {
margin: 0 24px;
}
.upcoming-empty {
background: $card; background: $card;
border-radius: $r; border-radius: $r;
padding: 48px 24px; padding: 36px 24px;
text-align: center; text-align: center;
box-shadow: $shadow-sm; box-shadow: $shadow-sm;
} }
.upcoming-empty-text { .todo-empty-text {
display: block;
font-size: 28px;
color: $tx2;
margin-bottom: 8px;
}
.upcoming-empty-hint {
display: block;
font-size: 24px; font-size: 24px;
color: $tx3; color: $tx2;
} }
.upcoming-list { .todo-list {
background: $card; background: $card;
border-radius: $r; border-radius: $r;
overflow: hidden; overflow: hidden;
box-shadow: $shadow-sm; box-shadow: $shadow-sm;
} }
.upcoming-item { .todo-item {
display: flex; display: flex;
align-items: center; align-items: center;
padding: 24px 24px; padding: 24px;
border-bottom: 1px solid $bd-l; border-bottom: 1px solid $bd-l;
&:last-child { &:last-child {
@@ -210,20 +222,36 @@
} }
} }
.upcoming-item-main { .todo-icon-wrap {
width: 48px;
height: 48px;
border-radius: $r-sm;
background: $pri-l;
@include flex-center;
margin-right: 16px;
flex-shrink: 0;
}
.todo-icon-char {
font-size: 24px;
font-weight: bold;
color: $pri;
}
.todo-info {
flex: 1; flex: 1;
min-width: 0; min-width: 0;
} }
.upcoming-item-title { .todo-title {
font-size: 28px; font-size: 28px;
color: $tx; color: $tx;
font-weight: 600;
display: block; display: block;
margin-bottom: 4px; margin-bottom: 4px;
font-weight: 500;
} }
.upcoming-item-sub { .todo-sub {
font-size: 22px; font-size: 22px;
color: $tx2; color: $tx2;
display: block; display: block;
@@ -232,158 +260,44 @@
white-space: nowrap; white-space: nowrap;
} }
.upcoming-item-tag { .todo-arrow {
font-size: 20px;
font-weight: 500;
padding: 4px 14px;
border-radius: $r-sm;
flex-shrink: 0;
margin-right: 12px;
&.tag-ok {
background: $acc-l;
color: $acc;
}
&.tag-warn {
background: $wrn-l;
color: $wrn;
}
&.tag-default {
background: $bd-l;
color: $tx2;
}
}
.upcoming-item-arrow {
font-size: 32px; font-size: 32px;
color: $tx3; color: $tx3;
flex-shrink: 0; flex-shrink: 0;
} }
/* ─── 健康空状态 ─── */ /* ─── 区域 5快捷操作 ─── */
.health-empty { .action-section {
background: $bg;
border-radius: $r-sm;
padding: 40px 24px;
text-align: center;
}
.health-empty-text {
display: block;
font-size: 28px;
color: $tx2;
margin-bottom: 8px;
}
.health-empty-action {
display: flex; display: flex;
justify-content: center; gap: 16px;
padding: 24px 0 0;
}
.health-empty-btn {
background: $pri;
border-radius: $r;
padding: 16px 40px;
}
.health-empty-btn-text {
color: #fff;
font-size: 26px;
font-weight: 500;
}
/* ─── 健康资讯 ─── */
.articles-section {
margin: 0 24px 24px; margin: 0 24px 24px;
} }
.article-card { .action-btn {
background: $card;
border-radius: $r;
padding: 24px;
margin-bottom: 16px;
box-shadow: $shadow-sm;
&:active {
opacity: 0.7;
}
}
.article-card-title {
font-size: 28px;
color: $tx;
display: block;
font-weight: 500;
margin-bottom: 8px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.article-card-meta {
font-size: 22px;
color: $tx3;
}
/* ─── 设备快捷入口 ─── */
.device-section {
margin: 0 24px 24px;
}
.device-entry {
display: flex;
align-items: center;
background: $card;
border-radius: $r;
padding: 20px 24px;
margin-bottom: 12px;
box-shadow: $shadow-sm;
&:active {
opacity: 0.7;
}
}
.device-entry-icon-wrap {
width: 64px;
height: 64px;
border-radius: $r;
background: $pri-l;
display: flex;
align-items: center;
justify-content: center;
margin-right: 20px;
flex-shrink: 0;
}
.device-entry-icon-text {
font-size: 32px;
}
.device-entry-info {
flex: 1; flex: 1;
min-width: 0; @include touch-target;
} height: $btn-primary-h;
border-radius: $r;
.device-entry-name {
font-size: 28px; font-size: 28px;
font-weight: 600; font-weight: 600;
color: $tx;
display: block; &:active {
margin-bottom: 4px; opacity: 0.85;
}
} }
.device-entry-desc { .action-primary {
font-size: 22px; background: $pri;
color: $tx3; color: #fff;
display: block;
} }
.device-entry-arrow { .action-outline {
font-size: 32px; background: transparent;
color: $tx3; color: $pri;
flex-shrink: 0; border: 2px solid $pri;
}
.action-btn-text {
font-size: 28px;
font-weight: 600;
} }

View File

@@ -3,29 +3,19 @@ import { useState } from 'react';
import Taro, { useDidShow } from '@tarojs/taro'; import Taro, { useDidShow } from '@tarojs/taro';
import { useAuthStore } from '../../stores/auth'; import { useAuthStore } from '../../stores/auth';
import { useHealthStore } from '../../stores/health'; import { useHealthStore } from '../../stores/health';
import ProgressRing from '../../components/ProgressRing';
import Loading from '../../components/Loading'; import Loading from '../../components/Loading';
import { trackPageView } from '@/services/analytics'; import { trackPageView } from '@/services/analytics';
import * as appointmentApi from '@/services/appointment'; import * as appointmentApi from '@/services/appointment';
import * as followupApi from '@/services/followup'; import * as followupApi from '@/services/followup';
import * as articleApi from '../../services/article';
import './index.scss'; import './index.scss';
const QUICK_SERVICES = [
{ label: '预约挂号', char: '约', path: '/pages/appointment/create/index' },
{ label: '健康录入', char: '录', path: '/pages/pkg-health/input/index' },
{ label: '健康趋势', char: '势', path: '/pages/pkg-health/trend/index' },
{ label: '健康告警', char: '警', path: '/pages/pkg-health/alerts/index' },
{ label: '资讯文章', char: '文', path: '/pages/article/index' },
{ label: 'AI 报告', char: 'AI', path: '/pages/ai-report/list/index' },
];
interface UpcomingItem { interface UpcomingItem {
id: string; id: string;
title: string; title: string;
subtitle: string; subtitle: string;
type: 'appointment' | 'followup'; type: 'appointment' | 'followup';
statusLabel: string; icon: string;
statusType: 'ok' | 'warn' | 'default';
} }
export default function Index() { export default function Index() {
@@ -33,24 +23,13 @@ export default function Index() {
const { todaySummary, loading, refreshToday } = useHealthStore(); const { todaySummary, loading, refreshToday } = useHealthStore();
const [upcomingItems, setUpcomingItems] = useState<UpcomingItem[]>([]); const [upcomingItems, setUpcomingItems] = useState<UpcomingItem[]>([]);
const [upcomingLoading, setUpcomingLoading] = useState(false); const [upcomingLoading, setUpcomingLoading] = useState(false);
const [articles, setArticles] = useState<articleApi.Article[]>([]);
useDidShow(() => { useDidShow(() => {
refreshToday(); refreshToday();
loadUpcoming(); loadUpcoming();
loadArticles();
trackPageView('home'); trackPageView('home');
}); });
const loadArticles = async () => {
try {
const res = await articleApi.listArticles({ page: 1, page_size: 2 });
setArticles(res.data || []);
} catch {
// 文章接口可能不可用
}
};
const loadUpcoming = async () => { const loadUpcoming = async () => {
const patientId = useAuthStore.getState().currentPatient?.id; const patientId = useAuthStore.getState().currentPatient?.id;
if (!patientId) return; if (!patientId) return;
@@ -62,32 +41,30 @@ export default function Index() {
followupApi.listTasks(patientId, 'pending'), followupApi.listTasks(patientId, 'pending'),
]); ]);
if (apptRes.status === 'fulfilled') { if (apptRes.status === 'fulfilled') {
for (const a of apptRes.value.data.slice(0, 3)) { for (const a of apptRes.value.data.slice(0, 2)) {
if (a.status === 'pending' || a.status === 'confirmed') { if (a.status === 'pending' || a.status === 'confirmed') {
items.push({ items.push({
id: a.id, id: a.id,
title: `${a.appointment_date} ${a.start_time}`, title: `${a.appointment_date} ${a.start_time}`,
subtitle: `${a.doctor_name || '医护'} · ${a.department || ''}`, subtitle: `${a.doctor_name || '医护'} · ${a.department || '门诊'}`,
type: 'appointment', type: 'appointment',
statusLabel: a.status === 'pending' ? '待确认' : '已确认', icon: '',
statusType: a.status === 'pending' ? 'warn' : 'ok',
}); });
} }
} }
} }
if (taskRes.status === 'fulfilled') { if (taskRes.status === 'fulfilled') {
for (const t of taskRes.value.data.slice(0, 2)) { for (const t of taskRes.value.data.slice(0, 1)) {
items.push({ items.push({
id: t.id, id: t.id,
title: t.follow_up_type, title: t.follow_up_type,
subtitle: `${t.content_template?.slice(0, 30) || ''} · 截止 ${t.planned_date}`, subtitle: `${t.content_template?.slice(0, 20) || '随访任务'} · 截止 ${t.planned_date}`,
type: 'followup', type: 'followup',
statusLabel: '进行中', icon: '',
statusType: 'default',
}); });
} }
} }
setUpcomingItems(items); setUpcomingItems(items.slice(0, 3));
} catch { } catch {
setUpcomingItems([]); setUpcomingItems([]);
} finally { } finally {
@@ -99,11 +76,29 @@ export default function Index() {
const greeting = hour < 12 ? '上午好' : hour < 18 ? '下午好' : '晚上好'; const greeting = hour < 12 ? '上午好' : hour < 18 ? '下午好' : '晚上好';
const displayName = user?.display_name || currentPatient?.name || '访客'; const displayName = user?.display_name || currentPatient?.name || '访客';
// 计算今日体征完成度4 个指标:血压/心率/血糖/体重)
const summary = todaySummary || {};
const indicators = [
!!summary.blood_pressure,
!!summary.heart_rate,
!!summary.blood_sugar,
!!summary.weight,
];
const completedCount = indicators.filter(Boolean).length;
const progressPercent = Math.round((completedCount / 4) * 100);
const indicatorCapsules = [
{ label: '血压', done: !!summary.blood_pressure },
{ label: '心率', done: !!summary.heart_rate },
{ label: '血糖', done: !!summary.blood_sugar },
{ label: '体重', done: !!summary.weight },
];
const healthItems = [ const healthItems = [
{ label: '血压', value: todaySummary?.blood_pressure ? `${todaySummary.blood_pressure.systolic}/${todaySummary.blood_pressure.diastolic}` : '--/--', unit: 'mmHg', status: todaySummary?.blood_pressure?.status }, { label: '血压', value: summary.blood_pressure ? `${summary.blood_pressure.systolic}/${summary.blood_pressure.diastolic}` : '', unit: 'mmHg', status: summary.blood_pressure?.status, indicator: 'blood_pressure_systolic' },
{ label: '心率', value: todaySummary?.heart_rate ? `${todaySummary.heart_rate.value}` : '--', unit: 'bpm', status: todaySummary?.heart_rate?.status }, { label: '心率', value: summary.heart_rate ? `${summary.heart_rate.value}` : '', unit: 'bpm', status: summary.heart_rate?.status, indicator: 'heart_rate' },
{ label: '血糖', value: todaySummary?.blood_sugar ? `${todaySummary.blood_sugar.value}` : '--', unit: 'mmol/L', status: todaySummary?.blood_sugar?.status }, { label: '血糖', value: summary.blood_sugar ? `${summary.blood_sugar.value}` : '', unit: 'mmol/L', status: summary.blood_sugar?.status, indicator: 'blood_sugar_fasting' },
{ label: '体重', value: todaySummary?.weight ? `${todaySummary.weight.value}` : '--', unit: 'kg', status: todaySummary?.weight?.status }, { label: '体重', value: summary.weight ? `${summary.weight.value}` : '', unit: 'kg', status: summary.weight?.status, indicator: 'weight' },
]; ];
const getStatusTag = (status?: string) => { const getStatusTag = (status?: string) => {
@@ -114,64 +109,67 @@ export default function Index() {
return ( return (
<View className='home-page'> <View className='home-page'>
{/* 问候区 */} {/* 区域 1问候 + 日期 + 消息入口 */}
<View className='greeting-section'> <View className='greeting-section'>
<View className='greeting-left'> <View className='greeting-left'>
<Text className='greeting-time'>{greeting}</Text> <Text className='greeting-text'>{greeting}{displayName}</Text>
<Text className='greeting-name'>{displayName}</Text> <Text className='greeting-date'>
{new Date().toLocaleDateString('zh-CN', { month: 'long', day: 'numeric', weekday: 'short' })}
</Text>
</View> </View>
<Text className='greeting-date'>{new Date().toLocaleDateString('zh-CN', { month: 'long', day: 'numeric', weekday: 'short' })}</Text> <View
</View> className='greeting-msg'
onClick={() => Taro.switchTab({ url: '/pages/messages/index' })}
{/* 设备快捷入口 — 点击直接跳转设备同步页面 */} >
<View className='device-section'> <Text className='greeting-msg-icon'></Text>
<View className='device-entry' onClick={() => Taro.navigateTo({ url: '/pages/device-sync/index' })}>
<View className='device-entry-icon-wrap'>
<Text className='device-entry-icon-text'>{'\u{1FA7A}'}</Text>
</View>
<View className='device-entry-info'>
<Text className='device-entry-name'></Text>
<Text className='device-entry-desc'> · </Text>
</View>
<Text className='device-entry-arrow'>{''}</Text>
</View>
<View className='device-entry' onClick={() => Taro.navigateTo({ url: '/pages/device-sync/index' })}>
<View className='device-entry-icon-wrap'>
<Text className='device-entry-icon-text'>{'\u{1FA78}'}</Text>
</View>
<View className='device-entry-info'>
<Text className='device-entry-name'></Text>
<Text className='device-entry-desc'> · </Text>
</View>
<Text className='device-entry-arrow'>{''}</Text>
</View> </View>
</View> </View>
{/* 今日健康 */} {/* 区域 2今日体征完成度 */}
<View className='health-section'> <View
<Text className='section-title'></Text> className='checkin-card'
onClick={() => Taro.switchTab({ url: '/pages/health/index' })}
>
<View className='checkin-left'>
<ProgressRing percent={progressPercent} />
</View>
<View className='checkin-right'>
<Text className='checkin-title'>
{completedCount === 4 ? '今日体征已全部记录' : completedCount === 0 ? '今日尚未记录体征' : `今日已记录 ${completedCount}/4 项`}
</Text>
<View className='checkin-capsules'>
{indicatorCapsules.map((cap) => (
<Text
key={cap.label}
className={`capsule ${cap.done ? 'capsule-done' : 'capsule-pending'}`}
>
{cap.done ? '✓' : ''}{cap.label}
</Text>
))}
</View>
</View>
</View>
{/* 区域 3今日体征 2x2 网格 */}
<View className='vitals-section'>
{loading && !todaySummary ? ( {loading && !todaySummary ? (
<Loading /> <Loading />
) : !todaySummary || (!todaySummary.blood_pressure && !todaySummary.heart_rate && !todaySummary.blood_sugar && !todaySummary.weight) ? (
<View className='health-empty'>
<Text className='health-empty-text'></Text>
<View className='health-empty-action'>
<View className='health-empty-btn' onClick={() => Taro.navigateTo({ url: '/pages/pkg-health/input/index' })}>
<Text className='health-empty-btn-text'></Text>
</View>
</View>
</View>
) : ( ) : (
<View className='health-grid'> <View className='vitals-grid'>
{healthItems.map((item) => { {healthItems.map((item) => {
const tag = getStatusTag(item.status); const tag = getStatusTag(item.status);
return ( return (
<View className='health-cell' key={item.label} onClick={() => Taro.navigateTo({ url: `/pages/pkg-health/trend/index?indicator=${item.label === '血压' ? 'blood_pressure_systolic' : item.label === '心率' ? 'heart_rate' : item.label === '血糖' ? 'blood_sugar_fasting' : 'weight'}` })}> <View
<Text className='health-cell-label'>{item.label}</Text> className='vital-card'
<Text className='health-cell-value'>{item.value}</Text> key={item.label}
<View className='health-cell-bottom'> onClick={() => Taro.navigateTo({ url: `/pages/pkg-health/trend/index?indicator=${item.indicator}` })}
<Text className='health-cell-unit'>{item.unit}</Text> >
{tag && <Text className={`health-cell-tag ${tag.cls}`}>{tag.label}</Text>} <Text className='vital-label'>{item.label}</Text>
<Text className='vital-value'>{item.value}</Text>
<View className='vital-bottom'>
<Text className='vital-unit'>{item.unit}</Text>
{tag && <Text className={`vital-tag ${tag.cls}`}>{tag.label}</Text>}
{!item.status && <Text className='vital-tag tag-empty'></Text>}
</View> </View>
</View> </View>
); );
@@ -180,37 +178,21 @@ export default function Index() {
)} )}
</View> </View>
{/* 快捷服务 */} {/* 区域 4今日待办≤3 条) */}
<View className='services-section'> <View className='todo-section'>
<Text className='section-title'></Text> <Text className='section-title'></Text>
<View className='services-row'>
{QUICK_SERVICES.map((svc) => (
<View className='service-btn' key={svc.label} onClick={() => Taro.navigateTo({ url: svc.path })}>
<View className='service-icon-wrap'>
<Text className='service-icon-text'>{svc.char}</Text>
</View>
<Text className='service-label'>{svc.label}</Text>
</View>
))}
</View>
</View>
{/* 待办事项 */}
<View className='upcoming-section'>
<Text className='section-title'></Text>
{upcomingLoading ? ( {upcomingLoading ? (
<Loading /> <Loading />
) : upcomingItems.length === 0 ? ( ) : upcomingItems.length === 0 ? (
<View className='upcoming-empty'> <View className='todo-empty'>
<Text className='upcoming-empty-text'></Text> <Text className='todo-empty-text'></Text>
<Text className='upcoming-empty-hint'></Text>
</View> </View>
) : ( ) : (
<View className='upcoming-list'> <View className='todo-list'>
{upcomingItems.map((item) => ( {upcomingItems.map((item) => (
<View <View
key={item.id} key={item.id}
className='upcoming-item' className='todo-item'
onClick={() => { onClick={() => {
if (item.type === 'appointment') { if (item.type === 'appointment') {
Taro.navigateTo({ url: '/pages/appointment/index' }); Taro.navigateTo({ url: '/pages/appointment/index' });
@@ -219,36 +201,35 @@ export default function Index() {
} }
}} }}
> >
<View className='upcoming-item-main'> <View className='todo-icon-wrap'>
<Text className='upcoming-item-title'>{item.title}</Text> <Text className='todo-icon-char'>{item.icon}</Text>
<Text className='upcoming-item-sub'>{item.subtitle}</Text>
</View> </View>
<Text className={`upcoming-item-tag tag-${item.statusType}`}>{item.statusLabel}</Text> <View className='todo-info'>
<Text className='upcoming-item-arrow'></Text> <Text className='todo-title'>{item.title}</Text>
<Text className='todo-sub'>{item.subtitle}</Text>
</View>
<Text className='todo-arrow'></Text>
</View> </View>
))} ))}
</View> </View>
)} )}
</View> </View>
{/* 健康资讯 */} {/* 区域 5快捷操作 */}
{articles.length > 0 && ( <View className='action-section'>
<View className='articles-section'> <View
<Text className='section-title'></Text> className='action-btn action-primary'
{articles.map((article) => ( onClick={() => Taro.switchTab({ url: '/pages/health/index' })}
<View >
className='article-card' <Text className='action-btn-text'></Text>
key={article.id}
onClick={() => Taro.navigateTo({ url: `/pages/article/detail/index?id=${article.id}` })}
>
<Text className='article-card-title'>{article.title}</Text>
<Text className='article-card-meta'>
{article.category_name || '健康'}{article.published_at ? ` · ${article.published_at.slice(0, 10)}` : ''}
</Text>
</View>
))}
</View> </View>
)} <View
className='action-btn action-outline'
onClick={() => Taro.navigateTo({ url: '/pages/appointment/create/index' })}
>
<Text className='action-btn-text'></Text>
</View>
</View>
</View> </View>
); );
} }

View File

@@ -0,0 +1,194 @@
@import '../../styles/variables.scss';
@import '../../styles/mixins.scss';
.messages-page {
min-height: 100vh;
background: $bg;
padding-bottom: calc(120px + env(safe-area-inset-bottom));
}
/* ─── 页头 ─── */
.messages-header {
padding: 24px 32px 8px;
}
.messages-title {
font-family: 'Georgia', 'Times New Roman', serif;
font-size: 36px;
font-weight: bold;
color: $tx;
}
/* ─── Tab 切换 ─── */
.msg-tabs {
display: flex;
padding: 16px 24px 0;
gap: 0;
}
.msg-tab {
flex: 1;
height: $tab-h;
@include flex-center;
&:active {
opacity: 0.85;
}
}
.msg-tab-text {
font-size: 28px;
font-weight: 600;
color: $tx2;
}
.msg-tab-active .msg-tab-text {
color: $pri;
}
.msg-tab-indicator {
padding: 0 24px;
height: 3px;
background: $bd-l;
margin-bottom: 16px;
}
.msg-tab-bar {
width: 50%;
height: 3px;
background: $pri;
border-radius: 2px;
transition: transform 0.2s;
&.msg-tab-bar-right {
transform: translateX(100%);
}
}
/* ─── 内容区 ─── */
.msg-content {
padding: 0 24px;
}
.msg-empty {
background: $card;
border-radius: $r;
padding: 48px 24px;
text-align: center;
box-shadow: $shadow-sm;
}
.msg-empty-text {
font-size: 26px;
color: $tx2;
}
/* ─── 咨询卡片 ─── */
.consult-card {
display: flex;
align-items: center;
background: $card;
border-radius: $r;
padding: 24px;
margin-bottom: 12px;
box-shadow: $shadow-sm;
&:active {
opacity: 0.85;
}
}
.consult-info {
flex: 1;
min-width: 0;
}
.consult-doctor {
font-size: 28px;
font-weight: 600;
color: $tx;
display: block;
margin-bottom: 6px;
}
.consult-preview {
font-size: 24px;
color: $tx2;
display: block;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.consult-meta {
display: flex;
flex-direction: column;
align-items: flex-end;
gap: 8px;
flex-shrink: 0;
margin-left: 16px;
}
.consult-time {
font-size: 22px;
color: $tx2;
}
.consult-badge {
min-width: 24px;
height: 24px;
border-radius: 12px;
background: $dan;
@include flex-center;
padding: 0 6px;
}
.consult-badge-text {
font-size: 18px;
color: #fff;
font-weight: 600;
}
/* ─── 通知卡片 ─── */
.notify-card {
display: flex;
align-items: center;
background: $card;
border-radius: $r;
padding: 24px;
margin-bottom: 12px;
box-shadow: $shadow-sm;
&:active {
opacity: 0.85;
}
}
.notify-info {
flex: 1;
min-width: 0;
}
.notify-title {
font-size: 28px;
font-weight: 600;
color: $tx;
display: block;
margin-bottom: 6px;
}
.notify-desc {
font-size: 24px;
color: $tx2;
display: block;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.notify-time {
font-size: 22px;
color: $tx2;
flex-shrink: 0;
margin-left: 16px;
}

View File

@@ -0,0 +1,154 @@
import { useState } from 'react';
import { View, Text } from '@tarojs/components';
import Taro, { useDidShow } from '@tarojs/taro';
import { listConsultations, ConsultationSession } from '../../services/consultation';
import Loading from '../../components/Loading';
import './index.scss';
type MsgTab = 'consultation' | 'notification';
interface NotificationItem {
id: string;
title: string;
desc: string;
time: string;
type: string;
}
export default function Messages() {
const [activeTab, setActiveTab] = useState<MsgTab>('consultation');
const [sessions, setSessions] = useState<ConsultationSession[]>([]);
const [notifications, setNotifications] = useState<NotificationItem[]>([]);
const [loading, setLoading] = useState(false);
useDidShow(() => {
loadData(activeTab);
});
const loadData = async (tab: MsgTab) => {
setLoading(true);
try {
if (tab === 'consultation') {
const res = await listConsultations({ page: 1, page_size: 20 });
setSessions(res.data || []);
} else {
// 通知目前从咨询中提取状态变化作为示例
// 后续可对接独立通知 API
setNotifications([]);
}
} catch {
if (tab === 'consultation') setSessions([]);
else setNotifications([]);
} finally {
setLoading(false);
}
};
const handleTabChange = (tab: MsgTab) => {
setActiveTab(tab);
loadData(tab);
};
const formatTime = (dateStr: string | null) => {
if (!dateStr) return '';
const d = new Date(dateStr);
const now = new Date();
const diffMs = now.getTime() - d.getTime();
const diffMin = Math.floor(diffMs / 60000);
if (diffMin < 60) return `${diffMin} 分钟前`;
const diffHour = Math.floor(diffMin / 60);
if (diffHour < 24) return `${diffHour} 小时前`;
return dateStr.slice(0, 10);
};
return (
<View className='messages-page'>
{/* 页头 */}
<View className='messages-header'>
<Text className='messages-title'></Text>
</View>
{/* Tab 切换 */}
<View className='msg-tabs'>
<View
className={`msg-tab ${activeTab === 'consultation' ? 'msg-tab-active' : ''}`}
onClick={() => handleTabChange('consultation')}
>
<Text className='msg-tab-text'></Text>
</View>
<View
className={`msg-tab ${activeTab === 'notification' ? 'msg-tab-active' : ''}`}
onClick={() => handleTabChange('notification')}
>
<Text className='msg-tab-text'></Text>
</View>
</View>
<View className='msg-tab-indicator'>
<View className={`msg-tab-bar ${activeTab === 'notification' ? 'msg-tab-bar-right' : ''}`} />
</View>
{/* 咨询列表 */}
{activeTab === 'consultation' && (
<View className='msg-content'>
{loading ? (
<Loading />
) : sessions.length === 0 ? (
<View className='msg-empty'>
<Text className='msg-empty-text'></Text>
</View>
) : (
sessions.map((session) => (
<View
key={session.id}
className='consult-card'
onClick={() => Taro.navigateTo({ url: `/pages/consultation/detail/index?id=${session.id}` })}
>
<View className='consult-info'>
<Text className='consult-doctor'>
{session.consultation_type === 'online' ? '在线咨询' : '门诊咨询'}
</Text>
<Text className='consult-preview'>
{session.last_message || session.subject || '暂无消息'}
</Text>
</View>
<View className='consult-meta'>
<Text className='consult-time'>{formatTime(session.last_message_at)}</Text>
{session.unread_count_patient > 0 && (
<View className='consult-badge'>
<Text className='consult-badge-text'>
{session.unread_count_patient > 99 ? '99+' : session.unread_count_patient}
</Text>
</View>
)}
</View>
</View>
))
)}
</View>
)}
{/* 通知列表 */}
{activeTab === 'notification' && (
<View className='msg-content'>
{loading ? (
<Loading />
) : notifications.length === 0 ? (
<View className='msg-empty'>
<Text className='msg-empty-text'></Text>
</View>
) : (
notifications.map((n) => (
<View key={n.id} className='notify-card'>
<View className='notify-info'>
<Text className='notify-title'>{n.title}</Text>
<Text className='notify-desc'>{n.desc}</Text>
</View>
<Text className='notify-time'>{n.time}</Text>
</View>
))
)}
</View>
)}
</View>
);
}

View File

@@ -63,7 +63,7 @@
} }
&.resolved { &.resolved {
@include tag($suc-l, $suc); @include tag($acc-l, $acc);
} }
&.chronic { &.chronic {

View File

@@ -46,7 +46,7 @@
color: rgba(255, 255, 255, 0.75); color: rgba(255, 255, 255, 0.75);
} }
/* ─── 积分统计 ─── */ /* ─── 积分 + 打卡横排 ─── */
.profile-stats { .profile-stats {
display: flex; display: flex;
align-items: center; align-items: center;
@@ -63,7 +63,7 @@
} }
} }
.stat-item { .stat-card {
flex: 1; flex: 1;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -72,12 +72,20 @@
.stat-value { .stat-value {
@include serif-number; @include serif-number;
font-size: 36px; font-size: 28px;
font-weight: bold; font-weight: bold;
color: #fff; color: #fff;
margin-bottom: 4px; margin-bottom: 4px;
} }
.stat-value-pri {
color: #FFD6C7;
}
.stat-value-acc {
color: #C8E6C9;
}
.stat-label { .stat-label {
font-size: 22px; font-size: 22px;
color: rgba(255, 255, 255, 0.7); color: rgba(255, 255, 255, 0.7);
@@ -103,6 +111,8 @@
display: flex; display: flex;
align-items: center; align-items: center;
padding: 28px 24px; padding: 28px 24px;
min-height: $menu-item-h;
box-sizing: border-box;
border-bottom: 1px solid $bd-l; border-bottom: 1px solid $bd-l;
&:last-child { &:last-child {
@@ -120,7 +130,7 @@
border-radius: $r-sm; border-radius: $r-sm;
background: $pri-l; background: $pri-l;
@include flex-center; @include flex-center;
margin-right: 16px; margin-right: 20px;
flex-shrink: 0; flex-shrink: 0;
} }
@@ -134,7 +144,7 @@
.menu-label { .menu-label {
flex: 1; flex: 1;
font-size: 30px; font-size: 28px;
color: $tx; color: $tx;
} }
@@ -159,6 +169,6 @@
} }
.logout-text { .logout-text {
font-size: 30px; font-size: 28px;
color: $dan; color: $dan;
} }

View File

@@ -4,8 +4,8 @@
background: $card; background: $card;
border-radius: $r; border-radius: $r;
box-shadow: $shadow-md; box-shadow: $shadow-md;
padding: 24px; padding: 28px;
margin: 0 24px 20px; margin: 0 24px 24px;
} }
@mixin flex-center { @mixin flex-center {
@@ -26,7 +26,7 @@
@mixin section-title { @mixin section-title {
font-family: 'Georgia', 'Times New Roman', serif; font-family: 'Georgia', 'Times New Roman', serif;
font-size: 30px; font-size: 26px;
font-weight: bold; font-weight: bold;
color: $tx; color: $tx;
margin-bottom: 20px; margin-bottom: 20px;
@@ -37,8 +37,48 @@
display: inline-block; display: inline-block;
padding: 4px 12px; padding: 4px 12px;
border-radius: $r-sm; border-radius: $r-sm;
font-size: 20px; font-size: 22px;
font-weight: 500; font-weight: 500;
background: $bg; background: $bg;
color: $color; color: $color;
} }
@mixin touch-target {
min-height: $touch-min;
min-width: $touch-min;
display: flex;
align-items: center;
justify-content: center;
}
@mixin btn-primary {
height: $btn-primary-h;
border-radius: $r;
background: $pri;
color: #FFFFFF;
font-size: 28px;
font-weight: 600;
border: none;
width: 100%;
@include touch-target;
&:active {
opacity: 0.85;
}
}
@mixin btn-outline {
height: $btn-primary-h;
border-radius: $r;
background: transparent;
color: $pri;
font-size: 28px;
font-weight: 600;
border: 2px solid $pri;
width: 100%;
@include touch-target;
&:active {
background: $pri-l;
}
}

View File

@@ -12,8 +12,8 @@ $bg: #F5F0EB; // 主背景 (warm cream)
$card: #FFFFFF; // 卡片白 $card: #FFFFFF; // 卡片白
$surface-alt: #EDE8E2; // 辅助底 $surface-alt: #EDE8E2; // 辅助底
$tx: #2D2A26; // 主文字 (warm black) $tx: #2D2A26; // 主文字 (warm black)
$tx2: #7A756E; // 次文字 (warm gray) $tx2: #5A554F; // 次文字 (warm gray) — AA 正文对比度 ~5.5:1
$tx3: #A8A29E; // 淡文字 $tx3: #78716C; // 淡文字 — AA 正文对比度 ~4.6:1仅 ≥24px
$bd: #E8E2DC; // 边框 $bd: #E8E2DC; // 边框
$bd-l: #F0EBE5; // 浅边框 $bd-l: #F0EBE5; // 浅边框
$dan: #B54A4A; // 危险 (muted red) $dan: #B54A4A; // 危险 (muted red)
@@ -22,11 +22,18 @@ $wrn: #C4873A; // 警告 (warm amber)
$wrn-l: #FFF3E0; // 警告浅 $wrn-l: #FFF3E0; // 警告浅
// ─── 圆角 ─── // ─── 圆角 ───
$r: 12px; $r: 16px;
$r-sm: 8px; $r-sm: 12px;
$r-lg: 16px; $r-lg: 20px;
$r-pill: 999px; $r-pill: 999px;
// ─── 老年友好触控参数 ───
$touch-min: 48px; // 最小触控区域
$btn-primary-h: 56px; // 主按钮高度
$menu-item-h: 64px; // 菜单项高度
$tab-h: 56px; // Tab 切换高度
$font-min: 22px; // 最小字号
// ─── 阴影 ─── // ─── 阴影 ───
$shadow-sm: 0 1px 4px rgba(45, 42, 38, 0.04); $shadow-sm: 0 1px 4px rgba(45, 42, 38, 0.04);
$shadow-md: 0 2px 12px rgba(45, 42, 38, 0.08); $shadow-md: 0 2px 12px rgba(45, 42, 38, 0.08);