123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506 |
- package Slic3r::GUI;
- use strict;
- use warnings;
- use utf8;
- use Wx 0.9901 qw(:bitmap :dialog :icon :id :misc :systemsettings :toplevelwindow
- :filedialog :font);
- use Wx::Event qw(EVT_MENU);
- BEGIN {
- # Wrap the Wx::_load_plugin() function which doesn't work with non-ASCII paths
- no warnings 'redefine';
- my $orig = *Wx::_load_plugin{CODE};
- *Wx::_load_plugin = sub {
- $_[0] = Slic3r::decode_path($_[0]);
- $orig->(@_);
- };
- }
- use File::Basename qw(basename);
- use FindBin;
- use List::Util qw(first any);
- use Slic3r::Geometry qw(X Y);
- use Slic3r::GUI::2DBed;
- use Slic3r::GUI::AboutDialog;
- use Slic3r::GUI::BedShapeDialog;
- use Slic3r::GUI::BonjourBrowser;
- use Slic3r::GUI::ConfigWizard;
- use Slic3r::GUI::Controller;
- use Slic3r::GUI::Controller::ManualControlDialog;
- use Slic3r::GUI::Controller::PrinterPanel;
- use Slic3r::GUI::MainFrame;
- use Slic3r::GUI::Notifier;
- use Slic3r::GUI::Plater;
- use Slic3r::GUI::Plater::2D;
- use Slic3r::GUI::Plater::2DToolpaths;
- use Slic3r::GUI::Plater::3D;
- use Slic3r::GUI::Plater::3DPreview;
- use Slic3r::GUI::Plater::ObjectPartsPanel;
- use Slic3r::GUI::Plater::ObjectCutDialog;
- use Slic3r::GUI::Plater::ObjectSettingsDialog;
- use Slic3r::GUI::Plater::LambdaObjectDialog;
- use Slic3r::GUI::Plater::OverrideSettingsPanel;
- use Slic3r::GUI::Plater::SplineControl;
- use Slic3r::GUI::Preferences;
- use Slic3r::GUI::ProgressStatusBar;
- use Slic3r::GUI::Projector;
- use Slic3r::GUI::OptionsGroup;
- use Slic3r::GUI::OptionsGroup::Field;
- use Slic3r::GUI::Preset;
- use Slic3r::GUI::PresetEditor;
- use Slic3r::GUI::PresetEditorDialog;
- use Slic3r::GUI::SLAPrintOptions;
- our $have_OpenGL = eval "use Slic3r::GUI::3DScene; 1";
- our $have_LWP = eval "use LWP::UserAgent; 1";
- use Wx::Event qw(EVT_IDLE EVT_COMMAND);
- use base 'Wx::App';
- use constant FILE_WILDCARDS => {
- known => 'Known files (*.stl, *.obj, *.amf, *.xml, *.3mf)|*.3mf;*.3MF;*.stl;*.STL;*.obj;*.OBJ;*.amf;*.AMF;*.xml;*.XML',
- stl => 'STL files (*.stl)|*.stl;*.STL',
- obj => 'OBJ files (*.obj)|*.obj;*.OBJ',
- amf => 'AMF files (*.amf)|*.amf;*.AMF;*.xml;*.XML',
- tmf => '3MF files (*.3mf)|*.3mf;*.3MF',
- ini => 'INI files *.ini|*.ini;*.INI',
- gcode => 'G-code files (*.gcode, *.gco, *.g, *.ngc)|*.gcode;*.GCODE;*.gco;*.GCO;*.g;*.G;*.ngc;*.NGC',
- svg => 'SVG files *.svg|*.svg;*.SVG',
- };
- use constant MODEL_WILDCARD => join '|', @{&FILE_WILDCARDS}{qw(known stl obj amf tmf)};
- use constant STL_MODEL_WILDCARD => join '|', @{&FILE_WILDCARDS}{qw(stl)};
- use constant AMF_MODEL_WILDCARD => join '|', @{&FILE_WILDCARDS}{qw(amf)};
- use constant TMF_MODEL_WILDCARD => join '|', @{&FILE_WILDCARDS}{qw(tmf)};
- our $datadir;
- # If set, the "Controller" tab for the control of the printer over serial line and the serial port settings are hidden.
- our $autosave;
- our $threads;
- our @cb;
- our $Settings = {
- _ => {
- version_check => 1,
- autocenter => 1,
- autoalignz => 1,
- invert_zoom => 0,
- background_processing => 0,
- threads => $Slic3r::Config::Options->{threads}{default},
- color_toolpaths_by => 'role',
- tabbed_preset_editors => 1,
- show_host => 0,
- nudge_val => 1
- },
- };
- our $have_button_icons = &Wx::wxVERSION_STRING =~ / (?:2\.9\.[1-9]|3\.)/;
- our $small_font = Wx::SystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
- $small_font->SetPointSize(11) if &Wx::wxMAC;
- our $small_bold_font = Wx::SystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
- $small_bold_font->SetPointSize(11) if &Wx::wxMAC;
- $small_bold_font->SetWeight(wxFONTWEIGHT_BOLD);
- our $medium_font = Wx::SystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT);
- $medium_font->SetPointSize(12);
- our $grey = Wx::Colour->new(200,200,200);
- # to use in ScrolledWindow::SetScrollRate(xstep, ystep)
- # step related to system font point size
- our $scroll_step = Wx::SystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT)->GetPointSize;
- our $VERSION_CHECK_EVENT : shared = Wx::NewEventType;
- our $DLP_projection_screen;
- sub OnInit {
- my ($self) = @_;
-
- $self->SetAppName('Slic3r');
- Slic3r::debugf "wxWidgets version %s, Wx version %s\n", &Wx::wxVERSION_STRING, $Wx::VERSION;
-
- $self->{notifier} = Slic3r::GUI::Notifier->new;
- $self->{presets} = { print => [], filament => [], printer => [] };
-
- # locate or create data directory
- # Unix: ~/.Slic3r
- # Windows: "C:\Users\username\AppData\Roaming\Slic3r" or "C:\Documents and Settings\username\Application Data\Slic3r"
- # Mac: "~/Library/Application Support/Slic3r"
- $datadir ||= Slic3r::decode_path(Wx::StandardPaths::Get->GetUserDataDir);
- my $enc_datadir = Slic3r::encode_path($datadir);
- Slic3r::debugf "Data directory: %s\n", $datadir;
-
- # just checking for existence of $datadir is not enough: it may be an empty directory
- # supplied as argument to --datadir; in that case we should still run the wizard
- my $run_wizard = (-d $enc_datadir && -e "$enc_datadir/slic3r.ini") ? 0 : 1;
- foreach my $dir ($enc_datadir, "$enc_datadir/print", "$enc_datadir/filament", "$enc_datadir/printer") {
- next if -d $dir;
- if (!mkdir $dir) {
- my $error = "Slic3r was unable to create its data directory at $dir ($!).";
- warn "$error\n";
- fatal_error(undef, $error);
- }
- }
-
- # load settings
- my $last_version;
- if (-f "$enc_datadir/slic3r.ini") {
- my $ini = eval { Slic3r::Config->read_ini("$datadir/slic3r.ini") };
- if ($ini) {
- $last_version = $ini->{_}{version};
- $ini->{_}{$_} = $Settings->{_}{$_}
- for grep !exists $ini->{_}{$_}, keys %{$Settings->{_}};
- $Settings = $ini;
- }
- delete $Settings->{_}{mode}; # handle legacy
- }
- $Settings->{_}{version} = $Slic3r::VERSION;
- $Settings->{_}{threads} = $threads if $threads;
- $self->save_settings;
-
- if (-f "$enc_datadir/simple.ini") {
- # The Simple Mode settings were already automatically duplicated to presets
- # named "Simple Mode" in each group, so we already support retrocompatibility.
- unlink "$enc_datadir/simple.ini";
- }
-
- $self->load_presets;
-
- # application frame
- Wx::Image::AddHandler(Wx::PNGHandler->new);
- $self->{mainframe} = my $frame = Slic3r::GUI::MainFrame->new;
- $self->SetTopWindow($frame);
-
- # load init bundle
- {
- my @dirs = ($FindBin::Bin);
- if (&Wx::wxMAC) {
- push @dirs, qw();
- } elsif (&Wx::wxMSW) {
- push @dirs, qw();
- }
- my $init_bundle = first { -e $_ } map "$_/.init_bundle.ini", @dirs;
- if ($init_bundle) {
- Slic3r::debugf "Loading config bundle from %s\n", $init_bundle;
- $self->{mainframe}->load_configbundle($init_bundle, 1);
- $run_wizard = 0;
- }
- }
-
- if (!$run_wizard && (!defined $last_version || $last_version ne $Slic3r::VERSION)) {
- # user was running another Slic3r version on this computer
- if (!defined $last_version || $last_version =~ /^0\./) {
- show_info($self->{mainframe}, "Hello! Support material was improved since the "
- . "last version of Slic3r you used. It is strongly recommended to revert "
- . "your support material settings to the factory defaults and start from "
- . "those. Enjoy and provide feedback!", "Support Material");
- }
- if (!defined $last_version || $last_version =~ /^(?:0|1\.[01])\./) {
- show_info($self->{mainframe}, "Hello! In this version a new Bed Shape option was "
- . "added. If the bed coordinates in the plater preview screen look wrong, go "
- . "to Print Settings and click the \"Set\" button next to \"Bed Shape\".", "Bed Shape");
- }
- }
- $self->{mainframe}->config_wizard if $run_wizard;
-
- $self->check_version
- if $self->have_version_check
- && ($Settings->{_}{version_check} // 1)
- && (!$Settings->{_}{last_version_check} || (time - $Settings->{_}{last_version_check}) >= 86400);
-
- EVT_IDLE($frame, sub {
- while (my $cb = shift @cb) {
- $cb->();
- }
- });
-
- EVT_COMMAND($self, -1, $VERSION_CHECK_EVENT, sub {
- my ($self, $event) = @_;
- my ($success, $response, $manual_check) = @{$event->GetData};
-
- if ($success) {
- if ($response =~ /^obsolete ?= ?([a-z0-9.-]+,)*\Q$Slic3r::VERSION\E(?:,|$)/) {
- my $res = Wx::MessageDialog->new(undef, "A new version is available. Do you want to open the Slic3r website now?",
- 'Update', wxYES_NO | wxCANCEL | wxYES_DEFAULT | wxICON_INFORMATION | wxICON_ERROR)->ShowModal;
- Wx::LaunchDefaultBrowser('http://slic3r.org/') if $res == wxID_YES;
- } else {
- Slic3r::GUI::show_info(undef, "You're using the latest version. No updates are available.") if $manual_check;
- }
- $Settings->{_}{last_version_check} = time();
- $self->save_settings;
- } else {
- Slic3r::GUI::show_error(undef, "Failed to check for updates. Try later.") if $manual_check;
- }
- });
-
- return 1;
- }
- sub about {
- my ($self) = @_;
-
- my $about = Slic3r::GUI::AboutDialog->new(undef);
- $about->ShowModal;
- $about->Destroy;
- }
- # static method accepting a wxWindow object as first parameter
- sub catch_error {
- my ($self, $cb, $message_dialog) = @_;
- if (my $err = $@) {
- $cb->() if $cb;
- $message_dialog
- ? $message_dialog->($err, 'Error', wxOK | wxICON_ERROR)
- : Slic3r::GUI::show_error($self, $err);
- return 1;
- }
- return 0;
- }
- # static method accepting a wxWindow object as first parameter
- sub show_error {
- my ($parent, $message) = @_;
- Wx::MessageDialog->new($parent, $message, 'Error', wxOK | wxICON_ERROR)->ShowModal;
- }
- # static method accepting a wxWindow object as first parameter
- sub show_info {
- my ($parent, $message, $title) = @_;
- Wx::MessageDialog->new($parent, $message, $title || 'Notice', wxOK | wxICON_INFORMATION)->ShowModal;
- }
- # static method accepting a wxWindow object as first parameter
- sub fatal_error {
- show_error(@_);
- exit 1;
- }
- # static method accepting a wxWindow object as first parameter
- sub 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;
- };
- }
- sub notify {
- my ($self, $message) = @_;
- my $frame = $self->GetTopWindow;
- # try harder to attract user attention on OS X
- $frame->RequestUserAttention(&Wx::wxMAC ? wxUSER_ATTENTION_ERROR : wxUSER_ATTENTION_INFO)
- unless ($frame->IsActive);
- $self->{notifier}->notify($message);
- }
- sub save_settings {
- my ($self) = @_;
- Slic3r::Config->write_ini("$datadir/slic3r.ini", $Settings);
- }
- sub presets { return $_[0]->{presets}; }
- sub load_presets {
- my ($self) = @_;
-
- for my $group (qw(printer filament print)) {
- my $presets = $self->{presets}{$group};
-
- # keep external or dirty presets
- @$presets = grep { ($_->external && $_->file_exists) || $_->dirty } @$presets;
-
- my $dir = "$Slic3r::GUI::datadir/$group";
- opendir my $dh, Slic3r::encode_path($dir)
- or die "Failed to read directory $dir (errno: $!)\n";
- foreach my $file (grep /\.ini$/i, readdir $dh) {
- $file = Slic3r::decode_path($file);
- my $name = basename($file);
- $name =~ s/\.ini$//i;
-
- # skip if we already have it
- next if any { $_->name eq $name } @$presets;
-
- push @$presets, Slic3r::GUI::Preset->new(
- group => $group,
- name => $name,
- file => "$dir/$file",
- );
- }
- closedir $dh;
-
- @$presets = sort { $a->name cmp $b->name } @$presets;
-
- unshift @$presets, Slic3r::GUI::Preset->new(
- group => $group,
- default => 1,
- name => '- default -',
- );
- }
- }
- sub add_external_preset {
- my ($self, $file) = @_;
-
- my $name = basename($file); # keep .ini suffix
- for my $group (qw(printer filament print)) {
- my $presets = $self->{presets}{$group};
-
- # remove any existing preset with the same name
- @$presets = grep { $_->name ne $name } @$presets;
-
- push @$presets, Slic3r::GUI::Preset->new(
- group => $group,
- name => $name,
- file => $file,
- external => 1,
- );
- }
- return $name;
- }
- sub have_version_check {
- my ($self) = @_;
-
- # return an explicit 0
- return ($Slic3r::have_threads && $Slic3r::VERSION !~ /-dev$/ && $have_LWP) || 0;
- }
- sub check_version {
- my ($self, $manual_check) = @_;
-
- Slic3r::debugf "Checking for updates...\n";
-
- @_ = ();
- threads->create(sub {
- my $ua = LWP::UserAgent->new;
- $ua->timeout(10);
- my $response = $ua->get('http://slic3r.org/updatecheck');
- Wx::PostEvent($self, Wx::PlThreadEvent->new(-1, $VERSION_CHECK_EVENT,
- threads::shared::shared_clone([ $response->is_success, $response->decoded_content, $manual_check ])));
-
- Slic3r::thread_cleanup();
- })->detach;
- }
- sub output_path {
- my ($self, $dir) = @_;
-
- return ($Settings->{_}{last_output_path} && $Settings->{_}{remember_output_path})
- ? $Settings->{_}{last_output_path}
- : $dir;
- }
- sub open_model {
- my ($self, $window) = @_;
-
- my $dir = $Slic3r::GUI::Settings->{recent}{skein_directory}
- || $Slic3r::GUI::Settings->{recent}{config_directory}
- || '';
-
- my $dialog = Wx::FileDialog->new($window // $self->GetTopWindow, 'Choose one or more files (STL/OBJ/AMF/3MF):', $dir, "",
- MODEL_WILDCARD, wxFD_OPEN | wxFD_MULTIPLE | wxFD_FILE_MUST_EXIST);
- if ($dialog->ShowModal != wxID_OK) {
- $dialog->Destroy;
- return;
- }
- my @input_files = map Slic3r::decode_path($_), $dialog->GetPaths;
- $dialog->Destroy;
-
- return @input_files;
- }
- sub CallAfter {
- my ($self, $cb) = @_;
- push @cb, $cb;
- }
- sub scan_serial_ports {
- my ($self) = @_;
-
- my @ports = ();
-
- if ($^O eq 'MSWin32') {
- # Windows
- if (eval "use Win32::TieRegistry; 1") {
- my $ts = Win32::TieRegistry->new("HKEY_LOCAL_MACHINE\\HARDWARE\\DEVICEMAP\\SERIALCOMM",
- { Access => 'KEY_READ' });
- if ($ts) {
- # when no serial ports are available, the registry key doesn't exist and
- # TieRegistry->new returns undef
- $ts->Tie(\my %reg);
- push @ports, sort values %reg;
- }
- }
- } else {
- # UNIX and OS X
- push @ports, glob '/dev/{ttyUSB,ttyACM,tty.,cu.,rfcomm}*';
- }
-
- return grep !/Bluetooth|FireFly/, @ports;
- }
- sub append_menu_item {
- my ($self, $menu, $string, $description, $cb, $id, $icon, $kind) = @_;
-
- $id //= &Wx::NewId();
- my $item = Wx::MenuItem->new($menu, $id, $string, $description // '', $kind // 0);
- $self->set_menu_item_icon($item, $icon);
- $menu->Append($item);
-
- EVT_MENU($self, $id, $cb);
- return $item;
- }
- sub append_submenu {
- my ($self, $menu, $string, $description, $submenu, $id, $icon) = @_;
-
- $id //= &Wx::NewId();
- my $item = Wx::MenuItem->new($menu, $id, $string, $description // '');
- $self->set_menu_item_icon($item, $icon);
- $item->SetSubMenu($submenu);
- $menu->Append($item);
-
- return $item;
- }
- sub set_menu_item_icon {
- my ($self, $menuItem, $icon) = @_;
-
- # SetBitmap was not available on OS X before Wx 0.9927
- if ($icon && $menuItem->can('SetBitmap')) {
- $menuItem->SetBitmap(Wx::Bitmap->new($Slic3r::var->($icon), wxBITMAP_TYPE_PNG));
- }
- }
- sub save_window_pos {
- my ($self, $window, $name) = @_;
-
- $Settings->{_}{"${name}_pos"} = join ',', $window->GetScreenPositionXY;
- $Settings->{_}{"${name}_size"} = join ',', $window->GetSizeWH;
- $Settings->{_}{"${name}_maximized"} = $window->IsMaximized;
- $self->save_settings;
- }
- sub restore_window_pos {
- my ($self, $window, $name) = @_;
-
- if (defined $Settings->{_}{"${name}_pos"}) {
- my $size = [ split ',', $Settings->{_}{"${name}_size"}, 2 ];
- $window->SetSize($size);
-
- my $display = Wx::Display->new->GetClientArea();
- my $pos = [ split ',', $Settings->{_}{"${name}_pos"}, 2 ];
- if (($pos->[X] + $size->[X]/2) < $display->GetRight && ($pos->[Y] + $size->[Y]/2) < $display->GetBottom) {
- $window->Move($pos);
- }
- $window->Maximize(1) if $Settings->{_}{"${name}_maximized"};
- }
- }
- 1;
|