《Windows游戏编程大师技巧》三、Windows高级编程


Windows编程很绝的地方在于:你不用了解太多细节,就可以完成很多工作。

使用资源

资源就是你的程序代码结合在一起的多块数据,可以被程序本身在运行时加载。
资源应当也放在程序的.EXE文件中的原因是:

1.同时包含代码和数据的.EXE文件更容易发布。
2.外力不容易任意删改程序的数据文件(如.BMP和.WAV文件)。

对于想编译进程序中的数据类型没有限制,下列这些预定义的资源类型就可以
满足大部分需要:

图标 - 小的位图文件
光标 - 鼠标指针的位图
字符串 - 可以硬编码在代码中,也可以集中放在这
声音 - 大部分Windows程序都使用.WAV格式
位图 - 这是标准的位图,使用.BMP扩展名
对话框 - 也可以作为资源来存储
图元 - 一系列图像操作记录的回放



要添加资源文件,必须有一个以ASCII形式的资源描述文件.RC。编译过程如下:



下面是如何在.RC脚本文件中定义一个ICON资源:

windowicon ICON star.ico (使用字符串)
124 ICON ship.ico          (使用数字)

使用字符串定义时会产生歧义,windowicon可能是个字符串也可能是个#define定义的符号常量。
所以还需要一个.H文件来解析符号索引。

RESOURCES.H的内容:

#define ID_ICON1 100
#define ID_ICON2 101
#define ID_ICON3 102

RESOURCES.RC的内容:

#include "RESOURCES.H"

ID_ICON1 ICON star.ico
ID_ICON2 ICON ball.ico
ID_ICON3 ICON cross.ico

现在,将RESOURCES.RC和.ICO文件添加到工程,并在程序中#include RESOURCES.H。
若使用字符串定义,则:winclass.hIcon = LoadIcon(hInstance, "icon_name");
若使用符号索引,则:winclass.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(ID_ICON1));
注意第一个参数不再是NULL而是hInstance。

其他资源也都是类似这样定义和使用的。
光标:LoadCursor(hInstance, MAKEINTRESOURCE(IDC_CURSOR1));
字符串表:LoadString(hInst, IDS_STRING109, string3, 20);


GDI简介

应该了解GDI,以便知道如何在不使用DirectX的情况下如何在Windows环境绘制各种图形。
而理解WM_PAINT消息对于标准的GDI图形和Windows编程来讲是非常重要的,因为大部分
Windows程序的显示都围绕该消息。例外是游戏编程中DirectDraw或Direct3D负责图形绘制。

PAINTSTRUCT ps;     // used in WM_PAINT
HDC hdc;                 // handle to a device context
case WM_PAINT:
{
// simply validate the window
hdc = BeginPaint(hwnd, &ps);
// do all your painting here
EndPaint(hwnd, &ps);
// return success
return (0);
}

参看下面图示,当一个窗口被移动、改变大小或被其他窗口“弄脏”时,该窗口的用户区的
部分或全部需要重画。这时,WM_PAINT消息就被发送了。



对BeginPaint()和EndPaint()函数的调用可以完成一系列任务。首先,它们使用户区有效;
其次,它们用该窗口创建时参照的Windows类中定义的背景刷来填充该窗口的背景。

你只能访问实际上需要刷新的该窗口用户区的一部分。无效矩阵区域的坐标都保存在
BeginPaint()函数返回值PAINTSTRUCT的rcPaint字段中。如果要访问整个用户区的话,
这就是一个问题。解决方法是通过GetDC()直接获得图形设备描述表。

但BeginPaint()-EndPaint()会向Windows发消息指示窗口有效,而GetDC()-ReleaseDC()不会,
所以WM_PAINT消息将一直不停地传递下去,因为必须使该窗口有效。因此在ReleaseDC()
后还要调用ValidateRect()。

PAINTSTRUCT ps;
HDC hdc;
RECT rect;
case WM_PAINT:
{
// simply validate the window
hdc = GetDC(hwnd);
// do all your painting here
ReleaseDC(hwnd, hdc);
// get client rectangle of window
GetClientRect(hwnd, &rect);
// validate window
ValidateRect(hwnd, &rect);
// return success
return(0);
}

注意GetClientRect是用来获取用户矩形区域的坐标。每个窗口都有两套坐标系:
Windows坐标系和用户坐标系。区别见下图:



你很可能会说:“非得这么麻烦吗”是的,非得如此,因为这是Windows。哈哈!
本书的大多数例程里,将在WM_PAINT消息以外的地方使用GetDC()-ReleaseDC(),
BeginPaint()-EndPaint()只用于WM_PAINT消息句柄中。


基本文本显示

Windows有最复杂且最强悍的文本渲染系统。当实际运用在实时游戏中时,用GDI文本
引擎输出文本就显得太慢了,还是要亲手设计基于DirectX的文本引擎。但先了解下有助于
调试和输出。

输出文本有两个常用函数:TextOut()和DrawText()。TextOut()是一个寒酸的文本输出函数,
而DrawText()则像凌志汽车一样豪华。我经常使用TextOut()因为它运行比较快。

