leocad/common/pieceinf.cpp

406 lines
10 KiB
C++
Raw Normal View History

#include "lc_global.h"
2012-03-29 03:10:55 +02:00
#include "lc_math.h"
2012-04-14 01:41:58 +02:00
#include "lc_mesh.h"
#include "lc_colors.h"
2012-10-12 01:55:55 +02:00
#include "lc_texture.h"
2011-09-07 23:06:51 +02:00
#include "pieceinf.h"
2012-10-02 03:23:44 +02:00
#include "lc_library.h"
2011-09-07 23:06:51 +02:00
#include "lc_application.h"
2014-12-24 16:52:52 +01:00
#include "lc_model.h"
#include "project.h"
2014-12-24 16:52:52 +01:00
#include "lc_context.h"
#include "lc_synth.h"
2016-05-02 21:13:54 +02:00
#include "lc_file.h"
2015-04-05 01:52:20 +02:00
#include <locale.h>
2011-09-07 23:06:51 +02:00
PieceInfo::PieceInfo()
2011-09-07 23:06:51 +02:00
{
mZipFileType = LC_NUM_ZIPFILES;
mZipFileIndex = -1;
mFlags = 0;
2017-01-23 04:28:05 +01:00
mState = LC_PIECEINFO_UNLOADED;
2012-10-12 01:55:55 +02:00
mRefCount = 0;
mMesh = NULL;
2014-12-24 16:52:52 +01:00
mModel = NULL;
mProject = NULL;
mSynthInfo = NULL;
2011-09-07 23:06:51 +02:00
}
2012-04-14 01:41:58 +02:00
PieceInfo::~PieceInfo()
2011-09-07 23:06:51 +02:00
{
delete mSynthInfo;
2017-01-23 04:28:05 +01:00
if (mState == LC_PIECEINFO_LOADED)
2012-10-12 02:21:04 +02:00
Unload();
2011-09-07 23:06:51 +02:00
}
2014-12-26 18:09:11 +01:00
QString PieceInfo::GetSaveID() const
{
if (mFlags & (LC_PIECE_MODEL | LC_PIECE_PROJECT))
return QString::fromLatin1(m_strName);
2014-12-26 18:09:11 +01:00
return QString::fromLatin1(m_strName) + QLatin1String(".DAT");
}
2017-01-23 04:28:05 +01:00
void PieceInfo::SetMesh(lcMesh* Mesh)
{
mBoundingBox = Mesh->mBoundingBox;
mMesh = Mesh;
}
void PieceInfo::SetPlaceholder()
{
2016-02-19 18:53:54 +01:00
mBoundingBox.Min = lcVector3(-10.0f, -10.0f, -24.0f);
mBoundingBox.Max = lcVector3(10.0f, 10.0f, 4.0f);
mFlags = LC_PIECE_PLACEHOLDER | LC_PIECE_HAS_DEFAULT | LC_PIECE_HAS_LINES;
mModel = NULL;
mProject = NULL;
delete mMesh;
mMesh = NULL;
}
void PieceInfo::SetModel(lcModel* Model, bool UpdateMesh)
2014-12-24 16:52:52 +01:00
{
if (mModel != Model)
{
mFlags = LC_PIECE_MODEL;
mModel = Model;
}
2014-12-24 16:52:52 +01:00
strncpy(m_strName, Model->GetProperties().mName.toUpper().toLatin1().data(), sizeof(m_strName));
m_strName[sizeof(m_strName)-1] = 0;
strncpy(m_strDescription, Model->GetProperties().mName.toLatin1().data(), sizeof(m_strDescription));
m_strDescription[sizeof(m_strDescription)-1] = 0;
2015-03-21 21:12:04 +01:00
const QStringList& MeshLines = Model->GetFileLines();
if (UpdateMesh && !MeshLines.isEmpty())
{
lcMemFile PieceFile;
2016-02-19 18:53:54 +01:00
foreach(const QString& Line, MeshLines)
{
QByteArray Buffer = Line.toLatin1();
PieceFile.WriteBuffer(Buffer.constData(), Buffer.size());
PieceFile.WriteBuffer("\r\n", 2);
}
lcLibraryMeshData MeshData;
lcArray<lcLibraryTextureMap> TextureStack;
PieceFile.Seek(0, SEEK_SET);
2017-02-21 17:09:13 +01:00
bool Ret = lcGetPiecesLibrary()->ReadMeshData(PieceFile, lcMatrix44Identity(), 16, false, TextureStack, MeshData, LC_MESHDATA_SHARED, true);
if (Ret && !MeshData.IsEmpty())
lcGetPiecesLibrary()->CreateMesh(this, MeshData);
}
2014-12-24 16:52:52 +01:00
}
void PieceInfo::SetProject(Project* Project, const char* PieceName)
{
if (mProject != Project)
{
mFlags = LC_PIECE_PROJECT;
mProject = Project;
2017-01-23 04:28:05 +01:00
mState = LC_PIECEINFO_LOADED;
}
strncpy(m_strName, PieceName, sizeof(m_strName));
m_strName[sizeof(m_strName)-1] = 0;
strncpy(m_strDescription, Project->GetFileName().toLatin1().data(), sizeof(m_strDescription));
m_strDescription[sizeof(m_strDescription)-1] = 0;
}
2014-12-24 16:52:52 +01:00
bool PieceInfo::IncludesModel(const lcModel* Model) const
{
if (mFlags & LC_PIECE_MODEL)
{
if (mModel == Model)
return true;
return mModel->IncludesModel(Model);
}
return false;
}
void PieceInfo::CreatePlaceholder(const char* Name)
{
strncpy(m_strName, Name, sizeof(m_strName));
m_strName[sizeof(m_strName)-1] = 0;
strncpy(m_strDescription, Name, sizeof(m_strDescription));
m_strDescription[sizeof(m_strDescription)-1] = 0;
SetPlaceholder();
}
2012-10-12 01:55:55 +02:00
void PieceInfo::Load()
2011-09-07 23:06:51 +02:00
{
2017-01-23 04:28:05 +01:00
if ((mFlags & (LC_PIECE_MODEL | LC_PIECE_PROJECT)) == 0)
2016-02-19 18:53:54 +01:00
{
2017-01-23 04:28:05 +01:00
mState = LC_PIECEINFO_LOADING; // todo: mutex lock when changing load state
if (mFlags & LC_PIECE_PLACEHOLDER)
{
2017-01-23 04:28:05 +01:00
if (lcGetPiecesLibrary()->LoadPieceData(this))
mFlags &= ~LC_PIECE_PLACEHOLDER;
else
{
mFlags |= LC_PIECE_HAS_DEFAULT | LC_PIECE_HAS_LINES;
mBoundingBox = gPlaceholderMesh->mBoundingBox;
}
}
2017-01-23 04:28:05 +01:00
else
lcGetPiecesLibrary()->LoadPieceData(this);
2016-02-19 18:53:54 +01:00
}
2017-01-23 04:28:05 +01:00
mState = LC_PIECEINFO_LOADED;
2014-09-11 21:55:34 +02:00
}
2011-09-07 23:06:51 +02:00
2012-10-12 01:55:55 +02:00
void PieceInfo::Unload()
2011-09-07 23:06:51 +02:00
{
2014-12-24 16:52:52 +01:00
if (mMesh)
2012-10-12 01:55:55 +02:00
{
2015-05-24 06:36:25 +02:00
for (int LodIdx = 0; LodIdx < LC_NUM_MESH_LODS; LodIdx++)
2014-12-24 16:52:52 +01:00
{
2015-05-24 06:36:25 +02:00
for (int SectionIdx = 0; SectionIdx < mMesh->mLods[LodIdx].NumSections; SectionIdx++)
{
lcMeshSection& Section = mMesh->mLods[LodIdx].Sections[SectionIdx];
2012-10-12 01:55:55 +02:00
2015-05-24 06:36:25 +02:00
if (Section.Texture)
Section.Texture->Release();
}
2014-12-24 16:52:52 +01:00
}
2012-10-12 01:55:55 +02:00
2014-12-24 16:52:52 +01:00
delete mMesh;
mMesh = NULL;
}
2017-01-23 04:28:05 +01:00
mState = LC_PIECEINFO_UNLOADED;
mModel = NULL;
2015-02-22 03:39:15 +01:00
if (IsModel())
lcGetPiecesLibrary()->RemovePiece(this);
else if (IsProject())
{
delete mProject;
mProject = NULL;
lcGetPiecesLibrary()->RemovePiece(this);
}
2011-09-07 23:06:51 +02:00
}
bool PieceInfo::MinIntersectDist(const lcVector3& Start, const lcVector3& End, float& MinDistance) const
2014-12-26 16:44:46 +01:00
{
bool Intersect = false;
2014-12-26 16:44:46 +01:00
if (mFlags & (LC_PIECE_PLACEHOLDER | LC_PIECE_MODEL | LC_PIECE_PROJECT))
{
float Distance;
if (!lcBoundingBoxRayIntersectDistance(mBoundingBox.Min, mBoundingBox.Max, Start, End, &Distance, NULL) || (Distance >= MinDistance))
return false;
if (mFlags & LC_PIECE_PLACEHOLDER)
return true;
if (mFlags & LC_PIECE_MODEL)
Intersect |= mModel->SubModelMinIntersectDist(Start, End, MinDistance);
else if (mFlags & LC_PIECE_PROJECT)
{
lcModel* Model = mProject->GetMainModel();
if (Model)
Intersect |= Model->SubModelMinIntersectDist(Start, End, MinDistance);
}
}
if (mMesh)
Intersect = mMesh->MinIntersectDist(Start, End, MinDistance);
return Intersect;
2014-12-26 16:44:46 +01:00
}
bool PieceInfo::BoxTest(const lcMatrix44& WorldMatrix, const lcVector4 WorldPlanes[6]) const
{
lcMatrix44 InverseWorldMatrix = lcMatrix44AffineInverse(WorldMatrix);
const int NumCorners = 8;
const int NumPlanes = 6;
lcVector4 LocalPlanes[NumPlanes];
for (int PlaneIdx = 0; PlaneIdx < NumPlanes; PlaneIdx++)
{
lcVector3 PlaneNormal = lcMul30(WorldPlanes[PlaneIdx], InverseWorldMatrix);
LocalPlanes[PlaneIdx] = lcVector4(PlaneNormal, WorldPlanes[PlaneIdx][3] - lcDot3(InverseWorldMatrix[3], PlaneNormal));
}
2016-02-19 18:53:54 +01:00
lcVector3 Box[NumCorners];
lcGetBoxCorners(mBoundingBox, Box);
2014-12-26 16:44:46 +01:00
int Outcodes[NumCorners];
for (int CornerIdx = 0; CornerIdx < NumCorners; CornerIdx++)
{
Outcodes[CornerIdx] = 0;
for (int PlaneIdx = 0; PlaneIdx < NumPlanes; PlaneIdx++)
{
if (lcDot3(Box[CornerIdx], LocalPlanes[PlaneIdx]) + LocalPlanes[PlaneIdx][3] > 0)
Outcodes[CornerIdx] |= 1 << PlaneIdx;
}
}
int OutcodesOR = 0, OutcodesAND = 0x3f;
for (int CornerIdx = 0; CornerIdx < NumCorners; CornerIdx++)
{
OutcodesAND &= Outcodes[CornerIdx];
OutcodesOR |= Outcodes[CornerIdx];
}
if (OutcodesAND != 0)
return false;
if (OutcodesOR == 0)
return true;
if (mFlags & LC_PIECE_PLACEHOLDER)
return gPlaceholderMesh->IntersectsPlanes(LocalPlanes);
if (mMesh && mMesh->IntersectsPlanes(LocalPlanes))
return true;
if (mFlags & LC_PIECE_MODEL)
return mModel->SubModelBoxTest(LocalPlanes);
else if (mFlags & LC_PIECE_PROJECT)
{
lcModel* Model = mProject->GetMainModel();
return Model ? Model->SubModelBoxTest(LocalPlanes) : false;
}
return false;
2014-12-26 16:44:46 +01:00
}
2011-09-07 23:06:51 +02:00
// Zoom extents for the preview window and print catalog
void PieceInfo::ZoomExtents(const lcMatrix44& ProjectionMatrix, lcMatrix44& ViewMatrix, float* EyePos) const
2011-09-07 23:06:51 +02:00
{
2016-02-19 18:53:54 +01:00
lcVector3 Points[8];
lcGetBoxCorners(mBoundingBox, Points);
lcVector3 Center = (mBoundingBox.Min + mBoundingBox.Max) / 2.0f;
2012-06-21 02:41:53 +02:00
lcVector3 Position;
2011-09-07 23:06:51 +02:00
2012-06-21 02:41:53 +02:00
if (EyePos)
Position = lcVector3(EyePos[0], EyePos[1], EyePos[2]);
else
2014-08-30 21:48:36 +02:00
Position = lcVector3(-250.0f, -250.0f, 75.0f);
2012-06-21 02:41:53 +02:00
Position += Center;
2011-09-07 23:06:51 +02:00
2012-06-21 02:41:53 +02:00
lcMatrix44 ModelView = lcMatrix44LookAt(Position, Center, lcVector3(0, 0, 1));
Position = lcZoomExtents(Position, ModelView, ProjectionMatrix, Points, 8);
ViewMatrix = lcMatrix44LookAt(Position, Center, lcVector3(0, 0, 1));
2011-09-07 23:06:51 +02:00
if (EyePos)
{
2012-06-21 02:41:53 +02:00
EyePos[0] = Position[0];
EyePos[1] = Position[1];
EyePos[2] = Position[2];
2011-09-07 23:06:51 +02:00
}
}
void PieceInfo::AddRenderMesh(lcScene& Scene)
2014-12-24 16:52:52 +01:00
{
if (!mMesh)
2014-12-24 16:52:52 +01:00
return;
2017-03-20 06:59:55 +01:00
lcRenderMesh& RenderMesh = Scene.mRenderMeshes.Add();
2014-12-24 16:52:52 +01:00
RenderMesh.WorldMatrix = lcMatrix44Identity();
RenderMesh.Mesh = mMesh;
RenderMesh.ColorIndex = gDefaultColor;
2015-05-04 02:51:41 +02:00
RenderMesh.State = LC_RENDERMESH_NONE;
2015-05-24 06:36:25 +02:00
RenderMesh.Distance = fabsf(Scene.mViewMatrix.r[3].z);
RenderMesh.LodIndex = mMesh->GetLodIndex(RenderMesh.Distance);
2014-12-24 16:52:52 +01:00
if (mFlags & (LC_PIECE_HAS_SOLID | LC_PIECE_HAS_DEFAULT | LC_PIECE_HAS_LINES))
2017-03-20 06:59:55 +01:00
Scene.mOpaqueMeshes.Add(Scene.mRenderMeshes.GetSize() - 1);
2014-12-24 16:52:52 +01:00
if (mFlags & LC_PIECE_HAS_TRANSLUCENT)
2017-03-20 06:59:55 +01:00
Scene.mTranslucentMeshes.Add(Scene.mRenderMeshes.GetSize() - 1);
2014-04-23 16:53:43 +02:00
}
2014-12-30 17:30:12 +01:00
void PieceInfo::AddRenderMeshes(lcScene& Scene, const lcMatrix44& WorldMatrix, int ColorIndex, bool Focused, bool Selected) const
{
if (mMesh || (mFlags & LC_PIECE_PLACEHOLDER))
{
2017-03-20 06:59:55 +01:00
lcRenderMesh& RenderMesh = Scene.mRenderMeshes.Add();
RenderMesh.WorldMatrix = WorldMatrix;
RenderMesh.Mesh = (mFlags & LC_PIECE_PLACEHOLDER) ? gPlaceholderMesh : mMesh;
RenderMesh.ColorIndex = ColorIndex;
2015-05-04 02:51:41 +02:00
RenderMesh.State = Focused ? LC_RENDERMESH_FOCUSED : (Selected ? LC_RENDERMESH_SELECTED : LC_RENDERMESH_NONE);
2015-05-24 06:36:25 +02:00
RenderMesh.Distance = fabsf(lcMul31(WorldMatrix[3], Scene.mViewMatrix).z);
2015-07-04 23:52:09 +02:00
RenderMesh.LodIndex = RenderMesh.Mesh->GetLodIndex(RenderMesh.Distance);
bool Translucent = lcIsColorTranslucent(ColorIndex);
if ((mFlags & (LC_PIECE_HAS_SOLID | LC_PIECE_HAS_LINES)) || ((mFlags & LC_PIECE_HAS_DEFAULT) && !Translucent))
2017-03-20 06:59:55 +01:00
Scene.mOpaqueMeshes.Add(Scene.mRenderMeshes.GetSize() - 1);
if ((mFlags & LC_PIECE_HAS_TRANSLUCENT) || ((mFlags & LC_PIECE_HAS_DEFAULT) && Translucent))
2017-03-20 06:59:55 +01:00
Scene.mTranslucentMeshes.Add(Scene.mRenderMeshes.GetSize() - 1);
}
if (mFlags & LC_PIECE_MODEL)
mModel->SubModelAddRenderMeshes(Scene, WorldMatrix, ColorIndex, Focused, Selected);
else if (mFlags & LC_PIECE_PROJECT)
{
lcModel* Model = mProject->GetMainModel();
if (Model)
Model->SubModelAddRenderMeshes(Scene, WorldMatrix, ColorIndex, Focused, Selected);
}
}
void PieceInfo::GetPartsList(int DefaultColorIndex, lcPartsList& PartsList) const
2014-12-30 17:30:12 +01:00
{
if (mFlags & LC_PIECE_MODEL)
mModel->GetPartsList(DefaultColorIndex, PartsList);
else if (mFlags & LC_PIECE_PROJECT)
{
lcModel* Model = mProject->GetMainModel();
if (Model)
Model->GetPartsList(DefaultColorIndex, PartsList);
}
else if (DefaultColorIndex < gNumUserColors)
PartsList[this][DefaultColorIndex]++;
2014-12-30 17:30:12 +01:00
}
void PieceInfo::GetModelParts(const lcMatrix44& WorldMatrix, int DefaultColorIndex, lcArray<lcModelPartsEntry>& ModelParts) const
{
if (mFlags & LC_PIECE_MODEL)
{
mModel->GetModelParts(WorldMatrix, DefaultColorIndex, ModelParts);
return;
}
else if (mFlags & LC_PIECE_PROJECT)
{
lcModel* Model = mProject->GetMainModel();
if (Model)
Model->GetModelParts(WorldMatrix, DefaultColorIndex, ModelParts);
return;
}
2014-12-30 17:30:12 +01:00
lcModelPartsEntry& ModelPartsEntry = ModelParts.Add();
ModelPartsEntry.WorldMatrix = WorldMatrix;
ModelPartsEntry.ColorIndex = DefaultColorIndex;
ModelPartsEntry.Info = const_cast<PieceInfo*>(this);
}
void PieceInfo::UpdateBoundingBox(lcArray<lcModel*>& UpdatedModels)
{
if (mFlags & LC_PIECE_MODEL)
mModel->UpdatePieceInfo(UpdatedModels);
else if (mFlags & LC_PIECE_PROJECT)
mProject->UpdatePieceInfo(this);
}