diff -r 09150d027e8c -r d79083b9f74d src/actions.cc
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/src/actions.cc Sat Mar 29 05:38:03 2014 +0200
@@ -0,0 +1,871 @@
+/*
+ * LDForge: LDraw parts authoring CAD
+ * Copyright (C) 2013, 2014 Santeri Piippo
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include "mainWindow.h"
+#include "ldDocument.h"
+#include "editHistory.h"
+#include "configDialog.h"
+#include "addObjectDialog.h"
+#include "miscallenous.h"
+#include "glRenderer.h"
+#include "dialogs.h"
+#include "primitives.h"
+#include "radioGroup.h"
+#include "colors.h"
+#include "glCompiler.h"
+#include "ui_newpart.h"
+
+extern_cfg (Bool, gl_wireframe);
+extern_cfg (Bool, gl_colorbfc);
+extern_cfg (String, ld_defaultname);
+extern_cfg (String, ld_defaultuser);
+extern_cfg (Int, ld_defaultlicense);
+extern_cfg (Bool, gl_drawangles);
+
+// =============================================================================
+//
+DEFINE_ACTION (New, CTRL_SHIFT (N))
+{
+ QDialog* dlg = new QDialog (g_win);
+ Ui::NewPartUI ui;
+ ui.setupUi (dlg);
+
+ QString authortext = ld_defaultname;
+
+ if (!ld_defaultuser.isEmpty())
+ authortext.append (format (" [%1]", ld_defaultuser));
+
+ ui.le_author->setText (authortext);
+
+ switch (ld_defaultlicense)
+ {
+ case 0:
+ ui.rb_license_ca->setChecked (true);
+ break;
+
+ case 1:
+ ui.rb_license_nonca->setChecked (true);
+ break;
+
+ case 2:
+ ui.rb_license_none->setChecked (true);
+ break;
+
+ default:
+ QMessageBox::warning (null, "Warning",
+ format ("Unknown ld_defaultlicense value %1!", ld_defaultlicense));
+ break;
+ }
+
+ if (dlg->exec() == false)
+ return;
+
+ newFile();
+
+ const LDBFC::Statement BFCType =
+ ui.rb_bfc_ccw->isChecked() ? LDBFC::CertifyCCW :
+ ui.rb_bfc_cw->isChecked() ? LDBFC::CertifyCW : LDBFC::NoCertify;
+
+ const QString license =
+ ui.rb_license_ca->isChecked() ? g_CALicense :
+ ui.rb_license_nonca->isChecked() ? g_nonCALicense : "";
+
+ getCurrentDocument()->addObjects (
+ {
+ new LDComment (ui.le_title->text()),
+ new LDComment ("Name: .dat"),
+ new LDComment (format ("Author: %1", ui.le_author->text())),
+ new LDComment (format ("!LDRAW_ORG Unofficial_Part")),
+ (license != "" ? new LDComment (license) : null),
+ new LDEmpty,
+ new LDBFC (BFCType),
+ new LDEmpty,
+ });
+
+ doFullRefresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewFile, CTRL (N))
+{
+ newFile();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Open, CTRL (O))
+{
+ QString name = QFileDialog::getOpenFileName (g_win, "Open File", "", "LDraw files (*.dat *.ldr)");
+
+ if (name.length() == 0)
+ return;
+
+ openMainFile (name);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Save, CTRL (S))
+{
+ save (getCurrentDocument(), false);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SaveAs, CTRL_SHIFT (S))
+{
+ save (getCurrentDocument(), true);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SaveAll, CTRL (L))
+{
+ for (LDDocument* file : g_loadedFiles)
+ {
+ if (file->isImplicit())
+ continue;
+
+ save (file, false);
+ }
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Close, CTRL (W))
+{
+ if (!getCurrentDocument()->isSafeToClose())
+ return;
+
+ delete getCurrentDocument();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (CloseAll, 0)
+{
+ if (!safeToCloseAll())
+ return;
+
+ closeAll();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Settings, 0)
+{
+ (new ConfigDialog)->exec();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SetLDrawPath, 0)
+{
+ (new LDrawPathDialog (true))->exec();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Exit, CTRL (Q))
+{
+ exit (0);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewSubfile, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::ESubfile, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewLine, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::ELine, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewTriangle, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::ETriangle, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewQuad, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::EQuad, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewCLine, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::ECondLine, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewComment, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::EComment, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewBFC, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::EBFC, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (NewVertex, 0)
+{
+ AddObjectDialog::staticDialog (LDObject::EVertex, null);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Edit, 0)
+{
+ if (selection().size() != 1)
+ return;
+
+ LDObject* obj = selection() [0];
+ AddObjectDialog::staticDialog (obj->type(), obj);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Help, KEY (F1))
+{
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (About, 0)
+{
+ AboutDialog().exec();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (AboutQt, 0)
+{
+ QMessageBox::aboutQt (g_win);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SelectAll, CTRL (A))
+{
+ for (LDObject* obj : getCurrentDocument()->objects())
+ obj->select();
+
+ updateSelection();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SelectByColor, CTRL_SHIFT (A))
+{
+ int colnum = getSelectedColor();
+
+ if (colnum == -1)
+ return; // no consensus on color
+
+ getCurrentDocument()->clearSelection();
+
+ for (LDObject* obj : getCurrentDocument()->objects())
+ if (obj->color() == colnum)
+ obj->select();
+
+ updateSelection();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SelectByType, 0)
+{
+ if (selection().isEmpty())
+ return;
+
+ LDObject::Type type = getUniformSelectedType();
+
+ if (type == LDObject::EUnidentified)
+ return;
+
+ // If we're selecting subfile references, the reference filename must also
+ // be uniform.
+ QString refName;
+
+ if (type == LDObject::ESubfile)
+ {
+ refName = static_cast (selection()[0])->fileInfo()->name();
+
+ for (LDObject* obj : selection())
+ if (static_cast (obj)->fileInfo()->name() != refName)
+ return;
+ }
+
+ getCurrentDocument()->clearSelection();
+
+ for (LDObject* obj : getCurrentDocument()->objects())
+ {
+ if (obj->type() != type)
+ continue;
+
+ if (type == LDObject::ESubfile && static_cast (obj)->fileInfo()->name() != refName)
+ continue;
+
+ obj->select();
+ }
+
+ updateSelection();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (GridCoarse, 0)
+{
+ grid = Grid::Coarse;
+ updateGridToolBar();
+}
+
+DEFINE_ACTION (GridMedium, 0)
+{
+ grid = Grid::Medium;
+ updateGridToolBar();
+}
+
+DEFINE_ACTION (GridFine, 0)
+{
+ grid = Grid::Fine;
+ updateGridToolBar();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (ResetView, CTRL (0))
+{
+ R()->resetAngles();
+ R()->update();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (InsertFrom, 0)
+{
+ QString fname = QFileDialog::getOpenFileName();
+ int idx = getInsertionPoint();
+
+ if (!fname.length())
+ return;
+
+ QFile f (fname);
+
+ if (!f.open (QIODevice::ReadOnly))
+ {
+ critical (format ("Couldn't open %1 (%2)", fname, f.errorString()));
+ return;
+ }
+
+ LDObjectList objs = loadFileContents (&f, null);
+
+ getCurrentDocument()->clearSelection();
+
+ for (LDObject* obj : objs)
+ {
+ getCurrentDocument()->insertObj (idx, obj);
+ obj->select();
+ R()->compileObject (obj);
+
+ idx++;
+ }
+
+ refresh();
+ scrollToSelection();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (ExportTo, 0)
+{
+ if (selection().isEmpty())
+ return;
+
+ QString fname = QFileDialog::getSaveFileName();
+
+ if (fname.length() == 0)
+ return;
+
+ QFile file (fname);
+
+ if (!file.open (QIODevice::WriteOnly | QIODevice::Text))
+ {
+ critical (format ("Unable to open %1 for writing (%2)", fname, file.errorString()));
+ return;
+ }
+
+ for (LDObject* obj : selection())
+ {
+ QString contents = obj->asText();
+ QByteArray data = contents.toUtf8();
+ file.write (data, data.size());
+ file.write ("\r\n", 2);
+ }
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (InsertRaw, 0)
+{
+ int idx = getInsertionPoint();
+
+ QDialog* const dlg = new QDialog;
+ QVBoxLayout* const layout = new QVBoxLayout;
+ QTextEdit* const te_edit = new QTextEdit;
+ QDialogButtonBox* const bbx_buttons = new QDialogButtonBox (QDialogButtonBox::Ok | QDialogButtonBox::Cancel);
+
+ layout->addWidget (te_edit);
+ layout->addWidget (bbx_buttons);
+ dlg->setLayout (layout);
+ dlg->setWindowTitle (APPNAME ": Insert Raw");
+ dlg->connect (bbx_buttons, SIGNAL (accepted()), dlg, SLOT (accept()));
+ dlg->connect (bbx_buttons, SIGNAL (rejected()), dlg, SLOT (reject()));
+
+ if (dlg->exec() == false)
+ return;
+
+ getCurrentDocument()->clearSelection();
+
+ for (QString line : QString (te_edit->toPlainText()).split ("\n"))
+ {
+ LDObject* obj = parseLine (line);
+
+ getCurrentDocument()->insertObj (idx, obj);
+ obj->select();
+ idx++;
+ }
+
+ refresh();
+ scrollToSelection();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Screenshot, 0)
+{
+ setlocale (LC_ALL, "C");
+
+ int w, h;
+ uchar* imgdata = R()->getScreencap (w, h);
+ QImage img = imageFromScreencap (imgdata, w, h);
+
+ QString root = basename (getCurrentDocument()->name());
+
+ if (root.right (4) == ".dat")
+ root.chop (4);
+
+ QString defaultname = (root.length() > 0) ? format ("%1.png", root) : "";
+ QString fname = QFileDialog::getSaveFileName (g_win, "Save Screencap", defaultname,
+ "PNG images (*.png);;JPG images (*.jpg);;BMP images (*.bmp);;All Files (*.*)");
+
+ if (fname.length() > 0 && !img.save (fname))
+ critical (format ("Couldn't open %1 for writing to save screencap: %2", fname, strerror (errno)));
+
+ delete[] imgdata;
+}
+
+// =============================================================================
+//
+extern_cfg (Bool, gl_axes);
+DEFINE_ACTION (Axes, 0)
+{
+ gl_axes = !gl_axes;
+ updateActions();
+ R()->update();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (VisibilityToggle, 0)
+{
+ for (LDObject* obj : selection())
+ obj->setHidden (!obj->isHidden());
+
+ refresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (VisibilityHide, 0)
+{
+ for (LDObject* obj : selection())
+ obj->setHidden (true);
+
+ refresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (VisibilityReveal, 0)
+{
+ for (LDObject* obj : selection())
+ obj->setHidden (false);
+ refresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (Wireframe, 0)
+{
+ gl_wireframe = !gl_wireframe;
+ R()->refresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SetOverlay, 0)
+{
+ OverlayDialog dlg;
+
+ if (!dlg.exec())
+ return;
+
+ R()->setupOverlay ((GL::EFixedCamera) dlg.camera(), dlg.fpath(), dlg.ofsx(),
+ dlg.ofsy(), dlg.lwidth(), dlg.lheight());
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (ClearOverlay, 0)
+{
+ R()->clearOverlay();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (ModeSelect, CTRL (1))
+{
+ R()->setEditMode (ESelectMode);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (ModeDraw, CTRL (2))
+{
+ R()->setEditMode (EDrawMode);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (ModeCircle, CTRL (3))
+{
+ R()->setEditMode (ECircleMode);
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (DrawAngles, 0)
+{
+ gl_drawangles = !gl_drawangles;
+ R()->refresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SetDrawDepth, 0)
+{
+ if (R()->camera() == GL::EFreeCamera)
+ return;
+
+ bool ok;
+ double depth = QInputDialog::getDouble (g_win, "Set Draw Depth",
+ format ("Depth value for %1 Camera:", R()->getCameraName()),
+ R()->getDepthValue(), -10000.0f, 10000.0f, 3, &ok);
+
+ if (ok)
+ R()->setDepthValue (depth);
+}
+
+#if 0
+// This is a test to draw a dummy axle. Meant to be used as a primitive gallery,
+// but I can't figure how to generate these pictures properly. Multi-threading
+// these is an immense pain.
+DEFINE_ACTION (testpic, "Test picture", "", "", (0))
+{
+ LDDocument* file = getFile ("axle.dat");
+ setlocale (LC_ALL, "C");
+
+ if (!file)
+ {
+ critical ("couldn't load axle.dat");
+ return;
+ }
+
+ int w, h;
+
+ GLRenderer* rend = new GLRenderer;
+ rend->resize (64, 64);
+ rend->setAttribute (Qt::WA_DontShowOnScreen);
+ rend->show();
+ rend->setFile (file);
+ rend->setDrawOnly (true);
+ rend->compileAllObjects();
+ rend->initGLData();
+ rend->drawGLScene();
+
+ uchar* imgdata = rend->screencap (w, h);
+ QImage img = imageFromScreencap (imgdata, w, h);
+
+ if (img.isNull())
+ {
+ critical ("Failed to create the image!\n");
+ }
+ else
+ {
+ QLabel* label = new QLabel;
+ QDialog* dlg = new QDialog;
+ label->setPixmap (QPixmap::fromImage (img));
+ QVBoxLayout* layout = new QVBoxLayout (dlg);
+ layout->addWidget (label);
+ dlg->exec();
+ }
+
+ delete[] imgdata;
+ rend->deleteLater();
+}
+#endif
+
+// =============================================================================
+//
+DEFINE_ACTION (ScanPrimitives, 0)
+{
+ PrimitiveScanner::start();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (BFCView, SHIFT (B))
+{
+ gl_colorbfc = !gl_colorbfc;
+ updateActions();
+ R()->refresh();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (JumpTo, CTRL (G))
+{
+ bool ok;
+ int defval = 0;
+ LDObject* obj;
+
+ if (selection().size() == 1)
+ defval = selection()[0]->lineNumber();
+
+ int idx = QInputDialog::getInt (null, "Go to line", "Go to line:", defval,
+ 1, getCurrentDocument()->getObjectCount(), 1, &ok);
+
+ if (!ok || (obj = getCurrentDocument()->getObject (idx - 1)) == null)
+ return;
+
+ getCurrentDocument()->clearSelection();
+ obj->select();
+ updateSelection();
+}
+
+// =============================================================================
+//
+DEFINE_ACTION (SubfileSelection, 0)
+{
+ if (selection().size() == 0)
+ return;
+
+ QString parentpath = getCurrentDocument()->fullPath();
+
+ // BFC type of the new subfile - it shall inherit the BFC type of the parent document
+ LDBFC::Statement bfctype = LDBFC::NoCertify;
+
+ // Dirname of the new subfile
+ QString subdirname = dirname (parentpath);
+
+ // Title of the new subfile
+ QString subtitle;
+
+ // Comment containing the title of the parent document
+ LDComment* titleobj = dynamic_cast (getCurrentDocument()->getObject (0));
+
+ // License text for the subfile
+ QString license = getLicenseText (ld_defaultlicense);
+
+ // LDraw code body of the new subfile (i.e. code of the selection)
+ QStringList code;
+
+ // Full path of the subfile to be
+ QString fullsubname;
+
+ // Where to insert the subfile reference?
+ int refidx = selection()[0]->lineNumber();
+
+ // Determine title of subfile
+ if (titleobj != null)
+ subtitle = "~" + titleobj->text();
+ else
+ subtitle = "~subfile";
+
+ // Remove duplicate tildes
+ while (subtitle[0] == '~' && subtitle[1] == '~')
+ subtitle.remove (0, 1);
+
+ // If this the parent document isn't already in s/, we need to stuff it into
+ // a subdirectory named s/. Ensure it exists!
+ QString topdirname = basename (dirname (getCurrentDocument()->fullPath()));
+
+ if (topdirname != "s")
+ {
+ QString desiredPath = subdirname + "/s";
+ QString title = tr ("Create subfile directory?");
+ QString text = format (tr ("The directory %1 is suggested for "
+ "subfiles. This directory does not exist, create it?"), desiredPath);
+
+ if (QDir (desiredPath).exists() || confirm (title, text))
+ {
+ subdirname = desiredPath;
+ QDir().mkpath (subdirname);
+ }
+ }
+
+ // Determine the body of the name of the subfile
+ if (!parentpath.isEmpty())
+ {
+ if (parentpath.endsWith (".dat"))
+ parentpath.chop (4);
+
+ // Remove the s?? suffix if it's there, otherwise we'll get filenames
+ // like s01s01.dat when subfiling subfiles.
+ QRegExp subfilesuffix ("s[0-9][0-9]$");
+ if (subfilesuffix.indexIn (parentpath) != -1)
+ parentpath.chop (subfilesuffix.matchedLength());
+
+ int subidx = 1;
+ QString digits;
+ QFile f;
+ QString testfname;
+
+ do
+ {
+ digits.setNum (subidx++);
+
+ // pad it with a zero
+ if (digits.length() == 1)
+ digits.prepend ("0");
+
+ fullsubname = subdirname + "/" + basename (parentpath) + "s" + digits + ".dat";
+ } while (findDocument ("s\\" + basename (fullsubname)) != null || QFile (fullsubname).exists());
+ }
+
+ // Determine the BFC winding type used in the main document - it is to
+ // be carried over to the subfile.
+ for (LDObject* obj : getCurrentDocument()->objects())
+ {
+ LDBFC* bfc = dynamic_cast (obj);
+
+ if (!bfc)
+ continue;
+
+ LDBFC::Statement a = bfc->statement();
+
+ if (a == LDBFC::CertifyCCW || a == LDBFC::CertifyCW || a == LDBFC::NoCertify)
+ {
+ bfctype = a;
+ break;
+ }
+ }
+
+ // Get the body of the document in LDraw code
+ for (LDObject* obj : selection())
+ code << obj->asText();
+
+ // Create the new subfile document
+ LDDocument* doc = new LDDocument;
+ doc->setImplicit (false);
+ doc->setFullPath (fullsubname);
+ doc->setName (LDDocument::shortenName (fullsubname));
+ doc->addObjects (
+ {
+ new LDComment (subtitle),
+ new LDComment ("Name: "),
+ new LDComment (format ("Author: %1 [%2]", ld_defaultname, ld_defaultuser)),
+ new LDComment (format ("!LDRAW_ORG Unofficial_Subpart")),
+ (license != "" ? new LDComment (license) : null),
+ new LDEmpty,
+ new LDBFC (bfctype),
+ new LDEmpty,
+ });
+
+ // Add the actual subfile code to the new document
+ for (QString line : code)
+ {
+ LDObject* obj = parseLine (line);
+ doc->addObject (obj);
+ }
+
+ // Try save it
+ if (save (doc, true))
+ {
+ // Remove the selection now
+ for (LDObject* obj : selection())
+ obj->destroy();
+
+ // Compile all objects in the new subfile
+ R()->compiler()->compileDocument (doc);
+ g_loadedFiles << doc;
+
+ // Add a reference to the new subfile to where the selection was
+ LDSubfile* ref = new LDSubfile();
+ ref->setColor (maincolor);
+ ref->setFileInfo (doc);
+ ref->setPosition (g_origin);
+ ref->setTransform (g_identity);
+ getCurrentDocument()->insertObj (refidx, ref);
+ R()->compileObject (ref);
+
+ // Refresh stuff
+ updateDocumentList();
+ doFullRefresh();
+ }
+ else
+ {
+ // Failed to save.
+ delete doc;
+ }
+}