goods.vue 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. <template>
  2. <div class="min-h-screen w-full">
  3. <div class="relative overflow-hidden py-4 pb-10 bg-[#fdedc5]">
  4. <img
  5. src="https://oss.hhmdtech.com/merchant/RmW4frEmQRr99JJ4RJ5hwdZR8FYDhTe8PKn6inRBijRJMRcA.png"
  6. mode="widthFix"
  7. class="absolute left-0 top-0 w-full"
  8. />
  9. <div class="relative">
  10. <div class="h-full w-full overflow-hidden px-4">
  11. <div
  12. class="flex h-full w-full rounded bg-[#ffc72f] transition-all duration-150"
  13. >
  14. <div class="flex-1 p-4">
  15. <div class="mb-2 max-w-[8em] truncate whitespace-nowrap text-2xl">
  16. {{ goodsDetail.memberCardName }}
  17. </div>
  18. <div
  19. v-if="goodsDetail?.originPrice"
  20. class="text-gray-500 line-through"
  21. >
  22. <span>¥</span>
  23. <span>
  24. {{ parseFloat((goodsDetail?.originPrice / 100).toFixed(2)) }}
  25. </span>
  26. <span>/{{ goodsDetail.memberCard?.lifespan }}天</span>
  27. </div>
  28. <div :class="goodsDetail?.originPrice ? 'pb-3' : 'py-3'">
  29. <span>¥</span>
  30. <span class="text-2xl">
  31. {{ parseFloat((goodsDetail?.price / 100).toFixed(2)) }}
  32. </span>
  33. <span>/{{ goodsDetail.memberCard?.lifespan }}天</span>
  34. </div>
  35. <div class="text-base text-gray-500">{{ goodsDetail.title }}</div>
  36. </div>
  37. <img src="@/assets/icon-vip.png" class="m-4 h-12 w-12" />
  38. </div>
  39. </div>
  40. </div>
  41. </div>
  42. <div class="relative -top-6 rounded rounded-t-2xl bg-white z-10">
  43. <div class="p-4 pb-20">
  44. <div class="" v-html="goodsDetail.saleSlogan"></div>
  45. </div>
  46. </div>
  47. <div
  48. class="z-[2000] fixed bottom-6 left-0 flex w-full items-center justify-center"
  49. >
  50. <div
  51. class="flex w-10/12 items-center justify-center overflow-hidden rounded-xl bg-[#2c2c2c]"
  52. >
  53. <button
  54. class="flex-1 whitespace-nowrap border-0 bg-[#2c2c2c] p-4 text-sm text-[#e59223] after:hidden"
  55. @click="buyHistory"
  56. >
  57. <span class="text-base">购买记录</span>
  58. </button>
  59. <div class="h-5 w-px bg-[#e59223]"></div>
  60. <button
  61. class="flex-1 whitespace-nowrap border-0 bg-[#2c2c2c] p-4 text-sm text-[#e59223] after:hidden"
  62. @click="buy"
  63. >
  64. <span class="text-base">立即购买</span>
  65. </button>
  66. </div>
  67. </div>
  68. <PaymentDialog
  69. :show="showDialog"
  70. :onClose="handleClose"
  71. :onConfirm="handleConfirm"
  72. />
  73. <PaymentHistoryDialog :show="showHistory" :openId="openId" />
  74. </div>
  75. </template>
  76. <script setup>
  77. import { ref, reactive, onMounted, computed, watch, provide } from "vue";
  78. import PaymentDialog from "./components/PaymentDialog.vue";
  79. import PaymentHistoryDialog from "./components/PaymentHistoryDialog.vue";
  80. import { useRoute, useRouter } from "vue-router";
  81. import { request } from "@/utils";
  82. const [route, router] = [useRoute(), useRouter()];
  83. const goodsDetail = ref({});
  84. provide("goodsDetail", goodsDetail);
  85. const getGoodsDetail = async () => {
  86. const { data } = await request.get(
  87. `archivesService/member/promotionGoods/detail`,
  88. {
  89. params: {
  90. promotionGoodsId: route.query.promotionGoodsId,
  91. },
  92. }
  93. );
  94. goodsDetail.value = data.detail;
  95. };
  96. getGoodsDetail();
  97. const showDialog = ref(false);
  98. const handleClose = () => {
  99. showDialog.value = false;
  100. };
  101. function onBridgeReady(paySign) {
  102. if (!WeixinJSBridge) {
  103. return showToast("请在微信浏览器中打开");
  104. }
  105. WeixinJSBridge.invoke(
  106. "getBrandWCPayRequest",
  107. {
  108. appId: appId.value, //公众号ID,由商户传入
  109. ...paySign,
  110. },
  111. function (res) {
  112. console.log(res);
  113. if (res.err_msg == "get_brand_wcpay_request:ok") {
  114. // 使用以上方式判断前端返回,微信团队郑重提示:
  115. //res.err_msg将在用户支付成功后返回ok,但并不保证它绝对可靠,商户需进一步调用后端查单确认支付结果。
  116. console.log("支付成功");
  117. router.push({
  118. name: "promotionPayResult",
  119. query: {
  120. promotionGoodsId: route.query.promotionGoodsId,
  121. promotionSiteId: route.query.promotionSiteId,
  122. },
  123. });
  124. } else {
  125. showToast("支付失败" + res.err_msg);
  126. }
  127. }
  128. );
  129. }
  130. const handleConfirm = async (mobile, captcha) => {
  131. console.log("Payment confirmed");
  132. showDialog.value = false;
  133. const { data } = await request.post(
  134. `/archivesService/member/promotionGoods/paymentParams`,
  135. {
  136. promotionGoodsId: route.query.promotionGoodsId,
  137. promotionSiteId: route.query.promotionSiteId,
  138. mobile,
  139. captcha,
  140. openId: openId.value,
  141. }
  142. );
  143. console.log(data.paymentParams);
  144. data.paymentParams?.paySign && onBridgeReady(data.paymentParams.paySign);
  145. };
  146. const showHistory = ref(false);
  147. const buyHistory = () => {
  148. showHistory.value = true;
  149. };
  150. const buy = () => {
  151. showDialog.value = true;
  152. };
  153. const preCheck = async () => {
  154. const { data } = await request.get(
  155. `/archivesService/member/promotionGoods/preData`,
  156. {
  157. params: {
  158. promotionGoodsId: route.query.promotionGoodsId,
  159. promotionSiteId: route.query.promotionSiteId,
  160. },
  161. }
  162. );
  163. return data;
  164. };
  165. // const jsSdkConfig = async () => {
  166. // const { data } = await request.post(
  167. // "archivesService/member/promotionGoods/getJssdk",
  168. // {
  169. // url: location.href,
  170. // promotionGoodsId: route.query.promotionGoodsId,
  171. // }
  172. // );
  173. // return data;
  174. // };
  175. // const wxConfig = async () => {
  176. // const { data } = await jsSdkConfig();
  177. // window["wx"]?.config?.(data.config);
  178. // window["wx"]?.error((res) => {
  179. // console.log(res);
  180. // });
  181. // };
  182. const appId = ref(sessionStorage.getItem("appId"));
  183. const openId = ref("KfZnJNKFaFXWWCagSlX0dT4tNkwsLEwQVlrzpHvo");
  184. const getWeChatAuth = async () => {
  185. const { appId, errMsg, paymentSettingId } = await preCheck();
  186. if (errMsg) {
  187. // return showToast(errMsg);
  188. router.push({
  189. name: "error",
  190. query: {
  191. msg: errMsg,
  192. },
  193. });
  194. return;
  195. }
  196. if (appId) {
  197. //缓存信息
  198. sessionStorage.setItem("appId", appId);
  199. sessionStorage.setItem("paymentSettingId", paymentSettingId);
  200. //跳转微信授权
  201. const redirectUri = encodeURIComponent(location.href);
  202. location.href = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=${appId}&redirect_uri=${redirectUri}&response_type=code&scope=snsapi_base&state=123#wechat_redirect`;
  203. return;
  204. }
  205. showToast("微信配置不存在");
  206. };
  207. const getOpenId = async () => {
  208. const { code } = route.query;
  209. const paymentSettingId = sessionStorage.getItem("paymentSettingId");
  210. if (!code) {
  211. return;
  212. }
  213. const { data } = await request.get(
  214. `archivesService/member/promotionGoods/getOpenId`,
  215. {
  216. params: {
  217. code,
  218. paymentSettingId,
  219. },
  220. }
  221. );
  222. if (data.openId) {
  223. openId.value = data.openId;
  224. } else {
  225. showToast("获取openId失败");
  226. }
  227. };
  228. const init = async () => {
  229. const { code, promotionGoodsId, promotionSiteId } = route.query;
  230. if (!promotionSiteId || !promotionGoodsId) {
  231. return router.push({
  232. name: "error",
  233. query: {
  234. msg: "参数异常",
  235. },
  236. });
  237. }
  238. const paymentSettingId = sessionStorage.getItem("paymentSettingId");
  239. if (!code || !paymentSettingId) {
  240. // return getWeChatAuth();
  241. }
  242. // getOpenId()
  243. // wxConfig();
  244. };
  245. onMounted(() => init());
  246. </script>
  247. <style lang="scss" scoped></style>