本文转载自微软嵌入式中文社区www.msembed.com
和以往一样我开始充满激情地研究WinCE的新版本:Windows Embedded Compact 7,尤其是操作系统的新功能。我用我们的参考平台Alioth(一个基于Marvel PXA 300的参考设计)BSP来研究WEC7的功能。我用到了WEC7(ie)的重要功能Silverlight Shell。Wow!!!微软为WEC7设计了新的图形界面。我为我的平台生成了新的OSDesign。我加入了"sample Home Screen Silverlight for Windows Embedded (SYSGEN_XRSHELL)"组件以及其它一些必要的组件,成功地编译出OS的镜像。
将镜像下载到目标板后,我有点失望,因为我没看到XRShell,取而代之的是一个灰色的空屏幕,如下图所示。
而与此同时,我的经理和其他一些Windows CE团队的成员却得到了成功的结果。我的脑袋又开始发热了。我开始收集他们的OSDesign,进行对比并找出在OSDesign的组件列表中的差异。我试着加入和移除我的OSDesign中的某些组件,最后终于找到了罪魁祸首。让XRShell不能运行的组件是"ActiveSync"。我找到了解决问题的线索。
我开始看XRShell的源代码,它们在WINCE700\public\ shell\ oak\ XRshell\ src\ CXRShell.cpp中。同时我也在观察ActiveSync在启动过程中的活动。起初我怀疑可能是ActiveSync中用到的SH_Shell API没有被实现。在检查CXRShell.cpp过程中我发现他们的确没有实现这个API,但是他们用一个"MissingAPI()"给所有的SH_Shell API设置一个占位函数,并在CXRShell::Execute()方法中对这个占位API进行初始化。
CXRShell::Execute()方法中shell生成桌面、场景等,包括对SH_Shell API进行初始化。用这个API的占位API,XRShell可以使其它需要用到SH_Shell API的应用程序得以运行。但我们的情况不同。这里XRShell被ActiveSync阻塞了。所以SH_Shell API的占位不是问题所在。
下一个被怀疑的活动是ActiveSync会在任务栏中显示提示图标,但不幸的是当前的XRShell中没有实现任务栏。最后我终于发现原来是ActiveSync在一直找任务栏,所以阻塞了XRShell的启动。
解决方案
我生成了一个起占位作用的空任务栏,并与XRShell相关联。太棒了!现在XRShell可以带着ActiveSync一起启动了。
下面我将解释如何生成一个占位用任务栏。我们可以从已有的标准shell的源代码中轻松地生成一个任务栏。如果你浏览了以下的文件,你可以轻松地生成一个占位任务栏。
• \Wince700\public\shell\oak\hpc\explorer\inc\taskbar.hxx
• \Wince700\public\shell\oak\hpc\explorer \taskbar.cpp
• \Wince700\public\shell\oak\hpc\explorer \main.cpp
以下是我生成任务栏的源代码。
//以下的宏定义取自taskbar.hxx文件。
#defineHHTASKBARDATA8
#defineHHTASKBARCLASSNAMETEXT(“HHTaskBar”)
#defineHHTASKBARID0
#defineHHTASKBARSTATE4
//为任务栏设立的一个哑WndProc
LRESULTCALLBACKXRshellTaskBarWndProc(HWNDhwnd,UINTmsg,WPARAMwParam,LPARAMlParam)
{
return0;
}
//以下代码通过参考taskbar.cpp中的CTaskBar::Register()方法而生成。
//函数XShellTaskBar()-用于生成一个占位任务栏。
voidXShellTaskBar(HINSTANCEhInstance)
{
RETAILMSG(1,(TEXT(“XRshellTaskBar++\r\n”)));
HWNDhwndTaskBar;
WNDCLASSwc;
wc.style=/*CS_HREDRAW|CS_VREDRAW|*/CS_DBLCLKS;
wc.lpfnWndProc=(WNDPROC)XRshellTaskBarWndProc;
wc.cbClsExtra=0;
wc.cbWndExtra=HHTASKBARDATA;
wc.hInstance=hInstance;
wc.hIcon=NULL;
wc.hCursor=0;
wc.hbrBackground=(HBRUSH)GetStockObject(LTGRAY_BRUSH);
wc.lpszMenuName=NULL;
wc.lpszClassName=HHTASKBARCLASSNAME;
RegisterClass(&wc);
//为任务栏生成一个窗口。参考自taskbar.cpp中的CtaskBar::Create()。
hwndTaskBar=CreateWindowEx(0,HHTASKBARCLASSNAME,L”",WS_POPUP,0,0,
10,10,NULL,NULL,hInstance,NULL);
//在成功生成窗口后,用RegisterTaskbar()API将此窗口注册为任务栏。参考自explorer.cpp中的WINAPICreateTaskBar()方法。该函数调用了CTaskbar::Register和RegisterTaskBar()API。
if(hwndTaskBar)
{
if(RegisterTaskBar(hwndTaskBar))
{
RETAILMSG(1,(TEXT(“XRShellTaskBar:RegisterTaskBarsuccess\r\n”)));
}
else
{
RETAILMSG(1,(TEXT(“XRShellTaskBar:RegisterTaskBarfailed\r\n”)));
}
}
else
{
RETAILMSG(1,(TEXT(“XRShellTaskBar:UnabletocreateTaskbarwindow\r\n”)));
}
RETAILMSG(1,(TEXT(“XshellTaskBar–\r\n”)));
}
占位任务栏已经准备好了,现在我们要将它加入到XRShell中。这部分比较简单,你可以将上面的代码加到CXRShell.cpp的开始部分然后在CXRShell::Execute()中调用XShellTaskBar()函数,就像下面做的这样。
unsignedlongCXRShell::Execute(intnCmdShow)
{
/*已有的代码*/
…..
InitializeAPI();
XShellTaskbar(m_hInstance);//我们为XRShell准备的黑色任务栏
CHR(ShowShell());//这是个阻塞调用
…..
}