case WM_PAINT: {
               hdc = BeginPaint (hwnd, &ps) ;
               GetClientRect (hwnd, &rect) ;
                  
               char buffer[50];
               static int counter = 0;
               sprintf(buffer, "WM_PAINT called %d times ", ++counter);
               TextOut(hdc, rect.left, rect.top, buffer, strlen(buffer));

               DrawText (hdc, TEXT (“Hello text”), -1, &rect,
                    DT_SINGLELINE | DT_CENTER | DT_VCENTER) ;               

               EndPaint (hwnd, &ps) ;
               return 0 ;
}

==========================================================

2012年4月22日 更新

补充学习处理重要事件:Window操作、键盘操作、鼠标操作。

1.Window操作

WM_CLOSE消息在WM_DESTROY和WM_QUIT之后发送,它表示用户正试图关闭窗口。
如果在WinProc()里仅return (0),那么用户不能关闭窗口。WM_SIZE消息对于窗口游戏
非常重要,当窗口尺寸改变时,必须调整图像显示来适应。


2.键盘操作

在Windows环境下,可以以多种方式访问键盘消息:
2.1 通过WM_CHAR消息:保存产生的ASCII码,如'a'或'A',字处理程序要关心这个。
2.2 通过WM_KEYDOWN/UP消息:保存产生的扫描码,即键盘上每一个按键的编码,
如ESC键VK_ESCAPE。游戏只需关心WASD移动,F开火而不必关心产生的是大写字符
还是小写字符,所以不必关心WM_CHAR消息。
2.3 通过调用GetAsyncKeyState():在状态表中跟踪该键的最后已知状态,使用它的
妙处是它与事件循环没有耦合,可以在任何地方测试按键。


3.鼠标操作

鼠标移动事件WM_MOUSEMOVE。lParam保存鼠标位置,wParam保存按键状态。

case WM_MOUSEMOVE:
{
int mouse_x = (int) LOWORD(lParam);
int mouse_y = (int) LOWORD(lParam);
int buttons = (int) wParam;
if (buttons & MK_LBUTTON)
...
if (buttons & MK_RBUTTON)
...
} break;

鼠标只是移动没有按键产生的事件。

case WM_LBUTTONDBLCLK:
{
int mouse_x = (int) LOWORD(lParam);
int mouse_y = (int) LOWORD(lParam);
     ...
// tell windows you handled it
return (0);
} break;


4.自行发送消息

自行传递消息有两种方法:
SendMessage()向窗口传递一个要求立即处理的消息。
PostMessage()将消息发往窗口的消息队列。

为什么要自行发送消息?因为Window的设计者希望你这样做,这也是窗口环境下的工作原理。
下一章中学习按键控件时将会看到,发送消息是和控件窗口交流的唯一途径!

转载于:https://www.cnblogs.com/xiaomaohai/archive/2012/04/18/6157844.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/262786.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

结构型模式--装饰模式

下面先用java,然后用Objective-C行对装饰模式的讲解: 对于java的装饰模式讲解和使用比较详细和难度有点偏高,而对于Objective-C的装饰模式讲解和使用方面比较简单,而且和java的装饰模式略有差异&#xff0c…

ArcGIS.Server.9.2.DotNet自带例子分析(三、一)

目的: 1.arcgis server9.2 ADF的AddGraphics。 准备工作: 1.用ArcGis Server Manager或者ArcCatalog发布一个叫world的Map Service,并且把这个Service启动起来。 2.找到DeveloperKit\SamplesNET\Server\Web_Applications目录下的Common_AddGraphicsCShar…

linux 使用timedatectl 修改时区和时间

使用timedatectl可以进行如下常见操作 1.查看当前时间/日期/时区:timedatectl或者timedatectl status 2.查看所有可用时区:timedatectl list-timezones 3.设置时区:timedatectl set-timezone “Asia/Shanghai” 或者 timedatectl set-time…

aspose将datatable导出excel 比自己拼好的多 Bug少-。.net

using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.IO; using System.Data; using Aspose.Cells; /// <summary> ///OutFileDao 的摘要说明 /// </summary> publicclass OutFileDao { public OutFileDa…

【MVC】AJAX+PartialView实现商城首页的楼层加载

使用AJAX实现楼层加载的例子已经非常多&#xff0c;但是html代码大都是手动拼接的&#xff0c;编写不便&#xff0c;而且难以维护。 下面就使用AJAXPartialView来实现 1.html代码 <!--楼层1开始--> <div class"floor" id"floor1"></div>…

VS2008常用快捷键

如果有一天&#xff0c;你的鼠标被猫叼走了的时候&#xff0c;你会发现这些快捷键很有意思&#xff0c;你的鼠标在的时候这些可爱的快捷键会让你更方便。 命令行&#xff1a; Devenv 启动VS Studion Isqlw 启动SQL2000查询分析器 Sqlwb 启动SQL2005企业管理器 Inet…

给控件动态添加方法

