【转】QpenGL加载3ds文件

 字体大小: 大 | 中 | 小

注释都很详细,可以直接拿来用.
#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <fstream>
#include <vector>
#include <gl/gl.h> 
#include <gl/glu.h> 
#include <gl/glaux.h>
#include <crtdbg.h>
 
#include <olectl.h> 
#include <math.h> 
 
using namespace std;
 
#define MAX_TEXTURES 100 // 最大的纹理数目
 
// 定义3D点的类,用于保存模型中的顶点
class CVector3 
{
public:
 float x, y, z;
};
 
// 定义2D点类,用于保存模型的UV纹理坐标
class CVector2 
{
public:
 float x, y;
};
 
// 面的结构定义
struct tFace
{
 int vertIndex[3]; // 顶点索引
 int coordIndex[3]; // 纹理坐标索引
};
 
// 材质信息结构体
struct tMaterialInfo
{
 char strName[255]; // 纹理名称
 char strFile[255]; // 如果存在纹理映射,则表示纹理文件名称
 BYTE color[3]; // 对象的RGB颜色
 int texureId; // 纹理ID
 float uTile; // u 重复
 float vTile; // v 重复
 float uOffset; // u 纹理偏移
 float vOffset; // v 纹理偏移
} ;
 
// 对象信息结构体
struct t3DObject 
{
 int numOfVerts; // 模型中顶点的数目
 int numOfFaces; // 模型中面的数目
 int numTexVertex; // 模型中纹理坐标的数目
 int materialID; // 纹理ID
 bool bHasTexture; // 是否具有纹理映射
 char strName[255]; // 对象的名称
 CVector3 *pVerts; // 对象的顶点
 CVector3 *pNormals; // 对象的法向量
 CVector2 *pTexVerts; // 纹理UV坐标
 tFace *pFaces; // 对象的面信息
};
 
// 模型信息结构体
struct t3DModel 
{
 UINT texture[MAX_TEXTURES];
 int numOfObjects; // 模型中对象的数目
 int numOfMaterials; // 模型中材质的数目
 vector<tMaterialInfo> pMaterials; // 材质链表信息
 vector<t3DObject> pObject; // 模型中对象链表信息
};
 
#ifndef _3DS_H
#define _3DS_H
 
// 基本块(Primary Chunk),位于文件的开始
#define PRIMARY 0x4D4D
 
// 主块(Main Chunks)
#define OBJECTINFO 0x3D3D // 网格对象的版本号
#define VERSION 0x0002 // .3ds文件的版本
#define EDITKEYFRAME 0xB000 // 所有关键帧信息的头部
 
// 对象的次级定义(包括对象的材质和对象)
#define MATERIAL 0xAFFF // 保存纹理信息
#define OBJECT 0x4000 // 保存对象的面、顶点等信息
 
// 材质的次级定义
#define MATNAME 0xA000 // 保存材质名称
#define MATDIFFUSE 0xA020 // 对象/材质的颜色
#define MATMAP 0xA200 // 新材质的头部
#define MATMAPFILE 0xA300 // 保存纹理的文件名
 
#define OBJECT_MESH 0x4100 // 新的网格对象
 
// OBJECT_MESH的次级定义
#define OBJECT_VERTICES 0x4110 // 对象顶点
#define OBJECT_FACES 0x4120 // 对象的面
#define OBJECT_MATERIAL 0x4130 // 对象的材质
#define OBJECT_UV 0x4140 // 对象的UV纹理坐标
 
struct tIndices 
 unsigned short a, b, c, bVisible; 
};
 
// 保存块信息的结构
struct tChunk
{
 unsigned short int ID; // 块的ID 
 unsigned int length; // 块的长度
 unsigned int bytesRead; // 需要读的块数据的字节数
};
 
// CLoad3DS类处理所有的装入代码
class CLoad3DS
{
public:
 CLoad3DS(); // 初始化数据成员
 // 装入3ds文件到模型结构中
 bool Import3DS(t3DModel *pModel, char *strFileName);
 // 读入一个纹理
 int BuildTexture(char *szPathName, GLuint &texid);
 
private:
 // 读一个字符串
 int GetString(char *);
 // 读下一个块
 void ReadChunk(tChunk *);
 // 读下一个块
 void ProcessNextChunk(t3DModel *pModel, tChunk *);
 // 读下一个对象块
 void ProcessNextObjectChunk(t3DModel *pModel, t3DObject *pObject, tChunk *);
 // 读下一个材质块
 void ProcessNextMaterialChunk(t3DModel *pModel, tChunk *);
 // 读对象颜色的RGB值
 void ReadColorChunk(tMaterialInfo *pMaterial, tChunk *pChunk);
 // 读对象的顶点
 void ReadVertices(t3DObject *pObject, tChunk *);
 // 读对象的面信息
 void ReadVertexIndices(t3DObject *pObject, tChunk *);
 // 读对象的纹理坐标
 void ReadUVCoordinates(t3DObject *pObject, tChunk *);
 // 读赋予对象的材质名称
 void ReadObjectMaterial(t3DModel *pModel, t3DObject *pObject, tChunk *pPreviousChunk);
 // 计算对象顶点的法向量
 void ComputeNormals(t3DModel *pModel);
 // 关闭文件,释放内存空间
 void CleanUp();
 // 文件指针
 FILE *m_FilePointer;
 
