Compare commits
	
		
			17 Commits
		
	
	
		
			934906bb75
			...
			d8b880cf47
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | d8b880cf47 | ||
|  | 88e8170755 | ||
|  | 0d087c0bc3 | ||
|  | f4df2d0a78 | ||
|  | c74ba7bcb3 | ||
|  | c445901806 | ||
|  | 8f38870c33 | ||
|  | 05cd427430 | ||
|  | 63e24791f2 | ||
|  | 635bca0fb6 | ||
|  | 6daf34856e | ||
|  | 59269a7547 | ||
|  | fc3e833605 | ||
|  | ad9899d716 | ||
|  | 77e08232f7 | ||
|  | ddbe15cfb1 | ||
|  | 7a59995bba | 
| @ -1,23 +1,25 @@ | ||||
| import { getHttp } from '~/api/http.js' | ||||
| import { request } from '@/api/http.js' | ||||
| 
 | ||||
| export async function senCode(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/user/send', { | ||||
| 
 | ||||
|     return await request({ | ||||
|         url:'/api/v1/m/user/send', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|         data | ||||
|     }) | ||||
| } | ||||
| export async function userLogin(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/user/login', { | ||||
| 
 | ||||
|     return await request( { | ||||
|         url:'/api/v1/m/user/login', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|         data | ||||
|     }) | ||||
| } | ||||
| export async function userUpdate(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/user/update', { | ||||
|     return await request( { | ||||
|         url:'/api/v1/m/user/update', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|         data | ||||
|     }) | ||||
| } | ||||
| @ -1,30 +1,32 @@ | ||||
| import { getHttp } from '~/api/http.js' | ||||
| import { request } from '@/api/http.js' | ||||
| 
 | ||||
