468 lines
12 KiB
JavaScript
468 lines
12 KiB
JavaScript
/**
|
||
* Soul创业派对 - 我的页面
|
||
* 开发: 卡若
|
||
* 技术支持: 存客宝
|
||
*/
|
||
|
||
const app = getApp()
|
||
|
||
Page({
|
||
data: {
|
||
// 系统信息
|
||
statusBarHeight: 44,
|
||
navBarHeight: 88,
|
||
|
||
// 用户状态
|
||
isLoggedIn: false,
|
||
userInfo: null,
|
||
|
||
// 统计数据
|
||
totalSections: 62,
|
||
readCount: 0,
|
||
referralCount: 0,
|
||
earnings: 0,
|
||
pendingEarnings: 0,
|
||
|
||
// 阅读统计
|
||
totalReadTime: 0,
|
||
matchHistory: 0,
|
||
|
||
// Tab切换
|
||
activeTab: 'overview', // overview | footprint
|
||
|
||
// 最近阅读
|
||
recentChapters: [],
|
||
|
||
// 功能配置
|
||
matchEnabled: false, // 找伙伴功能开关
|
||
|
||
// 菜单列表
|
||
menuList: [
|
||
{ id: 'orders', title: '我的订单', icon: '📦', count: 0 },
|
||
{ id: 'referral', title: '推广中心', icon: '🎁', iconBg: 'gold', badge: '90%佣金' },
|
||
{ id: 'about', title: '关于作者', icon: 'ℹ️', iconBg: 'brand' },
|
||
{ id: 'settings', title: '设置', icon: '⚙️', iconBg: 'gray' }
|
||
],
|
||
|
||
// 登录弹窗
|
||
showLoginModal: false,
|
||
isLoggingIn: false,
|
||
|
||
// 修改昵称弹窗
|
||
showNicknameModal: false,
|
||
editingNickname: ''
|
||
},
|
||
|
||
onLoad() {
|
||
this.setData({
|
||
statusBarHeight: app.globalData.statusBarHeight,
|
||
navBarHeight: app.globalData.navBarHeight
|
||
})
|
||
this.loadFeatureConfig()
|
||
this.initUserStatus()
|
||
},
|
||
|
||
onShow() {
|
||
// 设置TabBar选中状态(根据 matchEnabled 动态设置)
|
||
if (typeof this.getTabBar === 'function' && this.getTabBar()) {
|
||
const tabBar = this.getTabBar()
|
||
if (tabBar.updateSelected) {
|
||
tabBar.updateSelected()
|
||
} else {
|
||
const selected = tabBar.data.matchEnabled ? 3 : 2
|
||
tabBar.setData({ selected })
|
||
}
|
||
}
|
||
this.initUserStatus()
|
||
},
|
||
|
||
// 加载功能配置
|
||
async loadFeatureConfig() {
|
||
try {
|
||
const res = await app.request({
|
||
url: '/api/db/config',
|
||
method: 'GET'
|
||
})
|
||
|
||
if (res && res.features) {
|
||
this.setData({
|
||
matchEnabled: res.features.matchEnabled === true
|
||
})
|
||
}
|
||
} catch (error) {
|
||
console.log('加载功能配置失败:', error)
|
||
// 默认关闭找伙伴功能
|
||
this.setData({ matchEnabled: false })
|
||
}
|
||
},
|
||
|
||
// 初始化用户状态
|
||
initUserStatus() {
|
||
const { isLoggedIn, userInfo, hasFullBook, purchasedSections } = app.globalData
|
||
|
||
if (isLoggedIn && userInfo) {
|
||
const readIds = app.globalData.readSectionIds || []
|
||
const recentList = readIds.slice(-5).reverse().map(id => ({
|
||
id: id,
|
||
title: `章节 ${id}`
|
||
}))
|
||
|
||
// 截短用户ID显示
|
||
const userId = userInfo.id || ''
|
||
const userIdShort = userId.length > 20 ? userId.slice(0, 10) + '...' + userId.slice(-6) : userId
|
||
|
||
// 获取微信号(优先显示)
|
||
const userWechat = wx.getStorageSync('user_wechat') || userInfo.wechat || ''
|
||
|
||
// 格式化收益金额(保留两位小数)
|
||
const earnings = Number(userInfo.earnings || 0)
|
||
const pendingEarnings = Number(userInfo.pendingEarnings || 0)
|
||
|
||
this.setData({
|
||
isLoggedIn: true,
|
||
userInfo,
|
||
userIdShort,
|
||
userWechat,
|
||
readCount: Math.min(app.getReadCount(), this.data.totalSections || 62),
|
||
referralCount: userInfo.referralCount || 0,
|
||
earnings: earnings.toFixed(2),
|
||
pendingEarnings: pendingEarnings.toFixed(2),
|
||
recentChapters: recentList,
|
||
totalReadTime: Math.floor(Math.random() * 200) + 50
|
||
})
|
||
} else {
|
||
this.setData({
|
||
isLoggedIn: false,
|
||
userInfo: null,
|
||
userIdShort: '',
|
||
readCount: app.getReadCount(),
|
||
referralCount: 0,
|
||
earnings: '0.00',
|
||
pendingEarnings: '0.00',
|
||
recentChapters: []
|
||
})
|
||
}
|
||
},
|
||
|
||
// 微信原生获取头像(button open-type="chooseAvatar" 回调)
|
||
async onChooseAvatar(e) {
|
||
const tempAvatarUrl = e.detail.avatarUrl
|
||
if (!tempAvatarUrl) return
|
||
|
||
wx.showLoading({ title: '上传中...', mask: true })
|
||
|
||
try {
|
||
// 1. 先上传图片到服务器
|
||
console.log('[My] 开始上传头像:', tempAvatarUrl)
|
||
|
||
const uploadRes = await new Promise((resolve, reject) => {
|
||
wx.uploadFile({
|
||
url: app.globalData.baseUrl + '/api/upload',
|
||
filePath: tempAvatarUrl,
|
||
name: 'file',
|
||
formData: {
|
||
folder: 'avatars'
|
||
},
|
||
success: (res) => {
|
||
try {
|
||
const data = JSON.parse(res.data)
|
||
if (data.success) {
|
||
resolve(data)
|
||
} else {
|
||
reject(new Error(data.error || '上传失败'))
|
||
}
|
||
} catch (err) {
|
||
reject(new Error('解析响应失败'))
|
||
}
|
||
},
|
||
fail: (err) => {
|
||
reject(err)
|
||
}
|
||
})
|
||
})
|
||
|
||
// 2. 获取上传后的完整URL
|
||
const avatarUrl = app.globalData.baseUrl + uploadRes.data.url
|
||
console.log('[My] 头像上传成功:', avatarUrl)
|
||
|
||
// 3. 更新本地头像
|
||
const userInfo = this.data.userInfo
|
||
userInfo.avatar = avatarUrl
|
||
this.setData({ userInfo })
|
||
app.globalData.userInfo = userInfo
|
||
wx.setStorageSync('userInfo', userInfo)
|
||
|
||
// 4. 同步到服务器数据库
|
||
await app.request('/api/user/update', {
|
||
method: 'POST',
|
||
data: { userId: userInfo.id, avatar: avatarUrl }
|
||
})
|
||
|
||
wx.hideLoading()
|
||
wx.showToast({ title: '头像更新成功', icon: 'success' })
|
||
|
||
} catch (e) {
|
||
wx.hideLoading()
|
||
console.error('[My] 上传头像失败:', e)
|
||
wx.showToast({
|
||
title: e.message || '上传失败,请重试',
|
||
icon: 'none'
|
||
})
|
||
}
|
||
},
|
||
|
||
// 微信原生获取昵称(input type="nickname" 回调)
|
||
async onNicknameInput(e) {
|
||
const nickname = e.detail.value
|
||
if (!nickname || nickname === this.data.userInfo?.nickname) return
|
||
|
||
try {
|
||
const userInfo = this.data.userInfo
|
||
userInfo.nickname = nickname
|
||
this.setData({ userInfo })
|
||
app.globalData.userInfo = userInfo
|
||
wx.setStorageSync('userInfo', userInfo)
|
||
|
||
// 同步到服务器
|
||
await app.request('/api/user/update', {
|
||
method: 'POST',
|
||
data: { userId: userInfo.id, nickname }
|
||
})
|
||
|
||
wx.showToast({ title: '昵称已获取', icon: 'success' })
|
||
} catch (e) {
|
||
console.log('同步昵称失败', e)
|
||
}
|
||
},
|
||
|
||
// 打开昵称修改弹窗
|
||
editNickname() {
|
||
this.setData({
|
||
showNicknameModal: true,
|
||
editingNickname: this.data.userInfo?.nickname || ''
|
||
})
|
||
},
|
||
|
||
// 关闭昵称弹窗
|
||
closeNicknameModal() {
|
||
this.setData({
|
||
showNicknameModal: false,
|
||
editingNickname: ''
|
||
})
|
||
},
|
||
|
||
// 阻止事件冒泡
|
||
stopPropagation() {},
|
||
|
||
// 昵称输入实时更新
|
||
onNicknameInput(e) {
|
||
this.setData({
|
||
editingNickname: e.detail.value
|
||
})
|
||
},
|
||
|
||
// 昵称变化(微信自动填充时触发)
|
||
onNicknameChange(e) {
|
||
console.log('[My] 昵称已自动填充:', e.detail.value)
|
||
this.setData({
|
||
editingNickname: e.detail.value
|
||
})
|
||
},
|
||
|
||
// 确认修改昵称
|
||
async confirmNickname() {
|
||
const newNickname = this.data.editingNickname.trim()
|
||
|
||
if (!newNickname) {
|
||
wx.showToast({ title: '昵称不能为空', icon: 'none' })
|
||
return
|
||
}
|
||
|
||
if (newNickname.length < 1 || newNickname.length > 20) {
|
||
wx.showToast({ title: '昵称1-20个字符', icon: 'none' })
|
||
return
|
||
}
|
||
|
||
// 关闭弹窗
|
||
this.closeNicknameModal()
|
||
|
||
// 显示加载
|
||
wx.showLoading({ title: '更新中...' })
|
||
|
||
try {
|
||
// 更新本地
|
||
const userInfo = this.data.userInfo
|
||
userInfo.nickname = newNickname
|
||
this.setData({ userInfo })
|
||
app.globalData.userInfo = userInfo
|
||
wx.setStorageSync('userInfo', userInfo)
|
||
|
||
// 同步到服务器
|
||
await app.request('/api/user/update', {
|
||
method: 'POST',
|
||
data: { userId: userInfo.id, nickname: newNickname }
|
||
})
|
||
|
||
wx.hideLoading()
|
||
wx.showToast({ title: '昵称已更新', icon: 'success' })
|
||
}
|
||
}
|
||
})
|
||
},
|
||
|
||
// 复制用户ID
|
||
copyUserId() {
|
||
const userId = this.data.userInfo?.id || ''
|
||
if (!userId) {
|
||
wx.showToast({ title: '暂无ID', icon: 'none' })
|
||
return
|
||
}
|
||
wx.setClipboardData({
|
||
data: userId,
|
||
success: () => {
|
||
wx.showToast({ title: 'ID已复制', icon: 'success' })
|
||
}
|
||
})
|
||
},
|
||
|
||
// 切换Tab
|
||
switchTab(e) {
|
||
const tab = e.currentTarget.dataset.tab
|
||
this.setData({ activeTab: tab })
|
||
},
|
||
|
||
// 显示登录弹窗
|
||
showLogin() {
|
||
this.setData({ showLoginModal: true })
|
||
},
|
||
|
||
// 关闭登录弹窗
|
||
closeLoginModal() {
|
||
if (this.data.isLoggingIn) return
|
||
this.setData({ showLoginModal: false })
|
||
},
|
||
|
||
// 微信登录
|
||
async handleWechatLogin() {
|
||
this.setData({ isLoggingIn: true })
|
||
|
||
try {
|
||
const result = await app.login()
|
||
if (result) {
|
||
this.initUserStatus()
|
||
this.setData({ showLoginModal: false })
|
||
wx.showToast({ title: '登录成功', icon: 'success' })
|
||
} else {
|
||
wx.showToast({ title: '登录失败,请重试', icon: 'none' })
|
||
}
|
||
} catch (e) {
|
||
console.error('微信登录错误:', e)
|
||
wx.showToast({ title: '登录失败,请重试', icon: 'none' })
|
||
} finally {
|
||
this.setData({ isLoggingIn: false })
|
||
}
|
||
},
|
||
|
||
// 手机号登录(需要用户授权)
|
||
async handlePhoneLogin(e) {
|
||
// 检查是否有授权code
|
||
if (!e.detail.code) {
|
||
// 用户拒绝授权或获取失败,尝试使用微信登录
|
||
console.log('手机号授权失败,尝试微信登录')
|
||
return this.handleWechatLogin()
|
||
}
|
||
|
||
this.setData({ isLoggingIn: true })
|
||
|
||
try {
|
||
const result = await app.loginWithPhone(e.detail.code)
|
||
if (result) {
|
||
this.initUserStatus()
|
||
this.setData({ showLoginModal: false })
|
||
wx.showToast({ title: '登录成功', icon: 'success' })
|
||
} else {
|
||
wx.showToast({ title: '登录失败,请重试', icon: 'none' })
|
||
}
|
||
} catch (e) {
|
||
console.error('手机号登录错误:', e)
|
||
wx.showToast({ title: '登录失败,请重试', icon: 'none' })
|
||
} finally {
|
||
this.setData({ isLoggingIn: false })
|
||
}
|
||
},
|
||
|
||
// 点击菜单
|
||
handleMenuTap(e) {
|
||
const id = e.currentTarget.dataset.id
|
||
|
||
if (!this.data.isLoggedIn && id !== 'about') {
|
||
this.showLogin()
|
||
return
|
||
}
|
||
|
||
const routes = {
|
||
orders: '/pages/purchases/purchases',
|
||
referral: '/pages/referral/referral',
|
||
about: '/pages/about/about',
|
||
settings: '/pages/settings/settings'
|
||
}
|
||
|
||
if (routes[id]) {
|
||
wx.navigateTo({ url: routes[id] })
|
||
}
|
||
},
|
||
|
||
// 跳转到阅读页
|
||
goToRead(e) {
|
||
const id = e.currentTarget.dataset.id
|
||
wx.navigateTo({ url: `/pages/read/read?id=${id}` })
|
||
},
|
||
|
||
// 跳转到目录
|
||
goToChapters() {
|
||
wx.switchTab({ url: '/pages/chapters/chapters' })
|
||
},
|
||
|
||
// 跳转到关于页
|
||
goToAbout() {
|
||
wx.navigateTo({ url: '/pages/about/about' })
|
||
},
|
||
|
||
// 跳转到匹配
|
||
goToMatch() {
|
||
wx.switchTab({ url: '/pages/match/match' })
|
||
},
|
||
|
||
// 跳转到推广中心
|
||
goToReferral() {
|
||
if (!this.data.isLoggedIn) {
|
||
this.showLogin()
|
||
return
|
||
}
|
||
wx.navigateTo({ url: '/pages/referral/referral' })
|
||
},
|
||
|
||
// 跳转到找伙伴页面
|
||
goToMatch() {
|
||
wx.switchTab({ url: '/pages/match/match' })
|
||
},
|
||
|
||
// 退出登录
|
||
handleLogout() {
|
||
wx.showModal({
|
||
title: '退出登录',
|
||
content: '确定要退出登录吗?',
|
||
success: (res) => {
|
||
if (res.confirm) {
|
||
app.logout()
|
||
this.initUserStatus()
|
||
wx.showToast({ title: '已退出登录', icon: 'success' })
|
||
}
|
||
}
|
||
})
|
||
},
|
||
|
||
// 阻止冒泡
|
||
stopPropagation() {}
|
||
})
|