【转】vs2010中添加splashScreen

作者信息:罗树鹏  http://www.cnblogs.com/luoshupeng

由于笔者在实践过程中走了一些弯路,所以把这些情况记录下来,希望为后来者提供一些经验。

在VC6.0时代,可以通过组件为工程加入SplashScreen,具体方法是通过IDE中的菜单Project->Add to Project->Componentsand Controls,就可以从Visual C++ Components中选择Splash Screen这个组件插入工程。

但进入到VC. NET时代,这种方法就不行了,需要程序作者自己加入SplashScreen类,自己处理一些系统消息。下面笔者就把实践过程记录下来,并指出需要注意的地方。

一、            新建一个SplashScreen类,并声明成员和方法

新建基类为CWnd的CSplashWnd类(当然类名可以自由书写),并声明如下成员:

CBitmap m_bitmap;                                              //加载SplashScreen图片用

static CSplashWnd*c_pSplashWnd;        //CSplashWnd类的句柄,可以用来判定CSplashWnd是否已经创建或消毁

static BOOLc_bShowSplashWnd;             //标识是否显示显示用来,静态成员需要在类外进行初始化

类成员一般声明为保护类型(protected)。接下来声明几个静态方法来处理这些成员:

static BOOLc_bShowSplashWnd;             //标识是否显示显示用来,静态成员需要在类外进行初始化

static void ShowSplashScreen(CWnd* pParentWnd = NULL); //用来显示SplashScreen窗口

static BOOLPreTranslateAppMessage(MSG* pMsg);                       //用来处理一些消息  注意这里不是继承于CWnd类的PreTranslateMessage方法

这些方法一定要声明成公开类型的(public),因为要在外部调用这些方法。接下来再声明两个自定义方法:

BOOL Create(CWnd*pParentWnd = NULL);           //创建窗口

void HideSplashScreen(void);                              //隐藏窗口

我把这些方法声明为保护类型(protected)。接下来再声明一些处理系统消息的函数:

virtual void PostNcDestroy();

afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);

afx_msg void OnTimer(UINT_PTR nIDEvent);

afx_msg void OnPaint();

至此,类已经设计完毕。完整的头文件如下:

#pragma once

#include "afxwin.h"

//CSplashWnd

class CSplashWnd : public CWnd

{

DECLARE_DYNAMIC(CSplashWnd)

protected:

CSplashWnd();

public:

virtual~CSplashWnd();

protected:

CBitmap m_bitmap;                                              //加载SplashScreen图片用

staticCSplashWnd* c_pSplashWnd;        //CSplashWnd类的句柄,可以用来判定CSplashWnd是否已经创建或消毁

staticBOOL c_bShowSplashWnd;             //标识是否显示显示用来,静态成员需要在类外进行初始化

public:

static void EnableSplashScreen(BOOL bEnable = TRUE);        //用来设定c_bShowSplashWnd

static void ShowSplashScreen(CWnd* pParentWnd = NULL); //用来显示SplashScreen窗口

staticBOOL PreTranslateAppMessage(MSG* pMsg);                         //用来处理一些消息  注意这里不是继承于CWnd类的PreTranslateMessage方法

protected:

virtualvoid PostNcDestroy();

afx_msg intOnCreate(LPCREATESTRUCT lpCreateStruct);

afx_msg voidOnTimer(UINT_PTR nIDEvent);

afx_msg voidOnPaint();

BOOL Create(CWnd* pParentWnd = NULL);           //创建窗口

voidHideSplashScreen(void);                              //隐藏窗口

DECLARE_MESSAGE_MAP()

};

二、            CSplashWnd类方法的实现

这里需要注意的事项是在析构函数中一定要把c_pSplashWnd成员置为NULL类型,否则程序收到消息后会出现异常。其它没有什么了,直接看代码吧。

//Splash.cpp : implementation file

//

#include "stdafx.h"

#include "Splash.h"

CSplashWnd*CSplashWnd::c_pSplashWnd;

BOOLCSplashWnd::c_bShowSplashWnd;

//CSplashWnd

IMPLEMENT_DYNAMIC(CSplashWnd,CWnd)

CSplashWnd::CSplashWnd()

{

}

CSplashWnd::~CSplashWnd()

{

ASSERT(c_pSplashWnd == this);

c_pSplashWnd = NULL;

}

BEGIN_MESSAGE_MAP(CSplashWnd,CWnd)

ON_WM_CREATE()

ON_WM_TIMER()

ON_WM_PAINT()

END_MESSAGE_MAP()

//CSplashWnd message handlers

voidCSplashWnd::EnableSplashScreen(BOOL bEnable/* =TRUE*/)

{

c_bShowSplashWnd = bEnable;

}

voidCSplashWnd::ShowSplashScreen(CWnd* pParentWnd/* =NULL*/)

{

//如果不要显示SplashScreen或SplashWnd对象已经被创建则返回

if (!c_bShowSplashWnd || c_pSplashWnd!=NULL)

{

return;

}

c_pSplashWnd = newCSplashWnd;

if ( !c_pSplashWnd->Create(pParentWnd))

{

deletec_pSplashWnd;

}

else

{

c_pSplashWnd->UpdateWindow();

}

}

