DX笔记之五------游戏画面绘图之绘制位图

本系列文章由zhmxy555编写,转载请注明出处。 http://blog.csdn.net/zhmxy555/article/details/7335103

共四步

步骤一:加载位图

步骤二:建立与窗口DC兼容的内存DC

步骤三:选用位图对象

步骤四:贴图

详细步骤

步骤一:加载位图

要从文件加载位图,常常使用LoadImage()函数。

HANDLE LoadImage(

HINSTANCE hinst,  //包含目标位图的DLL或exe文件的模块句柄

LPCTSTR lpszName,

UINT uType,

int cxDesired,

int cyDesired,

UINT fuLoad

);

下面是该函数参数的详细说明。

▲HINSTANCE  来源实体:包含位图所在的实体,若要加载的位图在硬盘或者资源文件中,此项设置为"NULL"。

▲LPCTSTR 名称:要加载的位图所在的路径与文件名或者资源名称

▲UINT 位图类型:加载位图的类型,有下面三种:

★IMAGE_BITMAP: 加载的位图为一般图文件,扩展名为".bmp"

★IMAGE_CUSOR: 加载的位图为光标图标,扩展名为".cur"

★IMAGE_ICON: 加载的位图为图标,扩展名为".ico"

▲int 加载宽度:位图加载的宽度,单位为像素

▲int 加载高度:位图加载的高度,单位为像素

▲UINT 加载方式:设定位图的加载方式,若是从文件中加载位图,则设为"LR_LOADFROMFILE"

步骤二:建立与窗口DC兼容的内存DC

我们调用CreateCompatible()函数来建立内存DC

HDC CreateCompatibleDC(HDC hdc);  //建立兼容DC

函数中输出的唯一参数就是要与内存DC兼容的目的DC

跟窗口DC一样,内存DC使用后也必须进行释放的操作,释放内存DC所调用的函数为DeleteDC()

DeleteDC(HDC DC名称);            //释放DC

步骤三:选用位图对象

位图对象是GDI的6种对象之一,内存DC选用位图对象的方法和前面介绍的选用画笔或画刷的方式相同,都是通过调用SelectObject()函数来实现。

GDI对象有:画笔,画刷,位图,字体,区域及调色板等。

步骤四:贴图

把内存DC中的位图复制到显示的DC上,即"贴图"。这个操作使用的函数是BitBlt(),后面我们会经常用到他。这个函数的定义如下

BOOL BitBlt(

int x,            //  目的DC x坐标

int y,            //  目的DC y坐标

int nWidth,       //  贴到目的DC的宽度

int nHeight,      //  贴到目的DC的高度

CDC* pSrcDC,   //  来源DC

int xSrc,         //  来源DC x坐标

int ySrc,         //  来源DC y坐标

DWORD dwRop  //  贴图方式();

);

#include <windows.h>

//全局变量声明
HINSTANCE hInst;
HBITMAP hbmp;
HDC        mdc;

//全局函数的声明
ATOM                MyRegisterClass(HINSTANCE hInstance);
BOOL                InitInstance(HINSTANCE, int);
LRESULT CALLBACK    WndProc(HWND, UINT, WPARAM, LPARAM);
void                MyPaint(HDC hdc);

//****Winmain函数,程序入口点函数**************************************
int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
    MSG msg;

    MyRegisterClass(hInstance);

    if (!InitInstance (hInstance, nCmdShow))
    {
        return FALSE;
    }

    //消息循环
    while (GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    return msg.wParam;
}

//****﹚设计一个窗口类,类似填空题,使用窗口结构体*************************
ATOM MyRegisterClass(HINSTANCE hInstance)
{
    WNDCLASSEX wcex;

    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    = TEXT("canvas");
    wcex.hIconSm        = NULL;

    return RegisterClassEx(&wcex);
}

