3D.pm 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  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. __PACKAGE__->mk_accessors(qw(
  10. on_arrange on_rotate_object_left on_rotate_object_right on_scale_object_uniformly
  11. on_remove_object on_increase_objects on_decrease_objects));
  12. sub new {
  13. my $class = shift;
  14. my ($parent, $objects, $model, $print, $config) = @_;
  15. my $self = $class->SUPER::new($parent);
  16. $self->enable_picking(1);
  17. $self->enable_moving(1);
  18. $self->select_by('object');
  19. $self->drag_by('instance');
  20. $self->{objects} = $objects;
  21. $self->{model} = $model;
  22. $self->{print} = $print;
  23. $self->{config} = $config;
  24. $self->{on_select_object} = sub {};
  25. $self->{on_instances_moved} = sub {};
  26. $self->{on_wipe_tower_moved} = sub {};
  27. $self->on_select(sub {
  28. my ($volume_idx) = @_;
  29. $self->{on_select_object}->(($volume_idx == -1) ? undef : $self->volumes->[$volume_idx]->object_idx)
  30. if ($self->{on_select_object});
  31. });
  32. $self->on_move(sub {
  33. my @volume_idxs = @_;
  34. my %done = (); # prevent moving instances twice
  35. my $object_moved;
  36. my $wipe_tower_moved;
  37. foreach my $volume_idx (@volume_idxs) {
  38. my $volume = $self->volumes->[$volume_idx];
  39. my $obj_idx = $volume->object_idx;
  40. my $instance_idx = $volume->instance_idx;
  41. next if $done{"${obj_idx}_${instance_idx}"};
  42. $done{"${obj_idx}_${instance_idx}"} = 1;
  43. if ($obj_idx < 1000) {
  44. # Move a regular object.
  45. my $model_object = $self->{model}->get_object($obj_idx);
  46. $model_object
  47. ->instances->[$instance_idx]
  48. ->offset
  49. ->translate($volume->origin->x, $volume->origin->y); #))
  50. $model_object->invalidate_bounding_box;
  51. $object_moved = 1;
  52. } elsif ($obj_idx == 1000) {
  53. # Move a wipe tower proxy.
  54. $wipe_tower_moved = $volume->origin;
  55. }
  56. }
  57. $self->{on_instances_moved}->()
  58. if $object_moved && $self->{on_instances_moved};
  59. $self->{on_wipe_tower_moved}->($wipe_tower_moved)
  60. if $wipe_tower_moved && $self->{on_wipe_tower_moved};
  61. });
  62. EVT_KEY_DOWN($self, sub {
  63. my ($s, $event) = @_;
  64. if ($event->HasModifiers) {
  65. $event->Skip;
  66. } else {
  67. my $key = $event->GetKeyCode;
  68. if ($key == WXK_DELETE) {
  69. $self->on_remove_object->() if $self->on_remove_object;
  70. } else {
  71. $event->Skip;
  72. }
  73. }
  74. });
  75. EVT_CHAR($self, sub {
  76. my ($s, $event) = @_;
  77. if ($event->HasModifiers) {
  78. $event->Skip;
  79. } else {
  80. my $key = $event->GetKeyCode;
  81. if ($key == ord('a')) {
  82. $self->on_arrange->() if $self->on_arrange;
  83. } elsif ($key == ord('l')) {
  84. $self->on_rotate_object_left->() if $self->on_rotate_object_left;
  85. } elsif ($key == ord('r')) {
  86. $self->on_rotate_object_right->() if $self->on_rotate_object_right;
  87. } elsif ($key == ord('s')) {
  88. $self->on_scale_object_uniformly->() if $self->on_scale_object_uniformly;
  89. } elsif ($key == ord('+')) {
  90. $self->on_increase_objects->() if $self->on_increase_objects;
  91. } elsif ($key == ord('-')) {
  92. $self->on_decrease_objects->() if $self->on_decrease_objects;
  93. } else {
  94. $event->Skip;
  95. }
  96. }
  97. });
  98. return $self;
  99. }
  100. sub set_on_select_object {
  101. my ($self, $cb) = @_;
  102. $self->{on_select_object} = $cb;
  103. }
  104. sub set_on_double_click {
  105. my ($self, $cb) = @_;
  106. $self->on_double_click($cb);
  107. }
  108. sub set_on_right_click {
  109. my ($self, $cb) = @_;
  110. $self->on_right_click($cb);
  111. }
  112. sub set_on_arrange {
  113. my ($self, $cb) = @_;
  114. $self->on_arrange($cb);
  115. }
  116. sub set_on_rotate_object_left {
  117. my ($self, $cb) = @_;
  118. $self->on_rotate_object_left($cb);
  119. }
  120. sub set_on_rotate_object_right {
  121. my ($self, $cb) = @_;
  122. $self->on_rotate_object_right($cb);
  123. }
  124. sub set_on_scale_object_uniformly {
  125. my ($self, $cb) = @_;
  126. $self->on_scale_object_uniformly($cb);
  127. }
  128. sub set_on_increase_objects {
  129. my ($self, $cb) = @_;
  130. $self->on_increase_objects($cb);
  131. }
  132. sub set_on_decrease_objects {
  133. my ($self, $cb) = @_;
  134. $self->on_decrease_objects($cb);
  135. }
  136. sub set_on_remove_object {
  137. my ($self, $cb) = @_;
  138. $self->on_remove_object($cb);
  139. }
  140. sub set_on_instances_moved {
  141. my ($self, $cb) = @_;
  142. $self->{on_instances_moved} = $cb;
  143. }
  144. sub set_on_wipe_tower_moved {
  145. my ($self, $cb) = @_;
  146. $self->{on_wipe_tower_moved} = $cb;
  147. }
  148. sub set_on_model_update {
  149. my ($self, $cb) = @_;
  150. $self->on_model_update($cb);
  151. }
  152. sub reload_scene {
  153. my ($self, $force) = @_;
  154. $self->reset_objects;
  155. $self->update_bed_size;
  156. if (! $self->IsShown && ! $force) {
  157. $self->{reload_delayed} = 1;
  158. return;
  159. }
  160. $self->{reload_delayed} = 0;
  161. foreach my $obj_idx (0..$#{$self->{model}->objects}) {
  162. my @volume_idxs = $self->load_object($self->{model}, $self->{print}, $obj_idx);
  163. if ($self->{objects}[$obj_idx]->selected) {
  164. $self->select_volume($_) for @volume_idxs;
  165. }
  166. }
  167. if (defined $self->{config}->nozzle_diameter) {
  168. # Should the wipe tower be visualized?
  169. my $extruders_count = scalar @{ $self->{config}->nozzle_diameter };
  170. # Height of a print.
  171. my $height = $self->{model}->bounding_box->z_max;
  172. # Show at least a slab.
  173. $height = 10 if $height < 10;
  174. if ($extruders_count > 1 && $self->{config}->single_extruder_multi_material && $self->{config}->wipe_tower &&
  175. ! $self->{config}->complete_objects) {
  176. $self->volumes->load_wipe_tower_preview(1000,
  177. $self->{config}->wipe_tower_x, $self->{config}->wipe_tower_y,
  178. $self->{config}->wipe_tower_width, $self->{config}->wipe_tower_per_color_wipe * ($extruders_count - 1),
  179. $self->{model}->bounding_box->z_max, $self->{config}->wipe_tower_rotation_angle, $self->UseVBOs);
  180. }
  181. }
  182. }
  183. sub update_bed_size {
  184. my ($self) = @_;
  185. $self->set_bed_shape($self->{config}->bed_shape);
  186. }
  187. # Called by the Platter wxNotebook when this page is activated.
  188. sub OnActivate {
  189. my ($self) = @_;
  190. $self->reload_scene(1) if ($self->{reload_delayed});
  191. }
  192. 1;