This commit is contained in:
zoujiandong 2025-08-29 09:46:07 +08:00
parent cc0ddd0bc9
commit da6e4ae551
10 changed files with 1121 additions and 624 deletions

View File

@ -36,16 +36,130 @@ const api = {
getHomeData(){ getHomeData(){
return request('/expertAPI/index', {}, 'post', false); return request('/expertAPI/index', {}, 'post', false);
}, },
// 视频相关API //肝胆课件列表
ganDanFileByKeyWords(data){
return request('/expertAPI/ganDanFileByKeyWords', data, 'post', false);
},
//肝胆课件详情
ganDanFileDetials(data){
return request('/expertAPI/ganDanFileDetials', data, 'post', false);
},
//是否存在正在进行中的公益咨询
isConsultIng(data){
return request('/expertAPI/isConsultIng', data, 'post', false);
},
//视频浏览记录列表
videoWatchRecord(data){
return request('/expertAPI/VideoWatchRecord', data, 'post', false);
},
//修改备注和描述
updateNicknameNote(data){
return request('/expertAPI/updateNicknameNote', data, 'post', false);
},
//获取未读数量
unReadList(data){
return request('/expertAPI/unReadList', data, 'post', false);
},
//消息列表
appMesageList(data){
return request('/expertAPI/appMesageList', data, 'post', false);
},
//阅读消息
appMesageRead(data){
return request('/expertAPI/appMesageRead', data, 'post', false);
},
//阅读消息
appMesageRead(data){
return request('/expertAPI/appMesageRead', data, 'post', false);
},
//最近30天的随访记录
relationRecordLately(data){
return request('/expertAPI/relationRecordLately', data, 'post', false);
},
//我的页面(信息与统计)
getMyInfo(data){
return request('/expertAPI/my', data, 'post', false);
},
getVideoList(data) { getVideoList(data) {
return request('/video/list', data, 'get', true); return request('/video/list', data, 'get', true);
}, },
getVideoDetail(data) { getVideoDetail(data) {
return request('/video/detail', data, 'get', true); return request('/video/detail', data, 'get', true);
}, },
getBannerVideo(data) {
return request('/video/banner', data, 'get', true); // 新闻详情
getNewsDetail(data) {
return request('/expertAPI/newsDetail', data, 'post', false);
}, },
// 课件详情
getGandanFileDetail(data) {
return request('/expertAPI/gandanFileDetail', data, 'post', false);
},
// 直播详情
getLiveDetail(data) {
return request('/expertAPI/liveDetail', data, 'post', false);
},
// 视频详情
getVideoDetail2(data) {
return request('/expertAPI/videoDetail2', data, 'post', false);
},
// 搜索医院列表
getHospitalList(data) {
return request('/expertAPI/hospitalList', data, 'post', false);
},
// 省市地区列表
getAreaList(data) {
return request('/expertAPI/areaList', data, 'post', false);
},
// 积分商城相关API
// 商品列表
getGoodsList(data) {
return request('/expertAPI/goodsList', data, 'post', false);
},
// 商品列表_V2
getGoodsListV2(data) {
return request('/expertAPI/goodsListV2', data, 'post', false);
},
// 积分商城商品类型列表
getGoodsTagList(data) {
return request('/expertAPI/goodsTagList', data, 'post', false);
},
// 积分商城-查询快递100
getOrderTrack(data) {
return request('/expertAPI/getOrderTrack', data, 'post', false);
},
// 福利版块相关API
// 我的福利卡
getMyWelfareCard(data) {
return request('/expertAPI/myWelfareCard', data, 'post', false);
},
// 兑换福利卡
exchangeWelfareCard(data) {
return request('/expertAPI/exchangeWelfareCard', data, 'post', false);
},
// U盘福利剩余个数
getUpanWelfareCount(data) {
return request('/expertAPI/getUpanWelfareCount', data, 'post', false);
},
// 添加积分
addBonusPointsN(data) {
return request('/expertAPI/addBonusPointsN', data, 'post', false);
},
} }
export default api export default api

View File

