uniapp 微信小程序仿抖音评论区功能,支持展开收起

news2024/12/23 9:18:51

最近需要写一个评论区功能,所以打算仿照抖音做一个评论功能,支持展开和收起,

首先我们需要对功能做一个拆解,评论区功能,两个模块,一个是发表评论模块,一个是评论展示区。接下来对这两个模块进行详细描述。

使用到的技术 uniapp  uview2.0   文章最后我会贴上全部源码

一、发表评论模块

这个模块使用uview的两个组件来完成分别是u-popup弹出层和u-input输入框

下面是代码和展示图:

<u-popup :show="talkShow" mode="bottom" :customStyle="{'width':'100%','border-radius':'8rpx'}" @close="popclosed" @open="keyboard=true" :safeAreaInsetBottom="true">
			<view class="flex justify-between align-center" style="padding: 32rpx;">
				<div class="cirbOX padding-left padding-right-sm">
					<u--form labelPosition="left" :model="talkData" :rules="Rules" ref="Form" :borderBottom="false">
						<u-form-item label=" " prop="txt" :borderBottom="false" ref="item1" labelWidth='0'>
							<u--input :focus="keyboard" v-model="talkData.txt" cursorSpacing="30" maxlength="100" :placeholder="pinglunHolder" border="none" clearable></u--input>
						</u-form-item>
					</u--form>
				</div>
				<div class="submitpinglun" @click="submit">发布</div>
			</view>
		</u-popup>

这部分需要注意两点

1.input组件的focus属性的设置:

在弹出层弹出的时候 在open事件中对input的focus属性布尔值设置为true,close时候设置focus为false。这样做的目的是在弹出输入评论的弹窗时会拉起小键盘,这个交互方式是模仿的微信朋友圈发布评论的形式。

2.input的cursorSpacing属性(输入框聚焦时底部与键盘的距离)设置:

当键盘拉起时候整个输入框因为设置了cursorSpacing="30",故整体页面会被小键盘托起。 当收起小键盘时候,输入框有回归到手机底部,因为我们popup设置的是底部的弹出层。这样是和微信朋友圈发布评论是对标的。

二、展示评论区的功能

这一部分我封装成了组件,因为需求的要求需要下拉加载评论故在组件外部循环一级评论,组件内部展示一级评论和二级评论,其中二级评论是在组件内部去循环,

循环一级评论的时候需要注意,因为后续要获取pinglun组件的实例,所以在ref的设置上面起初我按照for循环提供的index来拼的字符串,这也导致后续bug的出现埋下伏笔,所以我后续调整了,选择了id这个唯一值作为组件实例的ref名字,这个很关键!

<div v-for="(item,index) in onePagePinglunList" :key="item.id" class="margin-bottom">
				<pinglun :ref="`pinglun-${item.levelOneCommentVo.id}`" :caseIdData="caseId" :data="item" :indexxx="index" @comment="goComment" @noLogin="sonNoLogin"></pinglun>
			</div>

组件代码:

<template>
	<div>
		<!-- 一级评论 -->
		<div class="flex justify-start align-start margin-bottom-sm">
			<div class="margin-right-xs">
				<d-image :dSrc="onePageList.levelOneCommentVo.userAvatar" dMode="aspectFit" dWidth="72rpx" dHeight="72rpx"></d-image>
			</div>
			<div class="flex-sub">
				<div class="flex justify-start align-center">
					<div class="name margin-right-sm">{{onePageList.levelOneCommentVo.userName}}</div>
					<div class="zuozhe flex justify-center align-center" v-if="onePageList.levelOneCommentVo.belongAuthor===1">作者</div>
				</div>
				<div class="flex justify-between align-center" @click="goPinglun(1,onePageList.levelOneCommentVo.id,onePageList.levelOneCommentVo.userName,onePageList.levelOneCommentVo.id)">
					<div class="content flex-sub">{{onePageList.levelOneCommentVo.content}}</div>
					<div class="flex flex-direction align-center" style="width: 68rpx;" @click.stop="likepinglun(1,'',onePageList.levelOneCommentVo.id)">
						<div class="margin-bottom-xs">
							<div v-show="onePageList.levelOneCommentVo.isCurrentUserLike===0">
								<u-icon name="heart" color="#667286" size="34rpx"></u-icon>
							</div>
							<div v-show="onePageList.levelOneCommentVo.isCurrentUserLike===1">
								<u-icon name="heart-fill" color="red" size="34rpx"></u-icon>
							</div>
						</div>
						<div class="likeNum">{{onePageList.levelOneCommentVo.likeCount}}</div>
					</div>
				</div>
				<div class="time">{{ $u.timeFrom(new Date(onePageList.levelOneCommentVo.createTime).getTime())}}</div>
			</div>
		</div>
		<!-- 二级评论 -->
		<div class="erpinglunBox" :style="{'height':`${pingjiaBoxMaxHeight}px`,'opacity':pinglunOpcity,}">
			<div class="pinglunDom">
				<div v-for="(item,index) in onePageList.twoLevelpinglun" :key="item.id" class="margin-bottom-sm">
					<div class="flex justify-start align-start">
						<div class="margin-right-xs">
							<d-image :dSrc="item.userAvatar" dMode="aspectFit" dWidth="72rpx" dHeight="72rpx"></d-image>
						</div>
						<div class="flex-sub">
							<div class="flex justify-start align-center">
								<div class="name margin-right-sm">{{item.userName}}</div>
								<div class="zuozhe flex justify-center align-center margin-right-sm" v-if="item.belongAuthor===1">作者</div>
								<div class="name" v-if="item.isReplayTwoComment===1">回复 {{item.replayLevelTwoCommentUser.userName}}</div>
							</div>
							<div class="flex justify-between align-center" @click="goPinglun(2,item.id,item.userName,onePageList.levelOneCommentVo.id)">
								<div class="content flex-sub">{{item.content}}</div>
								<div class="flex flex-direction align-center" style="width: 68rpx;" @click.stop="likepinglun(2,index,item.id)">
									<div class="margin-bottom-xs">
										<div v-show="item.isCurrentUserLike===0">
											<u-icon name="heart" color="#667286" size="34rpx"></u-icon>
										</div>
										<div v-show="item.isCurrentUserLike===1">
											<u-icon name="heart-fill" color="red" size="34rpx"></u-icon>
										</div>
									</div>
									<div class="likeNum">{{item.likeCount}}</div>
								</div>
							</div>
							<div class="time">{{ $u.timeFrom(new Date(item.createTime).getTime())}}</div>
						</div>
					</div>
				</div>
			</div>
		</div>
		<!-- 展开和收起按钮 -->
		<div class="flex justify-start align-center" style="padding-left: 84rpx;">
			<div class="seeMore padding-top-sm padding-bottom flex align-center" v-if="onePageList.levelTwoCommentCount > 0&&params.current <= totalPage" @click="$u.throttle(getTwoLevelPinglun, 1000,true)">
				<div class="margin-right-xs">查看更多回复</div>
				<u-icon name="arrow-down" color="#00875A" size="28rpx" :bold="true"></u-icon>
			</div>
			<div class="seeMore retract padding-top-sm padding-bottom margin-left flex justify-center align-center" v-if="params.current > 1" @click="$u.throttle(retract, 1000,true)">
				<div class="margin-right-xs">收起</div>
				<u-icon name="arrow-up" color="#00875A" size="28rpx" :bold="true"></u-icon>
			</div>
		</div>
	</div>
