#include "lc_global.h" #include "lc_scene.h" #include "lc_context.h" #include "pieceinf.h" #include "lc_texture.h" #include "lc_library.h" #include "lc_application.h" #include "object.h" lcScene::lcScene() : mRenderMeshes(0, 1024), mOpaqueMeshes(0, 1024), mTranslucentMeshes(0, 1024), mInterfaceObjects(0, 1024) { mActiveSubmodelInstance = nullptr; mAllowWireframe = true; } void lcScene::Begin(const lcMatrix44& ViewMatrix) { mViewMatrix = ViewMatrix; mActiveSubmodelInstance = nullptr; mDrawInterface = false; mRenderMeshes.RemoveAll(); mOpaqueMeshes.RemoveAll(); mTranslucentMeshes.RemoveAll(); mInterfaceObjects.RemoveAll(); mHasTexture = false; } void lcScene::End() { auto OpaqueMeshCompare = [this](int Index1, int Index2) { return mRenderMeshes[Index1].Mesh < mRenderMeshes[Index2].Mesh; }; std::sort(mOpaqueMeshes.begin(), mOpaqueMeshes.end(), OpaqueMeshCompare); auto TranslucentMeshCompare = [this](int Index1, int Index2) { return mRenderMeshes[Index1].Distance < mRenderMeshes[Index2].Distance; }; std::sort(mTranslucentMeshes.begin(), mTranslucentMeshes.end(), TranslucentMeshCompare); } void lcScene::AddMesh(lcMesh* Mesh, const lcMatrix44& WorldMatrix, int ColorIndex, lcRenderMeshState State, int Flags) { lcRenderMesh& RenderMesh = mRenderMeshes.Add(); RenderMesh.WorldMatrix = WorldMatrix; RenderMesh.Mesh = Mesh; RenderMesh.ColorIndex = ColorIndex; RenderMesh.State = State; RenderMesh.Distance = fabsf(lcMul31(WorldMatrix[3], mViewMatrix).z); RenderMesh.LodIndex = RenderMesh.Mesh->GetLodIndex(RenderMesh.Distance); bool Translucent = lcIsColorTranslucent(ColorIndex); if ((Flags & (LC_PIECE_HAS_SOLID | LC_PIECE_HAS_LINES)) || ((Flags & LC_PIECE_HAS_DEFAULT) && !Translucent)) mOpaqueMeshes.Add(mRenderMeshes.GetSize() - 1); if ((Flags & LC_PIECE_HAS_TRANSLUCENT) || ((Flags & LC_PIECE_HAS_DEFAULT) && Translucent)) mTranslucentMeshes.Add(mRenderMeshes.GetSize() - 1); if (Flags & LC_PIECE_HAS_TEXTURE) mHasTexture = true; } void lcScene::DrawRenderMeshes(lcContext* Context, int PrimitiveTypes, bool EnableNormals, bool DrawTranslucent, bool DrawTextured) const { const lcArray& Meshes = DrawTranslucent ? mTranslucentMeshes : mOpaqueMeshes; if (DrawTranslucent) Context->BeginTranslucent(); else Context->SetPolygonOffset(LC_POLYGON_OFFSET_OPAQUE); for (int MeshIndex : Meshes) { const lcRenderMesh& RenderMesh = mRenderMeshes[MeshIndex]; const lcMesh* Mesh = RenderMesh.Mesh; int LodIndex = RenderMesh.LodIndex; Context->BindMesh(Mesh); Context->SetWorldMatrix(RenderMesh.WorldMatrix); for (int SectionIdx = 0; SectionIdx < Mesh->mLods[LodIndex].NumSections; SectionIdx++) { lcMeshSection* Section = &Mesh->mLods[LodIndex].Sections[SectionIdx]; if ((Section->PrimitiveType & PrimitiveTypes) == 0 || (Section->Texture != nullptr) != DrawTextured) continue; int ColorIndex = Section->ColorIndex; if (Section->PrimitiveType & (LC_MESH_TRIANGLES | LC_MESH_TEXTURED_TRIANGLES)) { if (ColorIndex == gDefaultColor) ColorIndex = RenderMesh.ColorIndex; if (lcIsColorTranslucent(ColorIndex) != DrawTranslucent) continue; switch (RenderMesh.State) { case lcRenderMeshState::NORMAL: case lcRenderMeshState::HIGHLIGHT: Context->SetColorIndex(ColorIndex); break; case lcRenderMeshState::SELECTED: Context->SetColorIndexTinted(ColorIndex, LC_COLOR_SELECTED, 0.5f); break; case lcRenderMeshState::FOCUSED: Context->SetColorIndexTinted(ColorIndex, LC_COLOR_FOCUSED, 0.5f); break; case lcRenderMeshState::DISABLED: Context->SetColorIndexTinted(ColorIndex, LC_COLOR_DISABLED, 0.25f); break; } } else if (Section->PrimitiveType & (LC_MESH_LINES | LC_MESH_TEXTURED_LINES)) { switch (RenderMesh.State) { case lcRenderMeshState::NORMAL: if (ColorIndex == gEdgeColor) Context->SetEdgeColorIndex(RenderMesh.ColorIndex); else Context->SetColorIndex(ColorIndex); break; case lcRenderMeshState::SELECTED: Context->SetInterfaceColor(LC_COLOR_SELECTED); break; case lcRenderMeshState::FOCUSED: Context->SetInterfaceColor(LC_COLOR_FOCUSED); break; case lcRenderMeshState::HIGHLIGHT: Context->SetInterfaceColor(LC_COLOR_HIGHLIGHT); break; case lcRenderMeshState::DISABLED: Context->SetInterfaceColor(LC_COLOR_DISABLED); break; } } else if (Section->PrimitiveType == LC_MESH_CONDITIONAL_LINES) { lcMatrix44 WorldViewProjectionMatrix = lcMul(RenderMesh.WorldMatrix, lcMul(mViewMatrix, Context->GetProjectionMatrix())); lcVertex* VertexBuffer = (lcVertex*)Mesh->mVertexData; int IndexBufferOffset = Mesh->mIndexCacheOffset != -1 ? Mesh->mIndexCacheOffset : 0; int VertexBufferOffset = Mesh->mVertexCacheOffset != -1 ? Mesh->mVertexCacheOffset : 0; Context->SetVertexFormat(VertexBufferOffset, 3, 1, 0, 0, EnableNormals); if (Mesh->mIndexType == GL_UNSIGNED_SHORT) { quint16* Indices = (quint16*)((char*)Mesh->mIndexData + Section->IndexOffset); for (int i = 0; i < Section->NumIndices; i += 4) { lcVector3 p1 = lcMul31(VertexBuffer[Indices[i + 0]].Position, WorldViewProjectionMatrix); lcVector3 p2 = lcMul31(VertexBuffer[Indices[i + 1]].Position, WorldViewProjectionMatrix); lcVector3 p3 = lcMul31(VertexBuffer[Indices[i + 2]].Position, WorldViewProjectionMatrix); lcVector3 p4 = lcMul31(VertexBuffer[Indices[i + 3]].Position, WorldViewProjectionMatrix); if (((p1.y - p2.y) * (p3.x - p1.x) + (p2.x - p1.x) * (p3.y - p1.y)) * ((p1.y - p2.y) * (p4.x - p1.x) + (p2.x - p1.x) * (p4.y - p1.y)) >= 0) Context->DrawIndexedPrimitives(GL_LINES, 2, Mesh->mIndexType, IndexBufferOffset + Section->IndexOffset + i * sizeof(quint16)); } } else { quint32* Indices = (quint32*)((char*)Mesh->mIndexData + Section->IndexOffset); for (int i = 0; i < Section->NumIndices; i += 4) { lcVector3 p1 = lcMul31(VertexBuffer[Indices[i + 0]].Position, WorldViewProjectionMatrix); lcVector3 p2 = lcMul31(VertexBuffer[Indices[i + 1]].Position, WorldViewProjectionMatrix); lcVector3 p3 = lcMul31(VertexBuffer[Indices[i + 2]].Position, WorldViewProjectionMatrix); lcVector3 p4 = lcMul31(VertexBuffer[Indices[i + 3]].Position, WorldViewProjectionMatrix); if (((p1.y - p2.y) * (p3.x - p1.x) + (p2.x - p1.x) * (p3.y - p1.y)) * ((p1.y - p2.y) * (p4.x - p1.x) + (p2.x - p1.x) * (p4.y - p1.y)) >= 0) Context->DrawIndexedPrimitives(GL_LINES, 2, Mesh->mIndexType, IndexBufferOffset + Section->IndexOffset + i * sizeof(quint32)); } } continue; } lcTexture* Texture = Section->Texture; int VertexBufferOffset = Mesh->mVertexCacheOffset != -1 ? Mesh->mVertexCacheOffset : 0; int IndexBufferOffset = Mesh->mIndexCacheOffset != -1 ? Mesh->mIndexCacheOffset : 0; if (!Texture) { Context->SetVertexFormat(VertexBufferOffset, 3, 1, 0, 0, EnableNormals); } else { VertexBufferOffset += Mesh->mNumVertices * sizeof(lcVertex); Context->SetVertexFormat(VertexBufferOffset, 3, 1, 2, 0, EnableNormals); Context->BindTexture2D(Texture->mTexture); } GLenum DrawPrimitiveType = Section->PrimitiveType & (LC_MESH_TRIANGLES | LC_MESH_TEXTURED_TRIANGLES) ? GL_TRIANGLES : GL_LINES; Context->DrawIndexedPrimitives(DrawPrimitiveType, Section->NumIndices, Mesh->mIndexType, IndexBufferOffset + Section->IndexOffset); } #ifdef QT_DEBUG const bool DrawNormals = false; if (DrawNormals) { lcVertex* VertexBuffer = (lcVertex*)Mesh->mVertexData; lcVector3* Vertices = (lcVector3*)malloc(Mesh->mNumVertices * 2 * sizeof(lcVector3)); for (int VertexIdx = 0; VertexIdx < Mesh->mNumVertices; VertexIdx++) { Vertices[VertexIdx * 2] = VertexBuffer[VertexIdx].Position; Vertices[VertexIdx * 2 + 1] = VertexBuffer[VertexIdx].Position + lcUnpackNormal(VertexBuffer[VertexIdx].Normal); } Context->SetVertexBufferPointer(Vertices); Context->SetVertexFormatPosition(3); Context->DrawPrimitives(GL_LINES, 0, Mesh->mNumVertices * 2); free(Vertices); } #endif } if (DrawTranslucent) Context->EndTranslucent(); } void lcScene::Draw(lcContext* Context) const { // TODO: find a better place for these updates lcGetPiecesLibrary()->UpdateBuffers(Context); lcGetPiecesLibrary()->UploadTextures(Context); Context->SetViewMatrix(mViewMatrix); const bool DrawConditional = false; const lcPreferences& Preferences = lcGetPreferences(); lcShadingMode ShadingMode = Preferences.mShadingMode; if (ShadingMode == LC_SHADING_WIREFRAME && !mAllowWireframe) ShadingMode = LC_SHADING_FLAT; if (ShadingMode == LC_SHADING_WIREFRAME) { Context->BindTexture2D(0); Context->SetMaterial(LC_MATERIAL_UNLIT_COLOR); int UntexturedPrimitives = LC_MESH_LINES; if (DrawConditional) UntexturedPrimitives |= LC_MESH_CONDITIONAL_LINES; DrawRenderMeshes(Context, UntexturedPrimitives, false, false, false); if (mHasTexture) { Context->SetMaterial(LC_MATERIAL_UNLIT_TEXTURE_DECAL); DrawRenderMeshes(Context, LC_MESH_TEXTURED_LINES, false, false, false); Context->BindTexture2D(0); } } else if (ShadingMode == LC_SHADING_FLAT) { bool DrawLines = Preferences.mDrawEdgeLines && Preferences.mLineWidth != 0.0f; Context->BindTexture2D(0); Context->SetMaterial(LC_MATERIAL_UNLIT_COLOR); int UntexturedPrimitives = LC_MESH_TRIANGLES; if (DrawLines) { UntexturedPrimitives |= LC_MESH_LINES; if (DrawConditional) UntexturedPrimitives |= LC_MESH_CONDITIONAL_LINES; } DrawRenderMeshes(Context, UntexturedPrimitives, false, false, false); if (!mTranslucentMeshes.IsEmpty()) DrawRenderMeshes(Context, LC_MESH_TRIANGLES, false, true, false); if (mHasTexture) { Context->SetMaterial(LC_MATERIAL_UNLIT_TEXTURE_DECAL); if (DrawLines) DrawRenderMeshes(Context, LC_MESH_TEXTURED_LINES, false, false, false); DrawRenderMeshes(Context, LC_MESH_TEXTURED_TRIANGLES, false, false, true); if (!mTranslucentMeshes.IsEmpty()) DrawRenderMeshes(Context, LC_MESH_TEXTURED_TRIANGLES, false, true, true); Context->BindTexture2D(0); } } else { bool DrawLines = Preferences.mDrawEdgeLines && Preferences.mLineWidth != 0.0f; Context->BindTexture2D(0); if (DrawLines) { int LinePrimitives = LC_MESH_LINES; if (DrawConditional) LinePrimitives |= LC_MESH_CONDITIONAL_LINES; Context->SetMaterial(LC_MATERIAL_UNLIT_COLOR); DrawRenderMeshes(Context, LinePrimitives, false, false, false); } Context->SetMaterial(LC_MATERIAL_FAKELIT_COLOR); DrawRenderMeshes(Context, LC_MESH_TRIANGLES, true, false, false); if (!mTranslucentMeshes.IsEmpty()) DrawRenderMeshes(Context, LC_MESH_TRIANGLES, true, true, false); if (mHasTexture) { if (DrawLines) { Context->SetMaterial(LC_MATERIAL_UNLIT_TEXTURE_DECAL); DrawRenderMeshes(Context, LC_MESH_TEXTURED_LINES, false, false, true); } Context->SetMaterial(LC_MATERIAL_FAKELIT_TEXTURE_DECAL); DrawRenderMeshes(Context, LC_MESH_TEXTURED_TRIANGLES, true, false, true); if (!mTranslucentMeshes.IsEmpty()) DrawRenderMeshes(Context, LC_MESH_TEXTURED_TRIANGLES, true, true, true); Context->BindTexture2D(0); } } } void lcScene::DrawInterfaceObjects(lcContext* Context) const { for (const lcObject* Object : mInterfaceObjects) Object->DrawInterface(Context, *this); }