@ -9,12 +9,34 @@
<!-- 弹窗内容 --> <!-- 弹窗内容 -->
<view class="dialog-content"> <view class="dialog-content">
<slot name="content"> <slot name="content">
<text class="default-content">{{ content }}</text> <view class="default-content" >
<view v-for="item in formatContent(content)" :key="item">{{item}}</view>
</view>
</slot> </slot>
</view> </view>
<!-- 弹窗底部按钮 --> <!-- 弹窗底部按钮 -->
<view class="dialog-footer"> <view class="dialog-footer cancelFooter" v-if="showCancel">
<view
class="dialog-btn"
@click="oncancel"
>
<text class="btn-text">{{cancelText}}</text>
</view>
</view>
<view class="dialog-footer okFooter" v-else-if="showConfirm">
<view
class="dialog-btn okBtn"
@click="onconfirm"
>
<text class="btn-text">{{confirmText}}</text>
</view>
</view>
<view class="dialog-footer" v-else>
<view <view
class="dialog-btn cancel" class="dialog-btn cancel"
@ -49,13 +71,35 @@
type: String, type: String,
default: '' default: ''
}, },
cancelText:{
type: String,
default: '取消'
},
confirmText:{
type: String,
default: '确定'
},
showCancel:{
type:Boolean,
default:false,
},
showConfirm:{
type:Boolean,
default:false,
},
closeOnOverlay: { closeOnOverlay: {
type: Boolean, type: Boolean,
default: true default: true
} }
}) })
const formatContent=(val)=>{
console.log(val)
if(!val) return [];
// \n
const lines = val.split('<br>').filter(line => line.trim() !== '');
console.log('格式化后的内容数组:', lines);
return lines;
}
const emit = defineEmits(['close']) const emit = defineEmits(['close'])
@ -110,6 +154,7 @@
justify-content: center; justify-content: center;
.default-content { .default-content {
white-space:pre-wrap;
font-size: 32rpx; font-size: 32rpx;
color: #333; color: #333;
line-height: 1.5; line-height: 1.5;
@ -163,4 +208,7 @@
} }
} }
} }
.cancelFooter,.okFooter{
padding: 0;
}
</style> </style>

View File

@ -9,23 +9,19 @@
> >
<!-- 图标容器 --> <!-- 图标容器 -->
<view class="icon-container"> <view class="icon-container">
<!-- <uni-icons
:type="currentTab === index ? item.activeIcon : item.icon"
:size="currentTab === index ? 28 : 24"
:color="currentTab === index ? item.activeColor : item.color"
></uni-icons> -->
<image class="img" :src="currentTab === index ? item.activeIcon : item.icon" alt="" /> <image class="img" :src="currentTab === index ? item.activeIcon : item.icon" alt="" />
<!-- 徽章 --> <!-- 徽章 -->
<view class="badge" v-if="item.badge && item.badge > 0"> <!-- <view class="badge" v-if="item.badge && item.badge > 0">
<uni-badge <uni-badge
:text="item.badge > 99 ? '99+' : item.badge.toString()" :text="item.badge > 99 ? '99+' : item.badge.toString()"
type="error" type="error"
size="small" size="small"
></uni-badge> ></uni-badge>
</view> </view> -->
<!-- 红点 --> <!-- 红点 -->
<view class="red-dot" v-if="item.showRedDot"></view> <!-- <view class="red-dot" v-if="item.showRedDot"></view> -->
</view> </view>
<!-- 文字 --> <!-- 文字 -->
@ -82,9 +78,9 @@
text: '患教学堂', text: '患教学堂',
color: '#999999', color: '#999999',
activeColor: '#007aff', activeColor: '#007aff',
badge: 5, badge:0,
showRedDot: false, showRedDot: false,
pagePath: '/pages/education/education' pagePath: '/pages/patientClass/patientClass'
}, },
{ {
icon: live, icon: live,
@ -93,8 +89,8 @@
color: '#999999', color: '#999999',
activeColor: '#007aff', activeColor: '#007aff',
badge: 0, badge: 0,
showRedDot: true, showRedDot: false,
pagePath: '/pages/meeting/meeting' pagePath: '/pages/live/live'
}, },
{ {
icon: education, icon: education,
@ -102,9 +98,9 @@
text: '继续教育', text: '继续教育',
color: '#999999', color: '#999999',
activeColor: '#007aff', activeColor: '#007aff',
badge: 12, badge: 0,
showRedDot: false, showRedDot: false,
pagePath: '/pages/education/continuing' pagePath: '/pages/education/education'
}, },
{ {
icon: my, icon: my,
@ -114,7 +110,7 @@
activeColor: '#007aff', activeColor: '#007aff',
badge: 0, badge: 0,
showRedDot: false, showRedDot: false,
pagePath: '/pages/profile/profile' pagePath: '/pages/my/my'
} }
]); ]);

View File

