2018-11-08 19:18:40 +00:00
|
|
|
#ifndef slic3r_SLAPrint_hpp_
|
|
|
|
#define slic3r_SLAPrint_hpp_
|
|
|
|
|
2018-11-21 14:21:57 +00:00
|
|
|
#include <mutex>
|
2018-11-08 19:18:40 +00:00
|
|
|
#include "PrintBase.hpp"
|
2018-11-13 16:33:03 +00:00
|
|
|
#include "PrintExport.hpp"
|
2018-11-08 19:18:40 +00:00
|
|
|
#include "Point.hpp"
|
2018-11-21 14:21:57 +00:00
|
|
|
#include "MTUtils.hpp"
|
2018-11-08 19:18:40 +00:00
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
|
2018-11-13 17:44:30 +00:00
|
|
|
enum SLAPrintStep : unsigned int {
|
2018-11-08 19:18:40 +00:00
|
|
|
slapsRasterize,
|
2018-11-09 11:02:42 +00:00
|
|
|
slapsValidate,
|
2018-11-08 19:18:40 +00:00
|
|
|
slapsCount
|
|
|
|
};
|
|
|
|
|
2018-11-13 17:44:30 +00:00
|
|
|
enum SLAPrintObjectStep : unsigned int {
|
2018-11-09 11:02:42 +00:00
|
|
|
slaposObjectSlice,
|
2018-11-08 19:18:40 +00:00
|
|
|
slaposSupportPoints,
|
|
|
|
slaposSupportTree,
|
|
|
|
slaposBasePool,
|
2018-11-09 11:02:42 +00:00
|
|
|
slaposSliceSupports,
|
2018-11-26 13:43:28 +00:00
|
|
|
slaposIndexSlices,
|
2018-11-08 19:18:40 +00:00
|
|
|
slaposCount
|
|
|
|
};
|
|
|
|
|
|
|
|
class SLAPrint;
|
2018-11-12 16:35:57 +00:00
|
|
|
class GLCanvas;
|
2018-11-08 19:18:40 +00:00
|
|
|
|
2018-11-09 17:32:35 +00:00
|
|
|
using _SLAPrintObjectBase =
|
|
|
|
PrintObjectBaseWithState<SLAPrint, SLAPrintObjectStep, slaposCount>;
|
|
|
|
|
2018-11-29 17:12:40 +00:00
|
|
|
// Layers according to quantized height levels. This will be consumed by
|
|
|
|
// the printer (rasterizer) in the SLAPrint class.
|
|
|
|
using LevelID = long long;
|
|
|
|
|
2018-11-09 17:32:35 +00:00
|
|
|
class SLAPrintObject : public _SLAPrintObjectBase
|
2018-11-08 19:18:40 +00:00
|
|
|
{
|
|
|
|
private: // Prevents erroneous use by other classes.
|
2018-11-09 17:32:35 +00:00
|
|
|
using Inherited = _SLAPrintObjectBase;
|
2018-11-08 19:18:40 +00:00
|
|
|
|
|
|
|
public:
|
2018-11-21 16:35:35 +00:00
|
|
|
const SLAPrintObjectConfig& config() const { return m_config; }
|
|
|
|
const Transform3d& trafo() const { return m_trafo; }
|
2018-11-13 16:45:44 +00:00
|
|
|
|
|
|
|
struct Instance {
|
2018-11-13 18:22:05 +00:00
|
|
|
Instance(ModelID instance_id, const Point &shift, float rotation) : instance_id(instance_id), shift(shift), rotation(rotation) {}
|
2018-11-21 16:35:35 +00:00
|
|
|
bool operator==(const Instance &rhs) const { return this->instance_id == rhs.instance_id && this->shift == rhs.shift && this->rotation == rhs.rotation; }
|
2018-11-13 18:22:05 +00:00
|
|
|
// ID of the corresponding ModelInstance.
|
|
|
|
ModelID instance_id;
|
|
|
|
// Slic3r::Point objects in scaled G-code coordinates
|
2018-11-13 16:45:44 +00:00
|
|
|
Point shift;
|
|
|
|
// Rotation along the Z axis, in radians.
|
2018-11-21 16:35:35 +00:00
|
|
|
float rotation;
|
2018-11-13 18:22:05 +00:00
|
|
|
};
|
2018-11-13 16:45:44 +00:00
|
|
|
const std::vector<Instance>& instances() const { return m_instances; }
|
|
|
|
|
2018-11-17 16:23:56 +00:00
|
|
|
bool has_mesh(SLAPrintObjectStep step) const;
|
|
|
|
TriangleMesh get_mesh(SLAPrintObjectStep step) const;
|
|
|
|
|
2018-11-13 16:45:44 +00:00
|
|
|
// Get a support mesh centered around origin in XY, and with zero rotation around Z applied.
|
|
|
|
// Support mesh is only valid if this->is_step_done(slaposSupportTree) is true.
|
2018-11-21 14:21:57 +00:00
|
|
|
const TriangleMesh& support_mesh() const;
|
2018-11-13 16:45:44 +00:00
|
|
|
// Get a pad mesh centered around origin in XY, and with zero rotation around Z applied.
|
2018-11-26 13:43:28 +00:00
|
|
|
// Support mesh is only valid if this->is_step_done(slaposBasePool) is true.
|
2018-11-21 14:21:57 +00:00
|
|
|
const TriangleMesh& pad_mesh() const;
|
2018-11-09 11:02:42 +00:00
|
|
|
|
2018-11-15 17:05:47 +00:00
|
|
|
// This will return the transformed mesh which is cached
|
|
|
|
const TriangleMesh& transformed_mesh() const;
|
|
|
|
|
2019-01-30 07:26:23 +00:00
|
|
|
std::vector<sla::SupportPoint> transformed_support_points() const;
|
2018-11-16 10:34:19 +00:00
|
|
|
|
2018-11-16 16:25:23 +00:00
|
|
|
// Get the needed Z elevation for the model geometry if supports should be
|
|
|
|
// displayed. This Z offset should also be applied to the support
|
|
|
|
// geometries. Note that this is not the same as the value stored in config
|
|
|
|
// as the pad height also needs to be considered.
|
|
|
|
double get_elevation() const;
|
|
|
|
|
2018-11-22 12:57:13 +00:00
|
|
|
// This method returns the needed elevation according to the processing
|
|
|
|
// status. If the supports are not ready, it is zero, if they are and the
|
|
|
|
// pad is not, then without the pad, otherwise the full value is returned.
|
|
|
|
double get_current_elevation() const;
|
2018-11-16 16:25:23 +00:00
|
|
|
|
2018-11-26 13:43:28 +00:00
|
|
|
// These two methods should be callable on the client side (e.g. UI thread)
|
|
|
|
// when the appropriate steps slaposObjectSlice and slaposSliceSupports
|
|
|
|
// are ready. All the print objects are processed before slapsRasterize so
|
|
|
|
// it is safe to call them during and/or after slapsRasterize.
|
2018-11-21 14:21:57 +00:00
|
|
|
const std::vector<ExPolygons>& get_model_slices() const;
|
2018-11-26 13:43:28 +00:00
|
|
|
const std::vector<ExPolygons>& get_support_slices() const;
|
|
|
|
|
2018-12-21 11:35:20 +00:00
|
|
|
// This method returns the support points of this SLAPrintObject.
|
2019-02-17 12:05:22 +00:00
|
|
|
const std::vector<sla::SupportPoint>& get_support_points() const;
|
2018-12-21 11:35:20 +00:00
|
|
|
|
2018-11-26 14:03:57 +00:00
|
|
|
// An index record referencing the slices
|
|
|
|
// (get_model_slices(), get_support_slices()) where the keys are the height
|
|
|
|
// levels of the model in scaled-clipper coordinates. The levels correspond
|
|
|
|
// to the z coordinate of the object coordinate system.
|
2018-11-26 13:43:28 +00:00
|
|
|
struct SliceRecord {
|
2018-11-28 14:02:23 +00:00
|
|
|
using Key = float;
|
2018-11-26 13:43:28 +00:00
|
|
|
|
|
|
|
using Idx = size_t;
|
2018-11-26 14:03:57 +00:00
|
|
|
static const Idx NONE = Idx(-1); // this will be the max limit of size_t
|
2018-11-26 13:43:28 +00:00
|
|
|
|
|
|
|
Idx model_slices_idx = NONE;
|
|
|
|
Idx support_slices_idx = NONE;
|
|
|
|
};
|
|
|
|
|
|
|
|
using SliceIndex = std::map<SliceRecord::Key, SliceRecord>;
|
|
|
|
|
|
|
|
// Retrieve the slice index which is readable only after slaposIndexSlices
|
|
|
|
// is done.
|
|
|
|
const SliceIndex& get_slice_index() const;
|
2018-11-16 16:25:23 +00:00
|
|
|
|
2018-11-09 17:32:35 +00:00
|
|
|
// I refuse to grantee copying (Tamas)
|
|
|
|
SLAPrintObject(const SLAPrintObject&) = delete;
|
|
|
|
SLAPrintObject& operator=(const SLAPrintObject&) = delete;
|
|
|
|
|
2018-11-09 11:02:42 +00:00
|
|
|
protected:
|
|
|
|
// to be called from SLAPrint only.
|
|
|
|
friend class SLAPrint;
|
|
|
|
|
|
|
|
SLAPrintObject(SLAPrint* print, ModelObject* model_object);
|
2018-11-09 17:32:35 +00:00
|
|
|
~SLAPrintObject();
|
2018-11-09 11:02:42 +00:00
|
|
|
|
|
|
|
void config_apply(const ConfigBase &other, bool ignore_nonexistent = false) { this->m_config.apply(other, ignore_nonexistent); }
|
|
|
|
void config_apply_only(const ConfigBase &other, const t_config_option_keys &keys, bool ignore_nonexistent = false)
|
|
|
|
{ this->m_config.apply_only(other, keys, ignore_nonexistent); }
|
2018-11-21 14:21:57 +00:00
|
|
|
|
|
|
|
void set_trafo(const Transform3d& trafo) {
|
|
|
|
m_transformed_rmesh.invalidate([this, &trafo](){ m_trafo = trafo; });
|
|
|
|
}
|
2018-11-09 11:02:42 +00:00
|
|
|
|
2018-11-21 16:35:35 +00:00
|
|
|
void set_instances(const std::vector<Instance> &instances) { m_instances = instances; }
|
2018-11-09 11:02:42 +00:00
|
|
|
// Invalidates the step, and its depending steps in SLAPrintObject and SLAPrint.
|
|
|
|
bool invalidate_step(SLAPrintObjectStep step);
|
2018-11-21 16:35:35 +00:00
|
|
|
bool invalidate_all_steps();
|
|
|
|
// Invalidate steps based on a set of parameters changed.
|
|
|
|
bool invalidate_state_by_config_options(const std::vector<t_config_option_key> &opt_keys);
|
2018-11-08 19:18:40 +00:00
|
|
|
|
2019-02-21 07:44:07 +00:00
|
|
|
// Which steps have to be performed. Implicitly: all
|
|
|
|
// to be accessible from SLAPrint
|
|
|
|
std::vector<bool> m_stepmask;
|
|
|
|
|
2018-11-08 19:18:40 +00:00
|
|
|
private:
|
2018-11-09 11:02:42 +00:00
|
|
|
// Object specific configuration, pulled from the configuration layer.
|
|
|
|
SLAPrintObjectConfig m_config;
|
|
|
|
// Translation in Z + Rotation by Y and Z + Scaling / Mirroring.
|
|
|
|
Transform3d m_trafo = Transform3d::Identity();
|
|
|
|
std::vector<Instance> m_instances;
|
|
|
|
|
2019-01-15 13:25:28 +00:00
|
|
|
// Individual 2d slice polygons from lower z to higher z levels
|
2018-11-13 16:33:03 +00:00
|
|
|
std::vector<ExPolygons> m_model_slices;
|
2019-01-15 13:25:28 +00:00
|
|
|
|
|
|
|
// Exact (float) height levels mapped to the slices. Each record contains
|
|
|
|
// the index to the model and the support slice vectors.
|
2018-11-26 13:43:28 +00:00
|
|
|
SliceIndex m_slice_index;
|
2019-01-15 13:25:28 +00:00
|
|
|
|
|
|
|
// The height levels corrected and scaled up in integer values. This will
|
|
|
|
// be used at rasterization.
|
2018-11-29 17:12:40 +00:00
|
|
|
std::vector<LevelID> m_level_ids;
|
2018-11-08 19:18:40 +00:00
|
|
|
|
2018-11-15 17:05:47 +00:00
|
|
|
// Caching the transformed (m_trafo) raw mesh of the object
|
2018-11-21 14:21:57 +00:00
|
|
|
mutable CachedObject<TriangleMesh> m_transformed_rmesh;
|
2018-11-15 17:05:47 +00:00
|
|
|
|
2018-11-09 17:32:35 +00:00
|
|
|
class SupportData;
|
|
|
|
std::unique_ptr<SupportData> m_supportdata;
|
2018-11-08 19:18:40 +00:00
|
|
|
};
|
|
|
|
|
2018-11-12 16:35:57 +00:00
|
|
|
using PrintObjects = std::vector<SLAPrintObject*>;
|
|
|
|
|
|
|
|
class TriangleMesh;
|
|
|
|
|
2019-02-12 15:34:42 +00:00
|
|
|
struct SLAPrintStatistics
|
|
|
|
{
|
|
|
|
SLAPrintStatistics() { clear(); }
|
|
|
|
std::string estimated_print_time;
|
2019-02-13 14:35:41 +00:00
|
|
|
double objects_used_material;
|
|
|
|
double support_used_material;
|
2019-02-18 11:28:58 +00:00
|
|
|
size_t slow_layers_count;
|
|
|
|
size_t fast_layers_count;
|
2019-02-12 15:34:42 +00:00
|
|
|
double total_cost;
|
|
|
|
double total_weight;
|
|
|
|
|
|
|
|
// Config with the filled in print statistics.
|
|
|
|
DynamicConfig config() const;
|
|
|
|
// Config with the statistics keys populated with placeholder strings.
|
|
|
|
static DynamicConfig placeholders();
|
|
|
|
// Replace the print statistics placeholders in the path.
|
|
|
|
std::string finalize_output_path(const std::string &path_in) const;
|
|
|
|
|
|
|
|
void clear() {
|
|
|
|
estimated_print_time.clear();
|
2019-02-13 14:35:41 +00:00
|
|
|
objects_used_material = 0.;
|
|
|
|
support_used_material = 0.;
|
2019-02-18 11:28:58 +00:00
|
|
|
slow_layers_count = 0;
|
|
|
|
fast_layers_count = 0;
|
2019-02-12 15:34:42 +00:00
|
|
|
total_cost = 0.;
|
|
|
|
total_weight = 0.;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-11-08 19:18:40 +00:00
|
|
|
/**
|
|
|
|
* @brief This class is the high level FSM for the SLA printing process.
|
|
|
|
*
|
|
|
|
* It should support the background processing framework and contain the
|
|
|
|
* metadata for the support geometries and their slicing. It should also
|
|
|
|
* dispatch the SLA printing configuration values to the appropriate calculation
|
|
|
|
* steps.
|
|
|
|
*/
|
|
|
|
class SLAPrint : public PrintBaseWithState<SLAPrintStep, slapsCount>
|
|
|
|
{
|
|
|
|
private: // Prevents erroneous use by other classes.
|
|
|
|
typedef PrintBaseWithState<SLAPrintStep, slapsCount> Inherited;
|
|
|
|
|
|
|
|
public:
|
2018-11-13 10:53:54 +00:00
|
|
|
SLAPrint(): m_stepmask(slapsCount, true) {}
|
2018-11-13 16:33:03 +00:00
|
|
|
|
2018-12-13 14:33:39 +00:00
|
|
|
virtual ~SLAPrint() override { this->clear(); }
|
2018-11-08 19:18:40 +00:00
|
|
|
|
2018-12-13 14:33:39 +00:00
|
|
|
PrinterTechnology technology() const noexcept override { return ptSLA; }
|
2018-11-08 19:18:40 +00:00
|
|
|
|
|
|
|
void clear() override;
|
2018-11-22 09:35:04 +00:00
|
|
|
bool empty() const override { return m_objects.empty(); }
|
2018-11-08 19:18:40 +00:00
|
|
|
ApplyStatus apply(const Model &model, const DynamicPrintConfig &config) override;
|
2019-02-21 10:40:56 +00:00
|
|
|
void set_task(const TaskParams ¶ms) override;
|
2018-11-08 19:18:40 +00:00
|
|
|
void process() override;
|
2019-02-21 10:40:56 +00:00
|
|
|
void finalize() override;
|
2018-12-12 09:02:01 +00:00
|
|
|
// Returns true if an object step is done on all objects and there's at least one object.
|
2018-12-11 16:49:31 +00:00
|
|
|
bool is_step_done(SLAPrintObjectStep step) const;
|
|
|
|
// Returns true if the last step was finished with success.
|
2018-12-22 09:02:42 +00:00
|
|
|
bool finished() const override { return this->is_step_done(slaposIndexSlices) && this->Inherited::is_step_done(slapsRasterize); }
|
2018-11-08 19:18:40 +00:00
|
|
|
|
2018-11-13 16:33:03 +00:00
|
|
|
template<class Fmt> void export_raster(const std::string& fname) {
|
|
|
|
if(m_printer) m_printer->save<Fmt>(fname);
|
|
|
|
}
|
2018-11-13 16:45:44 +00:00
|
|
|
const PrintObjects& objects() const { return m_objects; }
|
|
|
|
|
2019-02-21 14:46:04 +00:00
|
|
|
std::string output_filename() const override;
|
2018-12-03 12:14:28 +00:00
|
|
|
|
2019-02-12 15:34:42 +00:00
|
|
|
const SLAPrintStatistics& print_statistics() const { return m_print_statistics; }
|
|
|
|
|
2018-11-08 19:18:40 +00:00
|
|
|
private:
|
2018-11-13 16:33:03 +00:00
|
|
|
using SLAPrinter = FilePrinter<FilePrinterFormat::SLA_PNGZIP>;
|
|
|
|
using SLAPrinterPtr = std::unique_ptr<SLAPrinter>;
|
|
|
|
|
2018-11-21 16:35:35 +00:00
|
|
|
// Invalidate steps based on a set of parameters changed.
|
|
|
|
bool invalidate_state_by_config_options(const std::vector<t_config_option_key> &opt_keys);
|
|
|
|
|
2019-02-12 15:34:42 +00:00
|
|
|
void fill_statistics();
|
|
|
|
|
2018-12-03 12:14:28 +00:00
|
|
|
SLAPrintConfig m_print_config;
|
2018-11-08 19:18:40 +00:00
|
|
|
SLAPrinterConfig m_printer_config;
|
|
|
|
SLAMaterialConfig m_material_config;
|
2018-11-21 16:35:35 +00:00
|
|
|
SLAPrintObjectConfig m_default_object_config;
|
|
|
|
|
2018-11-12 16:35:57 +00:00
|
|
|
PrintObjects m_objects;
|
2018-11-13 10:53:54 +00:00
|
|
|
std::vector<bool> m_stepmask;
|
2018-11-29 11:53:56 +00:00
|
|
|
|
|
|
|
// Definition of the print input map. It consists of the slices indexed
|
|
|
|
// with scaled (clipper) Z coordinates. Also contains the instance
|
|
|
|
// transformations in scaled and filtered version. This is enough for the
|
|
|
|
// rasterizer to be able to draw every layer in the right position
|
|
|
|
using Layer = ExPolygons;
|
|
|
|
using LayerCopies = std::vector<SLAPrintObject::Instance>;
|
|
|
|
struct LayerRef {
|
|
|
|
std::reference_wrapper<const Layer> lref;
|
|
|
|
std::reference_wrapper<const LayerCopies> copies;
|
|
|
|
LayerRef(const Layer& lyr, const LayerCopies& cp) :
|
|
|
|
lref(std::cref(lyr)), copies(std::cref(cp)) {}
|
|
|
|
};
|
|
|
|
|
2018-12-14 15:15:59 +00:00
|
|
|
std::vector<float> calculate_heights(const BoundingBoxf3& bb, float elevation, float initial_layer_height, float layer_height) const;
|
|
|
|
|
2018-11-29 11:53:56 +00:00
|
|
|
// One level may contain multiple slices from multiple objects and their
|
|
|
|
// supports
|
|
|
|
using LayerRefs = std::vector<LayerRef>;
|
|
|
|
std::map<LevelID, LayerRefs> m_printer_input;
|
|
|
|
|
|
|
|
// The printer itself
|
|
|
|
SLAPrinterPtr m_printer;
|
2018-11-08 19:18:40 +00:00
|
|
|
|
2019-02-12 15:34:42 +00:00
|
|
|
// Estimated print time, material consumed.
|
|
|
|
SLAPrintStatistics m_print_statistics;
|
|
|
|
|
2018-11-08 19:18:40 +00:00
|
|
|
friend SLAPrintObject;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Slic3r
|
|
|
|
|
|
|
|
#endif /* slic3r_SLAPrint_hpp_ */
|