#include "PhysicalPrinterDialog.hpp" #include "PresetComboBoxes.hpp" #include #include #include #include #include #include #include #include #include #include #include "libslic3r/libslic3r.h" #include "libslic3r/PrintConfig.hpp" #include "libslic3r/PresetBundle.hpp" #include "GUI.hpp" #include "GUI_App.hpp" #include "MainFrame.hpp" #include "format.hpp" #include "Tab.hpp" #include "wxExtensions.hpp" //#include "PrintHostDialogs.hpp" #include "../Utils/ASCIIFolding.hpp" #include "../Utils/PrintHost.hpp" #include "../Utils/FixModelByWin10.hpp" #include "../Utils/UndoRedo.hpp" #include "RemovableDriveManager.hpp" #include "BitmapCache.hpp" #include "BonjourDialog.hpp" #include "MsgDialog.hpp" namespace Slic3r { namespace GUI { #define BORDER_W FromDIP(10) //------------------------------------------ // PhysicalPrinterDialog //------------------------------------------ PhysicalPrinterDialog::PhysicalPrinterDialog(wxWindow* parent) : DPIDialog(parent, wxID_ANY, _L("Physical Printer"), wxDefaultPosition, wxSize(45 * wxGetApp().em_unit(), -1), wxDEFAULT_DIALOG_STYLE) { SetFont(wxGetApp().normal_font()); SetBackgroundColour(*wxWHITE); // input the preset name Tab *tab = wxGetApp().get_tab(Preset::TYPE_PRINTER); m_presets = tab->get_presets(); const Preset &sel_preset = m_presets->get_selected_preset(); std::string suffix = _CTX_utf8(L_CONTEXT("Copy", "PresetName"), "PresetName"); std::string preset_name = sel_preset.is_default ? "Untitled" : sel_preset.is_system ? (boost::format(("%1% - %2%")) % sel_preset.name % suffix).str() : sel_preset.name; auto input_sizer = new wxBoxSizer(wxVERTICAL); wxStaticText *label_top = new wxStaticText(this, wxID_ANY, from_u8((boost::format(_utf8(L("Save %s as"))) % into_u8(tab->title())).str())); label_top->SetFont(::Label::Body_13); label_top->SetForegroundColour(wxColour(38,46,48)); m_input_area = new RoundedRectangle(this, wxColour(172, 172, 172), wxDefaultPosition, wxSize(-1, -1), 3, 1); m_input_area->SetMinSize(wxSize(FromDIP(360), FromDIP(32))); wxBoxSizer *input_sizer_h = new wxBoxSizer(wxHORIZONTAL); wxBoxSizer *input_sizer_v = new wxBoxSizer(wxVERTICAL); m_input_ctrl = new wxTextCtrl(m_input_area, -1, from_u8(preset_name), wxDefaultPosition, wxSize(FromDIP(360), -1), 0 | wxBORDER_NONE); m_input_ctrl->SetBackgroundColour(*wxWHITE); m_input_ctrl->Bind(wxEVT_TEXT, [this](wxCommandEvent &) { update(); }); input_sizer_v->Add(m_input_ctrl, 0, wxALIGN_CENTER | wxLEFT | wxRIGHT, BORDER_W); input_sizer_h->Add(input_sizer_v, 0, wxALIGN_CENTER, 0); m_input_area->SetSizer(input_sizer_h); m_input_area->Layout(); m_valid_label = new wxStaticText(this, wxID_ANY, ""); m_valid_label->SetForegroundColour(wxColor(255, 111, 0)); input_sizer->Add(label_top, 0, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, BORDER_W); input_sizer->Add(m_input_area, 0, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, BORDER_W); input_sizer->Add(m_valid_label, 0, wxEXPAND | wxLEFT | wxRIGHT, BORDER_W); m_config = &wxGetApp().preset_bundle->printers.get_edited_preset().config; m_optgroup = new ConfigOptionsGroup(this, _L("Print Host upload"), m_config); build_printhost_settings(m_optgroup); auto button_sizer = new wxBoxSizer(wxHORIZONTAL); StateColor btn_bg_green(std::pair(wxColour(27, 136, 68), StateColor::Pressed), std::pair(wxColour(61, 203, 115), StateColor::Hovered), std::pair(AMS_CONTROL_BRAND_COLOUR, StateColor::Normal)); StateColor btn_bg_white(std::pair(wxColour(206, 206, 206), StateColor::Pressed), std::pair(wxColour(238, 238, 238), StateColor::Hovered), std::pair(*wxWHITE, StateColor::Normal)); m_button_ok = new Button(this, _L("OK")); m_button_ok->SetBackgroundColor(btn_bg_green); m_button_ok->SetBorderColor(*wxWHITE); m_button_ok->SetTextColor(*wxWHITE); m_button_ok->SetFont(Label::Body_12); m_button_ok->SetSize(wxSize(FromDIP(58), FromDIP(24))); m_button_ok->SetMinSize(wxSize(FromDIP(58), FromDIP(24))); m_button_ok->SetCornerRadius(FromDIP(12)); m_button_ok->Bind(wxEVT_LEFT_DOWN, &PhysicalPrinterDialog::OnOK, this); m_button_cancel = new Button(this, _L("Cancel")); m_button_cancel->SetBackgroundColor(btn_bg_white); m_button_cancel->SetBorderColor(wxColour(38, 46, 48)); m_button_cancel->SetFont(Label::Body_12); m_button_cancel->SetSize(wxSize(FromDIP(58), FromDIP(24))); m_button_cancel->SetMinSize(wxSize(FromDIP(58), FromDIP(24))); m_button_cancel->SetCornerRadius(FromDIP(12)); m_button_cancel->Bind(wxEVT_LEFT_DOWN, [this](auto &e) { this->EndModal(wxID_NO);}); button_sizer->AddStretchSpacer(); button_sizer->Add(m_button_ok, 0, wxALL, FromDIP(5)); button_sizer->Add(m_button_cancel, 0, wxALL, FromDIP(5)); wxBoxSizer *topSizer = new wxBoxSizer(wxVERTICAL); // topSizer->Add(label_top , 0, wxEXPAND | wxLEFT | wxTOP | wxRIGHT, BORDER_W); topSizer->Add(input_sizer, 0, wxEXPAND | wxALL, BORDER_W); topSizer->Add(m_optgroup->sizer, 1, wxEXPAND | wxLEFT | wxTOP | wxRIGHT, BORDER_W); topSizer->Add(button_sizer, 0, wxEXPAND | wxALL, BORDER_W); Bind(wxEVT_CLOSE_WINDOW, [this](auto& e) {this->EndModal(wxID_NO);}); SetSizer(topSizer); topSizer->SetSizeHints(this); this->CenterOnParent(); wxGetApp().UpdateDlgDarkUI(this); } PhysicalPrinterDialog::~PhysicalPrinterDialog() { } void PhysicalPrinterDialog::build_printhost_settings(ConfigOptionsGroup* m_optgroup) { m_optgroup->m_on_change = [this](t_config_option_key opt_key, boost::any value) { if (opt_key == "host_type" || opt_key == "printhost_authorization_type") this->update(); if (opt_key == "print_host") this->update_printhost_buttons(); }; m_optgroup->append_single_option_line("host_type"); auto create_sizer_with_btn = [](wxWindow* parent, ScalableButton** btn, const std::string& icon_name, const wxString& label) { *btn = new ScalableButton(parent, wxID_ANY, icon_name, label, wxDefaultSize, wxDefaultPosition, wxBU_LEFT | wxBU_EXACTFIT); (*btn)->SetFont(wxGetApp().normal_font()); auto sizer = new wxBoxSizer(wxHORIZONTAL); sizer->Add(*btn); return sizer; }; auto printhost_browse = [=](wxWindow* parent) { auto sizer = create_sizer_with_btn(parent, &m_printhost_browse_btn, "printer_host_browser", _L("Browse") + " " + dots); m_printhost_browse_btn->Bind(wxEVT_BUTTON, [=](wxCommandEvent& e) { BonjourDialog dialog(this, Preset::printer_technology(*m_config)); if (dialog.show_and_lookup()) { m_optgroup->set_value("print_host", dialog.get_selected(), true); m_optgroup->get_field("print_host")->field_changed(); } }); return sizer; }; auto print_host_test = [=](wxWindow* parent) { auto sizer = create_sizer_with_btn(parent, &m_printhost_test_btn, "printer_host_test", _L("Test")); m_printhost_test_btn->Bind(wxEVT_BUTTON, [this](wxCommandEvent& e) { std::unique_ptr host(PrintHost::get_print_host(m_config)); if (!host) { const wxString text = _L("Could not get a valid Printer Host reference"); show_error(this, text); return; } wxString msg; bool result; { // Show a wait cursor during the connection test, as it is blocking UI. wxBusyCursor wait; result = host->test(msg); } if (result) show_info(this, host->get_test_ok_msg(), _L("Success!")); else show_error(this, host->get_test_failed_msg(msg)); }); return sizer; }; auto print_host_printers = [this, create_sizer_with_btn](wxWindow* parent) { //add_scaled_button(parent, &m_printhost_port_browse_btn, "browse", _(L("Refresh Printers")), wxBU_LEFT | wxBU_EXACTFIT); auto sizer = create_sizer_with_btn(parent, &m_printhost_port_browse_btn, "monitor_signal_strong", _(L("Refresh Printers"))); ScalableButton* btn = m_printhost_port_browse_btn; btn->SetFont(Slic3r::GUI::wxGetApp().normal_font()); btn->Bind(wxEVT_BUTTON, [this](wxCommandEvent e) { update_printers(); }); return sizer; }; // Set a wider width for a better alignment Option option = m_optgroup->get_option("print_host"); option.opt.width = Field::def_width_wider(); Line host_line = m_optgroup->create_single_option_line(option); //do not support now host_line.append_widget(printhost_browse); host_line.append_widget(print_host_test); m_optgroup->append_line(host_line); option = m_optgroup->get_option("print_host_webui"); option.opt.width = Field::def_width_wider(); m_optgroup->append_single_option_line(option); m_optgroup->append_single_option_line("printhost_authorization_type"); option = m_optgroup->get_option("printhost_apikey"); option.opt.width = Field::def_width_wider(); m_optgroup->append_single_option_line(option); option = m_optgroup->get_option("printhost_port"); option.opt.width = Field::def_width_wider(); Line port_line = m_optgroup->create_single_option_line(option); port_line.append_widget(print_host_printers); m_optgroup->append_line(port_line); const auto ca_file_hint = _u8L("HTTPS CA file is optional. It is only needed if you use HTTPS with a self-signed certificate."); if (Http::ca_file_supported()) { option = m_optgroup->get_option("printhost_cafile"); option.opt.width = Field::def_width_wider(); Line cafile_line = m_optgroup->create_single_option_line(option); auto printhost_cafile_browse = [=](wxWindow* parent) { auto sizer = create_sizer_with_btn(parent, &m_printhost_cafile_browse_btn, "monitor_signal_strong", _L("Browse") + " " + dots); m_printhost_cafile_browse_btn->Bind(wxEVT_BUTTON, [this, m_optgroup](wxCommandEvent e) { static const auto filemasks = _L("Certificate files (*.crt, *.pem)|*.crt;*.pem|All files|*.*"); wxFileDialog openFileDialog(this, _L("Open CA certificate file"), "", "", filemasks, wxFD_OPEN | wxFD_FILE_MUST_EXIST); if (openFileDialog.ShowModal() != wxID_CANCEL) { m_optgroup->set_value("printhost_cafile", openFileDialog.GetPath(), true); m_optgroup->get_field("printhost_cafile")->field_changed(); } }); return sizer; }; cafile_line.append_widget(printhost_cafile_browse); m_optgroup->append_line(cafile_line); Line cafile_hint{ "", "" }; cafile_hint.full_width = 1; cafile_hint.widget = [ca_file_hint](wxWindow* parent) { auto txt = new wxStaticText(parent, wxID_ANY, ca_file_hint); auto sizer = new wxBoxSizer(wxHORIZONTAL); sizer->Add(txt); return sizer; }; m_optgroup->append_line(cafile_hint); } else { Line line{ "", "" }; line.full_width = 1; line.widget = [ca_file_hint](wxWindow* parent) { std::string info = _u8L("HTTPS CA File") + ":\n\t" + (boost::format(_u8L("On this system, %s uses HTTPS certificates from the system Certificate Store or Keychain.")) % SLIC3R_APP_NAME).str() + "\n\t" + _u8L("To use a custom CA file, please import your CA file into Certificate Store / Keychain."); //auto txt = new wxStaticText(parent, wxID_ANY, from_u8((boost::format("%1%\n\n\t%2%") % info % ca_file_hint).str())); auto txt = new wxStaticText(parent, wxID_ANY, from_u8((boost::format("%1%\n\t%2%") % info % ca_file_hint).str())); txt->SetFont(wxGetApp().normal_font()); auto sizer = new wxBoxSizer(wxHORIZONTAL); sizer->Add(txt, 1, wxEXPAND|wxALIGN_LEFT); return sizer; }; m_optgroup->append_line(line); } for (const std::string& opt_key : std::vector{ "printhost_user", "printhost_password" }) { option = m_optgroup->get_option(opt_key); option.opt.width = Field::def_width_wider(); m_optgroup->append_single_option_line(option); } #ifdef WIN32 option = m_optgroup->get_option("printhost_ssl_ignore_revoke"); option.opt.width = Field::def_width_wider(); m_optgroup->append_single_option_line(option); #endif m_optgroup->activate(); Field* printhost_field = m_optgroup->get_field("print_host"); if (printhost_field) { wxTextCtrl* temp = dynamic_cast(printhost_field->getWindow()); if (temp) temp->Bind(wxEVT_TEXT, ([printhost_field, temp](wxEvent& e) { #ifndef __WXGTK__ e.Skip(); temp->GetToolTip()->Enable(true); #endif // __WXGTK__ // Remove all leading and trailing spaces from the input std::string trimed_str, str = trimed_str = temp->GetValue().ToStdString(); boost::trim(trimed_str); if (trimed_str != str) temp->SetValue(trimed_str); TextCtrl* field = dynamic_cast(printhost_field); if (field) field->propagate_value(); }), temp->GetId()); } // Always fill in the "printhost_port" combo box from the config and select it. { Choice* choice = dynamic_cast(m_optgroup->get_field("printhost_port")); choice->set_values(std::vector({ m_config->opt_string("printhost_port") })); choice->set_selection(); } update(); } void PhysicalPrinterDialog::update_printhost_buttons() { std::unique_ptr host(PrintHost::get_print_host(m_config)); if (host) { m_printhost_test_btn->Enable(!m_config->opt_string("print_host").empty() && host->can_test()); m_printhost_browse_btn->Enable(host->has_auto_discovery()); } } void PhysicalPrinterDialog::update_preset_input() { m_preset_name = into_u8(m_input_ctrl->GetValue()); m_valid_type = Valid; wxString info_line; const char *unusable_symbols = "<>[]:/\\|?*\""; const std::string unusable_suffix = PresetCollection::get_suffix_modified(); //"(modified)"; for (size_t i = 0; i < std::strlen(unusable_symbols); i++) { if (m_preset_name.find_first_of(unusable_symbols[i]) != std::string::npos) { info_line = _L("Name is invalid;") + "\n" + _L("illegal characters:") + " " + unusable_symbols; m_valid_type = NoValid; break; } } if (m_valid_type == Valid && m_preset_name.find(unusable_suffix) != std::string::npos) { info_line = _L("Name is invalid;") + "\n" + _L("illegal suffix:") + "\n\t" + from_u8(PresetCollection::get_suffix_modified()); m_valid_type = NoValid; } if (m_valid_type == Valid && (m_preset_name == "Default Setting" || m_preset_name == "Default Filament" || m_preset_name == "Default Printer" || m_preset_name == "Default Config")) { info_line = _L("Name is unavailable."); m_valid_type = NoValid; } const Preset *existing = m_presets->find_preset(m_preset_name, false); if (m_valid_type == Valid && existing && (existing->is_default || existing->is_system)) { info_line = _L("Overwrite a system profile is not allowed"); m_valid_type = NoValid; } if (m_valid_type == Valid && existing && m_preset_name != m_presets->get_selected_preset_name()) { if (existing->is_compatible) info_line = from_u8((boost::format(_u8L("Preset \"%1%\" already exists.")) % m_preset_name).str()); else info_line = from_u8((boost::format(_u8L("Preset \"%1%\" already exists and is incompatible with current printer.")) % m_preset_name).str()); info_line += "\n" + _L("Please note that saving action will replace this preset"); m_valid_type = Warning; } if (m_valid_type == Valid && m_preset_name.empty()) { info_line = _L("The name is not allowed to be empty."); m_valid_type = NoValid; } if (m_valid_type == Valid && m_preset_name.find_first_of(' ') == 0) { info_line = _L("The name is not allowed to start with space character."); m_valid_type = NoValid; } if (m_valid_type == Valid && m_preset_name.find_last_of(' ') == m_preset_name.length() - 1) { info_line = _L("The name is not allowed to end with space character."); m_valid_type = NoValid; } if (m_valid_type == Valid && m_presets->get_preset_name_by_alias(m_preset_name) != m_preset_name) { info_line = _L("The name cannot be the same as a preset alias name."); m_valid_type = NoValid; } m_valid_label->SetLabel(info_line); m_input_area->Refresh(); m_valid_label->Show(!info_line.IsEmpty()); if (m_valid_type == NoValid) { if (m_button_ok) m_button_ok->Disable(); } else { if (m_button_ok) m_button_ok->Enable(); } } void PhysicalPrinterDialog::update(bool printer_change) { m_optgroup->reload_config(); const PrinterTechnology tech = Preset::printer_technology(*m_config); // Only offer the host type selection for FFF, for SLA it's always the SL1 printer (at the moment) bool supports_multiple_printers = false; if (tech == ptFFF) { update_host_type(printer_change); const auto opt = m_config->option>("host_type"); m_optgroup->show_field("host_type"); if (opt->value == htPrusaLink) { m_optgroup->show_field("printhost_authorization_type"); AuthorizationType auth_type = m_config->option>("printhost_authorization_type")->value; m_optgroup->show_field("printhost_apikey", auth_type == AuthorizationType::atKeyPassword); for (const char* opt_key : { "printhost_user", "printhost_password" }) m_optgroup->show_field(opt_key, auth_type == AuthorizationType::atUserPassword); } else { m_optgroup->hide_field("printhost_authorization_type"); m_optgroup->show_field("printhost_apikey", true); for (const std::string& opt_key : std::vector{ "printhost_user", "printhost_password" }) m_optgroup->hide_field(opt_key); supports_multiple_printers = opt && opt->value == htRepetier; } } else { m_optgroup->set_value("host_type", int(PrintHostType::htOctoPrint), false); m_optgroup->hide_field("host_type"); m_optgroup->show_field("printhost_authorization_type"); AuthorizationType auth_type = m_config->option>("printhost_authorization_type")->value; m_optgroup->show_field("printhost_apikey", auth_type == AuthorizationType::atKeyPassword); for (const char *opt_key : { "printhost_user", "printhost_password" }) m_optgroup->show_field(opt_key, auth_type == AuthorizationType::atUserPassword); } m_optgroup->show_field("printhost_port", supports_multiple_printers); m_printhost_port_browse_btn->Show(supports_multiple_printers); update_preset_input(); update_printhost_buttons(); this->SetSize(this->GetBestSize()); this->Layout(); } void PhysicalPrinterDialog::update_host_type(bool printer_change) { if (m_config == nullptr) return; bool all_presets_are_from_mk3_family = false; Field* ht = m_optgroup->get_field("host_type"); wxArrayString types; // Append localized enum_labels assert(ht->m_opt.enum_labels.size() == ht->m_opt.enum_values.size()); for (size_t i = 0; i < ht->m_opt.enum_labels.size(); i++) { if (ht->m_opt.enum_values[i] == "prusalink" && !all_presets_are_from_mk3_family) continue; types.Add(_(ht->m_opt.enum_labels[i])); } Choice* choice = dynamic_cast(ht); choice->set_values(types); auto set_to_choice_and_config = [this, choice](PrintHostType type) { choice->set_value(static_cast(type)); m_config->set_key_value("host_type", new ConfigOptionEnum(type)); }; if ((printer_change && all_presets_are_from_mk3_family) || all_presets_are_from_mk3_family) set_to_choice_and_config(htPrusaLink); else if ((printer_change && !all_presets_are_from_mk3_family) || (!all_presets_are_from_mk3_family && m_config->option>("host_type")->value == htPrusaLink)) set_to_choice_and_config(htOctoPrint); else choice->set_value(m_config->option("host_type")->getInt()); } void PhysicalPrinterDialog::update_printers() { wxBusyCursor wait; std::unique_ptr host(PrintHost::get_print_host(m_config)); wxArrayString printers; Field *rs = m_optgroup->get_field("printhost_port"); try { if (! host->get_printers(printers)) printers.clear(); } catch (const HostNetworkError &err) { printers.clear(); show_error(this, _L("Connection to printers connected via the print host failed.") + "\n\n" + from_u8(err.what())); } Choice *choice = dynamic_cast(rs); choice->set_values(printers); printers.empty() ? rs->disable() : rs->enable(); } void PhysicalPrinterDialog::on_dpi_changed(const wxRect& suggested_rect) { const int& em = em_unit(); m_printhost_browse_btn->msw_rescale(); m_printhost_test_btn->msw_rescale(); if (m_printhost_cafile_browse_btn) m_printhost_cafile_browse_btn->msw_rescale(); m_optgroup->msw_rescale(); msw_buttons_rescale(this, em, { wxID_OK, wxID_CANCEL }); const wxSize& size = wxSize(45 * em, 35 * em); SetMinSize(size); Fit(); Refresh(); } void PhysicalPrinterDialog::OnOK(wxMouseEvent& event) { wxGetApp().get_tab(Preset::TYPE_PRINTER)->save_preset("", false, false, true, m_preset_name ); this->EndModal(wxID_YES); } }} // namespace Slic3r::GUI