//#undef NDEBUGc #include #include "PresetBundle.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../../libslic3r/libslic3r.h" #include "../../libslic3r/PlaceholderParser.hpp" #include "../../libslic3r/Utils.hpp" namespace Slic3r { PresetBundle::PresetBundle() : prints(Preset::TYPE_PRINT, Preset::print_options()), filaments(Preset::TYPE_FILAMENT, Preset::filament_options()), printers(Preset::TYPE_PRINTER, Preset::printer_options()), m_bitmapCompatible(new wxBitmap), m_bitmapIncompatible(new wxBitmap) { ::wxInitAllImageHandlers(); // Create the ID config keys, as they are not part of the Static print config classes. this->prints.preset(0).config.opt_string("print_settings_id", true); this->filaments.preset(0).config.opt_string("filament_settings_id", true); this->printers.preset(0).config.opt_string("print_settings_id", true); // Create the "compatible printers" keys, as they are not part of the Static print config classes. this->filaments.preset(0).config.optptr("compatible_printers", true); this->prints.preset(0).config.optptr("compatible_printers", true); this->prints .load_bitmap_default("cog.png"); this->filaments.load_bitmap_default("spool.png"); this->printers .load_bitmap_default("printer_empty.png"); this->load_compatible_bitmaps(); } PresetBundle::~PresetBundle() { assert(m_bitmapCompatible != nullptr); assert(m_bitmapIncompatible != nullptr); delete m_bitmapCompatible; m_bitmapCompatible = nullptr; delete m_bitmapIncompatible; m_bitmapIncompatible = nullptr; for (std::pair &bitmap : m_mapColorToBitmap) delete bitmap.second; } void PresetBundle::setup_directories() { boost::filesystem::path dir = boost::filesystem::canonical(Slic3r::data_dir()); if (! boost::filesystem::is_directory(dir)) throw std::runtime_error(std::string("datadir does not exist: ") + Slic3r::data_dir()); std::initializer_list names = { "print", "filament", "printer" }; for (const char *name : names) { boost::filesystem::path subdir = (dir / name).make_preferred(); if (! boost::filesystem::is_directory(subdir) && ! boost::filesystem::create_directory(subdir)) throw std::runtime_error(std::string("Slic3r was unable to create its data directory at ") + subdir.string()); } } void PresetBundle::load_presets(const std::string &dir_path) { this->prints .load_presets(dir_path, "print"); this->filaments.load_presets(dir_path, "filament"); this->printers .load_presets(dir_path, "printer"); this->update_multi_material_filament_presets(); } static inline std::string remove_ini_suffix(const std::string &name) { std::string out = name; if (boost::iends_with(out, ".ini")) out.erase(out.end() - 4, out.end()); return out; } // Load selections (current print, current filaments, current printer) from config.ini // This is done just once on application start up. void PresetBundle::load_selections(const AppConfig &config) { prints.select_preset_by_name(remove_ini_suffix(config.get("presets", "print")), true); filaments.select_preset_by_name(remove_ini_suffix(config.get("presets", "filament")), true); printers.select_preset_by_name(remove_ini_suffix(config.get("presets", "printer")), true); auto *nozzle_diameter = dynamic_cast(printers.get_selected_preset().config.option("nozzle_diameter")); size_t num_extruders = nozzle_diameter->values.size(); this->set_filament_preset(0, filaments.get_selected_preset().name); for (unsigned int i = 1; i < (unsigned int)num_extruders; ++ i) { char name[64]; sprintf(name, "filament_%d", i); if (! config.has("presets", name)) break; this->set_filament_preset(i, remove_ini_suffix(config.get("presets", name))); } // Update visibility of presets based on their compatibility with the active printer. // This will switch the print or filament presets to compatible if the active presets are incompatible. this->update_compatible_with_printer(false); } // Export selections (current print, current filaments, current printer) into config.ini void PresetBundle::export_selections(AppConfig &config) { assert(filament_presets.size() >= 1); assert(filament_presets.size() > 1 || filaments.get_selected_preset().name == filament_presets.front()); config.clear_section("presets"); config.set("presets", "print", prints.get_selected_preset().name); config.set("presets", "filament", filament_presets.front()); for (int i = 1; i < filament_presets.size(); ++i) { char name[64]; sprintf(name, "filament_%d", i); config.set("presets", name, filament_presets[i]); } config.set("presets", "printer", printers.get_selected_preset().name); } void PresetBundle::export_selections(PlaceholderParser &pp) { assert(filament_presets.size() >= 1); assert(filament_presets.size() > 1 || filaments.get_selected_preset().name == filament_presets.front()); pp.set("print_preset", prints.get_selected_preset().name); pp.set("filament_preset", filament_presets); pp.set("printer_preset", printers.get_selected_preset().name); } bool PresetBundle::load_compatible_bitmaps() { const std::string path_bitmap_compatible = "flag-green-icon.png"; const std::string path_bitmap_incompatible = "flag-red-icon.png"; bool loaded_compatible = m_bitmapCompatible ->LoadFile( wxString::FromUTF8(Slic3r::var(path_bitmap_compatible).c_str()), wxBITMAP_TYPE_PNG); bool loaded_incompatible = m_bitmapIncompatible->LoadFile( wxString::FromUTF8(Slic3r::var(path_bitmap_incompatible).c_str()), wxBITMAP_TYPE_PNG); if (loaded_compatible) { prints .set_bitmap_compatible(m_bitmapCompatible); filaments.set_bitmap_compatible(m_bitmapCompatible); // printers .set_bitmap_compatible(m_bitmapCompatible); } if (loaded_incompatible) { prints .set_bitmap_incompatible(m_bitmapIncompatible); filaments.set_bitmap_incompatible(m_bitmapIncompatible); // printers .set_bitmap_incompatible(m_bitmapIncompatible); } return loaded_compatible && loaded_incompatible; } DynamicPrintConfig PresetBundle::full_config() const { DynamicPrintConfig out; out.apply(FullPrintConfig()); out.apply(this->prints.get_edited_preset().config); out.apply(this->printers.get_edited_preset().config); auto *nozzle_diameter = dynamic_cast(out.option("nozzle_diameter")); size_t num_extruders = nozzle_diameter->values.size(); if (num_extruders <= 1) { out.apply(this->filaments.get_edited_preset().config); } else { // Retrieve filament presets and build a single config object for them. // First collect the filament configurations based on the user selection of this->filament_presets. std::vector filament_configs; for (const std::string &filament_preset_name : this->filament_presets) filament_configs.emplace_back(&this->filaments.find_preset(filament_preset_name, true)->config); while (filament_configs.size() < num_extruders) filament_configs.emplace_back(&this->filaments.first_visible().config); // Option values to set a ConfigOptionVector from. std::vector filament_opts(num_extruders, nullptr); // loop through options and apply them to the resulting config. for (const t_config_option_key &key : this->filaments.default_preset().config.keys()) { if (key == "compatible_printers") continue; // Get a destination option. ConfigOption *opt_dst = out.option(key, false); if (opt_dst->is_scalar()) { // Get an option, do not create if it does not exist. const ConfigOption *opt_src = filament_configs.front()->option(key); if (opt_src != nullptr) opt_dst->set(opt_src); } else { // Setting a vector value from all filament_configs. for (size_t i = 0; i < filament_opts.size(); ++ i) filament_opts[i] = filament_configs[i]->option(key); static_cast(opt_dst)->set(filament_opts); } } } static const char *keys[] = { "perimeter", "infill", "solid_infill", "support_material", "support_material_interface" }; for (size_t i = 0; i < sizeof(keys) / sizeof(keys[0]); ++ i) { std::string key = std::string(keys[i]) + "_extruder"; auto *opt = dynamic_cast(out.option(key, false)); assert(opt != nullptr); opt->value = boost::algorithm::clamp(opt->value, 0, int(num_extruders)); } return out; } // Load an external config file containing the print, filament and printer presets. // Instead of a config file, a G-code may be loaded containing the full set of parameters. // In the future the configuration will likely be read from an AMF file as well. // If the file is loaded successfully, its print / filament / printer profiles will be activated. void PresetBundle::load_config_file(const std::string &path) { if (boost::iends_with(path, ".gcode") || boost::iends_with(path, ".g")) { DynamicPrintConfig config; config.apply(FullPrintConfig::defaults()); config.load_from_gcode(path); Preset::normalize(config); load_config_file_config(path, std::move(config)); return; } // 1) Try to load the config file into a boost property tree. boost::property_tree::ptree tree; try { boost::nowide::ifstream ifs(path); boost::property_tree::read_ini(ifs, tree); } catch (const std::ifstream::failure&) { throw std::runtime_error(std::string("The config file cannot be loaded: ") + path); } catch (const std::runtime_error&) { throw std::runtime_error(std::string("Failed loading the preset file: ") + path); } // 2) Continue based on the type of the configuration file. ConfigFileType config_file_type = guess_config_file_type(tree); switch (config_file_type) { case CONFIG_FILE_TYPE_UNKNOWN: throw std::runtime_error(std::string("Unknown configuration file type: ") + path); case CONFIG_FILE_TYPE_APP_CONFIG: throw std::runtime_error(std::string("Invalid configuration file: ") + path + ". This is an application config file."); case CONFIG_FILE_TYPE_CONFIG: { // Initialize a config from full defaults. DynamicPrintConfig config; config.apply(FullPrintConfig::defaults()); config.load(tree); Preset::normalize(config); load_config_file_config(path, std::move(config)); break; } case CONFIG_FILE_TYPE_CONFIG_BUNDLE: load_config_file_config_bundle(path, tree); break; } } // Load a config file from a boost property_tree. This is a private method called from load_config_file. void PresetBundle::load_config_file_config(const std::string &path, const DynamicPrintConfig &config) { // 1) Create a name from the file name. // Keep the suffix (.ini, .gcode, .amf, .3mf etc) to differentiate it from the normal profiles. std::string name = boost::filesystem::path(path).filename().string(); // 2) If the loading succeeded, split and load the config into print / filament / printer settings. // First load the print and printer presets. for (size_t i_group = 0; i_group < 2; ++ i_group) { PresetCollection &presets = (i_group == 0) ? this->prints : this->printers; presets.load_preset(path, name, config).is_external = true; } // 3) Now load the filaments. If there are multiple filament presets, split them and load them. auto *nozzle_diameter = dynamic_cast(config.option("nozzle_diameter")); auto *filament_diameter = dynamic_cast(config.option("filament_diameter")); size_t num_extruders = std::min(nozzle_diameter->values.size(), filament_diameter->values.size()); if (num_extruders <= 1) { this->filaments.load_preset(path, name, config).is_external = true; this->filament_presets.clear(); this->filament_presets.emplace_back(name); } else { // Split the filament presets, load each of them separately. std::vector configs(num_extruders, this->filaments.default_preset().config); // loop through options and scatter them into configs. for (const t_config_option_key &key : this->filaments.default_preset().config.keys()) { const ConfigOption *other_opt = config.option(key); if (other_opt == nullptr) continue; if (other_opt->is_scalar()) { for (size_t i = 0; i < configs.size(); ++ i) configs[i].option(key, false)->set(other_opt); } else { for (size_t i = 0; i < configs.size(); ++ i) static_cast(configs[i].option(key, false))->set_at(other_opt, 0, i); } } // Load the configs into this->filaments and make them active. this->filament_presets.clear(); for (size_t i = 0; i < configs.size(); ++ i) { char suffix[64]; if (i == 0) suffix[0] = 0; else sprintf(suffix, " (%d)", i); // Load all filament presets, but only select the first one in the preset dialog. this->filaments.load_preset(path, name + suffix, std::move(configs[i]), i == 0).is_external = true; this->filament_presets.emplace_back(name + suffix); } } } // Load the active configuration of a config bundle from a boost property_tree. This is a private method called from load_config_file. void PresetBundle::load_config_file_config_bundle(const std::string &path, const boost::property_tree::ptree &tree) { // 1) Load the config bundle into a temp data. PresetBundle tmp_bundle; tmp_bundle.load_configbundle(path); // 2) Extract active configs from the config bundle, copy them and activate them in this bundle. if (tmp_bundle.prints.get_selected_preset().is_default) this->prints.select_preset(0); else { std::string new_name = tmp_bundle.prints.get_selected_preset().name; Preset *existing = this->prints.find_preset(new_name, false); if (existing == nullptr) { // Save under the new_name. } else if (existing->config == tmp_bundle.prints.get_selected_preset().config) { // Don't save as the config exists in the current bundle and its content is the same. new_name.clear(); } else { // Generate a new unique name. } if (! new_name.empty()) this->prints.load_preset(path, new_name, std::move(tmp_bundle.prints.get_selected_preset().config)); } } // Load a config bundle file, into presets and store the loaded presets into separate files // of the local configuration directory. size_t PresetBundle::load_configbundle(const std::string &path) { // 1) Read the complete config file into a boost::property_tree. namespace pt = boost::property_tree; pt::ptree tree; boost::nowide::ifstream ifs(path); pt::read_ini(ifs, tree); // 2) Parse the property_tree, extract the active preset names and the profiles, save them into local config files. std::vector loaded_prints; std::vector loaded_filaments; std::vector loaded_printers; std::string active_print; std::vector active_filaments; std::string active_printer; size_t presets_loaded = 0; for (const auto §ion : tree) { PresetCollection *presets = nullptr; std::vector *loaded = nullptr; std::string preset_name; if (boost::starts_with(section.first, "print:")) { presets = &prints; loaded = &loaded_prints; preset_name = section.first.substr(6); } else if (boost::starts_with(section.first, "filament:")) { presets = &filaments; loaded = &loaded_filaments; preset_name = section.first.substr(9); } else if (boost::starts_with(section.first, "printer:")) { presets = &printers; loaded = &loaded_printers; preset_name = section.first.substr(8); } else if (section.first == "presets") { // Load the names of the active presets. for (auto &kvp : section.second) { if (kvp.first == "print") { active_print = kvp.second.data(); } else if (boost::starts_with(kvp.first, "filament")) { int idx = 0; if (kvp.first == "filament" || sscanf(kvp.first.c_str(), "filament_%d", &idx) == 1) { if (int(active_filaments.size()) <= idx) active_filaments.resize(idx + 1, std::string()); active_filaments[idx] = kvp.second.data(); } } else if (kvp.first == "printer") { active_printer = kvp.second.data(); } } } else if (section.first == "settings") { // Load the settings. for (auto &kvp : section.second) { if (kvp.first == "autocenter") { } } } else // Ignore an unknown section. continue; if (presets != nullptr) { // Load the print, filament or printer preset. DynamicPrintConfig config(presets->default_preset().config); for (auto &kvp : section.second) config.set_deserialize(kvp.first, kvp.second.data()); Preset::normalize(config); // Load the preset into the list of presets, save it to disk. presets->load_preset(Slic3r::config_path(presets->name(), preset_name), preset_name, std::move(config), false).save(); ++ presets_loaded; } } // 3) Activate the presets. if (! active_print.empty()) prints.select_preset_by_name(active_print, true); if (! active_printer.empty()) printers.select_preset_by_name(active_printer, true); // Activate the first filament preset. if (! active_filaments.empty() && ! active_filaments.front().empty()) filaments.select_preset_by_name(active_filaments.front(), true); this->update_multi_material_filament_presets(); for (size_t i = 0; i < std::min(this->filament_presets.size(), active_filaments.size()); ++ i) this->filament_presets[i] = filaments.find_preset(active_filaments[i], true)->name; return presets_loaded; } void PresetBundle::update_multi_material_filament_presets() { // Verify and select the filament presets. auto *nozzle_diameter = static_cast(printers.get_edited_preset().config.option("nozzle_diameter")); size_t num_extruders = nozzle_diameter->values.size(); // Verify validity of the current filament presets. for (size_t i = 0; i < std::min(this->filament_presets.size(), num_extruders); ++ i) this->filament_presets[i] = this->filaments.find_preset(this->filament_presets[i], true)->name; // Append the rest of filament presets. // if (this->filament_presets.size() < num_extruders) this->filament_presets.resize(num_extruders, this->filament_presets.empty() ? this->filaments.first_visible().name : this->filament_presets.back()); } void PresetBundle::update_compatible_with_printer(bool select_other_if_incompatible) { this->prints.update_compatible_with_printer(this->printers.get_selected_preset().name, select_other_if_incompatible); this->filaments.update_compatible_with_printer(this->printers.get_selected_preset().name, select_other_if_incompatible); if (select_other_if_incompatible) { // Verify validity of the current filament presets. for (std::string &filament_name : this->filament_presets) { Preset *preset = this->filaments.find_preset(filament_name, false); if (preset == nullptr || ! preset->is_compatible) filament_name = this->filaments.first_compatible().name; } } } void PresetBundle::export_configbundle(const std::string &path) //, const DynamicPrintConfig &settings { boost::nowide::ofstream c; c.open(path, std::ios::out | std::ios::trunc); // Put a comment at the first line including the time stamp and Slic3r version. c << "# " << Slic3r::header_slic3r_generated() << std::endl; // Export the print, filament and printer profiles. for (size_t i_group = 0; i_group < 3; ++ i_group) { const PresetCollection &presets = (i_group == 0) ? this->prints : (i_group == 1) ? this->filaments : this->printers; for (const Preset &preset : presets()) { if (preset.is_default || preset.is_external) // Only export the common presets, not external files or the default preset. continue; c << std::endl << "[" << presets.name() << ":" << preset.name << "]" << std::endl; for (const std::string &opt_key : preset.config.keys()) c << opt_key << " = " << preset.config.serialize(opt_key) << std::endl; } } // Export the names of the active presets. c << std::endl << "[presets]" << std::endl; c << "print = " << this->prints.get_selected_preset().name << std::endl; c << "printer = " << this->printers.get_selected_preset().name << std::endl; for (size_t i = 0; i < this->filament_presets.size(); ++ i) { char suffix[64]; if (i > 0) sprintf(suffix, "_%d", i); else suffix[0] = 0; c << "filament" << suffix << " = " << this->filament_presets[i] << std::endl; } #if 0 // Export the following setting values from the provided setting repository. static const char *settings_keys[] = { "autocenter" }; c << "[settings]" << std::endl; for (size_t i = 0; i < sizeof(settings_keys) / sizeof(settings_keys[0]); ++ i) c << settings_keys[i] << " = " << settings.serialize(settings_keys[i]) << std::endl; #endif c.close(); } // Set the filament preset name. As the name could come from the UI selection box, // an optional "(modified)" suffix will be removed from the filament name. void PresetBundle::set_filament_preset(size_t idx, const std::string &name) { if (idx >= filament_presets.size()) filament_presets.resize(idx + 1, filaments.default_preset().name); filament_presets[idx] = Preset::remove_suffix_modified(name); } static inline int hex_digit_to_int(const char c) { return (c >= '0' && c <= '9') ? int(c - '0') : (c >= 'A' && c <= 'F') ? int(c - 'A') + 10 : (c >= 'a' && c <= 'f') ? int(c - 'a') + 10 : -1; } static inline bool parse_color(const std::string &scolor, unsigned char *rgb_out) { rgb_out[0] = rgb_out[1] = rgb_out[2] = 0; const char *c = scolor.data() + 1; if (scolor.size() != 7 || scolor.front() != '#') return false; for (size_t i = 0; i < 3; ++ i) { int digit1 = hex_digit_to_int(*c ++); int digit2 = hex_digit_to_int(*c ++); if (digit1 == -1 || digit2 == -1) return false; rgb_out[i] = (unsigned char)(digit1 * 16 + digit2); } return true; } void PresetBundle::update_platter_filament_ui(unsigned int idx_extruder, wxBitmapComboBox *ui) { if (ui == nullptr) return; unsigned char rgb[3]; std::string extruder_color = this->printers.get_edited_preset().config.opt_string("extruder_colour", idx_extruder); if (! parse_color(extruder_color, rgb)) // Extruder color is not defined. extruder_color.clear(); // Fill in the list from scratch. ui->Freeze(); ui->Clear(); const Preset *selected_preset = this->filaments.find_preset(this->filament_presets[idx_extruder]); // Show wide icons if the currently selected preset is not compatible with the current printer, // and draw a red flag in front of the selected preset. bool wide_icons = selected_preset != nullptr && ! selected_preset->is_compatible && m_bitmapIncompatible != nullptr; assert(selected_preset != nullptr); for (int i = this->filaments().front().is_visible ? 0 : 1; i < int(this->filaments().size()); ++ i) { const Preset &preset = this->filaments.preset(i); bool selected = this->filament_presets[idx_extruder] == preset.name; if (! preset.is_visible || (! preset.is_compatible && ! selected)) continue; // Assign an extruder color to the selected item if the extruder color is defined. std::string filament_rgb = preset.config.opt_string("filament_colour", 0); std::string extruder_rgb = (selected && !extruder_color.empty()) ? extruder_color : filament_rgb; bool single_bar = filament_rgb == extruder_rgb; std::string bitmap_key = single_bar ? filament_rgb : filament_rgb + extruder_rgb; // If the filament preset is not compatible and there is a "red flag" icon loaded, show it left // to the filament color image. if (wide_icons) bitmap_key += preset.is_compatible ? "comp" : "notcomp"; auto it = m_mapColorToBitmap.find(bitmap_key); wxBitmap *bitmap = (it == m_mapColorToBitmap.end()) ? nullptr : it->second; if (bitmap == nullptr) { // Create the bitmap with color bars. bitmap = new wxBitmap((wide_icons ? 16 : 0) + 24, 16); bitmap->UseAlpha(); wxMemoryDC memDC; memDC.SelectObject(*bitmap); memDC.SetBackground(*wxTRANSPARENT_BRUSH); memDC.Clear(); if (wide_icons && ! preset.is_compatible) // Paint the red flag. memDC.DrawBitmap(*m_bitmapIncompatible, 0, 0, false); // Paint the color bars. parse_color(filament_rgb, rgb); wxImage image(24, 16); image.InitAlpha(); unsigned char* imgdata = image.GetData(); unsigned char* imgalpha = image.GetAlpha(); for (size_t i = 0; i < image.GetWidth() * image.GetHeight(); ++ i) { *imgdata ++ = rgb[0]; *imgdata ++ = rgb[1]; *imgdata ++ = rgb[2]; *imgalpha ++ = wxALPHA_OPAQUE; } if (! single_bar) { parse_color(extruder_rgb, rgb); imgdata = image.GetData(); for (size_t r = 0; r < 16; ++ r) { imgdata = image.GetData() + r * image.GetWidth() * 3; for (size_t c = 0; c < 16; ++ c) { *imgdata ++ = rgb[0]; *imgdata ++ = rgb[1]; *imgdata ++ = rgb[2]; } } } memDC.DrawBitmap(wxBitmap(image), wide_icons ? 16 : 0, 0, false); memDC.SelectObject(wxNullBitmap); m_mapColorToBitmap[bitmap_key] = bitmap; } ui->Append(wxString::FromUTF8((preset.name + (preset.is_dirty ? Preset::suffix_modified() : "")).c_str()), (bitmap == 0) ? wxNullBitmap : *bitmap); if (selected) ui->SetSelection(ui->GetCount() - 1); } ui->Thaw(); } void PresetBundle::set_default_suppressed(bool default_suppressed) { prints.set_default_suppressed(default_suppressed); filaments.set_default_suppressed(default_suppressed); printers.set_default_suppressed(default_suppressed); } } // namespace Slic3r