1 /* |
|
2 * LDForge: LDraw parts authoring CAD |
|
3 * Copyright (C) 2013 - 2015 Teemu Piippo |
|
4 * |
|
5 * This program is free software: you can redistribute it and/or modify |
|
6 * it under the terms of the GNU General Public License as published by |
|
7 * the Free Software Foundation, either version 3 of the License, or |
|
8 * (at your option) any later version. |
|
9 * |
|
10 * This program is distributed in the hope that it will be useful, |
|
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
13 * GNU General Public License for more details. |
|
14 * |
|
15 * You should have received a copy of the GNU General Public License |
|
16 * along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
17 */ |
|
18 |
|
19 #include <QProcess> |
|
20 #include <QTemporaryFile> |
|
21 #include <QDialog> |
|
22 #include <QDialogButtonBox> |
|
23 #include <QSpinBox> |
|
24 #include <QCheckBox> |
|
25 #include <QComboBox> |
|
26 #include <QGridLayout> |
|
27 #include <QFileInfo> |
|
28 #include "main.h" |
|
29 #include "configuration.h" |
|
30 #include "miscallenous.h" |
|
31 #include "mainwindow.h" |
|
32 #include "ldDocument.h" |
|
33 #include "radioGroup.h" |
|
34 #include "editHistory.h" |
|
35 #include "ui_ytruder.h" |
|
36 #include "ui_intersector.h" |
|
37 #include "ui_rectifier.h" |
|
38 #include "ui_coverer.h" |
|
39 #include "ui_isecalc.h" |
|
40 #include "ui_edger2.h" |
|
41 #include "dialogs.h" |
|
42 |
|
43 enum extprog |
|
44 { |
|
45 Isecalc, |
|
46 Intersector, |
|
47 Coverer, |
|
48 Ytruder, |
|
49 Rectifier, |
|
50 Edger2, |
|
51 }; |
|
52 |
|
53 // ============================================================================= |
|
54 // |
|
55 CFGENTRY (String, IsecalcPath, "") |
|
56 CFGENTRY (String, IntersectorPath, "") |
|
57 CFGENTRY (String, CovererPath, "") |
|
58 CFGENTRY (String, YtruderPath, "") |
|
59 CFGENTRY (String, RectifierPath, "") |
|
60 CFGENTRY (String, Edger2Path, "") |
|
61 |
|
62 QString* const g_extProgPaths[] = |
|
63 { |
|
64 &cfg::IsecalcPath, |
|
65 &cfg::IntersectorPath, |
|
66 &cfg::CovererPath, |
|
67 &cfg::YtruderPath, |
|
68 &cfg::RectifierPath, |
|
69 &cfg::Edger2Path, |
|
70 }; |
|
71 |
|
72 CFGENTRY (Bool, IsecalcUsesWine, false) |
|
73 CFGENTRY (Bool, IntersectorUsesWine, false) |
|
74 CFGENTRY (Bool, CovererUsesWine, false) |
|
75 CFGENTRY (Bool, YtruderUsesWine, false) |
|
76 CFGENTRY (Bool, RectifierUsesWine, false) |
|
77 CFGENTRY (Bool, Edger2UsesWine, false) |
|
78 |
|
79 bool* const g_extProgWine[] = |
|
80 { |
|
81 &cfg::IsecalcUsesWine, |
|
82 &cfg::IntersectorUsesWine, |
|
83 &cfg::CovererUsesWine, |
|
84 &cfg::YtruderUsesWine, |
|
85 &cfg::RectifierUsesWine, |
|
86 &cfg::Edger2UsesWine, |
|
87 }; |
|
88 |
|
89 const char* g_extProgNames[] = |
|
90 { |
|
91 "Isecalc", |
|
92 "Intersector", |
|
93 "Coverer", |
|
94 "Ytruder", |
|
95 "Rectifier", |
|
96 "Edger2" |
|
97 }; |
|
98 |
|
99 // ============================================================================= |
|
100 // |
|
101 static bool MakeTempFile (QTemporaryFile& tmp, QString& fname) |
|
102 { |
|
103 if (not tmp.open()) |
|
104 return false; |
|
105 |
|
106 fname = tmp.fileName(); |
|
107 tmp.close(); |
|
108 return true; |
|
109 } |
|
110 |
|
111 // ============================================================================= |
|
112 // |
|
113 static bool CheckExtProgramPath (const extprog prog) |
|
114 { |
|
115 QString& path = *g_extProgPaths[prog]; |
|
116 |
|
117 if (not path.isEmpty()) |
|
118 return true; |
|
119 |
|
120 ExtProgPathPrompt* dlg = new ExtProgPathPrompt (g_extProgNames[prog]); |
|
121 |
|
122 if (dlg->exec() and not dlg->getPath().isEmpty()) |
|
123 { |
|
124 path = dlg->getPath(); |
|
125 return true; |
|
126 } |
|
127 |
|
128 return false; |
|
129 } |
|
130 |
|
131 // ============================================================================= |
|
132 // |
|
133 static QString ProcessExtProgError (extprog prog, QProcess& proc) |
|
134 { |
|
135 switch (proc.error()) |
|
136 { |
|
137 case QProcess::FailedToStart: |
|
138 { |
|
139 QString wineblurb; |
|
140 |
|
141 #ifndef _WIN32 |
|
142 if (*g_extProgWine[prog]) |
|
143 wineblurb = "make sure Wine is installed and "; |
|
144 #else |
|
145 (void) prog; |
|
146 #endif |
|
147 |
|
148 return format ("Program failed to start, %1check your permissions", wineblurb); |
|
149 } break; |
|
150 |
|
151 case QProcess::Crashed: |
|
152 return "Crashed."; |
|
153 |
|
154 case QProcess::WriteError: |
|
155 case QProcess::ReadError: |
|
156 return "I/O error."; |
|
157 |
|
158 case QProcess::UnknownError: |
|
159 return "Unknown error"; |
|
160 |
|
161 case QProcess::Timedout: |
|
162 return format ("Timed out (30 seconds)"); |
|
163 } |
|
164 |
|
165 return ""; |
|
166 } |
|
167 |
|
168 // ============================================================================= |
|
169 // |
|
170 static void WriteObjects (const LDObjectList& objects, QFile& f) |
|
171 { |
|
172 for (LDObject* obj : objects) |
|
173 { |
|
174 if (obj->type() == OBJ_Subfile) |
|
175 { |
|
176 LDSubfile* ref = static_cast<LDSubfile*> (obj); |
|
177 LDObjectList objs = ref->inlineContents (true, false); |
|
178 |
|
179 WriteObjects (objs, f); |
|
180 |
|
181 for (LDObject* obj : objs) |
|
182 obj->destroy(); |
|
183 } |
|
184 else |
|
185 f.write ((obj->asText() + "\r\n").toUtf8()); |
|
186 } |
|
187 } |
|
188 |
|
189 // ============================================================================= |
|
190 // |
|
191 static void WriteObjects (const LDObjectList& objects, QString fname) |
|
192 { |
|
193 // Write the input file |
|
194 QFile f (fname); |
|
195 |
|
196 if (not f.open (QIODevice::WriteOnly | QIODevice::Text)) |
|
197 { |
|
198 Critical (format ("Couldn't open temporary file %1 for writing: %2\n", fname, f.errorString())); |
|
199 return; |
|
200 } |
|
201 |
|
202 WriteObjects (objects, f); |
|
203 f.close(); |
|
204 |
|
205 #ifdef DEBUG |
|
206 QFile::copy (fname, "debug_lastInput"); |
|
207 #endif |
|
208 } |
|
209 |
|
210 // ============================================================================= |
|
211 // |
|
212 void WriteSelection (QString fname) |
|
213 { |
|
214 WriteObjects (Selection(), fname); |
|
215 } |
|
216 |
|
217 // ============================================================================= |
|
218 // |
|
219 void WriteColorGroup (LDColor color, QString fname) |
|
220 { |
|
221 LDObjectList objects; |
|
222 |
|
223 for (LDObject* obj : CurrentDocument()->objects()) |
|
224 { |
|
225 if (not obj->isColored() or obj->color() != color) |
|
226 continue; |
|
227 |
|
228 objects << obj; |
|
229 } |
|
230 |
|
231 WriteObjects (objects, fname); |
|
232 } |
|
233 |
|
234 // ============================================================================= |
|
235 // |
|
236 bool RunExtProgram (extprog prog, QString path, QString argvstr) |
|
237 { |
|
238 QTemporaryFile input; |
|
239 QStringList argv = argvstr.split (" ", QString::SkipEmptyParts); |
|
240 |
|
241 #ifndef _WIN32 |
|
242 if (*g_extProgWine[prog]) |
|
243 { |
|
244 argv.insert (0, path); |
|
245 path = "wine"; |
|
246 } |
|
247 #endif // _WIN32 |
|
248 |
|
249 print ("Running command: %1 %2\n", path, argv.join (" ")); |
|
250 |
|
251 if (not input.open()) |
|
252 return false; |
|
253 |
|
254 QProcess proc; |
|
255 |
|
256 // Begin! |
|
257 proc.setStandardInputFile (input.fileName()); |
|
258 proc.start (path, argv); |
|
259 |
|
260 if (not proc.waitForStarted()) |
|
261 { |
|
262 Critical (format ("Couldn't start %1: %2\n", g_extProgNames[prog], ProcessExtProgError (prog, proc))); |
|
263 return false; |
|
264 } |
|
265 |
|
266 // Write an enter, the utility tools all expect one |
|
267 input.write ("\n"); |
|
268 |
|
269 // Wait while it runs |
|
270 proc.waitForFinished(); |
|
271 |
|
272 QString err = ""; |
|
273 |
|
274 if (proc.exitStatus() != QProcess::NormalExit) |
|
275 err = ProcessExtProgError (prog, proc); |
|
276 |
|
277 // Check the return code |
|
278 if (proc.exitCode() != 0) |
|
279 err = format ("Program exited abnormally (return code %1).", proc.exitCode()); |
|
280 |
|
281 if (not err.isEmpty()) |
|
282 { |
|
283 Critical (format ("%1 failed: %2\n", g_extProgNames[prog], err)); |
|
284 QString filename ("externalProgramOutput.txt"); |
|
285 QFile file (filename); |
|
286 |
|
287 if (file.open (QIODevice::WriteOnly | QIODevice::Text)) |
|
288 { |
|
289 file.write (proc.readAllStandardOutput()); |
|
290 file.write (proc.readAllStandardError()); |
|
291 print ("Wrote output and error logs to %1", QFileInfo (file).absoluteFilePath()); |
|
292 } |
|
293 else |
|
294 { |
|
295 print ("Couldn't open %1 for writing: %2", |
|
296 QFileInfo (filename).absoluteFilePath(), file.errorString()); |
|
297 } |
|
298 |
|
299 return false; |
|
300 } |
|
301 |
|
302 return true; |
|
303 } |
|
304 |
|
305 // ============================================================================= |
|
306 // |
|
307 static void InsertOutput (QString fname, bool replace, QList<LDColor> colorsToReplace) |
|
308 { |
|
309 #ifdef DEBUG |
|
310 QFile::copy (fname, "./debug_lastOutput"); |
|
311 #endif // RELEASE |
|
312 |
|
313 // Read the output file |
|
314 QFile f (fname); |
|
315 |
|
316 if (not f.open (QIODevice::ReadOnly)) |
|
317 { |
|
318 Critical (format ("Couldn't open temporary file %1 for reading.\n", fname)); |
|
319 return; |
|
320 } |
|
321 |
|
322 LDObjectList objs = LoadFileContents (&f, null); |
|
323 |
|
324 // If we replace the objects, delete the selection now. |
|
325 if (replace) |
|
326 g_win->deleteSelection(); |
|
327 |
|
328 for (LDColor color : colorsToReplace) |
|
329 g_win->deleteByColor (color); |
|
330 |
|
331 // Insert the new objects |
|
332 CurrentDocument()->clearSelection(); |
|
333 |
|
334 for (LDObject* obj : objs) |
|
335 { |
|
336 if (not obj->isScemantic()) |
|
337 { |
|
338 obj->destroy(); |
|
339 continue; |
|
340 } |
|
341 |
|
342 CurrentDocument()->addObject (obj); |
|
343 obj->select(); |
|
344 } |
|
345 |
|
346 g_win->doFullRefresh(); |
|
347 } |
|
348 |
|
349 // ============================================================================= |
|
350 // Interface for Ytruder |
|
351 // ============================================================================= |
|
352 void MainWindow::slot_actionYtruder() |
|
353 { |
|
354 setlocale (LC_ALL, "C"); |
|
355 |
|
356 if (not CheckExtProgramPath (Ytruder)) |
|
357 return; |
|
358 |
|
359 QDialog* dlg = new QDialog; |
|
360 Ui::YtruderUI ui; |
|
361 ui.setupUi (dlg); |
|
362 |
|
363 if (not dlg->exec()) |
|
364 return; |
|
365 |
|
366 // Read the user's choices |
|
367 const enum { Distance, Symmetry, Projection, Radial } mode = |
|
368 ui.mode_distance->isChecked() ? Distance : |
|
369 ui.mode_symmetry->isChecked() ? Symmetry : |
|
370 ui.mode_projection->isChecked() ? Projection : Radial; |
|
371 |
|
372 const Axis axis = |
|
373 ui.axis_x->isChecked() ? X : |
|
374 ui.axis_y->isChecked() ? Y : Z; |
|
375 |
|
376 const double depth = ui.planeDepth->value(), |
|
377 condAngle = ui.condAngle->value(); |
|
378 |
|
379 QTemporaryFile indat, outdat; |
|
380 QString inDATName, outDATName; |
|
381 |
|
382 // Make temp files for the input and output files |
|
383 if (not MakeTempFile (indat, inDATName) or not MakeTempFile (outdat, outDATName)) |
|
384 return; |
|
385 |
|
386 // Compose the command-line arguments |
|
387 QString argv = Join ( |
|
388 { |
|
389 (axis == X) ? "-x" : (axis == Y) ? "-y" : "-z", |
|
390 (mode == Distance) ? "-d" : (mode == Symmetry) ? "-s" : (mode == Projection) ? "-p" : "-r", |
|
391 depth, |
|
392 "-a", |
|
393 condAngle, |
|
394 inDATName, |
|
395 outDATName |
|
396 }); |
|
397 |
|
398 WriteSelection (inDATName); |
|
399 |
|
400 if (not RunExtProgram (Ytruder, cfg::YtruderPath, argv)) |
|
401 return; |
|
402 |
|
403 InsertOutput (outDATName, false, {}); |
|
404 } |
|
405 |
|
406 // ============================================================================= |
|
407 // Rectifier interface |
|
408 // ============================================================================= |
|
409 void MainWindow::slot_actionRectifier() |
|
410 { |
|
411 setlocale (LC_ALL, "C"); |
|
412 |
|
413 if (not CheckExtProgramPath (Rectifier)) |
|
414 return; |
|
415 |
|
416 QDialog* dlg = new QDialog; |
|
417 Ui::RectifierUI ui; |
|
418 ui.setupUi (dlg); |
|
419 |
|
420 if (not dlg->exec()) |
|
421 return; |
|
422 |
|
423 QTemporaryFile indat, outdat; |
|
424 QString inDATName, outDATName; |
|
425 |
|
426 // Make temp files for the input and output files |
|
427 if (not MakeTempFile (indat, inDATName) or not MakeTempFile (outdat, outDATName)) |
|
428 return; |
|
429 |
|
430 // Compose arguments |
|
431 QString argv = Join ( |
|
432 { |
|
433 (not ui.cb_condense->isChecked()) ? "-q" : "", |
|
434 (not ui.cb_subst->isChecked()) ? "-r" : "", |
|
435 (ui.cb_condlineCheck->isChecked()) ? "-a" : "", |
|
436 (ui.cb_colorize->isChecked()) ? "-c" : "", |
|
437 "-t", |
|
438 ui.dsb_coplthres->value(), |
|
439 inDATName, |
|
440 outDATName |
|
441 }); |
|
442 |
|
443 WriteSelection (inDATName); |
|
444 |
|
445 if (not RunExtProgram (Rectifier, cfg::RectifierPath, argv)) |
|
446 return; |
|
447 |
|
448 InsertOutput (outDATName, true, {}); |
|
449 } |
|
450 |
|
451 // ============================================================================= |
|
452 // Intersector interface |
|
453 // ============================================================================= |
|
454 void MainWindow::slot_actionIntersector() |
|
455 { |
|
456 setlocale (LC_ALL, "C"); |
|
457 |
|
458 if (not CheckExtProgramPath (Intersector)) |
|
459 return; |
|
460 |
|
461 QDialog* dlg = new QDialog; |
|
462 Ui::IntersectorUI ui; |
|
463 ui.setupUi (dlg); |
|
464 |
|
465 MakeColorComboBox (ui.cmb_incol); |
|
466 MakeColorComboBox (ui.cmb_cutcol); |
|
467 ui.cb_repeat->setWhatsThis ("If this is set, " APPNAME " runs Intersector a second time with inverse files to cut the " |
|
468 " cutter group with the input group. Both groups are cut by the intersection."); |
|
469 ui.cb_edges->setWhatsThis ("Makes " APPNAME " try run Isecalc to create edgelines for the intersection."); |
|
470 |
|
471 LDColor inCol, cutCol; |
|
472 const bool repeatInverse = ui.cb_repeat->isChecked(); |
|
473 |
|
474 forever |
|
475 { |
|
476 if (not dlg->exec()) |
|
477 return; |
|
478 |
|
479 inCol = ui.cmb_incol->itemData (ui.cmb_incol->currentIndex()).toInt(); |
|
480 cutCol = ui.cmb_cutcol->itemData (ui.cmb_cutcol->currentIndex()).toInt(); |
|
481 |
|
482 if (inCol == cutCol) |
|
483 { |
|
484 Critical ("Cannot use the same color group for both input and cutter!"); |
|
485 continue; |
|
486 } |
|
487 |
|
488 break; |
|
489 } |
|
490 |
|
491 // Five temporary files! |
|
492 // indat = input group file |
|
493 // cutdat = cutter group file |
|
494 // outdat = primary output |
|
495 // outdat2 = inverse output |
|
496 // edgesdat = edges output (isecalc) |
|
497 QTemporaryFile indat, cutdat, outdat, outdat2, edgesdat; |
|
498 QString inDATName, cutDATName, outDATName, outDAT2Name, edgesDATName; |
|
499 |
|
500 if (not MakeTempFile (indat, inDATName) or |
|
501 not MakeTempFile (cutdat, cutDATName) or |
|
502 not MakeTempFile (outdat, outDATName) or |
|
503 not MakeTempFile (outdat2, outDAT2Name) or |
|
504 not MakeTempFile (edgesdat, edgesDATName)) |
|
505 { |
|
506 return; |
|
507 } |
|
508 |
|
509 QString parms = Join ( |
|
510 { |
|
511 (ui.cb_colorize->isChecked()) ? "-c" : "", |
|
512 (ui.cb_nocondense->isChecked()) ? "-t" : "", |
|
513 "-s", |
|
514 ui.dsb_prescale->value() |
|
515 }); |
|
516 |
|
517 QString argv_normal = Join ( |
|
518 { |
|
519 parms, |
|
520 inDATName, |
|
521 cutDATName, |
|
522 outDATName |
|
523 }); |
|
524 |
|
525 QString argv_inverse = Join ( |
|
526 { |
|
527 parms, |
|
528 cutDATName, |
|
529 inDATName, |
|
530 outDAT2Name |
|
531 }); |
|
532 |
|
533 WriteColorGroup (inCol, inDATName); |
|
534 WriteColorGroup (cutCol, cutDATName); |
|
535 |
|
536 if (not RunExtProgram (Intersector, cfg::IntersectorPath, argv_normal)) |
|
537 return; |
|
538 |
|
539 InsertOutput (outDATName, false, {inCol}); |
|
540 |
|
541 if (repeatInverse and RunExtProgram (Intersector, cfg::IntersectorPath, argv_inverse)) |
|
542 InsertOutput (outDAT2Name, false, {cutCol}); |
|
543 |
|
544 if (ui.cb_edges->isChecked() and CheckExtProgramPath (Isecalc) and |
|
545 RunExtProgram (Isecalc, cfg::IsecalcPath, Join ({inDATName, cutDATName, edgesDATName}))) |
|
546 { |
|
547 InsertOutput (edgesDATName, false, {}); |
|
548 } |
|
549 } |
|
550 |
|
551 // ============================================================================= |
|
552 // |
|
553 void MainWindow::slot_actionCoverer() |
|
554 { |
|
555 setlocale (LC_ALL, "C"); |
|
556 |
|
557 if (not CheckExtProgramPath (Coverer)) |
|
558 return; |
|
559 |
|
560 QDialog* dlg = new QDialog; |
|
561 Ui::CovererUI ui; |
|
562 ui.setupUi (dlg); |
|
563 MakeColorComboBox (ui.cmb_col1); |
|
564 MakeColorComboBox (ui.cmb_col2); |
|
565 |
|
566 LDColor in1Col, in2Col; |
|
567 |
|
568 forever |
|
569 { |
|
570 if (not dlg->exec()) |
|
571 return; |
|
572 |
|
573 in1Col = ui.cmb_col1->itemData (ui.cmb_col1->currentIndex()).toInt(); |
|
574 in2Col = ui.cmb_col2->itemData (ui.cmb_col2->currentIndex()).toInt(); |
|
575 |
|
576 if (in1Col == in2Col) |
|
577 { |
|
578 Critical ("Cannot use the same color group for both inputs!"); |
|
579 continue; |
|
580 } |
|
581 |
|
582 break; |
|
583 } |
|
584 |
|
585 QTemporaryFile in1dat, in2dat, outdat; |
|
586 QString in1DATName, in2DATName, outDATName; |
|
587 |
|
588 if (not MakeTempFile (in1dat, in1DATName) or |
|
589 not MakeTempFile (in2dat, in2DATName) or |
|
590 not MakeTempFile (outdat, outDATName)) |
|
591 { |
|
592 return; |
|
593 } |
|
594 |
|
595 QString argv = Join ( |
|
596 { |
|
597 (ui.cb_oldsweep->isChecked() ? "-s" : ""), |
|
598 (ui.cb_reverse->isChecked() ? "-r" : ""), |
|
599 (ui.dsb_segsplit->value() != 0 ? format ("-l %1", ui.dsb_segsplit->value()) : ""), |
|
600 (ui.sb_bias->value() != 0 ? format ("-s %1", ui.sb_bias->value()) : ""), |
|
601 in1DATName, |
|
602 in2DATName, |
|
603 outDATName |
|
604 }); |
|
605 |
|
606 WriteColorGroup (in1Col, in1DATName); |
|
607 WriteColorGroup (in2Col, in2DATName); |
|
608 |
|
609 if (not RunExtProgram (Coverer, cfg::CovererPath, argv)) |
|
610 return; |
|
611 |
|
612 InsertOutput (outDATName, false, {}); |
|
613 } |
|
614 |
|
615 // ============================================================================= |
|
616 // |
|
617 void MainWindow::slot_actionIsecalc() |
|
618 { |
|
619 setlocale (LC_ALL, "C"); |
|
620 |
|
621 if (not CheckExtProgramPath (Isecalc)) |
|
622 return; |
|
623 |
|
624 Ui::IsecalcUI ui; |
|
625 QDialog* dlg = new QDialog; |
|
626 ui.setupUi (dlg); |
|
627 |
|
628 MakeColorComboBox (ui.cmb_col1); |
|
629 MakeColorComboBox (ui.cmb_col2); |
|
630 |
|
631 LDColor in1Col, in2Col; |
|
632 |
|
633 // Run the dialog and validate input |
|
634 forever |
|
635 { |
|
636 if (not dlg->exec()) |
|
637 return; |
|
638 |
|
639 in1Col = ui.cmb_col1->itemData (ui.cmb_col1->currentIndex()).toInt(); |
|
640 in2Col = ui.cmb_col2->itemData (ui.cmb_col2->currentIndex()).toInt(); |
|
641 |
|
642 if (in1Col == in2Col) |
|
643 { |
|
644 Critical ("Cannot use the same color group for both input and cutter!"); |
|
645 continue; |
|
646 } |
|
647 |
|
648 break; |
|
649 } |
|
650 |
|
651 QTemporaryFile in1dat, in2dat, outdat; |
|
652 QString in1DATName, in2DATName, outDATName; |
|
653 |
|
654 if (not MakeTempFile (in1dat, in1DATName) or |
|
655 not MakeTempFile (in2dat, in2DATName) or |
|
656 not MakeTempFile (outdat, outDATName)) |
|
657 { |
|
658 return; |
|
659 } |
|
660 |
|
661 QString argv = Join ( |
|
662 { |
|
663 in1DATName, |
|
664 in2DATName, |
|
665 outDATName |
|
666 }); |
|
667 |
|
668 WriteColorGroup (in1Col, in1DATName); |
|
669 WriteColorGroup (in2Col, in2DATName); |
|
670 RunExtProgram (Isecalc, cfg::IsecalcPath, argv); |
|
671 InsertOutput (outDATName, false, {}); |
|
672 } |
|
673 |
|
674 // ============================================================================= |
|
675 // |
|
676 void MainWindow::slot_actionEdger2() |
|
677 { |
|
678 setlocale (LC_ALL, "C"); |
|
679 |
|
680 if (not CheckExtProgramPath (Edger2)) |
|
681 return; |
|
682 |
|
683 QDialog* dlg = new QDialog; |
|
684 Ui::Edger2Dialog ui; |
|
685 ui.setupUi (dlg); |
|
686 |
|
687 if (not dlg->exec()) |
|
688 return; |
|
689 |
|
690 QTemporaryFile in, out; |
|
691 QString inName, outName; |
|
692 |
|
693 if (not MakeTempFile (in, inName) or not MakeTempFile (out, outName)) |
|
694 return; |
|
695 |
|
696 int unmatched = ui.unmatched->currentIndex(); |
|
697 |
|
698 QString argv = Join ( |
|
699 { |
|
700 format ("-p %1", ui.precision->value()), |
|
701 format ("-af %1", ui.flatAngle->value()), |
|
702 format ("-ac %1", ui.condAngle->value()), |
|
703 format ("-ae %1", ui.edgeAngle->value()), |
|
704 ui.delLines->isChecked() ? "-de" : "", |
|
705 ui.delCondLines->isChecked() ? "-dc" : "", |
|
706 ui.colored->isChecked() ? "-c" : "", |
|
707 ui.bfc->isChecked() ? "-b" : "", |
|
708 ui.convex->isChecked() ? "-cx" : "", |
|
709 ui.concave->isChecked() ? "-cv" : "", |
|
710 unmatched == 0 ? "-u+" : (unmatched == 2 ? "-u-" : ""), |
|
711 inName, |
|
712 outName, |
|
713 }); |
|
714 |
|
715 WriteSelection (inName); |
|
716 |
|
717 if (not RunExtProgram (Edger2, cfg::Edger2Path, argv)) |
|
718 return; |
|
719 |
|
720 InsertOutput (outName, true, {}); |
|
721 } |
|