edit.vue 14 KB

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