</template>

感觉唯一的难点在于因为展开收缩使用的过渡动画,大家应该都知道,想使用这个过渡必须设置有效值,也就是说比如我给高度写过渡动画,从一个高度到一个高度,都需要是具体的值,atuo这种被内容撑开的高度是不作数的。

这里拿高度,需要注意的是需要等待渲染完毕再去获取高度,不然拿到的值就是不准确的。

下面是我写的获取高度的函数。如果一个nexttick也获取不到准确高度,那么就再加个延时器,就差不多可以获取到准确高度了。

	updatHeight() {
				let that = this
				this.$nextTick(() => {
					// this.timer = setTimeout(() => {
					this.createSelectorQuery().select(".pinglunDom").boundingClientRect(function(rect) {
						// console.log(rect);
						that.pingjiaBoxMaxHeight = rect.height
						that.pinglunOpcity = 1
					}).exec();
					// }, 0)
				})
			},

还有一个需要注意的点,就是在一级评论发布之后,需要拿到所有pinglu组件的实例去调用这个方法,重置所有二级评论的高度。调用这个方法的前提也必须是一级评论的渲染完毕,不然还是不起作用

以下是代码:

this.$forceUpdate();
	this.$nextTick(() => {
		for (let i = 0; i < this.onePagePinglunList.length; i++) {
			this.$refs[`pinglun-${this.onePagePinglunList[i].levelOneCommentVo.id}`][0].updatHeight() 
			if (this.onePagePinglunList[i].twoLevelpinglun.length === 0) {
	        this.$refs[`pinglun-${this.onePagePinglunList[i].levelOneCommentVo.id}`][0].params.current = 1
			}
		}
	}) 

至此应该就没什么需要注意的地方了

可能也是第一次写这个功能,还有很多可以优化的地方。希望对各位有所帮助,接下来我把评论功能所有源码贴在下面。

因为我的评论功能是在案例详情里面的,所以有两个文件,一个是案例详情,一个就是封装的评论组件 

案例详情:

<template>
	<div :style="{'padding-bottom':`${(safeAreaBottom*2)+144}`+'rpx'}">
		<div style="padding: 32rpx 32rpx 50rpx;">
			<div class="margin-bottom-sm txt-1">成功蜕变历程</div>
			<div class="toplicheng flex justify-center align-center margin-bottom-sm">
				<div class="flex flex-direction align-center">
					<div class="flex align-center">
						<d-image dSrc="https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/caseDetailIcON2.png" dMode="aspectFit" dWidth="32rpx" dHeight="32rpx"></d-image>
						<div class="toptxt1 margin-left-xs">体重</div>
					</div>
					<div class="toptxt2 margin-top-xs">{{topweightcha||0}}kg</div>
				</div>
				<div class="midBox flex flex-direction align-center">
					<div class="xmonth flex justify-center align-center margin-bottom-xs">逆糖3个月</div>
					<div class="topshuxian"></div>
				</div>
				<div class="flex flex-direction align-center">
					<div class="flex align-center">
						<d-image dSrc="https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/caseDetailIcON1.png" dMode="aspectFit" dWidth="32rpx" dHeight="32rpx"></d-image>
						<div class="toptxt1 margin-left-xs">空腹血糖</div>
					</div>
					<div class="toptxt2 margin-top-xs">{{topxuetangcha||0}}mmol/L</div>
				</div>
			</div>
			<!-- 案例信息 -->
			<div class="caseBox">
				<div class="case-head-box flex justify-between align-center">
					<div class="head-left-box flex">
						<div class="avatarBox">
							<u-avatar :src="detailData.userInfoVo.userAvatar || 'https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/healthy.png'" size="72rpx" mode="aspectFill"></u-avatar>
						</div>
						<div>
							<div class="txt-1 margin-bottom-xs">{{detailData.userInfoVo.userName||'暂无昵称'}}</div>
							<div class="txt-2">{{detailData.isExistServicePack?detailData.servicePackVo.servicePackName:'暂无服务包'}}</div>
						</div>
					</div>
					<div class=" flex justify-center align-center" style="width: 80rpx;height: 80rpx;">
						<u-icon name="share-square" color="" size="34rpx"></u-icon>
					</div>
				</div>
				<div class="caseBoxContentBox">
					<u-row justify="space-start">
						<u-col span="4">
							<view class="txt-4 margin-bottom">项目</view>
						</u-col>
						<u-col span="4">
							<view class="txt-4 margin-bottom">管理前</view>
						</u-col>
						<u-col span="4">
							<view class="txt-4 margin-bottom">管理后</view>
						</u-col>
					</u-row>
					<u-row justify="space-start">
						<u-col span="4">
							<view>
								<div class="txt-3">服务时间</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="startTime">
									{{detailData.managementInfoVo.managementStartTime||'--'}}
								</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="endTime">
									{{detailData.managementInfoVo.managementEndTime||'--'}}
								</div>
							</view>
						</u-col>
					</u-row>
					<div class="line"></div>
					<u-row justify="space-start">
						<u-col span="4">
							<view>
								<div class="txt-3">体重</div>
								<div class="txt-7">kg</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="yellow-box flex justify-center align-center" style="position: relative;">
									{{detailData.managementInfoVo.beforeManagementWeight||'--'}}
									<div class="jiantou" v-if="detailData.managementInfoVo.beforeManagementFastingSugarBloodTrend===3">
										<d-image dSrc="https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/redUp.png" dMode="aspectFit" dWidth="24rpx" dHeight="24rpx"></d-image>
									</div>
									<div class="jiantou" v-else-if="detailData.managementInfoVo.beforeManagementFastingSugarBloodTrend===1">
										<d-image dSrc="https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/greenDown.png" dMode="aspectFit" dWidth="24rpx" dHeight="24rpx"></d-image>
									</div>
								</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="green-box flex justify-center align-center">{{detailData.managementInfoVo.afterManagementWeight||'--'}}</div>
							</view>
						</u-col>
					</u-row>
					<div class="line"></div>
					<u-row justify="space-start">
						<u-col span="4">
							<view>
								<div class="txt-3">空腹血糖</div>
								<div class="txt-7">mmol/L</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="yellow-box flex justify-center align-center" style="position: relative;">
									{{detailData.managementInfoVo.beforeManagementFastingSugarBlood||'--'}}
									<div class="jiantou"
										v-if="detailData.managementInfoVo.beforeManagementWeightTrend===3||detailData.managementInfoVo.beforeManagementWeightTrend===4||detailData.managementInfoVo.beforeManagementWeightTrend===5">
										<d-image dSrc="https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/redUp.png" dMode="aspectFit" dWidth="24rpx" dHeight="24rpx"></d-image>
									</div>
									<div class="jiantou" v-else-if="detailData.managementInfoVo.beforeManagementWeightTrend===1">
										<d-image dSrc="https://tj-data.oss-cn-hangzhou.aliyuncs.com/uploadFiles/wx/mzkHomeland/greenDown.png" dMode="aspectFit" dWidth="24rpx" dHeight="24rpx"></d-image>
									</div>
								</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="green-box flex justify-center align-center">
									{{detailData.managementInfoVo.afterManagementFastingSugarBlood||'--'}}
								</div>
							</view>
						</u-col>
					</u-row>
					<div class="line"></div>
					<u-row justify="space-start">
						<u-col span="4">
							<view>
								<div class="txt-3">用药数量</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="yellow-box flex justify-center align-center">
									{{detailData.managementInfoVo.beforeManagementMedicationCount||'0'}}
								</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="green-box flex justify-center align-center">
									{{detailData.managementInfoVo.afterManagementMedicationCount||'0'}}
								</div>
							</view>
						</u-col>
					</u-row>
					<div class="line"></div>
					<u-row justify="space-start">
						<u-col span="4">
							<view>
								<div class="txt-3">对比照片</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="margin-bottom-sm">
									<div>
										<div v-if="detailData.managementInfoVo.beforeManagementImagePhoto===''" class="noPicBox flex justify-center align-center">暂无</div>
										<div v-else>
											<d-image :dSrc="detailData.managementInfoVo.beforeManagementImagePhoto" dMode="aspectFit" dWidth="140rpx" dHeight="140rpx">
											</d-image>
										</div>
									</div>
								</div>
							</view>
						</u-col>
						<u-col span="4">
							<view>
								<div class="margin-bottom-sm">
									<div>
										<div v-if="detailData.managementInfoVo.afterManagementImagePhoto===''" class="noPicBox flex justify-center align-center">暂无</div>
										<div v-else>
											<d-image :dSrc="detailData.managementInfoVo.afterManagementImagePhoto" dMode="aspectFit" dWidth="140rpx" dHeight="140rpx">
											</d-image>
										</div>
									</div>
								</div>
							</view>
						</u-col>
					</u-row>
				</div>
			</div>
			<!-- 评价 -->
			<div class="evaluateBox margin-top-sm margin-bottom-sm">
				<div class="margin-bottom-sm txt-1">健康评价</div>
				<div class="margin-top-sm flex flex-wrap pingjiaBox">
					<view class="pingjiaDom">
						<div v-for="(item,index) in defaultPingjia" :key="item.id" class="flex ">
							<div class="tag margin-right-sm flex align-center margin-bottom-sm" :style="{'background':item.styleBg}">
								<div style="height: 100%; " class="flex align-start margin-right-sm">
									<d-image :dSrc="item.styleIcon" dMode="aspectFit" dWidth="26rpx" dHeight="26rpx">
									</d-image>
								</div>
								<text :style="{'max-width': '544rpx','word-break': 'break-all','color':item.styleColor}">{{item.content}}</text>
							</div>
						</div>
					</view>
				</div>
			</div>
			<!-- echart -->
			<!-- 	<div class="margin-bottom-sm" style="position: relative;overflow: hidden;" v-for="(item,index) in echartList" :key="index">
				<detailChart :echarType="item"></detailChart>
			</div> -->
		</div>
		<!-- 评论 -->
		<div style="padding: 0 32rpx 56rpx;background: #FFFFFF;">
			<div class="flex justify-between align-center">
				<div class="pingluntitle">{{detailData.interActionVo.commentCount||'0'}} 评论</div>
				<div></div>
			</div>
			<div v-for="(item,index) in onePagePinglunList" :key="item.id" class="margin-bottom">
				<pinglun :ref="`pinglun-${item.levelOneCommentVo.id}`" :caseIdData="caseId" :data="item" :indexxx="index" @comment="goComment" @noLogin="sonNoLogin" @shouqi="shouqiTwoPinglun"></pinglun>
			</div>
		</div>
		<div class="contentB">- 让每个人都能从知识中获得健康 -</div>
		<!-- 底部评价评论点赞收藏 -->
		<div class="pingjialikeBox flex justify-between align-center" :style="{'bottom':`${(safeAreaBottom*2)}`+'rpx'}">
			<div class="talksomething flex justify-center align-center" @click="openPinglun">说点什么吧</div>
			<div class="flex justify-around" style="width: 428rpx;">
				<div class="flex align-center btn" @click="$u.throttle(clickLike, 500,true)">
					<div v-show="detailData.interActionVo.isCurrentUserLike===0">
						<u-icon name="heart" color="" size="34rpx"></u-icon>
					</div>
					<div v-show="detailData.interActionVo.isCurrentUserLike===1">
						<u-icon name="heart-fill" color="red" size="34rpx"></u-icon>
					</div>
					<text class="margin-left-xs">{{detailData.interActionVo.likeCount||'0'}}</text>
				</div>
				<div class="flex align-center btn" @click="$u.throttle(clickCollect, 500,true)">
					<div v-show="detailData.interActionVo.isCurrentUserCollection===0">
						<u-icon name="star" color="" size="34rpx"></u-icon>
					</div>
					<div v-show="detailData.interActionVo.isCurrentUserCollection===1">
						<u-icon name="star-fill" color="#ff991f" size="34rpx"></u-icon>
					</div>
					<text class="margin-left-xs">{{detailData.interActionVo.collectionCount||'0'}}</text>
				</div>
				<div class=" flex align-center btn">
					<u-icon name="chat" color="" size="34rpx"></u-icon>
					<text class="margin-left-xs">{{detailData.interActionVo.commentCount||'0'}}</text>
				</div>
			</div>
		</div>
		<u-popup :show="talkShow" mode="bottom" :customStyle="{'width':'100%','border-radius':'8rpx'}" @close="popclosed" @open="keyboard=true" :safeAreaInsetBottom="true">
			<view class="flex justify-between align-center" style="padding: 32rpx;">
				<div class="cirbOX padding-left padding-right-sm">
					<u--form labelPosition="left" :model="talkData" :rules="Rules" ref="Form" :borderBottom="false">
						<u-form-item label=" " prop="txt" :borderBottom="false" ref="item1" labelWidth='0'>
							<u--input :focus="keyboard" v-model="talkData.txt" cursorSpacing="30" maxlength="100" :placeholder="pinglunHolder" border="none" clearable></u--input>
						</u-form-item>
					</u--form>
				</div>
				<div class="submitpinglun" @click="submit">发布</div>
			</view>
		</u-popup>
		<u-modal title="确认登录" :show="show1" width="608rpx" content="为了您的良好体验,建议您先确认登录 ~" :closeOnClickOverlay="true" @close="show1=false">
			<u-button slot="confirmButton" text="确定" shape="circle" color="#00875A" open-type="getPhoneNumber" @getphonenumber="getPhoneNumber" :customStyle="{'width':'264rpx','height':'68rpx'}"></u-button>
		</u-modal>
		<u-toast ref="uToast"></u-toast>
	</div>
</template>