 tChunk *m_CurrentChunk;
 tChunk *m_TempChunk;
};
 
#endif
 
// 读入一个纹理
int CLoad3DS::BuildTexture(char *szPathName, GLuint &texid)
{
 HDC hdcTemp; // The DC To Hold Our Bitmap
 HBITMAP hbmpTemp; // Holds The Bitmap Temporarily
 IPicture *pPicture; // IPicture Interface
 OLECHAR wszPath[MAX_PATH+1]; // Full Path To Picture (WCHAR)
 char szPath[MAX_PATH+1]; // Full Path To Picture
 long lWidth; // Width In Logical Units
 long lHeight; // Height In Logical Units
 long lWidthPixels; // Width In Pixels
 long lHeightPixels; // Height In Pixels
 GLint glMaxTexDim ; // Holds Maximum Texture Size
 
 if (strstr(szPathName, "http://")) // If PathName Contains http:// Then...
 {
 strcpy(szPath, szPathName); // Append The PathName To szPath
 }
 else // Otherwise... We Are Loading From A File
 {
 GetCurrentDirectory(MAX_PATH, szPath); // Get Our Working Directory
 strcat(szPath, "//"); // Append "/" After The Working Directory
 strcat(szPath, szPathName); // Append The PathName
 }
 
 MultiByteToWideChar(CP_ACP, 0, szPath, -1, wszPath, MAX_PATH); // Convert From ASCII To Unicode
 HRESULT hr = OleLoadPicturePath(wszPath, 0, 0, 0, IID_IPicture, (void**)&pPicture);
 
 if(FAILED(hr)) // If Loading Failed
 return FALSE; // Return False
 
 hdcTemp = CreateCompatibleDC(GetDC(0)); // Create The Windows Compatible Device Context
 if(!hdcTemp) // Did Creation Fail?
 {
 pPicture->Release(); // Decrements IPicture Reference Count
 return FALSE; // Return False (Failure)
 }
 
 glGetIntegerv(GL_MAX_TEXTURE_SIZE, &glMaxTexDim); // Get Maximum Texture Size Supported
 
 pPicture->get_Width(&lWidth); // Get IPicture Width (Convert To Pixels)
 lWidthPixels = MulDiv(lWidth, GetDeviceCaps(hdcTemp, LOGPIXELSX), 2540);
 pPicture->get_Height(&lHeight); // Get IPicture Height (Convert To Pixels)
 lHeightPixels = MulDiv(lHeight, GetDeviceCaps(hdcTemp, LOGPIXELSY), 2540);
 
 // Resize Image To Closest Power Of Two
 if (lWidthPixels <= glMaxTexDim) // Is Image Width Less Than Or Equal To Cards Limit
 lWidthPixels = 1 << (int)floor((log((double)lWidthPixels)/log(2.0f)) + 0.5f); 
 else // Otherwise Set Width To "Max Power Of Two" That The Card Can Handle
 lWidthPixels = glMaxTexDim;
 
 if (lHeightPixels <= glMaxTexDim) // Is Image Height Greater Than Cards Limit
 lHeightPixels = 1 << (int)floor((log((double)lHeightPixels)/log(2.0f)) + 0.5f);
 else // Otherwise Set Height To "Max Power Of Two" That The Card Can Handle
 lHeightPixels = glMaxTexDim;
 
 // Create A Temporary Bitmap
 BITMAPINFO bi = {0}; // The Type Of Bitmap We Request
 DWORD *pBits = 0; // Pointer To The Bitmap Bits
 
 bi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER); // Set Structure Size
 bi.bmiHeader.biBitCount = 32; // 32 Bit
 bi.bmiHeader.biWidth = lWidthPixels; // Power Of Two Width
 bi.bmiHeader.biHeight = lHeightPixels; // Make Image Top Up (Positive Y-Axis)
 bi.bmiHeader.biCompression = BI_RGB; // RGB Encoding
 bi.bmiHeader.biPlanes = 1; // 1 Bitplane
 
 // Creating A Bitmap This Way Allows Us To Specify Color Depth And Gives Us Imediate Access To The Bits
 hbmpTemp = CreateDIBSection(hdcTemp, &bi, DIB_RGB_COLORS, (void**)&pBits, 0, 0);
 
 if(!hbmpTemp) // Did Creation Fail?
 {
 DeleteDC(hdcTemp); // Delete The Device Context
 pPicture->Release(); // Decrements IPicture Reference Count
 return FALSE; // Return False (Failure)
 }
 
 SelectObject(hdcTemp, hbmpTemp); // Select Handle To Our Temp DC And Our Temp Bitmap Object
 
 // Render The IPicture On To The Bitmap
 pPicture->Render(hdcTemp, 0, 0, lWidthPixels, lHeightPixels, 0, lHeight, lWidth, -lHeight, 0);
 
 // Convert From BGR To RGB Format And Add An Alpha Value Of 255
 for(long i = 0; i < lWidthPixels * lHeightPixels; i++) // Loop Through All Of The Pixels
 {
 BYTE* pPixel = (BYTE*)(&pBits[i]); // Grab The Current Pixel
 BYTE temp = pPixel[0]; // Store 1st Color In Temp Variable (Blue)
 pPixel[0] = pPixel[2]; // Move Red Value To Correct Position (1st)
 pPixel[2] = temp; // Move Temp Value To Correct Blue Position (3rd)
 
 // This Will Make Any Black Pixels, Completely Transparent (You Can Hardcode The Value If You Wish)
 if ((pPixel[0]==0) && (pPixel[1]==0) && (pPixel[2]==0)) // Is Pixel Completely Black
 pPixel[3] = 0; // Set The Alpha Value To 0
 else // Otherwise
 pPixel[3] = 255; // Set The Alpha Value To 255
 }
 
 glGenTextures(1, &texid); // Create The Texture
 
 // Typical Texture Generation Using Data From The Bitmap
 glBindTexture(GL_TEXTURE_2D, texid); // Bind To The Texture ID
 glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR); // (Modify This For The Type Of Filtering You Want)
 glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR); // (Modify This For The Type Of Filtering You Want)
 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, lWidthPixels, lHeightPixels, 0, GL_RGBA, GL_UNSIGNED_BYTE, pBits); // (Modify This If You Want Mipmaps)
 
 DeleteObject(hbmpTemp); // Delete The Object
 DeleteDC(hdcTemp); // Delete The Device Context
 
 pPicture->Release(); // Decrements IPicture Reference Count
 
 return TRUE; // Return True (All Good)
 
}
 
