mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-24 17:21:11 -06:00 
			
		
		
		
	New way of starting arrange and rotation optimization.
To prevent segfaults when exiting while processing is running.
This commit is contained in:
		
							parent
							
								
									42ab22e243
								
							
						
					
					
						commit
						696ade15ca
					
				
					 7 changed files with 191 additions and 109 deletions
				
			
		|  | @ -44,7 +44,7 @@ std::array<double, 3> find_best_rotation(const ModelObject& modelobj, | ||||||
|     // call the status callback in each iteration but the actual value may be
 |     // call the status callback in each iteration but the actual value may be
 | ||||||
|     // the same for subsequent iterations (status goes from 0 to 100 but
 |     // the same for subsequent iterations (status goes from 0 to 100 but
 | ||||||
|     // iterations can be many more)
 |     // iterations can be many more)
 | ||||||
|     auto objfunc = [&emesh, &status, &statuscb, max_tries] |     auto objfunc = [&emesh, &status, &statuscb, &stopcond, max_tries] | ||||||
|             (double rx, double ry, double rz) |             (double rx, double ry, double rz) | ||||||
|     { |     { | ||||||
|         EigenMesh3D& m = emesh; |         EigenMesh3D& m = emesh; | ||||||
|  | @ -91,7 +91,7 @@ std::array<double, 3> find_best_rotation(const ModelObject& modelobj, | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // report status
 |         // report status
 | ||||||
|         statuscb( unsigned(++status * 100.0/max_tries) ); |         if(!stopcond()) statuscb( unsigned(++status * 100.0/max_tries) ); | ||||||
| 
 | 
 | ||||||
|         return score; |         return score; | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|  | @ -54,7 +54,7 @@ DPIFrame(NULL, wxID_ANY, wxString(SLIC3R_BUILD_ID) + " " + _(L("based on Slic3r" | ||||||
| #endif // _WIN32
 | #endif // _WIN32
 | ||||||
| 
 | 
 | ||||||
| 	// initialize status bar
 | 	// initialize status bar
 | ||||||
| 	m_statusbar = new ProgressStatusBar(this); | 	m_statusbar.reset(new ProgressStatusBar(this)); | ||||||
| 	m_statusbar->embed(this); | 	m_statusbar->embed(this); | ||||||
|     m_statusbar->set_status_text(_(L("Version")) + " " + |     m_statusbar->set_status_text(_(L("Version")) + " " + | ||||||
| 		SLIC3R_VERSION + | 		SLIC3R_VERSION + | ||||||
|  | @ -108,6 +108,8 @@ DPIFrame(NULL, wxID_ANY, wxString(SLIC3R_BUILD_ID) + " " + _(L("based on Slic3r" | ||||||
|         // In addition, there were some crashes due to the Paint events sent to already destructed windows.
 |         // In addition, there were some crashes due to the Paint events sent to already destructed windows.
 | ||||||
|         this->Show(false); |         this->Show(false); | ||||||
|          |          | ||||||
|  |         if(m_plater) m_plater->stop_jobs(); | ||||||
|  | 
 | ||||||
|         // Save the slic3r.ini.Usually the ini file is saved from "on idle" callback,
 |         // Save the slic3r.ini.Usually the ini file is saved from "on idle" callback,
 | ||||||
|         // but in rare cases it may not have been called yet.
 |         // but in rare cases it may not have been called yet.
 | ||||||
|         wxGetApp().app_config->save(); |         wxGetApp().app_config->save(); | ||||||
|  | @ -136,6 +138,8 @@ DPIFrame(NULL, wxID_ANY, wxString(SLIC3R_BUILD_ID) + " " + _(L("based on Slic3r" | ||||||
|     update_ui_from_settings();    // FIXME (?)
 |     update_ui_from_settings();    // FIXME (?)
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | MainFrame::~MainFrame() {} | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| void MainFrame::init_tabpanel() | void MainFrame::init_tabpanel() | ||||||
| { | { | ||||||
|  |  | ||||||
|  | @ -86,7 +86,7 @@ protected: | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|     MainFrame(); |     MainFrame(); | ||||||
|     ~MainFrame() {} |     ~MainFrame(); | ||||||
| 
 | 
 | ||||||
|     Plater*     plater() { return m_plater; } |     Plater*     plater() { return m_plater; } | ||||||
| 
 | 
 | ||||||
|  | @ -121,7 +121,7 @@ public: | ||||||
|     Plater*             m_plater { nullptr }; |     Plater*             m_plater { nullptr }; | ||||||
|     wxNotebook*         m_tabpanel { nullptr }; |     wxNotebook*         m_tabpanel { nullptr }; | ||||||
|     wxProgressDialog*   m_progress_dialog { nullptr }; |     wxProgressDialog*   m_progress_dialog { nullptr }; | ||||||
|     ProgressStatusBar*  m_statusbar { nullptr }; |     std::unique_ptr<ProgressStatusBar>  m_statusbar; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| } // GUI
 | } // GUI
 | ||||||
|  |  | ||||||
|  | @ -5,6 +5,8 @@ | ||||||
| #include <vector> | #include <vector> | ||||||
| #include <string> | #include <string> | ||||||
| #include <regex> | #include <regex> | ||||||
|  | #include <future> | ||||||
|  | 
 | ||||||
| #include <boost/algorithm/string/predicate.hpp> | #include <boost/algorithm/string/predicate.hpp> | ||||||
| #include <boost/algorithm/string/trim.hpp> | #include <boost/algorithm/string/trim.hpp> | ||||||
| #include <boost/optional.hpp> | #include <boost/optional.hpp> | ||||||
|  | @ -1219,8 +1221,107 @@ struct Plater::priv | ||||||
|     wxString project_filename; |     wxString project_filename; | ||||||
| 
 | 
 | ||||||
|     BackgroundSlicingProcess    background_process; |     BackgroundSlicingProcess    background_process; | ||||||
|     bool                        arranging; |      | ||||||
|     bool                        rotoptimizing; |     class Job: public wxEvtHandler { | ||||||
|  |         int m_range = 100; | ||||||
|  |         std::future<void> m_ftr; | ||||||
|  |         priv *m_plater = nullptr; | ||||||
|  |         std::atomic<bool> m_running {false}, m_canceled {false}; | ||||||
|  |          | ||||||
|  |         void run() {  | ||||||
|  |             m_running.store(true); process(); m_running.store(false);  | ||||||
|  |              | ||||||
|  |             // ensure to call the last status to finalize the job
 | ||||||
|  |             update_status(status_range(), ""); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |     protected: | ||||||
|  |          | ||||||
|  |         // status range for a particular job
 | ||||||
|  |         virtual int status_range() const { return 100; } | ||||||
|  |          | ||||||
|  |         void update_status(int st, const wxString& msg = "") {  | ||||||
|  |             auto evt = new wxThreadEvent(); evt->SetInt(st); evt->SetString(msg); | ||||||
|  |             wxQueueEvent(this, evt);  | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         priv& plater() { return *m_plater; } | ||||||
|  |         bool was_canceled() const { return m_canceled.load(); } | ||||||
|  |          | ||||||
|  |     public: | ||||||
|  |          | ||||||
|  |         Job(priv *_plater): m_plater(_plater) { | ||||||
|  |             Bind(wxEVT_THREAD, [this](const wxThreadEvent& evt){ | ||||||
|  |                 auto msg = evt.GetString(); | ||||||
|  |                 if(! msg.empty()) plater().statusbar()->set_status_text(msg); | ||||||
|  |                  | ||||||
|  |                 if(! m_range) return; | ||||||
|  |                  | ||||||
|  |                 plater().statusbar()->set_progress(evt.GetInt()); | ||||||
|  |                 if(evt.GetInt() == status_range()) { | ||||||
|  |                      | ||||||
|  |                     // set back the original range and cancel callback
 | ||||||
|  |                     plater().statusbar()->set_range(m_range); | ||||||
|  |                     plater().statusbar()->set_cancel_callback(); | ||||||
|  |                     wxEndBusyCursor(); | ||||||
|  |                      | ||||||
|  |                     // Do a full refresh of scene tree, including regenerating
 | ||||||
|  |                     // all the GLVolumes. FIXME The update function shall just
 | ||||||
|  |                     // reload the modified matrices.
 | ||||||
|  |                     if(! was_canceled()) plater().update(true); | ||||||
|  |                      | ||||||
|  |                     // dont do finalization again for the same process
 | ||||||
|  |                     m_range = 0; | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         virtual void process() = 0; | ||||||
|  |          | ||||||
|  |         void start() {  // only if not running
 | ||||||
|  |             if(! m_running.load()) { | ||||||
|  |                 m_range = plater().statusbar()->get_range(); | ||||||
|  |                 m_canceled.store(false); | ||||||
|  |                 plater().statusbar()->set_range(status_range()); | ||||||
|  |                 plater().statusbar()->set_cancel_callback( [this](){  | ||||||
|  |                     m_canceled.store(true); | ||||||
|  |                 }); | ||||||
|  |                 wxBeginBusyCursor(); | ||||||
|  |                 m_ftr = std::async(std::launch::async, &Job::run, this); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         bool join(int timeout_ms = 0) {  | ||||||
|  |             if(!m_ftr.valid()) return true; | ||||||
|  |              | ||||||
|  |             if(timeout_ms <= 0)  | ||||||
|  |                 m_ftr.wait(); | ||||||
|  |             else if(m_ftr.wait_for(std::chrono::milliseconds(timeout_ms)) ==  | ||||||
|  |                     std::future_status::timeout)  | ||||||
|  |                 return false; | ||||||
|  |              | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         bool is_running() const { return m_running.load(); } | ||||||
|  |         void cancel() { m_canceled.store(true); } | ||||||
|  |     }; | ||||||
|  |      | ||||||
|  |     class ArrangeJob: public Job { | ||||||
|  |         int count = 0; | ||||||
|  |     public: | ||||||
|  |         using Job::Job; | ||||||
|  |         int status_range() const override { return count; } | ||||||
|  |         void set_count(int c) { count = c; } | ||||||
|  |         void process() override; | ||||||
|  |     } arrange_job {this}; | ||||||
|  |      | ||||||
|  |     class RotoptimizeJob: public Job { | ||||||
|  |     public: | ||||||
|  |         using Job::Job; | ||||||
|  |         void process() override; | ||||||
|  |     } rotoptimize_job {this}; | ||||||
|  |      | ||||||
|     bool                        delayed_scene_refresh; |     bool                        delayed_scene_refresh; | ||||||
|     std::string                 delayed_error_message; |     std::string                 delayed_error_message; | ||||||
| 
 | 
 | ||||||
|  | @ -1389,8 +1490,6 @@ Plater::priv::priv(Plater *q, MainFrame *main_frame) | ||||||
| { | { | ||||||
| 	this->q->SetFont(Slic3r::GUI::wxGetApp().normal_font()); | 	this->q->SetFont(Slic3r::GUI::wxGetApp().normal_font()); | ||||||
| 
 | 
 | ||||||
|     arranging = false; |  | ||||||
|     rotoptimizing = false; |  | ||||||
|     background_process.set_fff_print(&fff_print); |     background_process.set_fff_print(&fff_print); | ||||||
| 	background_process.set_sla_print(&sla_print); | 	background_process.set_sla_print(&sla_print); | ||||||
|     background_process.set_gcode_preview_data(&gcode_preview_data); |     background_process.set_gcode_preview_data(&gcode_preview_data); | ||||||
|  | @ -1564,7 +1663,7 @@ void Plater::priv::update_ui_from_settings() | ||||||
| 
 | 
 | ||||||
| ProgressStatusBar* Plater::priv::statusbar() | ProgressStatusBar* Plater::priv::statusbar() | ||||||
| { | { | ||||||
|     return main_frame->m_statusbar; |     return main_frame->m_statusbar.get(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::string Plater::priv::get_config(const std::string &key) const | std::string Plater::priv::get_config(const std::string &key) const | ||||||
|  | @ -2089,46 +2188,37 @@ void Plater::priv::mirror(Axis axis) | ||||||
| 
 | 
 | ||||||
| void Plater::priv::arrange() | void Plater::priv::arrange() | ||||||
| { | { | ||||||
|     if (arranging) { return; } |     if(!arrange_job.is_running()) { | ||||||
|     arranging = true; |         int count = 0; | ||||||
|     Slic3r::ScopeGuard arranging_guard([this]() { arranging = false; }); |         for(auto obj : model.objects) count += int(obj->instances.size()); | ||||||
|  |         arrange_job.set_count(count); | ||||||
|  |         arrange_job.start(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|     wxBusyCursor wait; | // This method will find an optimal orientation for the currently selected item
 | ||||||
|  | // Very similar in nature to the arrange method above...
 | ||||||
|  | void Plater::priv::sla_optimize_rotation() { | ||||||
|      |      | ||||||
|     this->background_process.stop(); |     rotoptimize_job.start(); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|     unsigned count = 0; | void Plater::priv::ArrangeJob::process() { | ||||||
|     for(auto obj : model.objects) count += obj->instances.size(); |  | ||||||
|      |      | ||||||
|     auto prev_range = statusbar()->get_range(); |     // TODO: we should decide whether to allow arrange when the search is
 | ||||||
|     statusbar()->set_range(count); |     // running we should probably disable explicit slicing and background
 | ||||||
| 
 |     // processing
 | ||||||
|     auto statusfn = [this, count] (unsigned st, const std::string& msg) { |  | ||||||
|         /* // In case we would run the arrange asynchronously
 |  | ||||||
|         wxCommandEvent event(EVT_PROGRESS_BAR); |  | ||||||
|         event.SetInt(st); |  | ||||||
|         event.SetString(msg); |  | ||||||
|         wxQueueEvent(this->q, event.Clone()); */ |  | ||||||
|         statusbar()->set_progress(count - st); |  | ||||||
|         statusbar()->set_status_text(_(msg)); |  | ||||||
| 
 |  | ||||||
|         // ok, this is dangerous, but we are protected by the flag
 |  | ||||||
|         // 'arranging' and the arrange button is also disabled.
 |  | ||||||
|         // This call is needed for the cancel button to work.
 |  | ||||||
|         wxYieldIfNeeded(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     statusbar()->set_cancel_callback([this, statusfn](){ |  | ||||||
|         arranging = false; |  | ||||||
|         statusfn(0, L("Arranging canceled")); |  | ||||||
|     }); |  | ||||||
| 
 | 
 | ||||||
|     static const std::string arrangestr = L("Arranging"); |     static const std::string arrangestr = L("Arranging"); | ||||||
|      |      | ||||||
|  |     auto& config = plater().config; | ||||||
|  |     auto& view3D = plater().view3D; | ||||||
|  |     auto& model  = plater().model; | ||||||
|  | 
 | ||||||
|     // FIXME: I don't know how to obtain the minimum distance, it depends
 |     // FIXME: I don't know how to obtain the minimum distance, it depends
 | ||||||
|     // on printer technology. I guess the following should work but it crashes.
 |     // on printer technology. I guess the following should work but it crashes.
 | ||||||
|     double dist = 6; //PrintConfig::min_object_distance(config);
 |     double dist = 6; //PrintConfig::min_object_distance(config);
 | ||||||
|     if(printer_technology == ptFFF) { |     if(plater().printer_technology == ptFFF) { | ||||||
|         dist = PrintConfig::min_object_distance(config); |         dist = PrintConfig::min_object_distance(config); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -2141,7 +2231,7 @@ void Plater::priv::arrange() | ||||||
|     Polyline bed; bed.points.reserve(bedpoints.size()); |     Polyline bed; bed.points.reserve(bedpoints.size()); | ||||||
|     for(auto& v : bedpoints) bed.append(Point::new_scale(v(0), v(1))); |     for(auto& v : bedpoints) bed.append(Point::new_scale(v(0), v(1))); | ||||||
| 
 | 
 | ||||||
|     statusfn(0, arrangestr); |     update_status(0, arrangestr); | ||||||
| 
 | 
 | ||||||
|     arr::WipeTowerInfo wti = view3D->get_canvas3d()->get_wipe_tower_info(); |     arr::WipeTowerInfo wti = view3D->get_canvas3d()->get_wipe_tower_info(); | ||||||
| 
 | 
 | ||||||
|  | @ -2157,67 +2247,34 @@ void Plater::priv::arrange() | ||||||
|                      bed, |                      bed, | ||||||
|                      hint, |                      hint, | ||||||
|                      false, // create many piles not just one pile
 |                      false, // create many piles not just one pile
 | ||||||
|                      [statusfn](unsigned st) { statusfn(st, arrangestr); }, |                      [this](unsigned st) { if(st > 0) update_status(count - int(st), arrangestr); }, | ||||||
|                      [this] () { return !arranging; }); |                      [this] () { return was_canceled(); }); | ||||||
|     } catch(std::exception& /*e*/) { |     } catch(std::exception& /*e*/) { | ||||||
|         GUI::show_error(this->q, L("Could not arrange model objects! " |         GUI::show_error(plater().q, L("Could not arrange model objects! " | ||||||
|                                       "Some geometries may be invalid.")); |                                       "Some geometries may be invalid.")); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     update_status(count, was_canceled() ? L("Arranging canceled.") : L("Arranging done.")); | ||||||
|  | 
 | ||||||
|     // it remains to move the wipe tower:
 |     // it remains to move the wipe tower:
 | ||||||
|     view3D->get_canvas3d()->arrange_wipe_tower(wti);    |     view3D->get_canvas3d()->arrange_wipe_tower(wti);    | ||||||
| 
 |  | ||||||
|     statusfn(0, L("Arranging done.")); |  | ||||||
|     statusbar()->set_range(prev_range); |  | ||||||
|     statusbar()->set_cancel_callback(); // remove cancel button
 |  | ||||||
| 
 |  | ||||||
|     // Do a full refresh of scene tree, including regenerating all the GLVolumes.
 |  | ||||||
|     //FIXME The update function shall just reload the modified matrices.
 |  | ||||||
|     update(true); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // This method will find an optimal orientation for the currently selected item
 | void Plater::priv::RotoptimizeJob::process() | ||||||
| // Very similar in nature to the arrange method above...
 | { | ||||||
| void Plater::priv::sla_optimize_rotation() { |  | ||||||
| 
 | 
 | ||||||
|     // TODO: we should decide whether to allow arrange when the search is
 |     int obj_idx = plater().get_selected_object_idx(); | ||||||
|     // running we should probably disable explicit slicing and background
 |  | ||||||
|     // processing
 |  | ||||||
| 
 |  | ||||||
|     if (rotoptimizing) { return; } |  | ||||||
|     rotoptimizing = true; |  | ||||||
|     Slic3r::ScopeGuard rotoptimizing_guard([this]() { rotoptimizing = false; }); |  | ||||||
| 
 |  | ||||||
|     int obj_idx = get_selected_object_idx(); |  | ||||||
|     if (obj_idx < 0) { return; } |     if (obj_idx < 0) { return; } | ||||||
| 
 | 
 | ||||||
|     ModelObject * o = model.objects[size_t(obj_idx)]; |     ModelObject * o = plater().model.objects[size_t(obj_idx)]; | ||||||
| 
 |  | ||||||
|     background_process.stop(); |  | ||||||
| 
 |  | ||||||
|     auto prev_range = statusbar()->get_range(); |  | ||||||
|     statusbar()->set_range(100); |  | ||||||
| 
 |  | ||||||
|     auto stfn = [this] (unsigned st, const std::string& msg) { |  | ||||||
|         statusbar()->set_progress(int(st)); |  | ||||||
|         statusbar()->set_status_text(msg); |  | ||||||
| 
 |  | ||||||
|         // could be problematic, but we need the cancel button.
 |  | ||||||
|         wxYieldIfNeeded(); |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     statusbar()->set_cancel_callback([this, stfn](){ |  | ||||||
|         rotoptimizing = false; |  | ||||||
|         stfn(0, L("Orientation search canceled")); |  | ||||||
|     }); |  | ||||||
|      |      | ||||||
|     auto r = sla::find_best_rotation( |     auto r = sla::find_best_rotation( | ||||||
|                 *o, .005f, |                 *o, .005f, | ||||||
|                 [stfn](unsigned s) { stfn(s, L("Searching for optimal orientation")); }, |                 [this](unsigned s) { if(s < 100) update_status(int(s), L("Searching for optimal orientation")); }, | ||||||
|                 [this](){ return !rotoptimizing; } |                 [this](){ return was_canceled(); } | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     const auto *bed_shape_opt = config->opt<ConfigOptionPoints>("bed_shape"); |     const auto *bed_shape_opt = plater().config->opt<ConfigOptionPoints>("bed_shape"); | ||||||
|     assert(bed_shape_opt); |     assert(bed_shape_opt); | ||||||
| 
 | 
 | ||||||
|     auto& bedpoints = bed_shape_opt->values; |     auto& bedpoints = bed_shape_opt->values; | ||||||
|  | @ -2227,7 +2284,7 @@ void Plater::priv::sla_optimize_rotation() { | ||||||
|     double mindist = 6.0; // FIXME
 |     double mindist = 6.0; // FIXME
 | ||||||
|     double offs = mindist / 2.0 - EPSILON; |     double offs = mindist / 2.0 - EPSILON; | ||||||
| 
 | 
 | ||||||
|     if(rotoptimizing) // wasn't canceled
 |     if(! was_canceled()) // wasn't canceled
 | ||||||
|     for(ModelInstance * oi : o->instances) { |     for(ModelInstance * oi : o->instances) { | ||||||
|         oi->set_rotation({r[X], r[Y], r[Z]}); |         oi->set_rotation({r[X], r[Y], r[Z]}); | ||||||
| 
 | 
 | ||||||
|  | @ -2267,19 +2324,18 @@ void Plater::priv::sla_optimize_rotation() { | ||||||
| 
 | 
 | ||||||
|         Vec3d rt = oi->get_rotation(); rt(Z) += r; |         Vec3d rt = oi->get_rotation(); rt(Z) += r; | ||||||
|         oi->set_rotation(rt); |         oi->set_rotation(rt); | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|         arr::WipeTowerInfo wti; // useless in SLA context
 |         arr::WipeTowerInfo wti; // useless in SLA context
 | ||||||
|     arr::find_new_position(model, o->instances, coord_t(mindist/SCALING_FACTOR), bed, wti); |         arr::find_new_position(plater().model, o->instances,  | ||||||
|  |                                coord_t(mindist/SCALING_FACTOR), bed, wti); | ||||||
|      |      | ||||||
|         // Correct the z offset of the object which was corrupted be the rotation
 |         // Correct the z offset of the object which was corrupted be the rotation
 | ||||||
|         o->ensure_on_bed(); |         o->ensure_on_bed(); | ||||||
|      |      | ||||||
|     stfn(0, L("Orientation found.")); |         update_status(100, L("Orientation found.")); | ||||||
|     statusbar()->set_range(prev_range); |     } else { | ||||||
|     statusbar()->set_cancel_callback(); |         update_status(100, L("Orientation search canceled.")); | ||||||
| 
 |     } | ||||||
|     update(true); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Plater::priv::split_object() | void Plater::priv::split_object() | ||||||
|  | @ -2455,7 +2511,7 @@ unsigned int Plater::priv::update_background_process(bool force_validation) | ||||||
| // Restart background processing thread based on a bitmask of UpdateBackgroundProcessReturnState.
 | // Restart background processing thread based on a bitmask of UpdateBackgroundProcessReturnState.
 | ||||||
| bool Plater::priv::restart_background_process(unsigned int state) | bool Plater::priv::restart_background_process(unsigned int state) | ||||||
| { | { | ||||||
|     if (arranging || rotoptimizing) { |     if (arrange_job.is_running() || rotoptimize_job.is_running()) { | ||||||
|         // Avoid a race condition
 |         // Avoid a race condition
 | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  | @ -2686,7 +2742,7 @@ void Plater::priv::on_select_preset(wxCommandEvent &evt) | ||||||
| void Plater::priv::on_slicing_update(SlicingStatusEvent &evt) | void Plater::priv::on_slicing_update(SlicingStatusEvent &evt) | ||||||
| { | { | ||||||
|     if (evt.status.percent >= -1) { |     if (evt.status.percent >= -1) { | ||||||
|         if (arranging || rotoptimizing) { |         if (arrange_job.is_running() || rotoptimize_job.is_running()) { | ||||||
|             // Avoid a race condition
 |             // Avoid a race condition
 | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  | @ -3140,7 +3196,7 @@ bool Plater::priv::can_fix_through_netfabb() const | ||||||
| 
 | 
 | ||||||
| bool Plater::priv::can_increase_instances() const | bool Plater::priv::can_increase_instances() const | ||||||
| { | { | ||||||
|     if (arranging || rotoptimizing) { |     if (arrange_job.is_running() || rotoptimize_job.is_running()) { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -3150,7 +3206,7 @@ bool Plater::priv::can_increase_instances() const | ||||||
| 
 | 
 | ||||||
| bool Plater::priv::can_decrease_instances() const | bool Plater::priv::can_decrease_instances() const | ||||||
| { | { | ||||||
|     if (arranging || rotoptimizing) { |     if (arrange_job.is_running() || rotoptimize_job.is_running()) { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -3170,7 +3226,7 @@ bool Plater::priv::can_split_to_volumes() const | ||||||
| 
 | 
 | ||||||
| bool Plater::priv::can_arrange() const | bool Plater::priv::can_arrange() const | ||||||
| { | { | ||||||
|     return !model.objects.empty() && !arranging; |     return !model.objects.empty() && !arrange_job.is_running(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool Plater::priv::can_layers_editing() const | bool Plater::priv::can_layers_editing() const | ||||||
|  | @ -3292,6 +3348,21 @@ void Plater::load_files(const std::vector<std::string>& input_files, bool load_m | ||||||
| 
 | 
 | ||||||
| void Plater::update() { p->update(); } | void Plater::update() { p->update(); } | ||||||
| 
 | 
 | ||||||
|  | void Plater::stop_jobs() | ||||||
|  | { | ||||||
|  |     static const int ABORT_WAIT_MAX_MS = 10000; | ||||||
|  |     bool aborted = false; | ||||||
|  |      | ||||||
|  |     p->rotoptimize_job.cancel(); | ||||||
|  |     aborted = p->rotoptimize_job.join(ABORT_WAIT_MAX_MS); | ||||||
|  |          | ||||||
|  |     p->arrange_job.cancel(); | ||||||
|  |     aborted &= p->arrange_job.join(ABORT_WAIT_MAX_MS); | ||||||
|  |      | ||||||
|  |     if(!aborted)  | ||||||
|  |         BOOST_LOG_TRIVIAL(error) << "Could not abort an optimization job!"; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void Plater::update_ui_from_settings() { p->update_ui_from_settings(); } | void Plater::update_ui_from_settings() { p->update_ui_from_settings(); } | ||||||
| 
 | 
 | ||||||
| void Plater::select_view(const std::string& direction) { p->select_view(direction); } | void Plater::select_view(const std::string& direction) { p->select_view(direction); } | ||||||
|  |  | ||||||
|  | @ -143,6 +143,7 @@ public: | ||||||
|     void load_files(const std::vector<std::string>& input_files, bool load_model = true, bool load_config = true); |     void load_files(const std::vector<std::string>& input_files, bool load_model = true, bool load_config = true); | ||||||
| 
 | 
 | ||||||
|     void update(); |     void update(); | ||||||
|  |     void stop_jobs(); | ||||||
|     void select_view(const std::string& direction); |     void select_view(const std::string& direction); | ||||||
|     void select_view_3D(const std::string& name); |     void select_view_3D(const std::string& name); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -168,6 +168,11 @@ void ProgressStatusBar::set_status_text(const char *txt) | ||||||
|     this->set_status_text(wxString::FromUTF8(txt)); |     this->set_status_text(wxString::FromUTF8(txt)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | wxString ProgressStatusBar::get_status_text() const | ||||||
|  | { | ||||||
|  |     return self->GetStatusText(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ProgressStatusBar::show_cancel_button() | void ProgressStatusBar::show_cancel_button() | ||||||
| { | { | ||||||
|     if(m_cancelbutton) m_cancelbutton->Show(); |     if(m_cancelbutton) m_cancelbutton->Show(); | ||||||
|  |  | ||||||
|  | @ -51,6 +51,7 @@ public: | ||||||
|     void        set_status_text(const wxString& txt); |     void        set_status_text(const wxString& txt); | ||||||
|     void        set_status_text(const std::string& txt); |     void        set_status_text(const std::string& txt); | ||||||
|     void        set_status_text(const char *txt); |     void        set_status_text(const char *txt); | ||||||
|  |     wxString    get_status_text() const; | ||||||
| 
 | 
 | ||||||
|     // Temporary methods to satisfy Perl side
 |     // Temporary methods to satisfy Perl side
 | ||||||
|     void        show_cancel_button(); |     void        show_cancel_button(); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 tamasmeszaros
						tamasmeszaros