2015-07-01 18:14:05 +00:00
|
|
|
|
#include "GCode.hpp"
|
2015-07-01 21:14:40 +00:00
|
|
|
|
#include "ExtrusionEntity.hpp"
|
2016-09-12 14:25:15 +00:00
|
|
|
|
#include "EdgeGrid.hpp"
|
2017-05-03 16:28:22 +00:00
|
|
|
|
#include "Geometry.hpp"
|
2017-05-16 11:45:28 +00:00
|
|
|
|
#include "GCode/WipeTowerPrusaMM.hpp"
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
2015-07-02 16:57:40 +00:00
|
|
|
|
#include <algorithm>
|
2015-07-02 18:24:16 +00:00
|
|
|
|
#include <cstdlib>
|
2015-09-30 13:22:49 +00:00
|
|
|
|
#include <math.h>
|
2015-07-01 18:14:05 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
|
#include <boost/algorithm/string/find.hpp>
|
|
|
|
|
#include <boost/date_time/local_time/local_time.hpp>
|
|
|
|
|
#include <boost/foreach.hpp>
|
|
|
|
|
|
2016-09-12 14:25:15 +00:00
|
|
|
|
#include "SVG.hpp"
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
// Enable debugging and asserts, even in the release build.
|
|
|
|
|
#define DEBUG
|
|
|
|
|
#define _DEBUG
|
|
|
|
|
#undef NDEBUG
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
2015-07-01 18:14:05 +00:00
|
|
|
|
namespace Slic3r {
|
|
|
|
|
|
2017-05-05 07:59:56 +00:00
|
|
|
|
Polyline AvoidCrossingPerimeters::travel_to(GCode &gcodegen, Point point)
|
2015-07-01 18:14:05 +00:00
|
|
|
|
{
|
2017-05-05 07:59:56 +00:00
|
|
|
|
bool use_external = this->use_external_mp || this->use_external_mp_once;
|
|
|
|
|
Point scaled_origin = use_external ? Point(0, 0) : Point::new_scale(gcodegen.origin().x, gcodegen.origin().y);
|
|
|
|
|
Polyline result = (use_external ? m_external_mp.get() : m_layer_mp.get())->
|
|
|
|
|
shortest_path(gcodegen.last_pos() + scaled_origin, point + scaled_origin);
|
|
|
|
|
if (! use_external)
|
|
|
|
|
result.translate(scaled_origin.negative());
|
|
|
|
|
return result;
|
2015-07-01 18:14:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
std::string OozePrevention::pre_toolchange(GCode &gcodegen)
|
2015-07-02 13:02:20 +00:00
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
|
|
|
|
|
|
|
|
|
// move to the nearest standby point
|
|
|
|
|
if (!this->standby_points.empty()) {
|
|
|
|
|
// get current position in print coordinates
|
2017-05-03 16:28:22 +00:00
|
|
|
|
Pointf3 writer_pos = gcodegen.writer().get_position();
|
2015-07-02 13:02:20 +00:00
|
|
|
|
Point pos = Point::new_scale(writer_pos.x, writer_pos.y);
|
|
|
|
|
|
|
|
|
|
// find standby point
|
|
|
|
|
Point standby_point;
|
|
|
|
|
pos.nearest_point(this->standby_points, &standby_point);
|
|
|
|
|
|
|
|
|
|
/* We don't call gcodegen.travel_to() because we don't need retraction (it was already
|
|
|
|
|
triggered by the caller) nor avoid_crossing_perimeters and also because the coordinates
|
|
|
|
|
of the destination point must not be transformed by origin nor current extruder offset. */
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += gcodegen.writer().travel_to_xy(Pointf::new_unscale(standby_point),
|
2015-07-02 13:02:20 +00:00
|
|
|
|
"move to standby position");
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (gcodegen.config().standby_temperature_delta.value != 0) {
|
2015-07-02 13:02:20 +00:00
|
|
|
|
// we assume that heating is always slower than cooling, so no need to block
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += gcodegen.writer().set_temperature
|
|
|
|
|
(this->_get_temp(gcodegen) + gcodegen.config().standby_temperature_delta.value, false);
|
2015-07-02 13:02:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-26 12:24:31 +00:00
|
|
|
|
std::string OozePrevention::post_toolchange(GCode &gcodegen)
|
2015-07-02 13:02:20 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
return (gcodegen.config().standby_temperature_delta.value != 0) ?
|
|
|
|
|
gcodegen.writer().set_temperature(this->_get_temp(gcodegen), true) :
|
2017-04-26 12:24:31 +00:00
|
|
|
|
std::string();
|
2015-07-02 13:02:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
OozePrevention::_get_temp(GCode &gcodegen)
|
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
return (gcodegen.layer() != NULL && gcodegen.layer()->id() == 0)
|
|
|
|
|
? gcodegen.config().first_layer_temperature.get_at(gcodegen.writer().extruder()->id)
|
|
|
|
|
: gcodegen.config().temperature.get_at(gcodegen.writer().extruder()->id);
|
2015-07-02 13:02:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-07-01 21:00:52 +00:00
|
|
|
|
std::string
|
|
|
|
|
Wipe::wipe(GCode &gcodegen, bool toolchange)
|
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
|
|
|
|
|
|
|
|
|
/* Reduce feedrate a bit; travel speed is often too high to move on existing material.
|
|
|
|
|
Too fast = ripping of existing material; too slow = short wipe path, thus more blob. */
|
2017-05-03 16:28:22 +00:00
|
|
|
|
double wipe_speed = gcodegen.writer().config.travel_speed.value * 0.8;
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
// get the retraction length
|
|
|
|
|
double length = toolchange
|
2017-05-03 16:28:22 +00:00
|
|
|
|
? gcodegen.writer().extruder()->retract_length_toolchange()
|
|
|
|
|
: gcodegen.writer().extruder()->retract_length();
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
if (length > 0) {
|
|
|
|
|
/* Calculate how long we need to travel in order to consume the required
|
|
|
|
|
amount of retraction. In other words, how far do we move in XY at wipe_speed
|
|
|
|
|
for the time needed to consume retract_length at retract_speed? */
|
2017-05-03 16:28:22 +00:00
|
|
|
|
double wipe_dist = scale_(length / gcodegen.writer().extruder()->retract_speed() * wipe_speed);
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
/* Take the stored wipe path and replace first point with the current actual position
|
|
|
|
|
(they might be different, for example, in case of loop clipping). */
|
|
|
|
|
Polyline wipe_path;
|
|
|
|
|
wipe_path.append(gcodegen.last_pos());
|
|
|
|
|
wipe_path.append(
|
|
|
|
|
this->path.points.begin() + 1,
|
|
|
|
|
this->path.points.end()
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
wipe_path.clip_end(wipe_path.length() - wipe_dist);
|
|
|
|
|
|
|
|
|
|
// subdivide the retraction in segments
|
|
|
|
|
double retracted = 0;
|
|
|
|
|
Lines lines = wipe_path.lines();
|
|
|
|
|
for (Lines::const_iterator line = lines.begin(); line != lines.end(); ++line) {
|
|
|
|
|
double segment_length = line->length();
|
|
|
|
|
/* Reduce retraction length a bit to avoid effective retraction speed to be greater than the configured one
|
|
|
|
|
due to rounding (TODO: test and/or better math for this) */
|
|
|
|
|
double dE = length * (segment_length / wipe_dist) * 0.95;
|
2017-02-15 16:51:46 +00:00
|
|
|
|
//FIXME one shall not generate the unnecessary G1 Fxxx commands, here wipe_speed is a constant inside this cycle.
|
|
|
|
|
// Is it here for the cooling markers? Or should it be outside of the cycle?
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += gcodegen.writer().set_speed(wipe_speed*60, "", gcodegen.enable_cooling_markers() ? ";_WIPE" : "");
|
|
|
|
|
gcode += gcodegen.writer().extrude_to_xy(
|
2015-07-01 21:00:52 +00:00
|
|
|
|
gcodegen.point_to_gcode(line->b),
|
|
|
|
|
-dE,
|
2016-04-10 08:07:32 +00:00
|
|
|
|
"wipe and retract"
|
2015-07-01 21:00:52 +00:00
|
|
|
|
);
|
|
|
|
|
retracted += dE;
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcodegen.writer().extruder()->retracted += retracted;
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
// prevent wiping again on same path
|
|
|
|
|
this->reset_path();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
#define EXTRUDER_CONFIG(OPT) m_config.OPT.get_at(m_writer.extruder()->id)
|
2015-07-02 12:31:21 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
inline void write(FILE *file, const std::string &what)
|
2016-09-12 14:25:15 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
fwrite(what.data(), 1, what.size(), file);
|
2016-09-12 14:25:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
inline void writeln(FILE *file, const std::string &what)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
write(file, what);
|
|
|
|
|
fprintf(file, "\n");
|
2015-07-01 21:00:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
bool GCode::do_export(FILE *file, Print &print)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// How many times will be change_layer() called?
|
|
|
|
|
// change_layer() in turn increments the progress bar status.
|
|
|
|
|
m_layer_count = 0;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (print.config.complete_objects.value) {
|
|
|
|
|
// Add each of the object's layers separately.
|
|
|
|
|
for (auto object : print.objects) {
|
|
|
|
|
std::vector<coordf_t> zs;
|
|
|
|
|
zs.reserve(object->layers.size() + object->support_layers.size());
|
|
|
|
|
for (auto layer : object->layers)
|
|
|
|
|
zs.push_back(layer->print_z);
|
|
|
|
|
for (auto layer : object->support_layers)
|
|
|
|
|
zs.push_back(layer->print_z);
|
|
|
|
|
std::sort(zs.begin(), zs.end());
|
|
|
|
|
m_layer_count += (unsigned int)(object->copies().size() * (std::unique(zs.begin(), zs.end()) - zs.begin()));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Print all objects with the same print_z together.
|
|
|
|
|
std::vector<coordf_t> zs;
|
|
|
|
|
for (auto object : print.objects) {
|
|
|
|
|
zs.reserve(zs.size() + object->layers.size() + object->support_layers.size());
|
|
|
|
|
for (auto layer : object->layers)
|
|
|
|
|
zs.push_back(layer->print_z);
|
|
|
|
|
for (auto layer : object->support_layers)
|
|
|
|
|
zs.push_back(layer->print_z);
|
|
|
|
|
}
|
|
|
|
|
std::sort(zs.begin(), zs.end());
|
|
|
|
|
m_layer_count = (unsigned int)(std::unique(zs.begin(), zs.end()) - zs.begin());
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
|
|
|
|
m_enable_cooling_markers = true;
|
|
|
|
|
this->apply_print_config(print.config);
|
|
|
|
|
this->set_extruders(print.extruders());
|
|
|
|
|
|
|
|
|
|
// Initialize autospeed.
|
|
|
|
|
{
|
|
|
|
|
// get the minimum cross-section used in the print
|
|
|
|
|
std::vector<double> mm3_per_mm;
|
|
|
|
|
for (auto object : print.objects) {
|
|
|
|
|
for (size_t region_id = 0; region_id < print.regions.size(); ++ region_id) {
|
|
|
|
|
auto region = print.regions[region_id];
|
|
|
|
|
for (auto layer : object->layers) {
|
|
|
|
|
auto layerm = layer->regions[region_id];
|
|
|
|
|
if (region->config.get_abs_value("perimeter_speed" ) == 0 ||
|
|
|
|
|
region->config.get_abs_value("small_perimeter_speed" ) == 0 ||
|
|
|
|
|
region->config.get_abs_value("external_perimeter_speed" ) == 0 ||
|
|
|
|
|
region->config.get_abs_value("bridge_speed" ) == 0)
|
|
|
|
|
mm3_per_mm.push_back(layerm->perimeters.min_mm3_per_mm());
|
|
|
|
|
if (region->config.get_abs_value("infill_speed" ) == 0 ||
|
|
|
|
|
region->config.get_abs_value("solid_infill_speed" ) == 0 ||
|
|
|
|
|
region->config.get_abs_value("top_solid_infill_speed" ) == 0 ||
|
|
|
|
|
region->config.get_abs_value("bridge_speed" ) == 0)
|
|
|
|
|
mm3_per_mm.push_back(layerm->fills.min_mm3_per_mm());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (object->config.get_abs_value("support_material_speed" ) == 0 ||
|
|
|
|
|
object->config.get_abs_value("support_material_interface_speed" ) == 0)
|
|
|
|
|
for (auto layer : object->support_layers)
|
|
|
|
|
mm3_per_mm.push_back(layer->support_fills.min_mm3_per_mm());
|
|
|
|
|
}
|
|
|
|
|
// filter out 0-width segments
|
|
|
|
|
mm3_per_mm.erase(std::remove_if(mm3_per_mm.begin(), mm3_per_mm.end(), [](double v) { return v < 0.000001; }), mm3_per_mm.end());
|
|
|
|
|
if (! mm3_per_mm.empty()) {
|
|
|
|
|
// In order to honor max_print_speed we need to find a target volumetric
|
|
|
|
|
// speed that we can use throughout the print. So we define this target
|
|
|
|
|
// volumetric speed as the volumetric speed produced by printing the
|
|
|
|
|
// smallest cross-section at the maximum speed: any larger cross-section
|
|
|
|
|
// will need slower feedrates.
|
|
|
|
|
m_volumetric_speed = *std::min_element(mm3_per_mm.begin(), mm3_per_mm.end()) * print.config.max_print_speed.value;
|
|
|
|
|
// limit such volumetric speed with max_volumetric_speed if set
|
|
|
|
|
if (print.config.max_volumetric_speed.value > 0)
|
|
|
|
|
m_volumetric_speed = std::min(m_volumetric_speed, print.config.max_volumetric_speed.value);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:57:33 +00:00
|
|
|
|
m_cooling_buffer = make_unique<CoolingBuffer>(*this);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (print.config.spiral_vase.value)
|
2017-05-03 16:57:33 +00:00
|
|
|
|
m_spiral_vase = make_unique<SpiralVase>(print.config);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (print.config.max_volumetric_extrusion_rate_slope_positive.value > 0 ||
|
|
|
|
|
print.config.max_volumetric_extrusion_rate_slope_negative.value > 0)
|
2017-05-03 16:57:33 +00:00
|
|
|
|
m_pressure_equalizer = make_unique<PressureEqualizer>(&print.config);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_enable_extrusion_role_markers = (bool)m_pressure_equalizer;
|
|
|
|
|
|
|
|
|
|
// Write information on the generator.
|
|
|
|
|
{
|
|
|
|
|
const auto now = boost::posix_time::second_clock::local_time();
|
|
|
|
|
const auto date = now.date();
|
|
|
|
|
fprintf(file, "; generated by Slic3r %s on %04d-%02d-%02d at %02d:%02d:%02d\n\n",
|
|
|
|
|
SLIC3R_VERSION,
|
|
|
|
|
// Local date in an ANSII format.
|
|
|
|
|
int(now.date().year()), int(now.date().month()), int(now.date().day()),
|
|
|
|
|
int(now.time_of_day().hours()), int(now.time_of_day().minutes()), int(now.time_of_day().seconds()));
|
|
|
|
|
}
|
|
|
|
|
// Write notes (content of the Print Settings tab -> Notes)
|
|
|
|
|
{
|
|
|
|
|
std::list<std::string> lines;
|
|
|
|
|
boost::split(lines, print.config.notes.value, boost::is_any_of("\n"), boost::token_compress_off);
|
|
|
|
|
for (auto line : lines) {
|
|
|
|
|
// Remove the trailing '\r' from the '\r\n' sequence.
|
|
|
|
|
if (! line.empty() && line.back() == '\r')
|
|
|
|
|
line.pop_back();
|
|
|
|
|
fprintf(file, "; %s\n", line.c_str());
|
|
|
|
|
}
|
|
|
|
|
if (! lines.empty())
|
|
|
|
|
fprintf(file, "\n");
|
|
|
|
|
}
|
|
|
|
|
// Write some terse information on the slicing parameters.
|
|
|
|
|
{
|
|
|
|
|
const PrintObject *first_object = print.objects.front();
|
|
|
|
|
const double layer_height = first_object->config.layer_height.value;
|
|
|
|
|
for (size_t region_id = 0; region_id < print.regions.size(); ++ region_id) {
|
|
|
|
|
auto region = print.regions[region_id];
|
|
|
|
|
fprintf(file, "; external perimeters extrusion width = %.2fmm\n", region->flow(frExternalPerimeter, layer_height, false, false, -1., *first_object).width);
|
|
|
|
|
fprintf(file, "; perimeters extrusion width = %.2fmm\n", region->flow(frPerimeter, layer_height, false, false, -1., *first_object).width);
|
|
|
|
|
fprintf(file, "; infill extrusion width = %.2fmm\n", region->flow(frInfill, layer_height, false, false, -1., *first_object).width);
|
|
|
|
|
fprintf(file, "; solid infill extrusion width = %.2fmm\n", region->flow(frSolidInfill, layer_height, false, false, -1., *first_object).width);
|
|
|
|
|
fprintf(file, "; top infill extrusion width = %.2fmm\n", region->flow(frTopSolidInfill, layer_height, false, false, -1., *first_object).width);
|
|
|
|
|
if (print.has_support_material())
|
|
|
|
|
fprintf(file, "; support material extrusion width = %.2fmm\n", support_material_flow(first_object).width);
|
|
|
|
|
if (print.config.first_layer_extrusion_width.value > 0)
|
|
|
|
|
fprintf(file, "; first layer extrusion width = %.2fmm\n", region->flow(frPerimeter, layer_height, false, true, -1., *first_object).width);
|
|
|
|
|
fprintf(file, "\n");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Prepare the helper object for replacing placeholders in custom G-code and output filename.
|
|
|
|
|
m_placeholder_parser = print.placeholder_parser;
|
|
|
|
|
m_placeholder_parser.update_timestamp();
|
|
|
|
|
|
|
|
|
|
// Disable fan.
|
|
|
|
|
if (print.config.cooling.value && print.config.disable_fan_first_layers.value)
|
|
|
|
|
write(file, m_writer.set_fan(0, true));
|
|
|
|
|
|
|
|
|
|
// Set bed temperature if the start G-code does not contain any bed temp control G-codes.
|
|
|
|
|
if (print.config.first_layer_bed_temperature.value > 0 &&
|
|
|
|
|
boost::ifind_first(print.config.start_gcode.value, std::string("M140")).empty() &&
|
|
|
|
|
boost::ifind_first(print.config.start_gcode.value, std::string("M190")).empty())
|
|
|
|
|
write(file, m_writer.set_bed_temperature(print.config.first_layer_bed_temperature.value, true));
|
|
|
|
|
|
|
|
|
|
// Set extruder(s) temperature before and after start G-code.
|
|
|
|
|
this->_print_first_layer_extruder_temperatures(file, print, false);
|
|
|
|
|
fprintf(file, "%s\n", m_placeholder_parser.process(print.config.start_gcode.value).c_str());
|
|
|
|
|
this->_print_first_layer_extruder_temperatures(file, print, true);
|
|
|
|
|
|
|
|
|
|
// Set other general things.
|
|
|
|
|
write(file, this->preamble());
|
|
|
|
|
|
|
|
|
|
// Initialize a motion planner for object-to-object travel moves.
|
|
|
|
|
if (print.config.avoid_crossing_perimeters.value) {
|
|
|
|
|
//coord_t distance_from_objects = coord_t(scale_(1.));
|
|
|
|
|
// Compute the offsetted convex hull for each object and repeat it for each copy.
|
|
|
|
|
Polygons islands_p;
|
|
|
|
|
for (const PrintObject *object : print.objects) {
|
|
|
|
|
// Discard objects only containing thin walls (offset would fail on an empty polygon).
|
|
|
|
|
Polygons polygons;
|
|
|
|
|
for (const Layer *layer : object->layers)
|
|
|
|
|
for (const ExPolygon &expoly : layer->slices.expolygons)
|
|
|
|
|
polygons.push_back(expoly.contour);
|
|
|
|
|
if (! polygons.empty()) {
|
|
|
|
|
// Translate convex hull for each object copy and append it to the islands array.
|
|
|
|
|
for (const Point © : object->_shifted_copies)
|
|
|
|
|
for (Polygon poly : polygons) {
|
|
|
|
|
poly.translate(copy);
|
|
|
|
|
islands_p.emplace_back(std::move(poly));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
m_avoid_crossing_perimeters.init_external_mp(union_ex(islands_p));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Calculate wiping points if needed
|
|
|
|
|
if (print.config.ooze_prevention.value) {
|
|
|
|
|
Points skirt_points;
|
|
|
|
|
for (const ExtrusionEntity *ee : print.skirt.entities)
|
|
|
|
|
for (const ExtrusionPath &path : dynamic_cast<const ExtrusionLoop*>(ee)->paths)
|
|
|
|
|
append(skirt_points, path.polyline.points);
|
|
|
|
|
if (! skirt_points.empty()) {
|
|
|
|
|
Polygon outer_skirt = Slic3r::Geometry::convex_hull(skirt_points);
|
|
|
|
|
Polygons skirts;
|
|
|
|
|
for (unsigned int extruder_id : print.extruders()) {
|
|
|
|
|
const Pointf &extruder_offset = print.config.extruder_offset.get_at(extruder_id);
|
|
|
|
|
Polygon s(outer_skirt);
|
|
|
|
|
s.translate(-scale_(extruder_offset.x), -scale_(extruder_offset.y));
|
|
|
|
|
skirts.emplace_back(std::move(s));
|
|
|
|
|
}
|
|
|
|
|
m_ooze_prevention.enable = true;
|
|
|
|
|
m_ooze_prevention.standby_points =
|
|
|
|
|
offset(Slic3r::Geometry::convex_hull(skirts), scale_(3.f)).front().equally_spaced_points(scale_(10.));
|
|
|
|
|
#if 0
|
|
|
|
|
require "Slic3r/SVG.pm";
|
|
|
|
|
Slic3r::SVG::output(
|
|
|
|
|
"ooze_prevention.svg",
|
|
|
|
|
red_polygons => \@skirts,
|
|
|
|
|
polygons => [$outer_skirt],
|
|
|
|
|
points => $gcodegen->ooze_prevention->standby_points,
|
|
|
|
|
);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set initial extruder only after custom start G-code.
|
|
|
|
|
write(file, this->set_extruder(print.extruders().front()));
|
|
|
|
|
|
|
|
|
|
// Do all objects for each layer.
|
|
|
|
|
if (print.config.complete_objects.value) {
|
|
|
|
|
// Print objects from the smallest to the tallest to avoid collisions
|
|
|
|
|
// when moving onto next object starting point.
|
|
|
|
|
std::vector<PrintObject*> objects(print.objects);
|
|
|
|
|
std::sort(objects.begin(), objects.end(), [](const PrintObject* po1, const PrintObject* po2) { return po1->size.z < po2->size.z; });
|
|
|
|
|
size_t finished_objects = 0;
|
|
|
|
|
for (PrintObject *object : objects) {
|
|
|
|
|
for (const Point © : object->_shifted_copies) {
|
2017-05-10 09:25:57 +00:00
|
|
|
|
this->set_origin(unscale(copy.x), unscale(copy.y));
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (finished_objects > 0) {
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Move to the origin position for the copy we're going to print.
|
|
|
|
|
// This happens before Z goes down to layer 0 again, so that no collision happens hopefully.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_enable_cooling_markers = false; // we're not filtering these moves through CoolingBuffer
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp_once = true;
|
|
|
|
|
write(file, this->retract());
|
|
|
|
|
write(file, this->travel_to(Point(0, 0), erNone, "move to origin position for next object"));
|
|
|
|
|
m_enable_cooling_markers = true;
|
|
|
|
|
// Disable motion planner when traveling to first object point.
|
|
|
|
|
m_avoid_crossing_perimeters.disable_once = true;
|
|
|
|
|
// Ff we are printing the bottom layer of an object, and we have already finished
|
|
|
|
|
// another one, set first layer temperatures. This happens before the Z move
|
|
|
|
|
// is triggered, so machine has more time to reach such temperatures.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (print.config.first_layer_bed_temperature.value > 0)
|
|
|
|
|
write(file, m_writer.set_bed_temperature(print.config.first_layer_bed_temperature));
|
|
|
|
|
// Set first layer extruder.
|
|
|
|
|
this->_print_first_layer_extruder_temperatures(file, print, false);
|
|
|
|
|
}
|
2017-05-16 11:45:28 +00:00
|
|
|
|
// Get optimal tool ordering to minimize tool switches of a multi-exruder print.
|
|
|
|
|
std::vector<ToolOrdering::LayerTools> tool_ordering = ToolOrdering::tool_ordering(*object);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Pair the object layers with the support layers by z, extrude them.
|
|
|
|
|
size_t idx_object_layer = 0;
|
|
|
|
|
size_t idx_support_layer = 0;
|
|
|
|
|
std::vector<LayerToPrint> layers_to_print(1, LayerToPrint());
|
|
|
|
|
LayerToPrint &layer_to_print = layers_to_print.front();
|
|
|
|
|
while (idx_object_layer < object->layers.size() || idx_support_layer < object->support_layers.size()) {
|
|
|
|
|
layer_to_print.object_layer = (idx_object_layer < object->layers.size()) ? object->layers[idx_object_layer ++] : nullptr;
|
|
|
|
|
layer_to_print.support_layer = (idx_support_layer < object->support_layers.size()) ? object->support_layers[idx_support_layer ++] : nullptr;
|
|
|
|
|
if (layer_to_print.object_layer && layer_to_print.support_layer) {
|
|
|
|
|
if (layer_to_print.object_layer->print_z < layer_to_print.support_layer->print_z) {
|
|
|
|
|
layer_to_print.support_layer = nullptr;
|
|
|
|
|
-- idx_support_layer;
|
|
|
|
|
} else if (layer_to_print.support_layer->print_z < layer_to_print.object_layer->print_z) {
|
|
|
|
|
layer_to_print.object_layer = nullptr;
|
|
|
|
|
-- idx_object_layer;
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-16 11:45:28 +00:00
|
|
|
|
auto it_layer_tools = std::lower_bound(tool_ordering.begin(), tool_ordering.end(), ToolOrdering::LayerTools(layer_to_print.layer()->print_z));
|
|
|
|
|
assert(it_layer_tools != tool_ordering.end() && it_layer_tools->print_z == layer_to_print.layer()->print_z);
|
|
|
|
|
this->process_layer(file, print, layers_to_print, *it_layer_tools, © - object->_shifted_copies.data());
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
|
|
|
|
write(file, this->filter(m_cooling_buffer->flush(), true));
|
|
|
|
|
++ finished_objects;
|
|
|
|
|
// Flag indicating whether the nozzle temperature changes from 1st to 2nd layer were performed.
|
|
|
|
|
// Reset it when starting another object from 1st layer.
|
|
|
|
|
m_second_layer_things_done = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Order objects using a nearest neighbor search.
|
|
|
|
|
std::vector<size_t> object_indices;
|
|
|
|
|
Points object_reference_points;
|
|
|
|
|
for (PrintObject *object : print.objects)
|
|
|
|
|
object_reference_points.push_back(object->_shifted_copies.front());
|
2017-05-10 09:25:57 +00:00
|
|
|
|
Slic3r::Geometry::chained_path(object_reference_points, object_indices);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// Sort layers by Z.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// All extrusion moves with the same top layer height are extruded uninterrupted.
|
|
|
|
|
std::map<coordf_t, std::vector<LayerToPrint>> layers;
|
|
|
|
|
size_t object_order = 0;
|
|
|
|
|
for (size_t obj_idx : object_indices) {
|
|
|
|
|
PrintObject *print_object = print.objects[obj_idx];
|
|
|
|
|
for (Layer *layer : print_object->layers) {
|
|
|
|
|
std::vector<LayerToPrint> &object_layers_at_printz = layers[layer->print_z];
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (object_layers_at_printz.empty())
|
2017-05-10 09:25:57 +00:00
|
|
|
|
object_layers_at_printz.resize(print.objects.size(), LayerToPrint());
|
|
|
|
|
object_layers_at_printz[object_order].object_layer = layer;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
for (SupportLayer *layer : print_object->support_layers) {
|
|
|
|
|
std::vector<LayerToPrint> &object_layers_at_printz = layers[layer->print_z];
|
|
|
|
|
if (object_layers_at_printz.empty())
|
|
|
|
|
object_layers_at_printz.resize(print.objects.size(), LayerToPrint());
|
|
|
|
|
object_layers_at_printz[object_order].support_layer = layer;
|
|
|
|
|
}
|
|
|
|
|
++ object_order;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-16 11:45:28 +00:00
|
|
|
|
// Get optimal tool ordering to minimize tool switches of a multi-exruder print.
|
|
|
|
|
std::vector<ToolOrdering::LayerTools> tool_ordering = ToolOrdering::tool_ordering(print);
|
|
|
|
|
// Prusa Multi-Material wipe tower.
|
|
|
|
|
if (print.config.single_extruder_multi_material.value && print.config.wipe_tower.value &&
|
|
|
|
|
! tool_ordering.empty() && tool_ordering.front().wipe_tower_partitions > 0) {
|
|
|
|
|
// Initialize the wipe tower.
|
|
|
|
|
auto *wipe_tower = new WipeTowerPrusaMM(
|
|
|
|
|
float(print.config.wipe_tower_x.value), float(print.config.wipe_tower_y.value),
|
|
|
|
|
float(print.config.wipe_tower_width.value), float(print.config.wipe_tower_per_color_wipe.value));
|
|
|
|
|
//wipe_tower->set_retract();
|
|
|
|
|
//wipe_tower->set_zhop();
|
|
|
|
|
//wipe_tower->set_zhop();
|
|
|
|
|
// Set the extruder & material properties at the wipe tower object.
|
|
|
|
|
for (size_t i = 0; i < 4; ++ i)
|
|
|
|
|
wipe_tower->set_extruder(
|
|
|
|
|
i,
|
|
|
|
|
WipeTowerPrusaMM::parse_material(print.config.filament_type.get_at(i).c_str()),
|
|
|
|
|
print.config.temperature.get_at(i),
|
|
|
|
|
print.config.first_layer_temperature.get_at(i));
|
|
|
|
|
m_wipe_tower.reset(wipe_tower);
|
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Extrude the layers.
|
2017-05-16 11:45:28 +00:00
|
|
|
|
for (auto &layer : layers) {
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// layer.second is of type std::vector<LayerToPrint>,
|
|
|
|
|
// wher the objects are sorted by their sorted order given by object_indices.
|
2017-05-16 11:45:28 +00:00
|
|
|
|
auto it_layer_tools = std::lower_bound(tool_ordering.begin(), tool_ordering.end(), ToolOrdering::LayerTools(layer.first));
|
|
|
|
|
assert(it_layer_tools != tool_ordering.end() && layer.first);
|
|
|
|
|
if (m_wipe_tower) {
|
|
|
|
|
bool first_layer = layer.first == layers.begin()->first;
|
|
|
|
|
auto it_layer_tools_next = it_layer_tools;
|
|
|
|
|
++ it_layer_tools_next;
|
|
|
|
|
m_wipe_tower->set_layer(
|
|
|
|
|
layer.first,
|
|
|
|
|
first_layer ?
|
|
|
|
|
print.objects.front()->config.first_layer_height.get_abs_value(print.objects.front()->config.layer_height.value) :
|
|
|
|
|
print.objects.front()->config.layer_height.value,
|
|
|
|
|
it_layer_tools->wipe_tower_partitions,
|
|
|
|
|
first_layer,
|
|
|
|
|
it_layer_tools->wipe_tower_partitions == 0 || (it_layer_tools_next == tool_ordering.end() || it_layer_tools_next->wipe_tower_partitions == 0));
|
|
|
|
|
}
|
|
|
|
|
this->process_layer(file, print, layer.second, *it_layer_tools, size_t(-1));
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
write(file, this->filter(m_cooling_buffer->flush(), true));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// write end commands to file
|
2017-05-16 11:45:28 +00:00
|
|
|
|
if (m_wipe_tower) {
|
|
|
|
|
// Unload the current filament over the purge tower.
|
|
|
|
|
write(file, this->wipe_tower_tool_change(-1));
|
|
|
|
|
m_wipe_tower.release();
|
|
|
|
|
} else
|
|
|
|
|
write(file, this->retract()); // TODO: process this retract through PressureRegulator in order to discharge fully
|
2017-05-03 16:28:22 +00:00
|
|
|
|
write(file, m_writer.set_fan(false));
|
|
|
|
|
writeln(file, m_placeholder_parser.process(print.config.end_gcode));
|
|
|
|
|
write(file, m_writer.update_progress(m_layer_count, m_layer_count, true)); // 100%
|
|
|
|
|
write(file, m_writer.postamble());
|
2017-05-16 11:45:28 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// get filament stats
|
|
|
|
|
print.filament_stats.clear();
|
|
|
|
|
print.total_used_filament = 0.;
|
|
|
|
|
print.total_extruded_volume = 0.;
|
|
|
|
|
print.total_weight = 0.;
|
|
|
|
|
print.total_cost = 0.;
|
|
|
|
|
for (const Extruder &extruder : m_writer.extruders) {
|
|
|
|
|
double used_filament = extruder.used_filament();
|
|
|
|
|
double extruded_volume = extruder.extruded_volume();
|
|
|
|
|
double filament_weight = extruded_volume * extruder.filament_density() * 0.001;
|
|
|
|
|
double filament_cost = filament_weight * extruder.filament_cost() * 0.001;
|
|
|
|
|
print.filament_stats.insert(std::pair<size_t,float>(extruder.id, used_filament));
|
|
|
|
|
fprintf(file, "; filament used = %.1lfmm (%.1lfcm3)\n", used_filament, extruded_volume * 0.001);
|
|
|
|
|
if (filament_weight > 0.) {
|
|
|
|
|
print.total_weight = print.total_weight + filament_weight;
|
|
|
|
|
fprintf(file, "; filament used = %.1lf\n", filament_weight);
|
|
|
|
|
if (filament_cost > 0.) {
|
|
|
|
|
print.total_cost = print.total_cost + filament_cost;
|
|
|
|
|
fprintf(file, "; filament cost = %.1lf\n", filament_cost);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
print.total_used_filament = print.total_used_filament + used_filament;
|
|
|
|
|
print.total_extruded_volume = print.total_extruded_volume + extruded_volume;
|
|
|
|
|
}
|
|
|
|
|
fprintf(file, "; total filament cost = %.1lf\n", print.total_cost);
|
|
|
|
|
|
|
|
|
|
// Append full config.
|
|
|
|
|
fprintf(file, "\n");
|
|
|
|
|
for (const std::string &key : print.config.keys())
|
|
|
|
|
fprintf(file, "; %s = %s\n", key.c_str(), print.config.serialize(key).c_str());
|
|
|
|
|
for (const std::string &key : print.default_object_config.keys())
|
|
|
|
|
fprintf(file, "; %s = %s\n", key.c_str(), print.default_object_config.serialize(key).c_str());
|
|
|
|
|
|
|
|
|
|
return true;
|
2015-07-01 21:00:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// Write 1st layer extruder temperatures into the G-code.
|
|
|
|
|
// Only do that if the start G-code does not already contain any M-code controlling an extruder temperature.
|
|
|
|
|
// FIXME this does not work correctly for multi-extruder, single heater configuration as it emits multiple preheat commands for the same heater.
|
|
|
|
|
// M104 - Set Extruder Temperature
|
|
|
|
|
// M109 - Set Extruder Temperature and Wait
|
|
|
|
|
void GCode::_print_first_layer_extruder_temperatures(FILE *file, Print &print, bool wait)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (boost::ifind_first(print.config.start_gcode.value, std::string("M104")).empty() &&
|
|
|
|
|
boost::ifind_first(print.config.start_gcode.value, std::string("M109")).empty()) {
|
|
|
|
|
for (unsigned int tool_id : print.extruders()) {
|
|
|
|
|
int temp = print.config.first_layer_temperature.get_at(tool_id);
|
|
|
|
|
if (print.config.ooze_prevention.value)
|
|
|
|
|
temp += print.config.standby_temperature_delta.value;
|
|
|
|
|
if (temp > 0)
|
|
|
|
|
write(file, m_writer.set_temperature(temp, wait, tool_id));
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-07-01 21:00:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
inline GCode::ObjectByExtruder& object_by_extruder(
|
|
|
|
|
std::map<unsigned int, std::vector<GCode::ObjectByExtruder>> &by_extruder,
|
|
|
|
|
unsigned int extruder_id,
|
|
|
|
|
size_t object_idx,
|
|
|
|
|
size_t num_objects)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::vector<GCode::ObjectByExtruder> &objects_by_extruder = by_extruder[extruder_id];
|
|
|
|
|
if (objects_by_extruder.empty())
|
|
|
|
|
objects_by_extruder.assign(num_objects, GCode::ObjectByExtruder());
|
|
|
|
|
return objects_by_extruder[object_idx];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline std::vector<GCode::ObjectByExtruder::Island>& object_islands_by_extruder(
|
|
|
|
|
std::map<unsigned int, std::vector<GCode::ObjectByExtruder>> &by_extruder,
|
|
|
|
|
unsigned int extruder_id,
|
|
|
|
|
size_t object_idx,
|
|
|
|
|
size_t num_objects,
|
|
|
|
|
size_t num_islands)
|
|
|
|
|
{
|
|
|
|
|
std::vector<GCode::ObjectByExtruder::Island> &islands = object_by_extruder(by_extruder, extruder_id, object_idx, num_objects).islands;
|
|
|
|
|
if (islands.empty())
|
|
|
|
|
islands.assign(num_islands, GCode::ObjectByExtruder::Island());
|
|
|
|
|
return islands;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// In sequential mode, process_layer is called once per each object and its copy,
|
|
|
|
|
// therefore layers will contain a single entry and single_object_idx will point to the copy of the object.
|
|
|
|
|
// In non-sequential mode, process_layer is called per each print_z height with all object and support layers accumulated.
|
|
|
|
|
// For multi-material prints, this routine minimizes extruder switches by gathering extruder specific extrusion paths
|
|
|
|
|
// and performing the extruder specific extrusions together.
|
|
|
|
|
void GCode::process_layer(
|
|
|
|
|
// Write into the output file.
|
|
|
|
|
FILE *file,
|
|
|
|
|
const Print &print,
|
|
|
|
|
// Set of object & print layers of the same PrintObject and with the same print_z.
|
|
|
|
|
const std::vector<LayerToPrint> &layers,
|
2017-05-16 11:45:28 +00:00
|
|
|
|
const ToolOrdering::LayerTools &layer_tools,
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// If set to size_t(-1), then print all copies of all objects.
|
|
|
|
|
// Otherwise print a single copy of a single object.
|
|
|
|
|
const size_t single_object_idx)
|
|
|
|
|
{
|
|
|
|
|
assert(! layers.empty());
|
|
|
|
|
// Either printing all copies of all objects, or just a single copy of a single object.
|
|
|
|
|
assert(single_object_idx == size_t(-1) || layers.size() == 1);
|
|
|
|
|
|
|
|
|
|
// Extract 1st object_layer and support_layer of this set of layers with an equal print_z.
|
|
|
|
|
const Layer *object_layer = nullptr;
|
|
|
|
|
const SupportLayer *support_layer = nullptr;
|
|
|
|
|
for (const LayerToPrint &l : layers) {
|
|
|
|
|
if (l.object_layer != nullptr && object_layer == nullptr)
|
|
|
|
|
object_layer = l.object_layer;
|
|
|
|
|
if (l.support_layer != nullptr && support_layer == nullptr)
|
|
|
|
|
support_layer = l.support_layer;
|
|
|
|
|
}
|
|
|
|
|
const Layer &layer = (object_layer != nullptr) ? *object_layer : *support_layer;
|
|
|
|
|
coordf_t print_z = layer.print_z;
|
2017-05-16 11:45:28 +00:00
|
|
|
|
bool first_layer = layer.id() == 0;
|
|
|
|
|
unsigned int first_extruder_id = layer_tools.extruders.empty() ? 0 : layer_tools.extruders.front();
|
2017-05-10 09:25:57 +00:00
|
|
|
|
|
|
|
|
|
// Initialize config with the 1st object to be printed at this layer.
|
|
|
|
|
m_config.apply(layer.object()->config, true);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
|
|
|
|
// Check whether it is possible to apply the spiral vase logic for this layer.
|
2017-05-05 07:59:56 +00:00
|
|
|
|
// Just a reminder: A spiral vase mode is allowed for a single object, single material print only.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (m_spiral_vase && layers.size() == 1 && support_layer == nullptr) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
bool enable = (layer.id() > 0 || print.config.brim_width.value == 0.) && (layer.id() >= print.config.skirt_height.value && ! print.has_infinite_skirt());
|
|
|
|
|
if (enable) {
|
|
|
|
|
for (const LayerRegion *layer_region : layer.regions)
|
|
|
|
|
if (layer_region->region()->config.bottom_solid_layers.value > layer.id() ||
|
|
|
|
|
layer_region->perimeters.items_count() > 1 ||
|
|
|
|
|
layer_region->fills.items_count() > 0) {
|
|
|
|
|
enable = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
m_spiral_vase->enable = enable;
|
|
|
|
|
}
|
|
|
|
|
// If we're going to apply spiralvase to this layer, disable loop clipping
|
2017-05-10 09:25:57 +00:00
|
|
|
|
m_enable_loop_clipping = ! m_spiral_vase || ! m_spiral_vase->enable;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string gcode;
|
|
|
|
|
|
|
|
|
|
if (! first_layer && ! m_second_layer_things_done) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// Transition from 1st to 2nd layer. Adjust nozzle temperatures as prescribed by the nozzle dependent
|
|
|
|
|
// first_layer_temperature vs. temperature settings.
|
|
|
|
|
for (const Extruder &extruder : m_writer.extruders) {
|
|
|
|
|
int temperature = print.config.temperature.get_at(extruder.id);
|
|
|
|
|
if (temperature > 0 && temperature != print.config.first_layer_temperature.get_at(extruder.id))
|
|
|
|
|
gcode += m_writer.set_temperature(temperature, false, extruder.id);
|
|
|
|
|
}
|
|
|
|
|
if (print.config.bed_temperature.value > 0 && print.config.bed_temperature != print.config.first_layer_bed_temperature.value)
|
|
|
|
|
gcode += m_writer.set_bed_temperature(print.config.bed_temperature);
|
|
|
|
|
// Mark the temperature transition from 1st to 2nd layer to be finished.
|
|
|
|
|
m_second_layer_things_done = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set new layer - this will change Z and force a retraction if retract_layer_change is enabled.
|
|
|
|
|
if (! print.config.before_layer_gcode.value.empty()) {
|
|
|
|
|
PlaceholderParser pp(m_placeholder_parser);
|
|
|
|
|
pp.set("layer_num", m_layer_index + 1);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
pp.set("layer_z", print_z);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += pp.process(print.config.before_layer_gcode.value) + "\n";
|
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += this->change_layer(print_z); // this will increase m_layer_index
|
|
|
|
|
m_layer = &layer;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (! print.config.layer_gcode.value.empty()) {
|
|
|
|
|
PlaceholderParser pp(m_placeholder_parser);
|
|
|
|
|
pp.set("layer_num", m_layer_index);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
pp.set("layer_z", print_z);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += pp.process(print.config.layer_gcode.value) + "\n";
|
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
|
|
|
|
|
if (! m_brim_done)
|
|
|
|
|
// Switch the extruder to the extruder of the perimeters, so the perimeters extruder will be primed
|
|
|
|
|
// by the skirt before the brim is extruded with the same extruder.
|
2017-05-16 11:45:28 +00:00
|
|
|
|
gcode += this->set_extruder(layer_tools.extruders.front());
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
|
|
|
|
// Extrude skirt at the print_z of the raft layers and normal object layers
|
|
|
|
|
// not at the print_z of the interlaced support material layers.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
bool extrude_skirt =
|
|
|
|
|
! print.skirt.entities.empty() &&
|
|
|
|
|
// Not enough skirt layers printed yet.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
(m_skirt_done.size() < print.config.skirt_height.value || print.has_infinite_skirt()) &&
|
|
|
|
|
// This print_z has not been extruded yet
|
2017-05-10 09:25:57 +00:00
|
|
|
|
(m_skirt_done.empty() ? 0. : m_skirt_done.back()) < print_z - EPSILON &&
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// and this layer is the 1st layer, or it is an object layer, or it is a raft layer.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
(first_layer || object_layer != nullptr || support_layer->id() < m_config.raft_layers.value);
|
|
|
|
|
std::map<unsigned int, std::pair<size_t, size_t>> skirt_loops_per_extruder;
|
|
|
|
|
coordf_t skirt_height = 0.;
|
|
|
|
|
if (extrude_skirt) {
|
|
|
|
|
// Fill in skirt_loops_per_extruder.
|
|
|
|
|
skirt_height = print_z - (m_skirt_done.empty() ? 0. : m_skirt_done.back());
|
|
|
|
|
m_skirt_done.push_back(print_z);
|
|
|
|
|
if (first_layer) {
|
|
|
|
|
// Prime the extruders over the skirt lines.
|
|
|
|
|
std::vector<unsigned int> extruder_ids = m_writer.extruder_ids();
|
|
|
|
|
// Reorder the extruders, so that the last used extruder is at the front.
|
|
|
|
|
for (size_t i = 1; i < extruder_ids.size(); ++ i)
|
2017-05-16 11:45:28 +00:00
|
|
|
|
if (extruder_ids[i] == first_extruder_id) {
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Move the last extruder to the front.
|
2017-05-15 14:42:29 +00:00
|
|
|
|
memmove(extruder_ids.data() + 1, extruder_ids.data(), i * sizeof(unsigned int));
|
2017-05-16 11:45:28 +00:00
|
|
|
|
extruder_ids.front() = first_extruder_id;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
break;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
}
|
|
|
|
|
size_t n_loops = print.skirt.entities.size();
|
|
|
|
|
if (n_loops <= extruder_ids.size()) {
|
|
|
|
|
for (size_t i = 0; i < n_loops; ++i)
|
|
|
|
|
skirt_loops_per_extruder[extruder_ids[i]] = std::pair<size_t, size_t>(i, i + 1);
|
|
|
|
|
} else {
|
|
|
|
|
// Assign skirt loops to the extruders.
|
|
|
|
|
std::vector<unsigned int> extruder_loops(extruder_ids.size(), 1);
|
|
|
|
|
n_loops -= extruder_loops.size();
|
|
|
|
|
while (n_loops > 0) {
|
|
|
|
|
for (size_t i = 0; i < extruder_ids.size() && n_loops > 0; ++ i, -- n_loops)
|
|
|
|
|
++ extruder_loops[i];
|
|
|
|
|
}
|
|
|
|
|
for (size_t i = 0; i < extruder_ids.size(); ++ i)
|
|
|
|
|
skirt_loops_per_extruder[extruder_ids[i]] = std::make_pair<size_t, size_t>(
|
|
|
|
|
(i == 0) ? 0 : extruder_loops[i - 1],
|
|
|
|
|
((i == 0) ? 0 : extruder_loops[i - 1]) + extruder_loops[i]);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
} else
|
|
|
|
|
// Extrude all skirts with the current extruder.
|
2017-05-16 11:45:28 +00:00
|
|
|
|
skirt_loops_per_extruder[first_extruder_id] = std::pair<size_t, size_t>(0, print.config.skirts.value);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Group extrusions by an extruder, then by an object, an island and a region.
|
|
|
|
|
std::map<unsigned int, std::vector<ObjectByExtruder>> by_extruder;
|
|
|
|
|
|
|
|
|
|
for (const LayerToPrint &layer_to_print : layers) {
|
|
|
|
|
if (layer_to_print.support_layer != nullptr) {
|
|
|
|
|
const SupportLayer &support_layer = *layer_to_print.support_layer;
|
|
|
|
|
const PrintObject &object = *support_layer.object();
|
|
|
|
|
if (support_layer.support_fills.entities.size() > 0) {
|
|
|
|
|
// Both the support and the support interface are printed with the same extruder, therefore
|
|
|
|
|
// the interface may be interleaved with the support base.
|
|
|
|
|
// Don't change extruder if the extruder is set to 0. Use the current extruder instead.
|
|
|
|
|
bool single_extruder =
|
|
|
|
|
(object.config.support_material_extruder.value == object.config.support_material_interface_extruder.value ||
|
2017-05-16 11:45:28 +00:00
|
|
|
|
(object.config.support_material_extruder.value == int(first_extruder_id) && object.config.support_material_interface_extruder.value == 0) ||
|
|
|
|
|
(object.config.support_material_interface_extruder.value == int(first_extruder_id) && object.config.support_material_extruder.value == 0));
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Assign an extruder to the base.
|
|
|
|
|
ObjectByExtruder &obj = object_by_extruder(
|
|
|
|
|
by_extruder,
|
2017-05-16 11:45:28 +00:00
|
|
|
|
(object.config.support_material_extruder == 0) ? first_extruder_id : (object.config.support_material_extruder - 1),
|
2017-05-10 09:25:57 +00:00
|
|
|
|
&layer_to_print - layers.data(),
|
|
|
|
|
layers.size());
|
|
|
|
|
obj.support = &support_layer.support_fills;
|
|
|
|
|
obj.support_extrusion_role = single_extruder ? erMixed : erSupportMaterial;
|
|
|
|
|
if (! single_extruder) {
|
|
|
|
|
ObjectByExtruder &obj_interface = object_by_extruder(
|
|
|
|
|
by_extruder,
|
2017-05-16 11:45:28 +00:00
|
|
|
|
(object.config.support_material_interface_extruder == 0) ? first_extruder_id : (object.config.support_material_interface_extruder - 1),
|
2017-05-10 09:25:57 +00:00
|
|
|
|
&layer_to_print - layers.data(),
|
|
|
|
|
layers.size());
|
|
|
|
|
obj_interface.support = &support_layer.support_fills;
|
|
|
|
|
obj_interface.support_extrusion_role = erSupportMaterialInterface;
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
}
|
|
|
|
|
if (layer_to_print.object_layer != nullptr) {
|
|
|
|
|
const Layer &layer = *layer_to_print.object_layer;
|
|
|
|
|
// We now define a strategy for building perimeters and fills. The separation
|
|
|
|
|
// between regions doesn't matter in terms of printing order, as we follow
|
|
|
|
|
// another logic instead:
|
|
|
|
|
// - we group all extrusions by extruder so that we minimize toolchanges
|
|
|
|
|
// - we start from the last used extruder
|
|
|
|
|
// - for each extruder, we group extrusions by island
|
|
|
|
|
// - for each island, we extrude perimeters first, unless user set the infill_first
|
|
|
|
|
// option
|
|
|
|
|
// (Still, we have to keep track of regions because we need to apply their config)
|
|
|
|
|
size_t n_slices = layer.slices.expolygons.size();
|
|
|
|
|
std::vector<BoundingBox> layer_surface_bboxes;
|
|
|
|
|
layer_surface_bboxes.reserve(n_slices);
|
|
|
|
|
for (const ExPolygon &expoly : layer.slices.expolygons)
|
|
|
|
|
layer_surface_bboxes.push_back(get_extents(expoly.contour));
|
|
|
|
|
auto point_inside_surface = [&layer, &layer_surface_bboxes](const size_t i, const Point &point) {
|
|
|
|
|
const BoundingBox &bbox = layer_surface_bboxes[i];
|
|
|
|
|
return point.x >= bbox.min.x && point.x < bbox.max.x &&
|
|
|
|
|
point.y >= bbox.min.y && point.y < bbox.max.y &&
|
|
|
|
|
layer.slices.expolygons[i].contour.contains(point);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
for (size_t region_id = 0; region_id < print.regions.size(); ++ region_id) {
|
|
|
|
|
const LayerRegion *layerm = layer.regions[region_id];
|
|
|
|
|
if (layerm == nullptr)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
continue;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
const PrintRegion ®ion = *print.regions[region_id];
|
|
|
|
|
|
|
|
|
|
// process perimeters
|
|
|
|
|
for (const ExtrusionEntity *ee : layerm->perimeters.entities) {
|
|
|
|
|
// perimeter_coll represents perimeter extrusions of a single island.
|
|
|
|
|
const auto *perimeter_coll = dynamic_cast<const ExtrusionEntityCollection*>(ee);
|
|
|
|
|
if (perimeter_coll->entities.empty())
|
|
|
|
|
// This shouldn't happen but first_point() would fail.
|
|
|
|
|
continue;
|
|
|
|
|
// Init by_extruder item only if we actually use the extruder.
|
|
|
|
|
std::vector<ObjectByExtruder::Island> &islands = object_islands_by_extruder(
|
|
|
|
|
by_extruder,
|
|
|
|
|
std::max<int>(region.config.perimeter_extruder.value - 1, 0),
|
|
|
|
|
&layer_to_print - layers.data(),
|
|
|
|
|
layers.size(), n_slices+1);
|
|
|
|
|
for (size_t i = 0; i <= n_slices; ++ i)
|
|
|
|
|
if (// perimeter_coll->first_point does not fit inside any slice
|
|
|
|
|
i == n_slices ||
|
|
|
|
|
// perimeter_coll->first_point fits inside ith slice
|
|
|
|
|
point_inside_surface(i, perimeter_coll->first_point())) {
|
|
|
|
|
if (islands[i].by_region.empty())
|
|
|
|
|
islands[i].by_region.assign(print.regions.size(), ObjectByExtruder::Island::Region());
|
|
|
|
|
islands[i].by_region[region_id].perimeters.append(perimeter_coll->entities);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// process infill
|
|
|
|
|
// layerm->fills is a collection of Slic3r::ExtrusionPath::Collection objects (C++ class ExtrusionEntityCollection),
|
|
|
|
|
// each one containing the ExtrusionPath objects of a certain infill "group" (also called "surface"
|
|
|
|
|
// throughout the code). We can redefine the order of such Collections but we have to
|
|
|
|
|
// do each one completely at once.
|
|
|
|
|
for (const ExtrusionEntity *ee : layerm->fills.entities) {
|
|
|
|
|
// fill represents infill extrusions of a single island.
|
|
|
|
|
const auto *fill = dynamic_cast<const ExtrusionEntityCollection*>(ee);
|
|
|
|
|
if (fill->entities.empty())
|
|
|
|
|
// This shouldn't happen but first_point() would fail.
|
|
|
|
|
continue;
|
|
|
|
|
// init by_extruder item only if we actually use the extruder
|
|
|
|
|
int extruder_id = std::max<int>(0, (is_solid_infill(fill->entities.front()->role()) ? region.config.solid_infill_extruder : region.config.infill_extruder) - 1);
|
|
|
|
|
// Init by_extruder item only if we actually use the extruder.
|
|
|
|
|
std::vector<ObjectByExtruder::Island> &islands = object_islands_by_extruder(
|
|
|
|
|
by_extruder,
|
|
|
|
|
extruder_id,
|
|
|
|
|
&layer_to_print - layers.data(),
|
|
|
|
|
layers.size(), n_slices+1);
|
|
|
|
|
for (size_t i = 0; i <= n_slices; ++i)
|
|
|
|
|
if (// fill->first_point does not fit inside any slice
|
|
|
|
|
i == n_slices ||
|
|
|
|
|
// fill->first_point fits inside ith slice
|
|
|
|
|
point_inside_surface(i, fill->first_point())) {
|
|
|
|
|
if (islands[i].by_region.empty())
|
|
|
|
|
islands[i].by_region.assign(print.regions.size(), ObjectByExtruder::Island::Region());
|
|
|
|
|
islands[i].by_region[region_id].infills.append(fill->entities);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} // for regions
|
|
|
|
|
}
|
|
|
|
|
} // for objects
|
|
|
|
|
|
|
|
|
|
// Extrude the skirt, brim, support, perimeters, infill ordered by the extruders.
|
|
|
|
|
std::vector<std::unique_ptr<EdgeGrid::Grid>> lower_layer_edge_grids(layers.size());
|
2017-05-16 11:45:28 +00:00
|
|
|
|
for (unsigned int extruder_id : layer_tools.extruders)
|
2017-05-10 09:25:57 +00:00
|
|
|
|
{
|
|
|
|
|
gcode += this->set_extruder(extruder_id);
|
2017-05-16 11:45:28 +00:00
|
|
|
|
if (m_wipe_tower && ! m_wipe_tower->finished() && extruder_id == layer_tools.extruders.back())
|
|
|
|
|
// Last extruder change on the layer or no extruder change at all.
|
|
|
|
|
m_wipe_tower->close_layer();
|
2017-05-10 09:25:57 +00:00
|
|
|
|
|
|
|
|
|
if (extrude_skirt) {
|
|
|
|
|
auto loops_it = skirt_loops_per_extruder.find(extruder_id);
|
|
|
|
|
if (loops_it != skirt_loops_per_extruder.end()) {
|
|
|
|
|
const std::pair<size_t, size_t> loops = loops_it->second;
|
|
|
|
|
this->set_origin(0.,0.);
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp = true;
|
|
|
|
|
Flow skirt_flow = print.skirt_flow();
|
|
|
|
|
for (size_t i = loops.first; i < loops.second; ++ i) {
|
|
|
|
|
// Adjust flow according to this layer's layer height.
|
|
|
|
|
ExtrusionLoop loop = *dynamic_cast<const ExtrusionLoop*>(print.skirt.entities[i]);
|
|
|
|
|
Flow layer_skirt_flow(skirt_flow);
|
|
|
|
|
layer_skirt_flow.height = (float)skirt_height;
|
|
|
|
|
double mm3_per_mm = layer_skirt_flow.mm3_per_mm();
|
|
|
|
|
for (ExtrusionPath &path : loop.paths) {
|
|
|
|
|
path.height = (float)layer.height;
|
|
|
|
|
path.mm3_per_mm = mm3_per_mm;
|
|
|
|
|
}
|
|
|
|
|
gcode += this->extrude_loop(loop, "skirt", m_config.support_material_speed.value);
|
|
|
|
|
}
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp = false;
|
|
|
|
|
// Allow a straight travel move to the first object point if this is the first layer (but don't in next layers).
|
|
|
|
|
if (first_layer && loops.first == 0)
|
|
|
|
|
m_avoid_crossing_perimeters.disable_once = true;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Extrude brim with the extruder of the 1st region.
|
|
|
|
|
if (! m_brim_done) {
|
|
|
|
|
this->set_origin(0.f, 0.f);
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp = true;
|
|
|
|
|
for (const ExtrusionEntity *ee : print.brim.entities)
|
|
|
|
|
gcode += this->extrude_loop(*dynamic_cast<const ExtrusionLoop*>(ee), "brim", m_config.support_material_speed.value);
|
|
|
|
|
m_brim_done = true;
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp = false;
|
|
|
|
|
// Allow a straight travel move to the first object point.
|
|
|
|
|
m_avoid_crossing_perimeters.disable_once = true;
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
auto objects_by_extruder_it = by_extruder.find(extruder_id);
|
|
|
|
|
if (objects_by_extruder_it == by_extruder.end())
|
|
|
|
|
continue;
|
|
|
|
|
for (const ObjectByExtruder &object_by_extruder : objects_by_extruder_it->second) {
|
|
|
|
|
const size_t layer_id = &object_by_extruder - objects_by_extruder_it->second.data();
|
2017-05-15 09:32:59 +00:00
|
|
|
|
const PrintObject *print_object = layers[layer_id].object();
|
|
|
|
|
if (print_object == nullptr)
|
|
|
|
|
// This layer is empty for this particular object, it has neither object extrusions nor support extrusions at this print_z.
|
|
|
|
|
continue;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (m_enable_analyzer_markers) {
|
|
|
|
|
// Store the binary pointer to the layer object directly into the G-code to be accessed by the GCodeAnalyzer.
|
|
|
|
|
char buf[64];
|
|
|
|
|
sprintf(buf, ";_LAYEROBJ:%p\n", m_layer);
|
|
|
|
|
gcode += buf;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
2017-05-15 09:32:59 +00:00
|
|
|
|
m_config.apply(print_object->config, true);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
m_layer = layers[layer_id].layer();
|
|
|
|
|
if (m_config.avoid_crossing_perimeters)
|
|
|
|
|
m_avoid_crossing_perimeters.init_layer_mp(union_ex(m_layer->slices, true));
|
|
|
|
|
Points copies;
|
|
|
|
|
if (single_object_idx == size_t(-1))
|
2017-05-15 09:32:59 +00:00
|
|
|
|
copies = print_object->_shifted_copies;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
else
|
2017-05-15 09:32:59 +00:00
|
|
|
|
copies.push_back(print_object->_shifted_copies[single_object_idx]);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
for (const Point © : copies) {
|
|
|
|
|
// When starting a new object, use the external motion planner for the first travel move.
|
2017-05-15 09:32:59 +00:00
|
|
|
|
std::pair<const PrintObject*, Point> this_object_copy(print_object, copy);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (m_last_obj_copy != this_object_copy)
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp_once = true;
|
|
|
|
|
m_last_obj_copy = this_object_copy;
|
|
|
|
|
this->set_origin(unscale(copy.x), unscale(copy.y));
|
|
|
|
|
if (object_by_extruder.support != nullptr) {
|
|
|
|
|
m_layer = layers[layer_id].support_layer;
|
|
|
|
|
gcode += this->extrude_support(
|
|
|
|
|
// support_extrusion_role is erSupportMaterial, erSupportMaterialInterface or erMixed for all extrusion paths.
|
|
|
|
|
object_by_extruder.support->chained_path_from(m_last_pos, false, object_by_extruder.support_extrusion_role));
|
|
|
|
|
m_layer = layers[layer_id].layer();
|
|
|
|
|
}
|
|
|
|
|
for (const ObjectByExtruder::Island &island : object_by_extruder.islands) {
|
|
|
|
|
if (print.config.infill_first) {
|
|
|
|
|
gcode += this->extrude_infill(print, island.by_region);
|
|
|
|
|
gcode += this->extrude_perimeters(print, island.by_region, lower_layer_edge_grids[layer_id]);
|
|
|
|
|
} else {
|
|
|
|
|
gcode += this->extrude_perimeters(print, island.by_region, lower_layer_edge_grids[layer_id]);
|
|
|
|
|
gcode += this->extrude_infill(print, island.by_region);
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
|
|
|
|
|
// Apply spiral vase post-processing if this layer contains suitable geometry
|
|
|
|
|
// (we must feed all the G-code into the post-processor, including the first
|
|
|
|
|
// bottom non-spiral layers otherwise it will mess with positions)
|
|
|
|
|
// we apply spiral vase at this stage because it requires a full layer.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Just a reminder: A spiral vase mode is allowed for a single object, single material print only.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_spiral_vase)
|
|
|
|
|
gcode = m_spiral_vase->process_layer(gcode);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// Apply cooling logic; this may alter speeds.
|
|
|
|
|
if (m_cooling_buffer)
|
2017-05-15 09:32:59 +00:00
|
|
|
|
//FIXME Update the CoolingBuffer class to ignore the object ID, which does not make sense anymore
|
|
|
|
|
// once all extrusions of a layer are processed at once.
|
|
|
|
|
// Update the test cases.
|
|
|
|
|
gcode = m_cooling_buffer->append(gcode, 0, layer.id(), false);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
write(file, this->filter(std::move(gcode), false));
|
2015-07-01 21:00:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
std::string GCode::filter(std::string &&gcode, bool flush)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// apply pressure equalization if enabled;
|
|
|
|
|
// printf("G-code before filter:\n%s\n", gcode.c_str());
|
|
|
|
|
std::string out = m_pressure_equalizer ?
|
|
|
|
|
m_pressure_equalizer->process(gcode.c_str(), flush) :
|
|
|
|
|
std::move(gcode);
|
|
|
|
|
// printf("G-code after filter:\n%s\n", out.c_str());
|
|
|
|
|
return out;
|
2015-07-01 21:00:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
void GCode::apply_print_config(const PrintConfig &print_config)
|
2015-07-02 17:33:08 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_writer.apply_print_config(print_config);
|
|
|
|
|
m_config.apply(print_config);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void GCode::set_extruders(const std::vector<unsigned int> &extruder_ids)
|
|
|
|
|
{
|
|
|
|
|
m_writer.set_extruders(extruder_ids);
|
2015-07-02 17:33:08 +00:00
|
|
|
|
|
|
|
|
|
// enable wipe path generation if any extruder has wipe enabled
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_wipe.enable = false;
|
2017-03-29 15:45:38 +00:00
|
|
|
|
for (auto id : extruder_ids)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_config.wipe.get_at(id)) {
|
|
|
|
|
m_wipe.enable = true;
|
2015-07-02 17:33:08 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
void GCode::set_origin(const Pointf &pointf)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
|
|
|
|
// if origin increases (goes towards right), last_pos decreases because it goes towards left
|
2015-12-19 13:49:29 +00:00
|
|
|
|
const Point translate(
|
2017-05-03 16:28:22 +00:00
|
|
|
|
scale_(m_origin.x - pointf.x),
|
|
|
|
|
scale_(m_origin.y - pointf.y)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_last_pos.translate(translate);
|
|
|
|
|
m_wipe.path.translate(translate);
|
|
|
|
|
m_origin = pointf;
|
2015-07-01 21:00:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
std::string GCode::preamble()
|
2015-07-01 21:00:52 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
std::string gcode = m_writer.preamble();
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
/* Perform a *silent* move to z_offset: we need this to initialize the Z
|
|
|
|
|
position of our writer object so that any initial lift taking place
|
|
|
|
|
before the first layer change will raise the extruder from the correct
|
|
|
|
|
initial Z instead of 0. */
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_writer.travel_to_z(m_config.z_offset.value);
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// called by GCode::process_layer()
|
|
|
|
|
std::string GCode::change_layer(coordf_t print_z)
|
2015-07-02 17:33:08 +00:00
|
|
|
|
{
|
2016-09-26 10:52:40 +00:00
|
|
|
|
std::string gcode;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_layer_count > 0)
|
2017-05-10 09:25:57 +00:00
|
|
|
|
// Increment a progress bar indicator.
|
|
|
|
|
gcode += m_writer.update_progress(++ m_layer_index, m_layer_count);
|
|
|
|
|
coordf_t z = print_z + m_config.z_offset.value; // in unscaled coordinates
|
|
|
|
|
if (EXTRUDER_CONFIG(retract_layer_change) && m_writer.will_move_z(z))
|
2015-07-02 17:33:08 +00:00
|
|
|
|
gcode += this->retract();
|
2017-05-10 09:25:57 +00:00
|
|
|
|
|
2015-07-02 17:33:08 +00:00
|
|
|
|
{
|
|
|
|
|
std::ostringstream comment;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
comment << "move to next layer (" << m_layer_index << ")";
|
|
|
|
|
gcode += m_writer.travel_to_z(z, comment.str());
|
2015-07-02 17:33:08 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// forget last wiping path as wiping after raising Z is pointless
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_wipe.reset_path();
|
2015-07-02 17:33:08 +00:00
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-12 14:25:15 +00:00
|
|
|
|
static inline const char* ExtrusionRole2String(const ExtrusionRole role)
|
|
|
|
|
{
|
|
|
|
|
switch (role) {
|
|
|
|
|
case erNone: return "erNone";
|
|
|
|
|
case erPerimeter: return "erPerimeter";
|
|
|
|
|
case erExternalPerimeter: return "erExternalPerimeter";
|
|
|
|
|
case erOverhangPerimeter: return "erOverhangPerimeter";
|
|
|
|
|
case erInternalInfill: return "erInternalInfill";
|
|
|
|
|
case erSolidInfill: return "erSolidInfill";
|
|
|
|
|
case erTopSolidInfill: return "erTopSolidInfill";
|
|
|
|
|
case erBridgeInfill: return "erBridgeInfill";
|
|
|
|
|
case erGapFill: return "erGapFill";
|
|
|
|
|
case erSkirt: return "erSkirt";
|
|
|
|
|
case erSupportMaterial: return "erSupportMaterial";
|
|
|
|
|
case erSupportMaterialInterface: return "erSupportMaterialInterface";
|
2017-04-07 15:37:30 +00:00
|
|
|
|
case erMixed: return "erMixed";
|
2016-09-12 14:25:15 +00:00
|
|
|
|
default: return "erInvalid";
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline const char* ExtrusionLoopRole2String(const ExtrusionLoopRole role)
|
|
|
|
|
{
|
|
|
|
|
switch (role) {
|
|
|
|
|
case elrDefault: return "elrDefault";
|
|
|
|
|
case elrContourInternalPerimeter: return "elrContourInternalPerimeter";
|
|
|
|
|
case elrSkirt: return "elrSkirt";
|
|
|
|
|
default: return "elrInvalid";
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Return a value in <0, 1> of a cubic B-spline kernel centered around zero.
|
|
|
|
|
// The B-spline is re-scaled so it has value 1 at zero.
|
|
|
|
|
static inline float bspline_kernel(float x)
|
|
|
|
|
{
|
|
|
|
|
x = std::abs(x);
|
|
|
|
|
if (x < 1.f) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
return 1.f - (3.f / 2.f) * x * x + (3.f / 4.f) * x * x * x;
|
2016-09-12 14:25:15 +00:00
|
|
|
|
}
|
|
|
|
|
else if (x < 2.f) {
|
|
|
|
|
x -= 1.f;
|
|
|
|
|
float x2 = x * x;
|
|
|
|
|
float x3 = x2 * x;
|
|
|
|
|
return (1.f / 4.f) - (3.f / 4.f) * x + (3.f / 4.f) * x2 - (1.f / 4.f) * x3;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static float extrudate_overlap_penalty(float nozzle_r, float weight_zero, float overlap_distance)
|
|
|
|
|
{
|
|
|
|
|
// The extrudate is not fully supported by the lower layer. Fit a polynomial penalty curve.
|
|
|
|
|
// Solved by sympy package:
|
|
|
|
|
/*
|
|
|
|
|
from sympy import *
|
|
|
|
|
(x,a,b,c,d,r,z)=symbols('x a b c d r z')
|
|
|
|
|
p = a + b*x + c*x*x + d*x*x*x
|
|
|
|
|
p2 = p.subs(solve([p.subs(x, -r), p.diff(x).subs(x, -r), p.diff(x,x).subs(x, -r), p.subs(x, 0)-z], [a, b, c, d]))
|
|
|
|
|
from sympy.plotting import plot
|
|
|
|
|
plot(p2.subs(r,0.2).subs(z,1.), (x, -1, 3), adaptive=False, nb_of_points=400)
|
|
|
|
|
*/
|
|
|
|
|
if (overlap_distance < - nozzle_r) {
|
|
|
|
|
// The extrudate is fully supported by the lower layer. This is the ideal case, therefore zero penalty.
|
|
|
|
|
return 0.f;
|
|
|
|
|
} else {
|
|
|
|
|
float x = overlap_distance / nozzle_r;
|
|
|
|
|
float x2 = x * x;
|
|
|
|
|
float x3 = x2 * x;
|
|
|
|
|
return weight_zero * (1.f + 3.f * x + 3.f * x2 + x3);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static Points::iterator project_point_to_polygon_and_insert(Polygon &polygon, const Point &pt, double eps)
|
|
|
|
|
{
|
|
|
|
|
assert(polygon.points.size() >= 2);
|
|
|
|
|
if (polygon.points.size() <= 1)
|
|
|
|
|
if (polygon.points.size() == 1)
|
|
|
|
|
return polygon.points.begin();
|
|
|
|
|
|
|
|
|
|
Point pt_min;
|
|
|
|
|
double d_min = std::numeric_limits<double>::max();
|
|
|
|
|
size_t i_min = size_t(-1);
|
|
|
|
|
|
|
|
|
|
for (size_t i = 0; i < polygon.points.size(); ++ i) {
|
|
|
|
|
size_t j = i + 1;
|
|
|
|
|
if (j == polygon.points.size())
|
|
|
|
|
j = 0;
|
|
|
|
|
const Point &p1 = polygon.points[i];
|
|
|
|
|
const Point &p2 = polygon.points[j];
|
|
|
|
|
const Slic3r::Point v_seg = p1.vector_to(p2);
|
|
|
|
|
const Slic3r::Point v_pt = p1.vector_to(pt);
|
|
|
|
|
const int64_t l2_seg = int64_t(v_seg.x) * int64_t(v_seg.x) + int64_t(v_seg.y) * int64_t(v_seg.y);
|
|
|
|
|
int64_t t_pt = int64_t(v_seg.x) * int64_t(v_pt.x) + int64_t(v_seg.y) * int64_t(v_pt.y);
|
|
|
|
|
if (t_pt < 0) {
|
|
|
|
|
// Closest to p1.
|
|
|
|
|
double dabs = sqrt(int64_t(v_pt.x) * int64_t(v_pt.x) + int64_t(v_pt.y) * int64_t(v_pt.y));
|
|
|
|
|
if (dabs < d_min) {
|
|
|
|
|
d_min = dabs;
|
|
|
|
|
i_min = i;
|
|
|
|
|
pt_min = p1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (t_pt > l2_seg) {
|
|
|
|
|
// Closest to p2. Then p2 is the starting point of another segment, which shall be discovered in the next step.
|
|
|
|
|
continue;
|
|
|
|
|
} else {
|
|
|
|
|
// Closest to the segment.
|
|
|
|
|
assert(t_pt >= 0 && t_pt <= l2_seg);
|
|
|
|
|
int64_t d_seg = int64_t(v_seg.y) * int64_t(v_pt.x) - int64_t(v_seg.x) * int64_t(v_pt.y);
|
|
|
|
|
double d = double(d_seg) / sqrt(double(l2_seg));
|
|
|
|
|
double dabs = std::abs(d);
|
|
|
|
|
if (dabs < d_min) {
|
|
|
|
|
d_min = dabs;
|
|
|
|
|
i_min = i;
|
|
|
|
|
// Evaluate the foot point.
|
|
|
|
|
pt_min = p1;
|
|
|
|
|
double linv = double(d_seg) / double(l2_seg);
|
|
|
|
|
pt_min.x = pt.x - coord_t(floor(double(v_seg.y) * linv + 0.5));
|
|
|
|
|
pt_min.y = pt.y + coord_t(floor(double(v_seg.x) * linv + 0.5));
|
|
|
|
|
assert(Line(p1, p2).distance_to(pt_min) < scale_(1e-5));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(i_min != size_t(-1));
|
|
|
|
|
if (pt_min.distance_to(polygon.points[i_min]) > eps) {
|
|
|
|
|
// Insert a new point on the segment i_min, i_min+1.
|
|
|
|
|
return polygon.points.insert(polygon.points.begin() + (i_min + 1), pt_min);
|
|
|
|
|
}
|
|
|
|
|
return polygon.points.begin() + i_min;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::vector<float> polygon_parameter_by_length(const Polygon &polygon)
|
|
|
|
|
{
|
|
|
|
|
// Parametrize the polygon by its length.
|
|
|
|
|
std::vector<float> lengths(polygon.points.size()+1, 0.);
|
|
|
|
|
for (size_t i = 1; i < polygon.points.size(); ++ i)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
lengths[i] = lengths[i-1] + float(polygon.points[i].distance_to(polygon.points[i-1]));
|
|
|
|
|
lengths.back() = lengths[lengths.size()-2] + float(polygon.points.front().distance_to(polygon.points.back()));
|
2016-09-12 14:25:15 +00:00
|
|
|
|
return lengths;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::vector<float> polygon_angles_at_vertices(const Polygon &polygon, const std::vector<float> &lengths, float min_arm_length)
|
|
|
|
|
{
|
|
|
|
|
assert(polygon.points.size() + 1 == lengths.size());
|
|
|
|
|
if (min_arm_length > 0.25f * lengths.back())
|
|
|
|
|
min_arm_length = 0.25f * lengths.back();
|
|
|
|
|
|
|
|
|
|
// Find the initial prev / next point span.
|
|
|
|
|
size_t idx_prev = polygon.points.size();
|
|
|
|
|
size_t idx_curr = 0;
|
|
|
|
|
size_t idx_next = 1;
|
|
|
|
|
while (idx_prev > idx_curr && lengths.back() - lengths[idx_prev] < min_arm_length)
|
|
|
|
|
-- idx_prev;
|
|
|
|
|
while (idx_next < idx_prev && lengths[idx_next] < min_arm_length)
|
|
|
|
|
++ idx_next;
|
|
|
|
|
|
|
|
|
|
std::vector<float> angles(polygon.points.size(), 0.f);
|
|
|
|
|
for (; idx_curr < polygon.points.size(); ++ idx_curr) {
|
|
|
|
|
// Move idx_prev up until the distance between idx_prev and idx_curr is lower than min_arm_length.
|
|
|
|
|
if (idx_prev >= idx_curr) {
|
|
|
|
|
while (idx_prev < polygon.points.size() && lengths.back() - lengths[idx_prev] + lengths[idx_curr] > min_arm_length)
|
|
|
|
|
++ idx_prev;
|
|
|
|
|
if (idx_prev == polygon.points.size())
|
|
|
|
|
idx_prev = 0;
|
|
|
|
|
}
|
|
|
|
|
while (idx_prev < idx_curr && lengths[idx_curr] - lengths[idx_prev] > min_arm_length)
|
|
|
|
|
++ idx_prev;
|
|
|
|
|
// Move idx_prev one step back.
|
|
|
|
|
if (idx_prev == 0)
|
|
|
|
|
idx_prev = polygon.points.size() - 1;
|
|
|
|
|
else
|
|
|
|
|
-- idx_prev;
|
|
|
|
|
// Move idx_next up until the distance between idx_curr and idx_next is greater than min_arm_length.
|
|
|
|
|
if (idx_curr <= idx_next) {
|
|
|
|
|
while (idx_next < polygon.points.size() && lengths[idx_next] - lengths[idx_curr] < min_arm_length)
|
|
|
|
|
++ idx_next;
|
|
|
|
|
if (idx_next == polygon.points.size())
|
|
|
|
|
idx_next = 0;
|
|
|
|
|
}
|
|
|
|
|
while (idx_next < idx_curr && lengths.back() - lengths[idx_curr] + lengths[idx_next] < min_arm_length)
|
|
|
|
|
++ idx_next;
|
|
|
|
|
// Calculate angle between idx_prev, idx_curr, idx_next.
|
|
|
|
|
const Point &p0 = polygon.points[idx_prev];
|
|
|
|
|
const Point &p1 = polygon.points[idx_curr];
|
|
|
|
|
const Point &p2 = polygon.points[idx_next];
|
|
|
|
|
const Point v1 = p0.vector_to(p1);
|
|
|
|
|
const Point v2 = p1.vector_to(p2);
|
|
|
|
|
int64_t dot = int64_t(v1.x)*int64_t(v2.x) + int64_t(v1.y)*int64_t(v2.y);
|
|
|
|
|
int64_t cross = int64_t(v1.x)*int64_t(v2.y) - int64_t(v1.y)*int64_t(v2.x);
|
|
|
|
|
float angle = float(atan2(double(cross), double(dot)));
|
|
|
|
|
angles[idx_curr] = angle;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return angles;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::extrude_loop(ExtrusionLoop loop, std::string description, double speed, std::unique_ptr<EdgeGrid::Grid> *lower_layer_edge_grid)
|
2015-07-02 16:57:40 +00:00
|
|
|
|
{
|
2015-07-02 18:24:16 +00:00
|
|
|
|
// get a copy; don't modify the orientation of the original loop object otherwise
|
|
|
|
|
// next copies (if any) would not detect the correct orientation
|
2016-09-12 14:25:15 +00:00
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (m_layer->lower_layer != nullptr && lower_layer_edge_grid != nullptr) {
|
|
|
|
|
if (! *lower_layer_edge_grid) {
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// Create the distance field for a layer below.
|
|
|
|
|
const coord_t distance_field_resolution = scale_(1.f);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
*lower_layer_edge_grid = make_unique<EdgeGrid::Grid>();
|
|
|
|
|
(*lower_layer_edge_grid)->create(m_layer->lower_layer->slices, distance_field_resolution);
|
|
|
|
|
(*lower_layer_edge_grid)->calculate_sdf();
|
2016-09-12 14:25:15 +00:00
|
|
|
|
#if 0
|
|
|
|
|
{
|
|
|
|
|
static int iRun = 0;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
BoundingBox bbox = (*lower_layer_edge_grid)->bbox();
|
2016-09-12 14:25:15 +00:00
|
|
|
|
bbox.min.x -= scale_(5.f);
|
|
|
|
|
bbox.min.y -= scale_(5.f);
|
|
|
|
|
bbox.max.x += scale_(5.f);
|
|
|
|
|
bbox.max.y += scale_(5.f);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
EdgeGrid::save_png(*(*lower_layer_edge_grid), bbox, scale_(0.1f), debug_out_path("GCode_extrude_loop_edge_grid-%d.png", iRun++));
|
2016-09-12 14:25:15 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-02 18:24:16 +00:00
|
|
|
|
// extrude all loops ccw
|
|
|
|
|
bool was_clockwise = loop.make_counter_clockwise();
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
SeamPosition seam_position = m_config.seam_position;
|
2017-04-07 15:37:30 +00:00
|
|
|
|
if (loop.loop_role() == elrSkirt)
|
2017-02-07 17:46:02 +00:00
|
|
|
|
seam_position = spNearest;
|
2015-12-21 14:02:39 +00:00
|
|
|
|
|
2015-07-02 18:24:16 +00:00
|
|
|
|
// find the point of the loop that is closest to the current extruder position
|
|
|
|
|
// or randomize if requested
|
|
|
|
|
Point last_pos = this->last_pos();
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_config.spiral_vase) {
|
2017-02-02 17:49:33 +00:00
|
|
|
|
loop.split_at(last_pos, false);
|
2017-02-07 17:46:02 +00:00
|
|
|
|
} else if (seam_position == spNearest || seam_position == spAligned || seam_position == spRear) {
|
2016-09-12 14:25:15 +00:00
|
|
|
|
Polygon polygon = loop.polygon();
|
|
|
|
|
const coordf_t nozzle_dmr = EXTRUDER_CONFIG(nozzle_diameter);
|
|
|
|
|
const coord_t nozzle_r = scale_(0.5*nozzle_dmr);
|
|
|
|
|
|
|
|
|
|
// Retrieve the last start position for this object.
|
|
|
|
|
float last_pos_weight = 1.f;
|
2017-02-07 17:46:02 +00:00
|
|
|
|
switch (seam_position) {
|
|
|
|
|
case spAligned:
|
|
|
|
|
// Seam is aligned to the seam at the preceding layer.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_layer != NULL && m_seam_position.count(m_layer->object()) > 0) {
|
|
|
|
|
last_pos = m_seam_position[m_layer->object()];
|
2017-02-22 15:35:07 +00:00
|
|
|
|
last_pos_weight = 1.f;
|
2017-02-07 17:46:02 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case spRear:
|
2017-05-03 16:28:22 +00:00
|
|
|
|
last_pos = m_layer->object()->bounding_box().center();
|
|
|
|
|
last_pos.y += coord_t(3. * m_layer->object()->bounding_box().radius());
|
2016-09-12 14:25:15 +00:00
|
|
|
|
last_pos_weight = 5.f;
|
2017-02-07 17:46:02 +00:00
|
|
|
|
break;
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
2016-09-12 14:25:15 +00:00
|
|
|
|
|
|
|
|
|
// Insert a projection of last_pos into the polygon.
|
2017-02-07 17:46:02 +00:00
|
|
|
|
size_t last_pos_proj_idx;
|
|
|
|
|
{
|
|
|
|
|
Points::iterator it = project_point_to_polygon_and_insert(polygon, last_pos, 0.1 * nozzle_r);
|
|
|
|
|
last_pos_proj_idx = it - polygon.points.begin();
|
|
|
|
|
}
|
2016-09-12 14:25:15 +00:00
|
|
|
|
Point last_pos_proj = polygon.points[last_pos_proj_idx];
|
|
|
|
|
// Parametrize the polygon by its length.
|
|
|
|
|
std::vector<float> lengths = polygon_parameter_by_length(polygon);
|
|
|
|
|
|
|
|
|
|
// For each polygon point, store a penalty.
|
|
|
|
|
// First calculate the angles, store them as penalties. The angles are caluculated over a minimum arm length of nozzle_r.
|
|
|
|
|
std::vector<float> penalties = polygon_angles_at_vertices(polygon, lengths, nozzle_r);
|
|
|
|
|
// No penalty for reflex points, slight penalty for convex points, high penalty for flat surfaces.
|
|
|
|
|
const float penaltyConvexVertex = 1.f;
|
|
|
|
|
const float penaltyFlatSurface = 5.f;
|
|
|
|
|
const float penaltySeam = 1.3f;
|
|
|
|
|
const float penaltyOverhangHalf = 10.f;
|
|
|
|
|
// Penalty for visible seams.
|
|
|
|
|
for (size_t i = 0; i < polygon.points.size(); ++ i) {
|
|
|
|
|
float ccwAngle = penalties[i];
|
|
|
|
|
if (was_clockwise)
|
|
|
|
|
ccwAngle = - ccwAngle;
|
|
|
|
|
float penalty = 0;
|
|
|
|
|
// if (ccwAngle <- float(PI/3.))
|
|
|
|
|
if (ccwAngle <- float(0.6 * PI))
|
|
|
|
|
// Sharp reflex vertex. We love that, it hides the seam perfectly.
|
|
|
|
|
penalty = 0.f;
|
|
|
|
|
// else if (ccwAngle > float(PI/3.))
|
|
|
|
|
else if (ccwAngle > float(0.6 * PI))
|
|
|
|
|
// Seams on sharp convex vertices are more visible than on reflex vertices.
|
|
|
|
|
penalty = penaltyConvexVertex;
|
|
|
|
|
else if (ccwAngle < 0.f) {
|
|
|
|
|
// Interpolate penalty between maximum and zero.
|
|
|
|
|
penalty = penaltyFlatSurface * bspline_kernel(ccwAngle * (PI * 2. / 3.));
|
|
|
|
|
} else {
|
|
|
|
|
assert(ccwAngle >= 0.f);
|
|
|
|
|
// Interpolate penalty between maximum and the penalty for a convex vertex.
|
|
|
|
|
penalty = penaltyConvexVertex + (penaltyFlatSurface - penaltyConvexVertex) * bspline_kernel(ccwAngle * (PI * 2. / 3.));
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// Give a negative penalty for points close to the last point or the prefered seam location.
|
|
|
|
|
//float dist_to_last_pos_proj = last_pos_proj.distance_to(polygon.points[i]);
|
|
|
|
|
float dist_to_last_pos_proj = (i < last_pos_proj_idx) ?
|
|
|
|
|
std::min(lengths[last_pos_proj_idx] - lengths[i], lengths.back() - lengths[last_pos_proj_idx] + lengths[i]) :
|
|
|
|
|
std::min(lengths[i] - lengths[last_pos_proj_idx], lengths.back() - lengths[i] + lengths[last_pos_proj_idx]);
|
|
|
|
|
float dist_max = 0.1f * lengths.back(); // 5.f * nozzle_dmr
|
|
|
|
|
penalty -= last_pos_weight * bspline_kernel(dist_to_last_pos_proj / dist_max);
|
|
|
|
|
penalties[i] = std::max(0.f, penalty);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
2016-09-12 14:25:15 +00:00
|
|
|
|
|
|
|
|
|
// Penalty for overhangs.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (lower_layer_edge_grid && (*lower_layer_edge_grid)) {
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// Use the edge grid distance field structure over the lower layer to calculate overhangs.
|
|
|
|
|
coord_t nozzle_r = scale_(0.5*nozzle_dmr);
|
|
|
|
|
coord_t search_r = scale_(0.8*nozzle_dmr);
|
|
|
|
|
for (size_t i = 0; i < polygon.points.size(); ++ i) {
|
|
|
|
|
const Point &p = polygon.points[i];
|
|
|
|
|
coordf_t dist;
|
|
|
|
|
// Signed distance is positive outside the object, negative inside the object.
|
|
|
|
|
// The point is considered at an overhang, if it is more than nozzle radius
|
|
|
|
|
// outside of the lower layer contour.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
bool found = (*lower_layer_edge_grid)->signed_distance(p, search_r, dist);
|
|
|
|
|
// If the approximate Signed Distance Field was initialized over lower_layer_edge_grid,
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// then the signed distnace shall always be known.
|
|
|
|
|
assert(found);
|
|
|
|
|
penalties[i] += extrudate_overlap_penalty(nozzle_r, penaltyOverhangHalf, dist);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-09-12 14:25:15 +00:00
|
|
|
|
|
|
|
|
|
// Find a point with a minimum penalty.
|
|
|
|
|
size_t idx_min = std::min_element(penalties.begin(), penalties.end()) - penalties.begin();
|
|
|
|
|
|
2017-02-22 15:35:07 +00:00
|
|
|
|
// if (seam_position == spAligned)
|
|
|
|
|
// For all (aligned, nearest, rear) seams:
|
|
|
|
|
{
|
|
|
|
|
// Very likely the weight of idx_min is very close to the weight of last_pos_proj_idx.
|
|
|
|
|
// In that case use last_pos_proj_idx instead.
|
|
|
|
|
float penalty_aligned = penalties[last_pos_proj_idx];
|
|
|
|
|
float penalty_min = penalties[idx_min];
|
|
|
|
|
float penalty_diff_abs = std::abs(penalty_min - penalty_aligned);
|
|
|
|
|
float penalty_max = std::max(penalty_min, penalty_aligned);
|
|
|
|
|
float penalty_diff_rel = (penalty_max == 0.f) ? 0.f : penalty_diff_abs / penalty_max;
|
|
|
|
|
// printf("Align seams, penalty aligned: %f, min: %f, diff abs: %f, diff rel: %f\n", penalty_aligned, penalty_min, penalty_diff_abs, penalty_diff_rel);
|
|
|
|
|
if (penalty_diff_rel < 0.05) {
|
|
|
|
|
// Penalty of the aligned point is very close to the minimum penalty.
|
|
|
|
|
// Align the seams as accurately as possible.
|
|
|
|
|
idx_min = last_pos_proj_idx;
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_seam_position[m_layer->object()] = polygon.points[idx_min];
|
2017-02-22 15:35:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// Export the contour into a SVG file.
|
|
|
|
|
#if 0
|
|
|
|
|
{
|
|
|
|
|
static int iRun = 0;
|
2016-10-21 08:18:01 +00:00
|
|
|
|
SVG svg(debug_out_path("GCode_extrude_loop-%d.svg", iRun ++));
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_layer->lower_layer != NULL)
|
|
|
|
|
svg.draw(m_layer->lower_layer->slices.expolygons);
|
2016-09-12 14:25:15 +00:00
|
|
|
|
for (size_t i = 0; i < loop.paths.size(); ++ i)
|
|
|
|
|
svg.draw(loop.paths[i].as_polyline(), "red");
|
|
|
|
|
Polylines polylines;
|
|
|
|
|
for (size_t i = 0; i < loop.paths.size(); ++ i)
|
|
|
|
|
polylines.push_back(loop.paths[i].as_polyline());
|
|
|
|
|
Slic3r::Polygons polygons;
|
|
|
|
|
coordf_t nozzle_dmr = EXTRUDER_CONFIG(nozzle_diameter);
|
|
|
|
|
coord_t delta = scale_(0.5*nozzle_dmr);
|
|
|
|
|
Slic3r::offset(polylines, &polygons, delta);
|
|
|
|
|
// for (size_t i = 0; i < polygons.size(); ++ i) svg.draw((Polyline)polygons[i], "blue");
|
2017-02-07 17:46:02 +00:00
|
|
|
|
svg.draw(last_pos, "green", 3);
|
|
|
|
|
svg.draw(polygon.points[idx_min], "yellow", 3);
|
|
|
|
|
svg.Close();
|
2016-09-12 14:25:15 +00:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
// Split the loop at the point with a minium penalty.
|
|
|
|
|
if (!loop.split_at_vertex(polygon.points[idx_min]))
|
|
|
|
|
// The point is not in the original loop. Insert it.
|
2017-02-02 17:49:33 +00:00
|
|
|
|
loop.split_at(polygon.points[idx_min], true);
|
2016-09-12 14:25:15 +00:00
|
|
|
|
|
2015-12-21 14:02:39 +00:00
|
|
|
|
} else if (seam_position == spRandom) {
|
2017-04-07 15:37:30 +00:00
|
|
|
|
if (loop.loop_role() == elrContourInternalPerimeter) {
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// This loop does not contain any other loop. Set a random position.
|
|
|
|
|
// The other loops will get a seam close to the random point chosen
|
|
|
|
|
// on the inner most contour.
|
|
|
|
|
//FIXME This works correctly for inner contours first only.
|
|
|
|
|
//FIXME Better parametrize the loop by its length.
|
2015-07-02 18:24:16 +00:00
|
|
|
|
Polygon polygon = loop.polygon();
|
|
|
|
|
Point centroid = polygon.centroid();
|
|
|
|
|
last_pos = Point(polygon.bounding_box().max.x, centroid.y);
|
2015-09-30 13:22:49 +00:00
|
|
|
|
last_pos.rotate(fmod((float)rand()/16.0, 2.0*PI), centroid);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
2017-02-07 17:46:02 +00:00
|
|
|
|
// Find the closest point, avoid overhangs.
|
2017-02-02 17:49:33 +00:00
|
|
|
|
loop.split_at(last_pos, true);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// clip the path to avoid the extruder to get exactly on the first point of the loop;
|
|
|
|
|
// if polyline was shorter than the clipping distance we'd get a null polyline, so
|
|
|
|
|
// we discard it in that case
|
2017-05-03 16:28:22 +00:00
|
|
|
|
double clip_length = m_enable_loop_clipping ?
|
|
|
|
|
scale_(EXTRUDER_CONFIG(nozzle_diameter)) * LOOP_CLIPPING_LENGTH_OVER_NOZZLE_DIAMETER :
|
|
|
|
|
0;
|
|
|
|
|
|
2015-07-02 18:24:16 +00:00
|
|
|
|
// get paths
|
|
|
|
|
ExtrusionPaths paths;
|
|
|
|
|
loop.clip_end(clip_length, &paths);
|
|
|
|
|
if (paths.empty()) return "";
|
|
|
|
|
|
|
|
|
|
// apply the small perimeter speed
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (is_perimeter(paths.front().role()) && loop.length() <= SMALL_PERIMETER_LENGTH && speed == -1)
|
|
|
|
|
speed = m_config.small_perimeter_speed.get_abs_value(m_config.perimeter_speed);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
|
|
|
|
|
// extrude along the path
|
|
|
|
|
std::string gcode;
|
2017-02-15 16:51:46 +00:00
|
|
|
|
for (ExtrusionPaths::iterator path = paths.begin(); path != paths.end(); ++path) {
|
2017-04-07 15:37:30 +00:00
|
|
|
|
// description += ExtrusionLoopRole2String(loop.loop_role());
|
2016-09-12 14:25:15 +00:00
|
|
|
|
// description += ExtrusionRole2String(path->role);
|
2017-02-15 16:51:46 +00:00
|
|
|
|
path->simplify(SCALED_RESOLUTION);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
gcode += this->_extrude(*path, description, speed);
|
2017-02-15 16:51:46 +00:00
|
|
|
|
}
|
2015-07-02 18:24:16 +00:00
|
|
|
|
|
|
|
|
|
// reset acceleration
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.set_acceleration(m_config.default_acceleration.value);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_wipe.enable)
|
|
|
|
|
m_wipe.path = paths.front().polyline; // TODO: don't limit wipe to last path
|
2015-07-02 18:24:16 +00:00
|
|
|
|
|
|
|
|
|
// make a little move inwards before leaving loop
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (paths.back().role() == erExternalPerimeter && m_layer != NULL && m_config.perimeters.value > 1) {
|
2015-07-02 18:24:16 +00:00
|
|
|
|
// detect angle between last and first segment
|
|
|
|
|
// the side depends on the original winding order of the polygon (left for contours, right for holes)
|
|
|
|
|
Point a = paths.front().polyline.points[1]; // second point
|
|
|
|
|
Point b = *(paths.back().polyline.points.end()-3); // second to last point
|
|
|
|
|
if (was_clockwise) {
|
|
|
|
|
// swap points
|
|
|
|
|
Point c = a; a = b; b = c;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
double angle = paths.front().first_point().ccw_angle(a, b) / 3;
|
|
|
|
|
|
|
|
|
|
// turn left if contour, turn right if hole
|
|
|
|
|
if (was_clockwise) angle *= -1;
|
|
|
|
|
|
|
|
|
|
// create the destination point along the first segment and rotate it
|
|
|
|
|
// we make sure we don't exceed the segment length because we don't know
|
|
|
|
|
// the rotation of the second segment so we might cross the object boundary
|
|
|
|
|
Line first_segment(
|
|
|
|
|
paths.front().polyline.points[0],
|
|
|
|
|
paths.front().polyline.points[1]
|
|
|
|
|
);
|
|
|
|
|
double distance = std::min(
|
|
|
|
|
scale_(EXTRUDER_CONFIG(nozzle_diameter)),
|
|
|
|
|
first_segment.length()
|
|
|
|
|
);
|
|
|
|
|
Point point = first_segment.point_at(distance);
|
|
|
|
|
point.rotate(angle, first_segment.a);
|
|
|
|
|
|
|
|
|
|
// generate the travel move
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.travel_to_xy(this->point_to_gcode(point), "move inwards before travel");
|
2015-07-02 18:24:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::extrude_multi_path(ExtrusionMultiPath multipath, std::string description, double speed)
|
2017-01-19 12:35:55 +00:00
|
|
|
|
{
|
|
|
|
|
// extrude along the path
|
|
|
|
|
std::string gcode;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
for (ExtrusionPath path : multipath.paths) {
|
2017-04-07 15:37:30 +00:00
|
|
|
|
// description += ExtrusionLoopRole2String(loop.loop_role());
|
2017-01-19 12:35:55 +00:00
|
|
|
|
// description += ExtrusionRole2String(path->role);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
path.simplify(SCALED_RESOLUTION);
|
|
|
|
|
gcode += this->_extrude(path, description, speed);
|
2017-02-15 16:51:46 +00:00
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_wipe.enable) {
|
|
|
|
|
m_wipe.path = std::move(multipath.paths.back().polyline); // TODO: don't limit wipe to last path
|
|
|
|
|
m_wipe.path.reverse();
|
2017-02-15 16:51:46 +00:00
|
|
|
|
}
|
2017-01-19 12:35:55 +00:00
|
|
|
|
// reset acceleration
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.set_acceleration(m_config.default_acceleration.value);
|
2017-01-19 12:35:55 +00:00
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-15 09:32:59 +00:00
|
|
|
|
std::string GCode::extrude_entity(const ExtrusionEntity &entity, std::string description, double speed, std::unique_ptr<EdgeGrid::Grid> *lower_layer_edge_grid)
|
2015-07-02 18:24:16 +00:00
|
|
|
|
{
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (const ExtrusionPath* path = dynamic_cast<const ExtrusionPath*>(&entity))
|
|
|
|
|
return this->extrude_path(*path, description, speed);
|
|
|
|
|
else if (const ExtrusionMultiPath* multipath = dynamic_cast<const ExtrusionMultiPath*>(&entity))
|
|
|
|
|
return this->extrude_multi_path(*multipath, description, speed);
|
|
|
|
|
else if (const ExtrusionLoop* loop = dynamic_cast<const ExtrusionLoop*>(&entity))
|
2017-05-15 09:32:59 +00:00
|
|
|
|
return this->extrude_loop(*loop, description, speed, lower_layer_edge_grid);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
else {
|
2015-07-02 18:24:16 +00:00
|
|
|
|
CONFESS("Invalid argument supplied to extrude()");
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::extrude_path(ExtrusionPath path, std::string description, double speed)
|
2015-07-02 18:24:16 +00:00
|
|
|
|
{
|
2017-04-07 15:37:30 +00:00
|
|
|
|
// description += ExtrusionRole2String(path.role());
|
2017-02-15 16:51:46 +00:00
|
|
|
|
path.simplify(SCALED_RESOLUTION);
|
2015-07-02 18:24:16 +00:00
|
|
|
|
std::string gcode = this->_extrude(path, description, speed);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_wipe.enable) {
|
|
|
|
|
m_wipe.path = std::move(path.polyline);
|
|
|
|
|
m_wipe.path.reverse();
|
2017-05-10 09:25:57 +00:00
|
|
|
|
}
|
2015-07-02 16:57:40 +00:00
|
|
|
|
// reset acceleration
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += m_writer.set_acceleration((unsigned int)floor(m_config.default_acceleration.value + 0.5));
|
2015-07-02 16:57:40 +00:00
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
// Extrude perimeters: Decide where to put seams (hide or align seams).
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::extrude_perimeters(const Print &print, const std::vector<ObjectByExtruder::Island::Region> &by_region, std::unique_ptr<EdgeGrid::Grid> &lower_layer_edge_grid)
|
2017-04-07 15:37:30 +00:00
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
for (const ObjectByExtruder::Island::Region ®ion : by_region) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_config.apply(print.regions[®ion - &by_region.front()]->config);
|
|
|
|
|
for (ExtrusionEntity *ee : region.perimeters.entities)
|
2017-05-15 09:32:59 +00:00
|
|
|
|
gcode += this->extrude_entity(*ee, "perimeter", -1., &lower_layer_edge_grid);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Chain the paths hierarchically by a greedy algorithm to minimize a travel distance.
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::extrude_infill(const Print &print, const std::vector<ObjectByExtruder::Island::Region> &by_region)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
for (const ObjectByExtruder::Island::Region ®ion : by_region) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_config.apply(print.regions[®ion - &by_region.front()]->config);
|
|
|
|
|
ExtrusionEntityCollection chained = region.infills.chained_path_from(m_last_pos, false);
|
|
|
|
|
for (ExtrusionEntity *fill : chained.entities) {
|
|
|
|
|
auto *eec = dynamic_cast<ExtrusionEntityCollection*>(fill);
|
|
|
|
|
if (eec) {
|
|
|
|
|
ExtrusionEntityCollection chained2 = eec->chained_path_from(m_last_pos, false);
|
|
|
|
|
for (ExtrusionEntity *ee : chained2.entities)
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += this->extrude_entity(*ee, "infill");
|
2017-05-03 16:28:22 +00:00
|
|
|
|
} else
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += this->extrude_entity(*fill, "infill");
|
2017-05-03 16:28:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::extrude_support(const ExtrusionEntityCollection &support_fills)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
|
|
|
|
if (! support_fills.entities.empty()) {
|
2017-04-07 15:37:30 +00:00
|
|
|
|
const char *support_label = "support material";
|
|
|
|
|
const char *support_interface_label = "support material interface";
|
2017-05-03 16:28:22 +00:00
|
|
|
|
const double support_speed = m_config.support_material_speed.value;
|
|
|
|
|
const double support_interface_speed = m_config.support_material_interface_speed.get_abs_value(support_speed);
|
|
|
|
|
for (const ExtrusionEntity *ee : support_fills.entities) {
|
2017-04-07 15:37:30 +00:00
|
|
|
|
ExtrusionRole role = ee->role();
|
|
|
|
|
assert(role == erSupportMaterial || role == erSupportMaterialInterface);
|
|
|
|
|
const char *label = (role == erSupportMaterial) ? support_label : support_interface_label;
|
|
|
|
|
const double speed = (role == erSupportMaterial) ? support_speed : support_interface_speed;
|
|
|
|
|
const ExtrusionPath *path = dynamic_cast<const ExtrusionPath*>(ee);
|
|
|
|
|
if (path)
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += this->extrude_path(*path, label, speed);
|
2017-04-07 15:37:30 +00:00
|
|
|
|
else {
|
|
|
|
|
const ExtrusionMultiPath *multipath = dynamic_cast<const ExtrusionMultiPath*>(ee);
|
|
|
|
|
assert(multipath != nullptr);
|
|
|
|
|
if (multipath)
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += this->extrude_multi_path(*multipath, label, speed);
|
2017-04-07 15:37:30 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::_extrude(const ExtrusionPath &path, std::string description, double speed)
|
2015-07-02 16:57:40 +00:00
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
|
|
|
|
|
|
|
|
|
// go to first point of extrusion path
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (!m_last_pos_defined || !m_last_pos.coincides_with(path.first_point())) {
|
2015-07-02 16:57:40 +00:00
|
|
|
|
gcode += this->travel_to(
|
|
|
|
|
path.first_point(),
|
2017-04-07 15:37:30 +00:00
|
|
|
|
path.role(),
|
2015-07-02 16:57:40 +00:00
|
|
|
|
"move to first " + description + " point"
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// compensate retraction
|
|
|
|
|
gcode += this->unretract();
|
|
|
|
|
|
|
|
|
|
// adjust acceleration
|
|
|
|
|
{
|
|
|
|
|
double acceleration;
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (this->on_first_layer() && m_config.first_layer_acceleration.value > 0) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
acceleration = m_config.first_layer_acceleration.value;
|
|
|
|
|
} else if (m_config.perimeter_acceleration.value > 0 && is_perimeter(path.role())) {
|
|
|
|
|
acceleration = m_config.perimeter_acceleration.value;
|
|
|
|
|
} else if (m_config.bridge_acceleration.value > 0 && is_bridge(path.role())) {
|
|
|
|
|
acceleration = m_config.bridge_acceleration.value;
|
|
|
|
|
} else if (m_config.infill_acceleration.value > 0 && is_infill(path.role())) {
|
|
|
|
|
acceleration = m_config.infill_acceleration.value;
|
2015-07-02 16:57:40 +00:00
|
|
|
|
} else {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
acceleration = m_config.default_acceleration.value;
|
2015-07-02 16:57:40 +00:00
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
gcode += m_writer.set_acceleration((unsigned int)floor(acceleration + 0.5));
|
2015-07-02 16:57:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// calculate extrusion length per distance unit
|
2017-05-03 16:28:22 +00:00
|
|
|
|
double e_per_mm = m_writer.extruder()->e_per_mm3 * path.mm3_per_mm;
|
|
|
|
|
if (m_writer.extrusion_axis().empty()) e_per_mm = 0;
|
2015-07-02 16:57:40 +00:00
|
|
|
|
|
|
|
|
|
// set speed
|
|
|
|
|
if (speed == -1) {
|
2017-04-07 15:37:30 +00:00
|
|
|
|
if (path.role() == erPerimeter) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("perimeter_speed");
|
2017-04-07 15:37:30 +00:00
|
|
|
|
} else if (path.role() == erExternalPerimeter) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("external_perimeter_speed");
|
2017-04-07 15:37:30 +00:00
|
|
|
|
} else if (path.role() == erOverhangPerimeter || path.role() == erBridgeInfill) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("bridge_speed");
|
2017-04-07 15:37:30 +00:00
|
|
|
|
} else if (path.role() == erInternalInfill) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("infill_speed");
|
2017-04-07 15:37:30 +00:00
|
|
|
|
} else if (path.role() == erSolidInfill) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("solid_infill_speed");
|
2017-04-07 15:37:30 +00:00
|
|
|
|
} else if (path.role() == erTopSolidInfill) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("top_solid_infill_speed");
|
2017-04-07 15:37:30 +00:00
|
|
|
|
} else if (path.role() == erGapFill) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("gap_fill_speed");
|
2015-07-02 16:57:40 +00:00
|
|
|
|
} else {
|
|
|
|
|
CONFESS("Invalid speed");
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (this->on_first_layer())
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_config.get_abs_value("first_layer_speed", speed);
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (m_volumetric_speed != 0. && speed == 0)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
speed = m_volumetric_speed / path.mm3_per_mm;
|
|
|
|
|
if (m_config.max_volumetric_speed.value > 0) {
|
2015-07-02 16:57:40 +00:00
|
|
|
|
// cap speed with max_volumetric_speed anyway (even if user is not using autospeed)
|
|
|
|
|
speed = std::min(
|
|
|
|
|
speed,
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_config.max_volumetric_speed.value / path.mm3_per_mm
|
2015-07-02 16:57:40 +00:00
|
|
|
|
);
|
|
|
|
|
}
|
2016-11-01 12:41:24 +00:00
|
|
|
|
if (EXTRUDER_CONFIG(filament_max_volumetric_speed) > 0) {
|
|
|
|
|
// cap speed with max_volumetric_speed anyway (even if user is not using autospeed)
|
|
|
|
|
speed = std::min(
|
|
|
|
|
speed,
|
|
|
|
|
EXTRUDER_CONFIG(filament_max_volumetric_speed) / path.mm3_per_mm
|
|
|
|
|
);
|
|
|
|
|
}
|
2015-07-02 16:57:40 +00:00
|
|
|
|
double F = speed * 60; // convert mm/sec to mm/min
|
|
|
|
|
|
|
|
|
|
// extrude arc or line
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_enable_extrusion_role_markers || m_enable_analyzer_markers) {
|
|
|
|
|
if (path.role() != m_last_extrusion_role) {
|
|
|
|
|
m_last_extrusion_role = path.role();
|
2016-09-12 14:25:15 +00:00
|
|
|
|
char buf[32];
|
2017-04-07 15:37:30 +00:00
|
|
|
|
sprintf(buf, ";_EXTRUSION_ROLE:%d\n", int(path.role()));
|
2016-09-12 14:25:15 +00:00
|
|
|
|
gcode += buf;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (is_bridge(path.role()) && m_enable_cooling_markers)
|
2015-07-02 16:57:40 +00:00
|
|
|
|
gcode += ";_BRIDGE_FAN_START\n";
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.set_speed(F, "", m_enable_cooling_markers ? ";_EXTRUDE_SET_SPEED" : "");
|
2015-07-02 17:14:55 +00:00
|
|
|
|
double path_length = 0;
|
2015-07-02 16:57:40 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
std::string comment = m_config.gcode_comments ? description : "";
|
2017-05-10 09:25:57 +00:00
|
|
|
|
for (const Line &line : path.polyline.lines()) {
|
|
|
|
|
const double line_length = line.length() * SCALING_FACTOR;
|
2015-07-02 17:14:55 +00:00
|
|
|
|
path_length += line_length;
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.extrude_to_xy(
|
2017-05-10 09:25:57 +00:00
|
|
|
|
this->point_to_gcode(line.b),
|
2015-07-02 17:14:55 +00:00
|
|
|
|
e_per_mm * line_length,
|
2017-05-10 09:25:57 +00:00
|
|
|
|
comment);
|
2015-07-02 16:57:40 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (is_bridge(path.role()) && m_enable_cooling_markers)
|
2015-07-02 16:57:40 +00:00
|
|
|
|
gcode += ";_BRIDGE_FAN_END\n";
|
2015-07-02 17:14:55 +00:00
|
|
|
|
|
2015-07-02 16:57:40 +00:00
|
|
|
|
this->set_last_pos(path.last_point());
|
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_config.cooling)
|
2017-05-16 11:45:28 +00:00
|
|
|
|
m_elapsed_time += path_length / F * 60.f;
|
2015-07-02 16:57:40 +00:00
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-02 13:12:04 +00:00
|
|
|
|
// This method accepts &point in print coordinates.
|
|
|
|
|
std::string
|
|
|
|
|
GCode::travel_to(const Point &point, ExtrusionRole role, std::string comment)
|
|
|
|
|
{
|
|
|
|
|
/* Define the travel move as a line between current position and the taget point.
|
|
|
|
|
This is expressed in print coordinates, so it will need to be translated by
|
2015-07-02 16:57:40 +00:00
|
|
|
|
this->origin in order to get G-code coordinates. */
|
2015-07-02 13:12:04 +00:00
|
|
|
|
Polyline travel;
|
|
|
|
|
travel.append(this->last_pos());
|
|
|
|
|
travel.append(point);
|
|
|
|
|
|
|
|
|
|
// check whether a straight travel move would need retraction
|
|
|
|
|
bool needs_retraction = this->needs_retraction(travel, role);
|
|
|
|
|
|
|
|
|
|
// if a retraction would be needed, try to use avoid_crossing_perimeters to plan a
|
|
|
|
|
// multi-hop travel path inside the configuration space
|
|
|
|
|
if (needs_retraction
|
2017-05-03 16:28:22 +00:00
|
|
|
|
&& m_config.avoid_crossing_perimeters
|
|
|
|
|
&& ! m_avoid_crossing_perimeters.disable_once) {
|
|
|
|
|
travel = m_avoid_crossing_perimeters.travel_to(*this, point);
|
2015-07-02 13:12:04 +00:00
|
|
|
|
|
|
|
|
|
// check again whether the new travel path still needs a retraction
|
|
|
|
|
needs_retraction = this->needs_retraction(travel, role);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
//if (needs_retraction && m_layer_index > 1) exit(0);
|
2015-07-02 13:12:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Re-allow avoid_crossing_perimeters for the next travel moves
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_avoid_crossing_perimeters.disable_once = false;
|
|
|
|
|
m_avoid_crossing_perimeters.use_external_mp_once = false;
|
2015-07-02 13:12:04 +00:00
|
|
|
|
|
|
|
|
|
// generate G-code for the travel move
|
|
|
|
|
std::string gcode;
|
2017-03-29 15:45:38 +00:00
|
|
|
|
if (needs_retraction)
|
|
|
|
|
gcode += this->retract();
|
|
|
|
|
else
|
|
|
|
|
// Reset the wipe path when traveling, so one would not wipe along an old path.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_wipe.reset_path();
|
2015-07-02 13:12:04 +00:00
|
|
|
|
|
|
|
|
|
// use G1 because we rely on paths being straight (G0 may make round paths)
|
|
|
|
|
Lines lines = travel.lines();
|
2016-11-22 21:26:08 +00:00
|
|
|
|
for (Lines::const_iterator line = lines.begin(); line != lines.end(); ++line)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.travel_to_xy(this->point_to_gcode(line->b), comment);
|
2016-11-22 21:26:08 +00:00
|
|
|
|
|
|
|
|
|
/* While this makes the estimate more accurate, CoolingBuffer calculates the slowdown
|
|
|
|
|
factor on the whole elapsed time but only alters non-travel moves, thus the resulting
|
|
|
|
|
time is still shorter than the configured threshold. We could create a new
|
|
|
|
|
elapsed_travel_time but we would still need to account for bridges, retractions, wipe etc.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_config.cooling)
|
|
|
|
|
m_elapsed_time += unscale(travel.length()) / m_config.get_abs_value("travel_speed");
|
2016-11-22 21:26:08 +00:00
|
|
|
|
*/
|
2016-04-12 16:04:49 +00:00
|
|
|
|
|
2015-07-02 13:12:04 +00:00
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-01 21:14:40 +00:00
|
|
|
|
bool
|
|
|
|
|
GCode::needs_retraction(const Polyline &travel, ExtrusionRole role)
|
|
|
|
|
{
|
2015-07-02 12:31:21 +00:00
|
|
|
|
if (travel.length() < scale_(EXTRUDER_CONFIG(retract_before_travel))) {
|
2015-07-01 21:14:40 +00:00
|
|
|
|
// skip retraction if the move is shorter than the configured threshold
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (role == erSupportMaterial) {
|
2017-05-03 16:28:22 +00:00
|
|
|
|
const SupportLayer* support_layer = dynamic_cast<const SupportLayer*>(m_layer);
|
2017-03-07 12:03:14 +00:00
|
|
|
|
//FIXME support_layer->support_islands.contains should use some search structure!
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (support_layer != NULL && support_layer->support_islands.contains(travel))
|
2015-07-01 21:14:40 +00:00
|
|
|
|
// skip retraction if this is a travel move inside a support material island
|
2017-05-10 09:25:57 +00:00
|
|
|
|
//FIXME not retracting over a long path may cause oozing, which in turn may result in missing material
|
|
|
|
|
// at the end of the extrusion path!
|
2015-07-01 21:14:40 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2017-05-10 09:25:57 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_config.only_retract_when_crossing_perimeters && m_layer != nullptr) {
|
|
|
|
|
if (m_config.fill_density.value > 0
|
|
|
|
|
&& m_layer->any_internal_region_slice_contains(travel)) {
|
2015-07-01 21:14:40 +00:00
|
|
|
|
/* skip retraction if travel is contained in an internal slice *and*
|
|
|
|
|
internal infill is enabled (so that stringing is entirely not visible) */
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// retract if only_retract_when_crossing_perimeters is disabled or doesn't apply
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-01 21:00:52 +00:00
|
|
|
|
std::string
|
|
|
|
|
GCode::retract(bool toolchange)
|
|
|
|
|
{
|
|
|
|
|
std::string gcode;
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (m_writer.extruder() == nullptr)
|
2015-07-01 21:00:52 +00:00
|
|
|
|
return gcode;
|
|
|
|
|
|
|
|
|
|
// wipe (if it's enabled for this extruder and we have a stored wipe path)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (EXTRUDER_CONFIG(wipe) && m_wipe.has_path())
|
|
|
|
|
gcode += m_wipe.wipe(*this, toolchange);
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
/* The parent class will decide whether we need to perform an actual retraction
|
|
|
|
|
(the extruder might be already retracted fully or partially). We call these
|
|
|
|
|
methods even if we performed wipe, since this will ensure the entire retraction
|
|
|
|
|
length is honored in case wipe path was too short. */
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += toolchange ? m_writer.retract_for_toolchange() : m_writer.retract();
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += m_writer.reset_e();
|
|
|
|
|
if (m_writer.extruder()->retract_length() > 0 || m_config.use_firmware_retraction)
|
|
|
|
|
gcode += m_writer.lift();
|
2015-07-01 21:00:52 +00:00
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-10 09:25:57 +00:00
|
|
|
|
std::string GCode::set_extruder(unsigned int extruder_id)
|
2015-07-02 13:02:20 +00:00
|
|
|
|
{
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_placeholder_parser.set("current_extruder", extruder_id);
|
|
|
|
|
if (!m_writer.need_toolchange(extruder_id))
|
2015-07-02 13:02:20 +00:00
|
|
|
|
return "";
|
|
|
|
|
|
|
|
|
|
// if we are running a single-extruder setup, just set the extruder and return nothing
|
2017-05-10 09:25:57 +00:00
|
|
|
|
if (!m_writer.multiple_extruders)
|
2017-05-03 16:28:22 +00:00
|
|
|
|
return m_writer.toolchange(extruder_id);
|
2015-07-02 13:02:20 +00:00
|
|
|
|
|
|
|
|
|
// prepend retraction on the current extruder
|
|
|
|
|
std::string gcode = this->retract(true);
|
2017-03-29 15:45:38 +00:00
|
|
|
|
|
|
|
|
|
// Always reset the extrusion path, even if the tool change retract is set to zero.
|
2017-05-03 16:28:22 +00:00
|
|
|
|
m_wipe.reset_path();
|
2015-07-02 13:02:20 +00:00
|
|
|
|
|
|
|
|
|
// append custom toolchange G-code
|
2017-05-03 16:28:22 +00:00
|
|
|
|
if (m_writer.extruder() != NULL && !m_config.toolchange_gcode.value.empty()) {
|
|
|
|
|
PlaceholderParser pp = m_placeholder_parser;
|
|
|
|
|
pp.set("previous_extruder", m_writer.extruder()->id);
|
2015-07-02 13:02:20 +00:00
|
|
|
|
pp.set("next_extruder", extruder_id);
|
2017-05-03 16:28:22 +00:00
|
|
|
|
gcode += pp.process(m_config.toolchange_gcode.value) + '\n';
|
2015-07-02 13:02:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-16 11:45:28 +00:00
|
|
|
|
if (m_wipe_tower) {
|
|
|
|
|
assert(! m_wipe_tower->finished());
|
|
|
|
|
if (! m_wipe_tower->finished())
|
|
|
|
|
gcode += this->wipe_tower_tool_change(extruder_id);
|
|
|
|
|
} else {
|
|
|
|
|
// if ooze prevention is enabled, park current extruder in the nearest
|
|
|
|
|
// standby point and set it to the standby temperature
|
|
|
|
|
if (m_ooze_prevention.enable && m_writer.extruder() != NULL)
|
|
|
|
|
gcode += m_ooze_prevention.pre_toolchange(*this);
|
|
|
|
|
// append the toolchange command
|
|
|
|
|
gcode += m_writer.toolchange(extruder_id);
|
|
|
|
|
// set the new extruder to the operating temperature
|
|
|
|
|
if (m_ooze_prevention.enable)
|
|
|
|
|
gcode += m_ooze_prevention.post_toolchange(*this);
|
|
|
|
|
}
|
2015-07-02 13:02:20 +00:00
|
|
|
|
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-16 11:45:28 +00:00
|
|
|
|
std::string GCode::wipe_tower_tool_change(int extruder_id)
|
|
|
|
|
{
|
|
|
|
|
// Move over the wipe tower.
|
|
|
|
|
std::string gcode = m_writer.travel_to_xy(Pointf3(m_wipe_tower->position().x, m_wipe_tower->position().y));
|
|
|
|
|
gcode += m_writer.unlift();
|
|
|
|
|
// Let the tool change be executed by the wipe tower class.
|
|
|
|
|
std::pair<std::string, WipeTower::xy> code_and_pos = m_wipe_tower->tool_change(extruder_id);
|
|
|
|
|
// Inform the G-code writer about the changes done behind its back.
|
|
|
|
|
gcode += code_and_pos.first;
|
|
|
|
|
// A phony move to the end position at the wipe tower.
|
|
|
|
|
m_writer.travel_to_xy(Pointf(code_and_pos.second.x, code_and_pos.second.y));
|
|
|
|
|
return gcode;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-01 21:00:52 +00:00
|
|
|
|
// convert a model-space scaled point into G-code coordinates
|
2017-05-03 16:28:22 +00:00
|
|
|
|
Pointf GCode::point_to_gcode(const Point &point) const
|
2015-07-01 19:47:17 +00:00
|
|
|
|
{
|
2015-07-02 12:31:21 +00:00
|
|
|
|
Pointf extruder_offset = EXTRUDER_CONFIG(extruder_offset);
|
2015-07-01 21:00:52 +00:00
|
|
|
|
return Pointf(
|
2017-05-03 16:28:22 +00:00
|
|
|
|
unscale(point.x) + m_origin.x - extruder_offset.x,
|
|
|
|
|
unscale(point.y) + m_origin.y - extruder_offset.y
|
2015-07-01 21:00:52 +00:00
|
|
|
|
);
|
2015-07-01 19:47:17 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-07-01 18:14:05 +00:00
|
|
|
|
}
|