3D.pm 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. package Slic3r::GUI::Plater::3D;
  2. use strict;
  3. use warnings;
  4. use utf8;
  5. use List::Util qw();
  6. use Wx qw(:misc :pen :brush :sizer :font :cursor :keycode wxTAB_TRAVERSAL);
  7. use Wx::Event qw(EVT_KEY_DOWN EVT_CHAR);
  8. use base qw(Slic3r::GUI::3DScene Class::Accessor);
  9. use Wx::Locale gettext => 'L';
  10. __PACKAGE__->mk_accessors(qw(
  11. on_arrange on_rotate_object_left on_rotate_object_right on_scale_object_uniformly
  12. on_remove_object on_increase_objects on_decrease_objects on_enable_action_buttons));
  13. sub new {
  14. my $class = shift;
  15. my ($parent, $objects, $model, $print, $config) = @_;
  16. my $self = $class->SUPER::new($parent);
  17. #==============================================================================================================================
  18. Slic3r::GUI::_3DScene::enable_picking($self, 1);
  19. # $self->enable_picking(1);
  20. #==============================================================================================================================
  21. $self->enable_moving(1);
  22. $self->select_by('object');
  23. $self->drag_by('instance');
  24. $self->{objects} = $objects;
  25. $self->{model} = $model;
  26. $self->{print} = $print;
  27. $self->{config} = $config;
  28. #==============================================================================================================================
  29. Slic3r::GUI::_3DScene::set_config($self, $config);
  30. #==============================================================================================================================
  31. $self->{on_select_object} = sub {};
  32. $self->{on_instances_moved} = sub {};
  33. $self->{on_wipe_tower_moved} = sub {};
  34. $self->{objects_volumes_idxs} = [];
  35. $self->on_select(sub {
  36. my ($volume_idx) = @_;
  37. $self->{on_select_object}->(($volume_idx == -1) ? undef : $self->volumes->[$volume_idx]->object_idx)
  38. if ($self->{on_select_object});
  39. });
  40. $self->on_move(sub {
  41. my @volume_idxs = @_;
  42. my %done = (); # prevent moving instances twice
  43. my $object_moved;
  44. my $wipe_tower_moved;
  45. foreach my $volume_idx (@volume_idxs) {
  46. my $volume = $self->volumes->[$volume_idx];
  47. my $obj_idx = $volume->object_idx;
  48. my $instance_idx = $volume->instance_idx;
  49. next if $done{"${obj_idx}_${instance_idx}"};
  50. $done{"${obj_idx}_${instance_idx}"} = 1;
  51. if ($obj_idx < 1000) {
  52. # Move a regular object.
  53. my $model_object = $self->{model}->get_object($obj_idx);
  54. $model_object
  55. ->instances->[$instance_idx]
  56. ->offset
  57. ->translate($volume->origin->x, $volume->origin->y); #))
  58. $model_object->invalidate_bounding_box;
  59. $object_moved = 1;
  60. } elsif ($obj_idx == 1000) {
  61. # Move a wipe tower proxy.
  62. $wipe_tower_moved = $volume->origin;
  63. }
  64. }
  65. $self->{on_instances_moved}->()
  66. if $object_moved && $self->{on_instances_moved};
  67. $self->{on_wipe_tower_moved}->($wipe_tower_moved)
  68. if $wipe_tower_moved && $self->{on_wipe_tower_moved};
  69. });
  70. EVT_KEY_DOWN($self, sub {
  71. my ($s, $event) = @_;
  72. if ($event->HasModifiers) {
  73. $event->Skip;
  74. } else {
  75. my $key = $event->GetKeyCode;
  76. if ($key == WXK_DELETE) {
  77. $self->on_remove_object->() if $self->on_remove_object;
  78. } else {
  79. $event->Skip;
  80. }
  81. }
  82. });
  83. EVT_CHAR($self, sub {
  84. my ($s, $event) = @_;
  85. if ($event->HasModifiers) {
  86. $event->Skip;
  87. } else {
  88. my $key = $event->GetKeyCode;
  89. if ($key == ord('a')) {
  90. $self->on_arrange->() if $self->on_arrange;
  91. } elsif ($key == ord('l')) {
  92. $self->on_rotate_object_left->() if $self->on_rotate_object_left;
  93. } elsif ($key == ord('r')) {
  94. $self->on_rotate_object_right->() if $self->on_rotate_object_right;
  95. } elsif ($key == ord('s')) {
  96. $self->on_scale_object_uniformly->() if $self->on_scale_object_uniformly;
  97. } elsif ($key == ord('+')) {
  98. $self->on_increase_objects->() if $self->on_increase_objects;
  99. } elsif ($key == ord('-')) {
  100. $self->on_decrease_objects->() if $self->on_decrease_objects;
  101. } else {
  102. $event->Skip;
  103. }
  104. }
  105. });
  106. return $self;
  107. }
  108. sub set_on_select_object {
  109. my ($self, $cb) = @_;
  110. $self->{on_select_object} = $cb;
  111. }
  112. sub set_on_double_click {
  113. my ($self, $cb) = @_;
  114. $self->on_double_click($cb);
  115. }
  116. sub set_on_right_click {
  117. my ($self, $cb) = @_;
  118. $self->on_right_click($cb);
  119. }
  120. sub set_on_arrange {
  121. my ($self, $cb) = @_;
  122. $self->on_arrange($cb);
  123. }
  124. sub set_on_rotate_object_left {
  125. my ($self, $cb) = @_;
  126. $self->on_rotate_object_left($cb);
  127. }
  128. sub set_on_rotate_object_right {
  129. my ($self, $cb) = @_;
  130. $self->on_rotate_object_right($cb);
  131. }
  132. sub set_on_scale_object_uniformly {
  133. my ($self, $cb) = @_;
  134. $self->on_scale_object_uniformly($cb);
  135. }
  136. sub set_on_increase_objects {
  137. my ($self, $cb) = @_;
  138. $self->on_increase_objects($cb);
  139. }
  140. sub set_on_decrease_objects {
  141. my ($self, $cb) = @_;
  142. $self->on_decrease_objects($cb);
  143. }
  144. sub set_on_remove_object {
  145. my ($self, $cb) = @_;
  146. $self->on_remove_object($cb);
  147. }
  148. sub set_on_instances_moved {
  149. my ($self, $cb) = @_;
  150. $self->{on_instances_moved} = $cb;
  151. }
  152. sub set_on_wipe_tower_moved {
  153. my ($self, $cb) = @_;
  154. $self->{on_wipe_tower_moved} = $cb;
  155. }
  156. sub set_on_model_update {
  157. my ($self, $cb) = @_;
  158. $self->on_model_update($cb);
  159. }
  160. sub set_on_enable_action_buttons {
  161. my ($self, $cb) = @_;
  162. $self->on_enable_action_buttons($cb);
  163. }
  164. sub update_volumes_selection {
  165. my ($self) = @_;
  166. foreach my $obj_idx (0..$#{$self->{model}->objects}) {
  167. if ($self->{objects}[$obj_idx]->selected) {
  168. my $volume_idxs = $self->{objects_volumes_idxs}->[$obj_idx];
  169. #==============================================================================================================================
  170. Slic3r::GUI::_3DScene::select_volume($self, $_) for @{$volume_idxs};
  171. # $self->select_volume($_) for @{$volume_idxs};
  172. #==============================================================================================================================
  173. }
  174. }
  175. }
  176. sub reload_scene {
  177. my ($self, $force) = @_;
  178. #==============================================================================================================================
  179. Slic3r::GUI::_3DScene::reset_volumes($self);
  180. # $self->reset_objects;
  181. #==============================================================================================================================
  182. $self->update_bed_size;
  183. if (! $self->IsShown && ! $force) {
  184. $self->{reload_delayed} = 1;
  185. return;
  186. }
  187. $self->{reload_delayed} = 0;
  188. $self->{objects_volumes_idxs} = [];
  189. foreach my $obj_idx (0..$#{$self->{model}->objects}) {
  190. my @volume_idxs = $self->load_object($self->{model}, $self->{print}, $obj_idx);
  191. push(@{$self->{objects_volumes_idxs}}, \@volume_idxs);
  192. }
  193. $self->update_volumes_selection;
  194. if (defined $self->{config}->nozzle_diameter) {
  195. # Should the wipe tower be visualized?
  196. my $extruders_count = scalar @{ $self->{config}->nozzle_diameter };
  197. # Height of a print.
  198. my $height = $self->{model}->bounding_box->z_max;
  199. # Show at least a slab.
  200. $height = 10 if $height < 10;
  201. if ($extruders_count > 1 && $self->{config}->single_extruder_multi_material && $self->{config}->wipe_tower &&
  202. ! $self->{config}->complete_objects) {
  203. $self->volumes->load_wipe_tower_preview(1000,
  204. $self->{config}->wipe_tower_x, $self->{config}->wipe_tower_y, $self->{config}->wipe_tower_width,
  205. #$self->{config}->wipe_tower_per_color_wipe# 15 * ($extruders_count - 1), # this is just a hack when the config parameter became obsolete
  206. 15 * ($extruders_count - 1),
  207. $self->{model}->bounding_box->z_max, $self->{config}->wipe_tower_rotation_angle, $self->UseVBOs);
  208. }
  209. }
  210. $self->update_volumes_colors_by_extruder($self->{config});
  211. # checks for geometry outside the print volume to render it accordingly
  212. if (scalar @{$self->volumes} > 0)
  213. {
  214. my $contained = $self->volumes->check_outside_state($self->{config});
  215. if (!$contained) {
  216. #==============================================================================================================================
  217. Slic3r::GUI::_3DScene::enable_warning_texture($self, 1);
  218. # $self->set_warning_enabled(1);
  219. #==============================================================================================================================
  220. Slic3r::GUI::_3DScene::generate_warning_texture(L("Detected object outside print volume"));
  221. $self->on_enable_action_buttons->(0) if ($self->on_enable_action_buttons);
  222. } else {
  223. #==============================================================================================================================
  224. Slic3r::GUI::_3DScene::enable_warning_texture($self, 0);
  225. # $self->set_warning_enabled(0);
  226. #==============================================================================================================================
  227. $self->volumes->reset_outside_state();
  228. Slic3r::GUI::_3DScene::reset_warning_texture();
  229. $self->on_enable_action_buttons->(scalar @{$self->{model}->objects} > 0) if ($self->on_enable_action_buttons);
  230. }
  231. } else {
  232. #==============================================================================================================================
  233. Slic3r::GUI::_3DScene::enable_warning_texture($self, 0);
  234. # $self->set_warning_enabled(0);
  235. #==============================================================================================================================
  236. Slic3r::GUI::_3DScene::reset_warning_texture();
  237. }
  238. }
  239. sub update_bed_size {
  240. my ($self) = @_;
  241. #==============================================================================================================================
  242. Slic3r::GUI::_3DScene::set_bed_shape($self, $self->{config}->bed_shape);
  243. # $self->set_bed_shape($self->{config}->bed_shape);
  244. #==============================================================================================================================
  245. }
  246. # Called by the Platter wxNotebook when this page is activated.
  247. sub OnActivate {
  248. my ($self) = @_;
  249. $self->reload_scene(1) if ($self->{reload_delayed});
  250. }
  251. 1;