开发者社区> 史迪奇2号> 正文

[游戏模版14] Win32 键盘控制

简介:
+关注继续查看


 

>_<:compared with the previous article,this one only adds key-message listener.

>_<:up down left right control the roal movement.

复制代码
 1 LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
 2 {
 3     int wmId, wmEvent;
 4     PAINTSTRUCT ps;
 5 
 6     switch (message)
 7     {
 8     case WM_KEYDOWN:
 9         switch(wParam)
10         {
11              case VK_UP:
12                  y-=20;
13                  if(y<0)y+=480;
14                  break;
15              case VK_DOWN:
16                  y=(y+20)%480;
17                  break;
18              case VK_LEFT:
19                  speed-=10;
20                  if(speed<0)speed=0;
21                  SetTimer(hWnd,1,speed,NULL);
22                  break;
23              case VK_RIGHT:
24                  speed+=10;
25                  if(speed>300)speed=300;
26                  SetTimer(hWnd,1,speed,NULL);
27                  break;
28         }
29         break;
30     case WM_TIMER:
31         MyPaint(hdc);
32         break;
33     case WM_COMMAND:
34         wmId    = LOWORD(wParam);
35         wmEvent = HIWORD(wParam);
36         // 分析菜单选择:
37         switch (wmId)
38         {
39         case IDM_EXIT:
40             DestroyWindow(hWnd);
41             break;
42         default:
43             return DefWindowProc(hWnd, message, wParam, lParam);
44         }
45         break;
46     case WM_PAINT:
47         hdc = BeginPaint(hWnd, &ps);
48         // TODO: 在此添加任意绘图代码...
49         EndPaint(hWnd, &ps);
50         break;
51     case WM_DESTROY:
52         DeleteDC(mdc);
53         ReleaseDC(hWnd,hdc);
54         DeleteObject(bg);
55         DeleteObject(dra);
56         KillTimer(hWnd,1);
57         PostQuitMessage(0);
58         break;
59     default:
60         return DefWindowProc(hWnd, message, wParam, lParam);
61     }
62     return 0;
63 }
复制代码

 StdAfx.h
 resourse.h
复制代码
#include "stdafx.h"
#include "resourse.h"

#define MAX_LOADSTRING 100

// Global Variables:
HINSTANCE hInst;                                // current instance
TCHAR szTitle[MAX_LOADSTRING];                                // The title bar text
TCHAR szWindowClass[MAX_LOADSTRING];                                // The title bar text
HBITMAP dra,bg;//背景和恐龙
HDC hdc,mdc,bufdc;//窗口DC.存储贴到窗口上的内容并透明处理.选取位图用的DC
HWND hWnd;
DWORD tPre,tNow,tCheck;
int x,y,num,speed=200;//恐龙坐标

// Foward declarations of functions included in this code module:
ATOM                MyRegisterClass(HINSTANCE hInstance);
BOOL                InitInstance(HINSTANCE, int);
LRESULT CALLBACK    WndProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK    About(HWND, UINT, WPARAM, LPARAM);
void                MyPaint(HDC hdc);
//========================================================================================
int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
     // TODO: Place code here.
    MSG msg;

    MyRegisterClass(hInstance);//调用函数向系统注册窗口类别,输入参数hInstance是目前运行程序的对象代码;

    // 调用InitInstance函数,进行初始化操作;
    if (!InitInstance (hInstance, nCmdShow))
    {
        return FALSE;
    }

    // 消息循环(通过消息循环来获取信息,
    //进行必要的键盘信息转换而后将控制权交给操作系统,
    //有操作系统决定哪个程序的消息处理函数处理消息
    while (msg.message!=WM_QUIT) //不是窗口结束消息WM_QUIT,则继续循环
    {
        if(PeekMessage(&msg,NULL,0,0,PM_REMOVE))//PeekMessage()函数来检测目前是否有消息处理(检测到0,否则1)
        {                                       //------------|不能用GetMassage()换,因为它只有取得WM_QUIT时才返回0,其他返回非0,错误返回-1
            TranslateMessage(&msg);//转换伪码及字符
            DispatchMessage(&msg);//将控制权交给系统,再有系统决定负责处理消息的程序;

        }
        else//当此次循环运行与上次绘图时间相差0.1秒时再进行重绘操作
        {
            tNow=GetTickCount();//取得从开始到现在运行的时间百万分之一秒
            if(tNow-tPre>=200)MyPaint(hdc);    //tPre前次绘图的时间;计算上次绘图到这次循环之间的时间
        }                                   //------------|若相差100个单位进行一次绘图操作,通过这
    }                                       //------------|个操作可以调整运行快慢

    return msg.wParam;
}
//=====================================================================================