@ -299,6 +299,16 @@
} }
} }
}, },
{
"path": "webview/webview",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "网页",
"app": {
"bounce": "none"
}
}
},
{ {
"path": "pointGoods/pointGoods", "path": "pointGoods/pointGoods",
"style": { "style": {
@ -491,6 +501,16 @@
} }
} }
}, },
{
"path": "myVideo/myVideo",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "uni-app分页",
"app": {
"bounce": "none"
}
}
},
{ {
"path": "patientMsg/patientMsg", "path": "patientMsg/patientMsg",
"style": { "style": {

View File

@ -4,9 +4,8 @@
<view class="header" :style="{ backgroundColor: headerBgColor }"> <view class="header" :style="{ backgroundColor: headerBgColor }">
<view class="header-content"> <view class="header-content">
<view class="header-left"> <view class="header-left">
<view class="calendar-icon"> <view class="calendar-icon" @click="onSignClick">
<up-image :src="nosign" width="60rpx" height="60rpx" ></up-image> <up-image :src="isSignedIn ? sign : nosign" width="60rpx" height="60rpx"></up-image>
</view> </view>
</view> </view>
<view class="search-container" @click="focus"> <view class="search-container" @click="focus">
@ -22,7 +21,7 @@
<view class="header-right"> <view class="header-right">
<view class="message-icon"> <view class="message-icon">
<uni-icons type="email" size="32" color="#fff"></uni-icons> <uni-icons type="email" size="32" color="#fff"></uni-icons>
<view class="red-dot"></view> <view v-if="hasUnread" class="red-dot"></view>
</view> </view>
</view> </view>
</view> </view>
@ -38,28 +37,14 @@
:mode="mode" :mode="mode"
:dots-styles="dotsStyles" :dots-styles="dotsStyles"
> >
<swiper class="swiper-box" @change="change" :current="current"> <swiper class="swiper-box" @change="change" :current="current" :autoplay="true" :interval="4000" :duration="500" :circular="true">
<swiper-item v-for="(item, index) in bannerList" :key="index"> <swiper-item v-for="(item, index) in bannerList" :key="index">
<view class="banner-item" :style="{ background: item.background }"> <view class="banner-item" @click="onBannerClick(item,index)">
<view class="banner-content"> <view class="doctorInfo" v-if="index==0">
<view class="banner-text"> <view class="name">{{expertDetail.realName}}专家工作室</view>
<text class="banner-title">{{ item.title }}</text> <view class="hospital">{{expertDetail.hospitalName}}</view>
<text class="banner-subtitle">{{ item.subtitle }}</text>
</view>
<view class="banner-scene">
<view class="scene-left">
<view class="palm-tree">🌴</view>
</view>
<view class="scene-center">
<view class="beach-umbrella">🏖</view>
<view class="beach-ball">🏐</view>
</view>
<view class="scene-right">
<view class="doctor-desk">👨</view>
<view class="bookshelf">📚</view>
</view>
</view>
</view> </view>
<up-image :src="item.image" width="100%" height="400rpx"></up-image>
</view> </view>
</swiper-item> </swiper-item>
</swiper> </swiper>
@ -69,9 +54,9 @@
<!-- 功能网格 --> <!-- 功能网格 -->
<view class="grid-section"> <view class="grid-section">
<uni-grid :column="4" :highlight="true" @change="onClick"> <uni-grid :column="4" :highlight="true" >
<uni-grid-item v-for="(item, index) in gridList" :key="index"> <uni-grid-item v-for="(item, index) in gridList" :key="index">
<view class="grid-item"> <view class="grid-item" @click="onClick(index)">
<up-image :src="item.icon" width="75rpx" height="75rpx" ></up-image> <up-image :src="item.icon" width="75rpx" height="75rpx" ></up-image>
<text class="grid-text">{{ item.text }}</text> <text class="grid-text">{{ item.text }}</text>
</view> </view>
@ -79,7 +64,7 @@
</uni-grid> </uni-grid>
</view> </view>
<!-- 消息通知滚动播放 --> <!-- 消息通知滚动播放 -->
<view class="notice-section"> <view class="notice-section" v-if="noticeList.length > 0">
<view class="notice-container"> <view class="notice-container">
<view class="notice-icon"> <view class="notice-icon">
@ -101,7 +86,7 @@
<up-image :src="jing_sign" width="28rpx" height="28rpx" ></up-image> <up-image :src="jing_sign" width="28rpx" height="28rpx" ></up-image>
</view> </view>
<view class="date"> <view class="date">
8月6日 {{ formatToMonthDay(item.date || item.time) }}
</view> </view>
</view> </view>
<view class="bar"></view> <view class="bar"></view>
@ -121,17 +106,8 @@
<text class="title-text">专题E站</text> <text class="title-text">专题E站</text>
</view> </view>
<view class="special-content"> <view class="special-content">
<view class="special-item" v-for="(item, index) in specialList" :key="index"> <view class="special-item" v-for="(item, index) in specialList" :key="index" @click="onEsiteClick(item)">
<view class="special-card" :style="{ background: item.bgColor }"> <up-image class="special-img" :src="item.image" width="100%" height="auto" mode="widthFix"></up-image>
<view class="special-left">
<text class="special-title">{{ item.title }}</text>
<text class="special-desc">{{ item.desc }}</text>
</view>
<view class="special-right">
<view class="special-icon">{{ item.icon }}</view>
<view class="special-arrow"></view>
</view>
</view>
</view> </view>
</view> </view>
</view> </view>
@ -145,15 +121,15 @@
<uni-icons type="right" size="32rpx" color="#999"></uni-icons> <uni-icons type="right" size="32rpx" color="#999"></uni-icons>
</view> </view>
</view> </view>
<scroll-view class="course-scroll" scroll-x="true"> <scroll-view class="course-scroll" scroll-x="true" scroll-with-animation="true" show-scrollbar="false">
<view class="course-list"> <view class="course-list">
<view class="course-item" v-for="(item, index) in courseList" :key="index"> <view class="course-item" v-for="(item, index) in courseList" :key="index">
<view class="course-card"> <view class="course-card">
<up-image :src="item.avatar" width="464rpx" height="262rpx" ></up-image> <up-image :src="item.avatar" width="464rpx" height="262rpx" ></up-image>
<view class="course-content"> <view class="course-content">
<view class="course-title twoline">{{ item.title }}</view> <view class="course-title twoline">{{ item.title }}</view>
<view class="course-subtitle">{{ item.status }}</view> <view class="course-subtitle" v-if="item.upload_num==item.video_num">已完结</view>
<view class="course-subtitle" v-else>已更新{{item.upload_num}}课时</view>
</view> </view>
</view> </view>
@ -173,9 +149,7 @@
<view class="replay-grid"> <view class="replay-grid">
<view class="replay-item" v-for="(item, index) in replayList" :key="index" @click="onReplayClick(item)"> <view class="replay-item" v-for="(item, index) in replayList" :key="index" @click="onReplayClick(item)">
<view class="replay-card"> <view class="replay-card">
<up-image class="replay-img" :src="item.avatar" width="100%" height="auto" mode="widthFix"></up-image>
<up-image :src="item.avatar" width="330rpx" height="186rpx" ></up-image>
<!-- <image :src="item.avatar" mode="aspectFill" class="doctor-avatar"></image> -->
<view class="replay-content"> <view class="replay-content">
<text class="replay-text">{{ item.content }}</text> <text class="replay-text">{{ item.content }}</text>
@ -233,6 +207,7 @@
></custom-tabbar> ></custom-tabbar>
</view> </view>
<unidialog :visible="visible" :content="'有福利待领取'" @close="visible=false" ></unidialog> <unidialog :visible="visible" :content="'有福利待领取'" @close="visible=false" ></unidialog>
<unidialog :visible="hasSign" content="今日已签到,每天只能签到一次。<br>请明日继续哦~" @close="hasSign=false" :showCancel="true" cancelText="关闭"></unidialog>
<up-overlay :show="showSign" > <up-overlay :show="showSign" >
<view class="signwrap"> <view class="signwrap">
<view class="signbox"> <view class="signbox">
@ -263,6 +238,7 @@
import unidialog from "@/components/dialog/dialog.vue" import unidialog from "@/components/dialog/dialog.vue"
import api from '@/api/api.js' import api from '@/api/api.js'
import navTo from "@/utils/navTo.js"; import navTo from "@/utils/navTo.js";
import docUrl from "@/utils/docUrl.js";
import bg from "@/static/more_bg.png" import bg from "@/static/more_bg.png"
import patient from "@/static/icon_home_my_patient.png" import patient from "@/static/icon_home_my_patient.png"
import video from "@/static/icon_home_video.png" import video from "@/static/icon_home_video.png"
@ -276,11 +252,15 @@
import nosign from "@/static/home_no_qiandao_icon.png" import nosign from "@/static/home_no_qiandao_icon.png"
import sign from "@/static/home_qiandao_icon.png" import sign from "@/static/home_qiandao_icon.png"
import signImg from "@/static/sign_in_bng_big.png" import signImg from "@/static/sign_in_bng_big.png"
import dayjs from 'dayjs'
const expertDetail=reactive({})
const showSign=ref(false) const showSign=ref(false)
// refs // refs
const tabbarRef = ref(null); const tabbarRef = ref(null);
const visible=ref(false) const visible=ref(false)
const hasUnread = ref(false)
const isSignedIn = ref(false) // false-true-
const hasSign=ref(false)
// //
const scrollTop = ref(0); const scrollTop = ref(0);
const bannerHeight = 400; // rpx const bannerHeight = 400; // rpx
@ -318,16 +298,12 @@
backgroundColor: '#ddd', backgroundColor: '#ddd',
border: '1px solid #ddd', border: '1px solid #ddd',
color: '#fff', color: '#fff',
selectedBackgroundColor: '#ff0000', selectedBackgroundColor: '#8B2316',
selectedBorder: '1px solid #ff0000' selectedBorder: '1px solid #8B2316'
}); });
// //
const bannerList = reactive([{ const bannerList = reactive([]);
title: '邹建东专家工作室',
subtitle: '北京肝胆相照公益基金会',
background: 'linear-gradient(135deg, #007aff, #0056b3)'
}]);
// //
const gridList = reactive([ const gridList = reactive([
@ -370,98 +346,16 @@
]); ]);
// E // E
const specialList = reactive([ const specialList = reactive([]);
{
title: '肝病医生的临床计算器和决策辅助工具',
desc: '肝胆相照®——医学常用工具',
icon: '🔧',
bgColor: '#1e3a8a'
},
{
title: '徐医感染 疑难发热及感染分享',
desc: '',
icon: '🏥',
bgColor: '#10b981'
},
{
title: '爱肝微视 全国首档爱肝科普短视频',
desc: '免费领取福利',
icon: '📱',
bgColor: '#1e3a8a'
}
]);
// //
const noticeList = reactive([ const noticeList = reactive([]);
{
id: 1,
content: '欢迎使用肝胆相照医生版,新功能上线啦!',
time: '10:30',
type: 'system'
},
{
id: 2,
content: '您有3个新的患者咨询待回复',
time: '09:15',
type: 'patient'
},
{
id: 3,
content: '今日有2场直播会议即将开始',
time: '08:45',
type: 'meeting'
},
{
id: 4,
content: '新的医学指南已更新,请及时查看',
time: '昨天',
type: 'guide'
}
]);
// //
const courseList = reactive([ const courseList = reactive([]);
{
title: '王晓光医生发起了新的病例讨论,邀请您参与',
status:'已完结'
},
{
title: '王晓光医生发起了新的病例讨论,邀请您参与',
status:'已完结'
},
{
title: '王晓光医生发起了新的病例讨论,邀请您参与',
status:'已完结'
},
]);
// //
const replayList = reactive([ const replayList = reactive([]);
{
id: 1,
avatar: '/static/c1.png',
content: 'Xxx医生发起了新的病例讨论,邀请您参与',
type: 'case_discussion'
},
{
id: 2,
avatar: '/static/c2.png',
content: 'Xxx医生发起了新的',
type: 'new_topic'
},
{
id: 3,
avatar: '/static/c3.png',
content: 'Xxx医生发起了新的病例讨论,邀请与......',
type: 'case_discussion'
},
{
id: 4,
avatar: '/static/c4.png',
content: 'Xxx医生发起了新的病例讨论,邀请您参与',
type: 'case_discussion'
}
]);
// //
const guideTabs = reactive([ const guideTabs = reactive([
@ -472,44 +366,12 @@
// //
const currentGuideTab = ref(0); const currentGuideTab = ref(0);
// //
const guideList = reactive([ const guideList = reactive([]);
{
id: 1, //
title: '一学就会的keynote教程番外篇】keynote 线下', const guidesData = reactive([]);
description: '实训你们要的带练来了', const coursewareData = reactive([]);
actionType: 'view',
type: 'guide'
},
{
id: 2,
title: '胆源性肝脏疾病——专题研讨会成功举行',
description: '',
actionType: 'view',
type: 'guide'
},
{
id: 3,
title: '肝病学新领域:肝硬化合并慢性肾脏病|深度综',
description: '述大的并慢性肾脏病|深度综述并慢并慢性.......',
actionType: 'download',
type: 'guide'
},
{
id: 4,
title: '【一学就会的keynote教程番外篇】keynote 线',
description: '下实训你们要的带练来了',
actionType: 'download',
type: 'guide'
},
{
id: 5,
title: '段钟平教授:终末期肝病营养治疗的价值及研',
description: '究进展',
actionType: 'download',
type: 'guide'
}
]);
// //
const change = (e) => { const change = (e) => {
@ -521,6 +383,14 @@
console.log('消息通知切换:', e.detail.current); console.log('消息通知切换:', e.detail.current);
}; };
// 使 dayjs 86
const formatToMonthDay = (val) => {
if (!val) return '';
const d = dayjs(val);
if (!d.isValid()) return '';
return d.format('M月D日');
};
// //
const onNoticeClick = (item) => { const onNoticeClick = (item) => {
console.log('点击消息通知:', item); console.log('点击消息通知:', item);
@ -589,22 +459,29 @@
}; };
// //
const onClick = (e) => { const onClick = (index) => {
console.log('点击了第' + e.detail.index + '个'); console.log('点击了第' + index + '个');
const clickedItem = gridList[e.detail.index]; let url=''
if(index==0){
// url='/pages_app/patientMsg/patientMsg'
if (clickedItem.text === '开具发票') { }else if(index==1){
uni.navigateTo({ url='/pages_app/video/video'
url: '/pages_course/invoice/invoice' }else if(index==2){
}); url='/pages_app/consult/consult'
return; }else if(index==3){
url='/pages_app/zhinan/zhinan'
}else if(index==4){
url='/pages_app/news/news'
}else if(index==5){
url='/pages_app/ppt/ppt'
}else if(index==6){
url='/pages_course/index/index'
}else{
url='/pages_app/search/search'
} }
navTo({
uni.showToast({ url:url
title: `点击了${clickedItem.text}`, })
icon: 'none'
});
}; };
// Tab // Tab
@ -619,6 +496,13 @@
// //
const switchGuideTab = (index) => { const switchGuideTab = (index) => {
currentGuideTab.value = index; currentGuideTab.value = index;
// 0: -> guidesData1: -> coursewareData
guideList.length = 0;
if (index === 0) {
guideList.push(...guidesData);
} else {
guideList.push(...coursewareData);
}
}; };
// //
@ -629,23 +513,111 @@
}); });
}; };
//
const onBannerClick = (item,index) => {
if (!item) return;
if(index==0){
}else{
// path
if (item.path) {
// #ifdef H5
window.open(item.path, '_blank');
// #endif
// #ifdef APP-PLUS
plus.runtime.openURL(item.path);
// #endif
// #ifdef MP
const encoded = encodeURIComponent(item.path);
uni.navigateTo({
url: `/pages_app/webview/webview?url=${encoded}`
});
// #endif
return;
}
// uuidH5
if (item.uuid) {
const url = `https://dev-doc.igandan.com/app/html/news/${item.uuid}.html`;
// #ifdef H5
window.open(url, '_blank');
// #endif
// #ifdef APP-PLUS
plus.runtime.openURL(url);
// #endif
// #ifdef MP
const encoded = encodeURIComponent(url);
uni.navigateTo({ url: `/pages_app/webview/webview?url=${encoded}` });
// #endif
}
}
};
// E
const onEsiteClick = (item) => {
if (!item || !item.url) {
return;
}
// H5
// #ifdef H5
window.open(item.url, '_blank');
// #endif
// App
// #ifdef APP-PLUS
plus.runtime.openURL(item.url);
// #endif
// 使 webview
// #ifdef MP
const encoded = encodeURIComponent(item.url)
uni.navigateTo({
url: `/pages_app/webview/webview?url=${encoded}`
})
// #endif
};
// //
const onGuideClick = (item) => { const onGuideClick = (item) => {
console.log('点击实用指南:', item); // PDF
uni.showToast({ if (item.path) {
title: `查看${item.title}`, const pdfUrl = docUrl+item.path;
icon: 'none' console.log(pdfUrl)
}); if (pdfUrl) {
// // H5
if (item.type === 'guide') { // #ifdef H5
uni.navigateTo({ window.open(pdfUrl, '_blank');
url: '/pages/guide/detail?id=' + item.id // #endif
});
} else if (item.type === 'courseware') { // App
uni.navigateTo({ // #ifdef APP-PLUS
url: '/pages/courseware/list' plus.runtime.openURL(pdfUrl);
}); // #endif
// 使 webview
// #ifdef MP-WEIXIN
const encoded = encodeURIComponent(pdfUrl);
uni.navigateTo({
url: `/pages_app/webview/webview?url=${encoded}`
});
// #endif
return;
}
} }
// PDF
// if (item.type === 'guide') {
// uni.navigateTo({
// url: '/pages/guide/detail?id=' + item.id
// });
// } else if (item.type === 'courseware') {
// uni.navigateTo({
// url: '/pages/courseware/list'
// });
// }
}; };
// tabbar // tabbar
@ -685,47 +657,135 @@
if (res && res.data) { if (res && res.data) {
const data = res.data; const data = res.data;
Object.assign(expertDetail,data.expertDetail)
// //
if (data.bannerList && data.bannerList.length > 0) { hasUnread.value = !!data.has_unread;
bannerList.length = 0;
bannerList.push(...data.bannerList); //
isSignedIn.value = data.sign_in === 1;
//
if (data.welfare_notice && data.welfare_notice.receive_notice) {
visible.value = true;
} }
// // news_list+
if (data.gridList && data.gridList.length > 0) { bannerList.length = 0;
if (Array.isArray(data.news_list) && data.news_list.length > 0) {
bannerList.push(
...data.news_list.map(item => ({
image: item.headImg || '',
type: item.type, // 0: H5, 1:
path: item.path || '',
uuid: item.uuid || ''
}))
);
} else if (Array.isArray(data.esite_list) && data.esite_list.length > 0) {
bannerList.push({ image: data.esite_list[0].img_path || '' });
}
// icons_list -> gridList
if (Array.isArray(data.icons_list) && data.icons_list.length > 0) {
gridList.length = 0; gridList.length = 0;
gridList.push(...data.gridList); gridList.push(
...data.icons_list.map(item => ({
icon: item.img || '',
text: item.name || ''
}))
);
gridList.push({
icon:more,
text:'更多'
})
} }
// // meeting_list -> noticeList
if (data.noticeList && data.noticeList.length > 0) { if (Array.isArray(data.meeting_list) && data.meeting_list.length > 0) {
noticeList.length = 0; noticeList.length = 0;
noticeList.push(...data.noticeList); noticeList.push(
...data.meeting_list.map((item, idx) => ({
id: item.id || item.uuid || idx + 1,
content: item.title || item.name || '',
time: item.begin_date || item.start_time || item.time || item.create_date || '',
type: 'meeting',
path: item.path || ''
}))
);
} }
// E // Eesite_list -> specialList3
if (data.specialList && data.specialList.length > 0) { if (Array.isArray(data.esite_list) && data.esite_list.length > 0) {
specialList.length = 0; specialList.length = 0;
specialList.push(...data.specialList); specialList.push(
...data.esite_list.slice(0, 3).map(item => ({
image: item.img_path || '',
url: item.url || ''
}))
);
} }
// // excellencourse_list -> courseList
if (data.courseList && data.courseList.length > 0) { if (Array.isArray(data.excellencourse_list) && data.excellencourse_list.length > 0) {
courseList.length = 0; courseList.length = 0;
courseList.push(...data.courseList); courseList.push(
...data.excellencourse_list.map(item => ({
title: item.title || '',
status: item.video_num != null ? `${item.video_num}` : '',
avatar: item.index_img || ''
}))
);
} }
// // video_list -> replayList
if (data.replayList && data.replayList.length > 0) { if (Array.isArray(data.video_list) && data.video_list.length > 0) {
replayList.length = 0; replayList.length = 0;
replayList.push(...data.replayList); replayList.push(
...data.video_list.map(item => ({
id: item.uuid || item.polyv_uuid || Math.random().toString(36).slice(2),
avatar: item.imgpath || '',
content: item.name || '',
type: 'case_discussion'
}))
);
} }
// // guide_ist guidesData
if (data.guideList && data.guideList.length > 0) { guidesData.length = 0;
guideList.length = 0; if (Array.isArray(data.guide_ist) && data.guide_ist.length > 0) {
guideList.push(...data.guideList); guidesData.push(
...data.guide_ist.map(item => ({
id: item.article_uuid || '',
title: item.title || '',
description: '',
actionType: 'download',
type: 'courseware',
path: item.path || ''
}))
);
}
// gandanfile_list coursewareData
coursewareData.length = 0;
if (Array.isArray(data.gandanfile_list) && data.gandanfile_list.length > 0) {
coursewareData.push(
...data.gandanfile_list.map(item => ({
id: item.article_uuid || '',
title: item.title || '',
description: '',
actionType: 'download',
type: 'courseware',
path: item.path || ''
}))
);
}
//
guideList.length = 0;
if (currentGuideTab.value === 0) {
guideList.push(...guidesData);
} else {
guideList.push(...coursewareData);
} }
} }
} catch (error) { } catch (error) {
@ -774,11 +834,51 @@
}); });
}; };
//
const onSignClick = async () => {
//
if (isSignedIn.value) {
hasSign.value=true;
return;
}
try {
//
const res = await api.addBonusPointsN({
score_type:1
});
if (res && res.code === 200) {
//
isSignedIn.value = true;
uni.showToast({
title: '签到成功获得10积分',
icon: 'success'
});
//
showSign.value = true;
} else {
uni.showToast({
title: res?.msg || '签到失败',
icon: 'none'
});
}
} catch (error) {
console.error('签到失败:', error);
uni.showToast({
title: '签到失败,请重试',
icon: 'none'
});
}
};
// //
defineExpose({ defineExpose({
testTabbar, testTabbar,
getTabbarStatus, getTabbarStatus,
goToCourseHome goToCourseHome,
onSignClick
}); });
</script> </script>
@ -991,7 +1091,19 @@
overflow: hidden; overflow: hidden;
position: relative; position: relative;
} }
.doctorInfo{
position: absolute;
top:200rpx;
z-index:9;
left:30rpx;
right:30rpx;
color:#fff;
font-size: 40rpx;
}
.doctorInfo .hospital{
margin-top: 20rpx;
font-size: 32rpx;
}
.banner-content { .banner-content {
height: 100%; height: 100%;
display: flex; display: flex;
@ -1069,6 +1181,20 @@
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 20rpx; gap: 20rpx;
background-color: #fff;
padding: 20rpx;
border-radius: 10rpx;
}
/* 专题E站图片项宽度铺满高度自适应 */
.special-item {
position: relative;
width: 100%;
overflow: hidden;
}
.special-img {
width: 100%;
display: block;
} }
.special-card { .special-card {
@ -1148,19 +1274,20 @@
white-space: nowrap; white-space: nowrap;
margin-left: 30rpx; margin-left: 30rpx;
margin-right: 30rpx; margin-right: 30rpx;
overflow: hidden;
} }
.course-list { .course-list {
display: flex; display: inline-flex;
gap: 30rpx; gap: 30rpx;
width:auto; width: auto;
padding-right: 60rpx; /* 让最后一项与右边留出更大间距 */
} }
.course-item { .course-item {
flex:1; display: inline-block;
background-color: #f4f4f4; background-color: #f4f4f4;
width: 464rpx; width: 464rpx;
margin-right: 30rpx;
box-shadow: 0 4rpx 16rpx rgba(0,0,0,0.1); box-shadow: 0 4rpx 16rpx rgba(0,0,0,0.1);
border-radius: 24rpx; border-radius: 24rpx;
overflow: hidden; overflow: hidden;
@ -1185,15 +1312,19 @@
font-size: 20rpx; font-size: 20rpx;
} }
.course-content {
margin-bottom: 30rpx;
}
.course-title { .course-title {
white-space: normal; white-space: normal;
font-size: 30rpx; font-size: 30rpx;
color: #333333; color: #333333;
margin: 20rpx; margin: 20rpx;
/* 固定两行高度并省略 */
line-height: 40rpx;
height: 80rpx;
display: -webkit-box;
-webkit-box-orient: vertical;
-webkit-line-clamp: 2;
overflow: hidden;
text-overflow: ellipsis;
} }
.course-subtitle { .course-subtitle {
@ -1240,8 +1371,7 @@
.replay-item { .replay-item {
cursor: pointer; cursor: pointer;
height:305rpx; }
}
.replay-card { .replay-card {
background-color: white; background-color: white;
@ -1255,9 +1385,8 @@
transform: scale(0.98); transform: scale(0.98);
} }
.replay-image { .replay-image { /* 不再使用固定高容器,保留占位以防其它样式引用 */
width: 100%; width: 100%;
height: 200rpx;
overflow: hidden; overflow: hidden;
} }
@ -1267,6 +1396,13 @@
object-fit: cover; object-fit: cover;
} }
/* 新增:精彩回放图片自适应 */
.replay-img{
width: 100%;
height: auto;
display: block;
}
.replay-content { .replay-content {
padding: 20rpx; padding: 20rpx;
} }

