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

github.com/prusa3d/PrusaSlicer.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'src/slic3r/GUI/GUI.cpp')
-rw-r--r--src/slic3r/GUI/GUI.cpp1402
1 files changed, 1402 insertions, 0 deletions
diff --git a/src/slic3r/GUI/GUI.cpp b/src/slic3r/GUI/GUI.cpp
new file mode 100644
index 000000000..24d459921
--- /dev/null
+++ b/src/slic3r/GUI/GUI.cpp
@@ -0,0 +1,1402 @@
+#include "GUI.hpp"
+#include "WipeTowerDialog.hpp"
+
+#include <assert.h>
+#include <cmath>
+
+#include <boost/lexical_cast.hpp>
+#include <boost/algorithm/string.hpp>
+#include <boost/format.hpp>
+#include <boost/lexical_cast.hpp>
+
+#if __APPLE__
+#import <IOKit/pwr_mgt/IOPMLib.h>
+#elif _WIN32
+#include <Windows.h>
+// Undefine min/max macros incompatible with the standard library
+// For example, std::numeric_limits<std::streamsize>::max()
+// produces some weird errors
+#ifdef min
+#undef min
+#endif
+#ifdef max
+#undef max
+#endif
+#include "boost/nowide/convert.hpp"
+#endif
+
+#include <wx/app.h>
+#include <wx/button.h>
+#include <wx/dir.h>
+#include <wx/filename.h>
+#include <wx/frame.h>
+#include <wx/menu.h>
+#include <wx/notebook.h>
+#include <wx/panel.h>
+#include <wx/sizer.h>
+#include <wx/combo.h>
+#include <wx/window.h>
+#include <wx/msgdlg.h>
+#include <wx/settings.h>
+#include <wx/display.h>
+#include <wx/collpane.h>
+#include <wx/wupdlock.h>
+
+#include "wxExtensions.hpp"
+
+#include "Tab.hpp"
+#include "TabIface.hpp"
+#include "GUI_Preview.hpp"
+#include "GUI_PreviewIface.hpp"
+#include "AboutDialog.hpp"
+#include "AppConfig.hpp"
+#include "ConfigSnapshotDialog.hpp"
+#include "ProgressStatusBar.hpp"
+#include "Utils.hpp"
+#include "MsgDialog.hpp"
+#include "ConfigWizard.hpp"
+#include "Preferences.hpp"
+#include "PresetBundle.hpp"
+#include "UpdateDialogs.hpp"
+#include "FirmwareDialog.hpp"
+#include "GUI_ObjectParts.hpp"
+
+#include "../Utils/PresetUpdater.hpp"
+#include "../Config/Snapshot.hpp"
+
+#include "3DScene.hpp"
+#include "libslic3r/I18N.hpp"
+#include "Model.hpp"
+#include "LambdaObjectDialog.hpp"
+
+#include "../../libslic3r/Print.hpp"
+
+namespace Slic3r { namespace GUI {
+
+#if __APPLE__
+IOPMAssertionID assertionID;
+#endif
+
+void disable_screensaver()
+{
+ #if __APPLE__
+ CFStringRef reasonForActivity = CFSTR("Slic3r");
+ IOReturn success = IOPMAssertionCreateWithName(kIOPMAssertionTypeNoDisplaySleep,
+ kIOPMAssertionLevelOn, reasonForActivity, &assertionID);
+ // ignore result: success == kIOReturnSuccess
+ #elif _WIN32
+ SetThreadExecutionState(ES_DISPLAY_REQUIRED | ES_CONTINUOUS);
+ #endif
+}
+
+void enable_screensaver()
+{
+ #if __APPLE__
+ IOReturn success = IOPMAssertionRelease(assertionID);
+ #elif _WIN32
+ SetThreadExecutionState(ES_CONTINUOUS);
+ #endif
+}
+
+bool debugged()
+{
+ #ifdef _WIN32
+ return IsDebuggerPresent();
+ #else
+ return false;
+ #endif /* _WIN32 */
+}
+
+void break_to_debugger()
+{
+ #ifdef _WIN32
+ if (IsDebuggerPresent())
+ DebugBreak();
+ #endif /* _WIN32 */
+}
+
+// Passing the wxWidgets GUI classes instantiated by the Perl part to C++.
+wxApp *g_wxApp = nullptr;
+wxFrame *g_wxMainFrame = nullptr;
+ProgressStatusBar *g_progress_status_bar = nullptr;
+wxNotebook *g_wxTabPanel = nullptr;
+wxPanel *g_wxPlater = nullptr;
+AppConfig *g_AppConfig = nullptr;
+PresetBundle *g_PresetBundle= nullptr;
+PresetUpdater *g_PresetUpdater = nullptr;
+wxColour g_color_label_modified;
+wxColour g_color_label_sys;
+wxColour g_color_label_default;
+
+std::vector<Tab *> g_tabs_list;
+
+wxLocale* g_wxLocale;
+
+wxFont g_small_font;
+wxFont g_bold_font;
+
+std::vector <std::shared_ptr<ConfigOptionsGroup>> m_optgroups;
+double m_brim_width = 0.0;
+size_t m_label_width = 100;
+wxButton* g_wiping_dialog_button = nullptr;
+
+//showed/hided controls according to the view mode
+wxWindow *g_right_panel = nullptr;
+wxBoxSizer *g_frequently_changed_parameters_sizer = nullptr;
+wxBoxSizer *g_info_sizer = nullptr;
+wxBoxSizer *g_object_list_sizer = nullptr;
+std::vector<wxButton*> g_buttons;
+wxStaticBitmap *g_manifold_warning_icon = nullptr;
+bool g_show_print_info = false;
+bool g_show_manifold_warning_icon = false;
+
+PreviewIface* g_preview = nullptr;
+
+static void init_label_colours()
+{
+ auto luma = get_colour_approx_luma(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW));
+ if (luma >= 128) {
+ g_color_label_modified = wxColour(252, 77, 1);
+ g_color_label_sys = wxColour(26, 132, 57);
+ } else {
+ g_color_label_modified = wxColour(253, 111, 40);
+ g_color_label_sys = wxColour(115, 220, 103);
+ }
+ g_color_label_default = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT);
+}
+
+void update_label_colours_from_appconfig()
+{
+ if (g_AppConfig->has("label_clr_sys")){
+ auto str = g_AppConfig->get("label_clr_sys");
+ if (str != "")
+ g_color_label_sys = wxColour(str);
+ }
+
+ if (g_AppConfig->has("label_clr_modified")){
+ auto str = g_AppConfig->get("label_clr_modified");
+ if (str != "")
+ g_color_label_modified = wxColour(str);
+ }
+}
+
+static void init_fonts()
+{
+ g_small_font = wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
+ g_bold_font = wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT).Bold();
+#ifdef __WXMAC__
+ g_small_font.SetPointSize(11);
+ g_bold_font.SetPointSize(13);
+#endif /*__WXMAC__*/
+}
+
+static std::string libslic3r_translate_callback(const char *s) { return wxGetTranslation(wxString(s, wxConvUTF8)).utf8_str().data(); }
+
+void set_wxapp(wxApp *app)
+{
+ g_wxApp = app;
+ // Let the libslic3r know the callback, which will translate messages on demand.
+ Slic3r::I18N::set_translate_callback(libslic3r_translate_callback);
+ init_label_colours();
+ init_fonts();
+}
+
+void set_main_frame(wxFrame *main_frame)
+{
+ g_wxMainFrame = main_frame;
+}
+
+wxFrame* get_main_frame() { return g_wxMainFrame; }
+
+void set_progress_status_bar(ProgressStatusBar *prsb)
+{
+ g_progress_status_bar = prsb;
+}
+
+ProgressStatusBar* get_progress_status_bar() { return g_progress_status_bar; }
+
+void set_tab_panel(wxNotebook *tab_panel)
+{
+ g_wxTabPanel = tab_panel;
+}
+
+void set_plater(wxPanel *plater)
+{
+ g_wxPlater = plater;
+}
+
+void set_app_config(AppConfig *app_config)
+{
+ g_AppConfig = app_config;
+}
+
+void set_preset_bundle(PresetBundle *preset_bundle)
+{
+ g_PresetBundle = preset_bundle;
+}
+
+void set_preset_updater(PresetUpdater *updater)
+{
+ g_PresetUpdater = updater;
+}
+
+enum ActionButtons
+{
+ abExportGCode,
+ abReslice,
+ abPrint,
+ abSendGCode,
+};
+
+void set_objects_from_perl( wxWindow* parent,
+ wxBoxSizer *frequently_changed_parameters_sizer,
+ wxBoxSizer *info_sizer,
+ wxButton *btn_export_gcode,
+ wxButton *btn_reslice,
+ wxButton *btn_print,
+ wxButton *btn_send_gcode,
+ wxStaticBitmap *manifold_warning_icon)
+{
+ g_right_panel = parent->GetParent();
+ g_frequently_changed_parameters_sizer = frequently_changed_parameters_sizer;
+ g_info_sizer = info_sizer;
+
+ g_buttons.push_back(btn_export_gcode);
+ g_buttons.push_back(btn_reslice);
+ g_buttons.push_back(btn_print);
+ g_buttons.push_back(btn_send_gcode);
+
+ // Update font style for buttons
+ for (auto btn : g_buttons)
+ btn->SetFont(bold_font());
+
+ g_manifold_warning_icon = manifold_warning_icon;
+}
+
+void set_show_print_info(bool show)
+{
+ g_show_print_info = show;
+}
+
+void set_show_manifold_warning_icon(bool show)
+{
+ g_show_manifold_warning_icon = show;
+ if (!g_manifold_warning_icon)
+ return;
+
+ // update manifold_warning_icon showing
+ if (show && !g_info_sizer->IsShown(static_cast<size_t>(0)))
+ g_show_manifold_warning_icon = false;
+
+ g_manifold_warning_icon->Show(g_show_manifold_warning_icon);
+ g_manifold_warning_icon->GetParent()->Layout();
+}
+
+void set_objects_list_sizer(wxBoxSizer *objects_list_sizer){
+ g_object_list_sizer = objects_list_sizer;
+}
+
+std::vector<Tab *>& get_tabs_list()
+{
+ return g_tabs_list;
+}
+
+bool checked_tab(Tab* tab)
+{
+ bool ret = true;
+ if (find(g_tabs_list.begin(), g_tabs_list.end(), tab) == g_tabs_list.end())
+ ret = false;
+ return ret;
+}
+
+void delete_tab_from_list(Tab* tab)
+{
+ std::vector<Tab *>::iterator itr = find(g_tabs_list.begin(), g_tabs_list.end(), tab);
+ if (itr != g_tabs_list.end())
+ g_tabs_list.erase(itr);
+}
+
+bool select_language(wxArrayString & names,
+ wxArrayLong & identifiers)
+{
+ wxCHECK_MSG(names.Count() == identifiers.Count(), false,
+ _(L("Array of language names and identifiers should have the same size.")));
+ int init_selection = 0;
+ long current_language = g_wxLocale ? g_wxLocale->GetLanguage() : wxLANGUAGE_UNKNOWN;
+ for (auto lang : identifiers){
+ if (lang == current_language)
+ break;
+ else
+ ++init_selection;
+ }
+ if (init_selection == identifiers.size())
+ init_selection = 0;
+ long index = wxGetSingleChoiceIndex(_(L("Select the language")), _(L("Language")),
+ names, init_selection);
+ if (index != -1)
+ {
+ g_wxLocale = new wxLocale;
+ g_wxLocale->Init(identifiers[index]);
+ g_wxLocale->AddCatalogLookupPathPrefix(wxPathOnly(localization_dir()));
+ g_wxLocale->AddCatalog(g_wxApp->GetAppName());
+ wxSetlocale(LC_NUMERIC, "C");
+ Preset::update_suffix_modified();
+ return true;
+ }
+ return false;
+}
+
+bool load_language()
+{
+ wxString language = wxEmptyString;
+ if (g_AppConfig->has("translation_language"))
+ language = g_AppConfig->get("translation_language");
+
+ if (language.IsEmpty())
+ return false;
+ wxArrayString names;
+ wxArrayLong identifiers;
+ get_installed_languages(names, identifiers);
+ for (size_t i = 0; i < identifiers.Count(); i++)
+ {
+ if (wxLocale::GetLanguageCanonicalName(identifiers[i]) == language)
+ {
+ g_wxLocale = new wxLocale;
+ g_wxLocale->Init(identifiers[i]);
+ g_wxLocale->AddCatalogLookupPathPrefix(wxPathOnly(localization_dir()));
+ g_wxLocale->AddCatalog(g_wxApp->GetAppName());
+ wxSetlocale(LC_NUMERIC, "C");
+ Preset::update_suffix_modified();
+ return true;
+ }
+ }
+ return false;
+}
+
+void save_language()
+{
+ wxString language = wxEmptyString;
+ if (g_wxLocale)
+ language = g_wxLocale->GetCanonicalName();
+
+ g_AppConfig->set("translation_language", language.ToStdString());
+ g_AppConfig->save();
+}
+
+void get_installed_languages(wxArrayString & names,
+ wxArrayLong & identifiers)
+{
+ names.Clear();
+ identifiers.Clear();
+
+ wxDir dir(wxPathOnly(localization_dir()));
+ wxString filename;
+ const wxLanguageInfo * langinfo;
+ wxString name = wxLocale::GetLanguageName(wxLANGUAGE_DEFAULT);
+ if (!name.IsEmpty())
+ {
+ names.Add(_(L("Default")));
+ identifiers.Add(wxLANGUAGE_DEFAULT);
+ }
+ for (bool cont = dir.GetFirst(&filename, wxEmptyString, wxDIR_DIRS);
+ cont; cont = dir.GetNext(&filename))
+ {
+ langinfo = wxLocale::FindLanguageInfo(filename);
+ if (langinfo != NULL)
+ {
+ auto full_file_name = dir.GetName() + wxFileName::GetPathSeparator() +
+ filename + wxFileName::GetPathSeparator() +
+ g_wxApp->GetAppName() + wxT(".mo");
+ if (wxFileExists(full_file_name))
+ {
+ names.Add(langinfo->Description);
+ identifiers.Add(langinfo->Language);
+ }
+ }
+ }
+}
+
+enum ConfigMenuIDs {
+ ConfigMenuWizard,
+ ConfigMenuSnapshots,
+ ConfigMenuTakeSnapshot,
+ ConfigMenuUpdate,
+ ConfigMenuPreferences,
+ ConfigMenuModeSimple,
+ ConfigMenuModeExpert,
+ ConfigMenuLanguage,
+ ConfigMenuFlashFirmware,
+ ConfigMenuCnt,
+};
+
+ConfigMenuIDs get_view_mode()
+{
+ if (!g_AppConfig->has("view_mode"))
+ return ConfigMenuModeSimple;
+
+ const auto mode = g_AppConfig->get("view_mode");
+ return mode == "expert" ? ConfigMenuModeExpert : ConfigMenuModeSimple;
+}
+
+static wxString dots("…", wxConvUTF8);
+
+void add_config_menu(wxMenuBar *menu, int event_preferences_changed, int event_language_change)
+{
+ auto local_menu = new wxMenu();
+ wxWindowID config_id_base = wxWindow::NewControlId((int)ConfigMenuCnt);
+
+ const auto config_wizard_name = _(ConfigWizard::name().wx_str());
+ const auto config_wizard_tooltip = wxString::Format(_(L("Run %s")), config_wizard_name);
+ // Cmd+, is standard on OS X - what about other operating systems?
+ local_menu->Append(config_id_base + ConfigMenuWizard, config_wizard_name + dots, config_wizard_tooltip);
+ local_menu->Append(config_id_base + ConfigMenuSnapshots, _(L("Configuration Snapshots"))+dots, _(L("Inspect / activate configuration snapshots")));
+ local_menu->Append(config_id_base + ConfigMenuTakeSnapshot, _(L("Take Configuration Snapshot")), _(L("Capture a configuration snapshot")));
+// local_menu->Append(config_id_base + ConfigMenuUpdate, _(L("Check for updates")), _(L("Check for configuration updates")));
+ local_menu->AppendSeparator();
+ local_menu->Append(config_id_base + ConfigMenuPreferences, _(L("Preferences"))+dots+"\tCtrl+,", _(L("Application preferences")));
+ local_menu->AppendSeparator();
+ auto mode_menu = new wxMenu();
+ mode_menu->AppendRadioItem(config_id_base + ConfigMenuModeSimple, _(L("&Simple")), _(L("Simple View Mode")));
+ mode_menu->AppendRadioItem(config_id_base + ConfigMenuModeExpert, _(L("&Expert")), _(L("Expert View Mode")));
+ mode_menu->Check(config_id_base + get_view_mode(), true);
+ local_menu->AppendSubMenu(mode_menu, _(L("&Mode")), _(L("Slic3r View Mode")));
+ local_menu->AppendSeparator();
+ local_menu->Append(config_id_base + ConfigMenuLanguage, _(L("Change Application Language")));
+ local_menu->AppendSeparator();
+ local_menu->Append(config_id_base + ConfigMenuFlashFirmware, _(L("Flash printer firmware")), _(L("Upload a firmware image into an Arduino based printer")));
+ // TODO: for when we're able to flash dictionaries
+ // local_menu->Append(config_id_base + FirmwareMenuDict, _(L("Flash language file")), _(L("Upload a language dictionary file into a Prusa printer")));
+
+ local_menu->Bind(wxEVT_MENU, [config_id_base, event_language_change, event_preferences_changed](wxEvent &event){
+ switch (event.GetId() - config_id_base) {
+ case ConfigMenuWizard:
+ config_wizard(ConfigWizard::RR_USER);
+ break;
+ case ConfigMenuTakeSnapshot:
+ // Take a configuration snapshot.
+ if (check_unsaved_changes()) {
+ wxTextEntryDialog dlg(nullptr, _(L("Taking configuration snapshot")), _(L("Snapshot name")));
+ if (dlg.ShowModal() == wxID_OK)
+ g_AppConfig->set("on_snapshot",
+ Slic3r::GUI::Config::SnapshotDB::singleton().take_snapshot(
+ *g_AppConfig, Slic3r::GUI::Config::Snapshot::SNAPSHOT_USER, dlg.GetValue().ToUTF8().data()).id);
+ }
+ break;
+ case ConfigMenuSnapshots:
+ if (check_unsaved_changes()) {
+ std::string on_snapshot;
+ if (Config::SnapshotDB::singleton().is_on_snapshot(*g_AppConfig))
+ on_snapshot = g_AppConfig->get("on_snapshot");
+ ConfigSnapshotDialog dlg(Slic3r::GUI::Config::SnapshotDB::singleton(), on_snapshot);
+ dlg.ShowModal();
+ if (! dlg.snapshot_to_activate().empty()) {
+ if (! Config::SnapshotDB::singleton().is_on_snapshot(*g_AppConfig))
+ Config::SnapshotDB::singleton().take_snapshot(*g_AppConfig, Config::Snapshot::SNAPSHOT_BEFORE_ROLLBACK);
+ g_AppConfig->set("on_snapshot",
+ Config::SnapshotDB::singleton().restore_snapshot(dlg.snapshot_to_activate(), *g_AppConfig).id);
+ g_PresetBundle->load_presets(*g_AppConfig);
+ // Load the currently selected preset into the GUI, update the preset selection box.
+ load_current_presets();
+ }
+ }
+ break;
+ case ConfigMenuPreferences:
+ {
+ PreferencesDialog dlg(g_wxMainFrame, event_preferences_changed);
+ dlg.ShowModal();
+ break;
+ }
+ case ConfigMenuLanguage:
+ {
+ wxArrayString names;
+ wxArrayLong identifiers;
+ get_installed_languages(names, identifiers);
+ if (select_language(names, identifiers)) {
+ save_language();
+ show_info(g_wxTabPanel, _(L("Application will be restarted")), _(L("Attention!")));
+ if (event_language_change > 0) {
+ _3DScene::remove_all_canvases();// remove all canvas before recreate GUI
+ wxCommandEvent event(event_language_change);
+ g_wxApp->ProcessEvent(event);
+ }
+ }
+ break;
+ }
+ case ConfigMenuFlashFirmware:
+ FirmwareDialog::run(g_wxMainFrame);
+ break;
+ default:
+ break;
+ }
+ });
+ mode_menu->Bind(wxEVT_MENU, [config_id_base](wxEvent& event) {
+ std::string mode = event.GetId() - config_id_base == ConfigMenuModeExpert ?
+ "expert" : "simple";
+ g_AppConfig->set("view_mode", mode);
+ g_AppConfig->save();
+ update_mode();
+ });
+ menu->Append(local_menu, _(L("&Configuration")));
+}
+
+void add_menus(wxMenuBar *menu, int event_preferences_changed, int event_language_change)
+{
+ add_config_menu(menu, event_preferences_changed, event_language_change);
+}
+
+void open_model(wxWindow *parent, wxArrayString& input_files){
+ t_file_wild_card vec_FILE_WILDCARDS = get_file_wild_card();
+ std::vector<std::string> file_types = { "known", "stl", "obj", "amf", "3mf", "prusa" };
+ wxString MODEL_WILDCARD;
+ for (auto file_type : file_types)
+ MODEL_WILDCARD += vec_FILE_WILDCARDS.at(file_type) + "|";
+
+ auto dlg_title = _(L("Choose one or more files (STL/OBJ/AMF/3MF/PRUSA):"));
+ auto dialog = new wxFileDialog(parent /*? parent : GetTopWindow(g_wxMainFrame)*/, dlg_title,
+ g_AppConfig->get_last_dir(), "",
+ MODEL_WILDCARD, wxFD_OPEN | wxFD_MULTIPLE | wxFD_FILE_MUST_EXIST);
+ if (dialog->ShowModal() != wxID_OK) {
+ dialog->Destroy();
+ return ;
+ }
+
+ dialog->GetPaths(input_files);
+ dialog->Destroy();
+}
+
+// This is called when closing the application, when loading a config file or when starting the config wizard
+// to notify the user whether he is aware that some preset changes will be lost.
+bool check_unsaved_changes()
+{
+ std::string dirty;
+ for (Tab *tab : g_tabs_list)
+ if (tab->current_preset_is_dirty())
+ if (dirty.empty())
+ dirty = tab->name();
+ else
+ dirty += std::string(", ") + tab->name();
+ if (dirty.empty())
+ // No changes, the application may close or reload presets.
+ return true;
+ // Ask the user.
+ auto dialog = new wxMessageDialog(g_wxMainFrame,
+ _(L("You have unsaved changes ")) + dirty + _(L(". Discard changes and continue anyway?")),
+ _(L("Unsaved Presets")),
+ wxICON_QUESTION | wxYES_NO | wxNO_DEFAULT);
+ return dialog->ShowModal() == wxID_YES;
+}
+
+bool config_wizard_startup(bool app_config_exists)
+{
+ if (! app_config_exists || g_PresetBundle->printers.size() <= 1) {
+ config_wizard(ConfigWizard::RR_DATA_EMPTY);
+ return true;
+ } else if (g_AppConfig->legacy_datadir()) {
+ // Looks like user has legacy pre-vendorbundle data directory,
+ // explain what this is and run the wizard
+
+ MsgDataLegacy dlg;
+ dlg.ShowModal();
+
+ config_wizard(ConfigWizard::RR_DATA_LEGACY);
+ return true;
+ }
+ return false;
+}
+
+void config_wizard(int reason)
+{
+ // Exit wizard if there are unsaved changes and the user cancels the action.
+ if (! check_unsaved_changes())
+ return;
+
+ try {
+ ConfigWizard wizard(nullptr, static_cast<ConfigWizard::RunReason>(reason));
+ wizard.run(g_PresetBundle, g_PresetUpdater);
+ }
+ catch (const std::exception &e) {
+ show_error(nullptr, e.what());
+ }
+
+ // Load the currently selected preset into the GUI, update the preset selection box.
+ load_current_presets();
+}
+
+void open_preferences_dialog(int event_preferences)
+{
+ auto dlg = new PreferencesDialog(g_wxMainFrame, event_preferences);
+ dlg->ShowModal();
+}
+
+void create_preset_tabs(int event_value_change, int event_presets_changed)
+{
+ update_label_colours_from_appconfig();
+ add_created_tab(new TabPrint (g_wxTabPanel), event_value_change, event_presets_changed);
+ add_created_tab(new TabFilament (g_wxTabPanel), event_value_change, event_presets_changed);
+ add_created_tab(new TabSLAMaterial (g_wxTabPanel), event_value_change, event_presets_changed);
+ add_created_tab(new TabPrinter (g_wxTabPanel), event_value_change, event_presets_changed);
+}
+
+std::vector<PresetTab> preset_tabs = {
+ { "print", nullptr, ptFFF },
+ { "filament", nullptr, ptFFF },
+ { "sla_material", nullptr, ptSLA }
+};
+const std::vector<PresetTab>& get_preset_tabs() {
+ return preset_tabs;
+}
+
+Tab* get_tab(const std::string& name)
+{
+ std::vector<PresetTab>::iterator it = std::find_if(preset_tabs.begin(), preset_tabs.end(),
+ [name](PresetTab& tab){ return name == tab.name; });
+ return it != preset_tabs.end() ? it->panel : nullptr;
+}
+
+TabIface* get_preset_tab_iface(char *name)
+{
+ Tab* tab = get_tab(name);
+ if (tab) return new TabIface(tab);
+
+ for (size_t i = 0; i < g_wxTabPanel->GetPageCount(); ++ i) {
+ Tab *tab = dynamic_cast<Tab*>(g_wxTabPanel->GetPage(i));
+ if (! tab)
+ continue;
+ if (tab->name() == name) {
+ return new TabIface(tab);
+ }
+ }
+ return new TabIface(nullptr);
+}
+
+PreviewIface* create_preview_iface(wxNotebook* parent, DynamicPrintConfig* config, Print* print, GCodePreviewData* gcode_preview_data)
+{
+ if (g_preview == nullptr)
+ {
+ Preview* panel = new Preview(parent, config, print, gcode_preview_data);
+ g_preview = new PreviewIface(panel);
+ }
+
+ return g_preview;
+}
+
+// opt_index = 0, by the reason of zero-index in ConfigOptionVector by default (in case only one element)
+void change_opt_value(DynamicPrintConfig& config, const t_config_option_key& opt_key, const boost::any& value, int opt_index /*= 0*/)
+{
+ try{
+ switch (config.def()->get(opt_key)->type){
+ case coFloatOrPercent:{
+ std::string str = boost::any_cast<std::string>(value);
+ bool percent = false;
+ if (str.back() == '%'){
+ str.pop_back();
+ percent = true;
+ }
+ double val = stod(str);
+ config.set_key_value(opt_key, new ConfigOptionFloatOrPercent(val, percent));
+ break;}
+ case coPercent:
+ config.set_key_value(opt_key, new ConfigOptionPercent(boost::any_cast<double>(value)));
+ break;
+ case coFloat:{
+ double& val = config.opt_float(opt_key);
+ val = boost::any_cast<double>(value);
+ break;
+ }
+ case coPercents:{
+ ConfigOptionPercents* vec_new = new ConfigOptionPercents{ boost::any_cast<double>(value) };
+ config.option<ConfigOptionPercents>(opt_key)->set_at(vec_new, opt_index, opt_index);
+ break;
+ }
+ case coFloats:{
+ ConfigOptionFloats* vec_new = new ConfigOptionFloats{ boost::any_cast<double>(value) };
+ config.option<ConfigOptionFloats>(opt_key)->set_at(vec_new, opt_index, opt_index);
+ break;
+ }
+ case coString:
+ config.set_key_value(opt_key, new ConfigOptionString(boost::any_cast<std::string>(value)));
+ break;
+ case coStrings:{
+ if (opt_key.compare("compatible_printers") == 0) {
+ config.option<ConfigOptionStrings>(opt_key)->values =
+ boost::any_cast<std::vector<std::string>>(value);
+ }
+ else if (config.def()->get(opt_key)->gui_flags.compare("serialized") == 0){
+ std::string str = boost::any_cast<std::string>(value);
+ if (str.back() == ';') str.pop_back();
+ // Split a string to multiple strings by a semi - colon.This is the old way of storing multi - string values.
+ // Currently used for the post_process config value only.
+ std::vector<std::string> values;
+ boost::split(values, str, boost::is_any_of(";"));
+ if (values.size() == 1 && values[0] == "")
+ break;
+ config.option<ConfigOptionStrings>(opt_key)->values = values;
+ }
+ else{
+ ConfigOptionStrings* vec_new = new ConfigOptionStrings{ boost::any_cast<std::string>(value) };
+ config.option<ConfigOptionStrings>(opt_key)->set_at(vec_new, opt_index, 0);
+ }
+ }
+ break;
+ case coBool:
+ config.set_key_value(opt_key, new ConfigOptionBool(boost::any_cast<bool>(value)));
+ break;
+ case coBools:{
+ ConfigOptionBools* vec_new = new ConfigOptionBools{ (bool)boost::any_cast<unsigned char>(value) };
+ config.option<ConfigOptionBools>(opt_key)->set_at(vec_new, opt_index, 0);
+ break;}
+ case coInt:
+ config.set_key_value(opt_key, new ConfigOptionInt(boost::any_cast<int>(value)));
+ break;
+ case coInts:{
+ ConfigOptionInts* vec_new = new ConfigOptionInts{ boost::any_cast<int>(value) };
+ config.option<ConfigOptionInts>(opt_key)->set_at(vec_new, opt_index, 0);
+ }
+ break;
+ case coEnum:{
+ if (opt_key.compare("external_fill_pattern") == 0 ||
+ opt_key.compare("fill_pattern") == 0)
+ config.set_key_value(opt_key, new ConfigOptionEnum<InfillPattern>(boost::any_cast<InfillPattern>(value)));
+ else if (opt_key.compare("gcode_flavor") == 0)
+ config.set_key_value(opt_key, new ConfigOptionEnum<GCodeFlavor>(boost::any_cast<GCodeFlavor>(value)));
+ else if (opt_key.compare("support_material_pattern") == 0)
+ config.set_key_value(opt_key, new ConfigOptionEnum<SupportMaterialPattern>(boost::any_cast<SupportMaterialPattern>(value)));
+ else if (opt_key.compare("seam_position") == 0)
+ config.set_key_value(opt_key, new ConfigOptionEnum<SeamPosition>(boost::any_cast<SeamPosition>(value)));
+ else if (opt_key.compare("host_type") == 0)
+ config.set_key_value(opt_key, new ConfigOptionEnum<PrintHostType>(boost::any_cast<PrintHostType>(value)));
+ }
+ break;
+ case coPoints:{
+ if (opt_key.compare("bed_shape") == 0){
+ config.option<ConfigOptionPoints>(opt_key)->values = boost::any_cast<std::vector<Vec2d>>(value);
+ break;
+ }
+ ConfigOptionPoints* vec_new = new ConfigOptionPoints{ boost::any_cast<Vec2d>(value) };
+ config.option<ConfigOptionPoints>(opt_key)->set_at(vec_new, opt_index, 0);
+ }
+ break;
+ case coNone:
+ break;
+ default:
+ break;
+ }
+ }
+ catch (const std::exception &e)
+ {
+ int i = 0;//no reason, just experiment
+ }
+}
+
+void add_created_tab(Tab* panel, int event_value_change, int event_presets_changed)
+{
+ panel->create_preset_tab(g_PresetBundle);
+
+ // Load the currently selected preset into the GUI, update the preset selection box.
+ panel->load_current_preset();
+
+ panel->set_event_value_change(wxEventType(event_value_change));
+ panel->set_event_presets_changed(wxEventType(event_presets_changed));
+
+ const wxString& tab_name = panel->GetName();
+ bool add_panel = true;
+
+ auto it = std::find_if( preset_tabs.begin(), preset_tabs.end(),
+ [tab_name](PresetTab& tab){return tab.name == tab_name; });
+ if (it != preset_tabs.end()) {
+ it->panel = panel;
+ add_panel = it->technology == g_PresetBundle->printers.get_edited_preset().printer_technology();
+ }
+
+ if (add_panel)
+ g_wxTabPanel->AddPage(panel, panel->title());
+}
+
+void load_current_presets()
+{
+ for (Tab *tab : g_tabs_list) {
+ tab->load_current_preset();
+ }
+}
+
+void show_error(wxWindow* parent, const wxString& message) {
+ ErrorDialog msg(parent, message);
+ msg.ShowModal();
+}
+
+void show_error_id(int id, const std::string& message) {
+ auto *parent = id != 0 ? wxWindow::FindWindowById(id) : nullptr;
+ show_error(parent, wxString::FromUTF8(message.data()));
+}
+
+void show_info(wxWindow* parent, const wxString& message, const wxString& title){
+ wxMessageDialog msg_wingow(parent, message, title.empty() ? _(L("Notice")) : title, wxOK | wxICON_INFORMATION);
+ msg_wingow.ShowModal();
+}
+
+void warning_catcher(wxWindow* parent, const wxString& message){
+ if (message == "GLUquadricObjPtr | " + _(L("Attempt to free unreferenced scalar")) )
+ return;
+ wxMessageDialog msg(parent, message, _(L("Warning")), wxOK | wxICON_WARNING);
+ msg.ShowModal();
+}
+
+// Assign a Lambda to the print object to emit a wxWidgets Command with the provided ID
+// to deliver a progress status message.
+void set_print_callback_event(Print *print, int id)
+{
+ print->set_status_callback([id](int percent, const std::string &message){
+ wxCommandEvent event(id);
+ event.SetInt(percent);
+ event.SetString(message);
+ wxQueueEvent(g_wxMainFrame, event.Clone());
+ });
+}
+
+wxApp* get_app(){
+ return g_wxApp;
+}
+
+PresetBundle* get_preset_bundle()
+{
+ return g_PresetBundle;
+}
+
+const wxColour& get_label_clr_modified() {
+ return g_color_label_modified;
+}
+
+const wxColour& get_label_clr_sys() {
+ return g_color_label_sys;
+}
+
+void set_label_clr_modified(const wxColour& clr) {
+ g_color_label_modified = clr;
+ auto clr_str = wxString::Format(wxT("#%02X%02X%02X"), clr.Red(), clr.Green(), clr.Blue());
+ std::string str = clr_str.ToStdString();
+ g_AppConfig->set("label_clr_modified", str);
+ g_AppConfig->save();
+}
+
+void set_label_clr_sys(const wxColour& clr) {
+ g_color_label_sys = clr;
+ auto clr_str = wxString::Format(wxT("#%02X%02X%02X"), clr.Red(), clr.Green(), clr.Blue());
+ std::string str = clr_str.ToStdString();
+ g_AppConfig->set("label_clr_sys", str);
+ g_AppConfig->save();
+}
+
+const wxFont& small_font(){
+ return g_small_font;
+}
+
+const wxFont& bold_font(){
+ return g_bold_font;
+}
+
+const wxColour& get_label_clr_default() {
+ return g_color_label_default;
+}
+
+unsigned get_colour_approx_luma(const wxColour &colour)
+{
+ double r = colour.Red();
+ double g = colour.Green();
+ double b = colour.Blue();
+
+ return std::round(std::sqrt(
+ r * r * .241 +
+ g * g * .691 +
+ b * b * .068
+ ));
+}
+
+wxWindow* get_right_panel(){
+ return g_right_panel;
+}
+
+wxNotebook * get_tab_panel() {
+ return g_wxTabPanel;
+}
+
+const size_t& label_width(){
+ return m_label_width;
+}
+
+void create_combochecklist(wxComboCtrl* comboCtrl, std::string text, std::string items, bool initial_value)
+{
+ if (comboCtrl == nullptr)
+ return;
+
+ wxCheckListBoxComboPopup* popup = new wxCheckListBoxComboPopup;
+ if (popup != nullptr)
+ {
+ // FIXME If the following line is removed, the combo box popup list will not react to mouse clicks.
+ // On the other side, with this line the combo box popup cannot be closed by clicking on the combo button on Windows 10.
+ comboCtrl->UseAltPopupWindow();
+
+ comboCtrl->EnablePopupAnimation(false);
+ comboCtrl->SetPopupControl(popup);
+ popup->SetStringValue(from_u8(text));
+ popup->Bind(wxEVT_CHECKLISTBOX, [popup](wxCommandEvent& evt) { popup->OnCheckListBox(evt); });
+ popup->Bind(wxEVT_LISTBOX, [popup](wxCommandEvent& evt) { popup->OnListBoxSelection(evt); });
+ popup->Bind(wxEVT_KEY_DOWN, [popup](wxKeyEvent& evt) { popup->OnKeyEvent(evt); });
+ popup->Bind(wxEVT_KEY_UP, [popup](wxKeyEvent& evt) { popup->OnKeyEvent(evt); });
+
+ std::vector<std::string> items_str;
+ boost::split(items_str, items, boost::is_any_of("|"), boost::token_compress_off);
+
+ for (const std::string& item : items_str)
+ {
+ popup->Append(from_u8(item));
+ }
+
+ for (unsigned int i = 0; i < popup->GetCount(); ++i)
+ {
+ popup->Check(i, initial_value);
+ }
+ }
+}
+
+int combochecklist_get_flags(wxComboCtrl* comboCtrl)
+{
+ int flags = 0;
+
+ wxCheckListBoxComboPopup* popup = wxDynamicCast(comboCtrl->GetPopupControl(), wxCheckListBoxComboPopup);
+ if (popup != nullptr)
+ {
+ for (unsigned int i = 0; i < popup->GetCount(); ++i)
+ {
+ if (popup->IsChecked(i))
+ flags |= 1 << i;
+ }
+ }
+
+ return flags;
+}
+
+AppConfig* get_app_config()
+{
+ return g_AppConfig;
+}
+
+wxString L_str(const std::string &str)
+{
+ //! Explicitly specify that the source string is already in UTF-8 encoding
+ return wxGetTranslation(wxString(str.c_str(), wxConvUTF8));
+}
+
+wxString from_u8(const std::string &str)
+{
+ return wxString::FromUTF8(str.c_str());
+}
+
+void set_model_events_from_perl(Model &model,
+ int event_object_selection_changed,
+ int event_object_settings_changed,
+ int event_remove_object,
+ int event_update_scene)
+{
+ set_event_object_selection_changed(event_object_selection_changed);
+ set_event_object_settings_changed(event_object_settings_changed);
+ set_event_remove_object(event_remove_object);
+ set_event_update_scene(event_update_scene);
+ set_objects_from_model(model);
+ init_mesh_icons();
+
+// wxWindowUpdateLocker noUpdates(parent);
+
+// add_objects_list(parent, sizer);
+
+// add_collapsible_panes(parent, sizer);
+}
+
+void add_frequently_changed_parameters(wxWindow* parent, wxBoxSizer* sizer, wxFlexGridSizer* preset_sizer)
+{
+ DynamicPrintConfig* config = &g_PresetBundle->prints.get_edited_preset().config;
+ std::shared_ptr<ConfigOptionsGroup> optgroup = std::make_shared<ConfigOptionsGroup>(parent, "", config);
+ const wxArrayInt& ar = preset_sizer->GetColWidths();
+ m_label_width = ar.IsEmpty() ? 100 : ar.front()-4;
+ optgroup->label_width = m_label_width;
+
+ //Frequently changed parameters
+ optgroup->m_on_change = [config](t_config_option_key opt_key, boost::any value){
+ TabPrint* tab_print = nullptr;
+ for (size_t i = 0; i < g_wxTabPanel->GetPageCount(); ++i) {
+ Tab *tab = dynamic_cast<Tab*>(g_wxTabPanel->GetPage(i));
+ if (!tab)
+ continue;
+ if (tab->name() == "print"){
+ tab_print = static_cast<TabPrint*>(tab);
+ break;
+ }
+ }
+ if (tab_print == nullptr)
+ return;
+
+ if (opt_key == "fill_density"){
+ value = m_optgroups[ogFrequentlyChangingParameters]->get_config_value(*config, opt_key);
+ tab_print->set_value(opt_key, value);
+ tab_print->update();
+ }
+ else{
+ DynamicPrintConfig new_conf = *config;
+ if (opt_key == "brim"){
+ double new_val;
+ double brim_width = config->opt_float("brim_width");
+ if (boost::any_cast<bool>(value) == true)
+ {
+ new_val = m_brim_width == 0.0 ? 10 :
+ m_brim_width < 0.0 ? m_brim_width * (-1) :
+ m_brim_width;
+ }
+ else{
+ m_brim_width = brim_width * (-1);
+ new_val = 0;
+ }
+ new_conf.set_key_value("brim_width", new ConfigOptionFloat(new_val));
+ }
+ else{ //(opt_key == "support")
+ const wxString& selection = boost::any_cast<wxString>(value);
+
+ auto support_material = selection == _("None") ? false : true;
+ new_conf.set_key_value("support_material", new ConfigOptionBool(support_material));
+
+ if (selection == _("Everywhere"))
+ new_conf.set_key_value("support_material_buildplate_only", new ConfigOptionBool(false));
+ else if (selection == _("Support on build plate only"))
+ new_conf.set_key_value("support_material_buildplate_only", new ConfigOptionBool(true));
+ }
+ tab_print->load_config(new_conf);
+ }
+
+ tab_print->update_dirty();
+ };
+
+ Option option = optgroup->get_option("fill_density");
+ option.opt.sidetext = "";
+ option.opt.full_width = true;
+ optgroup->append_single_option_line(option);
+
+ ConfigOptionDef def;
+
+ def.label = L("Support");
+ def.type = coStrings;
+ def.gui_type = "select_open";
+ def.tooltip = L("Select what kind of support do you need");
+ def.enum_labels.push_back(L("None"));
+ def.enum_labels.push_back(L("Support on build plate only"));
+ def.enum_labels.push_back(L("Everywhere"));
+ std::string selection = !config->opt_bool("support_material") ?
+ "None" :
+ config->opt_bool("support_material_buildplate_only") ?
+ "Support on build plate only" :
+ "Everywhere";
+ def.default_value = new ConfigOptionStrings { selection };
+ option = Option(def, "support");
+ option.opt.full_width = true;
+ optgroup->append_single_option_line(option);
+
+ m_brim_width = config->opt_float("brim_width");
+ def.label = L("Brim");
+ def.type = coBool;
+ def.tooltip = L("This flag enables the brim that will be printed around each object on the first layer.");
+ def.gui_type = "";
+ def.default_value = new ConfigOptionBool{ m_brim_width > 0.0 ? true : false };
+ option = Option(def, "brim");
+ optgroup->append_single_option_line(option);
+
+
+ Line line = { "", "" };
+ line.widget = [config](wxWindow* parent){
+ g_wiping_dialog_button = new wxButton(parent, wxID_ANY, _(L("Purging volumes")) + dots, wxDefaultPosition, wxDefaultSize, wxBU_EXACTFIT);
+ auto sizer = new wxBoxSizer(wxHORIZONTAL);
+ sizer->Add(g_wiping_dialog_button);
+ g_wiping_dialog_button->Bind(wxEVT_BUTTON, ([parent](wxCommandEvent& e)
+ {
+ auto &config = g_PresetBundle->project_config;
+ const std::vector<double> &init_matrix = (config.option<ConfigOptionFloats>("wiping_volumes_matrix"))->values;
+ const std::vector<double> &init_extruders = (config.option<ConfigOptionFloats>("wiping_volumes_extruders"))->values;
+
+ WipingDialog dlg(parent,cast<float>(init_matrix),cast<float>(init_extruders));
+
+ if (dlg.ShowModal() == wxID_OK) {
+ std::vector<float> matrix = dlg.get_matrix();
+ std::vector<float> extruders = dlg.get_extruders();
+ (config.option<ConfigOptionFloats>("wiping_volumes_matrix"))->values = std::vector<double>(matrix.begin(),matrix.end());
+ (config.option<ConfigOptionFloats>("wiping_volumes_extruders"))->values = std::vector<double>(extruders.begin(),extruders.end());
+ g_on_request_update_callback.call();
+ }
+ }));
+ return sizer;
+ };
+ optgroup->append_line(line);
+
+ sizer->Add(optgroup->sizer, 0, wxEXPAND | wxBOTTOM | wxLEFT, 2);
+
+ m_optgroups.push_back(optgroup);// ogFrequentlyChangingParameters
+
+ // Object List
+ add_objects_list(parent, sizer);
+
+ // Frequently Object Settings
+ add_object_settings(parent, sizer);
+}
+
+void show_frequently_changed_parameters(bool show)
+{
+ g_frequently_changed_parameters_sizer->Show(show);
+ if (!show) return;
+
+ for (size_t i = 0; i < g_wxTabPanel->GetPageCount(); ++i) {
+ Tab *tab = dynamic_cast<Tab*>(g_wxTabPanel->GetPage(i));
+ if (!tab)
+ continue;
+ tab->update_wiping_button_visibility();
+ break;
+ }
+}
+
+void show_buttons(bool show)
+{
+ g_buttons[abReslice]->Show(show);
+ for (size_t i = 0; i < g_wxTabPanel->GetPageCount(); ++i) {
+ TabPrinter *tab = dynamic_cast<TabPrinter*>(g_wxTabPanel->GetPage(i));
+ if (!tab)
+ continue;
+ if (g_PresetBundle->printers.get_selected_preset().printer_technology() == ptFFF) {
+ g_buttons[abPrint]->Show(show && !tab->m_config->opt_string("serial_port").empty());
+ g_buttons[abSendGCode]->Show(show && !tab->m_config->opt_string("print_host").empty());
+ }
+ break;
+ }
+}
+
+void show_info_sizer(const bool show)
+{
+ g_info_sizer->Show(static_cast<size_t>(0), show);
+ g_info_sizer->Show(1, show && g_show_print_info);
+ g_manifold_warning_icon->Show(show && g_show_manifold_warning_icon);
+}
+
+void show_object_name(bool show)
+{
+ wxGridSizer* grid_sizer = get_optgroup(ogFrequentlyObjectSettings)->get_grid_sizer();
+ grid_sizer->Show(static_cast<size_t>(0), show);
+ grid_sizer->Show(static_cast<size_t>(1), show);
+}
+
+void update_mode()
+{
+ wxWindowUpdateLocker noUpdates(g_right_panel->GetParent());
+
+ ConfigMenuIDs mode = get_view_mode();
+
+ g_object_list_sizer->Show(mode == ConfigMenuModeExpert);
+ show_info_sizer(mode == ConfigMenuModeExpert);
+ show_buttons(mode == ConfigMenuModeExpert);
+ show_object_name(mode == ConfigMenuModeSimple);
+ show_manipulation_sizer(mode == ConfigMenuModeSimple);
+
+ // TODO There is a not the best place of it!
+ // *** Update showing of the collpane_settings
+// show_collpane_settings(mode == ConfigMenuModeExpert);
+ // *************************
+ g_right_panel->Layout();
+ g_right_panel->GetParent()->Layout();
+}
+
+bool is_expert_mode(){
+ return get_view_mode() == ConfigMenuModeExpert;
+}
+
+ConfigOptionsGroup* get_optgroup(size_t i)
+{
+ return m_optgroups[i].get();
+}
+
+std::vector <std::shared_ptr<ConfigOptionsGroup>>& get_optgroups() {
+ return m_optgroups;
+}
+
+wxButton* get_wiping_dialog_button()
+{
+ return g_wiping_dialog_button;
+}
+
+wxWindow* export_option_creator(wxWindow* parent)
+{
+ wxPanel* panel = new wxPanel(parent, -1);
+ wxSizer* sizer = new wxBoxSizer(wxHORIZONTAL);
+ wxCheckBox* cbox = new wxCheckBox(panel, wxID_HIGHEST + 1, L("Export print config"));
+ cbox->SetValue(true);
+ sizer->AddSpacer(5);
+ sizer->Add(cbox, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
+ panel->SetSizer(sizer);
+ sizer->SetSizeHints(panel);
+ return panel;
+}
+
+void add_export_option(wxFileDialog* dlg, const std::string& format)
+{
+ if ((dlg != nullptr) && (format == "AMF") || (format == "3MF"))
+ {
+ if (dlg->SupportsExtraControl())
+ dlg->SetExtraControlCreator(export_option_creator);
+ }
+}
+
+int get_export_option(wxFileDialog* dlg)
+{
+ if (dlg != nullptr)
+ {
+ wxWindow* wnd = dlg->GetExtraControl();
+ if (wnd != nullptr)
+ {
+ wxPanel* panel = dynamic_cast<wxPanel*>(wnd);
+ if (panel != nullptr)
+ {
+ wxWindow* child = panel->FindWindow(wxID_HIGHEST + 1);
+ if (child != nullptr)
+ {
+ wxCheckBox* cbox = dynamic_cast<wxCheckBox*>(child);
+ if (cbox != nullptr)
+ return cbox->IsChecked() ? 1 : 0;
+ }
+ }
+ }
+ }
+
+ return 0;
+
+}
+
+bool get_current_screen_size(wxWindow *window, unsigned &width, unsigned &height)
+{
+ const auto idx = wxDisplay::GetFromWindow(window);
+ if (idx == wxNOT_FOUND) {
+ return false;
+ }
+
+ wxDisplay display(idx);
+ const auto disp_size = display.GetClientArea();
+ width = disp_size.GetWidth();
+ height = disp_size.GetHeight();
+
+ return true;
+}
+
+void save_window_size(wxTopLevelWindow *window, const std::string &name)
+{
+ const wxSize size = window->GetSize();
+ const wxPoint pos = window->GetPosition();
+ const auto maximized = window->IsMaximized() ? "1" : "0";
+
+ g_AppConfig->set((boost::format("window_%1%_size") % name).str(), (boost::format("%1%;%2%") % size.GetWidth() % size.GetHeight()).str());
+ g_AppConfig->set((boost::format("window_%1%_maximized") % name).str(), maximized);
+}
+
+void restore_window_size(wxTopLevelWindow *window, const std::string &name)
+{
+ // XXX: This still doesn't behave nicely in some situations (mostly on Linux).
+ // The problem is that it's hard to obtain window position with respect to screen geometry reliably
+ // from wxWidgets. Sometimes wxWidgets claim a window is located on a different screen than on which
+ // it's actually visible. I suspect this has something to do with window initialization (maybe we
+ // restore window geometry too early), but haven't yet found a workaround.
+
+ const auto display_idx = wxDisplay::GetFromWindow(window);
+ if (display_idx == wxNOT_FOUND) { return; }
+
+ const auto display = wxDisplay(display_idx).GetClientArea();
+ std::vector<std::string> pair;
+
+ try {
+ const auto key_size = (boost::format("window_%1%_size") % name).str();
+ if (g_AppConfig->has(key_size)) {
+ if (unescape_strings_cstyle(g_AppConfig->get(key_size), pair) && pair.size() == 2) {
+ auto width = boost::lexical_cast<int>(pair[0]);
+ auto height = boost::lexical_cast<int>(pair[1]);
+
+ window->SetSize(width, height);
+ }
+ }
+ } catch(const boost::bad_lexical_cast &) {}
+
+ // Maximizing should be the last thing to do.
+ // This ensure the size and position are sane when the user un-maximizes the window.
+ const auto key_maximized = (boost::format("window_%1%_maximized") % name).str();
+ if (g_AppConfig->get(key_maximized) == "1") {
+ window->Maximize(true);
+ }
+}
+
+void enable_action_buttons(bool enable)
+{
+ if (g_buttons.empty())
+ return;
+
+ // Update background colour for buttons
+ const wxColour bgrd_color = enable ? wxColour(224, 224, 224/*255, 96, 0*/) : wxColour(204, 204, 204);
+
+ for (auto btn : g_buttons) {
+ btn->Enable(enable);
+ btn->SetBackgroundColour(bgrd_color);
+ }
+}
+
+void about()
+{
+ AboutDialog dlg;
+ dlg.ShowModal();
+ dlg.Destroy();
+}
+
+void desktop_open_datadir_folder()
+{
+ // Execute command to open a file explorer, platform dependent.
+ // FIXME: The const_casts aren't needed in wxWidgets 3.1, remove them when we upgrade.
+
+ const auto path = data_dir();
+#ifdef _WIN32
+ const auto widepath = wxString::FromUTF8(path.data());
+ const wchar_t *argv[] = { L"explorer", widepath.GetData(), nullptr };
+ ::wxExecute(const_cast<wchar_t**>(argv), wxEXEC_ASYNC, nullptr);
+#elif __APPLE__
+ const char *argv[] = { "open", path.data(), nullptr };
+ ::wxExecute(const_cast<char**>(argv), wxEXEC_ASYNC, nullptr);
+#else
+ const char *argv[] = { "xdg-open", path.data(), nullptr };
+
+ // Check if we're running in an AppImage container, if so, we need to remove AppImage's env vars,
+ // because they may mess up the environment expected by the file manager.
+ // Mostly this is about LD_LIBRARY_PATH, but we remove a few more too for good measure.
+ if (wxGetEnv("APPIMAGE", nullptr)) {
+ // We're running from AppImage
+ wxEnvVariableHashMap env_vars;
+ wxGetEnvMap(&env_vars);
+
+ env_vars.erase("APPIMAGE");
+ env_vars.erase("APPDIR");
+ env_vars.erase("LD_LIBRARY_PATH");
+ env_vars.erase("LD_PRELOAD");
+ env_vars.erase("UNION_PRELOAD");
+
+ wxExecuteEnv exec_env;
+ exec_env.env = std::move(env_vars);
+
+ wxString owd;
+ if (wxGetEnv("OWD", &owd)) {
+ // This is the original work directory from which the AppImage image was run,
+ // set it as CWD for the child process:
+ exec_env.cwd = std::move(owd);
+ }
+
+ ::wxExecute(const_cast<char**>(argv), wxEXEC_ASYNC, nullptr, &exec_env);
+ } else {
+ // Looks like we're NOT running from AppImage, we'll make no changes to the environment.
+ ::wxExecute(const_cast<char**>(argv), wxEXEC_ASYNC, nullptr, nullptr);
+ }
+#endif
+}
+
+} }