1_DirectX --D3D窗口初始化

《---------------------------------------》

欢迎转载,转载请注明

原文博客

《---------------------------------------》

学习D3D一个月了,

之前了解的是OpenGL,现在过来学习Direct3D,进度上大概进行得差不多,

好了,话不多说,直接开始今天的第一个小专题:《Direct3D窗口的初始化》


先来第一个文件:头文件d3dUtility.h:

#ifndef __d3dUtilityH__
#define __d3dUtilityH__

#include <d3dx9.h>
#include <string>

namespace d3d
{
	bool InitD3D(
		HINSTANCE hInstance,
		int width, int height,
		bool windowed,
		D3DDEVTYPE deviceType,
		IDirect3DDevice9** device);

	int EnterMsgLoop(
		bool(*ptr_display)(float timeDelta));

	LRESULT CALLBACK WndProc(
		HWND hwnd,
		UINT msg,
		WPARAM wParam,
		LPARAM lParam);

	template<class T>void Release(T t)
	{
		if(t)
		{
			t->Release();
			t = 0;
		}
	}

	template<class T>void Delete(T t)
	{
		if(t)
		{
			delete t;
			t = 0;
		}
	}
}

#endif


现在来第二个文件:d3dUtility.cpp

#include "d3dUtility.h"

bool d3d::InitD3D(
		HINSTANCE hInstance,
		int width, int height,
		bool windowed,
		D3DDEVTYPE deviceType,
		IDirect3DDevice9** device)
{
	WNDCLASS wc;

	wc.style         = CS_HREDRAW | CS_VREDRAW;
	wc.lpfnWndProc   = (WNDPROC)d3d::WndProc; 
	wc.cbClsExtra    = 0;
	wc.cbWndExtra    = 0;
	wc.hInstance     = hInstance;
	wc.hIcon         = LoadIcon(0, IDI_APPLICATION);
	wc.hCursor       = LoadCursor(0, IDC_ARROW);
	wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
	wc.lpszMenuName  = 0;
	wc.lpszClassName = "Direct3D9App";

	if(!RegisterClass(&wc))
	{
		::MessageBox(0,"RegisterClass-FAILED",0,0);
		return false;
	}

	HWND hwnd = 0;
	hwnd = ::CreateWindow("Direct3D9App", "Direct3D9App", 
		WS_EX_TOPMOST,
		0, 0, width, height,
		0 /*parent hwnd*/, 0 /* menu */, hInstance, 0 /*extra*/);

	if(!hwnd)
	{
		::MessageBox(0,"CreateWindow-FAILED",0,0);
		return false;
	}

	::ShowWindow(hwnd,SW_SHOW);
	::UpdateWindow(hwnd);

	HRESULT hr = 0;

	
	IDirect3D9* d3d9 = 0;
    d3d9 = Direct3DCreate9(D3D_SDK_VERSION);

    if( !d3d9 )
	{
		::MessageBox(0, "Direct3DCreate9() - FAILED", 0, 0);
		return false;
	}

	// Step 2: Check for hardware vp.

	D3DCAPS9 caps;
	d3d9->GetDeviceCaps(D3DADAPTER_DEFAULT, deviceType, &caps);

	int vp = 0;
	if( caps.DevCaps & D3DDEVCAPS_HWTRANSFORMANDLIGHT )
		vp = D3DCREATE_HARDWARE_VERTEXPROCESSING;
	else
		vp = D3DCREATE_SOFTWARE_VERTEXPROCESSING;

	D3DPRESENT_PARAMETERS d3dpp;
	d3dpp.BackBufferWidth            = width;
	d3dpp.BackBufferHeight           = height;
	d3dpp.BackBufferFormat           = D3DFMT_A8R8G8B8;
	d3dpp.BackBufferCount            = 1;
	d3dpp.MultiSampleType            = D3DMULTISAMPLE_NONE;
	d3dpp.MultiSampleQuality         = 0;
	d3dpp.SwapEffect                 = D3DSWAPEFFECT_DISCARD; 
	d3dpp.hDeviceWindow              = hwnd;
	d3dpp.Windowed                   = windowed;
	d3dpp.EnableAutoDepthStencil     = true; 
	d3dpp.AutoDepthStencilFormat     = D3DFMT_D24S8;
	d3dpp.Flags                      = 0;
	d3dpp.FullScreen_RefreshRateInHz = D3DPRESENT_RATE_DEFAULT;
	d3dpp.PresentationInterval       = D3DPRESENT_INTERVAL_IMMEDIATE;

hr = d3d9->CreateDevice(
		D3DADAPTER_DEFAULT, // primary adapter
		deviceType,         // device type
		hwnd,               // window associated with device
		vp,                 // vertex processing
	    &d3dpp,             // present parameters
	    device);            // return created device

	if( FAILED(hr) )
	{
		// try again using a 16-bit depth buffer
		d3dpp.AutoDepthStencilFormat = D3DFMT_D16;
		
		hr = d3d9->CreateDevice(
			D3DADAPTER_DEFAULT,
			deviceType,
			hwnd,
			vp,
			&d3dpp,
			device);

		if( FAILED(hr) )
		{
			d3d9->Release(); // done with d3d9 object
			::MessageBox(0, "CreateDevice() - FAILED", 0, 0);
			return false;
		}
	}

	d3d9->Release(); // done with d3d9 object
	
	return true;


}


