#include "lc_global.h" #include "lc_math.h" #include "lc_mesh.h" #include #include "opengl.h" #include "pieceinf.h" #include "lc_texture.h" #include "piece.h" #include "camera.h" #include "light.h" #include "group.h" #include "project.h" #include "image.h" #include "lc_mainwindow.h" #include "view.h" #include "lc_library.h" #include "lc_application.h" #include "lc_profile.h" #include "preview.h" #include "lc_qmodellistdialog.h" Project::Project() { mModified = false; mActiveModel = new lcModel(tr("Model #1")); mActiveModel->SetSaved(); mModels.Add(mActiveModel); } Project::~Project() { mModels.DeleteAll(); } bool Project::IsModified() const { if (mModified) return true; for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) if (mModels[ModelIdx]->IsModified()) return true; return false; } QString Project::GetTitle() const { return mFileName.isEmpty() ? tr("New Project.ldr") : QFileInfo(mFileName).fileName(); } void Project::SetActiveModel(int ModelIndex) { if (ModelIndex < 0 || ModelIndex >= mModels.GetSize()) return; for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) if (ModelIdx != ModelIndex) mModels[ModelIdx]->CalculateStep(LC_STEP_MAX); mActiveModel = mModels[ModelIndex]; mActiveModel->CalculateStep(mActiveModel->GetCurrentStep()); mActiveModel->UpdateInterface(); gMainWindow->UpdateModels(); const lcArray& Views = gMainWindow->GetViews(); for (int ViewIdx = 0; ViewIdx < Views.GetSize(); ViewIdx++) Views[ViewIdx]->SetModel(lcGetActiveModel()); } bool Project::IsModelNameValid(const QString& Name) const { if (Name.isEmpty()) return false; for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) if (mModels[ModelIdx]->GetProperties().mName == Name) return false; return true; } void Project::CreateNewModel() { const QString Prefix = tr("Model #"); int Max = 0; for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) { QString Name = mModels[ModelIdx]->GetProperties().mName; if (Name.startsWith(Prefix)) { QString NumberString = Name.mid(Prefix.length()); QTextStream Stream(&NumberString); int Number; Stream >> Number; Max = qMax(Max, Number); } } QString Name = Prefix + QString::number(Max + 1); for (;;) { bool Ok = false; Name = QInputDialog::getText(gMainWindow->mHandle, tr("New Model"), tr("Name:"), QLineEdit::Normal, Name, &Ok); if (!Ok) return; if (IsModelNameValid(Name)) break; if (Name.isEmpty()) QMessageBox::information(gMainWindow->mHandle, tr("Empty Name"), tr("The model name cannot be empty.")); else QMessageBox::information(gMainWindow->mHandle, tr("Duplicate Model"), tr("A model named '%1' already exists in this project, please enter an unique name.").arg(Name)); } if (!Name.isEmpty()) { mModified = true; lcModel* Model = new lcModel(Name); Model->SetSaved(); mModels.Add(Model); SetActiveModel(mModels.GetSize() - 1); gMainWindow->UpdateTitle(); } } void Project::ShowModelListDialog() { QList> Models; Models.reserve(mModels.GetSize()); for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) { lcModel* Model = mModels[ModelIdx]; Models.append(QPair(Model->GetProperties().mName, Model)); } lcQModelListDialog Dialog(gMainWindow->mHandle, Models); if (Dialog.exec() != QDialog::Accepted || Models.isEmpty()) return; lcArray NewModels; for (QList>::iterator it = Models.begin(); it != Models.end(); it++) { lcModel* Model = it->second; if (!Model) { Model = new lcModel(it->first); Model->SetSaved(); mModified = true; } else if (Model->GetProperties().mName != it->first) { Model->SetName(it->first); mModified = true; } NewModels.Add(Model); } for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) { lcModel* Model = mModels[ModelIdx]; if (NewModels.FindIndex(Model) == -1) { delete Model; mModified = true; } } mModels = NewModels; SetActiveModel(Dialog.mActiveModel); gMainWindow->UpdateTitle(); } bool Project::Load(const QString& FileName) { QFile File(FileName); if (!File.open(QIODevice::ReadOnly)) { QMessageBox::warning(gMainWindow->mHandle, tr("Error"), tr("Error reading file '%1':\n%2").arg(FileName, File.errorString())); return false; } mModels.DeleteAll(); QString Extension = QFileInfo(FileName).suffix().toLower(); if (Extension == QLatin1String("dat") || Extension == QLatin1String("ldr") || Extension == QLatin1String("mpd")) { QTextStream Stream(&File); while (!Stream.atEnd()) { lcModel* Model = new lcModel(QString()); mModels.Add(Model); Model->LoadLDraw(Stream); Model->SetSaved(); } } else { lcMemFile MemFile; QByteArray FileData = File.readAll(); MemFile.WriteBuffer(FileData.constData(), FileData.size()); MemFile.Seek(0, SEEK_SET); lcModel* Model = new lcModel(QString()); if (Model->LoadBinary(&MemFile)) { mModels.Add(Model); Model->SetSaved(); } else delete Model; } if (mModels.IsEmpty()) return false; for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) { lcModel* Model = mModels[ModelIdx]; if (Model->GetProperties().mName.isEmpty()) Model->SetName(tr("Model #%1").arg(QString::number(ModelIdx + 1))); // todo: validate model names Model->CreatePieceInfo(); } mActiveModel = mModels[0]; /* if (datfile || mpdfile) { gMainWindow->UpdateCurrentStep(); gMainWindow->UpdateFocusObject(GetFocusObject()); UpdateSelection(); const lcArray& Views = gMainWindow->GetViews(); for (int ViewIdx = 0; ViewIdx < Views.GetSize(); ViewIdx++) Views[ViewIdx]->ZoomExtents(); Success = true; } else { Success = FileLoad(&file, false, false); if (!bMerge) gMainWindow->UpdateFocusObject(GetFocusObject()); if (!bMerge) { const lcArray& Views = gMainWindow->GetViews(); for (int ViewIdx = 0; ViewIdx < Views.GetSize(); ViewIdx++) { View* view = Views[ViewIdx]; if (!view->mCamera->IsSimple()) view->SetDefaultCamera(); if (!bUndo) view->ZoomExtents(); } } gMainWindow->UpdateLockSnap(); gMainWindow->UpdateSnap(); gMainWindow->UpdateCameraMenu(); UpdateSelection(); gMainWindow->UpdateCurrentStep(); gMainWindow->UpdateAllViews(); } */ mFileName = FileName; mModified = false; return true; } bool Project::Save(const QString& FileName) { QFile File(FileName); if (!File.open(QIODevice::WriteOnly)) { QMessageBox::warning(gMainWindow->mHandle, tr("Error"), tr("Error writing to file '%1':\n%2").arg(FileName, File.errorString())); return false; } QTextStream Stream(&File); bool MPD = mModels.GetSize() > 1; for (int ModelIdx = 0; ModelIdx < mModels.GetSize(); ModelIdx++) { lcModel* Model = mModels[ModelIdx]; if (MPD) Stream << QLatin1String("0 FILE ") << Model->GetProperties().mName << QLatin1String("\r\n"); Model->SaveLDraw(Stream, false); Model->SetSaved(); if (MPD) Stream << QLatin1String("0 ENDFILE\r\n"); } mFileName = FileName; mModified = false; return true; } void Project::Merge(Project* Other) { for (int ModelIdx = 0; ModelIdx < Other->mModels.GetSize(); ModelIdx++) mModels.Add(Other->mModels[ModelIdx]); mModified = true; } /* void Project::LoadDefaults() // todo: Change the interface in SetProject() instead { mProperties.LoadDefaults(); gMainWindow->SetColorIndex(lcGetColorIndex(4)); gMainWindow->SetTool(LC_TOOL_SELECT); gMainWindow->SetAddKeys(false); gMainWindow->UpdateUndoRedo(NULL, NULL); gMainWindow->UpdateLockSnap(); gMainWindow->UpdateSnap(); mCurrentStep = 1; gMainWindow->UpdateCurrentStep(); const lcArray& Views = gMainWindow->GetViews(); for (int i = 0; i < Views.GetSize(); i++) if (!Views[i]->mCamera->IsSimple()) Views[i]->SetDefaultCamera(); gMainWindow->UpdateCameraMenu(); UpdateSelection(); gMainWindow->UpdateFocusObject(NULL); } */ void Project::SaveImage() { /* lcImageDialogOptions Options; lcStep LastStep = GetLastStep(); Options.Width = lcGetProfileInt(LC_PROFILE_IMAGE_WIDTH); Options.Height = lcGetProfileInt(LC_PROFILE_IMAGE_HEIGHT); Options.Start = mCurrentStep; Options.End = LastStep; if (!mFileName.isEmpty()) { Options.FileName = mFileName; QString Extension = QFileInfo(Options.FileName).suffix(); Options.FileName = Options.FileName.left(Options.FileName.length() - Extension.length()); } else Options.FileName = QLatin1String("image"); Options.FileName += lcGetProfileString(LC_PROFILE_IMAGE_EXTENSION); if (!gMainWindow->DoDialog(LC_DIALOG_SAVE_IMAGE, &Options)) return; QString Extension = QFileInfo(Options.FileName).suffix(); if (!Extension.isEmpty()) lcSetProfileString(LC_PROFILE_IMAGE_EXTENSION, Options.FileName.right(Extension.length() + 1)); lcSetProfileInt(LC_PROFILE_IMAGE_WIDTH, Options.Width); lcSetProfileInt(LC_PROFILE_IMAGE_HEIGHT, Options.Height); if (Options.Start != Options.End) Options.FileName = Options.FileName.insert(Options.FileName.length() - Extension.length() - 1, QLatin1String("%1")); SaveStepImages(Options.FileName, Options.Width, Options.Height, Options.Start, Options.End); */ } void Project::SaveStepImages(const QString& BaseName, int Width, int Height, lcStep Start, lcStep End) { /* gMainWindow->mPreviewWidget->MakeCurrent(); lcContext* Context = gMainWindow->mPreviewWidget->mContext; if (!Context->BeginRenderToTexture(Width, Height)) { gMainWindow->DoMessageBox("Error creating images.", LC_MB_ICONERROR | LC_MB_OK); return; } lcStep CurrentStep = mCurrentStep; View View(this); View.SetCamera(gMainWindow->GetActiveView()->mCamera, false); View.mWidth = Width; View.mHeight = Height; View.SetContext(Context); for (lcStep Step = Start; Step <= End; Step++) { SetCurrentStep(Step); View.OnDraw(); QString FileName = BaseName.arg(Step, 2, 10, QLatin1Char('0')); if (!Context->SaveRenderToTextureImage(FileName, Width, Height)) break; } SetCurrentStep(CurrentStep); Context->EndRenderToTexture(); */ } /* void Project::CreateHTMLPieceList(QTextStream& Stream, lcStep Step, bool Images, const QString& ImageExtension) { int* ColorsUsed = new int[gColorList.GetSize()]; memset(ColorsUsed, 0, sizeof(ColorsUsed[0]) * gColorList.GetSize()); int* PiecesUsed = new int[gColorList.GetSize()]; int NumColors = 0; for (int PieceIdx = 0; PieceIdx < mPieces.GetSize(); PieceIdx++) { lcPiece* Piece = mPieces[PieceIdx]; if ((Piece->GetStepShow() == Step) || (Step == 0)) ColorsUsed[Piece->mColorIndex]++; } Stream << QLatin1String("
\r\n"); for (int ColorIdx = 0; ColorIdx < gColorList.GetSize(); ColorIdx++) { if (ColorsUsed[ColorIdx]) { ColorsUsed[ColorIdx] = NumColors; NumColors++; Stream << QString("\n").arg(gColorList[ColorIdx].Name); } } NumColors++; Stream << QLatin1String("\n"); PieceInfo* pInfo; for (int j = 0; j < lcGetPiecesLibrary()->mPieces.GetSize(); j++) { bool Add = false; memset(PiecesUsed, 0, sizeof(PiecesUsed[0]) * gColorList.GetSize()); pInfo = lcGetPiecesLibrary()->mPieces[j]; for (int PieceIdx = 0; PieceIdx < mPieces.GetSize(); PieceIdx++) { lcPiece* Piece = mPieces[PieceIdx]; if ((Piece->mPieceInfo == pInfo) && ((Piece->GetStepShow() == Step) || (Step == 0))) { PiecesUsed[Piece->mColorIndex]++; Add = true; } } if (Add) { if (Images) Stream << QString("\n").arg(pInfo->m_strName, ImageExtension, pInfo->m_strDescription); else Stream << QString("\r\n").arg(pInfo->m_strDescription); int curcol = 1; for (int ColorIdx = 0; ColorIdx < gColorList.GetSize(); ColorIdx++) { if (PiecesUsed[ColorIdx]) { while (curcol != ColorsUsed[ColorIdx] + 1) { Stream << QLatin1String("\r\n"); curcol++; } Stream << QString("\r\n").arg(QString::number(PiecesUsed[ColorIdx])); curcol++; } } while (curcol != NumColors) { Stream << QLatin1String("\r\n"); curcol++; } Stream << QLatin1String("\r\n"); } } Stream << QLatin1String("
Piece
%1
\"%3\"
%1
-
%1
-
\r\n
"); delete[] PiecesUsed; delete[] ColorsUsed; } void Project::ExportHTML() { lcHTMLDialogOptions Options; if (!mFileName.isEmpty()) Options.PathName = QFileInfo(mFileName).canonicalPath(); int ImageOptions = lcGetProfileInt(LC_PROFILE_HTML_IMAGE_OPTIONS); int HTMLOptions = lcGetProfileInt(LC_PROFILE_HTML_OPTIONS); Options.ImageFormat = (LC_IMAGE_FORMAT)(ImageOptions & ~(LC_IMAGE_MASK)); Options.TransparentImages = (ImageOptions & LC_IMAGE_TRANSPARENT) != 0; Options.SinglePage = (HTMLOptions & LC_HTML_SINGLEPAGE) != 0; Options.IndexPage = (HTMLOptions & LC_HTML_INDEX) != 0; Options.StepImagesWidth = lcGetProfileInt(LC_PROFILE_HTML_IMAGE_WIDTH); Options.StepImagesHeight = lcGetProfileInt(LC_PROFILE_HTML_IMAGE_HEIGHT); Options.HighlightNewParts = (HTMLOptions & LC_HTML_HIGHLIGHT) != 0; Options.PartsListStep = (HTMLOptions & LC_HTML_LISTSTEP) != 0; Options.PartsListEnd = (HTMLOptions & LC_HTML_LISTEND) != 0; Options.PartsListImages = (HTMLOptions & LC_HTML_IMAGES) != 0; Options.PartImagesColor = lcGetColorIndex(lcGetProfileInt(LC_PROFILE_HTML_PARTS_COLOR)); Options.PartImagesWidth = lcGetProfileInt(LC_PROFILE_HTML_PARTS_WIDTH); Options.PartImagesHeight = lcGetProfileInt(LC_PROFILE_HTML_PARTS_HEIGHT); if (!gMainWindow->DoDialog(LC_DIALOG_EXPORT_HTML, &Options)) return; HTMLOptions = 0; if (Options.SinglePage) HTMLOptions |= LC_HTML_SINGLEPAGE; if (Options.IndexPage) HTMLOptions |= LC_HTML_INDEX; if (Options.HighlightNewParts) HTMLOptions |= LC_HTML_HIGHLIGHT; if (Options.PartsListStep) HTMLOptions |= LC_HTML_LISTSTEP; if (Options.PartsListEnd) HTMLOptions |= LC_HTML_LISTEND; if (Options.PartsListImages) HTMLOptions |= LC_HTML_IMAGES; ImageOptions = Options.ImageFormat; if (Options.TransparentImages) ImageOptions |= LC_IMAGE_TRANSPARENT; lcSetProfileInt(LC_PROFILE_HTML_IMAGE_OPTIONS, ImageOptions); lcSetProfileInt(LC_PROFILE_HTML_OPTIONS, HTMLOptions); lcSetProfileInt(LC_PROFILE_HTML_IMAGE_WIDTH, Options.StepImagesWidth); lcSetProfileInt(LC_PROFILE_HTML_IMAGE_HEIGHT, Options.StepImagesHeight); lcSetProfileInt(LC_PROFILE_HTML_PARTS_COLOR, lcGetColorCode(Options.PartImagesColor)); lcSetProfileInt(LC_PROFILE_HTML_PARTS_WIDTH, Options.PartImagesWidth); lcSetProfileInt(LC_PROFILE_HTML_PARTS_HEIGHT, Options.PartImagesHeight); QDir Dir(Options.PathName); Dir.mkpath(QLatin1String(".")); QString Title = GetTitle(); QString BaseName = Title.left(Title.length() - QFileInfo(Title).suffix().length() - 1); QString HTMLExtension = QLatin1String(".html"); QString ImageExtension; lcStep LastStep = GetLastStep(); switch (Options.ImageFormat) { case LC_IMAGE_BMP: ImageExtension = QLatin1String(".bmp"); break; case LC_IMAGE_JPG: ImageExtension = QLatin1String(".jpg"); break; default: case LC_IMAGE_PNG: ImageExtension = QLatin1String(".png"); break; } if (Options.SinglePage) { QString FileName = QFileInfo(Dir, BaseName + HTMLExtension).absoluteFilePath(); QFile File(FileName); if (!File.open(QIODevice::WriteOnly)) { QMessageBox::warning(gMainWindow->mHandle, tr("Error"), tr("Error writing to file '%1':\n%2").arg(FileName, File.errorString())); return; } QTextStream Stream(&File); Stream << QString("\r\n\r\nInstructions for %1\r\n\r\n
\r\n
\r\n").arg(Title); for (lcStep Step = 1; Step <= LastStep; Step++) { QString StepString = QString("%1").arg(Step, 2, 10, QLatin1Char('0')); Stream << QString("\"Step