//=============================================================================================
//在建立程序窗口实体之前,必须先定义一个窗口类别,其中包含所要建立窗口的信息,
//并向系统注册,这里的MyRegisterClass函数就是进行定义及注册窗口类别的函数。
//==============================================================================================
ATOM MyRegisterClass(HINSTANCE hInstance)
{
    WNDCLASSEX wcex;            //申请一个窗口类别“WNDCLASSEX”和结构”wcex“
                                //--------------------------------------------------------------
                                //定义vcex结构的各项信息,其中设定信息处理函数(lpfnWndProc)
                                //为WNDPROC,类别名称为(lpszClassName)为”fe";
                                //--------------------------------------------------------------
    wcex.cbSize = sizeof(WNDCLASSEX);

    wcex.style            = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc    = (WNDPROC)WndProc;
    wcex.cbClsExtra        = 0;
    wcex.cbWndExtra        = 0;
    wcex.hInstance        = hInstance;
    wcex.hIcon            = NULL;
    wcex.hCursor        = NULL;
    wcex.hCursor        = LoadCursor(NULL,IDC_ARROW);
    wcex.hbrBackground    = (HBRUSH)(COLOR_WINDOW+1);
    wcex.lpszMenuName    = NULL;
    wcex.lpszClassName    = "fe";
    wcex.hIconSm        = NULL;

    return RegisterClassEx(&wcex);//调用RegisterClassEx函数注册类别,返回一个“ATOM"形态的字符串
                                  //此字符串即为类别名称”fe";
}
//============================================================================================


//============================================================================================
//按照前面所定义的窗口类别来建立并显示实际的程序窗口
//============================================================================================
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
   char filename[20]="";
   HBITMAP bmp;
   hInst = hInstance; // 把instance handle 储存在全局变量中;

   hWnd = CreateWindow("fe","绘图窗口",WS_OVERLAPPEDWINDOW,
      CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
                      //-----------------------------------------------
                      //调用CreateWindow函数来建立一个窗口对象
                      //第一个参数就是窗口建立依据的类别名称
                      //-----------------------------------------------
   if (!hWnd)
   {
      return FALSE;
   }
   //------------------------------------------------
   //设定窗口的位置及窗口的大小,然后绘制显示在设备上
   //-------------------------------------------------
   MoveWindow(hWnd,10,10,640,480,true);//位置及大小
   ShowWindow(hWnd, nCmdShow);//改定窗口显示时的状态
   UpdateWindow(hWnd);//将窗口绘制在显示设备上

   hdc=GetDC(hWnd);
   mdc=CreateCompatibleDC(hdc);
   bufdc=CreateCompatibleDC(hdc);

   bmp=CreateCompatibleBitmap(hdc,640,480);//后面的位图处理是在bmp中进行
   SelectObject(mdc,bmp);

   dra=(HBITMAP)LoadImage(NULL,"dra.bmp",IMAGE_BITMAP,760,200,LR_LOADFROMFILE);
   bg=(HBITMAP)LoadImage(NULL,"bg.bmp",IMAGE_BITMAP,640,480,LR_LOADFROMFILE);

   num=0;
   x=500;//起始坐标
   y=300;
   SetTimer(hWnd,1,200,NULL);
   MyPaint(hdc);

   return TRUE;
}
//============================================================================================


//============================================================================================
//
//============================================================================================
void MyPaint(HDC hdc)
{
    if(num==8)num=0;
    if(tNow-tCheck>=200)//判断此次绘图时间由前一秒算起是否已经达到1秒钟的时间间隔
    {
        tCheck=tNow;//并重设下一次起始时间
    }

    SelectObject(bufdc,bg);//在mdc上贴上背景图
    BitBlt(mdc,0,0,640,480,bufdc,0,0,SRCCOPY);

    SelectObject(bufdc,dra);//在mdc上透明处理
    BitBlt(mdc,x,y,100,100,bufdc,num*100,100,SRCAND);
    BitBlt(mdc,x,y,100,100,bufdc,num*100,0,SRCPAINT);

    BitBlt(hdc,0,0,640,480,mdc,0,0,SRCCOPY);//将最后的画面显示在窗口上

    tPre=GetTickCount();//记录此次绘图的时间

    num++;

    x-=20;//记录下次贴图的坐标
    if(x<=-95)x=640;
}
//============================================================================================


