2018-09-20 06:40:22 +00:00
# include "GUI_App.hpp"
2018-10-05 21:29:15 +00:00
# include "GUI_ObjectList.hpp"
2018-10-04 14:43:10 +00:00
# include "GUI_ObjectManipulation.hpp"
2018-11-26 13:41:58 +00:00
# include "I18N.hpp"
2018-09-20 06:40:22 +00:00
2019-05-03 14:27:56 +00:00
# include <algorithm>
# include <iterator>
2019-02-14 14:28:18 +00:00
# include <exception>
2018-09-20 06:40:22 +00:00
# include <boost/lexical_cast.hpp>
# include <boost/algorithm/string.hpp>
2019-02-14 14:28:18 +00:00
# include <boost/log/trivial.hpp>
2018-09-20 06:40:22 +00:00
# include <wx/stdpaths.h>
# include <wx/imagpng.h>
# include <wx/display.h>
# include <wx/menu.h>
# include <wx/menuitem.h>
2018-09-20 11:12:35 +00:00
# include <wx/filedlg.h>
2019-03-26 09:08:57 +00:00
# include <wx/progdlg.h>
2018-10-02 11:23:38 +00:00
# include <wx/dir.h>
2018-11-26 13:41:58 +00:00
# include <wx/wupdlock.h>
2019-01-04 11:06:25 +00:00
# include <wx/filefn.h>
2019-03-18 11:48:39 +00:00
# include <wx/sysopt.h>
2019-04-09 09:28:11 +00:00
# include <wx/msgdlg.h>
# include <wx/log.h>
2019-05-03 14:27:56 +00:00
# include <wx/intl.h>
2018-11-26 13:41:58 +00:00
# include "libslic3r/Utils.hpp"
# include "libslic3r/Model.hpp"
# include "libslic3r/I18N.hpp"
2018-09-20 06:40:22 +00:00
# include "GUI.hpp"
2018-10-17 12:01:10 +00:00
# include "GUI_Utils.hpp"
2018-09-20 06:40:22 +00:00
# include "AppConfig.hpp"
# include "PresetBundle.hpp"
# include "3DScene.hpp"
# include "../Utils/PresetUpdater.hpp"
2018-12-11 09:33:11 +00:00
# include "../Utils/PrintHost.hpp"
2019-04-26 14:59:14 +00:00
# include "../Utils/MacDarkMode.hpp"
2019-04-17 12:56:00 +00:00
# include "ConfigWizard.hpp"
2018-09-20 23:33:41 +00:00
# include "slic3r/Config/Snapshot.hpp"
# include "ConfigSnapshotDialog.hpp"
# include "FirmwareDialog.hpp"
# include "Preferences.hpp"
# include "Tab.hpp"
2018-10-25 10:48:19 +00:00
# include "SysInfoDialog.hpp"
2018-12-19 12:06:24 +00:00
# include "KBShortcutsDialog.hpp"
2018-09-20 06:40:22 +00:00
2019-05-03 13:50:05 +00:00
# ifdef __WXMSW__
# include <Shlobj.h>
# endif // __WXMSW__
2018-09-20 06:40:22 +00:00
namespace Slic3r {
namespace GUI {
2018-10-04 09:12:55 +00:00
2018-12-06 16:32:49 +00:00
wxString file_wildcards ( FileType file_type , const std : : string & custom_extension )
{
2018-12-20 17:23:11 +00:00
static const std : : string defaults [ FT_SIZE ] = {
2019-05-15 08:23:02 +00:00
/* FT_STL */ " STL files (*.stl)|*.stl;*.STL " ,
/* FT_OBJ */ " OBJ files (*.obj)|*.obj;*.OBJ " ,
/* FT_AMF */ " AMF files (*.amf)|*.zip.amf;*.amf;*.AMF;*.xml;*.XML " ,
/* FT_3MF */ " 3MF files (*.3mf)|*.3mf;*.3MF; " ,
/* FT_PRUSA */ " Prusa Control files (*.prusa)|*.prusa;*.PRUSA " ,
/* FT_GCODE */ " G-code files (*.gcode, *.gco, *.g, *.ngc)|*.gcode;*.GCODE;*.gco;*.GCO;*.g;*.G;*.ngc;*.NGC " ,
/* FT_MODEL */ " Known files (*.stl, *.obj, *.amf, *.xml, *.3mf, *.prusa)|*.stl;*.STL;*.obj;*.OBJ;*.amf;*.AMF;*.xml;*.XML;*.3mf;*.3MF;*.prusa;*.PRUSA " ,
/* FT_PROJECT */ " Project files (*.3mf, *.amf)|*.3mf;*.3MF;*.amf;*.AMF " ,
2018-12-06 16:32:49 +00:00
2019-07-18 09:12:11 +00:00
/* FT_INI */ " INI files (*.ini)|*.ini;*.INI " ,
/* FT_SVG */ " SVG files (*.svg)|*.svg;*.SVG " ,
/* FT_TEX */ " Texture (*.png, *.svg)|*.png;*.PNG;*.svg;*.SVG " ,
/* FT_PNGZIP */ " Masked SLA files (*.sl1)|*.sl1;*.SL1 " ,
2018-12-06 16:32:49 +00:00
} ;
2018-12-20 17:23:11 +00:00
std : : string out = defaults [ file_type ] ;
2018-12-06 16:32:49 +00:00
if ( ! custom_extension . empty ( ) ) {
2018-12-20 17:23:11 +00:00
// Find the custom extension in the template.
if ( out . find ( std : : string ( " * " ) + custom_extension + " , " ) = = std : : string : : npos & & out . find ( std : : string ( " * " ) + custom_extension + " ) " ) = = std : : string : : npos ) {
// The custom extension was not found in the template.
// Append the custom extension to the wildcards, so that the file dialog would not add the default extension to it.
boost : : replace_first ( out , " )| " , std : : string ( " , * " ) + custom_extension + " )| " ) ;
out + = std : : string ( " ;* " ) + custom_extension ;
}
2018-12-06 16:32:49 +00:00
}
2019-01-07 18:17:10 +00:00
return from_u8 ( out ) ;
2018-12-06 16:32:49 +00:00
}
2018-10-04 09:12:55 +00:00
2018-10-01 13:09:31 +00:00
static std : : string libslic3r_translate_callback ( const char * s ) { return wxGetTranslation ( wxString ( s , wxConvUTF8 ) ) . utf8_str ( ) . data ( ) ; }
2019-04-02 10:00:50 +00:00
static void register_dpi_event ( )
{
# ifdef WIN32
enum { WM_DPICHANGED_ = 0x02e0 } ;
wxWindow : : MSWRegisterMessageHandler ( WM_DPICHANGED_ , [ ] ( wxWindow * win , WXUINT nMsg , WXWPARAM wParam , WXLPARAM lParam ) {
const int dpi = wParam & 0xffff ;
const auto rect = reinterpret_cast < PRECT > ( lParam ) ;
const wxRect wxrect ( wxPoint ( rect - > top , rect - > left ) , wxPoint ( rect - > bottom , rect - > right ) ) ;
DpiChangedEvent evt ( EVT_DPI_CHANGED , dpi , wxrect ) ;
win - > GetEventHandler ( ) - > AddPendingEvent ( evt ) ;
return true ;
} ) ;
# endif
}
2019-04-09 09:28:11 +00:00
static void generic_exception_handle ( )
{
// Note: Some wxWidgets APIs use wxLogError() to report errors, eg. wxImage
// - see https://docs.wxwidgets.org/3.1/classwx_image.html#aa249e657259fe6518d68a5208b9043d0
//
// wxLogError typically goes around exception handling and display an error dialog some time
// after an error is logged even if exception handling and OnExceptionInMainLoop() take place.
// This is why we use wxLogError() here as well instead of a custom dialog, because it accumulates
// errors if multiple have been collected and displays just one error message for all of them.
// Otherwise we would get multiple error messages for one missing png, for example.
//
// If a custom error message window (or some other solution) were to be used, it would be necessary
// to turn off wxLogError() usage in wx APIs, most notably in wxImage
// - see https://docs.wxwidgets.org/trunk/classwx_image.html#aa32e5d3507cc0f8c3330135bc0befc6a
try {
throw ;
} catch ( const std : : exception & ex ) {
wxLogError ( " Internal error: %s " , ex . what ( ) ) ;
BOOST_LOG_TRIVIAL ( error ) < < boost : : format ( " Uncaught exception: %1% " ) % ex . what ( ) ;
throw ;
} catch ( . . . ) {
wxLogError ( " Unknown internal error " ) ;
BOOST_LOG_TRIVIAL ( error ) < < " Uncaught exception: Unknown error " ;
}
}
2018-09-20 23:33:41 +00:00
IMPLEMENT_APP ( GUI_App )
2018-11-26 09:56:07 +00:00
GUI_App : : GUI_App ( )
: wxApp ( )
2019-02-21 17:59:21 +00:00
, m_em_unit ( 10 )
2019-04-01 12:12:05 +00:00
, m_imgui ( new ImGuiWrapper ( ) )
2018-11-26 09:56:07 +00:00
{ }
2019-06-27 12:42:55 +00:00
GUI_App : : ~ GUI_App ( )
{
if ( app_config ! = nullptr )
delete app_config ;
if ( preset_bundle ! = nullptr )
delete preset_bundle ;
if ( preset_updater ! = nullptr )
delete preset_updater ;
}
2018-09-20 06:40:22 +00:00
bool GUI_App : : OnInit ( )
2019-04-09 09:28:11 +00:00
{
try {
return on_init_inner ( ) ;
} catch ( . . . ) {
generic_exception_handle ( ) ;
return false ;
}
}
bool GUI_App : : on_init_inner ( )
2018-09-20 06:40:22 +00:00
{
2019-01-04 11:06:25 +00:00
// Verify resources path
const wxString resources_dir = from_u8 ( Slic3r : : resources_dir ( ) ) ;
wxCHECK_MSG ( wxDirExists ( resources_dir ) , false ,
wxString : : Format ( " Resources path does not exist or is not a directory: %s " , resources_dir ) ) ;
2018-10-31 09:19:44 +00:00
2019-05-13 10:42:40 +00:00
SetAppName ( SLIC3R_APP_KEY ) ;
2019-04-15 14:14:19 +00:00
SetAppDisplayName ( SLIC3R_APP_NAME ) ;
2018-09-20 06:40:22 +00:00
2019-03-18 19:54:01 +00:00
// Enable this to get the default Win32 COMCTRL32 behavior of static boxes.
2019-03-18 11:48:39 +00:00
// wxSystemOptions::SetOption("msw.staticbox.optimized-paint", 0);
2019-03-18 19:54:01 +00:00
// Enable this to disable Windows Vista themes for all wxNotebooks. The themes seem to lead to terrible
// performance when working on high resolution multi-display setups.
// wxSystemOptions::SetOption("msw.notebook.themed-background", 0);
2019-03-18 11:48:39 +00:00
2018-09-20 23:33:41 +00:00
// Slic3r::debugf "wxWidgets version %s, Wx version %s\n", wxVERSION_STRING, wxVERSION;
2018-09-20 06:40:22 +00:00
// Set the Slic3r data directory at the Slic3r XS module.
// Unix: ~/ .Slic3r
// Windows : "C:\Users\username\AppData\Roaming\Slic3r" or "C:\Documents and Settings\username\Application Data\Slic3r"
// Mac : "~/Library/Application Support/Slic3r"
2018-09-21 09:40:32 +00:00
if ( data_dir ( ) . empty ( ) )
2018-10-01 13:09:31 +00:00
set_data_dir ( wxStandardPaths : : Get ( ) . GetUserDataDir ( ) . ToUTF8 ( ) . data ( ) ) ;
2018-09-20 06:40:22 +00:00
2018-09-20 23:33:41 +00:00
app_config = new AppConfig ( ) ;
preset_bundle = new PresetBundle ( ) ;
2018-09-20 06:40:22 +00:00
// just checking for existence of Slic3r::data_dir is not enough : it may be an empty directory
// supplied as argument to --datadir; in that case we should still run the wizard
2019-04-09 09:28:11 +00:00
preset_bundle - > setup_directories ( ) ;
2018-12-03 12:14:28 +00:00
2019-05-17 15:35:15 +00:00
app_conf_exists = app_config - > exists ( ) ;
2018-09-20 06:40:22 +00:00
// load settings
2019-05-17 15:35:15 +00:00
app_conf_exists = app_config - > exists ( ) ;
if ( app_conf_exists ) {
2018-12-03 12:14:28 +00:00
app_config - > load ( ) ;
2019-05-17 15:35:15 +00:00
}
2018-10-01 13:09:31 +00:00
app_config - > set ( " version " , SLIC3R_VERSION ) ;
2018-09-20 23:33:41 +00:00
app_config - > save ( ) ;
2018-09-20 06:40:22 +00:00
2019-05-03 13:50:05 +00:00
# ifdef __WXMSW__
associate_3mf_files ( ) ;
# endif // __WXMSW__
2018-10-01 13:09:31 +00:00
preset_updater = new PresetUpdater ( ) ;
2019-02-14 15:51:41 +00:00
Bind ( EVT_SLIC3R_VERSION_ONLINE , [ this ] ( const wxCommandEvent & evt ) {
app_config - > set ( " version_online " , into_u8 ( evt . GetString ( ) ) ) ;
app_config - > save ( ) ;
} ) ;
2018-09-20 06:40:22 +00:00
2019-03-29 13:57:53 +00:00
// initialize label colors and fonts
init_label_colours ( ) ;
init_fonts ( ) ;
2018-09-20 23:33:41 +00:00
load_language ( ) ;
2018-09-20 06:40:22 +00:00
// Suppress the '- default -' presets.
2018-10-31 15:22:36 +00:00
preset_bundle - > set_default_suppressed ( app_config - > get ( " no_defaults " ) = = " 1 " ) ;
2019-04-09 09:28:11 +00:00
try {
preset_bundle - > load_presets ( * app_config ) ;
} catch ( const std : : exception & ex ) {
show_error ( nullptr , from_u8 ( ex . what ( ) ) ) ;
}
2018-09-20 06:40:22 +00:00
2019-04-02 10:00:50 +00:00
register_dpi_event ( ) ;
2018-10-01 13:09:31 +00:00
// Let the libslic3r know the callback, which will translate messages on demand.
Slic3r : : I18N : : set_translate_callback ( libslic3r_translate_callback ) ;
2018-09-20 06:40:22 +00:00
// application frame
2018-12-06 13:47:53 +00:00
if ( wxImage : : FindHandler ( wxBITMAP_TYPE_PNG ) = = nullptr )
wxImage : : AddHandler ( new wxPNGHandler ( ) ) ;
2018-12-19 16:38:41 +00:00
mainframe = new MainFrame ( ) ;
2018-10-05 21:29:15 +00:00
sidebar ( ) . obj_list ( ) - > init_objects ( ) ; // propagate model objects to object list
2019-01-24 11:11:01 +00:00
// update_mode(); // !!! do that later
2018-09-20 06:40:22 +00:00
SetTopWindow ( mainframe ) ;
2018-12-14 14:27:34 +00:00
m_printhost_job_queue . reset ( new PrintHostJobQueue ( mainframe - > printhost_queue_dlg ( ) ) ) ;
2018-10-02 11:23:38 +00:00
Bind ( wxEVT_IDLE , [ this ] ( wxIdleEvent & event )
{
2019-04-09 09:28:11 +00:00
if ( ! plater_ )
return ;
2019-03-11 10:03:13 +00:00
2019-02-22 16:18:58 +00:00
if ( app_config - > dirty ( ) & & app_config - > get ( " autosave " ) = = " 1 " )
2018-10-02 11:23:38 +00:00
app_config - > save ( ) ;
2018-12-20 19:12:26 +00:00
2019-03-11 10:03:13 +00:00
this - > obj_manipul ( ) - > update_if_dirty ( ) ;
2018-10-02 11:23:38 +00:00
2019-02-28 15:19:53 +00:00
// Preset updating & Configwizard are done after the above initializations,
// and after MainFrame is created & shown.
// The extra CallAfter() is needed because of Mac, where this is the only way
// to popup a modal dialog on start without screwing combo boxes.
2019-04-17 12:56:00 +00:00
// This is ugly but I honestly found no better way to do it.
2019-02-28 15:19:53 +00:00
// Neither wxShowEvent nor wxWindowCreateEvent work reliably.
static bool once = true ;
if ( once ) {
once = false ;
2019-05-17 15:35:15 +00:00
PresetUpdater : : UpdateResult updater_result ;
2019-02-28 15:19:53 +00:00
try {
2019-05-17 15:35:15 +00:00
updater_result = preset_updater - > config_update ( ) ;
if ( updater_result = = PresetUpdater : : R_INCOMPAT_EXIT ) {
2019-02-28 15:19:53 +00:00
mainframe - > Close ( ) ;
2019-05-17 15:35:15 +00:00
} else if ( updater_result = = PresetUpdater : : R_INCOMPAT_CONFIGURED ) {
app_conf_exists = true ;
2019-02-28 15:19:53 +00:00
}
} catch ( const std : : exception & ex ) {
2019-04-09 09:28:11 +00:00
show_error ( nullptr , from_u8 ( ex . what ( ) ) ) ;
2019-02-28 15:19:53 +00:00
}
CallAfter ( [ this ] {
2019-06-03 09:31:32 +00:00
config_wizard_startup ( app_conf_exists ) ;
preset_updater - > slic3r_update_notify ( ) ;
2019-02-28 15:19:53 +00:00
preset_updater - > sync ( preset_bundle ) ;
} ) ;
2018-10-02 11:23:38 +00:00
}
} ) ;
2019-03-11 10:49:24 +00:00
load_current_presets ( ) ;
2018-09-20 06:40:22 +00:00
mainframe - > Show ( true ) ;
2019-03-26 17:01:44 +00:00
/* Temporary workaround for the correct behavior of the Scrolled sidebar panel:
* change min hight of object list to the normal min value ( 15 * wxGetApp ( ) . em_unit ( ) )
* after first whole Mainframe updating / layouting
*/
if ( obj_list ( ) - > GetMinSize ( ) . GetY ( ) > 15 * em_unit ( ) )
obj_list ( ) - > SetMinSize ( wxSize ( - 1 , 15 * em_unit ( ) ) ) ;
2019-03-13 12:13:18 +00:00
update_mode ( ) ; // update view mode after fix of the object_list size
2019-03-26 17:01:44 +00:00
2019-02-28 15:19:53 +00:00
m_initialized = true ;
return true ;
2018-09-20 06:40:22 +00:00
}
2018-10-01 13:09:31 +00:00
unsigned GUI_App : : 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
) ) ;
}
2019-04-26 14:59:14 +00:00
bool GUI_App : : dark_mode ( )
{
const unsigned luma = get_colour_approx_luma ( wxSystemSettings : : GetColour ( wxSYS_COLOUR_WINDOW ) ) ;
return luma < 128 ;
}
bool GUI_App : : dark_mode_menus ( )
{
# if __APPLE__
return mac_dark_mode ( ) ;
# else
return dark_mode ( ) ;
# endif
}
2018-10-01 13:09:31 +00:00
void GUI_App : : init_label_colours ( )
{
2019-04-26 14:59:14 +00:00
if ( dark_mode ( ) ) {
2018-10-01 13:09:31 +00:00
m_color_label_modified = wxColour ( 253 , 111 , 40 ) ;
m_color_label_sys = wxColour ( 115 , 220 , 103 ) ;
}
2019-05-03 11:09:42 +00:00
else {
m_color_label_modified = wxColour ( 252 , 77 , 1 ) ;
m_color_label_sys = wxColour ( 26 , 132 , 57 ) ;
}
2018-10-01 13:09:31 +00:00
m_color_label_default = wxSystemSettings : : GetColour ( wxSYS_COLOUR_WINDOWTEXT ) ;
}
void GUI_App : : update_label_colours_from_appconfig ( )
{
2018-10-31 11:56:08 +00:00
if ( app_config - > has ( " label_clr_sys " ) ) {
2018-10-01 13:09:31 +00:00
auto str = app_config - > get ( " label_clr_sys " ) ;
if ( str ! = " " )
m_color_label_sys = wxColour ( str ) ;
}
2018-10-31 11:56:08 +00:00
if ( app_config - > has ( " label_clr_modified " ) ) {
2018-10-01 13:09:31 +00:00
auto str = app_config - > get ( " label_clr_modified " ) ;
if ( str ! = " " )
m_color_label_modified = wxColour ( str ) ;
}
}
void GUI_App : : init_fonts ( )
{
m_small_font = wxSystemSettings : : GetFont ( wxSYS_DEFAULT_GUI_FONT ) ;
m_bold_font = wxSystemSettings : : GetFont ( wxSYS_DEFAULT_GUI_FONT ) . Bold ( ) ;
2019-03-18 19:54:01 +00:00
m_normal_font = wxSystemSettings : : GetFont ( wxSYS_DEFAULT_GUI_FONT ) ;
2019-01-31 14:55:09 +00:00
2018-10-01 13:09:31 +00:00
# ifdef __WXMAC__
m_small_font . SetPointSize ( 11 ) ;
m_bold_font . SetPointSize ( 13 ) ;
# endif /*__WXMAC__*/
}
2019-05-06 16:28:23 +00:00
void GUI_App : : update_fonts ( const MainFrame * main_frame )
2019-04-13 21:46:52 +00:00
{
2019-04-18 00:03:40 +00:00
/* Only normal and bold fonts are used for an application rescale,
* because of under MSW small and normal fonts are the same .
* To avoid same rescaling twice , just fill this values
* from rescaled MainFrame
*/
2019-05-06 16:28:23 +00:00
if ( main_frame = = nullptr )
main_frame = this - > mainframe ;
m_normal_font = main_frame - > normal_font ( ) ;
m_small_font = m_normal_font ;
m_bold_font = main_frame - > normal_font ( ) . Bold ( ) ;
m_em_unit = main_frame - > em_unit ( ) ;
2019-04-13 21:46:52 +00:00
}
2018-10-01 13:09:31 +00:00
void GUI_App : : set_label_clr_modified ( const wxColour & clr ) {
m_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 ( ) ;
app_config - > set ( " label_clr_modified " , str ) ;
app_config - > save ( ) ;
}
void GUI_App : : set_label_clr_sys ( const wxColour & clr ) {
m_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 ( ) ;
app_config - > set ( " label_clr_sys " , str ) ;
app_config - > save ( ) ;
}
2019-05-22 11:51:02 +00:00
float GUI_App : : toolbar_icon_scale ( const bool is_limited /* = false*/ ) const
{
# ifdef __APPLE__
const float icon_sc = 1.0f ; // for Retina display will be used its own scale
# else
const float icon_sc = m_em_unit * 0.1f ;
# endif // __APPLE__
const std : : string & use_val = app_config - > get ( " use_custom_toolbar_size " ) ;
const std : : string & val = app_config - > get ( " custom_toolbar_size " ) ;
if ( val . empty ( ) | | use_val . empty ( ) | | use_val = = " 0 " )
return icon_sc ;
int int_val = atoi ( val . c_str ( ) ) ;
if ( is_limited & & int_val < 50 )
int_val = 50 ;
return 0.01f * int_val * icon_sc ;
}
2018-09-20 06:40:22 +00:00
void GUI_App : : recreate_GUI ( )
{
2019-03-26 09:08:57 +00:00
// Weird things happen as the Paint messages are floating around the windows being destructed.
// Avoid the Paint messages by hiding the main window.
// Also the application closes much faster without these unnecessary screen refreshes.
// In addition, there were some crashes due to the Paint events sent to already destructed windows.
mainframe - > Show ( false ) ;
const auto msg_name = _ ( L ( " Changing of an application language " ) ) + dots ;
wxProgressDialog dlg ( msg_name , msg_name ) ;
dlg . Pulse ( ) ;
2019-01-28 11:12:14 +00:00
// to make sure nobody accesses data from the soon-to-be-destroyed widgets:
tabs_list . clear ( ) ;
plater_ = nullptr ;
2019-01-24 14:32:50 +00:00
2019-03-26 09:08:57 +00:00
dlg . Update ( 10 , _ ( L ( " Recreating " ) ) + dots ) ;
2019-01-28 11:12:14 +00:00
MainFrame * topwindow = mainframe ;
2018-12-19 16:38:41 +00:00
mainframe = new MainFrame ( ) ;
2018-10-05 21:29:15 +00:00
sidebar ( ) . obj_list ( ) - > init_objects ( ) ; // propagate model objects to object list
2019-01-24 11:11:01 +00:00
2018-09-20 06:40:22 +00:00
if ( topwindow ) {
SetTopWindow ( mainframe ) ;
2019-03-26 09:08:57 +00:00
dlg . Update ( 30 , _ ( L ( " Recreating " ) ) + dots ) ;
2018-09-20 06:40:22 +00:00
topwindow - > Destroy ( ) ;
}
2019-04-15 14:14:19 +00:00
dlg . Update ( 80 , _ ( L ( " Loading of current presets " ) ) + dots ) ;
2019-03-26 09:08:57 +00:00
2018-12-19 16:38:41 +00:00
m_printhost_job_queue . reset ( new PrintHostJobQueue ( mainframe - > printhost_queue_dlg ( ) ) ) ;
2019-01-24 14:32:50 +00:00
load_current_presets ( ) ;
2018-12-19 16:38:41 +00:00
mainframe - > Show ( true ) ;
2019-03-26 09:08:57 +00:00
dlg . Update ( 90 , _ ( L ( " Loading of a mode view " ) ) + dots ) ;
2019-03-26 17:01:44 +00:00
/* Temporary workaround for the correct behavior of the Scrolled sidebar panel:
* change min hight of object list to the normal min value ( 15 * wxGetApp ( ) . em_unit ( ) )
* after first whole Mainframe updating / layouting
*/
if ( obj_list ( ) - > GetMinSize ( ) . GetY ( ) > 15 * em_unit ( ) )
obj_list ( ) - > SetMinSize ( wxSize ( - 1 , 15 * em_unit ( ) ) ) ;
2019-03-26 09:08:57 +00:00
update_mode ( ) ;
// #ys_FIXME_delete_after_testing Do we still need this ?
// CallAfter([]() {
// // Run the config wizard, don't offer the "reset user profile" checkbox.
// config_wizard_startup(true);
// });
2018-09-20 06:40:22 +00:00
}
void GUI_App : : system_info ( )
{
2018-10-25 10:48:19 +00:00
SysInfoDialog dlg ;
dlg . ShowModal ( ) ;
2018-09-20 06:40:22 +00:00
}
2018-12-19 12:06:24 +00:00
void GUI_App : : keyboard_shortcuts ( )
{
KBShortcutsDialog dlg ;
dlg . ShowModal ( ) ;
}
2018-09-20 06:40:22 +00:00
// static method accepting a wxWindow object as first parameter
bool GUI_App : : catch_error ( std : : function < void ( ) > cb ,
// wxMessageDialog* message_dialog,
2018-10-31 11:56:08 +00:00
const std : : string & err /*= ""*/ )
{
2018-09-20 06:40:22 +00:00
if ( ! err . empty ( ) ) {
if ( cb )
cb ( ) ;
// if (message_dialog)
// message_dialog->(err, "Error", wxOK | wxICON_ERROR);
show_error ( /*this*/ nullptr , err ) ;
return true ;
}
return false ;
}
// static method accepting a wxWindow object as first parameter
2018-10-31 11:56:08 +00:00
void fatal_error ( wxWindow * parent )
{
2018-09-20 06:40:22 +00:00
show_error ( parent , " " ) ;
// exit 1; // #ys_FIXME
}
// Called after the Preferences dialog is closed and the program settings are saved.
// Update the UI based on the current preferences.
2018-10-31 11:56:08 +00:00
void GUI_App : : update_ui_from_settings ( )
{
2018-09-20 06:40:22 +00:00
mainframe - > update_ui_from_settings ( ) ;
}
2019-04-29 14:55:41 +00:00
void GUI_App : : persist_window_geometry ( wxTopLevelWindow * window , bool default_maximized )
2019-01-11 17:09:21 +00:00
{
const std : : string name = into_u8 ( window - > GetName ( ) ) ;
window - > Bind ( wxEVT_CLOSE_WINDOW , [ = ] ( wxCloseEvent & event ) {
window_pos_save ( window , name ) ;
event . Skip ( ) ;
} ) ;
2019-04-29 14:55:41 +00:00
window_pos_restore ( window , name , default_maximized ) ;
2019-02-07 14:55:47 +00:00
on_window_geometry ( window , [ = ] ( ) {
window_pos_sanitize ( window ) ;
2019-01-11 17:09:21 +00:00
} ) ;
}
2019-07-17 13:48:53 +00:00
void GUI_App : : load_project ( wxWindow * parent , wxString & input_file ) const
2018-11-15 14:27:39 +00:00
{
input_file . Clear ( ) ;
wxFileDialog dialog ( parent ? parent : GetTopWindow ( ) ,
2019-05-15 08:23:02 +00:00
_ ( L ( " Choose one file (3MF/AMF): " ) ) ,
2018-11-15 14:27:39 +00:00
app_config - > get_last_dir ( ) , " " ,
2019-05-15 08:23:02 +00:00
file_wildcards ( FT_PROJECT ) , wxFD_OPEN | wxFD_FILE_MUST_EXIST ) ;
2018-09-20 11:12:35 +00:00
2018-11-15 14:27:39 +00:00
if ( dialog . ShowModal ( ) = = wxID_OK )
input_file = dialog . GetPath ( ) ;
}
2019-07-17 13:48:53 +00:00
void GUI_App : : import_model ( wxWindow * parent , wxArrayString & input_files ) const
2018-09-20 11:12:35 +00:00
{
2018-11-15 14:27:39 +00:00
input_files . Clear ( ) ;
wxFileDialog dialog ( parent ? parent : GetTopWindow ( ) ,
2018-09-20 11:12:35 +00:00
_ ( L ( " Choose one or more files (STL/OBJ/AMF/3MF/PRUSA): " ) ) ,
2019-01-02 14:11:05 +00:00
from_u8 ( app_config - > get_last_dir ( ) ) , " " ,
2018-12-06 16:32:49 +00:00
file_wildcards ( FT_MODEL ) , wxFD_OPEN | wxFD_MULTIPLE | wxFD_FILE_MUST_EXIST ) ;
2018-09-20 11:12:35 +00:00
2018-11-15 14:27:39 +00:00
if ( dialog . ShowModal ( ) = = wxID_OK )
dialog . GetPaths ( input_files ) ;
2018-09-20 11:12:35 +00:00
}
2018-09-20 06:40:22 +00:00
2019-04-29 16:27:46 +00:00
bool GUI_App : : switch_language ( )
{
2019-05-03 14:27:56 +00:00
if ( select_language ( ) ) {
2019-04-29 16:27:46 +00:00
save_language ( ) ;
_3DScene : : remove_all_canvases ( ) ;
recreate_GUI ( ) ;
return true ;
} else {
return false ;
}
}
2018-10-02 11:23:38 +00:00
// select language from the list of installed languages
2019-05-03 14:27:56 +00:00
bool GUI_App : : select_language ( )
2018-10-02 11:23:38 +00:00
{
2019-05-03 14:27:56 +00:00
const auto langs = get_installed_languages ( ) ;
wxArrayString names ;
names . Alloc ( langs . size ( ) ) ;
int init_selection = - 1 ;
const auto current_language = m_wxLocale ? m_wxLocale - > GetLanguage ( ) : wxLocale : : GetSystemLanguage ( ) ;
for ( size_t i = 0 ; i < langs . size ( ) ; i + + ) {
const auto lang = langs [ i ] - > Language ;
const bool is_english = lang > = wxLANGUAGE_ENGLISH & & lang < = wxLANGUAGE_ENGLISH_ZIMBABWE ;
if ( lang = = current_language | | ( current_language = = wxLANGUAGE_UNKNOWN & & is_english ) ) {
init_selection = i ;
}
names . Add ( langs [ i ] - > Description ) ;
2018-10-02 11:23:38 +00:00
}
2019-05-03 14:27:56 +00:00
const long index = wxGetSingleChoiceIndex (
_ ( L ( " Select the language " ) ) ,
_ ( L ( " Language " ) ) , names , init_selection > = 0 ? init_selection : 0 ) ;
if ( index ! = - 1 ) {
const wxLanguageInfo * lang = langs [ index ] ;
if ( lang - > Language = = current_language ) {
// There was no change
return false ;
}
m_wxLocale = new wxLocale ; // FIXME: leak?
m_wxLocale - > Init ( lang - > Language ) ;
2019-01-07 18:17:10 +00:00
m_wxLocale - > AddCatalogLookupPathPrefix ( from_u8 ( localization_dir ( ) ) ) ;
2019-05-13 10:13:28 +00:00
m_wxLocale - > AddCatalog ( SLIC3R_APP_KEY ) ;
2019-01-04 14:42:44 +00:00
//FIXME This is a temporary workaround, the correct solution is to switch to "C" locale during file import / export only.
wxSetlocale ( LC_NUMERIC , " C " ) ;
2018-10-02 11:23:38 +00:00
Preset : : update_suffix_modified ( ) ;
2019-05-03 14:27:56 +00:00
m_imgui - > set_language ( into_u8 ( lang - > CanonicalName ) ) ;
2018-10-02 11:23:38 +00:00
return true ;
}
2019-05-03 14:27:56 +00:00
2018-10-02 11:23:38 +00:00
return false ;
}
2019-05-10 12:43:35 +00:00
// Load gettext translation files and activate them at the start of the application,
// based on the "translation_language" key stored in the application config.
2018-09-20 23:33:41 +00:00
bool GUI_App : : load_language ( )
{
wxString language = wxEmptyString ;
if ( app_config - > has ( " translation_language " ) )
language = app_config - > get ( " translation_language " ) ;
2019-05-10 12:43:35 +00:00
if ( language . IsEmpty ( ) ) {
int lang = wxLocale : : GetSystemLanguage ( ) ;
if ( lang ! = wxLANGUAGE_UNKNOWN ) {
const wxLanguageInfo * info = wxLocale : : GetLanguageInfo ( lang ) ;
if ( info ! = nullptr )
language = info - > CanonicalName ;
2018-09-20 23:33:41 +00:00
}
}
2019-05-10 12:43:35 +00:00
const wxLanguageInfo * info = nullptr ;
if ( ! language . IsEmpty ( ) ) {
const auto langs = get_installed_languages ( ) ;
for ( const wxLanguageInfo * this_info : langs )
if ( this_info - > CanonicalName = = language ) {
info = this_info ;
break ;
}
}
m_wxLocale = new wxLocale ;
if ( info = = nullptr ) {
m_wxLocale - > Init ( wxLANGUAGE_DEFAULT ) ;
m_imgui - > set_language ( " en " ) ;
} else {
m_wxLocale - > Init ( info - > Language ) ;
m_wxLocale - > AddCatalogLookupPathPrefix ( from_u8 ( localization_dir ( ) ) ) ;
2019-05-13 10:13:28 +00:00
m_wxLocale - > AddCatalog ( SLIC3R_APP_KEY ) ;
2019-05-10 12:43:35 +00:00
m_imgui - > set_language ( into_u8 ( info - > CanonicalName ) ) ;
}
//FIXME This is a temporary workaround, the correct solution is to switch to "C" locale during file import / export only.
wxSetlocale ( LC_NUMERIC , " C " ) ;
Preset : : update_suffix_modified ( ) ;
return true ;
2018-09-20 23:33:41 +00:00
}
2018-10-02 11:23:38 +00:00
// save language at application config
void GUI_App : : save_language ( )
{
wxString language = wxEmptyString ;
if ( m_wxLocale )
language = m_wxLocale - > GetCanonicalName ( ) ;
2019-01-03 13:34:53 +00:00
app_config - > set ( " translation_language " , language . ToUTF8 ( ) . data ( ) ) ;
2018-10-02 11:23:38 +00:00
app_config - > save ( ) ;
}
2019-05-03 14:27:56 +00:00
// Get a list of installed languages
std : : vector < const wxLanguageInfo * > GUI_App : : get_installed_languages ( )
2018-10-02 11:23:38 +00:00
{
2019-05-03 14:27:56 +00:00
std : : vector < const wxLanguageInfo * > res ;
2018-10-02 11:23:38 +00:00
2019-01-07 18:17:10 +00:00
wxDir dir ( from_u8 ( localization_dir ( ) ) ) ;
2018-10-02 11:23:38 +00:00
wxString filename ;
const wxLanguageInfo * langinfo ;
wxString name = wxLocale : : GetLanguageName ( wxLANGUAGE_DEFAULT ) ;
2019-05-03 14:27:56 +00:00
if ( ! name . IsEmpty ( ) ) {
res . push_back ( wxLocale : : GetLanguageInfo ( wxLANGUAGE_DEFAULT ) ) ;
2018-10-02 11:23:38 +00:00
}
2019-05-03 14:27:56 +00:00
for ( bool cont = dir . GetFirst ( & filename , wxEmptyString , wxDIR_DIRS ) ; cont ; cont = dir . GetNext ( & filename ) ) {
2018-10-02 11:23:38 +00:00
langinfo = wxLocale : : FindLanguageInfo ( filename ) ;
2019-05-03 14:27:56 +00:00
if ( langinfo ! = NULL ) {
2018-10-02 11:23:38 +00:00
auto full_file_name = dir . GetName ( ) + wxFileName : : GetPathSeparator ( ) +
2019-05-13 10:42:40 +00:00
filename + wxFileName : : GetPathSeparator ( ) + SLIC3R_APP_KEY + wxT ( " .mo " ) ;
2019-05-03 14:27:56 +00:00
if ( wxFileExists ( full_file_name ) ) {
res . push_back ( langinfo ) ;
2018-10-02 11:23:38 +00:00
}
}
}
2019-05-03 14:27:56 +00:00
return res ;
2018-10-02 11:23:38 +00:00
}
2018-10-10 11:53:45 +00:00
Tab * GUI_App : : get_tab ( Preset : : Type type )
{
for ( Tab * tab : tabs_list )
if ( tab - > type ( ) = = type )
2019-03-07 15:36:39 +00:00
return tab - > complited ( ) ? tab : nullptr ; // To avoid actions with no-completed Tab
2018-10-10 11:53:45 +00:00
return nullptr ;
}
2019-01-10 10:05:58 +00:00
ConfigOptionMode GUI_App : : get_mode ( )
2018-09-20 23:33:41 +00:00
{
if ( ! app_config - > has ( " view_mode " ) )
2019-01-10 10:05:58 +00:00
return comSimple ;
2018-09-20 23:33:41 +00:00
const auto mode = app_config - > get ( " view_mode " ) ;
2019-01-10 10:05:58 +00:00
return mode = = " expert " ? comExpert :
mode = = " simple " ? comSimple : comAdvanced ;
2018-09-20 23:33:41 +00:00
}
2019-01-10 10:05:58 +00:00
void GUI_App : : save_mode ( const /*ConfigOptionMode*/ int mode )
{
const std : : string mode_str = mode = = comExpert ? " expert " :
mode = = comSimple ? " simple " : " advanced " ;
app_config - > set ( " view_mode " , mode_str ) ;
app_config - > save ( ) ;
update_mode ( ) ;
2018-12-21 08:19:00 +00:00
}
2018-10-03 13:14:52 +00:00
// Update view mode according to selected menu
void GUI_App : : update_mode ( )
{
2019-03-01 11:03:14 +00:00
sidebar ( ) . update_mode ( ) ;
2018-10-19 11:55:29 +00:00
for ( auto tab : tabs_list )
2019-03-20 15:22:01 +00:00
tab - > update_mode ( ) ;
2018-12-07 16:50:48 +00:00
plater ( ) - > update_object_menu ( ) ;
2018-10-03 13:14:52 +00:00
}
2018-09-20 23:33:41 +00:00
void GUI_App : : add_config_menu ( wxMenuBar * menu )
{
auto local_menu = new wxMenu ( ) ;
2019-05-21 12:06:43 +00:00
wxWindowID config_id_base = wxWindow : : NewControlId ( int ( ConfigMenuCnt ) ) ;
2018-09-20 23:33:41 +00:00
2019-01-21 11:34:28 +00:00
const auto config_wizard_name = _ ( ConfigWizard : : name ( true ) . wx_str ( ) ) ;
2018-09-20 23:33:41 +00:00
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 ) ;
2019-01-21 11:34:28 +00:00
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 " ) ) ) ;
2018-09-20 23:33:41 +00:00
// local_menu->Append(config_id_base + ConfigMenuUpdate, _(L("Check for updates")), _(L("Check for configuration updates")));
local_menu - > AppendSeparator ( ) ;
2019-02-03 11:12:26 +00:00
local_menu - > Append ( config_id_base + ConfigMenuPreferences , _ ( L ( " &Preferences " ) ) + dots +
# ifdef __APPLE__
" \t Ctrl+, " ,
# else
" \t Ctrl+P " ,
# endif
_ ( L ( " Application preferences " ) ) ) ;
2018-09-20 23:33:41 +00:00
local_menu - > AppendSeparator ( ) ;
auto mode_menu = new wxMenu ( ) ;
2018-11-21 09:21:12 +00:00
mode_menu - > AppendRadioItem ( config_id_base + ConfigMenuModeSimple , _ ( L ( " Simple " ) ) , _ ( L ( " Simple View Mode " ) ) ) ;
mode_menu - > AppendRadioItem ( config_id_base + ConfigMenuModeAdvanced , _ ( L ( " Advanced " ) ) , _ ( L ( " Advanced View Mode " ) ) ) ;
mode_menu - > AppendRadioItem ( config_id_base + ConfigMenuModeExpert , _ ( L ( " Expert " ) ) , _ ( L ( " Expert View Mode " ) ) ) ;
2019-05-21 12:06:43 +00:00
Bind ( wxEVT_UPDATE_UI , [ this ] ( wxUpdateUIEvent & evt ) { if ( get_mode ( ) = = comSimple ) evt . Check ( true ) ; } , config_id_base + ConfigMenuModeSimple ) ;
Bind ( wxEVT_UPDATE_UI , [ this ] ( wxUpdateUIEvent & evt ) { if ( get_mode ( ) = = comAdvanced ) evt . Check ( true ) ; } , config_id_base + ConfigMenuModeAdvanced ) ;
Bind ( wxEVT_UPDATE_UI , [ this ] ( wxUpdateUIEvent & evt ) { if ( get_mode ( ) = = comExpert ) evt . Check ( true ) ; } , config_id_base + ConfigMenuModeExpert ) ;
2019-02-06 09:47:19 +00:00
2019-04-26 08:52:38 +00:00
local_menu - > AppendSubMenu ( mode_menu , _ ( L ( " Mode " ) ) , wxString : : Format ( _ ( L ( " %s View Mode " ) ) , SLIC3R_APP_NAME ) ) ;
2018-09-20 23:33:41 +00:00
local_menu - > AppendSeparator ( ) ;
2019-01-21 11:34:28 +00:00
local_menu - > Append ( config_id_base + ConfigMenuLanguage , _ ( L ( " Change Application &Language " ) ) ) ;
2018-09-20 23:33:41 +00:00
local_menu - > AppendSeparator ( ) ;
2019-01-21 11:34:28 +00:00
local_menu - > Append ( config_id_base + ConfigMenuFlashFirmware , _ ( L ( " Flash printer &firmware " ) ) , _ ( L ( " Upload a firmware image into an Arduino based printer " ) ) ) ;
2018-09-20 23:33:41 +00:00
// 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")));
2018-10-31 11:56:08 +00:00
local_menu - > Bind ( wxEVT_MENU , [ this , config_id_base ] ( wxEvent & event ) {
2018-09-20 23:33:41 +00:00
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 " ) ) ) ;
2019-04-18 13:05:17 +00:00
// set current normal font for dialog children,
// because of just dlg.SetFont(normal_font()) has no result;
for ( auto child : dlg . GetChildren ( ) )
child - > SetFont ( normal_font ( ) ) ;
2018-09-20 23:33:41 +00:00
if ( dlg . ShowModal ( ) = = wxID_OK )
app_config - > set ( " on_snapshot " ,
Slic3r : : GUI : : Config : : SnapshotDB : : singleton ( ) . take_snapshot (
* app_config , 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 ( * app_config ) )
on_snapshot = app_config - > 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 ( * app_config ) )
Config : : SnapshotDB : : singleton ( ) . take_snapshot ( * app_config , Config : : Snapshot : : SNAPSHOT_BEFORE_ROLLBACK ) ;
app_config - > set ( " on_snapshot " ,
Config : : SnapshotDB : : singleton ( ) . restore_snapshot ( dlg . snapshot_to_activate ( ) , * app_config ) . id ) ;
preset_bundle - > load_presets ( * app_config ) ;
// Load the currently selected preset into the GUI, update the preset selection box.
load_current_presets ( ) ;
}
}
break ;
case ConfigMenuPreferences :
{
2018-10-02 11:23:38 +00:00
PreferencesDialog dlg ( mainframe ) ;
dlg . ShowModal ( ) ;
2018-09-20 23:33:41 +00:00
break ;
}
case ConfigMenuLanguage :
{
2019-03-28 09:54:56 +00:00
/* Before change application language, let's check unsaved changes on 3D-Scene
2019-03-26 17:01:44 +00:00
* and draw user ' s attention to the application restarting after a language change
*/
wxMessageDialog dialog ( nullptr ,
2019-05-09 13:24:08 +00:00
_ ( L ( " Switching the language will trigger application restart. \n "
2019-05-09 13:44:53 +00:00
" You will lose content of the plater. " ) ) + " \n \n " +
2019-05-09 13:24:08 +00:00
_ ( L ( " Do you want to proceed? " ) ) ,
wxString ( SLIC3R_APP_NAME ) + " - " + _ ( L ( " Language selection " ) ) ,
2019-03-28 09:54:56 +00:00
wxICON_QUESTION | wxOK | wxCANCEL ) ;
if ( dialog . ShowModal ( ) = = wxID_CANCEL )
2019-03-26 17:01:44 +00:00
return ;
2019-04-29 16:27:46 +00:00
switch_language ( ) ;
2018-09-20 23:33:41 +00:00
break ;
}
case ConfigMenuFlashFirmware :
FirmwareDialog : : run ( mainframe ) ;
break ;
default :
break ;
}
} ) ;
2019-05-21 12:06:43 +00:00
using std : : placeholders : : _1 ;
auto modfn = [ this ] ( int mode , wxCommandEvent & ) { if ( get_mode ( ) ! = mode ) save_mode ( mode ) ; } ;
mode_menu - > Bind ( wxEVT_MENU , std : : bind ( modfn , comSimple , _1 ) , config_id_base + ConfigMenuModeSimple ) ;
mode_menu - > Bind ( wxEVT_MENU , std : : bind ( modfn , comAdvanced , _1 ) , config_id_base + ConfigMenuModeAdvanced ) ;
mode_menu - > Bind ( wxEVT_MENU , std : : bind ( modfn , comExpert , _1 ) , config_id_base + ConfigMenuModeExpert ) ;
2018-09-20 23:33:41 +00:00
menu - > Append ( local_menu , _ ( L ( " &Configuration " ) ) ) ;
}
// 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.
2019-07-18 15:41:47 +00:00
bool GUI_App : : check_unsaved_changes ( const wxString & header )
2018-09-20 23:33:41 +00:00
{
2019-05-06 16:28:23 +00:00
wxString dirty ;
2018-10-31 15:22:36 +00:00
PrinterTechnology printer_technology = preset_bundle - > printers . get_edited_preset ( ) . printer_technology ( ) ;
2018-09-20 23:33:41 +00:00
for ( Tab * tab : tabs_list )
2018-10-31 15:22:36 +00:00
if ( tab - > supports_printer_technology ( printer_technology ) & & tab - > current_preset_is_dirty ( ) )
2018-09-20 23:33:41 +00:00
if ( dirty . empty ( ) )
2019-05-10 13:10:17 +00:00
dirty = tab - > title ( ) ;
2018-09-20 23:33:41 +00:00
else
2019-05-10 13:10:17 +00:00
dirty + = wxString ( " , " ) + tab - > title ( ) ;
2018-09-20 23:33:41 +00:00
if ( dirty . empty ( ) )
// No changes, the application may close or reload presets.
return true ;
// Ask the user.
2019-07-18 15:41:47 +00:00
wxString message ;
if ( ! header . empty ( ) )
message = header + " \n \n " ;
message + = _ ( L ( " The presets on the following tabs were modified " ) ) + " : " + dirty + " \n \n " + _ ( L ( " Discard changes and continue anyway? " ) ) ;
2019-03-26 17:01:44 +00:00
wxMessageDialog dialog ( mainframe ,
2019-07-18 15:41:47 +00:00
message ,
2019-05-09 17:24:21 +00:00
wxString ( SLIC3R_APP_NAME ) + " - " + _ ( L ( " Unsaved Presets " ) ) ,
2018-09-20 23:33:41 +00:00
wxICON_QUESTION | wxYES_NO | wxNO_DEFAULT ) ;
2019-03-26 17:01:44 +00:00
return dialog . ShowModal ( ) = = wxID_YES ;
2018-09-20 23:33:41 +00:00
}
2018-10-01 13:09:31 +00:00
bool GUI_App : : checked_tab ( Tab * tab )
{
bool ret = true ;
if ( find ( tabs_list . begin ( ) , tabs_list . end ( ) , tab ) = = tabs_list . end ( ) )
ret = false ;
return ret ;
}
// Update UI / Tabs to reflect changes in the currently loaded presets
void GUI_App : : load_current_presets ( )
{
2018-10-31 15:22:36 +00:00
PrinterTechnology printer_technology = preset_bundle - > printers . get_edited_preset ( ) . printer_technology ( ) ;
2018-12-03 12:14:28 +00:00
this - > plater ( ) - > set_printer_technology ( printer_technology ) ;
2018-10-31 15:22:36 +00:00
for ( Tab * tab : tabs_list )
2018-10-31 17:05:25 +00:00
if ( tab - > supports_printer_technology ( printer_technology ) ) {
2019-05-09 17:24:21 +00:00
if ( tab - > type ( ) = = Preset : : TYPE_PRINTER )
2018-10-31 17:05:25 +00:00
static_cast < TabPrinter * > ( tab ) - > update_pages ( ) ;
tab - > load_current_preset ( ) ;
}
2018-10-01 13:09:31 +00:00
}
2019-02-14 14:28:18 +00:00
bool GUI_App : : OnExceptionInMainLoop ( )
{
2019-04-09 09:28:11 +00:00
generic_exception_handle ( ) ;
2019-02-14 14:28:18 +00:00
return false ;
}
2019-01-09 14:40:12 +00:00
# ifdef __APPLE__
// wxWidgets override to get an event on open files.
void GUI_App : : MacOpenFiles ( const wxArrayString & fileNames )
{
std : : vector < std : : string > files ;
2019-01-09 15:05:36 +00:00
for ( size_t i = 0 ; i < fileNames . GetCount ( ) ; + + i )
2019-01-09 14:40:12 +00:00
files . emplace_back ( fileNames [ i ] . ToUTF8 ( ) . data ( ) ) ;
this - > plater ( ) - > load_files ( files , true , true ) ;
}
# endif /* __APPLE */
2018-10-04 14:43:10 +00:00
Sidebar & GUI_App : : sidebar ( )
{
2018-10-24 10:57:23 +00:00
return plater_ - > sidebar ( ) ;
2018-10-04 14:43:10 +00:00
}
ObjectManipulation * GUI_App : : obj_manipul ( )
{
2019-01-14 13:42:21 +00:00
// If this method is called before plater_ has been initialized, return nullptr (to avoid a crash)
return ( plater_ ! = nullptr ) ? sidebar ( ) . obj_manipul ( ) : nullptr ;
2018-10-04 14:43:10 +00:00
}
2018-11-09 17:39:07 +00:00
ObjectSettings * GUI_App : : obj_settings ( )
{
return sidebar ( ) . obj_settings ( ) ;
}
2018-10-05 21:29:15 +00:00
ObjectList * GUI_App : : obj_list ( )
2018-09-20 23:33:41 +00:00
{
2018-10-05 21:29:15 +00:00
return sidebar ( ) . obj_list ( ) ;
2018-09-20 23:33:41 +00:00
}
2019-05-28 14:38:04 +00:00
ObjectLayers * GUI_App : : obj_layers ( )
{
return sidebar ( ) . obj_layers ( ) ;
}
2018-10-08 14:27:38 +00:00
Plater * GUI_App : : plater ( )
{
2018-10-24 10:57:23 +00:00
return plater_ ;
2018-10-08 14:27:38 +00:00
}
2019-07-18 15:41:47 +00:00
Model & GUI_App : : model ( )
2018-10-05 21:29:15 +00:00
{
2019-07-18 15:41:47 +00:00
return plater_ - > model ( ) ;
2018-10-05 21:29:15 +00:00
}
2018-09-20 23:33:41 +00:00
2018-10-05 21:29:15 +00:00
wxNotebook * GUI_App : : tab_panel ( ) const
{
return mainframe - > m_tabpanel ;
}
2018-09-20 23:33:41 +00:00
2019-06-03 12:52:15 +00:00
// extruders count from selected printer preset
2019-03-19 13:36:32 +00:00
int GUI_App : : extruders_cnt ( ) const
{
const Preset & preset = preset_bundle - > printers . get_selected_preset ( ) ;
return preset . printer_technology ( ) = = ptSLA ? 1 :
preset . config . option < ConfigOptionFloats > ( " nozzle_diameter " ) - > values . size ( ) ;
}
2019-06-03 12:52:15 +00:00
// extruders count from edited printer preset
int GUI_App : : extruders_edited_cnt ( ) const
{
const Preset & preset = preset_bundle - > printers . get_edited_preset ( ) ;
return preset . printer_technology ( ) = = ptSLA ? 1 :
preset . config . option < ConfigOptionFloats > ( " nozzle_diameter " ) - > values . size ( ) ;
}
2019-07-29 15:55:50 +00:00
wxString GUI_App : : current_language_code_safe ( ) const
{
// Translate the language code to a code, for which Prusa Research maintains translations.
wxString language_code = this - > current_language_code ( ) ;
size_t idx_underscore = language_code . find ( language_code ) ;
if ( idx_underscore ! = wxString : : npos )
language_code = language_code . substr ( 0 , idx_underscore ) ;
const std : : map < wxString , wxString > mapping {
{ " cs " , " cs_CZ " , } ,
2019-08-06 15:27:36 +00:00
{ " sk " , " cs_CZ " , } ,
2019-07-29 15:55:50 +00:00
{ " de " , " de_DE " , } ,
{ " es " , " es_ES " , } ,
{ " fr " , " fr_FR " , } ,
{ " it " , " it_IT " , } ,
{ " ja " , " ja_JP " , } ,
{ " ko " , " ko_KR " , } ,
{ " pl " , " pl_PL " , } ,
{ " uk " , " uk_UA " , } ,
{ " zh " , " zh_CN " , } ,
} ;
auto it = mapping . find ( language_code ) ;
if ( it ! = mapping . end ( ) )
language_code = it - > second ;
else
language_code = " en_US " ;
return language_code ;
}
2019-05-13 14:34:41 +00:00
void GUI_App : : open_web_page_localized ( const std : : string & http_address )
{
2019-07-29 15:55:50 +00:00
wxLaunchDefaultBrowser ( http_address + " &lng= " + this - > current_language_code_safe ( ) ) ;
2019-05-13 14:34:41 +00:00
}
2019-01-11 17:09:21 +00:00
void GUI_App : : window_pos_save ( wxTopLevelWindow * window , const std : : string & name )
{
if ( name . empty ( ) ) { return ; }
const auto config_key = ( boost : : format ( " window_%1% " ) % name ) . str ( ) ;
WindowMetrics metrics = WindowMetrics : : from_window ( window ) ;
app_config - > set ( config_key , metrics . serialize ( ) ) ;
app_config - > save ( ) ;
}
2019-04-29 14:55:41 +00:00
void GUI_App : : window_pos_restore ( wxTopLevelWindow * window , const std : : string & name , bool default_maximized )
2019-01-11 17:09:21 +00:00
{
if ( name . empty ( ) ) { return ; }
const auto config_key = ( boost : : format ( " window_%1% " ) % name ) . str ( ) ;
2019-04-29 14:55:41 +00:00
if ( ! app_config - > has ( config_key ) ) {
window - > Maximize ( default_maximized ) ;
return ;
}
2019-01-11 17:09:21 +00:00
auto metrics = WindowMetrics : : deserialize ( app_config - > get ( config_key ) ) ;
2019-04-29 14:55:41 +00:00
if ( ! metrics ) {
window - > Maximize ( default_maximized ) ;
return ;
}
2019-01-11 17:09:21 +00:00
window - > SetSize ( metrics - > get_rect ( ) ) ;
window - > Maximize ( metrics - > get_maximized ( ) ) ;
}
void GUI_App : : window_pos_sanitize ( wxTopLevelWindow * window )
{
unsigned display_idx = wxDisplay : : GetFromWindow ( window ) ;
wxRect display ;
if ( display_idx = = wxNOT_FOUND ) {
display = wxDisplay ( 0u ) . GetClientArea ( ) ;
window - > Move ( display . GetTopLeft ( ) ) ;
} else {
display = wxDisplay ( display_idx ) . GetClientArea ( ) ;
}
auto metrics = WindowMetrics : : from_window ( window ) ;
metrics . sanitize_for_display ( display ) ;
if ( window - > GetScreenRect ( ) ! = metrics . get_rect ( ) ) {
window - > SetSize ( metrics . get_rect ( ) ) ;
}
}
2018-09-20 06:40:22 +00:00
// static method accepting a wxWindow object as first parameter
// void warning_catcher{
// my($self, $message_dialog) = @_;
// return sub{
// my $message = shift;
// return if $message = ~/ GLUquadricObjPtr | Attempt to free unreferenced scalar / ;
// my @params = ($message, 'Warning', wxOK | wxICON_WARNING);
// $message_dialog
// ? $message_dialog->(@params)
// : Wx::MessageDialog->new($self, @params)->ShowModal;
// };
// }
// Do we need this function???
2018-10-31 11:56:08 +00:00
// void GUI_App::notify(message) {
2018-09-20 06:40:22 +00:00
// auto frame = GetTopWindow();
// // try harder to attract user attention on OS X
// if (!frame->IsActive())
// frame->RequestUserAttention(defined(__WXOSX__/*&Wx::wxMAC */)? wxUSER_ATTENTION_ERROR : wxUSER_ATTENTION_INFO);
//
// // There used to be notifier using a Growl application for OSX, but Growl is dead.
// // The notifier also supported the Linux X D - bus notifications, but that support was broken.
// //TODO use wxNotificationMessage ?
// }
2019-05-03 13:50:05 +00:00
# ifdef __WXMSW__
void GUI_App : : associate_3mf_files ( )
{
// see as reference: https://stackoverflow.com/questions/20245262/c-program-needs-an-file-association
2019-05-17 09:35:35 +00:00
auto reg_set = [ ] ( HKEY hkeyHive , const wchar_t * pszVar , const wchar_t * pszValue ) - > bool
2019-05-03 13:50:05 +00:00
{
wchar_t szValueCurrent [ 1000 ] ;
DWORD dwType ;
DWORD dwSize = sizeof ( szValueCurrent ) ;
int iRC = : : RegGetValueW ( hkeyHive , pszVar , nullptr , RRF_RT_ANY , & dwType , szValueCurrent , & dwSize ) ;
bool bDidntExist = iRC = = ERROR_FILE_NOT_FOUND ;
if ( ( iRC ! = ERROR_SUCCESS ) & & ! bDidntExist )
// an error occurred
2019-05-17 09:35:35 +00:00
return false ;
2019-05-03 13:50:05 +00:00
if ( ! bDidntExist )
{
if ( dwType ! = REG_SZ )
// invalid type
2019-05-17 09:35:35 +00:00
return false ;
2019-05-03 13:50:05 +00:00
if ( : : wcscmp ( szValueCurrent , pszValue ) = = 0 )
// value already set
2019-05-17 09:35:35 +00:00
return false ;
2019-05-03 13:50:05 +00:00
}
DWORD dwDisposition ;
HKEY hkey ;
iRC = : : RegCreateKeyExW ( hkeyHive , pszVar , 0 , 0 , 0 , KEY_ALL_ACCESS , nullptr , & hkey , & dwDisposition ) ;
2019-05-17 09:35:35 +00:00
bool ret = false ;
2019-05-03 13:50:05 +00:00
if ( iRC = = ERROR_SUCCESS )
2019-05-17 09:35:35 +00:00
{
2019-05-03 13:50:05 +00:00
iRC = : : RegSetValueExW ( hkey , L " " , 0 , REG_SZ , ( BYTE * ) pszValue , ( : : wcslen ( pszValue ) + 1 ) * sizeof ( wchar_t ) ) ;
2019-05-17 09:35:35 +00:00
if ( iRC = = ERROR_SUCCESS )
ret = true ;
}
2019-05-03 13:50:05 +00:00
RegCloseKey ( hkey ) ;
2019-05-17 09:35:35 +00:00
return ret ;
2019-05-03 13:50:05 +00:00
} ;
wchar_t app_path [ MAX_PATH ] ;
: : GetModuleFileNameW ( nullptr , app_path , sizeof ( app_path ) ) ;
std : : wstring prog_path = L " \" " + std : : wstring ( app_path ) + L " \" " ;
std : : wstring prog_id = L " Prusa.Slicer.1 " ;
std : : wstring prog_desc = L " PrusaSlicer " ;
std : : wstring prog_command = prog_path + L " \" %1 \" " ;
std : : wstring reg_base = L " Software \\ Classes " ;
std : : wstring reg_extension = reg_base + L " \\ .3mf " ;
std : : wstring reg_prog_id = reg_base + L " \\ " + prog_id ;
std : : wstring reg_prog_id_command = reg_prog_id + L " \\ Shell \\ Open \\ Command " ;
2019-05-17 09:35:35 +00:00
bool is_new = false ;
is_new | = reg_set ( HKEY_CURRENT_USER , reg_extension . c_str ( ) , prog_id . c_str ( ) ) ;
is_new | = reg_set ( HKEY_CURRENT_USER , reg_prog_id . c_str ( ) , prog_desc . c_str ( ) ) ;
is_new | = reg_set ( HKEY_CURRENT_USER , reg_prog_id_command . c_str ( ) , prog_command . c_str ( ) ) ;
2019-05-03 13:50:05 +00:00
2019-05-17 09:35:35 +00:00
if ( is_new )
// notify Windows only when any of the values gets changed
: : SHChangeNotify ( SHCNE_ASSOCCHANGED , SHCNF_IDLIST , nullptr , nullptr ) ;
2019-05-03 13:50:05 +00:00
}
# endif // __WXMSW__
2018-09-20 06:40:22 +00:00
} // GUI
2018-11-12 17:09:47 +00:00
} //Slic3r