mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-26 02:01:12 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			2580 lines
		
	
	
	
		
			97 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			2580 lines
		
	
	
	
		
			97 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // FIXME: extract absolute units -> em
 | |
| 
 | |
| #include "ConfigWizard_private.hpp"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <numeric>
 | |
| #include <utility>
 | |
| #include <unordered_map>
 | |
| #include <stdexcept>
 | |
| #include <boost/format.hpp>
 | |
| #include <boost/log/trivial.hpp>
 | |
| #include <boost/algorithm/string/predicate.hpp>
 | |
| #include <boost/nowide/convert.hpp>
 | |
| 
 | |
| #include <wx/settings.h>
 | |
| #include <wx/stattext.h>
 | |
| #include <wx/textctrl.h>
 | |
| #include <wx/dcclient.h>
 | |
| #include <wx/statbmp.h>
 | |
| #include <wx/checkbox.h>
 | |
| #include <wx/statline.h>
 | |
| #include <wx/dataview.h>
 | |
| #include <wx/notebook.h>
 | |
| #include <wx/listbook.h>
 | |
| #include <wx/display.h>
 | |
| #include <wx/filefn.h>
 | |
| #include <wx/wupdlock.h>
 | |
| #include <wx/debug.h>
 | |
| 
 | |
| #ifdef _MSW_DARK_MODE
 | |
| #include <wx/msw/dark_mode.h>
 | |
| #endif // _MSW_DARK_MODE
 | |
| 
 | |
| #include "libslic3r/Platform.hpp"
 | |
| #include "libslic3r/Utils.hpp"
 | |
| #include "libslic3r/Config.hpp"
 | |
| #include "libslic3r/libslic3r.h"
 | |
| #include "libslic3r/Model.hpp"
 | |
| #include "GUI.hpp"
 | |
| #include "GUI_App.hpp"
 | |
| #include "GUI_Utils.hpp"
 | |
| #include "I18N.hpp"
 | |
| #include "Field.hpp"
 | |
| #include "DesktopIntegrationDialog.hpp"
 | |
| #include "slic3r/Config/Snapshot.hpp"
 | |
| #include "slic3r/Utils/PresetUpdater.hpp"
 | |
| #include "format.hpp"
 | |
| #include "MsgDialog.hpp"
 | |
| #include "UnsavedChangesDialog.hpp"
 | |
| 
 | |
| #if defined(__linux__) && defined(__WXGTK3__)
 | |
| #define wxLinux_gtk3 true
 | |
| #else
 | |
| #define wxLinux_gtk3 false
 | |
| #endif //defined(__linux__) && defined(__WXGTK3__)
 | |
| 
 | |
