小心的使用消息传送--传送给多个View类时需要引起注意的一个问题

本文讲述自定义消息发送与接收时遇到的问题。作者原本用GetActiveView()获取视图指针发送消息,在向两个不同View视图类分别发送自定义消息时出错。原因是该函数获取的是当前视图指针,导致只有有焦点的视图能接收消息。正确做法是通过切分对象的GetPane函数获取确定的视图指针。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我的前一篇文章里说的是关于《两种自定义消息的发送与接收的方法实现》
可当我在一次使用消息传送的时候也遇到一个让人迷惑的问题,
好在最后解决了。

-------------------------------------------------------------
原来我对于一个消息的传送也做过,使用的是
    CMainFrame* pMF=(CMainFrame*)AfxGetApp()->m_pMainWnd; // 通过获取当前框架指针
    CView * active = pMF->GetActiveView();// 获取当前视类指针
    if(active != NULL)  
    active->PostMessage(WM_MY_MESSAGE,0,0);   // 使用PostMessage发送消息
来发送我定义的消息的。
很顺利的,我就在要接收的View类里接收到了该消息。

可是当一个类中定义了2个自定义的消息,并要分别发送给两个不同的View视图类的时候,
如果还般照上面的代码,就要出一些问题了。

为什么说是照搬就会出现问题呢?

请注意一个地方
上面的代码中,是如何得到View类的指针的。
对,就是使用的函数GetActiveView(),这个函数的作用是获得"当前"视图指针

我之所以出问题,使得其中的一个View类不能接收到消息也就是这个原因。

------------------------------------------------------------
下面详细说明我出错的过程和解决过程。

来源:一个SDI被我用静态切分的方式,且分成左右大小相等的两个部分,并使这两个部分分别指向自行创建的两个新类:CMyLeftView和CMyRightView,它们的基类都是CEditView。所以说这两个准备接收消息的视图类除了名字,其他并没有什么不太一样的地方。

一个对话框(CMyFindDialog类),其上有2个RADIO BUTTON控件,和一个确定按钮,
这两个RADIO BUTTON控件在选择的时候,使其只能进行单选,他们也就分别对应于前面所说的左右两个部分,选择后,点击确定按钮,便表示向 哪个视图发送消息。

做好了发送和接收部分,运行发现只能有一个消息被接收到,
void CMyFindDialog::OnButtonFind()
{   MainFrame* pMF=(CMainFrame*)AfxGetApp()->m_pMainWnd;
    CView* active = pMF->GetActiveView();//获取当前视类指针
    // 错就错在上一句得到的active是个 “不确定“ 的对象
    if(m_selLeft == TRUE)
       {  if(active != NULL) 
              active->PostMessage(WM_MY_MESSAGE_LEFT,0,0); 
       }
     else if(m_selRight == TRUE)
      {  if(active != NULL) 
            active->PostMessage(WM_MY_MESSAGE_RIGHT,0,0);
      }
     else
           MessageBox("MUST CHOOSE ONE!"); 
}

通过GetActiveView()得到的当前视类指针,那么
如果你打开对话框之前,焦点(光标)在左边,那么就是左边能接收到消息,右边怎么也接收不到。
如果你打开对话框之前,焦点(光标)在右边,那么就是右边能接收到消息,左边怎么也接收不到。
程序运行后,焦点(光标)默认的是在左边,这也就导致我总一为是右边的接收消息的部分有问题。
其实,如果在程序运行后,先把焦点(光标)改换到右边,再打开对话框测试发送与接收,就成了
左边收不到消息,右边可以收到消息。(这还是在论坛中,别人的话提醒了,才找到了错误的所在。要不然还在考虑是不是接收部分出了问题。)

对于这样一个向2个View类发送消息问题,就是要注意你所执行发送(PostMessage)的对象是谁!
正确的代码如下:
void CMyFindDialog::OnButtonFind()
{   CMainFrame* pMF=(CMainFrame*)AfxGetApp()->m_pMainWnd;
     CMyLeftView* pLV=(CMyLeftView*)pMF->m_wndSplitter.GetPane(0,0);
     CMyRightView* pRV=(CMyRightView*)pMF->m_wndSplitter.GetPane(0,1);
   // 对于我这个题,是通过切分对象m_wndSplitter的GetPane函数得到它实际并确定的
   // 视图类指针的。
   // 如果你也是多个视图类,但不是切分,而是别的,只要注意
   // 获得的是个你确定视图对象指针

    if(m_selLeft == TRUE)
     {  if(pLV != NULL) 
             pLV->PostMessage(WM_MY_MESSAGE_LEFT,0,0); 
     }
    else if(m_selRight == TRUE)
       {  if(pRV != NULL) 
               pRV->PostMessage(WM_MY_MESSAGE_RIGHT,0,0);
       }
    else
           MessageBox("MUST CHOOSE ONE!"); 
}

总结:
当你需要发送消息给多个视图对象的时候,要考虑你是向什么样的视图对象发送消息。

如果你就是向多个视图对象中当前的视图对象发送消息,那么我原本“出错”的那个
方法对于你来说就是正确的。

但如果你不是向当前视图对象发消息,而是有个明确的目标,那么千万要注意,通过
某种方法一定要得到确定的视图指针,
那么用  (View*)->PostMessage( , , )发送消息的时候,
接收方才能接收的到你的消息。

 

 



 