//****初始化函数*************************************
// 1.建立与窗口DC兼容的内存DC
// 2.从文件加载位图并存至内存DC中
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
    HWND hWnd;
    HDC hdc;

    hInst = hInstance;

    hWnd = CreateWindow(TEXT("canvas"), TEXT("绘图窗口") , WS_OVERLAPPEDWINDOW,
        CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);

    if (!hWnd)
    {
        return FALSE;
    }

    MoveWindow(hWnd,10,10,800,600,true);
    ShowWindow(hWnd, nCmdShow);
    UpdateWindow(hWnd);

    hdc = GetDC(hWnd);//窗口DC
    mdc = CreateCompatibleDC(hdc);//与窗口兼容的内存DC

    hbmp = (HBITMAP)LoadImage(NULL,TEXT("bg.bmp"),IMAGE_BITMAP,800,600,LR_LOADFROMFILE); //Loads an icon, cursor, animated cursor, or bitmap
    //加载图标,光标,动画光标,或位图;加载位图

    SelectObject(mdc,hbmp);//选用GDI对象,返回先前使用的GDI对象
    //选用位图对象;

    MyPaint(hdc);//贴图

    ReleaseDC(hWnd,hdc);//删除GDI对象删除成功返回布尔值“ture”,若失败返回“FALSE”

    return TRUE;
}

//****自定义绘图函数*********************************
void MyPaint(HDC hdc)
{
    BitBlt(hdc,0,0,800,600,mdc,0,0,SRCCOPY);    //采用BitBlt函数贴图
}

//****消息处理函数**********************************
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    PAINTSTRUCT ps;
    HDC hdc;

    switch (message)
    {
        case WM_PAINT:                        //窗口重绘消息
            hdc = BeginPaint(hWnd, &ps);
            MyPaint(hdc);
            EndPaint(hWnd, &ps);
            break;
        case WM_DESTROY:                    //窗口结束消息
            DeleteDC(mdc);
            DeleteObject(hbmp);
            PostQuitMessage(0);
            break;
        default:                            //其他消息
            return DefWindowProc(hWnd, message, wParam, lParam);
   }
   return 0;
}

贴一下成果

感觉自己萌萌哒!感谢浅墨,哈哈哈。。。

时间: 2024-10-04 13:16:00

DX笔记之五------游戏画面绘图之绘制位图的相关文章

DX笔记之六------游戏画面绘图之透明特效的制作方法

原文链接:http://blog.csdn.net/zhmxy555/article/details/7338082 透明效果 由于所有的图文件都是以矩形来储存的,我们也许会需要把一张怪兽图片贴到窗口的背景图上,而这种情况下如果直接进行贴图,结果如下图: 这似乎不是我们想要的结果. 为了得到透明效果,我们需要运用到BitBlt()贴图函数以及其参数Raster的值来将图片中不必要的部分去掉(又称去背),使得图中的主题可以与背景完美融合. 我们以图中的恐龙为例子,首先准备一张位图,如下图. 图中的

matlab学习笔记之五种常见的图形绘制功能

分类: 离散数据图形绘制 函数图形绘制 网格图形绘制 曲面图形绘制 特殊图形绘制 本文重点介绍matlab五种图形绘制方法的后三种. 一.网格图形绘制 以绘制函数z=f(x,y)三维网格图为例,下面为绘制步骤: 确定自变量x和y的取值范围和取值间隔 x = x1:dx:x2; y = y1:dy:y2; 2.构成xoy平面上的自变量采样格点矩阵 1)  利用“格点”矩阵生成原理生成矩阵 X = ones(size(y))*x; Y = y*ones(size(x)); 2)  利用meshgri

游戏开发(三)——WIN32 黑白棋(三)——游戏画面的现实

