Bốcục của sách
Lời giới thiệu
PHẦN I. NHỮNG KIẾN THỨC NỀN TẢNG
Chương 1. DirectX là gì, tại sao và làm thếnào đểsửdụng nó
Chương 2. Ví dụ đầu tiên sửdụng DirectX
Chương 3. Phông nền, khung hình, hoạt cảnh.
PHẦN II. NHỮNG THÀNH PHẦN CƠ BẢN CỦA MỘT THẾ GIỚI 3D
Chương 4. Những kiến thức cơbản về3D
Chương 5. Ma trận, các phép biến đổi và phép xoay trong không gian
Chương 6. Bảng mầu, vật liệu phủvà ánh sáng trong không gian
Chương 7. Chia nhỏvà làm mịn đối tượng
Chương 8. Vật thể, điểm chèn và các hiệu ứng
PHẦN III. NHỮNG KIẾN THỨC BỔ XUNG
Chương 9.Sửdụng DirectInput
Chương 10.Hiệu ứng âm thanh bằng DirectSound
Chương 11.Xây dựng một dựán mẫu
PHẦN IV. PHỤ LỤC
Phụlục A.Giải đáp các bài tập cuối các chương
Phụlục B.Cách sửdụng CD-ROM
Giải thích các thuật ngữ
Chỉmục các từkhoá
178 trang |
Chia sẻ: maiphuongdc | Lượt xem: 1748 | Lượt tải: 1
Bạn đang xem trước 20 trang tài liệu Ebook Beginning DirectX9, để xem tài liệu hoàn chỉnh bạn click vào nút DOWNLOAD ở trên
ma trận biến đổi. Ma trận thứ
ba là vecto sau khi đã biến đổi.
Bạn có thể thực hiện phép nhân ma trận bằng cách nhân hàng của ma trận này với cột
của ma trận kía.
⎥⎥
⎥⎥
⎦
⎤
⎢⎢
⎢⎢
⎣
⎡
⎥⎥
⎥⎥
⎦
⎤
⎢⎢
⎢⎢
⎣
⎡
16151413
1211109
8765
4321
x
PONM
LKJI
HGFE
DCBA
Lấy hàng của ma trận bên trái nhân với cột của ma trận bên phải. Ví dụ ta lấy hàng 1 của
ma trận trái nhân cột 1 của ma trận phải ta sẽ có:
A × 1 + B × 5 + C × 9 + D × 13
Cách định nghĩa một ma trận trong Direct3D
Direct3D đơn giản hóa việc tạo ma trận thông qua cấu trúc D3DMATRIX như sau:
typedef struct _D3DMATRIX {
union {
struct {
float _11, _12, _13, _14;
float _21, _22, _23, _24;
float _31, _32, _33, _34;
float _41, _42, _43, _44;
};
float m[4][4];
};
} D3DMATRIX;
Sử dụng cấu trúc D3DMATRIX mà Direct3D cung cấp, bạn còn được cung cấp thêm nhiều
hàm tiện lợi khác khi thao tác trên ma trận ví dụ như hàm khởi tạo ma trận.
D3DX làm cho ma trận đơn giản hơn
Phần trên, ta đã được giới thiệu về cấu trúc D3DMATRIX mà Direct3D cung cấp. Nó giúp
đơn giản hóa việc định nghĩa và quản lý ma trận nhưng vẫn bắt bạn phải tự thực hiện tính
toán trên đó, tuy vậy thư viện D3DX sẽ giúp bạn điều này.
Thư viện D3DX cũng có một cấu trúc là D3DXMATRIX. Các thành phần trong
D3DXMATRIX cũng giống như của D3DMATRIX, nhưng D3DXMATRIX cung cấp thêm
Chú ý:
phép nhân ma trận không có tính hoán vị, nếu ta đổi chỗ hai ma trận này với nhau thì ta có
thể sẽ có một ma trận kết quả khác. Vì vậy, thứ tự của các ma trận là yếu tố rất quan trọng
trong phép nhân ma trận.
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
85
một vài tiện ích khác nữa. Nó cung cấp một vài hàm cho phép bạn thực hiện tính toán, so
sánh trên đó.
Cấu trúc D3DXMATRIX được định nghĩa như sau:
typedef struct D3DXMATRIX : public D3DMATRIX {
public:
D3DXMATRIX() {};
D3DXMATRIX( CONST FLOAT * );
D3DXMATRIX( CONST D3DMATRIX& );
D3DXMATRIX( FLOAT _11, FLOAT _12, FLOAT _13, FLOAT _14,
FLOAT _21, FLOAT _22, FLOAT _23, FLOAT _24,
FLOAT _31, FLOAT _32, FLOAT _33, FLOAT _34,
FLOAT _41, FLOAT _42, FLOAT _43, FLOAT _44 );
// access grants
FLOAT& operator () ( UINT Row, UINT Col );
FLOAT operator () ( UINT Row, UINT Col ) const;
// casting operators
operator FLOAT* ();
operator CONST FLOAT* () const;
// assignment operators
D3DXMATRIX& operator *= ( CONST D3DXMATRIX& );
D3DXMATRIX& operator += ( CONST D3DXMATRIX& );
D3DXMATRIX& operator -= ( CONST D3DXMATRIX& );
D3DXMATRIX& operator *= ( FLOAT );
D3DXMATRIX& operator /= ( FLOAT );
// unary operators
D3DXMATRIX operator + () const;
D3DXMATRIX operator - () const;
// binary operators
D3DXMATRIX operator * ( CONST D3DXMATRIX& ) const;
D3DXMATRIX operator + ( CONST D3DXMATRIX& ) const;
D3DXMATRIX operator - ( CONST D3DXMATRIX& ) const;
D3DXMATRIX operator * ( FLOAT ) const;
D3DXMATRIX operator / ( FLOAT ) const;
friend D3DXMATRIX operator * ( FLOAT, CONST D3DXMATRIX& );
BOOL operator == ( CONST D3DXMATRIX& ) const;
BOOL operator != ( CONST D3DXMATRIX& ) const;
} D3DXMATRIX, *LPD3DXMATRIX;
Điều bạn có thể thấy đầu tiên là D3DXMATRIX là một cấu trúc kế thừa từ D3DMATRIX
và bổ xung thêm một số hàm khiến cho nó giống như một lớp (class) ở trong C++. Theo
cách định nghĩa này, bạn chỉ có thể truy cập tới nó thông qua C++, và nó chỉ được sử
dụng như là một lớp (class) thật sự với các thành phần public.
Nhìn qua cấu trúc này, bạn có thể thấy nó viết chồng (overload) rất nhiều toán tử dùng
cho việc tính toán trên ma trận. Bởi vì, cấu trúc D3DXMATRIX rất hữu dụng, nên chúng ta
sẽ sử dụng nó nhiều ở các phần tiếp theo.
Điều khiển các đổi tượng 3D thông qua ma trận
Như vậy, ta đã biết sơ qua về khái niệm ma trận, ta sẽ tiếp tục xem xét tính hữu dụng của
nó. Ta sử dụng ma trận khi muốn kiểm soát các đối tượng trong không gian. Cho dù bạn
muốn di chuyển đối tượng lòng vòng hay chỉ đơn giản là xoay nó, bạn đều cần phải thực
hiện thông qua ma trận. D3DX cung cấp hàng loạt các hàm để bạn có thể kiểm soát các
đối tương dễ dàng hơn thông qua ma trận. Dưới đây là một trong số đó:
■ D3DXMatrixIdentity. Làm sạch ma trận (biến nó về lại ma trận đồng nhất).
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
86
■ D3DXMatrixRotationX. Xoay đối tượng quanh trục X.
■ D3DXMatrixRotationY. Xoay đối tượng quanh trục Y.
■ D3DXMatrixScaling. Phép tỉ lệ (thu phóng) một đỗi tượng.
■ D3DXMatrixTranslation. Dịch chuyển đối tượng theo các trục.
Di chuyển đối tượng
Để di chuyển đối tượng lòng vòng trong game, bạn cần phải thực hiện phép tịnh tiến.
Phép tịnh tiến sẽ tác động làm cho đối tượng di chuyển theo các trục tọa độ. Nếu bạn
muốn di chuyển đối tượng sang phải chẳng hạn, bạn có thể tịnh tiến nó dọc theo trục X về
hướng dương.
Tịnh tiến các vật thể được thực hiện thông qua hàm D3DXMatrixTranslation như dưới đây:
D3DXMATRIX *D3DXMatrixTranslation(
D3DXMATRIX *pOut,
FLOAT x,
FLOAT y,
FLOAT z
);
Hàm D3DXMatrixTranslation cần 4 đối số.
■ pOut. Ma trận đầu ra. Con trỏ đến đối tượng D3DXMATRIX.
■ x. Khoảng dịch chuyển theo trục X. Chấp nhận cả âm hoặc dương.
■ y. Khoảng dịch chuyển theo trục Y.
■ z. Khoảng dịch chuyển theo trục Z.
Đoạn code sau cho thấy cách sử dụng hàm D3DXMatrixTranslation
D3DXMATRIX matTranslate;
D3DXMATRIX matFinal;
// Đưa ma trận matFinal về ma trận đồng nhất
D3DXMatrixIdentity(&matFinal);
// tịnh tiến đối tượng 64 đơn vị về bên phải theo trục X.
// Ma trận kết quả lưu vào matTranslate
D3DXMatrixTranslation(&matTranslate, 64.0f, 0.0f, 0.0f);
// Nhân ma trận tịnh tiến và ma trận đồng nhất để có ma trận biến đổi
// ma trận biến đổi lưu trữ trong finalMat
D3DXMatrixMultiply(&finalMat, &finalMat, & matTranslate);
// thực hiện phép biến đổi đối tượng trong không gian thực
pd3dDevice->SetTransform(D3DTS_WORLD, &finalMat);
Hàm D3DXMatrixTranslation được sử dụng để tịnh tiến đối tượng 64 đơn vị về bên phải
trục X. Để thực hiện phép biến đổi này với đối tượng, ta nhân ma trận tịnh tiến với ma
trận đồng nhất, sau đó đối tượng được quy đổi về không gian thực.
Xoay đối tượng
Phép tịnh tiến đối tượng thật tuyệt, nhưng bạn vẫn chưa thực sự làm được những thứ mà
game của bạn đòi hỏi. Sẽ còn gì là thú vị nếu như trong một trò đua xe bạn không thể
lượn chiếc xe quanh một vòng cua … bởi lý do xe của bạn chỉ có thể di chuyển trên một
đường thẳng? Đó là lý do mà bạn cần đến phép xoay. Nếu chiếc xe có khả năng xoay thì
thì nó có thể ngoặt, và lượn theo các khúc cua.
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
87
Kết hợp giữa phép xoay và tịnh tiến các đối tượng 3D cho phép nhân vật của bạn có thể
di chuyển tự do trong không gian game. Phép xoay cho phép những chiếc bánh xe ô tô có
thể quay, cánh tay có thể cử động, quả bóng chày có thể xoay tròn khi bay…
Phép xoay là một quá trình quay một đối tượng xung quanh các trục tọa độ. Bởi vì phép
xoay được thực hiện thông qua ma trận, nên thư viện D3DX đã cung cấp một vài hàm
hữu ich để đơn giản hóa quá trình đó.
Phép xoay có thể thực hiện trên các trục vào bất kì thời điểm nào và trên bất kì trục nào
trong 3 trục tọa độ. D3DX cung cấp cho mỗi trục tọa độ một hàm xoay. Ví dụ như, nếu
bạn muốn xoay đỗi tượng quanh trục X, bạn có thể dùng hàm D3DXMatrixRotationX,
như định nghĩa dưới đây:
D3DXMATRIX *D3DXMatrixRotationX(
D3DXMATRIX *pOut,
FLOAT Angle
);
Hàm D3DXMatrixRotationX chỉ có 2 đối số:
■ pOut. Con trỏ đến đối tượng kiểu D3DXMATRIX. Chứa ma trận trả về.
■ Angle. Góc xoay đối tượng (dạng radian).
Sử dụng hàm D3DXMatrixRotationX hay bất kì gì có liên quan là rất đơn giản. Đầu tiên,
ta định nghĩa một cấu trúc D3DXMATRIX chứa ma trận kết quả, sau đó đưa vào góc xoay
là xong. Đoạn code sau cho thấy cách sử dụng hàm này:
D3DXMATRIX matRotate; // ma trận kết quả
D3DXMatrixRotationX(&matRotate, D3DXToRadian(45.0f));
Bạn định nghĩa một ma trận đầu ra và gọi hàm
D3DXMatrixRotationX. Bạn có thể thấy, với tham
số thứ hai ta đã sử dụng một lệnh trợ giúp là
D3DXToRadian. Lệnh này nhận vào một góc
trong khoảng 0 đên 360 độ và chuyển nó thành
dạng radian. Trong ví dụ trên, góc xoay là 45 độ.
Kết quả của phép xoay là một đối tượng được xoay
quanh trục X 45 độ.
Hình 5.6 biểu diễn một hình hộp được xoay quanh
trục Y.
Đoạn code sau cho thấy những bước cần làm để
xoay hình hộp quanh trục Y. Phép xoay sẽ thực
hiện dựa trên bộ đếm thời gian và hình hộp Hình 5.6 Hình hộp được
sẽ được xoay liên tục. xoay quanh trục Y
/************************************************************************
* render
************************************************************************/
void render(void)
{
// xóa back buffer về màu đen
pd3dDevice->Clear( 0,
NULL,
D3DCLEAR_TARGET,
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
88
D3DCOLOR_XRGB(255,255,255),
1.0f,
0 );
pd3dDevice->BeginScene();
// đặt luồng vecto
pd3dDevice->SetStreamSource( 0, vertexBuffer, 0, sizeof(CUSTOMVERTEX) );
// định dạng vecto
pd3dDevice->SetFVF( D3DFVF_CUSTOMVERTEX );
// gán meshMat về ma trận đơn vị
D3DXMatrixIdentity(&objMat);
// cài đặt ma trận xoay
D3DXMatrixRotationY(&matRotate, timeGetTime()/1000.0f);
// nhân ma trận tỉ lệ và ma trận xoay để có ma trận objMat
D3DXMatrixMultiply(&finalMat, &objMat, &matRotate);
// thưc hiện phép biến đổi trong không gian thực
pd3dDevice->SetTransform(D3DTS_WORLD, &finalMat);
// Render hình hộp bằng kiểu triangle strips
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 0, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 4, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 8, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 12, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 16, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 20, 2 );
pd3dDevice->EndScene();
// biểu diễn ra front buffer
pd3dDevice->Present( NULL, NULL, NULL, NULL );
}
Có 3 biến được khai báo ở đầu của hàm render là objMat, matRotate, và finalMat. Những
biến này là các ma trận lưu trữ thông tin về hình hộp. Phần trước bạn đã được học cách
đưa một ma trận về dạng đồng nhất , và ở đây ma trận objMat cần được làm như vậy mỗi
lần hàm render được gọi. Điều đó nhằm mục đich làm cho phép xoay luôn thực hiện ở
gốc tọa độ. Như vậy objMat biểu diễn vị trí thực của hình hộp.
D3DXMatrixIdentity(&objMat);
Ma trận thứ hai là matRotate, chứa thông tin về phép xoay hình hộp. Bởi vì hình hộp cần
được xoay liên tục, do đó bạn cần cập nhật mới ma trận matRotate cho mỗi lần xoay với
một vị trí mới của hình hộp. Phép xoay được thực hiện thông qua
D3DXMatrixRotationY, là một hàm trong thư viện D3DX. Những hàm xoay trong thư
viện D3DX sẽ viết đè thông tin trên ma trận qua mỗi lần xoay, vì thế mà bạn không cần
gọi D3DXMatrixIdentity để đồng nhất hóa ma trận này.
D3DXMatrixRotationY(&matRotate, timeGetTime()/1000.0f);
Hàm timeGetTime sử dụng thời gian hiện tại và chia cho 1000.0f để xoay hình hộp trơn
chu hơn.
Tóm lại, bạn có hai ma trận, một cái biểu diễn vị trí của đối tượng và cái kia biểu diễn sự
vận động của đối tượng, bạn cần nhân hai ma trận này với nhau để có được ma trận cuối
cùng biểu diễn qua finalMat.
Ma trận kết quả quy đổi hình hộp về không gian thực qua hàm SetTransform dưới đây:
pd3dDevice->SetTransform(D3DTS_WORLD, &finalMat);
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
89
Kết quả trả về từ SetTransform là một hình hộp được đặt ở vị trí mới và định hướng trong
không gian thực. Hàm render sẽ vẽ hình hộp qua các lời gọi DrawPrimitive.
Bạn có thể tìm thấy mã nguồn về các phép xoay đối tượng trong thư mục
chapter5\example2 trên đĩa CD.
Tâm của phép xoay
Tâm của phép xoay một đối tựợng phụ thuộc
vào trục mà nó xoay quanh. Nếu một đối
tượng, ví như hình hộp trong hình 5.6 đã
xoay , tâm xoay của nó làm cho nó quay tròn
xung quanh gốc tọa độ. Nếu một đối tựợng
được tịnh tiến ra xa gốc tọa độ và dọc theo
một trục nào đó, thì tâm xoay của cũng dịch
chuyển dọc trục đó làm cho đối tượng bị dịch
chuyển sang vị trí khác trong quá trình thực
hiện phép xoay.
Nhìn vào hinh 5.7, ta thấy một hình hộp được
tịnh tiến dọc theo trục X và Y trước khi bị
xoay. Khi hình hộp đó
được xoay quanh trục X, thì nó cũng bị tịnh Hình 5.7: Hình hộp được xoay quanh trục X
tiến trong suốt quá trình xoay này. sau khi được tịnh tiến ra xa gốc tọa độ
Để thay đổi tâm phép xoay, bạn cần phải tịnh
tiến đối tượng ra xa gốc tọa độ trước khi tiến
hành phép xoay.
Đoạn code sau cho thấy cách mà tâm xoay thay đổi khi tịnh tiến đổi tượng.
/************************************************************************
* render
************************************************************************/
void render(void)
{
// xóa back buffer về màu đen
pd3dDevice->Clear( 0,
NULL,
D3DCLEAR_TARGET,
D3DCOLOR_XRGB(255,255,255),
1.0f,
0 );
pd3dDevice->BeginScene();
pd3dDevice->SetStreamSource( 0, vertexBuffer, 0, sizeof(CUSTOMVERTEX) );
pd3dDevice->SetFVF( D3DFVF_CUSTOMVERTEX );
// tịnh tiến đối tượng ra xa gốc tọa độ
D3DXMatrixTranslation(&matTranslate, 64.0f, 0.0f, 0.0f);
// cài đặt phép quay
D3DXMatrixRotationY(&matRotate, timeGetTime()/1000.0f);
// Nhân ma trận tịnh tiến và ma trận xoay để có ma trận objMat
D3DXMatrixMultiply(&objMat, &matTranslate, &matRotate);
// thực hiện phép biến đổi trong không gian thực
pd3dDevice->SetTransform(D3DTS_WORLD, &objMat);
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 0, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 4, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 8, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 12, 2 );
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
90
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 16, 2 );
pd3dDevice->DrawPrimitive( D3DPT_TRIANGLESTRIP, 20, 2 );
pd3dDevice->EndScene();
// đưa kết quả ra front buffer
pd3dDevice->Present( NULL, NULL, NULL, NULL );
}
Sự thay đổi lớn nhất trong hàm render là hàm D3DXMatrixTranslation. Hàm này dịch
chuyển hình hộp ra xa gốc tọa độ 64 đơn vị.
Trong trường hợp này, hình hộp sẽ được tịnh tiến ra xa gôc tọa độ dọc theo trục X và sau
đó mới xoay. Hai ma trận được dùng ở đây là : matTranslate và matRotate. Hai ma trận
này được nhân với nhau để có ma trận objMat, chứa vị trí sau cùng của hình hộp. Kết quả
là ta có một hình hộp được xoay ở cách xa gốc tọa độ
Phép tỉ lệ
Phép tỉ lệ cho phép bạn thay đổi kích thước đối tượng bằng cách nhân các vecto của đối
tượng với một lượng nào đó. Để thực hiện phép tỉ lệ trên đối tượng, bạn cần tạo một ma
trận chứa các hệ số tỉ lệ. Những hệ số này cho biết các vecto sẽ được phóng to hay thu
nhỏ bao nhiêu. Như đã đề cập ở trên, các vị trí 1, 6, 11 là các hệ số tỉ lệ theo các phương
X, Y và Z. Mặc định, các số đó là 1.0f nghĩa là các đối tượng có kích thước như nó vốn
có. Thay đổi bất kì hệ số nào sẽ làm thay đổi kích thước của đối tượng. Nếu các hệ số này
lớn hơn 1.0f thì đối tượng sẽ được phóng to ra, ngược lại, nếu hệ số này nhỏ hơn 1.0f thì
đối tượng sẽ bị thu nhỏ lại.
⎥⎥
⎥⎥
⎦
⎤
⎢⎢
⎢⎢
⎣
⎡
16151413
12109
875
442
Z
Y
X
Như đã đề cập ở trên, phép tỉ lệ được điều khiển thông qua các giá trị trong ma trận. Để
tạo một ma trận tỉ lệ, ta chỉ cần định nghĩa một ma trận đồng nhất và thay đổi các hệ số
như đã nói ở trên. Bạn vừa có thể tự mình thay đổi các giá trị này vừa có thể sử dụng
thông qua hàm D3DXMatrixScaling như định nghĩa dưới đây:
D3DXMATRIX *D3DXMatrixScaling(
D3DXMATRIX *pOut,
FLOAT sx,
FLOAT sy,
FLOAT sz
);
Hàm D3DXMatrixScaling cần 4 đối số:
■ pOut. Con trỏ đến đối tượng D3DXMATRIX chứa ma trận tỉ lệ
■ sx. Hệ số tỉ lệ theo phương X
■ sy. Hệ số tỉ lệ theo phương Y
■ sz. Hệ số tỉ lệ theo phương Z
Đoạn code sau cho thấy cách dùng hàm D3DXMatrixScaling để tăng gấp đôi kích thước
một vật thể.
D3DXMATRIX matScale;
// đặt hệ số tỉ lệ
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
91
D3DXMatrixScaling(&matScale, 2.0f, 2.0f, 2.0f);
// Nhân ma trận đối tượng với ma trận tỉ lệ
D3DXMatrixMultiply(&objMat, & objMat, &matScaling);
Biến objMat ở trên biểu diễn ma trận gốc của đối tượng. nhân ma trận của đối tượng với
ma trận tỉ lệ ta sẽ có thể thu phóng đối tượng khi vẽ.
Thứ tự trong các phép tính toán ma trận
Thứ tự trong các phép toán là rất quan trọng. Ví dụ như, nếu bạn muốn xoay một đối
tượng quanh tâm của nó và sau đó dịch chuyển nó đi đâu đó, thì trước tiên bạn cần tính
toán với ma trận xoay trước tiếp đó là đến ma trận tịnh tiến. Nếu hai phép tính này được
đổi chỗ cho nhau, thì đối tượng sẽ được tịnh tiến đến 1 vị trí khác sau đó mới được xoay
quanh gốc tọa độ. Điều này có thể làm cho đối tượng được đặt ở một ví trí khác và hướng
theo một hướng khác trong không gian. Đoạn code sau chỉ ra trình tự đúng để thực hiện
phép xoay và tịnh tiến đối tượng:
D3DXMATRIX objRotate;
D3DXMATRIX objTranslation;
D3DXMATRIX objFinal;
// phép xoay
D3DXMatrixRotationY(&objRotate, D3DXToRadian(45));
// phép tịnh tiến
D3DXMatrixTranslation(&objTranslation, 1.0f, 0.0f, 0.0f);
// nhân ma trận xoay và ma trận tịnh tiến với nhau
D3DXMatrixMultiply(&objFinal, &objRotate, &objTranslation);
// thực hiện phép biến đổi trong không gian thực
pd3dDevice->SetTransform(D3DTS_WORLD, &objFinal);
Bước thứ nhất là tạo ma trận xoay đối tượng, objRatate. Sử dụng hàm
D3DXMatrixRotationY như trên, đối tượng sẽ được quay một góc 45 độ quanh trục Y.
Tiếp theo ta tịnh tiến đối tượng đã xoay một đơn vị về bên phải bằng cách sử dụng hàm
D3DXMatrixTranslation.
Cuối cùng, ta tạo ra ma trận biến đổi bằng cách nhân ma trận xoay và ma trận tịnh tiến
với nhau với hàm D3DXMatrixMultiply. Nếu ma trận xoay và ma trận được đảo chỗ cho
nhau (hoán vị) trong lời gọi hàm D3DXMatrixMultiply, thì phép tịnh tiến sẽ được thực
hiện trước phép xoay, và đối tượng bị rời sang vị trí khác.
Tạo một camera bằng các phép chiếu
Bạn có thể tạo ra một camera trong Direct3D bằng cách định nghĩa một ma trận cho bước
(projection transformation). Ma trận này định nghĩa một vùng nhìn cho camera, hệ số co,
mặt phẳng clipping xa và gần.
Sau khi bạn đã tạo được ma trận chiếu, bạn thiết lập nó thông qua hàm SetTransform. Bạn
có thể thấy rằng, hàm SetTransform đã được sử dụng ở ví dụ trước. SetTransform, định
nghĩa ở dưới đây, cho phép thiết lập ma trận cho các bước của hệ thống chuyển đổi hình
học. Ví dụ như, khi bạn thiết lập ma trận cho một camera, cũng có nghĩa là bạn đang quy
định cho scene sẽ được quan sát ra sao trong suốt giai đoạn chiếu. Giai đoạn này cũng là
giai đoạn cuối cùng của hệ chuyển đổi hình học, nó quy định cách mà khung cảnh 3D sẽ
được render dưới dạng 2D.
HRESULT SetTransform(
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
92
D3DTRANSFORMSTATETYPE State,
CONST D3DMATRIX *pMatrix
);
Hàm SetTransform cần 2 đối số:
■ State. Xác định giai đoạn của hệ chuyển đổi hình học cần chỉnh sửa.
■ pMatrix. Con trỏ có cấu trúc D3DMATRIX được dùng để thiết lập cho giai đoạn trên.
Đoạn code sau cho thấy cách tạo và định nghĩa một ma trận cho giai đoạn chiếu.
D3DXMATRIX matProj; // ma trận chiếu
/**********************************************************************
* createCamera
* tạo camera ảo
***********************************************************************/
void createCamera(float nearClip, float farClip)
{
// Xác định vùng nhìn, hệ số co và mặt phẳng clipping xa, gần
D3DXMatrixPerspectiveFovLH(&matProj, D3DX_PI/4, 640/480, nearClip, farClip);
// thiết lập ma trận matProj cho giai đoạn chiếu
pd3dDevice->SetTransform(D3DTS_PROJECTION, &matProj);
}
Thay vì tự mình tạo một ma trận chiếu, ta đã dùng hàm D3DXMatrixPerspectiveFovLH do
D3DX cung cấp. Hàm này tạo một ma trận đầu ra chứa trong matProj được khai báo ở
trên, nó cho phép bạn xác định góc nhìn phối cảnh, hệ số co, và mặt phẳng clipping chỉ
trong một lời gọi hàm.
Sau khi bạn đã tạo đươc ma trận chiếu, bạn có thể thiết lập nó cho hệ chuyển đổi hình học
thông qua hàm SetTransform. Bởi vì ma trận này tác dụng lên giai đoạn chiếu, cho nên
tham số đưa vào là D3DTS_PROJECTION.
Vị trí và hướng của camera
Đến thời điểm này, bạn đã có thể sử dụng được camera rồi. Camera tác động lên mọi thứ
trong khung cảnh cũng giống như là các đối tượng được chuyển qua giai đoạn chiếu trong
hệ chuyển đổi hình học vậy. Chỉ có một điều là, camera của ta đang được đặt ở gốc tọa
độ. Bởi vì camera trong thế giới thực là một vật thể có thể chuyển động được, nên ta cũng
cần làm cho camera ảo có thể làm được giống như vậy. Nó cần có khả năng chuyển động
trong khung cảnh và cũng có thể thay đổi hướng nhìn. Để đạt được 2 tiêu chí này, bạn cần
thay đổi ma trận tương ứng với giai đoạn View transformation. Mặc định, ma trận này
được đặt là ma trận đồng nhất cố định camera ảo ở gốc tọa độ. Để thay đổi vị trí và hướng
của camera, bạn cần tạo một ma trận mới. Cách đơn giản nhất để làm điều đó là sử dụng
hàm trợ giúp D3DXMatrixLookAtLH của D3DX.
Hàm D3DXMatrixLookAtLH cho phép bạn chỉ ra vị trí của camera (định nghĩa như một
vecto dạng D3DXVECTOR3), nơi mà camera nhìn vào (cũng dạng D3DXVECTOR3), và
hướng của camera (cũng là dạng D3DXVECTOR3).
Đoạn code sau chỉ ra cách tạo ma trận quan sát (view).
D3DXMATRIX matView; // ma trận view
/*************************************************************************
* pointCamera
* đặt điểm nhìn cho camera thông qua 1 vecto
*************************************************************************/
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
93
void pointCamera(D3DXVECTOR3 cameraPosition, D3DXVECTOR3 cameraLook)
{
D3DXMatrixLookAtLH ( &matView,
&cameraPosition, // vị trí camera
&cameraLook, // điểm nhìn
&D3DXVECTOR3 (0.0f, 1.0f, 0.0f)); // hướng lên trên
// thiết lập ma trận này cho giai đoạn view
pd3dDevice->SetTransform (D3DTS_VIEW, &matView);
}
Hàm pointCamera cần hai đối số: cameraLook và cameraPosition.
Biến cameraPosition chứa vị trí hiện tại của camera. Ví dụ như, nếu camera được đặt ở
cách gốc tọa độ 2 đơn vị dọc theo trục Y, cameraPosition sẽ có dạng (0.0f, -2.0f, 0.0f).
Biến cameraLook thông báo cho camera nơi cần hướng tới và nó có quan hệ với vị trí đặt
camera. Ví dụ như, nếu coi camera đặt ở vị trí10 đơn vị dương dọc theo trục Y,10 đơn vị
âm theo trục Z và tưởng tượng rằng ta muốn camera hướng về phía gốc tọa độ. Bởi vì lúc
này camera đã được đặt ở phía trên so với gốc tọa độ, do đó muốn thấy được gốc tọa độ
nó cần phải nhìn xuống dưới. Ta cần thiết lập cho vecto cameraLook giá trị là (0.0f, -
10.0f; 0.0f), điều này có nghĩa là camera hướng thẳng theo trục Y và hướng xuống dưới.
Camera lúc đó có thể nhìn các thấy đối tượng đặt ở gốc tọa độ và từ phía trên đầu của
chúng.
Ma trận cuối cùng mà D3DXMatrixLookAtLH tạo ra được lưu trữ trong matView và được
thiết lập cho giai đoạn view của hệ chuyển đổi. Giá trị D3DTS_VIEW được truyền cho
hàm SetTransform thông báo cho Direct3D rằng ma trận view cần được cập nhật lại.
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
94
Tổng kết chương
Trong chương này, ta đã giới thiệu những khái niệm chung cần thiết khi xây dựng một
ứng dụng 3D. Khi bạn muốn tạo một chương trình quan sát mô hình, hay một game nhập
vai góc quay thứ nhất, ma trận và các phép biến đổi chính là nền móng mà game của bạn
được xây dựng trên đó.
Những gì bạn đã được học
Trong chương này, bạn đã được học:
■ Cách chuyển các đối tượng 3D qua hệ chuyển đổi hình học.
■ Ma trận là gì khi nào và làm thế nào sử dụng chúng
■ Cách dịch chuyển và xoay các đối tượng.
■ Tại sao thứ tự trong phép nhân ma trận lại quan trọng đến vậy.
■ Cách tạo và sử dụng camera để quan sát các đối tượng 3D.
Câu hỏi kiểm tra
Bạn có thể tìm thấy đáp án cho phần này và phần bài tập tự làm trong phụ lục “Đáp án
phần bài tập cuối chương”.
1. Chỉ mục của đối tượng được lưu trữ trong bộ đệm (buffer) nào ?
2. Ma trận là gi?
3. Những bước trong hệ chuyển đổi hình học?
4. Ma trận đồng nhất dùng để làm gì?
5. Thay đổi hệ số co của camera tác động đến phần nào của hệ chuyển đổi hình học?
Bài tập tự làm
1. Sử dụng hàm D3DXMatrixMultiply, để xoay đối tượng và sau đó tịnh tiến nó 5 đơn vị
theo trục X.
2. Viết một hàm render xoay liên tục một đối tượng quanh trục Y.
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
95
(Chuong 6...)
Beginning DirectX9 Dịch bởi TransTeam diễn đàn Gamedev.VN
96
CHIA NHỎ VÀ LÀM MỊN
CÁC ĐỐI TƯỢNG
ạn đã học được cách làm thế nào để tạo các object 3D bằng code và biểu diễn nó
lên màn hình. Có lẽ bạn đang nghĩ rằng đây là một quá trình thật nhàm chán và
chưa có cách nào để có thể tạo tất cả các object bằng code.
Các file đính kèm theo tài liệu này:
- beginningdirectx9tiengviet.pdf