<script>
	import { mapState } from 'vuex';
	import detailChart from "./detailEchart.vue"
	import pinglun from "./pinglun.vue"
	import { sub } from '@/utils/tool.js'
	import { getDetail, getCurveModule, collection, like, savecomment, getLevelOnePage } from '@/api/case/case.js'
	export default {
		data() {
			return {
				show1: false,
				keyboard: false,
				pinglunHolder: '说点什么吧',
				Rules: {
					'txt': [{
							required: true,
							type: 'any',
							message: '评论不能为空',
							trigger: ['blur', 'change']
						},
						{
							validator: (rule, value, callback) => {
								return value.length < 100;
							},
							message: '您评论的字数超过100,请调整您的字数~',
							trigger: ['blur', 'change']
						}
					],
				},
				talkShow: false,
				talkData: { //弹框form值 评论
					txt: ""
				},
				caseId: null,
				echartList: [],
				detailData: {},
				defaultPingjia: [],
				topweightcha: null,
				topxuetangcha: null,
				pinglunForm: { //接口参数
					caseId: null,
					caseLevelOneCommentId: '', //对一级评论进行回复时不可为空
					caseLevelTwoCommentId: '', //对二级评论进行回复时不可为空
					content: null,
					userId: null,
				},
				params: {
					current: 1,
					limit: 5,
					likeSort: 2, //点赞数排序 1:升序 2:降序
					timeSort: 2, //创建时间排序 1:升序 2:降序
				},
				totalPage: 1,
				onePagePinglunList: [],
				pinglunType: null, //判断用户评论的类型是案例评论(3)还是一级评论(1)还是二级评论(2)
				erpinglunIndex: 0, //暂存二级评论发送给父级组件给的一级评论的index
			}
		},
		components: { detailChart, pinglun },
		computed: {
			...mapState(["hasLogin", "safeAreaBottom", "userInfo", ])
		},
		// 发送给朋友
		onShareAppMessage(res) {
			return {
				title: '妙智健康案例',
				path: `/pages-caseStory/caseDetail/index?caseId=${this.caseId}&title=${this.detailData.userInfoVo.userName}`
			}
		},
		//分享到朋友圈
		onShareTimeline(res) {
			return {
				title: '妙智健康案例',
				query: `caseId=${this.caseId}&title=${this.detailData.userInfoVo.userName}`,
				path: `/pages-caseStory/caseDetail/index` //自定义路径拼参数会导致接收参数会失败
			}
		},
		onReady() {},
		onLoad(option) {
			this.$refs.Form.setRules(this.Rules)
			this.caseId = option?.caseId
			uni.setNavigationBarTitle({
				title: `${option.title||''}案例`
			});
			this.getdetailData()
			this.getLevelOnePageData()
		},
		onUnload() {
			// this.$store.commit('set_caseShareEchartPicList', [])
		},
		onShow() {

		},
		methods: {
			sub,
			async getPhoneNumber(e) {
				// console.log("获取手机号code", e) //获取手机号已经不需要先进行wx.login(最新文档)
				if (!e.detail.code) {
					uni.showToast({
						title: '登录需要获取您的手机号',
						icon: 'none',
						duration: 2500
					})
					return
				}
				let resss = await this.$store.dispatch("loginFn", e) //能同步拿到vux中mutaion的resolve,然后给fourDataNum赋值(登录后数据更新)
				console.log(resss);
				if (resss.state === 1) { //登录成功
					this.show1 = false
				}
			},
			async getdetailData() {
				uni.showLoading({
					title: '加载中...'
				})
				try {
					let res = await getDetail({
						caseId: this.caseId,
						userId: this.hasLogin ? this.userInfo?.userId : ''
					})

					let echarRes = await getCurveModule({
						caseId: this.caseId,
					})
					if (res.state === 1) {
						this.detailData = res.content
						this.defaultPingjia = res.content.personnelEvaluationVoList
						this.topweightcha = this.sub(this.detailData.managementInfoVo.afterManagementWeight, this.detailData.managementInfoVo.beforeManagementWeight)
						this.topxuetangcha = this.sub(this.detailData.managementInfoVo.afterManagementFastingSugarBlood, this.detailData.managementInfoVo.beforeManagementFastingSugarBlood)
					}
					if (echarRes.state === 1) {
						this.echartList = echarRes.content.map(item => {
							let obj = {
								title: item.caseCurveType === 1 ? '硅基' : (item.caseCurveType === 2 ? "微策" : "体重"),
								defaultTime: item.caseCurveType === 1 ? item.curveStartDate : [item.curveStartDate, item.curveEndDate],
								userId: res.content.userInfoVo.userId
							}
							return obj
						})
					}
					uni.hideLoading();
				} catch (e) {
					uni.hideLoading();
					uni.$u.toast(e)
				}
			},
			async clickLike(item, index) {
				if (!this.hasLogin) {
					this.show1 = true
					return
				}
				try {
					let res = await like({
						userId: this.userInfo?.userId,
						caseId: this.caseId,
					})
					if (res.state === 1) {
						if (this.detailData.interActionVo.isCurrentUserLike === 1) {
							this.detailData.interActionVo.isCurrentUserLike = 0
							this.detailData.interActionVo.likeCount--
						} else if (this.detailData.interActionVo.isCurrentUserLike === 0) {
							this.detailData.interActionVo.isCurrentUserLike = 1
							this.detailData.interActionVo.likeCount++
						}
					}
				} catch (e) {
					uni.$u.toast(e)
				}
			},
			async clickCollect(item) {
				if (!this.hasLogin) {
					this.show1 = true
					return
				}
				try {
					let res = await collection({
						userId: this.userInfo?.userId,
						caseId: this.caseId,
					})
					if (res.state === 1) {
						if (this.detailData.interActionVo.isCurrentUserCollection === 1) {
							this.detailData.interActionVo.isCurrentUserCollection = 0
							this.detailData.interActionVo.collectionCount--
						} else if (this.detailData.interActionVo.isCurrentUserCollection === 0) {
							this.detailData.interActionVo.isCurrentUserCollection = 1
							this.detailData.interActionVo.collectionCount++
						}
					}
				} catch (e) {
					uni.$u.toast(e)
				}
			},
			popclosed() {
				this.talkData.txt = ''
				this.keyboard = false
				this.talkShow = false
				console.log(this.keyboard);
			},
			async getLevelOnePageData() {
				uni.showLoading({
					title: '加载中...'
				})
				try {
					let res = await getLevelOnePage({ userId: this.hasLogin ? this.userInfo?.userId : '', caseId: Number(this.caseId), ...this.params })

					if (res.state === 1) {
						for (let i = 0; i < res.content.records.length; i++) {
							res.content.records[i].twoLevelpinglun = []
							if (this.onePagePinglunList.some(item => item.levelOneCommentVo.id === res.content.records[i].levelOneCommentVo.id)) { //删除重复项
								res.content.records.splice(i, 1)
							}
						}
						this.onePagePinglunList = [...this.onePagePinglunList, ...res.content.records]
						this.totalPage = Math.ceil(res.content.total / this.params.limit) //总页数=总数量/每页数量
					}
					uni.hideLoading();
				} catch (e) {
					uni.hideLoading();
					uni.$u.toast(e)
				}
			},
			openPinglun() {
				if (!this.hasLogin) {
					this.show1 = true
					return
				}
				this.pinglunType = 3 //案例评论
				this.pinglunHolder = '说点什么吧'
				this.pinglunForm.caseLevelOneCommentId = '' //不回复一二级评论时候置空该字段
				this.pinglunForm.caseLevelTwoCommentId = '' //不回复一二级评论时候置空该字段

				this.talkShow = true
				// this.keyboard = true
			},
			goComment(e) { //处理回复一级二级评论  案例的顶级评论在openPinglun()函数中
				console.log(e);
				this.pinglunType = e.type //评论类型
				console.log('this.pinglunType', this.pinglunType);
				if (e.type === 1) { //一级评论
					this.pinglunForm.caseLevelOneCommentId = e.id
					this.erpinglunIndex = e.index
					this.pinglunHolder = `回复 @${e.replyName}`
				} else if (e.type === 2) { //二级评论
					this.pinglunForm.caseLevelTwoCommentId = e.id
					this.erpinglunIndex = e.index
					this.pinglunHolder = `回复 @${e.replyName}`
				}
				console.log("点击的item", this.$refs[`pinglun-${this.erpinglunIndex}`][0].onePageList);
				this.talkShow = true
				// this.keyboard = true
				console.log(this.keyboard);
			},
			async submit() {
				this.pinglunForm.userId = this.userInfo.userId
				this.pinglunForm.content = this.talkData.txt
				this.pinglunForm.caseId = this.caseId
				// let form = {
				// 	caseId: this.caseId,
				// 	caseLevelOneCommentId: '', //对一级评论进行回复时不可为空
				// 	caseLevelTwoCommentId: '', //对二级评论进行回复时不可为空
				// 	content: this.talkData.txt,
				// 	userId: this.userInfo.userId,
				// }
				try {
					let res = await savecomment(this.pinglunForm)
					if (res.state === 1) {

						if (this.pinglunType === 3) { //案例评论
							this.onePagePinglunList.unshift({
								twoLevelpinglun: [],
								...res.content.caseCommentHomeVo
							})
							this.$forceUpdate();
							this.$nextTick(() => {
								for (let i = 0; i < this.onePagePinglunList.length; i++) {
									this.$refs[`pinglun-${this.onePagePinglunList[i].levelOneCommentVo.id}`][0].updatHeight() //重置子组件内部height 不管新增一级还是二级都需要全部重置高度不然会出现bug
									if (this.onePagePinglunList[i].twoLevelpinglun.length === 0) {
										this.$refs[`pinglun-${this.onePagePinglunList[i].levelOneCommentVo.id}`][0].params.current = 1
									}
								}
								console.log(this.$refs[`pinglun-${127}`][0].onePageList);
							})
							this.$forceUpdate();
						} else if (this.pinglunType === 1 || this.pinglunType === 2) { //回复一级评论或二级评论  push完需要注意的是在获取分页数据时候去重,因为这个push操作是模拟更新数据,后端并不知晓所以后端未做去重
							console.log(this.$refs[`pinglun-${this.erpinglunIndex}`][0].onePageList.twoLevelpinglun);
							this.$refs[`pinglun-${this.erpinglunIndex}`][0].onePageList.twoLevelpinglun.push({
								...res.content.caseLevelTwoCommentVo
							})
							console.log(this.$refs[`pinglun-${this.erpinglunIndex}`][0].onePageList.twoLevelpinglun);
							let indxx;
							for (let i = 0; i < this.onePagePinglunList.length; i++) {
								if (this.onePagePinglunList[i].levelOneCommentVo.id === this.erpinglunIndex) {
									indxx = i
									break
								}
							}
							// 目的是防止发表一级评论之后父级向下重新注入数据,会触发pinglun组件内部的watch,导致会重置组件内部的twoLevelpinglun为[],
							this.onePagePinglunList[indxx].twoLevelpinglun = this.$refs[`pinglun-${this.erpinglunIndex}`][0].onePageList.twoLevelpinglun
							this.onePagePinglunList[indxx].levelTwoCommentCount++

							this.$refs[`pinglun-${this.erpinglunIndex}`][0].updatHeight() //重置子组件内部height

						}
						this.talkData.txt = '' //重置评论
						this.keyboard = false //自动聚焦设为false
						this.talkShow = false //关闭弹窗
						this.updatePinglunNum() //更新评论数量
						console.log(this.keyboard);
						this.$refs.uToast.show({
							type: 'success',
							message: "已发送评论~",
							duration: 1200,
						})
					}
				} catch (e) {
					uni.$u.toast(e)
				}
			},
			sonNoLogin(e) {
				console.log(e);
				if (!this.hasLogin) {
					this.show1 = true
					return
				}
			},
			async updatePinglunNum() {
				try {
					let res = await getDetail({
						caseId: this.caseId,
						userId: this.hasLogin ? this.userInfo?.userId : ''
					})
					if (res.state === 1) {
						this.detailData = res.content
					}
				} catch (e) {
					uni.$u.toast(e)
				}
			},
			//因为每次发表二级评论都会往父级的twoLevelpinglun添加属性,也就是submit函数里面的565行代码,会导致一个bug 就是发布二级评论后,
			// 因为同时给父级也赋值了,故收起二级评论后,再发表一级评论,重置了了渲染,会将父级被赋值的twoLevelpinglun同步到二级评论的twoLevelpinglun,相当于之前收起操作白重置了twoLevelpinglun
			shouqiTwoPinglun(index) {
				let indxx;
				for (let i = 0; i < this.onePagePinglunList.length; i++) {
					if (this.onePagePinglunList[i].levelOneCommentVo.id === index) {
						indxx = i
						break
					}
				}
				this.onePagePinglunList[indxx].twoLevelpinglun = []
			},
		},
		// 上拉加载
		async onReachBottom() {
			if (this.params.current > this.totalPage) {
				this.$refs.uToast.show({
					type: 'warning',
					message: "已经到底啦~",
					duration: 1200,
				})
				return
			}
			this.params.current += 1
			await this.getLevelOnePageData()
			uni.stopPullDownRefresh() //停止上拉加载
		},
		// 下拉刷新触发
		async onPullDownRefresh() {
			this.params.current = 1 //重置页码
			this.onePagePinglunList = []
			await this.getLevelOnePageData()
			this.$refs.uToast.show({
				type: 'success',
				message: "刷新成功",
				duration: 1200,
			})
			uni.stopPullDownRefresh() //停止下拉刷新
		},

	}
