现在我们要创建一个Windows外壳程序,将单调的Windows运行细节隐藏,
Win32/DirectX编程就变得类似于32位DOS的编程过程。在构建好的代码结构下,
我们只需关心Game_*几个有关游戏逻辑的方法就行了。
1.Game_Init()
Game_Init()在WinMain()中的主事件循环之前被调用,并且仅调用一次。它用于初始化游戏
所需的各种资源,如获得资源设备描述表DC。
2.Game_Main()
Game_Main()在主事件循环中每次处理Windows消息之后被调用一次,应当在Game_Main()中
处理所有的图像渲染、声音、人工智能等内容。需要注意的是你必须仅绘制一帧画面然后就
返回。
3.Game_Shutdown()
Game_Shutdown()在主事件循环退出后被调用,清除游戏中被分配的所有资源。
#define WIN32_LEAN_AND_MEAN #include #include // DEFINE AND GLOBALS // #define WINDOW_CLASS_NAME "WINCLASS1" #define KEYDOWN(vk_code) ((GetAsyncKeyState(vk_code) & 0x8000) ? 1 : 0) #define KEYUP(vk_code) ((GetAsyncKeyState(vk_code) & 0x8000) ? 0 : 1) HWND main_window_handle = NULL; HINSTANCE hinstance_app = NULL; char buffer[80]; // Function prototype // LRESULT CALLBACK WindowProc(HWND, UINT, WPARAM, LPARAM); int Game_Init(void *parms = NULL, int num_parms = 0); int Game_Main(void *parms = NULL, int num_parms = 0); int Game_Shutdown(void *parms = NULL, int num_parms = 0); // WINMAIN int WINAPI WinMain(HINSTANCE hinstance, HINSTANCE hprevinstance, LPSTR lpcmdline, int ncmdshow) { WNDCLASSEX winclass; HWND hwnd; MSG msg; // 1. Fill in the window class structure winclass.cbSize = sizeof(WNDCLASSEX); winclass.style = CS_DBLCLKS | CS_OWNDC | CS_HREDRAW | CS_VREDRAW; winclass.lpfnWndProc = WindowProc; winclass.cbClsExtra = 0; winclass.cbWndExtra = 0; winclass.hInstance = hinstance; winclass.hIcon = LoadIcon(NULL, IDI_APPLICATION); winclass.hCursor = LoadCursor(NULL, IDC_ARROW); winclass.hbrBackground = (HBRUSH) GetStockObject(BLACK_BRUSH); winclass.lpszMenuName = NULL; winclass.lpszClassName = WINDOW_CLASS_NAME; winclass.hIconSm = LoadIcon(NULL, IDI_APPLICATION); // 2. Register the window class if (!RegisterClassEx(&winclass)) return (0); // 3. Create window if (!(hwnd = CreateWindowEx(NULL, WINDOW_CLASS_NAME, "T3D Game Console Version 1.0", WS_OVERLAPPEDWINDOW | WS_VISIBLE, 0, 0, 400, 300, NULL, NULL, hinstance, NULL))) return (0); // Save global handle main_window_handle = hwnd; hinstance_app = hinstance; // Initialize game here Game_Init(); while (TRUE) { // Retrieve message if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { if (msg.message == WM_QUIT) break; TranslateMessage(&msg); DispatchMessage(&msg); } // Main game processing goes here Game_Main(); } // Closedown game here Game_Shutdown(); return msg.wParam; } /// LRESULT CALLBACK WindowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) { PAINTSTRUCT ps; HDC hdc; switch(msg) { case WM_CREATE: return (0); case WM_PAINT: hdc = BeginPaint(hwnd, &ps); EndPaint(hwnd, &ps); return (0); case WM_DESTROY: PostQuitMessage(0); return (0); default: break; } return (DefWindowProc(hwnd, msg, wparam, lparam)); } /// int Game_Init(void *parms, int num_parms) { return (1); } /// int Game_Main(void *parms, int num_parms) { DWORD start_time = GetTickCount(); // Lock time to 30 fps (1s / 30 = 33ms) while ((GetTickCount() - start_time) < 33); if (KEYDOWN(VK_ESCAPE)) SendMessage(main_window_handle, WM_CLOSE, 0, 0); return (1); } int Game_Shutdown(void *parms, int num_parms) { return (1); }
下面是一个流星的具体实例,我们在Game_Init()随机初始化256个小星星。之后
每一帧绘制时,首先擦除掉星星位置上的点,之后将星星向屏幕右侧移动并检测
是否到达屏幕边界,最后在新位置上绘制星星。这样就可以进入下一轮的主事件
循环了。
/// #define NUM_STARS 256 #define WINDOW_WIDTH 400 #define WINDOW_HEIGHT 300 typedef struct STAR_TYP { int x, y; int vel; COLORREF col; } STAR, *STAR_PTR; HDC global_dc = NULL; STAR stars[NUM_STARS]; void Erase_Stars(void); void Draw_Stars(void); void Move_Stars(void); void Init_Stars(void); /// int Game_Init(void *parms, int num_parms) { global_dc = GetDC(main_window_handle); Init_Stars(); return (1); } /// int Game_Main(void *parms, int num_parms) { DWORD start_time = GetTickCount(); Erase_Stars(); Move_Stars(); Draw_Stars(); // Lock time to 30 fps (1s / 30 = 33ms) while ((GetTickCount() - start_time) < 33); if (KEYDOWN(VK_ESCAPE)) SendMessage(main_window_handle, WM_CLOSE, 0, 0); return (1); } int Game_Shutdown(void *parms, int num_parms) { ReleaseDC(main_window_handle, global_dc); return (1); } void Init_Stars(void) { for (int index = 0; index < NUM_STARS; index++) { stars[index].x = rand() % WINDOW_WIDTH; stars[index].y = rand() % WINDOW_HEIGHT; // random velocity 1-16 stars[index].vel = 1 + rand() % 16; int intensity = 15 * (17 - stars[index].vel); stars[index].col = RGB(intensity, intensity, intensity); } } void Erase_Stars(void) { for (int index = 0; index < NUM_STARS; index++) SetPixel(global_dc, stars[index].x, stars[index].y, RGB(0, 0, 0)); } void Draw_Stars(void) { for (int index = 0; index < NUM_STARS; index++) SetPixel(global_dc, stars[index].x, stars[index].y, stars[index].col); } void Move_Stars(void) { for (int index = 0; index < NUM_STARS; index++) { stars[index].x += stars[index].vel; if (stars[index].x >= WINDOW_WIDTH) stars[index].x -= WINDOW_WIDTH; } }
运行起来就是这个样子了。
这是一个使用GDI绘图的不错的小程序,该程序演示了擦除、移动、绘制动画的循环。
第一部分总结
在第2章我们通过一个简单的Windows程序学习了创建一个最简单Windows窗体的步骤:
创建Windows类,注册它,然后创建窗体,编写事件处理函数,最后编写主事件循环
从事件队列获取并分发消息到事件处理函数。并且我们还学会了使用PeekMessage()
而不是阻塞的GetMessage()编写实时的事件循环。
第3章我们学会了如何创建图标、光标、字符串等等各种资源,并在程序中引用它们。
同时我们还学习了GDI基础知识,获得DC句柄,调用TextOut输出文字。以及重要的
窗口、键盘、鼠标事件处理。
第4章我们学习了GDI高级部分绘制各种图形,定时器,常用的按钮等小控件。至此我们
已经掌握了足够的Windows编程知识来应对游戏编程,下面是总结的知识列表,留作日后
复习使用:
1.Windows程序构建步骤:WNDCLASS类、注册、创建窗体、事件处理函数、主事件循环。
2.用PeekMessage()编写实时的事件循环
3.使用图标、光标、字符串等资源,并了解VS自动产生的.RC文件和资源索引文件中有什么。
4.处理常见窗口、键盘、鼠标消息,如WM_CLOSE,WM_SIZE,获得按键,WM_MOUSE*。
5.用GDI绘制文本、点线及各种图形。
6.通过检查系统时钟来控制程序刷新的帧频。
7.使用简单的小控件,如按钮、文本框等等。
接下来我们将进入DirectX的神奇世界!