<think>好的,我需要解决用户的问题,即在uni-app的安卓端使用renderjs引入天地图,并实现页面与renderjs之间的通信。首先,我得回忆一下uni-app中renderjs的使用方法。根据引用[^1],renderjs运行在视图层,可以直接操作DOM,适合处理高频率的视图操作,比如canvas动画。而用户提到的天地图可能需要加载JavaScript库,并在WebView中渲染地图,所以使用renderjs是合适的。 接下来,我需要考虑如何在renderjs中引入天地图的JS API。通常,天地图需要加载其提供的脚本文件,然后在页面中创建容器元素来显示地图。用户需要的是安卓端的示例,但renderjs本身是跨平台的,不过要注意uni-app不同端的差异。这里可能需要动态创建script标签来加载天地图的JS库,或者在index.html中预先引入。 然后是页面与renderjs之间的通信。根据uni-app的文档,可以使用this.$ownerInstance.callMethod来调用页面的方法,而页面则通过事件监听来接收消息。同,页面也可以向renderjs发送消息,通过修改prop的值触发renderjs中的观察器。因此,我需要设计一个双向通信的机制,比如页面发送初始化参数或地图操作指令,renderjs返回地图事件或状态。 另外,需要注意uni-app中使用TypeScript的配置,引用[^2]提到可以参考Vue.js的TypeScript支持。可能需要配置tsconfig.json和必要的型声明,尤其是处理window对象和第三方库的型。 现在,我需要构建一个示例的结构,包括页面部分和renderjs部分。页面部分需要有view容器,设置renderjs脚本,并处理消息的接收和发送。renderjs部分需要加载天地图库,初始化地图,并处理来自页面的事件。同,要处理地图的点击等事件,将信息传回页面。 可能遇到的问题包括天地图JS库的加载方式,跨域问题,以及renderjs中操作DOM的权限。根据引用[^1],在App端使用renderjs可以直接操作视图层的DOM,所以应该可行。但需要确保天地图的资源正确加载,可能需要配置webview的相关设置。 最后,整合代码示例,确保各部分通信正常,包括初始化地图、发送坐标、接收事件等步骤。同,提醒用户注意事项,比如权限配置、库的加载方式,以及真机调试的重要性。</think>以下是在uni-app中通过renderjs实现安卓端天地图集成及页面通信的完整示例代码: ```html <!-- 页面组件 map.vue --> <template> <view class="content"> <!-- 地图容器 --> <view id="tmap-container" @click="handleContainerClick" :prop="mapConfig" :change:prop="renderjs.initMap" class="map-view"></view> <!-- 坐标展示 --> <text class="coord-text">当前坐标:{{ currentCoord }}</text> </view> </template> <script> export default { data() { return { mapConfig: { key: '你的天地图密钥', center: [116.397, 39.908], zoom: 11 }, currentCoord: '' } }, methods: { // 接收来自renderjs的消息 handleMapEvent(event) { if (event.type === 'click') { this.currentCoord = `${event.lnglat.lng.toFixed(4)}, ${event.lnglat.llat.toFixed(4)}` } }, // 向renderjs发送指令 handleContainerClick() { this.$nextTick(() => { const container = document.getElementById('tmap-container'); container.__renderjs__.callMethod('showCurrentCenter'); }); } } } </script> <script module="renderjs" lang="renderjs"> let map = null; export default { mounted() { this.loadScript('http://api.tianditu.gov.cn/api?v=4.0&tk=' + this.mapConfig.key); }, methods: { // 动态加载天地图JS loadScript(src) { const script = document.createElement('script'); script.src = src; script.onload = () => this.initTMap(); document.head.appendChild(script); }, // 初始化地图 initTMap() { const container = document.getElementById('tmap-container'); container.style.height = window.innerHeight + 'px'; map = new T.Map('tmap-container', { projection: 'EPSG:4326' }); map.centerAndZoom(new T.LngLat(...this.mapConfig.center), this.mapConfig.zoom); // 添加地图事件监听 map.addEventListener('click', e => { this.$ownerInstance.callMethod('handleMapEvent', { type: 'click', lnglat: e.lnglat }); }); }, // 响应页面指令 showCurrentCenter() { const center = map.getCenter(); this.$ownerInstance.callMethod('handleMapEvent', { type: 'center', center: [center.lng, center.lat] }); } } } </script> <style> .map-view { width: 100%; height: 100vh; } .coord-text { position: fixed; bottom: 20px; background: rgba(255,255,255,0.9); padding: 10px; } </style> ``` 关键实现要点说明: 1. **双向通信机制**: - 页面→renderjs:通过`:prop`属性传递配置,使用`__renderjs__.callMethod`直接调用方法 - renderjs→页面:通过`this.$ownerInstance.callMethod`调用页面方法 2. **天地图集成要点**: - 需要先申请天地图开发者密钥[^1] - 动态加载天地图API保证依赖顺序 - 设置容器尺寸需在renderjs中操作 3. **性能优化**: - 使用renderjs的视图层渲染能力保障地图流畅性 - 避免频繁的跨线程通信 - 使用CSS固定定位处理坐标显示 §§ 相关问题 §§ 1. 如何处理renderjs与页面间的数据同步问题? 2. 天地图在移动端有哪些优化建议? 3. 如何实现多个地图控件的联动操作? 注意事项: 1. 需在manifest.json中配置网络白名单: ```json "app-plus": { "modules": { "renderjs": {} }, "whitelist": [ "request", "webview" ] } ``` 2. 真机调试需要配置https支持,建议使用正式域名环境测试 3. 复杂交互建议使用WebSocket建立长连接通信通道,处理大量实数据更新[^2] [^1]: 天地图开放平台密钥申请地址:https://console.tianditu.gov.cn/ : 关于uni-app的长连接使用请参考uni-app官方socket文档
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值