int d3d::EnterMsgLoop( bool (*ptr_display)(float timeDelta) )
{
	MSG msg;
	::ZeroMemory(&msg, sizeof(MSG));

	static float lastTime = (float)timeGetTime(); 

	while(msg.message != WM_QUIT)
	{
		if(::PeekMessage(&msg, 0, 0, 0, PM_REMOVE))
		{
			::TranslateMessage(&msg);
			::DispatchMessage(&msg);
		}
		else
        {	
			float currTime  = (float)timeGetTime();
			float timeDelta = (currTime - lastTime)*0.001f;

			ptr_display(timeDelta);

			lastTime = currTime;
        }
    }
    return msg.wParam;
}


ok、现在来第三个文件:Init.cpp


//
// 
// File: d3dinit.cpp
// 
// Author: Frank Luna (C) All Rights Reserved
//
// System: AMD Athlon 1800+ XP, 512 DDR, Geforce 3, Windows XP, MSVC++ 7.0 
//
// Desc: Demonstrates how to initialize Direct3D, how to use the book's framework
//       functions, and how to clear the screen to black.  Note that the Direct3D
//       initialization code is in the d3dUtility.h/.cpp files.
//          
//

#include "d3dUtility.h"

//
// Globals
//

IDirect3DDevice9* Device = 0; 

//
// Framework Functions
//

bool Setup()
{
	// Nothing to setup in this sample.

	return true;
}

void Cleanup()
{
	// Nothing to cleanup in this sample.
}

bool Display(float timeDelta)
{
	if( Device ) // Only use Device methods if we have a valid device.
	{
		// Instruct the device to set each pixel on the back buffer black -
		// D3DCLEAR_TARGET: 0x00000000 (black) - and to set each pixel on
		// the depth buffer to a value of 1.0 - D3DCLEAR_ZBUFFER: 1.0f.
		Device->Clear(0, 0, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, 0x00000000, 1.0f, 0);

		// Swap the back and front buffers.
		Device->Present(0, 0, 0, 0);
	}
	return true;
}

//
// WndProc
//
LRESULT CALLBACK d3d::WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
	switch( msg )
	{
	case WM_DESTROY:
		::PostQuitMessage(0);
		break;
		
	case WM_KEYDOWN:
		if( wParam == VK_ESCAPE )
			::DestroyWindow(hwnd);
		break;
	}
	return ::DefWindowProc(hwnd, msg, wParam, lParam);
}

//
// WinMain
//
int WINAPI WinMain(HINSTANCE hinstance,
				   HINSTANCE prevInstance, 
				   PSTR cmdLine,
				   int showCmd)
{
	if(!d3d::InitD3D(hinstance,
		640, 480, true, D3DDEVTYPE_HAL, &Device))
	{
		::MessageBox(0, "InitD3D() - FAILED", 0, 0);
		return 0;
	}
		
	if(!Setup())
	{
		::MessageBox(0, "Setup() - FAILED", 0, 0);
		return 0;
	}

	d3d::EnterMsgLoop( Display );

	Cleanup();

	Device->Release();

	return 0;
}


ok、现在基本完毕















  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