整个游戏分3部分介绍. 1.棋局的现实 2.AI的现实 3.游戏画面的现实 提供一下完整项目下载 这是第三部分:画面的显示 这部分其实就比较简单的,说白了就是api的堆砌. 主要了解下windows的消息机制,以及怎么画图 主要是分别封装了下对棋盘,棋子,以及当前轮到谁,当前比分是多少,就是游戏画面上不同的部分的绘制. void DrawReversiBoard(); void DrawReversiPieces(EnumReversiPiecesType type, int row_y, in

DX笔记之一---Direct3D基础

一.预备知识 1.表面 表面就是Direct3D用于储存2D图像数据的一个像素矩阵.width和height以像素为单位,pitch以字节单位,用接口IDirect3DSurface来描述表面 LockRect:该方法用于获取指向表面存储区的指针,通过通过指针运算,可对每个像素进行读写操作: UnlockRect:配对使用,调用lock必须unlock,解除对表面存储区的锁定: GetDesc: 获取表面的描述信息,通过填充D3DSURFACE_DESC; //Assume _surface i

MySQL学习笔记之五 有关数据表操作

MySQL在创建表的时候,创建一个.frm文件保存表和列定义.索引存储在一个有.MYI(MYindex)扩展名的文件并且数据存储在有.MYD(MYData)扩展名的文件中.   一.用SHOW/ DESCRIBE语句显示数据表的信息 语法: SHOW TABLES [FROM db_name] [LIKE wild] or SHOW COLUMNS FROM tbl_name [FROM db_name] [LIKE wild] or SHOW INDEX FROM tbl_name [FROM

Citrix XenMobile学习笔记之五:XenMoble架构组件

XenMobile Enterprise包含了电子邮件和网页浏览等安全的移动生产应用程序,并且保障档共享.同步处理和编辑的安全.Citrix凭借这个无可比拟的整合式移动服务,成功在企业移动办公市场上脱颖而出,傲视同侪. 现今的工作模式讲求从各种移动装置上安全和顺利地存取应用程序与资料.员工则需从易用的统一化企业app store,获取移动.Windows.网络及软件即服务(Software as a service,SaaS)应用程序.员工需要能够随时随地检视.编辑.同步处理和分享资料,并在各种

5、蛤蟆的数据结构笔记之五链栈实现

5.蛤蟆的数据结构笔记之五链栈实现 本篇名言:"人生就像奕棋,一步失误,全盘皆输." 昨天对栈和队列进行了定义.这次我们来看下如何使用代码来实现链栈和链队列,后续蛤蟆会记录如何将栈应用到实际问题中. 栈一般是顺序结构,但是也可以采用链式存储结构,具体如下实现. 欢迎转载,转载请标明出处: 1.  定义结构体 #define MAX_STACKS10 typedef struct { intkey; /*otherfields */ }element; typedef struct st

linux网络编程学习笔记之五 -----并发机制与线程?

进程线程分配方式 简述下常见的进程和线程分配方式:(好吧,我仅仅是举几个样例作为笔记...并发的水太深了,不敢妄谈...) 1.进程线程预分配 简言之,当I/O开销大于计算开销且并发量较大时,为了节省每次都要创建和销毁进程和线程的开销.能够在请求到达前预先进行分配. 2.进程线程延迟分配 预分配节省了处理时的负担,但操作系统管理这些进程线程也会带来一定的开销.由此,有个折中的方法是,当某个处理须要花费较长时间的时候,我们创建一个并发的进程或线程来处理该请求.实现也非常easy,在主线程中定时,定

linux网络编程学习笔记之五 -----并发机制与线程池

进程线程分配方式 简述下常见的进程和线程分配方式:(好吧,我只是举几个例子作为笔记...并发的水太深了,不敢妄谈...) 1.进程线程预分配 简言之,当I/O开销大于计算开销且并发量较大时,为了节省每次都要创建和销毁进程和线程的开销.可以在请求到达前预先进行分配. 2.进程线程延迟分配 预分配节省了处理时的负担,但操作系统管理这些进程线程也会带来一定的开销.由此,有个折中的方法是,当某个处理需要花费较长时间的时候,我们创建一个并发的进程或线程来处理该请求.实现也很简单,在主线程中定时,定时到期,