File diff suppressed because it is too large Load Diff

View File

@ -2,6 +2,17 @@
<view class="flower-page"> <view class="flower-page">
<!-- 顶部统计栏与截图一致两项 --> <!-- 顶部统计栏与截图一致两项 -->
<uni-nav-bar
left-icon="left"
title="我的鲜花"
@clickLeft="goBack"
fixed
color="#8B2316"
height="140rpx"
:border="false"
backgroundColor="#eeeeee"
/>
<view class="stats-bar"> <view class="stats-bar">
<view class="stat"> <view class="stat">
<up-image :src="flowerImg" width="36rpx" height="36rpx" ></up-image> <up-image :src="flowerImg" width="36rpx" height="36rpx" ></up-image>

View File

@ -0,0 +1,40 @@
<template>
<view class="container">
<!-- H5/APP 使用 web-view 也可 plus 下直接 openURL 更流畅这里统一 -->
<!-- 小程序端必须使用 web-view -->
<!-- #ifdef MP -->
<web-view :src="safeUrl"></web-view>
<!-- #endif -->
<!-- #ifndef MP -->
<view class="tip">仅小程序内使用内嵌浏览器其它端请直接打开外部浏览器</view>
<!-- #endif -->
</view>
</template>
<script setup>
import { ref } from 'vue'
import { onLoad } from '@dcloudio/uni-app'
const safeUrl = ref('')
onLoad((query) => {
// url
const raw = query && (query.url || '')
try {
safeUrl.value = decodeURIComponent(raw)
} catch (e) {
safeUrl.value = raw
}
})
</script>
<style>
.container{
min-height: 100vh;
}
.tip{
padding: 24rpx;
color: #666;
font-size: 28rpx;
}
</style>