// 构造函数的功能是初始化tChunk数据
CLoad3DS::CLoad3DS()
{
 m_CurrentChunk = new tChunk; // 初始化并为当前的块分配空间
 m_TempChunk = new tChunk; // 初始化一个临时块并分配空间
}
 
// 打开一个3ds文件,读出其中的内容,并释放内存
bool CLoad3DS::Import3DS(t3DModel *pModel, char *strFileName)
{
 char strMessage[255] = {0};
 
 // 打开一个3ds文件
 m_FilePointer = fopen(strFileName, "rb");
 
 // 确保所获得的文件指针合法
 if(!m_FilePointer) 
 {
 sprintf(strMessage, "Unable to find the file: %s!", strFileName);
 MessageBox(NULL, strMessage, "Error", MB_OK);
 return false;
 }
 
 // 当文件打开之后,首先应该将文件最开始的数据块读出以判断是否是一个3ds文件
 // 如果是3ds文件的话,第一个块ID应该是PRIMARY
 
 // 将文件的第一块读出并判断是否是3ds文件
 ReadChunk(m_CurrentChunk);
 
 // 确保是3ds文件
 if (m_CurrentChunk->ID != PRIMARY)
 {
 sprintf(strMessage, "Unable to load PRIMARY chuck from file: %s!", strFileName);
 MessageBox(NULL, strMessage, "Error", MB_OK);
 return false;
 }
 
 // 现在开始读入数据,ProcessNextChunk()是一个递归函数
 
 // 通过调用下面的递归函数,将对象读出
 ProcessNextChunk(pModel, m_CurrentChunk);
 
 // 在读完整个3ds文件之后,计算顶点的法线
 ComputeNormals(pModel);
 
 // 释放内存空间
 CleanUp();
 
 return true;
}
 
// 下面的函数释放所有的内存空间,并关闭文件
void CLoad3DS::CleanUp()
{
 
 fclose(m_FilePointer); // 关闭当前的文件指针
 delete m_CurrentChunk; // 释放当前块
 delete m_TempChunk; // 释放临时块
}
 
