You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

389 lines
9.2 KiB

<template>
<view class="daily-page">
<view class="top-safe" :style="{ height: menuButtonInfo.top + 'px' }"></view>
<view class="nav">
<view class="back" @tap="back"></view>
<view class="title">每日同频一题</view>
<view class="ghost"></view>
</view>
<view class="question-card" v-if="question.exists">
<view class="q-deco"> 今日全校同题 </view>
<view class="q-text">{{ question.content }}</view>
<view class="q-count">已有 {{ question.answerCount }} 人回答</view>
</view>
<view class="question-card" v-else>
<view class="q-text">{{ loading ? '正在取今天的题目...' : '今天还没有题目,明天再来吧' }}</view>
</view>
<view class="answer-box" v-if="question.exists && !question.answered">
<textarea class="answer-input" v-model="draft" maxlength="200"
placeholder="写下你的回答,答完就能看到大家的答案(200字以内)"></textarea>
<view class="answer-submit" :class="{ disabled: !draft.trim() || submitting }" @tap="submit">
提交回答,解锁大家的答案 ✨
</view>
</view>
<view class="my-answer" v-if="question.exists && question.answered">
<view class="my-answer-label">我的回答</view>
<view class="my-answer-text">{{ question.myAnswer }}</view>
</view>
<view class="feed" v-if="question.answered">
<view class="feed-title">大家的回答</view>
<view class="answer-card" v-for="item in answers" :key="item.id">
<view class="card-head">
<image class="card-avatar-img" v-if="item.avatarUrl" :src="item.avatarUrl" mode="aspectFill"></image>
<view class="card-avatar" v-else>{{ item.avatarText || '◌' }}</view>
<view class="card-user">
<view class="card-name">
{{ item.anonymousName || '半匿名漂流者' }}
<text class="school-badge" v-if="item.sameSchool">同校</text>
<text class="mine-badge" v-if="item.mine">我</text>
</view>
<view class="card-mode">{{ item.currentMode === 'e' ? 'e 人 · 轻轻热闹' : 'i 人 · 安静陪伴' }}</view>
</view>
<view class="card-chat" v-if="!item.mine" @tap="chatWith(item)">和TA聊聊</view>
</view>
<view class="card-content">{{ item.content }}</view>
</view>
<view class="feed-status">{{ loadingAnswers ? '正在加载...' : (hasMore ? '上滑加载更多' : (answers.length ? '今天先到这里' : '还没有人回答,抢个沙发')) }}</view>
</view>
<view class="locked-tip" v-else-if="question.exists">
<view class="locked-icon">🔒</view>
<view class="locked-text">回答之后才能看到大家的答案</view>
</view>
</view>
</template>
<script>
import { getIeDailyQuestion, answerIeDailyQuestion, pageIeDailyAnswers, matchIeByAnswer } from '@/common/ieApi.js'
export default {
data() {
return {
menuButtonInfo: { top: 44 },
loading: true,
question: { exists: false },
draft: '',
submitting: false,
answers: [],
pageNumber: 1,
hasMore: true,
loadingAnswers: false,
matchingAnswerId: null
}
},
onLoad() {
if (uni.getMenuButtonBoundingClientRect) {
this.menuButtonInfo = uni.getMenuButtonBoundingClientRect()
}
this.loadQuestion()
},
onReachBottom() {
if (this.question.answered && this.hasMore && !this.loadingAnswers) {
this.loadAnswers()
}
},
methods: {
back() {
uni.navigateBack({ fail: () => uni.redirectTo({ url: '/package1/ieBrowser/index' }) })
},
async loadQuestion() {
this.loading = true
const res = await getIeDailyQuestion().catch(() => null)
this.loading = false
if (!res) return
this.question = res
if (res.answered) {
this.pageNumber = 1
this.answers = []
this.hasMore = true
this.loadAnswers()
}
},
async submit() {
const content = this.draft.trim()
if (!content || this.submitting) return
this.submitting = true
try {
const res = await answerIeDailyQuestion(content)
if (!res) return
this.draft = ''
uni.showToast({ title: '已提交,去看看大家的回答', icon: 'none' })
this.loadQuestion()
} finally {
this.submitting = false
}
},
async loadAnswers() {
if (this.loadingAnswers) return
this.loadingAnswers = true
try {
const res = await pageIeDailyAnswers(this.pageNumber, 10)
if (!res) return
const records = res.records || []
this.answers = this.pageNumber === 1 ? records : this.answers.concat(records)
this.hasMore = this.answers.length < (res.total || 0)
this.pageNumber += 1
} finally {
this.loadingAnswers = false
}
},
async chatWith(item) {
if (this.matchingAnswerId) return
this.matchingAnswerId = item.id
try {
const match = await matchIeByAnswer(item.id)
if (!match || !match.roomId) return
uni.navigateTo({
url: '/package1/ieBrowser/chat?mode=' + (match.mode || 'i') +
'&roomId=' + match.roomId +
'&targetUserId=' + match.targetUserId +
'&name=' + encodeURIComponent(match.anonymousName || item.anonymousName || '') +
'&avatar=' + encodeURIComponent(match.avatarText || item.avatarText || '') +
'&avatarUrl=' + encodeURIComponent(match.avatarUrl || item.avatarUrl || '') +
'&quote=' + encodeURIComponent(match.quoteText || '')
})
} finally {
this.matchingAnswerId = null
}
}
}
}
</script>
<style scoped>
.daily-page {
min-height: 100vh;
padding: 0 32rpx 60rpx;
box-sizing: border-box;
color: #161b2e;
background: radial-gradient(circle at 18% 10%, rgba(167, 139, 250, .26), transparent 320rpx),
radial-gradient(circle at 86% 24%, rgba(255, 184, 209, .26), transparent 320rpx),
linear-gradient(180deg, #fbfdff, #eef4ff 62%, #fff4e8);
}
.nav {
display: flex;
align-items: center;
height: 88rpx;
}
.back {
width: 62rpx;
font-size: 52rpx;
color: rgba(22, 27, 46, .62);
}
.title {
flex: 1;
text-align: center;
font-size: 31rpx;
font-weight: 800;
}
.ghost {
width: 62rpx;
}
.question-card {
margin-top: 18rpx;
padding: 36rpx 30rpx;
border-radius: 34rpx;
background: rgba(255, 255, 255, .8);
border: 1rpx solid rgba(255, 255, 255, .9);
box-shadow: 0 18rpx 44rpx rgba(96, 112, 160, .14);
text-align: center;
}
.q-deco {
font-size: 22rpx;
font-weight: 800;
color: #7a4dff;
letter-spacing: 4rpx;
}
.q-text {
margin-top: 16rpx;
font-size: 36rpx;
font-weight: 800;
line-height: 1.5;
}
.q-count {
margin-top: 14rpx;
font-size: 23rpx;
color: rgba(22, 27, 46, .5);
}
.answer-box {
margin-top: 24rpx;
}
.answer-input {
width: 100%;
height: 220rpx;
padding: 24rpx;
box-sizing: border-box;
border-radius: 28rpx;
background: rgba(255, 255, 255, .85);
border: 1rpx solid rgba(255, 255, 255, .9);
font-size: 26rpx;
}
.answer-submit {
margin-top: 20rpx;
height: 92rpx;
line-height: 92rpx;
text-align: center;
border-radius: 999rpx;
font-size: 28rpx;
font-weight: 800;
color: #fff;
background: linear-gradient(135deg, #7a4dff, #ff7eb3);
box-shadow: 0 16rpx 36rpx rgba(122, 77, 255, .3);
}
.answer-submit.disabled {
opacity: .5;
box-shadow: none;
}
.my-answer {
margin-top: 24rpx;
padding: 24rpx 26rpx;
border-radius: 26rpx;
background: rgba(122, 77, 255, .08);
border: 1rpx solid rgba(122, 77, 255, .16);
}
.my-answer-label {
font-size: 22rpx;
font-weight: 800;
color: #7a4dff;
}
.my-answer-text {
margin-top: 8rpx;
font-size: 26rpx;
line-height: 1.5;
}
.feed {
margin-top: 28rpx;
}
.feed-title {
font-size: 28rpx;
font-weight: 800;
margin-bottom: 16rpx;
}
.answer-card {
margin-bottom: 18rpx;
padding: 24rpx;
border-radius: 28rpx;
background: rgba(255, 255, 255, .82);
border: 1rpx solid rgba(255, 255, 255, .9);
box-shadow: 0 12rpx 30rpx rgba(96, 112, 160, .1);
}
.card-head {
display: flex;
align-items: center;
}
.card-avatar,
.card-avatar-img {
width: 76rpx;
height: 76rpx;
border-radius: 50%;
flex-shrink: 0;
}
.card-avatar {
line-height: 76rpx;
text-align: center;
font-size: 30rpx;
font-weight: 800;
color: #5a4632;
background: linear-gradient(135deg, #ffe9c9, #ffd6a5);
}
.card-user {
flex: 1;
margin-left: 16rpx;
min-width: 0;
}
.card-name {
font-size: 27rpx;
font-weight: 800;
}
.school-badge,
.mine-badge {
margin-left: 10rpx;
padding: 2rpx 12rpx;
border-radius: 999rpx;
font-size: 18rpx;
font-weight: 800;
vertical-align: 4rpx;
}
.school-badge {
color: #1d7a4f;
background: rgba(61, 220, 151, .16);
}
.mine-badge {
color: #7a4dff;
background: rgba(122, 77, 255, .12);
}
.card-mode {
margin-top: 4rpx;
font-size: 21rpx;
color: rgba(22, 27, 46, .48);
}
.card-chat {
flex-shrink: 0;
padding: 12rpx 24rpx;
border-radius: 999rpx;
font-size: 22rpx;
font-weight: 800;
color: #fff;
background: linear-gradient(135deg, #7a4dff, #ff7eb3);
}
.card-content {
margin-top: 16rpx;
font-size: 26rpx;
line-height: 1.55;
color: rgba(22, 27, 46, .82);
}
.feed-status {
padding: 24rpx 0 10rpx;
text-align: center;
font-size: 23rpx;
color: rgba(22, 27, 46, .42);
}
.locked-tip {
margin-top: 60rpx;
text-align: center;
}
.locked-icon {
font-size: 64rpx;
}
.locked-text {
margin-top: 14rpx;
font-size: 25rpx;
color: rgba(22, 27, 46, .5);
}
</style>