| namespace Slic3r {
 | |
| namespace GUI {
 | |
| 
 | |
| 
 | |
| using Config::Snapshot;
 | |
| using Config::SnapshotDB;
 | |
| 
 | |
| 
 | |
| // Configuration data structures extensions needed for the wizard
 | |
| //BBS: set BBL as default
 | |
| bool Bundle::load(fs::path source_path, bool ais_in_resources, bool ais_bbl_bundle)
 | |
| {
 | |
|     this->preset_bundle = std::make_unique<PresetBundle>();
 | |
|     this->is_in_resources = ais_in_resources;
 | |
|     this->is_bbl_bundle = ais_bbl_bundle;
 | |
| 
 | |
|     std::string path_string = source_path.string();
 | |
|     std::string parent_path = source_path.parent_path().string();
 | |
|     //BBS: add json logic for vendor bundles
 | |
|     std::string vendor_name = source_path.filename().string();
 | |
|     if (Slic3r::is_json_file(path_string)) {
 | |
|         // Remove the .json suffix.
 | |
|         vendor_name.erase(vendor_name.size() - 5);
 | |
|     }
 | |
|     else
 | |
|         return false;
 | |
| 
 | |
|     // Throw when parsing invalid configuration. Only valid configuration is supposed to be provided over the air.
 | |
|     //BBS: add json logic for vendor bundles
 | |
|     auto [config_substitutions, presets_loaded] = preset_bundle->load_vendor_configs_from_json(
 | |
|         parent_path, vendor_name, PresetBundle::LoadConfigBundleAttribute::LoadSystem, ForwardCompatibilitySubstitutionRule::Disable);
 | |
|     UNUSED(config_substitutions);
 | |
|     // No substitutions shall be reported when loading a system config bundle, no substitutions are allowed.
 | |
|     assert(config_substitutions.empty());
 | |
|     auto first_vendor = preset_bundle->vendors.begin();
 | |
|     if (first_vendor == preset_bundle->vendors.end()) {
 | |
|         BOOST_LOG_TRIVIAL(error) << boost::format("Vendor bundle: `%1%`: No vendor information defined, cannot install.") % path_string;
 | |
|         return false;
 | |
|     }
 | |
|     if (presets_loaded == 0) {
 | |
|         BOOST_LOG_TRIVIAL(error) << boost::format("Vendor bundle: `%1%`: No profile loaded.") % path_string;
 | |
|         return false;
 | |
|     } 
 | |
| 
 | |
|     BOOST_LOG_TRIVIAL(trace) << boost::format("Vendor bundle: `%1%`: %2% profiles loaded.") % path_string % presets_loaded;
 | |
|     this->vendor_profile = &first_vendor->second;
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| //BBS: set BBL as default
 | |
| Bundle::Bundle(Bundle &&other)
 | |
|     : preset_bundle(std::move(other.preset_bundle))
 | |
|     , vendor_profile(other.vendor_profile)
 | |
|     , is_in_resources(other.is_in_resources)
 | |
|     , is_bbl_bundle(other.is_bbl_bundle)
 | |
| {
 | |
|     other.vendor_profile = nullptr;
 | |
| }
 | |
| 
 | |
| BundleMap BundleMap::load()
 | |
| {
 | |
|     BundleMap res;
 | |
| 
 | |
|     //BBS: change directories by design
 | |
|     const auto vendor_dir = (boost::filesystem::path(Slic3r::data_dir()) / PRESET_SYSTEM_DIR).make_preferred();
 | |
|     const auto rsrc_vendor_dir = (boost::filesystem::path(resources_dir()) / "profiles").make_preferred();
 | |
| 
 | |
|     //BBS: add BBL as default
 | |
|     //BBS: add json logic for vendor bundle
 | |
|     auto bbl_bundle_path = (vendor_dir / PresetBundle::BBL_BUNDLE).replace_extension(".json");
 | |
|     auto bbl_bundle_rsrc = false;
 | |
|     if (!boost::filesystem::exists(bbl_bundle_path)) {
 | |
|         bbl_bundle_path = (rsrc_vendor_dir / PresetBundle::BBL_BUNDLE).replace_extension(".json");
 | |
|         bbl_bundle_rsrc = true;
 | |
|     }
 | |
|     {
 | |
|         Bundle bbl_bundle;
 | |
|         if (bbl_bundle.load(std::move(bbl_bundle_path), bbl_bundle_rsrc, true))
 | |
|             res.emplace(PresetBundle::BBL_BUNDLE, std::move(bbl_bundle));
 | |
|     }
 | |
| 
 | |
|     // Load the other bundles in the datadir/vendor directory
 | |
|     // and then additionally from resources/profiles.
 | |
|     bool is_in_resources = false;
 | |
|     for (auto dir : { &vendor_dir, &rsrc_vendor_dir }) {
 | |
|         for (const auto &dir_entry : boost::filesystem::directory_iterator(*dir)) {
 | |
|             //BBS: add json logic for vendor bundle
 | |
|             if (Slic3r::is_json_file(dir_entry.path().string())) {
 | |
|                 std::string id = dir_entry.path().stem().string();  // stem() = filename() without the trailing ".json" part
 | |
| 
 | |
|                 // Don't load this bundle if we've already loaded it.
 | |
|                 if (res.find(id) != res.end()) { continue; }
 | |
| 
 | |
|                 Bundle bundle;
 | |
|                 if (bundle.load(dir_entry.path(), is_in_resources))
 | |
|                     res.emplace(std::move(id), std::move(bundle));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         is_in_resources = true;
 | |
|     }
 | |
| 
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| Bundle& BundleMap::bbl_bundle()
 | |
| {
 | |
|     //BBS: add BBL as default
 | |
|     auto it = find(PresetBundle::BBL_BUNDLE);
 | |
|     if (it == end()) {
 | |
|         throw Slic3r::RuntimeError("ConfigWizard: Internal error in BundleMap: BBL_BUNDLE not loaded");
 | |
|     }
 | |
| 
 | |
|     return it->second;
 | |
| }
 | |
| 
 | |
| const Bundle& BundleMap::bbl_bundle() const
 | |
| {
 | |
|     return const_cast<BundleMap*>(this)->bbl_bundle();
 | |
| }
 | |
| 
 | |
| 
 | |
| // Printer model picker GUI control
 | |
| 
 | |
| struct PrinterPickerEvent : public wxEvent
 | |
| {
 | |
|     std::string vendor_id;
 | |
|     std::string model_id;
 | |
|     std::string variant_name;
 | |
|     bool enable;
 | |
| 
 | |
|     PrinterPickerEvent(wxEventType eventType, int winid, std::string vendor_id, std::string model_id, std::string variant_name, bool enable)
 | |
|         : wxEvent(winid, eventType)
 | |
|         , vendor_id(std::move(vendor_id))
 | |
|         , model_id(std::move(model_id))
 | |
|         , variant_name(std::move(variant_name))
 | |
|         , enable(enable)
 | |
|     {}
 | |
| 
 | |
|     virtual wxEvent *Clone() const
 | |
|     {
 | |
|         return new PrinterPickerEvent(*this);
 | |
|     }
 | |
| };
 | |
| 
 | |
| wxDEFINE_EVENT(EVT_PRINTER_PICK, PrinterPickerEvent);
 | |
| 
 | |
| const std::string PrinterPicker::PRINTER_PLACEHOLDER = "printer_placeholder.png";
 | |
| 
 | |
| PrinterPicker::PrinterPicker(wxWindow *parent, const VendorProfile &vendor, wxString title, size_t max_cols, const AppConfig &appconfig, const ModelFilter &filter)
 | |
|     : wxPanel(parent)
 | |
|     , vendor_id(vendor.id)
 | |
|     , width(0)
 | |
| {
 | |
|     wxGetApp().UpdateDarkUI(this);
 | |
|     const auto &models = vendor.models;
 | |
| 
 | |
|     auto *sizer = new wxBoxSizer(wxVERTICAL);
 | |
| 
 | |
|     const auto font_title = GetFont().MakeBold().Scaled(1.3f);
 | |
|     const auto font_name = GetFont().MakeBold();
 | |
|     const auto font_alt_nozzle = GetFont().Scaled(0.9f);
 | |
| 
 | |
|     // wxGrid appends widgets by rows, but we need to construct them in columns.
 | |
|     // These vectors are used to hold the elements so that they can be appended in the right order.
 | |
|     std::vector<wxStaticText*> titles;
 | |
|     std::vector<wxStaticBitmap*> bitmaps;
 | |
|     std::vector<wxPanel*> variants_panels;
 | |
| 
 | |
|     int max_row_width = 0;
 | |
|     int current_row_width = 0;
 | |
| 
 | |
|     bool is_variants = false;
 | |
| 
 | |
|     for (const auto &model : models) {
 | |
|         if (! filter(model)) { continue; }
 | |
| 
 | |
|         wxBitmap bitmap;
 | |
|         int bitmap_width = 0;
 | |
|         auto load_bitmap = [](const wxString& bitmap_file, wxBitmap& bitmap, int& bitmap_width)->bool {
 | |
|             if (wxFileExists(bitmap_file)) {
 | |
|                 bitmap.LoadFile(bitmap_file, wxBITMAP_TYPE_PNG);
 | |
|                 bitmap_width = bitmap.GetWidth();
 | |
|                 return true;
 | |
|             }
 | |
|             return false;
 | |
|         };
 | |
|         if (!load_bitmap(GUI::from_u8(Slic3r::data_dir() + "/vendor/" + vendor.id + "/" + model.id + "_thumbnail.png"), bitmap, bitmap_width)) {
 | |
|             if (!load_bitmap(GUI::from_u8(Slic3r::resources_dir() + "/profiles/" + vendor.id + "/" + model.id + "_thumbnail.png"), bitmap, bitmap_width)) {
 | |
|                 BOOST_LOG_TRIVIAL(warning) << boost::format("Can't find bitmap file `%1%` for vendor `%2%`, printer `%3%`, using placeholder icon instead")
 | |
|                     % (Slic3r::resources_dir() + "/profiles/" + vendor.id + "/" + model.id + "_thumbnail.png")
 | |
|                     % vendor.id
 | |
|                     % model.id;
 | |
|                 load_bitmap(Slic3r::var(PRINTER_PLACEHOLDER), bitmap, bitmap_width);
 | |
|             }
 | |
|         }
 | |
|         auto *title = new wxStaticText(this, wxID_ANY, from_u8(model.name), wxDefaultPosition, wxDefaultSize, wxALIGN_LEFT);
 | |
|         title->SetFont(font_name);
 | |
|         const int wrap_width = std::max((int)MODEL_MIN_WRAP, bitmap_width);
 | |
|         title->Wrap(wrap_width);
 | |
| 
 | |
|         current_row_width += wrap_width;
 | |
|         if (titles.size() % max_cols == max_cols - 1) {
 | |
|             max_row_width = std::max(max_row_width, current_row_width);
 | |
|             current_row_width = 0;
 | |
|         }
 | |
| 
 | |
|         titles.push_back(title);
 | |
| 
 | |
|         auto *bitmap_widget = new wxStaticBitmap(this, wxID_ANY, bitmap);
 | |
|         bitmaps.push_back(bitmap_widget);
 | |
| 
 | |
|         auto *variants_panel = new wxPanel(this);
 | |
|         wxGetApp().UpdateDarkUI(variants_panel);
 | |
|         auto *variants_sizer = new wxBoxSizer(wxVERTICAL);
 | |
|         variants_panel->SetSizer(variants_sizer);
 | |
|         const auto model_id = model.id;
 | |
| 
 | |
|         for (size_t i = 0; i < model.variants.size(); i++) {
 | |
|             const auto &variant = model.variants[i];
 | |
| 
 | |
|             const auto label = model.technology == ptFFF
 | |
|                 ? from_u8((boost::format("%1% %2% %3%") % variant.name % _utf8(L("mm")) % _utf8(L("nozzle"))).str())
 | |
|                 : from_u8(model.name);
 | |
| 
 | |
|             if (i == 1) {
 | |
|                 auto *alt_label = new wxStaticText(variants_panel, wxID_ANY, _L("Alternate nozzles:"));
 | |
|                 alt_label->SetFont(font_alt_nozzle);
 | |
|                 variants_sizer->Add(alt_label, 0, wxBOTTOM, 3);
 | |
|                 is_variants = true;
 | |
|             }
 | |
| 
 | |
|             auto *cbox = new Checkbox(variants_panel, label, model_id, variant.name);
 | |
|             i == 0 ? cboxes.push_back(cbox) : cboxes_alt.push_back(cbox);
 | |
| 
 | |
|             const bool enabled = appconfig.get_variant(vendor.id, model_id, variant.name);
 | |
|             cbox->SetValue(enabled);
 | |
| 
 | |
|             variants_sizer->Add(cbox, 0, wxBOTTOM, 3);
 | |
| 
 | |
|             cbox->Bind(wxEVT_CHECKBOX, [this, cbox](wxCommandEvent &event) {
 | |
|                 on_checkbox(cbox, event.IsChecked());
 | |
|             });
 | |
|         }
 | |
| 
 | |
|         variants_panels.push_back(variants_panel);
 | |
|     }
 | |
| 
 | |
|     width = std::max(max_row_width, current_row_width);
 | |
| 
 | |
|     const size_t cols = std::min(max_cols, titles.size());
 | |
| 
 | |
|     auto *printer_grid = new wxFlexGridSizer(cols, 0, 20);
 | |
|     printer_grid->SetFlexibleDirection(wxVERTICAL | wxHORIZONTAL);
 | |
| 
 | |
|     if (titles.size() > 0) {
 | |
|         const size_t odd_items = titles.size() % cols;
 | |
| 
 | |
|         for (size_t i = 0; i < titles.size() - odd_items; i += cols) {
 | |
|             for (size_t j = i; j < i + cols; j++) { printer_grid->Add(bitmaps[j], 0, wxBOTTOM, 20); }
 | |
|             for (size_t j = i; j < i + cols; j++) { printer_grid->Add(titles[j], 0, wxBOTTOM, 3); }
 | |
|             for (size_t j = i; j < i + cols; j++) { printer_grid->Add(variants_panels[j]); }
 | |
| 
 | |
|             // Add separator space to multiliners
 | |
|             if (titles.size() > cols) {
 | |
|                 for (size_t j = i; j < i + cols; j++) { printer_grid->Add(1, 30); }
 | |
|             }
 | |
|         }
 | |
|         if (odd_items > 0) {
 | |
|             const size_t rem = titles.size() - odd_items;
 | |
| 
 | |
|             for (size_t i = rem; i < titles.size(); i++) { printer_grid->Add(bitmaps[i], 0, wxBOTTOM, 20); }
 | |
|             for (size_t i = 0; i < cols - odd_items; i++) { printer_grid->AddSpacer(1); }
 | |
|             for (size_t i = rem; i < titles.size(); i++) { printer_grid->Add(titles[i], 0, wxBOTTOM, 3); }
 | |
|             for (size_t i = 0; i < cols - odd_items; i++) { printer_grid->AddSpacer(1); }
 | |
|             for (size_t i = rem; i < titles.size(); i++) { printer_grid->Add(variants_panels[i]); }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     auto *title_sizer = new wxBoxSizer(wxHORIZONTAL);
 | |
|     if (! title.IsEmpty()) {
 | |
|         auto *title_widget = new wxStaticText(this, wxID_ANY, title);
 | |
|         title_widget->SetFont(font_title);
 | |
|         title_sizer->Add(title_widget);
 | |
|     }
 | |
|     title_sizer->AddStretchSpacer();
 | |
| 
 | |
|     if (titles.size() > 1 || is_variants) {
 | |
|         // It only makes sense to add the All / None buttons if there's multiple printers
 | |
|         // All Standard button is added when there are more variants for at least one printer
 | |
|         auto *sel_all_std = new wxButton(this, wxID_ANY, titles.size() > 1 ? _L("All standard") : _L("Standard"));
 | |
|         auto *sel_all = new wxButton(this, wxID_ANY, _L("All"));
 | |
|         auto *sel_none = new wxButton(this, wxID_ANY, _L("None"));
 | |
|         if (is_variants) 
 | |
|             sel_all_std->Bind(wxEVT_BUTTON, [this](const wxCommandEvent& event) { this->select_all(true, false); });
 | |
|         sel_all->Bind(wxEVT_BUTTON, [this](const wxCommandEvent &event) { this->select_all(true, true); });
 | |
|         sel_none->Bind(wxEVT_BUTTON, [this](const wxCommandEvent &event) { this->select_all(false); });
 | |
|         if (is_variants) 
 | |
|             title_sizer->Add(sel_all_std, 0, wxRIGHT, BTN_SPACING);
 | |
|         title_sizer->Add(sel_all, 0, wxRIGHT, BTN_SPACING);
 | |
|         title_sizer->Add(sel_none);
 | |
| 
 | |
|         wxGetApp().UpdateDarkUI(sel_all_std);
 | |
|         wxGetApp().UpdateDarkUI(sel_all);
 | |
|         wxGetApp().UpdateDarkUI(sel_none);
 | |
| 
 | |
|         // fill button indexes used later for buttons rescaling
 | |
|         if (is_variants)
 | |
|             m_button_indexes = { sel_all_std->GetId(), sel_all->GetId(), sel_none->GetId() };
 | |
|         else {
 | |
|             sel_all_std->Destroy();
 | |
|             m_button_indexes = { sel_all->GetId(), sel_none->GetId() };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     sizer->Add(title_sizer, 0, wxEXPAND | wxBOTTOM, BTN_SPACING);
 | |
|     sizer->Add(printer_grid);
 | |
| 
 | |
|     SetSizer(sizer);
 | |
| }
 | |
| 
 | |
| PrinterPicker::PrinterPicker(wxWindow *parent, const VendorProfile &vendor, wxString title, size_t max_cols, const AppConfig &appconfig)
 | |
|     : PrinterPicker(parent, vendor, std::move(title), max_cols, appconfig, [](const VendorProfile::PrinterModel&) { return true; })
 | |
| {}
 | |
| 
 | |
| void PrinterPicker::select_all(bool select, bool alternates)
 | |
| {
 | |
|     for (const auto &cb : cboxes) {
 | |
|         if (cb->GetValue() != select) {
 | |
|             cb->SetValue(select);
 | |
|             on_checkbox(cb, select);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (! select) { alternates = false; }
 | |
| 
 | |
|     for (const auto &cb : cboxes_alt) {
 | |
|         if (cb->GetValue() != alternates) {
 | |
|             cb->SetValue(alternates);
 | |
|             on_checkbox(cb, alternates);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void PrinterPicker::select_one(size_t i, bool select)
 | |
| {
 | |
|     if (i < cboxes.size() && cboxes[i]->GetValue() != select) {
 | |
|         cboxes[i]->SetValue(select);
 | |
|         on_checkbox(cboxes[i], select);
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool PrinterPicker::any_selected() const
 | |
| {
 | |
|     for (const auto &cb : cboxes) {
 | |
|         if (cb->GetValue()) { return true; }
 | |
|     }
 | |
| 
 | |
|     for (const auto &cb : cboxes_alt) {
 | |
|         if (cb->GetValue()) { return true; }
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| std::set<std::string> PrinterPicker::get_selected_models() const 
 | |
| {
 | |
|     std::set<std::string> ret_set;
 | |
| 
 | |
|     for (const auto& cb : cboxes)
 | |
|         if (cb->GetValue())
 | |
|             ret_set.emplace(cb->model);
 | |
| 
 | |
|     for (const auto& cb : cboxes_alt)
 | |
|         if (cb->GetValue())
 | |
|             ret_set.emplace(cb->model);
 | |
| 
 | |
|     return ret_set;
 | |
| }
 | |
| 
 | |
| void PrinterPicker::on_checkbox(const Checkbox *cbox, bool checked)
 | |
| {
 | |
|     PrinterPickerEvent evt(EVT_PRINTER_PICK, GetId(), vendor_id, cbox->model, cbox->variant, checked);
 | |
|     AddPendingEvent(evt);
 | |
| }
 | |
| 
 | |
| 
 | |
| // Wizard page base
 | |
| 
 | |
| ConfigWizardPage::ConfigWizardPage(ConfigWizard *parent, wxString title, wxString shortname, unsigned indent)
 | |
|     : wxPanel(parent->p->hscroll)
 | |
|     , parent(parent)
 | |
|     , shortname(std::move(shortname))
 | |
|     , indent(indent)
 | |
| {
 | |
|     wxGetApp().UpdateDarkUI(this);
 | |
| 
 | |
|     auto *sizer = new wxBoxSizer(wxVERTICAL);
 | |
| 
 | |
|     auto *text = new wxStaticText(this, wxID_ANY, std::move(title), wxDefaultPosition, wxDefaultSize, wxALIGN_LEFT);
 | |
|     const auto font = GetFont().MakeBold().Scaled(1.5);
 | |
|     text->SetFont(font);
 | |
|     sizer->Add(text, 0, wxALIGN_LEFT, 0);
 | |
|     sizer->AddSpacer(10);
 | |
| 
 | |
|     content = new wxBoxSizer(wxVERTICAL);
 | |
|     sizer->Add(content, 1, wxEXPAND);
 | |
| 
 | |
|     SetSizer(sizer);
 | |
| 
 | |
|     // There is strange layout on Linux with GTK3, 
 | |
|     // So, non-active pages will be hidden later, on wxEVT_SHOW, after completed Layout() for all pages 
 | |
|     if (!wxLinux_gtk3)
 | |
|         this->Hide();
 | |
| 
 | |
|     Bind(wxEVT_SIZE, [this](wxSizeEvent &event) {
 | |
|         this->Layout();
 | |
|         event.Skip();
 | |
|     });
 | |
| }
 | |
| 
 | |
| ConfigWizardPage::~ConfigWizardPage() {}
 | |
| 
 | |
| wxStaticText* ConfigWizardPage::append_text(wxString text)
 | |
| {
 | |
|     auto *widget = new wxStaticText(this, wxID_ANY, text, wxDefaultPosition, wxDefaultSize, wxALIGN_LEFT);
 | |
|     widget->Wrap(WRAP_WIDTH);
 | |
|     widget->SetMinSize(wxSize(WRAP_WIDTH, -1));
 | |
|     append(widget);
 | |
|     return widget;
 | |
| }
 | |
| 
 | |
| void ConfigWizardPage::append_spacer(int space)
 | |
| {
 | |
|     // FIXME: scaling
 | |
|     content->AddSpacer(space);
 | |
| }
 | |
| 
 | |
| // Wizard pages
 | |
| 
 | |
| PageWelcome::PageWelcome(ConfigWizard *parent)
 | |
|     : ConfigWizardPage(parent, from_u8((boost::format(
 | |
| #ifdef __APPLE__
 | |
|             _utf8(L("Welcome to the %s Configuration Assistant"))
 | |
| #else
 | |
|             _utf8(L("Welcome to the %s Configuration Wizard"))
 | |
| #endif
 | |
|             ) % SLIC3R_APP_NAME).str()), _L("Welcome"))
 | |
|     , welcome_text(append_text(from_u8((boost::format(
 | |
|         _utf8(L("Hello, welcome to %s! This %s helps you with the initial configuration; just a few settings and you will be ready to print.")))
 | |
|         % SLIC3R_APP_NAME
 | |
|         % _utf8(ConfigWizard::name())).str())
 | |
|     ))
 | |
|     , cbox_reset(append(
 | |
|         new wxCheckBox(this, wxID_ANY, _L("Remove user profiles (a snapshot will be taken beforehand)"))
 | |
|     ))
 | |
|     , cbox_integrate(append(
 | |
|         new wxCheckBox(this, wxID_ANY, _L("Perform desktop integration (Sets this binary to be searchable by the system)."))
 | |
|     ))
 | |
| {
 | |
|     welcome_text->Hide();
 | |
|     cbox_reset->Hide();
 | |
|     cbox_integrate->Hide();    
 | |
| }
 | |
| 
 | |
| void PageWelcome::set_run_reason(ConfigWizard::RunReason run_reason)
 | |
| {
 | |
|     const bool data_empty = run_reason == ConfigWizard::RR_DATA_EMPTY;
 | |
|     welcome_text->Show(data_empty);
 | |
|     cbox_reset->Show(!data_empty);
 | |
| #if defined(__linux__) && defined(SLIC3R_DESKTOP_INTEGRATION)
 | |
|     if (!DesktopIntegrationDialog::is_integrated())
 | |
|         cbox_integrate->Show(true);
 | |
|     else
 | |
|         cbox_integrate->Hide();
 | |
| #else
 | |
|     cbox_integrate->Hide();
 | |
| #endif
 | |
| }
 | |
| 
 | |
| 
 | |
| PagePrinters::PagePrinters(ConfigWizard *parent,
 | |
|     wxString title,
 | |
|     wxString shortname,
 | |
|     const VendorProfile &vendor,
 | |
|     unsigned indent,
 | |
|     Technology technology)
 | |
|     : ConfigWizardPage(parent, std::move(title), std::move(shortname), indent)
 | |
|     , technology(technology)
 | |
|     , install(false)   // only used for 3rd party vendors
 | |
| {
 | |
|     enum {
 | |
|         COL_SIZE = 200,
 | |
|     };
 | |
| 
 | |
|     AppConfig *appconfig = &this->wizard_p()->appconfig_new;
 | |
| 
 | |
|     const auto families = vendor.families();
 | |
|     for (const auto &family : families) {
 | |
|         const auto filter = [&](const VendorProfile::PrinterModel &model) {
 | |
|             return ((model.technology == ptFFF && technology & T_FFF)
 | |
|                     || (model.technology == ptSLA && technology & T_SLA))
 | |
|                 && model.family == family;
 | |
|         };
 | |
| 
 | |
|         if (std::find_if(vendor.models.begin(), vendor.models.end(), filter) == vendor.models.end()) {
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         const auto picker_title = family.empty() ? wxString() : from_u8((boost::format(_utf8(L("%s Family"))) % family).str());
 | |
|         auto *picker = new PrinterPicker(this, vendor, picker_title, MAX_COLS, *appconfig, filter);
 | |
| 
 | |
|         picker->Bind(EVT_PRINTER_PICK, [this, appconfig](const PrinterPickerEvent &evt) {
 | |
|             appconfig->set_variant(evt.vendor_id, evt.model_id, evt.variant_name, evt.enable);
 | |
|             wizard_p()->on_printer_pick(this, evt);
 | |
|         });
 | |
| 
 | |
|         append(new StaticLine(this));
 | |
| 
 | |
|         append(picker);
 | |
|         printer_pickers.push_back(picker);
 | |
|         has_printers = true;
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| void PagePrinters::select_all(bool select, bool alternates)
 | |
| {
 | |
|     for (auto picker : printer_pickers) {
 | |
|         picker->select_all(select, alternates);
 | |
|     }
 | |
| }
 | |
| 
 | |
| int PagePrinters::get_width() const
 | |
| {
 | |
|     return std::accumulate(printer_pickers.begin(), printer_pickers.end(), 0,
 | |
|         [](int acc, const PrinterPicker *picker) { return std::max(acc, picker->get_width()); });
 | |
| }
 | |
| 
 | |
| bool PagePrinters::any_selected() const
 | |
| {
 | |
|     for (const auto *picker : printer_pickers) {
 | |
|         if (picker->any_selected()) { return true; }
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| std::set<std::string> PagePrinters::get_selected_models()
 | |
| {
 | |
|     std::set<std::string> ret_set;
 | |
| 
 | |
|     for (const auto *picker : printer_pickers)
 | |
|     {
 | |
|         std::set<std::string> tmp_models = picker->get_selected_models();
 | |
|         ret_set.insert(tmp_models.begin(), tmp_models.end());
 | |
|     }
 | |
| 
 | |
|     return ret_set;
 | |
| }
 | |
| 
 | |
| void PagePrinters::set_run_reason(ConfigWizard::RunReason run_reason)
 | |
| {
 | |
|     //BBS: add BBL as default
 | |
|     if (is_primary_printer_page
 | |
|         && (run_reason == ConfigWizard::RR_DATA_EMPTY || run_reason == ConfigWizard::RR_DATA_LEGACY)
 | |
|         && printer_pickers.size() > 0 
 | |
|         && printer_pickers[0]->vendor_id == PresetBundle::BBL_BUNDLE) {
 | |
|         //BBS: select alll bbs machine by default
 | |
|         //printer_pickers[0]->select_one(0, true);
 | |
|         printer_pickers[0]->select_all(true);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| const std::string PageMaterials::EMPTY;
 | |
| 
 | |
| PageMaterials::PageMaterials(ConfigWizard *parent, Materials *materials, wxString title, wxString shortname, wxString list1name)
 | |
|     : ConfigWizardPage(parent, std::move(title), std::move(shortname))
 | |
|     , materials(materials)
 | |
| 	, list_printer(new  StringList(this, wxLB_MULTIPLE))
 | |
|     , list_type(new StringList(this))
 | |
|     , list_vendor(new StringList(this))
 | |
|     , list_profile(new PresetList(this))
 | |
| {
 | |
|     append_spacer(VERTICAL_SPACING);
 | |
| 
 | |
|     const int em = parent->em_unit();
 | |
|     const int list_h = 30*em;
 | |
| 
 | |
| 
 | |
| 	list_printer->SetMinSize(wxSize(23*em, list_h));
 | |
|     list_type->SetMinSize(wxSize(13*em, list_h));
 | |
|     list_vendor->SetMinSize(wxSize(13*em, list_h));
 | |
|     list_profile->SetMinSize(wxSize(23*em, list_h));
 | |
| 
 | |
| 
 | |
| 
 | |
|     grid = new wxFlexGridSizer(4, em/2, em);
 | |
|     grid->AddGrowableCol(3, 1);
 | |
|     grid->AddGrowableRow(1, 1);
 | |
| 
 | |
| 	grid->Add(new wxStaticText(this, wxID_ANY, _L("Printer:")));
 | |
|     grid->Add(new wxStaticText(this, wxID_ANY, list1name));
 | |
|     grid->Add(new wxStaticText(this, wxID_ANY, _L("Vendor:")));
 | |
|     grid->Add(new wxStaticText(this, wxID_ANY, _L("Profile:")));
 | |
| 
 | |
| 	grid->Add(list_printer, 0, wxEXPAND);
 | |
|     grid->Add(list_type, 0, wxEXPAND);
 | |
|     grid->Add(list_vendor, 0, wxEXPAND);
 | |
|     grid->Add(list_profile, 1, wxEXPAND);
 | |
| 
 | |
|     auto *btn_sizer = new wxBoxSizer(wxHORIZONTAL);
 | |
|     auto *sel_all = new wxButton(this, wxID_ANY, _L("All"));
 | |
|     auto *sel_none = new wxButton(this, wxID_ANY, _L("None"));
 | |
|     btn_sizer->Add(sel_all, 0, wxRIGHT, em / 2);
 | |
|     btn_sizer->Add(sel_none);
 | |
| 
 | |
|     wxGetApp().UpdateDarkUI(list_printer);
 | |
|     wxGetApp().UpdateDarkUI(list_type);
 | |
|     wxGetApp().UpdateDarkUI(list_vendor);
 | |
|     wxGetApp().UpdateDarkUI(sel_all);
 | |
|     wxGetApp().UpdateDarkUI(sel_none);
 | |
| 
 | |
|     grid->Add(new wxBoxSizer(wxHORIZONTAL));
 | |
|     grid->Add(new wxBoxSizer(wxHORIZONTAL));
 | |
|     grid->Add(new wxBoxSizer(wxHORIZONTAL));
 | |
|     grid->Add(btn_sizer, 0, wxALIGN_RIGHT);
 | |
| 
 | |
|     append(grid, 1, wxEXPAND);
 | |
| 
 | |
|     append_spacer(VERTICAL_SPACING);
 | |
| 
 | |
|     html_window = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition,
 | |
|         wxSize(60 * em, 20 * em), wxHW_SCROLLBAR_AUTO);
 | |
|     append(html_window, 0, wxEXPAND);
 | |
| 
 | |
| 	list_printer->Bind(wxEVT_LISTBOX, [this](wxCommandEvent& evt) {
 | |
| 		update_lists(list_type->GetSelection(), list_vendor->GetSelection(), evt.GetInt());
 | |
| 		});
 | |
|     list_type->Bind(wxEVT_LISTBOX, [this](wxCommandEvent &) {
 | |
|         update_lists(list_type->GetSelection(), list_vendor->GetSelection());
 | |
|     });
 | |
|     list_vendor->Bind(wxEVT_LISTBOX, [this](wxCommandEvent &) {
 | |
|         update_lists(list_type->GetSelection(), list_vendor->GetSelection());
 | |
|     });
 | |
| 
 | |
|     list_profile->Bind(wxEVT_CHECKLISTBOX, [this](wxCommandEvent &evt) { select_material(evt.GetInt()); });
 | |
|     list_profile->Bind(wxEVT_LISTBOX, [this](wxCommandEvent& evt) { on_material_highlighted(evt.GetInt()); });
 | |
| 
 | |
|     sel_all->Bind(wxEVT_BUTTON, [this](wxCommandEvent &) { select_all(true); });
 | |
|     sel_none->Bind(wxEVT_BUTTON, [this](wxCommandEvent &) { select_all(false); });
 | |
|     /*
 | |
|     Bind(wxEVT_PAINT, [this](wxPaintEvent& evt) {on_paint();});
 | |
| 
 | |
|     list_profile->Bind(wxEVT_MOTION, [this](wxMouseEvent& evt) { on_mouse_move_on_profiles(evt); });
 | |
|     list_profile->Bind(wxEVT_ENTER_WINDOW, [this](wxMouseEvent& evt) { on_mouse_enter_profiles(evt); });
 | |
|     list_profile->Bind(wxEVT_LEAVE_WINDOW, [this](wxMouseEvent& evt) { on_mouse_leave_profiles(evt); });
 | |
|     */
 | |
|     reload_presets();
 | |
|     set_compatible_printers_html_window(std::vector<std::string>(), false);
 | |
| }
 | |
| void PageMaterials::on_paint()
 | |
| {
 | |
| }
 | |
| void PageMaterials::on_mouse_move_on_profiles(wxMouseEvent& evt)
 | |
| {
 | |
|     const wxClientDC dc(list_profile);
 | |
|     const wxPoint pos = evt.GetLogicalPosition(dc);
 | |
|     int item = list_profile->HitTest(pos);
 | |
|     on_material_hovered(item);
 | |
| }
 | |
| void PageMaterials::on_mouse_enter_profiles(wxMouseEvent& evt)
 | |
| {}
 | |
| void PageMaterials::on_mouse_leave_profiles(wxMouseEvent& evt)
 | |
| {
 | |
|     on_material_hovered(-1);
 | |
| }
 | |
| void PageMaterials::reload_presets()
 | |
| {
 | |
|     clear();
 | |
| 
 | |
| 	list_printer->append(_L("(All)"), &EMPTY);
 | |
|     //list_printer->SetLabelMarkup("<b>bald</b>");
 | |
| 	for (const Preset* printer : materials->printers) {
 | |
| 		list_printer->append(printer->name, &printer->name);
 | |
| 	}
 | |
|     sort_list_data(list_printer, true, false);
 | |
|     if (list_printer->GetCount() > 0) {
 | |
|         list_printer->SetSelection(0);
 | |
|         sel_printers_prev.Clear();
 | |
|         sel_type_prev = wxNOT_FOUND;
 | |
|         sel_vendor_prev = wxNOT_FOUND;
 | |
|         update_lists(0, 0, 0);
 | |
|     }
 | |
| 
 | |
|     presets_loaded = true;
 | |
| }
 | |
| 
 | |
| void PageMaterials::set_compatible_printers_html_window(const std::vector<std::string>& printer_names, bool all_printers)
 | |
| {
 | |
|     const auto bgr_clr = 
 | |
| #if defined(__APPLE__)
 | |
|         html_window->GetParent()->GetBackgroundColour();
 | |
| #else 
 | |
| #if defined(_WIN32)
 | |
|         wxGetApp().get_window_default_clr();
 | |
| #else
 | |
|         wxSystemSettings::GetColour(wxSYS_COLOUR_MENU);
 | |
| #endif
 | |
| #endif
 | |
|     const auto bgr_clr_str = wxString::Format(wxT("#%02X%02X%02X"), bgr_clr.Red(), bgr_clr.Green(), bgr_clr.Blue());
 | |
|     const auto text_clr = wxGetApp().get_label_clr_default();//wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT);
 | |
|     const auto text_clr_str = wxString::Format(wxT("#%02X%02X%02X"), text_clr.Red(), text_clr.Green(), text_clr.Blue());
 | |
|     wxString first_line = format_wxstr(_L("%1% marked with <b>*</b> are <b>not</b> compatible with some installed printers."), materials->technology == T_FFF ? _L("Filaments") : _L("SLA materials"));
 | |
|     wxString text;
 | |
|     if (all_printers) {
 | |
|         wxString second_line = format_wxstr(_L("All installed printers are compatible with the selected %1%."), materials->technology == T_FFF ? _L("filament") : _L("SLA material"));
 | |
|         text = wxString::Format(
 | |
|             "<html>"
 | |
|             "<style>"
 | |
|             "table{border-spacing: 1px;}"
 | |
|             "</style>"
 | |
|             "<body bgcolor= %s>"
 | |
|             "<font color=%s>"
 | |
|             "<font size=\"3\">"
 | |
|             "%s<br /><br />%s"
 | |
|             "</font>"
 | |
|             "</font>"
 | |
|             "</body>"
 | |
|             "</html>"
 | |
|             , bgr_clr_str
 | |
|             , text_clr_str
 | |
|             , first_line
 | |
|             , second_line
 | |
|             );
 | |
|     } else {
 | |
|         wxString second_line;
 | |
|         if (!printer_names.empty())
 | |
|             second_line = (materials->technology == T_FFF ?
 | |
|                           _L("Only the following installed printers are compatible with the selected filaments") :
 | |
|                           _L("Only the following installed printers are compatible with the selected SLA materials")) + ":";
 | |
|         text = wxString::Format(
 | |
|             "<html>"
 | |
|             "<style>"
 | |
|             "table{border-spacing: 1px;}"
 | |
|             "</style>"
 | |
|             "<body bgcolor= %s>"
 | |
|             "<font color=%s>"
 | |
|             "<font size=\"3\">"
 | |
|             "%s<br /><br />%s"
 | |
|             "<table>"
 | |
|             "<tr>"
 | |
|             , bgr_clr_str
 | |
|             , text_clr_str
 | |
|             , first_line
 | |
|             , second_line);
 | |
|         for (size_t i = 0; i < printer_names.size(); ++i)
 | |
|         {
 | |
|             text += wxString::Format("<td>%s</td>", boost::nowide::widen(printer_names[i]));
 | |
|             if (i % 3 == 2) {
 | |
|                 text += wxString::Format(
 | |
|                     "</tr>"
 | |
|                     "<tr>");
 | |
|             }
 | |
|         }
 | |
|         text += wxString::Format(
 | |
|             "</tr>"
 | |
|             "</table>"
 | |
|             "</font>"
 | |
|             "</font>"
 | |
|             "</body>"
 | |
|             "</html>"
 | |
|         );
 | |
|     }
 | |
|    
 | |
|     wxFont font = get_default_font_for_dpi(this, get_dpi_for_window(this));
 | |
|     const int fs = font.GetPointSize();
 | |
|     int size[] = { fs,fs,fs,fs,fs,fs,fs };
 | |
|     html_window->SetFonts(font.GetFaceName(), font.GetFaceName(), size);
 | |
|     html_window->SetPage(text);
 | |
| }
 | |
| 
 | |
| void PageMaterials::clear_compatible_printers_label()
 | |
| {
 | |
|     set_compatible_printers_html_window(std::vector<std::string>(), false);
 | |
| }
 | |
| 
 | |
| void PageMaterials::on_material_hovered(int sel_material)
 | |
| {
 | |
| 
 | |
| }
 | |
| 
 | |
| void PageMaterials::on_material_highlighted(int sel_material)
 | |
| {
 | |
|     if (sel_material == last_hovered_item)
 | |
|         return;
 | |
|     if (sel_material == -1) {
 | |
|         clear_compatible_printers_label();
 | |
|         return;
 | |
|     }
 | |
|     last_hovered_item = sel_material;
 | |
|     std::vector<std::string> tabs;
 | |
|     tabs.push_back(std::string());
 | |
|     tabs.push_back(std::string());
 | |
|     tabs.push_back(std::string());
 | |
|     //selected material string
 | |
|     std::string material_name = list_profile->get_data(sel_material);
 | |
|     // get material preset
 | |
|     const std::vector<const Preset*> matching_materials = materials->get_presets_by_alias(material_name);
 | |
|     if (matching_materials.empty())
 | |
|     {
 | |
|         clear_compatible_printers_label();
 | |
|         return;
 | |
|     }
 | |
|     //find matching printers
 | |
|     std::vector<std::string> names;
 | |
|     for (const Preset* printer : materials->printers) {
 | |
|         for (const Preset* material : matching_materials) {
 | |
|             if (is_compatible_with_printer(PresetWithVendorProfile(*material, material->vendor), PresetWithVendorProfile(*printer, printer->vendor))) {
 | |
|                 names.push_back(printer->name);
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     set_compatible_printers_html_window(names, names.size() == materials->printers.size());
 | |
| }
 | |
| 
 | |
| void PageMaterials::update_lists(int sel_type, int sel_vendor, int last_selected_printer/* = -1*/)
 | |
| {
 | |
| 	wxWindowUpdateLocker freeze_guard(this);
 | |
| 	(void)freeze_guard;
 | |
| 
 | |
| 	wxArrayInt sel_printers;
 | |
| 	int sel_printers_count = list_printer->GetSelections(sel_printers);
 | |
| 
 | |
|     // Does our wxWidgets version support operator== for wxArrayInt ?
 | |
| #if wxCHECK_VERSION(3, 1, 1)
 | |
|     if (sel_printers != sel_printers_prev) {
 | |
| #else
 | |
|     auto are_equal = [](const wxArrayInt& arr_first, const wxArrayInt& arr_second) {
 | |
|         if (arr_first.GetCount() != arr_second.GetCount())
 | |
|             return false;
 | |
|         for (size_t i = 0; i < arr_first.GetCount(); i++)
 | |
|             if (arr_first[i] != arr_second[i])
 | |
|                 return false;
 | |
|         return true;
 | |
|     };
 | |
|     if (!are_equal(sel_printers, sel_printers_prev)) {
 | |
| #endif
 | |
| 
 | |
|         // Refresh type list
 | |
| 		list_type->Clear();
 | |
| 		list_type->append(_L("(All)"), &EMPTY);
 | |
| 		if (sel_printers_count > 0) {
 | |
|             // If all is selected with other printers
 | |
|             // unselect "all" or all printers depending on last value
 | |
|             if (sel_printers[0] == 0 && sel_printers_count > 1) {
 | |
|                 if (last_selected_printer == 0) {
 | |
|                     list_printer->SetSelection(wxNOT_FOUND);
 | |
|                     list_printer->SetSelection(0);
 | |
|                 } else {
 | |
|                     list_printer->SetSelection(0, false);
 | |
|                     sel_printers_count = list_printer->GetSelections(sel_printers);
 | |
|                 }
 | |
|             }
 | |
| 			if (sel_printers[0] != 0) {
 | |
|                 for (int i = 0; i < sel_printers_count; i++) {
 | |
| 					const std::string& printer_name = list_printer->get_data(sel_printers[i]);
 | |
| 					const Preset* printer = nullptr;
 | |
| 					for (const Preset* it : materials->printers) {
 | |
| 						if (it->name == printer_name) {
 | |
| 							printer = it;
 | |
| 							break;
 | |
| 						}
 | |
| 					}
 | |
| 					materials->filter_presets(printer, EMPTY, EMPTY, [this](const Preset* p) {
 | |
| 						const std::string& type = this->materials->get_type(p);
 | |
| 						if (list_type->find(type) == wxNOT_FOUND) {
 | |
| 							list_type->append(type, &type);
 | |
| 						}
 | |
| 						});
 | |
| 				}
 | |
| 			} else {
 | |
|                 //clear selection except "ALL"
 | |
|                 list_printer->SetSelection(wxNOT_FOUND);
 | |
|                 list_printer->SetSelection(0);
 | |
|                 sel_printers_count = list_printer->GetSelections(sel_printers);
 | |
| 
 | |
| 				materials->filter_presets(nullptr, EMPTY, EMPTY, [this](const Preset* p) {
 | |
| 					const std::string& type = this->materials->get_type(p);
 | |
| 					if (list_type->find(type) == wxNOT_FOUND) {
 | |
| 						list_type->append(type, &type);
 | |
| 					}
 | |
| 					});
 | |
| 			}
 | |
|             sort_list_data(list_type, true, true);
 | |
| 		}
 | |
| 
 | |
| 		sel_printers_prev = sel_printers;
 | |
| 		sel_type = 0;
 | |
| 		sel_type_prev = wxNOT_FOUND;
 | |
| 		list_type->SetSelection(sel_type);
 | |
| 		list_profile->Clear();
 | |
| 	}
 | |
| 	
 | |
| 	if (sel_type != sel_type_prev) {
 | |
| 		// Refresh vendor list
 | |
| 
 | |
| 		// XXX: The vendor list is created with quadratic complexity here,
 | |
| 		// but the number of vendors is going to be very small this shouldn't be a problem.
 | |
| 
 | |
| 		list_vendor->Clear();
 | |
| 		list_vendor->append(_L("(All)"), &EMPTY);
 | |
| 		if (sel_printers_count != 0 && sel_type != wxNOT_FOUND) {
 | |
| 			const std::string& type = list_type->get_data(sel_type);
 | |
| 			// find printer preset
 | |
|             for (int i = 0; i < sel_printers_count; i++) {
 | |
| 				const std::string& printer_name = list_printer->get_data(sel_printers[i]);
 | |
| 				const Preset* printer = nullptr;
 | |
| 				for (const Preset* it : materials->printers) {
 | |
| 					if (it->name == printer_name) {
 | |
| 						printer = it;
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 				materials->filter_presets(printer, type, EMPTY, [this](const Preset* p) {
 | |
| 					const std::string& vendor = this->materials->get_vendor(p);
 | |
| 					if (list_vendor->find(vendor) == wxNOT_FOUND) {
 | |
| 						list_vendor->append(vendor, &vendor);
 | |
| 					}
 | |
| 					});
 | |
| 			}
 | |
|             sort_list_data(list_vendor, true, false);
 | |
| 		}
 | |
| 
 | |
| 		sel_type_prev = sel_type;
 | |
| 		sel_vendor = 0;
 | |
| 		sel_vendor_prev = wxNOT_FOUND;
 | |
| 		list_vendor->SetSelection(sel_vendor);
 | |
| 		list_profile->Clear();
 | |
| 	}
 | |
|          
 | |
| 	if (sel_vendor != sel_vendor_prev) {
 | |
| 		// Refresh material list
 | |
| 
 | |
| 		list_profile->Clear();
 | |
|         clear_compatible_printers_label();
 | |
| 		if (sel_printers_count != 0 && sel_type != wxNOT_FOUND && sel_vendor != wxNOT_FOUND) {
 | |
| 			const std::string& type = list_type->get_data(sel_type);
 | |
| 			const std::string& vendor = list_vendor->get_data(sel_vendor);
 | |
| 			// finst printer preset
 | |
|             std::vector<ProfilePrintData> to_list;
 | |
|             for (int i = 0; i < sel_printers_count; i++) {
 | |
| 				const std::string& printer_name = list_printer->get_data(sel_printers[i]);
 | |
| 				const Preset* printer = nullptr;
 | |
| 				for (const Preset* it : materials->printers) {
 | |
| 					if (it->name == printer_name) {
 | |
| 						printer = it;
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				materials->filter_presets(printer, type, vendor, [this, &to_list](const Preset* p) {
 | |
| 					const std::string& section = materials->appconfig_section();
 | |
|                     bool checked = wizard_p()->appconfig_new.has(section, p->name);
 | |
|                     bool was_checked = false;
 | |
| 
 | |
|                     int cur_i = list_profile->find(p->alias);
 | |
|                     if (cur_i == wxNOT_FOUND) {
 | |
|                         cur_i = list_profile->append(p->alias + (materials->get_omnipresent(p) ? "" : " *"), &p->alias);
 | |
|                         to_list.emplace_back(p->alias, materials->get_omnipresent(p), checked);
 | |
|                     }
 | |
|                     else {
 | |
|                         was_checked = list_profile->IsChecked(cur_i);
 | |
|                         to_list[cur_i].checked = checked || was_checked;
 | |
|                     }
 | |
|                     list_profile->Check(cur_i, checked || was_checked);
 | |
| 
 | |
| 					/* Update preset selection in config.
 | |
| 					 * If one preset from aliases bundle is selected,
 | |
| 					 * than mark all presets with this aliases as selected
 | |
| 					 * */
 | |
| 					if (checked && !was_checked)
 | |
| 						wizard_p()->update_presets_in_config(section, p->alias, true);
 | |
| 					else if (!checked && was_checked)
 | |
| 						wizard_p()->appconfig_new.set(section, p->name, "true");
 | |
| 					});
 | |
| 			}
 | |
|             sort_list_data(list_profile, to_list);
 | |
| 		}
 | |
| 
 | |
| 		sel_vendor_prev = sel_vendor;
 | |
| 	}
 | |
|     wxGetApp().UpdateDarkUI(list_profile);
 | |
| }
 | |
| 
 | |
| void PageMaterials::sort_list_data(StringList* list, bool add_All_item, bool material_type_ordering)
 | |
| {
 | |
| // get data from list
 | |
| // sort data
 | |
| // first should be <all>
 | |
| // then prusa profiles
 | |
| // then the rest
 | |
| // in alphabetical order
 | |
|     
 | |
|     std::vector<std::reference_wrapper<const std::string>> prusa_profiles;
 | |
|     std::vector<std::reference_wrapper<const std::string>> other_profiles;
 | |
|     for (int i = 0 ; i < list->size(); ++i) {
 | |
|         const std::string& data = list->get_data(i);
 | |
|         if (data == EMPTY) // do not sort <all> item
 | |
|             continue;
 | |
|         if (!material_type_ordering && data.find("Bambu") != std::string::npos)
 | |
|             prusa_profiles.push_back(data);
 | |
|         else 
 | |
|             other_profiles.push_back(data);
 | |
|     }
 | |
|     if(material_type_ordering) {
 | |
|         
 | |
|         const ConfigOptionDef* def = print_config_def.get("filament_type");
 | |
|         std::vector<std::string>enum_values = def->enum_values;
 | |
|         size_t end_of_sorted = 0;
 | |
|         for (size_t vals = 0; vals < enum_values.size(); vals++) {
 | |
|             for (size_t profs = end_of_sorted; profs < other_profiles.size(); profs++)
 | |
|             {
 | |
|                 // find instead compare because PET vs PETG
 | |
|                 if (other_profiles[profs].get().find(enum_values[vals]) != std::string::npos) {
 | |
|                     //swap
 | |
|                     if(profs != end_of_sorted) {
 | |
|                         std::reference_wrapper<const std::string> aux = other_profiles[end_of_sorted];
 | |
|                         other_profiles[end_of_sorted] = other_profiles[profs];
 | |
|                         other_profiles[profs] = aux;
 | |
|                     }
 | |
|                     end_of_sorted++;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         std::sort(prusa_profiles.begin(), prusa_profiles.end(), [](std::reference_wrapper<const std::string> a, std::reference_wrapper<const std::string> b) {
 | |
|             return a.get() < b.get();
 | |
|             });
 | |
|         std::sort(other_profiles.begin(), other_profiles.end(), [](std::reference_wrapper<const std::string> a, std::reference_wrapper<const std::string> b) {
 | |
|             return a.get() < b.get();
 | |
|             });
 | |
|     }
 | |
|     
 | |
|     list->Clear();
 | |
|     if (add_All_item)
 | |
|         list->append(_L("(All)"), &EMPTY);
 | |
|     for (const auto& item : prusa_profiles)
 | |
|         list->append(item, &const_cast<std::string&>(item.get()));
 | |
|     for (const auto& item : other_profiles)
 | |
|         list->append(item, &const_cast<std::string&>(item.get()));
 | |
| }     
 | |
| 
 | |
| void PageMaterials::sort_list_data(PresetList* list, const std::vector<ProfilePrintData>& data)
 | |
| {
 | |
|     // sort data
 | |
|     // then prusa profiles
 | |
|     // then the rest
 | |
|     // in alphabetical order
 | |
|     std::vector<ProfilePrintData> prusa_profiles;
 | |
|     std::vector<ProfilePrintData> other_profiles;
 | |
|     //for (int i = 0; i < data.size(); ++i) {
 | |
|     for (const auto& item : data) {
 | |
|         const std::string& name = item.name;
 | |
|         if (name.find("Bambu") != std::string::npos)
 | |
|             prusa_profiles.emplace_back(item);
 | |
|         else
 | |
|             other_profiles.emplace_back(item);
 | |
|     }
 | |
|     std::sort(prusa_profiles.begin(), prusa_profiles.end(), [](ProfilePrintData a, ProfilePrintData b) {
 | |
|         return a.name.get() < b.name.get();
 | |
|         });
 | |
|     std::sort(other_profiles.begin(), other_profiles.end(), [](ProfilePrintData a, ProfilePrintData b) {
 | |
|         return a.name.get() < b.name.get();
 | |
|         });
 | |
|     list->Clear();
 | |
|     for (size_t i = 0; i < prusa_profiles.size(); ++i) {
 | |
|         list->append(std::string(prusa_profiles[i].name) + (prusa_profiles[i].omnipresent ? "" : " *"), &const_cast<std::string&>(prusa_profiles[i].name.get()));
 | |
|         list->Check(i, prusa_profiles[i].checked);
 | |
|     }
 | |
|     for (size_t i = 0; i < other_profiles.size(); ++i) {
 | |
|         list->append(std::string(other_profiles[i].name) + (other_profiles[i].omnipresent ? "" : " *"), &const_cast<std::string&>(other_profiles[i].name.get()));
 | |
|         list->Check(i + prusa_profiles.size(), other_profiles[i].checked);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void PageMaterials::select_material(int i)
 | |
| {
 | |
|     const bool checked = list_profile->IsChecked(i);
 | |
| 
 | |
|     const std::string& alias_key = list_profile->get_data(i);
 | |
|     wizard_p()->update_presets_in_config(materials->appconfig_section(), alias_key, checked);
 | |
| }
 | |
| 
 | |
| void PageMaterials::select_all(bool select)
 | |
| {
 | |
|     wxWindowUpdateLocker freeze_guard(this);
 | |
|     (void)freeze_guard;
 | |
| 
 | |
|     for (unsigned i = 0; i < list_profile->GetCount(); i++) {
 | |
|         const bool current = list_profile->IsChecked(i);
 | |
|         if (current != select) {
 | |
|             list_profile->Check(i, select);
 | |
|             select_material(i);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void PageMaterials::clear()
 | |
| {
 | |
| 	list_printer->Clear();
 | |
|     list_type->Clear();
 | |
|     list_vendor->Clear();
 | |
|     list_profile->Clear();
 | |
| 	sel_printers_prev.Clear();
 | |
|     sel_type_prev = wxNOT_FOUND;
 | |
|     sel_vendor_prev = wxNOT_FOUND;
 | |
|     presets_loaded = false;
 | |
| }
 | |
| 
 | |
| void PageMaterials::on_activate()
 | |
| {
 | |
|     if (! presets_loaded) {
 | |
|         wizard_p()->update_materials(materials->technology);
 | |
|         reload_presets();
 | |
|     }
 | |
|     first_paint = true;
 | |
| }
 | |
| 
 | |
| 
 | |
| const char *PageCustom::default_profile_name = "My Settings";
 | |
| 
 | |
| PageCustom::PageCustom(ConfigWizard *parent)
 | |
|     : ConfigWizardPage(parent, _L("Custom Printer Setup"), _L("Custom Printer"))
 | |
| {
 | |
|     cb_custom = new wxCheckBox(this, wxID_ANY, _L("Define a custom printer profile"));
 | |
|     tc_profile_name = new wxTextCtrl(this, wxID_ANY, default_profile_name);
 | |
|     auto *label = new wxStaticText(this, wxID_ANY, _L("Custom profile name:"));
 | |
| 
 | |
|     wxGetApp().UpdateDarkUI(tc_profile_name);
 | |
| 
 | |
|     tc_profile_name->Enable(false);
 | |
|     tc_profile_name->Bind(wxEVT_KILL_FOCUS, [this](wxFocusEvent &evt) {
 | |
|         if (tc_profile_name->GetValue().IsEmpty()) {
 | |
|             if (profile_name_prev.IsEmpty()) { tc_profile_name->SetValue(default_profile_name); }
 | |
|             else { tc_profile_name->SetValue(profile_name_prev); }
 | |
|         } else {
 | |
|             profile_name_prev = tc_profile_name->GetValue();
 | |
|         }
 | |
|         evt.Skip();
 | |
|     });
 | |
| 
 | |
|     cb_custom->Bind(wxEVT_CHECKBOX, [this](wxCommandEvent &event) {
 | |
|         tc_profile_name->Enable(custom_wanted());
 | |
|         wizard_p()->on_custom_setup(custom_wanted());
 | |
| 		
 | |
|     });
 | |
| 
 | |
|     append(cb_custom);
 | |
|     append(label);
 | |
|     append(tc_profile_name);
 | |
| }
 | |
| 
 | |
| #ifdef _WIN32
 | |
| PageFilesAssociation::PageFilesAssociation(ConfigWizard* parent)
 | |
|     : ConfigWizardPage(parent, _L("Files association"), _L("Files association"))
 | |
| {
 | |
|     cb_3mf = new wxCheckBox(this, wxID_ANY, _L("Associate with 3mf files"));
 | |
|     cb_stl = new wxCheckBox(this, wxID_ANY, _L("Associate with stl files"));
 | |
|     //cb_gcode = new wxCheckBox(this, wxID_ANY, _L("Associate with gcode files"));
 | |
| 
 | |
|     append(cb_3mf);
 | |
|     append(cb_stl);
 | |
|     //append(cb_gcode);
 | |
| }
 | |
| #endif // _WIN32
 | |
| 
 | |
| PageVendors::PageVendors(ConfigWizard *parent)
 | |
|     : ConfigWizardPage(parent, _L("Other Vendors"), _L("Other Vendors"))
 | |
| {
 | |
|     const AppConfig &appconfig = this->wizard_p()->appconfig_new;
 | |
| 
 | |
|     append_text(_L("Pick another vendor:"));
 | |
| 
 | |
|     auto boldfont = wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
 | |
|     boldfont.SetWeight(wxFONTWEIGHT_BOLD);
 | |
| 
 | |
|     for (const auto &pair : wizard_p()->bundles) {
 | |
|         const VendorProfile *vendor = pair.second.vendor_profile;
 | |
|         //BBS: add BBL as default
 | |
|         if (vendor->id == PresetBundle::BBL_BUNDLE) { continue; }
 | |
| 
 | |
|         auto *cbox = new wxCheckBox(this, wxID_ANY, vendor->name);
 | |
|         cbox->Bind(wxEVT_CHECKBOX, [=](wxCommandEvent &event) {
 | |
|             wizard_p()->on_3rdparty_install(vendor, cbox->IsChecked());
 | |
|         });
 | |
| 
 | |
|         const auto &vendors = appconfig.vendors();
 | |
|         const bool enabled = vendors.find(pair.first) != vendors.end();
 | |
|         if (enabled) {
 | |
|             cbox->SetValue(true);
 | |
| 
 | |
|             auto pages = wizard_p()->pages_3rdparty.find(vendor->id);
 | |
|             wxCHECK_RET(pages != wizard_p()->pages_3rdparty.end(), _L("Internal error: third party vendor printers not created"));
 | |
| 
 | |
|             for (PagePrinters* page : { pages->second.first, pages->second.second })
 | |
|                 if (page) page->install = true;
 | |
|         }
 | |
| 
 | |
|         append(cbox);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*PageBedShape::PageBedShape(ConfigWizard *parent)
 | |
|     : ConfigWizardPage(parent, _L("Bed Shape and Size"), _L("Bed Shape"), 1)
 | |
|     , shape_panel(new BedShapePanel(this))
 | |
| {
 | |
|     append_text(_L("Set the shape of your printer's bed."));
 | |
|     shape_panel->build_panel(*wizard_p()->custom_config->option<ConfigOptionPoints>("printable_area"), {}, {});
 | |
|     append(shape_panel);
 | |
| }
 | |
| 
 | |
| void PageBedShape::apply_custom_config(DynamicPrintConfig &config)
 | |
| {
 | |
|     const std::vector<Vec2d>& points = shape_panel->get_shape();
 | |
|     config.set_key_value("printable_area", new ConfigOptionPoints(points));
 | |
| }*/
 | |
| 
 | |
| static void focus_event(wxFocusEvent& e, wxTextCtrl* ctrl, double def_value) 
 | |
| {
 | |
|     e.Skip();
 | |
|     wxString str = ctrl->GetValue();
 | |
| 
 | |
|     const char dec_sep = is_decimal_separator_point() ? '.' : ',';
 | |
|     const char dec_sep_alt = dec_sep == '.' ? ',' : '.';
 | |
|     // Replace the first incorrect separator in decimal number.
 | |
|     bool was_replaced = str.Replace(dec_sep_alt, dec_sep, false) != 0;
 | |
| 
 | |
|     double val = 0.0;
 | |
|     if (!str.ToDouble(&val)) {
 | |
|         if (val == 0.0)
 | |
|             val = def_value;
 | |
|         ctrl->SetValue(double_to_string(val));
 | |
|         show_error(nullptr, _L("Invalid numeric input."));
 | |
|         ctrl->SetFocus();
 | |
|     }
 | |
|     else if (was_replaced)
 | |
|         ctrl->SetValue(double_to_string(val));
 | |
| }
 | |
| 
 | |
| class DiamTextCtrl : public wxTextCtrl
 | |
| {
 | |
| public:
 | |
|     DiamTextCtrl(wxWindow* parent)
 | |
|     {
 | |
| #ifdef _WIN32
 | |
|         long style = wxBORDER_SIMPLE;
 | |
| #else
 | |
|         long style = 0;
 | |
| #endif
 | |
|         Create(parent, wxID_ANY, wxEmptyString, wxDefaultPosition, wxSize(Field::def_width_thinner() * wxGetApp().em_unit(), wxDefaultCoord), style);
 | |
|         wxGetApp().UpdateDarkUI(this);
 | |
|     }
 | |
|     ~DiamTextCtrl() {}
 | |
| };
 | |
| 
 | |
| PageDiameters::PageDiameters(ConfigWizard *parent)
 | |
|     : ConfigWizardPage(parent, _L("Filament and Nozzle Diameters"), _L("Print Diameters"), 1)
 | |
|     , diam_nozzle(new DiamTextCtrl(this))
 | |
|     , diam_filam (new DiamTextCtrl(this))
 | |
| {
 | |
|     auto *default_nozzle = print_config_def.get("nozzle_diameter")->get_default_value<ConfigOptionFloats>();
 | |
|     wxString value = double_to_string(default_nozzle != nullptr && default_nozzle->size() > 0 ? default_nozzle->get_at(0) : 0.5);
 | |
|     diam_nozzle->SetValue(value);
 | |
| 
 | |
|     auto *default_filam = print_config_def.get("filament_diameter")->get_default_value<ConfigOptionFloats>();
 | |
|     value = double_to_string(default_filam != nullptr && default_filam->size() > 0 ? default_filam->get_at(0) : 3.0);
 | |
|     diam_filam->SetValue(value);
 | |
| 
 | |
|     diam_nozzle->Bind(wxEVT_KILL_FOCUS, [this](wxFocusEvent& e) { focus_event(e, diam_nozzle, 0.5); }, diam_nozzle->GetId());
 | |
|     diam_filam ->Bind(wxEVT_KILL_FOCUS, [this](wxFocusEvent& e) { focus_event(e, diam_filam , 3.0); }, diam_filam->GetId());
 | |
| 
 | |
|     append_text(_L("Enter the diameter of your printer's hot end nozzle."));
 | |
| 
 | |
|     auto *sizer_nozzle = new wxFlexGridSizer(3, 5, 5);
 | |
|     auto *text_nozzle = new wxStaticText(this, wxID_ANY, _L("Nozzle Diameter:"));
 | |
|     auto *unit_nozzle = new wxStaticText(this, wxID_ANY, _L("mm"));
 | |
|     sizer_nozzle->AddGrowableCol(0, 1);
 | |
|     sizer_nozzle->Add(text_nozzle, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     sizer_nozzle->Add(diam_nozzle);
 | |
|     sizer_nozzle->Add(unit_nozzle, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     append(sizer_nozzle);
 | |
| 
 | |
|     append_spacer(VERTICAL_SPACING);
 | |
| 
 | |
|     append_text(_L("Enter the diameter of your filament."));
 | |
|     append_text(_L("Good precision is required, so use a caliper and do multiple measurements along the filament, then compute the average."));
 | |
| 
 | |
|     auto *sizer_filam = new wxFlexGridSizer(3, 5, 5);
 | |
|     auto *text_filam = new wxStaticText(this, wxID_ANY, _L("Filament Diameter:"));
 | |
|     auto *unit_filam = new wxStaticText(this, wxID_ANY, _L("mm"));
 | |
|     sizer_filam->AddGrowableCol(0, 1);
 | |
|     sizer_filam->Add(text_filam, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     sizer_filam->Add(diam_filam);
 | |
|     sizer_filam->Add(unit_filam, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     append(sizer_filam);
 | |
| }
 | |
| 
 | |
| void PageDiameters::apply_custom_config(DynamicPrintConfig &config)
 | |
| {
 | |
|     double val = 0.0;
 | |
|     diam_nozzle->GetValue().ToDouble(&val);
 | |
|     auto *opt_nozzle = new ConfigOptionFloats(1, val);
 | |
|     config.set_key_value("nozzle_diameter", opt_nozzle);
 | |
| 
 | |
|     val = 0.0;
 | |
|     diam_filam->GetValue().ToDouble(&val);
 | |
|     auto * opt_filam = new ConfigOptionFloats(1, val);
 | |
|     config.set_key_value("filament_diameter", opt_filam);
 | |
| 
 | |
|     auto set_extrusion_width = [&config, opt_nozzle](const char *key, double dmr) {
 | |
|         char buf[64]; // locales don't matter here (sprintf/atof)
 | |
|         sprintf(buf, "%.2lf", dmr * opt_nozzle->values.front() / 0.4);
 | |
|         config.set_key_value(key, new ConfigOptionFloat(atof(buf)));
 | |
|     };
 | |
| 
 | |
|     set_extrusion_width("support_line_width", 0.35);
 | |
|     set_extrusion_width("top_surface_line_width",		  0.40);
 | |
|     set_extrusion_width("initial_layer_line_width",		  0.42);
 | |
| 
 | |
|     set_extrusion_width("line_width",					  0.45);
 | |
|     set_extrusion_width("inner_wall_line_width",		  0.45);
 | |
|     set_extrusion_width("outer_wall_line_width", 0.45);
 | |
|     set_extrusion_width("sparse_infill_line_width",			  0.45);
 | |
|     set_extrusion_width("internal_solid_infill_line_width",       0.45);
 | |
| }
 | |
| 
 | |
| class SpinCtrlDouble: public wxSpinCtrlDouble
 | |
| {
 | |
| public:
 | |
|     SpinCtrlDouble(wxWindow* parent)
 | |
|     {
 | |
| #ifdef _WIN32
 | |
|         long style = wxSP_ARROW_KEYS | wxBORDER_SIMPLE;
 | |
| #else
 | |
|         long style = wxSP_ARROW_KEYS;
 | |
| #endif
 | |
|         Create(parent, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize, style);
 | |
| #ifdef _WIN32
 | |
|         wxGetApp().UpdateDarkUI(this->GetText());
 | |
| #endif
 | |
|         this->Refresh();
 | |
|     }
 | |
|     ~SpinCtrlDouble() {}
 | |
| };
 | |
| 
 | |
| PageTemperatures::PageTemperatures(ConfigWizard *parent)
 | |
|     : ConfigWizardPage(parent, _L("Nozzle and Bed Temperatures"), _L("Temperatures"), 1)
 | |
|     , spin_extr(new SpinCtrlDouble(this))
 | |
|     , spin_bed (new SpinCtrlDouble(this))
 | |
| {
 | |
|     // BBS: FIXME
 | |
| #if 0
 | |
|     spin_extr->SetIncrement(5.0);
 | |
|     const auto &def_extr = *print_config_def.get("nozzle_temperature");
 | |
|     spin_extr->SetRange(def_extr.min, def_extr.max);
 | |
|     auto *default_extr = def_extr.get_default_value<ConfigOptionInts>();
 | |
|     spin_extr->SetValue(default_extr != nullptr && default_extr->size() > 0 ? default_extr->get_at(0) : 200);
 | |
| 
 | |
|     spin_bed->SetIncrement(5.0);
 | |
|     const auto &def_bed = *print_config_def.get("bed_temperature");
 | |
|     spin_bed->SetRange(def_bed.min, def_bed.max);
 | |
|     auto *default_bed = def_bed.get_default_value<ConfigOptionInts>();
 | |
|     spin_bed->SetValue(default_bed != nullptr && default_bed->size() > 0 ? default_bed->get_at(0) : 0);
 | |
| 
 | |
|     append_text(_L("Enter the nozzle_temperature needed for extruding your filament."));
 | |
|     append_text(_L("A rule of thumb is 160 to 230 °C for PLA, and 215 to 250 °C for ABS."));
 | |
| #endif
 | |
| 
 | |
|     auto *sizer_extr = new wxFlexGridSizer(3, 5, 5);
 | |
|     auto *text_extr = new wxStaticText(this, wxID_ANY, _L("Extrusion Temperature:"));
 | |
|     auto *unit_extr = new wxStaticText(this, wxID_ANY, _L("°C"));
 | |
|     sizer_extr->AddGrowableCol(0, 1);
 | |
|     sizer_extr->Add(text_extr, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     sizer_extr->Add(spin_extr);
 | |
|     sizer_extr->Add(unit_extr, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     append(sizer_extr);
 | |
| 
 | |
|     append_spacer(VERTICAL_SPACING);
 | |
| 
 | |
|     append_text(_L("Enter the bed temperature needed for getting your filament to stick to your heated bed."));
 | |
|     append_text(_L("A rule of thumb is 60 °C for PLA and 110 °C for ABS. Leave zero if you have no heated bed."));
 | |
| 
 | |
|     auto *sizer_bed = new wxFlexGridSizer(3, 5, 5);
 | |
|     auto *text_bed = new wxStaticText(this, wxID_ANY, _L("Bed Temperature:"));
 | |
|     auto *unit_bed = new wxStaticText(this, wxID_ANY, _L("°C"));
 | |
|     sizer_bed->AddGrowableCol(0, 1);
 | |
|     sizer_bed->Add(text_bed, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     sizer_bed->Add(spin_bed);
 | |
|     sizer_bed->Add(unit_bed, 0, wxALIGN_CENTRE_VERTICAL);
 | |
|     append(sizer_bed);
 | |
| }
 | |
| 
 | |
| void PageTemperatures::apply_custom_config(DynamicPrintConfig &config)
 | |
| {
 | |
|     // BBS
 | |
| #if 0
 | |
|     auto *opt_extr = new ConfigOptionInts(1, spin_extr->GetValue());
 | |
|     config.set_key_value("nozzle_temperature", opt_extr);
 | |
|     auto *opt_extr1st = new ConfigOptionInts(1, spin_extr->GetValue());
 | |
|     config.set_key_value("nozzle_temperature_initial_layer", opt_extr1st);
 | |
|     auto *opt_bed = new ConfigOptionInts(1, spin_bed->GetValue());
 | |
|     config.set_key_value("bed_temperature", opt_bed);
 | |
|     auto *opt_bed1st = new ConfigOptionInts(1, spin_bed->GetValue());
 | |
|     config.set_key_value("bed_temperature_initial_layer", opt_bed1st);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| // Materials
 | |
| 
 | |
| const std::string Materials::UNKNOWN = "(Unknown)";
 | |
| 
 | |
| void Materials::push(const Preset *preset)
 | |
| {
 | |
|     presets.emplace_back(preset);
 | |
|     types.insert(technology & T_FFF
 | |
|         ? Materials::get_filament_type(preset)
 | |
|         : Materials::get_material_type(preset));
 | |
| }
 | |
| 
 | |
| void  Materials::add_printer(const Preset* preset)
 | |
| {
 | |
| 	printers.insert(preset);
 | |
| }
 | |
| 
 | |
| void Materials::clear()
 | |
| {
 | |
|     presets.clear();
 | |
|     types.clear();
 | |
| 	printers.clear();
 | |
|     compatibility_counter.clear();
 | |
| }
 | |
| 
 | |
| const std::string& Materials::appconfig_section() const
 | |
| {
 | |
|     return (technology & T_FFF) ? AppConfig::SECTION_FILAMENTS : AppConfig::SECTION_MATERIALS;
 | |
| }
 | |
| 
 | |
| const std::string& Materials::get_type(const Preset *preset) const
 | |
| {
 | |
|     return (technology & T_FFF) ? get_filament_type(preset) : get_material_type(preset);
 | |
| }
 | |
| 
 | |
| const std::string& Materials::get_vendor(const Preset *preset) const
 | |
| {
 | |
|     return (technology & T_FFF) ? get_filament_vendor(preset) : get_material_vendor(preset);
 | |
| }
 | |
| 
 | |
| const std::string& Materials::get_filament_type(const Preset *preset)
 | |
| {
 | |
|     const auto *opt = preset->config.opt<ConfigOptionStrings>("filament_type");
 | |
|     if (opt != nullptr && opt->values.size() > 0) {
 | |
|         return opt->values[0];
 | |
|     } else {
 | |
|         return UNKNOWN;
 | |
|     }
 | |
| }
 | |
| 
 | |
| //BBS: change filament_vendor to strings for multiple filaments case requirements
 | |
| const std::string& Materials::get_filament_vendor(const Preset *preset)
 | |
| {
 | |
|     const auto *opt = preset->config.opt<ConfigOptionStrings>("filament_vendor");
 | |
|     if (opt != nullptr && opt->values.size() > 0) {
 | |
|         return opt->values[0];
 | |
|     } else {
 | |
|         return UNKNOWN;
 | |
|     }
 | |
| }
 | |
| 
 | |
| const std::string& Materials::get_material_type(const Preset *preset)
 | |
| {
 | |
|     const auto *opt = preset->config.opt<ConfigOptionString>("material_type");
 | |
|     if (opt != nullptr) {
 | |
|         return opt->value;
 | |
|     } else {
 | |
|         return UNKNOWN;
 | |
|     }
 | |
| }
 | |
| 
 | |
| const std::string& Materials::get_material_vendor(const Preset *preset)
 | |
| {
 | |
|     const auto *opt = preset->config.opt<ConfigOptionString>("material_vendor");
 | |
|     return opt != nullptr ? opt->value : UNKNOWN;
 | |
| }
 | |
| 
 | |
| // priv
 | |
| 
 | |
| static const std::unordered_map<std::string, std::pair<std::string, std::string>> legacy_preset_map {{
 | |
| }};
 | |
| 
 | |
| void ConfigWizard::priv::load_pages()
 | |
| {
 | |
|     wxWindowUpdateLocker freeze_guard(q);
 | |
|     (void)freeze_guard;
 | |
| 
 | |
|     // there should to be selected at least one printer
 | |
|     btn_finish->Enable();
 | |
| 
 | |
|     q->Layout();
 | |
| // This Refresh() is needed to avoid ugly artifacts after printer selection, when no one vendor was selected from the very beginnig
 | |
|     q->Refresh();
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::init_dialog_size()
 | |
| {
 | |
|     // Clamp the Wizard size based on screen dimensions
 | |
| 
 | |
|     const auto idx = wxDisplay::GetFromWindow(q);
 | |
|     wxDisplay display(idx != wxNOT_FOUND ? idx : 0u);
 | |
| 
 | |
|     const auto disp_rect = display.GetClientArea();
 | |
|     wxRect window_rect(
 | |
|         disp_rect.x + disp_rect.width / 20,
 | |
|         disp_rect.y + disp_rect.height / 20,
 | |
|         9*disp_rect.width / 10,
 | |
|         9*disp_rect.height / 10);
 | |
| 
 | |
|     q->SetSize(window_rect);
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::load_vendors()
 | |
| {
 | |
|     bundles = BundleMap::load();
 | |
| 
 | |
|     // Load up the set of vendors / models / variants the user has had enabled up till now
 | |
|     AppConfig *app_config = wxGetApp().app_config;
 | |
|     if (! app_config->legacy_datadir()) {
 | |
|         appconfig_new.set_vendors(*app_config);
 | |
|     } else {
 | |
|         // In case of legacy datadir, try to guess the preference based on the printer preset files that are present
 | |
|         //BBS: change directories by design
 | |
|         const auto printer_dir = fs::path(Slic3r::data_dir()) / PRESET_SYSTEM_DIR / PRESET_PRINTER_NAME;
 | |
|         for (auto &dir_entry : boost::filesystem::directory_iterator(printer_dir))
 | |
|             if (Slic3r::is_ini_file(dir_entry)) {
 | |
|                 auto needle = legacy_preset_map.find(dir_entry.path().filename().string());
 | |
|                 if (needle == legacy_preset_map.end()) { continue; }
 | |
| 
 | |
|                 const auto &model = needle->second.first;
 | |
|                 const auto &variant = needle->second.second;
 | |
|                 appconfig_new.set_variant("BambuResearch", model, variant, true);
 | |
|             }
 | |
|     }
 | |
| 
 | |
|     // Initialize the is_visible flag in printer Presets
 | |
|     for (auto &pair : bundles) {
 | |
|         pair.second.preset_bundle->load_installed_printers(appconfig_new);
 | |
|     }
 | |
| 
 | |
|     // Copy installed filaments and SLA material names from app_config to appconfig_new
 | |
|     // while resolving current names of profiles, which were renamed in the meantime.
 | |
|     for (PrinterTechnology technology : { ptFFF, ptSLA }) {
 | |
|     	const std::string §ion_name = (technology == ptFFF) ? AppConfig::SECTION_FILAMENTS : AppConfig::SECTION_MATERIALS;
 | |
| 		std::map<std::string, std::string> section_new;
 | |
| 		if (app_config->has_section(section_name)) {
 | |
| 			const std::map<std::string, std::string> §ion_old = app_config->get_section(section_name);
 | |
|             for (const auto& material_name_and_installed : section_old)
 | |
| 				if (material_name_and_installed.second == "true") {
 | |
| 					// Material is installed. Resolve it in bundles.
 | |
|                     size_t num_found = 0;
 | |
| 					const std::string &material_name = material_name_and_installed.first;
 | |
| 				    for (auto &bundle : bundles) {
 | |
| 				    	const PresetCollection &materials = bundle.second.preset_bundle->materials(technology);
 | |
| 				    	const Preset           *preset    = materials.find_preset(material_name);
 | |
| 				    	if (preset == nullptr) {
 | |
| 				    		// Not found. Maybe the material preset is there, bu it was was renamed?
 | |
| 							const std::string *new_name = materials.get_preset_name_renamed(material_name);
 | |
| 							if (new_name != nullptr)
 | |
| 								preset = materials.find_preset(*new_name);
 | |
| 				    	}
 | |
|                         if (preset != nullptr) {
 | |
|                             // Materal preset was found, mark it as installed.
 | |
|                             section_new[preset->name] = "true";
 | |
|                             ++ num_found;
 | |
|                         }
 | |
| 				    }
 | |
|                     if (num_found == 0)
 | |
|             	        BOOST_LOG_TRIVIAL(error) << boost::format("Profile %1% was not found in installed vendor Preset Bundles.") % material_name;
 | |
|                     else if (num_found > 1)
 | |
|             	        BOOST_LOG_TRIVIAL(error) << boost::format("Profile %1% was found in %2% vendor Preset Bundles.") % material_name % num_found;
 | |
|                 }
 | |
| 		}
 | |
|         appconfig_new.set_section(section_name, section_new);
 | |
|     };
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::add_page(ConfigWizardPage *page)
 | |
| {
 | |
|     const int proportion = (page->shortname == _L("Filaments")) || (page->shortname == _L("SLA Materials")) ? 1 : 0;
 | |
|     hscroll_sizer->Add(page, proportion, wxEXPAND);
 | |
|     all_pages.push_back(page);
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::enable_next(bool enable)
 | |
| {
 | |
|     btn_next->Enable(enable);
 | |
|     btn_finish->Enable(enable);
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::set_start_page(ConfigWizard::StartPage start_page)
 | |
| {
 | |
|     switch (start_page) {
 | |
|         case ConfigWizard::SP_PRINTERS: 
 | |
|             btn_next->SetFocus();
 | |
|             break;
 | |
|         case ConfigWizard::SP_FILAMENTS:
 | |
|             btn_finish->SetFocus();
 | |
|             break;
 | |
|         case ConfigWizard::SP_MATERIALS:
 | |
|             btn_finish->SetFocus();
 | |
|             break;
 | |
|         default:
 | |
|             btn_next->SetFocus();
 | |
|             break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::create_3rdparty_pages()
 | |
| {
 | |
|     for (const auto &pair : bundles) {
 | |
|         const VendorProfile *vendor = pair.second.vendor_profile;
 | |
|         //BBS: add BBL as default
 | |
|         if (vendor->id == PresetBundle::BBL_BUNDLE) { continue; }
 | |
| 
 | |
|         bool is_fff_technology = false;
 | |
|         bool is_sla_technology = false;
 | |
| 
 | |
|         for (auto& model: vendor->models)
 | |
|         {
 | |
|             if (!is_fff_technology && model.technology == ptFFF)
 | |
|                  is_fff_technology = true;
 | |
|             if (!is_sla_technology && model.technology == ptSLA)
 | |
|                  is_sla_technology = true;
 | |
|         }
 | |
| 
 | |
|         PagePrinters* pageFFF = nullptr;
 | |
|         PagePrinters* pageSLA = nullptr;
 | |
| 
 | |
|         if (is_fff_technology) {
 | |
|             pageFFF = new PagePrinters(q, vendor->name + " " +_L("FFF Technology Printers"), vendor->name+" FFF", *vendor, 1, T_FFF);
 | |
|             add_page(pageFFF);
 | |
|         }
 | |
| 
 | |
|         if (is_sla_technology) {
 | |
|             pageSLA = new PagePrinters(q, vendor->name + " " + _L("SLA Technology Printers"), vendor->name+" MSLA", *vendor, 1, T_SLA);
 | |
|             add_page(pageSLA);
 | |
|         }
 | |
| 
 | |
|         pages_3rdparty.insert({vendor->id, {pageFFF, pageSLA}});
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::set_run_reason(RunReason run_reason)
 | |
| {
 | |
|     this->run_reason = run_reason;
 | |
|     for (auto &page : all_pages) {
 | |
|         page->set_run_reason(run_reason);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::update_materials(Technology technology)
 | |
| {
 | |
|     if (any_fff_selected && (technology & T_FFF)) {
 | |
|         filaments.clear();
 | |
|         aliases_fff.clear();
 | |
|         // Iterate filaments in all bundles
 | |
|         for (const auto &pair : bundles) {
 | |
|             for (const auto &filament : pair.second.preset_bundle->filaments) {
 | |
|                 // Check if filament is already added
 | |
|                 if (filaments.containts(&filament))
 | |
| 					continue;
 | |
|                 // Iterate printers in all bundles
 | |
|                 for (const auto &printer : pair.second.preset_bundle->printers) {
 | |
| 					if (!printer.is_visible || printer.printer_technology() != ptFFF)
 | |
| 						continue;
 | |
|                     // Filter out inapplicable printers
 | |
| 					if (is_compatible_with_printer(PresetWithVendorProfile(filament, filament.vendor), PresetWithVendorProfile(printer, printer.vendor))) {
 | |
| 						if (!filaments.containts(&filament)) {
 | |
| 							filaments.push(&filament);
 | |
| 							if (!filament.alias.empty())
 | |
| 								aliases_fff[filament.alias].insert(filament.name); 
 | |
| 						} 
 | |
| 						filaments.add_printer(&printer);
 | |
|                     }
 | |
| 				}
 | |
| 				
 | |
|             }
 | |
|         }
 | |
|         // count compatible printers
 | |
|         for (const auto& preset : filaments.presets) {
 | |
| 
 | |
|             const auto filter = [preset](const std::pair<std::string, size_t> element) {
 | |
|                 return preset->alias == element.first;
 | |
|             };
 | |
|             if (std::find_if(filaments.compatibility_counter.begin(), filaments.compatibility_counter.end(), filter) != filaments.compatibility_counter.end()) {
 | |
|                 continue;
 | |
|             }
 | |
|             std::vector<size_t> idx_with_same_alias;
 | |
|             for (size_t i = 0; i < filaments.presets.size(); ++i) {
 | |
|                 if (preset->alias == filaments.presets[i]->alias)
 | |
|                     idx_with_same_alias.push_back(i);
 | |
|             }
 | |
|             size_t counter = 0;
 | |
|             for (const auto& printer : filaments.printers) {
 | |
|                 if (!(*printer).is_visible || (*printer).printer_technology() != ptFFF)
 | |
|                     continue;
 | |
|                 bool compatible = false;
 | |
|                 // Test otrher materials with same alias
 | |
|                 for (size_t i = 0; i < idx_with_same_alias.size() && !compatible; ++i) {
 | |
|                     const Preset& prst = *(filaments.presets[idx_with_same_alias[i]]);
 | |
|                     const Preset& prntr = *printer;
 | |
|                     if (is_compatible_with_printer(PresetWithVendorProfile(prst, prst.vendor), PresetWithVendorProfile(prntr, prntr.vendor))) {
 | |
|                         compatible = true;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|                 if (compatible)
 | |
|                     counter++;
 | |
|             }
 | |
|             filaments.compatibility_counter.emplace_back(preset->alias, counter);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (any_sla_selected && (technology & T_SLA)) {
 | |
|         sla_materials.clear();
 | |
|         aliases_sla.clear();
 | |
| 
 | |
|         // Iterate SLA materials in all bundles
 | |
|         for (const auto &pair : bundles) {
 | |
|             for (const auto &material : pair.second.preset_bundle->sla_materials) {
 | |
|                 // Check if material is already added
 | |
|                 if (sla_materials.containts(&material))
 | |
|                 	continue;
 | |
|                 // Iterate printers in all bundles
 | |
| 				// For now, we only allow the profiles to be compatible with another profiles inside the same bundle.
 | |
|                 for (const auto& printer : pair.second.preset_bundle->printers) {
 | |
|                     if(!printer.is_visible || printer.printer_technology() != ptSLA)
 | |
|                         continue;
 | |
|                     // Filter out inapplicable printers
 | |
|                     if (is_compatible_with_printer(PresetWithVendorProfile(material, nullptr), PresetWithVendorProfile(printer, nullptr))) {
 | |
|                         // Check if material is already added
 | |
|                         if(!sla_materials.containts(&material)) {
 | |
|                             sla_materials.push(&material);
 | |
|                             if (!material.alias.empty())
 | |
|                                 aliases_sla[material.alias].insert(material.name);
 | |
|                         }
 | |
|                         sla_materials.add_printer(&printer);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         // count compatible printers        
 | |
|         for (const auto& preset : sla_materials.presets) {
 | |
|             
 | |
|             const auto filter = [preset](const std::pair<std::string, size_t> element) {
 | |
|                 return preset->alias == element.first;
 | |
|             };
 | |
|             if (std::find_if(sla_materials.compatibility_counter.begin(), sla_materials.compatibility_counter.end(), filter) != sla_materials.compatibility_counter.end()) {
 | |
|                 continue;
 | |
|             }
 | |
|             std::vector<size_t> idx_with_same_alias;
 | |
|             for (size_t i = 0; i < sla_materials.presets.size(); ++i) {
 | |
|                 if(preset->alias == sla_materials.presets[i]->alias)
 | |
|                     idx_with_same_alias.push_back(i);
 | |
|             }
 | |
|             size_t counter = 0;
 | |
|             for (const auto& printer : sla_materials.printers) {
 | |
|                 if (!(*printer).is_visible || (*printer).printer_technology() != ptSLA)
 | |
|                     continue;
 | |
|                 bool compatible = false;
 | |
|                 // Test otrher materials with same alias
 | |
|                 for (size_t i = 0; i < idx_with_same_alias.size() && !compatible; ++i) {
 | |
|                     const Preset& prst = *(sla_materials.presets[idx_with_same_alias[i]]);
 | |
|                     const Preset& prntr = *printer;
 | |
|                     if (is_compatible_with_printer(PresetWithVendorProfile(prst, prst.vendor), PresetWithVendorProfile(prntr, prntr.vendor))) {
 | |
|                         compatible = true;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|                 if (compatible)
 | |
|                     counter++;
 | |
|             }
 | |
|             sla_materials.compatibility_counter.emplace_back(preset->alias, counter);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::on_custom_setup(const bool custom_wanted)
 | |
| {
 | |
| 	custom_printer_selected = custom_wanted;
 | |
|     load_pages();
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::on_printer_pick(PagePrinters *page, const PrinterPickerEvent &evt)
 | |
| {
 | |
|     if (check_sla_selected() != any_sla_selected ||
 | |
|         check_fff_selected() != any_fff_selected) {
 | |
|         any_fff_selected = check_fff_selected();
 | |
|         any_sla_selected = check_sla_selected();
 | |
| 
 | |
|         load_pages();
 | |
|     }
 | |
| 
 | |
|     // Update the is_visible flag on relevant printer profiles
 | |
|     for (auto &pair : bundles) {
 | |
|         if (pair.first != evt.vendor_id) { continue; }
 | |
| 
 | |
|         for (auto &preset : pair.second.preset_bundle->printers) {
 | |
|             if (preset.config.opt_string("printer_model") == evt.model_id
 | |
|                 && preset.config.opt_string("printer_variant") == evt.variant_name) {
 | |
|                 preset.is_visible = evt.enable;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // When a printer model is picked, but there is no material installed compatible with this printer model,
 | |
|         // install default materials for selected printer model silently.
 | |
| 		check_and_install_missing_materials(page->technology, evt.model_id);
 | |
|     }
 | |
| 
 | |
|     if (page->technology & T_FFF) {
 | |
|         page_filaments->clear();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::select_default_materials_for_printer_model(const VendorProfile::PrinterModel &printer_model, Technology technology)
 | |
| {
 | |
|     PageMaterials* page_materials = technology & T_FFF ? page_filaments : page_sla_materials;
 | |
|     for (const std::string& material : printer_model.default_materials)
 | |
|         appconfig_new.set(page_materials->materials->appconfig_section(), material, "true");
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::select_default_materials_for_printer_models(Technology technology, const std::set<const VendorProfile::PrinterModel*> &printer_models)
 | |
| {
 | |
|     PageMaterials     *page_materials    = technology & T_FFF ? page_filaments : page_sla_materials;
 | |
|     const std::string &appconfig_section = page_materials->materials->appconfig_section();
 | |
| 
 | |
|     // Iterate printer_models and select default materials. If none available -> msg to user.
 | |
|     std::vector<const VendorProfile::PrinterModel*> models_without_default;
 | |
|     for (const VendorProfile::PrinterModel* printer_model : printer_models) {
 | |
|         if (printer_model->default_materials.empty()) {
 | |
|             models_without_default.emplace_back(printer_model);
 | |
|         } else {
 | |
|             for (const std::string& material : printer_model->default_materials)
 | |
|                 appconfig_new.set(appconfig_section, material, "true");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!models_without_default.empty()) {
 | |
|         std::string printer_names = "\n\n";
 | |
|         for (const VendorProfile::PrinterModel* printer_model : models_without_default) {
 | |
|             printer_names += printer_model->name + "\n";
 | |
|         }
 | |
|         printer_names += "\n\n";
 | |
|         std::string message = (technology & T_FFF ?
 | |
|             GUI::format(_L("Following printer profiles has no default filament: %1%Please select one manually."), printer_names) :
 | |
|             GUI::format(_L("Following printer profiles has no default material: %1%Please select one manually."), printer_names));
 | |
|         MessageDialog msg(q, message, _L("Notice"), wxOK);
 | |
|         msg.ShowModal();
 | |
|     }
 | |
| 
 | |
|     update_materials(technology);
 | |
|     ((technology & T_FFF) ? page_filaments : page_sla_materials)->reload_presets();
 | |
| }
 | |
| 
 | |
| void ConfigWizard::priv::on_3rdparty_install(const VendorProfile *vendor, bool install)
 | |
| {
 | |
|     auto it = pages_3rdparty.find(vendor->id);
 | |
|     wxCHECK_RET(it != pages_3rdparty.end(), "Internal error: GUI page not found for 3rd party vendor profile");
 | |
| 
 | |
|     for (PagePrinters* page : { it->second.first, it->second.second }) 
 | |
|         if (page) {
 | |
|             if (page->install && !install)
 | |
|                 page->select_all(false);
 | |
|             page->install = install;
 | |
|             // if some 3rd vendor is selected, select first printer for them
 | |
|             if (install)
 | |
|                 page->printer_pickers[0]->select_one(0, true);
 | |
|             page->Layout();
 | |
|         }
 | |
| 
 | |
|     load_pages();
 | |
| }
 | |
| 
 | |
| bool ConfigWizard::priv::on_bnt_finish()
 | |
| {
 | |
|     wxBusyCursor wait;
 | |
|     /* When Filaments or Sla Materials pages are activated, 
 | |
|      * materials for this pages are automaticaly updated and presets are reloaded.
 | |
|      * 
 | |
|      * But, if _Finish_ button was clicked without activation of those pages 
 | |
|      * (for example, just some printers were added/deleted), 
 | |
|      * than last changes wouldn't be updated for filaments/materials.
 | |
|      * SO, do that before close of Wizard
 | |
|      */
 | |
|     update_materials(T_ANY);
 | |
|     if (any_fff_selected)
 | |
|         page_filaments->reload_presets();
 | |
| 
 | |
| 	// theres no need to check that filament is selected if we have only custom printer
 | |
|     if (custom_printer_selected && !any_fff_selected && !any_sla_selected) return true;
 | |
|     // check, that there is selected at least one filament/material
 | |
|     return check_and_install_missing_materials(T_ANY);
 | |
| }
 | |
| 
 | |
| // This allmighty method verifies, whether there is at least a single compatible filament or SLA material installed
 | |
| // for each Printer preset of each Printer Model installed.
 | |
| //
 | |
| // In case only_for_model_id is set, then the test is done for that particular printer model only, and the default materials are installed silently.
 | |
| // Otherwise the user is quieried whether to install the missing default materials or not.
 | |
| // 
 | |
| // Return true if the tested Printer Models already had materials installed.
 | |
| // Return false if there were some Printer Models with missing materials, independent from whether the defaults were installed for these
 | |
| // respective Printer Models or not.
 | |
| bool ConfigWizard::priv::check_and_install_missing_materials(Technology technology, const std::string &only_for_model_id)
 | |
| {
 | |
| 	// Walk over all installed Printer presets and verify whether there is a filament or SLA material profile installed at the same PresetBundle,
 | |
| 	// which is compatible with it.
 | |
|     const auto printer_models_missing_materials = [this, only_for_model_id](PrinterTechnology technology, const std::string §ion)
 | |
|     {
 | |
| 		const std::map<std::string, std::string> &appconfig_presets = appconfig_new.has_section(section) ? appconfig_new.get_section(section) : std::map<std::string, std::string>();
 | |
|     	std::set<const VendorProfile::PrinterModel*> printer_models_without_material;
 | |
|         for (const auto &pair : bundles) {
 | |
|         	const PresetCollection &materials = pair.second.preset_bundle->materials(technology);
 | |
|         	for (const auto &printer : pair.second.preset_bundle->printers) {
 | |
|                 if (printer.is_visible && printer.printer_technology() == technology) {
 | |
| 	            	const VendorProfile::PrinterModel *printer_model = PresetUtils::system_printer_model(printer);
 | |
| 	            	assert(printer_model != nullptr);
 | |
| 	            	if ((only_for_model_id.empty() || only_for_model_id == printer_model->id) &&
 | |
| 	            		printer_models_without_material.find(printer_model) == printer_models_without_material.end()) {
 | |
|                     	bool has_material = false;
 | |
|                         for (const auto& preset : appconfig_presets) {
 | |
| 			            	if (preset.second == "1" || preset.second == "true") {
 | |
| 			            		const Preset *material = materials.find_preset(preset.first, false);
 | |
| 			            		if (material != nullptr && is_compatible_with_printer(PresetWithVendorProfile(*material, nullptr), PresetWithVendorProfile(printer, nullptr))) {
 | |
| 				                	has_material = true;
 | |
| 				                    break;
 | |
| 				                }
 | |
| 			                }
 | |
| 			            }
 | |
| 			            if (! has_material)
 | |
| 			            	printer_models_without_material.insert(printer_model);
 | |
| 			        }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         assert(printer_models_without_material.empty() || only_for_model_id.empty() || only_for_model_id == (*printer_models_without_material.begin())->id);
 | |
|         return printer_models_without_material;
 | |
|     };
 | |
| 
 | |
|     const auto ask_and_select_default_materials = [this](const wxString &message, const std::set<const VendorProfile::PrinterModel*> &printer_models, Technology technology)
 | |
|     {
 | |
|         //wxMessageDialog msg(q, message, _L("Notice"), wxYES_NO);
 | |
|         MessageDialog msg(q, message, _L("Notice"), wxYES_NO);
 | |
|         if (msg.ShowModal() == wxID_YES)
 | |
|             select_default_materials_for_printer_models(technology, printer_models);
 | |
|     };
 | |
| 
 | |
|     const auto printer_model_list = [](const std::set<const VendorProfile::PrinterModel*> &printer_models) -> wxString {
 | |
|     	wxString out;
 | |
|     	for (const VendorProfile::PrinterModel *printer_model : printer_models) {
 | |
|             wxString name = from_u8(printer_model->name);
 | |
|     		out += "\t\t";
 | |
|     		out += name;
 | |
|     		out += "\n";
 | |
|     	}
 | |
|     	return out;
 | |
|     };
 | |
| 
 | |
|     if (any_fff_selected && (technology & T_FFF)) {
 | |
|     	std::set<const VendorProfile::PrinterModel*> printer_models_without_material = printer_models_missing_materials(ptFFF, AppConfig::SECTION_FILAMENTS);
 | |
|     	if (! printer_models_without_material.empty()) {
 | |
| 			if (only_for_model_id.empty())
 | |
| 				ask_and_select_default_materials(
 | |
| 					_L("The following FFF printer models have no filament selected:") +
 | |
| 					"\n\n\t" +
 | |
| 					printer_model_list(printer_models_without_material) +
 | |
| 					"\n\n\t" +
 | |
| 					_L("Do you want to select default filaments for these FFF printer models?"),
 | |
| 					printer_models_without_material,
 | |
| 					T_FFF);
 | |
| 			else
 | |
| 				select_default_materials_for_printer_model(**printer_models_without_material.begin(), T_FFF);
 | |
| 			return false;
 | |
| 		}
 | |
|     }
 | |
| 
 | |
|     /* BBS skip SLA
 | |
|     if (any_sla_selected && (technology & T_SLA)) {
 | |
|     	std::set<const VendorProfile::PrinterModel*> printer_models_without_material = printer_models_missing_materials(ptSLA, AppConfig::SECTION_MATERIALS);
 | |
|     	if (! printer_models_without_material.empty()) {
 | |
| 	        if (only_for_model_id.empty())
 | |
| 	            ask_and_select_default_materials(
 | |
| 					_L("The following SLA printer models have no materials selected:") +
 | |
| 	            	"\n\n\t" +
 | |
| 				   	printer_model_list(printer_models_without_material) +
 | |
| 					"\n\n\t" +
 | |
| 					_L("Do you want to select default SLA materials for these printer models?"),
 | |
| 					printer_models_without_material,
 | |
| 	            	T_SLA);
 | |
| 	        else
 | |
| 				select_default_materials_for_printer_model(**printer_models_without_material.begin(), T_SLA);
 | |
| 	        return false;
 | |
| 	    }
 | |
|     }
 | |
|     */
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static std::set<std::string> get_new_added_presets(const std::map<std::string, std::string>& old_data, const std::map<std::string, std::string>& new_data) 
 | |
| {
 | |
|     auto get_aliases = [](const std::map<std::string, std::string>& data) {
 | |
|         std::set<std::string> old_aliases;
 | |
|         for (auto item : data) {
 | |
|             const std::string& name = item.first;
 | |
|             size_t pos = name.find("@");
 | |
|             old_aliases.emplace(pos == std::string::npos ? name : name.substr(0, pos-1));
 | |
|         }
 | |
|         return old_aliases;
 | |
|     };
 | |
| 
 | |
|     std::set<std::string> old_aliases = get_aliases(old_data);
 | |
|     std::set<std::string> new_aliases = get_aliases(new_data);
 | |
|     std::set<std::string> diff;
 | |
|     std::set_difference(new_aliases.begin(), new_aliases.end(), old_aliases.begin(), old_aliases.end(), std::inserter(diff, diff.begin()));
 | |
| 
 | |
|     return diff;
 | |
| }
 | |
| 
 | |
| static std::string get_first_added_preset(const std::map<std::string, std::string>& old_data, const std::map<std::string, std::string>& new_data)
 | |
| {
 | |
|     std::set<std::string> diff = get_new_added_presets(old_data, new_data);
 | |
|     if (diff.empty())
 | |
|         return std::string();
 | |
|     return *diff.begin();
 | |
| }
 | |
| 
 | |
| bool ConfigWizard::priv::apply_config(AppConfig *app_config, PresetBundle *preset_bundle, const PresetUpdater *updater, bool& apply_keeped_changes)
 | |
| {
 | |
|     wxString header, caption = _L("Configuration is edited in ConfigWizard");
 | |
|     const auto enabled_vendors = appconfig_new.vendors();
 | |
| 
 | |
|     bool suppress_sla_printer = model_has_multi_part_objects(wxGetApp().model());
 | |
|     PrinterTechnology preferred_pt = ptAny;
 | |
|     auto get_preferred_printer_technology = [enabled_vendors, suppress_sla_printer](const std::string& bundle_name, const Bundle& bundle) {
 | |
|         const auto config = enabled_vendors.find(bundle_name);
 | |
|         PrinterTechnology pt = ptAny;
 | |
|         if (config != enabled_vendors.end()) {
 | |
|             for (const auto& model : bundle.vendor_profile->models) {
 | |
|                 if (const auto model_it = config->second.find(model.id);
 | |
|                     model_it != config->second.end() && model_it->second.size() > 0) {
 | |
|                     if (pt == ptAny)
 | |
|                         pt = model.technology;
 | |
|                     // if preferred printer model has SLA printer technology it's important to check the model for multypart state
 | |
|                     if (pt == ptSLA && suppress_sla_printer)
 | |
|                         continue;
 | |
|                     else
 | |
|                         return pt;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return pt;
 | |
|     };
 | |
|     // Prusa printers are considered first, then 3rd party.
 | |
|     if (preferred_pt = get_preferred_printer_technology("BBL", bundles.bbl_bundle());
 | |
|         preferred_pt == ptAny || (preferred_pt == ptSLA && suppress_sla_printer)) {
 | |
|         for (const auto& bundle : bundles) {
 | |
|             //BBS: set BBL as default
 | |
|             if (bundle.second.is_bbl_bundle) { continue; }
 | |
|             if (PrinterTechnology pt = get_preferred_printer_technology(bundle.first, bundle.second); pt == ptAny)
 | |
|                 continue;
 | |
|             else if (preferred_pt == ptAny)
 | |
|                 preferred_pt = pt;
 | |
|             if(!(preferred_pt == ptAny || (preferred_pt == ptSLA && suppress_sla_printer)))
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (preferred_pt == ptSLA && !wxGetApp().may_switch_to_SLA_preset(caption))
 | |
|         return false;
 | |
| 
 | |
|     bool check_unsaved_preset_changes = false;
 | |
|     if (check_unsaved_preset_changes)
 | |
|         header = _L("All user presets will be deleted.");
 | |
|     int act_btns = UnsavedChangesDialog::ActionButtons::KEEP;
 | |
|     if (!check_unsaved_preset_changes)
 | |
|         act_btns |= UnsavedChangesDialog::ActionButtons::SAVE;
 | |
| 
 | |
|     // Install bundles from resources if needed:
 | |
|     std::vector<std::string> install_bundles;
 | |
|     for (const auto &pair : bundles) {
 | |
|         if (! pair.second.is_in_resources) { continue; }
 | |
| 
 | |
|         //BBS: set BBL as default
 | |
|         if (pair.second.is_bbl_bundle) {
 | |
|             // Always install Prusa bundle, because it has a lot of filaments/materials
 | |
|             // likely to be referenced by other profiles.
 | |
|             install_bundles.emplace_back(pair.first);
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         const auto vendor = enabled_vendors.find(pair.first);
 | |
|         if (vendor == enabled_vendors.end()) { continue; }
 | |
| 
 | |
|         size_t size_sum = 0;
 | |
|         for (const auto &model : vendor->second) { size_sum += model.second.size(); }
 | |
| 
 | |
|         if (size_sum > 0) {
 | |
|             // This vendor needs to be installed
 | |
|             install_bundles.emplace_back(pair.first);
 | |
|         }
 | |
|     }
 | |
|     if (!check_unsaved_preset_changes)
 | |
|         if ((check_unsaved_preset_changes = install_bundles.size() > 0))
 | |
|             header = _L_PLURAL("A new vendor was installed and one of its printers will be activated", "New vendors were installed and one of theirs printers will be activated", install_bundles.size());
 | |
| 
 | |
|     // Decide whether to create snapshot based on run_reason and the reset profile checkbox
 | |
|     bool snapshot = true;
 | |
|     Snapshot::Reason snapshot_reason = Snapshot::SNAPSHOT_UPGRADE;
 | |
|     switch (run_reason) {
 | |
|         case ConfigWizard::RR_DATA_EMPTY:
 | |
|             snapshot = false;
 | |
|             break;
 | |
|         case ConfigWizard::RR_DATA_LEGACY:
 | |
|             snapshot = true;
 | |
|             break;
 | |
|         case ConfigWizard::RR_DATA_INCOMPAT:
 | |
|             // In this case snapshot has already been taken by
 | |
|             // PresetUpdater with the appropriate reason
 | |
|             snapshot = false;
 | |
|             break;
 | |
|         case ConfigWizard::RR_USER:
 | |
|             snapshot = false;
 | |
|             snapshot_reason = Snapshot::SNAPSHOT_USER;
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     //BBS: remove snapshot logic
 | |
|     /*if (snapshot && ! take_config_snapshot_cancel_on_error(*app_config, snapshot_reason, "", _u8L("Do you want to continue changing the configuration?")))
 | |
|         return false;*/
 | |
| 
 | |
|     if (check_unsaved_preset_changes &&
 | |
|         !wxGetApp().check_and_keep_current_preset_changes(caption, header, act_btns, &apply_keeped_changes))
 | |
|         return false;
 | |
| 
 | |
|     if (install_bundles.size() > 0) {
 | |
|         // Install bundles from resources.
 | |
|         // Don't create snapshot - we've already done that above if applicable.
 | |
|         if (! updater->install_bundles_rsrc(std::move(install_bundles), false))
 | |
|             return false;
 | |
|     } else {
 | |
|         BOOST_LOG_TRIVIAL(info) << "No bundles need to be installed from resources";
 | |
|     }
 | |
| 
 | |
|     std::string preferred_model;
 | |
|     std::string preferred_variant;
 | |
|     const auto enabled_vendors_old = app_config->vendors();
 | |
|     auto get_preferred_printer_model = [enabled_vendors, enabled_vendors_old, preferred_pt](const std::string& bundle_name, const Bundle& bundle, std::string& variant) {
 | |
|         const auto config = enabled_vendors.find(bundle_name);
 | |
|         if (config == enabled_vendors.end())
 | |
|             return std::string();
 | |
|         for (const auto& model : bundle.vendor_profile->models) {
 | |
|             if (const auto model_it = config->second.find(model.id);
 | |
|                 model_it != config->second.end() && model_it->second.size() > 0 &&
 | |
|                 preferred_pt == model.technology) {
 | |
|                 variant = *model_it->second.begin();
 | |
|                 const auto config_old = enabled_vendors_old.find(bundle_name);
 | |
|                 if (config_old == enabled_vendors_old.end())
 | |
|                     return model.id;
 | |
|                 const auto model_it_old = config_old->second.find(model.id);
 | |
|                 if (model_it_old == config_old->second.end())
 | |
|                     return model.id;
 | |
|                 else if (model_it_old->second != model_it->second) {
 | |
|                     for (const auto& var : model_it->second)
 | |
|                         if (model_it_old->second.find(var) == model_it_old->second.end()) {
 | |
|                             variant = var;
 | |
|                             return model.id;
 | |
|                         }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         if (!variant.empty())
 | |
|             variant.clear();
 | |
|         return std::string();
 | |
|     };
 | |
|     // Prusa printers are considered first, then 3rd party.
 | |
|     if (preferred_model = get_preferred_printer_model("BBL", bundles.bbl_bundle(), preferred_variant);
 | |
|         preferred_model.empty()) {
 | |
|         for (const auto& bundle : bundles) {
 | |
|             if (bundle.second.is_bbl_bundle) { continue; }
 | |
|             if (preferred_model = get_preferred_printer_model(bundle.first, bundle.second, preferred_variant);
 | |
|                 !preferred_model.empty())
 | |
|                     break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // if unsaved changes was not cheched till this moment
 | |
|     if (!check_unsaved_preset_changes) {
 | |
|         if ((check_unsaved_preset_changes = !preferred_model.empty())) {
 | |
|             header = _L("A new Printer was installed and it will be activated.");
 | |
|             if (!wxGetApp().check_and_keep_current_preset_changes(caption, header, act_btns, &apply_keeped_changes))
 | |
|                 return false;
 | |
|         }
 | |
|         else if ((check_unsaved_preset_changes = enabled_vendors_old != enabled_vendors)) {
 | |
|             header = _L("Some Printers were uninstalled.");
 | |
|             if (!wxGetApp().check_and_keep_current_preset_changes(caption, header, act_btns, &apply_keeped_changes))
 | |
|                 return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     std::string first_added_filament, first_added_sla_material;
 | |
|     auto get_first_added_material_preset = [this, app_config](const std::string& section_name, std::string& first_added_preset) {
 | |
|         if (appconfig_new.has_section(section_name)) {
 | |
|             // get first of new added preset names
 | |
|             const std::map<std::string, std::string>& old_presets = app_config->has_section(section_name) ? app_config->get_section(section_name) : std::map<std::string, std::string>();
 | |
|             first_added_preset = get_first_added_preset(old_presets, appconfig_new.get_section(section_name));
 | |
|         }
 | |
|     };
 | |
|     get_first_added_material_preset(AppConfig::SECTION_FILAMENTS, first_added_filament);
 | |
| 
 | |
|     //BBS skip sla get_first_added_material_preset(AppConfig::SECTION_MATERIALS, first_added_sla_material);
 | |
| 
 | |
|     // if unsaved changes was not cheched till this moment
 | |
|     if (!check_unsaved_preset_changes) {
 | |
|         if ((check_unsaved_preset_changes = !first_added_filament.empty() || !first_added_sla_material.empty())) {
 | |
|             header = !first_added_filament.empty() ? 
 | |
|                      _L("A new filament was installed and it will be activated.") :
 | |
|                      _L("A new SLA material was installed and it will be activated.");
 | |
|             if (!wxGetApp().check_and_keep_current_preset_changes(caption, header, act_btns, &apply_keeped_changes))
 | |
|                 return false;
 | |
|         }
 | |
|         else {
 | |
|             //BBS
 | |
|             bool is_filaments_changed = app_config->get_section(AppConfig::SECTION_FILAMENTS) != appconfig_new.get_section(AppConfig::SECTION_FILAMENTS);
 | |
|             if ((check_unsaved_preset_changes = is_filaments_changed)) {
 | |
|                 header = _L("Some filaments were uninstalled.");
 | |
|                 if (!wxGetApp().check_and_keep_current_preset_changes(caption, header, act_btns, &apply_keeped_changes))
 | |
|                     return false;
 | |
|             }
 | |
| 
 | |
|             /*bool is_filaments_changed = app_config->get_section(AppConfig::SECTION_FILAMENTS) != appconfig_new.get_section(AppConfig::SECTION_FILAMENTS);
 | |
|             bool is_sla_materials_changed = app_config->get_section(AppConfig::SECTION_MATERIALS) != appconfig_new.get_section(AppConfig::SECTION_MATERIALS);
 | |
|             if ((check_unsaved_preset_changes = is_filaments_changed || is_sla_materials_changed)) {
 | |
|                 header = is_filaments_changed ? _L("Some filaments were uninstalled.") : _L("Some SLA materials were uninstalled.");
 | |
|                 if (!wxGetApp().check_and_keep_current_preset_changes(caption, header, act_btns, &apply_keeped_changes))
 | |
|                     return false;
 | |
|             }*/
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // apply materials in app_config
 | |
|     //BBS skip sla
 | |
|     for (const std::string& section_name : {AppConfig::SECTION_FILAMENTS})
 | |
|         app_config->set_section(section_name, appconfig_new.get_section(section_name));
 | |
| 
 | |
|     app_config->set_vendors(appconfig_new);
 | |
| 
 | |
| #ifdef _WIN32
 | |
|     app_config->set_bool("associate_3mf", page_files_association->associate_3mf());
 | |
|     app_config->set_bool("associate_stl", page_files_association->associate_stl());
 | |
| 
 | |
|     if (wxGetApp().is_editor()) {
 | |
|         if (page_files_association->associate_3mf())
 | |
|             wxGetApp().associate_files(L"3mf");
 | |
|         if (page_files_association->associate_stl())
 | |
|             wxGetApp().associate_files(L"stl");
 | |
|     }
 | |
| //    else {
 | |
| //        if (page_files_association->associate_gcode())
 | |
| //            wxGetApp().associate_gcode_files();
 | |
| //    }
 | |
| 
 | |
| #endif // _WIN32
 | |
| 
 | |
|     if (check_unsaved_preset_changes)
 | |
|         preset_bundle->load_presets(*app_config, ForwardCompatibilitySubstitutionRule::EnableSilentDisableSystem, 
 | |
|                                     {preferred_model, preferred_variant, first_added_filament, first_added_sla_material});
 | |
| 
 | |
|     // Update the selections from the compatibilty.
 | |
|     preset_bundle->export_selections(*app_config);
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| void ConfigWizard::priv::update_presets_in_config(const std::string& section, const std::string& alias_key, bool add)
 | |
| {
 | |
|     const PresetAliases& aliases = section == AppConfig::SECTION_FILAMENTS ? aliases_fff : aliases_sla;
 | |
| 
 | |
|     auto update = [this, add](const std::string& s, const std::string& key) {
 | |
|     	assert(! s.empty());
 | |
|         if (add)
 | |
|             appconfig_new.set(s, key, "true");
 | |
|         else
 | |
|             appconfig_new.erase(s, key); 
 | |
|     };
 | |
| 
 | |
|     // add or delete presets had a same alias 
 | |
|     auto it = aliases.find(alias_key);
 | |
|     if (it != aliases.end())
 | |
|         for (const std::string& name : it->second)
 | |
|             update(section, name);
 | |
| }
 | |
| 
 | |
| bool ConfigWizard::priv::check_fff_selected()
 | |
| {
 | |
|     bool ret = page_fff->any_selected();
 | |
|     for (const auto& printer: pages_3rdparty)
 | |
|         if (printer.second.first)               // FFF page
 | |
|             ret |= printer.second.first->any_selected();
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| bool ConfigWizard::priv::check_sla_selected()
 | |
| {
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| // Public
 | |
| 
 | |
| ConfigWizard::ConfigWizard(wxWindow *parent)
 | |
|     : DPIDialog(parent, wxID_ANY, wxString(SLIC3R_APP_NAME) + " - " + _(name()), wxDefaultPosition, wxDefaultSize, wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
 | |
|     , p(new priv(this))
 | |
| {
 | |
|     this->SetFont(wxGetApp().normal_font());
 | |
| 
 | |
|     p->load_vendors();
 | |
|     //BBS: add bed exclude areas
 | |
|     p->custom_config.reset(DynamicPrintConfig::new_from_defaults_keys({
 | |
|         "gcode_flavor", "printable_area", "bed_exclude_area", "nozzle_diameter", "filament_diameter", "nozzle_temperature", /*"bed_temperature",*/
 | |
|     }));
 | |
| 
 | |
|     auto *vsizer = new wxBoxSizer(wxVERTICAL);
 | |
|     auto *topsizer = new wxBoxSizer(wxHORIZONTAL);
 | |
|     auto* hline = new StaticLine(this);
 | |
|     p->btnsizer = new wxBoxSizer(wxHORIZONTAL);
 | |
| 
 | |
|     // Initially we _do not_ SetScrollRate in order to figure out the overall width of the Wizard  without scrolling.
 | |
|     // Later, we compare that to the size of the current screen and set minimum width based on that (see below).
 | |
|     p->hscroll = new wxScrolledWindow(this);
 | |
|     p->hscroll_sizer = new wxBoxSizer(wxHORIZONTAL);
 | |
|     p->hscroll->SetSizer(p->hscroll_sizer);
 | |
| 
 | |
|     topsizer->AddSpacer(INDEX_MARGIN);
 | |
|     topsizer->Add(p->hscroll, 1, wxEXPAND);
 | |
| 
 | |
|     p->btn_sel_all = new wxButton(this, wxID_ANY, _L("Select all standard printers"));
 | |
|     p->btnsizer->Add(p->btn_sel_all);
 | |
| 
 | |
|     p->btn_prev = new wxButton(this, wxID_ANY, _L("< &Back"));
 | |
|     p->btn_next = new wxButton(this, wxID_ANY, _L("&Next >"));
 | |
|     p->btn_finish = new wxButton(this, wxID_APPLY, _L("&Finish"));
 | |
|     p->btn_cancel = new wxButton(this, wxID_CANCEL, _L("Cancel"));   // Note: The label needs to be present, otherwise we get accelerator bugs on Mac
 | |
|     p->btnsizer->AddStretchSpacer();
 | |
|     p->btnsizer->Add(p->btn_prev, 0, wxLEFT, BTN_SPACING);
 | |
|     p->btnsizer->Add(p->btn_next, 0, wxLEFT, BTN_SPACING);
 | |
|     p->btnsizer->Add(p->btn_finish, 0, wxLEFT, BTN_SPACING);
 | |
|     p->btnsizer->Add(p->btn_cancel, 0, wxLEFT, BTN_SPACING);
 | |
| 
 | |
|     wxGetApp().UpdateDarkUI(p->btn_sel_all);
 | |
|     wxGetApp().UpdateDarkUI(p->btn_prev);
 | |
|     wxGetApp().UpdateDarkUI(p->btn_next);
 | |
|     wxGetApp().UpdateDarkUI(p->btn_finish);
 | |
|     wxGetApp().UpdateDarkUI(p->btn_cancel);
 | |
| 
 | |
|     //BBS: add BBL as default
 | |
|     const auto bbl_it = p->bundles.find("BBL");
 | |
|     wxCHECK_RET(bbl_it != p->bundles.cend(), "Vendor BambooLab not found");
 | |
|     const VendorProfile * vendor_bbl = bbl_it->second.vendor_profile;
 | |
|     
 | |
|     p->page_fff = new PagePrinters(this, _L("BBL FFF Technology Printers"), "BBL FFF", *vendor_bbl, 0, T_FFF);
 | |
|     p->only_sla_mode = false;
 | |
|     if (!p->only_sla_mode) {
 | |
|         p->add_page(p->page_fff);
 | |
|         p->page_fff->is_primary_printer_page = true;
 | |
|     }
 | |
| 
 | |
|     if (!p->only_sla_mode) {
 | |
| 	    // Pages for 3rd party vendors
 | |
| 	    p->create_3rdparty_pages();   // Needs to be done _before_ creating PageVendors
 | |
| 	    p->add_page(p->page_vendors = new PageVendors(this));
 | |
| 	    //p->add_page(p->page_custom = new PageCustom(this));
 | |
|         //p->custom_printer_selected = p->page_custom->custom_wanted();
 | |
|     }
 | |
| 
 | |
|     p->any_sla_selected = p->check_sla_selected();
 | |
|     if (p->only_sla_mode)
 | |
|         p->any_fff_selected = p->check_fff_selected();
 | |
| 
 | |
|     p->update_materials(T_ANY);
 | |
|     if (!p->only_sla_mode)
 | |
|         p->add_page(p->page_filaments = new PageMaterials(this, &p->filaments,
 | |
|             _L("Filament Profiles Selection"), _L("Filaments"), _L("Type:") ));
 | |
| 
 | |
| #ifdef _WIN32
 | |
|     p->add_page(p->page_files_association = new PageFilesAssociation(this));
 | |
| #endif // _WIN32
 | |
|     //p->add_page(p->page_bed      = new PageBedShape(this));
 | |
|     p->add_page(p->page_diams    = new PageDiameters(this));
 | |
|     p->add_page(p->page_temps    = new PageTemperatures(this));
 | |
| 
 | |
|     p->load_pages();
 | |
| 
 | |
|     vsizer->Add(topsizer, 1, wxEXPAND | wxALL, DIALOG_MARGIN);
 | |
|     vsizer->Add(hline, 0, wxEXPAND | wxLEFT | wxRIGHT, VERTICAL_SPACING);
 | |
|     vsizer->Add(p->btnsizer, 0, wxEXPAND | wxALL, DIALOG_MARGIN);
 | |
| 
 | |
|     SetSizer(vsizer);
 | |
|     SetSizerAndFit(vsizer);
 | |
| 
 | |
|     // We can now enable scrolling on hscroll
 | |
|     p->hscroll->SetScrollRate(30, 30);
 | |
| 
 | |
|     on_window_geometry(this, [this]() {
 | |
|         p->init_dialog_size();
 | |
|     });
 | |
| 
 | |
|     p->btn_next->Bind(wxEVT_BUTTON, [this](const wxCommandEvent &)
 | |
|     {
 | |
|         ;
 | |
|     });
 | |
| 
 | |
|     p->btn_finish->Bind(wxEVT_BUTTON, [this](const wxCommandEvent &)
 | |
|     {
 | |
|         if (p->on_bnt_finish())
 | |
|             this->EndModal(wxID_OK);
 | |
|     });
 | |
| 
 | |
|     p->btn_sel_all->Bind(wxEVT_BUTTON, [this](const wxCommandEvent &) {
 | |
|         p->any_sla_selected = true;
 | |
|         p->load_pages();
 | |
|         p->page_fff->select_all(true, false);
 | |
|     });
 | |
| 
 | |
|     if (wxLinux_gtk3)
 | |
|         this->Bind(wxEVT_SHOW, [this, vsizer](const wxShowEvent& e) {
 | |
|             ;
 | |
|         });
 | |
| }
 | |
| 
 | |
| ConfigWizard::~ConfigWizard() {}
 | |
| 
 | |
| bool ConfigWizard::run(RunReason reason, StartPage start_page)
 | |
| {
 | |
|     BOOST_LOG_TRIVIAL(info) << boost::format("Running ConfigWizard, reason: %1%, start_page: %2%") % reason % start_page;
 | |
| 
 | |
|     GUI_App &app = wxGetApp();
 | |
| 
 | |
|     p->set_run_reason(reason);
 | |
|     p->set_start_page(start_page);
 | |
| 
 | |
|     if (ShowModal() == wxID_OK) {
 | |
|         bool apply_keeped_changes = false;
 | |
|         if (! p->apply_config(app.app_config, app.preset_bundle, app.preset_updater, apply_keeped_changes))
 | |
|             return false;
 | |
| 
 | |
|         if (apply_keeped_changes)
 | |
|             app.apply_keeped_preset_modifications();
 | |
| 
 | |
|         app.app_config->set_legacy_datadir(false);
 | |
|         app.update_mode();
 | |
|         // BBS
 | |
|         //app.obj_manipul()->update_ui_from_settings();
 | |
|         BOOST_LOG_TRIVIAL(info) << "ConfigWizard applied";
 | |
|         return true;
 | |
|     } else {
 | |
|         BOOST_LOG_TRIVIAL(info) << "ConfigWizard cancelled";
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| const wxString& ConfigWizard::name(const bool from_menu/* = false*/)
 | |
| {
 | |
|     // A different naming convention is used for the Wizard on Windows & GTK vs. OSX.
 | |
|     // Note: Don't call _() macro here.
 | |
|     //       This function just return the current name according to the OS.
 | |
|     //       Translation is implemented inside GUI_App::add_config_menu()
 | |
| #if __APPLE__
 | |
|     static const wxString config_wizard_name =  L("Configuration Assistant");
 | |
|     static const wxString config_wizard_name_menu = L("Configuration &Assistant");
 | |
| #else
 | |
|     static const wxString config_wizard_name = L("Configuration Wizard");
 | |
|     static const wxString config_wizard_name_menu = L("Configuration &Wizard");
 | |
| #endif
 | |
|     return from_menu ? config_wizard_name_menu : config_wizard_name;
 | |
| }
 | |
| 
 | |
| void ConfigWizard::on_dpi_changed(const wxRect &suggested_rect)
 | |
| {
 | |
|     const int em = em_unit();
 | |
| 
 | |
|     msw_buttons_rescale(this, em, { wxID_APPLY, 
 | |
|                                     wxID_CANCEL,
 | |
|                                     p->btn_sel_all->GetId(),
 | |
|                                     p->btn_next->GetId(),
 | |
|                                     p->btn_prev->GetId() });
 | |
| 
 | |
|     for (auto printer_picker: p->page_fff->printer_pickers)
 | |
|         msw_buttons_rescale(this, em, printer_picker->get_button_indexes());
 | |
| 
 | |
|     p->init_dialog_size();
 | |
| 
 | |
|     Refresh();
 | |
| }
 | |
| 
 | |
| void ConfigWizard::on_sys_color_changed()
 | |
| {
 | |
|     wxGetApp().UpdateDlgDarkUI(this);
 | |
|     Refresh();
 | |
| }
 | |
| 
 | |
| }
 | |
| }
 | 
