目前,在开始开发程序之前,我正在使用DirectX做一些测试用例。现在,我已经花了一天左右的时间处理下面的代码。我在网上查找类似的问题时修改了代码。但最终,它也没有按照我想要的方式工作。谁能告诉我我到底做错了什么?也许其他人可以解决这个问题。
// TetrisClone.cpp : Defines the entry point for the application.
//
#include "stdafx.h"
#include "TetrisClone.h"
#include <d3d9.h>
#include <d3dx9.h>
#define MAX_LOADSTRING 100
#define SCREEN_WIDTH 800
#define SCREEN_HEIGHT 600
#define D3DFVF_CUSTOMVERTEX (D3DFVF_XYZ|D3DFVF_DIFFUSE)
//for testing purposes
struct CUSTOMVERTEX
{
FLOAT x, y, z; //position
DWORD color; //Color
};
// global declarations for Direct3d
LPDIRECT3D9 d3d; // the pointer to our Direct3D interface
LPDIRECT3DDEVICE9 d3ddev; // the pointer to the device class
LPDIRECT3DVERTEXBUFFER9 g_pVB = NULL;
// function prototypes
void initD3D(HWND hWnd); // sets up and initializes Direct3D
void render_frame(void); // renders a single frame
void cleanD3D(void); // closes Direct3D and releases memory
void init_graphics(void); // 3D declarations
// Global Variables:
HINSTANCE hInst; // current instance
TCHAR szTitle[MAX_LOADSTRING]; // The title bar text
TCHAR szWindowClass[MAX_LOADSTRING]; // the main window class name
// Forward declarations of functions included in this code module:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);
int APIENTRY _tWinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPTSTR lpCmdLine,
int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
// TODO: Place code here.
MSG msg;
HACCEL hAccelTable;
// Initialize global strings
LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadString(hInstance, IDC_TETRISCLONE, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance);
// Perform application initialization:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
}
hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_TETRISCLONE));
// Main message loop:
while(TRUE)
{
// Check to see if any messages are waiting in the queue
while(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
{
// Translate the message and dispatch it to WindowProc()
TranslateMessage(&msg);
DispatchMessage(&msg);
}
// If the message is WM_QUIT, exit the while loop
if(msg.message == WM_QUIT)
break;
render_frame();
}
cleanD3D();
return (int) msg.wParam;
}
//
// FUNCTION: MyRegisterClass()
//
// PURPOSE: Registers the window class.
//
// COMMENTS:
//
// This function and its usage are only necessary if you want this code
// to be compatible with Win32 systems prior to the 'RegisterClassEx'
// function that was added to Windows 95. It is important to call this function
// so that the application will get 'well formed' small icons associated
// with it.
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEX wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_TETRISCLONE));
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_GRAYTEXT);
wcex.lpszMenuName = MAKEINTRESOURCE(IDC_TETRISCLONE);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));
return RegisterClassEx(&wcex);
}
//
// FUNCTION: InitInstance(HINSTANCE, int)
//
// PURPOSE: Saves instance handle and creates main window
//
// COMMENTS:
//
// In this function, we save the instance handle in a global variable and
// create and display the main program window.
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
HWND hWnd;
hInst = hInstance; // Store instance handle in our global variable
hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT, SCREEN_WIDTH, SCREEN_HEIGHT, NULL, NULL, hInstance, NULL);
if (!hWnd)
{
return FALSE;
}
ShowWindow(hWnd, nCmdShow);
// set up and initialize Direct3D
initD3D(hWnd);
UpdateWindow(hWnd);
return TRUE;
}
//
// FUNCTION: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// PURPOSE: Processes messages for the main window.
//
// WM_COMMAND - process the application menu
// WM_PAINT - Paint the main window
// WM_DESTROY - post a quit message and return
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
int wmId, wmEvent;
PAINTSTRUCT ps;
HDC hdc;
switch (message)
{
case WM_COMMAND:
wmId = LOWORD(wParam);
wmEvent = HIWORD(wParam);
// Parse the menu selections:
switch (wmId)
{
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
break;
case WM_PAINT:
hdc = BeginPaint(hWnd, &ps);
// TODO: Add any drawing code here...
EndPaint(hWnd, &ps);
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
// Message handler for about box.
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE;
case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
/***
Direct3d functions
***/
// this function initializes and prepares Direct3D for use
void initD3D(HWND hWnd)
{
d3d = Direct3DCreate9(D3D_SDK_VERSION); // create the Direct3D interface
D3DPRESENT_PARAMETERS d3dpp; // create a struct to hold various device information
ZeroMemory(&d3dpp, sizeof(d3dpp)); // clear out the struct for use
d3dpp.Windowed = TRUE; // program windowed, not fullscreen
d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD; // discard old frames
d3dpp.hDeviceWindow = hWnd; // set the window to be used by Direct3D
// create a device class using this information and information from the d3dpp stuct
d3d->CreateDevice(D3DADAPTER_DEFAULT,
D3DDEVTYPE_HAL,
hWnd,
D3DCREATE_SOFTWARE_VERTEXPROCESSING,
&d3dpp,
&d3ddev);
init_graphics();
}
// this is the function used to render a single frame
void render_frame(void)
{
// clear the window to a deep blue
d3ddev->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(0, 40, 100), 1.0f, 0);
d3ddev->BeginScene(); // begins the 3D scene
//set FVF
d3ddev->SetFVF(D3DFVF_CUSTOMVERTEX);
//setting stream source
d3ddev->SetStreamSource(0, g_pVB, 0, sizeof(CUSTOMVERTEX));
// do 3D rendering on the back buffer here
//drawing triangle
d3ddev->DrawPrimitive(D3DPT_TRIANGLELIST, 0, 1);
d3ddev->EndScene(); // ends the 3D scene
d3ddev->Present(NULL, NULL, NULL, NULL); // displays the created frame
}
// this is the function that puts the 3D models into video RAM
void init_graphics(void)
{
// create the vertices using the CUSTOMVERTEX struct
CUSTOMVERTEX vertices[] =
{
{ 400.0f, 62.5f, 0.0f, D3DCOLOR_XRGB(0, 0, 255), },
{ 650.0f, 500.0f, 0.0f, D3DCOLOR_XRGB(0, 255, 0), },
{ 150.0f, 500.0f, 0.0f, D3DCOLOR_XRGB(255, 0, 0), },
};
// create a vertex buffer interface called v_buffer
d3ddev->CreateVertexBuffer(3*sizeof(CUSTOMVERTEX),
0,
D3DFVF_CUSTOMVERTEX,
D3DPOOL_MANAGED,
&g_pVB,
NULL);
VOID* pVoid; // a void pointer
// lock v_buffer and load the vertices into it
g_pVB->Lock(0, 0, (void**)&pVoid, 0);
memcpy(pVoid, vertices, sizeof(vertices));
g_pVB->Unlock();
}
// this is the function that cleans up Direct3D and COM
void cleanD3D(void)
{
g_pVB->Release();
d3ddev->Release(); // close and release the 3D device
d3d->Release(); // close and release Direct3D
}发布于 2012-11-15 21:47:06
因为你没有设置相机,也没有对你的顶点做任何变换,所以,你应该把你的顶点设置在长方体的范围内(透视投影变换之后),这是
-1 << 1 << 1
-1 << y << 1
0 << z << 1以下更改将使您的三角形可见。
CUSTOMVERTEX vertices[] =
{
{ 0.0f, 0.0, 0.0f, D3DCOLOR_XRGB(0, 0, 255), },
{ 0.0f, 1.0f, 0.0f, D3DCOLOR_XRGB(0, 255, 0), },
{ 1.0f, 0.0f, 0.0f, D3DCOLOR_XRGB(255, 0, 0), },
};有很多原因会导致几何/模型在Direct3D程序中不可见,我列出了一些原因如下。
通常,我们需要在渲染期间启用照明,但如果您使用LT (照明和变换)顶点格式,这意味着您定义顶点的颜色并使用它的屏幕坐标,这告诉Direct3D引擎:“嘿,不要照明和变换我的顶点,我会这样做的!”,在您的代码中您没有使用LT格式,这意味着您应该转换和照明您的顶点,但您没有任何灯光设置,所以您将得到一个黑色三角形,事实上,您没有得到任何东西,因为顶点坐标是看不见的。
我们通常在对象空间中指定顶点坐标,然后将它们转换到世界空间->视图空间->投影空间->屏幕空间,在上面的代码中,您没有做任何传输,但是您的顶点坐标太大,透视变换后超出了长方体的范围,范围是- 1 << x << 1,-1 << y << 1,0 << z << 1。
这也是一个需要考虑的非常重要的因素,默认情况下,Direct3D的使用将按逆时针方向剔除所有定义的面,请参阅此线程Odd behaviour when trying to create triangle using TriangleStrip Topologu了解详细信息。
首先,确保你的相机面对你正在绘制的东西,然后设置一个正确的距离。您为顶点使用了较大的坐标,因此,您应该将摄影机设置为远离要渲染的几何体,否则您将什么也看不到。
这看起来有点可笑,但它确实发生了,当我们想要测试渲染是否工作时,我们总是注释掉渲染函数中的一些代码并绘制其他东西,我们也可以测试程序是否可以在没有任何绘制代码的情况下工作,然后我们忘记了取消注释绘制代码,这导致屏幕上没有任何内容。我在刚开始学习D3D时就遇到过这个问题。
发布于 2012-11-15 14:17:16
我可以通过一些调整来渲染你的场景!
我注意到的第一件事是你的FVF不是以我习惯的方式定义的,没有相机就可以渲染。(不是100% )在你的情况下,如果你只想要位置和颜色,而不是摄像头,你会使用
#define D3DFVF_CUSTOMVERTEX (D3DFVF_XYZRHW|D3DFVF_DIFFUSE)而不是你所拥有的。
这也意味着您需要更改您的自定义顶点结构以添加w分量,此链接可能有助于您理解w的用途。在基本术语中,XYZRHW意味着点已经被转换到屏幕空间,而不是在世界空间中。(快速谷歌搜索:http://www.gamedev.net/topic/200077-xyz-rhw/ )
struct CUSTOMVERTEX
{
FLOAT x, y, z, w; //position
DWORD color; //Color
};有了这个改变,你还需要在init_graphics中改变你的顶点,试着这样做,(你已经有了,但添加了w组件)我还删除了一些额外的逗号。
CUSTOMVERTEX vertices[] =
{
{ 400.0f, 62.5f, 0.0f, 1, D3DCOLOR_XRGB(0,0,255) },
{ 650.0f, 500.0f, 0.0f, 1, D3DCOLOR_XRGB(0,255,0) },
{ 150.0f, 500.0f, 0.0f, 1, D3DCOLOR_XRGB(255,0,0) }
};其次,在您的init_graphics函数中,您的顶点缓冲区锁锁定的大小为0!更改它以锁定您拥有的顶点数,在本例中为3。
g_pVB->Lock(0, 3*sizeof(CUSTOMVERTEX), (void**)&pVoid, 0);小心使用memcpy的方式,如果你在函数之间传递顶点数据,你的指针会降低,导致sizeof只检索数组中一个元素的大小。在您当前的代码中根本没有必要更改这一点,但这可能在未来很重要,您应该知道这一点。在我看来,使用这样的东西将使这一点更具未来的证据。
memcpy(pVoid, vertices, 3*sizeof(CUSTOMVERTEX));如果我没有忘记任何东西,这应该足以让您启动并运行。祝好运!
https://stackoverflow.com/questions/13392021
复制相似问题