BOOLCSplashWnd::PreTranslateAppMessage(MSG* pMsg)

{

if(c_pSplashWnd == NULL)

returnFALSE;

if(pMsg->message == WM_KEYDOWN

|| pMsg->message ==WM_SYSKEYDOWN

|| pMsg->message ==WM_LBUTTONDOWN

|| pMsg->message ==WM_RBUTTONDOWN

|| pMsg->message ==WM_MBUTTONDOWN

|| pMsg->message ==WM_NCLBUTTONDOWN

|| pMsg->message ==WM_NCRBUTTONDOWN

|| pMsg->message ==WM_NCMBUTTONDOWN)

{

c_pSplashWnd->HideSplashScreen();

returnTRUE;

}

returnFALSE;

}

void CSplashWnd::PostNcDestroy()

{

delete this;

}

intCSplashWnd::OnCreate(LPCREATESTRUCT lpCreateStruct)

{

if(CWnd::OnCreate(lpCreateStruct) == -1)

return-1;

// TODO:  Add your specialized creation code here

CenterWindow();

SetTimer(1,3000,NULL);

return0;

}

voidCSplashWnd::OnTimer(UINT_PTR nIDEvent)

{

// TODO: Addyour message handler code here and/or call default

if(nIDEvent == 1)

{

HideSplashScreen();

}

/*CWnd::OnTimer(nIDEvent);*/

}

voidCSplashWnd::OnPaint()

{

CPaintDC dc(this);// device context for painting

// TODO: Addyour message handler code here

// Do notcall CWnd::OnPaint() for painting messages

CDC dcImg;

if ( !dcImg.CreateCompatibleDC(&dc))

{

return;

}

BITMAP bm;

m_bitmap.GetBitmap(&bm);

// paint theimage

CBitmap* pOldBit =dcImg.SelectObject(&m_bitmap);

dc.BitBlt(0,0,bm.bmWidth,bm.bmHeight,&dcImg,0,0,SRCCOPY);

dcImg.SelectObject(pOldBit);

}

BOOLCSplashWnd::Create(CWnd* pParentWnd)

{

if ( !m_bitmap.LoadBitmap(IDB_SPLASH))

{

returnFALSE;

}

BITMAP bm;

m_bitmap.GetBitmap(&bm);

returnCreateEx(0,

AfxRegisterWndClass(0,AfxGetApp()->LoadStandardCursor(IDC_ARROW)),

NULL,

WS_POPUP|WS_VISIBLE,

0,

0,

bm.bmWidth,

bm.bmHeight,

pParentWnd->GetSafeHwnd(),

NULL);

}

void CSplashWnd::HideSplashScreen(void)

{

DestroyWindow();

AfxGetMainWnd()->UpdateWindow();

}

三、            在SDI或者MDI中使用CSplashWnd类

(1)  在CWinApp::InitInstance()中调用CSplashWnd::EnableSplashScreen()设置c_bShowSplashWnd;
在PreTranslateMessage()中调用CSplashWnd::PreTranslateAppMessage(),将键盘和鼠标消息传递给CSplashWnd对象

(2)  (2)在CMainFrame对象的OnCreate()中调用CSplashWnd::ShowSplashScreen()创建一个静态的SplashScreen窗口对象c_pSplashWnd,并设置其父窗口为CMainFrame.

代码如下:

BOOLCDrawApp::InitInstance()

{

//用来处理是否显示SplashScreen

{

CCommandLineInfo cmdinfo;

ProcessShellCommand(cmdinfo);

CSplashWnd::EnableSplashScreen(cmdinfo.m_bShowSplash);

}

………..

}

BOOLCDrawApp::PreTranslateMessage(MSG* pMsg)

{

if(CSplashWnd::PreTranslateAppMessage(pMsg))

returnTRUE;

returnCWinAppEx::PreTranslateMessage(pMsg);

}

intCMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)

{

if(CMDIFrameWndEx::OnCreate(lpCreateStruct) == -1)

return-1;

………..

CSplashWnd::ShowSplashScreen(this);

return0;

}

在此过程中不会出现什么错误,但编译时会提示:IDB_SPLASH没有定义。这需要在资源中加入个位图资源,并将”resource.h” 头文件包含到CSlashWnd.cpp文件中。

四、            在对话框中使用CSplashWnd类

在对话框中使用和在SDI或MDI中使用基本相似,首先在CWinApp的继承类中处理InitInstance()和PreTranslateMessage(MSG* pMsg)两个消息函数,然后在对话框类的WM_CREATE响应函数中显示SplashScreen。

BOOLCDLGApp::InitInstance()

{

//用来处理是否显示SplashScreen

{

CCommandLineInfo cmdinfo;

ProcessShellCommand(cmdinfo);

CSplashWnd::EnableSplashScreen(cmdinfo.m_bShowSplash);

}

……….

}

BOOLCDLGApp::PreTranslateMessage(MSG* pMsg)

{

if (CSplashWnd::PreTranslateAppMessage(pMsg) )

{

returnTRUE;

}

returnCWinApp::PreTranslateMessage(pMsg);

}

