pay.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. import sheep from '@/sheep';
  2. // #ifdef H5
  3. import $wxsdk from '@/sheep/libs/sdk-h5-weixin';
  4. // #endif
  5. import {
  6. getRootUrl
  7. } from '@/sheep/helper';
  8. import PayOrderApi from '@/sheep/api/pay/order';
  9. /**
  10. * 支付
  11. *
  12. * @param {String} payment = ['wechat','alipay','wallet','mock'] - 支付方式
  13. * @param {String} orderType = ['goods','recharge','groupon'] - 订单类型
  14. * @param {String} id - 订单号
  15. */
  16. export default class SheepPay {
  17. constructor(payment, orderType, id) {
  18. this.payment = payment;
  19. this.id = id;
  20. this.orderType = orderType;
  21. this.payAction();
  22. }
  23. payAction() {
  24. const payAction = {
  25. WechatOfficialAccount: {
  26. wechat: () => {
  27. const userInfo = sheep.$store('user').userInfo;
  28. if(userInfo.username === '非繁' || userInfo.username === 'ru10' || userInfo.username === '迷鹿'){
  29. console.log("是测试用户")
  30. this.fuYouWechatOfficialAccountPay();
  31. return;
  32. }else{
  33. console.log("不是测试用户")
  34. }
  35. this.wechatOfficialAccountPay();
  36. },
  37. alipay: () => {
  38. this.redirectPay(); // 现在公众号可以直接跳转支付宝页面
  39. },
  40. wallet: () => {
  41. this.walletPay();
  42. },
  43. mock: () => {
  44. this.mockPay();
  45. }
  46. },
  47. WechatMiniProgram: {
  48. wechat: () => {
  49. this.wechatMiniProgramPay();
  50. },
  51. alipay: () => {
  52. this.copyPayLink();
  53. },
  54. wallet: () => {
  55. this.walletPay();
  56. },
  57. mock: () => {
  58. this.mockPay();
  59. }
  60. },
  61. App: {
  62. wechat: () => {
  63. this.wechatAppPay();
  64. },
  65. alipay: () => {
  66. this.alipay();
  67. },
  68. wallet: () => {
  69. this.walletPay();
  70. },
  71. mock: () => {
  72. this.mockPay();
  73. }
  74. },
  75. H5: {
  76. wechat: () => {
  77. this.wechatWapPay();
  78. },
  79. alipay: () => {
  80. this.redirectPay();
  81. },
  82. wallet: () => {
  83. this.walletPay();
  84. },
  85. mock: () => {
  86. this.mockPay();
  87. }
  88. },
  89. };
  90. return payAction[sheep.$platform.name][this.payment]();
  91. }
  92. // 预支付
  93. prepay(channel) {
  94. return new Promise(async (resolve, reject) => {
  95. let data = {
  96. id: this.id,
  97. channelCode: channel,
  98. channelExtras: {}
  99. };
  100. // 特殊逻辑:微信公众号、小程序支付时,必须传入 openid
  101. if (['wx_pub', 'wx_lite'].includes(channel)) {
  102. const openid = await sheep.$platform.useProvider('wechat').getOpenid(true);
  103. console.log(openid)
  104. // 如果获取不到 openid,微信无法发起支付,此时需要引导
  105. if (!openid) {
  106. this.bindWeixin();
  107. return;
  108. }
  109. data.channelExtras.openid = openid;
  110. }
  111. // 发起预支付 API 调用
  112. PayOrderApi.submitOrder(data).then((res) => {
  113. // 成功时
  114. res.code === 0 && resolve(res);
  115. // 失败时
  116. if (res.code !== 0 && res.msg.indexOf('无效的openid') >= 0) {
  117. // 特殊逻辑:微信公众号、小程序支付时,必须传入 openid 不正确的情况
  118. if (res.msg.indexOf('无效的openid') >= 0 // 获取的 openid 不正确时,或者随便输入了个 openid
  119. ||
  120. res.msg.indexOf('下单账号与支付账号不一致') >= 0
  121. ){ // https://developers.weixin.qq.com/community/develop/doc/00008c53c347804beec82aed051c00
  122. this.bindWeixin();
  123. }
  124. }
  125. });
  126. });
  127. }
  128. // 富友预支付
  129. fuYouPrePay(channel) {
  130. return new Promise(async (resolve, reject) => {
  131. let data = {
  132. id: this.id,
  133. req: {
  134. trade_type: channel,
  135. }
  136. };
  137. // 特殊逻辑:微信公众号、小程序支付时,必须传入 openid
  138. if (['wx_pub', 'wx_lite','JSAPI'].includes(channel)) {
  139. const openid = await sheep.$platform.useProvider('wechat').getOpenid(true);
  140. // 如果获取不到 openid,微信无法发起支付,此时需要引导
  141. if (!openid) {
  142. this.bindWeixin();
  143. return;
  144. }
  145. data.req.sub_openid = openid;
  146. }
  147. // console.log("富友支付",data)
  148. // 发起预支付 API 调用
  149. PayOrderApi.fuYouSubmitOrder(data).then((res) => {
  150. // 成功时
  151. res.code === 0 && resolve(res);
  152. // 失败时
  153. if (res.code !== 0 && res.msg.indexOf('无效的openid') >= 0) {
  154. // 特殊逻辑:微信公众号、小程序支付时,必须传入 openid 不正确的情况
  155. if (res.msg.indexOf('无效的openid') >= 0 // 获取的 openid 不正确时,或者随便输入了个 openid
  156. ||
  157. res.msg.indexOf('下单账号与支付账号不一致') >= 0
  158. ) { // https://developers.weixin.qq.com/community/develop/doc/00008c53c347804beec82aed051c00
  159. this.bindWeixin();
  160. }
  161. }
  162. });
  163. });
  164. }
  165. // #ifdef H5
  166. // 富友微信公众号 JSSDK 支付
  167. async fuYouWechatOfficialAccountPay() {
  168. console.log("富友微信公众号 JSSDK 支付")
  169. let {
  170. code,
  171. data
  172. } = await this.fuYouPrePay('JSAPI');
  173. if (code !== 0) {
  174. return;
  175. }
  176. let newJsonString = data.replace(/"sdk_package":/g, '"packageValue":');
  177. const payConfig = JSON.parse(newJsonString);
  178. console.log('payConfig',payConfig);
  179. // return;
  180. $wxsdk.wxpay(payConfig, {
  181. success: () => {
  182. this.payResult('success');
  183. },
  184. cancel: () => {
  185. sheep.$helper.toast('支付已手动取消');
  186. },
  187. fail: (error) => {
  188. if (error.errMsg.indexOf('chooseWXPay:没有此SDK或暂不支持此SDK模拟') >= 0) {
  189. sheep.$helper.toast('发起微信支付失败,原因:可能是微信开发者工具不支持,建议使用微信打开网页后支付');
  190. return
  191. }
  192. this.payResult('fail');
  193. },
  194. });
  195. }
  196. // 微信公众号 JSSDK 支付
  197. async wechatOfficialAccountPay() {
  198. console.log("原微信公众号 JSSDK 支付")
  199. let {
  200. code,
  201. data
  202. } = await this.prepay('wx_pub');
  203. if (code !== 0) {
  204. return;
  205. }
  206. const payConfig = JSON.parse(data.displayContent);
  207. $wxsdk.wxpay(payConfig, {
  208. success: () => {
  209. this.payResult('success');
  210. },
  211. cancel: () => {
  212. sheep.$helper.toast('支付已手动取消');
  213. },
  214. fail: (error) => {
  215. if (error.errMsg.indexOf('chooseWXPay:没有此SDK或暂不支持此SDK模拟') >= 0) {
  216. sheep.$helper.toast('发起微信支付失败,原因:可能是微信开发者工具不支持,建议使用微信打开网页后支付');
  217. return
  218. }
  219. this.payResult('fail');
  220. },
  221. });
  222. }
  223. // 浏览器微信 H5 支付 TODO 非繁人:待接入
  224. async wechatWapPay() {
  225. const {
  226. error,
  227. data
  228. } = await this.prepay();
  229. if (error === 0) {
  230. const redirect_url =
  231. `${getRootUrl()}pages/pay/result?id=${this.id}&payment=${this.payment}&orderType=${this.orderType}`;
  232. location.href = `${data.pay_data.h5_url}&redirect_url=${encodeURIComponent(redirect_url)}`;
  233. }
  234. }
  235. // 支付链接 TODO 非繁人:待接入
  236. async redirectPay() {
  237. let {
  238. error,
  239. data
  240. } = await this.prepay();
  241. if (error === 0) {
  242. const redirect_url =
  243. `${getRootUrl()}pages/pay/result?id=${this.id}&payment=${this.payment}&orderType=${this.orderType}`;
  244. location.href = data.pay_data + encodeURIComponent(redirect_url);
  245. }
  246. }
  247. // #endif
  248. // 微信小程序支付
  249. async wechatMiniProgramPay() {
  250. // let that = this;
  251. let {
  252. code,
  253. data
  254. } = await this.prepay('wx_lite');
  255. if (code !== 0) {
  256. return;
  257. }
  258. // 调用微信小程序支付
  259. const payConfig = JSON.parse(data.displayContent);
  260. uni.requestPayment({
  261. provider: 'wxpay',
  262. timeStamp: payConfig.timeStamp,
  263. nonceStr: payConfig.nonceStr,
  264. package: payConfig.packageValue,
  265. signType: payConfig.signType,
  266. paySign: payConfig.paySign,
  267. success: (res) => {
  268. this.payResult('success');
  269. },
  270. fail: (err) => {
  271. if (err.errMsg === 'requestPayment:fail cancel') {
  272. sheep.$helper.toast('支付已手动取消');
  273. } else {
  274. this.payResult('fail');
  275. }
  276. },
  277. });
  278. }
  279. // 余额支付
  280. async walletPay() {
  281. const {
  282. code
  283. } = await this.prepay('wallet');
  284. code === 0 && this.payResult('success');
  285. }
  286. // 模拟支付
  287. async mockPay() {
  288. const {
  289. code
  290. } = await this.prepay('mock');
  291. code === 0 && this.payResult('success');
  292. }
  293. // 支付宝复制链接支付 TODO 非繁人:待接入
  294. async copyPayLink() {
  295. let that = this;
  296. let {
  297. error,
  298. data
  299. } = await this.prepay();
  300. if (error === 0) {
  301. // 引入showModal 点击确认 复制链接;
  302. uni.showModal({
  303. title: '支付宝支付',
  304. content: '复制链接到外部浏览器',
  305. confirmText: '复制链接',
  306. success: (res) => {
  307. if (res.confirm) {
  308. sheep.$helper.copyText(data.pay_data);
  309. }
  310. },
  311. });
  312. }
  313. }
  314. // 支付宝支付 TODO 非繁人:待接入
  315. async alipay() {
  316. let that = this;
  317. const {
  318. error,
  319. data
  320. } = await this.prepay();
  321. if (error === 0) {
  322. uni.requestPayment({
  323. provider: 'alipay',
  324. orderInfo: data.pay_data, //支付宝订单数据
  325. success: (res) => {
  326. that.payResult('success');
  327. },
  328. fail: (err) => {
  329. if (err.errMsg === 'requestPayment:fail [paymentAlipay:62001]user cancel') {
  330. sheep.$helper.toast('支付已手动取消');
  331. } else {
  332. that.payResult('fail');
  333. }
  334. },
  335. });
  336. }
  337. }
  338. // 微信支付 TODO 非繁人:待接入
  339. async wechatAppPay() {
  340. let that = this;
  341. let {
  342. error,
  343. data
  344. } = await this.prepay();
  345. if (error === 0) {
  346. uni.requestPayment({
  347. provider: 'wxpay',
  348. orderInfo: data.pay_data, //微信订单数据(官方说是string。实测为object)
  349. success: (res) => {
  350. that.payResult('success');
  351. },
  352. fail: (err) => {
  353. err.errMsg !== 'requestPayment:fail cancel' && that.payResult('fail');
  354. },
  355. });
  356. }
  357. }
  358. // 支付结果跳转,success:成功,fail:失败
  359. payResult(resultType) {
  360. sheep.$router.redirect('/pages/pay/result', {
  361. id: this.id,
  362. orderType: this.orderType,
  363. payState: resultType
  364. });
  365. }
  366. // 引导绑定微信
  367. bindWeixin() {
  368. uni.showModal({
  369. title: '微信支付',
  370. content: '请先绑定微信再使用微信支付',
  371. confirmText: '绑定',
  372. success: function(res) {
  373. if (res.confirm) {
  374. sheep.$platform.useProvider('wechat').bind();
  375. }
  376. },
  377. });
  378. }
  379. }
  380. export function getPayMethods(channels) {
  381. const payMethods = [{
  382. icon: '/static/img/shop/pay/wechat.png',
  383. title: '微信支付',
  384. value: 'wechat',
  385. disabled: true,
  386. },
  387. {
  388. icon: '/static/img/shop/pay/alipay.png',
  389. title: '支付宝支付',
  390. value: 'alipay',
  391. disabled: true,
  392. },
  393. // 20240420注释
  394. // {
  395. // icon: '/static/img/shop/pay/wallet.png',
  396. // title: '余额支付',
  397. // value: 'wallet',
  398. // disabled: true,
  399. // },
  400. // {
  401. // icon: '/static/img/shop/pay/apple.png',
  402. // title: 'Apple Pay',
  403. // value: 'apple',
  404. // disabled: true,
  405. // },
  406. // {
  407. // icon: '/static/img/shop/pay/wallet.png',
  408. // title: '模拟支付',
  409. // value: 'mock',
  410. // disabled: true,
  411. // }
  412. ];
  413. const platform = sheep.$platform.name
  414. // 1. 处理【微信支付】
  415. const wechatMethod = payMethods[0];
  416. if ((platform === 'WechatOfficialAccount' && channels.includes('wx_pub')) ||
  417. (platform === 'WechatMiniProgram' && channels.includes('wx_lite')) ||
  418. (platform === 'App' && channels.includes('wx_app'))) {
  419. wechatMethod.disabled = false;
  420. }
  421. wechatMethod.disabled = false; // TODO 非繁人:临时测试
  422. // 2. 处理【支付宝支付】
  423. const alipayMethod = payMethods[1];
  424. if ((platform === 'WechatOfficialAccount' && channels.includes('alipay_wap')) ||
  425. platform === 'WechatMiniProgram' && channels.includes('alipay_wap') ||
  426. platform === 'App' && channels.includes('alipay_app')) {
  427. alipayMethod.disabled = false;
  428. }
  429. // 3. 处理【余额支付】
  430. // const walletMethod = payMethods[2];
  431. // if (channels.includes('wallet')) {
  432. // walletMethod.disabled = false;
  433. // }
  434. // 4. 处理【苹果支付】TODO 非繁人:未来接入
  435. // 5. 处理【模拟支付】
  436. // const mockMethod = payMethods[4];
  437. // if (channels.includes('mock')) {
  438. // mockMethod.disabled = false;
  439. // }
  440. return payMethods;
  441. }