支持微信支付和支付寶支付的收銀臺(tái)(支持微信環(huán)境/瀏覽器環(huán)境/App)

demo圖

image.png

前置操作

1、微信平臺(tái)要配置支付域名
2、申請(qǐng)appid

支付流程

1涨冀、微信支付在微信環(huán)境會(huì)直接喚起支付
2、微信支付在瀏覽器會(huì)先跳轉(zhuǎn)到微信支付麦萤、支付成功與否都會(huì)調(diào)轉(zhuǎn)到你寫的頁(yè)面鹿鳖、加個(gè)彈窗、讓用戶點(diǎn)擊是否支付來(lái)獲取支付狀態(tài)
3频鉴、支付寶在任何環(huán)境下都可以支付

<template>
    <div class="pay-page">
        <div class="pay-moneny">
            <span>¥</span>
            <span class="pay-account">{{ amount }}</span>
        </div>
        <div class="pay-title">
            {{ tradeItem }}
        </div>
        <div class="pay-type">
            <p>選擇支付方式</p>
            <van-radio-group v-model="formData.payPlate">
                <div class="pay-list">
                    <div class="left">
                        <img src="@/assets/images/wxpay.png" alt="微信">
                        <span>微信</span>
                    </div>
                    <van-radio checked-color="rgb(206,51,58)" name="wechat" />
                </div>
                <div class="pay-list">
                    <div class="left">
                        <img src="@/assets/images/zfpay.png" alt="支付寶">
                        <span>支付寶</span>
                    </div>
                    <van-radio checked-color="rgb(206,51,58)" name="alipay" />
                </div>
            </van-radio-group>
        </div>
        <div class="btn-bottom">
            <van-button style="background: rgb(206,51,58);color: #ffffff;" block @click="pay">立即支付</van-button>
        </div>
    </div>
</template>

<script setup>
import { ref, onMounted, nextTick } from "vue"
import { showToast, showConfirmDialog } from 'vant';
import { getOpenId, allPay, getOrderInfo, getOrderList } from '@/api/pay.ts'  // 接口API
import { useRouter, useRoute } from 'vue-router'

useHead({
  title: '支付',
  meta: [
      { hid: 'keywords', name: 'keywords', content: ' bingbing' }
  ]
})

const router = useRouter()
const route = useRoute()

const formData = reactive({
    openId: '',
    payPlate: 'wechat',
    productId: '',
    tradeId: '',
    flowId: '',
    tradeType: ''
})

const appId = ref() // 自己申請(qǐng)
const loading = ref(false)
const amount = ref()
const tradeItem = ref()
const redirectUrl = ref()

onMounted(() => {
    const query = route.query
    const openId = localStorage.getItem('openId')
    if(getEnvironment() == 'weixin'){
        if(openId) {
            formData.openId = openId
        }else {
            checkCode()
        }
    }
   if(query.tradeId) {
        formData.tradeId = query.tradeId
        getOrder()
        if(query.fromWx == 'y') {
            isPay()
        }
    }
    if(query.out_trade_no) {
        formData.flowId = route.query.out_trade_no
        isPay()
    }
})

// 獲取訂單信息
const getOrder = async () => {
    const res = await getOrderInfo({
        tradeId: formData.tradeId,
        flowId: formData.flowId
    })
    if(res.code === 'SUCCESS'){
        amount.value = res.data.amount
        tradeItem.value = res.data.tradeItem
        redirectUrl.value = res.data.redirectUrl
    }else{
        showToast('訂單獲取失敗')
    }
}

//判斷當(dāng)前環(huán)境(微信/其他)
const getEnvironment = () => {
    let ua = navigator.userAgent.toLowerCase();
    if (ua.match(/MicroMessenger/i) == "micromessenger") {
        return 'weixin'
    }else {
        return 'other'
    }
}