// 下面的函数读出3ds文件的主要部分
void CLoad3DS::ProcessNextChunk(t3DModel *pModel, tChunk *pPreviousChunk)
{
 t3DObject newObject = {0}; // 用来添加到对象链表
 tMaterialInfo newTexture = {0}; // 用来添加到材质链表
 unsigned int version = 0; // 保存文件版本
 int buffer[50000] = {0}; // 用来跳过不需要的数据
 
 m_CurrentChunk = new tChunk; // 为新的块分配空间 
 
 // 下面每读一个新块,都要判断一下块的ID,如果该块是需要的读入的,则继续进行
 // 如果是不需要读入的块,则略过
 
 // 继续读入子块,直到达到预定的长度
 while (pPreviousChunk->bytesRead < pPreviousChunk->length)
 {
 // 读入下一个块
 ReadChunk(m_CurrentChunk);
 
 // 判断块的ID号
 switch (m_CurrentChunk->ID)
 {
 case VERSION: // 文件版本号
 
 // 在该块中有一个无符号短整型数保存了文件的版本
 
 // 读入文件的版本号,并将字节数添加到bytesRead变量中
 m_CurrentChunk->bytesRead += fread(&version, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 
 // 如果文件版本号大于3,给出一个警告信息
 if (version > 0x03)
 MessageBox(NULL, "This 3DS file is over version 3 so it may load incorrectly", "Warning", MB_OK);
 break;
 
 case OBJECTINFO: // 网格版本信息
 
 // 读入下一个块
 ReadChunk(m_TempChunk);
 
 // 获得网格的版本号
 m_TempChunk->bytesRead += fread(&version, 1, m_TempChunk->length - m_TempChunk->bytesRead, m_FilePointer);
 
 // 增加读入的字节数
 m_CurrentChunk->bytesRead += m_TempChunk->bytesRead;
 
 // 进入下一个块
 ProcessNextChunk(pModel, m_CurrentChunk);
 break;
 
 case MATERIAL: // 材质信息
 
 // 材质的数目递增
 pModel->numOfMaterials++;
 
 // 在纹理链表中添加一个空白纹理结构
 pModel->pMaterials.push_back(newTexture);
 
 // 进入材质装入函数
 ProcessNextMaterialChunk(pModel, m_CurrentChunk);
 break;
 
 case OBJECT: // 对象的名称
 
 // 该块是对象信息块的头部,保存了对象了名称
 
 // 对象数递增
 pModel->numOfObjects++;
 
 // 添加一个新的tObject节点到对象链表中
 pModel->pObject.push_back(newObject);
 
 // 初始化对象和它的所有数据成员
 memset(&(pModel->pObject[pModel->numOfObjects - 1]), 0, sizeof(t3DObject));
 
 // 获得并保存对象的名称,然后增加读入的字节数
 m_CurrentChunk->bytesRead += GetString(pModel->pObject[pModel->numOfObjects - 1].strName);
 
 // 进入其余的对象信息的读入
 ProcessNextObjectChunk(pModel, &(pModel->pObject[pModel->numOfObjects - 1]), m_CurrentChunk);
 break;
 
 case EDITKEYFRAME:
 
 // 跳过关键帧块的读入,增加需要读入的字节数
 m_CurrentChunk->bytesRead += fread(buffer, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 break;
 
 default: 
 
 // 跳过所有忽略的块的内容的读入,增加需要读入的字节数
 m_CurrentChunk->bytesRead += fread(buffer, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 break;
 }
 
 // 增加从最后块读入的字节数
 pPreviousChunk->bytesRead += m_CurrentChunk->bytesRead;
 }
 
 // 释放当前块的内存空间
 delete m_CurrentChunk;
 m_CurrentChunk = pPreviousChunk;
}
 
// 下面的函数处理所有的文件中对象的信息
void CLoad3DS::ProcessNextObjectChunk(t3DModel *pModel, t3DObject *pObject, tChunk *pPreviousChunk)
{
 int buffer[50000] = {0}; // 用于读入不需要的数据
 
 // 对新的块分配存储空间
 m_CurrentChunk = new tChunk;
 
 // 继续读入块的内容直至本子块结束
 while (pPreviousChunk->bytesRead < pPreviousChunk->length)
 {
 // 读入下一个块
 ReadChunk(m_CurrentChunk);
 
 // 区别读入是哪种块
 switch (m_CurrentChunk->ID)
 {
 case OBJECT_MESH: // 正读入的是一个新块
 
 // 使用递归函数调用,处理该新块
 ProcessNextObjectChunk(pModel, pObject, m_CurrentChunk);
 break;
 
 case OBJECT_VERTICES: // 读入是对象顶点
 ReadVertices(pObject, m_CurrentChunk);
 break;
 
 case OBJECT_FACES: // 读入的是对象的面
 ReadVertexIndices(pObject, m_CurrentChunk);
 break;
 
 case OBJECT_MATERIAL: // 读入的是对象的材质名称
 
 // 该块保存了对象材质的名称,可能是一个颜色,也可能是一个纹理映射。同时在该块中也保存了
 // 纹理对象所赋予的面
 
 // 下面读入对象的材质名称
 ReadObjectMaterial(pModel, pObject, m_CurrentChunk); 
 break;
 
 case OBJECT_UV: // 读入对象的UV纹理坐标
 
 // 读入对象的UV纹理坐标
 ReadUVCoordinates(pObject, m_CurrentChunk);
 break;
 
 default: 
 
 // 略过不需要读入的块
 m_CurrentChunk->bytesRead += fread(buffer, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 break;
 }
 
 // 添加从最后块中读入的字节数到前面的读入的字节中
 pPreviousChunk->bytesRead += m_CurrentChunk->bytesRead;
 }
 
 // 释放当前块的内存空间,并把当前块设置为前面块
 delete m_CurrentChunk;
 m_CurrentChunk = pPreviousChunk;
}
 
// 下面的函数处理所有的材质信息
void CLoad3DS::ProcessNextMaterialChunk(t3DModel *pModel, tChunk *pPreviousChunk)
{
 int buffer[50000] = {0}; // 用于读入不需要的数据
 
 // 给当前块分配存储空间
 m_CurrentChunk = new tChunk;
 
 // 继续读入这些块,知道该子块结束
 while (pPreviousChunk->bytesRead < pPreviousChunk->length)
 {
 // 读入下一块
 ReadChunk(m_CurrentChunk);
 
 // 判断读入的是什么块
 switch (m_CurrentChunk->ID)
 {
 case MATNAME: // 材质的名称
 
 // 读入材质的名称
 m_CurrentChunk->bytesRead += fread(pModel->pMaterials[pModel->numOfMaterials - 1].strName, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 break;
 
 case MATDIFFUSE: // 对象的R G B颜色
 ReadColorChunk(&(pModel->pMaterials[pModel->numOfMaterials - 1]), m_CurrentChunk);
 break;
 
 case MATMAP: // 纹理信息的头部
 
 // 进入下一个材质块信息
 ProcessNextMaterialChunk(pModel, m_CurrentChunk);
 break;
 
 case MATMAPFILE: // 材质文件的名称
 
 // 读入材质的文件名称
 m_CurrentChunk->bytesRead += fread(pModel->pMaterials[pModel->numOfMaterials - 1].strFile, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 break;
 
 default: 
 
 // 掠过不需要读入的块
 m_CurrentChunk->bytesRead += fread(buffer, 1, m_CurrentChunk->length - m_CurrentChunk->bytesRead, m_FilePointer);
 break;
 }
 
 // 添加从最后块中读入的字节数
 pPreviousChunk->bytesRead += m_CurrentChunk->bytesRead;
 }
 
 // 删除当前块,并将当前块设置为前面的块
 delete m_CurrentChunk;
 m_CurrentChunk = pPreviousChunk;
}
 
// 下面函数读入块的ID号和它的字节长度
void CLoad3DS::ReadChunk(tChunk *pChunk)
{
 // 读入块的ID号,占用了2个字节。块的ID号象OBJECT或MATERIAL一样,说明了在块中所包含的内容
 pChunk->bytesRead = fread(&pChunk->ID, 1, 2, m_FilePointer);
 
 // 然后读入块占用的长度,包含了四个字节
 pChunk->bytesRead += fread(&pChunk->length, 1, 4, m_FilePointer);
}
 
// 下面的函数读入一个字符串
int CLoad3DS::GetString(char *pBuffer)
{
 int index = 0;
 
 // 读入一个字节的数据
 fread(pBuffer, 1, 1, m_FilePointer);
 
 // 直到结束
 while (*(pBuffer + index++) != 0) {
 
 // 读入一个字符直到NULL
 fread(pBuffer + index, 1, 1, m_FilePointer);
 }
 
 // 返回字符串的长度
 return strlen(pBuffer) + 1;
}
 
// 下面的函数读入RGB颜色
void CLoad3DS::ReadColorChunk(tMaterialInfo *pMaterial, tChunk *pChunk)
{
 // 读入颜色块信息
 ReadChunk(m_TempChunk);
 
 // 读入RGB颜色
 m_TempChunk->bytesRead += fread(pMaterial->color, 1, m_TempChunk->length - m_TempChunk->bytesRead, m_FilePointer);
 
 // 增加读入的字节数
 pChunk->bytesRead += m_TempChunk->bytesRead;
}
 
// 下面的函数读入顶点索引
void CLoad3DS::ReadVertexIndices(t3DObject *pObject, tChunk *pPreviousChunk)
{
 unsigned short index = 0; // 用于读入当前面的索引
 
 // 读入该对象中面的数目
 pPreviousChunk->bytesRead += fread(&pObject->numOfFaces, 1, 2, m_FilePointer);
 
 // 分配所有面的存储空间,并初始化结构
 pObject->pFaces = new tFace [pObject->numOfFaces];
 memset(pObject->pFaces, 0, sizeof(tFace) * pObject->numOfFaces);
 
 // 遍历对象中所有的面
 for(int i = 0; i < pObject->numOfFaces; i++)
 {
 for(int j = 0; j < 4; j++)
 {
 // 读入当前面的第一个点 
 pPreviousChunk->bytesRead += fread(&index, 1, sizeof(index), m_FilePointer);
 
 if(j < 3)
 {
 // 将索引保存在面的结构中
 pObject->pFaces[i].vertIndex[j] = index;
 }
 }
 }
}
 
// 下面的函数读入对象的UV坐标
void CLoad3DS::ReadUVCoordinates(t3DObject *pObject, tChunk *pPreviousChunk)
{
 // 为了读入对象的UV坐标,首先需要读入UV坐标的数量,然后才读入具体的数据
 
 // 读入UV坐标的数量
 pPreviousChunk->bytesRead += fread(&pObject->numTexVertex, 1, 2, m_FilePointer);
 
 // 分配保存UV坐标的内存空间
 pObject->pTexVerts = new CVector2 [pObject->numTexVertex];
 
 // 读入纹理坐标
 pPreviousChunk->bytesRead += fread(pObject->pTexVerts, 1, pPreviousChunk->length - pPreviousChunk->bytesRead, m_FilePointer);
}
 
// 读入对象的顶点
void CLoad3DS::ReadVertices(t3DObject *pObject, tChunk *pPreviousChunk)
{
 // 在读入实际的顶点之前,首先必须确定需要读入多少个顶点。
 
 // 读入顶点的数目
 pPreviousChunk->bytesRead += fread(&(pObject->numOfVerts), 1, 2, m_FilePointer);
 
 // 分配顶点的存储空间,然后初始化结构体
 pObject->pVerts = new CVector3 [pObject->numOfVerts];
 memset(pObject->pVerts, 0, sizeof(CVector3) * pObject->numOfVerts);
 
 // 读入顶点序列
 pPreviousChunk->bytesRead += fread(pObject->pVerts, 1, pPreviousChunk->length - pPreviousChunk->bytesRead, m_FilePointer);
 
 // 现在已经读入了所有的顶点。
 // 因为3D Studio Max的模型的Z轴是指向上的,因此需要将y轴和z轴翻转过来。
 // 具体的做法是将Y轴和Z轴交换,然后将Z轴反向。
 
 // 遍历所有的顶点
 for(int i = 0; i < pObject->numOfVerts; i++)
 {
 // 保存Y轴的值
 float fTempY = pObject->pVerts[i].y;
 
 // 设置Y轴的值等于Z轴的值
 pObject->pVerts[i].y = pObject->pVerts[i].z;
 
 // 设置Z轴的值等于-Y轴的值 
 pObject->pVerts[i].z = -fTempY;
 }
}
 
// 下面的函数读入对象的材质名称
void CLoad3DS::ReadObjectMaterial(t3DModel *pModel, t3DObject *pObject, tChunk *pPreviousChunk)
{
 char strMaterial[255] = {0}; // 用来保存对象的材质名称
 int buffer[50000] = {0}; // 用来读入不需要的数据
 
 // 材质或者是颜色,或者是对象的纹理,也可能保存了象明亮度、发光度等信息。
 
 // 下面读入赋予当前对象的材质名称
 pPreviousChunk->bytesRead += GetString(strMaterial);
 
 // 遍历所有的纹理
 for(int i = 0; i < pModel->numOfMaterials; i++)
 {
 //如果读入的纹理与当前的纹理名称匹配
 if(strcmp(strMaterial, pModel->pMaterials[i].strName) == 0)
 {
 // 设置材质ID
 pObject->materialID = i;
 
 // 判断是否是纹理映射,如果strFile是一个长度大于1的字符串,则是纹理
 if(strlen(pModel->pMaterials[i].strFile) > 0) {
 //载入纹理
 BuildTexture(pModel->pMaterials[i].strFile, pModel->texture[pObject->materialID]);
    // 设置对象的纹理映射标志
 pObject->bHasTexture = true;
 } 
 break;
 }
 else
 {
 // 如果该对象没有材质,则设置ID为-1
 pObject->materialID = -1;
 }
 }
 
 pPreviousChunk->bytesRead += fread(buffer, 1, pPreviousChunk->length - pPreviousChunk->bytesRead, m_FilePointer);
 
// 下面的这些函数主要用来计算顶点的法向量,顶点的法向量主要用来计算光照
// 下面的宏定义计算一个矢量的长度
#define Mag(Normal) (sqrt(Normal.x*Normal.x + Normal.y*Normal.y + Normal.z*Normal.z))
 
// 下面的函数求两点决定的矢量
CVector3 Vector(CVector3 vPoint1, CVector3 vPoint2)
{
 CVector3 vVector; 
 
 vVector.x = vPoint1.x - vPoint2.x; 
 vVector.y = vPoint1.y - vPoint2.y; 
 vVector.z = vPoint1.z - vPoint2.z; 
 
 return vVector; 
}
 
// 下面的函数两个矢量相加
CVector3 AddVector(CVector3 vVector1, CVector3 vVector2)
{
 CVector3 vResult; 
 
 vResult.x = vVector2.x + vVector1.x; 
 vResult.y = vVector2.y + vVector1.y; 
 vResult.z = vVector2.z + vVector1.z; 
 
 return vResult; 
}
 
// 下面的函数处理矢量的缩放
CVector3 DivideVectorByScaler(CVector3 vVector1, float Scaler)
{
 CVector3 vResult; 
 
 vResult.x = vVector1.x / Scaler; 
 vResult.y = vVector1.y / Scaler; 
 vResult.z = vVector1.z / Scaler; 
 
 return vResult; 
}
 
// 下面的函数返回两个矢量的叉积
CVector3 Cross(CVector3 vVector1, CVector3 vVector2)
{
 CVector3 vCross; 
 
 vCross.x = ((vVector1.y * vVector2.z) - (vVector1.z * vVector2.y));
 
 vCross.y = ((vVector1.z * vVector2.x) - (vVector1.x * vVector2.z));
 
 vCross.z = ((vVector1.x * vVector2.y) - (vVector1.y * vVector2.x));
 
 return vCross; 
}
 
// 下面的函数规范化矢量
CVector3 Normalize(CVector3 vNormal)
{
 double Magnitude; 
 
 Magnitude = Mag(vNormal); // 获得矢量的长度
 
 vNormal.x /= (float)Magnitude; 
 vNormal.y /= (float)Magnitude; 
 vNormal.z /= (float)Magnitude; 
 
 return vNormal; 
}
 
// 下面的函数用于计算对象的法向量
void CLoad3DS::ComputeNormals(t3DModel *pModel)
{
 CVector3 vVector1, vVector2, vNormal, vPoly[3];
 
 // 如果模型中没有对象,则返回
 if(pModel->numOfObjects <= 0)
 return;
 
 // 遍历模型中所有的对象
 for(int index = 0; index < pModel->numOfObjects; index++)
 {
 // 获得当前的对象
 t3DObject *pObject = &(pModel->pObject[index]);
 
 // 分配需要的存储空间
 CVector3 *pNormals = new CVector3 [pObject->numOfFaces];
 CVector3 *pTempNormals = new CVector3 [pObject->numOfFaces];
 pObject->pNormals = new CVector3 [pObject->numOfVerts];
 
 // 遍历对象的所有面
 for(int i=0; i < pObject->numOfFaces; i++)
 { 
 vPoly[0] = pObject->pVerts[pObject->pFaces[i].vertIndex[0]];
 vPoly[1] = pObject->pVerts[pObject->pFaces[i].vertIndex[1]];
 vPoly[2] = pObject->pVerts[pObject->pFaces[i].vertIndex[2]];
 
 // 计算面的法向量
 
 vVector1 = Vector(vPoly[0], vPoly[2]); // 获得多边形的矢量
 vVector2 = Vector(vPoly[2], vPoly[1]); // 获得多边形的第二个矢量
 
 vNormal = Cross(vVector1, vVector2); // 获得两个矢量的叉积
 pTempNormals[i] = vNormal; // 保存非规范化法向量
 vNormal = Normalize(vNormal); // 规范化获得的叉积
 
 pNormals[i] = vNormal; // 将法向量添加到法向量列表中
 }
 
 // 下面求顶点法向量
 CVector3 vSum = {0.0, 0.0, 0.0};
 CVector3 vZero = vSum;
 int shared=0;
 // 遍历所有的顶点
 for (i = 0; i < pObject->numOfVerts; i++) 
 {
 for (int j = 0; j < pObject->numOfFaces; j++) // 遍历所有的三角形面
 { // 判断该点是否与其它的面共享
 if (pObject->pFaces[j].vertIndex[0] == i || 
 pObject->pFaces[j].vertIndex[1] == i || 
 pObject->pFaces[j].vertIndex[2] == i)
 {
 vSum = AddVector(vSum, pTempNormals[j]);
 shared++; 
 }
 } 
 
 pObject->pNormals[i] = DivideVectorByScaler(vSum, float(-shared));
 
 // 规范化最后的顶点法向
 pObject->pNormals[i] = Normalize(pObject->pNormals[i]); 
 
 vSum = vZero; 
 shared = 0; 
 }
 
 // 释放存储空间,开始下一个对象
 delete [] pTempNormals;
 delete [] pNormals;
 }
}
 
调用示例:
t3DModel PetalModel;
.....................
CLoad3DS *loader=NULL;
loader=new(CLoad3DS);
  
loader->Import3DS(&Model, FullPathFileName);
 
delete loader;
.......................
//画这个模型
 
void DrawModel(t3DModel& Model)
{
  // 遍历模型中所有的对象
  for(int i = 0; i < Model.numOfObjects; i++)
  {
    // 如果对象的大小小于0,则退出
    if(Model.pObject.size() <= 0) break;
 
    // 获得当前显示的对象
    t3DObject *pObject = &Model.pObject[i];
      
    // 判断该对象是否有纹理映射
    if(pObject->bHasTexture) {
 
      // 打开纹理映射
      glEnable(GL_TEXTURE_2D);
      glColor3ub(255, 255, 255);
      glBindTexture(GL_TEXTURE_2D, Model.texture[pObject->materialID]);
    } else {
 
      // 关闭纹理映射
      glDisable(GL_TEXTURE_2D);
      glColor3ub(255, 255, 255);
    }
    // 开始以g_ViewMode模式绘制
    glBegin(GL_TRIANGLES);          
      // 遍历所有的面
      for(int j = 0; j < pObject->numOfFaces; j++)
      {
        // 遍历三角形的所有点
        for(int whichVertex = 0; whichVertex < 3; whichVertex++)
        {
          // 获得面对每个点的索引
          int index = pObject->pFaces[j].vertIndex[whichVertex];
      
          // 给出法向量
          glNormal3f(pObject->pNormals[ index ].x, pObject->pNormals[ index ].y, pObject->pNormals[ index ].z);
        
          // 如果对象具有纹理
          if(pObject->bHasTexture) {
 
            // 确定是否有UVW纹理坐标
            if(pObject->pTexVerts) {
              glTexCoord2f(pObject->pTexVerts[ index ].x, pObject->pTexVerts[ index ].y);
            }
          } else {
 
            if(Model.pMaterials.size() && pObject->materialID >= 0) 
            {
              BYTE *pColor = Model.pMaterials[pObject->materialID].color;
              glColor3ub(pColor[0], pColor[1], pColor[2]);
            }
          }
          glVertex3f(pObject->pVerts[ index ].x, pObject->pVerts[ index ].y, pObject->pVerts[ index ].z);
        }
      }
 
    glEnd();                // 绘制结束
  }
}
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 3ds文件3D Studio Max的文件格式,而stp是一种通用的3D CAD文件格式。要将3ds文件换为stp,你需要使用一个3D文件换器软件,例如Autodesk Inventor或SolidWorks等软件。以下是使用Autodesk Inventor进行换的步骤: 1. 打开Autodesk Inventor软件并创建一个新文件。 2. 在“文件”菜单下,选择“导入”,然后选择你的3ds文件。 3. 在“导入”对话框中,选择“STP”作为输出文件格式,并设置其他选项。 4. 点击“导入”按钮,程序将开始换你的3ds文件为stp格式。 5. 保存你的stp文件,并在你的CAD软件中打开它进行查看和编辑。 请注意,换的结果可能不会完全符合你的期望,因为3ds和stp文件格式之间存在差异。检查并调整换结果是很重要的。 ### 回答2: 3DS文件和STP文件分别是不同的3D模型文件格式。3DS文件3D Studio Max软件常用的文件格式,而STP文件是Standard for the Exchange of Product model data(STEP)标准的文件格式。如果想将3DS文件换为STP文件,可以通过以下步骤进行: 1. 找到适合的换工具:在互联网上搜索并找到适合的3DS到STP文件格式换工具。有许多在线换工具和软件可以进行文件格式换。一些常用的工具包括Autodesk Fusion 360、FreeCAD、SolidWorks等。 2. 下载并安装换工具:根据所选的工具,下载并安装相应的软件。 3. 打开换工具:启动所下载的换工具,然后导入要换的3DS文件。通常,在软件的菜单栏中会有一个“导入”或“打开”选项,点击这些选项可以找到并选择3DS文件。 4. 确认换设置:在导入3DS文件之后,换工具通常会显示一些设置选项,例如文件的单位制、尺寸、材质等。根据需要进行相应的设置,并确保换选项设置为STP文件格式。 5. 进行文件换:设置完毕后,点击换按钮或菜单选项执行文件换操作。换时间可能取决于3DS文件的大小和计算机性能。 6. 保存换后的文件换完成后,换工具通常会要求您选择换后的文件保存路径和文件名。选择合适的位置和文件名,并保存STP格式的文件。 7. 检查换结果:在换完成后,您可以打开STP文件,使用相应的3D软件来验证换结果。确保文件换过程中没有丢失或损坏。 请注意,换的结果可能会因所使用的换工具和软件版本而有所不同。为了获得最佳的换结果,建议使用专业的CAD软件和最新版本的换工具。 ### 回答3: 要将3ds文件换成stp文件,需要使用专业的三维建模软件进行操作。以下是一种可能的换方法: 首先,打开支持3ds和stp文件格式的建模软件,例如SolidWorks、Autodesk Inventor或CATIA等。 在软件界面上,选择导入(Import)或打开(Open)选项,然后浏览系统中的文件,找到要换的3ds文件并选择打开。 软件会自动将3ds文件导入到场景中。请注意,由于3ds文件和stp文件的差异,部分对象可能需要进行调整或重建。 在软件中,选择导出(Export)选项,然后选择stp格式作为输出格式。 设置导出选项。通常可以选择导出的对象、单位制、尺寸等。根据实际需求进行设置并确认。 选择导出的文件路径和文件名,然后点击导出或确认按钮。 软件会开始将3ds文件换成stp格式,并将生成的stp文件保存到指定的路径中。 换完成后,可以在指定路径中找到已换的stp文件,可以使用stp文件在其他软件中进一步使用和编辑三维模型。 需要注意的是,由于3ds和stp是不同的文件格式,换过程中可能会有部分细节的损失或不一致。因此,在进行换前最好备份原始3ds文件,以免意外损失数据。 此外,还可以使用在线换工具或其他第三方软件来进行3ds到stp的换,具体步骤可能会有所不同。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值