//============================================================================================
//在前面定义类别的时候把WndProc定义为消息处理函数(当某些外部消息发生时,会按消息的类型
//来决定该如何进行处理。此外该函数也是一个回叫函数(CALLBACK)(windows系统函数)每一个
//程序都会接收信息,选择性接受、处理;
//============================================================================================
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    int wmId, wmEvent;
    PAINTSTRUCT ps;

    switch (message)
    {
    case WM_KEYDOWN:
        switch(wParam)
        {
             case VK_UP:
                 y-=20;
                 if(y<0)y+=480;
                 break;
             case VK_DOWN:
                 y=(y+20)%480;
                 break;
             case VK_LEFT:
                 speed-=10;
                 if(speed<0)speed=0;
                 SetTimer(hWnd,1,speed,NULL);
                 break;
             case VK_RIGHT:
                 speed+=10;
                 if(speed>300)speed=300;
                 SetTimer(hWnd,1,speed,NULL);
                 break;
        }
        break;
    case WM_TIMER:
        MyPaint(hdc);
        break;
    case WM_COMMAND:
        wmId    = LOWORD(wParam);
        wmEvent = HIWORD(wParam);
        // 分析菜单选择:
        switch (wmId)
        {
        case IDM_EXIT:
            DestroyWindow(hWnd);
            break;
        default:
            return DefWindowProc(hWnd, message, wParam, lParam);
        }
        break;
    case WM_PAINT:
        hdc = BeginPaint(hWnd, &ps);
        // TODO: 在此添加任意绘图代码...
        EndPaint(hWnd, &ps);
        break;
    case WM_DESTROY:
        DeleteDC(mdc);
        ReleaseDC(hWnd,hdc);
        DeleteObject(bg);
        DeleteObject(dra);
        KillTimer(hWnd,1);
        PostQuitMessage(0);
        break;
    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

//============================================================================================
复制代码


版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
使用NAT网关轻松为单台云服务器设置多个公网IP
在应用中,有时会遇到用户询问如何使单台云服务器具备多个公网IP的问题。 具体如何操作呢,有了NAT网关这个也不是难题。
33749 0
使用OpenApi弹性释放和设置云服务器ECS释放
云服务器ECS的一个重要特性就是按需创建资源。您可以在业务高峰期按需弹性的自定义规则进行资源创建,在完成业务计算的时候释放资源。本篇将提供几个Tips帮助您更加容易和自动化的完成云服务器的释放和弹性设置。
17759 0
腾讯云服务器 设置ngxin + fastdfs +tomcat 开机自启动
在tomcat中新建一个可以启动的 .sh 脚本文件 /usr/local/tomcat7/bin/ export JAVA_HOME=/usr/local/java/jdk7 export PATH=$JAVA_HOME/bin/:$PATH export CLASSPATH=.
12193 0
阿里云服务器ECS远程登录用户名密码查询方法
阿里云服务器ECS远程连接登录输入用户名和密码,阿里云没有默认密码,如果购买时没设置需要先重置实例密码,Windows用户名是administrator,Linux账号是root,阿小云来详细说下阿里云服务器远程登录连接用户名和密码查询方法
19791 0
如何设置阿里云服务器安全组?阿里云安全组规则详细解说
阿里云安全组设置详细图文教程(收藏起来) 阿里云服务器安全组设置规则分享,阿里云服务器安全组如何放行端口设置教程。阿里云会要求客户设置安全组,如果不设置,阿里云会指定默认的安全组。那么,这个安全组是什么呢?顾名思义,就是为了服务器安全设置的。安全组其实就是一个虚拟的防火墙,可以让用户从端口、IP的维度来筛选对应服务器的访问者,从而形成一个云上的安全域。
15594 0
阿里云服务器安全组设置内网互通的方法
虽然0.0.0.0/0使用非常方便,但是发现很多同学使用它来做内网互通,这是有安全风险的,实例有可能会在经典网络被内网IP访问到。下面介绍一下四种安全的内网互联设置方法。 购买前请先:领取阿里云幸运券,有很多优惠,可到下文中领取。
17482 0
阿里云服务器ECS登录用户名是什么?系统不同默认账号也不同
阿里云服务器Windows系统默认用户名administrator,Linux镜像服务器用户名root
11323 0
阿里云服务器怎么设置密码?怎么停机?怎么重启服务器?
如果在创建实例时没有设置密码,或者密码丢失,您可以在控制台上重新设置实例的登录密码。本文仅描述如何在 ECS 管理控制台上修改实例登录密码。
18329 0
阿里云服务器如何登录?阿里云服务器的三种登录方法
购买阿里云ECS云服务器后如何登录?场景不同,阿里云优惠总结大概有三种登录方式: 登录到ECS云服务器控制台 在ECS云服务器控制台用户可以更改密码、更换系.
23611 0
阿里云服务器如何登录?阿里云服务器的三种登录方法
购买阿里云ECS云服务器后如何登录?场景不同,云吞铺子总结大概有三种登录方式: 登录到ECS云服务器控制台 在ECS云服务器控制台用户可以更改密码、更换系统盘、创建快照、配置安全组等操作如何登录ECS云服务器控制台? 1、先登录到阿里云ECS服务器控制台 2、点击顶部的“控制台” 3、通过左侧栏,切换到“云服务器ECS”即可,如下图所示 通过ECS控制台的远程连接来登录到云服务器 阿里云ECS云服务器自带远程连接功能,使用该功能可以登录到云服务器,简单且方便,如下图:点击“远程连接”,第一次连接会自动生成6位数字密码,输入密码即可登录到云服务器上。
31958 0
862
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
OceanBase 入门到实战教程
立即下载
阿里云图数据库GDB,加速开启“图智”未来.ppt
立即下载
实时数仓Hologres技术实战一本通2.0版(下)
立即下载