Compare commits

..

1 Commits
main ... ycysp

Author SHA1 Message Date
xiaogang 1a576ed7a3 新增邮储音视频的分支 2025-07-09 17:48:43 +08:00
67 changed files with 2114 additions and 2084 deletions

View File

@ -1,4 +1,4 @@
NODE_ENV = 'development'
# 邮储奶茶活动
VITE_BASE_URL = 'http://milk.test.api.cdlsxd.cn'
# 邮储音视频
VITE_BASE_URL = 'https://gateway.dev.cdlsxd.cn/ycav'

View File

@ -1,7 +1,6 @@
NODE_ENV = 'production'
# 邮储奶茶活动(正式环境)
VITE_BASE_URL = 'https://milk.api.cdlsxd.cn'
# 邮储音视频活动(生产环境)
VITE_BASE_URL = 'https://mv.h5.cdlsxd.cn'
VITE_YCNC_MERCH_CODE = '100510102303326'

View File

@ -1,5 +1,5 @@
NODE_ENV = 'test'
# 邮储奶茶活动(测试环境)
VITE_BASE_URL = 'http://milk.test.api.cdlsxd.cn'
# 邮储音视频活动(测试环境)
VITE_BASE_URL = 'https://gateway.dev.cdlsxd.cn'
VITE_YCNC_MERCH_CODE = '100310100018908'

View File

@ -4,7 +4,7 @@ import path from 'path'
const config = {
pages:'./src/pages',
pagesInclude:['ycnc']
pagesInclude:['ycysp/home.vue','ycysp/pay.vue','ycysp/order.vue','ycysp/orderDetail.vue',]
}
const { pages, pagesInclude } = config

View File

@ -1,31 +0,0 @@
import http from "../utils/http";
export const login = (params) => http({
url:'/v1/xy/login',
method:'POST',
...params
})
export const getProduceList = (params) => http({
url:'/v1/auth/activity',
method:'POST',
...params
})
export const goBuy = (params) => http({
url:'/v1/auth/order/buy',
method:'POST',
...params
})
export const getTicketList = (params) => http({
url:'/v1/auth/order/list',
method:'POST',
...params
})
export const queryOrder = (params) => http({
url:'/v1/auth/order/query',
...params
})

View File

@ -1,60 +0,0 @@
import http from "../utils/http";
export const login = (params) => http({
url:'/api/v1/UnionLogin',
method:'POST',
noToken:true,
...params
})
export const getProductList = (params) => http({
url:'/api/v1/VoucherList',
method:'POST',
...params
})
export const queryDetail = (params) => http({
url:'/api/v1/product/detail',
method:'POST',
...params
})
export const getOrderList = (params) => http({
url:'/api/v1/auth/order/list',
method:'POST',
...params
})
export const goPay = (params) => http({
url:'/api/v1/auth/order/create',
method:'POST',
...params
})
export const queryOrderDetail = (params) => http({
url:'/api/v1/auth/order/query',
method:'POST',
...params
})
export const deleteOrder = (params) => http({
url:'/api/v1/auth/order/delete',
method:'POST',
...params
})
export const refundOrder = (params) => http({
url:'/api/v1/auth/order/refund',
method:'POST',
...params
})
//查询订单状态,主要针对短连接
export const queryOrderState = (params) => http({
url:'/api/v1/auth/order/state',
method:'POST',
...params
})

View File

