Tab.pm 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929
  1. package Slic3r::GUI::Tab;
  2. use strict;
  3. use warnings;
  4. use utf8;
  5. use File::Basename qw(basename);
  6. use List::Util qw(first);
  7. use Wx qw(:bookctrl :dialog :keycode :icon :id :misc :panel :sizer :treectrl :window);
  8. use Wx::Event qw(EVT_BUTTON EVT_CHOICE EVT_KEY_DOWN EVT_TREE_SEL_CHANGED);
  9. use base 'Wx::Panel';
  10. sub new {
  11. my $class = shift;
  12. my ($parent, %params) = @_;
  13. my $self = $class->SUPER::new($parent, -1, wxDefaultPosition, wxDefaultSize, wxBK_LEFT | wxTAB_TRAVERSAL);
  14. $self->{options} = []; # array of option names handled by this tab
  15. $self->{$_} = $params{$_} for qw(on_value_change on_presets_changed);
  16. # horizontal sizer
  17. $self->{sizer} = Wx::BoxSizer->new(wxHORIZONTAL);
  18. $self->{sizer}->SetSizeHints($self);
  19. $self->SetSizer($self->{sizer});
  20. # left vertical sizer
  21. my $left_sizer = Wx::BoxSizer->new(wxVERTICAL);
  22. $self->{sizer}->Add($left_sizer, 0, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, 3);
  23. my $left_col_width = 150;
  24. # preset chooser
  25. {
  26. # choice menu
  27. $self->{presets_choice} = Wx::Choice->new($self, -1, wxDefaultPosition, [$left_col_width, -1], []);
  28. $self->{presets_choice}->SetFont($Slic3r::GUI::small_font);
  29. # buttons
  30. $self->{btn_save_preset} = Wx::BitmapButton->new($self, -1, Wx::Bitmap->new("$Slic3r::var/disk.png", wxBITMAP_TYPE_PNG));
  31. $self->{btn_delete_preset} = Wx::BitmapButton->new($self, -1, Wx::Bitmap->new("$Slic3r::var/delete.png", wxBITMAP_TYPE_PNG));
  32. $self->{btn_save_preset}->SetToolTipString("Save current " . lc($self->title));
  33. $self->{btn_delete_preset}->SetToolTipString("Delete this preset");
  34. $self->{btn_delete_preset}->Disable;
  35. ### These cause GTK warnings:
  36. ###my $box = Wx::StaticBox->new($self, -1, "Presets:", wxDefaultPosition, [$left_col_width, 50]);
  37. ###my $hsizer = Wx::StaticBoxSizer->new($box, wxHORIZONTAL);
  38. my $hsizer = Wx::BoxSizer->new(wxHORIZONTAL);
  39. $left_sizer->Add($hsizer, 0, wxEXPAND | wxBOTTOM, 5);
  40. $hsizer->Add($self->{presets_choice}, 1, wxRIGHT | wxALIGN_CENTER_VERTICAL, 3);
  41. $hsizer->Add($self->{btn_save_preset}, 0, wxALIGN_CENTER_VERTICAL);
  42. $hsizer->Add($self->{btn_delete_preset}, 0, wxALIGN_CENTER_VERTICAL);
  43. }
  44. # tree
  45. $self->{treectrl} = Wx::TreeCtrl->new($self, -1, wxDefaultPosition, [$left_col_width, -1], wxTR_NO_BUTTONS | wxTR_HIDE_ROOT | wxTR_SINGLE | wxTR_NO_LINES | wxBORDER_SUNKEN | wxWANTS_CHARS);
  46. $left_sizer->Add($self->{treectrl}, 1, wxEXPAND);
  47. $self->{icons} = Wx::ImageList->new(16, 16, 1);
  48. $self->{treectrl}->AssignImageList($self->{icons});
  49. $self->{iconcount} = -1;
  50. $self->{treectrl}->AddRoot("root");
  51. $self->{pages} = [];
  52. $self->{treectrl}->SetIndent(0);
  53. EVT_TREE_SEL_CHANGED($parent, $self->{treectrl}, sub {
  54. my $page = first { $_->{title} eq $self->{treectrl}->GetItemText($self->{treectrl}->GetSelection) } @{$self->{pages}}
  55. or return;
  56. $_->Hide for @{$self->{pages}};
  57. $page->Show;
  58. $self->{sizer}->Layout;
  59. $self->Refresh;
  60. });
  61. EVT_KEY_DOWN($self->{treectrl}, sub {
  62. my ($treectrl, $event) = @_;
  63. if ($event->GetKeyCode == WXK_TAB) {
  64. $treectrl->Navigate($event->ShiftDown ? &Wx::wxNavigateBackward : &Wx::wxNavigateForward);
  65. } else {
  66. $event->Skip;
  67. }
  68. });
  69. EVT_CHOICE($parent, $self->{presets_choice}, sub {
  70. $self->on_select_preset;
  71. $self->on_presets_changed;
  72. });
  73. EVT_BUTTON($self, $self->{btn_save_preset}, sub { $self->save_preset });
  74. EVT_BUTTON($self, $self->{btn_delete_preset}, sub {
  75. my $i = $self->{presets_choice}->GetSelection;
  76. return if $i == 0; # this shouldn't happen but let's trap it anyway
  77. my $res = Wx::MessageDialog->new($self, "Are you sure you want to delete the selected preset?", 'Delete Preset', wxYES_NO | wxNO_DEFAULT | wxICON_QUESTION)->ShowModal;
  78. return unless $res == wxID_YES;
  79. if (-e $self->{presets}[$i]{file}) {
  80. unlink $self->{presets}[$i]{file};
  81. }
  82. splice @{$self->{presets}}, $i, 1;
  83. $self->set_dirty(0);
  84. $self->{presets_choice}->Delete($i);
  85. $self->{presets_choice}->SetSelection(0);
  86. $self->on_select_preset;
  87. $self->on_presets_changed;
  88. });
  89. $self->{config} = Slic3r::Config->new;
  90. $self->build;
  91. if ($self->hidden_options) {
  92. $self->{config}->apply(Slic3r::Config->new_from_defaults($self->hidden_options));
  93. push @{$self->{options}}, $self->hidden_options;
  94. }
  95. $self->load_presets;
  96. return $self;
  97. }
  98. sub current_preset {
  99. my $self = shift;
  100. return $self->{presets}[ $self->{presets_choice}->GetSelection ];
  101. }
  102. sub get_preset {
  103. my $self = shift;
  104. return $self->{presets}[ $_[0] ];
  105. }
  106. sub save_preset {
  107. my ($self, $name) = @_;
  108. # since buttons (and choices too) don't get focus on Mac, we set focus manually
  109. # to the treectrl so that the EVT_* events are fired for the input field having
  110. # focus currently. is there anything better than this?
  111. $self->{treectrl}->SetFocus;
  112. if (!defined $name) {
  113. my $preset = $self->current_preset;
  114. my $default_name = $preset->{default} ? 'Untitled' : basename($preset->{name});
  115. $default_name =~ s/\.ini$//i;
  116. my $dlg = Slic3r::GUI::SavePresetWindow->new($self,
  117. title => lc($self->title),
  118. default => $default_name,
  119. values => [ map { my $name = $_->{name}; $name =~ s/\.ini$//i; $name } @{$self->{presets}} ],
  120. );
  121. return unless $dlg->ShowModal == wxID_OK;
  122. $name = $dlg->get_name;
  123. }
  124. $self->config->save(sprintf "$Slic3r::GUI::datadir/%s/%s.ini", $self->name, $name);
  125. $self->set_dirty(0);
  126. $self->load_presets;
  127. $self->{presets_choice}->SetSelection(first { basename($self->{presets}[$_]{file}) eq $name . ".ini" } 1 .. $#{$self->{presets}});
  128. $self->on_select_preset;
  129. $self->on_presets_changed;
  130. }
  131. # propagate event to the parent
  132. sub on_value_change {
  133. my $self = shift;
  134. $self->{on_value_change}->(@_) if $self->{on_value_change};
  135. }
  136. sub on_presets_changed {
  137. my $self = shift;
  138. $self->{on_presets_changed}->([$self->{presets_choice}->GetStrings], $self->{presets_choice}->GetSelection)
  139. if $self->{on_presets_changed};
  140. }
  141. sub on_preset_loaded {}
  142. sub hidden_options {}
  143. sub config { $_[0]->{config}->clone }
  144. sub select_default_preset {
  145. my $self = shift;
  146. $self->{presets_choice}->SetSelection(0);
  147. }
  148. sub select_preset {
  149. my $self = shift;
  150. $self->{presets_choice}->SetSelection($_[0]);
  151. $self->on_select_preset;
  152. }
  153. sub on_select_preset {
  154. my $self = shift;
  155. if (defined $self->{dirty}) {
  156. my $name = $self->{dirty} == 0 ? 'Default preset' : "Preset \"$self->{presets}[$self->{dirty}]{name}\"";
  157. my $confirm = Wx::MessageDialog->new($self, "$name has unsaved changes. Discard changes and continue anyway?",
  158. 'Unsaved Changes', wxYES_NO | wxNO_DEFAULT | wxICON_QUESTION);
  159. if ($confirm->ShowModal == wxID_NO) {
  160. $self->{presets_choice}->SetSelection($self->{dirty});
  161. return;
  162. }
  163. $self->set_dirty(0);
  164. }
  165. my $preset = $self->current_preset;
  166. my $preset_config = $self->get_preset_config($preset);
  167. eval {
  168. local $SIG{__WARN__} = Slic3r::GUI::warning_catcher($self);
  169. foreach my $opt_key (@{$self->{options}}) {
  170. $self->{config}->set($opt_key, $preset_config->get($opt_key))
  171. if $preset_config->has($opt_key);
  172. }
  173. ($preset->{default} || $preset->{external})
  174. ? $self->{btn_delete_preset}->Disable
  175. : $self->{btn_delete_preset}->Enable;
  176. $self->on_preset_loaded;
  177. $self->reload_values;
  178. $self->set_dirty(0);
  179. $Slic3r::GUI::Settings->{presets}{$self->name} = $preset->{file} ? basename($preset->{file}) : '';
  180. };
  181. if ($@) {
  182. $@ = "I was unable to load the selected config file: $@";
  183. Slic3r::GUI::catch_error($self);
  184. $self->select_default_preset;
  185. }
  186. Slic3r::GUI->save_settings;
  187. }
  188. sub get_preset_config {
  189. my $self = shift;
  190. my ($preset) = @_;
  191. if ($preset->{default}) {
  192. return Slic3r::Config->new_from_defaults(@{$self->{options}});
  193. } else {
  194. if (!-e $preset->{file}) {
  195. Slic3r::GUI::show_error($self, "The selected preset does not exist anymore ($preset->{file}).");
  196. return;
  197. }
  198. # apply preset values on top of defaults
  199. my $external_config = Slic3r::Config->load($preset->{file});
  200. my $config = Slic3r::Config->new;
  201. $config->set($_, $external_config->get($_))
  202. for grep $external_config->has($_), @{$self->{options}};
  203. return $config;
  204. }
  205. }
  206. sub add_options_page {
  207. my $self = shift;
  208. my ($title, $icon, %params) = @_;
  209. if ($icon) {
  210. my $bitmap = Wx::Bitmap->new("$Slic3r::var/$icon", wxBITMAP_TYPE_PNG);
  211. $self->{icons}->Add($bitmap);
  212. $self->{iconcount}++;
  213. }
  214. {
  215. # get all config options being added to the current page; remove indexes; associate defaults
  216. my @options = map { $_ =~ s/#.+//; $_ } grep !ref($_), map @{$_->{options}}, @{$params{optgroups}};
  217. my %defaults_to_set = map { $_ => 1 } @options;
  218. # apply default values for the options we don't have already
  219. delete $defaults_to_set{$_} for @{$self->{options}};
  220. $self->{config}->apply(Slic3r::Config->new_from_defaults(keys %defaults_to_set)) if %defaults_to_set;
  221. # append such options to our list
  222. push @{$self->{options}}, @options;
  223. }
  224. my $page = Slic3r::GUI::Tab::Page->new($self, $title, $self->{iconcount}, %params, on_change => sub {
  225. $self->on_value_change(@_);
  226. $self->set_dirty(1);
  227. $self->on_presets_changed;
  228. });
  229. $page->Hide;
  230. $self->{sizer}->Add($page, 1, wxEXPAND | wxLEFT, 5);
  231. push @{$self->{pages}}, $page;
  232. $self->update_tree;
  233. return $page;
  234. }
  235. sub set_value {
  236. my $self = shift;
  237. my ($opt_key, $value) = @_;
  238. my $changed = 0;
  239. foreach my $page (@{$self->{pages}}) {
  240. $changed = 1 if $page->set_value($opt_key, $value);
  241. }
  242. return $changed;
  243. }
  244. sub reload_values {
  245. my $self = shift;
  246. $self->set_value($_, $self->{config}->get($_)) for keys %{$self->{config}};
  247. }
  248. sub update_tree {
  249. my $self = shift;
  250. my ($select) = @_;
  251. $select //= 0; #/
  252. my $rootItem = $self->{treectrl}->GetRootItem;
  253. $self->{treectrl}->DeleteChildren($rootItem);
  254. foreach my $page (@{$self->{pages}}) {
  255. my $itemId = $self->{treectrl}->AppendItem($rootItem, $page->{title}, $page->{iconID});
  256. $self->{treectrl}->SelectItem($itemId) if $self->{treectrl}->GetChildrenCount($rootItem) == $select + 1;
  257. }
  258. }
  259. sub set_dirty {
  260. my $self = shift;
  261. my ($dirty) = @_;
  262. my $selection = $self->{presets_choice}->GetSelection;
  263. my $i = $self->{dirty} // $selection; #/
  264. my $text = $self->{presets_choice}->GetString($i);
  265. if ($dirty) {
  266. $self->{dirty} = $i;
  267. if ($text !~ / \(modified\)$/) {
  268. $self->{presets_choice}->SetString($i, "$text (modified)");
  269. $self->{presets_choice}->SetSelection($selection); # http://trac.wxwidgets.org/ticket/13769
  270. }
  271. } else {
  272. $self->{dirty} = undef;
  273. $text =~ s/ \(modified\)$//;
  274. $self->{presets_choice}->SetString($i, $text);
  275. $self->{presets_choice}->SetSelection($selection); # http://trac.wxwidgets.org/ticket/13769
  276. }
  277. $self->on_presets_changed;
  278. }
  279. sub is_dirty {
  280. my $self = shift;
  281. return (defined $self->{dirty});
  282. }
  283. sub load_presets {
  284. my $self = shift;
  285. $self->{presets} = [{
  286. default => 1,
  287. name => '- default -',
  288. }];
  289. opendir my $dh, "$Slic3r::GUI::datadir/" . $self->name or die "Failed to read directory $Slic3r::GUI::datadir/" . $self->name . " (errno: $!)\n";
  290. foreach my $file (sort grep /\.ini$/i, readdir $dh) {
  291. my $name = basename($file);
  292. $name =~ s/\.ini$//;
  293. push @{$self->{presets}}, {
  294. file => "$Slic3r::GUI::datadir/" . $self->name . "/$file",
  295. name => $name,
  296. };
  297. }
  298. closedir $dh;
  299. $self->{presets_choice}->Clear;
  300. $self->{presets_choice}->Append($_->{name}) for @{$self->{presets}};
  301. {
  302. # load last used preset
  303. my $i = first { basename($self->{presets}[$_]{file}) eq ($Slic3r::GUI::Settings->{presets}{$self->name} || '') } 1 .. $#{$self->{presets}};
  304. $self->{presets_choice}->SetSelection($i || 0);
  305. $self->on_select_preset;
  306. }
  307. $self->on_presets_changed;
  308. }
  309. sub load_config_file {
  310. my $self = shift;
  311. my ($file) = @_;
  312. # look for the loaded config among the existing menu items
  313. my $i = first { $self->{presets}[$_]{file} eq $file && $self->{presets}[$_]{external} } 1..$#{$self->{presets}};
  314. if (!$i) {
  315. my $preset_name = basename($file); # keep the .ini suffix
  316. push @{$self->{presets}}, {
  317. file => $file,
  318. name => $preset_name,
  319. external => 1,
  320. };
  321. $self->{presets_choice}->Append($preset_name);
  322. $i = $#{$self->{presets}};
  323. }
  324. $self->{presets_choice}->SetSelection($i);
  325. $self->on_select_preset;
  326. $self->on_presets_changed;
  327. }
  328. package Slic3r::GUI::Tab::Print;
  329. use base 'Slic3r::GUI::Tab';
  330. sub name { 'print' }
  331. sub title { 'Print Settings' }
  332. sub build {
  333. my $self = shift;
  334. $self->add_options_page('Layers and perimeters', 'layers.png', optgroups => [
  335. {
  336. title => 'Layer height',
  337. options => [qw(layer_height first_layer_height)],
  338. },
  339. {
  340. title => 'Vertical shells',
  341. options => [qw(perimeters spiral_vase)],
  342. },
  343. {
  344. title => 'Horizontal shells',
  345. options => [qw(top_solid_layers bottom_solid_layers)],
  346. lines => [
  347. {
  348. label => 'Solid layers',
  349. options => [qw(top_solid_layers bottom_solid_layers)],
  350. },
  351. ],
  352. },
  353. {
  354. title => 'Quality (slower slicing)',
  355. options => [qw(extra_perimeters avoid_crossing_perimeters start_perimeters_at_concave_points start_perimeters_at_non_overhang thin_walls overhangs)],
  356. lines => [
  357. Slic3r::GUI::OptionsGroup->single_option_line('extra_perimeters'),
  358. Slic3r::GUI::OptionsGroup->single_option_line('avoid_crossing_perimeters'),
  359. {
  360. label => 'Start perimeters at',
  361. options => [qw(start_perimeters_at_concave_points start_perimeters_at_non_overhang)],
  362. },
  363. Slic3r::GUI::OptionsGroup->single_option_line('thin_walls'),
  364. Slic3r::GUI::OptionsGroup->single_option_line('overhangs'),
  365. ],
  366. },
  367. {
  368. title => 'Advanced',
  369. options => [qw(randomize_start external_perimeters_first)],
  370. },
  371. ]);
  372. $self->add_options_page('Infill', 'shading.png', optgroups => [
  373. {
  374. title => 'Infill',
  375. options => [qw(fill_density fill_pattern solid_fill_pattern)],
  376. },
  377. {
  378. title => 'Reducing printing time',
  379. options => [qw(infill_every_layers infill_only_where_needed)],
  380. },
  381. {
  382. title => 'Advanced',
  383. options => [qw(solid_infill_every_layers fill_angle
  384. solid_infill_below_area only_retract_when_crossing_perimeters infill_first)],
  385. },
  386. ]);
  387. $self->add_options_page('Speed', 'time.png', optgroups => [
  388. {
  389. title => 'Speed for print moves',
  390. options => [qw(perimeter_speed small_perimeter_speed external_perimeter_speed infill_speed solid_infill_speed top_solid_infill_speed support_material_speed bridge_speed gap_fill_speed)],
  391. },
  392. {
  393. title => 'Speed for non-print moves',
  394. options => [qw(travel_speed)],
  395. },
  396. {
  397. title => 'Modifiers',
  398. options => [qw(first_layer_speed)],
  399. },
  400. {
  401. title => 'Acceleration control (advanced)',
  402. options => [qw(perimeter_acceleration infill_acceleration bridge_acceleration first_layer_acceleration default_acceleration)],
  403. },
  404. ]);
  405. $self->add_options_page('Skirt and brim', 'box.png', optgroups => [
  406. {
  407. title => 'Skirt',
  408. options => [qw(skirts skirt_distance skirt_height min_skirt_length)],
  409. },
  410. {
  411. title => 'Brim',
  412. options => [qw(brim_width)],
  413. },
  414. ]);
  415. $self->add_options_page('Support material', 'building.png', optgroups => [
  416. {
  417. title => 'Support material',
  418. options => [qw(support_material support_material_threshold support_material_enforce_layers)],
  419. },
  420. {
  421. title => 'Raft',
  422. options => [qw(raft_layers)],
  423. },
  424. {
  425. title => 'Options for support material and raft',
  426. options => [qw(support_material_pattern support_material_spacing support_material_angle
  427. support_material_interface_layers support_material_interface_spacing)],
  428. },
  429. ]);
  430. $self->add_options_page('Notes', 'note.png', optgroups => [
  431. {
  432. title => 'Notes',
  433. no_labels => 1,
  434. options => [qw(notes)],
  435. },
  436. ]);
  437. $self->add_options_page('Output options', 'page_white_go.png', optgroups => [
  438. {
  439. title => 'Sequential printing',
  440. options => [qw(complete_objects extruder_clearance_radius extruder_clearance_height)],
  441. lines => [
  442. Slic3r::GUI::OptionsGroup->single_option_line('complete_objects'),
  443. {
  444. label => 'Extruder clearance (mm)',
  445. options => [qw(extruder_clearance_radius extruder_clearance_height)],
  446. },
  447. ],
  448. },
  449. {
  450. title => 'Output file',
  451. options => [qw(gcode_comments output_filename_format)],
  452. },
  453. {
  454. title => 'Post-processing scripts',
  455. no_labels => 1,
  456. options => [qw(post_process)],
  457. },
  458. ]);
  459. $self->add_options_page('Multiple Extruders', 'funnel.png', optgroups => [
  460. {
  461. title => 'Extruders',
  462. options => [qw(perimeter_extruder infill_extruder support_material_extruder support_material_interface_extruder)],
  463. },
  464. {
  465. title => 'Ooze prevention',
  466. options => [qw(ooze_prevention standby_temperature_delta)],
  467. },
  468. ]);
  469. $self->add_options_page('Advanced', 'wrench.png', optgroups => [
  470. {
  471. title => 'Extrusion width',
  472. label_width => 180,
  473. options => [qw(extrusion_width first_layer_extrusion_width perimeter_extrusion_width infill_extrusion_width solid_infill_extrusion_width top_infill_extrusion_width support_material_extrusion_width)],
  474. },
  475. {
  476. title => 'Flow',
  477. options => [qw(bridge_flow_ratio)],
  478. },
  479. {
  480. title => 'Other',
  481. options => [($Slic3r::have_threads ? qw(threads) : ()), qw(resolution)],
  482. },
  483. ]);
  484. }
  485. sub hidden_options { !$Slic3r::have_threads ? qw(threads) : () }
  486. package Slic3r::GUI::Tab::Filament;
  487. use base 'Slic3r::GUI::Tab';
  488. sub name { 'filament' }
  489. sub title { 'Filament Settings' }
  490. sub build {
  491. my $self = shift;
  492. $self->add_options_page('Filament', 'spool.png', optgroups => [
  493. {
  494. title => 'Filament',
  495. options => ['filament_diameter#0', 'extrusion_multiplier#0'],
  496. },
  497. {
  498. title => 'Temperature (°C)',
  499. options => ['temperature#0', 'first_layer_temperature#0', qw(bed_temperature first_layer_bed_temperature)],
  500. lines => [
  501. {
  502. label => 'Extruder',
  503. options => ['first_layer_temperature#0', 'temperature#0'],
  504. },
  505. {
  506. label => 'Bed',
  507. options => [qw(first_layer_bed_temperature bed_temperature)],
  508. },
  509. ],
  510. },
  511. ]);
  512. $self->add_options_page('Cooling', 'hourglass.png', optgroups => [
  513. {
  514. title => 'Enable',
  515. options => [qw(fan_always_on cooling)],
  516. lines => [
  517. Slic3r::GUI::OptionsGroup->single_option_line('fan_always_on'),
  518. Slic3r::GUI::OptionsGroup->single_option_line('cooling'),
  519. {
  520. label => '',
  521. widget => ($self->{description_line} = Slic3r::GUI::OptionsGroup::StaticTextLine->new),
  522. },
  523. ],
  524. },
  525. {
  526. title => 'Fan settings',
  527. options => [qw(min_fan_speed max_fan_speed bridge_fan_speed disable_fan_first_layers)],
  528. lines => [
  529. {
  530. label => 'Fan speed',
  531. options => [qw(min_fan_speed max_fan_speed)],
  532. },
  533. Slic3r::GUI::OptionsGroup->single_option_line('bridge_fan_speed'),
  534. Slic3r::GUI::OptionsGroup->single_option_line('disable_fan_first_layers'),
  535. ],
  536. },
  537. {
  538. title => 'Cooling thresholds',
  539. label_width => 250,
  540. options => [qw(fan_below_layer_time slowdown_below_layer_time min_print_speed)],
  541. },
  542. ]);
  543. }
  544. sub _update_description {
  545. my $self = shift;
  546. my $config = $self->config;
  547. my $msg = "";
  548. my $fan_other_layers = $config->fan_always_on
  549. ? sprintf "will always run at %d%%%s.", $config->min_fan_speed,
  550. ($config->disable_fan_first_layers > 1
  551. ? " except for the first " . $config->disable_fan_first_layers . " layers"
  552. : $config->disable_fan_first_layers == 1
  553. ? " except for the first layer"
  554. : "")
  555. : "will be turned off.";
  556. if ($config->cooling) {
  557. $msg = sprintf "If estimated layer time is below ~%ds, fan will run at %d%% and print speed will be reduced so that no less than %ds are spent on that layer (however, speed will never be reduced below %dmm/s).",
  558. $config->slowdown_below_layer_time, $config->max_fan_speed, $config->slowdown_below_layer_time, $config->min_print_speed;
  559. if ($config->fan_below_layer_time > $config->slowdown_below_layer_time) {
  560. $msg .= sprintf "\nIf estimated layer time is greater, but still below ~%ds, fan will run at a proportionally decreasing speed between %d%% and %d%%.",
  561. $config->fan_below_layer_time, $config->max_fan_speed, $config->min_fan_speed;
  562. }
  563. $msg .= "\nDuring the other layers, fan $fan_other_layers"
  564. } else {
  565. $msg = "Fan $fan_other_layers";
  566. }
  567. $self->{description_line}->SetText($msg);
  568. }
  569. sub on_value_change {
  570. my $self = shift;
  571. my ($opt_key) = @_;
  572. $self->SUPER::on_value_change(@_);
  573. $self->_update_description;
  574. }
  575. package Slic3r::GUI::Tab::Printer;
  576. use base 'Slic3r::GUI::Tab';
  577. sub name { 'printer' }
  578. sub title { 'Printer Settings' }
  579. sub build {
  580. my $self = shift;
  581. $self->{extruders_count} = 1;
  582. $self->add_options_page('General', 'printer_empty.png', optgroups => [
  583. {
  584. title => 'Size and coordinates',
  585. options => [qw(bed_size print_center z_offset)],
  586. },
  587. {
  588. title => 'Firmware',
  589. options => [qw(gcode_flavor use_relative_e_distances)],
  590. },
  591. {
  592. title => 'Capabilities',
  593. options => [
  594. {
  595. opt_key => 'extruders_count',
  596. label => 'Extruders',
  597. tooltip => 'Number of extruders of the printer.',
  598. type => 'i',
  599. min => 1,
  600. default => 1,
  601. on_change => sub { $self->{extruders_count} = $_[0] },
  602. },
  603. ],
  604. },
  605. {
  606. title => 'Advanced',
  607. options => [qw(use_firmware_retraction vibration_limit)],
  608. },
  609. ]);
  610. $self->add_options_page('Custom G-code', 'cog.png', optgroups => [
  611. {
  612. title => 'Start G-code',
  613. no_labels => 1,
  614. options => [qw(start_gcode)],
  615. },
  616. {
  617. title => 'End G-code',
  618. no_labels => 1,
  619. options => [qw(end_gcode)],
  620. },
  621. {
  622. title => 'Layer change G-code',
  623. no_labels => 1,
  624. options => [qw(layer_gcode)],
  625. },
  626. {
  627. title => 'Tool change G-code',
  628. no_labels => 1,
  629. options => [qw(toolchange_gcode)],
  630. },
  631. ]);
  632. $self->{extruder_pages} = [];
  633. $self->_build_extruder_pages;
  634. }
  635. sub _extruder_options { qw(nozzle_diameter extruder_offset retract_length retract_lift retract_speed retract_restart_extra retract_before_travel wipe
  636. retract_layer_change retract_length_toolchange retract_restart_extra_toolchange) }
  637. sub config {
  638. my $self = shift;
  639. my $config = $self->SUPER::config(@_);
  640. # remove all unused values
  641. foreach my $opt_key ($self->_extruder_options) {
  642. splice @{ $config->{$opt_key} }, $self->{extruders_count};
  643. }
  644. return $config;
  645. }
  646. sub _build_extruder_pages {
  647. my $self = shift;
  648. foreach my $extruder_idx (0 .. $self->{extruders_count}-1) {
  649. # build page if it doesn't exist
  650. $self->{extruder_pages}[$extruder_idx] ||= $self->add_options_page("Extruder " . ($extruder_idx + 1), 'funnel.png', optgroups => [
  651. {
  652. title => 'Size',
  653. options => ['nozzle_diameter#' . $extruder_idx],
  654. },
  655. {
  656. title => 'Position (for multi-extruder printers)',
  657. options => ['extruder_offset#' . $extruder_idx],
  658. },
  659. {
  660. title => 'Retraction',
  661. options => [
  662. map "${_}#${extruder_idx}",
  663. qw(retract_length retract_lift retract_speed retract_restart_extra retract_before_travel retract_layer_change wipe)
  664. ],
  665. },
  666. {
  667. title => 'Retraction when tool is disabled (advanced settings for multi-extruder setups)',
  668. options => [
  669. map "${_}#${extruder_idx}",
  670. qw(retract_length_toolchange retract_restart_extra_toolchange)
  671. ],
  672. },
  673. ]);
  674. $self->{extruder_pages}[$extruder_idx]{disabled} = 0;
  675. }
  676. # rebuild page list
  677. @{$self->{pages}} = (
  678. (grep $_->{title} !~ /^Extruder \d+/, @{$self->{pages}}),
  679. @{$self->{extruder_pages}}[ 0 .. $self->{extruders_count}-1 ],
  680. );
  681. }
  682. sub on_value_change {
  683. my $self = shift;
  684. my ($opt_key) = @_;
  685. $self->SUPER::on_value_change(@_);
  686. if ($opt_key eq 'extruders_count') {
  687. # remove unused pages from list
  688. my @unused_pages = @{ $self->{extruder_pages} }[$self->{extruders_count} .. $#{$self->{extruder_pages}}];
  689. for my $page (@unused_pages) {
  690. @{$self->{pages}} = grep $_ ne $page, @{$self->{pages}};
  691. $page->{disabled} = 1;
  692. }
  693. # add extra pages
  694. $self->_build_extruder_pages;
  695. # update page list and select first page (General)
  696. $self->update_tree(0);
  697. }
  698. }
  699. # this gets executed after preset is loaded and before GUI fields are updated
  700. sub on_preset_loaded {
  701. my $self = shift;
  702. # update the extruders count field
  703. {
  704. # update the GUI field according to the number of nozzle diameters supplied
  705. $self->set_value('extruders_count', scalar @{ $self->{config}->nozzle_diameter });
  706. # update extruder page list
  707. $self->on_value_change('extruders_count');
  708. }
  709. }
  710. sub load_config_file {
  711. my $self = shift;
  712. $self->SUPER::load_config_file(@_);
  713. Slic3r::GUI::warning_catcher($self)->(
  714. "Your configuration was imported. However, Slic3r is currently only able to import settings "
  715. . "for the first defined filament. We recommend you don't use exported configuration files "
  716. . "for multi-extruder setups and rely on the built-in preset management system instead.")
  717. if @{ $self->{config}->nozzle_diameter } > 1;
  718. }
  719. package Slic3r::GUI::Tab::Page;
  720. use Wx qw(:misc :panel :sizer);
  721. use base 'Wx::ScrolledWindow';
  722. sub new {
  723. my $class = shift;
  724. my ($parent, $title, $iconID, %params) = @_;
  725. my $self = $class->SUPER::new($parent, -1, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
  726. $self->{optgroups} = [];
  727. $self->{title} = $title;
  728. $self->{iconID} = $iconID;
  729. $self->SetScrollbars(1, 1, 1, 1);
  730. $self->{vsizer} = Wx::BoxSizer->new(wxVERTICAL);
  731. $self->SetSizer($self->{vsizer});
  732. if ($params{optgroups}) {
  733. $self->append_optgroup(
  734. %$_,
  735. config => $parent->{config},
  736. on_change => $params{on_change},
  737. ) for @{$params{optgroups}};
  738. }
  739. return $self;
  740. }
  741. sub append_optgroup {
  742. my $self = shift;
  743. my %params = @_;
  744. my $class = $params{class} || 'Slic3r::GUI::ConfigOptionsGroup';
  745. my $optgroup = $class->new(
  746. parent => $self,
  747. config => $self->GetParent->{config},
  748. label_width => 200,
  749. %params,
  750. );
  751. $self->{vsizer}->Add($optgroup->sizer, 0, wxEXPAND | wxALL, 5);
  752. push @{$self->{optgroups}}, $optgroup;
  753. }
  754. sub set_value {
  755. my $self = shift;
  756. my ($opt_key, $value) = @_;
  757. my $changed = 0;
  758. foreach my $optgroup (@{$self->{optgroups}}) {
  759. $changed = 1 if $optgroup->set_value($opt_key, $value);
  760. }
  761. return $changed;
  762. }
  763. package Slic3r::GUI::SavePresetWindow;
  764. use Wx qw(:combobox :dialog :id :misc :sizer);
  765. use Wx::Event qw(EVT_BUTTON EVT_TEXT_ENTER);
  766. use base 'Wx::Dialog';
  767. sub new {
  768. my $class = shift;
  769. my ($parent, %params) = @_;
  770. my $self = $class->SUPER::new($parent, -1, "Save preset", wxDefaultPosition, wxDefaultSize);
  771. my $text = Wx::StaticText->new($self, -1, "Save " . lc($params{title}) . " as:", wxDefaultPosition, wxDefaultSize);
  772. $self->{combo} = Wx::ComboBox->new($self, -1, $params{default}, wxDefaultPosition, wxDefaultSize, $params{values},
  773. wxTE_PROCESS_ENTER);
  774. my $buttons = $self->CreateStdDialogButtonSizer(wxOK | wxCANCEL);
  775. my $sizer = Wx::BoxSizer->new(wxVERTICAL);
  776. $sizer->Add($text, 0, wxEXPAND | wxTOP | wxLEFT | wxRIGHT, 10);
  777. $sizer->Add($self->{combo}, 0, wxEXPAND | wxLEFT | wxRIGHT, 10);
  778. $sizer->Add($buttons, 0, wxEXPAND | wxBOTTOM | wxLEFT | wxRIGHT, 10);
  779. EVT_BUTTON($self, wxID_OK, \&accept);
  780. EVT_TEXT_ENTER($self, $self->{combo}, \&accept);
  781. $self->SetSizer($sizer);
  782. $sizer->SetSizeHints($self);
  783. return $self;
  784. }
  785. sub accept {
  786. my ($self, $event) = @_;
  787. if (($self->{chosen_name} = $self->{combo}->GetValue)) {
  788. if ($self->{chosen_name} =~ /^[^<>:\/\\|?*\"]+$/i) {
  789. $self->EndModal(wxID_OK);
  790. } else {
  791. Slic3r::GUI::show_error($self, "The supplied name is not valid; the following characters are not allowed: <>:/\|?*\"");
  792. }
  793. }
  794. }
  795. sub get_name {
  796. my $self = shift;
  797. return $self->{chosen_name};
  798. }
  799. 1;