mirror of
https://github.com/SoftFever/OrcaSlicer.git
synced 2025-10-21 15:51:10 -06:00

* Rework UI jobs to make them more understandable and flexible. * Update Orca specific jobs * Fix progress issue * Fix dark mode and window radius * Update cereal version from 1.2.2 to 1.3.0 (cherry picked from commit prusa3d/PrusaSlicer@057232a275) * Initial port of Emboss gizmo * Bump up CGAL version to 5.4 (cherry picked from commit prusa3d/PrusaSlicer@1bf9dee3e7) * Fix text rotation * Fix test dragging * Add text gizmo to right click menu * Initial port of SVG gizmo * Fix text rotation * Fix Linux build * Fix "from surface" * Fix -90 rotation * Fix icon path * Fix loading font with non-ascii name * Fix storing non-utf8 font descriptor in 3mf file * Fix filtering with non-utf8 characters * Emboss: Use Orca style input dialog * Fix build on macOS * Fix tooltip color in light mode * InputText: fixed incorrect padding when FrameBorder > 0. (ocornut/imgui#4794, ocornut/imgui#3781) InputTextMultiline: fixed vertical tracking with large values of FramePadding.y. (ocornut/imgui#3781, ocornut/imgui#4794) (cherry picked from commit ocornut/imgui@072caa4a90) (cherry picked from commit ocornut/imgui@bdd2a94315) * SVG: Use Orca style input dialog * Fix job progress update * Fix crash when select editing text in preview screen * Use Orca checkbox style * Fix issue that toolbar icons are kept regenerated * Emboss: Fix text & icon alignment * SVG: Fix text & icon alignment * Emboss: fix toolbar icon mouse hover state * Add a simple subtle outline effect by drawing back faces using wireframe mode * Disable selection outlines * Show outline in white if the model color is too dark * Make the outline algorithm more reliable * Enable cull face, which fix render on Linux * Fix `disable_cullface` * Post merge fix * Optimize selection rendering * Fix scale gizmo * Emboss: Fix text rotation if base object is scaled * Fix volume synchronize * Fix emboss rotation * Emboss: Fix advance toggle * Fix text position after reopened the project * Make font style preview darker * Make font style preview selector height shorter --------- Co-authored-by: tamasmeszaros <meszaros.q@gmail.com> Co-authored-by: ocornut <omarcornut@gmail.com> Co-authored-by: SoftFever <softfeverever@gmail.com>
186 lines
5 KiB
C++
186 lines
5 KiB
C++
///|/ Copyright (c) Prusa Research 2021 Tomáš Mészáros @tamasmeszaros
|
|
///|/
|
|
///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
|
|
///|/
|
|
#include <exception>
|
|
|
|
#include "BoostThreadWorker.hpp"
|
|
|
|
namespace Slic3r { namespace GUI {
|
|
|
|
void BoostThreadWorker::WorkerMessage::deliver(BoostThreadWorker &runner)
|
|
{
|
|
switch(MsgType(get_type())) {
|
|
case Empty: break;
|
|
case Status: {
|
|
auto info = boost::get<StatusInfo>(m_data);
|
|
if (runner.get_pri()) {
|
|
runner.get_pri()->set_progress(info.status);
|
|
runner.get_pri()->set_status_text(info.msg.c_str());
|
|
}
|
|
break;
|
|
}
|
|
case Finalize: {
|
|
auto& entry = boost::get<JobEntry>(m_data);
|
|
entry.job->finalize(entry.canceled, entry.eptr);
|
|
|
|
// Unhandled exceptions are rethrown without mercy.
|
|
if (entry.eptr)
|
|
std::rethrow_exception(entry.eptr);
|
|
|
|
break;
|
|
}
|
|
case MainThreadCall: {
|
|
auto &calldata = boost::get<MainThreadCallData >(m_data);
|
|
calldata.fn();
|
|
calldata.promise.set_value();
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void BoostThreadWorker::run()
|
|
{
|
|
bool stop = false;
|
|
while (!stop) {
|
|
m_input_queue
|
|
.consume_one(BlockingWait{0, &m_running}, [this, &stop](JobEntry &e) {
|
|
if (!e.job)
|
|
stop = true;
|
|
else {
|
|
m_canceled.store(false);
|
|
|
|
try {
|
|
e.job->process(*this);
|
|
} catch (...) {
|
|
e.eptr = std::current_exception();
|
|
}
|
|
|
|
e.canceled = m_canceled.load();
|
|
m_output_queue.push(std::move(e)); // finalization message
|
|
}
|
|
m_running.store(false);
|
|
});
|
|
};
|
|
}
|
|
|
|
void BoostThreadWorker::update_status(int st, const std::string &msg)
|
|
{
|
|
m_output_queue.push(st, msg);
|
|
}
|
|
|
|
std::future<void> BoostThreadWorker::call_on_main_thread(std::function<void ()> fn)
|
|
{
|
|
MainThreadCallData cbdata{std::move(fn), {}};
|
|
std::future<void> future = cbdata.promise.get_future();
|
|
|
|
m_output_queue.push(std::move(cbdata));
|
|
|
|
return future;
|
|
}
|
|
|
|
BoostThreadWorker::BoostThreadWorker(std::shared_ptr<ProgressIndicator> pri,
|
|
boost::thread::attributes &attribs,
|
|
const char * name)
|
|
: m_progress(std::move(pri)), m_name{name}
|
|
{
|
|
if (m_progress)
|
|
m_progress->set_cancel_callback([this](){ cancel(); });
|
|
|
|
m_thread = create_thread(attribs, [this] { this->run(); });
|
|
|
|
std::string nm{name};
|
|
if (!nm.empty()) set_thread_name(m_thread, name);
|
|
}
|
|
|
|
constexpr int ABORT_WAIT_MAX_MS = 10000;
|
|
|
|
BoostThreadWorker::~BoostThreadWorker()
|
|
{
|
|
bool joined = false;
|
|
try {
|
|
cancel_all();
|
|
wait_for_idle(ABORT_WAIT_MAX_MS);
|
|
m_input_queue.push(JobEntry{nullptr});
|
|
joined = join(ABORT_WAIT_MAX_MS);
|
|
} catch(...) {}
|
|
|
|
if (!joined)
|
|
BOOST_LOG_TRIVIAL(error)
|
|
<< "Could not join worker thread '" << m_name << "'";
|
|
}
|
|
|
|
bool BoostThreadWorker::join(int timeout_ms)
|
|
{
|
|
if (!m_thread.joinable())
|
|
return true;
|
|
|
|
if (timeout_ms <= 0) {
|
|
m_thread.join();
|
|
}
|
|
else if (m_thread.try_join_for(boost::chrono::milliseconds(timeout_ms))) {
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void BoostThreadWorker::process_events()
|
|
{
|
|
while (m_output_queue.consume_one([this](WorkerMessage &msg) {
|
|
msg.deliver(*this);
|
|
}));
|
|
}
|
|
|
|
bool BoostThreadWorker::wait_for_current_job(unsigned timeout_ms)
|
|
{
|
|
bool ret = true;
|
|
|
|
if (!is_idle()) {
|
|
bool was_finish = false;
|
|
bool timeout_reached = false;
|
|
while (!timeout_reached && !was_finish) {
|
|
timeout_reached =
|
|
!m_output_queue.consume_one(BlockingWait{timeout_ms},
|
|
[this, &was_finish](
|
|
WorkerMessage &msg) {
|
|
msg.deliver(*this);
|
|
if (msg.get_type() ==
|
|
WorkerMessage::Finalize)
|
|
was_finish = true;
|
|
});
|
|
}
|
|
|
|
ret = !timeout_reached;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool BoostThreadWorker::wait_for_idle(unsigned timeout_ms)
|
|
{
|
|
bool timeout_reached = false;
|
|
while (!timeout_reached && !is_idle()) {
|
|
timeout_reached = !m_output_queue
|
|
.consume_one(BlockingWait{timeout_ms},
|
|
[this](WorkerMessage &msg) {
|
|
msg.deliver(*this);
|
|
});
|
|
}
|
|
|
|
return !timeout_reached;
|
|
}
|
|
|
|
bool BoostThreadWorker::push(std::unique_ptr<Job> job)
|
|
{
|
|
if (!job)
|
|
return false;
|
|
|
|
m_input_queue.push(JobEntry{std::move(job)});
|
|
return true;
|
|
}
|
|
|
|
}} // namespace Slic3r::GUI
|