2015-12-04 20:25:45 +00:00
|
|
|
#include "GUI.hpp"
|
|
|
|
|
2017-11-02 15:21:34 +00:00
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
2017-11-02 15:29:03 +00:00
|
|
|
#include <boost/filesystem.hpp>
|
2018-01-07 17:41:40 +00:00
|
|
|
#include <boost/lexical_cast.hpp>
|
2017-11-02 15:21:34 +00:00
|
|
|
|
2015-12-04 20:25:45 +00:00
|
|
|
#if __APPLE__
|
|
|
|
#import <IOKit/pwr_mgt/IOPMLib.h>
|
|
|
|
#elif _WIN32
|
|
|
|
#include <Windows.h>
|
2017-12-13 13:45:10 +00:00
|
|
|
// 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
|
2017-11-02 15:21:34 +00:00
|
|
|
#include "boost/nowide/convert.hpp"
|
2015-12-04 20:25:45 +00:00
|
|
|
#pragma comment(lib, "user32.lib")
|
|
|
|
#endif
|
|
|
|
|
2017-12-04 09:48:40 +00:00
|
|
|
#include <wx/app.h>
|
|
|
|
#include <wx/button.h>
|
|
|
|
#include <wx/frame.h>
|
|
|
|
#include <wx/menu.h>
|
|
|
|
#include <wx/notebook.h>
|
|
|
|
#include <wx/panel.h>
|
|
|
|
#include <wx/sizer.h>
|
2017-12-13 13:45:10 +00:00
|
|
|
#include <wx/window.h>
|
2017-12-04 09:48:40 +00:00
|
|
|
|
2018-01-23 10:42:04 +00:00
|
|
|
#include "Tab.hpp"
|
2018-01-23 10:37:19 +00:00
|
|
|
#include "TabIface.hpp"
|
2018-01-05 14:11:33 +00:00
|
|
|
#include "AppConfig.hpp"
|
2017-12-05 14:54:01 +00:00
|
|
|
|
2015-12-04 20:25:45 +00:00
|
|
|
namespace Slic3r { namespace GUI {
|
|
|
|
|
2015-12-05 10:48:24 +00:00
|
|
|
#if __APPLE__
|
2015-12-04 20:25:45 +00:00
|
|
|
IOPMAssertionID assertionID;
|
2015-12-05 10:48:24 +00:00
|
|
|
#endif
|
2015-12-04 20:25:45 +00:00
|
|
|
|
2017-10-25 10:53:31 +00:00
|
|
|
void disable_screensaver()
|
2015-12-04 20:25:45 +00:00
|
|
|
{
|
|
|
|
#if __APPLE__
|
|
|
|
CFStringRef reasonForActivity = CFSTR("Slic3r");
|
|
|
|
IOReturn success = IOPMAssertionCreateWithName(kIOPMAssertionTypeNoDisplaySleep,
|
|
|
|
kIOPMAssertionLevelOn, reasonForActivity, &assertionID);
|
|
|
|
// ignore result: success == kIOReturnSuccess
|
|
|
|
#elif _WIN32
|
2015-12-06 10:17:50 +00:00
|
|
|
SetThreadExecutionState(ES_DISPLAY_REQUIRED | ES_CONTINUOUS);
|
2015-12-04 20:25:45 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-10-25 10:53:31 +00:00
|
|
|
void enable_screensaver()
|
2015-12-04 20:25:45 +00:00
|
|
|
{
|
|
|
|
#if __APPLE__
|
|
|
|
IOReturn success = IOPMAssertionRelease(assertionID);
|
|
|
|
#elif _WIN32
|
2015-12-06 10:17:50 +00:00
|
|
|
SetThreadExecutionState(ES_CONTINUOUS);
|
2015-12-04 20:25:45 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-11-02 15:21:34 +00:00
|
|
|
std::vector<std::string> scan_serial_ports()
|
|
|
|
{
|
|
|
|
std::vector<std::string> out;
|
|
|
|
#ifdef _WIN32
|
|
|
|
// 1) Open the registry key SERIALCOM.
|
|
|
|
HKEY hKey;
|
|
|
|
LONG lRes = ::RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"HARDWARE\\DEVICEMAP\\SERIALCOMM", 0, KEY_READ, &hKey);
|
|
|
|
assert(lRes == ERROR_SUCCESS);
|
|
|
|
if (lRes == ERROR_SUCCESS) {
|
|
|
|
// 2) Get number of values of SERIALCOM key.
|
|
|
|
DWORD cValues; // number of values for key
|
|
|
|
{
|
|
|
|
TCHAR achKey[255]; // buffer for subkey name
|
|
|
|
DWORD cbName; // size of name string
|
|
|
|
TCHAR achClass[MAX_PATH] = TEXT(""); // buffer for class name
|
|
|
|
DWORD cchClassName = MAX_PATH; // size of class string
|
|
|
|
DWORD cSubKeys=0; // number of subkeys
|
|
|
|
DWORD cbMaxSubKey; // longest subkey size
|
|
|
|
DWORD cchMaxClass; // longest class string
|
|
|
|
DWORD cchMaxValue; // longest value name
|
|
|
|
DWORD cbMaxValueData; // longest value data
|
|
|
|
DWORD cbSecurityDescriptor; // size of security descriptor
|
|
|
|
FILETIME ftLastWriteTime; // last write time
|
|
|
|
// Get the class name and the value count.
|
|
|
|
lRes = RegQueryInfoKey(
|
|
|
|
hKey, // key handle
|
|
|
|
achClass, // buffer for class name
|
|
|
|
&cchClassName, // size of class string
|
|
|
|
NULL, // reserved
|
|
|
|
&cSubKeys, // number of subkeys
|
|
|
|
&cbMaxSubKey, // longest subkey size
|
|
|
|
&cchMaxClass, // longest class string
|
|
|
|
&cValues, // number of values for this key
|
|
|
|
&cchMaxValue, // longest value name
|
|
|
|
&cbMaxValueData, // longest value data
|
|
|
|
&cbSecurityDescriptor, // security descriptor
|
|
|
|
&ftLastWriteTime); // last write time
|
|
|
|
assert(lRes == ERROR_SUCCESS);
|
|
|
|
}
|
|
|
|
// 3) Read the SERIALCOM values.
|
|
|
|
{
|
|
|
|
DWORD dwIndex = 0;
|
|
|
|
for (int i = 0; i < cValues; ++ i, ++ dwIndex) {
|
|
|
|
wchar_t valueName[2048];
|
|
|
|
DWORD valNameLen = 2048;
|
|
|
|
DWORD dataType;
|
|
|
|
wchar_t data[2048];
|
|
|
|
DWORD dataSize = 4096;
|
|
|
|
lRes = ::RegEnumValueW(hKey, dwIndex, valueName, &valNameLen, nullptr, &dataType, (BYTE*)&data, &dataSize);
|
|
|
|
if (lRes == ERROR_SUCCESS && dataType == REG_SZ && valueName[0] != 0)
|
|
|
|
out.emplace_back(boost::nowide::narrow(data));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
::RegCloseKey(hKey);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// UNIX and OS X
|
|
|
|
std::initializer_list<const char*> prefixes { "ttyUSB" , "ttyACM", "tty.", "cu.", "rfcomm" };
|
2017-11-02 15:40:25 +00:00
|
|
|
for (auto &dir_entry : boost::filesystem::directory_iterator(boost::filesystem::path("/dev"))) {
|
|
|
|
std::string name = dir_entry.path().filename().string();
|
2017-11-02 15:35:46 +00:00
|
|
|
for (const char *prefix : prefixes) {
|
|
|
|
if (boost::starts_with(name, prefix)) {
|
2017-11-02 15:40:25 +00:00
|
|
|
out.emplace_back(dir_entry.path().string());
|
2017-11-02 15:21:34 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-11-02 15:35:46 +00:00
|
|
|
}
|
|
|
|
}
|
2017-11-02 15:21:34 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
out.erase(std::remove_if(out.begin(), out.end(),
|
|
|
|
[](const std::string &key){
|
|
|
|
return boost::starts_with(key, "Bluetooth") || boost::starts_with(key, "FireFly");
|
|
|
|
}),
|
|
|
|
out.end());
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2017-10-25 10:53:31 +00:00
|
|
|
bool debugged()
|
2016-04-13 18:45:44 +00:00
|
|
|
{
|
2016-04-13 18:51:03 +00:00
|
|
|
#ifdef _WIN32
|
2016-04-13 18:45:44 +00:00
|
|
|
return IsDebuggerPresent();
|
2016-04-13 18:51:03 +00:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif /* _WIN32 */
|
2016-04-13 18:45:44 +00:00
|
|
|
}
|
|
|
|
|
2017-10-25 10:53:31 +00:00
|
|
|
void break_to_debugger()
|
2016-04-13 18:45:44 +00:00
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
if (IsDebuggerPresent())
|
|
|
|
DebugBreak();
|
|
|
|
#endif /* _WIN32 */
|
|
|
|
}
|
|
|
|
|
2017-12-04 09:48:40 +00:00
|
|
|
// Passing the wxWidgets GUI classes instantiated by the Perl part to C++.
|
|
|
|
wxApp *g_wxApp = nullptr;
|
|
|
|
wxFrame *g_wxMainFrame = nullptr;
|
|
|
|
wxNotebook *g_wxTabPanel = nullptr;
|
|
|
|
|
|
|
|
void set_wxapp(wxApp *app)
|
|
|
|
{
|
|
|
|
g_wxApp = app;
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_main_frame(wxFrame *main_frame)
|
|
|
|
{
|
|
|
|
g_wxMainFrame = main_frame;
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_tab_panel(wxNotebook *tab_panel)
|
|
|
|
{
|
|
|
|
g_wxTabPanel = tab_panel;
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_debug_menu(wxMenuBar *menu)
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
auto debug_menu = new wxMenu();
|
|
|
|
debug_menu->Append(wxWindow::NewControlId(1), "Some debug");
|
|
|
|
menu->Append(debug_menu, _T("&Debug"));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-01-26 00:44:34 +00:00
|
|
|
void create_preset_tabs(PresetBundle *preset_bundle, AppConfig *app_config,
|
|
|
|
bool no_controller, bool is_disabled_button_browse, bool is_user_agent,
|
|
|
|
int event_value_change, int event_presets_changed,
|
|
|
|
int event_button_browse, int event_button_test)
|
2018-01-05 14:11:33 +00:00
|
|
|
{
|
2018-01-26 00:44:34 +00:00
|
|
|
add_created_tab(new TabPrint (g_wxTabPanel, no_controller), preset_bundle, app_config);
|
|
|
|
add_created_tab(new TabFilament (g_wxTabPanel, no_controller), preset_bundle, app_config);
|
|
|
|
add_created_tab(new TabPrinter (g_wxTabPanel, no_controller, is_disabled_button_browse, is_user_agent),
|
|
|
|
preset_bundle, app_config);
|
2018-01-12 11:41:13 +00:00
|
|
|
g_wxTabPanel->Bind(wxEVT_NOTEBOOK_PAGE_CHANGED, ([](wxCommandEvent e){
|
|
|
|
Tab* panel = (Tab*)g_wxTabPanel->GetCurrentPage();
|
2018-01-25 20:44:22 +00:00
|
|
|
if (panel->GetName().compare("Print Settings")==0 ||
|
|
|
|
panel->GetName().compare("Filament Settings") == 0)
|
2018-01-12 11:41:13 +00:00
|
|
|
panel->OnActivate();
|
|
|
|
}), g_wxTabPanel->GetId() );
|
2018-01-21 22:35:00 +00:00
|
|
|
for (size_t i = 0; i < g_wxTabPanel->GetPageCount(); ++ i) {
|
|
|
|
Tab *tab = dynamic_cast<Tab*>(g_wxTabPanel->GetPage(i));
|
|
|
|
if (! tab)
|
|
|
|
continue;
|
|
|
|
tab->set_event_value_change(wxEventType(event_value_change));
|
|
|
|
tab->set_event_presets_changed(wxEventType(event_presets_changed));
|
2018-01-26 00:44:34 +00:00
|
|
|
if (tab->name() == "printer"){
|
|
|
|
TabPrinter* tab_printer = static_cast<TabPrinter*>(tab);
|
|
|
|
tab_printer->set_event_button_browse(wxEventType(event_button_browse));
|
|
|
|
tab_printer->set_event_button_test(wxEventType(event_button_test));
|
|
|
|
}
|
2018-01-21 22:35:00 +00:00
|
|
|
}
|
2017-12-22 10:50:28 +00:00
|
|
|
}
|
|
|
|
|
2018-01-23 10:37:19 +00:00
|
|
|
TabIface* get_preset_tab_iface(char *name)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < g_wxTabPanel->GetPageCount(); ++ i) {
|
|
|
|
Tab *tab = dynamic_cast<Tab*>(g_wxTabPanel->GetPage(i));
|
|
|
|
if (! tab)
|
|
|
|
continue;
|
2018-01-25 20:44:22 +00:00
|
|
|
if (tab->name() == name) {
|
2018-01-23 10:37:19 +00:00
|
|
|
return new TabIface(tab);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new TabIface(nullptr);
|
|
|
|
}
|
|
|
|
|
2018-01-07 17:41:40 +00:00
|
|
|
void change_opt_value(DynamicPrintConfig& config, t_config_option_key opt_key, boost::any value)
|
|
|
|
{
|
|
|
|
try{
|
|
|
|
switch (config.def()->get(opt_key)->type){
|
2018-01-09 12:52:01 +00:00
|
|
|
case coFloatOrPercent:{
|
|
|
|
const auto &val = *config.option<ConfigOptionFloatOrPercent>(opt_key);
|
2018-01-11 09:33:17 +00:00
|
|
|
config.set_key_value(opt_key, new ConfigOptionFloatOrPercent(boost::any_cast<double>(value), val.percent));
|
2018-01-09 12:52:01 +00:00
|
|
|
break;}
|
2018-01-07 17:41:40 +00:00
|
|
|
case coPercent:
|
2018-01-11 09:33:17 +00:00
|
|
|
config.set_key_value(opt_key, new ConfigOptionPercent(boost::any_cast<double>(value)));
|
2018-01-09 12:52:01 +00:00
|
|
|
break;
|
|
|
|
case coFloat:{
|
2018-01-07 17:41:40 +00:00
|
|
|
double& val = config.opt_float(opt_key);
|
2018-01-09 08:41:07 +00:00
|
|
|
val = boost::any_cast<double>(value);
|
2018-01-07 17:41:40 +00:00
|
|
|
break;
|
|
|
|
}
|
2018-01-09 12:52:01 +00:00
|
|
|
case coPercents:
|
|
|
|
case coFloats:{
|
|
|
|
double& val = config.opt_float(opt_key, 0);
|
|
|
|
val = boost::any_cast<double>(value);
|
|
|
|
break;
|
|
|
|
}
|
2018-01-07 17:41:40 +00:00
|
|
|
case coString:
|
2018-01-09 12:52:01 +00:00
|
|
|
config.set_key_value(opt_key, new ConfigOptionString(boost::any_cast<std::string>(value)));
|
2018-01-07 17:41:40 +00:00
|
|
|
break;
|
2018-01-12 11:41:13 +00:00
|
|
|
case coStrings:{
|
|
|
|
if (opt_key.compare("compatible_printers") == 0){
|
2018-01-25 12:46:04 +00:00
|
|
|
config.option<ConfigOptionStrings>(opt_key)->values.resize(0);
|
|
|
|
for (auto el : boost::any_cast<std::vector<std::string>>(value))
|
|
|
|
config.option<ConfigOptionStrings>(opt_key)->values.push_back(el);
|
2018-01-12 11:41:13 +00:00
|
|
|
}
|
|
|
|
else{
|
2018-01-25 12:46:04 +00:00
|
|
|
ConfigOptionStrings* vec_new = new ConfigOptionStrings{ boost::any_cast<std::string>(value) };
|
|
|
|
config.option<ConfigOptionStrings>(opt_key)->set_at(vec_new, 0, 0);
|
2018-01-12 11:41:13 +00:00
|
|
|
}
|
|
|
|
}
|
2018-01-07 17:41:40 +00:00
|
|
|
break;
|
|
|
|
case coBool:
|
|
|
|
config.set_key_value(opt_key, new ConfigOptionBool(boost::any_cast<bool>(value)));
|
|
|
|
break;
|
2018-01-09 12:52:01 +00:00
|
|
|
case coBools:{
|
|
|
|
ConfigOptionBools* vec_new = new ConfigOptionBools{ boost::any_cast<bool>(value) };
|
|
|
|
config.option<ConfigOptionBools>(opt_key)->set_at(vec_new, 0, 0);
|
|
|
|
break;}
|
2018-01-07 17:41:40 +00:00
|
|
|
case coInt:
|
|
|
|
config.set_key_value(opt_key, new ConfigOptionInt(boost::any_cast<int>(value)));
|
|
|
|
break;
|
2018-01-12 11:41:13 +00:00
|
|
|
case coInts:{
|
|
|
|
ConfigOptionInts* vec_new = new ConfigOptionInts{ boost::any_cast<int>(value) };
|
|
|
|
config.option<ConfigOptionInts>(opt_key)->set_at(vec_new, 0, 0);
|
|
|
|
}
|
2018-01-07 17:41:40 +00:00
|
|
|
break;
|
2018-01-11 09:33:17 +00:00
|
|
|
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)));
|
|
|
|
}
|
2018-01-07 17:41:40 +00:00
|
|
|
break;
|
2018-01-30 11:10:12 +00:00
|
|
|
case coPoints:{
|
|
|
|
ConfigOptionPoints points;
|
|
|
|
points.values = boost::any_cast<std::vector<Pointf>>(value);
|
|
|
|
config.set_key_value(opt_key, new ConfigOptionPoints(points));
|
|
|
|
}
|
2018-01-07 17:41:40 +00:00
|
|
|
break;
|
|
|
|
case coNone:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (const std::exception &e)
|
|
|
|
{
|
2018-01-09 12:52:01 +00:00
|
|
|
int i = 0;//no reason, just experiment
|
2018-01-07 17:41:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-26 00:44:34 +00:00
|
|
|
void add_created_tab(Tab* panel, PresetBundle *preset_bundle, AppConfig *app_config)
|
2017-12-22 10:50:28 +00:00
|
|
|
{
|
2018-01-16 15:28:01 +00:00
|
|
|
panel->m_show_btn_incompatible_presets = app_config->get("show_incompatible_presets").empty();
|
2018-01-05 14:11:33 +00:00
|
|
|
panel->create_preset_tab(preset_bundle);
|
|
|
|
|
2018-01-16 15:28:01 +00:00
|
|
|
// Load the currently selected preset into the GUI, update the preset selection box.
|
|
|
|
panel->load_current_preset();
|
2017-12-22 10:50:28 +00:00
|
|
|
g_wxTabPanel->AddPage(panel, panel->title());
|
|
|
|
}
|
|
|
|
|
|
|
|
void show_error(wxWindow* parent, std::string message){
|
|
|
|
auto msg_wingow = new wxMessageDialog(parent, message, "Error", wxOK | wxICON_ERROR);
|
|
|
|
msg_wingow->ShowModal();
|
|
|
|
}
|
|
|
|
|
|
|
|
void show_info(wxWindow* parent, std::string message, std::string title){
|
2018-01-26 00:44:34 +00:00
|
|
|
auto msg_wingow = new wxMessageDialog(parent, message, title.empty() ? "Notice" : title, wxOK | wxICON_INFORMATION);
|
2017-12-22 10:50:28 +00:00
|
|
|
msg_wingow->ShowModal();
|
2017-12-04 09:48:40 +00:00
|
|
|
}
|
|
|
|
|
2018-02-05 15:12:16 +00:00
|
|
|
wxApp* get_app(){
|
|
|
|
return g_wxApp;
|
|
|
|
}
|
|
|
|
|
2015-12-04 20:25:45 +00:00
|
|
|
} }
|