diff --git a/DX11Starter.vcxproj b/DX11Starter.vcxproj
index b0e319f..066bcbf 100644
--- a/DX11Starter.vcxproj
+++ b/DX11Starter.vcxproj
@@ -168,6 +168,113 @@
5.0
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
+ false
+ true
+ Document
+ false
+ true
+ false
+ true
+ false
+ true
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+ $(OutDir)/Assets/Models
+
+
diff --git a/DX11Starter.vcxproj.filters b/DX11Starter.vcxproj.filters
index 301d76a..c4edff6 100644
--- a/DX11Starter.vcxproj.filters
+++ b/DX11Starter.vcxproj.filters
@@ -16,6 +16,12 @@
{e76f01f5-08db-40d4-8720-b5b21f7ec0a3}
+
+ {27304946-4bdd-48b3-86d6-d0de49df247a}
+
+
+ {70d904c1-abb7-4ffe-a6fd-58c67ea6f72b}
+
@@ -89,4 +95,27 @@
Shaders
+
+
+ Assets\Models
+
+
+ Assets\Models
+
+
+ Assets\Models
+
+
+ Assets\Models
+
+
+ Assets\Models
+
+
+ Assets\Models
+
+
+ Assets\Models
+
+
\ No newline at end of file
diff --git a/Game.cpp b/Game.cpp
index e57c673..88c85b1 100644
--- a/Game.cpp
+++ b/Game.cpp
@@ -98,37 +98,32 @@ void Game::LoadShaders()
// --------------------------------------------------------
void Game::CreateBasicGeometry()
{
- // Create some temporary variables to represent colors
- // - Not necessary, just makes things more readable
- XMFLOAT4 red = XMFLOAT4(1.0f, 0.0f, 0.0f, 1.0f);
- XMFLOAT4 green = XMFLOAT4(0.0f, 1.0f, 0.0f, 1.0f);
- XMFLOAT4 blue = XMFLOAT4(0.0f, 0.0f, 1.0f, 1.0f);
- XMFLOAT4 black = XMFLOAT4(0.0f, 0.0f, 0.0f, 1.0f);
- XMFLOAT4 white = XMFLOAT4(1.0f, 1.0f, 1.0f, 1.0f);
+ XMFLOAT3 normals = XMFLOAT3(0, 0, -1);
+ XMFLOAT2 uvs = XMFLOAT2(0, 0);
Vertex verts1[] = {
- { XMFLOAT3(+0.00f, +0.00f, +0.25f), white },
- { XMFLOAT3(-0.25f, -0.25f, -0.25f), red },
- { XMFLOAT3(+0.00f, +0.25f, -0.25f), green },
- { XMFLOAT3(+0.25f, -0.25f, -0.25f), blue },
+ { XMFLOAT3(+0.00f, +0.00f, +0.25f), normals, uvs },
+ { XMFLOAT3(-0.25f, -0.25f, -0.25f), normals, uvs },
+ { XMFLOAT3(+0.00f, +0.25f, -0.25f), normals, uvs },
+ { XMFLOAT3(+0.25f, -0.25f, -0.25f), normals, uvs },
};
unsigned int ind1[] = { 0,1,2 , 0,2,3 , 0,3,1 , 3,2,1 };
Vertex verts2[] = {
- { XMFLOAT3(-0.75f, +0.50f, +0.00f), red },
- { XMFLOAT3(-0.50f, +0.50f, +0.00f), blue },
- { XMFLOAT3(-0.50f, +0.20f, +0.00f), red },
- { XMFLOAT3(-0.75f, +0.20f, +0.00f), blue },
+ { XMFLOAT3(-0.75f, +0.50f, +0.00f), normals, uvs },
+ { XMFLOAT3(-0.50f, +0.50f, +0.00f), normals, uvs },
+ { XMFLOAT3(-0.50f, +0.20f, +0.00f), normals, uvs },
+ { XMFLOAT3(-0.75f, +0.20f, +0.00f), normals, uvs },
};
unsigned int ind2[] = { 0,1,2, 0,2,3 , 3,2,0 , 2,1,0 };
Vertex verts3[] = {
- { XMFLOAT3(+0.00f, +0.30f, +0.15f), white },
- { XMFLOAT3(+0.30f, +0.15f, +0.00f), black },
- { XMFLOAT3(+0.30f, -0.15f, +0.00f), white },
- { XMFLOAT3(+0.00f, -0.30f, +0.15f), black },
- { XMFLOAT3(-0.30f, -0.15f, +0.00f), white },
- { XMFLOAT3(-0.30f, +0.15f, +0.00f), black },
+ { XMFLOAT3(+0.00f, +0.30f, +0.15f), normals, uvs },
+ { XMFLOAT3(+0.30f, +0.15f, +0.00f), normals, uvs },
+ { XMFLOAT3(+0.30f, -0.15f, +0.00f), normals, uvs },
+ { XMFLOAT3(+0.00f, -0.30f, +0.15f), normals, uvs },
+ { XMFLOAT3(-0.30f, -0.15f, +0.00f), normals, uvs },
+ { XMFLOAT3(-0.30f, +0.15f, +0.00f), normals, uvs },
};
unsigned int ind3[] = { 0,1,5 , 1,2,5 , 2,3,4 , 2,4,5 , 5,4,2 , 4,3,2 , 5,2,1 , 5,1,0 };
diff --git a/Mesh.cpp b/Mesh.cpp
index 0c80d8b..f1a0bbd 100644
--- a/Mesh.cpp
+++ b/Mesh.cpp
@@ -1,8 +1,239 @@
#include "Mesh.h"
+#include
+#include
+
using namespace DirectX;
Mesh::Mesh(Vertex* _vertices, int _vertexCount, unsigned int* _indices, int _indexCount, Microsoft::WRL::ComPtr _device, Microsoft::WRL::ComPtr _context)
+{
+ CreateMesh(_vertices, _vertexCount, _indices, _indexCount, _device, _context);
+}
+
+Mesh::Mesh(const char* _file, Microsoft::WRL::ComPtr _device, Microsoft::WRL::ComPtr _context)
+{
+ /// BEGIN OBJLOADER ///
+
+ // Author: Chris Cascioli
+ // Purpose: Basic .OBJ 3D model loading, supporting positions, uvs and normals
+ //
+ // - You are allowed to directly copy/paste this into your code base
+ // for assignments, given that you clearly cite that this is not
+ // code of your own design.
+ //
+ // - NOTE: You'll need to #include
+
+
+ // File input object
+ std::ifstream obj(_file);
+
+ // Check for successful open
+ if (!obj.is_open())
+ return;
+
+ // Variables used while reading the file
+ std::vector positions; // Positions from the file
+ std::vector normals; // Normals from the file
+ std::vector uvs; // UVs from the file
+ std::vector verts; // Verts we're assembling
+ std::vector indices; // Indices of these verts
+ int vertCounter = 0; // Count of vertices
+ int indexCounter = 0; // Count of indices
+ char chars[100]; // String for line reading
+
+ // Still have data left?
+ while (obj.good())
+ {
+ // Get the line (100 characters should be more than enough)
+ obj.getline(chars, 100);
+
+ // Check the type of line
+ if (chars[0] == 'v' && chars[1] == 'n')
+ {
+ // Read the 3 numbers directly into an XMFLOAT3
+ XMFLOAT3 norm;
+ sscanf_s(
+ chars,
+ "vn %f %f %f",
+ &norm.x, &norm.y, &norm.z);
+
+ // Add to the list of normals
+ normals.push_back(norm);
+ }
+ else if (chars[0] == 'v' && chars[1] == 't')
+ {
+ // Read the 2 numbers directly into an XMFLOAT2
+ XMFLOAT2 uv;
+ sscanf_s(
+ chars,
+ "vt %f %f",
+ &uv.x, &uv.y);
+
+ // Add to the list of uv's
+ uvs.push_back(uv);
+ }
+ else if (chars[0] == 'v')
+ {
+ // Read the 3 numbers directly into an XMFLOAT3
+ XMFLOAT3 pos;
+ sscanf_s(
+ chars,
+ "v %f %f %f",
+ &pos.x, &pos.y, &pos.z);
+
+ // Add to the positions
+ positions.push_back(pos);
+ }
+ else if (chars[0] == 'f')
+ {
+ // Read the face indices into an array
+ // NOTE: This assumes the given obj file contains
+ // vertex positions, uv coordinates AND normals.
+ unsigned int i[12];
+ int numbersRead = sscanf_s(
+ chars,
+ "f %d/%d/%d %d/%d/%d %d/%d/%d %d/%d/%d",
+ &i[0], &i[1], &i[2],
+ &i[3], &i[4], &i[5],
+ &i[6], &i[7], &i[8],
+ &i[9], &i[10], &i[11]);
+
+ // If we only got the first number, chances are the OBJ
+ // file has no UV coordinates. This isn't great, but we
+ // still want to load the model without crashing, so we
+ // need to re-read a different pattern (in which we assume
+ // there are no UVs denoted for any of the vertices)
+ if (numbersRead == 1)
+ {
+ // Re-read with a different pattern
+ numbersRead = sscanf_s(
+ chars,
+ "f %d//%d %d//%d %d//%d %d//%d",
+ &i[0], &i[2],
+ &i[3], &i[5],
+ &i[6], &i[8],
+ &i[9], &i[11]);
+
+ // The following indices are where the UVs should
+ // have been, so give them a valid value
+ i[1] = 1;
+ i[4] = 1;
+ i[7] = 1;
+ i[10] = 1;
+
+ // If we have no UVs, create a single UV coordinate
+ // that will be used for all vertices
+ if (uvs.size() == 0)
+ uvs.push_back(XMFLOAT2(0, 0));
+ }
+
+ // - Create the verts by looking up
+ // corresponding data from vectors
+ // - OBJ File indices are 1-based, so
+ // they need to be adusted
+ Vertex v1;
+ v1.Position = positions[i[0] - 1];
+ v1.UV = uvs[i[1] - 1];
+ v1.Normal = normals[i[2] - 1];
+
+ Vertex v2;
+ v2.Position = positions[i[3] - 1];
+ v2.UV = uvs[i[4] - 1];
+ v2.Normal = normals[i[5] - 1];
+
+ Vertex v3;
+ v3.Position = positions[i[6] - 1];
+ v3.UV = uvs[i[7] - 1];
+ v3.Normal = normals[i[8] - 1];
+
+ // The model is most likely in a right-handed space,
+ // especially if it came from Maya. We want to convert
+ // to a left-handed space for DirectX. This means we
+ // need to:
+ // - Invert the Z position
+ // - Invert the normal's Z
+ // - Flip the winding order
+ // We also need to flip the UV coordinate since DirectX
+ // defines (0,0) as the top left of the texture, and many
+ // 3D modeling packages use the bottom left as (0,0)
+
+ // Flip the UV's since they're probably "upside down"
+ v1.UV.y = 1.0f - v1.UV.y;
+ v2.UV.y = 1.0f - v2.UV.y;
+ v3.UV.y = 1.0f - v3.UV.y;
+
+ // Flip Z (LH vs. RH)
+ v1.Position.z *= -1.0f;
+ v2.Position.z *= -1.0f;
+ v3.Position.z *= -1.0f;
+
+ // Flip normal's Z
+ v1.Normal.z *= -1.0f;
+ v2.Normal.z *= -1.0f;
+ v3.Normal.z *= -1.0f;
+
+ // Add the verts to the vector (flipping the winding order)
+ verts.push_back(v1);
+ verts.push_back(v3);
+ verts.push_back(v2);
+ vertCounter += 3;
+
+ // Add three more indices
+ indices.push_back(indexCounter); indexCounter += 1;
+ indices.push_back(indexCounter); indexCounter += 1;
+ indices.push_back(indexCounter); indexCounter += 1;
+
+ // Was there a 4th face?
+ // - 12 numbers read means 4 faces WITH uv's
+ // - 8 numbers read means 4 faces WITHOUT uv's
+ if (numbersRead == 12 || numbersRead == 8)
+ {
+ // Make the last vertex
+ Vertex v4;
+ v4.Position = positions[i[9] - 1];
+ v4.UV = uvs[i[10] - 1];
+ v4.Normal = normals[i[11] - 1];
+
+ // Flip the UV, Z pos and normal's Z
+ v4.UV.y = 1.0f - v4.UV.y;
+ v4.Position.z *= -1.0f;
+ v4.Normal.z *= -1.0f;
+
+ // Add a whole triangle (flipping the winding order)
+ verts.push_back(v1);
+ verts.push_back(v4);
+ verts.push_back(v3);
+ vertCounter += 3;
+
+ // Add three more indices
+ indices.push_back(indexCounter); indexCounter += 1;
+ indices.push_back(indexCounter); indexCounter += 1;
+ indices.push_back(indexCounter); indexCounter += 1;
+ }
+ }
+ }
+
+ // Close the file and create the actual buffers
+ obj.close();
+
+ // - At this point, "verts" is a vector of Vertex structs, and can be used
+ // directly to create a vertex buffer: &verts[0] is the address of the first vert
+ //
+ // - The vector "indices" is similar. It's a vector of unsigned ints and
+ // can be used directly for the index buffer: &indices[0] is the address of the first int
+ //
+ // - "vertCounter" is the number of vertices
+ // - "indexCounter" is the number of indices
+ // - Yes, these are effectively the same since OBJs do not index entire vertices! This means
+ // an index buffer isn't doing much for us. We could try to optimize the mesh ourselves
+ // and detect duplicate vertices, but at that point it would be better to use a more
+ // sophisticated model loading library like TinyOBJLoader or AssImp (yes, that's its name)
+
+ /// END OBJLOADER ///
+ CreateMesh(&verts[0], verts.size(), &indices[0], indices.size(), _device, _context);
+}
+
+void Mesh::CreateMesh(Vertex* _vertices, int _vertexCount, unsigned int* _indices, int _indexCount, Microsoft::WRL::ComPtr _device, Microsoft::WRL::ComPtr _context)
{
// Create the VERTEX BUFFER description
D3D11_BUFFER_DESC vbd = {};
diff --git a/Mesh.h b/Mesh.h
index 8d118cd..ad90b4d 100644
--- a/Mesh.h
+++ b/Mesh.h
@@ -6,11 +6,6 @@
class Mesh
{
-private:
- Microsoft::WRL::ComPtr bufferVertex;
- Microsoft::WRL::ComPtr bufferIndex;
- Microsoft::WRL::ComPtr deviceContext;
- int countIndex;
public:
Mesh(
Vertex* _vertices,
@@ -19,10 +14,28 @@ public:
int _indexCount,
Microsoft::WRL::ComPtr _device,
Microsoft::WRL::ComPtr _context);
+ Mesh(
+ const char* _file,
+ Microsoft::WRL::ComPtr _device,
+ Microsoft::WRL::ComPtr _context);
~Mesh();
void Draw();
Microsoft::WRL::ComPtr* GetVertexBuffer();
Microsoft::WRL::ComPtr* GetIndexBuffer();
int GetIndexCount();
+
+private:
+ Microsoft::WRL::ComPtr bufferVertex;
+ Microsoft::WRL::ComPtr bufferIndex;
+ Microsoft::WRL::ComPtr deviceContext;
+ int countIndex;
+
+ void CreateMesh(
+ Vertex* _vertices,
+ int _vertexCount,
+ unsigned int* _indices,
+ int _indexCount,
+ Microsoft::WRL::ComPtr _device,
+ Microsoft::WRL::ComPtr _context);
};
diff --git a/Vertex.h b/Vertex.h
index 4f06bc0..fe94c18 100644
--- a/Vertex.h
+++ b/Vertex.h
@@ -9,6 +9,7 @@
// --------------------------------------------------------
struct Vertex
{
- DirectX::XMFLOAT3 Position; // The local position of the vertex
- DirectX::XMFLOAT4 Color; // The color of the vertex
+ DirectX::XMFLOAT3 Position;
+ DirectX::XMFLOAT3 Normal;
+ DirectX::XMFLOAT2 UV;
};
\ No newline at end of file
diff --git a/VertexShader.hlsl b/VertexShader.hlsl
index dc17a70..e33cda5 100644
--- a/VertexShader.hlsl
+++ b/VertexShader.hlsl
@@ -18,8 +18,9 @@ struct VertexShaderInput
// | Name Semantic
// | | |
// v v v
- float3 localPosition : POSITION; // XYZ position
- float4 color : COLOR; // RGBA color
+ float3 localPosition : POSITION;
+ float3 color : NORMAL;
+ float2 uv : UV;
};
// Struct representing the data we're sending down the pipeline
@@ -66,7 +67,7 @@ VertexToPixel main( VertexShaderInput input )
// Pass the color through
// - The values will be interpolated per-pixel by the rasterizer
// - We don't need to alter it here, but we do need to send it to the pixel shader
- output.color = input.color * colorTint;
+ output.color = colorTint;
// Whatever we return will make its way through the pipeline to the
// next programmable stage we're using (the pixel shader for now)