intCDLGDlg::OnCreate(LPCREATESTRUCT lpCreateStruct)

{

if(CDialogEx::OnCreate(lpCreateStruct) == -1)

return-1;

// TODO:  Add your specialized creation code here

CSplashWnd::ShowSplashScreen(this);

return0;

}

时间: 2024-10-08 09:45:30

【转】vs2010中添加splashScreen的相关文章

VS2010中添加lib库引用

一.VS2010中添加lib库引用:1 菜单  项目---> 属性--->配置属性-->链接器---->输入---附加依赖项,  加入库名,如: my_API.lib; 或是在cpp源文件中用代码#pragma comment(lib,"my_API.lib")代替. 此时再编译会提示错误:fatal error LNK1104: 无法打开文件"my_API.lib" ,  原因应该是编译器不知道去哪里找我们的这个库,下面就来解决2 然后给项

vs2010中添加dll文件

1.更改设置 1.1   project->properties->configuration properties->C/C++->General->Addtional Include Directories->放dll文件的路径 1.2  project->properties->configuration properties->Linker->General->Addtional Library Directories->放d

VS2010的MFC对话框程序中添加菜单栏的过程

VS2010的MFC对话框程序中添加菜单栏的过程 最近在看一个用MFC写的界面的项目的代码,在代码和界面中一直没有看到关于菜单控件是如何添加进对话框的,于是就百度了下.结果,与其它控件(Button等)添加的方式不一样: VS2010的MFC对话框程序中添加菜单栏的过程大致分了这五步. 一.将Menu加入Resource视图中 在WorkSpace中的Resource视图下,在左边目录的任意位置上,先右键-->选择Insert Resource(插入资源)选项,在弹出的对话框中选择Menu以后,

VS2010在空解决方案中添加项目

如题,在空解决方案中添加第一个项目的时候会看不到那个solution解决方案文件,而是你当前添加的项目,当你再添加其他项目的时候就悲催了,找不到这个solution,只能在这个项目文件上新加文件,很郁闷. 原来有个选项 工具-选项-项目和解决方案-总是显示解决方案,把这个选项勾选了就可以在solution里面添加新的项目了.截图如下. VS2010在空解决方案中添加项目,布布扣,bubuko.com

VS2010编译错误:是否忘记了向源中添加“#include "stdafx.h

VS2010编译错误:是否忘记了向源中添加“#include "stdafx.h 编译提示:fatal error C1010: 在查找预编译头时遇到意外的文件结尾.是否忘记了向源中添加“#include "stdafx.h"”? 错误分析:此错误发生的原因是编译器在寻找预编译指示头文件(默认#include "stdafx.h")时,文件未预期结束.没有找到预编译指示信息的头文件"stdafx.h".    (因为工程中的每个cpp文

[整理]VS2010中如何添加“依赖","库目录","包含目录"

VS2010中如何添加“依赖","库目录","包含目录" 1. 添加编译所需要(依赖)的 lib 文件[解决方案资源管理器]“项目->属性->配置属性->连接器->输入->附加依赖项”里填写“winsock.lib”,多个 lib 以空格隔开.(等同于“#pragma comment(lib, "winsock.lib") ”语句)2. 添加库(Libs)文件目录    方法 1:[解决方案资源管理器]“项

如何在VS2010的VC++ 基于对话框的MFC程序中添加菜单

方法1:亲测 成功  转载自https://social.msdn.microsoft.com/Forums/vstudio/zh-CN/48338f6b-e5d9-4c0c-8b17-05ca3ef1f761/vs2010dialog 你可以创建菜单作为一种资源,然后将它附加到对话框.打开dialog“属性”对话框,你可以看到一个Menu属性,最终我们将在这里指定一个我们自己创建的菜单ID. 具体步骤如下: •在资源视图中右键单击工程名,插入一个新的菜单.这个菜单会有一个ID(例如:IDR_M

VS2010中的快捷键

一. VS2010中的快捷键 1: Ctrl + Enter (在光标指定位置的上+ K + C (注释) Ctrl + E + U (取消注释) <=> Ctrl + K + U (取消注释) 5: Tab  (增加缩进)面添加一行,并将光标移至新添加行的行首位置) Ctrl + Shift + Enter (在光标指定位置的下面添加一行,并将光标移至新添加行的行首位置) 2: Ctrl + Shift + L (删除当前行) 3: Ctrl + M + O (折叠所有的函数) Ctrl +

VS2010中如何将动态链接库改成静态链接库

VS2010中如何将动态链接库改成静态链接库 VS2010静态编译生成的.exe可执行文件,可以免安装免DLL在其他电脑直接运行. 静态编译:就是在编译可执行文件的时候,将可执行文件需要调用的对应动态链接库(.so)中的部分提取出来,链接到可执行文件中去,使可执行文件在运行的时候不依赖动态链接库. 1. 编译实现方式有两种 [1]debug方式的静态链接: 设置: (1)项目 -> 配置属性->常规->MFC的使用:在静态库中使用MFC(如果有使用MFC). (2)项目 -> 配置