\r\n").arg(BaseName, StepString, ImageExtension, StepString, QString::number(Options.StepImagesWidth), QString::number(Options.StepImagesHeight)); if (Options.PartsListStep) CreateHTMLPieceList(Stream, Step, Options.PartsListImages, ImageExtension); } if (Options.PartsListEnd) CreateHTMLPieceList(Stream, 0, Options.PartsListImages, ImageExtension); Stream << QLatin1String("
\n


Created by LeoCAD
\r\n"); } else { if (Options.IndexPage) { QString FileName = QFileInfo(Dir, BaseName + QLatin1String("-index") + HTMLExtension).absoluteFilePath(); QFile File(FileName); if (!File.open(QIODevice::WriteOnly)) { QMessageBox::warning(gMainWindow->mHandle, tr("Error"), tr("Error writing to file '%1':\n%2").arg(FileName, File.errorString())); return; } QTextStream Stream(&File); Stream << QString("\r\n\r\nInstructions for %1\r\n\r\n
\r\n
\r\n").arg(Title); for (lcStep Step = 1; Step <= LastStep; Step++) Stream << QString("Step %3
\r\n
").arg(BaseName, QString("%1").arg(Step, 2, 10, QLatin1Char('0')), QString::number(Step)); if (Options.PartsListEnd) Stream << QString("Pieces Used
\r\n").arg(BaseName); Stream << QLatin1String("
\r\n