下面是使用DirectX 12渲染超大图片的基本代码: ```cpp #include <d3d12.h> #include <dxgi1_4.h> #include <DirectXMath.h> #include <DirectXTex.h> #include <wrl/client.h> using Microsoft::WRL::ComPtr; // 渲染窗口大小 const UINT WindowWidth = 800; const UINT WindowHeight = 600; // 超大图片大小 const UINT ImageWidth = 10000; const UINT ImageHeight = 10000; // 超大图片文件名 const wchar_t* ImageFilename = L"UltraLargeImage.jpg"; // 全局变量 ComPtr<ID3D12Device> g_Device; ComPtr<IDXGISwapChain3> g_SwapChain; ComPtr<ID3D12CommandQueue> g_CommandQueue; ComPtr<ID3D12CommandAllocator> g_CommandAllocator; ComPtr<ID3D12GraphicsCommandList> g_CommandList; ComPtr<ID3D12Resource> g_BackBuffer[2]; ComPtr<ID3D12Fence> g_Fence; UINT64 g_FenceValue = 0; HANDLE g_FenceEvent; // 初始化设备、命令队列等 bool InitializeDevice() { // 创建设备 HRESULT hr = D3D12CreateDevice(nullptr, D3D_FEATURE_LEVEL_11_0, IID_PPV_ARGS(&g_Device)); if (FAILED(hr)) return false; // 创建命令队列 D3D12_COMMAND_QUEUE_DESC commandQueueDesc = {}; commandQueueDesc.Type = D3D12_COMMAND_LIST_TYPE_DIRECT; commandQueueDesc.Priority = D3D12_COMMAND_QUEUE_PRIORITY_NORMAL; commandQueueDesc.Flags = D3D12_COMMAND_QUEUE_FLAG_NONE; commandQueueDesc.NodeMask = 0; hr = g_Device->CreateCommandQueue(&commandQueueDesc, IID_PPV_ARGS(&g_CommandQueue)); if (FAILED(hr)) return false; // 创建交换链 DXGI_SWAP_CHAIN_DESC1 swapChainDesc = {}; swapChainDesc.Width = WindowWidth; swapChainDesc.Height = WindowHeight; swapChainDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; swapChainDesc.Stereo = FALSE; swapChainDesc.SampleDesc.Count = 1; swapChainDesc.SampleDesc.Quality = 0; swapChainDesc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; swapChainDesc.BufferCount = 2; swapChainDesc.Scaling = DXGI_SCALING_STRETCH; swapChainDesc.SwapEffect = DXGI_SWAP_EFFECT_FLIP_DISCARD; swapChainDesc.AlphaMode = DXGI_ALPHA_MODE_UNSPECIFIED; swapChainDesc.Flags = DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH; ComPtr<IDXGIFactory4> dxgiFactory; hr = CreateDXGIFactory1(IID_PPV_ARGS(&dxgiFactory)); if (FAILED(hr)) return false; hr = dxgiFactory->CreateSwapChainForHwnd(g_CommandQueue.Get(), GetForegroundWindow(), &swapChainDesc, nullptr, nullptr, &g_SwapChain); if (FAILED(hr)) return false; // 创建命令分配器、命令列表 hr = g_Device->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&g_CommandAllocator)); if (FAILED(hr)) return false; hr = g_Device->CreateCommandList(0, D3D12_COMMAND_LIST_TYPE_DIRECT, g_CommandAllocator.Get(), nullptr, IID_PPV_ARGS(&g_CommandList)); if (FAILED(hr)) return false; // 创建RTV CD3DX12_CPU_DESCRIPTOR_HANDLE rtvHandle(g_Device->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_RTV), 0); for (UINT i = 0; i < 2; i++) { hr = g_SwapChain->GetBuffer(i, IID_PPV_ARGS(&g_BackBuffer[i])); if (FAILED(hr)) return false; g_Device->CreateRenderTargetView(g_BackBuffer[i].Get(), nullptr, rtvHandle); rtvHandle.Offset(1, g_Device->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_RTV)); } // 创建围栏 hr = g_Device->CreateFence(0, D3D12_FENCE_FLAG_NONE, IID_PPV_ARGS(&g_Fence)); if (FAILED(hr)) return false; g_FenceEvent = CreateEvent(nullptr, FALSE, FALSE, nullptr); if (g_FenceEvent == nullptr) return false; return true; } // 加载超大图片 ComPtr<ID3D12Resource> LoadImage() { // 加载图片 std::unique_ptr<DirectX::ScratchImage> image(new DirectX::ScratchImage()); HRESULT hr = DirectX::LoadFromWICFile(ImageFilename, DirectX::WIC_FLAGS_NONE, nullptr, *image); if (FAILED(hr)) return nullptr; // 创建纹理资源 ComPtr<ID3D12Resource> texture; D3D12_RESOURCE_DESC textureDesc = {}; textureDesc.Dimension = D3D12_RESOURCE_DIMENSION_TEXTURE2D; textureDesc.Alignment = 0; textureDesc.Width = ImageWidth; textureDesc.Height = ImageHeight; textureDesc.DepthOrArraySize = 1; textureDesc.MipLevels = 1; textureDesc.Format = image->GetMetadata().format; textureDesc.SampleDesc.Count = 1; textureDesc.SampleDesc.Quality = 0; textureDesc.Layout = D3D12_TEXTURE_LAYOUT_UNKNOWN; textureDesc.Flags = D3D12_RESOURCE_FLAG_NONE; hr = g_Device->CreateCommittedResource(&CD3DX12_HEAP_PROPERTIES(D3D12_HEAP_TYPE_DEFAULT), D3D12_HEAP_FLAG_NONE, &textureDesc, D3D12_RESOURCE_STATE_COPY_DEST, nullptr, IID_PPV_ARGS(&texture)); if (FAILED(hr)) return nullptr; // 上传图片数据 const DirectX::Image* imageData = image->GetImage(0, 0, 0); D3D12_SUBRESOURCE_DATA textureData = {}; textureData.pData = imageData->pixels; textureData.RowPitch = imageData->rowPitch; textureData.SlicePitch = imageData->slicePitch; UpdateSubresources(g_CommandList.Get(), texture.Get(), g_BackBuffer[0].Get(), 0, 0, 1, &textureData); // 转换资源状态 CD3DX12_RESOURCE_BARRIER barrier = CD3DX12_RESOURCE_BARRIER::Transition(texture.Get(), D3D12_RESOURCE_STATE_COPY_DEST, D3D12_RESOURCE_STATE_PIXEL_SHADER_RESOURCE); g_CommandList->ResourceBarrier(1, &barrier); return texture; } // 渲染函数 void Render() { // 等待上一帧完成 const UINT currentBackBufferIndex = g_SwapChain->GetCurrentBackBufferIndex(); if (g_Fence->GetCompletedValue() < g_FenceValue) { hr = g_Fence->SetEventOnCompletion(g_FenceValue, g_FenceEvent); if (FAILED(hr)) return; WaitForSingleObject(g_FenceEvent, INFINITE); } // 开始渲染 g_CommandAllocator->Reset(); g_CommandList->Reset(g_CommandAllocator.Get(), nullptr); // 设置渲染目标 CD3DX12_RESOURCE_BARRIER barrierToRenderTarget = CD3DX12_RESOURCE_BARRIER::Transition(g_BackBuffer[currentBackBufferIndex].Get(), D3D12_RESOURCE_STATE_PRESENT, D3D12_RESOURCE_STATE_RENDER_TARGET); g_CommandList->ResourceBarrier(1, &barrierToRenderTarget); CD3DX12_CPU_DESCRIPTOR_HANDLE rtvHandle(g_Device->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_RTV), currentBackBufferIndex); g_CommandList->OMSetRenderTargets(1, &rtvHandle, FALSE, nullptr); // 清空渲染目标 const float clearColor[] = { 0.0f, 0.0f, 0.0f, 1.0f }; g_CommandList->ClearRenderTargetView(rtvHandle, clearColor, 0, nullptr); // 绘制超大图片 // ... // 结束渲染 CD3DX12_RESOURCE_BARRIER barrierToPresent = CD3DX12_RESOURCE_BARRIER::Transition(g_BackBuffer[currentBackBufferIndex].Get(), D3D12_RESOURCE_STATE_RENDER_TARGET, D3D12_RESOURCE_STATE_PRESENT); g_CommandList->ResourceBarrier(1, &barrierToPresent); g_CommandList->Close(); // 提交命令列表 ID3D12CommandList* commandLists[] = { g_CommandList.Get() }; g_CommandQueue->ExecuteCommandLists(_countof(commandLists), commandLists); // 信号围栏 g_FenceValue++; hr = g_CommandQueue->Signal(g_Fence.Get(), g_FenceValue); if (FAILED(hr)) return; // 呈现画面 g_SwapChain->Present(1, 0); } // 程序入口 int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { // 初始化窗口等 // ... // 初始化设备、命令队列等 if (!InitializeDevice()) return -1; // 加载超大图片 ComPtr<ID3D12Resource> texture = LoadImage(); if (texture == nullptr) return -1; // 进入消息循环 // ... } ``` 其中,`LoadImage`函数用于加载超大图片,返回一个纹理资源。在绘制超大图片时,可以使用多次绘制来分批渲染,或者使用视口和裁剪矩形来渲染部分区域,以避免一次渲染过多数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值