3DPreview.pm 17 KB

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