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(){
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) {
return request('/video/list', data, 'get', true);
},
getVideoDetail(data) {
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

View File

@ -9,12 +9,34 @@
<!-- 弹窗内容 -->
<view class="dialog-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>
</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
class="dialog-btn cancel"
@ -49,13 +71,35 @@
type: String,
default: ''
},
cancelText:{
type: String,
default: '取消'
},
confirmText:{
type: String,
default: '确定'
},
showCancel:{
type:Boolean,
default:false,
},
showConfirm:{
type:Boolean,
default:false,
},
closeOnOverlay: {
type: Boolean,
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'])
@ -110,6 +154,7 @@
justify-content: center;
.default-content {
white-space:pre-wrap;
font-size: 32rpx;
color: #333;
line-height: 1.5;
@ -163,4 +208,7 @@
}
}
}
.cancelFooter,.okFooter{
padding: 0;
}
</style>

View File

@ -9,23 +9,19 @@
>
<!-- 图标容器 -->
<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="" />
<!-- 徽章 -->
<view class="badge" v-if="item.badge && item.badge > 0">
<!-- <view class="badge" v-if="item.badge && item.badge > 0">
<uni-badge
:text="item.badge > 99 ? '99+' : item.badge.toString()"
type="error"
size="small"
></uni-badge>
</view>
</view> -->
<!-- 红点 -->
<view class="red-dot" v-if="item.showRedDot"></view>
<!-- <view class="red-dot" v-if="item.showRedDot"></view> -->
</view>
<!-- 文字 -->
@ -82,9 +78,9 @@
text: '患教学堂',
color: '#999999',
activeColor: '#007aff',
badge: 5,
badge:0,
showRedDot: false,
pagePath: '/pages/education/education'
pagePath: '/pages/patientClass/patientClass'
},
{
icon: live,
@ -93,8 +89,8 @@
color: '#999999',
activeColor: '#007aff',
badge: 0,
showRedDot: true,
pagePath: '/pages/meeting/meeting'
showRedDot: false,
pagePath: '/pages/live/live'
},
{
icon: education,
@ -102,9 +98,9 @@
text: '继续教育',
color: '#999999',
activeColor: '#007aff',
badge: 12,
badge: 0,
showRedDot: false,
pagePath: '/pages/education/continuing'
pagePath: '/pages/education/education'
},
{
icon: my,
@ -114,7 +110,7 @@
activeColor: '#007aff',
badge: 0,
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",
"style": {
@ -491,6 +501,16 @@
}
}
},
{
"path": "myVideo/myVideo",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "uni-app分页",
"app": {
"bounce": "none"
}
}
},
{
"path": "patientMsg/patientMsg",
"style": {

View File

@ -4,9 +4,8 @@
<view class="header" :style="{ backgroundColor: headerBgColor }">
<view class="header-content">
<view class="header-left">
<view class="calendar-icon">
<up-image :src="nosign" width="60rpx" height="60rpx" ></up-image>
<view class="calendar-icon" @click="onSignClick">
<up-image :src="isSignedIn ? sign : nosign" width="60rpx" height="60rpx"></up-image>
</view>
</view>
<view class="search-container" @click="focus">
@ -22,7 +21,7 @@
<view class="header-right">
<view class="message-icon">
<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>
@ -38,28 +37,14 @@
:mode="mode"
: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">
<view class="banner-item" :style="{ background: item.background }">
<view class="banner-content">
<view class="banner-text">
<text class="banner-title">{{ item.title }}</text>
<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 class="banner-item" @click="onBannerClick(item,index)">
<view class="doctorInfo" v-if="index==0">
<view class="name">{{expertDetail.realName}}专家工作室</view>
<view class="hospital">{{expertDetail.hospitalName}}</view>
</view>
<up-image :src="item.image" width="100%" height="400rpx"></up-image>
</view>
</swiper-item>
</swiper>
@ -69,9 +54,9 @@
<!-- 功能网格 -->
<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">
<view class="grid-item">
<view class="grid-item" @click="onClick(index)">
<up-image :src="item.icon" width="75rpx" height="75rpx" ></up-image>
<text class="grid-text">{{ item.text }}</text>
</view>
@ -79,7 +64,7 @@
</uni-grid>
</view>
<!-- 消息通知滚动播放 -->
<view class="notice-section">
<view class="notice-section" v-if="noticeList.length > 0">
<view class="notice-container">
<view class="notice-icon">
@ -101,7 +86,7 @@
<up-image :src="jing_sign" width="28rpx" height="28rpx" ></up-image>
</view>
<view class="date">
8月6日
{{ formatToMonthDay(item.date || item.time) }}
</view>
</view>
<view class="bar"></view>
@ -121,17 +106,8 @@
<text class="title-text">专题E站</text>
</view>
<view class="special-content">
<view class="special-item" v-for="(item, index) in specialList" :key="index">
<view class="special-card" :style="{ background: item.bgColor }">
<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 class="special-item" v-for="(item, index) in specialList" :key="index" @click="onEsiteClick(item)">
<up-image class="special-img" :src="item.image" width="100%" height="auto" mode="widthFix"></up-image>
</view>
</view>
</view>
@ -145,15 +121,15 @@
<uni-icons type="right" size="32rpx" color="#999"></uni-icons>
</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-item" v-for="(item, index) in courseList" :key="index">
<view class="course-card">
<up-image :src="item.avatar" width="464rpx" height="262rpx" ></up-image>
<view class="course-content">
<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>
@ -173,9 +149,7 @@
<view class="replay-grid">
<view class="replay-item" v-for="(item, index) in replayList" :key="index" @click="onReplayClick(item)">
<view class="replay-card">
<up-image :src="item.avatar" width="330rpx" height="186rpx" ></up-image>
<!-- <image :src="item.avatar" mode="aspectFill" class="doctor-avatar"></image> -->
<up-image class="replay-img" :src="item.avatar" width="100%" height="auto" mode="widthFix"></up-image>
<view class="replay-content">
<text class="replay-text">{{ item.content }}</text>
@ -233,6 +207,7 @@
></custom-tabbar>
</view>
<unidialog :visible="visible" :content="'有福利待领取'" @close="visible=false" ></unidialog>
<unidialog :visible="hasSign" content="今日已签到,每天只能签到一次。<br>请明日继续哦~" @close="hasSign=false" :showCancel="true" cancelText="关闭"></unidialog>
<up-overlay :show="showSign" >
<view class="signwrap">
<view class="signbox">
@ -263,6 +238,7 @@
import unidialog from "@/components/dialog/dialog.vue"
import api from '@/api/api.js'
import navTo from "@/utils/navTo.js";
import docUrl from "@/utils/docUrl.js";
import bg from "@/static/more_bg.png"
import patient from "@/static/icon_home_my_patient.png"
import video from "@/static/icon_home_video.png"
@ -276,11 +252,15 @@
import nosign from "@/static/home_no_qiandao_icon.png"
import sign from "@/static/home_qiandao_icon.png"
import signImg from "@/static/sign_in_bng_big.png"
import dayjs from 'dayjs'
const expertDetail=reactive({})
const showSign=ref(false)
// refs
const tabbarRef = ref(null);
const visible=ref(false)
const hasUnread = ref(false)
const isSignedIn = ref(false) // false-true-
const hasSign=ref(false)
//
const scrollTop = ref(0);
const bannerHeight = 400; // rpx
@ -318,16 +298,12 @@
backgroundColor: '#ddd',
border: '1px solid #ddd',
color: '#fff',
selectedBackgroundColor: '#ff0000',
selectedBorder: '1px solid #ff0000'
selectedBackgroundColor: '#8B2316',
selectedBorder: '1px solid #8B2316'
});
//
const bannerList = reactive([{
title: '邹建东专家工作室',
subtitle: '北京肝胆相照公益基金会',
background: 'linear-gradient(135deg, #007aff, #0056b3)'
}]);
//
const bannerList = reactive([]);
//
const gridList = reactive([
@ -370,98 +346,16 @@
]);
// E
const specialList = reactive([
{
title: '肝病医生的临床计算器和决策辅助工具',
desc: '肝胆相照®——医学常用工具',
icon: '🔧',
bgColor: '#1e3a8a'
},
{
title: '徐医感染 疑难发热及感染分享',
desc: '',
icon: '🏥',
bgColor: '#10b981'
},
{
title: '爱肝微视 全国首档爱肝科普短视频',
desc: '免费领取福利',
icon: '📱',
bgColor: '#1e3a8a'
}
]);
const specialList = 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 noticeList = reactive([]);
//
const courseList = reactive([
{
title: '王晓光医生发起了新的病例讨论,邀请您参与',
status:'已完结'
},
{
title: '王晓光医生发起了新的病例讨论,邀请您参与',
status:'已完结'
},
{
title: '王晓光医生发起了新的病例讨论,邀请您参与',
status:'已完结'
},
]);
const courseList = 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 replayList = reactive([]);
//
const guideTabs = reactive([
@ -472,44 +366,12 @@
//
const currentGuideTab = ref(0);
//
const guideList = reactive([
{
id: 1,
title: '一学就会的keynote教程番外篇】keynote 线下',
description: '实训你们要的带练来了',
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 guideList = reactive([]);
//
const guidesData = reactive([]);
const coursewareData = reactive([]);
//
const change = (e) => {
@ -521,6 +383,14 @@
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) => {
console.log('点击消息通知:', item);
@ -589,22 +459,29 @@
};
//
const onClick = (e) => {
console.log('点击了第' + e.detail.index + '个');
const clickedItem = gridList[e.detail.index];
//
if (clickedItem.text === '开具发票') {
uni.navigateTo({
url: '/pages_course/invoice/invoice'
});
return;
const onClick = (index) => {
console.log('点击了第' + index + '个');
let url=''
if(index==0){
url='/pages_app/patientMsg/patientMsg'
}else if(index==1){
url='/pages_app/video/video'
}else if(index==2){
url='/pages_app/consult/consult'
}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'
}
uni.showToast({
title: `点击了${clickedItem.text}`,
icon: 'none'
});
navTo({
url:url
})
};
// Tab
@ -619,6 +496,13 @@
//
const switchGuideTab = (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) => {
console.log('点击实用指南:', item);
uni.showToast({
title: `查看${item.title}`,
icon: 'none'
});
//
if (item.type === 'guide') {
uni.navigateTo({
url: '/pages/guide/detail?id=' + item.id
});
} else if (item.type === 'courseware') {
uni.navigateTo({
url: '/pages/courseware/list'
});
// PDF
if (item.path) {
const pdfUrl = docUrl+item.path;
console.log(pdfUrl)
if (pdfUrl) {
// H5
// #ifdef H5
window.open(pdfUrl, '_blank');
// #endif
// App
// #ifdef APP-PLUS
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
@ -685,47 +657,135 @@
if (res && res.data) {
const data = res.data;
Object.assign(expertDetail,data.expertDetail)
//
if (data.bannerList && data.bannerList.length > 0) {
bannerList.length = 0;
bannerList.push(...data.bannerList);
//
hasUnread.value = !!data.has_unread;
//
isSignedIn.value = data.sign_in === 1;
//
if (data.welfare_notice && data.welfare_notice.receive_notice) {
visible.value = true;
}
//
if (data.gridList && data.gridList.length > 0) {
// news_list+
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.push(...data.gridList);
gridList.push(
...data.icons_list.map(item => ({
icon: item.img || '',
text: item.name || ''
}))
);
gridList.push({
icon:more,
text:'更多'
})
}
//
if (data.noticeList && data.noticeList.length > 0) {
// meeting_list -> noticeList
if (Array.isArray(data.meeting_list) && data.meeting_list.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
if (data.specialList && data.specialList.length > 0) {
// Eesite_list -> specialList3
if (Array.isArray(data.esite_list) && data.esite_list.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 || ''
}))
);
}
//
if (data.courseList && data.courseList.length > 0) {
// excellencourse_list -> courseList
if (Array.isArray(data.excellencourse_list) && data.excellencourse_list.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 || ''
}))
);
}
//
if (data.replayList && data.replayList.length > 0) {
// video_list -> replayList
if (Array.isArray(data.video_list) && data.video_list.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'
}))
);
}
//
if (data.guideList && data.guideList.length > 0) {
guideList.length = 0;
guideList.push(...data.guideList);
// guide_ist guidesData
guidesData.length = 0;
if (Array.isArray(data.guide_ist) && data.guide_ist.length > 0) {
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) {
@ -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({
testTabbar,
getTabbarStatus,
goToCourseHome
goToCourseHome,
onSignClick
});
</script>
@ -991,7 +1091,19 @@
overflow: hidden;
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 {
height: 100%;
display: flex;
@ -1069,6 +1181,20 @@
display: flex;
flex-direction: column;
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 {
@ -1148,19 +1274,20 @@
white-space: nowrap;
margin-left: 30rpx;
margin-right: 30rpx;
overflow: hidden;
}
.course-list {
display: flex;
display: inline-flex;
gap: 30rpx;
width:auto;
width: auto;
padding-right: 60rpx; /* 让最后一项与右边留出更大间距 */
}
.course-item {
flex:1;
background-color: #f4f4f4;
display: inline-block;
background-color: #f4f4f4;
width: 464rpx;
margin-right: 30rpx;
box-shadow: 0 4rpx 16rpx rgba(0,0,0,0.1);
border-radius: 24rpx;
overflow: hidden;
@ -1185,15 +1312,19 @@
font-size: 20rpx;
}
.course-content {
margin-bottom: 30rpx;
}
.course-title {
white-space: normal;
font-size: 30rpx;
color: #333333;
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 {
@ -1240,8 +1371,7 @@
.replay-item {
cursor: pointer;
height:305rpx;
}
}
.replay-card {
background-color: white;
@ -1255,9 +1385,8 @@
transform: scale(0.98);
}
.replay-image {
.replay-image { /* 不再使用固定高容器,保留占位以防其它样式引用 */
width: 100%;
height: 200rpx;
overflow: hidden;
}
@ -1267,6 +1396,13 @@
object-fit: cover;
}
/* 新增:精彩回放图片自适应 */
.replay-img{
width: 100%;
height: auto;
display: block;
}
.replay-content {
padding: 20rpx;
}

File diff suppressed because it is too large Load Diff

View File

@ -2,6 +2,17 @@
<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="stat">
<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 freeList=['/manager/getSystemTimeStamp','/expertAPI/smsLogin','/login/mobile','/expertAPI/login','/expertAPI/index']
let postData={
...data
}
if(freeList.indexOf(url)!=-1){
if(freeList[4].indexOf(url)!=-1){
if(!token){
header['Authorization']='Bearer ' + ''
}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{
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) {
let timestamp = Date.now();
uni.request({
data: {...data},
data: {...data,...defaultData},
url: url.indexOf('http') != -1 ? url : encodeURI(BASE_URL + url),
method: method,
sslVerify: false,