Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/prusa3d/PrusaSlicer.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'src/slic3r/GUI/Preset.cpp')
-rw-r--r--src/slic3r/GUI/Preset.cpp1451
1 files changed, 0 insertions, 1451 deletions
diff --git a/src/slic3r/GUI/Preset.cpp b/src/slic3r/GUI/Preset.cpp
deleted file mode 100644
index 883dc438a..000000000
--- a/src/slic3r/GUI/Preset.cpp
+++ /dev/null
@@ -1,1451 +0,0 @@
-#include <cassert>
-
-#include "Preset.hpp"
-#include "AppConfig.hpp"
-#include "I18N.hpp"
-
-#ifdef _MSC_VER
- #define WIN32_LEAN_AND_MEAN
- #define NOMINMAX
- #include <Windows.h>
-#endif /* _MSC_VER */
-
-#include <algorithm>
-#include <fstream>
-#include <stdexcept>
-#include <unordered_map>
-#include <boost/format.hpp>
-#include <boost/filesystem.hpp>
-#include <boost/filesystem/fstream.hpp>
-#include <boost/algorithm/string/predicate.hpp>
-
-#include <boost/nowide/cenv.hpp>
-#include <boost/nowide/convert.hpp>
-#include <boost/nowide/cstdio.hpp>
-#include <boost/nowide/fstream.hpp>
-#include <boost/property_tree/ini_parser.hpp>
-#include <boost/property_tree/ptree.hpp>
-#include <boost/locale.hpp>
-#include <boost/log/trivial.hpp>
-
-#include "libslic3r/libslic3r.h"
-#include "libslic3r/Utils.hpp"
-#include "libslic3r/PlaceholderParser.hpp"
-
-using boost::property_tree::ptree;
-
-namespace Slic3r {
-
-ConfigFileType guess_config_file_type(const ptree &tree)
-{
- size_t app_config = 0;
- size_t bundle = 0;
- size_t config = 0;
- for (const ptree::value_type &v : tree) {
- if (v.second.empty()) {
- if (v.first == "background_processing" ||
- v.first == "last_output_path" ||
- v.first == "no_controller" ||
- v.first == "no_defaults")
- ++ app_config;
- else if (v.first == "nozzle_diameter" ||
- v.first == "filament_diameter")
- ++ config;
- } else if (boost::algorithm::starts_with(v.first, "print:") ||
- boost::algorithm::starts_with(v.first, "filament:") ||
- boost::algorithm::starts_with(v.first, "printer:") ||
- v.first == "settings")
- ++ bundle;
- else if (v.first == "presets") {
- ++ app_config;
- ++ bundle;
- } else if (v.first == "recent") {
- for (auto &kvp : v.second)
- if (kvp.first == "config_directory" || kvp.first == "skein_directory")
- ++ app_config;
- }
- }
- return (app_config > bundle && app_config > config) ? CONFIG_FILE_TYPE_APP_CONFIG :
- (bundle > config) ? CONFIG_FILE_TYPE_CONFIG_BUNDLE : CONFIG_FILE_TYPE_CONFIG;
-}
-
-
-VendorProfile VendorProfile::from_ini(const boost::filesystem::path &path, bool load_all)
-{
- ptree tree;
- boost::filesystem::ifstream ifs(path);
- boost::property_tree::read_ini(ifs, tree);
- return VendorProfile::from_ini(tree, path, load_all);
-}
-
-static const std::unordered_map<std::string, std::string> pre_family_model_map {{
- { "MK3", "MK3" },
- { "MK3MMU2", "MK3" },
- { "MK2.5", "MK2.5" },
- { "MK2.5MMU2", "MK2.5" },
- { "MK2S", "MK2" },
- { "MK2SMM", "MK2" },
- { "SL1", "SL1" },
-}};
-
-VendorProfile VendorProfile::from_ini(const ptree &tree, const boost::filesystem::path &path, bool load_all)
-{
- static const std::string printer_model_key = "printer_model:";
- static const std::string filaments_section = "default_filaments";
- static const std::string materials_section = "default_sla_materials";
-
- const std::string id = path.stem().string();
-
- if (! boost::filesystem::exists(path)) {
- throw std::runtime_error((boost::format("Cannot load Vendor Config Bundle `%1%`: File not found: `%2%`.") % id % path).str());
- }
-
- VendorProfile res(id);
-
- // Helper to get compulsory fields
- auto get_or_throw = [&](const ptree &tree, const std::string &key) -> ptree::const_assoc_iterator
- {
- auto res = tree.find(key);
- if (res == tree.not_found()) {
- throw std::runtime_error((boost::format("Vendor Config Bundle `%1%` is not valid: Missing secion or key: `%2%`.") % id % key).str());
- }
- return res;
- };
-
- // Load the header
- const auto &vendor_section = get_or_throw(tree, "vendor")->second;
- res.name = get_or_throw(vendor_section, "name")->second.data();
-
- auto config_version_str = get_or_throw(vendor_section, "config_version")->second.data();
- auto config_version = Semver::parse(config_version_str);
- if (! config_version) {
- throw std::runtime_error((boost::format("Vendor Config Bundle `%1%` is not valid: Cannot parse config_version: `%2%`.") % id % config_version_str).str());
- } else {
- res.config_version = std::move(*config_version);
- }
-
- // Load URLs
- const auto config_update_url = vendor_section.find("config_update_url");
- if (config_update_url != vendor_section.not_found()) {
- res.config_update_url = config_update_url->second.data();
- }
-
- const auto changelog_url = vendor_section.find("changelog_url");
- if (changelog_url != vendor_section.not_found()) {
- res.changelog_url = changelog_url->second.data();
- }
-
- if (! load_all) {
- return res;
- }
-
- // Load printer models
- for (auto &section : tree) {
- if (boost::starts_with(section.first, printer_model_key)) {
- VendorProfile::PrinterModel model;
- model.id = section.first.substr(printer_model_key.size());
- model.name = section.second.get<std::string>("name", model.id);
-
- const char *technology_fallback = boost::algorithm::starts_with(model.id, "SL") ? "SLA" : "FFF";
-
- auto technology_field = section.second.get<std::string>("technology", technology_fallback);
- if (! ConfigOptionEnum<PrinterTechnology>::from_string(technology_field, model.technology)) {
- BOOST_LOG_TRIVIAL(error) << boost::format("Vendor bundle: `%1%`: Invalid printer technology field: `%2%`") % id % technology_field;
- model.technology = ptFFF;
- }
-
- model.family = section.second.get<std::string>("family", std::string());
- if (model.family.empty() && res.name == "Prusa Research") {
- // If no family is specified, it can be inferred for known printers
- const auto from_pre_map = pre_family_model_map.find(model.id);
- if (from_pre_map != pre_family_model_map.end()) { model.family = from_pre_map->second; }
- }
-#if 0
- // Remove SLA printers from the initial alpha.
- if (model.technology == ptSLA)
- continue;
-#endif
- section.second.get<std::string>("variants", "");
- const auto variants_field = section.second.get<std::string>("variants", "");
- std::vector<std::string> variants;
- if (Slic3r::unescape_strings_cstyle(variants_field, variants)) {
- for (const std::string &variant_name : variants) {
- if (model.variant(variant_name) == nullptr)
- model.variants.emplace_back(VendorProfile::PrinterVariant(variant_name));
- }
- } else {
- BOOST_LOG_TRIVIAL(error) << boost::format("Vendor bundle: `%1%`: Malformed variants field: `%2%`") % id % variants_field;
- }
- auto default_materials_field = section.second.get<std::string>("default_materials", "");
- if (default_materials_field.empty())
- default_materials_field = section.second.get<std::string>("default_filaments", "");
- if (Slic3r::unescape_strings_cstyle(default_materials_field, model.default_materials)) {
- Slic3r::sort_remove_duplicates(model.default_materials);
- if (! model.default_materials.empty() && model.default_materials.front().empty())
- // An empty material was inserted into the list of default materials. Remove it.
- model.default_materials.erase(model.default_materials.begin());
- } else {
- BOOST_LOG_TRIVIAL(error) << boost::format("Vendor bundle: `%1%`: Malformed default_materials field: `%2%`") % id % default_materials_field;
- }
- model.bed_model = section.second.get<std::string>("bed_model", "");
- model.bed_texture = section.second.get<std::string>("bed_texture", "");
- if (! model.id.empty() && ! model.variants.empty())
- res.models.push_back(std::move(model));
- }
- }
-
- // Load filaments and sla materials to be installed by default
- const auto filaments = tree.find(filaments_section);
- if (filaments != tree.not_found()) {
- for (auto &pair : filaments->second) {
- if (pair.second.data() == "1") {
- res.default_filaments.insert(pair.first);
- }
- }
- }
- const auto materials = tree.find(materials_section);
- if (materials != tree.not_found()) {
- for (auto &pair : materials->second) {
- if (pair.second.data() == "1") {
- res.default_sla_materials.insert(pair.first);
- }
- }
- }
-
- return res;
-}
-
-std::vector<std::string> VendorProfile::families() const
-{
- std::vector<std::string> res;
- unsigned num_familiies = 0;
-
- for (auto &model : models) {
- if (std::find(res.begin(), res.end(), model.family) == res.end()) {
- res.push_back(model.family);
- num_familiies++;
- }
- }
-
- return res;
-}
-
-// Suffix to be added to a modified preset name in the combo box.
-static std::string g_suffix_modified = " (modified)";
-const std::string& Preset::suffix_modified()
-{
- return g_suffix_modified;
-}
-
-void Preset::update_suffix_modified()
-{
- g_suffix_modified = (" (" + _(L("modified")) + ")").ToUTF8().data();
-}
-// Remove an optional "(modified)" suffix from a name.
-// This converts a UI name to a unique preset identifier.
-std::string Preset::remove_suffix_modified(const std::string &name)
-{
- return boost::algorithm::ends_with(name, g_suffix_modified) ?
- name.substr(0, name.size() - g_suffix_modified.size()) :
- name;
-}
-
-// Update new extruder fields at the printer profile.
-void Preset::normalize(DynamicPrintConfig &config)
-{
- auto *nozzle_diameter = dynamic_cast<const ConfigOptionFloats*>(config.option("nozzle_diameter"));
- if (nozzle_diameter != nullptr)
- // Loaded the FFF Printer settings. Verify, that all extruder dependent values have enough values.
- config.set_num_extruders((unsigned int)nozzle_diameter->values.size());
- if (config.option("filament_diameter") != nullptr) {
- // This config contains single or multiple filament presets.
- // Ensure that the filament preset vector options contain the correct number of values.
- size_t n = (nozzle_diameter == nullptr) ? 1 : nozzle_diameter->values.size();
- const auto &defaults = FullPrintConfig::defaults();
- for (const std::string &key : Preset::filament_options()) {
- if (key == "compatible_prints" || key == "compatible_printers")
- continue;
- auto *opt = config.option(key, false);
- /*assert(opt != nullptr);
- assert(opt->is_vector());*/
- if (opt != nullptr && opt->is_vector())
- static_cast<ConfigOptionVectorBase*>(opt)->resize(n, defaults.option(key));
- }
- // The following keys are mandatory for the UI, but they are not part of FullPrintConfig, therefore they are handled separately.
- for (const std::string &key : { "filament_settings_id" }) {
- auto *opt = config.option(key, false);
- assert(opt == nullptr || opt->type() == coStrings);
- if (opt != nullptr && opt->type() == coStrings)
- static_cast<ConfigOptionStrings*>(opt)->values.resize(n, std::string());
- }
- }
-}
-
-std::string Preset::remove_invalid_keys(DynamicPrintConfig &config, const DynamicPrintConfig &default_config)
-{
- std::string incorrect_keys;
- for (const std::string &key : config.keys())
- if (! default_config.has(key)) {
- if (incorrect_keys.empty())
- incorrect_keys = key;
- else {
- incorrect_keys += ", ";
- incorrect_keys += key;
- }
- config.erase(key);
- }
- return incorrect_keys;
-}
-
-void Preset::save()
-{
- this->config.save(this->file);
-}
-
-// Return a label of this preset, consisting of a name and a "(modified)" suffix, if this preset is dirty.
-std::string Preset::label() const
-{
- return this->name + (this->is_dirty ? g_suffix_modified : "");
-}
-
-bool is_compatible_with_print(const PresetWithVendorProfile &preset, const PresetWithVendorProfile &active_print, const PresetWithVendorProfile &active_printer)
-{
- if (preset.vendor != nullptr && preset.vendor != active_printer.vendor)
- // The current profile has a vendor assigned and it is different from the active print's vendor.
- return false;
- auto &condition = preset.preset.compatible_prints_condition();
- auto *compatible_prints = dynamic_cast<const ConfigOptionStrings*>(preset.preset.config.option("compatible_prints"));
- bool has_compatible_prints = compatible_prints != nullptr && ! compatible_prints->values.empty();
- if (! has_compatible_prints && ! condition.empty()) {
- try {
- return PlaceholderParser::evaluate_boolean_expression(condition, active_print.preset.config);
- } catch (const std::runtime_error &err) {
- //FIXME in case of an error, return "compatible with everything".
- printf("Preset::is_compatible_with_print - parsing error of compatible_prints_condition %s:\n%s\n", active_print.preset.name.c_str(), err.what());
- return true;
- }
- }
- return preset.preset.is_default || active_print.preset.name.empty() || ! has_compatible_prints ||
- std::find(compatible_prints->values.begin(), compatible_prints->values.end(), active_print.preset.name) !=
- compatible_prints->values.end();
-}
-
-bool is_compatible_with_printer(const PresetWithVendorProfile &preset, const PresetWithVendorProfile &active_printer, const DynamicPrintConfig *extra_config)
-{
- if (preset.vendor != nullptr && preset.vendor != active_printer.vendor)
- // The current profile has a vendor assigned and it is different from the active print's vendor.
- return false;
- auto &condition = preset.preset.compatible_printers_condition();
- auto *compatible_printers = dynamic_cast<const ConfigOptionStrings*>(preset.preset.config.option("compatible_printers"));
- bool has_compatible_printers = compatible_printers != nullptr && ! compatible_printers->values.empty();
- if (! has_compatible_printers && ! condition.empty()) {
- try {
- return PlaceholderParser::evaluate_boolean_expression(condition, active_printer.preset.config, extra_config);
- } catch (const std::runtime_error &err) {
- //FIXME in case of an error, return "compatible with everything".
- printf("Preset::is_compatible_with_printer - parsing error of compatible_printers_condition %s:\n%s\n", active_printer.preset.name.c_str(), err.what());
- return true;
- }
- }
- return preset.preset.is_default || active_printer.preset.name.empty() || ! has_compatible_printers ||
- std::find(compatible_printers->values.begin(), compatible_printers->values.end(), active_printer.preset.name) !=
- compatible_printers->values.end();
-}
-
-bool is_compatible_with_printer(const PresetWithVendorProfile &preset, const PresetWithVendorProfile &active_printer)
-{
- DynamicPrintConfig config;
- config.set_key_value("printer_preset", new ConfigOptionString(active_printer.preset.name));
- const ConfigOption *opt = active_printer.preset.config.option("nozzle_diameter");
- if (opt)
- config.set_key_value("num_extruders", new ConfigOptionInt((int)static_cast<const ConfigOptionFloats*>(opt)->values.size()));
- return is_compatible_with_printer(preset, active_printer, &config);
-}
-
-void Preset::set_visible_from_appconfig(const AppConfig &app_config)
-{
- if (vendor == nullptr) { return; }
-
- if (type == TYPE_PRINTER) {
- const std::string &model = config.opt_string("printer_model");
- const std::string &variant = config.opt_string("printer_variant");
- if (model.empty() || variant.empty())
- return;
- is_visible = app_config.get_variant(vendor->id, model, variant);
- } else if (type == TYPE_FILAMENT || type == TYPE_SLA_MATERIAL) {
- const std::string &section_name = (type == TYPE_FILAMENT) ? AppConfig::SECTION_FILAMENTS : AppConfig::SECTION_MATERIALS;
- if (app_config.has_section(section_name)) {
- // Check whether this profile is marked as "installed" in PrusaSlicer.ini,
- // or whether a profile is marked as "installed", which this profile may have been renamed from.
- const std::map<std::string, std::string> &installed = app_config.get_section(section_name);
- auto has = [&installed](const std::string &name) {
- auto it = installed.find(name);
- return it != installed.end() && ! it->second.empty();
- };
- is_visible = has(this->name);
- for (auto it = this->renamed_from.begin(); ! is_visible && it != this->renamed_from.end(); ++ it)
- is_visible = has(*it);
- }
- }
-}
-
-const std::vector<std::string>& Preset::print_options()
-{
- static std::vector<std::string> s_opts {
- "layer_height", "first_layer_height", "perimeters", "spiral_vase", "slice_closing_radius",
- "top_solid_layers", "top_solid_min_thickness", "bottom_solid_layers", "bottom_solid_min_thickness",
- "extra_perimeters", "ensure_vertical_shell_thickness", "avoid_crossing_perimeters", "thin_walls", "overhangs",
- "seam_position", "external_perimeters_first", "fill_density", "fill_pattern", "top_fill_pattern", "bottom_fill_pattern",
- "infill_every_layers", "infill_only_where_needed", "solid_infill_every_layers", "fill_angle", "bridge_angle",
- "solid_infill_below_area", "only_retract_when_crossing_perimeters", "infill_first",
- "ironing", "ironing_type", "ironing_flowrate", "ironing_speed", "ironing_spacing",
- "max_print_speed", "max_volumetric_speed",
-#ifdef HAS_PRESSURE_EQUALIZER
- "max_volumetric_extrusion_rate_slope_positive", "max_volumetric_extrusion_rate_slope_negative",
-#endif /* HAS_PRESSURE_EQUALIZER */
- "perimeter_speed", "small_perimeter_speed", "external_perimeter_speed", "infill_speed", "solid_infill_speed",
- "top_solid_infill_speed", "support_material_speed", "support_material_xy_spacing", "support_material_interface_speed",
- "bridge_speed", "gap_fill_speed", "travel_speed", "first_layer_speed", "perimeter_acceleration", "infill_acceleration",
- "bridge_acceleration", "first_layer_acceleration", "default_acceleration", "skirts", "skirt_distance", "skirt_height", "draft_shield",
- "min_skirt_length", "brim_width", "support_material", "support_material_auto", "support_material_threshold", "support_material_enforce_layers",
- "raft_layers", "support_material_pattern", "support_material_with_sheath", "support_material_spacing",
- "support_material_synchronize_layers", "support_material_angle", "support_material_interface_layers",
- "support_material_interface_spacing", "support_material_interface_contact_loops", "support_material_contact_distance",
- "support_material_buildplate_only", "dont_support_bridges", "notes", "complete_objects", "extruder_clearance_radius",
- "extruder_clearance_height", "gcode_comments", "gcode_label_objects", "output_filename_format", "post_process", "perimeter_extruder",
- "infill_extruder", "solid_infill_extruder", "support_material_extruder", "support_material_interface_extruder",
- "ooze_prevention", "standby_temperature_delta", "interface_shells", "extrusion_width", "first_layer_extrusion_width",
- "perimeter_extrusion_width", "external_perimeter_extrusion_width", "infill_extrusion_width", "solid_infill_extrusion_width",
- "top_infill_extrusion_width", "support_material_extrusion_width", "infill_overlap", "bridge_flow_ratio", "clip_multipart_objects",
- "elefant_foot_compensation", "xy_size_compensation", "threads", "resolution", "wipe_tower", "wipe_tower_x", "wipe_tower_y",
- "wipe_tower_width", "wipe_tower_rotation_angle", "wipe_tower_bridging", "single_extruder_multi_material_priming",
- "wipe_tower_no_sparse_layers", "compatible_printers", "compatible_printers_condition", "inherits"
- };
- return s_opts;
-}
-
-const std::vector<std::string>& Preset::filament_options()
-{
- static std::vector<std::string> s_opts {
- "filament_colour", "filament_diameter", "filament_type", "filament_soluble", "filament_notes", "filament_max_volumetric_speed",
- "extrusion_multiplier", "filament_density", "filament_cost", "filament_loading_speed", "filament_loading_speed_start", "filament_load_time",
- "filament_unloading_speed", "filament_unloading_speed_start", "filament_unload_time", "filament_toolchange_delay", "filament_cooling_moves",
- "filament_cooling_initial_speed", "filament_cooling_final_speed", "filament_ramming_parameters", "filament_minimal_purge_on_wipe_tower",
- "temperature", "first_layer_temperature", "bed_temperature", "first_layer_bed_temperature", "fan_always_on", "cooling", "min_fan_speed",
- "max_fan_speed", "bridge_fan_speed", "disable_fan_first_layers", "fan_below_layer_time", "slowdown_below_layer_time", "min_print_speed",
- "start_filament_gcode", "end_filament_gcode",
- // Retract overrides
- "filament_retract_length", "filament_retract_lift", "filament_retract_lift_above", "filament_retract_lift_below", "filament_retract_speed", "filament_deretract_speed", "filament_retract_restart_extra", "filament_retract_before_travel",
- "filament_retract_layer_change", "filament_wipe", "filament_retract_before_wipe",
- // Profile compatibility
- "filament_vendor", "compatible_prints", "compatible_prints_condition", "compatible_printers", "compatible_printers_condition", "inherits"
- };
- return s_opts;
-}
-
-const std::vector<std::string>& Preset::printer_options()
-{
- static std::vector<std::string> s_opts;
- if (s_opts.empty()) {
- s_opts = {
- "printer_technology",
- "bed_shape", "bed_custom_texture", "bed_custom_model", "z_offset", "gcode_flavor", "use_relative_e_distances", "serial_port", "serial_speed",
- "use_firmware_retraction", "use_volumetric_e", "variable_layer_height",
- "host_type", "print_host", "printhost_apikey", "printhost_cafile",
- "single_extruder_multi_material", "start_gcode", "end_gcode", "before_layer_gcode", "layer_gcode", "toolchange_gcode",
- "color_change_gcode", "pause_print_gcode", "template_custom_gcode",
- "between_objects_gcode", "printer_vendor", "printer_model", "printer_variant", "printer_notes", "cooling_tube_retraction",
- "cooling_tube_length", "high_current_on_filament_swap", "parking_pos_retraction", "extra_loading_move", "max_print_height",
- "default_print_profile", "inherits",
- "remaining_times", "silent_mode", "machine_max_acceleration_extruding", "machine_max_acceleration_retracting",
- "machine_max_acceleration_x", "machine_max_acceleration_y", "machine_max_acceleration_z", "machine_max_acceleration_e",
- "machine_max_feedrate_x", "machine_max_feedrate_y", "machine_max_feedrate_z", "machine_max_feedrate_e",
- "machine_min_extruding_rate", "machine_min_travel_rate",
- "machine_max_jerk_x", "machine_max_jerk_y", "machine_max_jerk_z", "machine_max_jerk_e",
- "thumbnails"
- };
- s_opts.insert(s_opts.end(), Preset::nozzle_options().begin(), Preset::nozzle_options().end());
- }
- return s_opts;
-}
-
-// The following nozzle options of a printer profile will be adjusted to match the size
-// of the nozzle_diameter vector.
-const std::vector<std::string>& Preset::nozzle_options()
-{
- return print_config_def.extruder_option_keys();
-}
-
-const std::vector<std::string>& Preset::sla_print_options()
-{
- static std::vector<std::string> s_opts;
- if (s_opts.empty()) {
- s_opts = {
- "layer_height",
- "faded_layers",
- "supports_enable",
- "support_head_front_diameter",
- "support_head_penetration",
- "support_head_width",
- "support_pillar_diameter",
- "support_max_bridges_on_pillar",
- "support_pillar_connection_mode",
- "support_buildplate_only",
- "support_pillar_widening_factor",
- "support_base_diameter",
- "support_base_height",
- "support_base_safety_distance",
- "support_critical_angle",
- "support_max_bridge_length",
- "support_max_pillar_link_distance",
- "support_object_elevation",
- "support_points_density_relative",
- "support_points_minimal_distance",
- "slice_closing_radius",
- "pad_enable",
- "pad_wall_thickness",
- "pad_wall_height",
- "pad_brim_size",
- "pad_max_merge_distance",
- // "pad_edge_radius",
- "pad_wall_slope",
- "pad_object_gap",
- "pad_around_object",
- "pad_around_object_everywhere",
- "pad_object_connector_stride",
- "pad_object_connector_width",
- "pad_object_connector_penetration",
- "hollowing_enable",
- "hollowing_min_thickness",
- "hollowing_quality",
- "hollowing_closing_distance",
- "output_filename_format",
- "default_sla_print_profile",
- "compatible_printers",
- "compatible_printers_condition",
- "inherits"
- };
- }
- return s_opts;
-}
-
-const std::vector<std::string>& Preset::sla_material_options()
-{
- static std::vector<std::string> s_opts;
- if (s_opts.empty()) {
- s_opts = {
- "material_type",
- "initial_layer_height",
- "bottle_cost",
- "bottle_volume",
- "bottle_weight",
- "material_density",
- "exposure_time",
- "initial_exposure_time",
- "material_correction",
- "material_notes",
- "material_vendor",
- "default_sla_material_profile",
- "compatible_prints", "compatible_prints_condition",
- "compatible_printers", "compatible_printers_condition", "inherits"
- };
- }
- return s_opts;
-}
-
-const std::vector<std::string>& Preset::sla_printer_options()
-{
- static std::vector<std::string> s_opts;
- if (s_opts.empty()) {
- s_opts = {
- "printer_technology",
- "bed_shape", "bed_custom_texture", "bed_custom_model", "max_print_height",
- "display_width", "display_height", "display_pixels_x", "display_pixels_y",
- "display_mirror_x", "display_mirror_y",
- "display_orientation",
- "fast_tilt_time", "slow_tilt_time", "area_fill",
- "relative_correction",
- "absolute_correction",
- "elefant_foot_compensation",
- "elefant_foot_min_width",
- "gamma_correction",
- "min_exposure_time", "max_exposure_time",
- "min_initial_exposure_time", "max_initial_exposure_time",
- "print_host", "printhost_apikey", "printhost_cafile",
- "printer_notes",
- "inherits"
- };
- }
- return s_opts;
-}
-
-PresetCollection::PresetCollection(Preset::Type type, const std::vector<std::string> &keys, const Slic3r::StaticPrintConfig &defaults, const std::string &default_name) :
- m_type(type),
- m_edited_preset(type, "", false),
- m_idx_selected(0)
-{
- // Insert just the default preset.
- this->add_default_preset(keys, defaults, default_name);
- m_edited_preset.config.apply(m_presets.front().config);
-}
-
-PresetCollection::~PresetCollection()
-{
-}
-
-void PresetCollection::reset(bool delete_files)
-{
- if (m_presets.size() > m_num_default_presets) {
- if (delete_files) {
- // Erase the preset files.
- for (Preset &preset : m_presets)
- if (! preset.is_default && ! preset.is_external && ! preset.is_system)
- boost::nowide::remove(preset.file.c_str());
- }
- // Don't use m_presets.resize() here as it requires a default constructor for Preset.
- m_presets.erase(m_presets.begin() + m_num_default_presets, m_presets.end());
- this->select_preset(0);
- }
- m_map_alias_to_profile_name.clear();
- m_map_system_profile_renamed.clear();
-}
-
-void PresetCollection::add_default_preset(const std::vector<std::string> &keys, const Slic3r::StaticPrintConfig &defaults, const std::string &preset_name)
-{
- // Insert just the default preset.
- m_presets.emplace_back(Preset(this->type(), preset_name, true));
- m_presets.back().config.apply_only(defaults, keys.empty() ? defaults.keys() : keys);
- m_presets.back().loaded = true;
- ++ m_num_default_presets;
-}
-
-// Load all presets found in dir_path.
-// Throws an exception on error.
-void PresetCollection::load_presets(const std::string &dir_path, const std::string &subdir)
-{
- boost::filesystem::path dir = boost::filesystem::canonical(boost::filesystem::path(dir_path) / subdir).make_preferred();
- m_dir_path = dir.string();
- std::string errors_cummulative;
- // Store the loaded presets into a new vector, otherwise the binary search for already existing presets would be broken.
- // (see the "Preset already present, not loading" message).
- std::deque<Preset> presets_loaded;
- for (auto &dir_entry : boost::filesystem::directory_iterator(dir))
- if (Slic3r::is_ini_file(dir_entry)) {
- std::string name = dir_entry.path().filename().string();
- // Remove the .ini suffix.
- name.erase(name.size() - 4);
- if (this->find_preset(name, false)) {
- // This happens when there's is a preset (most likely legacy one) with the same name as a system preset
- // that's already been loaded from a bundle.
- BOOST_LOG_TRIVIAL(warning) << "Preset already present, not loading: " << name;
- continue;
- }
- try {
- Preset preset(m_type, name, false);
- preset.file = dir_entry.path().string();
- // Load the preset file, apply preset values on top of defaults.
- try {
- DynamicPrintConfig config;
- config.load_from_ini(preset.file);
- // Find a default preset for the config. The PrintPresetCollection provides different default preset based on the "printer_technology" field.
- const Preset &default_preset = this->default_preset_for(config);
- preset.config = default_preset.config;
- preset.config.apply(std::move(config));
- Preset::normalize(preset.config);
- // Report configuration fields, which are misplaced into a wrong group.
- std::string incorrect_keys = Preset::remove_invalid_keys(config, default_preset.config);
- if (! incorrect_keys.empty())
- BOOST_LOG_TRIVIAL(error) << "Error in a preset file: The preset \"" <<
- preset.file << "\" contains the following incorrect keys: " << incorrect_keys << ", which were removed";
- preset.loaded = true;
- } catch (const std::ifstream::failure &err) {
- throw std::runtime_error(std::string("The selected preset cannot be loaded: ") + preset.file + "\n\tReason: " + err.what());
- } catch (const std::runtime_error &err) {
- throw std::runtime_error(std::string("Failed loading the preset file: ") + preset.file + "\n\tReason: " + err.what());
- }
- presets_loaded.emplace_back(preset);
- } catch (const std::runtime_error &err) {
- errors_cummulative += err.what();
- errors_cummulative += "\n";
- }
- }
- m_presets.insert(m_presets.end(), std::make_move_iterator(presets_loaded.begin()), std::make_move_iterator(presets_loaded.end()));
- std::sort(m_presets.begin() + m_num_default_presets, m_presets.end());
- this->select_preset(first_visible_idx());
- if (! errors_cummulative.empty())
- throw std::runtime_error(errors_cummulative);
-}
-
-// Load a preset from an already parsed config file, insert it into the sorted sequence of presets
-// and select it, losing previous modifications.
-Preset& PresetCollection::load_preset(const std::string &path, const std::string &name, const DynamicPrintConfig &config, bool select)
-{
- DynamicPrintConfig cfg(this->default_preset().config);
- cfg.apply_only(config, cfg.keys(), true);
- return this->load_preset(path, name, std::move(cfg), select);
-}
-
-enum class ProfileHostParams
-{
- Same,
- Different,
- Anonymized,
-};
-
-static ProfileHostParams profile_host_params_same_or_anonymized(const DynamicPrintConfig &cfg_old, const DynamicPrintConfig &cfg_new)
-{
- auto opt_print_host_old = cfg_old.option<ConfigOptionString>("print_host");
- auto opt_printhost_apikey_old = cfg_old.option<ConfigOptionString>("printhost_apikey");
- auto opt_printhost_cafile_old = cfg_old.option<ConfigOptionString>("printhost_cafile");
-
- auto opt_print_host_new = cfg_new.option<ConfigOptionString>("print_host");
- auto opt_printhost_apikey_new = cfg_new.option<ConfigOptionString>("printhost_apikey");
- auto opt_printhost_cafile_new = cfg_new.option<ConfigOptionString>("printhost_cafile");
-
- // If the new print host data is undefined, use the old data.
- bool new_print_host_undefined = (opt_print_host_new == nullptr || opt_print_host_new ->empty()) &&
- (opt_printhost_apikey_new == nullptr || opt_printhost_apikey_new ->empty()) &&
- (opt_printhost_cafile_new == nullptr || opt_printhost_cafile_new ->empty());
- if (new_print_host_undefined)
- return ProfileHostParams::Anonymized;
-
- auto opt_same = [](const ConfigOptionString *l, const ConfigOptionString *r) {
- return ((l == nullptr || l->empty()) && (r == nullptr || r->empty())) ||
- (l != nullptr && r != nullptr && l->value == r->value);
- };
- return (opt_same(opt_print_host_old, opt_print_host_new) && opt_same(opt_printhost_apikey_old, opt_printhost_apikey_new) &&
- opt_same(opt_printhost_cafile_old, opt_printhost_cafile_new)) ? ProfileHostParams::Same : ProfileHostParams::Different;
-}
-
-static bool profile_print_params_same(const DynamicPrintConfig &cfg_old, const DynamicPrintConfig &cfg_new)
-{
- t_config_option_keys diff = cfg_old.diff(cfg_new);
- // Following keys are used by the UI, not by the slicing core, therefore they are not important
- // when comparing profiles for equality. Ignore them.
- for (const char *key : { "compatible_prints", "compatible_prints_condition",
- "compatible_printers", "compatible_printers_condition", "inherits",
- "print_settings_id", "filament_settings_id", "sla_print_settings_id", "sla_material_settings_id", "printer_settings_id",
- "printer_model", "printer_variant", "default_print_profile", "default_filament_profile", "default_sla_print_profile", "default_sla_material_profile",
- "print_host", "printhost_apikey", "printhost_cafile" })
- diff.erase(std::remove(diff.begin(), diff.end(), key), diff.end());
- // Preset with the same name as stored inside the config exists.
- return diff.empty() && profile_host_params_same_or_anonymized(cfg_old, cfg_new) != ProfileHostParams::Different;
-}
-
-// Load a preset from an already parsed config file, insert it into the sorted sequence of presets
-// and select it, losing previous modifications.
-// In case
-Preset& PresetCollection::load_external_preset(
- // Path to the profile source file (a G-code, an AMF or 3MF file, a config file)
- const std::string &path,
- // Name of the profile, derived from the source file name.
- const std::string &name,
- // Original name of the profile, extracted from the loaded config. Empty, if the name has not been stored.
- const std::string &original_name,
- // Config to initialize the preset from.
- const DynamicPrintConfig &config,
- // Select the preset after loading?
- bool select)
-{
- // Load the preset over a default preset, so that the missing fields are filled in from the default preset.
- DynamicPrintConfig cfg(this->default_preset_for(config).config);
- cfg.apply_only(config, cfg.keys(), true);
- // Is there a preset already loaded with the name stored inside the config?
- std::deque<Preset>::iterator it = this->find_preset_internal(original_name);
- bool found = it != m_presets.end() && it->name == original_name;
- if (! found) {
- // Try to match the original_name against the "renamed_from" profile names of loaded system profiles.
- it = this->find_preset_renamed(original_name);
- found = it != m_presets.end();
- }
- if (found) {
- if (profile_print_params_same(it->config, cfg)) {
- // The preset exists and it matches the values stored inside config.
- if (select)
- this->select_preset(it - m_presets.begin());
- return *it;
- }
- if (profile_host_params_same_or_anonymized(it->config, cfg) == ProfileHostParams::Anonymized) {
- // The project being loaded is anonymized. Replace the empty host keys of the loaded profile with the data from the original profile.
- // See "Octoprint Settings when Opening a .3MF file" GH issue #3244
- auto opt_update = [it, &cfg](const std::string &opt_key) {
- auto opt = it->config.option<ConfigOptionString>(opt_key);
- if (opt != nullptr)
- cfg.set_key_value(opt_key, opt->clone());
- };
- opt_update("print_host");
- opt_update("printhost_apikey");
- opt_update("printhost_cafile");
- }
- }
- // Update the "inherits" field.
- std::string &inherits = Preset::inherits(cfg);
- if (found && inherits.empty()) {
- // There is a profile with the same name already loaded. Should we update the "inherits" field?
- if (it->vendor == nullptr)
- inherits = it->inherits();
- else
- inherits = it->name;
- }
- // The external preset does not match an internal preset, load the external preset.
- std::string new_name;
- for (size_t idx = 0;; ++ idx) {
- std::string suffix;
- if (original_name.empty()) {
- if (idx > 0)
- suffix = " (" + std::to_string(idx) + ")";
- } else {
- if (idx == 0)
- suffix = " (" + original_name + ")";
- else
- suffix = " (" + original_name + "-" + std::to_string(idx) + ")";
- }
- new_name = name + suffix;
- it = this->find_preset_internal(new_name);
- if (it == m_presets.end() || it->name != new_name)
- // Unique profile name. Insert a new profile.
- break;
- if (profile_print_params_same(it->config, cfg)) {
- // The preset exists and it matches the values stored inside config.
- if (select)
- this->select_preset(it - m_presets.begin());
- return *it;
- }
- // Form another profile name.
- }
- // Insert a new profile.
- Preset &preset = this->load_preset(path, new_name, std::move(cfg), select);
- preset.is_external = true;
- if (&this->get_selected_preset() == &preset)
- this->get_edited_preset().is_external = true;
-
- return preset;
-}
-
-Preset& PresetCollection::load_preset(const std::string &path, const std::string &name, DynamicPrintConfig &&config, bool select)
-{
- auto it = this->find_preset_internal(name);
- if (it == m_presets.end() || it->name != name) {
- // The preset was not found. Create a new preset.
- it = m_presets.emplace(it, Preset(m_type, name, false));
- }
- Preset &preset = *it;
- preset.file = path;
- preset.config = std::move(config);
- preset.loaded = true;
- preset.is_dirty = false;
- if (select)
- this->select_preset_by_name(name, true);
- return preset;
-}
-
-void PresetCollection::save_current_preset(const std::string &new_name, bool detach)
-{
- // 1) Find the preset with a new_name or create a new one,
- // initialize it with the edited config.
- auto it = this->find_preset_internal(new_name);
- if (it != m_presets.end() && it->name == new_name) {
- // Preset with the same name found.
- Preset &preset = *it;
- if (preset.is_default || preset.is_external || preset.is_system)
- // Cannot overwrite the default preset.
- return;
- // Overwriting an existing preset.
- preset.config = std::move(m_edited_preset.config);
- // The newly saved preset will be activated -> make it visible.
- preset.is_visible = true;
- if (detach) {
- // Clear the link to the parent profile.
- preset.vendor = nullptr;
- preset.inherits().clear();
- preset.alias.clear();
- preset.renamed_from.clear();
- }
- } else {
- // Creating a new preset.
- Preset &preset = *m_presets.insert(it, m_edited_preset);
- std::string &inherits = preset.inherits();
- std::string old_name = preset.name;
- preset.name = new_name;
- preset.file = this->path_from_name(new_name);
- preset.vendor = nullptr;
- preset.alias.clear();
- preset.renamed_from.clear();
- if (detach) {
- // Clear the link to the parent profile.
- inherits.clear();
- } else if (preset.is_system) {
- // Inheriting from a system preset.
- inherits = /* preset.vendor->name + "/" + */ old_name;
- } else if (inherits.empty()) {
- // Inheriting from a user preset. Link the new preset to the old preset.
- // inherits = old_name;
- } else {
- // Inherited from a user preset. Just maintain the "inherited" flag,
- // meaning it will inherit from either the system preset, or the inherited user preset.
- }
- preset.is_default = false;
- preset.is_system = false;
- preset.is_external = false;
- // The newly saved preset will be activated -> make it visible.
- preset.is_visible = true;
- // Just system presets have aliases
- preset.alias.clear();
- }
- // 2) Activate the saved preset.
- this->select_preset_by_name(new_name, true);
- // 2) Store the active preset to disk.
- this->get_selected_preset().save();
-}
-
-bool PresetCollection::delete_current_preset()
-{
- const Preset &selected = this->get_selected_preset();
- if (selected.is_default)
- return false;
- if (! selected.is_external && ! selected.is_system) {
- // Erase the preset file.
- boost::nowide::remove(selected.file.c_str());
- }
- // Remove the preset from the list.
- m_presets.erase(m_presets.begin() + m_idx_selected);
- // Find the next visible preset.
- size_t new_selected_idx = m_idx_selected;
- if (new_selected_idx < m_presets.size())
- for (; new_selected_idx < m_presets.size() && ! m_presets[new_selected_idx].is_visible; ++ new_selected_idx) ;
- if (new_selected_idx == m_presets.size())
- for (--new_selected_idx; new_selected_idx > 0 && !m_presets[new_selected_idx].is_visible; --new_selected_idx);
- this->select_preset(new_selected_idx);
- return true;
-}
-
-bool PresetCollection::delete_preset(const std::string& name)
-{
- auto it = this->find_preset_internal(name);
-
- const Preset& preset = *it;
- if (preset.is_default)
- return false;
- if (!preset.is_external && !preset.is_system) {
- // Erase the preset file.
- boost::nowide::remove(preset.file.c_str());
- }
- m_presets.erase(it);
- return true;
-}
-
-const Preset* PresetCollection::get_selected_preset_parent() const
-{
- if (this->get_selected_idx() == size_t(-1))
- // This preset collection has no preset activated yet. Only the get_edited_preset() is valid.
- return nullptr;
-
- const Preset &selected_preset = this->get_selected_preset();
- if (selected_preset.is_system || selected_preset.is_default)
- return &selected_preset;
-
- const Preset &edited_preset = this->get_edited_preset();
- const std::string &inherits = edited_preset.inherits();
- const Preset *preset = nullptr;
- if (inherits.empty()) {
- if (selected_preset.is_external)
- return nullptr;
- preset = &this->default_preset(m_type == Preset::Type::TYPE_PRINTER && edited_preset.printer_technology() == ptSLA ? 1 : 0);
- } else
- preset = this->find_preset(inherits, false);
- if (preset == nullptr) {
- // Resolve the "renamed_from" field.
- assert(! inherits.empty());
- auto it = this->find_preset_renamed(inherits);
- if (it != m_presets.end())
- preset = &(*it);
- }
- return (preset == nullptr/* || preset->is_default*/ || preset->is_external) ? nullptr : preset;
-}
-
-const Preset* PresetCollection::get_preset_parent(const Preset& child) const
-{
- const std::string &inherits = child.inherits();
- if (inherits.empty())
-// return this->get_selected_preset().is_system ? &this->get_selected_preset() : nullptr;
- return nullptr;
- const Preset* preset = this->find_preset(inherits, false);
- if (preset == nullptr) {
- auto it = this->find_preset_renamed(inherits);
- if (it != m_presets.end())
- preset = &(*it);
- }
- return (preset == nullptr/* || preset->is_default */|| preset->is_external) ? nullptr : preset;
-}
-
-// Return vendor of the first parent profile, for which the vendor is defined, or null if such profile does not exist.
-PresetWithVendorProfile PresetCollection::get_preset_with_vendor_profile(const Preset &preset) const
-{
- const Preset *p = &preset;
- const VendorProfile *v = nullptr;
- do {
- if (p->vendor != nullptr) {
- v = p->vendor;
- break;
- }
- p = this->get_preset_parent(*p);
- } while (p != nullptr);
- return PresetWithVendorProfile(preset, v);
-}
-
-const std::string& PresetCollection::get_preset_name_by_alias(const std::string& alias) const
-{
- for (
- // Find the 1st profile name with the alias.
- auto it = Slic3r::lower_bound_by_predicate(m_map_alias_to_profile_name.begin(), m_map_alias_to_profile_name.end(), [&alias](auto &l){ return l.first < alias; });
- // Continue over all profile names with the same alias.
- it != m_map_alias_to_profile_name.end() && it->first == alias; ++ it)
- if (auto it_preset = this->find_preset_internal(it->second);
- it_preset != m_presets.end() && it_preset->name == it->second &&
- it_preset->is_visible && (it_preset->is_compatible || size_t(it_preset - m_presets.begin()) == m_idx_selected))
- return it_preset->name;
- return alias;
-}
-
-const std::string* PresetCollection::get_preset_name_renamed(const std::string &old_name) const
-{
- auto it_renamed = m_map_system_profile_renamed.find(old_name);
- if (it_renamed != m_map_system_profile_renamed.end())
- return &it_renamed->second;
- return nullptr;
-}
-
-const std::string& PresetCollection::get_suffix_modified() {
- return g_suffix_modified;
-}
-
-// Return a preset by its name. If the preset is active, a temporary copy is returned.
-// If a preset is not found by its name, null is returned.
-Preset* PresetCollection::find_preset(const std::string &name, bool first_visible_if_not_found)
-{
- Preset key(m_type, name, false);
- auto it = this->find_preset_internal(name);
- // Ensure that a temporary copy is returned if the preset found is currently selected.
- return (it != m_presets.end() && it->name == key.name) ? &this->preset(it - m_presets.begin()) :
- first_visible_if_not_found ? &this->first_visible() : nullptr;
-}
-
-// Return index of the first visible preset. Certainly at least the '- default -' preset shall be visible.
-size_t PresetCollection::first_visible_idx() const
-{
- size_t idx = m_default_suppressed ? m_num_default_presets : 0;
- for (; idx < this->m_presets.size(); ++ idx)
- if (m_presets[idx].is_visible)
- break;
- if (idx == m_presets.size())
- idx = 0;
- return idx;
-}
-
-void PresetCollection::set_default_suppressed(bool default_suppressed)
-{
- if (m_default_suppressed != default_suppressed) {
- m_default_suppressed = default_suppressed;
- bool default_visible = ! default_suppressed || m_idx_selected < m_num_default_presets;
- for (size_t i = 0; i < m_num_default_presets; ++ i)
- m_presets[i].is_visible = default_visible;
- }
-}
-
-size_t PresetCollection::update_compatible_internal(const PresetWithVendorProfile &active_printer, const PresetWithVendorProfile *active_print, PresetSelectCompatibleType unselect_if_incompatible)
-{
- DynamicPrintConfig config;
- config.set_key_value("printer_preset", new ConfigOptionString(active_printer.preset.name));
- const ConfigOption *opt = active_printer.preset.config.option("nozzle_diameter");
- if (opt)
- config.set_key_value("num_extruders", new ConfigOptionInt((int)static_cast<const ConfigOptionFloats*>(opt)->values.size()));
- bool some_compatible = false;
- for (size_t idx_preset = m_num_default_presets; idx_preset < m_presets.size(); ++ idx_preset) {
- bool selected = idx_preset == m_idx_selected;
- Preset &preset_selected = m_presets[idx_preset];
- Preset &preset_edited = selected ? m_edited_preset : preset_selected;
-
- const PresetWithVendorProfile this_preset_with_vendor_profile = this->get_preset_with_vendor_profile(preset_edited);
- bool was_compatible = preset_edited.is_compatible;
- preset_edited.is_compatible = is_compatible_with_printer(this_preset_with_vendor_profile, active_printer, &config);
- some_compatible |= preset_edited.is_compatible;
- if (active_print != nullptr)
- preset_edited.is_compatible &= is_compatible_with_print(this_preset_with_vendor_profile, *active_print, active_printer);
- if (! preset_edited.is_compatible && selected &&
- (unselect_if_incompatible == PresetSelectCompatibleType::Always || (unselect_if_incompatible == PresetSelectCompatibleType::OnlyIfWasCompatible && was_compatible)))
- m_idx_selected = size_t(-1);
- if (selected)
- preset_selected.is_compatible = preset_edited.is_compatible;
- }
- // Update visibility of the default profiles here if the defaults are suppressed, the current profile is not compatible and we don't want to select another compatible profile.
- if (m_idx_selected >= m_num_default_presets && m_default_suppressed)
- for (size_t i = 0; i < m_num_default_presets; ++ i)
- m_presets[i].is_visible = ! some_compatible;
- return m_idx_selected;
-}
-
-// Save the preset under a new name. If the name is different from the old one,
-// a new preset is stored into the list of presets.
-// All presets are marked as not modified and the new preset is activated.
-//void PresetCollection::save_current_preset(const std::string &new_name);
-
-// Delete the current preset, activate the first visible preset.
-//void PresetCollection::delete_current_preset();
-
-// Update a dirty flag of the current preset
-// Return true if the dirty flag changed.
-bool PresetCollection::update_dirty()
-{
- bool was_dirty = this->get_selected_preset().is_dirty;
- bool is_dirty = current_is_dirty();
- this->get_selected_preset().is_dirty = is_dirty;
- this->get_edited_preset().is_dirty = is_dirty;
-
- return was_dirty != is_dirty;
-}
-
-template<class T>
-void add_correct_opts_to_diff(const std::string &opt_key, t_config_option_keys& vec, const ConfigBase &other, const ConfigBase &this_c)
-{
- const T* opt_init = static_cast<const T*>(other.option(opt_key));
- const T* opt_cur = static_cast<const T*>(this_c.option(opt_key));
- int opt_init_max_id = opt_init->values.size() - 1;
- for (int i = 0; i < int(opt_cur->values.size()); i++)
- {
- int init_id = i <= opt_init_max_id ? i : 0;
- if (opt_cur->values[i] != opt_init->values[init_id])
- vec.emplace_back(opt_key + "#" + std::to_string(i));
- }
-}
-
-// Use deep_diff to correct return of changed options, considering individual options for each extruder.
-inline t_config_option_keys deep_diff(const ConfigBase &config_this, const ConfigBase &config_other)
-{
- t_config_option_keys diff;
- for (const t_config_option_key &opt_key : config_this.keys()) {
- const ConfigOption *this_opt = config_this.option(opt_key);
- const ConfigOption *other_opt = config_other.option(opt_key);
- if (this_opt != nullptr && other_opt != nullptr && *this_opt != *other_opt)
- {
- if (opt_key == "bed_shape" || opt_key == "thumbnails" || opt_key == "compatible_prints" || opt_key == "compatible_printers") {
- diff.emplace_back(opt_key);
- continue;
- }
- switch (other_opt->type())
- {
- case coInts: add_correct_opts_to_diff<ConfigOptionInts >(opt_key, diff, config_other, config_this); break;
- case coBools: add_correct_opts_to_diff<ConfigOptionBools >(opt_key, diff, config_other, config_this); break;
- case coFloats: add_correct_opts_to_diff<ConfigOptionFloats >(opt_key, diff, config_other, config_this); break;
- case coStrings: add_correct_opts_to_diff<ConfigOptionStrings >(opt_key, diff, config_other, config_this); break;
- case coPercents:add_correct_opts_to_diff<ConfigOptionPercents >(opt_key, diff, config_other, config_this); break;
- case coPoints: add_correct_opts_to_diff<ConfigOptionPoints >(opt_key, diff, config_other, config_this); break;
- default: diff.emplace_back(opt_key); break;
- }
- }
- }
- return diff;
-}
-
-std::vector<std::string> PresetCollection::dirty_options(const Preset *edited, const Preset *reference, const bool deep_compare /*= false*/)
-{
- std::vector<std::string> changed;
- if (edited != nullptr && reference != nullptr) {
- changed = deep_compare ?
- deep_diff(edited->config, reference->config) :
- reference->config.diff(edited->config);
- // The "compatible_printers" option key is handled differently from the others:
- // It is not mandatory. If the key is missing, it means it is compatible with any printer.
- // If the key exists and it is empty, it means it is compatible with no printer.
- std::initializer_list<const char*> optional_keys { "compatible_prints", "compatible_printers" };
- for (auto &opt_key : optional_keys) {
- if (reference->config.has(opt_key) != edited->config.has(opt_key))
- changed.emplace_back(opt_key);
- }
- }
- return changed;
-}
-
-// Select a new preset. This resets all the edits done to the currently selected preset.
-// If the preset with index idx does not exist, a first visible preset is selected.
-Preset& PresetCollection::select_preset(size_t idx)
-{
- for (Preset &preset : m_presets)
- preset.is_dirty = false;
- if (idx >= m_presets.size())
- idx = first_visible_idx();
- m_idx_selected = idx;
- m_edited_preset = m_presets[idx];
- bool default_visible = ! m_default_suppressed || m_idx_selected < m_num_default_presets;
- for (size_t i = 0; i < m_num_default_presets; ++i)
- m_presets[i].is_visible = default_visible;
- return m_presets[idx];
-}
-
-bool PresetCollection::select_preset_by_name(const std::string &name_w_suffix, bool force)
-{
- std::string name = Preset::remove_suffix_modified(name_w_suffix);
- // 1) Try to find the preset by its name.
- auto it = this->find_preset_internal(name);
- size_t idx = 0;
- if (it != m_presets.end() && it->name == name && it->is_visible)
- // Preset found by its name and it is visible.
- idx = it - m_presets.begin();
- else {
- // Find the first visible preset.
- for (size_t i = m_default_suppressed ? m_num_default_presets : 0; i < m_presets.size(); ++ i)
- if (m_presets[i].is_visible) {
- idx = i;
- break;
- }
- // If the first visible preset was not found, return the 0th element, which is the default preset.
- }
-
- // 2) Select the new preset.
- if (m_idx_selected != idx || force) {
- this->select_preset(idx);
- return true;
- }
-
- return false;
-}
-
-bool PresetCollection::select_preset_by_name_strict(const std::string &name)
-{
- // 1) Try to find the preset by its name.
- auto it = this->find_preset_internal(name);
- size_t idx = (size_t)-1;
- if (it != m_presets.end() && it->name == name && it->is_visible)
- // Preset found by its name.
- idx = it - m_presets.begin();
- // 2) Select the new preset.
- if (idx != (size_t)-1) {
- this->select_preset(idx);
- return true;
- }
- m_idx_selected = idx;
- return false;
-}
-
-// Merge one vendor's presets with the other vendor's presets, report duplicates.
-std::vector<std::string> PresetCollection::merge_presets(PresetCollection &&other, const VendorMap &new_vendors)
-{
- std::vector<std::string> duplicates;
- for (Preset &preset : other.m_presets) {
- if (preset.is_default || preset.is_external)
- continue;
- Preset key(m_type, preset.name);
- auto it = std::lower_bound(m_presets.begin() + m_num_default_presets, m_presets.end(), key);
- if (it == m_presets.end() || it->name != preset.name) {
- if (preset.vendor != nullptr) {
- // Re-assign a pointer to the vendor structure in the new PresetBundle.
- auto it = new_vendors.find(preset.vendor->id);
- assert(it != new_vendors.end());
- preset.vendor = &it->second;
- }
- this->m_presets.emplace(it, std::move(preset));
- } else
- duplicates.emplace_back(std::move(preset.name));
- }
- return duplicates;
-}
-
-void PresetCollection::update_map_alias_to_profile_name()
-{
- m_map_alias_to_profile_name.clear();
- for (const Preset &preset : m_presets)
- m_map_alias_to_profile_name.emplace_back(preset.alias, preset.name);
- std::sort(m_map_alias_to_profile_name.begin(), m_map_alias_to_profile_name.end(), [](auto &l, auto &r) { return l.first < r.first; });
-}
-
-void PresetCollection::update_map_system_profile_renamed()
-{
- m_map_system_profile_renamed.clear();
- for (Preset &preset : m_presets)
- for (const std::string &renamed_from : preset.renamed_from) {
- const auto [it, success] = m_map_system_profile_renamed.insert(std::pair<std::string, std::string>(renamed_from, preset.name));
- if (! success)
- BOOST_LOG_TRIVIAL(error) << boost::format("Preset name \"%1%\" was marked as renamed from \"%2%\", though preset name \"%3%\" was marked as renamed from \"%2%\" as well.") % preset.name % renamed_from % it->second;
- }
-}
-
-std::string PresetCollection::name() const
-{
- switch (this->type()) {
- case Preset::TYPE_PRINT: return L("print");
- case Preset::TYPE_FILAMENT: return L("filament");
- case Preset::TYPE_SLA_PRINT: return L("SLA print");
- case Preset::TYPE_SLA_MATERIAL: return L("SLA material");
- case Preset::TYPE_PRINTER: return L("printer");
- default: return "invalid";
- }
-}
-
-std::string PresetCollection::section_name() const
-{
- switch (this->type()) {
- case Preset::TYPE_PRINT: return "print";
- case Preset::TYPE_FILAMENT: return "filament";
- case Preset::TYPE_SLA_PRINT: return "sla_print";
- case Preset::TYPE_SLA_MATERIAL: return "sla_material";
- case Preset::TYPE_PRINTER: return "printer";
- default: return "invalid";
- }
-}
-
-std::vector<std::string> PresetCollection::system_preset_names() const
-{
- size_t num = 0;
- for (const Preset &preset : m_presets)
- if (preset.is_system)
- ++ num;
- std::vector<std::string> out;
- out.reserve(num);
- for (const Preset &preset : m_presets)
- if (preset.is_system)
- out.emplace_back(preset.name);
- std::sort(out.begin(), out.end());
- return out;
-}
-
-// Generate a file path from a profile name. Add the ".ini" suffix if it is missing.
-std::string PresetCollection::path_from_name(const std::string &new_name) const
-{
- std::string file_name = boost::iends_with(new_name, ".ini") ? new_name : (new_name + ".ini");
- return (boost::filesystem::path(m_dir_path) / file_name).make_preferred().string();
-}
-
-const Preset& PrinterPresetCollection::default_preset_for(const DynamicPrintConfig &config) const
-{
- const ConfigOptionEnumGeneric *opt_printer_technology = config.opt<ConfigOptionEnumGeneric>("printer_technology");
- return this->default_preset((opt_printer_technology == nullptr || opt_printer_technology->value == ptFFF) ? 0 : 1);
-}
-
-const Preset* PrinterPresetCollection::find_by_model_id(const std::string &model_id) const
-{
- if (model_id.empty()) { return nullptr; }
-
- const auto it = std::find_if(cbegin(), cend(), [&](const Preset &preset) {
- return preset.config.opt_string("printer_model") == model_id;
- });
-
- return it != cend() ? &*it : nullptr;
-}
-/*
-PhysicalPrinter& PhysicalPrinterCollection::load_external_printer(
- // Path to the profile source file (a G-code, an AMF or 3MF file, a config file)
- const std::string& path,
- // Name of the profile, derived from the source file name.
- const std::string& name,
- // Original name of the profile, extracted from the loaded config. Empty, if the name has not been stored.
- const std::string& original_name,
- // Config to initialize the preset from.
- const DynamicPrintConfig& config,
- // Select the preset after loading?
- bool select)
-{
- // Load the preset over a default preset, so that the missing fields are filled in from the default preset.
- DynamicPrintConfig cfg(this->default_printer().config);
- cfg.apply_only(config, cfg.keys(), true);
- // Is there a preset already loaded with the name stored inside the config?
- std::deque<PhysicalPrinter>::iterator it = this->find_printer_internal(original_name);
- bool found = it != m_printers.end() && it->name == original_name;
- if (!found) {
- // Try to match the original_name against the "renamed_from" profile names of loaded system profiles.
- / *
- it = this->find_preset_renamed(original_name);
- found = it != m_presets.end();
- * /
- }
- if (found) {
- if (profile_print_params_same(it->config, cfg)) {
- // The preset exists and it matches the values stored inside config.
- if (select)
- this->select_printer(it - m_printers.begin());
- return *it;
- }
- if (profile_host_params_same_or_anonymized(it->config, cfg) == ProfileHostParams::Anonymized) {
- // The project being loaded is anonymized. Replace the empty host keys of the loaded profile with the data from the original profile.
- // See "Octoprint Settings when Opening a .3MF file" GH issue #3244
- auto opt_update = [it, &cfg](const std::string& opt_key) {
- auto opt = it->config.option<ConfigOptionString>(opt_key);
- if (opt != nullptr)
- cfg.set_key_value(opt_key, opt->clone());
- };
- opt_update("print_host");
- opt_update("printhost_apikey");
- opt_update("printhost_cafile");
- }
- }
- // The external preset does not match an internal preset, load the external preset.
- std::string new_name;
- for (size_t idx = 0;; ++idx) {
- std::string suffix;
- if (original_name.empty()) {
- if (idx > 0)
- suffix = " (" + std::to_string(idx) + ")";
- }
- else {
- if (idx == 0)
- suffix = " (" + original_name + ")";
- else
- suffix = " (" + original_name + "-" + std::to_string(idx) + ")";
- }
- new_name = name + suffix;
- it = this->find_printer_internal(new_name);
- if (it == m_printers.end() || it->name != new_name)
- // Unique profile name. Insert a new profile.
- break;
- if (profile_print_params_same(it->config, cfg)) {
- // The preset exists and it matches the values stored inside config.
- if (select)
- this->select_printer(it - m_printers.begin());
- return *it;
- }
- // Form another profile name.
- }
- // Insert a new profile.
- PhysicalPrinter& printer = this->load_printer(path, new_name, std::move(cfg), select);
-
- return printer;
-}
-
-void PhysicalPrinterCollection::save_printer(const std::string& new_name)
-{
- // 1) Find the printer with a new_name or create a new one,
- // initialize it with the edited config.
- auto it = this->find_printer_internal(new_name);
- if (it != m_printers.end() && it->name == new_name) {
- // Preset with the same name found.
- PhysicalPrinter& printer = *it;
- // Overwriting an existing preset.
- printer.config = std::move(m_edited_printer.config);
- }
- else {
- // Creating a new printer.
- PhysicalPrinter& printer = *m_printers.insert(it, m_edited_printer);
- std::string old_name = printer.name;
- printer.name = new_name;
- }
- // 2) Activate the saved preset.
- this->select_printer_by_name(new_name, true);
- // 3) Store the active preset to disk.
- this->get_selected_preset().save();
-}
-*/
-namespace PresetUtils {
- const VendorProfile::PrinterModel* system_printer_model(const Preset &preset)
- {
- const VendorProfile::PrinterModel *out = nullptr;
- if (preset.vendor != nullptr) {
- auto *printer_model = preset.config.opt<ConfigOptionString>("printer_model");
- if (printer_model != nullptr && ! printer_model->value.empty()) {
- auto it = std::find_if(preset.vendor->models.begin(), preset.vendor->models.end(), [printer_model](const VendorProfile::PrinterModel &pm) { return pm.id == printer_model->value; });
- if (it != preset.vendor->models.end())
- out = &(*it);
- }
- }
- return out;
- }
-} // namespace PresetUtils
-
-} // namespace Slic3r