wxapp/pages/user/index.vue
2024-08-03 19:57:17 +08:00

807 lines
25 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<view v-if="!isLoading" class="container">
<!-- 页面头部 -->
<view class="main-header">
<!-- 用户信息 -->
<view class="user-info">
<!--头像-->
<view class="user-avatar" @click="onUserInfo">
<image class="image" :src="userInfo.avatar ? userInfo.avatar : '/static/default-avatar.png'"></image>
</view>
<view class="user-content" @click="onUserInfo">
<!-- 会员昵称 -->
<view class="nick-name">{{ userInfo.name ? userInfo.name : '未登录'}}</view>
<view class="login-tips" v-show="!isLogin">(点击头像登录)</view>
<!-- 会员等级 -->
<!-- <view v-if="userInfo.gradeId > 0 && gradeInfo" class="user-grade">
<view class="user-grade_icon">
<image class="image" src=""></image>
</view>
<view class="user-grade_name">
<text>{{ gradeInfo.name }}</text>
</view>
</view> -->
<!-- 会员无等级时显示手机号 -->
<!-- <view v-else class="mobile">{{ userInfo.mobile }}</view> -->
<!-- <view class="active-time" v-if="gradeEndTime">{{ gradeEndTime }}</view> -->
<view class="active-time" v-show="userInfo.groupInfo">{{userInfo.groupInfo? userInfo.groupInfo.name:"会员"}}卡</view>
</view>
<view class="pay-qr" @click="toMemberCode(userInfo.id ? userInfo.id : 0)">
<view class="qrcode iconfont icon-qr-extract"></view>
</view>
<!-- <view class="amount-info" @click="toMemberWallet(userInfo.id ? userInfo.id : 0)">
<view class="amount-tip">可用余额(元)</view>
<view class="amount-num" v-if="isLogin">{{ userInfo.balance.toFixed(2) }}</view>
<view class="amount-num" v-if="!isLogin">0.00</view>
<view class="point-amount" @click="onTargetPoints">积分 {{ userInfo.point ? userInfo.point : 0 }}</view>
</view> -->
</view>
<view class="user-no">
<view class="no" v-if="userInfo.userNo">会员卡号:{{ userInfo.userNo ? userInfo.userNo : '-'}}</view>
<!-- <view class="recharge" @click="toRecharge(userInfo.id ? userInfo.id : 0)">储值有礼</view> -->
</view>
</view>
<!--会员升级 start-->
<!-- <view class="member-update" v-if="memberGrade.length > 0">
<view class="update-title">
<text>会员升级</text>
</view>
<scroll-view scroll-x>
<view class="recharge">
<view class="recharge-item" :class="current == index ? 'recharge-item-active': ''" v-for="(item, index) in memberGrade" :key="index" :style="{marginLeft: !index ? '30rpx': ''}" @click="onShowPopup(index)">
<view class="recharge-tag">
<text class="recharge-tag-text" v-if="parseInt(item.validDay) > 0">{{ item.validDay }}天有效期</text>
<text class="recharge-tag-text" v-else>永久有效期</text>
</view>
<text class="recharge-item-duration">{{ item.name }}</text>
<view class="recharge-item-price">
<text class="rmb">¥</text>
<text class="recharge-item-price-text">{{ item.catchValue }}</text>
</view>
<text class="recharge-item-des" v-if="item.discount > 0">买单{{ item.discount }}折</text>
<text class="recharge-item-des" v-if="item.speedPoint > 0">积分翻{{ item.speedPoint }}倍</text>
</view>
</view>
</scroll-view>
</view> -->
<!-- 弹窗 -->
<!-- <Popup v-if="!isLoading" v-model="showPopup" @onPaySuccess="getPageData" :memberGrade="curGrade"/> -->
<!--会员升级 end-->
<!-- 我的资产 -->
<view class="my-asset">
<view class="asset-left flex-box dis-flex flex-x-center">
<!-- <view class="asset-left-item" @click="onTargetMyCoupon('C')"> -->
<view class="asset-left-item">
<view class="item-value dis-flex flex-x-center">
<text>{{ isLogin && userInfo.balance ? userInfo.balance.toFixed(2) : '0' }}</text>
</view>
<view class="item-name dis-flex flex-x-center">
<text>账户余额(元)</text>
</view>
</view>
<view class="asset-left-item">
<!-- <view class="asset-left-item" @click="onTargetMyCoupon('P')"> -->
<view class="item-value dis-flex flex-x-center">
<text>{{ isLogin && assets.totalPrePayment ? assets.totalPrePayment.toFixed(2) : '0' }}</text>
</view>
<view class="item-name dis-flex flex-x-center">
<text>预付金额(元)</text>
</view>
</view>
<view class="asset-left-item">
<!-- <view class="asset-left-item" @click="onTargetMyCoupon('T')"> -->
<view class="item-value dis-flex flex-x-center">
<text>{{ isLogin && userInfo.balance ? ((userInfo.balance*100 - assets.totalPrePayment*100)/100).toFixed(2) : '0' }}</text>
</view>
<view class="item-name dis-flex flex-x-center">
<text>可用金额(元)</text>
</view>
</view>
</view>
</view>
<!-- 订单操作 -->
<view class="order-navbar">
<view class="order-navbar-item" v-for="(item, index) in orderNavbar" :key="index" @click="onTargetOrder(item)">
<view class="item-icon">
<text class="iconfont" :class="[`icon-${item.icon}`]"></text>
</view>
<view class="item-name">{{ item.name }}</view>
<text class="order-badge" v-if="item.count && item.count > 0">{{ item.count }}</text>
</view>
</view>
<!-- 我的服务 -->
<!-- <view class="my-service">
<view class="service-title">我的服务</view>
<view class="service-content clearfix">
<block v-for="(item, index) in service" :key="index">
<view v-if="item.type == 'link'" class="service-item" @click="handleService(item)">
<view class="item-icon">
<text class="iconfont" :class="[`icon-${item.icon}`]"></text>
</view>
<view class="item-name">{{ item.name }}</view>
</view>
<view v-if="item.type == 'button' && $platform == 'MP-WEIXIN'" class="service-item">
<button class="btn-normal" :open-type="item.openType">
<view class="item-icon">
<text class="iconfont" :class="[`icon-${item.icon}`]"></text>
</view>
<view class="item-name">{{ item.name }}</view>
</button>
</view>
</block>
<block>
<view v-if="isMerchant == true" class="service-item" @click="handleService({'url': 'pages/merchant/index'})">
<view class="item-icon">
<text class="iconfont icon-dianpu"></text>
</view>
<view class="item-name">商户管理</view>
</view>
</block>
</view>
</view> -->
<view class="copyright"><view>技术支持成都关键元素科技有限公司</view></view>
</view>
</template>
<script>
import SettingKeyEnum from '@/common/enum/setting/Key'
import SettingModel from '@/common/model/Setting'
import * as UserApi from '@/api/user'
import * as OrderApi from '@/api/order'
import * as MessageApi from '@/api/message'
import { checkLogin, showMessage } from '@/utils/app'
import Popup from './components/Popup'
// 订单操作
const orderNavbar = [
{ id: 'allll', name: '全部订单', icon: 'qpdingdan' },
{ id: 'paid', name: '已支付', icon: 'daishouhuo', count: 0 },
{ id: 'prepay', name: '预支付', icon: 'daifukuan', count: 0 },
{ id: 'refund', name: '已退款', icon: 'fuwu', count: 0 }
]
/**
* 我的服务
* id: 标识; name: 标题名称; icon: 图标; type 类型(link和button); url: 跳转的链接
*/
const service = [
{ id: 'myCoupon', name: '卡券兑换', icon: 'youhuiquan', type: 'link', url: 'pages/coupon/receive' },
{ id: 'coupon', name: '转赠记录', icon: 'lingquan', type: 'link', url: 'pages/give/index' },
{ id: 'points', name: '我的积分', icon: 'jifen', type: 'link', url: 'pages/points/detail' },
{ id: 'help', name: '我的帮助', icon: 'bangzhu', type: 'link', url: 'pages/help/index' },
{ id: 'contact', name: '在线客服', icon: 'kefu', type: 'button', openType: 'contact' },
{ id: 'address', name: '收货地址', icon: 'shouhuodizhi', type: 'link', url: 'pages/address/index' },
{ id: 'refund', name: '售后服务', icon: 'shouhou', type: 'link', url: 'pages/refund/index' },
{ id: 'setting', name: '个人信息', icon: 'shezhi1', type: 'link', url: 'pages/user/setting' },
]
export default {
components: {
Popup
},
data() {
return {
// 枚举类
SettingKeyEnum,
// 当前运行的终端 (此处并不冗余,因为微信小程序端view层无法直接读取$platform)
$platform: this.$platform,
// 正在加载
isLoading: true,
// 是否已登录
isLogin: false,
// 系统设置
setting: {},
// 当前用户信息
userInfo: { id: 0, name: '', avatar: '', gradeId: 0, mobile: '', balance: 0 },
gradeInfo: {},
isMerchant: false,
gradeEndTime: '',
// 账户资产
assets: { prestore: '0', timer: '0', coupon: '0' },
// 我的服务
service,
// 订单操作
orderNavbar,
// 当前用户待处理的订单数量
todoCounts: { payment: 0 },
current: 0,
// 显示、隐藏弹窗
showPopup: false,
memberGrade: [],
curGrade: {}
}
},
/**
* 生命周期函数--监听页面显示
*/
onShow(options) {
// 获取页面数据
this.getPageData()
// 判断是否已登录
this.isLogin = checkLogin()
// 消息显示
//showMessage();
},
methods: {
// 获取页面数据
getPageData(callback) {
const app = this
app.isLoading = true
Promise.all([app.getSetting(), app.getUserInfo(), app.getTodoCounts()])
.then(result => {
app.isLoading = false
// 初始化我的服务数据
app.initService()
// 初始化订单操作数据
app.initOrderTabbar()
// 执行回调函数
callback && callback()
})
.catch(err => {
console.log('catch', err)
})
},
// 初始化我的服务数据
initService() {
const app = this
const newService = []
service.forEach(item => {
if (item.id === 'points') {
item.name = '我的积分'
}
newService.push(item)
})
app.service = newService
},
// 初始化订单操作数据
initOrderTabbar() {
const app = this
const newOrderNavbar = []
orderNavbar.forEach(item => {
if (item.hasOwnProperty('count')) {
item.count = app.isLogin ? app.todoCounts[item.id] : 0
}
newOrderNavbar.push(item)
})
app.orderNavbar = newOrderNavbar
},
// 获取设置
getSetting() {
const app = this
app.setting = {}
},
// 获取当前用户信息
getUserInfo() {
const app = this
app.showPopup = false;
return new Promise((resolve, reject) => {
UserApi.info()
.then(result => {
if (result.data.userInfo) {
app.userInfo = result.data.userInfo
app.getUserAssets(app.userInfo.id)
app.isLogin = true
} else {
app.isLogin = false
app.userInfo = { id: 0, name: '', avatar: '', gradeId: 0, mobile: '', balance: 0 }
}
app.gradeInfo = result.data.gradeInfo
app.memberGrade = result.data.memberGrade
app.gradeEndTime = result.data.gradeEndTime
app.isMerchant = result.data.isMerchant
resolve(app.userInfo)
resolve(app.gradeInfo)
resolve(isMerchant)
})
.catch(err => {
if (err.result && err.result.status == 1001) {
app.isLogin = false
resolve(null)
} else {
reject(err)
}
})
})
},
// 获取账户资产
getUserAssets(id) {
const app = this
return new Promise((resolve, reject) => {
UserApi.assets(id)
.then(result => {
app.assets = result.data.asset
resolve(app.assets)
})
.catch(err => {
if (err.result && err.result.status == 1001) {
app.isLogin = false
resolve(null)
} else {
reject(err)
}
})
})
},
// 获取当前用户待处理的事项数量
getTodoCounts() {
const app = this
return new Promise((resolve, reject) => {
!app.isLogin ? resolve(null) : OrderApi.todoCounts()
.then(result => {
app.todoCounts = result.data
resolve(app.todoCounts)
})
})
},
// 会员等级
onShowPopup(index) {
this.showPopup = !this.showPopup
this.current = index
this.curGrade = this.memberGrade[index]
},
// 跳转到会员码
toMemberCode(userId) {
!this.isLogin && this.$navTo('pages/login/index')
this.$navTo('pages/user/code', { userId: userId})
},
// 跳转我的余额
toMemberWallet(userId) {
!this.isLogin && this.$navTo('pages/login/index')
this.$navTo('pages/wallet/index', { userId: userId})
},
// 跳转充值
toRecharge(userId) {
!this.isLogin && this.$navTo('pages/login/index')
this.$navTo('pages/wallet/recharge/index', { userId: userId})
},
// 跳转到订单页
onTargetOrder(item) {
!this.isLogin && this.$navTo('pages/login/index')
this.$navTo('pages/order/index', { dataType: item.id })
},
// 跳转到我的积分页面
onTargetPoints() {
!this.isLogin && this.$navTo('pages/login/index')
this.$navTo('pages/points/detail')
},
// 跳转到我的卡券列表页
onTargetMyCoupon(type) {
const app = this
if (app.isLogin) {
// #ifdef MP-WEIXIN
MessageApi.getSubTemplate({keys: "couponExpire,couponArrival"}).then(result => {
const templateIds = result.data
wx.requestSubscribeMessage({tmplIds: templateIds,
success(res) {
console.log("调用成功!")
}, fail(res) {
console.log("调用失败:", res)
}, complete() {
app.$navTo('pages/my-coupon/index?type='+type)
}})
})
// #endif
// #ifndef MP-WEIXIN
app.$navTo('pages/my-coupon/index?type='+type)
// #endif
} else {
app.$navTo('pages/login/index')
}
},
// 跳转会员设置页面
onUserInfo() {
if (!this.isLogin) {
this.$navTo('pages/login/index')
} else {
this.$navTo('pages/user/setting')
}
},
// 跳转到服务页面
handleService({ url }) {
this.$navTo(url)
}
},
/**
* 下拉刷新
*/
onPullDownRefresh() {
// 获取首页数据
this.getPageData(() => {
uni.stopPullDownRefresh()
})
}
}
</script>
<style lang="scss" scoped>
.container{
height: 100vh;
}
// 页面头部
.main-header {
background: url('~@/static/background/user-header.png') no-repeat;
height: 380rpx;
background-size: cover;
overflow: hidden;
display: block;
align-items: center;
margin: 10rpx 25rpx 10rpx 25rpx;
border-radius: 10rpx;
.user-info {
display: block;
height: 200rpx;
margin: 20rpx;
margin-left: 20rpx;
.user-avatar {
padding-top: 10rpx;
width: 50rpx;
margin-top: 10rpx;
float: left;
.image {
display: block;
width: 100rpx;
height: 100rpx;
border-radius: 999rpx;
}
}
.user-content {
display: block;
justify-content: center;
margin-top: 20rpx;
margin-left: 60rpx;
float: left;
color: #ffffff;
max-width: 300rpx;
.nick-name {
font-size: 32rpx;
font-weight: bold;
max-width: 270rpx;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.mobile {
margin-top: 15rpx;
font-size: 26rpx;
}
.user-grade {
display: block;
align-items: center;
background: #3c3c3c;
margin-top: 8rpx;
border-radius: 10rpx;
padding: 5rpx 12rpx;
width: 80%;
min-width: 160rpx;
height: 40rpx;
.user-grade_icon .image {
display: block;
width: 32rpx;
height: 32rpx;
float: left;
}
.user-grade_name {
margin-left: 5rpx;
font-size: 24rpx;
color: #EEE0C3;
float: left;
}
}
.active-time {
margin-top: 3rpx;
}
.login-tips {
margin-top: 9rpx;
font-size: 25rpx;
}
}
.amount-info {
margin-top: 80rpx;
margin-left: 70rpx;
color: #fff;
display: block;
float: left;
max-width: 180rpx;
.amount-tip {
font-size: 24rpx;
}
.amount-num {
margin-top: 10rpx;
font-weight: bold;
font-size: 48rpx;
}
.point-amount {
display: block;
margin-top: 2px;
width: 100px;
}
}
.pay-qr {
color:#ffffff;
margin-top: 10rpx;
margin-left: 50rpx;
text-align: center;
width: 50rpx;
float: right;
.qrcode {
display: block;
font-size: 40rpx;
}
}
}
.user-no {
display: block;
margin: 100rpx 0rpx 0rpx 30rpx;
color: #ffffff;
.no {
font-size: 30rpx;
float: left;
}
.recharge {
float: right;
margin-right: 20rpx;
}
}
}
// 我的资产
.my-asset {
display: flex;
margin: 20rpx auto 10rpx auto;
padding: 20rpx 0;
width: 94%;
box-shadow: 0 1rpx 5rpx 0px rgba(0, 0, 0, 0.05);
font-size: 30rpx;
border-radius: 5rpx;
background: #fff;
border: 2rpx #f5f5f5 solid;
.asset-right {
width: 200rpx;
border-left: 1rpx solid #eee;
}
.asset-left-item {
text-align: center;
color: #666;
//padding: 0 72rpx;
width: 33%;
.item-value {
font-size: 35rpx;
color: #f03c3c;
font-weight: bold;
}
.item-name {
font-size: 25rpx;
margin-top: 6rpx;
}
}
}
// 订单操作
.order-navbar {
display: flex;
margin: 10rpx auto 10rpx auto;
padding: 20rpx 0;
width: 94%;
box-shadow: 0 1rpx 5rpx 0px rgba(0, 0, 0, 0.05);
font-size: 30rpx;
border-radius: 5rpx;
background: #fff;
border: 2rpx #f5f5f5 solid;
&-item {
position: relative;
width: 33%;
.item-icon {
text-align: center;
margin: 0 auto;
padding: 10rpx 0;
color: #545454;
font-size: 40rpx;
}
.item-name {
font-size: 24rpx;
color: #545454;
text-align: center;
margin-right: 10rpx;
}
.order-badge {
position: absolute;
top: 0;
right: 58rpx;
font-size: 20rpx;
background: #fa5151;
text-align: center;
line-height: 30rpx;
color: #fff;
border-radius: 50%;
min-width: 36rpx;
padding: 6rpx 13rpx 6rpx 13rpx;
}
}
}
// 我的服务
.my-service {
margin: 0rpx auto 20rpx auto;
border: 2rpx #f5f5f5 solid;
background: #FFF;
padding: 10rpx 0rpx;
width: 94%;
box-shadow: 0 1rpx 5rpx 0px rgba(0, 0, 0, 0.05);
border-radius: 5rpx;
display: block;
.service-title {
padding-left: 20rpx;
margin-bottom: 30rpx;
font-size: 28rpx;
}
.service-content {
.service-item {
width: 25%;
float: left;
margin-bottom: 25rpx;
.item-icon {
text-align: center;
margin: 0 auto;
padding: 10rpx 0;
color: #ff3800;
font-size: 40rpx;
}
.item-name {
font-size: 24rpx;
color: #545454;
text-align: center;
margin-right: 10rpx;
}
}
}
}
// 推荐信息
.my-recommend {
height: 20rpx;
}
.copyright{
position: absolute;
bottom: 20rpx;
left: 50%;
transform: translateX(-50%);
width: 100%;
text-align: center;
color: #c0c0c0;
font-size: 24rpx;
}
// 会员升级
.member-update {
margin: 22rpx auto 22rpx auto;
padding: 20rpx 0;
border-radius: 5rpx;
box-shadow: 0 1rpx 5rpx 0px rgba(0, 0, 0, 0.05);
background: #fff;
width: 94%;
text-align: center;
.update-title {
padding-left: 20rpx;
margin-bottom: 30rpx;
font-size: 28rpx;
text-align: left;
}
.recharge {
position: relative;
margin-bottom: 35rpx;
display: flex;
flex-direction: row;
align-items: center;
&-tag {
position: absolute;
top: -2rpx;
left: -2rpx;
width: 170rpx;
height: 36rpx;
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
background-image: url('~@/static/user/tag.png');
background-size: 100%;
&-text {
font-size: 20rpx;
color: #FFFFFF;
text-align: center;
}
}
&-item {
position: relative;
padding: 40rpx 0;
margin-left: 15rpx;
width: 29.33%;
height: 270rpx;
flex-shrink: 0;
display: flex;
flex-direction: column;
align-items: center;
border: solid 1rpx #CBCCCE;
border-radius: 12rpx;
&-active {
border: solid 2rpx #EDD2A9;
background-color: #FBF1E5;
}
&-duration {
margin-bottom: 30rpx;
font-size: 26rpx;
color: #1C1C1C;
}
&-price {
margin-bottom: 20rpx;
display: flex;
flex-direction: row;
align-items: baseline;
&-text {
font-size: 48rpx;
color: #E3BE83;
}
}
&-des {
font-size: 22rpx;
color: #A5A3A2;
}
}
}
}
</style>