18
utils/docUrl.js Normal file
View File

@ -0,0 +1,18 @@
let DOC_URL=''
if(process.env.UNI_PLATFORM =="h5"){
if (window.location.href.indexOf('//casedata.igandan.com')>-1){
DOC_URL='https://doc.igandan.com/app/'
}else{
DOC_URL='https://dev-doc.igandan.com/app/'
}
}else if(process.env.UNI_PLATFORM =="mp-weixin"){
const { envVersion } = uni.getAccountInfoSync().miniProgram;
if (envVersion == "release") {
DOC_URL='https://app.igandan.com/app/'
}else{
DOC_URL='https://dev-doc.igandan.com/app/'
}
}else{
DOC_URL='https://dev-doc.igandan.com/app/'
}
export default DOC_URL

View File

@ -40,31 +40,54 @@ export const request = (url, data = {}, method = 'post', loading = false, conten
let defaultData={} let defaultData={}
let freeList=['/manager/getSystemTimeStamp','/expertAPI/smsLogin','/login/mobile','/expertAPI/login','/expertAPI/index'] let freeList=['/manager/getSystemTimeStamp','/expertAPI/smsLogin','/login/mobile','/expertAPI/login','/expertAPI/index']
let postData={
...data
}
if(freeList.indexOf(url)!=-1){ if(freeList.indexOf(url)!=-1){
if(freeList[4].indexOf(url)!=-1){ if(freeList[4].indexOf(url)!=-1){
if(!token){ if(!token){
header['Authorization']='Bearer ' + '' header['Authorization']='Bearer ' + ''
}else{ }else{
header['Authorization']='Bearer ' + token header['Authorization']='Bearer ' + token;
let userInfo= uni.getStorageSync('userInfo')
defaultData = {
version: '4.0.0',
user_uuid:userInfo.uuid,
client_type: 'A', //client_type,
}
postData={
...data,
...defaultData
}
} }
} }
}else{ }else{
if(token){ if(token){
header['Authorization']='Bearer ' +token header['Authorization']='Bearer ' +token;
let userInfo= uni.getStorageSync('userInfo')
defaultData = {
version: '4.0.0',
user_uuid:userInfo.uuid,
client_type: 'A', //client_type,
}
defaultData = {
version: '4.0.0',
user_uuid:userInfo.uuid,
client_type: 'A', //client_type,
}
postData={
...data,
...defaultData
}
} }
} }
// defaultData = {
// version: '4.0.0',
// user_uuid: 'Rj3zuTY6zP6YTjHsYEz',
// client_type: 'A', //client_type,
// }
return new Promise(function(e, n) { return new Promise(function(e, n) {
let timestamp = Date.now(); let timestamp = Date.now();
uni.request({ uni.request({
data: {...data}, data: {...data,...defaultData},
url: url.indexOf('http') != -1 ? url : encodeURI(BASE_URL + url), url: url.indexOf('http') != -1 ? url : encodeURI(BASE_URL + url),
method: method, method: method,
sslVerify: false, sslVerify: false,