Created by LeoCAD
\r\n"); } for (lcStep Step = 1; Step <= LastStep; Step++) { QString StepString = QString("%1").arg(Step, 2, 10, QLatin1Char('0')); QString FileName = QFileInfo(Dir, BaseName + QLatin1String("-") + StepString + HTMLExtension).absoluteFilePath(); QFile File(FileName); if (!File.open(QIODevice::WriteOnly)) { QMessageBox::warning(gMainWindow->mHandle, tr("Error"), tr("Error writing to file '%1':\n%2").arg(FileName, File.errorString())); return; } QTextStream Stream(&File); Stream << QString("\r\n\r\n%1 - Step %2\r\n\r\n
\r\n
\r\n").arg(Title, QString::number(Step)); Stream << QString("\"Step

\r\n").arg(BaseName, StepString, ImageExtension, StepString, QString::number(Options.StepImagesWidth), QString::number(Options.StepImagesHeight)); if (Options.PartsListStep) CreateHTMLPieceList(Stream, Step, Options.PartsListImages, ImageExtension); Stream << QLatin1String("
\r\n


"); if (Step != 1) Stream << QString("Previous ").arg(BaseName, QString("%1").arg(Step - 1, 2, 10, QLatin1Char('0'))); if (Options.IndexPage) Stream << QString("Index ").arg(BaseName); if (Step != LastStep) Stream << QString("Next").arg(BaseName, QString("%1").arg(Step + 1, 2, 10, QLatin1Char('0'))); else if (Options.PartsListEnd) Stream << QString("Pieces Used").arg(BaseName); Stream << QLatin1String("
\r\n"); } if (Options.PartsListEnd) { QString FileName = QFileInfo(Dir, BaseName + QLatin1String("-pieces") + HTMLExtension).absoluteFilePath(); QFile File(FileName); if (!File.open(QIODevice::WriteOnly)) { QMessageBox::warning(gMainWindow->mHandle, tr("Error"), tr("Error writing to file '%1':\n%2").arg(FileName, File.errorString())); return; } QTextStream Stream(&File); Stream << QString("\r\n\r\nPieces used by %1\r\n\r\n
\r\n
\n").arg(Title); CreateHTMLPieceList(Stream, 0, Options.PartsListImages, ImageExtension); Stream << QLatin1String("
\n


