edit.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. <template>
  2. <view class="app">
  3. <view>
  4. <view class="label">支付单号:</view>
  5. <view><input v-model="out_trade_no" /></view>
  6. </view>
  7. <view>
  8. <view class="label">支付金额(单位分,100=1元):</view>
  9. <view><input v-model.number="total_fee" type="number" /></view>
  10. </view>
  11. <!-- #endif -->
  12. <!-- #ifdef MP-WEIXIN || H5 || APP -->
  13. <button type="primary" @click="createOrder('wxpay')">发起支付(微信)</button>
  14. <!-- #endif -->
  15. <!-- #ifdef MP-WEIXIN -->
  16. <button @click="pageTo('/pages/weixin-virtual-payment/weixin-virtual-payment')">微信小程序虚拟支付示例</button>
  17. <!-- #endif -->
  18. <!--
  19. <button @click="refund">发起退款</button>
  20. <view class="tips">发起退款需要admin权限,本示例未对接登录功能</view>
  21. <button @click="getRefund">查询退款状态</button>
  22. <button @click="closeOrder">关闭订单</button>
  23. -->
  24. <!-- #ifdef H5 -->
  25. <button v-if="h5Env === 'h5-weixin'" @click="getWeiXinJsCode('snsapi_base')">公众号获取openid示例</button>
  26. <!-- #endif -->
  27. <!-- 统一支付组件,注意:vue3下ref不可以等于组件名,因此这里ref="pay" 而不能是 ref="uniPay" -->
  28. <uni-pay ref="pay" :adpid="adpid" height="70vh" return-url="/pages/order-detail/order-detail" logo="/static/logo.png" @success="onSuccess" @create="onCreate" @fail="onFail"></uni-pay>
  29. </view>
  30. </template>
  31. <script>
  32. export default {
  33. data() {
  34. return {
  35. total_fee: 1000, // 支付金额,单位分 100 = 1元
  36. order_no: "", // 业务系统订单号(即你自己业务系统的订单表的订单号)
  37. out_trade_no: "", // 插件支付单号
  38. description: "测试订单", // 支付描述
  39. type: "test", // 支付回调类型 如 recharge 代表余额充值 goods 代表商品订单(可自定义,任意英文单词都可以,只要你在 uni-pay-co/notify/目录下创建对应的 xxx.js文件进行编写对应的回调逻辑即可)
  40. //qr_code: true, // 是否强制使用扫码支付
  41. openid:"", // 微信公众号需要
  42. custom:{
  43. a: "a",
  44. b: 1
  45. },
  46. adpid: "1000000001", // uni-ad的广告位id
  47. transaction_id:"", // 查询订单接口的查询条件
  48. getOrderRes:{}, // 查询订单支付成功后的返回值
  49. }
  50. },
  51. onLoad(options={}) {
  52. // #ifdef H5
  53. // 微信公众号特殊逻辑开始-----------------------------------------------------------
  54. // 以下代码仅为获取openid,正常你自己项目应该是登录后才能支付,登录后已经拿到openid,无需编写下面的代码
  55. if (this.h5Env === 'h5-weixin') {
  56. let openid = uni.getStorageSync("uni-pay-weixin-h5-openid");
  57. let code = uni.getStorageSync("uni-pay-weixin-h5-code");
  58. if (openid) {
  59. this.openid = openid;
  60. }
  61. // 如果code和state有值,且此code没有被使用过,则执行获取微信公众号的openid
  62. if (options.code && options.state && code !== options.code) {
  63. // 获取微信公众号的openid
  64. setTimeout(() => {
  65. this.getOpenid({
  66. provider: "wxpay",
  67. code: options.code
  68. });
  69. }, 300);
  70. } else if (!openid){
  71. // 如果openid为空,则执行微信公众号的网页授权登录逻辑
  72. setTimeout(() => {
  73. this.getWeiXinJsCode('snsapi_base');
  74. }, 300);
  75. }
  76. }
  77. this.order_no = `test`+Date.now();
  78. this.out_trade_no = `${this.order_no}-1`;
  79. this.user = this.getUser();
  80. // 微信公众号特殊逻辑结束-----------------------------------------------------------
  81. // #endif
  82. },
  83. methods: {
  84. /**
  85. * 发起支付(唤起收银台,如果只有一种支付方式,则收银台不会弹出来,会直接使用此支付方式)
  86. * 在调用此api前,你应该先创建自己的业务系统订单,并获得订单号 order_no,把order_no当参数传给此api,而示例中为了简化跟支付插件无关的代码,这里直接已时间戳生成了order_no
  87. */
  88. open() {
  89. this.order_no = `test`+Date.now();
  90. this.out_trade_no = `${this.order_no}-1`;
  91. // 打开支付收银台
  92. this.$refs.pay.open({
  93. total_fee: this.total_fee, // 支付金额,单位分 100 = 1元
  94. order_no: this.order_no, // 业务系统订单号(即你自己业务系统的订单表的订单号)
  95. out_trade_no: this.out_trade_no, // 插件支付单号
  96. description: this.description, // 支付描述
  97. type: this.type, // 支付回调类型
  98. qr_code: this.qr_code, // 是否强制使用扫码支付
  99. openid: this.openid, // 微信公众号需要
  100. custom: this.custom, // 自定义数据
  101. });
  102. },
  103. /**
  104. * 发起支付(不唤起收银台,手动指定支付方式)
  105. * 在调用此api前,你应该先创建自己的业务系统订单,并获得订单号 order_no,把order_no当参数传给此api,而示例中为了简化跟支付插件无关的代码,这里直接已时间戳生成了order_no
  106. */
  107. createOrder(provider){
  108. /*
  109. // 发起支付
  110. this.$refs.pay.createOrder({
  111. provider: provider, // 支付供应商
  112. total_fee: this.total_fee, // 支付金额,单位分 100 = 1元
  113. order_no: this.order_no, // 业务系统订单号(即你自己业务系统的订单表的订单号)
  114. out_trade_no: this.out_trade_no, // 插件支付单号
  115. description: this.description, // 支付描述
  116. type: this.type, // 支付回调类型
  117. qr_code: this.qr_code, // 是否强制使用扫码支付
  118. openid: this.openid, // 微信公众号需要
  119. custom: this.custom, // 自定义数据
  120. });
  121. */
  122. this.http.request({
  123. url: '/level-one-server/app/WalletManage/topupSave',
  124. data: {
  125. amount: this.total_fee, // 支付金额,单位分 100 = 1元
  126. transactionId: this.order_no, // 业务系统订单号(即你自己业务系统的订单表的订单号)
  127. outTradeNo: this.out_trade_no, // 插件支付单号
  128. },
  129. success: res => {
  130. this.user.wallet = this.user.wallet + this.total_fee/100;
  131. console.log("this.user",this.user)
  132. uni.setStorageSync('info', this.user);
  133. uni.showToast({
  134. title: '充值成功!'
  135. });
  136. uni.navigateBack({
  137. data:1
  138. });
  139. }
  140. });
  141. },
  142. /**
  143. * 生成支付独立二维码(只返回支付二维码)
  144. * 在调用此api前,你应该先创建自己的业务系统订单,并获得订单号 order_no,把order_no当参数传给此api,而示例中为了简化跟支付插件无关的代码,这里直接已时间戳生成了order_no
  145. */
  146. createQRcode(provider){
  147. this.order_no = `test`+Date.now();
  148. this.out_trade_no = `${this.order_no}-1`;
  149. // 发起支付
  150. this.$refs.pay.createOrder({
  151. provider: provider, // 支付供应商
  152. total_fee: this.total_fee, // 支付金额,单位分 100 = 1元
  153. order_no: this.order_no, // 业务系统订单号(即你自己业务系统的订单表的订单号)
  154. out_trade_no: this.out_trade_no, // 插件支付单号
  155. description: this.description, // 支付描述
  156. type: this.type, // 支付回调类型
  157. qr_code: true, // 是否强制使用扫码支付
  158. cancel_popup: true, // 配合qr_code:true使用,是否只生成支付二维码,没有二维码弹窗
  159. openid: this.openid, // 微信公众号需要
  160. custom: this.custom, // 自定义数据
  161. });
  162. },
  163. /**
  164. * 前往自定义收银台页面
  165. * 在调用此api前,你应该先创建自己的业务系统订单,并获得订单号 order_no,把order_no当参数传给此api,而示例中为了简化跟支付插件无关的代码,这里直接已时间戳生成了order_no
  166. */
  167. toPayDesk(){
  168. this.order_no = `test`+Date.now();
  169. this.out_trade_no = `${this.order_no}-1`;
  170. let options = {
  171. total_fee: this.total_fee, // 支付金额,单位分 100 = 1元
  172. order_no: this.order_no, // 业务系统订单号(即你自己业务系统的订单表的订单号)
  173. out_trade_no: this.out_trade_no, // 插件支付单号
  174. description: this.description, // 支付描述
  175. type: this.type, // 支付回调类型
  176. qr_code: this.qr_code, // 是否强制使用扫码支付
  177. openid: this.openid, // 微信公众号需要
  178. custom: this.custom, // 自定义数据
  179. };
  180. let optionsStr = encodeURI(JSON.stringify(options));
  181. uni.navigateTo({
  182. url:`/uni_modules/uni-pay/pages/pay-desk/pay-desk?options=${optionsStr}`
  183. });
  184. },
  185. // 打开查询订单的弹窗
  186. getOrderPopup(key){
  187. if (key) {
  188. this.$refs.getOrderPopup.open();
  189. } else {
  190. this.$refs.getOrderPopup.close();
  191. }
  192. },
  193. // 查询支付状态
  194. async getOrder() {
  195. this.getOrderRes = {};
  196. let res = await this.$refs.pay.getOrder({
  197. //out_trade_no: this.out_trade_no, // 插件支付单号 两者传1个即可
  198. transaction_id: this.transaction_id, // 第三方单号 两者传1个即可
  199. await_notify: true
  200. });
  201. if (res) {
  202. this.getOrderRes = res.pay_order;
  203. let obj = {
  204. "-1": "已关闭",
  205. "1": "已支付",
  206. "0": "未支付",
  207. "2": "已部分退款",
  208. "3": "已全额退款"
  209. };
  210. uni.showToast({
  211. title: obj[res.status] || res.errMsg,
  212. icon: "none"
  213. });
  214. }
  215. },
  216. // 发起退款
  217. async refund() {
  218. let res = await this.$refs.pay.refund({
  219. out_trade_no: this.out_trade_no, // 插件支付单号
  220. });
  221. if (res) {
  222. uni.showToast({
  223. title: res.errMsg,
  224. icon: "none"
  225. });
  226. }
  227. },
  228. // 查询退款状态
  229. async getRefund() {
  230. let res = await this.$refs.pay.getRefund({
  231. out_trade_no: this.out_trade_no, // 插件支付单号
  232. });
  233. if (res) {
  234. uni.showModal({
  235. content: res.errMsg,
  236. showCancel: false
  237. });
  238. }
  239. },
  240. // 关闭订单
  241. async closeOrder() {
  242. let res = await this.$refs.pay.closeOrder({
  243. out_trade_no: this.out_trade_no, // 插件支付单号
  244. });
  245. if (res) {
  246. uni.showModal({
  247. content: res.errMsg,
  248. showCancel: false
  249. });
  250. }
  251. },
  252. // 获取公众号code
  253. async getWeiXinJsCode(scope="snsapi_base") {
  254. let res = await this.$refs.pay.getProviderAppId({
  255. provider: "wxpay",
  256. provider_pay_type: "jsapi"
  257. });
  258. if (res.appid) {
  259. let appid = res.appid;
  260. let redirect_uri = window.location.href.split("?")[0];
  261. let url = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=${appid}&redirect_uri=${redirect_uri}&response_type=code&scope=${scope}&state=STATE#wechat_redirect`;
  262. window.location.href = url;
  263. }
  264. },
  265. // 获取公众号openid
  266. async getOpenid(data={}) {
  267. let res = await this.$refs.pay.getOpenid(data);
  268. if (res) {
  269. this.openid = res.openid;
  270. // 将openid缓存到本地
  271. uni.setStorageSync("uni-pay-weixin-h5-openid", this.openid);
  272. uni.setStorageSync("uni-pay-weixin-h5-code", data.code);
  273. uni.showToast({
  274. title: "已获取到openid,可以开始支付",
  275. icon: "none"
  276. });
  277. }
  278. },
  279. // 监听事件 - 支付订单创建成功(此时用户还未支付)
  280. onCreate(res){
  281. console.log('create: ', res);
  282. // 如果只是想生成支付二维码,不需要组件自带的弹窗,则在这里可以获取到支付二维码 qr_code_image
  283. },
  284. // 监听事件 - 支付成功
  285. onSuccess(res){
  286. console.log('success: ', res);
  287. if (res.user_order_success) {
  288. // 代表用户已付款,且你自己写的回调成功并正确执行了
  289. } else {
  290. // 代表用户已付款,但你自己写的回调执行失败(通常是因为你的回调代码有问题)
  291. }
  292. },
  293. onFail(err){
  294. console.log('err: ', err)
  295. },
  296. pageTo(url){
  297. uni.navigateTo({
  298. url
  299. });
  300. },
  301. providerFormat(provider){
  302. let providerObj = {
  303. "wxpay":"微信支付",
  304. "alipay":"支付宝支付",
  305. "appleiap":"ios内购"
  306. };
  307. let providerStr = providerObj[provider] || "未知";
  308. return providerStr;
  309. },
  310. /**
  311. * 日期格式化
  312. * @params {Date || Number} date 需要格式化的时间
  313. * timeFormat(new Date(),"yyyy-MM-dd hh:mm:ss");
  314. */
  315. timeFormat(time, fmt = 'yyyy-MM-dd hh:mm:ss', targetTimezone = 8){
  316. try {
  317. if (!time) {
  318. return "";
  319. }
  320. if (typeof time === "string" && !isNaN(time)) time = Number(time);
  321. // 其他更多是格式化有如下:
  322. // yyyy-MM-dd hh:mm:ss|yyyy年MM月dd日 hh时MM分等,可自定义组合
  323. let date;
  324. if (typeof time === "number") {
  325. if (time.toString().length == 10) time *= 1000;
  326. date = new Date(time);
  327. } else {
  328. date = time;
  329. }
  330. const dif = date.getTimezoneOffset();
  331. const timeDif = dif * 60 * 1000 + (targetTimezone * 60 * 60 * 1000);
  332. const east8time = date.getTime() + timeDif;
  333. date = new Date(east8time);
  334. let opt = {
  335. "M+": date.getMonth() + 1, //月份
  336. "d+": date.getDate(), //日
  337. "h+": date.getHours(), //小时
  338. "m+": date.getMinutes(), //分
  339. "s+": date.getSeconds(), //秒
  340. "q+": Math.floor((date.getMonth() + 3) / 3), //季度
  341. "S": date.getMilliseconds() //毫秒
  342. };
  343. if (/(y+)/.test(fmt)) {
  344. fmt = fmt.replace(RegExp.$1, (date.getFullYear() + "").substr(4 - RegExp.$1.length));
  345. }
  346. for (let k in opt) {
  347. if (new RegExp("(" + k + ")").test(fmt)) {
  348. fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (opt[k]) : (("00" + opt[k]).substr(("" + opt[k]).length)));
  349. }
  350. }
  351. return fmt;
  352. } catch (err) {
  353. // 若格式错误,则原值显示
  354. return time;
  355. }
  356. },
  357. },
  358. computed: {
  359. h5Env(){
  360. // #ifdef H5
  361. let ua = window.navigator.userAgent.toLowerCase();
  362. if (ua.match(/MicroMessenger/i) == 'micromessenger' && (ua.match(/miniprogram/i) == 'miniprogram')) {
  363. // 微信小程序
  364. return "mp-weixin";
  365. }
  366. if (ua.match(/MicroMessenger/i) == 'micromessenger') {
  367. // 微信公众号
  368. return "h5-weixin";
  369. }
  370. if (ua.match(/alipay/i) == 'alipay' && ua.match(/miniprogram/i) == 'miniprogram') {
  371. return "mp-alipay";
  372. }
  373. if (ua.match(/alipay/i) == 'alipay') {
  374. return "h5-alipay";
  375. }
  376. // 外部 H5
  377. return "h5";
  378. // #endif
  379. },
  380. // 计算当前是否是ios app
  381. isIosAppCom(){
  382. let info = uni.getSystemInfoSync();
  383. return info.uniPlatform === 'app' && info.osName === 'ios' ? true : false;
  384. },
  385. // 计算当前是否是PC环境
  386. isPcCom(){
  387. // #ifdef H5
  388. let info = uni.getSystemInfoSync();
  389. return info.deviceType === 'pc' ? true : false;
  390. // #endif
  391. return false;
  392. }
  393. },
  394. }
  395. </script>
  396. <style lang="scss" scoped>
  397. .app{
  398. padding: 30rpx;
  399. }
  400. input {
  401. border: 1px solid #f3f3f3;
  402. padding: 10rpx;
  403. width: 100%;
  404. box-sizing: border-box;
  405. height: 80rpx;
  406. }
  407. button {
  408. margin-top: 20rpx;
  409. }
  410. .label{
  411. margin: 10rpx 0;
  412. }
  413. .tips{
  414. margin-top: 20rpx;
  415. font-size: 24rpx;
  416. color: #565656;
  417. }
  418. .get-order-popup{
  419. background-color: #ffffff;
  420. padding: 30rpx;
  421. height: 60vh;
  422. border-radius: 30rpx 30rpx 0 0;
  423. overflow: hidden;
  424. }
  425. .mt20{
  426. margin-top: 20rpx;
  427. }
  428. .pd2030{
  429. padding: 20rpx 30rpx;
  430. }
  431. .table{
  432. font-size: 24rpx;
  433. }
  434. .align-left{
  435. text-align: left;
  436. width: 50%;
  437. }
  438. .align-right{
  439. text-align: right;
  440. width: 50%;
  441. }
  442. </style>