index.vue 68 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100
  1. <template>
  2. <view class="product-con">
  3. <up-navbar
  4. class="goods-nav-bar"
  5. :bgColor="`rgba(255, 255, 255, ${opacity})`"
  6. >
  7. <template #left>
  8. <view class="nav-slot">
  9. <uni-icons
  10. @click="goBack"
  11. type="left"
  12. size="22"
  13. color="#606266"
  14. ></uni-icons>
  15. <up-line
  16. direction="column"
  17. :hairline="false"
  18. length="16"
  19. margin="0 8px"
  20. ></up-line>
  21. <uni-icons
  22. @click="goHome"
  23. type="home"
  24. id="home"
  25. size="22"
  26. color="#606266"
  27. ></uni-icons>
  28. </view>
  29. </template>
  30. <template #center> </template>
  31. <template #right>
  32. <!-- 分享 -->
  33. <!-- <view @click="handleShare" class="share share-top">-->
  34. <!-- <uni-icons size="24" type="redo"></uni-icons>-->
  35. <!-- </view>-->
  36. </template>
  37. </up-navbar>
  38. <!-- <view class='iconfont icon-xiangzuo' :style="'top:'+navH/2+'rpx'" @tap='returns'></view> -->
  39. <view>
  40. <scroll-view
  41. :scroll-top="scrollTop"
  42. scroll-y="true"
  43. scroll-with-animation="true"
  44. :style="'height:' + height + 'px;'"
  45. @scroll="handleScroll"
  46. >
  47. <view id="past0">
  48. <product-con-swiper
  49. :imgUrls="sliderImage"
  50. :videoline="productInfo.videoLink"
  51. @click="previewImage(sliderImage)"
  52. >
  53. </product-con-swiper>
  54. <view class="pad30 container">
  55. <view class="wrapper mb30 borRadius14" style="margin-top: 0;">
  56. <view class="product-info">
  57. <view style="width:100%;">
  58. <view class="share acea-row row-between row-bottom">
  59. <view class="money font-color">
  60. <text class="num">¥{{ attr.productSelect.storePrice }}</text>
  61. </view>
  62. <view @click="listenerActionSheet" class="share share-top">
  63. <uni-icons size="24" type="redo"></uni-icons>
  64. </view>
  65. </view>
  66. <view class="label acea-row row-between-wrapper">
  67. <view>工费:{{ attr.productSelect.price || 0 }}元/g</view>
  68. <view>重量:{{ attr.productSelect.weight }}g</view>
  69. <view>
  70. 附加费:{{ attr.productSelect.additionalAmount || "" }}
  71. </view>
  72. <view >
  73. 销量:{{
  74. Number(productInfo?.sales || 0) + productInfo.ficti || "0"
  75. }}件
  76. </view>
  77. </view>
  78. </view>
  79. <!-- <view class="sales-container">-->
  80. <!-- <text class="sales-text">-->
  81. <!-- 销量:{{-->
  82. <!-- Number(productInfo?.sales || 0) + productInfo.ficti || "0"-->
  83. <!-- }}件-->
  84. <!-- </text>-->
  85. <!-- </view>-->
  86. </view>
  87. <view class="introduce" style="margin-bottom: 0;">{{ productInfo.storeName }}</view>
  88. <!-- <view class='coupon acea-row row-between-wrapper' v-if="productInfo.give_integral > 0">
  89. <view class='hide line1 acea-row'>
  90. 赠积分:
  91. <view class='activity'>赠送 {{productInfo.give_integral}} 积分</view>
  92. </view>
  93. </view> -->
  94. <!-- <view-->
  95. <!-- v-if="type == 'normal'"-->
  96. <!-- class="coupon acea-row row-between-wrapper"-->
  97. <!-- @click="handleCoupon"-->
  98. <!-- >-->
  99. <!-- <view class="hide line1 acea-row">-->
  100. <!-- 优惠券:-->
  101. <!-- <view class="activity">-->
  102. <!-- 满{{ coupon.list[0]?.minPrice || "0.00" }}减{{-->
  103. <!-- coupon.list[0]?.money || "0.00"-->
  104. <!-- }}-->
  105. <!-- </view>-->
  106. <!-- </view>-->
  107. <!-- <view class="iconfont icon-jiantou"></view>-->
  108. <!-- </view>-->
  109. </view>
  110. <!-- <view-->
  111. <!-- class="attribute acea-row row-between-wrapper mb30 borRadius14"-->
  112. <!-- @click="selecAttr"-->
  113. <!-- >-->
  114. <!-- <view class="line1"-->
  115. <!-- >{{ attrTxt }}:-->
  116. <!-- <text class="atterTxt">{{ attrValue }}</text>-->
  117. <!-- </view>-->
  118. <!-- <view class="iconfont icon-jiantou"></view>-->
  119. <!-- </view>-->
  120. <view class="product-card" @click="selecAttr">
  121. <!-- 材质和克重行 -->
  122. <view class="material-row">
  123. <view class="material-item" style="width: 60rpx;">
  124. <image style="width: 40rpx;" src="@/static/images/order.png" mode="widthFix"></image>
  125. </view>
  126. <view class="material-item">
  127. <text class="material-value">{{ metalTypeFormatter(productInfo.metalType) }}</text>
  128. <text class="material-label">材质</text>
  129. </view>
  130. <view class="material-item" style="width: 2rpx;">
  131. <view class="line"></view>
  132. </view>
  133. <view class="material-item">
  134. <text class="material-value">{{ selectValue().val }}</text>
  135. <text class="material-label">{{ selectValue().name }}</text>
  136. </view>
  137. <view class="material-item" style="width: 2rpx;">
  138. <view class="line"></view>
  139. </view>
  140. <view class="material-item">
  141. <text class="material-value">{{ attr.productSelect.price || 0 }}元/g</text>
  142. <text class="material-label">工费</text>
  143. </view>
  144. <view class="material-item" style="width: 2rpx;">
  145. <view class="line"></view>
  146. </view>
  147. <view class="material-item">
  148. <text class="material-value">{{ attr.productSelect.additionalAmount || "" }}</text>
  149. <text class="material-label">附加费</text>
  150. </view>
  151. </view>
  152. <!-- 分割线 -->
  153. <view class="divider"></view>
  154. <!-- 规格选择区域 -->
  155. <view class="spec-section" @click="handleSpecSelect">
  156. <view style="width: 60rpx;text-align: center;">
  157. <image style="width: 40rpx;" src="@/static/images/2-001.png" mode="widthFix"></image>
  158. </view>
  159. <view class="spec-info">
  160. <view class="spec-label">请选择规格</view>
  161. </view>
  162. <uni-icons type="right" size="16" color="#999999"></uni-icons>
  163. </view>
  164. </view>
  165. <!-- <view class="row-block mb30 borRadius14">-->
  166. <!-- <view class="row-express">-->
  167. <!-- <view class="left-box">-->
  168. <!-- <uni-icons class="icon" type="cart" size="24"></uni-icons>-->
  169. <!-- <text class="text">48小时送达</text>-->
  170. <!-- </view>-->
  171. <!-- <view class="express-price">-->
  172. <!-- <text class="express-place">广东深圳</text>-->
  173. <!-- &lt;!&ndash; <up-line-->
  174. <!-- color="#ccc"-->
  175. <!-- direction="column"-->
  176. <!-- :hairline="false"-->
  177. <!-- length="14px"-->
  178. <!-- margin="0 8px"-->
  179. <!-- ></up-line> &ndash;&gt;-->
  180. <!-- &lt;!&ndash; <text class="express-place">快递费:8元</text> &ndash;&gt;-->
  181. <!-- </view>-->
  182. <!-- </view>-->
  183. <!-- <up-line color="#ccc" length="100%" margin="10px 0"></up-line>-->
  184. <!-- <view class="tip-text">-->
  185. <!-- <uni-icons size="24" type="hand-up"></uni-icons>-->
  186. <!-- <view class="text">-->
  187. <!-- <text class="t1">买的放心,用的称心</text>-->
  188. <!-- <text class="t2">平台有保障</text>-->
  189. <!-- </view>-->
  190. <!-- </view>-->
  191. <!-- </view>-->
  192. <view class="store-card" @click="toStore">
  193. <view class="left">
  194. <image class="storeImg" :src="sbMerchantInfo.merchantLogo" mode="aspectFit"></image>
  195. </view>
  196. <view class="center">
  197. <view class="name line1">{{sbMerchantInfo.merchantName}}</view>
  198. <view class="desc line1">{{sbMerchantInfo.merchantDescribe}}</view>
  199. </view>
  200. <view class="right">
  201. <uni-icons type="right" size="16" color="#999999"></uni-icons>
  202. </view>
  203. </view>
  204. <view class="product-intro" id="past3">
  205. <view class="title" style="text-align: left;height:80rpx;line-height: 80rpx;font-size: 32rpx;color: #333;">
  206. <!-- <image src="/static/images/xzuo.png"></image>-->
  207. <span class="sp">商品详情</span>
  208. <!-- <image src="/static/images/xyou.png"></image>-->
  209. </view>
  210. <view class="conter" style="border-radius: 16rpx;overflow: hidden;">
  211. <up-parse :content="description"></up-parse>
  212. </view>
  213. </view>
  214. </view>
  215. </view>
  216. <view style="height: 120rpx"></view>
  217. </scroll-view>
  218. </view>
  219. <view class="footer acea-row row-between-wrapper">
  220. <!-- <navigator-->
  221. <!-- open-type="switchTab"-->
  222. <!-- class="animated item"-->
  223. <!-- :class="animated == true ? 'bounceIn' : ''"-->
  224. <!-- url="/pages/mall/index"-->
  225. <!-- hover-class="none"-->
  226. <!-- >-->
  227. <!-- <uni-icons-->
  228. <!-- size="22"-->
  229. <!-- color="#666"-->
  230. <!-- customPrefix="iconfont"-->
  231. <!-- type="icon-shouye"-->
  232. <!-- ></uni-icons>-->
  233. <!-- <view>首页</view>-->
  234. <!-- </navigator>-->
  235. <button
  236. @click="toMessagePage"
  237. open-type="contact"
  238. hover-class="none"
  239. class="item"
  240. >
  241. <uni-icons
  242. size="22"
  243. color="#666"
  244. customPrefix="iconfont"
  245. type="icon-kefu1"
  246. ></uni-icons>
  247. <view>客服</view>
  248. </button>
  249. <button
  250. @click="setCollect"
  251. hover-class="none"
  252. class="item"
  253. >
  254. <uni-icons
  255. size="22"
  256. color="#F8C008"
  257. customPrefix="iconfont"
  258. type="icon-shoucangxuanzhong"
  259. v-if="userCollect"
  260. ></uni-icons>
  261. <uni-icons
  262. size="22"
  263. color="#666"
  264. customPrefix="iconfont"
  265. type="icon-shoucang"
  266. v-else
  267. ></uni-icons>
  268. <view>收藏</view>
  269. </button>
  270. <block v-if="type === 'normal'">
  271. <view
  272. class="animated item"
  273. :class="animated == true ? 'bounceIn' : ''"
  274. url="/pages/order_addcart/order_addcart"
  275. hover-class="none"
  276. @click="toShopCart"
  277. >
  278. <uni-icons
  279. size="22"
  280. color="#666"
  281. customPrefix="iconfont"
  282. type="icon-gouwuche"
  283. class="icon-item"
  284. >
  285. <text v-if="Math.floor(CartCount) > 0" class="num bg-color">{{
  286. CartCount
  287. }}</text>
  288. </uni-icons>
  289. <!-- <view class="iconfont icon-gouwuche1">
  290. <text v-if="Math.floor(CartCount) > 0" class="num bg-color">{{
  291. CartCount
  292. }}</text>
  293. </view> -->
  294. <view>购物车</view>
  295. </view>
  296. <view class="bnt acea-row" v-if="attr.productSelect.stock <= 0">
  297. <form @submit="joinCart" report-submit="true">
  298. <button class="joinCart bnts" form-type="submit">加入购物车</button>
  299. </form>
  300. <form report-submit="true">
  301. <button class="buy bnts bg-color-hui" form-type="submit">
  302. 已售罄
  303. </button>
  304. </form>
  305. </view>
  306. <view class="bnt acea-row" v-else>
  307. <form @submit="joinCart" report-submit="true">
  308. <button class="joinCart bnts" form-type="submit">加入购物车</button>
  309. </form>
  310. <form @submit="goBuy" report-submit="true">
  311. <button class="buy bnts" form-type="submit">立即购买</button>
  312. </form>
  313. </view>
  314. </block>
  315. <!-- <view
  316. class="bnt bntVideo acea-row"
  317. v-if="attr.productSelect.stock <= 0 && type === 'video'"
  318. >
  319. <form report-submit="true">
  320. <button class="buy bnts bg-color-hui" form-type="submit">
  321. 已售罄
  322. </button>
  323. </form>
  324. </view>
  325. <view
  326. class="bnt bntVideo acea-row"
  327. v-if="attr.productSelect.stock > 0 && type === 'video'"
  328. >
  329. <form @submit="goBuy" report-submit="true">
  330. <button class="buy bnts" form-type="submit">立即购买</button>
  331. </form>
  332. </view> -->
  333. </view>
  334. <!-- 组件 -->
  335. <productWindow
  336. :attr="attr"
  337. :isShow="1"
  338. :iSplus="1"
  339. :showPopup="showProductPopup"
  340. @closePopup="closeProductPopup"
  341. @myevent="onMyEvent"
  342. @ChangeAttr="ChangeAttr"
  343. @ChangeCartNum="ChangeCartNum"
  344. @attrVal="attrVal"
  345. @iptCartNum="iptCartNum"
  346. @updatePrice="handlePriceUpdate"
  347. @submit="handleSubmit"
  348. id="product-window"
  349. >
  350. </productWindow>
  351. <couponListWindow
  352. :coupon="coupon"
  353. :showPopup="showCouponPopup"
  354. @close="closeCouponPopup"
  355. @ChangCoupons="ChangCoupons"
  356. @ChangCouponsUseState="ChangCouponsUseState"
  357. @tabCouponType="tabCouponType"
  358. >
  359. </couponListWindow>
  360. <!-- 分享按钮 -->
  361. <view class="generate-posters acea-row row-middle" :class="posters ? 'on' : ''">
  362. <!-- #ifndef MP -->
  363. <view class="item" hover-class='none' v-if="weixinStatus === true" @click="H5ShareBox = true">
  364. <view>
  365. <image class="shareImg" src="@/static/images/wexin.png" mode="widthFix"></image>
  366. </view>
  367. <view class="">发送给朋友</view>
  368. </view>
  369. <!-- #endif -->
  370. <!-- #ifdef MP -->
  371. <view class="item" open-type="share" hover-class='none' @click="goFriend">
  372. <view class="">
  373. <image class="shareImg" src="@/static/images/wexin.png" mode="widthFix"></image>
  374. </view>
  375. <view class="">发送给朋友</view>
  376. </view>
  377. <!-- #endif -->
  378. <view class="item" hover-class='none' @click="goPoster">
  379. <view style="display: flex;justify-content: center;align-items: center;">
  380. <view class="shareImg1">
  381. <image class="img" src="@/static/images/picture.png" mode="widthFix"></image>
  382. </view>
  383. </view>
  384. <view class="">生成海报</view>
  385. </view>
  386. </view>
  387. <view class="mask" v-if="posters" @click="closePosters"></view>
  388. </view>
  389. </template>
  390. <script setup>
  391. import { onReady, onLoad, onShow } from "@dcloudio/uni-app";
  392. import { ref, computed, watch, getCurrentInstance, toRaw } from "vue";
  393. import { useToast } from "@/hooks/useToast";
  394. import { useAppStore } from "@/stores/app.js";
  395. // import uQRCode from '@/js_sdk/Sansnn-uQRCode/uqrcode.js';
  396. import UQRCode from "uqrcodejs";
  397. import { getProductDetail, postCartAdd,collectDel,collectAdd } from "@/api/store.js";
  398. import { spread } from "@/api/user";
  399. import { getCoupons, getQrcode } from "@/api/api.js";
  400. import { getCartCounts } from "@/api/order.js";
  401. import { toLogin } from "@/libs/login.js";
  402. import { imageBase64 } from "@/api/public";
  403. import wechat from "@/libs/wechat.js";
  404. import { getPreOrder } from "@/libs/order";
  405. import productConSwiper from "@/components/productConSwiper";
  406. import couponListWindow from "@/components/couponListWindow";
  407. import productWindow from "@/components/productWindow";
  408. import { isTabBarPage } from "@/utils/util.js";
  409. import $util from "@/utils/util.js";
  410. import { HTTP_REQUEST_URL_SHARE } from "@/config/app.js";
  411. import UniIcons from "../../uni_modules/uni-icons/components/uni-icons/uni-icons.vue";
  412. const { Toast } = useToast();
  413. const app = getApp();
  414. const instance = getCurrentInstance();
  415. const appStore = useAppStore();
  416. const isLogin = computed(() => appStore.isLogin);
  417. const uid = computed(() => appStore.uid);
  418. const coupon = ref({
  419. coupon: false,
  420. type: 1,
  421. list: [],
  422. count: [],
  423. });
  424. const attrTxt = ref("请选择规格");
  425. const attrValue = ref("");
  426. const animated = ref(false);
  427. const id = ref(0);
  428. const productInfo = ref({});
  429. const productValue = ref([]);
  430. const couponList = ref([]);
  431. const cart_num = ref(1);
  432. const isOpen = ref(false);
  433. const storeImage = ref("");
  434. const PromotionCode = ref("");
  435. const posterbackgd = ref("/static/images/posterbackgd.png");
  436. const sharePacket = ref({ isState: true });
  437. const clientHeight = ref("");
  438. const good_list = ref([]);
  439. const CartCount = ref(0);
  440. const posters = ref(false);
  441. const attr = ref({
  442. cartAttr: false,
  443. productAttr: [],
  444. productSelect: {},
  445. });
  446. const description = ref("");
  447. const navActive = ref(0);
  448. const activityH5 = ref([]);
  449. const retunTop = ref(true);
  450. const navH = ref("");
  451. const opacity = ref(0);
  452. const scrollY = ref(0);
  453. const topArr = ref([]);
  454. const height = ref(0);
  455. const heightArr = ref([]);
  456. const lock = ref(false);
  457. const scrollTop = ref(0);
  458. const sliderImage = ref([]);
  459. const canvasStatus = ref(false);
  460. const imagePath = ref("");
  461. const imgTop = ref("");
  462. const errT = ref("");
  463. const homeTop = ref(20);
  464. const userCollect = ref(false);
  465. const returnShow = ref(true);
  466. const type = ref("");
  467. const showProductPopup = ref(false); // 商品规格弹窗
  468. const showCouponPopup = ref(false); // 优惠券弹窗
  469. const handleBtnTpe = ref(""); // "buy" or "cart"
  470. const sbMerchantInfo = ref({});
  471. watch(
  472. isLogin,
  473. (newV) => {
  474. if (newV) {
  475. getCouponList();
  476. getCartCount();
  477. }
  478. },
  479. {
  480. deep: true,
  481. }
  482. );
  483. onLoad((options) => {
  484. const pages = getCurrentPages();
  485. returnShow.value = pages.length > 1;
  486. retunTop.value = pages.length > 1;
  487. navH.value = app.globalData.navHeight;
  488. // #ifdef MP || APP-PLUS
  489. setTimeout(() => {
  490. if (options.spread) {
  491. app.globalData.spread = options.spread;
  492. spread(options.spread).catch(() => {});
  493. }
  494. }, 2000);
  495. // #endif
  496. uni.getSystemInfo({
  497. success(res) {
  498. height.value = res.windowHeight;
  499. },
  500. });
  501. if (!options.scene && !options.id) {
  502. Toast({
  503. title: "缺少参数无法查看商品",
  504. icon: "none",
  505. });
  506. uni.navigateTo({
  507. url: "/pages/mall/index",
  508. });
  509. return;
  510. }
  511. if (options.id || options.scene) {
  512. if (options.scene) {
  513. const qrCodeValue = $util.getUrlParams(decodeURIComponent(options.scene));
  514. const mapeMpQrCodeValue = $util.formatMpQrCodeData(qrCodeValue);
  515. app.globalData.spread = mapeMpQrCodeValue.spread;
  516. id.value = mapeMpQrCodeValue.id;
  517. setTimeout(() => {
  518. spread(mapeMpQrCodeValue.spread).catch(() => {});
  519. }, 2000);
  520. } else {
  521. id.value = options.id;
  522. }
  523. type.value = options.type ?? "normal";
  524. appStore.$patch({
  525. productType: type.value,
  526. });
  527. }
  528. getGoodsDetails();
  529. getCouponList();
  530. });
  531. onShow(() => {
  532. getCartCount();
  533. });
  534. onReady(() => {
  535. // #ifdef MP
  536. const menuButton = uni.getMenuButtonBoundingClientRect();
  537. const query = uni.createSelectorQuery().in(instance.proxy);
  538. query
  539. .select("#home")
  540. .boundingClientRect((data) => {
  541. console.log(data)
  542. homeTop.value = menuButton.top * 2 + menuButton.height - data.height;
  543. })
  544. .exec();
  545. // #endif
  546. });
  547. const iptCartNum = (e) => {
  548. attr.value.productSelect.cart_num = e || 1;
  549. };
  550. const handleScroll = (e) => {
  551. const scrollYVal = e.detail.scrollTop;
  552. const opacityVal = scrollYVal / 350 > 1 ? 1 : scrollYVal / 350;
  553. opacity.value = opacityVal;
  554. scrollY.value = scrollYVal;
  555. if (lock.value) {
  556. lock.value = false;
  557. return;
  558. }
  559. for (let i = 0; i < topArr.value.length; i++) {
  560. if (
  561. scrollYVal <
  562. topArr.value[i] - app.globalData.navHeight / 2 + heightArr.value[i]
  563. ) {
  564. navActive.value = i;
  565. break;
  566. }
  567. }
  568. };
  569. const closeCouponPopup = () => {
  570. showCouponPopup.value = false;
  571. // coupon.value.coupon = false;
  572. };
  573. const ChangeCartNum = (changeValue) => {
  574. let productSelect =
  575. productValue.value[attrValue.value] ||
  576. (attr.value.productAttr.length ? undefined : attr.value.productSelect);
  577. if (!productSelect) return;
  578. const stock = productSelect.stock || 0;
  579. let num = attr.value.productSelect;
  580. num.cart_num = changeValue
  581. ? Math.min(num.cart_num + 1, stock)
  582. : Math.max(num.cart_num - 1, 1);
  583. attr.value.productSelect.cart_num = num.cart_num;
  584. cart_num.value = num.cart_num;
  585. };
  586. const attrVal = (val) => {
  587. attr.value.productAttr[val.indexw].index =
  588. attr.value.productAttr[val.indexw].attrValues[val.indexn];
  589. };
  590. const ChangeAttr = (res) => {
  591. const productSelect = productValue.value[res];
  592. if (productSelect) {
  593. attr.value.productSelect = {
  594. ...attr.value.productSelect,
  595. image: productSelect.image,
  596. sales: productSelect.sales,
  597. weight: productSelect.weight,
  598. price: productSelect.price,
  599. storePrice: productSelect.storePrice,
  600. stock: productSelect.stock,
  601. unique: productSelect.id,
  602. cart_num: 1,
  603. additionalAmount: productSelect.additionalAmount,
  604. };
  605. console.log(res);
  606. attrValue.value = res;
  607. attrTxt.value = "已选择";
  608. } else {
  609. attr.value.productSelect = {
  610. ...attr.value.productSelect,
  611. image: productInfo.value.image,
  612. weight: productSelect.weight,
  613. price: productInfo.value.price,
  614. storePrice: productSelect.storePrice,
  615. stock: 0,
  616. unique: productInfo.value.id,
  617. cart_num: 1,
  618. };
  619. attrValue.value = "";
  620. attrTxt.value = "请选择";
  621. }
  622. };
  623. const ChangCoupons = (coupon) => {
  624. couponList.value = couponList.value.filter((item) => item.id !== coupon.id);
  625. getCouponList();
  626. };
  627. const setClientHeight = () => {
  628. if (!good_list.value.length) return;
  629. const query = uni.createSelectorQuery().in(instance.proxy);
  630. query
  631. .select("#list0")
  632. .fields(
  633. {
  634. size: true,
  635. },
  636. (data) => {
  637. clientHeight.value = data.height + 20;
  638. }
  639. )
  640. .exec();
  641. };
  642. const getGoodsDetails = async () => {
  643. try {
  644. const res = await getProductDetail(id.value, type.value);
  645. const product = res.data.productInfo;
  646. sliderImage.value = JSON.parse(product.sliderImage);
  647. productInfo.value = product;
  648. description.value = product.content;
  649. userCollect.value = res.data.userCollect;
  650. attr.value.productAttr = res.data.productAttr;
  651. productValue.value = res.data.productValue;
  652. sharePacket.value.priceName = res.data.priceName;
  653. sharePacket.value.isState = Math.floor(res.data.priceName) !== 0;
  654. activityH5.value = res.data.activityAllH5 || [];
  655. sbMerchantInfo.value = res.data?.sbMerchant || {};
  656. uni.setNavigationBarTitle({
  657. title: product.storeName.substring(0, 7) + "...",
  658. });
  659. attr.value.productAttr = attr.value.productAttr.map((item) => ({
  660. attrName: item.attrName,
  661. attrValues: item.attrValues.split(","),
  662. id: item.id,
  663. isDel: item.isDel,
  664. productId: item.productId,
  665. type: item.type,
  666. }));
  667. if (isLogin.value) {
  668. // #ifdef H5
  669. // make(uid.value);
  670. // ShareInfo();
  671. getImageBase64(productInfo.value.image);
  672. // #endif
  673. // #ifdef MP
  674. getQrcodeFn();
  675. // #endif
  676. }
  677. // nextTick(() => {
  678. // infoScroll();
  679. // });
  680. // #ifdef MP
  681. imgTop.value = productInfo.value.image;
  682. // #endif
  683. // #ifndef H5
  684. downloadFilestoreImage();
  685. // #endif
  686. DefaultSelect();
  687. } catch (err) {
  688. Toast({ title: err.toString(), icon: "none" });
  689. // uni.navigateBack();
  690. }
  691. };
  692. const DefaultSelect = () => {
  693. let value = [];
  694. const keys = Object.keys(productValue.value);
  695. for (let i = 0; i < keys.length; i++) {
  696. const key = keys[i];
  697. if (productValue.value[key].stock > 0) {
  698. value = attr.value.productAttr.length ? key.split(",") : [];
  699. break;
  700. }
  701. }
  702. attr.value.productAttr.forEach((item, i) => {
  703. item.index = value[i];
  704. });
  705. const productSelect = productValue.value[value.join(",")];
  706. if (productSelect && attr.value.productAttr.length) {
  707. attr.value.productSelect = {
  708. ...attr.value.productSelect,
  709. storeName: productInfo.value.storeName,
  710. image: productSelect.image,
  711. additionalAmount: productSelect.additionalAmount,
  712. sales: productSelect.sales,
  713. weight: productSelect.weight,
  714. price: productSelect.price,
  715. storePrice: productSelect.storePrice,
  716. stock: productSelect.stock,
  717. unique: productSelect.id,
  718. cart_num: 1,
  719. };
  720. attrValue.value = value.join(",");
  721. attrTxt.value = "已选择";
  722. } else if (!productSelect && attr.value.productAttr.length) {
  723. attr.value.productSelect = {
  724. ...attr.value.productSelect,
  725. storeName: productInfo.value.storeName,
  726. additionalAmount: productSelect.additionalAmount,
  727. image: productInfo.value.image,
  728. sales: productSelect.sales,
  729. weight: productSelect.weight,
  730. price: productInfo.value.price,
  731. storePrice: productSelect.storePrice,
  732. stock: 0,
  733. unique: productInfo.value.id,
  734. cart_num: 1,
  735. };
  736. attrValue.value = "";
  737. attrTxt.value = "请选择";
  738. } else if (!productSelect && !attr.value.productAttr.length) {
  739. attr.value.productSelect = {
  740. ...attr.value.productSelect,
  741. storeName: productInfo.value.storeName,
  742. additionalAmount: productSelect.additionalAmount,
  743. image: productInfo.value.image,
  744. sales: productSelect.sales,
  745. weight: productSelect.weight,
  746. price: productInfo.value.price,
  747. storePrice: productSelect.storePrice,
  748. stock: productInfo.value.stock,
  749. unique: productInfo.value.id || "",
  750. cart_num: 1,
  751. };
  752. attrValue.value = "";
  753. attrTxt.value = "请选择";
  754. }
  755. };
  756. const getCouponList = async (type = "") => {
  757. try {
  758. const obj = { page: 1, limit: 20, productId: id.value, type };
  759. const { data } = await getCoupons(obj);
  760. coupon.value.list = data;
  761. } catch (err) {
  762. console.error("getCouponList", err);
  763. }
  764. };
  765. const tabCouponType = (type) => {
  766. coupon.value.type = type;
  767. getCouponList(type);
  768. };
  769. const ChangCouponsUseState = (index) => {
  770. coupon.value.list[index].isUse = true;
  771. coupon.value.coupon = false;
  772. };
  773. const selecAttr = () => {
  774. // attr.value.cartAttr = true;
  775. // isOpen.value = true;
  776. showProductPopup.value = true;
  777. handleBtnTpe.value = "buy";
  778. };
  779. // 购买/加入购物车 提交事件
  780. const handleSubmit = async () => {
  781. if (!isLogin.value) {
  782. return toLogin();
  783. }
  784. const productSelect = productValue.value[attrValue.value];
  785. // if (attrValue.value) {
  786. // attr.value.cartAttr = isOpen.value ? attr.value.cartAttr : true;
  787. // } else {
  788. // attr.value.cartAttr = isOpen.value ? true : !attr.value.cartAttr;
  789. // }
  790. // if (attr.value.cartAttr && !isOpen.value) {
  791. // isOpen.value = true;
  792. // return;
  793. // }
  794. if (
  795. attr.value.productAttr.length &&
  796. productSelect?.stock === 0 &&
  797. isOpen.value
  798. ) {
  799. return Toast({ title: "产品库存不足,请选择其它", icon: "none" });
  800. }
  801. if (handleBtnTpe.value === "buy") {
  802. showProductPopup.value = false;
  803. getPreOrderFn();
  804. } else if (handleBtnTpe.value === "cart") {
  805. try {
  806. const params = {
  807. productId: parseFloat(id.value),
  808. cartNum: parseFloat(attr.value.productSelect.cart_num),
  809. isNew: false,
  810. productAttrUnique:
  811. attr.value.productSelect?.unique ?? productInfo.value.id,
  812. };
  813. await postCartAdd(params);
  814. Toast({
  815. title: "添加购物车成功",
  816. icon: "success",
  817. success: () => getCartCount(true),
  818. });
  819. showProductPopup.value = false;
  820. } catch (res) {
  821. showProductPopup.value = false;
  822. Toast({ title: res.message, icon: "none" });
  823. }
  824. }
  825. };
  826. function handlePriceUpdate(price) {
  827. attr.value.productSelect.storePrice = price;
  828. }
  829. // 点击优惠券事件
  830. const handleCoupon = () => {
  831. if (!isLogin.value) {
  832. toLogin();
  833. } else {
  834. getCouponList(1);
  835. showCouponPopup.value = true;
  836. }
  837. };
  838. const onMyEvent = () => {
  839. attr.value.cartAttr = false;
  840. isOpen.value = false;
  841. };
  842. // 关闭规格弹窗
  843. const closeProductPopup = () => {
  844. showProductPopup.value = false;
  845. attr.value.cartAttr = false;
  846. isOpen.value = false;
  847. };
  848. // 立即购买
  849. const goBuy = () => {
  850. if (!isLogin.value) {
  851. toLogin();
  852. } else {
  853. handleBtnTpe.value = "buy";
  854. showProductPopup.value = true;
  855. }
  856. };
  857. // 加入购物车
  858. const joinCart = (e) => {
  859. if (!isLogin.value) {
  860. toLogin();
  861. } else {
  862. handleBtnTpe.value = "cart";
  863. showProductPopup.value = true;
  864. }
  865. };
  866. // 跳到购物车页
  867. const toShopCart = () => {
  868. if (!isLogin.value) {
  869. toLogin();
  870. } else {
  871. uni.switchTab({
  872. url: "/pages/order_addcart/order_addcart",
  873. });
  874. }
  875. };
  876. // 获取购物车数量统计
  877. const getCartCount = async (isAnima = false) => {
  878. if (!isLogin.value) return;
  879. try {
  880. const res = await getCartCounts(true, "total");
  881. CartCount.value = res.data.count;
  882. if (isAnima) {
  883. animated.value = true;
  884. setTimeout(() => {
  885. animated.value = false;
  886. }, 1000);
  887. }
  888. } catch (err) {
  889. console.error(err);
  890. }
  891. };
  892. // 立即购买 加载预订单
  893. const getPreOrderFn = () => {
  894. const params = {
  895. mallType: 0,
  896. preOrderType: type.value === "normal" ? "buyNow" : "video",
  897. orderDetails: [
  898. {
  899. attrValueId: parseFloat(attr.value.productSelect.unique),
  900. productId: parseFloat(id.value),
  901. productNum: parseFloat(attr.value.productSelect.cart_num),
  902. },
  903. ],
  904. };
  905. getPreOrder(params);
  906. };
  907. const posterImageClose = () => {
  908. canvasStatus.value = false;
  909. };
  910. const setDomain = (url) => {
  911. url = url ? url.toString() : "";
  912. return url.includes("https://") ? url : url.replace("http://", "https://");
  913. };
  914. const downloadFilestoreImage = async () => {
  915. try {
  916. const res = await uni.downloadFile({
  917. url: setDomain(productInfo.value.image),
  918. });
  919. storeImage.value = res.tempFilePath;
  920. } catch {
  921. storeImage.value = "";
  922. }
  923. };
  924. const goFriend = () => {
  925. posters.value = false;
  926. };
  927. const getQrcodeFn = async () => {
  928. try {
  929. const data = {
  930. pid: uid.value,
  931. id: id.value,
  932. path: "pages/goods_details/index",
  933. };
  934. const res = await getQrcode(data);
  935. base64src(res.data.code, (res) => {
  936. PromotionCode.value = res;
  937. });
  938. } catch (err) {
  939. errT.value = err;
  940. }
  941. };
  942. const make = (uid) => {
  943. const href = `${location.href.split("?")[0]}?id=${id.value}&spread=${uid}`;
  944. var qr = new UQRCode();
  945. // 设置二维码内容
  946. qr.data = "https://uqrcode.cn/doc";
  947. // 设置二维码大小,必须与canvas设置的宽高一致
  948. qr.size = 200;
  949. // 设置二维码前景图,可以是路径
  950. qr.foregroundImageSrc =
  951. "";
  952. // 调用制作二维码方法
  953. qr.make();
  954. // qr.make({
  955. // canvasId: 'qrcode',
  956. // text: href,
  957. // size: qrcodeSize.value,
  958. // margin: 10,
  959. // success: (res) => {
  960. // PromotionCode.value = res;
  961. // },
  962. // fail: () => {
  963. // Toast({ title: '海报二维码生成失败!', icon: 'none' });
  964. // },
  965. // });
  966. };
  967. const getImageBase64 = async (images) => {
  968. try {
  969. const res = await imageBase64({
  970. url: images,
  971. });
  972. imgTop.value = res.data.code;
  973. } catch (err) {
  974. console.error(err);
  975. }
  976. };
  977. const goPoster = async () => {
  978. uni.showLoading({
  979. title: "海报生成中",
  980. mask: true,
  981. });
  982. posters.value = false;
  983. console.log('PromotionCode.value',PromotionCode.value)
  984. console.log('imgTop.value',imgTop.value)
  985. if (!PromotionCode.value) {
  986. uni.hideLoading();
  987. Toast({
  988. title: errT.value,
  989. icon: "none",
  990. });
  991. return;
  992. }
  993. setTimeout(() => {
  994. if (!imgTop.value) {
  995. uni.hideLoading();
  996. Toast({
  997. title: "无法生成商品海报!",
  998. icon: "none",
  999. });
  1000. return;
  1001. }
  1002. }, 1000);
  1003. try {
  1004. const res = await uni.downloadFile({
  1005. url: imgTop.value,
  1006. });
  1007. const arrImages = [
  1008. posterbackgd.value,
  1009. res.tempFilePath,
  1010. PromotionCode.value,
  1011. ];
  1012. const storeName = productInfo.value.storeName;
  1013. const price = productInfo.value.storePrice;
  1014. setTimeout(() => {
  1015. $util.PosterCanvas(
  1016. arrImages,
  1017. storeName,
  1018. price,
  1019. productInfo.value.otPrice,
  1020. (tempFilePath) => {
  1021. imagePath.value = tempFilePath;
  1022. canvasStatus.value = true;
  1023. uni.hideLoading();
  1024. }
  1025. );
  1026. }, 500);
  1027. } catch {
  1028. uni.hideLoading();
  1029. }
  1030. };
  1031. // #ifdef MP
  1032. const savePosterPath = () => {
  1033. uni.getSetting({
  1034. success(res) {
  1035. if (!res.authSetting["scope.writePhotosAlbum"]) {
  1036. uni.authorize({
  1037. scope: "scope.writePhotosAlbum",
  1038. success() {
  1039. uni.saveImageToPhotosAlbum({
  1040. filePath: imagePath.value,
  1041. success() {
  1042. posterImageClose();
  1043. Toast({
  1044. title: "保存成功",
  1045. icon: "success",
  1046. });
  1047. },
  1048. fail() {
  1049. Toast({
  1050. title: "保存失败",
  1051. icon: "none",
  1052. });
  1053. },
  1054. });
  1055. },
  1056. });
  1057. } else {
  1058. uni.saveImageToPhotosAlbum({
  1059. filePath: imagePath.value,
  1060. success() {
  1061. posterImageClose();
  1062. Toast({
  1063. title: "保存成功",
  1064. icon: "success",
  1065. });
  1066. },
  1067. fail() {
  1068. Toast({
  1069. title: "保存失败",
  1070. icon: "none",
  1071. });
  1072. },
  1073. });
  1074. }
  1075. },
  1076. });
  1077. };
  1078. // #endif
  1079. const ShareInfo = async () => {
  1080. const data = productInfo.value;
  1081. let href = location.href;
  1082. if (wechat.isWeixin()) {
  1083. href = href.includes("?")
  1084. ? `${href}&spread=${uid.value}`
  1085. : `${href}?spread=${uid.value}`;
  1086. const configAppMessage = {
  1087. desc: data.storeInfo,
  1088. title: data.storeName,
  1089. link: href,
  1090. imgUrl: data.image,
  1091. };
  1092. try {
  1093. await wechat.wechatEvevt(
  1094. [
  1095. "updateAppMessageShareData",
  1096. "updateTimelineShareData",
  1097. "onMenuShareAppMessage",
  1098. "onMenuShareTimeline",
  1099. ],
  1100. configAppMessage
  1101. );
  1102. } catch (err) {
  1103. console.error(err);
  1104. }
  1105. }
  1106. };
  1107. // 分享至微信
  1108. const handleShare = () => {
  1109. console.log("uni.getSystemInfoSync()", uni.getSystemInfoSync());
  1110. if (uni.getSystemInfoSync().romName !== "HarmonyOS") {
  1111. uni.shareWithSystem({
  1112. href: `${HTTP_REQUEST_URL_SHARE}/share/#/pages/goods_details/index?articleId=${id.value}`,
  1113. title: productInfo.value.storeName,
  1114. success() {
  1115. // 分享完成,请注意此时不一定是成功分享
  1116. },
  1117. fail() {
  1118. // 分享失败
  1119. },
  1120. });
  1121. } else {
  1122. uni.share({
  1123. provider: "weixin",
  1124. scene: "WXSceneSession",
  1125. type: 0,
  1126. href: `${HTTP_REQUEST_URL_SHARE}/share/#/pages/goods_details/index?articleId=${id.value}`,
  1127. title: productInfo.value.storeName,
  1128. imageUrl:
  1129. "https://sb-admin.oss-cn-shenzhen.aliyuncs.com/crmebimage/public/maintain/2025/08/28/eb3953f78a0f468fa2e4c7721c2a6ca2b9pirw5y37.png",
  1130. // imageUrl:"",
  1131. success: function (res) {
  1132. console.log("success:" + JSON.stringify(res));
  1133. },
  1134. fail: function (err) {
  1135. console.log("fail:" + JSON.stringify(err));
  1136. },
  1137. });
  1138. }
  1139. };
  1140. /**
  1141. * 分享打开
  1142. *
  1143. */
  1144. function listenerActionSheet () {
  1145. if (isLogin === false) {
  1146. toLogin();
  1147. } else {
  1148. // #ifdef H5
  1149. if (wechat.isWeixin() === true) {
  1150. weixinStatus = true;
  1151. }
  1152. // #endif
  1153. posters.value = true;
  1154. }
  1155. }
  1156. function closePosters() {
  1157. posters.value = false;
  1158. }
  1159. const goBack = () => {
  1160. if (isTabBarPage()) {
  1161. uni.switchTab({
  1162. url: "/pages/index/index",
  1163. });
  1164. } else {
  1165. uni.navigateBack();
  1166. }
  1167. };
  1168. const goHome = () => {
  1169. uni.switchTab({
  1170. url: "/pages/index/index",
  1171. });
  1172. };
  1173. function toMessagePage() {
  1174. uni.navigateTo({ url: "/pages/users/customer_service_message/index" });
  1175. }
  1176. const previewImage = (item) => {
  1177. // 预览图片,支持多张图片传数组
  1178. uni.previewImage({
  1179. urls: item,
  1180. current: item[0],
  1181. });
  1182. };
  1183. const calcNumPrice = (productSelect) => {
  1184. const { price, cart_num, additionalAmount, weight } = productSelect;
  1185. ``;
  1186. // 计算总价并保留两位小数
  1187. const total =
  1188. Number(price) * Number(weight) * Number(cart_num) + additionalAmount;
  1189. return total.toFixed(2);
  1190. };
  1191. const selectValue = () => {
  1192. let val = '',name='';
  1193. const arr = attrValue.value.split(',');
  1194. // 优先查找 '克重' 或 '重量'
  1195. const weightItem = attr.value.productAttr.find((item, index) => {
  1196. if (item.attrName == '克重' || item.attrName == '重量') {
  1197. val = arr == '' ? item.attrValues : arr[index];
  1198. name = item.attrName;
  1199. return true;
  1200. }
  1201. return false;
  1202. });
  1203. // 如果没有找到 '克重' 或 '重量',取第一个对象
  1204. if (!weightItem && attr.value.productAttr.length > 0) {
  1205. val = arr == '' ? attr.value.productAttr[0].attrValues : arr[0];
  1206. name = attr.value.productAttr[0].attrName;
  1207. }
  1208. return {
  1209. val,
  1210. name
  1211. };
  1212. }
  1213. const metalTypeFormatter = (val)=> {
  1214. let str = '';
  1215. if(val == 1){
  1216. str = '黄金'
  1217. }else if(val == 3){
  1218. str = '铂金'
  1219. }else{
  1220. str = '白银'
  1221. }
  1222. return str;
  1223. }
  1224. const toStore = () => {
  1225. uni.navigateTo({ url: "/pages/merchant/index?merchantId="+sbMerchantInfo.value.id });
  1226. }
  1227. /**
  1228. *
  1229. *
  1230. * 收藏商品
  1231. */
  1232. function setCollect () {
  1233. if (isLogin.value === false) {
  1234. toLogin();
  1235. } else {
  1236. if (userCollect.value) {
  1237. collectDel(productInfo.value.id).then(res => {
  1238. userCollect.value = !userCollect.value;
  1239. })
  1240. } else {
  1241. collectAdd(productInfo.value.id).then(res => {
  1242. userCollect.value = !userCollect.value;
  1243. })
  1244. }
  1245. }
  1246. }
  1247. </script>
  1248. <style scoped lang="scss">
  1249. .product-con {
  1250. height: 100%;
  1251. .mask {
  1252. z-index: 88;
  1253. }
  1254. .footer {
  1255. padding: 0 20rpx 0 30rpx;
  1256. position: fixed;
  1257. bottom: 0;
  1258. width: 100%;
  1259. box-sizing: border-box;
  1260. height: 100rpx;
  1261. background-color: #fff;
  1262. z-index: 277;
  1263. border-top: 1rpx solid #f0f0f0;
  1264. text-align: center;
  1265. .item {
  1266. font-size: 18rpx;
  1267. color: #666;
  1268. .iconfont {
  1269. text-align: center;
  1270. font-size: 40rpx;
  1271. &.icon-shoucang1 {
  1272. color: #f00;
  1273. }
  1274. }
  1275. .icon-item {
  1276. font-size: 40rpx;
  1277. position: relative;
  1278. .num {
  1279. color: #fff;
  1280. position: absolute;
  1281. font-size: 18rpx;
  1282. padding: 2rpx 8rpx 3rpx;
  1283. border-radius: 200rpx;
  1284. top: -10rpx;
  1285. right: -10rpx;
  1286. }
  1287. }
  1288. }
  1289. .bnt {
  1290. width: 444rpx;
  1291. height: 76rpx;
  1292. .bnts {
  1293. width: 222rpx;
  1294. text-align: center;
  1295. line-height: 76rpx;
  1296. color: #fff;
  1297. font-size: 28rpx;
  1298. }
  1299. .joinCart {
  1300. border-radius: 50rpx 0 0 50rpx;
  1301. background-image: linear-gradient(to right, #fea10f 0%, #fa8013 100%);
  1302. }
  1303. .buy {
  1304. border-radius: 0 50rpx 50rpx 0;
  1305. background-image: linear-gradient(to right, #fa6514 0%, #e93323 100%);
  1306. }
  1307. }
  1308. }
  1309. .store-info {
  1310. margin-top: 20rpx;
  1311. background-color: #fff;
  1312. .title {
  1313. padding: 0 30rpx;
  1314. font-size: 28rpx;
  1315. color: #282828;
  1316. height: 80rpx;
  1317. line-height: 80rpx;
  1318. border-bottom: 1px solid #f5f5f5;
  1319. }
  1320. .info {
  1321. padding: 0 30rpx;
  1322. height: 126rpx;
  1323. .picTxt {
  1324. width: 615rpx;
  1325. .pictrue {
  1326. width: 76rpx;
  1327. height: 76rpx;
  1328. image {
  1329. width: 100%;
  1330. height: 100%;
  1331. border-radius: 6rpx;
  1332. }
  1333. }
  1334. .text {
  1335. width: 522rpx;
  1336. .name {
  1337. font-size: 30rpx;
  1338. color: #282828;
  1339. }
  1340. .address {
  1341. font-size: 24rpx;
  1342. color: #666;
  1343. margin-top: 3rpx;
  1344. .iconfont {
  1345. color: #707070;
  1346. font-size: 18rpx;
  1347. margin-left: 10rpx;
  1348. }
  1349. .addressTxt {
  1350. max-width: 480rpx;
  1351. }
  1352. }
  1353. }
  1354. }
  1355. .iconfont {
  1356. font-size: 40rpx;
  1357. }
  1358. }
  1359. }
  1360. .superior {
  1361. background-color: #fff;
  1362. margin-top: 30rpx;
  1363. padding: 0 24rpx 30rpx 24rpx;
  1364. .title {
  1365. height: 98rpx;
  1366. image {
  1367. width: 20rpx;
  1368. height: 20rpx;
  1369. }
  1370. .titleTxt {
  1371. margin: 0 10rpx;
  1372. font-size: 30rpx;
  1373. color: #333333;
  1374. // background-image: linear-gradient(to right, #f57a37 0%, #f21b07 100%);
  1375. // -webkit-background-clip: text;
  1376. // -webkit-text-fill-color: transparent;
  1377. }
  1378. }
  1379. .slider-banner {
  1380. width: 100%;
  1381. margin: 0 auto;
  1382. position: relative;
  1383. swiper,
  1384. swiper-item {
  1385. height: 100%;
  1386. width: 100%;
  1387. }
  1388. .list {
  1389. width: 100%;
  1390. .item {
  1391. width: 198rpx;
  1392. margin: 0 22rpx 30rpx 0;
  1393. font-size: 26rpx;
  1394. &:nth-of-type(3n) {
  1395. margin-right: 0;
  1396. }
  1397. .pictrue {
  1398. position: relative;
  1399. width: 100%;
  1400. height: 198rpx;
  1401. image {
  1402. width: 100%;
  1403. height: 100%;
  1404. border-radius: 6rpx;
  1405. }
  1406. }
  1407. .name {
  1408. color: #282828;
  1409. margin-top: 12rpx;
  1410. }
  1411. }
  1412. }
  1413. .swiper-pagination-bullet {
  1414. background-color: #999;
  1415. }
  1416. .swiper-pagination-bullet-active {
  1417. background-color: $theme-color;
  1418. }
  1419. }
  1420. }
  1421. }
  1422. .activityName {
  1423. line-height: 44rpx;
  1424. }
  1425. .bntVideo {
  1426. width: auto !important;
  1427. .buy {
  1428. border-radius: 50rpx !important;
  1429. }
  1430. }
  1431. .row-block {
  1432. background-color: #fff;
  1433. padding: 20rpx;
  1434. font-size: 0.8125rem;
  1435. color: #000;
  1436. .row-express {
  1437. // width: 100%;
  1438. display: flex;
  1439. justify-content: space-between;
  1440. // margin: 20rpx 10rpx 10rpx;
  1441. .left-box {
  1442. .icon {
  1443. vertical-align: middle;
  1444. }
  1445. .text {
  1446. margin: 0 0 0 14rpx;
  1447. font-size: 28rpx;
  1448. vertical-align: middle;
  1449. }
  1450. }
  1451. .express-price {
  1452. display: flex;
  1453. align-items: center;
  1454. }
  1455. }
  1456. .tip-text {
  1457. display: flex;
  1458. align-items: center;
  1459. .text {
  1460. display: flex;
  1461. flex-direction: column;
  1462. margin: 0 0 0 20rpx;
  1463. .t2 {
  1464. font-size: 24rpx;
  1465. color: #666;
  1466. }
  1467. }
  1468. }
  1469. }
  1470. .attribute {
  1471. .line1 {
  1472. width: 600rpx;
  1473. }
  1474. }
  1475. .chat-btn {
  1476. background-color: antiquewhite !important;
  1477. }
  1478. .activity_pin,
  1479. .activity_miao,
  1480. .activity_kan {
  1481. width: auto;
  1482. height: 44rpx;
  1483. line-height: 44rpx;
  1484. padding: 0 15rpx;
  1485. opacity: 1;
  1486. border-radius: 22rpx;
  1487. }
  1488. .activity_pin {
  1489. background: linear-gradient(
  1490. 90deg,
  1491. rgba(233, 51, 35, 1) 0%,
  1492. rgba(250, 101, 20, 1) 100%
  1493. );
  1494. }
  1495. .activity_miao {
  1496. background: linear-gradient(
  1497. 90deg,
  1498. rgba(250, 102, 24, 1) 0%,
  1499. rgba(254, 161, 15, 1) 100%
  1500. );
  1501. margin-left: 19rpx;
  1502. }
  1503. .activity_kan {
  1504. background: linear-gradient(
  1505. 90deg,
  1506. rgba(254, 159, 15, 1) 0%,
  1507. rgba(254, 178, 15, 1) 100%
  1508. );
  1509. margin-left: 19rpx;
  1510. }
  1511. .iconfonts {
  1512. color: #fff !important;
  1513. font-size: 28rpx;
  1514. }
  1515. .activity_title {
  1516. font-size: 24rpx;
  1517. color: #fff;
  1518. }
  1519. .mask {
  1520. position: fixed;
  1521. top: 0;
  1522. left: 0;
  1523. right: 0;
  1524. bottom: 0;
  1525. background-color: rgba(0, 0, 0, 0.6);
  1526. z-index: 9;
  1527. &.mask {
  1528. z-index: 300 !important;
  1529. }
  1530. }
  1531. .head-bar {
  1532. background: #fff;
  1533. }
  1534. .generate-posters {
  1535. width: 100%;
  1536. height: 170rpx;
  1537. background-color: #fff;
  1538. position: fixed;
  1539. left: 0;
  1540. bottom: 0;
  1541. z-index: 388;
  1542. transform: translate3d(0, 100%, 0);
  1543. transition: all 0.3s cubic-bezier(0.25, 0.5, 0.5, 0.9);
  1544. border-top: 1rpx solid #eee;
  1545. &.on {
  1546. transform: translate3d(0, 0, 0);
  1547. }
  1548. .item {
  1549. flex: 50%;
  1550. text-align: center;
  1551. font-size: 30rpx;
  1552. .iconfont {
  1553. font-size: 80rpx;
  1554. color: #5eae72;
  1555. &.icon-haibao {
  1556. color: #5391f1;
  1557. }
  1558. }
  1559. }
  1560. }
  1561. .pictrue_log {
  1562. width: 80upx;
  1563. height: 40upx;
  1564. border-radius: 10upx 0 12upx 0;
  1565. line-height: 40upx;
  1566. font-size: 24upx;
  1567. }
  1568. .pictrue_log_class {
  1569. z-index: 3;
  1570. background: linear-gradient(
  1571. 90deg,
  1572. rgba(246, 122, 56, 1) 0%,
  1573. rgba(241, 27, 9, 1) 100%
  1574. );
  1575. opacity: 1;
  1576. position: absolute;
  1577. top: 0;
  1578. left: 0;
  1579. color: #fff;
  1580. text-align: center;
  1581. }
  1582. .navbar {
  1583. position: fixed;
  1584. background-color: #fff;
  1585. top: 0;
  1586. left: 0;
  1587. z-index: 99;
  1588. width: 100%;
  1589. .navbarH {
  1590. position: relative;
  1591. .navbarCon {
  1592. position: absolute;
  1593. bottom: 0;
  1594. height: 100rpx;
  1595. width: 100%;
  1596. }
  1597. }
  1598. .header {
  1599. height: 96rpx;
  1600. font-size: 30rpx;
  1601. color: #050505;
  1602. background-color: #fff;
  1603. /* #ifdef MP */
  1604. padding-right: 95rpx;
  1605. /* #endif */
  1606. .item {
  1607. position: relative;
  1608. margin: 0 25rpx;
  1609. &.on:before {
  1610. position: absolute;
  1611. width: 60rpx;
  1612. height: 5rpx;
  1613. background-repeat: no-repeat;
  1614. content: "";
  1615. background-image: linear-gradient(to right, #ff3366 0%, #ff6533 100%);
  1616. bottom: -10rpx;
  1617. left: 50%;
  1618. margin-left: -28rpx;
  1619. }
  1620. }
  1621. }
  1622. }
  1623. .icon-xiangzuo {
  1624. margin-top: var(--status-bar-height);
  1625. /* #ifdef H5 */
  1626. top: 20rpx !important;
  1627. /* #endif */
  1628. color: #000;
  1629. position: fixed;
  1630. font-size: 36rpx;
  1631. width: 100rpx;
  1632. height: 56rpx;
  1633. line-height: 54rpx;
  1634. z-index: 1000;
  1635. left: -5rpx;
  1636. }
  1637. .share-box {
  1638. z-index: 1000;
  1639. position: fixed;
  1640. left: 0;
  1641. top: 0;
  1642. width: 100%;
  1643. height: 100%;
  1644. image {
  1645. width: 100%;
  1646. height: 100%;
  1647. }
  1648. }
  1649. .pro-wrapper {
  1650. .iconn {
  1651. background-image: url("");
  1652. width: 100rpx;
  1653. height: 100rpx;
  1654. background-repeat: no-repeat;
  1655. background-size: 100% 100%;
  1656. margin: 0 auto;
  1657. &.iconn1 {
  1658. background-image: url("");
  1659. }
  1660. }
  1661. }
  1662. .canvas {
  1663. position: fixed;
  1664. z-index: -5;
  1665. opacity: 0;
  1666. }
  1667. .poster-pop {
  1668. position: fixed;
  1669. width: 450rpx;
  1670. height: 714rpx;
  1671. top: 50%;
  1672. left: 50%;
  1673. transform: translateX(-50%);
  1674. margin-top: -432rpx;
  1675. z-index: 399;
  1676. image {
  1677. width: 100%;
  1678. height: 100%;
  1679. display: block;
  1680. }
  1681. .close {
  1682. width: 46rpx;
  1683. height: 75rpx;
  1684. position: fixed;
  1685. right: 0;
  1686. top: -73rpx;
  1687. display: block;
  1688. }
  1689. .save-poster {
  1690. background-color: #df2d0a;
  1691. font-size: 22rpx;
  1692. color: #fff;
  1693. text-align: center;
  1694. height: 76rpx;
  1695. line-height: 76rpx;
  1696. width: 100%;
  1697. }
  1698. .keep {
  1699. color: #fff;
  1700. text-align: center;
  1701. font-size: 25rpx;
  1702. margin-top: 10rpx;
  1703. }
  1704. }
  1705. button {
  1706. padding: 0;
  1707. margin: 0;
  1708. line-height: normal;
  1709. background-color: #fff;
  1710. &::after {
  1711. border: 0;
  1712. }
  1713. }
  1714. action-sheet-item {
  1715. padding: 0;
  1716. height: 240rpx;
  1717. align-items: center;
  1718. display: flex;
  1719. }
  1720. .contact {
  1721. font-size: 16px;
  1722. width: 50%;
  1723. background-color: #fff;
  1724. padding: 8rpx 0;
  1725. border-radius: 0;
  1726. margin: 0;
  1727. line-height: 2;
  1728. &::after {
  1729. border: none;
  1730. }
  1731. }
  1732. .action-sheet {
  1733. font-size: 17px;
  1734. line-height: 1.8;
  1735. width: 50%;
  1736. position: absolute;
  1737. top: 0;
  1738. right: 0;
  1739. padding: 25rpx 0;
  1740. }
  1741. .share-top {
  1742. background-color: rgba(255, 255, 255, 0.8);
  1743. padding: 5rpx 10rpx;
  1744. border-radius: 40rpx;
  1745. }
  1746. .container{
  1747. margin-top: -40rpx;
  1748. position: relative;
  1749. z-index: 2;
  1750. background-color: #F9F7F0;
  1751. border-radius: 14rpx 14rpx 0 0;
  1752. padding-top: 30rpx; /* 顶部内边距 */
  1753. }
  1754. .product-info {
  1755. display: flex;
  1756. justify-content: space-between;
  1757. align-items: center;
  1758. width: 100%;
  1759. }
  1760. .sales-container {
  1761. width: 20%;
  1762. display: flex;
  1763. justify-content: flex-end;
  1764. align-items: center;
  1765. text-align: right;
  1766. }
  1767. .sales-text {
  1768. font-size: 24rpx;
  1769. color: #666;
  1770. text-align: right;
  1771. white-space: nowrap;
  1772. }
  1773. .product-card {
  1774. background: #ffffff;
  1775. border-radius: 16rpx;
  1776. padding: 20rpx;
  1777. }
  1778. .material-row {
  1779. display: flex;
  1780. justify-content: flex-start;
  1781. align-items: center;
  1782. padding: 24rpx 0;
  1783. }
  1784. .material-item {
  1785. width: 22%;
  1786. display: flex;
  1787. flex-direction: column;
  1788. align-items: center;
  1789. }
  1790. .material-item .line{
  1791. height: 72rpx;
  1792. width: 2rpx;
  1793. background-color: #F1F3F8;
  1794. }
  1795. .material-label {
  1796. font-size: 26rpx;
  1797. color: #999;
  1798. margin-bottom: 8rpx;
  1799. }
  1800. .material-value {
  1801. font-size: 28rpx;
  1802. color: #333;
  1803. font-weight: 500;
  1804. }
  1805. .divider {
  1806. height: 1rpx;
  1807. background-color: #F1F3F8;
  1808. }
  1809. .craft-fee-section {
  1810. padding: 24rpx 0;
  1811. }
  1812. .craft-fee-header {
  1813. display: flex;
  1814. justify-content: space-between;
  1815. align-items: center;
  1816. margin-bottom: 16rpx;
  1817. }
  1818. .craft-fee-label {
  1819. font-size: 28rpx;
  1820. color: #333;
  1821. font-weight: 500;
  1822. }
  1823. .unit-price {
  1824. background: #f8f8f8;
  1825. padding: 8rpx 16rpx;
  1826. border-radius: 20rpx;
  1827. }
  1828. .unit-price-text {
  1829. font-size: 24rpx;
  1830. color: #666;
  1831. }
  1832. .total-fee {
  1833. text-align: right;
  1834. }
  1835. .total-fee-text {
  1836. font-size: 32rpx;
  1837. color: #e4393c;
  1838. font-weight: 600;
  1839. }
  1840. .spec-section {
  1841. display: flex;
  1842. justify-content: space-between;
  1843. align-items: center;
  1844. padding-top: 24rpx;
  1845. min-height: 120rpx;
  1846. }
  1847. .spec-info {
  1848. width: 100%;
  1849. display: flex;
  1850. justify-items: space-between;
  1851. }
  1852. .spec-label {
  1853. font-size: 28rpx;
  1854. color: #333;
  1855. font-weight: 500;
  1856. margin-bottom: 8rpx;
  1857. }
  1858. .spec-hint {
  1859. font-size: 24rpx;
  1860. color: #999;
  1861. float: right;
  1862. }
  1863. .store-card{
  1864. background: #ffffff;
  1865. border-radius: 16rpx;
  1866. padding: 20rpx;
  1867. display: flex;
  1868. justify-content: center;
  1869. align-items: center;
  1870. margin-top: 30rpx;
  1871. .left{
  1872. width: 140rpx;
  1873. .storeImg{
  1874. width: 100rpx;
  1875. height: 100rpx;
  1876. }
  1877. }
  1878. .center{
  1879. width: 70%;
  1880. .name{
  1881. font-size: 32rpx;
  1882. color: #333;line-height: 60rpx;
  1883. }
  1884. .desc{
  1885. font-size: 28rpx;
  1886. color: #666;
  1887. }
  1888. }
  1889. }
  1890. .product-intro{
  1891. background: #ffffff;
  1892. border-radius: 16rpx;
  1893. padding: 20rpx;
  1894. margin-top: 30rpx;
  1895. }
  1896. .shareImg{
  1897. width: 70rpx;
  1898. height: 70rpx;
  1899. border-radius: 50%;
  1900. }
  1901. .shareImg1{
  1902. width: 70rpx;
  1903. height: 70rpx;
  1904. border-radius: 50%;
  1905. display: flex;
  1906. justify-content: center;
  1907. align-items: center;
  1908. background-color: #5590EB;
  1909. .img{
  1910. width: 40rpx;
  1911. }
  1912. }
  1913. </style>
  1914. <style>
  1915. .goods-nav-bar {
  1916. background-color: rgba(255, 255, 255, 0.3) !important;
  1917. }
  1918. .nav-slot {
  1919. padding: 5rpx 10rpx !important;
  1920. border-radius: 40rpx;
  1921. border: 1px solid #ccc;
  1922. display: flex;
  1923. justify-content: space-between;
  1924. align-items: center;
  1925. background-color: rgba(255, 255, 255, 0.8);
  1926. }
  1927. </style>