</script>

<style lang="scss" scoped>
	@import '@/pages-caseStory/style/caseCommon.scss';

	.pingluntitle {
		font-size: 28rpx;
		font-family: PingFangSC;
		color: #1F3253;
		height: 90rpx;
		line-height: 90rpx;
	}

	.contentB {
		margin: 42rpx 0;
		font-size: 24rpx;
		font-weight: 400;
		color: #667286;
		text-align: center;
	}

	.pingjialikeBox {
		padding: 32rpx;
		width: 100%;
		height: 144rpx;
		background: #FFFFFF;
		position: fixed;
		left: 0;

		.btn {
			height: 70rpx;
			width: 78rpx;
		}
	}

	.toplicheng {
		height: 160rpx;
		background: linear-gradient(47deg, rgba(23, 144, 109, 0.84) 0%, #5DC063 100%);
		border-radius: 12rpx;
	}

	.topshuxian {
		width: 1rpx;
		height: 80rpx;
		opacity: 0.5;
		border: 2rpx solid #FFFFFF;
	}

	.xmonth {
		width: 156rpx;
		height: 47rpx;
		background: #FFFFFF;
		border-radius: 0rpx 0rpx 14rpx 14rpx;
		opacity: 0.8;
		font-size: 24rpx;
		font-weight: 400;
		color: #00875A;
	}

	.toptxt1 {
		font-size: 28rpx;
		font-weight: 400;
		color: #FFFFFF;
	}

	.toptxt2 {
		font-size: 36rpx;
		font-weight: bold;
		color: #E2FFF5;
	}

	.midBox {
		width: 156rpx;
		height: 160rpx;
		margin-left: 70rpx;
		margin-right: 23rpx;
	}

	.noPicBox {
		width: 140rpx;
		height: 140rpx;
		background: #E7EFF6;
	}

	.jiantou {
		position: absolute;
		top: 0;
		right: 0;
	}

	.evaluateBox {
		background: #FFFFFF;
		border-radius: 12rpx;
		padding: 40rpx 32rpx 78rpx;

		.tag {
			background: #FFF6E9;
			border-radius: 30rpx;
			padding: 20rpx 30rpx 20rpx 20rpx;
			vertical-align: center;

		}
	}

	.talksomething {
		width: 248rpx;
		height: 80rpx;
		background: #F4F5F7;
		border-radius: 40rpx;
		font-size: 28rpx;
		font-weight: 400;
		color: #697588;
	}

	.cirbOX {
		width: 600rpx;
		height: 80rpx;
		background: #F4F5F7;
		border-radius: 40rpx;
	}

	.submitpinglun {
		height: 80rpx;
		width: 64rpx;
		font-size: 32rpx;
		font-weight: 500;
		color: #00875A;
		line-height: 80rpx;
	}
</style>

pinglun组件:

<template>
	<div>
		<!-- 一级评论 -->
		<div class="flex justify-start align-start margin-bottom-sm">
			<div class="margin-right-xs">
				<d-image :dSrc="onePageList.levelOneCommentVo.userAvatar" dMode="aspectFit" dWidth="72rpx" dHeight="72rpx"></d-image>
			</div>
			<div class="flex-sub">
				<div class="flex justify-start align-center">
					<div class="name margin-right-sm">{{onePageList.levelOneCommentVo.userName}}</div>
					<div class="zuozhe flex justify-center align-center" v-if="onePageList.levelOneCommentVo.belongAuthor===1">作者</div>
				</div>
				<div class="flex justify-between align-center" @click="goPinglun(1,onePageList.levelOneCommentVo.id,onePageList.levelOneCommentVo.userName,onePageList.levelOneCommentVo.id)">
					<div class="content flex-sub">{{onePageList.levelOneCommentVo.content}}</div>
					<div class="flex flex-direction align-center" style="width: 68rpx;" @click.stop="likepinglun(1,'',onePageList.levelOneCommentVo.id)">
						<div class="margin-bottom-xs">
							<div v-show="onePageList.levelOneCommentVo.isCurrentUserLike===0">
								<u-icon name="heart" color="#667286" size="34rpx"></u-icon>
							</div>
							<div v-show="onePageList.levelOneCommentVo.isCurrentUserLike===1">
								<u-icon name="heart-fill" color="red" size="34rpx"></u-icon>
							</div>
						</div>
						<div class="likeNum">{{onePageList.levelOneCommentVo.likeCount}}</div>
					</div>
				</div>
				<div class="time">{{ $u.timeFrom(new Date(onePageList.levelOneCommentVo.createTime).getTime())}}</div>
			</div>
		</div>
		<!-- 二级评论 -->
		<div class="erpinglunBox" :style="{'height':`${pingjiaBoxMaxHeight}px`,'opacity':pinglunOpcity,}">
			<div class="pinglunDom">
				<div v-for="(item,index) in onePageList.twoLevelpinglun" :key="item.id" class="margin-bottom-sm">
					<div class="flex justify-start align-start">
						<div class="margin-right-xs">
							<d-image :dSrc="item.userAvatar" dMode="aspectFit" dWidth="72rpx" dHeight="72rpx"></d-image>
						</div>
						<div class="flex-sub">
							<div class="flex justify-start align-center">
								<div class="name margin-right-sm">{{item.userName}}</div>
								<div class="zuozhe flex justify-center align-center margin-right-sm" v-if="item.belongAuthor===1">作者</div>
								<div class="name" v-if="item.isReplayTwoComment===1">回复 {{item.replayLevelTwoCommentUser.userName}}</div>
							</div>
							<div class="flex justify-between align-center" @click="goPinglun(2,item.id,item.userName,onePageList.levelOneCommentVo.id)">
								<div class="content flex-sub">{{item.content}}</div>
								<div class="flex flex-direction align-center" style="width: 68rpx;" @click.stop="likepinglun(2,index,item.id)">
									<div class="margin-bottom-xs">
										<div v-show="item.isCurrentUserLike===0">
											<u-icon name="heart" color="#667286" size="34rpx"></u-icon>
										</div>
										<div v-show="item.isCurrentUserLike===1">
											<u-icon name="heart-fill" color="red" size="34rpx"></u-icon>
										</div>
									</div>
									<div class="likeNum">{{item.likeCount}}</div>
								</div>
							</div>
							<div class="time">{{ $u.timeFrom(new Date(item.createTime).getTime())}}</div>
						</div>
					</div>
				</div>
			</div>
		</div>
		<!-- 展开和收起按钮 -->
		<div class="flex justify-start align-center" style="padding-left: 84rpx;">
			<div class="seeMore padding-top-sm padding-bottom flex align-center" v-if="onePageList.levelTwoCommentCount > 0&&params.current <= totalPage" @click="$u.throttle(getTwoLevelPinglun, 1000,true)">
				<div class="margin-right-xs">查看更多回复</div>
				<u-icon name="arrow-down" color="#00875A" size="28rpx" :bold="true"></u-icon>
			</div>
			<div class="seeMore retract padding-top-sm padding-bottom margin-left flex justify-center align-center" v-if="params.current > 1" @click="$u.throttle(retract, 1000,true)">
				<div class="margin-right-xs">收起</div>
				<u-icon name="arrow-up" color="#00875A" size="28rpx" :bold="true"></u-icon>
			</div>
		</div>
	</div>
</template>

<script>
	import { mapState } from 'vuex';
	import { commentlike, getLevelOnePage, getLevelTwoPage } from '@/api/case/case.js'
	export default {
		data() {
			return {
				caseId: null,
				indexxxx: null, //一级评论的index
				params: {
					current: 1,
					limit: 5,
					timeSort: 1, //创建时间排序 1:升序 2:降序
				},
				totalPage: 1,
				onePageList: {},
				pingjiaBoxMaxHeight: 0,
				pinglunOpcity: 0,
				timer: null,
				timer1: null,
			}
		},
		props: {
			caseIdData: {
				type: Number,
				// 定义是否必须传
				required: true,
				// 定义默认值
				default: 0
			},
			data: {
				type: Object,
				// 定义是否必须传
				required: true,
				// 定义默认值
				default: {}
			},
			indexxx: {
				type: Number,
				// 定义是否必须传
				required: true,
				// 定义默认值
				default: 0
			}
		},
		watch: {
			caseIdData: {
				immediate: true,
				handler(val) {
					this.caseId = val;
				}
			},
			data: {
				immediate: true,
				handler(val) {
					this.onePageList = val;
				}
			},
			indexxx: {
				immediate: true,
				handler(val) {
					this.indexxxx = val;
				}
			}
		},
		components: {},
		computed: {
			...mapState(["hasLogin", "userInfo"])
		},
		mounted() {},
		beforeDestroy() {
			clearTimeout(this.timer)
			clearTimeout(this.timer1)
		},
		methods: {
			async likepinglun(type, index, id) {
				if (!this.hasLogin) {
					this.$emit('noLogin', '一二级评论点赞未登录')
					return
				}

				let form = {};
				if (type === 1) {
					form.caseLevelOneCommentId = id
				} else if (type === 2) {
					form.caseLevelTwoCommentId = id
				}
				try {
					let res = await commentlike({ ...form, userId: this.userInfo?.userId })
					if (res.state === 1) { //如果接口回调成功
						if (type === 1) { //如果点击的是一级评论的点赞
							if (this.onePageList.levelOneCommentVo.isCurrentUserLike === 0) { //判断点赞之前是0还是1 然后取反  并且对应点赞数量同步加减
								this.onePageList.levelOneCommentVo.isCurrentUserLike = 1
								this.onePageList.levelOneCommentVo.likeCount++
							} else if (this.onePageList.levelOneCommentVo.isCurrentUserLike === 1) {
								this.onePageList.levelOneCommentVo.isCurrentUserLike = 0
								this.onePageList.levelOneCommentVo.likeCount--
							}
						} else if (type === 2) { //如果点击的是二级评论的点赞
							if (this.onePageList.twoLevelpinglun[index].isCurrentUserLike === 0) {
								this.onePageList.twoLevelpinglun[index].isCurrentUserLike = 1
								this.onePageList.twoLevelpinglun[index].likeCount++
							} else if (this.onePageList.twoLevelpinglun[index].isCurrentUserLike === 1) {
								this.onePageList.twoLevelpinglun[index].isCurrentUserLike = 0
								this.onePageList.twoLevelpinglun[index].likeCount--
							}
						}
					}
				} catch (e) {
					uni.$u.toast(e)
				}
			},
			goPinglun(e, id, name, indexx) {
				if (e === 1) { //一级评论
					this.$emit('comment', {
						type: e,
						id: id,
						// index: this.indexxxx,
						index: indexx,
						replyName: name, //点击的谁的评论进行回复,用于在输入框的placeholder回显
					})
				} else if (e === 2) { //二级评论
					this.$emit('comment', {
						type: e,
						id: id,
						// index: this.indexxxx,
						index: indexx,
						replyName: name, //点击的谁的评论进行回复,用于在输入框的placeholder回显
					})
				}
			},
			async getTwoLevelPinglun() {
				try {
					let res = await getLevelTwoPage({ userId: this.hasLogin ? this.userInfo?.userId : '', caseLevelOneCommentId: this.onePageList.levelOneCommentVo.id, ...this.params })
					if (res.state === 1) {
						for (let i = 0; i < res.content.records.length; i++) {
							res.content.records[i].twoLevelpinglun = []
							if (this.onePageList.twoLevelpinglun.some(item => item.id === res.content.records[i].id)) { //删除重复项
								res.content.records.splice(i, 1)
								console.log("发现重复项,删除他!!!");
							}
						}

						this.onePageList.twoLevelpinglun = [...this.onePageList.twoLevelpinglun, ...res.content.records]
						this.totalPage = Math.ceil(res.content.total / this.params.limit) //总页数=总数量/每页数量
						this.params.current += 1
						this.updatHeight()
					}
				} catch (e) {
					uni.$u.toast(e)
				}
			},
			retract() {
				this.pingjiaBoxMaxHeight = 0
				this.pinglunOpcity = 0
				this.params.current = 1
				this.onePageList.twoLevelpinglun = []
				this.timer1 = setTimeout(() => { //因为展开动画需要1s 故 在收起的时候延迟置空数组,
					console.log(this.onePageList);
					this.$emit('shouqi', this.onePageList.levelOneCommentVo.id)
				}, 800)
			},
			updatHeight() {
				let that = this
				this.$nextTick(() => {
					// this.timer = setTimeout(() => {
					this.createSelectorQuery().select(".pinglunDom").boundingClientRect(function(rect) {
						// console.log(rect);
						that.pingjiaBoxMaxHeight = rect.height
						that.pinglunOpcity = 1
					}).exec();
					// }, 0)
				})
			},
		}
	}
</script>

<style lang="scss" scoped>
	.name {
		font-size: 24rpx;
		font-weight: 400;
		color: #667286;
	}

	.content {
		font-size: 28rpx;
		font-weight: 400;
		color: #1F3253;
	}

	.time {
		font-size: 20rpx;
		font-weight: 400;
		color: #AFAFAF;
	}

	.likeNum {
		font-size: 20rpx;
		font-weight: 400;
		color: #667286;
	}

	.zuozhe {
		width: 60rpx;
		height: 28rpx;
		background: #FFFFFF;
		border-radius: 18rpx;
		border: 1rpx solid #00875A;
		font-size: 20rpx;
		font-weight: 400;
		color: #00875A;
	}

	.seeMore {
		font-size: 24rpx;
		font-weight: 400;
		color: #00875A;
	}

	.retract {
		width: 150rpx;
		text-align: center;
	}

	.erpinglunBox {
		padding-left: 84rpx;
		transition: height 1s, opacity 2s;
		overflow: hidden;
	}
</style>

案例详情引入的scss文件:

	.font-20 {
		font-size: 20rpx;
		font-weight: 400;
	}

	.font-24 {
		font-size: 24rpx;
		font-weight: 400;
	}


	.txt-1 {
		font-size: 28rpx;
		font-weight: 500;
		color: #0F2C50;
	}

	.txt-2 {
		@extend .font-20;
		color: #667286;
	}

	.txt-3 {
		@extend .font-24;
		color: #667286;
	}

	.txt-4 {
		font-size: 24rpx;
		font-weight: 500;
		color: #667286;
	}

	.txt-5 {
		font-size: 36rpx;
		font-weight: bold;
	}

	.txt-6 {
		@extend .font-24;
		color: #9CADC6;
	}

	.txt-7 {
		@extend .font-20;
		color: #B7BCC3;
	}

	.txt-8 {
		@extend .font-24;
		color: #fff;
	}
	.txt-9 {
		@extend .font-24;
		color: #0F2C50;
	}
	.txt-10 {
		font-size: 28rpx;
		font-weight: 400;
		color: #667286;
	}

	.yell-green-base {
		width: 140rpx;
		height: 52rpx;
		border-radius: 2rpx;
		font-size: 36rpx;
		font-weight: bold;
	}

	.yellow-box {
		@extend .yell-green-base;
		background-color: #FFF4CD;
		color: #FF991F;
	}

	.green-box {
		@extend .yell-green-base;
		background: #E2FFEE;
		color: #00875A;
	}
	.timeFont{
		font-size: 32rpx;
		font-family: DINAlternate-Bold, DINAlternate;
		font-weight: bold;
	}
  .startTime{
		@extend .timeFont;
		color: #FF991F;
  }
  .endTime{
		@extend .timeFont;
		color: #00875A;
  }
	.line {
		height: 1rpx;
		border: 1rpx solid #E6E6E6;
		margin: 16rpx 0;
	}
.caseBox {
		background: #FFFFFF;
		border-radius: 12rpx;
		margin-top: 20rpx;
		padding: 0 32rpx;

		.case-head-box {
			height: 140rpx;
		}

		.avatarBox {
			width: 72rpx;
			height: 72rpx;
			margin-right: 16rpx;
		}

		.caseDetailBtn {
			width: 100rpx;
			height: 44rpx;
			background: #00875A;
			border-radius: 22rpx;
		}

		.rateBox {
			height: 80rpx;
		}

		.mar-80 {
			margin-right: 80rpx;
		}
	}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/954470.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

论文于祥读及复现——《Multi-level Map Construction for Dynamic Scenes》

论文祥读之——动态场景的多层次地图构建 0. 出发点&#xff08;暨摘要&#xff09;1. 引言2. 相关工作3.主要内容概括3.1 几何地图的构建3.1.1 密集点云地图和八叉图的构建3.1.2 平面地图的构建 3.2 对象地图的构建3.2.1 对象参数化和数据关联3.2.2 对象的更新与优化 4. 实验4…

数组——双指针法

双指针法 用两个同向或者反向的指针来代替两重循环。 提醒&#xff1a;不要老想着用同向双指针&#xff0c;有时候&#xff0c;相向双指针更容易解决问题。 LeetCode 27 class Solution {public int removeElement(int[] nums, int val) {int j0;for(int i0;i<nums.leng…

Hibernate(Spring-Data)3种实体继承创建表方式指南

文章目录 引言1. Hibernate 实体继承概述1.1 继承的概念和作用1.2 Hibernate 中的实体继承方式1.3 基础注解 2. 单表继承策略2.1 概述2.2 表结构设计2.3 实体类映射配置 3. 具体类继承策略3.1 概述3.2 表结构设计3.3 实体类映射配置 4. 映射超类策略(每个类一张表)4.1 概述4.2 …

vue3动态路由警告问题

{ path: "/:pathMatch(.*)*", // 必备 component: () > import("/views/error/404.vue"), }, 路由里添加

通达信接口查询持仓数据步骤(一)

股市里用到的查询通达信接口数据通常需要经历以下步骤&#xff1a; 1. 连接接口&#xff1a;首先需要使用相应的接口来连接到交易所或证券公司的服务器。这通常需要使用开发包或API提供的相关函数或方法进行连接。 2. 登录认证&#xff1a;在连接成功后&#xff0c;需要使用登录…

第八周第四天学习总结

测试linux基础并复习基础命令

含纽扣电池的产品出口澳洲需要做哪些认证?认证标准是什么?

澳大利亚含纽扣电池产品新规 01纽扣电池安全问题<<<< 在澳大利亚&#xff0c;已有儿童因为误食纽扣电池而导致死亡&#xff0c;且每月至少有一名儿童因吞咽或插入纽扣/硬币电池而严重受伤&#xff0c;导致其中一些儿童永久性损伤&#xff0c;而全世界数以百万计的…

一起来学shiny把(5)—反应式

什么是shiny&#xff1f;Shiny是一个R包&#xff0c;可让您轻松地直接从 R 构建交互式 Web 应用程序&#xff08;应用程序&#xff09;。本系列是个长教程&#xff0c;带你由浅入深学习shiny。 上一节我们在文章《R语言系列教程—–一起来学shiny吧&#xff08;4&#xff09;》…

《Go 语言第一课》课程学习笔记(十四)

接口 认识接口类型 接口类型是由 type 和 interface 关键字定义的一组方法集合&#xff0c;其中&#xff0c;方法集合唯一确定了这个接口类型所表示的接口。type MyInterface interface {M1(int) errorM2(io.Writer, ...string) }我们在接口类型的方法集合中声明的方法&#…

风力发电场集中监控系统解决方案

作为清洁能源之一&#xff0c;风力发电场近几年装机容量快速增长。8月17日&#xff0c;国家能源局发布1-7月份全国电力工业统计数据。截至7月底&#xff0c;全国累计发电装机容量约27.4亿千瓦&#xff0c;同比增长11.5%。其中&#xff0c;太阳能发电装机容量约4.9亿千瓦&#x…

有指针或者多维数组时,如何写 拷贝构造函数,移动构造函数,拷贝赋值运算符,移动赋值运算符

当成员变量里面有指针或者多维数组时&#xff0c;如何写 拷贝构造函数&#xff0c;移动构造函数&#xff0c;拷贝赋值运算符&#xff0c;移动赋值运算符 头文件 #pragma once #include<iostream> using namespace std; const int num 5; /* 重写C默认函数 */ class De…

低代码赋能| 你真的了解物联网操作系统嘛?点进来看干货!

在聊物联网操作系统之前&#xff0c;我们先来聊物联网。 什么是物联网&#xff1f;物联网&#xff08;IoT&#xff09;是实现万物互联的技术&#xff0c;它赋予物品以控制力、感知力和决策力&#xff0c;推动各类生活场景向智能化方向发展。从架构的层面来说&#xff0c;物联网…

技术科普:汽车开放系统架构AUTOSAR

01.AUTOSAR简介 汽车是现代人类实现“千里江陵一日还”的交通工具&#xff0c;而计算机则是使人脱离繁杂重复脑力劳动的生产技术&#xff0c;两者的结合催生了汽车电子产业的蓬勃发展。 21世纪初&#xff0c;随着汽车电子应用需求的不断增多与硬件资源不断丰富&#xff0c;软…

大数据时代下的精准营销

在大数据时代&#xff0c;人们的信息越来越透明&#xff0c;留在网络上的各种数据也是企业进行营销的一个重要的生产要素。一直以来&#xff0c;营销的科学性正是因为运用了自然科学中一级互联网中的数据收集手段&#xff0c;严谨的记录、搜集和分析消费者的各项数据和日常生活…

Linux以系统服务的方式启动Kafka(其他服务同理)

最终效果&#xff1a; 先回顾命令行的启动方式&#xff1a; kafka的启动 进入kafka的安装目录 1、首先启动zookeeper服务&#xff1a; bin/zookeeper-server-start.sh config/zookeeper.properties2、再启动kafka bin/kafka-server-start.sh config/server.properties &…

vue3 DOM元素渲染完成之后执行

在Vue 3中&#xff0c;可以使用nextTick函数来在DOM元素渲染完成之后执行代码。nextTick函数会在下次DOM更新循环结束之后执行提供的回调函数。 例如&#xff0c;在Vue 3的组件中&#xff0c;可以这样使用nextTick函数&#xff1a; import { nextTick } from vue;export defa…

Vue实现Antv/X6中的示例,以及一些er图开发场景

通过Vue实现Antv X6中的示例&#xff0c;以及一些开发场景&#xff0c;代码已经丢到仓库里了。 lwstudy/antv-x6-vue-demo: Vue实现Antv X6中的示例&#xff0c;以及一些开发场景 (github.com)learn-antv-x6: antv/X6学习 (gitee.com) 介绍 使用脚手架&#xff08;自动生成接…

为了他的鸟,做件很叛逆很酷的事儿

有种鸟儿&#xff0c;叫隐鹮&#xff08;Geronticus eremita&#xff09;&#xff0c;大小如鹅&#xff0c;头部光秃&#xff0c;嘴巴巨大&#xff0c;一个字&#xff0c;丑。可是&#xff0c;它还有一个特点&#xff0c;面临濒危。 为了能在欧洲冬季存活&#xff0c;这种鸟儿需…

mfc140u.dll丢失如何修复?解析mfc140u.dll是什么文件跟修复方法分享

大家好&#xff01;今天&#xff0c;我将和大家分享一下关于计算机中mfc140u.dll丢失的6种解决方法。希望我的分享能对大家在计算机使用过程中遇到问题时提供一些帮助。 首先&#xff0c;我想请大家了解一下什么是mfc140u.dll文件。mfc140u.dll是一个动态链接库文件&#xff0…