聊天室结构和样式
<script setup lang="ts">
import { ref, onMounted } from 'vue'
import { flagOptions, timeOptions } from '@/services/constants'
import { MsgType, type IllnessTime } from '@/enums'
import type { Message } from '@/types/room'
import { ImagePreview, showImagePreview } from 'vant'
import { useUserStore } from '@/stores'
import type { Image } from '@/types/consult'
import dayjs from 'dayjs'
const formatTime = (time: string) => dayjs(time).format('HH:mm')
const store = useUserStore()
defineProps<{ list: Message[] }>()
const getIllnessTimeText = (time: IllnessTime) =>
timeOptions.find((item) => item.value === time)?.label
const getConsultFlagText = (flag: 0 | 1) =>
flagOptions.find((item) => item.value === flag)?.label
const previewImg = (pictures?: Image[]) => {
if (pictures && pictures.length)
// let a = pictures!.map((item) => item.url)
// console.log(a)
// showImagePreview(a)
showImagePreview(pictures!.map((item) => item.url))
}
const loadMy = (notScroll?: boolean) => {
if (notScroll === true) return
window.scrollTo(0, document.body.scrollHeight)
}
</script>
<template>
<template
v-for="{ msgType, msg, createTime, from, toAvatar, fromAvatar } in list"
:key="msg.id"
>
<!-- 病情描述 -->
<div class="msg msg-illness" v-if="msgType === MsgType.CardPat">
<div class="patient van-hairline--bottom" v-if="msg.consultRecord">
<p>
{{ msg.consultRecord.patientInfo.name }}
{{ msg.consultRecord.patientInfo.genderValue }}
{{ msg.consultRecord.patientInfo.age }}岁
</p>
<p>
{{ getIllnessTimeText(msg.consultRecord.illnessTime) }} |
{{ getConsultFlagText(msg.consultRecord.consultFlag) }}
</p>
</div>
<van-row>
<van-col span="6">病情描述</van-col>
<van-col span="18">{{ msg.consultRecord?.illnessDesc }}</van-col>
<van-col span="6">图片</van-col>
<van-col span="18" @click="previewImg(msg.consultRecord?.pictures)">
点击查看
</van-col>
</van-row>
</div>
<!-- 温馨提示 -->
<div class="msg msg-tip" v-if="msgType === MsgType.NotifyTip">
<div class="content">
<span class="green">温馨提示:</span>
<span>{{ msg.content }}</span>
</div>
</div>
<!-- 通用通知 -->
<div class="msg msg-tip" v-if="msgType === 31">
<div class="content">
<span>{{ msg.content }}</span>
</div>
</div>
<!-- 我发的消息 -->
<div
class="msg msg-to"
v-if="msgType === MsgType.MsgText && store.user?.id === from"
>
<div class="content">
<div class="time">{{ formatTime(createTime) }}</div>
<div class="pao">{{ msg.content }}</div>
</div>
<van-image :src="store.user?.avatar || toAvatar" />
</div>
<!-- 医生发的消息 -->
<div
class="msg msg-from"
v-if="msgType === MsgType.MsgText && store.user?.id !== from"
>
<van-image :src="fromAvatar" />
<div class="content">
<div class="time">{{ formatTime(createTime) }}</div>
<div class="pao">{{ msg.content }}</div>
</div>
</div>
<div
class="msg msg-to"
v-if="msgType === MsgType.MsgImage && store.user?.id === from"
>
<div class="content">
<div class="time">{{ formatTime(createTime) }}</div>
<van-image @load="loadMy" fit="contain" :src="msg.picture?.url" />
</div>
<van-image :src="store.user?.avatar || toAvatar" />
</div>
<div
class="msg msg-from"
v-if="msgType === MsgType.MsgImage && store.user?.id !== from"
>
<van-image :src="fromAvatar" />
<div class="content">
<div class="time">{{ formatTime(createTime) }}</div>
<van-image @load="loadMy" fit="contain" :src="msg.picture?.url" />
</div>
</div>
</template>
<!-- 患者卡片 -->
<!-- <div class="msg msg-illness">
<div class="patient van-hairline--bottom">
<p>李富贵 男 31岁</p>
<p>一周内 | 未去医院就诊</p>
</div>
<van-row>
<van-col span="6">病情描述</van-col>
<van-col span="18">头痛、头晕、恶心</van-col>
<van-col span="6">图片</van-col>
<van-col span="18">点击查看</van-col>
</van-row>
</div> -->
<!-- 通知-通用 -->
<!-- <div class="msg msg-tip">
<div class="content">
<span>医护人员正在赶来,请耐心等候</span>
</div>
</div> -->
<!-- 通知-温馨提示 -->
<!-- <div class="msg msg-tip">
<div class="content">
<span class="green">温馨提示:</span>
<span>在线咨询不能代替面诊,医护人员建议仅供参考</span>
</div>
</div> -->
<!-- 通知-结束 -->
<!-- <div class="msg msg-tip msg-tip-cancel">
<div class="content">
<span>订单取消</span>
</div>
</div> -->
<!-- 发送文字 -->
<!-- <div class="msg msg-to">
<div class="content">
<div class="time">20:12</div>
<div class="pao">大夫你好?</div>
</div>
<van-image
src="https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/popular_3.jpg"
/>
</div> -->
<!-- 发送图片 -->
<!-- <div class="msg msg-to">
<div class="content">
<div class="time">20:12</div>
<van-image
fit="contain"
src="https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/popular_3.jpg"
/>
</div>
<van-image
src="https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/popular_3.jpg"
/>
</div> -->
<!-- 接收文字 -->
<!-- <div class="msg msg-from">
<van-image
src="https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/popular_3.jpg"
/>
<div class="content">
<div class="time">20:12</div>
<div class="pao">哪里不舒服</div>
</div>
</div> -->
<!-- 接收图片 -->
<!-- <div class="msg msg-from">
<van-image
src="https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/popular_3.jpg"
/>
<div class="content">
<div class="time">20:12</div>
<van-image
fit="contain"
src="https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/popular_3.jpg"
/>
</div>
</div> -->
<!-- 处方卡片 -->
<!-- <div class="msg msg-recipe">
<div class="content">
<div class="head van-hairline--bottom">
<div class="head-tit">
<h3>电子处方</h3>
<p>原始处方 <van-icon name="arrow"></van-icon></p>
</div>
<p>李富贵 男 31岁 血管性头痛</p>
<p>开方时间:2022-01-15 14:21:42</p>
</div>
<div class="body">
<div class="body-item" v-for="i in 2" :key="i">
<div class="durg">
<p>优赛明 维生素E乳</p>
<p>口服,每次1袋,每天3次,用药3天</p>
</div>
<div class="num">x1</div>
</div>
</div>
<div class="foot"><span>购买药品</span></div>
</div>
</div> -->
<!-- 评价卡片,后期实现 -->
</template>
<style lang="scss" scoped>
@import '@/styles/room.scss';
</style>
样式
.msg {
display: flex;
padding: 15px;
// 医生消息+图片
&-from {
.content {
max-width: 240px;
min-width: 52px;
.time {
color: var(--cp-tip);
margin-bottom: 5px;
}
.pao {
padding: 15px;
background-color: #fff;
color: var(--cp-text3);
font-size: 15px;
border-radius: 8px;
position: relative;
&::before {
content: '';
position: absolute;
left: -13px;
top: 10px;
width: 13px;
height: 16px;
background: #fff;
border-top-left-radius: 13px 3px;
}
&::after {
content: '';
position: absolute;
left: -13px;
top: 13px;
width: 13px;
height: 13px;
background: var(--cp-bg);
border-top-right-radius: 13px 13px;
}
}
.van-image {
max-height: 160px;
max-width: 160px;
border-radius: 8px;
overflow: hidden;
border: 1px solid var(--cp-line);
}
}
> .van-image {
width: 38px;
height: 38px;
border-radius: 50%;
overflow: hidden;
margin-right: 13px;
}
}
// 患者消息+图片
&-to {
justify-content: flex-end;
.content {
max-width: 240px;
min-width: 52px;
.time {
color: var(--cp-tip);
margin-bottom: 5px;
text-align: right;
}
.pao {
padding: 15px;
background-color: var(--cp-primary);
color: #fff;
font-size: 15px;
border-radius: 8px;
position: relative;
&::before {
content: '';
position: absolute;
right: -13px;
top: 10px;
width: 13px;
height: 16px;
background: var(--cp-primary);
border-top-right-radius: 13px 3px;
}
&::after {
content: '';
position: absolute;
right: -13px;
top: 13px;
width: 13px;
height: 13px;
background: var(--cp-bg);
border-top-left-radius: 13px 13px;
}
}
.van-image {
max-height: 160px;
max-width: 160px;
border-radius: 8px;
overflow: hidden;
border: 1px solid var(--cp-line);
}
}
> .van-image {
width: 38px;
height: 38px;
border-radius: 50%;
overflow: hidden;
margin-left: 13px;
}
}
&-tip {
justify-content: center;
font-size: 12px;
.content {
height: 34px;
line-height: 34px;
background-color: #fff;
color: var(--cp-tip);
font-size: 12px;
border-radius: 17px;
padding: 0 16px;
max-width: 100%;
.green {
color: var(--cp-primary);
}
}
&-cancel {
.content {
background-color: #ededed;
}
}
}
&-illness {
display: block;
background-color: #fff;
margin: 15px;
border-radius: 8px;
font-size: 12px;
.patient {
padding-bottom: 15px;
margin-bottom: 15px;
> p {
&:first-child {
font-size: 16px;
}
&:last-child {
margin-top: 5px;
color: var(--cp-tip);
}
}
}
.van-col {
&:nth-child(-n + 2) {
margin-bottom: 5px;
}
&:nth-child(2n) {
color: var(--cp-tip);
}
}
}
&-recipe {
padding: 15px;
.content {
background-color: #fff;
border-radius: 8px;
color: var(--cp-tip);
font-size: 12px;
flex: 1;
.head {
padding: 15px;
.head-tit {
display: flex;
justify-content: space-between;
> h3 {
font-weight: normal;
font-size: 16px;
color: var(--cp-text1);
}
}
p {
margin-top: 5px;
}
}
.body {
padding: 15px 15px 0 15px;
&-item {
display: flex;
margin-bottom: 15px;
.durg {
flex: 1;
> p {
&:first-child {
font-size: 14px;
color: var(--cp-text1);
margin-bottom: 5px;
}
}
}
.num {
color: var(--cp-text1);
}
}
}
.foot {
background-color: var(--cp-plain);
color: var(--cp-primary);
font-size: 16px;
text-align: center;
height: 42px;
line-height: 42px;
}
}
}
}
本文来自博客园,作者:jialiangzai,转载请注明原文链接:https://www.cnblogs.com/zsnhweb/articles/17292064.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· C#/.NET/.NET Core技术前沿周刊 | 第 29 期(2025年3.1-3.9)
· 从HTTP原因短语缺失研究HTTP/2和HTTP/3的设计差异