"); Stream << QString("Previous ").arg(BaseName, QString("%1").arg(LastStep, 2, 10, QLatin1Char('0'))); if (Options.IndexPage) Stream << QString("Index ").arg(BaseName); Stream << QLatin1String("
\r\n"); } } QString StepImageBaseName = QFileInfo(Dir, BaseName + QLatin1String("-%1") + ImageExtension).absoluteFilePath(); SaveStepImages(StepImageBaseName, Options.StepImagesWidth, Options.StepImagesHeight, 1, LastStep); if (Options.PartsListImages) { gMainWindow->mPreviewWidget->MakeCurrent(); lcContext* Context = gMainWindow->mPreviewWidget->mContext; int Width = Options.PartImagesWidth; int Height = Options.PartImagesHeight; if (!Context->BeginRenderToTexture(Width, Height)) { gMainWindow->DoMessageBox("Error creating images.", LC_MB_ICONERROR | LC_MB_OK); return; } float aspect = (float)Width/(float)Height; Context->SetViewport(0, 0, Width, Height); for (int PieceIdx = 0; PieceIdx < mPieces.GetSize(); PieceIdx++) { lcPiece* Piece = mPieces[PieceIdx]; bool Skip = false; PieceInfo* Info = Piece->mPieceInfo; for (int CheckIdx = 0; CheckIdx < PieceIdx; CheckIdx++) { if (mPieces[CheckIdx]->mPieceInfo == Info) { Skip = true; break; } } if (Skip) continue; glClearColor(1.0f, 1.0f, 1.0f, 0.0f); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); Info->ZoomExtents(30.0f, aspect); Info->RenderPiece(Options.PartImagesColor); glFinish(); QString FileName = QFileInfo(Dir, Info->m_strName + ImageExtension).absoluteFilePath(); if (!Context->SaveRenderToTextureImage(FileName, Width, Height)) break; } Context->EndRenderToTexture(); } } */