index.vue 70 KB

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