如何套用效果
注意
針對Windows 10上的應用程式,我們建議使用 Windows.UI.Composition API,而不是 DirectComposition。 如需詳細資訊,請參閱 使用視覺層將傳統型應用程式現代化。
本主題示範如何使用 Microsoft DirectComposition 將效果和 3D 轉換套用至視覺效果。 本主題中的範例會變更視覺效果的不透明度,並將它繞著位於視覺效果中央的垂直軸旋轉。 若要深入瞭解 DirectComposition 支援的其他效果,請參閱 效果。
您所需了解的事情
技術
必要條件
- C/C++
- Microsoft Win32
- 元件物件模型 (COM)
指示
步驟 1:初始化 DirectComposition 物件
- 建立裝置物件和組合目標物件。
- 建立視覺效果、設定其內容,並將其新增至視覺化樹狀結構。
如需詳細資訊,請參閱 如何初始化 DirectComposition。
步驟 2:建立 3D 旋轉轉換物件、效果群組物件和動畫物件
使用 IDCompositionDevice::CreateRotateTransform3D 方法來建立 3D 旋轉轉換物件,並使用 CreateEffectGroup 方法來建立效果群組物件。 此範例也會使用 CreateAnimation 方法來建立動畫物件,以動畫顯示 3D 旋轉轉換。 若要深入瞭解如何套用動畫,請參閱 如何套用動畫。
HRESULT hr = S_OK;
IDCompositionAnimation *pAnimation = nullptr;
IDCompositionRotateTransform3D *pRotate3D = nullptr;
IDCompositionEffectGroup *pEffectGroup = nullptr;
// Create a 3D rotate transform object.
hr = m_pDevice->CreateRotateTransform3D(&pRotate3D);
if (SUCCEEDED(hr))
{
// Create an effect group object.
hr = m_pDevice->CreateEffectGroup(&pEffectGroup);
}
if (SUCCEEDED(hr))
{
// Create an animation object.
hr = m_pDevice->CreateAnimation(&pAnimation);
}
步驟 3:定義動畫函式
使用 IDCompositionAnimation 物件的方法來定義動畫函式。
下列範例會定義簡單的動畫函式。 套用至物件屬性時,動畫函式會累加地將屬性值從 0 變更為一秒的 degrees 引數值。
if (SUCCEEDED(hr))
{
// Define the animation function.
pAnimation->AddCubic(0.0f, 0.0f, degrees, 0.0f, 0.0f);
pAnimation->End(1.0f, degrees);
步驟 4:設定 3D 旋轉轉換的屬性
- 呼叫 IDCompositionRotateTransform3D::SetAngle 方法,將動畫函式套用至 3D 旋轉轉換的 Angle 屬性。
- 呼叫 IDCompositionRotateTransform3D::SetAxisX、 SetAxisY和 SetAxisZ 方法,以設定 3D 旋轉轉換的旋轉軸。
- 呼叫 IDCompositionRotateTransform3D::SetCenterX 和 SetCenterY 方法,以設定 3D 旋轉轉換的旋轉中心。
下列範例會設定 3D 旋轉轉換,以繞著位於視覺效果中央的垂直軸旋轉視覺效果。 m_bitmapWidth和m_bitmapHeight參數是點陣圖的寬度和高度,以圖元為單位。
// Set the properties of the rotate transform object.
//
// Apply the animation object to the Angle property so that
// the visual will appear to spin around an axis.
pRotate3D->SetAngle(pAnimation);
// Set a vertical axis through the center of the visual's bitmap.
pRotate3D->SetAxisX(0.0f);
pRotate3D->SetAxisY(1.0f);
pRotate3D->SetAxisZ(0.0f);
// Set the center of rotation to the center of the visual's bitmap.
pRotate3D->SetCenterX(m_bitmapWidth / 2.0f);
pRotate3D->SetCenterY(m_bitmapHeight / 2.0f);
}
步驟 5:設定效果群組物件的屬性
- 呼叫 IDCompositionEffectGroup::SetTransform3D 方法,將 3D 旋轉轉換物件套用至效果群組物件的 Transform3D 屬性。
- 呼叫 IDCompositionEffectGroup::SetOpacity來設定效果群組物件的 Opacity 屬性。
if (SUCCEEDED(hr))
{
// Apply the rotate transform object to the Tranform3D property
// of the effect group object.
hr = pEffectGroup->SetTransform3D(pRotate3D);
}
if (SUCCEEDED(hr))
{
// Set the Opacity of the effect group object.
hr = pEffectGroup->SetOpacity(opacity);
}
步驟 6:將效果群組物件套用至視覺效果的 Effect 屬性
呼叫 IDCompositionVisual::SetEffect 方法,將效果群組物件套用至視覺效果。
if (SUCCEEDED(hr))
{
// Apply the effect group object to the Effect property of the visual.
hr = pVisual->SetEffect(pEffectGroup);
}
步驟 7:認可組合
呼叫 IDCompositionDevice::Commit 方法,將命令批次認可至 DirectComposition 進行處理。 產生的組合會出現在目標視窗中。
if (SUCCEEDED(hr))
{
// Commit the visual to DirectComposition.
hr = m_pDevice->Commit();
}
步驟 8:釋放 DirectComposition 物件
當您不再需要動畫物件時,請務必釋放動畫物件、3D 旋轉轉換物件和效果群組物件。 下列範例會呼叫應用程式定義的 SafeRelease 宏來釋放物件。
// Release the DirectComposition objects.
SafeRelease(&pAnimation);
SafeRelease(&pRotate3D);
SafeRelease(&pEffectGroup);
也請記得在應用程式結束之前釋放裝置物件、組合目標物件和視覺效果。 如需詳細資訊,請參閱 如何初始化 DirectComposition。
完整範例
//
// ApplyEffects.h
//
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
// ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
// PARTICULAR PURPOSE.
//
// Copyright (c) Microsoft Corporation. All rights reserved
#pragma once
// Modify the following definitions if you need to target a platform prior to the ones specified below.
#ifndef WINVER // Allow use of features specific to Windows 7 or later.
#define WINVER 0x0700 // Change this to the appropriate value to target other versions of Windows.
#endif
#ifndef _WIN32_WINNT // Allow use of features specific to Windows 7 or later.
#define _WIN32_WINNT 0x0700 // Change this to the appropriate value to target other versions of Windows.
#endif
#ifndef UNICODE
#define UNICODE
#endif
#define WIN32_LEAN_AND_MEAN // Exclude rarely-used items from Windows headers
// Windows Header Files:
#include <windows.h>
// C RunTime Header Files
#include <math.h>
// DirectComposition and Direct3D Header Files
#include <dcomp.h>
#include <d3d11.h>
/******************************************************************
* *
* Macros *
* *
******************************************************************/
template<class Interface>
inline void
SafeRelease(
Interface **ppInterfaceToRelease
)
{
if (*ppInterfaceToRelease != NULL)
{
(*ppInterfaceToRelease)->Release();
(*ppInterfaceToRelease) = NULL;
}
}
#ifndef HINST_THISCOMPONENT
EXTERN_C IMAGE_DOS_HEADER __ImageBase;
#define HINST_THISCOMPONENT ((HINSTANCE)&__ImageBase)
#endif
/******************************************************************
* *
* DemoApp *
* *
******************************************************************/
class DemoApp
{
public:
DemoApp();
~DemoApp();
HRESULT Initialize();
void RunMessageLoop();
private:
HRESULT InitializeDirectCompositionDevice();
HRESULT CreateResources();
void DiscardResources();
HRESULT OnPaint();
HRESULT OnClientClick(int xPos, int yPos);
HRESULT OnMouseMove(int xPos, int yPos);
HRESULT LoadResourceGDIBitmap(
PCWSTR resourceName,
HBITMAP &hbmp
);
HRESULT MyCreateGDIRenderedDCompSurface(HBITMAP hBitmap,
IDCompositionSurface **ppSurface);
HRESULT SetVisualOpacity(IDCompositionVisual *pVisual, float opacity);
HRESULT RotateVisual(IDCompositionVisual *pVisual,float degrees);
static LRESULT CALLBACK WndProc(
HWND hWnd,
UINT message,
WPARAM wParam,
LPARAM lParam
);
private:
HWND m_hwnd;
HBITMAP m_hBitmap;
int m_bitmapWidth;
int m_bitmapHeight;
ID3D11Device *m_pD3D11Device;
IDCompositionDevice *m_pDevice;
IDCompositionTarget *m_pCompTarget;
IDCompositionVisual *m_pVisual;
};
//
// ApplyEffects.cpp
//
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF
// ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
// PARTICULAR PURPOSE.
//
// Copyright (c) Microsoft Corporation. All rights reserved
// Instructions: Hover over the image to see the opacity change. Click
// the image to apply a 3D rotation.
#include "ApplyEffects.h"
#define OFFSET_X 20
#define OFFSET_Y 20
#define TRANSPARENT 0.5
#define OPAQUE 1.0
/******************************************************************
* *
* The application entry point. *
* *
******************************************************************/
int WINAPI WinMain(
HINSTANCE /* hInstance */,
HINSTANCE /* hPrevInstance */,
LPSTR /* lpCmdLine */,
int /* nCmdShow */
)
{
// Ignore the return value because we want to run the program even in the
// unlikely event that HeapSetInformation fails.
HeapSetInformation(NULL, HeapEnableTerminationOnCorruption, NULL, 0);
if (SUCCEEDED(CoInitialize(NULL)))
{
{
DemoApp app;
if (SUCCEEDED(app.Initialize()))
{
app.RunMessageLoop();
}
}
CoUninitialize();
}
return 0;
}
/******************************************************************
* *
* DemoApp::DemoApp constructor *
* *
* Initialize member data. *
* *
******************************************************************/
DemoApp::DemoApp() :
m_hwnd(NULL),
m_hBitmap(NULL),
m_pDevice(nullptr),
m_pCompTarget(nullptr),
m_pD3D11Device(nullptr),
m_pVisual(nullptr),
m_bitmapHeight(0),
m_bitmapWidth(0)
{
}
/******************************************************************
* *
* Release resources. *
* *
******************************************************************/
DemoApp::~DemoApp()
{
SafeRelease(&m_pDevice);
SafeRelease(&m_pCompTarget);
SafeRelease(&m_pD3D11Device);
SafeRelease(&m_pVisual);
}
/*******************************************************************
* *
* Create the application window. *
* *
*******************************************************************/
HRESULT DemoApp::Initialize()
{
HRESULT hr;
// Register the window class.
WNDCLASSEX wcex = { sizeof(WNDCLASSEX) };
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = DemoApp::WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = sizeof(LONG_PTR);
wcex.hInstance = HINST_THISCOMPONENT;
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);;
wcex.lpszMenuName = NULL;
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
wcex.lpszClassName = L"DirectCompDemoApp";
RegisterClassEx(&wcex);
// Create the application window.
//
// Because the CreateWindow function takes its size in pixels, we
// obtain the system DPI and use it to scale the window size.
int dpiX = 0;
int dpiY = 0;
HDC hdc = GetDC(NULL);
if (hdc)
{
dpiX = GetDeviceCaps(hdc, LOGPIXELSX);
dpiY = GetDeviceCaps(hdc, LOGPIXELSY);
ReleaseDC(NULL, hdc);
}
m_hwnd = CreateWindow(
L"DirectCompDemoApp",
L"DirectComposition Demo Application",
WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT,
CW_USEDEFAULT,
static_cast<UINT>(ceil(640.f * dpiX / 96.f)),
static_cast<UINT>(ceil(480.f * dpiY / 96.f)),
NULL,
NULL,
HINST_THISCOMPONENT,
this
);
hr = m_hwnd ? S_OK : E_FAIL;
if (SUCCEEDED(hr))
{
// Initialize DirectComposition resources, such as the
// device object and composition target object.
hr = InitializeDirectCompositionDevice();
}
if (SUCCEEDED(hr))
{
hr = CreateResources();
}
if (SUCCEEDED(hr))
{
ShowWindow(m_hwnd, SW_SHOWNORMAL);
UpdateWindow(m_hwnd);
}
return hr;
}
/******************************************************************
* *
* This method creates the DirectComposition device object and *
* and the composition target object. These objects endure for *
* the lifetime of the application. *
* *
******************************************************************/
HRESULT DemoApp::InitializeDirectCompositionDevice()
{
HRESULT hr = S_OK;
D3D_FEATURE_LEVEL featureLevelSupported;
// Create the D3D device object.
D3D11CreateDevice(
nullptr,
D3D_DRIVER_TYPE_HARDWARE,
NULL,
D3D11_CREATE_DEVICE_BGRA_SUPPORT,
NULL,
0,
D3D11_SDK_VERSION,
&m_pD3D11Device,
&featureLevelSupported,
NULL);
IDXGIDevice *pDXGIDevice = nullptr;
hr = (m_pD3D11Device == nullptr) ? E_UNEXPECTED : S_OK;
if (SUCCEEDED(hr))
{
// Create the DXGI device used to create bitmap surfaces.
hr = m_pD3D11Device->QueryInterface(&pDXGIDevice);
}
if (SUCCEEDED(hr))
{
// Create the DirectComposition device object.
hr = DCompositionCreateDevice(pDXGIDevice,
__uuidof(IDCompositionDevice),
reinterpret_cast<void **>(&m_pDevice));
}
if (SUCCEEDED(hr))
{
// Create the composition target object.
hr = m_pDevice->CreateTargetForHwnd(m_hwnd, TRUE, &m_pCompTarget);
}
SafeRelease(&pDXGIDevice);
return hr;
}
/******************************************************************
* *
* This method creates the GDI bitmap that the application gives *
* to DirectComposition to be composed. *
* *
******************************************************************/
HRESULT DemoApp::CreateResources()
{
HRESULT hr = S_OK;
hr = LoadResourceGDIBitmap(L"Penguins", m_hBitmap);
return hr;
}
/******************************************************************
* *
* Discard device-specific resources. *
* *
******************************************************************/
void DemoApp::DiscardResources()
{
DeleteObject(m_hBitmap);
}
/******************************************************************
* *
* The main window's message loop. *
* *
******************************************************************/
void DemoApp::RunMessageLoop()
{
MSG msg;
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
/******************************************************************
* *
* Called when the application's main window is painted. This *
* method builds a simple visual tree and passes it to *
* DirectComposition. *
* *
******************************************************************/
HRESULT DemoApp::OnPaint()
{
HRESULT hr = S_OK;
IDCompositionSurface *pSurface = nullptr;
// Create a visual object.
hr = m_pDevice->CreateVisual(&m_pVisual);
if (SUCCEEDED(hr))
{
// Create a composition surface and render a GDI bitmap
// to the surface.
hr = MyCreateGDIRenderedDCompSurface(m_hBitmap, &pSurface);
}
if (SUCCEEDED(hr))
{
// Set the bitmap content of the visual.
hr = m_pVisual->SetContent(pSurface);
}
if (SUCCEEDED(hr))
{
// Set the horizontal and vertical position of the visual relative
// to the upper-left corner of the composition target window.
m_pVisual->SetOffsetX(OFFSET_X);
m_pVisual->SetOffsetY(OFFSET_Y);
hr = SetVisualOpacity(m_pVisual, TRANSPARENT);
}
if (SUCCEEDED(hr))
{
// Set the visual to be the root of the visual tree.
hr = m_pCompTarget->SetRoot(m_pVisual);
}
if (SUCCEEDED(hr))
{
// Commit the visual to be composed and displayed.
hr = m_pDevice->Commit();
}
return hr;
}
/******************************************************************
* *
* Called when the mouse moves in the main window's client area. *
* This method determines whether the mouse cursor is over the *
* visual, and calls an application-defined function to set the *
* visual's opacity. *
* *
******************************************************************/
HRESULT DemoApp::OnMouseMove(int xPos, int yPos)
{
HRESULT hr = S_OK;
static BOOL fOverImage = FALSE;
// Determine whether the cursor is over the visual.
if ((xPos >= OFFSET_X && xPos <= (OFFSET_X + m_bitmapWidth))
&& (yPos >= OFFSET_Y && yPos <= (OFFSET_Y + m_bitmapHeight)))
{
if (!fOverImage)
{
// The cursor has moved over the visual, so make the visual
// 100% opaque.
hr = SetVisualOpacity(m_pVisual, OPAQUE);
fOverImage = TRUE;
}
}
else if (fOverImage)
{
// The cursor has moved off the visual, so make the visual
// 50% opaque.
hr = SetVisualOpacity(m_pVisual, TRANSPARENT);
fOverImage = FALSE;
}
return hr;
}
/******************************************************************
* *
* Changes the opacity of a visual. *
* *
******************************************************************/
HRESULT DemoApp::SetVisualOpacity(IDCompositionVisual *pVisual, float opacity)
{
HRESULT hr = S_OK;
IDCompositionEffectGroup *pEffectGroup = nullptr;
// Validate the input arguments.
if (pVisual == NULL || (opacity > 1.0f || opacity < 0.0f))
return E_INVALIDARG;
// Create an effect group object.
hr = m_pDevice->CreateEffectGroup(&pEffectGroup);
if (SUCCEEDED(hr))
{
// Set the Opacity of the effect group object.
hr = pEffectGroup->SetOpacity(opacity);
}
if (SUCCEEDED(hr))
{
// Apply the effect group object to the Effect property of the visual.
hr = m_pVisual->SetEffect(pEffectGroup);
}
if (SUCCEEDED(hr))
{
// Commit the visual to DirectComposition.
hr = m_pDevice->Commit();
}
// Free the effect group object.
SafeRelease(&pEffectGroup);
return hr;
}
/******************************************************************
* *
* Called when the user clicks in the main window's client area. *
* This method determines whether the mouse cursor is over the *
* visual and calls an application-defined function to rotate the *
* visual. *
* *
******************************************************************/
HRESULT DemoApp::OnClientClick(int xPos, int yPos)
{
HRESULT hr = S_OK;
// Determine whether the mouse cursor is over the visual. If so,
// rotate the visual.
if ((xPos >= OFFSET_X && xPos <= (OFFSET_X + m_bitmapWidth))
&& (yPos >= OFFSET_Y && yPos <= (OFFSET_Y + m_bitmapHeight)))
{
hr = RotateVisual(m_pVisual, 360.0f);
}
return hr;
}
/******************************************************************
* *
* Performs an animated 3D rotation of a visual. *
* *
******************************************************************/
HRESULT DemoApp::RotateVisual(IDCompositionVisual *pVisual, float degrees)
{
HRESULT hr = S_OK;
IDCompositionAnimation *pAnimation = nullptr;
IDCompositionRotateTransform3D *pRotate3D = nullptr;
IDCompositionEffectGroup *pEffectGroup = nullptr;
// Validate the input arguments.
if (pVisual == NULL || (degrees > 360.0f || degrees < -360.0f))
return E_INVALIDARG;
// Create a 3D rotate transform object.
hr = m_pDevice->CreateRotateTransform3D(&pRotate3D);
if (SUCCEEDED(hr))
{
// Create an effect group object.
hr = m_pDevice->CreateEffectGroup(&pEffectGroup);
}
if (SUCCEEDED(hr))
{
// Create an animation object.
hr = m_pDevice->CreateAnimation(&pAnimation);
}
if (SUCCEEDED(hr))
{
// Define the animation function.
pAnimation->AddCubic(0.0f, 0.0f, degrees, 0.0f, 0.0f);
pAnimation->End(1.0f, degrees);
// Set the properties of the rotate transform object.
//
// Apply the animation object to the Angle property so that
// the visual will appear to spin around an axis.
pRotate3D->SetAngle(pAnimation);
// Set a vertical axis through the center of the visual's bitmap.
pRotate3D->SetAxisX(0.0f);
pRotate3D->SetAxisY(1.0f);
pRotate3D->SetAxisZ(0.0f);
// Set the center of rotation to the center of the visual's bitmap.
pRotate3D->SetCenterX(m_bitmapWidth / 2.0f);
pRotate3D->SetCenterY(m_bitmapHeight / 2.0f);
}
if (SUCCEEDED(hr))
{
// Apply the rotate transform object to the Tranform3D property
// of the effect group object.
hr = pEffectGroup->SetTransform3D(pRotate3D);
}
if (SUCCEEDED(hr))
{
// Apply the effect group object to the Effect property of the visual.
hr = pVisual->SetEffect(pEffectGroup);
}
if (SUCCEEDED(hr))
{
// Commit the visual to DirectComposition.
hr = m_pDevice->Commit();
}
// Release the DirectComposition objects.
SafeRelease(&pAnimation);
SafeRelease(&pRotate3D);
SafeRelease(&pEffectGroup);
return hr;
}
/******************************************************************
* *
* The window's message handler. *
* *
******************************************************************/
LRESULT CALLBACK DemoApp::WndProc(HWND hwnd, UINT message,
WPARAM wParam, LPARAM lParam)
{
LRESULT result = 0;
if (message == WM_CREATE)
{
LPCREATESTRUCT pcs = (LPCREATESTRUCT)lParam;
DemoApp *pDemoApp = (DemoApp *)pcs->lpCreateParams;
::SetWindowLongPtrW(
hwnd,
GWLP_USERDATA,
PtrToUlong(pDemoApp)
);
result = 1;
}
else
{
DemoApp *pDemoApp = reinterpret_cast<DemoApp *>(static_cast<LONG_PTR>(
::GetWindowLongPtrW(
hwnd,
GWLP_USERDATA
)));
bool wasHandled = false;
if (pDemoApp)
{
switch (message)
{
case WM_LBUTTONDOWN:
{
pDemoApp->OnClientClick(LOWORD(lParam), HIWORD(lParam));
}
wasHandled = true;
result = 0;
break;
case WM_MOUSEMOVE:
{
pDemoApp->OnMouseMove(LOWORD(lParam), HIWORD(lParam));
}
wasHandled = true;
result = 0;
break;
case WM_PAINT:
{
pDemoApp->OnPaint();
ValidateRect(hwnd, NULL);
}
wasHandled = true;
result = 0;
break;
case WM_DISPLAYCHANGE:
{
InvalidateRect(hwnd, NULL, FALSE);
}
wasHandled = true;
result = 0;
break;
case WM_DESTROY:
{
PostQuitMessage(0);
pDemoApp->DiscardResources();
}
wasHandled = true;
result = 1;
break;
}
}
if (!wasHandled)
{
result = DefWindowProc(hwnd, message, wParam, lParam);
}
}
return result;
}
/******************************************************************
* *
* This method loads the specified GDI bitmap from the *
* application resources, creates a new bitmap that is in a *
* format that DirectComposition can use, and copies the contents *
* of the original bitmap to the new bitmap. *
* *
******************************************************************/
HRESULT DemoApp::LoadResourceGDIBitmap(PCWSTR resourceName, HBITMAP &hbmp)
{
hbmp = static_cast<HBITMAP>(LoadImageW(HINST_THISCOMPONENT, resourceName,
IMAGE_BITMAP, 0, 0, LR_DEFAULTCOLOR));
return hbmp ? S_OK : E_FAIL;
}
// CreateGDIRenderedDCompSurface - Creates a DirectComposition surface and
// copies the bitmap to the surface.
//
// Parameters:
// hBitmap - a GDI bitmap.
// ppSurface - the composition surface object.
//
HRESULT DemoApp::MyCreateGDIRenderedDCompSurface(HBITMAP hBitmap,
IDCompositionSurface **ppSurface)
{
HRESULT hr = S_OK;
int bmpSize = 0;
BITMAP bmp = { };
HBITMAP hBitmapOld = NULL;
HDC hSurfaceDC = NULL;
HDC hBitmapDC = NULL;
IDXGISurface1 *pDXGISurface = nullptr;
IDCompositionSurface *pDCSurface = nullptr;
POINT pointOffset = { };
hr = hBitmap ? S_OK : E_FAIL;
if (SUCCEEDED(hr))
{
// Get information about the bitmap.
bmpSize = GetObject(hBitmap, sizeof(BITMAP), &bmp);
}
hr = bmpSize ? S_OK : E_FAIL;
if (SUCCEEDED(hr))
{
// Save the bitmap dimensions.
m_bitmapWidth = bmp.bmWidth;
m_bitmapHeight = bmp.bmHeight;
// Create a DirectComposition-compatible surface that is the same size
// as the bitmap.
hr = m_pDevice->CreateSurface(m_bitmapWidth, m_bitmapHeight,
DXGI_FORMAT_B8G8R8A8_UNORM, DXGI_ALPHA_MODE_IGNORE, &pDCSurface);
}
hr = pDCSurface ? S_OK : E_FAIL;
if (SUCCEEDED(hr))
{
// Begin rendering to the surface.
hr = pDCSurface->BeginDraw(NULL, __uuidof(IDXGISurface1),
reinterpret_cast<void**>(&pDXGISurface), &pointOffset);
}
if (SUCCEEDED(hr))
{
// Get the device context (DC) for the surface.
pDXGISurface->GetDC(FALSE, &hSurfaceDC);
}
hr = hSurfaceDC ? S_OK : E_FAIL;
if (SUCCEEDED(hr))
{
// Create a compatible (DC) and select the surface
// into the DC.
hBitmapDC = CreateCompatibleDC(hSurfaceDC);
if (hBitmapDC != NULL)
{
hBitmapOld = (HBITMAP)SelectObject(hBitmapDC, hBitmap);
BitBlt(hSurfaceDC, pointOffset.x, pointOffset.y,
m_bitmapWidth, m_bitmapHeight, hBitmapDC, 0, 0, SRCCOPY);
if (hBitmapOld)
{
SelectObject(hBitmapDC, hBitmapOld);
}
DeleteDC(hBitmapDC);
}
pDXGISurface->ReleaseDC(NULL);
}
// End the rendering.
pDCSurface->EndDraw();
*ppSurface = pDCSurface;
SafeRelease(&pDXGISurface);
return hr;
}
相關主題