Compare commits

...

3 Commits

Author SHA1 Message Date
d897dc1079 2025-10-17
商城详情页
收获地址
2025-10-17 14:10:49 +08:00
980cc496ba 2025-10-16
收货地址页面40%
2025-10-16 17:23:57 +08:00
28e36b5625 2025-10-16
商城主页
商城分类
我的
商品详情40%
2025-10-16 17:07:47 +08:00
34 changed files with 1305 additions and 159 deletions

View File

@@ -2,8 +2,12 @@ import {
http http
} from "../util/api"; } from "../util/api";
const baseUrl = "http://192.168.0.26:3000"
export const mallAPI = { export const mallAPI = {
getMallList: (params) => http.get('/mall', params) getMallList: (params) => http.get(baseUrl + '/api/products', params),
getCategory: () => http.get(baseUrl + '/api/category'),
getMallDetail: (id) => http.get(baseUrl + '/api/products/' + id)
} }
export default { export default {

View File

@@ -2,7 +2,8 @@ import {
http http
} from "../util/api" } from "../util/api"
const baseURL = "http://192.168.0.12:3005/chat" // const baseURL = "http://192.168.0.12:3005/chat"
const baseURL = "http://192.168.0.15:3007"
// 项目相关API // 项目相关API
export const messageAPI = { export const messageAPI = {

View File

@@ -58,41 +58,58 @@
} }
}, },
{ {
"path" : "pages/mall/mall", "path": "pages/mall/mall",
"style" : "style": {
{ "navigationBarTitleText": "商城",
"navigationBarTitleText" : "商城",
"navigationStyle": "custom" "navigationStyle": "custom"
} }
}, },
{ {
"path" : "pages/message/chat", "path": "pages/message/chat",
"style" : "style": {
{ "navigationBarTitleText": "聊天页面",
"navigationBarTitleText" : "聊天页面",
"navigationStyle": "custom", "navigationStyle": "custom",
"enablePullDownRefresh": true "enablePullDownRefresh": true
} }
}, },
{ {
"path" : "pages/message/systemMessage", "path": "pages/message/systemMessage",
"style" : "style": {
{ "navigationBarTitleText": "系统通知",
"navigationBarTitleText" : "系统通知",
"navigationBarBackgroundColor": "#F0F3FF", "navigationBarBackgroundColor": "#F0F3FF",
"navigationBarTextStyle": "#000", "navigationBarTextStyle": "#000",
"app-plus": { "app-plus": {
"titleNView": { "titleNView": {
"buttons": [ "buttons": [{
{
"type": "menu", "type": "menu",
"color": "#000" "color": "#000"
} }
] ]
} }
} }
} }
},
{
"path": "pages/home/mallDetail",
"style": {
"navigationBarTitleText": "商品详情",
"navigationStyle": "custom"
}
},
{
"path": "pages/home/mallCategory",
"style": {
"navigationBarTitleText": "商品分类",
"navigationStyle": "custom"
}
},
{
"path": "pages/my/shippingAddress",
"style": {
"navigationBarTitleText": "收获地址",
"navigationStyle": "custom"
}
} }
], ],
"globalStyle": { "globalStyle": {

View File

@@ -1,174 +1,339 @@
<template> <template>
<view class="message-container"> <view class="message-container">
<!-- 固定背景的容器 --> <scroll-view scroll-y="true" style="height: 100%;">
<view class="background-container"></view> <view class="header-search">
<u-search :action-style="searchBtn" shape="square" bg-color="#CADBFF" placeholder="输入商品名称或商品种类"
<!-- 可滚动的内容区域 --> v-model="keyword"></u-search>
<view class="content-container" :style="'height:'+height+'px'"> </view>
<u-swiper class="u-m-t-60" :list="list" name="url" height="324"></u-swiper>
<view class="menu-list"> <view class="menu-list">
<swiper class="swiper" @change="changeMenu"> <view class="menu-item" @click="handleCategory">
<swiper-item> <u-image src="/static/shop/01.png" :width="iconWidth" mode="widthFix"></u-image>
<u-grid :col="4" :border="false"> <view class="menu-text">
<u-grid-item bg-color="transparent"> 商品分类
<u-icon name="/static/home/1.png" :size="120"></u-icon>
<view class="grid-text">精选直播</view>
</u-grid-item>
<u-grid-item bg-color="transparent">
<u-icon name="/static/home/2.png" :size="120"></u-icon>
<view class="grid-text">订单数据</view>
</u-grid-item>
<u-grid-item bg-color="transparent" @click="openMall">
<u-icon name="/static/home/3.png" :size="120"></u-icon>
<view class="grid-text">商城好物</view>
</u-grid-item>
<u-grid-item bg-color="transparent">
<u-icon name="/static/home/4.png" :size="120"></u-icon>
<view class="grid-text">获得积分</view>
</u-grid-item>
</u-grid>
</swiper-item>
<swiper-item>
<u-grid :col="4" :border="false">
<u-grid-item bg-color="transparent">
<u-icon name="/static/home/5.png" :size="120"></u-icon>
<view class="grid-text">微信读书</view>
</u-grid-item>
</u-grid>
</swiper-item>
</swiper>
<view class="indicator-dots">
<view class="indicator-dots-item" :class="current==0?'indicator-dots-active':''">
</view> </view>
<view class="indicator-dots-item" :class="current==1?'indicator-dots-active':''"> </view>
<view class="menu-item">
<u-image src="/static/shop/02.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
观看直播
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/03.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
违规通告
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/04.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
百亿补贴
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/05.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
限时秒杀
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/06.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
会员专区
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/07.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
商品收藏
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/08.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
充值缴费
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/09.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
资质认证
</view>
</view>
<view class="menu-item">
<u-image src="/static/shop/10.png" :width="iconWidth" mode="widthFix"></u-image>
<view class="menu-text">
联系客服
</view> </view>
</view> </view>
</view> </view>
</view>
<Tabbar id="tabbarId"></Tabbar>
<view class="mall-list">
<u-waterfall v-model="mallList" ref="mallListRef">
<template v-slot:left="{leftList}">
<view class="mall-item u-m-r-10" v-for="(item, index) in leftList" :key="index"
@click="handleCheck(item)">
<u-lazy-load threshold="-450" border-radius="10" :image="getImageUrl(item.image)"
:index="index"></u-lazy-load>
<view class="mall-title u-m-l-5 u-m-r-5">
{{item.name}}
</view>
<del class="u-m-l-5 u-m-r-5" style="white-space: nowrap;">{{item.price}}</del>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="RDType(item.payment_methods)">
<image src="/static/icon/rongdou.png" class="icon" mode=""></image>
{{item.rongdou_price}}
</view>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="pointsType(item.payment_methods)">
<u-icon name="integral"></u-icon>
{{item.points_price}}
</view>
<view class="mall-tag u-m-l-5 u-m-r-5">
<view class="mall-tag-text" v-if="RDType(item.payment_methods)">
融豆
</view>
<view class="mall-tag-owner" v-if="pointsType(item.payment_methods)">
积分
</view>
</view>
<view class="mall-shop u-m-l-5 u-m-r-5">
{{item.category}}
</view>
</view>
</template>
<template v-slot:right="{rightList}">
<view class="mall-item u-m-l-10" v-for="(item, index) in rightList" :key="index"
@click="handleCheck(item)">
<u-lazy-load threshold="-450" border-radius="10" :image="getImageUrl(item.image)"
:index="index"></u-lazy-load>
<view class="mall-title u-m-l-5 u-m-r-5">
{{item.name}}
</view>
<del class="u-m-l-5 u-m-r-5" style="white-space: nowrap;">{{item.price}}</del>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="RDType(item.payment_methods)">
<image src="/static/icon/rongdou.png" class="icon" mode=""></image>
{{item.rongdou_price}}
</view>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="pointsType(item.payment_methods)">
<u-icon name="integral"></u-icon>
{{item.points_price}}
</view>
<view class="mall-tag u-m-l-5 u-m-r-5">
<view class="mall-tag-text" v-if="RDType(item.payment_methods)">
融豆
</view>
<view class="mall-tag-owner" v-if="pointsType(item.payment_methods)">
积分
</view>
</view>
<view class="mall-shop u-m-l-5 u-m-r-5">
{{item.category}}
</view>
</view>
</template>
</u-waterfall>
<u-loadmore margin-top="20" :status="loadStatus"></u-loadmore>
</view>
<view class="box-div"></view>
</scroll-view>
</view> </view>
<Tabbar id="tabbarId"></Tabbar>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, computed, getCurrentInstance } from 'vue';
import { import {
onReady ref, onMounted
} from '@dcloudio/uni-app'; } from 'vue';
import { onReady as onUniReady } from '@dcloudio/uni-app'; import { mallAPI } from '../../api/mall';
const instance = getCurrentInstance(); import { getImageUrl } from '../../util/common';
const height = ref(0) const keyword = ref('')
const searchBtn = {
background: 'linear-gradient(100.84deg, #7B99FF 0%, #002CC2 99.52%)',
width: '128rpx',
borderRadius: '198rpx',
color: '#fff',
fontSize: '28rpx',
boxShadow: '0px 4px 4px 0px #FFFFFF40 inset',
}
const iconWidth = "65%"
const loadHeight = () => { const handleCategory = () => {
uni.getSystemInfo({ uni.navigateTo({
success(res) { url: '/pages/home/mallCategory'
let screenHeight = res.screenHeight })
uni.createSelectorQuery().in(instance.proxy).select("#tabbarId").boundingClientRect((data : any) => { }
height.value = screenHeight - data.height
}).exec()
const params = ref({
keyword: '',
page: 1,
limit: 5,
category: ''
})
const mallList = ref([])
const mallListRef = ref()
const loadStatus = ref('loadmore') // nomore
const maxPage = ref(1)
// 积分兑换
const pointsType = (val : any) => {
if (val && val.indexOf("points") >= 0) return true
return false
}
// 融豆兑换
const RDType = (val : any) => {
if (val && val.indexOf("rongdou") >= 0) return true
return false
}
// 加载数据
const loadMallData = () => {
if (loadStatus.value == 'nomore') return
mallAPI.getMallList(params.value).then((res) => {
console.log(res.data);
mallList.value = mallList.value.concat(res.data.products)
maxPage.value = res.data.pagination.pages
params.value.page++
if (maxPage.value < params.value.page) {
loadStatus.value = 'nomore'
} }
}) })
} }
const list = [ const handleCheck = (item : any) => {
{ url: '/static/banner/1.png' },
{ url: '/static/banner/2.png' },
{ url: '/static/banner/3.png' },
]
const current = ref(0)
const changeMenu = (val : any) => {
current.value = val.detail.current
}
const openMall = () => {
uni.navigateTo({ uni.navigateTo({
url: '/pages/mall/mall' url: '/pages/home/mallDetail?id=' + item.id
}) })
} }
onUniReady(() => { onMounted(() => {
loadHeight() loadMallData()
}) })
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
.message-container { .message-container {
width: 100%; width: 100%;
position: fixed; height: 100vh;
top: 0; padding: 60rpx 20rpx;
left: 0; background: #79AFFF;
height: 100%;
overflow: hidden;
/* 防止容器本身滚动 */
/* 固定背景 */ .header-search {
.background-container { .search-btn {
position: absolute; ;
top: 0; box-shadow: 0px -4px 4px 0px #00000040 inset;
left: 0;
width: 100%; }
height: 100%;
background: linear-gradient(180deg, #2F75F9 0%, #F0F3FF 34.13%);
z-index: 1;
/* 确保背景在内容下方 */
} }
/* 内容滚动区域 */ .menu-list {
.content-container { background: #FFFFFF80;
position: relative; margin-top: 32rpx;
z-index: 2; padding: 20rpx 0 0 0;
/* 确保内容在背景上方 */ border-radius: 20rpx;
width: 100%; display: flex;
overflow-y: scroll; flex-wrap: wrap;
height: 100%;
// padding: 32rpx;
.menu-list { .menu-item {
border-top-left-radius: 32rpx; display: flex;
border-top-right-radius: 32rpx; flex-direction: column;
box-shadow: 0rpx -8rpx 20rpx 0rpx #628AC080; align-items: center;
padding-bottom: 20rpx; width: 20%;
margin-bottom: 20rpx;
.swiper { .menu-text {
height: 200rpx; font-family: Work Sans;
font-weight: 400;
font-style: Regular;
font-size: 24rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
color: #2B2B2B;
margin-top: 10rpx;
}
}
}
.grid-text {
font-family: Work Sans; // 商城列表
font-weight: 400; .mall-list {
font-style: Regular; margin-top: 20rpx;
font-size: 28rpx;
leading-trim: NONE; .mall-item {
line-height: 32rpx; border-radius: 16rpx;
letter-spacing: -2%; background: #F0F5FF;
text-align: center; position: relative;
margin-top: 20rpx;
box-shadow: 0px 4px 4px 0px #00000040;
padding-bottom: 10rpx;
.mall-title {
font-size: 30rpx;
margin-top: 10rpx;
color: $u-main-color;
}
.mall-price {
font-size: 30rpx;
color: $u-type-error;
margin-top: 10rpx;
.icon {
height: 30rpx;
width: 30rpx;
} }
} }
.indicator-dots {
// margin-top: 40rpx; .mall-tag {
display: flex; display: flex;
justify-content: center; margin-top: 5px;
align-items: center;
.mall-tag-owner {
background-color: $u-type-error;
color: #FFFFFF;
display: flex;
align-items: center;
padding: 4rpx 14rpx;
border-radius: 50rpx;
font-size: 20rpx;
line-height: 1;
}
.mall-tag-text {
margin-right: 10px;
border: 1px solid $u-type-primary;
color: $u-type-primary;
border-radius: 50rpx;
line-height: 1;
padding: 4rpx 14rpx;
display: flex;
align-items: center;
border-radius: 50rpx;
font-size: 20rpx;
}
} }
.indicator-dots-item {
background-color: $u-tips-color;
height: 6px; .mall-shop {
width: 6px; font-size: 22rpx;
border-radius: 10px; color: $u-tips-color;
margin: 0 3px; margin-top: 5px;
} }
.indicator-dots-active {
background-color: $u-type-primary;
}
} }
} }
.box-div {
padding: 30rpx 0rpx;
}
} }
</style> </style>

319
pages/home/mallCategory.vue Normal file
View File

@@ -0,0 +1,319 @@
<template>
<view class="category-container">
<u-navbar id="uNavbarId" title="商品详情" :background="{background: 'transparent' }" :border-bottom="false"
back-icon-color="#000" title-color="#000">
<template v-slot:right>
<image class="collection" src="/static/icon/Settings.png" mode=""></image>
</template>
</u-navbar>
<scroll-view scroll-y="true" :style="'height:'+scrollHeight+'px'">
<view class="scroll-view">
<view class="menu-level1">
<view class="text" :class="{'active': currentCate==index}" v-for="(item, index) in categoryList"
@click="handleChangeCategory(index)">
{{item.name}}
</view>
</view>
<view class="menu-level2" v-if="categoryList[currentCate]?.relative.length!=0">
<view class="menu-item" v-for="item in categoryList[currentCate]?.relative">
<u-image v-if="item.img" :fade="false" :src="getImageUrl(item.img)" :width="iconWidth"
mode="widthFix"></u-image>
<view class="menu-text">
{{item.name}}
</view>
</view>
</view>
<view class="mall-list">
<u-waterfall v-model="mallList" ref="mallListRef">
<template v-slot:left="{leftList}">
<view class="mall-item u-m-r-10" v-for="(item, index) in leftList" :key="index"
@click="handleCheck(item)">
<u-lazy-load threshold="-450" border-radius="10" :image="getImageUrl(item.image)"
:index="index"></u-lazy-load>
<view class="mall-title u-m-l-5 u-m-r-5">
{{item.name}}
</view>
<del class="u-m-l-5 u-m-r-5" style="white-space: nowrap;">{{item.price}}</del>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="RDType(item.payment_methods)">
<image src="/static/icon/rongdou.png" class="icon" mode=""></image>
{{item.rongdou_price}}
</view>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="pointsType(item.payment_methods)">
<u-icon name="integral"></u-icon>
{{item.points_price}}
</view>
<view class="mall-tag u-m-l-5 u-m-r-5">
<view class="mall-tag-text" v-if="RDType(item.payment_methods)">
融豆
</view>
<view class="mall-tag-owner" v-if="pointsType(item.payment_methods)">
积分
</view>
</view>
<view class="mall-shop u-m-l-5 u-m-r-5">
{{item.category}}
</view>
</view>
</template>
<template v-slot:right="{rightList}">
<view class="mall-item u-m-l-10" v-for="(item, index) in rightList" :key="index"
@click="handleCheck(item)">
<u-lazy-load threshold="-450" border-radius="10" :image="getImageUrl(item.image)"
:index="index"></u-lazy-load>
<view class="mall-title u-m-l-5 u-m-r-5">
{{item.name}}
</view>
<del class="u-m-l-5 u-m-r-5" style="white-space: nowrap;">{{item.price}}</del>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="RDType(item.payment_methods)">
<image src="/static/icon/rongdou.png" class="icon" mode=""></image>
{{item.rongdou_price}}
</view>
<view class="mall-price u-m-l-5 u-m-r-5" v-if="pointsType(item.payment_methods)">
<u-icon name="integral"></u-icon>
{{item.points_price}}
</view>
<view class="mall-tag u-m-l-5 u-m-r-5">
<view class="mall-tag-text" v-if="RDType(item.payment_methods)">
融豆
</view>
<view class="mall-tag-owner" v-if="pointsType(item.payment_methods)">
积分
</view>
</view>
<view class="mall-shop u-m-l-5 u-m-r-5">
{{item.category}}
</view>
</view>
</template>
</u-waterfall>
<u-loadmore margin-top="20" :status="loadStatus"></u-loadmore>
</view>
</view>
</scroll-view>
</view>
</template>
<script setup lang="ts">
import {
ref, onMounted, getCurrentInstance
} from 'vue';
import { mallAPI } from '../../api/mall';
import { getImageUrl } from '../../util/common';
const instance = getCurrentInstance();
const currentCate = ref(0)
const categoryList = ref([])
const iconWidth = "65%"
const loadCategory = () => {
mallAPI.getCategory().then(res => {
categoryList.value = res.data
})
}
const handleChangeCategory = (index) => {
currentCate.value = index
}
const params = ref({
keyword: '',
page: 1,
limit: 5,
category: ''
})
const mallList = ref([])
const mallListRef = ref()
const loadStatus = ref('loadmore') // nomore
const maxPage = ref(1)
// 积分兑换
const pointsType = (val : any) => {
if (val && val.indexOf("points") >= 0) return true
return false
}
// 融豆兑换
const RDType = (val : any) => {
if (val && val.indexOf("rongdou") >= 0) return true
return false
}
// 加载数据
const loadMallData = () => {
if (loadStatus.value == 'nomore') return
mallAPI.getMallList(params.value).then((res) => {
mallList.value = mallList.value.concat(res.data.products)
maxPage.value = res.data.pagination.pages
params.value.page++
if (maxPage.value < params.value.page) {
loadStatus.value = 'nomore'
}
})
}
const handleCheck = (item : any) => {
uni.navigateTo({
url: '/pages/home/mallDetail?id=' + item.id
})
}
const scrollHeight = ref(0)
const loadHeight = () => {
uni.getSystemInfo({
success(res) {
let screenHeight = res.screenHeight
uni.createSelectorQuery().in(instance.proxy).select("#uNavbarId").boundingClientRect((data : any) => {
scrollHeight.value = screenHeight - data.height
}).exec()
}
})
}
onMounted(() => {
loadHeight()
loadCategory()
loadMallData()
})
</script>
<style lang="scss" scoped>
.category-container {
width: 100%;
height: 100vh;
background: #B9D5FF;
.collection {
width: 48rpx;
height: 48rpx;
margin-right: 24rpx;
}
.scroll-view {
.menu-level1 {
display: flex;
white-space: nowrap;
overflow: hidden;
overflow-x: scroll;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
font-family: Work Sans;
font-weight: 600;
font-style: SemiBold;
.text {
margin: 22rpx;
font-family: Work Sans;
font-weight: 600;
font-style: SemiBold;
font-size: 14px;
color: #F5F8FF;
}
.active {
font-size: 16px;
color: var(--wz, #2938E0);
}
}
.menu-level2 {
background: #F5F8FF;
// background: #FFFFFF80;
padding: 20rpx 0 0 0;
border-radius: 20rpx;
display: flex;
flex-wrap: wrap;
.menu-item {
display: flex;
flex-direction: column;
align-items: center;
width: 20%;
margin-bottom: 20rpx;
.menu-text {
font-family: Work Sans;
font-weight: 400;
font-style: Regular;
font-size: 24rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
color: #2B2B2B;
margin-top: 10rpx;
}
}
}
// 商城列表
.mall-list {
margin-top: 20rpx;
padding: 0 20rpx;
.mall-item {
border-radius: 16rpx;
background: #F0F5FF;
position: relative;
margin-top: 20rpx;
box-shadow: 0px 4px 4px 0px #00000040;
padding-bottom: 10rpx;
.mall-title {
font-size: 30rpx;
margin-top: 10rpx;
color: $u-main-color;
}
.mall-price {
font-size: 30rpx;
color: $u-type-error;
margin-top: 10rpx;
.icon {
height: 30rpx;
width: 30rpx;
}
}
.mall-tag {
display: flex;
margin-top: 5px;
.mall-tag-owner {
background-color: $u-type-error;
color: #FFFFFF;
display: flex;
align-items: center;
padding: 4rpx 14rpx;
border-radius: 50rpx;
font-size: 20rpx;
line-height: 1;
}
.mall-tag-text {
margin-right: 10px;
border: 1px solid $u-type-primary;
color: $u-type-primary;
border-radius: 50rpx;
line-height: 1;
padding: 4rpx 14rpx;
display: flex;
align-items: center;
border-radius: 50rpx;
font-size: 20rpx;
}
}
.mall-shop {
font-size: 22rpx;
color: $u-tips-color;
margin-top: 5px;
}
}
}
}
}
</style>

161
pages/home/mallDetail.vue Normal file
View File

@@ -0,0 +1,161 @@
<template>
<view class="detail-container">
<u-navbar id="uNavbarId" title="商品详情" :background="{background: 'transparent' }" :border-bottom="false"
back-icon-color="#000" title-color="#000">
<template v-slot:right>
<image class="collection" src="/static/icon/Settings.png" mode=""></image>
</template>
</u-navbar>
<scroll-view scroll-y="true" :style="'height:'+scrollHeight+'px'">
111
</scroll-view>
<view class="bottom-view" id="bottomViewId">
<view class="icon-btn">
<view class="item">
<u-image width="100%" :fade="false" src="/static/mall/Home.png" mode="widthFix"></u-image>
店铺
</view>
<view class="item">
<u-image width="100%" :fade="false" src="/static/mall/Twitch.png" mode="widthFix"></u-image>
客服
</view>
<view class="item">
<u-image width="100%" :fade="false" src="/static/mall/Star.png" mode="widthFix"></u-image>
收藏
</view>
</view>
<view class="text-btn">
<u-button class="add-car common" :hair-line="false" hover-class="none">加入购物车</u-button>
<u-button class="buy common" :hair-line="false" hover-class="none">领券购买</u-button>
</view>
</view>
</view>
</template>
<script setup lang="ts">
import { onMounted, ref, getCurrentInstance } from 'vue';
import {
onLoad
} from '@dcloudio/uni-app';
import { mallAPI } from '../../api/mall';
const instance = getCurrentInstance();
const scrollHeight = ref(0)
const loadHeight = () => {
uni.getSystemInfo({
success(res) {
let screenHeight = res.screenHeight
uni.createSelectorQuery().in(instance.proxy).select("#uNavbarId").boundingClientRect((data : any) => {
scrollHeight.value = screenHeight - data.height
}).exec()
uni.createSelectorQuery().in(instance.proxy).select("#bottomViewId").boundingClientRect((data : any) => {
scrollHeight.value = scrollHeight.value - data.height
}).exec()
}
})
}
const dataId = ref()
const loadData = () => {
mallAPI.getMallDetail(dataId.value).then(res => {
console.log(res);
})
}
onLoad((val) => {
dataId.value = val.id
})
onMounted(() => {
loadHeight()
loadData()
})
</script>
<style scoped lang="scss">
.detail-container {
width: 100%;
height: 100vh;
background: linear-gradient(180deg, #2F75F9 0%, #F0F3FF 34.13%);
.collection {
width: 48rpx;
height: 48rpx;
margin-right: 24rpx;
}
.bottom-view {
position: absolute;
bottom: 0;
width: 100%;
height: 116rpx;
background: #F5F8FF;
padding: 0 29rpx;
display: flex;
flex-wrap: nowrap;
justify-content: space-between;
align-items: center;
.icon-btn {
display: flex;
width: 228rpx;
width: 30%;
flex-wrap: nowrap;
.item {
margin-right: 20rpx;
white-space: nowrap;
font-family: Work Sans;
font-weight: 400;
font-style: Regular;
font-size: 26rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
}
}
.text-btn {
display: flex;
flex: 1;
justify-content: flex-end;
.common {
border: none;
width: 200rpx;
height: 70rpx;
margin: 0;
color: #fff;
font-family: Work Sans;
font-weight: 400;
font-style: Regular;
font-size: 26rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
}
.buy {
background: #6287FF;
border-top-right-radius: 12rpx;
border-bottom-right-radius: 12rpx;
border-top-left-radius: 0;
border-bottom-left-radius: 0;
}
.add-car {
background: #A8BCFF;
border-top-right-radius: 0rpx;
border-bottom-right-radius: 0rpx;
border-top-left-radius: 12rpx;
border-bottom-left-radius: 12rpx;
}
}
}
}
</style>

View File

@@ -7,7 +7,7 @@
<view class="header-avatar"> <view class="header-avatar">
<u-avatar :src="getImageUrl(user.avatar)"></u-avatar> <u-avatar :src="getImageUrl(user.avatar)"></u-avatar>
</view> </view>
<view class="header-username u-m-l-10"> <view class="header-username u-m-l-20">
{{user.real_name}} {{user.real_name}}
</view> </view>
</view> </view>
@@ -15,7 +15,7 @@
<image src="/static/icon/Settings.png" style="width: 100%;height: 100%;" mode=""></image> <image src="/static/icon/Settings.png" style="width: 100%;height: 100%;" mode=""></image>
</view> </view>
</view> </view>
<view class="statistics"> <view class="statistics">
<view class="item"> <view class="item">
<view class="count"> <view class="count">
@@ -51,7 +51,89 @@
</view> </view>
</view> </view>
<u-button @click="loginOut">退出登录</u-button> <view class="open-vip">
<view class="text-vip">
<view class="title">
vip专享
</view>
<view class="sub-title">
立享价值1000元权益
</view>
</view>
<u-button class="btn-vip" hover-class="none" :hair-line="false">298元开通</u-button>
</view>
<view class="white-block">
<view class="item">
<view class="title">
会员权益
</view>
<view class="sub-title">
领取升级福利
</view>
</view>
<view class="item">
<view class="title">
积分商城
</view>
<view class="sub-title">
赚积分享福利
</view>
</view>
</view>
<view class="menu-list">
<view class="menu-item">
<u-image src="/static/my/01.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
领券中心
</view>
</view>
<view class="menu-item">
<u-image src="/static/my/02.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
积分兑换
</view>
</view>
<view class="menu-item">
<u-image src="/static/my/03.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
我的客户
</view>
</view>
<view class="menu-item">
<u-image src="/static/my/04.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
常见问题
</view>
</view>
<view class="menu-item">
<u-image src="/static/my/05.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
充值中心
</view>
</view>
<view class="menu-item" @click="handleShippingAddress">
<u-image src="/static/my/06.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
收货地址
</view>
</view>
<view class="menu-item">
<u-image src="/static/my/07.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
我的订单
</view>
</view>
<view class="menu-item">
<u-image src="/static/my/08.png" width="40%" mode="widthFix"></u-image>
<view class="menu-text">
成为代理
</view>
</view>
</view>
<u-button class="u-m-t-20" @click="loginOut">退出登录</u-button>
</view> </view>
</scroll-view> </scroll-view>
</view> </view>
@@ -65,6 +147,12 @@
const user = ref() const user = ref()
const handleShippingAddress = () => {
uni.navigateTo({
url: '/pages/my/shippingAddress'
})
}
onMounted(() => { onMounted(() => {
user.value = uni.getStorageSync("user") user.value = uni.getStorageSync("user")
console.log(user.value); console.log(user.value);
@@ -92,10 +180,11 @@
justify-content: flex-end; justify-content: flex-end;
.header-user { .header-user {
padding: 20rpx 10rpx;
flex: 1; flex: 1;
display: flex; display: flex;
align-items: center; align-items: center;
height: 170rpx;
padding-left: 60rpx;
} }
.header-setting { .header-setting {
@@ -103,17 +192,139 @@
height: 52rpx; height: 52rpx;
} }
} }
.statistics{ .statistics {
display: flex; display: flex;
.item{ font-family: SF Pro;
font-weight: 510;
font-style: Medium;
font-size: 32rpx;
leading-trim: NONE;
line-height: 48rpx;
letter-spacing: 0%;
margin: 40rpx 0 20rpx;
.item {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
justify-content: center; justify-content: center;
align-items: center; align-items: center;
width: 25%; width: 25%;
border: 1rpx solid #000;
}
}
.open-vip {
display: flex;
justify-content: space-between;
background: #2B2B2B;
align-items: center;
padding: 24rpx 0;
border-top-left-radius: 24rpx;
border-top-right-radius: 24rpx;
.text-vip {
margin-left: 32rpx;
display: flex;
flex-direction: column;
color: #FFDD9E;
justify-content: flex-start;
.title {
font-family: SF Pro;
font-weight: 700;
font-style: Bold;
font-size: 32rpx;
leading-trim: NONE;
line-height: 48rpx;
letter-spacing: 0%;
}
.sub-title {
font-family: SF Pro;
font-weight: 274;
font-style: Light;
font-size: 32rpx;
leading-trim: NONE;
line-height: 48rpx;
letter-spacing: 0%;
}
}
.btn-vip {
margin-right: 42rpx;
width: 180rpx;
height: 56rpx;
border: none;
border-radius: 198rpx;
background: linear-gradient(90deg, #FFCC5E 0%, #FFE9BA 47.27%, #FFCC5E 100%);
}
}
.white-block {
margin-top: 44rpx;
display: flex;
justify-content: space-between;
.item {
width: 48%;
background: #F5F8FF;
display: flex;
flex-direction: column;
padding: 30rpx 32rpx;
border-radius: 24rpx;
.title {
font-family: SF Pro;
font-weight: 700;
font-style: Bold;
font-size: 32rpx;
leading-trim: NONE;
line-height: 48rpx;
letter-spacing: 0%;
}
.sub-title {
margin-top: 14rpx;
font-family: Work Sans;
font-weight: 400;
font-size: 26rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
color: #8C8C8C;
}
}
}
.menu-list {
background: #F5F8FF;
margin-top: 20rpx;
padding: 20rpx 0 0 0;
border-radius: 24rpx;
display: flex;
flex-wrap: wrap;
.menu-item {
display: flex;
flex-direction: column;
align-items: center;
width: 25%;
margin-bottom: 20rpx;
.menu-text {
font-family: Work Sans;
font-weight: 400;
font-style: Regular;
font-size: 20rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
}
} }
} }
} }

View File

@@ -0,0 +1,268 @@
<template>
<view class="shipping-container">
<u-navbar id="uNavbarId" back-text="收货地址" :back-text-style="navBarTitle"
:background="{background: 'transparent' }" :border-bottom="false" back-icon-color="#000" title-color="#000">
<template v-slot:right>
<view class="right-menu">
<view @click="handleManage">
{{isManage?'取消':'管理'}}
</view>
<view v-if="isManage" class="u-m-l-10 del-text" @click="showDelModel">
删除
</view>
<view class="u-m-l-10 add-text">
新增地址
</view>
</view>
</template>
</u-navbar>
<scroll-view scroll-y="true" :style="'height:'+scrollHeight+'px'" @scrolltolower="loadData">
<view class="data-list">
<view class="data-item" v-for="item in dataList" @click="handleClick(item)">
<view class="item-text">
<view class="address u-m-b-18">
{{item.province + item.city + item.district}}
</view>
<view class="specific-address u-m-b-18">
{{item.detailed_address}}
</view>
<view class="info">
<div class="u-m-r-20">{{item.receiver_name}}</div>
<div class="u-m-r-20">{{item.receiver_phone}}</div>
<span v-if="item.is_default" class="u-m-r-10">默认</span>
<span class="u-m-r-10">{{item.label}}</span>
</view>
</view>
<view class="item-icon">
<image v-if="!isManage" style="width: 100%;height: 100%;" src="/static/icon/Edit.png" mode="">
</image>
<template v-else>
<image v-if="!isIncludeId(item.id)" style="width: 100%;height: 100%;"
src="/static/icon/chose.png" mode=""></image>
<image v-else style="width: 100%;height: 100%;" src="/static/icon/chose-active.png" mode="">
</image>
</template>
</view>
</view>
<u-loadmore :status="status" />
</view>
</scroll-view>
<!-- 确认是否删除 -->
<u-modal v-model="showDel" content="是否删除选中的地址" :show-cancel-button="true" @confirm="handleDel"></u-modal>
</view>
</template>
<script setup lang="ts">
import { computed, getCurrentInstance, onMounted, ref } from 'vue'
const navBarTitle = {
fontWeight: '510',
fontStyle: 'Medium',
fontSize: '40rpx',
lineHeight: '52px'
}
const isManage = ref(false)
const showDel = ref(false)
const handleManage = () => {
if (isManage.value) {
ids.value = []
}
isManage.value = !isManage.value
}
const showDelModel = () => {
showDel.value = true
}
const handleDel = () => {
// TODO 删除
}
const scrollHeight = ref(0)
const instance = getCurrentInstance();
const loadHeight = () => {
uni.getSystemInfo({
success(res) {
let screenHeight = res.screenHeight
uni.createSelectorQuery().in(instance.proxy).select("#uNavbarId").boundingClientRect((data : any) => {
scrollHeight.value = screenHeight - data.height
}).exec()
}
})
}
const defaultSize = 10
const params = ref({
page: 1,
size: defaultSize,
keyword: ''
})
const status = ref('loadmore')
const maxPage = ref()
const dataList = ref([
{
id: 1,
receiver_name: '张三',
receiver_phone: '13800138000',
province: '广东省',
city: '深圳市',
district: '南山区',
detailed_address: '科技园南区1栋101室',
label: '家',
is_default: true,
created_at: '2023-10-01 12:00:00'
},
{
id: 2,
receiver_name: '李四',
receiver_phone: '13900139000',
province: '北京市',
city: '北京市',
district: '朝阳区',
detailed_address: '建国门外大街1号',
label: '公司',
is_default: false,
created_at: '2023-10-02 14:30:00'
},
{
id: 3,
receiver_name: '王五',
receiver_phone: '13700137000',
province: '上海市',
city: '上海市',
district: '浦东新区',
detailed_address: '陆家嘴环路1000号',
label: '公司',
is_default: false,
created_at: '2023-10-03 09:15:00'
},
{
id: 4,
receiver_name: '赵六',
receiver_phone: '13600136000',
province: '浙江省',
city: '杭州市',
district: '西湖区',
detailed_address: '文三路100号',
label: '学校',
is_default: false,
created_at: '2023-10-04 16:45:00'
},
{
id: 5,
receiver_name: '钱七',
receiver_phone: '13500135000',
province: '江苏省',
city: '南京市',
district: '鼓楼区',
detailed_address: '中山路1号',
label: '家',
is_default: false,
created_at: '2023-10-05 11:20:00'
}
])
const loadData = () => {
if (status.value == 'nomore') return
// programAPI.getList(params.value).then(res => {
// programList.value = programList.value.concat(res.data.list)
// maxPage.value = res.data.pages
// params.value.page++
// if (params.value.page > maxPage.value) {
// status.value = 'nomore'
// }
// }).finally(() => {
// uni.stopPullDownRefresh()
// })
}
const ids = ref([])
const handleClick = (item : any) => {
if (isManage.value) { // 开始管理
ids.value.push(item.id)
}
}
const isIncludeId = (id : any) => {
return ids.value.includes(id);
}
onMounted(() => {
loadHeight()
loadData()
})
</script>
<style scoped lang="scss">
.shipping-container {
width: 100%;
height: 100vh;
background: #E5ECFD;
.right-menu {
display: flex;
margin-right: 24rpx;
font-family: Work Sans;
font-weight: 500;
font-style: Medium;
font-size: 32rpx;
leading-trim: NONE;
line-height: 100%;
letter-spacing: -2%;
.del-text {
color: red;
}
.add-text {
color: #305DEF;
}
}
.data-list {
.data-item {
width: 100%;
display: flex;
justify-content: space-between;
padding: 28rpx 22rpx;
background: #F5F8FF;
margin-bottom: 4rpx;
align-items: center;
.item-text {
font-weight: 400;
line-height: 100%;
.address {
font-size: 20rpx;
color: #7B7E8F;
}
.specific-address {
font-size: 26rpx;
}
.info {
font-size: 24rpx;
display: flex;
align-items: center;
span {
padding: 2rpx 8rpx;
font-size: 16rpx;
background: #DADDEA;
color: #305DEF;
border-radius: 2rpx;
}
}
}
.item-icon {
width: 40rpx;
height: 40rpx;
}
}
}
}
</style>

View File

@@ -201,7 +201,7 @@
} from '../../api/program'; } from '../../api/program';
import { import {
onLoad onLoad
} from '@dcloudio/uni-app' } from '@dcloudio/uni-app';
import { import {
groupAPI groupAPI
} from '../../api/group'; } from '../../api/group';

BIN
static/icon/Edit.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 732 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.3 KiB

BIN
static/icon/chose.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.6 KiB

BIN
static/mall/Home.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 834 B

BIN
static/mall/Star.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
static/mall/Twitch.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 654 B

BIN
static/my/01.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 KiB

BIN
static/my/02.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.4 KiB

BIN
static/my/03.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
static/my/04.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

BIN
static/my/05.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
static/my/06.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

BIN
static/my/07.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.0 KiB

BIN
static/my/08.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.6 KiB

BIN
static/shop/01.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

BIN
static/shop/02.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.3 KiB

BIN
static/shop/03.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.1 KiB

BIN
static/shop/04.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.4 KiB

BIN
static/shop/05.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.2 KiB

BIN
static/shop/06.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.1 KiB

BIN
static/shop/07.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.5 KiB

BIN
static/shop/08.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.8 KiB

BIN
static/shop/09.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.4 KiB

BIN
static/shop/10.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.1 KiB

View File

@@ -7,18 +7,18 @@ export const validatePhone = (phone) => {
export const getImageUrl = (imagePath) => { export const getImageUrl = (imagePath) => {
if (!imagePath) return '' if (!imagePath) return ''
// 如果图片路径以/uploads开头直接返回原路径
const baseURL = "https://minio.zrbjr.com"
if (imagePath.startsWith('/uploads')) {
return `${baseURL}/jurongquan${imagePath}`
}
if (imagePath.startsWith('http')) return imagePath if (imagePath.startsWith('http')) return imagePath
// const baseURL = "http://192.168.1.43:3000" // const baseURL = "http://192.168.1.43:3000"
// const baseURL = "https://www.zrbjr.com" // const baseURL = "https://www.zrbjr.com"
const baseURL = "https://minio.zrbjr.com"
// 如果图片路径以/uploads开头直接返回原路径 return baseURL + imagePath
if (imagePath.startsWith('/uploads')) {
return `${baseURL}/jurongquan${imagePath}`
}
return fullUrl
} }
export const getUserInfo = () => { export const getUserInfo = () => {