让 Vant 弹出层适配 Uniapp Webview 返回键

问题背景

在 UniApp Webview 中使用 Vant 组件库时,返回键的行为往往不符合用户预期:

  • 当 Popup、Dialog、ActionSheet 等弹出层打开时,用户按下返回键会直接返回上一页,而不是关闭弹出层
  • 多层弹出层叠加时,无法按层级顺序依次关闭
  • Vant 内置的 closeOnPopstate 仅会在页面回退时自动关闭弹窗,而不会阻止页面回退

这导致用户体验与原生应用存在明显差距。

解决方案

@vue-spark/back-handler 提供了基于栈的返回键处理机制,可以与 Vant 组件无缝集成,让弹出层正确响应 UniApp 的返回键事件。

核心思路:将每个需要响应返回键的弹出层注册到全局栈中,按后进先出的顺序处理返回事件。

使用方式

1. 安装依赖

bash 复制代码
npm install @vue-spark/back-handler

2. 初始化插件(UniApp 适配)

ts 复制代码
// main.ts
import { BackHandler } from '@vue-spark/back-handler'
import { createApp } from 'vue'
import App from './App.vue'
import router from './router'

createApp(App)
  .use((app) => {
    const routerHistory = router.options.history

    let initialPosition = 0
    const hasRouteHistory = () => {
      // 当 vue-router 内部记录的位置不是初始位置时认为还存在历史记录
      return routerHistory.state.position !== initialPosition
    }

    router.isReady().then(() => {
      // 记录初始位置
      initialPosition = routerHistory.state.position as number

      router.afterEach(() => {
        // 每次页面变更后通知 uniapp 是否需要阻止返回键默认行为
        uni.postMessage({
          type: 'preventBackPress',
          data: hasRouteHistory(),
        })
      })
    })

    // 注册插件
    BackHandler.install(app, {
      // 每次增加栈记录时通知 uniapp 阻止返回键默认行为
      onPush() {
        uni.postMessage({
          type: 'preventBackPress',
          data: true,
        })
      },

      // 每次移除栈记录时通知 uniapp 是否阻止返回键默认行为
      onRemove() {
        uni.postMessage({
          type: 'preventBackPress',
          data: BackHandler.stack.length > 0 || hasRouteHistory(),
        })
      },

      // 栈为空时尝试页面回退
      fallback() {
        hasRouteHistory() && router.back()
      },

      // 这里绑定 uniapp webview 触发的 backbutton 事件
      bind(handler) {
        window.addEventListener('uni:backbutton', handler)
      },
    })
  })

  // 在这里注册 router
  .use(router)

  // 挂载应用
  .mount('#app')

通过扩展 Popup 的 setup 函数,让所有基于 Popup 的组件(ActionSheet、ShareSheet、Picker 等)都支持返回键关闭:

ts 复制代码
import { useBackHandler } from '@vue-spark/back-handler'
import { Dialog, Popup } from 'vant'
import { callInterceptor } from 'vant/es/utils'
import { getCurrentInstance, watch } from 'vue'

const { setup } = Popup

// 变更 closeOnPopstate 默认值为 true
Popup.props.closeOnPopstate = {
  type: Boolean,
  default: true,
}

Popup.setup = (props, ctx) => {
  const { emit } = ctx
  const vm = getCurrentInstance()!

  // Dialog 组件基于 Popup,这里需要排除,否则会重复注册
  if (vm.parent?.type !== Dialog) {
    const close = () => {
      return new Promise<void>((resolve, reject) => {
        if (!props.show) {
          return resolve()
        }

        callInterceptor(props.beforeClose, {
          done() {
            emit('close')
            emit('update:show', false)
            resolve()
          },
          canceled() {
            reject(new Error('canceled'))
          },
        })
      })
    }

    const { push, remove } = useBackHandler(
      () => props.show,
      // closeOnPopstate 用于控制是否响应返回键
      () => !!props.closeOnPopstate && close(),
    )

    watch(
      () => props.show,
      (value) => (value ? push() : remove()),
      { immediate: true, flush: 'sync' },
    )
  }

  return setup!(props, ctx)
}

4. 适配 Vant Dialog

Dialog 需要单独适配,因为它基于 Popup 但有独立的关闭逻辑:

ts 复制代码
import { useBackHandler } from '@vue-spark/back-handler'
import { Dialog, showLoadingToast } from 'vant'
import { callInterceptor } from 'vant/es/utils'
import { watch } from 'vue'

// Dialog 的 closeOnPopstate 默认为 true,可以不修改默认值
const { setup } = Dialog

Dialog.setup = (props, ctx) => {
  const { emit } = ctx
  const updateShow = (value: boolean) => emit('update:show', value)

  const close = (action: 'cancel') => {
    updateShow(false)
    props.callback?.(action)
  }

  const getActionHandler = (action: 'cancel') => () => {
    return new Promise<void>((resolve, reject) => {
      if (!props.show) {
        return resolve()
      }

      emit(action)

      if (props.beforeClose) {
        const toast = showLoadingToast({})
        callInterceptor(props.beforeClose, {
          args: [action],
          done() {
            close(action)
            toast.close()
            resolve()
          },
          canceled() {
            toast.close()
            reject(new Error('canceled'))
          },
        })
      } else {
        close(action)
        resolve()
      }
    })
  }

  const { push, remove } = useBackHandler(
    () => props.show,
    // closeOnPopstate 用于控制是否响应返回键
    () => !!props.closeOnPopstate && getActionHandler('cancel')(),
  )

  watch(
    () => props.show,
    (value) => (value ? push() : remove()),
    { immediate: true, flush: 'sync' },
  )

  return setup!(props, ctx)
}

效果

完成上述配置后:

  • Popup、ActionSheet、ShareSheet、Picker、Dialog 等弹出层在打开时,按返回键会关闭弹出层而不是退出页面
  • 多层弹出层会按打开顺序的逆序依次关闭
  • 支持 beforeClose 拦截器进行异步确认

相关链接

相关推荐
格子软件19 小时前
2026年GEO优化系统源码级状态机与多模型调度拆解
java·前端·vue.js·人工智能·vue·geo
HUMHSX20 小时前
Vue 项目启动全流程解析:从入口文件到全局指令注册与页面渲染
前端·javascript·vue.js
有颜有货20 小时前
PMC生产排产的4种算法,一次讲清
java·服务器·前端
小虎牙00720 小时前
Android kotlin图片库Coil源码详解
android·前端
随风一样自由20 小时前
【前端领域】前端开发核心应用场景与落地实践
前端·前端框架
an3174221 小时前
弹窗数据流设计的两种高阶架构实践
前端·vue.js·架构
谢尔登21 小时前
【React】 状态管理方案
前端·react.js·前端框架
用户21366100357221 小时前
Vue商品详情与放大镜组件
前端·javascript
半个落月21 小时前
从Tapas小Demo理清localStorage、事件与this
前端·javascript
李明卫杭州21 小时前
Vue2 中 v-model 处理不同数据结构的技巧
前端·javascript·vue.js