新建一个窗体,添加如下方法 public void init() { FormBuildStringControl formStringControl; FormBuildButtonControl formButtonControl; FormBuildGroupControl formGroupControl; ; // Adding a group formGroupControl this.form().addControl(FormControlType::Group, …

整理:Android apk 框架 布局 集锦

2019独角兽企业重金招聘Python工程师标准>>> 看到好的技术教程就想分享一下&#xff0c;不喜勿喷&#xff01;谢谢配合&#xff0c;仅供菜鸟学习研究(^o^)/~ 友情推荐《爱加密》Android apk加密保护视频教程剪辑&#xff1a;http://www.ijiami.cn/Video?v3 Andro…

IE8不兼容你的网页 怎么办? - 简单开启兼容模式

自从用了IE8 整个世界都变了形.   呵呵,问题没那么严重,如果你的网站还来不及修改以适合IE8访问的时候,咱们可以通过非常简单的方法,加几行代码就可以让访问的IE8自动调用IE7的渲染模式[/b],这样可以保证最大的兼容性,方法如下:   只需要在页面中加入如下HTTP meta-tag:  …

springboot打war包汇总

概述 第一次用maven工具打war包&#xff0c;出现各种各样的问题&#xff0c;做个问题记录方便下次查看 maven 一开始用的maven是springboot默认的&#xff0c;在.m2下&#xff0c;要打包时才发现没有mvn指令。索性自己就重新装个maven&#xff0c;去官网下载&#xff0c;我安装…

学习Nutch不错的系列文章

1&#xff09;Nutch1.2二次开发详细攻略 &#xff08;1&#xff09;Windows平台下Cygwin环境的搭建 地址&#xff1a;http://www.cnblogs.com/streamhope/archive/2011/07/27/2118397.html &#xff08;2&#xff09;Windows平台下Nutch1.2的搭建 地址&#xff1a;http://www.c…

[CareerCup] 9.4 Subsets 子集合

9.4 Write a method to return all subsets of a set. LeetCode上的原题&#xff0c;请参见我之前的博客Subsets 子集合和Subsets II 子集合之二。 解法一&#xff1a; class Solution { public:vector<vector<int> > getSubsets(vector<int> &S) {vecto…

新增一自增列

/// <summary> /// 为m_DataTable新增一自增列 /// </summary> private void AddAutoIncrementComunm() { DataColumn col new DataColumn(NewColumnName, typeof(System.Int32)); col.AutoIncrement t…

解决Lync2010错误:无法同步通讯簿信息

1、修改注册表&#xff1a;HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Lsa 新建DWORD值起名为DisableLoopbackCheck 数值为1&#xff1b; 2、检查一下你的iis 里面的虚拟目录abs&#xff0c;和你共享文件夹的的权限是否允许所有的域用户访问。 3、设置好以后重启i…

Java 8 stream forEach filter anyMatch

forEach的循 someObjects.forEach(obj -> { //to do something }) 如果这个循环的目标是找到匹配某个谓词的第一个元素 Optional<SomeObject> result someObjects.stream().filter(obj -> some_condition_met).findFirst(); 如果你只是想知道集合中是否有一个元素…

大数据初探——Hadoop历史

Hadoop是一个开源的分布式框架&#xff0c;是Apache下的一个开源项目。Hadoop运行可以在成千上万个普通机器节点组成的集群上&#xff0c;通过分布式的计算模型和存储模型来处理大数据集。Hadoop具有高容错性、工作在普通的机器节点上扩展性强等众多的优点&#xff0c;是企业选…

Dynamic Data Web Application编译是报GetActionPath调用模糊解决办法

新建的Dynamic Data Web Application编译时报错 Error 3 The call is ambiguous between the following methods or properties: System.Web.DynamicData.MetaTable.GetActionPath(string, System.Collections.Generic.IList<object>) and System.Web.DynamicData.MetaTa…

BMP格式图像的显示

使用多文档编程 也可以使用单文档编程 建立一个DIB图像的显示类 ImageDib 成员变量&#xff1a; 4个指针&#xff1a; LPBYTE m_lpDib; //指向DIB的指针    LPBITMAPINFOHEADER m_lpBmpInfoHead; //图像信息头指针 LPRGBQUAD m_lpColorTable; //图像颜色表指针 …

JPA多条件复杂SQL动态分页查询

概述 ORM映射为我们带来便利的同时&#xff0c;也失去了较大灵活性&#xff0c;如果SQL较复杂&#xff0c;要进行动态查询&#xff0c;那必定是一件头疼的事情&#xff08;也可能是lz还没发现好的方法&#xff09;&#xff0c;记录下自己用的三种复杂查询方式。 环境 springBoo…

上下文环境

##执行上下文环境 在一段JS真正运行之前浏览器已经做了一些准备工作&#xff1a; 1.变量&#xff1a;变量的申明&#xff0c;默认值为undefined&#xff1b; 2.this&#xff1a;赋值&#xff1b; 3.函数申明&#xff1a;赋值 变量&#xff1a; console.log(a) ---…