#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 = {}; vbd.Usage = D3D11_USAGE_IMMUTABLE; vbd.ByteWidth = sizeof(Vertex) * _vertexCount; vbd.BindFlags = D3D11_BIND_VERTEX_BUFFER; vbd.CPUAccessFlags = 0; vbd.MiscFlags = 0; vbd.StructureByteStride = 0; // Create the proper struct to hold the initial vertex data // - This is how we put the initial data into the buffer D3D11_SUBRESOURCE_DATA initialVertexData = {}; initialVertexData.pSysMem = _vertices; // Actually create the buffer with the initial data _device->CreateBuffer(&vbd, &initialVertexData, bufferVertex.GetAddressOf()); // Create the INDEX BUFFER description D3D11_BUFFER_DESC ibd = {}; ibd.Usage = D3D11_USAGE_IMMUTABLE; ibd.ByteWidth = sizeof(unsigned int) * _indexCount; ibd.BindFlags = D3D11_BIND_INDEX_BUFFER; ibd.CPUAccessFlags = 0; ibd.MiscFlags = 0; ibd.StructureByteStride = 0; // Create the proper struct to hold the initial index data D3D11_SUBRESOURCE_DATA initialIndexData = {}; initialIndexData.pSysMem = _indices; countIndex = _indexCount; // Create the buffer with the initial data _device->CreateBuffer(&ibd, &initialIndexData, bufferIndex.GetAddressOf()); deviceContext = _context; } Mesh::~Mesh() { // Because this is using ComPtr, destructor for now is unnecessary } void Mesh::Draw() { // Set buffers in the input assembler UINT stride = sizeof(Vertex); UINT offset = 0; deviceContext->IASetVertexBuffers(0, 1, bufferVertex.GetAddressOf(), &stride, &offset); deviceContext->IASetIndexBuffer(bufferIndex.Get(), DXGI_FORMAT_R32_UINT, 0); // Do the actual drawing deviceContext->DrawIndexed( countIndex, // The number of indices to use (we could draw a subset if we wanted) 0, // Offset to the first index we want to use 0); // Offset to add to each index when looking up vertices } Microsoft::WRL::ComPtr* Mesh::GetVertexBuffer() { return &bufferVertex; } Microsoft::WRL::ComPtr* Mesh::GetIndexBuffer() { return &bufferIndex; } int Mesh::GetIndexCount() { return countIndex; }