3DPreview.pm 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. package Slic3r::GUI::Plater::3DPreview;
  2. use strict;
  3. use warnings;
  4. use utf8;
  5. use Slic3r::Print::State ':steps';
  6. use Wx qw(:misc :sizer :slider :statictext :keycode wxWHITE wxCB_READONLY);
  7. use Wx::Event qw(EVT_SLIDER EVT_KEY_DOWN EVT_CHECKBOX EVT_CHOICE EVT_CHECKLISTBOX);
  8. use base qw(Wx::Panel Class::Accessor);
  9. __PACKAGE__->mk_accessors(qw(print gcode_preview_data enabled _loaded canvas slider_low slider_high single_layer auto_zoom));
  10. sub new {
  11. my $class = shift;
  12. my ($parent, $print, $gcode_preview_data, $config) = @_;
  13. my $self = $class->SUPER::new($parent, -1, wxDefaultPosition);
  14. $self->{config} = $config;
  15. $self->{number_extruders} = 1;
  16. # Show by feature type by default.
  17. $self->{preferred_color_mode} = 'feature';
  18. $self->auto_zoom(1);
  19. # init GUI elements
  20. my $canvas = Slic3r::GUI::3DScene->new($self);
  21. $canvas->use_plain_shader(1);
  22. $self->canvas($canvas);
  23. my $slider_low = Wx::Slider->new(
  24. $self, -1,
  25. 0, # default
  26. 0, # min
  27. # we set max to a bogus non-zero value because the MSW implementation of wxSlider
  28. # will skip drawing the slider if max <= min:
  29. 1, # max
  30. wxDefaultPosition,
  31. wxDefaultSize,
  32. wxVERTICAL | wxSL_INVERSE,
  33. );
  34. $self->slider_low($slider_low);
  35. my $slider_high = Wx::Slider->new(
  36. $self, -1,
  37. 0, # default
  38. 0, # min
  39. # we set max to a bogus non-zero value because the MSW implementation of wxSlider
  40. # will skip drawing the slider if max <= min:
  41. 1, # max
  42. wxDefaultPosition,
  43. wxDefaultSize,
  44. wxVERTICAL | wxSL_INVERSE,
  45. );
  46. $self->slider_high($slider_high);
  47. my $z_label_low = $self->{z_label_low} = Wx::StaticText->new($self, -1, "", wxDefaultPosition,
  48. [40,-1], wxALIGN_CENTRE_HORIZONTAL);
  49. $z_label_low->SetFont($Slic3r::GUI::small_font);
  50. my $z_label_high = $self->{z_label_high} = Wx::StaticText->new($self, -1, "", wxDefaultPosition,
  51. [40,-1], wxALIGN_CENTRE_HORIZONTAL);
  52. $z_label_high->SetFont($Slic3r::GUI::small_font);
  53. $self->single_layer(0);
  54. my $checkbox_singlelayer = $self->{checkbox_singlelayer} = Wx::CheckBox->new($self, -1, "1 Layer");
  55. my $label_view_type = $self->{label_view_type} = Wx::StaticText->new($self, -1, "View");
  56. my $choice_view_type = $self->{choice_view_type} = Wx::Choice->new($self, -1);
  57. $choice_view_type->Append("Feature type");
  58. $choice_view_type->Append("Height");
  59. $choice_view_type->Append("Width");
  60. $choice_view_type->Append("Speed");
  61. $choice_view_type->Append("Tool");
  62. $choice_view_type->SetSelection(0);
  63. my $label_show_features = $self->{label_show_features} = Wx::StaticText->new($self, -1, "Show");
  64. my $combochecklist_features = $self->{combochecklist_features} = Wx::ComboCtrl->new();
  65. $combochecklist_features->Create($self, -1, "Feature types", wxDefaultPosition, [200, -1], wxCB_READONLY);
  66. #FIXME If the following line is removed, the combo box popup list will not react to mouse clicks.
  67. # On the other side, with this line the combo box popup cannot be closed by clicking on the combo button on Windows 10.
  68. $combochecklist_features->UseAltPopupWindow();
  69. $combochecklist_features->EnablePopupAnimation(0);
  70. my $feature_text = "Feature types";
  71. my $feature_items = "Perimeter|External perimeter|Overhang perimeter|Internal infill|Solid infill|Top solid infill|Bridge infill|Gap fill|Skirt|Support material|Support material interface|Wipe tower";
  72. Slic3r::GUI::create_combochecklist($combochecklist_features, $feature_text, $feature_items, 1);
  73. my $checkbox_travel = $self->{checkbox_travel} = Wx::CheckBox->new($self, -1, "Travel");
  74. my $checkbox_retractions = $self->{checkbox_retractions} = Wx::CheckBox->new($self, -1, "Retractions");
  75. my $checkbox_unretractions = $self->{checkbox_unretractions} = Wx::CheckBox->new($self, -1, "Unretractions");
  76. my $checkbox_shells = $self->{checkbox_shells} = Wx::CheckBox->new($self, -1, "Shells");
  77. my $hsizer = Wx::BoxSizer->new(wxHORIZONTAL);
  78. my $vsizer = Wx::BoxSizer->new(wxVERTICAL);
  79. my $vsizer_outer = Wx::BoxSizer->new(wxVERTICAL);
  80. $vsizer->Add($slider_low, 3, 0, 0);
  81. $vsizer->Add($z_label_low, 0, 0, 0);
  82. $hsizer->Add($vsizer, 0, wxEXPAND, 0);
  83. $vsizer = Wx::BoxSizer->new(wxVERTICAL);
  84. $vsizer->Add($slider_high, 3, 0, 0);
  85. $vsizer->Add($z_label_high, 0, 0, 0);
  86. $hsizer->Add($vsizer, 0, wxEXPAND, 0);
  87. $vsizer_outer->Add($hsizer, 3, wxALIGN_CENTER_HORIZONTAL, 0);
  88. $vsizer_outer->Add($checkbox_singlelayer, 0, wxTOP | wxALIGN_CENTER_HORIZONTAL, 5);
  89. my $bottom_sizer = Wx::BoxSizer->new(wxHORIZONTAL);
  90. $bottom_sizer->Add($label_view_type, 0, wxALIGN_CENTER_VERTICAL, 5);
  91. $bottom_sizer->Add($choice_view_type, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
  92. $bottom_sizer->AddSpacer(10);
  93. $bottom_sizer->Add($label_show_features, 0, wxALIGN_CENTER_VERTICAL, 5);
  94. $bottom_sizer->Add($combochecklist_features, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
  95. $bottom_sizer->AddSpacer(20);
  96. $bottom_sizer->Add($checkbox_travel, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
  97. $bottom_sizer->AddSpacer(10);
  98. $bottom_sizer->Add($checkbox_retractions, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
  99. $bottom_sizer->AddSpacer(10);
  100. $bottom_sizer->Add($checkbox_unretractions, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
  101. $bottom_sizer->AddSpacer(10);
  102. $bottom_sizer->Add($checkbox_shells, 0, wxEXPAND | wxALL | wxALIGN_CENTER_VERTICAL, 5);
  103. my $sizer = Wx::BoxSizer->new(wxHORIZONTAL);
  104. $sizer->Add($canvas, 1, wxALL | wxEXPAND, 0);
  105. $sizer->Add($vsizer_outer, 0, wxTOP | wxBOTTOM | wxEXPAND, 5);
  106. my $main_sizer = Wx::BoxSizer->new(wxVERTICAL);
  107. $main_sizer->Add($sizer, 1, wxALL | wxEXPAND, 0);
  108. $main_sizer->Add($bottom_sizer, 0, wxALL | wxEXPAND, 0);
  109. EVT_SLIDER($self, $slider_low, sub {
  110. $slider_high->SetValue($slider_low->GetValue) if $self->single_layer;
  111. $self->set_z_idx_low ($slider_low ->GetValue)
  112. });
  113. EVT_SLIDER($self, $slider_high, sub {
  114. $slider_low->SetValue($slider_high->GetValue) if $self->single_layer;
  115. $self->set_z_idx_high($slider_high->GetValue)
  116. });
  117. EVT_KEY_DOWN($canvas, sub {
  118. my ($s, $event) = @_;
  119. my $key = $event->GetKeyCode;
  120. if ($event->HasModifiers) {
  121. $event->Skip;
  122. } else {
  123. if ($key == ord('U')) {
  124. $slider_high->SetValue($slider_high->GetValue + 1);
  125. $slider_low->SetValue($slider_high->GetValue) if ($event->ShiftDown());
  126. $self->set_z_idx_high($slider_high->GetValue);
  127. } elsif ($key == ord('D')) {
  128. $slider_high->SetValue($slider_high->GetValue - 1);
  129. $slider_low->SetValue($slider_high->GetValue) if ($event->ShiftDown());
  130. $self->set_z_idx_high($slider_high->GetValue);
  131. } elsif ($key == ord('S')) {
  132. $checkbox_singlelayer->SetValue(! $checkbox_singlelayer->GetValue());
  133. $self->single_layer($checkbox_singlelayer->GetValue());
  134. if ($self->single_layer) {
  135. $slider_low->SetValue($slider_high->GetValue);
  136. $self->set_z_idx_high($slider_high->GetValue);
  137. }
  138. } else {
  139. $event->Skip;
  140. }
  141. }
  142. });
  143. EVT_KEY_DOWN($slider_low, sub {
  144. my ($s, $event) = @_;
  145. my $key = $event->GetKeyCode;
  146. if ($event->HasModifiers) {
  147. $event->Skip;
  148. } else {
  149. if ($key == WXK_LEFT) {
  150. } elsif ($key == WXK_RIGHT) {
  151. $slider_high->SetFocus;
  152. } else {
  153. $event->Skip;
  154. }
  155. }
  156. });
  157. EVT_KEY_DOWN($slider_high, sub {
  158. my ($s, $event) = @_;
  159. my $key = $event->GetKeyCode;
  160. if ($event->HasModifiers) {
  161. $event->Skip;
  162. } else {
  163. if ($key == WXK_LEFT) {
  164. $slider_low->SetFocus;
  165. } elsif ($key == WXK_RIGHT) {
  166. } else {
  167. $event->Skip;
  168. }
  169. }
  170. });
  171. EVT_CHECKBOX($self, $checkbox_singlelayer, sub {
  172. $self->single_layer($checkbox_singlelayer->GetValue());
  173. if ($self->single_layer) {
  174. $slider_low->SetValue($slider_high->GetValue);
  175. $self->set_z_idx_high($slider_high->GetValue);
  176. }
  177. });
  178. EVT_CHOICE($self, $choice_view_type, sub {
  179. my $selection = $choice_view_type->GetCurrentSelection();
  180. $self->{preferred_color_mode} = ($selection == 4) ? 'tool' : 'feature';
  181. $self->gcode_preview_data->set_type($selection);
  182. $self->auto_zoom(0);
  183. $self->reload_print;
  184. $self->auto_zoom(1);
  185. });
  186. EVT_CHECKLISTBOX($self, $combochecklist_features, sub {
  187. my $flags = Slic3r::GUI::combochecklist_get_flags($combochecklist_features);
  188. $self->gcode_preview_data->set_extrusion_flags($flags);
  189. $self->auto_zoom(0);
  190. $self->refresh_print;
  191. $self->auto_zoom(1);
  192. });
  193. EVT_CHECKBOX($self, $checkbox_travel, sub {
  194. $self->gcode_preview_data->set_travel_visible($checkbox_travel->IsChecked());
  195. $self->auto_zoom(0);
  196. $self->refresh_print;
  197. $self->auto_zoom(1);
  198. });
  199. EVT_CHECKBOX($self, $checkbox_retractions, sub {
  200. $self->gcode_preview_data->set_retractions_visible($checkbox_retractions->IsChecked());
  201. $self->auto_zoom(0);
  202. $self->refresh_print;
  203. $self->auto_zoom(1);
  204. });
  205. EVT_CHECKBOX($self, $checkbox_unretractions, sub {
  206. $self->gcode_preview_data->set_unretractions_visible($checkbox_unretractions->IsChecked());
  207. $self->auto_zoom(0);
  208. $self->refresh_print;
  209. $self->auto_zoom(1);
  210. });
  211. EVT_CHECKBOX($self, $checkbox_shells, sub {
  212. $self->gcode_preview_data->set_shells_visible($checkbox_shells->IsChecked());
  213. $self->auto_zoom(0);
  214. $self->refresh_print;
  215. $self->auto_zoom(1);
  216. });
  217. $self->SetSizer($main_sizer);
  218. $self->SetMinSize($self->GetSize);
  219. $sizer->SetSizeHints($self);
  220. # init canvas
  221. $self->print($print);
  222. $self->gcode_preview_data($gcode_preview_data);
  223. # sets colors for gcode preview extrusion roles
  224. my @extrusion_roles_colors = (
  225. 'Perimeter' => 'FFA500',
  226. 'External perimeter' => 'FFFF66',
  227. 'Overhang perimeter' => '0000FF',
  228. 'Internal infill' => 'FF0000',
  229. 'Solid infill' => 'CD00CD',
  230. 'Top solid infill' => 'FF3333',
  231. 'Bridge infill' => '9999FF',
  232. 'Gap fill' => 'FFFFFF',
  233. 'Skirt' => '7F0000',
  234. 'Support material' => '00FF00',
  235. 'Support material interface' => '008000',
  236. 'Wipe tower' => 'B3E3AB',
  237. );
  238. $self->gcode_preview_data->set_extrusion_paths_colors(\@extrusion_roles_colors);
  239. $self->show_hide_ui_elements('none');
  240. $self->reload_print;
  241. return $self;
  242. }
  243. sub reload_print {
  244. my ($self, $force) = @_;
  245. $self->canvas->reset_objects;
  246. $self->_loaded(0);
  247. if (! $self->IsShown && ! $force) {
  248. # $self->{reload_delayed} = 1;
  249. return;
  250. }
  251. $self->load_print;
  252. }
  253. sub refresh_print {
  254. my ($self) = @_;
  255. $self->_loaded(0);
  256. if (! $self->IsShown) {
  257. return;
  258. }
  259. $self->load_print;
  260. }
  261. sub load_print {
  262. my ($self) = @_;
  263. return if $self->_loaded;
  264. # we require that there's at least one object and the posSlice step
  265. # is performed on all of them (this ensures that _shifted_copies was
  266. # populated and we know the number of layers)
  267. my $n_layers = 0;
  268. if ($self->print->object_step_done(STEP_SLICE)) {
  269. my %z = (); # z => 1
  270. foreach my $object (@{$self->{print}->objects}) {
  271. foreach my $layer (@{$object->layers}, @{$object->support_layers}) {
  272. $z{$layer->print_z} = 1;
  273. }
  274. }
  275. $self->{layers_z} = [ sort { $a <=> $b } keys %z ];
  276. $n_layers = scalar(@{$self->{layers_z}});
  277. }
  278. if ($n_layers == 0) {
  279. $self->enabled(0);
  280. $self->set_z_range(0,0);
  281. $self->slider_low->Hide;
  282. $self->slider_high->Hide;
  283. $self->{z_label_low}->SetLabel("");
  284. $self->{z_label_high}->SetLabel("");
  285. $self->canvas->reset_legend_texture();
  286. $self->canvas->Refresh; # clears canvas
  287. return;
  288. }
  289. my $z_idx_low = $self->slider_low->GetValue;
  290. my $z_idx_high = $self->slider_high->GetValue;
  291. $self->enabled(1);
  292. $self->slider_low->SetRange(0, $n_layers - 1);
  293. $self->slider_high->SetRange(0, $n_layers - 1);
  294. if ($z_idx_high < $n_layers && ($self->single_layer || $z_idx_high != 0)) {
  295. # use $z_idx
  296. } else {
  297. # Out of range. Disable 'single layer' view.
  298. $self->single_layer(0);
  299. $self->{checkbox_singlelayer}->SetValue(0);
  300. $z_idx_low = 0;
  301. $z_idx_high = $n_layers - 1;
  302. }
  303. if ($self->single_layer) {
  304. $z_idx_low = $z_idx_high;
  305. } elsif ($z_idx_low > $z_idx_high) {
  306. $z_idx_low = 0;
  307. }
  308. $self->slider_low->SetValue($z_idx_low);
  309. $self->slider_high->SetValue($z_idx_high);
  310. $self->slider_low->Show;
  311. $self->slider_high->Show;
  312. $self->Layout;
  313. if ($self->{preferred_color_mode} eq 'tool_or_feature') {
  314. # It is left to Slic3r to decide whether the print shall be colored by the tool or by the feature.
  315. # Color by feature if it is a single extruder print.
  316. my $extruders = $self->{print}->extruders;
  317. my $type = (scalar(@{$extruders}) > 1) ? 4 : 0;
  318. $self->gcode_preview_data->set_type($type);
  319. $self->{choice_view_type}->SetSelection($type);
  320. # If the ->SetSelection changed the following line, revert it to "decide yourself".
  321. $self->{preferred_color_mode} = 'tool_or_feature';
  322. }
  323. # Collect colors per extruder.
  324. my @colors = ();
  325. if (! $self->gcode_preview_data->empty() || $self->gcode_preview_data->type == 4) {
  326. my @extruder_colors = @{$self->{config}->extruder_colour};
  327. my @filament_colors = @{$self->{config}->filament_colour};
  328. for (my $i = 0; $i <= $#extruder_colors; $i += 1) {
  329. my $color = $extruder_colors[$i];
  330. $color = $filament_colors[$i] if (! defined($color) || $color !~ m/^#[[:xdigit:]]{6}/);
  331. $color = '#FFFFFF' if (! defined($color) || $color !~ m/^#[[:xdigit:]]{6}/);
  332. push @colors, $color;
  333. }
  334. }
  335. if ($self->IsShown) {
  336. if ($self->gcode_preview_data->empty) {
  337. # load skirt and brim
  338. $self->canvas->load_print_toolpaths($self->print, \@colors);
  339. $self->canvas->load_wipe_tower_toolpaths($self->print, \@colors);
  340. foreach my $object (@{$self->print->objects}) {
  341. $self->canvas->load_print_object_toolpaths($object, \@colors);
  342. # Show the objects in very transparent color.
  343. #my @volume_ids = $self->canvas->load_object($object->model_object);
  344. #$self->canvas->volumes->[$_]->color->[3] = 0.2 for @volume_ids;
  345. }
  346. $self->show_hide_ui_elements('simple');
  347. } else {
  348. $self->canvas->load_gcode_preview($self->print, $self->gcode_preview_data, \@colors);
  349. $self->show_hide_ui_elements('full');
  350. }
  351. if ($self->auto_zoom) {
  352. $self->canvas->zoom_to_volumes;
  353. }
  354. $self->_loaded(1);
  355. }
  356. $self->set_z_range($self->{layers_z}[$z_idx_low], $self->{layers_z}[$z_idx_high]);
  357. }
  358. sub set_z_range
  359. {
  360. my ($self, $z_low, $z_high) = @_;
  361. return if !$self->enabled;
  362. $self->{z_label_low}->SetLabel(sprintf '%.2f', $z_low);
  363. $self->{z_label_high}->SetLabel(sprintf '%.2f', $z_high);
  364. $self->canvas->set_toolpaths_range($z_low - 1e-6, $z_high + 1e-6);
  365. $self->canvas->Refresh if $self->IsShown;
  366. }
  367. sub set_z_idx_low
  368. {
  369. my ($self, $idx_low) = @_;
  370. if ($self->enabled) {
  371. my $idx_high = $self->slider_high->GetValue;
  372. if ($idx_low >= $idx_high) {
  373. $idx_high = $idx_low;
  374. $self->slider_high->SetValue($idx_high);
  375. }
  376. $self->set_z_range($self->{layers_z}[$idx_low], $self->{layers_z}[$idx_high]);
  377. }
  378. }
  379. sub set_z_idx_high
  380. {
  381. my ($self, $idx_high) = @_;
  382. if ($self->enabled) {
  383. my $idx_low = $self->slider_low->GetValue;
  384. if ($idx_low > $idx_high) {
  385. $idx_low = $idx_high;
  386. $self->slider_low->SetValue($idx_low);
  387. }
  388. $self->set_z_range($self->{layers_z}[$idx_low], $self->{layers_z}[$idx_high]);
  389. }
  390. }
  391. sub set_bed_shape {
  392. my ($self, $bed_shape) = @_;
  393. $self->canvas->set_bed_shape($bed_shape);
  394. }
  395. sub set_number_extruders {
  396. my ($self, $number_extruders) = @_;
  397. if ($self->{number_extruders} != $number_extruders) {
  398. $self->{number_extruders} = $number_extruders;
  399. my $type = ($number_extruders > 1) ?
  400. 4 # color by a tool number
  401. : 0; # color by a feature type
  402. $self->{choice_view_type}->SetSelection($type);
  403. $self->gcode_preview_data->set_type($type);
  404. $self->{preferred_color_mode} = ($type == 4) ? 'tool_or_feature' : 'feature';
  405. }
  406. }
  407. sub show_hide_ui_elements {
  408. my ($self, $what) = @_;
  409. my $method = ($what eq 'full') ? 'Enable' : 'Disable';
  410. $self->{$_}->$method for qw(label_show_features combochecklist_features checkbox_travel checkbox_retractions checkbox_unretractions checkbox_shells);
  411. $method = ($what eq 'none') ? 'Disable' : 'Enable';
  412. $self->{$_}->$method for qw(label_view_type choice_view_type);
  413. }
  414. # Called by the Platter wxNotebook when this page is activated.
  415. sub OnActivate {
  416. # my ($self) = @_;
  417. # $self->reload_print(1) if ($self->{reload_delayed});
  418. }
  419. 1;