index.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728
  1. <template>
  2. <view class="app-container">
  3. <HomePageA :homePage="homePage" :templateInfo="templateInfo" v-if="homePage === 1" />
  4. <HomePageB :homePage="homePage" :templateInfo="templateInfo" v-if="homePage === 2" />
  5. <HomePageC :homePage="homePage" :templateInfo="templateInfo" v-if="homePage === 3" />
  6. <view v-if="isSharePyq">
  7. <image src="https://zfire-train.oss-cn-guangzhou.aliyuncs.com/train/pic/1657090568346be106ca9-dd33-48d2-b06b-e4a02b06f2a2.png" mode="widthFix" style="width: 100%" @tap="clickPyq"></image>
  8. </view>
  9. <!-- 购买弹窗 -->
  10. <view class="global-mask" v-show="isBuyDialog"></view>
  11. <view class="buyDialog" v-show="isBuyDialog">
  12. <view class="close"><image src="@/static/icon/close.png" @tap="isBuyDialog = false"></image></view>
  13. <view class="goods" v-if="orderDetail.orderDetails && orderDetail.orderDetails.length == 1">
  14. <image :src="orderDetail.orderDetails[0].imgUrl" mode="aspectFill"></image>
  15. <view class="name ellipsis-2">{{orderDetail.orderDetails[0].goodsName}}</view>
  16. <view class="des ellipsis">{{orderDetail.orderDetails[0].goodsSpecValue}}</view>
  17. </view>
  18. <view class="goodsList" v-if="orderDetail.orderDetails && orderDetail.orderDetails.length > 1">
  19. <block v-for="(item, index) in orderDetail.orderDetails" :key='index'>
  20. <view class="item">
  21. <image :src="item.imgUrl" mode="aspectFill"></image>
  22. <view class="right">
  23. <view class="name ellipsis-2">{{item.goodsName}}</view>
  24. <view class="des ellipsis">{{item.goodsSpecValue}}</view>
  25. </view>
  26. </view>
  27. </block>
  28. </view>
  29. <view class="num">共<text>{{orderDetail.totalNum}}</text>件商品</view>
  30. <view class="price">需支付金额:<text>¥{{orderDetail.payAmount}}</text></view>
  31. <view class="button" @tap="payment">立即支付</view>
  32. </view>
  33. <!-- 优惠券提醒 -->
  34. <view class="global-mask" v-show="isCouponDialog"></view>
  35. <view class="couponDialog" v-show="isCouponDialog">
  36. <image src="@/static/home/cp_dialog.png" mode="widthFix"></image>
  37. <view class="bottom">
  38. <view class="text">你有优惠券即将到期,请尽快使用!</view>
  39. <view class="btn-group">
  40. <view class="button left" @tap="isCouponDialog = false">关闭</view>
  41. <view class="button right" @tap="toMyCoupon">去看看</view>
  42. </view>
  43. </view>
  44. </view>
  45. <!-- 领取优惠券弹窗 -->
  46. <view class="global-mask" v-show="isReceiveDialog"></view>
  47. <view class="couponDialog" v-show="isReceiveDialog">
  48. <image src="@/static/home/cp_dialog.png" mode="widthFix"></image>
  49. <view class="bottom">
  50. <view class="text">你有一张优惠券可领取!</view>
  51. <view class="btn-group onlyOne">
  52. <view class="button right" @tap="receiveCoupon">立即领取</view>
  53. </view>
  54. </view>
  55. </view>
  56. <modal-dialog showTitle="提示" showText="领取成功" :isShowDialog="isReceiveCouponDialog && isReceiveSuccess" @cancel="isReceiveCouponDialog = false" @confirm="toMyCoupon()" confirmText="去看看"></modal-dialog>
  57. <modal-dialog showTitle="提示" :showText="'领取失败:'+failMessage" :isShowDialog="isReceiveCouponDialog && !isReceiveSuccess" @cancel="isReceiveCouponDialog = false" :isShowConfirm="false" cancelText="关闭"></modal-dialog>
  58. <drag-button :isDock="true" :customBar="false" ref="dragButton"></drag-button>
  59. </view>
  60. </template>
  61. <script>
  62. import HomePageA from "@/pages/index/components/home-page-a";
  63. import HomePageB from "@/pages/index/components/home-page-b";
  64. import HomePageC from "@/pages/index/components/home-page-c";
  65. import {mapState} from 'vuex';
  66. import { appId } from '@/utils/config.js';
  67. import modalDialog from '@/components/modalDialog.vue';
  68. import dragButton from '@/components/drag-button.vue';
  69. export default {
  70. components:{
  71. HomePageA, HomePageB, HomePageC,
  72. modalDialog,
  73. dragButton
  74. },
  75. data() {
  76. return {
  77. configInfo: uni.getStorageSync('configInfo'),
  78. templateInfo: uni.getStorageSync('templateInfo'),
  79. isBuyDialog: false, // 是否显示购买弹窗(业务员代客下单时产生)
  80. orderDetail: {}, // 订单详情(购买弹窗需要)
  81. scene: { // 页面参数(扫码进入小程序时产生)
  82. type: '',
  83. goodsId: '',
  84. orderId: '',
  85. serviceId: '',
  86. couponId: '',
  87. },
  88. isCouponDialog: false, // 是否显示过期优惠券弹窗
  89. isReceiveCouponDialog: false,
  90. isReceiveSuccess: true,
  91. failMessage: '',
  92. isReceiveDialog: false,
  93. isSmsReceive: false,
  94. homePage: 0,
  95. isSharePyq: false,
  96. }
  97. },
  98. computed:{
  99. ...mapState(['userInfo', 'isLogin', 'userId'])
  100. },
  101. watch: {
  102. nowDate() {
  103. let hh = this.nowDate.getHours(),
  104. mm = this.nowDate.getMinutes(),
  105. ss = this.nowDate.getSeconds();
  106. let hs = [10, 12, 15, 18, 20];
  107. if(mm == 0 && ss == 0 && hs.indexOf(hh) >= 0) {
  108. setTimeout(() => {
  109. this.getSeckillTimeList();
  110. }, 1000)
  111. }
  112. }
  113. },
  114. // 页面滚动
  115. onPageScroll(res) {
  116. uni.$emit('onPageScroll', res.scrollTop);
  117. },
  118. // 下拉刷新
  119. onPullDownRefresh() {
  120. if(this.isSharePyq) {
  121. return false;
  122. }
  123. uni.$emit('onPullDownRefresh');
  124. this.getHomePage();
  125. },
  126. // 上拉加载
  127. onReachBottom() {
  128. uni.$emit('onReachBottom');
  129. },
  130. // 页面隐藏
  131. onHide() {
  132. uni.$emit('onHide');
  133. },
  134. // 页面显示
  135. async onShow() {
  136. var obj = wx.getLaunchOptionsSync()
  137. console.log(obj);
  138. if(obj.scene == 1154) {
  139. return this.isSharePyq = true;
  140. }
  141. await this.$onLaunched;
  142. await this.getHomePage();
  143. let timer = setInterval(() =>{
  144. if(this.homePage) {
  145. clearInterval(timer);
  146. setTimeout(()=> {
  147. uni.$emit('onShow');
  148. }, 200)
  149. }
  150. }, 100)
  151. this.$refs.dragButton.init();
  152. },
  153. // 页面加载
  154. async onLoad({scene, goodsId, serviceId, mobile, smsServiceId, otherType, appOrderId}) {
  155. await this.$onLaunched;
  156. uni.showLoading({
  157. title: '加载中'
  158. });
  159. this.getConfig();
  160. this.checkCoupon();
  161. // 点击短信链接进入小程序
  162. if(smsServiceId) {
  163. this.bindUser(smsServiceId);
  164. if(this.userInfo.type !== "SERVICE") {
  165. this.isSmsReceive = true;
  166. this.isReceiveDialog = true;
  167. }
  168. }
  169. // 从师傅端app跳转进入小程序
  170. if(appOrderId) {
  171. this.appOrderPay(appOrderId);
  172. }
  173. // 点击商品链接进入小程序
  174. if(serviceId) {
  175. this.bindUser(serviceId);
  176. }
  177. if(goodsId) {
  178. this.toGoodsDetail(goodsId);
  179. }
  180. if(otherType) {
  181. if(otherType === 'groupbuyList') {
  182. uni.navigateTo({
  183. url: '/pages/mine/groupbuy/list'
  184. })
  185. }
  186. }
  187. // 扫码进入小程序
  188. if(scene) {
  189. console.log(scene);
  190. // 拿scene的id去获取
  191. this.$axios({
  192. url: '/common/scene',
  193. method: 'get',
  194. params: {
  195. scene: scene,
  196. }
  197. }).then(res => {
  198. let newScene = res.data.split("&");
  199. this.scene.type = newScene[0];
  200. this.scene.serviceId = newScene[2];
  201. // 如果是商品
  202. if(newScene[0] == 'goods') {
  203. this.scene.goodsId = newScene[1];
  204. this.toGoodsDetail(this.scene.goodsId);
  205. }
  206. // 如果是订单
  207. else if(newScene[0] == 'order') {
  208. this.scene.orderId = newScene[1];
  209. this.getOrderInfo();
  210. }
  211. // 如果是优惠券
  212. else if(newScene[0] == 'coupon') {
  213. this.scene.couponId = newScene[1];
  214. this.isReceiveDialog = true;
  215. // this.receiveCoupon(this.scene.couponId);
  216. }
  217. // 如果是商城分享码
  218. else if(newScene[0] == 'bind') {
  219. }
  220. // 如果是团购分享码
  221. else if(newScene[0] == 'promotion_group') {
  222. uni.navigateTo({
  223. url: '/pages/mine/groupbuy/list'
  224. })
  225. }
  226. console.log(this.scene);
  227. this.bindUser(this.scene.serviceId);
  228. })
  229. }
  230. },
  231. // 分享
  232. onShareAppMessage(options) {
  233. if (options && options.from == 'button') {
  234. // 来自页面内的转发按钮
  235. } else {
  236. // 点击微信右上角的分享按钮
  237. }
  238. return {
  239. title: `${this.userInfo.nickName}向你推荐了「${this.configInfo.minAppName}」小程序`,
  240. // imageUrl: this.detail.imgUrl,
  241. path: '/pages/index/index?serviceId=' + this.userId,
  242. query: {
  243. // id: this.goodsId,
  244. },
  245. success: function(res) {
  246. if(res.errMsg == 'shareAppMessage:ok'){
  247. this.$successToast('分享完成');
  248. }
  249. }
  250. }
  251. },
  252. onShareTimeline() {
  253. return {
  254. title: `${this.userInfo.nickName}向你推荐了「${this.configInfo.minAppName}」小程序`,
  255. // imageUrl: this.detail.imgUrl,
  256. // path: '/pages/index/index?serviceId=' + this.userId,
  257. query: 'serviceId=' + this.userId,
  258. success: function(res) {
  259. if(res.errMsg == 'shareAppMessage:ok'){
  260. this.$successToast('分享完成');
  261. }
  262. }
  263. }
  264. },
  265. methods: {
  266. // 获取页面模版
  267. async getHomePage() {
  268. const result = await new Promise((resolve, reject)=>{
  269. this.$axios({
  270. url: '/renovation/detail',
  271. method: 'get',
  272. }).then(res => {
  273. if(res.data) {
  274. uni.setStorageSync('templateInfo', res.data);
  275. this.templateInfo = res.data;
  276. // res.data.templateType = 3;
  277. this.homePage = res.data.templateType;
  278. resolve(res.data.templateType);
  279. }
  280. }).catch(res => {
  281. this.homePage = 1;
  282. resolve(1);
  283. })
  284. })
  285. return result;
  286. },
  287. // 获取配置信息
  288. getConfig() {
  289. this.$axios({
  290. url: '/common/config/get',
  291. method: 'get',
  292. }).then(res => {
  293. if(res.data) {
  294. uni.setStorageSync('configInfo', res.data);
  295. }
  296. })
  297. },
  298. // 绑定用户关系
  299. bindUser(serviceId) {
  300. this.$axios({
  301. url: '/user/bind',
  302. params: {
  303. serviceId: serviceId,
  304. userId: this.userId,
  305. }
  306. }).then(res => {
  307. console.log('绑定成功:' + res.message);
  308. })
  309. },
  310. // 点击分享朋友圈的图
  311. clickPyq() {
  312. this.$toast('请前往小程序使用完整服务');
  313. },
  314. // app订单支付
  315. appOrderPay(appOrderId){
  316. let that = this;
  317. this.$axios({
  318. url: '/common/online/pay',
  319. method: 'post',
  320. params: {
  321. key: appOrderId,
  322. openId: uni.getStorageSync('userInfo').openId,
  323. }
  324. }).then(res => {
  325. if(!res.data) return false;
  326. let data = JSON.parse(res.data.codeUrl);
  327. wx.requestPayment({
  328. timeStamp: data.timeStamp,
  329. nonceStr: data.nonceStr,
  330. package: data.package,
  331. signType: 'RSA',
  332. paySign: data.paySign,
  333. success: (res) => {
  334. that.$successToast('支付成功');
  335. },
  336. fail: (err) => {
  337. that.$toast('支付失败');
  338. }
  339. })
  340. })
  341. },
  342. // 进入商品详情
  343. toGoodsDetail(id) {
  344. if(!id) {
  345. return false;
  346. }
  347. uni.navigateTo({
  348. url: '/packageGoods/pages/detail?id=' + id
  349. })
  350. },
  351. // 检查是否有优惠券过期
  352. checkCoupon() {
  353. this.$axios({
  354. url: '/coupon/timeout',
  355. method: 'get',
  356. params: {},
  357. }).then(res => {
  358. if(res.data) {
  359. this.isCouponDialog = true;
  360. }
  361. })
  362. },
  363. // 进入我的优惠券
  364. toMyCoupon() {
  365. this.isCouponDialog = false;
  366. this.isReceiveCouponDialog = false;
  367. uni.navigateTo({
  368. url:'/pages/mine/coupon/list'
  369. })
  370. },
  371. // 领取优惠券
  372. receiveCoupon(id) {
  373. if(!this.isLogin) {
  374. return uni.navigateTo({
  375. url: '/pages/login/index'
  376. })
  377. }
  378. if(this.isSmsReceive) {
  379. this.$axios({
  380. url: '/coupon/obtain',
  381. method: 'get',
  382. params: {
  383. userId: '',
  384. couponId: ''
  385. }
  386. }).then(res => {
  387. this.isReceiveDialog = false;
  388. this.isReceiveSuccess = true;
  389. this.isReceiveCouponDialog = true;
  390. }).catch(res => {
  391. this.failMessage = res.message;
  392. this.isReceiveDialog = false;
  393. this.isReceiveSuccess = false;
  394. this.isReceiveCouponDialog = true;
  395. })
  396. }else {
  397. this.$axios({
  398. url: '/coupon/transfer/coupon',
  399. method: 'get',
  400. params: {
  401. userCouponId: this.scene.couponId
  402. }
  403. }).then(res => {
  404. this.isReceiveDialog = false;
  405. this.isReceiveSuccess = true;
  406. this.isReceiveCouponDialog = true;
  407. }).catch(res => {
  408. this.failMessage = res.message;
  409. this.isReceiveDialog = false;
  410. this.isReceiveSuccess = false;
  411. this.isReceiveCouponDialog = true;
  412. })
  413. }
  414. },
  415. // 获取订单信息
  416. getOrderInfo() {
  417. this.$axios({
  418. url: '/order/detail',
  419. method: 'get',
  420. params: {
  421. orderId: this.scene.orderId
  422. }
  423. }).then(res => {
  424. if(res.data && res.data.orderStatus === 'NOPAY') {
  425. this.orderDetail = res.data;
  426. this.isBuyDialog = true;
  427. }else {
  428. return this.$toast('二维码已失效,请重新生成');
  429. }
  430. })
  431. },
  432. // 支付
  433. payment() {
  434. let that = this;
  435. if(!this.isLogin) {
  436. uni.navigateTo({
  437. url: '/pages/login/index'
  438. })
  439. }else {
  440. this.$axios({
  441. url: '/order/wait/pay',
  442. params: {
  443. userId: this.userId,
  444. orderId: this.scene.orderId,
  445. },
  446. isLoading: 1,
  447. }).then(res => {
  448. uni.getProvider({
  449. service: 'payment',
  450. success: (e) => {
  451. uni.requestPayment({
  452. provider: e.provider[0],
  453. orderInfo: res.data,
  454. timeStamp: res.data.timeStamp,
  455. nonceStr: res.data.nonceStr,
  456. package: res.data.payPackage,
  457. signType: 'MD5',
  458. paySign: res.data.paySign,
  459. success: (res) => {
  460. that.$successToast('支付成功');
  461. that.isBuyDialog = false;
  462. that.requestMessage();
  463. },
  464. fail: (err) => {
  465. that.$toast('支付失败');
  466. }
  467. })
  468. }
  469. })
  470. })
  471. }
  472. },
  473. // 消息推送
  474. requestMessage() {
  475. let that = this;
  476. uni.showModal({
  477. title: '温馨提示',
  478. content: '为更好的促进您与买家的交流,需要在您的订单成交时向您发送消息',
  479. confirmText: "同意",
  480. cancelText: "拒绝",
  481. success: function (res) {
  482. if (res.confirm) {
  483. let tmplIds = [that.configInfo.template];
  484. uni.requestSubscribeMessage({
  485. tmplIds: tmplIds,
  486. success (res) {
  487. let status = null;
  488. tmplIds.map((item, index) => {
  489. if(res[item] == 'accept') {
  490. status = 'accept';
  491. }
  492. })
  493. if(status == 'accept') {
  494. that.$successToast('订阅成功');
  495. }else {
  496. that.$toast('订阅取消');
  497. }
  498. },
  499. fail (res) {
  500. console.log(res);
  501. that.$toast('订阅失败');
  502. }
  503. })
  504. } else if (res.cancel) {
  505. uni.showModal({
  506. title: '温馨提示',
  507. content: '拒绝后您将无法获取实时的与卖家(买家)的交易消息',
  508. confirmText: "知道了",
  509. showCancel: false,
  510. success: function (res) {
  511. }
  512. });
  513. }
  514. }
  515. });
  516. },
  517. }
  518. }
  519. </script>
  520. <style lang="scss">
  521. .app-container {
  522. background: #F4F2F2;
  523. box-sizing: border-box;
  524. }
  525. .couponDialog {
  526. position: fixed;
  527. top: 30%;
  528. left: 75rpx;
  529. z-index: 999;
  530. width: 600rpx;
  531. background: #FFFFFF;
  532. border-radius: 20rpx;
  533. image {
  534. width: 600rpx;
  535. }
  536. .bottom {
  537. padding: 0 40rpx 50rpx;
  538. .text {
  539. font-size: 32rpx;
  540. line-height: 48rpx;
  541. text-align: center;
  542. }
  543. .btn-group {
  544. display: flex;
  545. justify-content: space-between;
  546. margin-top: 50rpx;
  547. .button {
  548. width: 240rpx;
  549. height: 72rpx;
  550. border-radius: 72rpx;
  551. border: 1px solid #FF3F42;
  552. font-size: 32rpx;
  553. line-height: 72rpx;
  554. text-align: center;
  555. &.left {
  556. color: #FF3F42;
  557. }
  558. &.right {
  559. color: #FFFFFF;
  560. background: #FF3F42;
  561. }
  562. }
  563. &.onlyOne {
  564. justify-content: center;
  565. .button {
  566. width: 300rpx;
  567. }
  568. }
  569. }
  570. }
  571. }
  572. .buyDialog {
  573. position: fixed;
  574. top: 30%;
  575. left: 100rpx;
  576. z-index: 999;
  577. width: 550rpx;
  578. background: #FFFFFF;
  579. border-radius: 10rpx;
  580. box-sizing: border-box;
  581. padding: 20rpx 20rpx 30rpx;
  582. display: flex;
  583. flex-direction: column;
  584. align-items: center;
  585. .close {
  586. width: 100%;
  587. display: flex;
  588. justify-content: flex-end;
  589. image {
  590. width: 32rpx;
  591. height: 32rpx;
  592. display: block;
  593. }
  594. }
  595. .goods {
  596. margin-top: 10rpx;
  597. image {
  598. width: 410rpx;
  599. height: 300rpx;
  600. display: block;
  601. }
  602. .name {
  603. font-size: 28rpx;
  604. line-height: 34rpx;
  605. margin-top: 16rpx;
  606. }
  607. .des {
  608. font-size: 26rpx;
  609. color: #666666;
  610. line-height: 30rpx;
  611. margin-top: 10rpx;
  612. }
  613. }
  614. .goodsList {
  615. max-height: 400rpx;
  616. overflow-y: scroll;
  617. margin-top: 10rpx;
  618. .item {
  619. display: flex;
  620. padding: 10rpx 20rpx;
  621. image {
  622. width: 120rpx;
  623. height: 120rpx;
  624. display: block;
  625. flex-shrink: 0;
  626. }
  627. .right {
  628. width: 334rpx;
  629. margin-left: 16rpx;
  630. .name {
  631. font-size: 28rpx;
  632. line-height: 34rpx;
  633. margin-top: 8rpx;
  634. }
  635. .des {
  636. font-size: 26rpx;
  637. color: #666666;
  638. line-height: 30rpx;
  639. margin-top: 10rpx;
  640. }
  641. }
  642. }
  643. }
  644. .num {
  645. margin-top: 10rpx;
  646. line-height: 40rpx;
  647. text {
  648. color: #FF3F42;
  649. font-size: 32rpx;
  650. font-weight: 600;
  651. margin: 0 10rpx;
  652. }
  653. }
  654. .price {
  655. margin-top: 8rpx;
  656. line-height: 40rpx;
  657. text {
  658. color: #FF3F42;
  659. font-size: 32rpx;
  660. font-weight: 600;
  661. }
  662. }
  663. .button {
  664. width: 300rpx;
  665. height: 64rpx;
  666. line-height: 64rpx;
  667. text-align: center;
  668. background: #FF3F42;
  669. border-radius: 64rpx;
  670. font-size: 28rpx;
  671. color: #FFFFFF;
  672. margin-top: 20rpx;
  673. }
  674. }
  675. </style>