@ -1,25 +1,16 @@
{
"pages": [
{
"path": "pages/ycnc/index",
"path": "pages/ycysp/home",
"type": "home",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "奶茶活动",
"navigationBarTitleText": "音视频专区",
"navigationBarBackgroundColor": "#FFF"
}
},
{
"path": "pages/ycnc/detail",
"type": "page",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "商品详情",
"navigationBarBackgroundColor": "#FFF"
}
},
{
"path": "pages/ycnc/order",
"path": "pages/ycysp/order",
"type": "page",
"style": {
"navigationStyle": "custom",
@ -28,13 +19,22 @@
}
},
{
"path": "pages/ycnc/orderDetail",
"path": "pages/ycysp/orderDetail",
"type": "page",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "订单详情",
"navigationBarBackgroundColor": "#FFF"
}
},
{
"path": "pages/ycysp/pay",
"type": "page",
"style": {
"navigationStyle": "custom",
"navigationBarTitleText": "支付确认",
"navigationBarBackgroundColor": "#FFF"
}
}
],
"globalStyle": {

View File

@ -1,87 +0,0 @@
<template>
<view class="product-container">
<view class="product-tips">
支付享立减1元
</view>
<view class="product-price display">
<view class="price">
<text>{{detail.price}}</text>/{{detail.brand}}
</view>
<view class="ori">
原价{{detail.show_price}}
</view>
</view>
<view class="product-type display">
{{detail.name}}
</view>
<view class="product-btn" @click="$emit('buy-event',detail)">
立即购买
</view>
</view>
</template>
<script setup>
const props = defineProps({
detail:{
type:Object,
require:true,
}
})
</script>
<style scoped lang="scss">
.product-container{
width:292rpx;
height:318rpx;
background: url('../../../static/xyyk/bg-product.png') no-repeat;
background-size: 100% 100%;
margin-top:30rpx;
.product-tips{
font-weight: 400;
font-size: 20rpx;
color: #B05619;
box-sizing: border-box;
padding-left:40rpx;
}
.display{
display: flex;
flex-direction: column;
align-items: center;
}
.product-price{
color: #FD613F;
.price{
font-weight: 400;
font-size: 40rpx;
text{
font-weight:bold;
font-size: 60rpx;
}
}
.ori{
font-size: 24rpx;
color: #E18B72;
font-weight:400;
text-decoration-line: line-through;
}
}
.product-type{
font-weight: 400;
font-size: 28rpx;
color: #FFFFFF;
margin-top:40rpx;
}
.product-btn{
width: 218rpx;
height: 60rpx;
background: linear-gradient( 90deg, #FFD888 0%, #FFE39E 100%);
border-radius: 30rpx;
font-weight: 400;
font-size: 30rpx;
color: #A83203;
text-align: center;
line-height: 60rpx;
margin:16rpx auto 0;
}
}
</style>

View File

@ -1,80 +0,0 @@
<template>
<view class="ticket-container">
<view class="ticket-type">{{detail.product_name}}</view>
<view class="ticket-content">
<image src="../../../static/xyyk/icon-ticket.png" mode=""></image>
<view class="ticket-name">{{detail.product_name}}</view>
<view class="ticket-btn" @click="$emit('view-event',detail)">
立即查看
</view>
</view>
</view>
</template>
<script setup>
const props = defineProps({
detail:{
type:Object,
require:true
}
})
</script>
<style scoped lang="scss">
.ticket-container{
width: 694rpx;
height: 240rpx;
background: #FFFFFF;
border-radius: 24rpx 24rpx 24rpx 24rpx;
font-weight: 500;
font-size: 32rpx;
color: #1A1A1A;
box-sizing: border-box;
padding:32rpx;
display: flex;
flex-direction: column;
justify-content: space-between;
margin-top:32rpx;
.ticket-type{
width:100%;
height:32rpx;
font-weight: 500;
font-size: 32rpx;
color: #1A1A1A;
line-height: 32rpx;
}
.ticket-content{
height:92rpx;
display: flex;
align-items: center;
position: relative;
image{
width:92rpx;
height:92rpx;
margin-right: 22rpx;
}
.ticket-name{
font-weight: 500;
font-size: 28rpx;
color: #333333;
width:50%;
overflow: hidden; /* 隐藏溢出内容 */
text-overflow: ellipsis; /* 当文本溢出时显示省略号 */
white-space: nowrap; /* 禁止文本换行 */
}
.ticket-btn{
position: absolute;
right:0;
width: 150rpx;
height: 54rpx;
background: #47A5FF;
border-radius: 27rpx;
font-weight: 500;
font-size: 24rpx;
color: #FFFFFF;
text-align: center;
line-height:54rpx ;
}
}
}
</style>

View File

@ -1,241 +0,0 @@
<template>
<view class="container">
<image class="title" src="../../static/xyyk/bg-title.png" mode="widthFix"></image>
<view class="tips">
支付享立减1元
</view>
<view class="tickets" @click="toTickets">
我的券包
</view>
<view class="bottom-banner">
<view class="banner-title">
兑换商品列表
</view>
<view class="banner-tips">
<view>兴业优酷半价购送周卡</view>
<view><text class="line"></text>支付享立减1元<text class="line"></text></view>
</view>
<scroll-view class="banner-content" scroll-y>
<view class="banner-container">
<view v-for="(item,index) in products" :key="item.id">
<ProductItem :detail="item" @buy-event="handleBuy"/>
</view>
</view>
</scroll-view>
</view>
<uni-popup ref="popupRef" type="center" :is-mask-click="false" :mask-click="false">
<view class="popup-content">
<view class="popup-title">购买{{payResult ? '成功' : '失败'}}</view>
<view class="popup-tips" v-if="payResult">恭喜购买成功请稍后查看结果</view>
<view class="popup-tips" v-else>支付失败了</view>
<view class="btn" @click="closePopup">
确定
</view>
</view>
</uni-popup>
</view>
</template>
<script setup>
import { onMounted, ref } from 'vue';
import ProductItem from './components/product-item';
import {getQueryString} from '../../utils/utils';
import {login,getProduceList,goBuy,queryOrder} from '../../api/xyyk';
const popupRef = ref(null);
const products = ref([]);
const payResult = ref(false);
const orderNo = "SN202406131136518803"
const code = 'MDQ4MDM2ZWYwNzUxYmMwYmM3YTk4ZjdmNjEzZWNmOGNjZDk0OGQ5ZjM0NjY1NjY1NDMyYjY2MzVmNjkwZTE5MjU1MTdhOTc1MDkzOGVhMWJhNDE5YWVlMjQ5NGMwZjFmMTg4NzgyZDBiN2UyOGYwZDFjNzIzOGIzOTQ1ZGRiMWU0M2FiNjhjOTRlZjEyODE1MmM5NjIyMTQzZTBhYmQzZGFiOTRjM2ZhNTNlYTVlYTRiOGY0ZTIxZjFkN2I4ZjFlYzBmODQ1NzlmYTY2MDJmMmViZDIxMzc4ZDMwMmM1OGExZTZiMTQ1N2Y0MjY5NWJlNDRhZWU5YzE5OTViMmQ1MjAxNWY5NWM4ZTQ0OGI2NGEyZDI5MTkyNWNkZjA0YWIwZjFlZGE3MGJmZjk3YzViZTYwNTYzZDgxMjk2YmE5MWE4ZDQ0ZDZiZGY3MmJlMzdmN2NmMjI3YjU5NmQwNmMwYWM4MmZkYjc5MTk0MmRlNzE0NWU3ZmYxMGNkNjhmN2RhODMwMzU3NGUzMTM0ZGU3OTc1NWIwNDM0ODBmMjdlMzliZDEwZjViNWI4OTE1N2RhN2EzNmU5ZGNmZjQ3NzhiYjBhOGFmOTNlNTM2M2NiNmIzY2UwODJjZDU5NzFjMTVmZGI2ZA=='
onMounted(()=>{
//code,
const code = getQueryString('code');
const orderNo = getQueryString('order_no');
// if(code){
// getProductConfig()
// }else if(orderNo){
// queryOrderStatus()
// }
getProductConfig()
})
const getProductConfig = async () => {
const params = {data:code}
const resData = await login({params})
uni.setStorageSync('token',resData.token);
const productConfig = await getProduceList();
const {name,description,product} = productConfig;
products.value = product;
}
const queryOrderStatus = () => {
const params = {
order_no:orderNo
}
queryOrder({params}).then(res => {
const {state} = res
payResult.value = state == '2' ? true : false;
popupRef.value.open()
})
}
const handleBuy = (item) => {
const {id} = item;
const params = {product_id:id}
goBuy({params}).then(res => {
const {payToken,payOrderId,mchtOrderId} = res;
window.location.href = payToken;
})
}
const toTickets = () => {
uni.navigateTo({url:'/pages/xyyk/tickets'})
}
const closePopup = () => {
popupRef.value.close()
}
</script>
<style scoped lang="scss">
.container{
position: relative;
display: flex;
flex-direction: column;
align-items: center;
width:100vw;
height:100vh;
background: url('../../static/xyyk/bg.png') no-repeat;
background-size: 100% 100%;
.title{
width:640rpx;
margin: 32rpx auto 0;
}
.tips{
width: 215rpx;
height: 42rpx;
background: linear-gradient( 90deg, #E5F5FF 0%, #FFFEFF 100%);
border-radius: 21rpx;
margin-top:16rpx;
font-weight: 400;
font-size: 24rpx;
color: #00A5EB;
text-align: center;
line-height: 42rpx;
}
.tickets{
position: fixed;
right:0;
top:150rpx;
width: 102rpx;
height: 32rpx;
background: linear-gradient( 90deg, #E7F6FF 0%, #FEFEFF 100%);
border-radius: 52rpx 0px 0px 52rpx;
font-weight: 400;
font-size: 20rpx;
color: #00A5EB;
line-height: 28rpx;
text-align: center;
line-height: 28rpx;
}
.bottom-banner{
position: absolute;
bottom:10rpx;
width:730rpx;
height:620rpx;
background: url('../../static/xyyk/bg-banner.png') no-repeat;
background-size: 100% 100%;
display: flex;
flex-direction: column;
.banner-title{
width:350rpx;
height:78rpx;
text-align: center;
line-height: 78rpx;
font-weight: 400;
font-size: 32rpx;
color: #FFFFFF;
margin:0 auto;
padding-left:50rpx;
}
.banner-tips{
display: flex;
flex-direction: column;
align-items: center;
margin-top:30rpx;
.line{
display: block;
width: 40rpx;
height: 0;
border: 1rpx solid #3F85BC;
&:first-child{
margin-right:14rpx;
}
&:last-child{
margin-left:14rpx;
}
}
> :first-child{
font-weight: 600;
font-size: 32rpx;
color: #135BC3;
}
> :last-child{
display: flex;
align-items: center;
font-weight: 400;
font-size: 28rpx;
color: #3F85BC;
}
}
.banner-content{
box-sizing: border-box;
padding:0 60rpx 40rpx;
flex:1;
overflow: auto;
.banner-container{
display: flex;
justify-content: space-between;
flex-wrap: wrap;
}
}
}
.popup-content{
width:578rpx;
height:700rpx;
background: url("../../static/xyyk/bg-popup.png") no-repeat;
background-size:100% 700rpx;
display: flex;
flex-direction: column;
align-items: center;
box-sizing: border-box;
padding-top:290rpx;
.popup-title{
font-weight:700;
color:#111B1F;
font-size:60rpx;
margin-top:24rpx;
}
.popup-tips{
font-weight: 400;
font-size: 28rpx;
color: #00A5EB;
margin-top:34rpx;
opacity: .6;
}
.btn{
width: 236rpx;
height: 72rpx;
background: linear-gradient( 180deg, #57C3FB 7%, #2D98F7 100%);
border-radius: 36rpx;
text-align: center;
line-height: 72rpx;
font-weight: 400;
font-size: 32rpx;
color: #FFFFFF;
margin-top:100rpx;
}
}
}
</style>

View File

@ -1,66 +0,0 @@
<template>
<scroll-view scroll-y="true" class="container" @scrolltolower="getNextPage">
<view class="wrapper">
<view v-for="(item,index) in list" :key="item.id">
<TicketItem :detail="item" @view-event="handleView"/>
</view>
</view>
</scroll-view>
</template>
<script setup>
import { onMounted , ref, unref } from 'vue';
import TicketItem from './components/ticket-item';
import {getTicketList} from '../../api/xyyk';
const page = ref(1);
const pageSize = 10;
const total = ref(0);
const list = ref([]);
onMounted(()=> {
const params = {
page:unref(page),
pageSize:unref(pageSize)
}
getTicketList({params}).then(res => {
const {count,data} = res
total.value = count;
list.value = data;
})
})
const isLastPage = () => unref(page) === Math.ceil(unref(total) / pageSize)
const getNextPage = () => {
if(isLastPage()){
console.log('已经是最后一页');
return false;
};
const params = {
page:unref(page) + 1,
pageSize:unref(pageSize),
}
}
const handleView = (item) => {
const {voucher_link} = item
console.log(item)
}
</script>
<style scoped lang="scss">
.container{
width:100vw;
height:100vh;
background-color: #f7f7f7;
overflow-y: auto;
.wrapper{
width:100%;
position: relative;
display: flex;
flex-direction: column;
align-items: center;
padding-bottom:32rpx;
}
}
</style>

View File

@ -1,59 +0,0 @@
<template>
<scroll-view scroll-y class="pro-container box-border" :style="{backgroundImage: 'url('+config[brandName]?.src+')',height:`${config[brandName].height}rpx`}">
<view class="pro-wrapper flex flex-wrap">
<view v-for="(item,index) in products" :key="index" class="product-container">
<ProductItem :detail="item" :index="index"/>
</view>
</view>
</scroll-view>
</template>
<script setup>
import ProductItem from './product-item'
import config from '../config'
const props = defineProps({
products:{
type:Array,
required: true,
default:() => ([])
},
brandName:{
type:String,
required:true,
default:''
}
})
</script>
<style lang="scss" scoped>
.pro-container{
width:780rpx;
padding: 136rpx 60rpx 72rpx;
// margin-bottom: -24rpx;
background-repeat: no-repeat;
background-size: contain;
overflow-y: auto;
}
.pro-wrapper{
.product-container{
width: calc((100% - 40rpx) / 3);
margin-right: 20rpx;
margin-bottom: 20rpx;
}
// view:nth-child(3n) > .pro-item-wrapper{
// margin-right:0 !important;
// }
view:nth-child(3n) {
margin-right:0 !important;
}
// view:nth-child(3n+1):nth-last-child(-n+3) > .pro-item-wrapper,
// view:nth-child(3n+1):nth-last-child(-n+3)~view > .pro-item-wrapper{
// margin-bottom: 0 !important;
// }
view:nth-child(3n+1):nth-last-child(-n+3),
view:nth-child(3n+1):nth-last-child(-n+3)~view{
margin-bottom: 0 !important;
}
}
</style>

View File

@ -1,178 +0,0 @@
<template>
<view class="order-item-wrapper flex flex-col box-border" @click="goDetail">
<view class="nav-info flex flex-justify-between" >
<view>订单编号{{ detail.order_no }}</view>
<view :style="{color:stateColor[detail.state] || ''}">{{ stateConfig[detail.state] }}</view>
</view>
<view class="order-info">
<view class="content flex">
<image
:src="detail.main_image"
class="pro-img"
/>
<view class="flex flex-1 flex-col">
<view class="flex flex-justify-between name info-item">
<view class="text-over">{{ detail.product_name }}</view>
<view class="amount"><text></text>{{ detail.price }}<text></text></view>
</view>
<view class="flex flex-justify-between num">
<view>数量</view>
<view>x1</view>
</view>
</view>
</view>
<view class="time">下单时间{{ detail.create_time }}</view>
</view>
<view class="btns flex flex-justify-end">
<view class="btn del" @click="goDel">删除订单</view>
<view class="btn pay" v-if="[1].includes(detail.state)" @click="goPay">立即付款</view>
<view class="btn view" v-if="[3].includes(detail.state) && !!detail.voucher_link" @click="goPwd">查看卡券</view>
<view class="btn pay" @click="goRefund" v-if="[4].includes(detail.state)">申请退款</view>
</view>
</view>
</template>
<script setup>
import { stateConfig, stateColor } from '../config';
const props = defineProps({
detail:{
type:Object,
required: true,
default:() => ({})
}
})
const emits = defineEmits(['pay-event','detail-event','pwd-event','del-event'])
function goPay(e){
if(e.stopPropagation) { //W3C
e.stopPropagation();
}
emits('pay-event', props.detail);
}
function goDetail(e){
emits('detail-event', props.detail);
}
function goPwd(e){
if(e.stopPropagation) { //W3C
e.stopPropagation();
}
emits('pwd-event', props.detail);
}
function goDel(e){
if(e.stopPropagation) { //W3C
e.stopPropagation();
}
emits('del-event', props.detail);
}
function goRefund(e){
if(e.stopPropagation) { //W3C
e.stopPropagation();
}
emits('refund-event', props.detail);
}
</script>
<style lang="scss">
.order-item-wrapper{
width: 702rpx;
// height: 386rpx;
background: #FFFFFF;
border-radius: 24rpx;
padding:24rpx;
margin:24rpx 0 16rpx;
}
.nav-info{
font-size:22rpx;
font-weight:400;
:first-child{
color: #333333;
}
:last-child{
color: #787878;
}
}
.order-info{
margin-top:24rpx;
.content{
margin-bottom:24rpx;
font-weight: 400;
.pro-img{
display: block;
width:90rpx;
height:90rpx;
margin-right:34rpx;
}
}
.time{
font-size: 22rpx;
color: #6C6C6C;
}
.info-item{
font-weight: bold;
margin-bottom:12rpx;
}
.text-over{
max-width:80%;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
}
.amount{
color: #333333;
font-weight: bold;
font-size:24rpx;
text:nth-child(1){
font-size:20rpx;
font-weight: 400;
}
text:nth-child(2){
font-size:22rpx;
}
}
.name{
font-size: 24rpx;
color: #333333;
font-weight: 700;
}
.num{
font-size: 20rpx;
color: #9E9E9E;
font-weight: 400;
}
}
.btns{
border-top: 1rpx solid #F0E1E1;
padding:36rpx 0 12rpx;
margin-top:24rpx;
.btn{
width: 168rpx;
height: 64rpx;
border-radius: 68rpx;
font-weight: 400;
font-size: 28rpx;
display: flex;
align-items: center;
justify-content: center;
margin-left:48rpx;
}
.del{
color: #545454;
border: 2rpx solid #B9C8C7;
}
.pay{
color: #FFFFFF;
background: #EA0000;
}
.view{
background: #FFFFFF;
border-radius: 68rpx;
border: 2rpx solid #EA0000;
color:#EA0000;
}
}
</style>

View File

@ -1,131 +0,0 @@
<template>
<view class="pro-item-wrapper flex flex-col box-border flex-justify-between" @click="toDetail">
<view class="img-container">
<image
:src="detail?.voucherIcon"
mode="scaleToFill"
class="img"
/>
</view>
<view class="bottom-container">
<view class="brand">{{ config[detail.brandFlag].name}}</view>
<view class="name">{{ detail.voucherTitle }}</view>
<view class="flex flex-justify-between flex-items-end">
<view class="price">{{ detail.voucherAmount }}<text></text>
</view>
<view class="ori">{{ detail.voucherOriginalPrice }}</view>
</view>
<view class="btn flex flex-justify-center flex-items-center">
<view class="btn-content flex flex-justify-center flex-items-center" @click="goBuy">点击购买</view>
</view>
</view>
</view>
</template>
<script setup>
import { inject } from 'vue';
import config from '../config'
const { handleBuy, goDetail } = inject('custom-events')
const props = defineProps({
detail:{
type:Object,
required: true,
default:() => ({})
},
index:{
type:Number,
required:true,
default:0,
}
})
const goBuy = (e) => {
if(e.stopPropagation) { //W3C
e.stopPropagation();
}
handleBuy(props.detail)
}
const toDetail = () => {
goDetail(props.detail)
}
</script>
<style lang="scss">
.pro-item-wrapper{
// width:206rpx;
width:100%;
height:370rpx;
background: url('/static/ycnc/bg-product.png') no-repeat;
background-size: 100% 100%;
padding: 6rpx 6rpx 10rpx;
margin-right: 20rpx;
margin-bottom: 20rpx;
}
.bottom-container{
overflow: hidden;
flex:1;
padding: 18rpx 8rpx 0;
display: flex;
flex-direction: column;
// justify-content: space-between;
}
.brand{
font-weight: 400;
font-size: 22rpx;
color: #887F6E;
}
.img-container{
width:100%;
height:150rpx !important;
box-sizing: border-box;
border-radius: 18rpx;
overflow: hidden;
.img{
width:100%;
height: 100%;
}
}
.name{
font-weight: 400;
font-size: 28rpx;
color: #3B2609;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
margin:10rpx 0 6rpx;
}
.price{
font-weight: 400;
font-size: 32rpx;
color: #F71924;
text{
font-size: 18rpx;
}
}
.ori{
font-weight: 300;
font-size: 24rpx;
color: #888365;
text-decoration: line-through;
text{
font-size: 16rpx;
// text-decoration: line-through;
}
}
.btn{
width: 166rpx;
height: 56rpx;
background: linear-gradient( 180deg, #FFF8EA 0%, #FFD9BC 100%);
box-shadow: 0rpx 1 2rpx 0rpx #FAD5A8;
border-radius: 28rpx;
margin-top:20rpx;
.btn-content {
width: 158rpx;
height: 48rpx;
background: linear-gradient( 180deg, #FB3E3A 0%, #F01617 100%);
border-radius: 24rpx;
font-weight: 400;
font-size: 26rpx;
color: #FFFFFF;
}
}
</style>

View File

@ -1,94 +0,0 @@
import MX from '/static/ycnc/bg-mxbc.png'
import BW from '/static/ycnc/bg-bwcj.png'
import NX from '/static/ycnc/bg-nxdc.png'
import CoCo from '/static/ycnc/bg-coco.png'
import CBD from '/static/ycnc/bg-cbd.png'
import GM from '/static/ycnc/bg-gm.png'
import SYSXC from '/static/ycnc/bg-sysxc.png'
// import CYYS from '/static/ycnc/bg-cyys.png'
const config = {
"MX":{
name:'蜜雪冰城',
src:MX,
height:974,
order:1
},
"BW":{
name:'霸王茶姬',
src:BW,
height:974,
order:2
},
"NX":{
name:'奈雪的茶',
src:NX,
height:584,
order:5
},
"CoCo":{
name:'CoCo',
src:CoCo,
height:584,
order:6
},
"CBD":{
name:'茶百道',
src:CBD,
height:584,
order:7
},
"YCGM":{
name:'古茗',
src:GM,
height:584,
order:3,
},
"YCSY":{
name:'书亦烧仙草',
src:SYSXC,
height:584,
order:4
},
// "CYYS":{
// name:'茶颜悦色',
// src:CYYS,
// height:974,
// },
}
export const stateConfig = {
1: '待支付',
2: '已支付',
3: '充值完成',
4: '充值异常',
5: '订单核销',
6: '退款中',
7: '退款完成',
8: '退款失败' ,
9: '订单取消'
}
export const stateColor = {
2: '#FF8400',
7: '#FF0000',
}
export const tabs = [{
tabName:'全部订单',
state:0,
},{
tabName:'待付款',
state:1,
},{
tabName:'处理中',
state:2,
},{
tabName:'已完成',
state:3,
},{
tabName:"已退款",
state:7
}]
export default config

View File

@ -1,216 +0,0 @@
<route lang="json5" type="page">
{
style: {
navigationStyle: 'custom',
navigationBarTitleText: '商品详情',
navigationBarBackgroundColor:'#FFF',
},
}
</route>
<template>
<view class="w-full h-full flex flex-col">
<scroll-view scroll-y class="w-full h-full overflow-y-auto flex-1">
<view class="container">
<image class="banner" :src="detailObj.main_image" @click="testEvent"></image>
<view class="detail flex flex-col flex-justify-between">
<view class="proname">{{ detailObj.name }}</view>
<view class="num">
<text class="price">{{ detailObj.price }}</text>
<text class="ori">{{ detailObj.show_price }}</text>
</view>
</view>
<view class="desc" v-html="detailObj.description">
</view>
<!-- <view class="desc">
<div class="big">购买须知</div>
<div class="small">有效期</div>
<p class="prog">
代金券到账后10天有效逾期视为自动放弃不退不补
</p>
</view> -->
</view>
</scroll-view>
<view class="pay-container">
<view class="pay-btn" @click="toPay">
立即支付
</view>
</view>
</view>
</template>
<script setup>
import { onMounted, reactive, ref, unref } from 'vue';
import { onLoad } from '@dcloudio/uni-app';
import { queryDetail, goPay, login } from '../../api/ycnc';
import usePay from './hooks/usePay';
import useCode from './hooks/useCode';
const id = ref('')
const authCode = ref('');
const detailObj = reactive({})
onLoad((options) => {
const {product_id} = options;
id.value = product_id
})
onMounted(() => {
getDetail()
})
const getDetail = () => {
const params = {
product_id:unref(id)
}
queryDetail({params}).then(res => {
Object.assign(detailObj,res)
})
}
const toPay = async () => {
const token = window.localStorage.getItem('token') || '';
if(!token){
const code = await useCode();
authCode.value = code
const {token} = await login({params:{code:unref(authCode)}});
window.localStorage.setItem('token',token);
}
const params = {
product_id:unref(id)
}
goPay({params}).then(res => {
const {order_no,notify_url,sign,plain_text} = res;
const {payFunc} = usePay();
payFunc({order_no,notify_url,TranAmt:detailObj.price,MerName:detailObj.brand,sign,plain_text})
}).catch(err => {
console.log(err);
})
}
var clickCount = 0
var timer = null
const testEvent = () => {
if (clickCount < 6) {
clickCount++
console.log(`Clicked ${clickCount} times`)
//
if (clickCount === 1) {
timer = setTimeout(() => {
clickCount = 0 //
}, 2000)
}
// 6
if (clickCount >= 6) {
clearTimeout(timer)
clickCount = 0 //
window.location.href = 'https://22233.cn/AAALM4HwLrAfdSCx'
}
}
}
</script>
<style lang='scss' scoped>
.banner{
width: 100%;
height:600rpx;
display: block;
box-sizing: border-box;
}
.detail{
width: 694rpx;
// height: 138rpx;
border-bottom: 2rpx solid rgba(0,0,0,0.04);
margin: 0 auto;
box-sizing: border-box;
padding:25rpx 0;
.proname{
font-weight: bold;
font-size: 30rpx;
color: #333333;
margin-bottom:16rpx;
}
.price{
font-weight: 700;
font-size: 28rpx;
color: #EA0000;
margin-right:18rpx;
}
.ori{
font-weight: 400;
font-size: 28rpx;
color: #A5A5A5;
text-decoration: line-through;
}
}
.desc{
width: 694rpx;
margin:0 auto;
padding-top:44rpx;
padding-bottom:24rpx;
//
&:deep(.big-title) {
font-weight: bold;
font-size: 40rpx;
color: #3D3D3D;
margin-bottom:16rpx;
}
&:deep(.small-area) {
margin-bottom:16rpx;
}
&:deep(.small-title){
font-weight: 400;
font-size: 32rpx;
color: #333333;
margin-bottom:16rpx;
}
&:deep(.small-content){
font-weight: 400;
font-size: 28rpx;
color: #4D4D4D;
line-height: 1.6;
}
//
&:deep(.big){
font-weight: bold;
font-size: 40rpx;
color: #3D3D3D;
margin-bottom:16rpx;
}
&:deep(.small){
font-weight: 400;
font-size: 32rpx;
color: #333333;
margin-bottom:16rpx;
}
&:deep(.prog){
font-weight: 400;
font-size: 28rpx;
color: #4D4D4D;
margin-bottom:16rpx;
line-height: 1.6;
}
}
.pay-container{
width:100%;
height:138rpx;
border-top:1px solid rgba(0,0,0,0.1);
}
.pay-btn{
width: 600rpx;
height: 88rpx;
background: #EA0000;
border-radius: 54rpx 54rpx 54rpx 54rpx;
display: flex;
align-items: center;
justify-content: center;
font-weight: 500;
font-size: 32rpx;
color: #FFFFFF;
margin:25rpx auto;
}
</style>

View File

@ -1,29 +0,0 @@
import md5 from 'js-md5';
export default function useCode(){
return new Promise((resolve,reject) => {
const authCallback = (params) => {
console.log('authCallback-use',params);
const {code} = params
resolve(code)
}
const getNewAuthorization = () => {
let appId = import.meta.env.VITE_YCNC_APPID;
let time = Date.now().toString();
let secret = import.meta.env.VITE_YCNC_SECRET;
let signBefore = appId + time + secret;
let sign = md5(signBefore);//普通md5
let param = {
appId: appId,
sign: sign,
time: time,
tran_code: "157",
fn: "authCallback",
needBind: ""
};
console.log('auth-param157-use',param);
Fw.device.api.getNewAuthorization(param)
};
window.authCallback = authCallback;
getNewAuthorization()
})
}

View File

@ -1,41 +0,0 @@
import dayjs from "dayjs";
import md5 from 'js-md5';
const handleParams = (obj) => Object.entries(obj).reduce((total,curr) => {
if(!total){
total += `${curr[0]}=${curr[1]}`
}else{
total += `|${curr[0]}=${curr[1]}`
}
return total
},'')
export default function usePay(){
const payFunc = (args) => {
const {order_no,notify_url,TranAmt,MerName,sign,plain_text} = args;
const MercCode = import.meta.env.VITE_YCNC_MERCH_CODE
const tranAmt = Number(TranAmt).toFixed(2)
// const plainText = handleParams({MercCode,TranAmt:tranAmt,TermSsn:order_no})
const Plain = {
MercUrl:notify_url,
TranAmt:tranAmt,
TermSsn:order_no,
BackLink:encodeURIComponent(`${window.location.origin}/#/pages/ycnc/orderDetail?order_no=${order_no}&isPayBack=true`),
psbcmcc:'LSXD',
TxnDt:dayjs(Date.now()).format('YYYY-MM-DD'),
MercCode:MercCode,
IsIntegral:'1',
MerName:MerName,
}
const params = {
Plain:handleParams(Plain),
plainText:plain_text,
sign:sign,
Signature: '',
}
console.log('handCodePay-plain',Plain);
console.log('handCodePay-params',params);
Fw.device.api.handCodePay(params);
}
return {payFunc}
}

View File

@ -1,145 +0,0 @@
<route lang="json5" type="home">
{
style: {
navigationStyle: 'custom',
navigationBarTitleText: '奶茶活动',
navigationBarBackgroundColor:'#FFF',
},
}
</route>
<template>
<scroll-view scroll-y class="page-wrapper">
<view class="nav-btn" @click="toOrder">我的订单</view>
<!-- <view style="opacity: 0;width: 100%;height:200rpx;position:absolute;top:230rpx" @click="test"></view> -->
<view class="wrapper flex flex-col flex-items-center box-border">
<view v-for="(item,key) in productList" :key="key">
<brand :products="item" :brandName="key"/>
</view>
</view>
</scroll-view>
</template>
<script setup>
import brand from './components/brand';
import { getProductList, goPay, login } from '../../api/ycnc'
import { onMounted, ref, provide, unref } from 'vue';
// import { getQueryString , isIOS, isAndroid } from '../../utils/utils';
import useCode from './hooks/useCode';
import usePay from './hooks/usePay';
import config from './config';
const productList = ref([]);
const authCode = ref('');
const handleBuy = (productData) => {
pay(productData)
}
const goDetail = (productData) => {
const {ProductId} = productData
uni.navigateTo({
url:`/pages/ycnc/detail?product_id=${ProductId}`
})
}
const pay = (productData) => {
const {ProductId,voucherAmount,brandFlag} = productData
console.log('商品数据',productData);
const params = {
product_id:ProductId
}
goPay({params}).then(res => {
const {order_no,notify_url,sign,plain_text} = res;
const {payFunc} = usePay();
payFunc({order_no,notify_url,TranAmt:voucherAmount,MerName:config[brandFlag].name,sign,plain_text})
}).catch(err => {
console.log(err);
})
}
provide('custom-events',{
handleBuy,
goDetail
})
const toOrder = () => {
uni.navigateTo({
url:'/pages/ycnc/order'
})
};
onMounted(async ()=>{
queryProducts();
const token = window.localStorage.getItem('token') || '';
if(!token){
const code = await useCode();
authCode.value = code
const {token} = await login({params:{code:unref(authCode)}});
window.localStorage.setItem('token',token);
}
});
function handleData(arg,args){
return arg.reduce((total,current) =>{
const flag = current.brandFlag;
const flagData = args.filter(item => item.brandFlag === flag)
if(Array.isArray(total[flag])){
total[flag].push(...flagData)
}else{
total[flag] = [...flagData]
}
return total
},{})
}
//
const queryProducts = () => {
const params = {GID:'4894651'}
getProductList({params}).then(res => {
const {milkList,
milkVoucherList} = res
const sortMilkList = milkList.sort((a,b)=>{
const orderA = config[a.brandFlag] ? config[a.brandFlag].order : Infinity;
const orderB = config[b.brandFlag] ? config[b.brandFlag].order : Infinity;
return orderA - orderB
})
productList.value = handleData(sortMilkList,milkVoucherList);
}).catch(err=>{
console.log(err);
})
}
</script>
<style lang='scss'>
.page-wrapper{
// width: 100vw;
// height: 100vh;
width:100%;
height: 100%;
overflow-y: auto;
}
.nav-btn{
position: absolute;
right: 0;
top:42rpx;
width: 42rpx;
height: 132rpx;
background: #FFFFFF;
border-radius: 16rpx 0rpx 0rpx 16rpx;
font-weight: normal;
font-size: 22rpx;
color: #4BB8FF;
writing-mode: vertical-lr;
display: flex;
align-items: center;
justify-content: center;
}
.wrapper{
width:100%;
min-height:100vh;
background: url('/static/ycnc/bg.png') no-repeat;
background-size:cover;
padding-top:730rpx;
}
</style>

View File

@ -1,279 +0,0 @@
<route lang="json5" type="page">
{
style: {
navigationStyle: 'custom',
navigationBarTitleText: '我的订单',
navigationBarBackgroundColor:'#FFF',
},
}
</route>
<template>
<wd-tabs color="#333" inactiveColor="#888" @click="handleClick">
<block v-for="item in tabs" :key="item.state">
<wd-tab :title="item.tabName" :name="item.state">
<scroll-view scroll-y class="page-wrapper" @scrolltolower="queryNext">
<!-- <view class="order-wrapper flex flex-col flex-items-center" v-if="Array.isArray(list[item.state]) && list[item.state].length > 0"> -->
<!-- <view v-for="(ele,index) in list[item.state]" :key="`${ele.id}-${item.state}`">
<OrderItem :detail="ele" @pay-event="pay" @detail-event="viewDetail" @pwd-event="viewPwd"/>
</view> -->
<view class="order-wrapper flex flex-col flex-items-center" v-if="Array.isArray(list) && list.length > 0">
<view v-for="(ele,index) in list" :key="`${ele.id}-${item.state}`">
<OrderItem
:detail="ele"
@pay-event="pay"
@detail-event="viewDetail"
@pwd-event="viewPwd"
@del-event="removeOrder"
@refund-event="refund"
/>
</view>
<view class="no-more" v-if="isLastPage">没有更多了~</view>
</view>
<view class="no-data" v-else>
<image
src="/static/ycnc/bg-nodata.png"
mode="scaleToFill"
class="no-img"
/>
<text class="no-text">暂无订单快去下单吧~</text>
</view>
</scroll-view>
</wd-tab>
</block>
</wd-tabs>
<wd-message-box />
</template>
<script setup>
import { useMessage } from 'wot-design-uni'
import OrderItem from './components/order-item';
import usePay from './hooks/usePay';
import { getOrderList,deleteOrder,refundOrder,queryOrderDetail } from '../../api/ycnc';
import { onMounted, ref, unref,reactive,computed } from 'vue';
import { tabs } from './config';
import {deepClone} from '../../utils/utils'
const message = useMessage()
const pageSize = 10;
const activeName = ref(0);
const total = ref(0);
const page = ref(1);
const list = ref([]);
const replace = ref(false);
//
// const enumPage = tabs.reduce((prev,cur) => {
// prev[cur.state] = 1
// return prev
// },{})
// const page = ref(enumPage)
// //
// const enumObj = tabs.reduce((prev,cur) => {
// prev[cur.state] = 0
// return prev
// },{})
// const total = reactive(enumObj);
//
// const enumOrder = tabs.reduce((prev,cur) => {
// prev[cur.state] = []
// return prev
// },{})
// const list = reactive(enumOrder)
//
const queryOrderList = () => {
const activeTab = unref(activeName);
const params = {
page:unref(page),
pageSize:pageSize,
...(activeTab !== 0 && {state:activeTab})
}
getOrderList({params}).then(res => {
const {count,data} = res
total.value = count
// Object.assign(list,{[activeTab]:[...list[activeTab],...data]})
if(replace.value){
const prevPage = page.value - 1;
const prevList = list.value.slice(0, prevPage * pageSize);
list.value = [...prevList,...data];
}else{
list.value = [...list.value,...data]
}
replace.value = false
}).catch(err => {
total.value = 0;
// Object.assign(list,{[activeTab]:[]})
list.value = []
})
}
function handleClick({index, name}){
page.value = 1;
total.value = 0;
// Object.assign(list,{[name]:[]})
list.value = [];
activeName.value = name;
// name
queryOrderList()
}
async function pay(orderData){
console.log(orderData);
const {order_no,notify_url,price,brand,sign,plain_text,id} = orderData;
// const params = {
// order_id:id,
// order_no: order_no
// }
// const res = await queryOrderDetail({params})
// if(res.third_status === '03'){
// uni.showToast({
// title: '~',
// icon: 'none'
// });
// replace.value = true;
// queryOrderList()
// return
// }
const {payFunc} = usePay();
payFunc({order_no,notify_url,TranAmt:price,MerName:brand,sign,plain_text})
}
function viewDetail(orderData){
const {id,order_no} = orderData;
uni.navigateTo({
url:`/pages/ycnc/orderDetail?order_id=${id}&order_no=${order_no}`
})
}
function viewPwd(orderData){
const {voucher_link} = orderData
if(voucher_link){
console.log('跳转外部链接--->',voucher_link);
window.location.href = voucher_link
}else{
console.error(`${voucher_link}无有效值`);
}
}
function removeOrder(orderData){
message
.confirm({
msg: '删除订单后无法恢复,确认继续吗?',
title: '提示'
})
.then(() => {
const {id} = orderData
const params = {order_id:id}
deleteOrder({params}).then(res => {
replace.value = true;
queryOrderList()
}).catch(err => {
console.log(err);
})
})
.catch(() => {
console.log('点击了取消按钮')
})
}
function refund(orderData){
const {id} = orderData
if(!id){
uni.showToast({
icon:'none',
title: '无有效的订单id',
});
return
}
const params = {order_id:id}
refundOrder({params}).then(res => {
uni.showToast({
icon:'success',
title: '退款成功',
});
replace.value = true;
queryOrderList();
}).catch(err => {
console.log(err);
})
}
onMounted(()=>{
queryOrderList()
})
const isLastPage = computed(()=>{
return unref(page) === Math.ceil(unref(total) / pageSize)
})
function queryNext(){
if(unref(isLastPage)){
console.error('已经是最后一页');
return false;
};
page.value = page.value + 1
queryOrderList()
}
</script>
<style lang='scss' scoped>
.page-wrapper{
width:100%;
// height: 100%;
height: calc(100vh - 42rpx);
overflow-y: auto;
background: #FAFBFD;
box-sizing: border-box;
padding-bottom:30rpx
}
.order-wrapper{
display: flex;
flex-direction: column;
width:100%;
}
.no-more{
font-weight: normal;
font-size: 18rpx;
color: #999999;
height:146rpx;
padding: 28rpx 0;
display: flex;
align-items: flex-end;
justify-content: center;
box-sizing: border-box;
}
.no-data{
height: 100%;
display: flex;
align-items: center;
justify-content: center;
flex-direction: column;
.no-img{
width:446rpx;
height: 446rpx;
}
.no-text{
font-weight: normal;
font-size: 24rpx;
color: #999999;
}
}
/* 修改标签页下划线颜色*/
// :deep(.wd-button.is-primary){
// background: #EA0000;
// }
:deep(.wd-tabs__line) {
background: #EA0000;
}
:deep(.wd-tabs){
display: flex;
flex-direction: column;
height:100%;
}
:deep(.wd-tab__body){
height:calc(100vh - 42rpx);
}
:deep(.wd-tabs__container){
flex:1;
}
</style>

View File

@ -1,311 +0,0 @@
<route lang="json5" type="page">
{
style: {
navigationStyle: 'custom',
navigationBarTitleText: '订单详情',
navigationBarBackgroundColor:'#FFF',
},
}
</route>
<template>
<view class="w-full h-full flex flex-col flex-items-center page">
<view class="wrapper" v-if="pageType === 1">
<view class="pro-info">
<view class="title">订单详情</view>
<view class="content flex">
<image class="pro-img" :src="orderDetail.main_image" />
<view class="flex flex-1 flex-col">
<view class="flex flex-justify-between name" style="margin-bottom:12rpx">
<view class="text-over">{{ orderDetail.product_name }}</view>
<view class="amount"><text></text>{{ orderDetail.price }}<text></text></view>
</view>
<view class="flex flex-justify-between num">
<view>数量</view>
<view>x1</view>
</view>
</view>
</view>
</view>
<view class="order-info">
<view class="info-item">订单编号{{ orderDetail.order_no }}</view>
<view class="info-item" v-if="!!orderDetail.exchange_time && ![9].includes(orderDetail.state)">支付时间{{
orderDetail.exchange_time }}</view>
<view class="info-item">下单时间{{ orderDetail.create_time }}</view>
<view class="info-item" style="margin-bottom: 24rpx;">订单金额{{ orderDetail.price }}</view>
</view>
</view>
<view class="btns" v-if="pageType === 1">
<view class="btn pwd" @click="viewPwd(orderDetail)"
v-if="[3].includes(orderDetail.state) && orderDetail.voucher_link">查看卡券</view>
<view class="btn pwd" @click="pay(orderDetail)" v-if="[1].includes(orderDetail.state)">立即付款</view>
<view class="btn pwd" v-if="[4].includes(orderDetail.state)" @click="goRefund">申请退款</view>
<view class="btn back" @click="backIndex">返回首页</view>
</view>
</view>
</template>
<script setup>
import { onShow, onHide, onUnload } from '@dcloudio/uni-app';
import { onMounted, reactive, ref, unref } from 'vue';
import { queryOrderDetail, refundOrder, queryOrderState,login } from '../../api/ycnc';
import { getQueryString } from '../../utils/utils'
import usePay from './hooks/usePay';
import useCode from './hooks/useCode';
const authCode = ref('');
const id = ref('');
const orderNo = ref('');
const orderDetail = reactive({});
const pageType = ref(0)
let timer = undefined
// onshow h5
onShow(async () => {
const token = window.localStorage.getItem('token') || '';
if(!token){
const code = await useCode();
authCode.value = code
const {token} = await login({params:{code:unref(authCode)}});
window.localStorage.setItem('token',token);
}
const order_id = getQueryString('order_id')
const order_no = getQueryString('order_no')
const isPayBack = getQueryString('isPayBack')
if (!order_id && !order_no) return
id.value = order_id
orderNo.value = order_no;
if (isPayBack) {
queryEvent()
} else {
getDetail();
}
})
onHide(() => {
timer && clearInterval(timer)
})
onUnload(() => {
timer && clearInterval(timer)
})
const backIndex = () => {
uni.navigateTo({
url: `/pages/ycnc/index`
})
}
const timerEvent = () => {
timer && clearInterval(timer)
uni.showLoading({
title: "加载中",
mask: true,
});
timer = setInterval(()=>{
queryEvent()
}, 2500)
}
const queryEvent = () => {
uni.hideLoading()
const params = {
order_id: unref(id),
order_no: unref(orderNo)
}
queryOrderState({ params }).then(res => {
const { state, voucher_link } = res
if([3].includes(state)){
if(voucher_link){
timer && clearInterval(timer)
window.location.replace(voucher_link)
}else{
timerEvent()
}
}else if([1,2].includes(state)){
timerEvent()
}else{
timer && clearInterval(timer)
getDetail()
}
})
}
const viewPwd = (detailData) => {
const { voucher_link } = detailData
if (voucher_link) {
console.log(`跳转外部链接-->${voucher_link}`);
window.location.href = voucher_link
} else {
console.error(`voucher_link无有效值`);
}
}
const goRefund = () => {
const { id } = orderDetail
if (!id) {
uni.showToast({
icon: 'none',
title: '无有效的订单id',
});
return
}
const params = { order_id: id }
refundOrder({ params }).then(res => {
uni.showToast({
icon: 'success',
title: '退款成功',
});
getDetail();
}).catch(err => {
console.log(err);
})
}
async function pay(orderData) {
console.log(orderData);
const { order_no, notify_url, price,brand,sign,plain_text,id } = orderData;
// const params = {
// order_id:id,
// order_no: order_no
// }
// const res = await queryOrderDetail({params})
// if(res.third_status === '03'){
// uni.showToast({
// title: '~',
// icon: 'none'
// });
// getDetail()
// return
// }
const { payFunc } = usePay();
payFunc({ order_no, notify_url, TranAmt: price,MerName:brand,sign,plain_text })
}
onMounted(() => {
// getDetail()
})
const getDetail = () => {
const params = {
order_id: unref(id),
order_no: unref(orderNo)
}
pageType.value = 1
queryOrderDetail({ params }).then(res => {
Object.assign(orderDetail, res)
})
}
</script>
<style lang="scss" scoped>
.page {
background-color: #fafafa;
}
.wrapper {
width: 702rpx;
height: 486rpx;
background: #FFFFFF;
border-radius: 24rpx;
margin-top: 24rpx;
box-sizing: border-box;
padding: 32rpx 24rpx;
.pro-info {
border-bottom: 1rpx solid #F0E1E1;
.title {
font-weight: 500;
font-size: 28rpx;
color: #333333;
margin-bottom: 28rpx;
}
.content {
margin-bottom: 24rpx;
font-weight: 400;
.pro-img {
display: block;
width: 90rpx;
height: 90rpx;
margin-right: 34rpx;
}
.text-over {
max-width: 80%;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
}
.name {
font-size: 24rpx;
color: #333333;
font-weight: 700;
}
.amount {
color: #333333;
font-weight: 400;
font-size: 24rpx;
text:nth-child(1) {
font-size: 20rpx;
}
text:nth-child(2) {
font-size: 22rpx;
}
}
.num {
font-weight: 400;
font-size: 24rpx;
color: #9E9E9E;
}
}
}
.order-info {
border-bottom: 1rpx solid #F0E1E1;
.info-item {
font-weight: 400;
font-size: 22rpx;
color: #6C6C6C;
margin-top: 24rpx;
}
}
}
.btns {
margin-top: 146rpx;
display: flex;
.btn {
width: 312rpx;
height: 76rpx;
border-radius: 38rpx;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.pwd {
font-weight: 500;
font-size: 32rpx;
color: #FFFFFF;
background: #EA0000;
margin-right: 10rpx;
}
.back {
font-weight: 400;
font-size: 32rpx;
color: #B9C8C7;
border: 2rpx solid #B9C8C7;
}
}
</style>

View File

@ -0,0 +1,85 @@
<template>
<view class="flex coupon-wrapper">
<view class="flex flex-justify-center flex-items-center left">
<text></text>
<text>10</text>
</view>
<view class="flex flex-justify-between flex-items-center right">
<view class="coupon-info">
<view class="coupon-name">芒果全屏会员年卡适用</view>
<view class="coupon-time">2024.09.19 -2024.10.31</view>
<view class="coupon-time">单日限领取1张</view>
</view>
<view class="flex flex-justify-center flex-items-center btn"
>立即领取</view
>
</view>
</view>
</template>
<script setup>
import { ref, reactive, unref, onMounted } from "vue";
</script>
<style scoped lang="scss">
.coupon-wrapper {
width: 686rpx;
height: 178rpx;
box-sizing: border-box;
background: url("@/static/ycysp/coupon.webp") no-repeat;
background-size: 100% 100%;
}
.left {
width: 150rpx;
height: 100%;
color: #0e9bf2;
letter-spacing: 0.04em;
font-weight: bold;
line-height: normal;
margin-right: 20rpx;
text:first-child {
font-size: 26rpx;
}
text:last-child {
font-size: 74rpx;
}
}
.right {
flex: 1;
height: 100%;
box-sizing: border-box;
padding-left: 10rpx;
padding-right: 20rpx;
overflow: hidden;
.coupon-info {
width: calc(100% - 30rpx - 162rpx);
font-weight: normal;
line-height: normal;
letter-spacing: 0.04em;
}
.coupon-name {
width: 100%;
font-size: 28rpx;
color: #1f93da;
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
}
.coupon-time {
font-size: 24rpx;
color: #78839e;
margin-top: 6rpx;
}
.btn {
width: 162rpx;
height: 64rpx;
border-radius: 27rpx;
opacity: 1;
font-size: 28rpx;
font-weight: normal;
line-height: 32rpx;
color: #ffffff;
background: linear-gradient(270deg, #9ad2ff 0%, #0e9bf2 100%);
}
}
</style>

View File

@ -0,0 +1,86 @@
<template>
<view
class="wrapper"
:style="{ backgroundImage: `url(${detail.brand_logo})` }"
>
<svg
xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink"
fill="none"
version="1.1"
viewBox="0 0 13 13"
class="svg"
@click="checkedEvent"
>
<g>
<ellipse
cx="6.5"
cy="6.5"
rx="6.5"
ry="6.5"
fill="#FFFFFF"
fill-opacity="1"
/>
<path
d="M6.85467,0.16026C6.65747,-0.049906,6.33416,-0.0538585,6.13257,0.151333L2.65289,3.69208L0.855769,1.81926C0.669146,1.62458,0.356251,1.63538,0.156706,1.84316C-0.0428302,2.05098,-0.0532221,2.37714,0.133625,2.57177L2.33038,4.86098C2.51696,5.05539,2.82989,5.04459,3.0294,4.83682C3.07511,4.78922,3.10934,4.73525,3.13501,4.67819C3.15477,4.66321,3.17447,4.649,3.19236,4.63059L6.84614,0.91282C7.048,0.707426,7.05164,0.370686,6.85467,0.16026L6.85467,0.16026Z"
:fill="checkedColor[detail.id]"
fill-opacity="1"
class="gou"
v-if="detail.isChecked"
/>
<ellipse
cx="6.5"
cy="6.5"
rx="6.2"
ry="6.2"
fill-opacity="0"
stroke-opacity="1"
:stroke="checkedColor[detail.id]"
fill="none"
stroke-width="0.6"
/>
</g>
</svg>
</view>
</template>
<script setup lang="ts">
import { ref, reactive, unref, onMounted } from "vue";
import { checkedColor } from "../config";
const props = defineProps({
detail: {
type: Object,
required: true,
},
});
const emits = defineEmits(["checked"]);
const checkedEvent = () => {
emits("checked", props.detail.id);
};
</script>
<style scoped lang="scss">
.wrapper {
// width:226rpx;
width: 100%;
height: 110rpx;
box-sizing: border-box;
// background: url('@/static/ycysp/cloud.webp') no-repeat;
background-repeat: no-repeat;
background-size: 100% 100%;
position: relative;
}
.gou {
width: 14rpx;
height: 10rpx;
transform: translate(6rpx, 8rpx);
}
.svg {
position: absolute;
right: 8rpx;
top: 8rpx;
width: 27rpx;
height: 27rpx;
}
</style>

View File

@ -0,0 +1,236 @@
<template>
<view class="order-item-wrapper flex flex-col box-border" @click="goDetail">
<view class="nav-info flex flex-justify-between">
<view>订单编号{{ detail.order_no }}</view>
<view :style="{ color: stateColor[detail.status] || '' }">{{
stateConfig[detail.status]
}}</view>
</view>
<view class="order-info">
<view class="content flex">
<image :src="detail.main_image" class="pro-img" />
<view class="flex flex-1 flex-col">
<view class="flex flex-justify-between name info-item">
<view class="text-over">{{ detail.product_name }}</view>
<view class="amount"
><text></text>{{ detail.order_amount }}<text></text></view
>
</view>
<view class="flex num">
<view>充值账号</view>
<view>{{ maskPhoneNumber(detail.account) }}</view>
</view>
<view class="flex flex-justify-between num">
<view>数量</view>
<view>x1</view>
</view>
</view>
</view>
<view class="flex flex-justify-between time">
<view>下单时间{{ detail.create_time }}</view>
<!-- <view v-if="[1].includes(detail.status)">优惠金额¥{{ discount }}</view> -->
</view>
</view>
<view class="line"></view>
<view class="bottoms">
<!-- <view class="pay-des" v-if="[1].includes(detail.status)">
<view class="count-item"> 1 应付总额<text style="text-decoration: line-through;">{{ detail.order_amount }}</text></view>
<view class="number-item">实际支付¥{{ detail.wait_amount }}</view>
</view> -->
<view class="btns">
<!-- <view class="btn del" @click="goDel">删除订单</view> -->
<view class="btn pay" v-if="[1].includes(detail.status)" @click="goPay"
>立即付款</view
>
<view
class="btn pay"
@click="goRefund"
v-if="[4].includes(detail.status)"
>申请退款</view
>
</view>
</view>
</view>
</template>
<script setup>
import { computed } from "vue";
import { stateConfig, stateColor } from "../config";
const props = defineProps({
detail: {
type: Object,
required: true,
default: () => ({}),
},
});
const emits = defineEmits([
"pay-event",
"detail-event",
"pwd-event",
"del-event",
]);
function goPay(e) {
if (e.stopPropagation) {
//W3C
e.stopPropagation();
}
emits("pay-event", props.detail);
}
const discount = computed(() => {
if (!Object.keys(props.detail).length) return 0;
return (
(Number(props.detail.order_amount) * 100 -
Number(props.detail.wait_amount) * 100) /
100
);
});
function maskPhoneNumber(phoneNumber) {
return phoneNumber.replace(/(\d{3})\d{4}(\d{4})/, "$1****$2");
}
function goDetail(e) {
emits("detail-event", props.detail);
}
function goDel(e) {
if (e.stopPropagation) {
//W3C
e.stopPropagation();
}
emits("del-event", props.detail);
}
function goRefund(e) {
if (e.stopPropagation) {
//W3C
e.stopPropagation();
}
emits("refund-event", props.detail);
}
</script>
<style lang="scss">
.order-item-wrapper {
width: 702rpx;
// height: 386rpx;
background: #ffffff;
border-radius: 24rpx;
padding: 24rpx;
margin: 24rpx 0 16rpx;
}
.nav-info {
font-size: 22rpx;
font-weight: 400;
:first-child {
color: #333333;
}
:last-child {
color: #787878;
}
}
.order-info {
margin-top: 24rpx;
.content {
margin-bottom: 24rpx;
font-weight: 400;
.pro-img {
display: block;
width: 90rpx;
height: 90rpx;
margin-right: 34rpx;
}
}
.time {
font-size: 22rpx;
color: #6c6c6c;
}
.info-item {
font-weight: bold;
margin-bottom: 12rpx;
}
.text-over {
max-width: 80%;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
}
.amount {
color: #333333;
font-weight: bold;
font-size: 24rpx;
text:nth-child(1) {
font-size: 20rpx;
font-weight: 400;
}
text:nth-child(2) {
font-size: 22rpx;
}
}
.name {
font-size: 24rpx;
color: #333333;
font-weight: 700;
}
.num {
font-size: 20rpx;
color: #9e9e9e;
font-weight: 400;
margin-bottom: 10rpx;
}
}
.count-item {
font-size: 22rpx;
color: #878787;
text-align: right;
margin-top: 8rpx;
}
.number-item {
font-size: 24rpx;
color: #2e92ff;
text-align: right;
margin-top: 16rpx;
}
.bottoms {
padding: 10rpx 0 12rpx;
}
.btns {
display: flex;
justify-content: flex-end;
margin-top: 24rpx;
.btn {
width: 168rpx;
height: 64rpx;
border-radius: 68rpx;
font-weight: 400;
font-size: 28rpx;
display: flex;
align-items: center;
justify-content: center;
margin-left: 48rpx;
}
.del {
color: #545454;
border: 2rpx solid #b9c8c7;
}
.pay {
color: #ffffff;
background: linear-gradient(270deg, #2c91ff 0%, #9ad2ff 100%);
}
.view {
background: #ffffff;
border-radius: 68rpx;
border: 2rpx solid #ea0000;
color: #ea0000;
}
}
.line {
margin-top: 24rpx;
width: 662rpx;
height: 1rpx;
background-color: #e1e1f0;
transform: scaleY(0.5);
}
</style>

View File

@ -0,0 +1,79 @@
<template>
<view
class="flex flex-col flex-items-center flex-justify-center re-wrapper"
:class="{ 'is-checked': detail.isSelected }"
@click="selectEvent"
>
<view class="type-name">{{ detail.product_type }}</view>
<view class="flex type-amount">
<view class="flex flex-col flex-justify-end icon"></view>
<view class="num">{{ detail.sell_price }}</view>
</view>
<view class="type-ori">原价{{ detail.price }}</view>
</view>
</template>
<script setup lang="ts">
import { ref, reactive, unref, onMounted } from "vue";
const props = defineProps({
detail: {
type: Object,
required: true,
},
});
const emits = defineEmits(["selected"]);
const selectEvent = () => {
emits("selected", props.detail.id);
};
</script>
<style scoped lang="scss">
.re-wrapper {
width: 100%;
height: 200rpx;
box-sizing: border-box;
border-radius: 10rpx;
opacity: 1;
background: #fff;
box-sizing: border-box;
border: 2rpx solid #ececec;
}
.is-checked {
background: #dcefff;
border: 1.6rpx solid #2d91ff;
}
.type-name {
width: 80%;
font-size: 26rpx;
font-weight: normal;
line-height: normal;
color: #070b1b;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
text-align: center;
}
.type-amount {
margin-top: 6rpx;
color: $sub-price-color;
font-weight: 600;
line-height: normal;
.icon {
font-size: 28rpx;
margin-bottom: 8rpx;
margin-right:4rpx;
}
.num {
/* 价格 */
font-size: 48rpx;
}
}
.type-ori {
font-size: 24rpx;
font-weight: normal;
text-decoration: line-through;
color: #a4c4fa;
margin-top: 4rpx;
}
</style>

View File

@ -0,0 +1,167 @@
<template>
<view
class="flex pos-relative wrapper flex-justify-between"
:class="{
small: ['small'].includes(type),
large: ['large'].includes(type),
}"
>
<view
class="flex flex-items-center flex-justify-center pos-absolute tips"
v-if="['small'].includes(type)"
>每日限量50张</view
>
<view
class="amount"
:style="{ width: ['small'].includes(type) ? '120rpx' : '140rpx' }"
>
<view class="icon"></view>
<view class="number">100</view>
</view>
<view class="operation" v-if="['small'].includes(type)">
<view class="name">优酷喵喵年卡适用</view>
<view class="flex flex-items-center flex-justify-center btn"
>立即领取</view
>
</view>
<view class="operation-large" v-else>
<view class="name-tips">
<view class="name-large">芒果全屏会员年卡适用</view>
<view class="tips-large">每日限量50张</view>
</view>
<view class="flex flex-items-center flex-justify-center btn-large"
>立即领取</view
>
</view>
</view>
</template>
<script setup>
import { ref, reactive, unref, onMounted } from "vue";
const props = defineProps({
type: {
type: String,
default: "small",
},
});
</script>
<style scoped lang="scss">
.wrapper {
height: 136rpx;
}
.small {
width: 322rpx;
background: url("@/static/ycysp/ticket-small.webp") no-repeat;
margin-bottom: 18rpx;
background-size: cover;
}
.large {
width: 662rpx;
background: url("@/static/ycysp/ticket-large.webp") no-repeat;
background-size: cover;
}
.tips {
border-radius: 20rpx 0rpx 20rpx 0rpx;
opacity: 1;
background: #69c5ff;
width: 100rpx;
height: 20rpx;
width: 100rpx;
font-size: 12rpx;
font-weight: normal;
line-height: normal;
letter-spacing: 0.04em;
color: #ffffff;
}
.amount {
display: flex;
justify-content: center;
height: 100%;
font-weight: bold;
line-height: normal;
letter-spacing: 0.04em;
color: $price-color;
}
.icon {
font-size: 18rpx;
display: flex;
align-items: center;
margin-top: 20rpx;
}
.number {
font-size: 48rpx;
display: flex;
align-items: center;
}
.operation {
width: 180rpx;
height: 100%;
display: flex;
flex-direction: column;
justify-content: center;
box-sizing: border-box;
padding: 0 15rpx;
}
.name {
width: 100%;
font-size: 22rpx;
font-weight: normal;
line-height: normal;
letter-spacing: 0em;
color: $text-color;
overflow: hidden; /*文本超出隐藏*/
display: -webkit-box; /*盒子模型微弹性伸缩模型*/
-webkit-box-orient: vertical; /*伸缩盒子的子元素垂直排列*/
-webkit-line-clamp: 2; /*文本显示3行*/
}
.btn {
margin-top: 10rpx;
width: 134rpx;
height: 34rpx;
background: linear-gradient(270deg, #9ad2ff 0%, #0e9bf2 100%);
border-radius: 38rpx;
font-size: 20rpx;
font-weight: normal;
line-height: 24rpx;
color: #ffffff;
}
.operation-large {
width: 500rpx;
height: 100%;
display: flex;
align-items: center;
justify-content: space-between;
box-sizing: border-box;
padding: 0 24rpx 0 10rpx;
}
.name-tips {
display: flex;
flex-direction: column;
}
.name-large {
font-size: 28rpx;
font-weight: normal;
line-height: normal;
letter-spacing: 0.04em;
color: #1f93da;
}
.tips-large {
font-size: 20rpx;
font-weight: normal;
line-height: normal;
letter-spacing: 0.04em;
color: #8ca1d1;
margin-top: 4rpx;
}
.btn-large {
width: 162rpx;
height: 64rpx;
background: linear-gradient(270deg, #9ad2ff 0%, #0e9bf2 100%);
border-radius: 68rpx;
font-size: 28rpx;
font-weight: normal;
line-height: 64rpx;
color: #ffffff;
}
</style>

84
src/pages/ycysp/config.js Normal file
View File

@ -0,0 +1,84 @@
const config = {
1:{
name:'优酷',
},
2:{
name:'腾讯',
},
3:{
name:'网易云',
},
4:{
name:'芒果TV',
},
5:{
name:'QQ音乐',
},
6:{
name:'喜马拉雅',
},
7:{
name:'哔哩哔哩',
},
8:{
name:'爱奇艺',
},
}
export const stateConfig = {
1: "待支付",
2: "已支付",
3: "充值中",
4: "充值失败",
5: "退款中",
6: "退款失败",
7: "退款成功",
8: "充值成功",
9: "订单作废",
10:'支付失败'
};
export const stateColor = {
2: "#FF911A",
3: "#FF8400",
4: "#FF0000",
7: "#EA722D",
8: "#FF0000",
10: "#FF0000"
};
export const checkedColor = {
1: "#4A98FB",
2: "#4A98FB",
3: "#FE3B34",
4: "#FF8103",
5: "#17BE73",
6: "#F82801",
7: "#FE306E",
8: "#61D93F",
};
export const tabs = [
{
tabName: "全部订单",
status: 0,
},
{
tabName: "待付款",
status: 1,
},
{
tabName: "处理中",
status: 2,
},
{
tabName: "已完成",
status: 8,
},
{
tabName: "已退款",
status: 7,
},
];
export default config

View File

@ -0,0 +1,17 @@
<template>
<view class="page-wrapper">
<couponItem />
</view>
</template>
<script setup>
import { ref, reactive, unref, onMounted} from 'vue';
import couponItem from './components/coupon-item.vue';
</script>
<style scoped lang="scss">
.page-wrapper{
width:100vw;
height:100vh;
}
</style>

454
src/pages/ycysp/home.vue Normal file
View File

@ -0,0 +1,454 @@
<route lang="json5" type="home">
{
style: {
navigationStyle: "custom",
navigationBarTitleText: "音视频专区",
navigationBarBackgroundColor: "#FFF",
},
}
</route>
<template>
<scroll-view class="page-wrapper" scroll-y>
<view class="w-full box-border banner-container">
<image
src="@/static/ycysp/banner.webp"
mode="scaleToFill"
class="w-full h-full"
/>
</view>
<!-- // -->
<!-- <view class="ticket-container">
<view class="title">
<view class="title-line"></view>
<view class="title-text">领券更划算</view>
<view class="title-more">我的优惠券</view>
</view>
<view class="flex flex-wrap flex-justify-between">
<ticketRecieve />
<ticketRecieve />
<ticketRecieve type="large"/>
</view>
</view> -->
<!-- // -->
<view class="member-container">
<view class="title">
<view class="title-text">选择影音会员</view>
</view>
<scroll-view scroll-x class="scroll" :scroll-left="scrollLeft" scroll-with-animation>
<view class="swiper">
<view v-for="(item, index) in brandArr" class="member-item">
<member :key="item.id" :detail="item" @checked="handleChecked" />
</view>
</view>
</scroll-view>
</view>
<!-- // -->
<view class="recharge-container">
<view class="title">
<view class="title-text">充值选项</view>
</view>
<scroll-view scroll-x class="recharge-scroll">
<view v-for="(item, index) in productArr" class="recharge-item">
<rechargeItem
:key="item.id"
:detail="item"
@selected="handleSelected"
/>
</view>
</scroll-view>
</view>
<!-- // -->
<view class="input-container">
<input
type="text"
placeholder="请输入手机号充值"
placeholder-class="pcs"
v-model="account"
/>
</view>
<!-- 下一步 -->
<view
class="flex flex-items-center flex-justify-center next-btn"
@click="nextEvent"
>
下一步
</view>
<!-- 描述 -->
<view class="description">
<view class="small-title">产品介绍</view>
<view class="small-text"
>1充值后可登录官方APP<text class="small-red"
>查看会员状态及有效期</text
>过期不可使用</view
>
<view class="small-text"
>2<text class="small-red"
>充值前请您确认充值信息是否正确一旦提交无法取消或更改</text
>如因账号信息输入错误导致充值至他人账户仍视为充值成功无法补券</view
>
<view class="small-text"
>3此权益中包含的优酷黄金会员芒果视频会员可在手机电脑平板上使用不支持电视端不支持投屏<text
class="small-red"
>QQ音乐支持QQ号或手机号</text
>其余会员账号均为手机号</view
>
<view class="small-text"
>4本产品仅适用于中国大陆地区用户暂不支持开通中国港澳台海外地区服务</view
>
<view class="small-text"
>5如有疑问请联系成都蓝色兄弟网络科技有限公司<text class="small-red"
>客服电话400-623-9566</text
>服务时间09:00-18:00
</view>
</view>
<view style="height: 40rpx"></view>
<!-- 我的订单 -->
<view
class="flex flex-justify-center flex-items-center order-btn"
@click="toOrder"
>我的<br />订单</view
>
</scroll-view>
</template>
<script setup>
import { ref, reactive, unref, onMounted } from "vue";
import {onLoad} from "@dcloudio/uni-app";
// import ticketRecieve from './components/ticket-recieve.vue';
import member from "./components/member.vue";
import rechargeItem from "./components/recharge-item.vue";
import { getBrandList, getProductList, login } from "@/api/ycysp";
import { generateUUID,getQueryString } from "@/utils/utils";
import { createOrder } from "@/api/ycysp";
import useCode from "./hooks/useCode";
// const reg =
// /^(((13[0-9]{1})|(15[0-9]{1})|(16[0-9]{1})|(17[3-8]{1})|(18[0-9]{1})|(19[0-9]{1})|(14[5-7]{1}))+\d{8})$/;
// const _reg = /^[1-9][0-9]{4,}$/;
const brandArr = ref([]);
const productArr = ref([]);
const account = ref("");
const authCode = ref("");
const scrollLeft = ref(0);
//
const getBrand = () => {
return getBrandList()
.then((res) => {
const { list } = res;
const is_anchor_brand_id = getQueryString('brand_id')
brandArr.value = list.map((item, index) => ({
...item,
isChecked: is_anchor_brand_id ? item.id === Number(is_anchor_brand_id) : index === 0,
}));
setTimeout(()=>{
scrollLeft.value = 60
},500)
})
.catch((err) => {
console.error(err);
});
};
//
const getProduct = () => {
const checkedBrand = brandArr.value.find((item) => item.isChecked);
const params = { brand_id: checkedBrand.id };
getProductList({ params })
.then((res) => {
const { list } = res;
const is_anchor_product_id = getQueryString('product_id')
productArr.value = list.map((item, index) => ({
...item,
isSelected: is_anchor_product_id ? item.id === Number(is_anchor_product_id) : index === 0,
}));
})
.catch((err) => {
console.error(err);
});
};
//
const handleChecked = (checkedId) => {
brandArr.value = brandArr.value.map((item, index) => {
const { id, isChecked } = item;
if (checkedId === id) {
return { ...item, isChecked: !isChecked };
}
return { ...item, isChecked: false };
});
getProduct();
};
//
const handleSelected = (selectedId) => {
productArr.value = productArr.value.map((item, index) => {
const { id, isSelected } = item;
if (selectedId === id) {
return { ...item, isSelected: !isSelected };
}
return { ...item, isSelected: false };
});
};
//
const nextEvent = () => {
const selectProduct = productArr.value.find((item) => item.isSelected);
if (!selectProduct) {
uni.showToast({
title: "请选择充值选项",
icon: "none",
});
return;
}
if (!account.value) {
uni.showToast({
title: "请填写充值手机号",
icon: "none",
});
return;
}
// if (!reg.test(account.value)) {
// uni.showToast({
// title: ",",
// icon: "none",
// });
// return;
// }
addOrder(selectProduct.id);
};
const addOrder = (id) => {
const params = {
account: unref(account),
product_id: id,
};
createOrder({ params })
.then((res) => {
const { order_no, notify_url, sign, plain_text } = res;
uni.navigateTo({
url: `/pages/ycysp/pay?order_no=${order_no}`,
});
})
.catch((err) => {
console.error(err);
});
};
const toOrder = () => {
uni.navigateTo({
url: `/pages/ycysp/order`,
});
};
onLoad((options) => {
const {brand_id,product_id} = options;
console.log(options)
})
onMounted(async () => {
await getBrand();
getProduct();
const token = window.localStorage.getItem("token") || "";
if (!token) {
const code = await useCode();
authCode.value = code
// authCode.value = generateUUID();
const { token } = await login({ params: { code: unref(authCode) } });
window.localStorage.setItem("token", token);
}
});
</script>
<style scoped lang="scss">
.page-wrapper {
width: 100vw;
height: 100vh;
opacity: 1;
// background: linear-gradient(180deg, #FFFFFF 18%, #F3F3F3 40%, #FAFBFD 60%, #FAFBFD 61%, #FAFBFD 65%);
background: #ffffff;
overflow-y: auto;
}
.banner-container {
height: 338rpx;
padding: 32rpx 20rpx 0;
background: url("@/static/ycysp/banner-bg.webp") no-repeat;
background-size: 100% auto;
}
.ticket-container {
width: 710rpx;
height: 416rpx;
border-radius: 28rpx;
opacity: 1;
background: #ffffff;
box-shadow: 0rpx 4rpx 8rpx 0rpx rgba(0, 0, 0, 0.04);
margin: 40rpx auto 0;
box-sizing: border-box;
padding: 24rpx;
}
.title {
width: 100%;
height: 50rpx;
display: flex;
align-items: center;
position: relative;
margin-bottom: 2rpx;
.title-line {
width: 10rpx;
height: 34rpx;
border-radius: 0rpx 6rpx 6rpx 0rpx;
opacity: 1;
background: linear-gradient(180deg, #95ceff 0%, #439fff 100%);
}
.title-text {
font-size: 36rpx;
font-weight: 400;
color: #212f3b;
margin-left: 14rpx;
font-family: PingFang SC-Bold;
}
.title-more {
position: absolute;
right: 0;
font-size: 26rpx;
font-weight: normal;
color: #666e7e;
}
}
.member-container {
width: 100%;
padding-left: 20rpx;
box-sizing: border-box;
margin: 40rpx auto 0;
}
.scroll {
width: 100%;
// height:250rpx;
overflow-x: auto;
transition: all 1s linear;
}
.swiper {
display: flex;
flex-wrap: wrap;
width: calc((100vw - 68rpx) / 3 * 4 + 64rpx);
}
.member-item {
margin: 16rpx 16rpx 0 0;
width: calc((100vw - 68rpx) / 3);
}
.recharge-container {
width: 100%;
margin: 40rpx auto 0;
padding-left: 20rpx;
box-sizing: border-box;
}
.recharge-scroll {
width: 100%;
overflow-x: auto;
white-space: nowrap;
}
.recharge-item {
display: inline-flex;
margin: 20rpx 16rpx 0 0;
// width: calc((100vw - 68rpx) / 3.5);
width: 204rpx;
}
.input-container {
width: 710rpx;
height: 88rpx;
border-radius: 84rpx;
opacity: 1;
background: #eef7ff;
margin: 36rpx auto 0;
display: flex;
align-items: center;
padding: 0 40rpx;
box-sizing: border-box;
input {
width: 100%;
font-size: 32rpx;
font-weight: normal;
color: #8dbff6;
}
}
.pcs {
font-size: 32rpx;
font-weight: normal;
color: #8dbff6;
}
.next-btn {
width: 710rpx;
height: 88rpx;
padding: 12rpx 28rpx;
margin: 20rpx auto 0;
font-size: 32rpx;
font-weight: normal;
line-height: normal;
letter-spacing: 0em;
color: #ffffff;
background: linear-gradient(270deg, #3194ff 0%, #96cfff 100%);
box-sizing: border-box;
border-radius: 392rpx;
}
.description {
width: 100%;
box-sizing: border-box;
padding: 0 20rpx;
margin-top: 40rpx;
}
.small-title {
/* 标题二 */
font-size: 28rpx;
font-weight: normal;
line-height: 32rpx;
letter-spacing: 0px;
color: #0f0f0f;
margin-bottom: 20rpx;
}
.small-text {
font-size: 24rpx;
font-weight: normal;
line-height: 44rpx;
color: #383838;
}
.small-red {
color: #ff0000;
}
.order-btn {
position: fixed;
right: 20rpx;
bottom: 100rpx;
width: 96rpx;
height: 96rpx;
font-size: 24rpx;
font-weight: normal;
line-height: 32rpx;
letter-spacing: 0px;
color: #ffffff;
opacity: 1;
background: rgba(45, 145, 255, 0.6);
border-radius: 96rpx;
}
</style>

View File

@ -0,0 +1,29 @@
import md5 from "js-md5";
export default function useCode() {
return new Promise((resolve, reject) => {
const authCallback = (params) => {
console.log("authCallback-use", params);
const { code } = params;
resolve(code);
};
const getNewAuthorization = () => {
let appId = import.meta.env.VITE_YCNC_APPID;
let time = Date.now().toString();
let secret = import.meta.env.VITE_YCNC_SECRET;
let signBefore = appId + time + secret;
let sign = md5(signBefore); //普通md5
let param = {
appId: appId,
sign: sign,
time: time,
tran_code: "157",
fn: "authCallback",
needBind: "",
};
console.log("auth-param157-use", param);
Fw.device.api.getNewAuthorization(param);
};
window.authCallback = authCallback;
getNewAuthorization();
});
}

View File

@ -0,0 +1,45 @@
import dayjs from "dayjs";
import md5 from "js-md5";
const handleParams = (obj) =>
Object.entries(obj).reduce((total, curr) => {
if (!total) {
total += `${curr[0]}=${curr[1]}`;
} else {
total += `|${curr[0]}=${curr[1]}`;
}
return total;
}, "");
export default function usePay() {
const payFunc = (args) => {
const { order_no, notify_url, TranAmt, MerName, sign, plain_text } = args;
const MercCode = import.meta.env.VITE_YCNC_MERCH_CODE;
const tranAmt = Number(TranAmt).toFixed(2);
const Plain = {
MercUrl: notify_url,
TranAmt: tranAmt,
TermSsn: order_no,
BackLink: encodeURIComponent(
`${window.location.origin}${
window.location.pathname || ""
}/#/pages/ycysp/orderDetail?order_no=${order_no}&isPayBack=true`
),
psbcmcc: "LSXD",
TxnDt: dayjs(Date.now()).format("YYYY-MM-DD"),
MercCode: MercCode,
IsIntegral: "1",
MerName: MerName,
};
const params = {
Plain: handleParams(Plain),
plainText: plain_text,
sign: sign,
Signature: "",
};
console.log("CodePay-plain", Plain);
console.log("CodePay-params", params);
Fw.device.api.handCodePay(params);
};
return { payFunc };
}

251
src/pages/ycysp/order.vue Normal file
View File

@ -0,0 +1,251 @@
<route lang="json5" type="page">
{
style: {
navigationStyle: "custom",
navigationBarTitleText: "我的订单",
navigationBarBackgroundColor: "#FFF",
},
}
</route>
<template>
<wd-tabs color="#333" inactiveColor="#888" @click="handleClick">
<block v-for="item in tabs" :key="item.status">
<wd-tab :title="item.tabName" :name="item.status">
<scroll-view scroll-y class="page-wrapper" @scrolltolower="queryNext">
<view
class="order-wrapper flex flex-col flex-items-center"
v-if="Array.isArray(listData) && listData.length > 0"
>
<view
v-for="(ele, index) in listData"
:key="`${ele.id}-${item.status}`"
>
<OrderItem
:detail="ele"
@pay-event="pay"
@detail-event="viewDetail"
@del-event="removeOrder"
@refund-event="refund"
/>
</view>
<view class="no-more" v-if="isLastPage">没有更多了~</view>
</view>
<view class="no-data" v-else>
<image
src="/static/ycysp/bg-nodata.png"
mode="scaleToFill"
class="no-img"
/>
<text class="no-text">暂无订单快去下单吧~</text>
</view>
</scroll-view>
</wd-tab>
</block>
</wd-tabs>
<wd-message-box />
</template>
<script setup>
import { useMessage } from "wot-design-uni";
import OrderItem from "./components/order-item";
import usePay from "./hooks/usePay";
import { getOrderList, refundOrder } from "../../api/ycysp";
import { onMounted, ref, unref, reactive, computed } from "vue";
import { tabs } from "./config";
import { deepClone } from "../../utils/utils";
const message = useMessage();
const pageSize = 10;
const activeName = ref(0);
const totalData = ref(0);
const page = ref(1);
const listData = ref([]);
const replace = ref(false);
//
const queryOrderList = () => {
const activeTab = unref(activeName);
const params = {
page: unref(page),
page_size: pageSize,
...(activeTab !== 0 && { status: activeTab }),
};
getOrderList({ params })
.then((res) => {
const { total, list } = res;
totalData.value = total;
if (replace.value) {
const prevPage = page.value - 1;
const prevList = listData.value.slice(0, prevPage * pageSize);
listData.value = [...prevList, ...list];
} else {
listData.value = [...listData.value, ...list];
}
replace.value = false;
})
.catch((err) => {
totalData.value = 0;
listData.value = [];
});
};
function handleClick({ index, name }) {
page.value = 1;
totalData.value = 0;
listData.value = [];
activeName.value = name;
// name
queryOrderList();
}
function pay(orderData) {
console.log(orderData);
const { order_no, notify_url, sign, plain_text } =
orderData.pay_info;
const price = orderData.order_amount
const MerName = orderData.brandName
const { payFunc } = usePay();
payFunc({
order_no,
notify_url,
TranAmt: price,
MerName: MerName,
sign,
plain_text,
});
}
function viewDetail(orderData) {
const { id, order_no } = orderData;
uni.navigateTo({
url: `/pages/ycysp/orderDetail?order_id=${id}&order_no=${order_no}`,
});
}
function removeOrder(orderData) {
message
.confirm({
msg: "删除订单后无法恢复,确认继续吗?",
title: "提示",
})
.then(() => {
const { id } = orderData;
const params = { order_id: id };
deleteOrder({ params })
.then((res) => {
replace.value = true;
queryOrderList();
})
.catch((err) => {
console.log(err);
});
})
.catch(() => {
console.log("点击了取消按钮");
});
}
function refund(orderData) {
const { id } = orderData;
if (!id) {
uni.showToast({
icon: "none",
title: "无有效的订单id",
});
return;
}
const params = { id: id };
refundOrder({ params })
.then((res) => {
uni.showToast({
icon: "success",
title: "退款成功",
});
replace.value = true;
queryOrderList();
})
.catch((err) => {
console.log(err);
});
}
onMounted(() => {
queryOrderList();
});
const isLastPage = computed(() => {
return unref(page) === Math.ceil(unref(totalData) / pageSize);
});
function queryNext() {
console.log(unref(isLastPage));
if (unref(isLastPage)) {
console.error("已经是最后一页");
return false;
}
page.value = page.value + 1;
queryOrderList();
}
</script>
<style lang="scss" scoped>
.page-wrapper {
width: 100%;
// height: 100%;
height: calc(100vh - 42rpx);
overflow-y: auto;
background: #fafbfd;
box-sizing: border-box;
padding-bottom: 30rpx;
}
.order-wrapper {
display: flex;
flex-direction: column;
width: 100%;
}
.no-more {
font-weight: normal;
font-size: 18rpx;
color: #999999;
height: 146rpx;
padding: 28rpx 0;
display: flex;
align-items: flex-end;
justify-content: center;
box-sizing: border-box;
}
.no-data {
height: 100%;
display: flex;
align-items: center;
justify-content: center;
flex-direction: column;
.no-img {
width: 446rpx;
height: 446rpx;
}
.no-text {
font-weight: normal;
font-size: 24rpx;
color: #999999;
}
}
/* 修改标签页下划线颜色*/
// :deep(.wd-button.is-primary){
// background: #EA0000;
// }
:deep(.wd-tabs__line) {
background: linear-gradient(180deg, #99d1ff 0%, #3194ff 100%);
}
:deep(.wd-tabs) {
display: flex;
flex-direction: column;
height: 100%;
}
:deep(.wd-tab__body) {
height: calc(100vh - 42rpx);
}
:deep(.wd-tabs__container) {
flex: 1;
}
</style>

View File

@ -0,0 +1,325 @@
<route lang="json5" type="page">
{
style: {
navigationStyle: "custom",
navigationBarTitleText: "订单详情",
navigationBarBackgroundColor: "#FFF",
},
}
</route>
<template>
<view class="w-full h-full flex flex-col flex-items-center page">
<view class="wait" v-if="isWaiting">
<wd-loading :size="40"/>
<text class="wait-text">订单查询中...</text>
</view>
<view class="wrapper" v-if="!isWaiting">
<view class="pro-info">
<view class="title">订单详情</view>
<view class="content flex">
<image class="pro-img" :src="orderDetail.main_image" />
<view class="flex flex-1 flex-col">
<view
class="flex flex-justify-between name"
style="margin-bottom: 12rpx"
>
<view class="text-over">{{ orderDetail.product_name }}</view>
<view class="amount"
><text></text>{{ orderDetail.order_amount
}}<text></text></view
>
</view>
<view class="flex num">
<view>充值账号</view>
<view>{{ orderDetail.account ? maskPhoneNumber(orderDetail.account) : '' }}</view>
</view>
<view class="flex flex-justify-between num">
<view>数量</view>
<view>x1</view>
</view>
</view>
</view>
</view>
<view class="line"></view>
<view class="order-info">
<view class="info-item">订单编号{{ orderDetail.order_no }}</view>
<view
class="info-item"
v-if="
!!orderDetail.exchange_time && ![9].includes(orderDetail.status)
"
>支付时间{{ orderDetail.exchange_time }}</view
>
<view class="info-item">下单时间{{ orderDetail.create_time }}</view>
<view class="info-item" style="margin-bottom: 24rpx"
>订单金额{{ orderDetail.order_amount }}</view
>
<!-- <view class="info-item" style="margin-bottom: 24rpx;">优惠金额{{ orderDetail.order_amount}}</view> -->
<view
class="flex flex-justify-between info-item"
style="margin-bottom: 24rpx"
v-if="[3, 4, 8].includes(orderDetail.status)"
>
<view>充值状态</view>
<view :style="{ color: stateColor[orderDetail.status] || '' }">{{
stateConfig[orderDetail.status]
}}</view>
</view>
</view>
<view class="line"></view>
</view>
<view class="btns" v-if="!isWaiting">
<view
class="btn pwd"
@click="pay(orderDetail)"
v-if="[1].includes(orderDetail.status)"
>立即付款</view
>
<view
class="btn pwd"
v-if="[4].includes(orderDetail.status)"
@click="goRefund"
>申请退款</view
>
<view class="btn back" @click="backIndex">返回首页</view>
</view>
</view>
</template>
<script setup>
import { onShow, onHide, onUnload } from "@dcloudio/uni-app";
import { onMounted, reactive, ref, unref } from "vue";
import { getOrderList, refundOrder,login } from "@/api/ycysp";
import { getQueryString } from "@/utils/utils";
import { stateConfig, stateColor } from "./config";
import usePay from "./hooks/usePay";
import useCode from "./hooks/useCode";
const id = ref("");
const orderNo = ref("");
const orderDetail = reactive({});
const isWaiting = ref(false)
// onshow h5
onShow(async () => {
const isPayBack = getQueryString("isPayBack");
if(isPayBack){
isWaiting.value = true
await sleep(3000)
isWaiting.value = false
}
const order_id = getQueryString("order_id");
const order_no = getQueryString("order_no");
if (!order_id && !order_no) return;
id.value = order_id;
orderNo.value = order_no;
getDetail();
});
const backIndex = () => {
uni.navigateTo({
url: `/pages/ycysp/home`,
});
};
function sleep(ms) {
return new Promise((resolve) => {
setTimeout(resolve, ms);
});
}
const goRefund = () => {
const { id } = orderDetail;
if (!id) {
uni.showToast({
icon: "none",
title: "无有效的订单id",
});
return;
}
const params = { id: id };
refundOrder({ params })
.then((res) => {
uni.showToast({
icon: "success",
title: "退款成功",
});
getDetail();
})
.catch((err) => {
console.log(err);
});
};
function maskPhoneNumber(phoneNumber) {
return phoneNumber.replace(/(\d{3})\d{4}(\d{4})/, "$1****$2");
}
function pay(orderData) {
const { order_no, notify_url,sign, plain_text } = orderData.pay_info;
const price = orderData.order_amount
const MerName = orderData.brandName
const { payFunc } = usePay();
payFunc({
order_no,
notify_url,
TranAmt: price,
MerName: MerName,
sign,
plain_text,
});
}
const getDetail = async () => {
const token = window.localStorage.getItem('token') || '';
if(!token){
const code = await useCode();
const {token} = await login({params:{code:code}});
window.localStorage.setItem('token',token);
}
const params = {
id: unref(id),
order_no: unref(orderNo),
};
getOrderList({ params }).then((res) => {
const { total, list } = res;
Object.assign(orderDetail, res.list[0]);
});
};
</script>
<style lang="scss" scoped>
.wait{
position: absolute;
top:50%;
left:50%;
transform: translate(-50%,-50%);
display: flex;
flex-direction: column;
align-items: center;
.wait-text{
font-weight: 400;
font-size: 28rpx;
color: #333333;
margin-top:20rpx;
}
}
.page {
background-color: #fafafa;
}
.line {
width: 662rpx;
height: 1rpx;
background-color: #e1e1f0;
transform: scaleY(0.5);
}
.wrapper {
width: 702rpx;
// height: 542rpx;
min-height: 442rpx;
background: #ffffff;
border-radius: 24rpx;
margin-top: 24rpx;
box-sizing: border-box;
padding: 32rpx 24rpx;
.pro-info {
// border-bottom: 1rpx solid #F0E1E1;
.title {
font-weight: 500;
font-size: 28rpx;
color: #333333;
margin-bottom: 28rpx;
}
.content {
margin-bottom: 24rpx;
font-weight: 400;
.pro-img {
display: block;
width: 90rpx;
height: 90rpx;
margin-right: 34rpx;
}
.text-over {
max-width: 80%;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
}
.name {
font-size: 24rpx;
color: #333333;
font-weight: 700;
}
.amount {
color: #333333;
font-weight: 400;
font-size: 24rpx;
text:nth-child(1) {
font-size: 20rpx;
}
text:nth-child(2) {
font-size: 22rpx;
}
}
.num {
font-weight: 400;
font-size: 24rpx;
color: #9e9e9e;
margin-bottom: 10rpx;
}
}
}
.order-info {
// border-bottom: 1rpx solid #F0E1E1;
.info-item {
font-weight: 400;
font-size: 22rpx;
color: #6c6c6c;
margin-top: 24rpx;
}
}
}
.btns {
margin-top: 146rpx;
display: flex;
.btn {
width: 312rpx;
height: 76rpx;
border-radius: 38rpx;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.pwd {
font-weight: 500;
font-size: 32rpx;
color: #ffffff;
background: linear-gradient(270deg, #2c91ff 0%, #9ad2ff 100%);
margin-right: 10rpx;
}
.back {
font-weight: 400;
font-size: 32rpx;
color: #b9c8c7;
border: 2rpx solid #b9c8c7;
}
}
</style>

217
src/pages/ycysp/pay.vue Normal file
View File

@ -0,0 +1,217 @@
<route lang="json5" type="page">
{
style: {
navigationStyle: "custom",
navigationBarTitleText: "支付确认",
navigationBarBackgroundColor: "#FFF",
},
}
</route>
<template>
<view
class="flex flex-col flex-items-center flex-justify-between page-wrapper"
>
<view class="info">
<view class="title-small">订单详情</view>
<view class="flex detail-info">
<view class="detail-img">
<image :src="orderDetail.main_image" mode="scaleToFill" />
</view>
<view class="detail-other">
<view class="flex flex-justify-between name-info">
<text>{{ orderDetail.product_name }}</text>
<text
><text style="font-size: 20rpx"></text
>{{ orderDetail.order_amount }}</text
>
</view>
<view class="flex flex-justify-between count-info">
<text>数量</text>
<text>x1</text>
</view>
</view>
</view>
<view class="line"></view>
<view class="order-info">
<view class="order-item"
>充值账户{{ orderDetail.account ? maskPhoneNumber(orderDetail.account) : '' }}</view
>
<view class="order-item">充值类型{{ orderDetail.product_type }}</view>
<!-- <view class="order-item">优惠金额{{ discount }}</view> -->
</view>
<view class="line"></view>
<!-- <view class="order-info">
<view class="count-item"> 1 应付总额<text style="text-decoration: line-through;">{{ detail.order_amount }}</text></view>
<view class="number-item">实际支付¥{{ detail.order_amount }}</view>
</view> -->
</view>
<view class="pay-btn" @click="goPay">立即支付</view>
</view>
</template>
<script setup>
import { onShow, onLoad } from "@dcloudio/uni-app";
import { ref, reactive, unref, onMounted, computed } from "vue";
import { createOrder, getOrderList } from "@/api/ycysp";
import usePay from "./hooks/usePay";
import config from './config';
const orderDetail = reactive({});
const _order_no = ref("");
onLoad((options) => {
const { order_no } = options;
_order_no.value = order_no;
});
const discount = computed(() => {
if (!unref(_product)) return 0;
return (
(Number(unref(_product).price) * 100 -
Number(unref(_product).sell_price) * 100) /
100
);
});
function maskPhoneNumber(phoneNumber) {
return phoneNumber.replace(/(\d{3})\d{4}(\d{4})/, "$1****$2");
}
const goPay = () => {
const { payFunc } = usePay();
const { order_no, notify_url, sign, plain_text } = orderDetail.pay_info;
payFunc({
order_no,
notify_url,
TranAmt: orderDetail.order_amount,
MerName: orderDetail.brandName,
sign,
plain_text,
});
};
onMounted(() => {
getDetail();
});
const getDetail = () => {
const params = {
order_no: unref(_order_no),
};
getOrderList({ params }).then((res) => {
const { total, list } = res;
Object.assign(orderDetail, res.list[0]);
});
};
</script>
<style scoped lang="scss">
.page-wrapper {
width: 100vw;
height: 100vh;
opacity: 1;
// background: linear-gradient(180deg, #FFFFFF 18%, #F3F3F3 40%, #FAFBFD 60%, #FAFBFD 61%, #FAFBFD 65%);
background: #fafbfd;
padding: 33rpx 0;
box-sizing: border-box;
}
.info {
width: 702rpx;
// height: 514rpx;
height: 374rpx;
border-radius: 24rpx;
opacity: 1;
background: #ffffff;
box-sizing: border-box;
padding: 32rpx 24rpx;
}
.line {
width: 100%;
height: 1rpx;
background: #e1e1f0;
transform: scaleY(0.5);
}
.title-small {
font-size: 28rpx;
font-weight: normal;
line-height: 32rpx;
letter-spacing: 0rpx;
color: #333333;
}
.detail-info {
padding: 24rpx 0;
}
.detail-img {
width: 92rpx;
height: 92rpx;
margin-right: 30rpx;
image {
width: 100%;
height: 100%;
}
}
.detail-other {
flex: 1;
font-weight: 400;
line-height: 32rpx;
letter-spacing: 0rpx;
.name-info {
font-size: 24rpx;
color: #333333;
}
.count-info {
font-size: 20rpx;
color: #9e9e9e;
margin-top: 12rpx;
}
}
.order-info {
padding-bottom: 24rpx;
}
.order-item {
font-size: 22rpx;
font-weight: normal;
line-height: 32rpx;
letter-spacing: 0rpx;
color: #6c6c6c;
margin-top: 24rpx;
}
.bottom {
font-weight: 400;
line-height: 32rpx;
text-align: right;
font-style: normal;
text-transform: none;
}
.count-item {
font-size: 22rpx;
color: #878787;
text-align: right;
margin-top: 16rpx;
}
.number-item {
font-size: 24rpx;
color: #2e92ff;
text-align: right;
margin-top: 16rpx;
}
.pay-btn {
width: 710rpx;
height: 88rpx;
/* 自动布局 */
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
padding: 12rpx 28rpx;
border-radius: 392rpx;
opacity: 1;
background: linear-gradient(270deg, #2c91ff 0%, #9ad2ff 100%);
box-sizing: border-box;
font-size: 32rpx;
font-weight: normal;
line-height: normal;
letter-spacing: 0em;
color: #ffffff;
}
</style>

Binary file not shown.

Before

Width:  |  Height:  |  Size: 26 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 32 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 268 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.4 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 9.0 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 44 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 17 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 377 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

View File

Before

Width:  |  Height:  |  Size: 11 KiB

After

Width:  |  Height:  |  Size: 11 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" fill="none" version="1.1" width="13" height="13" viewBox="0 0 13 13"><g><ellipse cx="6.5" cy="6.5" rx="6.5" ry="6.5" fill="#FFFFFF" fill-opacity="1"/><ellipse cx="6.5" cy="6.5" rx="6.199999988079071" ry="6.199999988079071" fill-opacity="0" stroke-opacity="1" stroke="#4A98FB" fill="none" stroke-width="0.6000000238418579"/></g></svg>

After

Width:  |  Height:  |  Size: 416 B

BIN
src/static/ycysp/cloud.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

1
src/static/ycysp/gou.svg Normal file
View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" fill="none" version="1.1" width="7" height="5" viewBox="0 0 7 5"><g><path d="M6.85467,0.16026C6.65747,-0.049906,6.33416,-0.0538585,6.13257,0.151333L2.65289,3.69208L0.855769,1.81926C0.669146,1.62458,0.356251,1.63538,0.156706,1.84316C-0.0428302,2.05098,-0.0532221,2.37714,0.133625,2.57177L2.33038,4.86098C2.51696,5.05539,2.82989,5.04459,3.0294,4.83682C3.07511,4.78922,3.10934,4.73525,3.13501,4.67819C3.15477,4.66321,3.17447,4.649,3.19236,4.63059L6.84614,0.91282C7.048,0.707426,7.05164,0.370686,6.85467,0.16026L6.85467,0.16026Z" fill="#4B99FB" fill-opacity="1"/></g></svg>

After

Width:  |  Height:  |  Size: 652 B

BIN
src/static/ycysp/mgtv.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

BIN
src/static/ycysp/qqyy.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

BIN
src/static/ycysp/txsp.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

BIN
src/static/ycysp/xmly.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.8 KiB

BIN
src/static/ycysp/yksp.webp Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.2 KiB

View File

@ -1,8 +1,10 @@
//邮储银行(奶茶)专用的
import useCode from "../pages/ycnc/hooks/useCode";
import {login} from '../api/ycnc'
//邮储银行(音视频)专用的
import useCode from "../pages/ycysp/hooks/useCode";
import {login} from '../api/ycysp'
//邮储音视频baseURL专用
const baseURL = '/ycav'
function http(options) {
uni.showLoading({
@ -10,13 +12,13 @@ function http(options) {
mask: true,
});
return new Promise((resolve, reject) => {
//邮储音视频添加前缀
const newOptions = {...options,url:baseURL + options.url}
uni.request({
...options,
...newOptions,
header: {
//邮储奶茶
'content-type': 'application/x-www-form-urlencoded',
//邮储音视频
'content-type': 'application/json',
},
responseType: 'json',
success: async (res) => {
@ -39,8 +41,8 @@ function http(options) {
const params = {code:authCode};
const {token} = await login({params});
window.localStorage.setItem('token',token);
//邮储奶茶
http(options).then(_res => {
//邮储音视频专用的
http(newOptions).then(_res => {
resolve(_res)
})
}else{

8
uni-pages.d.ts vendored
View File

@ -4,10 +4,10 @@
// Generated by vite-plugin-uni-pages
interface NavigateToOptions {
url: "/pages/ycnc/index" |
"/pages/ycnc/detail" |
"/pages/ycnc/order" |
"/pages/ycnc/orderDetail";
url: "/pages/ycysp/home" |
"/pages/ycysp/order" |
"/pages/ycysp/orderDetail" |
"/pages/ycysp/pay";
}
interface RedirectToOptions extends NavigateToOptions {}