闪屏图形特技效果的实现
闪屏其实就是窗口,其基类是CWnd,与一般的窗口没有什么区别,只不过这个窗口在开始显示而已。利用类向导生成一个新类,其基类为CWnd。在新类中增加如下变量。
CDC MemDC; //创建内存DCBITMAP bm; //创建位图结构变量CBitmap m_bitmap; //创建位图对象CBitmap *old_bitmap; //创建位图对象指针
在资源中导入一张位图,色彩可以大于256色,只不过在VC中不能编辑256以上的位图。在构造函数初始化。 m_bitmap.LoadBitmap(IDB_BITMAP1);//拷贝资源位图m_bitmap.GetBitmap(&bm);//得到位图结构中的大小信息
并在类的头文件中#include "resource.h",否则编译时会出现IDB_BITMAP1没有定义的错误。然后在类中增加一个函数CreatLjxWnd()用来生成窗口。在里面增加如下代码: CreateEx(0, AfxRegisterWndClass(0,AfxGetApp()->LoadStandardCursor(IDC_ARROW)),//注册类 "animatesplash",//窗口标题 WS_POPUP,//窗口为弹出式 0,0,bm.bmWidth,bm.bmHeight, //建立大小与位图大小相同的窗口 NULL, NULL, NULL );
这一下我们可以把位图显示在窗口里了。特技显示有很多,像淡入,半透明等等。如想得到更多的特效算法,可以参考化境编程界网站(www.5xsoft.com)有一篇《Visual C++中的图形特技》和《精通VC++图像编程》,它们里面的算法写得很详细。我们这里采用随机积木法。原理是:将内存设备情境对象(如MemDC)中的位图数据分成纵横十等份共一百组数据,然后随机地取出这一百组数据中的某一组显示到目标设备(如ClientDC)中待显示位图的相应位置,如此反复直到所有一百组数据均显示完毕为止。我们增加WM_PAINT消息,vc++自动生成响应此消息的对应函数,特技显示代码也在此函数执行。在函数中增加代码: MemDC.CreateCompatibleDC(NULL);//建立一个和dc兼容的内存DC放置位图old_bitmap=MemDC.SelectObject(&m_bitmap);//将创建的位图选入内存DC//随机积木图像显示特技算法int i,j,stepx,stepy,dispnum,x,y; int ljx[20][20]; //数组记录已显示过的数据组 for ( i=0; i<20; i++ ) for ( j=0; j<20; j++ ) ljx[i][j]=0; stepx=bm.bmWidth/20; stepy=bm.bmHeight/20; srand( (unsigned)time( NULL ) ); dispnum=0; //记录已显示过的数据组的个数 while(1) { x=rand() % 20; y=rand() % 20; if ( ljx[x][y] ) //如果为1,则已经显示了,跳出循环。 continue; ljx[x][y]=1; //显示,设置为1 dc.StretchBlt(x*stepx, //目标设备逻辑横坐标 y*stepy, //目标设备逻辑纵坐标 stepx, //显示位图的像素宽 stepy, //显示位图的像素高度 &MemDC, //位图内存设备对象 x*stepx, //位图的起始横 y*stepy, //位图的起始纵坐标 stepx, //位图的像素宽 stepy, //位图的像素高度 SRCCOPY); dispnum++; if ( dispnum >=400 ) break; Sleep(10); }MemDC.SelectObject(old_bitmap);
到此,我们这个类已经大功告成了。现在就是要在对话框应用程序显示之前显示它出来。接下来我们要把这个类实例化再显示。我们找到应用程序的初始化函数InitInstance(),在里面增加如下代码: BOOL CAnimatesplashApp::InitInstance(){ AfxEnableControlContainer();#ifdef _AFXDLL Enable3dControls(); // Call this when using MFC in a shared DLL#else Enable3dControlsStatic(); // Call this when linking to MFC statically#endif //增加的代码 CLjxWnd *ljxljx=new CLjxWnd; //建立一个新窗口对象 ljxljx->CreatLjxWnd (); //创建窗口 ljxljx->CenterWindow (); //在屏幕中央 ljxljx->ShowWindow (SW_SHOW); //显示窗口 ljxljx->UpdateWindow (); //更新窗口,激活OnPait函数 Sleep(2000); //等待函数指定秒钟 if (ljxljx!=NULL) ljxljx->SendMessage (WM_CLOSE); //关闭窗口 //代码结束 CAnimatesplashDlg dlg; m_pMainWnd = &dlg; int nResponse = dlg.DoModal(); if (nResponse == IDOK) { // TODO: Place code here to handle when the dialog is // dismissed with OK } else if (nResponse == IDCANCEL) { // TODO: Place code here to handle when the dialog is // dismissed with Cancel } // Since the dialog has been closed, return FALSE so that we exit the // application, rather than start the application''s message pump. return FALSE;}
到这里就算是完成了,编译执行看看效果吧。你也可以对这个代码进行改进。是它的功能更加强大。