edit.vue 14 KB

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