12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196 |
- <template>
- <view class="uni-pay">
- <!-- PC版收银台弹窗开始 -->
- <uni-pay-popup v-if="modeCom == 'pc'" ref="payPopup" type="center" :safe-area="false">
- <view class="pc-pay-popup">
- <view class="pc-pay-popup-title"><text class="text">收银台</text></view>
- <view class="pc-pay-popup-flex">
- <view class="pc-pay-popup-qrcode-box">
- <image class="pc-pay-popup-qrcode-image" :src="orderRes['qr_code_image']"></image>
- <view class="pc-pay-popup-amount-box">
- <view class="pc-pay-popup-amount-tips">
- <text class="text" v-if="orderRes['provider'] == 'wxpay'">微信扫一扫付款</text>
- <text class="text" v-else-if="orderRes['provider'] == 'alipay'">支付宝扫一扫付款</text>
- <text class="text" v-else>扫一扫付款</text>
- </view>
- <view class="pc-pay-popup-amount"><text class="text">{{ (totalFeeCom / 100).toFixed(2) }}</text></view>
- </view>
- <view class="pc-pay-popup-complete-button" v-if="orderRes['qr_code_image']">
- <button type="primary" @click="_getOrder()">我已完成支付</button>
- </view>
- </view>
- <view class="pc-pay-popup-provider-list">
- <view class="pc-pay-popup-provider-item" v-if="currentProviders.indexOf('wxpay') > -1" :class="myOptions['provider'] == 'wxpay' ? 'active' : ''"
- @click="_pcChooseProvider('wxpay')">
- <image :src="images['wxpay']" class="pc-pay-popup-provider-image"></image>
- <text class="pc-pay-popup-provider-text">微信支付</text>
- </view>
- <view class="pc-pay-popup-provider-item" v-if="currentProviders.indexOf('alipay') > -1" :class="myOptions['provider'] == 'alipay' ? 'active' : ''"
- @click="_pcChooseProvider('alipay')">
- <image :src="images['alipay']" class="pc-pay-popup-provider-image"></image>
- <text class="pc-pay-popup-provider-text">支付宝支付</text>
- </view>
- <view class="pc-pay-popup-logo">
- <image class="image" :src="logo" mode="widthFix"></image>
- </view>
- </view>
- </view>
- </view>
- </uni-pay-popup>
- <!-- PC版收银台弹窗结束 -->
- <!-- 手机版收银台弹窗开始 -->
- <uni-pay-popup v-else ref="payPopup" type="bottom" :safe-area="false">
- <view class="mobile-pay-popup" :style="'min-height: '+height+';'">
- <view class="mobile-pay-popup-title"><text class="text">收银台</text></view>
- <view class="mobile-pay-popup-amount-box">
- <view><text class="text">待支付金额:</text></view>
- <view class="mobile-pay-popup-amount"><text class="text">{{ (totalFeeCom / 100).toFixed(2) }}</text></view>
- </view>
- <view class="mobile-pay-popup-provider-list">
- <view class="uni-list">
- <!-- #ifdef MP-WEIXIN || H5 || APP -->
- <view class="uni-list-item" v-if="currentProviders.indexOf('wxpay') > -1" @click="createOrderByProvider('wxpay')">
- <view class="uni-list-item__container container--right">
- <view class="uni-list-item__header">
- <image :src="images['wxpay']" class="image"></image>
- </view>
- <view class="uni-list-item__content uni-list-item__content--center">
- <text class="text">微信支付</text>
- </view>
- </view>
- <view class="arrowright"></view>
- </view>
- <!-- #endif -->
- <!-- #ifdef MP-ALIPAY || H5 || APP -->
- <view class="uni-list-item" v-if="currentProviders.indexOf('alipay') > -1" @click="createOrderByProvider('alipay')">
- <view class="uni-list-item__container container--right">
- <view class="uni-list-item__header">
- <image :src="images['alipay']" class="image"></image>
- </view>
- <view class="uni-list-item__content uni-list-item__content--center">
- <text class="text">支付宝</text>
- </view>
- </view>
- <view class="arrowright"></view>
- </view>
- <!-- #endif -->
- </view>
- </view>
- </view>
- </uni-pay-popup>
- <!-- 手机版收银台弹窗结束 -->
- <!-- 二维码支付弹窗开始 -->
- <uni-pay-popup ref="qrcodePopup" type="center" :safe-area="false" :animation="false" :mask-click="false" @close="clearQrcode">
- <view class="qrcode-popup-content">
- <image :src="orderRes['qr_code_image']" class="qrcode-image"></image>
- <view class="qrcode-popup-info">
- <view class="qrcode-popup-info-fee-box">
- <view class="qrcode-popup-info-fee">
- <text class="text">{{ (totalFeeCom / 100).toFixed(2) }}</text>
- </view>
- <view class="qrcode-popup-info-fee-unit">
- <text class="text">元</text>
- </view>
- </view>
- <view v-if="myOptions['provider'] == 'wxpay'"><text>请用微信扫码支付</text></view>
- <view v-else-if="myOptions['provider'] == 'alipay'"><text>请用支付宝扫码支付</text></view>
- </view>
- <button type="primary" @click="_getOrder()" class="qrcode-popup-btn-primary">我已完成支付</button>
- <view class="qrcode-popup-cancel" @click="closePopup('qrcodePopup')"><text class="qrcode-popup-cancel-text">暂不支付</text></view>
- </view>
- </uni-pay-popup>
- <!-- 二维码支付弹窗结束 -->
- <!-- 外部浏览器确认支付弹窗开始 -->
- <uni-pay-popup ref="payConfirmPopup" type="center" :safe-area="false" :animation="false" :mask-click="false">
- <view class="pay-confirm-popup-content">
- <view class="pay-confirm-popup-title"><text>请确认支付是否已完成</text></view>
- <view><button type="primary" @click="_getOrder()">已完成支付</button></view>
- <view class="pay-confirm-popup-refresh"><button type="default" @click="_afreshPayment()">支付遇到问题,重新支付</button></view>
- <view class="pay-confirm-popup-cancel" @click="closePopup('payConfirmPopup')"><text>暂不支付</text></view>
- </view>
- </uni-pay-popup>
- <!-- 外部浏览器确认支付弹窗结束 -->
- </view>
- </template>
- <script>
- import { checkPlatform, objectAssign, getWeixinCode, getAlipayCode } from "../../js_sdk/js_sdk"
- export default {
- name: "uni-pay",
- emits: ["success", "cancel", "fail", "create", "mounted", "qrcode"],
- props: {
- /**
- * Banner广告位id
- */
- adpid: {
- type: String,
- default: ""
- },
- /**
- * 是否自动跳转到插件内置的支付成功页面(具有看广告功能,可以增加开发者收益)默认true
- */
- toSuccessPage: {
- type: Boolean,
- default: true
- },
- /**
- * 支付成功后,点击查看订单按钮时跳转的页面地址
- */
- returnUrl: {
- type: String,
- default: ""
- },
- /**
- * 支付结果页主色调,默认支付宝小程序为#108ee9,其他端均为#01be6e
- * 建议:绿色系 #01be6e 蓝色系 #108ee9 咖啡色 #816a4e 粉红 #fe4070 橙黄 #ffac0c 橘黄 #ff7100
- */
- mainColor: {
- type: String,
- default: ""
- },
- /**
- * 收银台模式
- * mobile 手机版
- * pc 电脑版
- */
- mode: {
- type: String,
- default: ""
- },
- /**
- * PC收银台模式时,展示的logo
- */
- logo: {
- type: String,
- default: "/static/logo.png"
- },
- /**
- * 收银台高度(默认70vh)
- */
- height: {
- type: [String],
- default: "70vh"
- },
- /**
- * 是否打印运行过程日志
- */
- debug: {
- type: Boolean,
- default: false
- }
- },
- data() {
- return {
- // 支付参数
- myOptions: {} as UTSJSONObject,
- // 支付云对象返回结果
- orderRes: {
- order: "",
- order_no: "",
- out_trade_no: "",
- provider: "",
- provider_pay_type: "",
- errCode: 0,
- } as UTSJSONObject,
- images: {
- wxpay: "",
- alipay: ""
- } as UTSJSONObject,
- originalRroviders: ["wxpay", "alipay"] as Array<string>,
- currentProviders: ["wxpay", "alipay"] as Array<string>,
- openid: ""
- }
- },
- mounted() {
- // #ifdef MP-WEIXIN || MP-ALIPAY
- if (this.openid == "") {
- new Promise(async (resolve, reject) => {
- let code = await this.getCode();
- let res = await this.getOpenid({
- // #ifdef MP-WEIXIN
- provider: "wxpay",
- // #endif
- // #ifdef MP-ALIPAY
- provider: "alipay",
- // #endif
- code
- });
- resolve(res);
- }).then((res) => {
- if (res['openid'] != null) {
- this.openid = res['openid'];
- }
- }).catch((err) => {
- reject(err);
- })
- }
- // #endif
- // #ifdef MP-WEIXIN
- // 如果是微信小程序,则设置只支持微信支付
- this.originalRroviders = ["wxpay"];
- this.currentProviders = JSON.parse(JSON.stringify(this.originalRroviders));
- // #endif
- // #ifdef MP-ALIPAY
- // 如果是支付宝小程序,则设置只支持支付宝支付
- this.originalRroviders = ["alipay"];
- this.currentProviders = JSON.parse(JSON.stringify(this.originalRroviders));
- // #endif
- let insideData = {
- images: this.images as UTSJSONObject,
- originalRroviders: this.originalRroviders as Array<string>,
- currentProviders: this.currentProviders as Array<string>,
- } as UTSJSONObject;
- this.$emit("mounted", insideData);
- },
- methods: {
- // 发起支付 - 打开支付选项弹窗
- async open(options : UTSJSONObject) : Promise<void> {
- let provider = options['provider'] as string | null;
- if (provider != null && provider != "") {
- let providers : Array<string> = [];
- this.originalRroviders.map((item : string) => {
- if (provider == item) {
- providers.push(item);
- }
- });
- this.currentProviders = providers;
- options['provider'] = "";
- } else {
- this.currentProviders = JSON.parse(JSON.stringify(this.originalRroviders)) as Array<string>;
- }
- this.myOptions = options;
- if (this.currentProviders.length == 1) {
- this.createOrder({ provider: this.currentProviders[0] });
- } else {
- if (this.modeCom == "pc") {
- this._pcChooseProvider(this.currentProviders[0]);
- }
- this.openPopup("payPopup");
- }
- },
- createOrderByProvider(provider : string) {
- this.createOrder({
- provider
- })
- },
- // 创建支付
- async createOrder(data : UTSJSONObject) : Promise<void> {
- let options = this.myOptions;
- options['qr_code'] = false;
- options = objectAssign(options, data);
- if (options['provider'] == "appleiap") {
- // 苹果虚拟支付走特殊逻辑
- // #ifdef APP-IOS
- return this._appleiapCreateOrder(options);
- // #endif
- // #ifndef APP-IOS
- uni.showModal({
- title: "提示",
- content: "请在iOS系统中执行",
- showCancel: false
- })
- // #endif
- }
- // #ifdef APP
- if (options['provider'] == "wxpay") {
- // #ifdef uniVersion < 4.11
- uni.showModal({
- title: "提示",
- content: "请先升级HBX至4.11",
- showCancel: false
- })
- return;
- // #endif
- }
- // #endif
- // #ifdef H5
- // 判断如果是pc访问,则强制扫码模式
- if (checkPlatform() == "pc") {
- options['qr_code'] = true;
- }
- // #endif
- let createOrderData = {
- provider: options['provider'],
- total_fee: options.getNumber('total_fee'),
- order_no: options['order_no'],
- out_trade_no: options['out_trade_no'],
- description: options['description'],
- type: options['type'],
- qr_code: options.getBoolean('qr_code'),
- custom: options.getJSON('custom'),
- other: options.getJSON('other'),
- wxpay_virtual: options.getJSON('wxpay_virtual'), // 微信小程序虚拟支付需要
- } as UTSJSONObject;
- // #ifdef H5
- if (options['openid'] != "" && options['provider'] == "wxpay") {
- createOrderData['openid'] = options['openid'];
- }
- // #endif
- // #ifdef MP
- if (this.openid != "") {
- createOrderData['openid'] = this.openid;
- }
- // #endif
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.createOrder(createOrderData);
- if (res['errCode'] == 0) {
- this.$emit("create", res);
- if (res['qr_code'] != null && res['qr_code'] == true && options.getBoolean('cancel_popup') != true) {
- this.orderRes = res;
- // 展示组件内置的二维码弹窗
- if (this.modeCom == "pc") {
- this.openPopup("payPopup");
- this._pcChooseProvider(options['provider'] as string);
- } else {
- this.openPopup("qrcodePopup");
- }
- } else {
- // 调起支付
- this.orderPayment(res);
- }
- } else {
- this.$emit("fail", res);
- }
- } catch (err) {
- this.$emit("fail", {
- errCode: -1,
- errMsg: (err as Error).message
- });
- }
- },
- // 调起支付
- orderPayment(res : UTSJSONObject) {
- this.orderRes = res;
- if (res['qr_code'] != null && res['qr_code'] != "") {
- this.$emit("qrcode", res);
- }
- // #ifdef H5
- let order = res.get('order') as UTSJSONObject;
- if (res['provider_pay_type'] == "jsapi") {
- // 微信公众号支付
- (window.WeixinJSBridge as any).invoke("getBrandWCPayRequest", order, (res : any) => {
- if (res.err_msg == "get_brand_wcpay_request:ok") {
- // 用户支付成功回调
- this._getOrder();
- } else if (res.err_msg == "get_brand_wcpay_request:cancel") {
- // 用户取消支付回调
- this.$emit("cancel", res);
- } else if (res.err_msg == "get_brand_wcpay_request:fail") {
- // 用户支付失败回调
- console.error('getBrandWCPayRequest-fail: ', res);
- this.$emit("fail", res);
- }
- });
- } else {
- // 外部浏览器支付
- let codeUrl = order['codeUrl'];
- let mwebUrl = order['h5Url'] || order['mwebUrl'] || order['mweb_url'];
- setTimeout(() => {
- this.openPopup("payConfirmPopup");
- window.location.href = codeUrl || mwebUrl;
- }, 200);
- }
- // #endif
- // #ifndef H5
- let _order = res.get('order');
- let orderStr = typeof _order == "string" ? _order as string : JSON.stringify(_order) as string;
- console.log('orderStr: ', orderStr)
- uni.requestPayment({
- provider: res['provider'] as string,
- orderInfo: orderStr,
- // #ifdef MP-WEIXIN
- ..._order,
- // #endif
- success: (res) => {
- console.log("requestPaymentSuccess", JSON.stringify(res))
- this._getOrder();
- },
- fail: (err) => {
- console.log("requestPaymentFail", JSON.stringify(err))
- let errCode = err.errCode;
- let errMsg = err.errMsg;
- if (errCode == 700601 || errMsg.indexOf("fail cancel") > -1) {
- // 用户取消支付
- this.$emit("cancel", {
- errCode: errCode,
- errMsg: errMsg
- });
- } else {
- // 发起支付失败
- console.error("uni.requestPayment:fail", err);
- this.$emit("fail", {
- errCode: errCode,
- errMsg: errMsg
- });
- }
- }
- });
- // #endif
- },
- // 打开弹窗
- openPopup(name : string) {
- let popupRef = this.$refs[name] as UniPayPopupComponentPublicInstance;
- popupRef.open();
- },
- // 关闭弹窗
- closePopup(name : string) {
- let popupRef = this.$refs[name] as UniPayPopupComponentPublicInstance;
- popupRef.close();
- },
- // 查询订单(查询支付情况)
- async getOrder(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.getOrder(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 发起退款(此接口需要admin角色才可以访问)
- async refund(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.refund(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 查询退款(查询退款情况)
- async getRefund(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.getRefund(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 关闭订单
- async closeOrder(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.closeOrder(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 获取支持的支付供应商
- async getPayProviderFromCloud(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.getPayProviderFromCloud(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 获取支付配置内的appid(主要用于获取获取微信公众号的appid,用以获取code)
- async getProviderAppId(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.getProviderAppId(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 根据code获取openid
- async getOpenid(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.getOpenid(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 验证iosIap苹果内购支付凭据
- async verifyReceiptFromAppleiap(data : UTSJSONObject) : Promise<UTSJSONObject> {
- try {
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.verifyReceiptFromAppleiap(data);
- return res;
- } catch (err) {
- return {
- errCode: -1,
- errMsg: (err as Error).message
- } as UTSJSONObject
- }
- },
- // 支付成功后的逻辑
- paySuccess(res : UTSJSONObject) {
- this.closePopup("payPopup");
- this.closePopup("payConfirmPopup");
- this.clearQrcode();
- let toSuccessPage = this.toSuccessPage as boolean;
- if (toSuccessPage) {
- // 跳转到支付成功的内置页面
- this.pageToSuccess(res);
- }
- this.$emit("success", res);
- },
- pageToSuccess(res : UTSJSONObject) {
- let out_trade_no = res['out_trade_no'] as string;
- let pay_order = res.getJSON('pay_order') as UTSJSONObject;
- let order_no = pay_order['order_no'] as string;
- //let pay_date = pay_order['pay_date'];
- let total_fee = pay_order.getNumber('total_fee');
- if (total_fee == null) {
- total_fee = 0;
- }
- let returnUrl = this.returnUrl as string;
- let adpid = this.adpid as string;
- let mainColor = this.mainColor as string;
- if (this.modeCom != "pc") {
- uni.navigateTo({
- url: `/uni_modules/uni-pay-x/pages/success/success?out_trade_no=${out_trade_no}&order_no=${order_no}&total_fee=${total_fee}&adpid=${adpid}&return_url=${returnUrl}&main_color=${mainColor}`
- });
- } else {
- if (returnUrl != "") {
- let url = returnUrl + `?out_trade_no=${out_trade_no}&order_no=${order_no}`;
- if (url.indexOf("/") != 0) url = `/${url}`;
- uni.navigateTo({
- url
- });
- }
- }
- },
- // 监听 - 关闭二维码弹窗
- clearQrcode() {
- this.orderRes["codeUrl"] = "";
- this.orderRes["qr_code_image"] = "";
- },
- // 内部函数查询支付状态
- async _getOrder() : Promise<void> {
- let out_trade_no = this.orderRes["out_trade_no"] as string;
- let res = await this.getOrder({
- out_trade_no,
- await_notify: true
- });
- if (res['errCode'] == 0) {
- let has_paid = res.getBoolean('has_paid');
- if (has_paid != null && has_paid == true) {
- this.closePopup("qrcodePopup");
- this.paySuccess(res);
- }
- }
- },
- // 重新发起支付
- _afreshPayment() {
- this.orderPayment(this.orderRes);
- },
- // pc版弹窗选择支付方式
- _pcChooseProvider(provider : string) {
- let _provider : string = this.myOptions["provider"] as string;
- if (provider != _provider) {
- this.createOrder({ provider: provider })
- }
- },
- // 苹果虚拟支付支付逻辑
- async _appleiapCreateOrder(options : UTSJSONObject) : Promise<void>{
- // #ifndef APP-IOS
- uni.showToast({
- title: "请在iOS系统中打开",
- icon: "none"
- })
- // #endif
- // #ifdef APP-IOS
- const virtualPaymentManager = uni.getVirtualPaymentManager();
- if (options.apple_virtual['product_type'] == 'nonconsumable') {
- uni.showLoading({
- title: "请求中...",
- mask: true
- })
- let purchased = await new Promise((resolve, reject) => {
- // 如果是非消耗性产品只能购买一次,需要先判断下
- virtualPaymentManager.restoreTransactions({
- success: (res) => {
- let transactions = res.transactions;
- let index = res.transactions.findIndex(transaction => {
- return transaction.productId == options.apple_virtual['product_id']
- })
- resolve(index > -1 ? true : false);
- }
- });
- });
- uni.hideLoading()
- if (purchased) {
- uni.showModal({
- title: "提示",
- content: "您已购买过此商品,请勿重复购买",
- showCancel: false
- });
- return;
- }
- }
- let createOrderData = {
- provider: options.provider,
- total_fee: options.total_fee,
- order_no: options.order_no,
- out_trade_no: options.out_trade_no,
- description: options.description,
- type: options.type,
- apple_virtual: options.apple_virtual,
- custom: options.custom,
- } as UTSJSONObject;
- // 引入支付云对象
- const uniPayCo = uniCloud.importObject("uni-pay-co");
- let res = await uniPayCo.createOrder(createOrderData);
- if (res.errCode == 0) {
- this.$emit("create", res);
- this.res = res;
- uni.showLoading({
- title: '支付请求中...'
- });
- try {
- // 请求苹果支付
- if (this.debug) console.log("正在请求苹果服务器", res.out_trade_no);
- uni.requestVirtualPayment({
- apple: {
- productId: options.getJSON('apple_virtual')!.getString('product_id')!,
- appAccountToken: res.appleiap_account_token,
- quantity: options.getJSON('apple_virtual')!.getNumber('buy_quantity')! || 1,
- },
- success: async (requestPaymentRes) => {
- uni.hideLoading()
- let transaction = requestPaymentRes?.apple;
- if (this.debug) console.log('用户支付成功', transaction);
- let transactionIdentifier : string = transaction.transactionIdentifier;
- let transactionDate : string = transaction.transactionDate;
- let outTradeNo : string = res.out_trade_no;
- uni.showLoading({
- title: '正在处理支付结果...'
- });
- // 云端请求苹果服务器验证票据
- let verifyRes = await this.verifyReceiptFromAppleiap({
- out_trade_no: outTradeNo,
- transaction_receipt: transaction.jsonRepresentation,
- transaction_identifier: transactionIdentifier
- });
- if (verifyRes.errCode == 0) {
- if (verifyRes.repeat) {
- uni.showModal({
- title: "提示",
- content: `当前道具只能购买一次`,
- showCancel: false,
- confirmText: "好的"
- });
- } else {
- //经过开发者server验证成功后请结束该交易
- virtualPaymentManager.finishTransaction({
- transaction: transaction,
- success: (r) => {
- if (this.debug) console.log("关单成功, 该productId= " + transaction.productId)
- },
- fail: (e) => {
- if (this.debug) console.log("关单失败, 该productId= " + transaction.roductId)
- }
- });
- uni.hideLoading();
- this.paySuccess(verifyRes);
- }
- } else {
- if (this.debug) console.log('verifyRes: ', verifyRes)
- }
- },
- fail: (err) => {
- uni.hideLoading();
- if (this.debug) console.log("购买失败:errSubject= " + err.errSubject + ", errCode= " + err.errCode + ", errMsg= " + err.errMsg);
- if ([700601].indexOf(err.errCode) > -1 || err.errMsg.indexOf("cancel") > -1) {
- this.$emit("cancel", err);
- } else {
- this.$emit("fail", {
- errCode: err.errCode,
- errMsg: err.errMsg
- });
- }
- }
- });
- } catch (err) {
- let code = err.errCode || err.code;
- if (code == 2) {
- // 用户取消支付
- if (this.debug) console.log("用户取消支付");
- this.$emit("cancel", err);
- } else {
- // 发起支付失败
- console.error("appleiapCreateOrder:fail", err);
- this.$emit("fail", {
- errCode: code,
- errMsg: err.errMsg || err.message
- });
- }
- uni.hideLoading();
- }
- }
- // #endif
- },
- // 苹果虚拟支付未完成订单检测
- appleiapRestore() {
- // #ifdef APP-IOS
- uni.showLoading({
- title: "",
- mask: true
- });
- try {
- const virtualPaymentManager = uni.getVirtualPaymentManager();
- virtualPaymentManager.getUnfinishedTransactions({
- success: async (res) => {
- uni.hideLoading()
- console.log("获取未结束的订单列表个数:" + res.transactions.length)
- res.transactions.forEach(async transaction => {
- console.log("getUnfinishedTransactions成功的交易productId= " + transaction.productId);
- let appAccountToken : string = transaction.appAccountToken;
- let transactionIdentifier : string = transaction.transactionIdentifier;
- //let originalTransactionIdentifier : string = transaction.originalTransactionIdentifier;
- let transactionDate : string = transaction.transactionDate;
- // 云端请求苹果服务器验证票据
- let verifyRes = await this.verifyReceiptFromAppleiap({
- appleiap_account_token: appAccountToken,
- transaction_receipt: transaction.jsonRepresentation,
- transaction_identifier: transactionIdentifier,
- });
- if (verifyRes.errCode == 0 || !appAccountToken) {
- // 经过开发者server验证成功后请结束该交易
- virtualPaymentManager.finishTransaction({
- transaction: transaction,
- success: (r) => {
- if (this.debug) console.log("关单成功, 该productId= " + transaction.productId)
- },
- fail: (e) => {
- if (this.debug) console.log("关单失败, 该productId= " + transaction.productId)
- }
- });
- uni.hideLoading();
- // 如果是自动续期,则不跳页面
- if (!verifyRes.is_subscribe && verifyRes.pay_order) {
- this.paySuccess(verifyRes);
- }
- } else {
- if (this.debug) console.log('verifyRes: ', verifyRes)
- }
- })
- },
- fail: (e) => {
- uni.hideLoading()
- console.log("获取未结束的订单列表失败:errSubject= " + e.errSubject + ", errCode= " + e.errCode + ", errMsg= " + e.errMsg)
- uni.showToast({
- title: "获取未结束的订单列表失败:errCode= " + e.errCode,
- icon: 'error'
- });
- }
- })
- } catch(err){
- console.error('err: ', err)
- uni.hideLoading()
- }
- // #endif
- },
- // #ifdef MP-WEIXIN || MP-ALIPAY
- // 获取code
- async getCode() : Promise<string>{
- // #ifdef MP-WEIXIN
- return getWeixinCode();
- // #endif
- // #ifdef MP-ALIPAY
- return getAlipayCode();
- // #endif
- },
- // #endif
- },
- watch: {
- },
- computed: {
- modeCom() : string {
- let mode = this.mode as string;
- if (mode != "") return mode;
- let systemInfo = uni.getDeviceInfo();
- return systemInfo.deviceType == "pc" ? "pc" : "mobile";
- },
- totalFeeCom() : number {
- let totalFee = this.myOptions.getNumber('total_fee');
- return totalFee != null ? totalFee : 0;
- }
- },
- }
- </script>
- <style lang="scss" scoped>
- $bgcolor: #f3f3f3;
- .uni-pay {}
- /* 手机版收银台弹窗开始 */
- .mobile-pay-popup {
- width: 100%;
- min-height: 450px;
- background-color: $bgcolor;
- border-radius: 15px 15px 0 0;
- overflow: hidden;
- .mobile-pay-popup-title {
- background-color: #ffffff;
- padding: 10px;
- .text {
- text-align: center;
- font-weight: bold;
- font-size: 20px;
- }
- }
- .mobile-pay-popup-amount-box {
- background-color: #ffffff;
- padding: 15px;
- .mobile-pay-popup-amount {
- margin-top: 10px;
- .text {
- color: #e43d33;
- font-size: 30px;
- }
- }
- }
- .mobile-pay-popup-provider-list {
- background-color: #ffffff;
- margin-top: 10px;
- .uni-list {
- display: flex;
- background-color: #fff;
- position: relative;
- flex-direction: column;
- .uni-list-item {
- display: flex;
- position: relative;
- justify-content: space-between;
- align-items: center;
- background-color: #fff;
- flex-direction: row;
- border-bottom: 1px solid #f8f8f8;
- /* #ifdef H5 */
- cursor: pointer;
- /* #endif */
- &:hover {
- background-color: #f1f1f1;
- }
- .uni-list-item__container {
- position: relative;
- display: flex;
- flex-direction: row;
- padding: 12px 15px;
- padding-left: 15px;
- flex: 1;
- overflow: hidden;
- .uni-list-item__header {
- display: flex;
- flex-direction: row;
- align-items: center;
- .image {
- width: 26px;
- height: 26px;
- margin-right: 9px;
- }
- }
- }
- .container--right {
- padding-right: 0;
- }
- .uni-list-item__content {
- display: flex;
- padding-right: 8px;
- flex: 1;
- flex-direction: column;
- justify-content: space-between;
- overflow: hidden;
- .text {
- color: #3b4144;
- font-size: 14px;
- }
- }
- .uni-list-item__content--center {
- justify-content: center;
- }
- .arrowright {
- border-top: 1px solid #bbbbbb;
- border-right: 1px solid #bbbbbb;
- width: 8px;
- height: 8px;
- margin-right: 15px;
- transform: rotate(45deg);
- }
- }
- }
- }
- }
- /* 手机版收银台弹窗结束 */
- /* PC版收银台弹窗开始 */
- .pc-pay-popup {
- width: 800px;
- height: 600px;
- background-color: $bgcolor;
- border-radius: 10px;
- overflow: hidden;
- .pc-pay-popup-title {
- background-color: #ffffff;
- height: 66px;
- .text {
- text-align: center;
- font-weight: bold;
- font-size: 20px;
- line-height: 66px;
- }
- }
- .pc-pay-popup-flex {
- width: 100%;
- display: flex;
- flex-direction: row;
- .pc-pay-popup-qrcode-box {
- height: 534px;
- flex: 1;
- background-color: #ffffff;
- display: flex;
- flex-direction: row;
- flex-direction: column;
- justify-content: center;
- align-items: center;
- .pc-pay-popup-qrcode-image {
- width: 225px;
- height: 225px;
- }
- .pc-pay-popup-amount-box {
- .pc-pay-popup-amount-tips {
- margin-top: 20px;
- .text {
- text-align: center;
- color: #333;
- font-size: 20px;
- }
- }
- .pc-pay-popup-amount {
- margin-top: 20px;
- .text {
- text-align: center;
- color: #dd524d;
- font-weight: bold;
- font-size: 32px;
- }
- }
- }
- .pc-pay-popup-complete-button {
- margin-top: 20px;
- }
- }
- .pc-pay-popup-provider-list {
- width: 300px;
- display: flex;
- flex-direction: column;
- .pc-pay-popup-provider-item {
- padding: 20px;
- display: flex;
- flex-direction: row;
- align-items: center;
- .pc-pay-popup-provider-image {
- width: 60px;
- height: 60px;
- }
- .pc-pay-popup-provider-text {
- color: #333;
- font-size: 20px;
- margin-left: 10px;
- }
- }
- .pc-pay-popup-provider-item.active {
- background-color: #ffffff;
- }
- .pc-pay-popup-provider-item:hover {
- background-color: #ffffff;
- /* #ifdef H5 */
- cursor: pointer;
- /* #endif */
- }
- .pc-pay-popup-logo {
- flex: 1;
- display: flex;
- flex-direction: row;
- align-items: center;
- justify-content: center;
- .image {
- width: 120px;
- }
- }
- }
- }
- }
- /* PC版收银台弹窗结束 */
- /* 二维码支付弹窗开始 */
- .qrcode-popup-content {
- width: 300px;
- background-color: #ffffff;
- border-radius: 5px;
- padding: 20px;
- box-sizing: border-box;
- text-align: center;
- display: flex;
- flex-direction: column;
- align-items: center;
- justify-content: center;
- .qrcode-image {
- width: 225px;
- height: 225px;
- }
- .qrcode-popup-info {
- padding: 10px;
- display: flex;
- flex-direction: column;
- align-items: center;
- .qrcode-popup-info-fee-box {
- display: flex;
- flex-direction: row;
- margin-bottom: 3px;
- .qrcode-popup-info-fee {
- .text {
- color: red;
- font-size: 30px;
- font-weight: bold;
- text-align: center;
- }
- }
- .qrcode-popup-info-fee-unit {
- display: flex;
- flex-direction: row;
- align-items: flex-end;
- padding-bottom: 5px;
- }
- }
- }
- .qrcode-popup-btn-primary {
- width: 260px;
- }
- .qrcode-popup-cancel {
- margin-top: 10px;
- width: 260px;
- .qrcode-popup-cancel-text {
- text-align: center;
- }
- }
- }
- /* 二维码支付弹窗结束 */
- /* 外部浏览器H5支付弹窗确认开始 */
- .pay-confirm-popup-content {
- width: 275px;
- background-color: #ffffff;
- border-radius: 5px;
- padding: 20px;
- .pay-confirm-popup-title {
- text-align: center;
- padding: 10px 0;
- margin-bottom: 15px;
- }
- .pay-confirm-popup-refresh {
- margin-top: 10px;
- }
- .pay-confirm-popup-cancel {
- margin-top: 10px;
- text-align: center;
- }
- }
- /* 外部浏览器H5支付弹窗确认结束 */
- </style>
|