| export async function artworkList(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/auction/default/artwork/list', { | ||||
|     return await request( { | ||||
|         url:'/api/v1/m/auction/default/artwork/list', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|         data | ||||
|     }) | ||||
| } | ||||
| export async function defaultDetail(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/auction/default/detail', { | ||||
|     return await request ({ | ||||
|         url:'/api/v1/m/auction/default/detail', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|        data | ||||
|     }) | ||||
| } | ||||
| export async function artworkDetail(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/artwork/detail', { | ||||
| 
 | ||||
|     return await request( { | ||||
|         url:'/api/v1/m/artwork/detail', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|     data, | ||||
|     }) | ||||
| } | ||||
| export async function userArtworks(data) { | ||||
|     const http = getHttp() | ||||
|     return await http('/api/v1/m/user/artworks', { | ||||
| 
 | ||||
|     return await request( { | ||||
|         url:'/api/v1/m/user/artworks', | ||||
|         method: 'POST', | ||||
|         body: data, | ||||
|         data | ||||
|     }) | ||||
| } | ||||
							
								
								
									
										137
									
								
								app/api/http.js
									
									
									
									
									
								
							
							
						
						
									
										137
									
								
								app/api/http.js
									
									
									
									
									
								
							| @ -1,52 +1,109 @@ | ||||
| 
 | ||||
| import { useRuntimeConfig } from '#app' | ||||
| import { ofetch } from 'ofetch' | ||||
| import {useRuntimeConfig} from '#app' | ||||
| import {ofetch} from 'ofetch' | ||||
| import {message} from '@/components/x-message/useMessage.js' | ||||
| import {authStore} from "@/stores/auth/index.js"; | ||||
| let httpStatusErrorHandler | ||||
| import {authStore} from "@/stores/auth/index.js" | ||||
| 
 | ||||
| let httpStatusErrorHandler | ||||
| let http | ||||
| 
 | ||||
| // HTTP 状态码映射
 | ||||
| const HTTP_STATUS_MAP = { | ||||
|   400: '请求参数错误', | ||||
|   401: '未授权或登录过期', | ||||
|   403: '访问被禁止', | ||||
|   404: '请求的资源不存在', | ||||
|   500: '服务器内部错误', | ||||
|   502: '网关错误', | ||||
|   503: '服务暂时不可用', | ||||
|   504: '网关超时' | ||||
| } | ||||
| 
 | ||||
| export function setupHttp() { | ||||
|   if (http) | ||||
|     return http | ||||
|   if (http) return http | ||||
| 
 | ||||
|   const config = useRuntimeConfig() | ||||
|   const baseURL = config.public.NUXT_PUBLIC_API_BASE | ||||
|   const {token}= authStore() | ||||
|   const { token } = authStore() | ||||
|   const router = useRouter() | ||||
|   http = ofetch.create({ | ||||
| 
 | ||||
|   const defaultOptions = { | ||||
|     baseURL, | ||||
|     headers: { 'Content-Type': 'application/json' }, | ||||
|     async onRequest({ options }) { | ||||
|       options.headers = { | ||||
|         ...options.headers, | ||||
|         Authorization:token.value | ||||
|       } | ||||
|     }, | ||||
|     async onResponse({ response }) { | ||||
|      if (response._data.status===1){ | ||||
|        message.error(response._data.msg) | ||||
|      } | ||||
|      if (response._data.status===401){ | ||||
|        router.replace('/login') | ||||
|      } | ||||
|     }, | ||||
|     async onResponseError({ response }) { | ||||
|       const { message } = response._data | ||||
|       if (Array.isArray(message)) { | ||||
|         message.forEach((item) => { | ||||
|           httpStatusErrorHandler?.(item, response.status) | ||||
|         }) | ||||
|       } | ||||
|       else { | ||||
|         httpStatusErrorHandler?.(message, response.status) | ||||
|       } | ||||
|       return Promise.reject(response._data) | ||||
|     }, | ||||
|     timeout: 15000, // 15秒超时
 | ||||
|     retry: 3, | ||||
|     retryDelay: 1000, | ||||
|   } | ||||
| 
 | ||||
|   http = ofetch.create({ | ||||
|     ...defaultOptions, | ||||
| 
 | ||||
|     // 请求拦截
 | ||||
|     async onRequest({ options, request }) { | ||||
|       // 添加 token
 | ||||
|       options.headers = { | ||||
|         ...options.headers, | ||||
|         Authorization: token.value | ||||
|       } | ||||
| 
 | ||||
|       // GET 请求添加时间戳防止缓存
 | ||||
|       if (request.toLowerCase().includes('get')) { | ||||
|         options.params = { | ||||
|           ...options.params, | ||||
|           _t: Date.now() | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
| 
 | ||||
|     // 响应拦截
 | ||||
|     async onResponse({ response }) { | ||||
|       const data = response._data | ||||
| 
 | ||||
|       // 处理业务错误
 | ||||
|       if (data.status === 1) { | ||||
|         message.error(data.msg || '操作失败') | ||||
|       } | ||||
| 
 | ||||
|       // 处理登录失效
 | ||||
|       if (data.status === 401) { | ||||
|         message.error('登录已过期,请重新登录') | ||||
|         token.value = '' // 清除 token
 | ||||
|         router.replace('/login') | ||||
|       } | ||||
| 
 | ||||
|       return response | ||||
|     }, | ||||
| 
 | ||||
|     // 响应错误处理
 | ||||
|     async onResponseError({ response, request }) { | ||||
|       // 网络错误
 | ||||
|       if (!response) { | ||||
|         message.error('网络连接失败,请检查网络设置') | ||||
|         return Promise.reject(new Error('网络错误')) | ||||
|       } | ||||
|       const status = response.status | ||||
|       const data = response._data | ||||
| 
 | ||||
|       // 处理 HTTP 状态错误
 | ||||
|       const errorMessage = data.msg || HTTP_STATUS_MAP[status] || '请求失败' | ||||
| 
 | ||||
|       if (Array.isArray(data.msg)) { | ||||
|         data.msg.forEach(item => { | ||||
|           httpStatusErrorHandler?.(item, status) | ||||
|         }) | ||||
|       } else { | ||||
|         httpStatusErrorHandler?.(errorMessage, status) | ||||
|       } | ||||
| 
 | ||||
|       message.error(errorMessage) | ||||
|       return Promise.reject(data) | ||||
|     }, | ||||
|   }) | ||||
| 
 | ||||
|   return http | ||||
| } | ||||
| 
 | ||||
| export function createAbortController() { | ||||
|   return new AbortController() | ||||
| } | ||||
| 
 | ||||
| export function injectHttpStatusErrorHandler(handler) { | ||||
| @ -59,3 +116,13 @@ export function getHttp() { | ||||
|   } | ||||
|   return http | ||||
| } | ||||
| 
 | ||||
| // 导出请求工具函数
 | ||||
| export async function request({url,...options}) { | ||||
|   const http = getHttp() | ||||
|   try { | ||||
|     return await http(url, {...options,body:options.data}) | ||||
|   } catch (error) { | ||||
|     throw error | ||||
|   } | ||||
| } | ||||
							
								
								
									
										46
									
								
								app/components/x-popup/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								app/components/x-popup/index.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,46 @@ | ||||
| <script setup> | ||||
| const props = defineProps({ | ||||
|   show: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|   }, | ||||
|   title:'' | ||||
| }) | ||||
| const emit = defineEmits(['update:show']) | ||||
| const close=()=>{ | ||||
|   emit('update:show',false) | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <van-popup | ||||
|       :show="show" | ||||
|       position="bottom" | ||||
|       :style="{ height: '74%' }" | ||||
|       v-bind="{...$attrs,...$props}" | ||||
|       @close="close" | ||||
|   > | ||||
|     <div class="flex flex-col h-full"> | ||||
|       <!-- 固定的标题栏 --> | ||||
|       <div class="flex items-center pl-16px pr-19px h-40px border-b-1px border-b-[#D3D3D3] shrink-0"> | ||||
|         <template v-if="$slots.title"> | ||||
|         <slot name="title"></slot> | ||||
|         </template> | ||||
|         <template v-else> | ||||
|           <div class="text-#000 text-16px">{{title}}</div> | ||||
|         </template> | ||||
|         <div class="ml-auto flex items-center"> | ||||
|           <van-icon size="20" name="cross" color="#939393" @click="close"/> | ||||
|         </div> | ||||
|       </div> | ||||
|       <!-- 可滚动的内容区域 --> | ||||
|       <div class="flex-1 overflow-y-auto "> | ||||
|         <slot/> | ||||
|       </div> | ||||
|     </div> | ||||
|   </van-popup> | ||||
| </template> | ||||
| 
 | ||||
| <style scoped> | ||||
| 
 | ||||
| </style> | ||||
| @ -1,3 +1,3 @@ | ||||
| 
 | ||||
| export const useAppFooterRouteNames= ['home', 'profile'] | ||||
| export const useAppHeaderRouteNames= ['home', 'profile','login'] | ||||
| export const useAppHeaderRouteNames= ['home', 'profile','login','collectCode-login','collectCode-mine'] | ||||
|  | ||||
| @ -2,14 +2,10 @@ | ||||
|   <main class="flex flex-col min-h-svh"> | ||||
|     <AppHeader class="h-[var(--van-nav-bar-height)]" /> | ||||
|     <div class="flex-1 flex flex-col"> | ||||
|       <keep-alive> | ||||
|         <slot /> | ||||
|       </keep-alive> | ||||
|       <slot /> | ||||
|     </div> | ||||
|     <AppFooter  /> | ||||
|   </main> | ||||
| </template> | ||||
| <script setup > | ||||
| import { goodStore } from "@/stores/goods/index.js"; | ||||
| const { fullLive } = goodStore() | ||||
| </script> | ||||
							
								
								
									
										114
									
								
								app/pages/LiveRoom/components/Broadcast/tang.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								app/pages/LiveRoom/components/Broadcast/tang.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,114 @@ | ||||
| <script setup> | ||||
| import xPopup from '@/components/x-popup/index.vue' | ||||
| const props = defineProps({ | ||||
|   show: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|   } | ||||
| }) | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <x-popup :show="show"> | ||||
|     <template #title> | ||||
|       <div class="text-#000 text-16px">拍品列表</div> | ||||
|       <div class="text-#939393 text-16px ml-14px">共188个拍品</div> | ||||
|     </template> | ||||
|     <div class="px-16px py-18px"> | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作,日落而息撒打算撒打算撒打决赛多久啊是世奥兰日落而息撒打算撒打算撒打决赛多久啊是世奥兰 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作,日落而息撒打算撒打算撒打决赛多久啊是世奥兰日落而息撒打算撒打算撒打决赛多久啊是世奥兰 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
| 
 | ||||
|       <div class="flex mb-21px"> | ||||
|         <div class="mr-10px flex-shrink-0 rounded-4px overflow-hidden"> | ||||
|           <img class="w-80px h-80px" src="@/static/images/ddfdcaer.png" alt=""> | ||||
|         </div> | ||||
|         <div> | ||||
|           <div class="ellipsis line-height-20px text-16px font-600 min-h-40px"> | ||||
|             张天赐 | 日出而作 | ||||
|           </div> | ||||
|           <div class="text-14px text-#575757">起拍价:RMB 1,000</div> | ||||
|           <div class="text-14px text-#B58047">成交价:等待更新</div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </x-popup> | ||||
| </template> | ||||
| 
 | ||||
| <style scoped> | ||||
| 
 | ||||
| </style> | ||||
| @ -35,7 +35,6 @@ const confirm=()=>{ | ||||
|           <div class="text-#000 text-16px font-600 ">支付部分</div> | ||||
|           <input class="w-272px h-48px bg-#F3F3F3 px-11px text-16px" type="text" placeholder="最多RMB5,000"> | ||||
|         </template> | ||||
| 
 | ||||
|         <div class="text-#2B53AC text-14px" @click="changePayStatus">{{payStatus===0 ? '支付部分' : '支付全部'}}</div> | ||||
|       </div> | ||||
|     </van-dialog> | ||||
|  | ||||
| @ -40,5 +40,6 @@ const { quoteStatus, changeStatus,show,show1 } = liveStore(); | ||||
|         <div class="text-10px">去支付</div> | ||||
|       </div> | ||||
|     </PressableButton> | ||||
| 
 | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| @ -110,11 +110,11 @@ watch(()=>{ | ||||
|         <div class="absolute left-1/2 transform -translate-x-1/2 flex flex-col items-center" style="bottom:calc(var(--safe-area-inset-bottom) + 26px)"> | ||||
|           <div class="text-16px text-#FFB25F font-600"> | ||||
|             当前价:RMB | ||||
|             <van-rolling-text class="my-rolling-text" :start-num="0" :target-num="3000" direction="up"/> | ||||
|             <van-rolling-text class="my-rolling-text" :start-num="0" :duration="0.5" :target-num="3000" direction="up"/> | ||||
|           </div> | ||||
|           <div class="text-16px text-#fff font-600"> | ||||
|             下口价:RMB | ||||
|             <van-rolling-text class="my-rolling-text1" :start-num="0" :target-num="3500" direction="up"/> | ||||
|             <van-rolling-text class="my-rolling-text1" :start-num="0" :duration="0.5" :target-num="3500" direction="up"/> | ||||
|           </div> | ||||
|           <PressableButton> | ||||
|             <div | ||||
|  | ||||
							
								
								
									
										211
									
								
								app/pages/collectCode/login/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										211
									
								
								app/pages/collectCode/login/index.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,211 @@ | ||||
| <script setup> | ||||
| import { useRouter, useRoute } from 'vue-router'; | ||||
| import { useI18n } from 'vue-i18n' | ||||
| import { senCode, userLogin } from "@/api/auth/index.js"; | ||||
| import { authStore } from "~/stores/auth/index.js"; | ||||
| import { message } from '@/components/x-message/useMessage.js' | ||||
| // ... 现有导入 ... | ||||
| import FingerprintJS from '@fingerprintjs/fingerprintjs' | ||||
| const { userInfo, token,fingerprint } = authStore() | ||||
| const router = useRouter(); | ||||
| const route = useRoute(); | ||||
| const { locale } = useI18n() | ||||
| definePageMeta({ | ||||
|   title: '登录', | ||||
|   i18n: 'login.title', | ||||
| }) | ||||
| const loadingRef = ref({ | ||||
|   loading1: false, | ||||
|   loading2: false, | ||||
| }) | ||||
| const password = ref('') | ||||
| const loginType = ref(0) | ||||
| const interval = ref(null) | ||||
| const startCountdown = () => { | ||||
|   if (interval.value) { | ||||
|     clearInterval(interval.value); | ||||
|   } | ||||
|   countdown.value = 60; | ||||
|   interval.value = setInterval(() => { | ||||
|     if (countdown.value > 0) { | ||||
|       countdown.value--; | ||||
|     } else { | ||||
|       clearInterval(interval.value); | ||||
|     } | ||||
|   }, 1000); | ||||
| } | ||||
| const countdown = ref(0); | ||||
| const phoneNum = ref('17630920520') | ||||
| const code = ref('123789') | ||||
| const pane = ref(0) | ||||
| const showKeyboard = ref(false); | ||||
| const getFingerprint = async () => { | ||||
|   const fp = await FingerprintJS.load() | ||||
|   const result = await fp.get() | ||||
|   return result.visitorId // 稳定的指纹哈希值 | ||||
| } | ||||
| 
 | ||||
| // 如果指纹存在,且指纹和指纹库中的指纹一致,则直接登录 | ||||
| const checkFingerprint = async () => { | ||||
|   const tempFingerprint = await getFingerprint() | ||||
|   if (fingerprint && fingerprint === tempFingerprint) { | ||||
|     await router.push('/collectCode/mine') | ||||
|   } | ||||
| } | ||||
| checkFingerprint() | ||||
| const vanSwipeRef = ref(null) | ||||
| const getCode = async () => { | ||||
|   loadingRef.value.loading1 = true | ||||
|   const res = await senCode({ | ||||
|     telNum: phoneNum.value, | ||||
|     zone: '86' | ||||
|   }) | ||||
|   loadingRef.value.loading1 = false | ||||
|   if (res.status === 0) { | ||||
| 
 | ||||
| 
 | ||||
|   } | ||||
|   pane.value = 1 | ||||
|   vanSwipeRef.value?.swipeTo(pane.value) | ||||
|   showKeyboard.value = true | ||||
|   startCountdown(); | ||||
|   /*  pane.value = 1 | ||||
|     vanSwipeRef.value?.swipeTo(pane.value) | ||||
|     showKeyboard.value=true | ||||
|     startCountdown();*/ | ||||
| 
 | ||||
| } | ||||
| const changeToPwd = async () => { | ||||
|   loginType.value = loginType.value === 0 ? 1 : 0 | ||||
| } | ||||
| const goBack = () => { | ||||
|   code.value = '' | ||||
|   pane.value = 0 | ||||
|   vanSwipeRef.value?.swipeTo(pane.value) | ||||
| } | ||||
| const goLogin = async () => { | ||||
|   loadingRef.value.loading2 = true | ||||
|   const res = await userLogin({ | ||||
|     telNum: phoneNum.value, | ||||
|     zone: '86', | ||||
|     code: code.value | ||||
|   }) | ||||
|   if (res.status === 0) { | ||||
|     userInfo.value = res.data.accountInfo | ||||
|     token.value = res.data.token | ||||
|     fingerprint.value = await getFingerprint() | ||||
| 
 | ||||
|     await router.push('/collectCode/mine'); | ||||
|    | ||||
|   } | ||||
|   loadingRef.value.loading2 = false | ||||
| } | ||||
| 
 | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="h-[100vh] w-[100vw] bg-[url('@/static/images/asdfsdd.png')] bg-cover px-[31px] pt-[86px]"> | ||||
|     <div class="w-full flex justify-center mb-[100px] flex-col items-center"> | ||||
|       <img class="h-[105px] w-[189px]" src="@/static/images/ghfggff.png" alt=""> | ||||
|       <img class="h-[29px] w-[108px]" src="@/static/images/qrcodetext.png" alt=""> | ||||
|     </div> | ||||
|     <van-swipe ref="vanSwipeRef" :show-indicators="false" :touchable="false" :lazy-render="true" :loop="false"> | ||||
|       <van-swipe-item> | ||||
|         <div v-show="pane === 0"> | ||||
|           <div class=""> | ||||
|             <div class="border-b-[1.7px] mt-[8px]"> | ||||
|               <van-field v-model="phoneNum" clearable :placeholder="$t('login.phonePlaceholder')"> | ||||
|                 <template #label> | ||||
|                   <div class="text-[16px] text-[#1A1A1A] flex align-center justify-start"> | ||||
|                     手机号 | ||||
|                   </div> | ||||
|                 </template> | ||||
|               </van-field> | ||||
| 
 | ||||
|             </div> | ||||
|             <div class="border-b-[1.7px] mt-[8px]" v-show="loginType === 1"> | ||||
|               <van-field v-model="password" clearable :placeholder="$t('login.passwordPlaceholder')"> | ||||
|                 <template #label> | ||||
|                   <div class="text-[16px] text-[#1A1A1A] flex align-center justify-start"> | ||||
|                     密码 | ||||
|                   </div> | ||||
|                 </template> | ||||
|               </van-field> | ||||
| 
 | ||||
|             </div> | ||||
|             <div class="flex justify-end mt-[10px]" @click="changeToPwd"> | ||||
|               <div class="text-[14px] text-[#2B53AC]"> | ||||
|                 {{ loginType === 0 ? '密码登录' : '验证码登录' }} | ||||
|               </div> | ||||
| 
 | ||||
|             </div> | ||||
|             <div /> | ||||
|           </div> | ||||
|           <div class="mt-[55px]"> | ||||
|             <div v-if="loginType === 0"> | ||||
|               <van-button :loading="loadingRef.loading1" v-if="phoneNum" :loading-text="$t('login.getCode')" | ||||
|                 type="primary" block style="height: 48px" @click="getCode">{{ $t('login.getCode') | ||||
|                 }}</van-button> | ||||
|               <van-button v-else type="primary" color="#D3D3D3" block style="height: 48px">{{ | ||||
|                 $t('login.getCode') | ||||
|               }}</van-button> | ||||
|             </div> | ||||
|             <div v-else> | ||||
|               <van-button type="primary" v-if="password" block :loading="loadingRef.loading2" :loading-text="$t('login.login')" | ||||
|                 style="height: 48px;margin-top:10px" @click="goLogin">{{ | ||||
|                   $t('login.login') | ||||
|                 }}</van-button> | ||||
|               <van-button v-else type="primary" color="#D3D3D3" block style="height: 48px">{{ | ||||
|                 $t('login.login') | ||||
|               }}</van-button> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </van-swipe-item> | ||||
|       <van-swipe-item> | ||||
|         <div v-show="pane === 1"> | ||||
|           <div class="flex mb-[16px]"> | ||||
|             <div class="text-[16px] text-[#BDBDBD] mr-[10px]">{{ $t('login.hasSendTo') }}</div> | ||||
|             <div class="text-[16px] text-[#000]">+{{ selectedZone }} {{ phoneNum }}</div> | ||||
|           </div> | ||||
|           <van-password-input :value="code" :gutter="10" :mask="false" focused @focus="showKeyboard = true" /> | ||||
|           <div :class="`${countdown > 0 ? 'text-#BDBDBD' : 'text-#2B53AC'}  text-14px`"> | ||||
|             {{ $t('login.reSend') }}<span v-if="countdown > 0">({{ countdown }})</span> | ||||
|           </div> | ||||
|           <div class="mt-[17px]"> | ||||
| 
 | ||||
|             <van-button v-if="code.length === 6" type="primary" block :loading="loadingRef.loading2" | ||||
|               :loading-text="$t('login.login')" style="height: 48px" @click="goLogin">{{ | ||||
|                 $t('login.login') | ||||
|               }}</van-button> | ||||
|             <van-button v-else type="primary" color="#D3D3D3" block style="height: 48px">{{ | ||||
|               $t('login.login') | ||||
|             }}</van-button> | ||||
|           </div> | ||||
|           <div class="mt-[17px]"> | ||||
|             <van-button type="primary" @click="goBack" block style="height: 48px">{{ $t('login.back') | ||||
|               }}</van-button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </van-swipe-item> | ||||
|     </van-swipe> | ||||
|     <van-number-keyboard v-model="code" :show="showKeyboard" @blur="showKeyboard = false" /> | ||||
| 
 | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <style scoped lang="scss"> | ||||
| :deep(.van-cell.van-field) { | ||||
|   padding-left: 0; | ||||
| } | ||||
| 
 | ||||
| :deep(.van-password-input) { | ||||
|   margin: 0; | ||||
| } | ||||
| 
 | ||||
| :deep(.van-password-input__item) { | ||||
|   border: 1px solid #E5E5E5; | ||||
|   width: 41px; | ||||
|   height: 41px; | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										55
									
								
								app/pages/collectCode/mine/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								app/pages/collectCode/mine/index.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | ||||
| <script setup> | ||||
| import { userArtworks } from "~/api/goods/index.js"; | ||||
| import { authStore } from "~/stores/auth/index.js"; | ||||
| 
 | ||||
| definePageMeta({ | ||||
|   layout: 'default', | ||||
|   title: '收款二维码', | ||||
|   i18n: 'menu.profile', | ||||
| }) | ||||
| const { userInfo } = authStore() | ||||
| const initData = async () => { | ||||
|   const res = await userArtworks({}) | ||||
|   if (res.status === 0) { | ||||
| 
 | ||||
|   } | ||||
| } | ||||
| initData() | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="w-[100vw] bg-[url('@/static/images/3532@2x.png')] bg-cover pt-43px flex-grow-1 flex flex-col"> | ||||
|     <div class="flex items-center px-16px mb-43px"> | ||||
|       <div class="mr-23px"> | ||||
|         <img class="w-57px h-57px" src="@/static/images/5514@2x.png" alt=""> | ||||
|       </div> | ||||
|       <div class="flex flex-col"> | ||||
|         <div class="text-18px text-#181818">{{ userInfo.realName }}</div> | ||||
|         <div class="text-#575757 text-14px">{{ userInfo.telNum }}</div> | ||||
|       </div> | ||||
|       <div class="flex-grow-1 flex justify-end"> | ||||
|         <img class="w-40px h-40px" src="@/static/images/logout.png" alt=""> | ||||
|       </div> | ||||
|     </div> | ||||
|     <div class="flex-grow-1"> | ||||
|       <div class="border-b-1px border-b-#D3D3D3  px-16px flex"> | ||||
|         <div class="text-#000 text-16px border-b-3 border-b-#2B53AC h-36px">线下付款二维码 </div> | ||||
|       </div> | ||||
|       <van-pull-refresh> | ||||
|         <van-list finished-text="没有更多了"> | ||||
|           <van-swipe-cell> | ||||
| 
 | ||||
|             <van-cell :border="false" title="单元格" value="内容" /> | ||||
|             <template #right> | ||||
|               <van-button square type="danger" text="删除" /> | ||||
|               <van-button square type="primary" text="收藏" /> | ||||
|             </template> | ||||
|           </van-swipe-cell> | ||||
|         </van-list> | ||||
|       </van-pull-refresh> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| <style scoped> | ||||
| 
 | ||||
| </style> | ||||
| @ -1,3 +1,19 @@ | ||||
| 
 | ||||
| <script setup> | ||||
| const props = defineProps({ | ||||
|   items: Array, | ||||
|   colIndex: { | ||||
|     type: Number, | ||||
|     default: 0 | ||||
|   } | ||||
| }); | ||||
| 
 | ||||
| const emit = defineEmits(['openShow']); | ||||
| 
 | ||||
| const openShow = (item,index) => { | ||||
|   emit('openShow', item,index); | ||||
| }; | ||||
| </script> | ||||
| <template> | ||||
|   <div class="flex flex-1 flex-col gap-[16px]"> | ||||
|     <div | ||||
| @ -35,19 +51,3 @@ | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| const props = defineProps({ | ||||
|   items: Array, | ||||
|   colIndex: { | ||||
|     type: Number, | ||||
|     default: 0 | ||||
|   } | ||||
| }); | ||||
| 
 | ||||
| const emit = defineEmits(['openShow']); | ||||
| 
 | ||||
| const openShow = (item,index) => { | ||||
|   emit('openShow', item,index); | ||||
| }; | ||||
| </script> | ||||
							
								
								
									
										96
									
								
								app/pages/home/components/ItemDetailSheet/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								app/pages/home/components/ItemDetailSheet/index.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,96 @@ | ||||
| <template> | ||||
|   <van-popup | ||||
|       v-model:show="props.show" | ||||
|       position="bottom" | ||||
|       :style="{ height: props.height }" | ||||
|       round | ||||
|       closeable | ||||
|       @closed="handleClose" | ||||
|   > | ||||
|     <div class="p-[16px] overflow-y-auto h-full"> | ||||
|       <!-- 商品基本信息 --> | ||||
|       <div class="flex flex-col gap-[12px]"> | ||||
|         <img | ||||
|             :src="props.detail?.artwork?.hdPic" | ||||
|             class="w-full rounded-[4px] object-cover" | ||||
|             :alt="props.detail?.name" | ||||
|         /> | ||||
|         <div class="text-[16px] font-medium text-[#000]"> | ||||
|           {{ props.detail?.name }} | ||||
|         </div> | ||||
| 
 | ||||
|         <!-- 价格信息 --> | ||||
|         <div class="flex flex-col gap-[8px]"> | ||||
|           <div class="text-[14px] text-[#575757]"> | ||||
|             起拍价:{{ formatPrice(props.detail?.startPrice) }} | ||||
|           </div> | ||||
|           <div v-if="props.detail?.soldPrice" class="text-[14px] text-[#b58047]"> | ||||
|             成交价:{{ formatPrice(props.detail?.soldPrice) }} | ||||
|           </div> | ||||
|         </div> | ||||
| 
 | ||||
|         <!-- 商品详情 --> | ||||
|         <div class="mt-[16px]"> | ||||
|           <div class="text-[16px] font-medium mb-[12px]">商品详情</div> | ||||
|           <div class="flex flex-col gap-[8px]"> | ||||
|             <div class="flex justify-between text-[14px]"> | ||||
|               <span class="text-[#575757]">尺寸</span> | ||||
|               <span>{{ props.detail?.artwork?.size || '-' }}</span> | ||||
|             </div> | ||||
|             <div class="flex justify-between text-[14px]"> | ||||
|               <span class="text-[#575757]">年代</span> | ||||
|               <span>{{ props.detail?.artwork?.year || '-' }}</span> | ||||
|             </div> | ||||
|             <div class="flex justify-between text-[14px]"> | ||||
|               <span class="text-[#575757]">材质</span> | ||||
|               <span>{{ props.detail?.artwork?.material || '-' }}</span> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
| 
 | ||||
|         <!-- 作品描述 --> | ||||
|         <div v-if="props.detail?.artwork?.description" class="mt-[16px]"> | ||||
|           <div class="text-[16px] font-medium mb-[12px]">作品描述</div> | ||||
|           <div class="text-[14px] text-[#575757] whitespace-pre-wrap"> | ||||
|             {{ props.detail?.artwork?.description }} | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </van-popup> | ||||
| </template> | ||||
| 
 | ||||
| <script setup> | ||||
| import { formatPrice } from '~/utils/format' | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   show: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|   }, | ||||
|   height: { | ||||
|     type: String, | ||||
|     default: '90vh' | ||||
|   }, | ||||
|   detail: { | ||||
|     type: Object, | ||||
|     default: () => ({}) | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const emit = defineEmits(['update:show']) | ||||
| 
 | ||||
| const handleClose = () => { | ||||
|   emit('update:show', false) | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <style scoped> | ||||
| :deep(.van-popup) { | ||||
|   max-height: 90vh; | ||||
| } | ||||
| 
 | ||||
| :deep(.van-popup__close-icon) { | ||||
|   color: #000; | ||||
| } | ||||
| </style> | ||||
| @ -1,78 +1,115 @@ | ||||
| <script setup> | ||||
| import Column from "~/pages/home/components/Column/index.vue"; | ||||
| import {goodStore} from "~/stores/goods"; | ||||
| import {artworkDetail, artworkList} from "~/api/goods"; | ||||
| import {useRect} from "@vant/use"; | ||||
| const {itemList, pageRef,auctionDetail,liveRef,artWorkDetail,currentItem} = goodStore(); | ||||
| const loading = ref(false); | ||||
| const showHeight = ref(''); | ||||
| const show = ref(false); | ||||
| const loading1 = ref(false); | ||||
| const finished = ref(false); | ||||
| const getArtworkList=async ()=>{ | ||||
|   const res= await artworkList({auctionUuid: auctionDetail.value.uuid,...pageRef.value}) | ||||
|   if (res.status===0){ | ||||
|     itemList.value.push(...res.data.data) | ||||
|     pageRef.value.itemCount=res.data.count | ||||
|   } | ||||
| } | ||||
| const loadData = async () => { | ||||
|   pageRef.value.page++ | ||||
|   await getArtworkList() | ||||
|   loading.value = false; | ||||
|   if (pageRef.value.itemCount <= itemList.value.length) { | ||||
|     finished.value = true | ||||
|   } | ||||
| }; | ||||
| import { ref, computed } from 'vue' | ||||
| import { useRect } from "@vant/use" | ||||
| import { goodStore } from "@/stores/goods" | ||||
| import Column from "../Column/index.vue" | ||||
| import ItemDetail from "@/components/itemDetail/index.vue" | ||||
| 
 | ||||
| const { | ||||
|   itemList, | ||||
|   pageRef, | ||||
|   auctionDetail, | ||||
|   liveRef, | ||||
|   artWorkDetail, | ||||
|   currentItem, | ||||
|   loading: storeLoading, | ||||
|   getArtworkList, | ||||
|   getArtworkDetail | ||||
| } = goodStore() | ||||
| 
 | ||||
| const localState = ref({ | ||||
|   finished: false, | ||||
|   refreshing: false, | ||||
|   showDetail: false, | ||||
|   showHeight: '' | ||||
| }) | ||||
| 
 | ||||
| // 计算列数据 | ||||
| const columns = computed(() => { | ||||
|   const result = [[], []]; | ||||
|   const result = [[], []] | ||||
|   itemList.value.forEach((item, index) => { | ||||
|     result[index % 2].push({...item, index}); | ||||
|   }); | ||||
|   return result; | ||||
| }); | ||||
| const refreshData = async () => { | ||||
|   pageRef.value.page = 1 | ||||
|   finished.value=false | ||||
|   const res= await artworkList({auctionUuid: auctionDetail.value.uuid,...pageRef.value}) | ||||
|   if (res.status===0){ | ||||
|     itemList.value=res.data.data | ||||
|     pageRef.value.itemCount=res.data.count | ||||
|   } | ||||
|   loading1.value = false | ||||
|     result[index % 2].push({ ...item, index }) | ||||
|   }) | ||||
|   return result | ||||
| }) | ||||
| 
 | ||||
| // 加载更多 | ||||
| const loadMore = async () => { | ||||
|   pageRef.value.page++ | ||||
|   const { finished } = await getArtworkList() | ||||
|   localState.value.finished = finished | ||||
| } | ||||
| 
 | ||||
| const getDetail=async ()=>{ | ||||
|  const res=await artworkDetail({uuid:currentItem.value.artworkUuid}) | ||||
|   if (res.status===0){ | ||||
|     artWorkDetail.value | ||||
| // 刷新 | ||||
| const onRefresh = async () => { | ||||
|   try { | ||||
|     localState.value.refreshing = true | ||||
|     localState.value.finished = false | ||||
|     const { finished } = await getArtworkList(true) | ||||
|     localState.value.finished = finished | ||||
|   } finally { | ||||
|     localState.value.refreshing = false | ||||
|   } | ||||
| } | ||||
| const openShow = (item,index) => { | ||||
|   currentItem.value=item | ||||
|   getDetail() | ||||
|   const rect = useRect(liveRef.value.$el); | ||||
|   showHeight.value = rect.height; | ||||
| 
 | ||||
| // 打开详情 | ||||
| const openShow = async (item) => { | ||||
|   currentItem.value = item | ||||
|   await getArtworkDetail(item.uuid) | ||||
| 
 | ||||
|   const rect = useRect(liveRef.value.$el) | ||||
|   localState.value.showHeight = rect.height | ||||
| 
 | ||||
|   nextTick(() => { | ||||
|     show.value = true; | ||||
|   }); | ||||
| }; | ||||
|     localState.value.showDetail = true | ||||
|   }) | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="px-[16px] pt-[16px]"> | ||||
|     <van-pull-refresh v-model="loading1" success-text="刷新成功" @refresh="refreshData"> | ||||
|       <van-list :immediate-check="false" v-model:loading="loading" :finished="finished" finished-text="没有更多了" | ||||
|                 @load="loadData"> | ||||
|     <van-pull-refresh | ||||
|         v-model="localState.refreshing" | ||||
|         success-text="刷新成功" | ||||
|         @refresh="onRefresh" | ||||
|     > | ||||
|       <van-list | ||||
|           v-model:loading="storeLoading" | ||||
|           :finished="localState.finished" | ||||
|           finished-text="没有更多了" | ||||
|           @load="loadMore" | ||||
|       > | ||||
|         <div class="w-full flex gap-[16px]"> | ||||
|           <Column v-for="(column, colIndex) in columns" :key="colIndex" :colIndex="colIndex" :items="column" @openShow="openShow" /> | ||||
|           <Column | ||||
|               v-for="(column, colIndex) in columns" | ||||
|               :key="colIndex" | ||||
|               :colIndex="colIndex" | ||||
|               :items="column" | ||||
|               @openShow="openShow" | ||||
|           /> | ||||
|         </div> | ||||
|       </van-list> | ||||
|     </van-pull-refresh> | ||||
|     <van-action-sheet teleport="#__nuxt" :round="false" v-model:show="show" title="拍品详情"> | ||||
|       <div class="content bg-[#F0F0F0]" :style="`height: calc(100vh - ${showHeight + 85}px)`"> | ||||
|         <itemDetail/> | ||||
| 
 | ||||
|     <van-action-sheet | ||||
|         teleport="#__nuxt" | ||||
|         :round="false" | ||||
|         v-model:show="localState.showDetail" | ||||
|         title="拍品详情" | ||||
|     > | ||||
|       <div | ||||
|           class="content bg-[#F0F0F0]" | ||||
|           :style="`height: calc(100vh - ${localState.showHeight + 85}px)`" | ||||
|       > | ||||
|         <ItemDetail /> | ||||
|       </div> | ||||
|     </van-action-sheet> | ||||
|   </div> | ||||
| </template> | ||||
| 
 | ||||
| <style scoped> | ||||
| .content { | ||||
|   overflow-y: auto; | ||||
|   -webkit-overflow-scrolling: touch; | ||||
| } | ||||
| </style> | ||||
| @ -3,8 +3,11 @@ import LiveRoom from '@/pages/LiveRoom/index.client.vue'; | ||||
| import {goodStore} from "@/stores/goods/index.js"; | ||||
| import ItemList from './components/ItemList/index.vue' | ||||
| import Cescribe from './components/Cescribe/index.vue' | ||||
| 
 | ||||
| import {artworkList} from "~/api/goods/index.js"; | ||||
| const {fullLive,getAuctionDetail,auctionDetail,itemList,pageRef,liveRef} = goodStore(); | ||||
| import {ref} from "vue"; | ||||
| 
 | ||||
| const {fullLive, getAuctionDetail, auctionDetail, itemList, pageRef, liveRef} = goodStore(); | ||||
| definePageMeta({ | ||||
|   layout: 'default', | ||||
|   i18n: 'menu.home', | ||||
| @ -12,21 +15,15 @@ definePageMeta({ | ||||
| const changeLive = () => { | ||||
|   fullLive.value = true; | ||||
| }; | ||||
| const initData=async ()=>{ | ||||
|   await getAuctionDetail() | ||||
|  const res= await artworkList({auctionUuid: auctionDetail.value.uuid,...pageRef.value}) | ||||
|   if (res.status===0){ | ||||
|     itemList.value=res.data.data | ||||
|     pageRef.value.itemCount=res.data.count | ||||
|   } | ||||
| } | ||||
| initData() | ||||
| const showBottom = ref(true) | ||||
| 
 | ||||
| </script> | ||||
| 
 | ||||
| <template> | ||||
|   <div class="flex-grow-1"> | ||||
|     <client-only> | ||||
|       <LiveRoom @click="changeLive" :class="['changeLive', fullLive ? 'expanded' : 'collapsed']" ref="liveRef" :fullLive="fullLive"/> | ||||
|       <LiveRoom @click="changeLive" :class="['changeLive', fullLive ? 'expanded' : 'collapsed']" ref="liveRef" | ||||
|                 :fullLive="fullLive"/> | ||||
|     </client-only> | ||||
|     <div v-show="!fullLive" class="bg-#fff"> | ||||
|       <van-tabs sticky animated> | ||||
| @ -39,6 +36,7 @@ initData() | ||||
|       </van-tabs> | ||||
|       <van-back-top right="15vw" bottom="10vh"/> | ||||
|     </div> | ||||
| 
 | ||||
|   </div> | ||||
| </template> | ||||
| <style> | ||||
| @ -47,6 +45,13 @@ initData() | ||||
| } | ||||
| </style> | ||||
| <style scoped lang="scss"> | ||||
| .ellipsis { | ||||
|   display: -webkit-box; | ||||
|   -webkit-box-orient: vertical; | ||||
|   -webkit-line-clamp: 2; | ||||
|   overflow: hidden; | ||||
|   text-overflow: ellipsis; | ||||
| } | ||||
| :deep(.van-swipe__indicator) { | ||||
|   width: 8px; | ||||
|   height: 8px; | ||||
|  | ||||
| @ -92,12 +92,13 @@ const getCode =async () => { | ||||
|  }) | ||||
|   loadingRef.value.loading1=false | ||||
|   if (res.status===0){ | ||||
|     pane.value = 1 | ||||
|     vanSwipeRef.value?.swipeTo(pane.value) | ||||
|     showKeyboard.value=true | ||||
|     startCountdown(); | ||||
| 
 | ||||
| 
 | ||||
|   } | ||||
|   pane.value = 1 | ||||
|   vanSwipeRef.value?.swipeTo(pane.value) | ||||
|   showKeyboard.value=true | ||||
|   startCountdown(); | ||||
| /*  pane.value = 1 | ||||
|   vanSwipeRef.value?.swipeTo(pane.value) | ||||
|   showKeyboard.value=true | ||||
|  | ||||
							
								
								
									
										
											BIN
										
									
								
								app/static/images/logout.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/static/images/logout.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 2.5 KiB | 
							
								
								
									
										
											BIN
										
									
								
								app/static/images/qrcodetext.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								app/static/images/qrcodetext.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 7.3 KiB | 
| @ -3,9 +3,11 @@ export const authStore = createGlobalState(() => { | ||||
|     const token=useLocalStorage('token','') | ||||
|     const RefreshToken=useLocalStorage('RefreshToken','') | ||||
|     const userInfo=useLocalStorage('userInfo',{}) | ||||
|     const fingerprint=useLocalStorage('fingerprint','') | ||||
|     return{ | ||||
|         userInfo, | ||||
|         RefreshToken, | ||||
|         token | ||||
|         token, | ||||
|         fingerprint | ||||
|     } | ||||
| }) | ||||
| @ -1,40 +1,118 @@ | ||||
| import {createGlobalState} from '@vueuse/core' | ||||
| import {artworkList, defaultDetail} from "~/api/goods/index.js"; | ||||
| import { createGlobalState } from '@vueuse/core' | ||||
| import { ref } from 'vue' | ||||
| import { artworkList, defaultDetail, artworkDetail } from "~/api/goods/index.js" | ||||
| 
 | ||||
| export const goodStore = createGlobalState(() => { | ||||
|     // 状态定义
 | ||||
|     const actionDetails = ref({}) | ||||
|     const fullLive = ref(false) | ||||
|     const liveRef = ref(null); | ||||
|     const currentItem=ref({}) | ||||
|     const myArtWorks=ref([]) | ||||
|     const liveRef = ref(null) | ||||
|     const currentItem = ref({}) | ||||
|     const myArtWorks = ref([]) | ||||
|     const pageRef = ref({ | ||||
|         page: 1, | ||||
|         page: 0, | ||||
|         pageSize: 5, | ||||
|         itemCount: 0 | ||||
|     }) | ||||
|     const artWorkDetail=ref(null) | ||||
|     const artWorkDetail = ref(null) | ||||
|     const itemList = ref([]) | ||||
|     const auctionDetail = ref({}) | ||||
|     const loading = ref(false) | ||||
|     const error = ref(null) | ||||
| 
 | ||||
|     // 重置分页
 | ||||
|     const resetPage = () => { | ||||
|         pageRef.value.page = 1 | ||||
|         pageRef.value.pageSize=5 | ||||
|         pageRef.value.itemCount = 0 | ||||
|         itemList.value = [] | ||||
|     } | ||||
| 
 | ||||
|     // 获取拍卖详情
 | ||||
|     const getAuctionDetail = async () => { | ||||
|         const res = await defaultDetail({}) | ||||
|         if (res.status === 0) { | ||||
|             auctionDetail.value = res.data | ||||
|         try { | ||||
|             loading.value = true | ||||
|             const res = await defaultDetail({}) | ||||
|             if (res.status === 0) { | ||||
|                 auctionDetail.value = res.data | ||||
|             } | ||||
|         } catch (err) { | ||||
|             console.error('获取拍卖详情错误:', err) | ||||
|         } finally { | ||||
|             loading.value = false | ||||
|         } | ||||
|     } | ||||
|     const getArtworkList = async (page = pageRef.value) => { | ||||
|         return artworkList({auctionUuid: auctionDetail.value.uuid, ...page}) | ||||
| 
 | ||||
|     // 获取艺术品列表
 | ||||
|     const getArtworkList = async (isRefresh = false) => { | ||||
|         if (isRefresh) { | ||||
|             resetPage() | ||||
|         } | ||||
|         try { | ||||
|             loading.value = true | ||||
|             const res = await artworkList({ | ||||
|                 auctionUuid: auctionDetail.value.uuid, | ||||
|                 page: pageRef.value.page, | ||||
|                 pageSize: pageRef.value.pageSize | ||||
|             }) | ||||
|             console.log('res',res) | ||||
| 
 | ||||
|             if (res.status === 0) { | ||||
|                 const newItems = res.data.data || [] | ||||
| 
 | ||||
|                 if (isRefresh) { | ||||
|                     itemList.value = newItems | ||||
|                 } else { | ||||
|                     itemList.value = [...itemList.value, ...newItems] | ||||
|                 } | ||||
| 
 | ||||
|                 pageRef.value.itemCount = res.data.count || 0 | ||||
|                 return { | ||||
|                     finished: !newItems.length || newItems.length < pageRef.value.pageSize, | ||||
|                     items: newItems | ||||
|                 } | ||||
|             } | ||||
|             return { finished: true, items: [] } | ||||
|         } catch (err) { | ||||
|             console.error('获取艺术品列表错误:', err) | ||||
|             return { finished: true, items: [] } | ||||
|         } finally { | ||||
|             loading.value = false | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // 获取艺术品详情
 | ||||
|     const getArtworkDetail = async (uuid) => { | ||||
|         try { | ||||
|             loading.value = true | ||||
|             const res = await artworkDetail({ uuid }) | ||||
|             if (res.status === 0) { | ||||
|                 artWorkDetail.value = res.data | ||||
|             } | ||||
|         } catch (err) { | ||||
|             console.error('获取艺术品详情错误:', err) | ||||
|         } finally { | ||||
|             loading.value = false | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|         myArtWorks, | ||||
|         currentItem, | ||||
|         artWorkDetail, | ||||
|         liveRef, | ||||
|         pageRef, | ||||
|         getArtworkList, | ||||
|         auctionDetail, | ||||
|         getAuctionDetail, | ||||
|         // 状态
 | ||||
|         actionDetails, | ||||
|         fullLive, | ||||
|         liveRef, | ||||
|         currentItem, | ||||
|         myArtWorks, | ||||
|         pageRef, | ||||
|         artWorkDetail, | ||||
|         itemList, | ||||
|         fullLive | ||||
|         auctionDetail, | ||||
|         loading, | ||||
|         error, | ||||
|         // 方法
 | ||||
|         getAuctionDetail, | ||||
|         getArtworkList, | ||||
|         getArtworkDetail, | ||||
|         resetPage | ||||
|     } | ||||
| }) | ||||
							
								
								
									
										44
									
								
								app/utils/format.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								app/utils/format.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | ||||
| /** | ||||
|  * 格式化价格 | ||||
|  * @param {number} price - 价格数值 | ||||
|  * @param {string} currency - 货币符号,默认为 ¥ | ||||
|  * @returns {string} 格式化后的价格字符串 | ||||
|  */ | ||||
| export const formatPrice = (price, currency = '¥') => { | ||||
|   if (price == null || isNaN(price)) return `${currency}0` | ||||
|    | ||||
|   // 将价格转换为数字
 | ||||
|   const numPrice = Number(price) | ||||
|    | ||||
|   // 处理小数点,保留两位小数
 | ||||
|   const formattedPrice = numPrice.toFixed(2) | ||||
|    | ||||
|   // 添加千位分隔符
 | ||||
|   const parts = formattedPrice.toString().split('.') | ||||
|   parts[0] = parts[0].replace(/\B(?=(\d{3})+(?!\d))/g, ',') | ||||
|    | ||||
|   return `${currency}${parts.join('.')}` | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 格式化数字 | ||||
|  * @param {number} num - 需要格式化的数字 | ||||
|  * @returns {string} 格式化后的数字字符串 | ||||
|  */ | ||||
| export const formatNumber = (num) => { | ||||
|   if (num == null || isNaN(num)) return '0' | ||||
|    | ||||
|   return num.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ',') | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * 格式化百分比 | ||||
|  * @param {number} value - 需要格式化的值 | ||||
|  * @param {number} decimals - 小数位数,默认为 2 | ||||
|  * @returns {string} 格式化后的百分比字符串 | ||||
|  */ | ||||
| export const formatPercent = (value, decimals = 2) => { | ||||
|   if (value == null || isNaN(value)) return '0%' | ||||
|    | ||||
|   return `${(Number(value) * 100).toFixed(decimals)}%` | ||||
| }  | ||||
| @ -10,9 +10,7 @@ const publicConfig = Object.entries(process.env) | ||||
|     config[key] = value | ||||
|     return config | ||||
|   }, {}) | ||||
| 
 | ||||
| export default defineNuxtConfig({ | ||||
| 
 | ||||
|   hooks: { | ||||
|     'pages:extend'(pages) { | ||||
|       const indexPage = pages.findIndex(page => page.path === '/') | ||||
| @ -22,7 +20,7 @@ export default defineNuxtConfig({ | ||||
|       pages.push({ | ||||
|         name: 'home', | ||||
|         path: '/', | ||||
|         file: '~/pages/home/index.vue' | ||||
|         file: '@/pages/home/index.vue' | ||||
|       }) | ||||
|     } | ||||
|   }, | ||||
|  | ||||
| @ -6,7 +6,7 @@ | ||||
|   "scripts": { | ||||
|     "build": "nuxt build", | ||||
|     "dev": "cross-env ENV_FILE=.env.test nuxt dev", | ||||
|     "dev:prod": "NODE_ENV=production ENV_FILE=.env.prod nuxt dev", | ||||
|     "dev:prod": "cross-env ENV_FILE=.env.prod nuxt dev", | ||||
|     "build:test": "cross-env ENV_FILE=.env.test nuxt build", | ||||
|     "build:prod": "cross-env ENV_FILE=.env.prod nuxt build", | ||||
|     "generate": "nuxt generate", | ||||
| @ -17,6 +17,7 @@ | ||||
|     "start": "cross-env ENV_FILE=.env.prod nuxt start" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@fingerprintjs/fingerprintjs": "^4.5.1", | ||||
|     "@nuxtjs/color-mode": "^3.5.2", | ||||
|     "@nuxtjs/i18n": "^9.1.1", | ||||
|     "@vueuse/core": "^12.4.0", | ||||
|  | ||||
| @ -11,6 +11,9 @@ importers: | ||||
| 
 | ||||
|   .: | ||||
|     dependencies: | ||||
|       '@fingerprintjs/fingerprintjs': | ||||
|         specifier: ^4.5.1 | ||||
|         version: 4.5.1 | ||||
|       '@nuxtjs/color-mode': | ||||
|         specifier: ^3.5.2 | ||||
|         version: 3.5.2(magicast@0.3.5)(rollup@4.29.1) | ||||
| @ -714,6 +717,9 @@ packages: | ||||
|     resolution: {integrity: sha512-zSkKow6H5Kdm0ZUQUB2kV5JIXqoG0+uH5YADhaEHswm664N9Db8dXSi0nMJpacpMf+MyyglF1vnZohpEg5yUtg==} | ||||
|     engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} | ||||
| 
 | ||||
|   '@fingerprintjs/fingerprintjs@4.5.1': | ||||
|     resolution: {integrity: sha512-hKJaRoLHNeUUPhb+Md3pTlY/Js2YR4aXjroaDHpxrjoM8kGnEFyZVZxXo6l3gRyKnQN52Uoqsycd3M73eCdMzw==} | ||||
| 
 | ||||
|   '@humanfs/core@0.19.1': | ||||
|     resolution: {integrity: sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==} | ||||
|     engines: {node: '>=18.18.0'} | ||||
| @ -3792,6 +3798,9 @@ packages: | ||||
|   ts-interface-checker@0.1.13: | ||||
|     resolution: {integrity: sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==} | ||||
| 
 | ||||
|   tslib@2.8.1: | ||||
|     resolution: {integrity: sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==} | ||||
| 
 | ||||
|   tsx@4.19.2: | ||||
|     resolution: {integrity: sha512-pOUl6Vo2LUq/bSa8S5q7b91cgNSjctn9ugq/+Mvow99qW6x/UZYwzxy/3NmqoT66eHYfCVvFvACC58UBPFf28g==} | ||||
|     engines: {node: '>=18.0.0'} | ||||
| @ -4719,6 +4728,10 @@ snapshots: | ||||
|     dependencies: | ||||
|       levn: 0.4.1 | ||||
| 
 | ||||
|   '@fingerprintjs/fingerprintjs@4.5.1': | ||||
|     dependencies: | ||||
|       tslib: 2.8.1 | ||||
| 
 | ||||
|   '@humanfs/core@0.19.1': {} | ||||
| 
 | ||||
|   '@humanfs/node@0.16.6': | ||||
| @ -8373,6 +8386,8 @@ snapshots: | ||||
| 
 | ||||
|   ts-interface-checker@0.1.13: {} | ||||
| 
 | ||||
|   tslib@2.8.1: {} | ||||
| 
 | ||||
|   tsx@4.19.2: | ||||
|     dependencies: | ||||
|       esbuild: 0.23.1 | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user