// 立即支付
const pay = async () => {
    let payInfo = {}
    loading.value = true
    const res = await allPay({
        ...formData,
        tradeType: formData.payPlate == 'alipay' ? 'MWEB' : getEnvironment() == 'weixin' ? 'JSAPI' : 'MWEB'
    })
    loading.value = false
    if(res.code === 'SUCCESS') {
        payInfo = {
            nonceStr: res.data.nonceStr,
            prepayIdPackage: res.data.prepayIdPackage,
            paySign: res.data.paySign,
            timeStamp: res.data.timeStamp,
            signType: res.data.signType,
            mwebUrl: res.data.mwebUrl,
            formContent: res.data.formContent
        }
    }else {
        showToast(res.message)
        return false
    }
    if(formData.payPlate == 'wechat') {
        // 微信支付的兩種方式
        if(getEnvironment() == 'weixin') {
            pullWchatPay(payInfo)
        }else {
            // 微信回調(diào)地址
            location.href = payInfo.mwebUrl + '&redirect_url=' + encodeURIComponent(window.location.href + '&fromWx=y')
        }
    }else if (formData.payPlate == 'alipay') {
        // 支付寶支付
        pullAliPay(payInfo.formContent)
    }
    
}

// 獲取微信用戶code
const checkCode = () =>{
    let link  = window.location.href;
    let code = null;
    if(link.indexOf('code=') === -1){
        let url = encodeURIComponent(link)
        let authUrl = `https://open.weixin.qq.com/connect/oauth2/authorize?appid=${appId.value}&redirect_uri=${url}&response_type=code&scope=snsapi_base&state=STATE#wechat_redirect`
        window.location.href = authUrl
    }else{
        let temp = link.split("code=")[1]
        code = temp.split("&")[0]
        // 通過(guò)code去拿openId
        // code作為換取access_token的票據(jù)栓辜,每次用戶授權(quán)帶上的code將不一樣,code只能使用一次垛孔,5分鐘未被使用自動(dòng)過(guò)期
        getUserOpenId(code)
    }
}

// 獲取openId
const getUserOpenId = async (code) => {
    const res = await getOpenId({code})
    if(res.code === 'SUCCESS'){
        formData.openId = res.data.openId
        localStorage.setItem('openId', formData.openId)
    }else{
        showToast('openId獲取失敗')
    }
}

// 關(guān)閉回調(diào)
const beforeClose = async () => {
    const res = await getOrderList({tradeId: formData.tradeId})
    if(res.code === 'SUCCESS') {
        if(res.data.status == 'success') {
            router.push({path: '/result', query: {redirectUrl: redirectUrl.value}})
        }else {
            showToast('未支付成功,請(qǐng)重新支付藕甩!')
        }
    }
    return true
}

// 是否支付彈窗
const isPay = () => {
    showConfirmDialog({
    title: '',
    confirmButtonText: '支付完成',
    cancelButtonText: '未支付',
    confirmButtonColor: 'rgb(206, 51, 58)',
    beforeClose,
    message:
        '請(qǐng)確認(rèn)支付是否完成!',
    })
}

// 支付寶支付
const pullAliPay = (dom) => {
    window.open(dom, '_self')
    // document.querySelector('body').innerHTML = dom
    // nextTick(()=>{
    //     window.document.forms[0].submit()
    //     isPay()
    // })
}

//拉取微信支付
const pullWchatPay = ({nonceStr, prepayIdPackage, paySign, timeStamp, signType}) => {
    let onBridgeReady = () => {
        if(WeixinJSBridge && appId.value){
            WeixinJSBridge.invoke('getBrandWCPayRequest',{
                    appId: appId.value,
                    nonceStr,
                    package: prepayIdPackage,
                    paySign,
                    signType,
                    timeStamp
                },
                function (res) { 
                    if(res.err_msg==='get_brand_wcpay_request:ok'){
                        router.push({path: '/result', query: {redirectUrl: redirectUrl.value}})
                    }else{ 
                        showToast('未支付成功!')
                    }
                },
            )
        }
    }
    // 檢測(cè)支付環(huán)境中的 WeixinJSBridge
    if (typeof WeixinJSBridge == "undefined"){
        if (document.addEventListener) {
            //android機(jī)型,調(diào)取支付支付環(huán)境偶爾有延遲狭莱,這里增加延時(shí)器處理
            var timev=setTimeout(()=>{
                clearTimeout(timev)
                document.addEventListener('WeixinJSBridgeReady', onBridgeReady(),     
                false);
            },1000)
        } else if (document.attachEvent) {
            document.attachEvent('WeixinJSBridgeReady', onBridgeReady());
            document.attachEvent('onWeixinJSBridgeReady', onBridgeReady());
        }
    } else {
        onBridgeReady();
    }
}

