GoogleEarth二次开发(资料二)

GoogleEarth二次开发(资料二)

 

在163博客上发布了《GoogleEarth二次开发难点和技巧》一文之后,很多朋友发邮件询问怎样将googleearth的地图控件引用到自定义的窗体中去,因为某些原因一直没有给予详细解答。其实一些公司早已经实现了这种技术,上帝之眼很早就发布了这样一个ge控件,不过控件上有个很大的logo,看着很不爽。后来因为公司业务的需要,也钻研了一下,发现其实实现那样一种效果并不难,下面请听我细细道来。

 

这里用到一种黑客的手段–hook–来实现,将ge的地图显示部分强行抢夺到我们所指定的一个winform控件上去, 并将ge的主窗体隐藏起来。GE的com api为实现hook提供了方便,通过IApplicationGE.GetMainHwnd()和IApplicationGE.GetRenderHwnd()两个函数我们可以获取到GE主窗体和地图显示部分的句柄,然后利用windows api的几个函数来操作这两个句柄,就能实现我们所期待的效果了。

以VS2005为例(其他windows开发平台都可以),首先新建一个winform窗体,将GE的com组件引用进来,并在winform上添加一个panel控件(命名为pnlEarth),然后引入我们所需要的几个window API函数和常量。(这些API的具体作用和使用方法请自己查资料,这里不再啰嗦)

接下来,我们开始最关键的部分–GE地图控件截获:

 

代码
1 //---------------------------------
2  
3 [DllImport("user32.dll", CharSet = CharSet.Auto)]
4  public extern static bool SetWindowPos(int hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, uint uFlags);
5
6 [DllImport("user32.dll", CharSet = CharSet.Auto)]
7  public static extern IntPtr PostMessage(int hWnd, int msg, int wParam, int lParam);
8
9 [DllImport("user32", CharSet = CharSet.Auto)]
10  public extern static IntPtr GetParent(IntPtr hWnd);
11
12 [DllImport("user32", CharSet = CharSet.Auto)]
13  public extern static bool MoveWindow(IntPtr hWnd, int X, int Y, int nWidth, int nHeight, bool bRepaint);
14
15 [DllImport("user32", CharSet = CharSet.Auto)]
16  public extern static IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);
17
18  private static readonly IntPtr HWND_BOTTOM = new IntPtr(1);
19  private static readonly IntPtr HWND_NOTOPMOST = new IntPtr(-2);
20  private static readonly IntPtr HWND_TOP = new IntPtr(0);
21 private static readonly IntPtr HWND_TOPMOST = new IntPtr(-1);
22 private static readonly UInt32 SWP_NOSIZE = 1;
23 private static readonly UInt32 SWP_HIDEWINDOW = 128;
24 private static readonly Int32 WM_QUIT = 0x0012;
25 private static readonly Int32 WM_HIDE = 0x0;
26 //-------------------------------------------

 

 

GE地图控件截获
1 //-----------------------------------
2 private IntPtr _GEHrender;
3 private IntPtr _GEParentHrender;
4 private IntPtr _GEMainHandler;
5 private IApplicationGE _googleEarth;
6 private System.Windows.Forms.Control _parentControl;
7
8 /// <summary>
9 /// 将GE的地图控件挂载到指定的控件中去
10 /// </summary>
11 /// <param name="parentControl">c#控件</param>
12 /// <param name="geApplication">ge应用程序</param>
13 public void SetGEHandlerToControl(System.Windows.Forms.Control parentControl, IApplicationGE geApplication)
14 {
15 this._parentControl = parentControl;
16 this._googleEarth = geApplication;
17
18 //获取GE的主窗体句柄
19 this._GEMainHandler = (IntPtr)this._googleEarth.GetMainHwnd();
20 //将GE主窗体的高宽设置为0,隐藏掉GE主窗体
21 SetWindowPos((int)this._GEMainHandler, HWND_BOTTOM, 0, 0, 0, 0, SWP_NOSIZE + SWP_HIDEWINDOW);
22
23 //获取GE地图控件句柄
24 this._GEHrender = (IntPtr)_googleEarth.GetRenderHwnd();
25 //获取GE地图控件的父窗体句柄
26 this._GEParentHrender = GetParent(this._GEHrender);
27 //将GE地图控件的父窗体设置为不可见
28 //(考虑到GE地图控件可能被其他程序截获,加上这一句以应万全)
29 PostMessage((int)this._GEParentHrender, WM_HIDE, 0, 0);
30
31 //设置GE地图控件的父窗体句柄为winform上的控件
32 SetParent(this._GEHrender, parentControl.Handle);
33 }
34 //----------------------------------------------

 

 

 

在winform的load事件中加入SetGEHandlerToControl函数,编译并打开程序 ,这时候你发现你已经成功了。

但是,还不能高兴的太早, 还有两个问题需要解决。你会发现从GE截获的地图控件并不会随着你自定义窗体的大小改变而改变,另外,你自定义窗体关闭之后,GE的进程还没有被杀掉,进而引起下次启动GE的时候地图控件会消失掉。不用急,是问题总会有解决的办法。再定义下面两个函数

 

代码
//----------------------------------
/// <summary>
/// 使GE控件的大小和父窗体的大小保持一致
/// </summary>
public void ResizeGEControl()
{
if (this._parentControl != null)
{
//设置GE地图控件的大小等于父窗体大小
MoveWindow(this._GEHrender, 0, 0, this._parentControl.Width, this._parentControl.Height, true);
}
}

/// <summary>
/// 释放GE句柄
/// </summary>
public void RealseGEHandler()
{
try
{
if (this._parentControl != null)
{
//将GE地图控件的句柄还原到GE主窗体上去
SetParent(this._GEHrender, this._GEParentHrender);
//关闭GE主程序
PostMessage(this._googleEarth.GetMainHwnd(), WM_QUIT, 0, 0);
}
}
finally
{
//为防本程序的进程不能成功退出而导致GE出现问题,强制杀掉本程序的进程
System.Diagnostics.Process geProcess = System.Diagnostics.Process.GetCurrentProcess();
geProcess.Kill();
}
}
//----------------------------------

 

 

ResizeGEControl函数放在pnlEarth控件的SizeChanged事件中,这样每次panel大小改变,GE的地图控件的大小也会随之改变; RealseGEHandler函数放在winform的FormClosing事件中,以释放GE的窗体句柄。

经过上面几个步骤,就实现了GE地图控件的截获。接下来要做一些基于GE的开发,我们就可以摆脱GE那一成不变的摸样了。

posted @ 2010-06-18 08:34  晓染霜林醉  阅读(4643)  评论(13编辑  收藏  举报