问题探索:某些场景下H5页面能够触发,而uniapp微信小程序不能触发

问题重现

在使用uniapp开发微信小程序时,甲方要求将toast改为弹窗展示,并增加按钮用于跳转到电话界面,跳转到电话界面的逻辑很好做,使用uniapp自带的如下命令或者找AI写能够轻松搞定:

javascript 复制代码
const contact = () => {
  uni.makePhoneCall({
    phoneNumber: '123456789',
    fail: (err) => {
      console.error('拨打电话失败:', err);
    },
  });
}

但是toast改为弹窗展示的这一部分遇到了一些问题,最初我使用传统的vue3代码逻辑来写,先用props+emits的方法来解决,部分伪代码如下:

子组件

html 复制代码
<template>
  <su-popup :show="props?.show"@close="props.show = false">
    ...
</template>

<script setup>
import { reactive, computed, defineProps, defineEmits } from 'vue';
...
// 负责展示
const props = defineProps({
  show: false,
});

const emits = defineEmits();

const closeModal = () => {
  emits('close');
};
...
</script>

父组件

html 复制代码
<template>
    <s-lvl-modal :show="state?.show" @close="state?.show=false" />
</template>
<script setup>
const state= reactive({ show: false, })
// 触发弹窗的逻辑
state?.show=true
...
</script>

但是这种方式下H5模式下可以跳出弹窗,而微信小程序无法弹窗

问题探索

为了解决这个问题,我先求助于deepseek,它给出的代码是这样的

父组件

html 复制代码
<!-- 父组件修改后 -->
<template>
    <s-lvl-modal :show="state.show" @close="closeModal" />
    <!-- 其他内容不变 -->
</template>

<script setup>
import { reactive } from 'vue';
import sLvlModal from './s-lvl-modal.vue'; // 确保路径正确

const state = reactive({
    show: false
});

const closeModal = () => {
    state.show = false;
}
// 其他逻辑不变
</script>

子组件

html 复制代码
<!-- 子组件修改后 -->
<script setup>
import { ref, watch } from 'vue';

// props定义
const props = defineProps({
  show: Boolean
});

// 创建响应式变量
const showProp = ref(props.show);

// 监听props变化
watch(() => props.show, (val) => {
  showProp.value = val;
});

// emits定义不变
</script>

<template>
  <su-popup 
    :show="showProp"
    type="center" 
    round="10"
    @close="closeModal"
    :isMaskClick="false"
    maskBackgroundColor="rgba(0, 0, 0, 0.7)"
  >
    <!-- 内容不变 -->
  </su-popup>
</template>

结果报错如下:

ini 复制代码
AppOnError: TypeError: Cannot read property 'openModal' of null
    at toCommission (<anonymous>:47:26)
    at callWithErrorHandling (vendor.js?t=wechat&s=1753756936479&v=edf8366d112d96098f5089ed6f3e1306:3616)
    at callWithAsyncErrorHandling (vendor.js?t=wechat&s=1753756936479&v=edf8366d112d96098f5089ed6f3e1306:3623)
    at t.invoke2 (vendor.js?t=wechat&s=1753756936479&v=edf8366d112d96098f5089ed6f3e1306:7035)
    at f (WASubContext.js?t=wechat&s=1753756936479&v=3.8.9:1)
    at WASubContext.js?t=wechat&s=1753756936479&v=3.8.9:1
    at t.<anonymous> (WASubContext.js?t=wechat&s=1753756936479&v=3.8.9:1)
    at Function.<anonymous> (WASubContext.js?t=wechat&s=1753756936479&v=3.8.9:1)

解决方法

最后迫不得已,使用expose的方法进行解决,去掉props和emits的传递,子组件伪代码如下:

html 复制代码
<template>
  <su-popup :show="state.show" @close="closeModal()">
  ...
</template>
<script setup>
import { reactive,defineExpose } from 'vue';
...
const state = reactive({
  show: false,
  ...
});

const openModal = () => {
  state.show = true;
}

const closeModal = () => {
  state.show = false;
};
...
defineExpose({
  openModal,
  closeModal
});
</script>

父组件的伪代码如下:

html 复制代码
<template>
    <s-lvl-modal ref="lvlModalRef" />
</template>
<script setup>
import { ref } from 'vue';
import lvlModal from './s-lvl-modal.vue';
const lvlModalRef = ref(null);
// 触发弹窗的逻辑
lvlModalRef.value?.openModal();
...
</script>

之后再用微信开发者工具进行测试,解决

相关推荐
少年姜太公5 小时前
什么?还不知道git cherry pick?
前端·javascript·git
白兰地空瓶7 小时前
🏒 前端 AI 应用实战:用 Vue3 + Coze,把宠物一键变成冰球运动员!
前端·vue.js·coze
Liu.7748 小时前
vue3使用vue3-print-nb打印
前端·javascript·vue.js
松涛和鸣9 小时前
Linux Makefile : From Basic Syntax to Multi-File Project Compilation
linux·运维·服务器·前端·windows·哈希算法
dly_blog9 小时前
Vue 逻辑复用的多种方案对比!
前端·javascript·vue.js
万少9 小时前
HarmonyOS6 接入分享,原来也是三分钟的事情
前端·harmonyos
烛阴9 小时前
C# 正则表达式:量词与锚点——从“.*”到精确匹配
前端·正则表达式·c#
wyzqhhhh10 小时前
京东啊啊啊啊啊
开发语言·前端·javascript
JIngJaneIL10 小时前
基于java+ vue助农电商系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
想学后端的前端工程师10 小时前
【Java集合框架深度解析:从入门到精通-后端技术栈】
前端·javascript·vue.js