问题探索:某些场景下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>

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

相关推荐
若梦plus2 分钟前
Nuxt.js基础与进阶
前端·vue.js
樱花开了几轉8 分钟前
React中为甚么强调props的不可变性
前端·javascript·react.js
风清云淡_A8 分钟前
【REACT18.x】CRA+TS+ANTD5.X实现useImperativeHandle让父组件修改子组件的数据
前端·react.js
小飞大王6669 分钟前
React与Rudex的合奏
前端·react.js·前端框架
若梦plus37 分钟前
React之react-dom中的dom-server与dom-client
前端·react.js
若梦plus39 分钟前
react-router-dom中的几种路由详解
前端·react.js
若梦plus39 分钟前
Vue服务端渲染
前端·vue.js
Mr...Gan1 小时前
VUE3(四)、组件通信
前端·javascript·vue.js
OEC小胖胖1 小时前
渲染篇(二):解密Diff算法:如何用“最少的操作”更新UI
前端·算法·ui·状态模式·web
万少1 小时前
AI编程神器!Trae+Claude4.0 简单配置 让HarmonyOS开发效率飙升 - 坚果派
前端·aigc·harmonyos