</script>

<style lang="scss" scoped>
.pay-page {
    width: 100%;
    .pay-moneny {
        display: flex;
        justify-content: center;
        margin-top: 40px;
        margin-bottom: 8px;
        span {
            font-size: 24px;
            font-weight: 600;
            box-sizing: border-box;
            vertical-align: bottom;
            display: flex;
            align-items: flex-end;
            line-height: 1;
        }
        .pay-account {
            font-size: 40px;
            font-weight: 600;
        }
    }
    .pay-title {
        display: flex;
        justify-content: center;
        font-size: 12px;
        color: rgba(91, 103, 124, 1);
    }
    .pay-type {
        background: rgba(255, 255, 255, 1);
        border-radius: 2px;
        padding: 12px;
        margin-top: 24px;
        p {
            font-size: 14px;
            color: rgba(5, 5, 6, 1);
            line-height: 1;
            margin-bottom: 12px;
        }
        .pay-list {
            display: flex;
            justify-content: space-between;
            align-items: center;
            padding: 12px 0;
            .left {
                display: flex;
                align-items: center;
                img {
                    width: 24px;
                    height: 24px;
                    margin-right: 8px;
                }
                span {
                    font-size: 14px;
                    color: rgba(24, 24, 24, 1);
                }
            }
        }
    }
    .btn-bottom {
        width: calc(100% - 48px);
        padding: 0 24px;
        position: fixed;
        bottom: 70px;
        left: 0;
    }
}
@media screen and (min-width: 1164px) {
    .btn-bottom {
        position: static !important;
        margin-top: 20px;
        padding: 0 !important;
        width: 100% !important;
    }
}
</style>

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末僵娃,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子腋妙,更是在濱河造成了極大的恐慌默怨,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,248評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件骤素,死亡現(xiàn)場(chǎng)離奇詭異匙睹,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)济竹,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門痕檬,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人送浊,你說(shuō)我怎么就攤上這事梦谜。” “怎么了袭景?”我有些...
    開封第一講書人閱讀 153,443評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵唁桩,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我耸棒,道長(zhǎng)荒澡,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,475評(píng)論 1 279
  • 正文 為了忘掉前任榆纽,我火速辦了婚禮仰猖,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘奈籽。我一直安慰自己饥侵,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,458評(píng)論 5 374
  • 文/花漫 我一把揭開白布衣屏。 她就那樣靜靜地躺著躏升,像睡著了一般。 火紅的嫁衣襯著肌膚如雪狼忱。 梳的紋絲不亂的頭發(fā)上膨疏,一...
    開封第一講書人閱讀 49,185評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音钻弄,去河邊找鬼佃却。 笑死,一個(gè)胖子當(dāng)著我的面吹牛窘俺,可吹牛的內(nèi)容都是我干的饲帅。 我是一名探鬼主播,決...
    沈念sama閱讀 38,451評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼灶泵!你這毒婦竟也來(lái)了育八?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,112評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤赦邻,失蹤者是張志新(化名)和其女友劉穎髓棋,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體惶洲,經(jīng)...
    沈念sama閱讀 43,609評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡按声,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,083評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了湃鹊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片儒喊。...
    茶點(diǎn)故事閱讀 38,163評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖币呵,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情侨颈,我是刑警寧澤余赢,帶...
    沈念sama閱讀 33,803評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站哈垢,受9級(jí)特大地震影響妻柒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜耘分,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,357評(píng)論 3 307
  • 文/蒙蒙 一举塔、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧求泰,春花似錦央渣、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至卜朗,卻和暖如春拔第,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背场钉。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工蚊俺, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人逛万。 一個(gè)月前我還...
    沈念sama閱讀 45,636評(píng)論 2 355
  • 正文 我出身青樓泳猬,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子暂殖,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,925評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容