123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271 |
- #include <catch2/catch.hpp>
- #include "libslic3r/libslic3r.h"
- #include "libslic3r/GCodeReader.hpp"
- #include "test_data.hpp"
- #include <algorithm>
- #include <boost/regex.hpp>
- using namespace Slic3r;
- using namespace Slic3r::Test;
- boost::regex perimeters_regex("G1 X[-0-9.]* Y[-0-9.]* E[-0-9.]* ; perimeter");
- boost::regex infill_regex("G1 X[-0-9.]* Y[-0-9.]* E[-0-9.]* ; infill");
- boost::regex skirt_regex("G1 X[-0-9.]* Y[-0-9.]* E[-0-9.]* ; skirt");
- SCENARIO( "PrintGCode basic functionality", "[PrintGCode]") {
- GIVEN("A default configuration and a print test object") {
- WHEN("the output is executed with no support material") {
- Slic3r::Print print;
- Slic3r::Model model;
- Slic3r::Test::init_print({TestMesh::cube_20x20x20}, print, model, {
- { "layer_height", 0.2 },
- { "first_layer_height", 0.2 },
- { "first_layer_extrusion_width", 0 },
- { "gcode_comments", true },
- { "start_gcode", "" }
- });
- std::string gcode = Slic3r::Test::gcode(print);
- THEN("Some text output is generated.") {
- REQUIRE(gcode.size() > 0);
- }
- THEN("Exported text contains slic3r version") {
- REQUIRE(gcode.find(SLIC3R_VERSION) != std::string::npos);
- }
- //THEN("Exported text contains git commit id") {
- // REQUIRE(gcode.find("; Git Commit") != std::string::npos);
- // REQUIRE(gcode.find(SLIC3R_BUILD_ID) != std::string::npos);
- //}
- THEN("Exported text contains extrusion statistics.") {
- REQUIRE(gcode.find("; external perimeters extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; perimeters extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; solid infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; top infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; support material extrusion width") == std::string::npos);
- REQUIRE(gcode.find("; first layer extrusion width") == std::string::npos);
- }
- THEN("Exported text does not contain cooling markers (they were consumed)") {
- REQUIRE(gcode.find(";_EXTRUDE_SET_SPEED") == std::string::npos);
- }
- THEN("GCode preamble is emitted.") {
- REQUIRE(gcode.find("G21 ; set units to millimeters") != std::string::npos);
- }
- THEN("Config options emitted for print config, default region config, default object config") {
- REQUIRE(gcode.find("; first_layer_temperature") != std::string::npos);
- REQUIRE(gcode.find("; layer_height") != std::string::npos);
- REQUIRE(gcode.find("; fill_density") != std::string::npos);
- }
- THEN("Infill is emitted.") {
- boost::smatch has_match;
- REQUIRE(boost::regex_search(gcode, has_match, infill_regex));
- }
- THEN("Perimeters are emitted.") {
- boost::smatch has_match;
- REQUIRE(boost::regex_search(gcode, has_match, perimeters_regex));
- }
- THEN("Skirt is emitted.") {
- boost::smatch has_match;
- REQUIRE(boost::regex_search(gcode, has_match, skirt_regex));
- }
- THEN("final Z height is 20mm") {
- double final_z = 0.0;
- GCodeReader reader;
- reader.apply_config(print.config());
- reader.parse_buffer(gcode, [&final_z] (GCodeReader& self, const GCodeReader::GCodeLine& line) {
- final_z = std::max<double>(final_z, static_cast<double>(self.z())); // record the highest Z point we reach
- });
- REQUIRE(final_z == Approx(20.));
- }
- }
- WHEN("output is executed with complete objects and two differently-sized meshes") {
- Slic3r::Print print;
- Slic3r::Model model;
- Slic3r::Test::init_print({TestMesh::cube_20x20x20,TestMesh::cube_20x20x20}, print, model, {
- { "first_layer_extrusion_width", 0 },
- { "first_layer_height", 0.3 },
- { "layer_height", 0.2 },
- { "support_material", false },
- { "raft_layers", 0 },
- { "complete_objects", true },
- { "gcode_comments", true },
- { "between_objects_gcode", "; between-object-gcode" }
- });
- std::string gcode = Slic3r::Test::gcode(print);
- THEN("Some text output is generated.") {
- REQUIRE(gcode.size() > 0);
- }
- THEN("Infill is emitted.") {
- boost::smatch has_match;
- REQUIRE(boost::regex_search(gcode, has_match, infill_regex));
- }
- THEN("Perimeters are emitted.") {
- boost::smatch has_match;
- REQUIRE(boost::regex_search(gcode, has_match, perimeters_regex));
- }
- THEN("Skirt is emitted.") {
- boost::smatch has_match;
- REQUIRE(boost::regex_search(gcode, has_match, skirt_regex));
- }
- THEN("Between-object-gcode is emitted.") {
- REQUIRE(gcode.find("; between-object-gcode") != std::string::npos);
- }
- THEN("final Z height is 20.1mm") {
- double final_z = 0.0;
- GCodeReader reader;
- reader.apply_config(print.config());
- reader.parse_buffer(gcode, [&final_z] (GCodeReader& self, const GCodeReader::GCodeLine& line) {
- final_z = std::max(final_z, static_cast<double>(self.z())); // record the highest Z point we reach
- });
- REQUIRE(final_z == Approx(20.1));
- }
- THEN("Z height resets on object change") {
- double final_z = 0.0;
- bool reset = false;
- GCodeReader reader;
- reader.apply_config(print.config());
- reader.parse_buffer(gcode, [&final_z, &reset] (GCodeReader& self, const GCodeReader::GCodeLine& line) {
- if (final_z > 0 && std::abs(self.z() - 0.3) < 0.01 ) { // saw higher Z before this, now it's lower
- reset = true;
- } else {
- final_z = std::max(final_z, static_cast<double>(self.z())); // record the highest Z point we reach
- }
- });
- REQUIRE(reset == true);
- }
- THEN("Shorter object is printed before taller object.") {
- double final_z = 0.0;
- bool reset = false;
- GCodeReader reader;
- reader.apply_config(print.config());
- reader.parse_buffer(gcode, [&final_z, &reset] (GCodeReader& self, const GCodeReader::GCodeLine& line) {
- if (final_z > 0 && std::abs(self.z() - 0.3) < 0.01 ) {
- reset = (final_z > 20.0);
- } else {
- final_z = std::max(final_z, static_cast<double>(self.z())); // record the highest Z point we reach
- }
- });
- REQUIRE(reset == true);
- }
- }
- WHEN("the output is executed with support material") {
- std::string gcode = ::Test::slice({TestMesh::cube_20x20x20}, {
- { "first_layer_extrusion_width", 0 },
- { "support_material", true },
- { "raft_layers", 3 },
- { "gcode_comments", true }
- });
- THEN("Some text output is generated.") {
- REQUIRE(gcode.size() > 0);
- }
- THEN("Exported text contains extrusion statistics.") {
- REQUIRE(gcode.find("; external perimeters extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; perimeters extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; solid infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; top infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; support material extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; first layer extrusion width") == std::string::npos);
- }
- THEN("Raft is emitted.") {
- REQUIRE(gcode.find("; raft") != std::string::npos);
- }
- }
- WHEN("the output is executed with a separate first layer extrusion width") {
- std::string gcode = ::Test::slice({ TestMesh::cube_20x20x20 }, {
- { "first_layer_extrusion_width", "0.5" }
- });
- THEN("Some text output is generated.") {
- REQUIRE(gcode.size() > 0);
- }
- THEN("Exported text contains extrusion statistics.") {
- REQUIRE(gcode.find("; external perimeters extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; perimeters extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; solid infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; top infill extrusion width") != std::string::npos);
- REQUIRE(gcode.find("; support material extrusion width") == std::string::npos);
- REQUIRE(gcode.find("; first layer extrusion width") != std::string::npos);
- }
- }
- WHEN("Cooling is enabled and the fan is disabled.") {
- std::string gcode = ::Test::slice({ TestMesh::cube_20x20x20 }, {
- { "cooling", true },
- { "disable_fan_first_layers", 5 }
- });
- THEN("GCode to disable fan is emitted."){
- REQUIRE(gcode.find("M107") != std::string::npos);
- }
- }
- WHEN("end_gcode exists with layer_num and layer_z") {
- std::string gcode = ::Test::slice({ TestMesh::cube_20x20x20 }, {
- { "end_gcode", "; Layer_num [layer_num]\n; Layer_z [layer_z]" },
- { "layer_height", 0.1 },
- { "first_layer_height", 0.1 }
- });
- THEN("layer_num and layer_z are processed in the end gcode") {
- REQUIRE(gcode.find("; Layer_num 199") != std::string::npos);
- REQUIRE(gcode.find("; Layer_z 20") != std::string::npos);
- }
- }
- WHEN("current_extruder exists in start_gcode") {
- {
- std::string gcode = ::Test::slice({ TestMesh::cube_20x20x20 }, {
- { "start_gcode", "; Extruder [current_extruder]" }
- });
- THEN("current_extruder is processed in the start gcode and set for first extruder") {
- REQUIRE(gcode.find("; Extruder 0") != std::string::npos);
- }
- }
- {
- DynamicPrintConfig config = DynamicPrintConfig::full_print_config();
- config.set_num_extruders(4);
- config.set_deserialize_strict({
- { "start_gcode", "; Extruder [current_extruder]" },
- { "infill_extruder", 2 },
- { "solid_infill_extruder", 2 },
- { "perimeter_extruder", 2 },
- { "support_material_extruder", 2 },
- { "support_material_interface_extruder", 2 }
- });
- std::string gcode = Slic3r::Test::slice({TestMesh::cube_20x20x20}, config);
- THEN("current_extruder is processed in the start gcode and set for second extruder") {
- REQUIRE(gcode.find("; Extruder 1") != std::string::npos);
- }
- }
- }
- WHEN("layer_num represents the layer's index from z=0") {
- std::string gcode = ::Test::slice({ TestMesh::cube_20x20x20, TestMesh::cube_20x20x20 }, {
- { "complete_objects", true },
- { "gcode_comments", true },
- { "layer_gcode", ";Layer:[layer_num] ([layer_z] mm)" },
- { "layer_height", 0.1 },
- { "first_layer_height", 0.1 }
- });
- // End of the 1st object.
- std::string token = ";Layer:199 ";
- size_t pos = gcode.find(token);
- THEN("First and second object last layer is emitted") {
- // First object
- REQUIRE(pos != std::string::npos);
- pos += token.size();
- REQUIRE(pos < gcode.size());
- double z = 0;
- REQUIRE((sscanf(gcode.data() + pos, "(%lf mm)", &z) == 1));
- REQUIRE(z == Approx(20.));
- // Second object
- pos = gcode.find(";Layer:399 ", pos);
- REQUIRE(pos != std::string::npos);
- pos += token.size();
- REQUIRE(pos < gcode.size());
- REQUIRE((sscanf(gcode.data() + pos, "(%lf mm)", &z) == 1));
- REQUIRE(z == Approx(20.));
- }
- }
- }
- }
|