ObjectPartsPanel.pm 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645
  1. # Configuration of mesh modifiers and their parameters.
  2. # This panel is inserted into ObjectSettingsDialog.
  3. package Slic3r::GUI::Plater::ObjectPartsPanel;
  4. use strict;
  5. use warnings;
  6. use utf8;
  7. use File::Basename qw(basename);
  8. use Wx qw(:misc :sizer :treectrl :button :keycode wxTAB_TRAVERSAL wxSUNKEN_BORDER wxBITMAP_TYPE_PNG wxID_CANCEL wxMOD_CONTROL
  9. wxTheApp);
  10. use Wx::Event qw(EVT_BUTTON EVT_TREE_ITEM_COLLAPSING EVT_TREE_SEL_CHANGED EVT_TREE_KEY_DOWN EVT_KEY_DOWN);
  11. use List::Util qw(max);
  12. use base 'Wx::Panel';
  13. use constant ICON_OBJECT => 0;
  14. use constant ICON_SOLIDMESH => 1;
  15. use constant ICON_MODIFIERMESH => 2;
  16. use constant ICON_SUPPORT_ENFORCER => 3;
  17. use constant ICON_SUPPORT_BLOCKER => 4;
  18. sub new {
  19. my ($class, $parent, %params) = @_;
  20. my $self = $class->SUPER::new($parent, -1, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL);
  21. # C++ type Slic3r::ModelObject
  22. my $object = $self->{model_object} = $params{model_object};
  23. # Save state for sliders.
  24. $self->{move_options} = {
  25. x => 0,
  26. y => 0,
  27. z => 0,
  28. };
  29. $self->{last_coords} = {
  30. x => 0,
  31. y => 0,
  32. z => 0,
  33. };
  34. # create TreeCtrl
  35. my $tree = $self->{tree} = Wx::TreeCtrl->new($self, -1, wxDefaultPosition, [300, 100],
  36. wxTR_NO_BUTTONS | wxSUNKEN_BORDER | wxTR_HAS_VARIABLE_ROW_HEIGHT
  37. | wxTR_SINGLE);
  38. {
  39. $self->{tree_icons} = Wx::ImageList->new(16, 16, 1);
  40. $tree->AssignImageList($self->{tree_icons});
  41. $self->{tree_icons}->Add(Wx::Bitmap->new(Slic3r::var("brick.png"), wxBITMAP_TYPE_PNG)); # ICON_OBJECT
  42. $self->{tree_icons}->Add(Wx::Bitmap->new(Slic3r::var("package.png"), wxBITMAP_TYPE_PNG)); # ICON_SOLIDMESH
  43. $self->{tree_icons}->Add(Wx::Bitmap->new(Slic3r::var("plugin.png"), wxBITMAP_TYPE_PNG)); # ICON_MODIFIERMESH
  44. $self->{tree_icons}->Add(Wx::Bitmap->new(Slic3r::var("support_enforcer.png"), wxBITMAP_TYPE_PNG)); # ICON_SUPPORT_ENFORCER
  45. $self->{tree_icons}->Add(Wx::Bitmap->new(Slic3r::var("support_blocker.png"), wxBITMAP_TYPE_PNG)); # ICON_SUPPORT_BLOCKER
  46. my $rootId = $tree->AddRoot("Object", ICON_OBJECT);
  47. $tree->SetPlData($rootId, { type => 'object' });
  48. }
  49. # buttons
  50. $self->{btn_load_part} = Wx::Button->new($self, -1, "Load part…", wxDefaultPosition, wxDefaultSize, wxBU_LEFT);
  51. $self->{btn_load_modifier} = Wx::Button->new($self, -1, "Load modifier…", wxDefaultPosition, wxDefaultSize, wxBU_LEFT);
  52. $self->{btn_load_lambda_modifier} = Wx::Button->new($self, -1, "Load generic…", wxDefaultPosition, wxDefaultSize, wxBU_LEFT);
  53. $self->{btn_delete} = Wx::Button->new($self, -1, "Delete part", wxDefaultPosition, wxDefaultSize, wxBU_LEFT);
  54. $self->{btn_split} = Wx::Button->new($self, -1, "Split part", wxDefaultPosition, wxDefaultSize, wxBU_LEFT);
  55. $self->{btn_move_up} = Wx::Button->new($self, -1, "", wxDefaultPosition, [40, -1], wxBU_LEFT);
  56. $self->{btn_move_down} = Wx::Button->new($self, -1, "", wxDefaultPosition, [40, -1], wxBU_LEFT);
  57. $self->{btn_load_part}->SetBitmap(Wx::Bitmap->new(Slic3r::var("brick_add.png"), wxBITMAP_TYPE_PNG));
  58. $self->{btn_load_modifier}->SetBitmap(Wx::Bitmap->new(Slic3r::var("brick_add.png"), wxBITMAP_TYPE_PNG));
  59. $self->{btn_load_lambda_modifier}->SetBitmap(Wx::Bitmap->new(Slic3r::var("brick_add.png"), wxBITMAP_TYPE_PNG));
  60. $self->{btn_delete}->SetBitmap(Wx::Bitmap->new(Slic3r::var("brick_delete.png"), wxBITMAP_TYPE_PNG));
  61. $self->{btn_split}->SetBitmap(Wx::Bitmap->new(Slic3r::var("shape_ungroup.png"), wxBITMAP_TYPE_PNG));
  62. $self->{btn_move_up}->SetBitmap(Wx::Bitmap->new(Slic3r::var("bullet_arrow_up.png"), wxBITMAP_TYPE_PNG));
  63. $self->{btn_move_down}->SetBitmap(Wx::Bitmap->new(Slic3r::var("bullet_arrow_down.png"), wxBITMAP_TYPE_PNG));
  64. # buttons sizer
  65. my $buttons_sizer = Wx::GridSizer->new(2, 3);
  66. $buttons_sizer->Add($self->{btn_load_part}, 0, wxEXPAND | wxBOTTOM | wxRIGHT, 5);
  67. $buttons_sizer->Add($self->{btn_load_modifier}, 0, wxEXPAND | wxBOTTOM | wxRIGHT, 5);
  68. $buttons_sizer->Add($self->{btn_load_lambda_modifier}, 0, wxEXPAND | wxBOTTOM, 5);
  69. $buttons_sizer->Add($self->{btn_delete}, 0, wxEXPAND | wxRIGHT, 5);
  70. $buttons_sizer->Add($self->{btn_split}, 0, wxEXPAND | wxRIGHT, 5);
  71. {
  72. my $up_down_sizer = Wx::GridSizer->new(1, 2);
  73. $up_down_sizer->Add($self->{btn_move_up}, 0, wxEXPAND | wxRIGHT, 5);
  74. $up_down_sizer->Add($self->{btn_move_down}, 0, wxEXPAND, 5);
  75. $buttons_sizer->Add($up_down_sizer, 0, wxEXPAND, 5);
  76. }
  77. $self->{btn_load_part}->SetFont($Slic3r::GUI::small_font);
  78. $self->{btn_load_modifier}->SetFont($Slic3r::GUI::small_font);
  79. $self->{btn_load_lambda_modifier}->SetFont($Slic3r::GUI::small_font);
  80. $self->{btn_delete}->SetFont($Slic3r::GUI::small_font);
  81. $self->{btn_split}->SetFont($Slic3r::GUI::small_font);
  82. $self->{btn_move_up}->SetFont($Slic3r::GUI::small_font);
  83. $self->{btn_move_down}->SetFont($Slic3r::GUI::small_font);
  84. # part settings panel
  85. $self->{settings_panel} = Slic3r::GUI::Plater::OverrideSettingsPanel->new($self, on_change => sub {
  86. my ($key, $value) = @_;
  87. wxTheApp->CallAfter(sub {
  88. $self->set_part_type($value) if ($key eq "part_type");
  89. $self->{part_settings_changed} = 1;
  90. $self->_update_canvas;
  91. });
  92. });
  93. my $settings_sizer = Wx::StaticBoxSizer->new($self->{staticbox} = Wx::StaticBox->new($self, -1, "Part Settings"), wxVERTICAL);
  94. $settings_sizer->Add($self->{settings_panel}, 1, wxEXPAND | wxALL, 0);
  95. my $optgroup_movers;
  96. $optgroup_movers = $self->{optgroup_movers} = Slic3r::GUI::OptionsGroup->new(
  97. parent => $self,
  98. title => 'Move',
  99. on_change => sub {
  100. my ($opt_id) = @_;
  101. # There seems to be an issue with wxWidgets 3.0.2/3.0.3, where the slider
  102. # genates tens of events for a single value change.
  103. # Only trigger the recalculation if the value changes
  104. # or a live preview was activated and the mesh cut is not valid yet.
  105. if ($self->{move_options}{$opt_id} != $optgroup_movers->get_value($opt_id)) {
  106. $self->{move_options}{$opt_id} = $optgroup_movers->get_value($opt_id);
  107. wxTheApp->CallAfter(sub {
  108. $self->_update;
  109. });
  110. }
  111. },
  112. label_width => 20,
  113. );
  114. $optgroup_movers->append_single_option_line(Slic3r::GUI::OptionsGroup::Option->new(
  115. opt_id => 'x',
  116. type => 'slider',
  117. label => 'X',
  118. default => 0,
  119. min => -($self->{model_object}->bounding_box->size->x)*4,
  120. max => $self->{model_object}->bounding_box->size->x*4,
  121. full_width => 1,
  122. ));
  123. $optgroup_movers->append_single_option_line(Slic3r::GUI::OptionsGroup::Option->new(
  124. opt_id => 'y',
  125. type => 'slider',
  126. label => 'Y',
  127. default => 0,
  128. min => -($self->{model_object}->bounding_box->size->y)*4,
  129. max => $self->{model_object}->bounding_box->size->y*4,
  130. full_width => 1,
  131. ));
  132. $optgroup_movers->append_single_option_line(Slic3r::GUI::OptionsGroup::Option->new(
  133. opt_id => 'z',
  134. type => 'slider',
  135. label => 'Z',
  136. default => 0,
  137. min => -($self->{model_object}->bounding_box->size->z)*4,
  138. max => $self->{model_object}->bounding_box->size->z*4,
  139. full_width => 1,
  140. ));
  141. # left pane with tree
  142. my $left_sizer = Wx::BoxSizer->new(wxVERTICAL);
  143. $left_sizer->Add($tree, 3, wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM, 10);
  144. $left_sizer->Add($buttons_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT | wxBOTTOM, 10);
  145. $left_sizer->Add($settings_sizer, 5, wxEXPAND | wxALL, 0);
  146. $left_sizer->Add($optgroup_movers->sizer, 0, wxEXPAND | wxBOTTOM | wxLEFT | wxRIGHT, 10);
  147. # right pane with preview canvas
  148. my $canvas;
  149. if ($Slic3r::GUI::have_OpenGL) {
  150. $canvas = $self->{canvas} = Slic3r::GUI::3DScene->new($self);
  151. Slic3r::GUI::_3DScene::enable_picking($canvas, 1);
  152. Slic3r::GUI::_3DScene::set_select_by($canvas, 'volume');
  153. Slic3r::GUI::_3DScene::register_on_select_object_callback($canvas, sub {
  154. my ($volume_idx) = @_;
  155. $self->reload_tree($volume_idx);
  156. });
  157. Slic3r::GUI::_3DScene::load_model_object($canvas, $self->{model_object}, 0, [0]);
  158. Slic3r::GUI::_3DScene::set_auto_bed_shape($canvas);
  159. Slic3r::GUI::_3DScene::set_axes_length($canvas, 2.0 * max(@{ Slic3r::GUI::_3DScene::get_volumes_bounding_box($canvas)->size }));
  160. $canvas->SetSize([500,700]);
  161. Slic3r::GUI::_3DScene::set_config($canvas, $self->GetParent->GetParent->GetParent->{config});
  162. Slic3r::GUI::_3DScene::update_volumes_colors_by_extruder($canvas);
  163. Slic3r::GUI::_3DScene::enable_force_zoom_to_bed($canvas, 1);
  164. }
  165. $self->{sizer} = Wx::BoxSizer->new(wxHORIZONTAL);
  166. $self->{sizer}->Add($left_sizer, 0, wxEXPAND | wxALL, 0);
  167. $self->{sizer}->Add($canvas, 1, wxEXPAND | wxALL, 0) if $canvas;
  168. $self->SetSizer($self->{sizer});
  169. $self->{sizer}->SetSizeHints($self);
  170. # attach events
  171. EVT_TREE_ITEM_COLLAPSING($self, $tree, sub {
  172. my ($self, $event) = @_;
  173. $event->Veto;
  174. });
  175. EVT_TREE_SEL_CHANGED($self, $tree, sub {
  176. my ($self, $event) = @_;
  177. return if $self->{disable_tree_sel_changed_event};
  178. $self->selection_changed;
  179. });
  180. EVT_TREE_KEY_DOWN($self, $tree, \&on_tree_key_down);
  181. EVT_BUTTON($self, $self->{btn_load_part}, sub { $self->on_btn_load(0) });
  182. EVT_BUTTON($self, $self->{btn_load_modifier}, sub { $self->on_btn_load(1) });
  183. EVT_BUTTON($self, $self->{btn_load_lambda_modifier}, sub { $self->on_btn_lambda(1) });
  184. EVT_BUTTON($self, $self->{btn_delete}, \&on_btn_delete);
  185. EVT_BUTTON($self, $self->{btn_split}, \&on_btn_split);
  186. EVT_BUTTON($self, $self->{btn_move_up}, \&on_btn_move_up);
  187. EVT_BUTTON($self, $self->{btn_move_down}, \&on_btn_move_down);
  188. EVT_KEY_DOWN($canvas, sub {
  189. my ($canvas, $event) = @_;
  190. if ($event->GetKeyCode == WXK_DELETE) {
  191. $canvas->GetParent->on_btn_delete;
  192. } else {
  193. $event->Skip;
  194. }
  195. });
  196. $self->reload_tree;
  197. return $self;
  198. }
  199. sub reload_tree {
  200. my ($self, $selected_volume_idx) = @_;
  201. $selected_volume_idx //= -1;
  202. my $object = $self->{model_object};
  203. my $tree = $self->{tree};
  204. my $rootId = $tree->GetRootItem;
  205. # despite wxWidgets states that DeleteChildren "will not generate any events unlike Delete() method",
  206. # the MSW implementation of DeleteChildren actually calls Delete() for each item, so
  207. # EVT_TREE_SEL_CHANGED is being called, with bad effects (the event handler is called; this
  208. # subroutine is never continued; an invisible EndModal is called on the dialog causing Plater
  209. # to continue its logic and rescheduling the background process etc. GH #2774)
  210. $self->{disable_tree_sel_changed_event} = 1;
  211. $tree->DeleteChildren($rootId);
  212. $self->{disable_tree_sel_changed_event} = 0;
  213. my $selectedId = $rootId;
  214. foreach my $volume_id (0..$#{$object->volumes}) {
  215. my $volume = $object->volumes->[$volume_id];
  216. my $icon =
  217. $volume->modifier ? ICON_MODIFIERMESH :
  218. $volume->support_enforcer ? ICON_SUPPORT_ENFORCER :
  219. $volume->support_blocker ? ICON_SUPPORT_BLOCKER :
  220. ICON_SOLIDMESH;
  221. my $itemId = $tree->AppendItem($rootId, $volume->name || $volume_id, $icon);
  222. if ($volume_id == $selected_volume_idx) {
  223. $selectedId = $itemId;
  224. }
  225. $tree->SetPlData($itemId, {
  226. type => 'volume',
  227. volume_id => $volume_id,
  228. });
  229. }
  230. $tree->ExpandAll;
  231. Slic3r::GUI->CallAfter(sub {
  232. $self->{tree}->SelectItem($selectedId);
  233. # SelectItem() should trigger EVT_TREE_SEL_CHANGED as per wxWidgets docs,
  234. # but in fact it doesn't if the given item is already selected (this happens
  235. # on first load)
  236. $self->selection_changed;
  237. });
  238. }
  239. sub get_selection {
  240. my ($self) = @_;
  241. my $nodeId = $self->{tree}->GetSelection;
  242. if ($nodeId->IsOk) {
  243. return $self->{tree}->GetPlData($nodeId);
  244. }
  245. return undef;
  246. }
  247. sub selection_changed {
  248. my ($self) = @_;
  249. # deselect all meshes
  250. if ($self->{canvas}) {
  251. Slic3r::GUI::_3DScene::deselect_volumes($self->{canvas});
  252. }
  253. # disable things as if nothing is selected
  254. $self->{'btn_' . $_}->Disable for (qw(delete move_up move_down split));
  255. $self->{settings_panel}->disable;
  256. $self->{settings_panel}->set_config(undef);
  257. # reset move sliders
  258. $self->{optgroup_movers}->set_value("x", 0);
  259. $self->{optgroup_movers}->set_value("y", 0);
  260. $self->{optgroup_movers}->set_value("z", 0);
  261. $self->{move_options} = {
  262. x => 0,
  263. y => 0,
  264. z => 0,
  265. };
  266. $self->{last_coords} = {
  267. x => 0,
  268. y => 0,
  269. z => 0,
  270. };
  271. if (my $itemData = $self->get_selection) {
  272. my ($config, @opt_keys);
  273. my $type = Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_OBJECT;
  274. my $support = 0;
  275. if ($itemData->{type} eq 'volume') {
  276. # select volume in 3D preview
  277. if ($self->{canvas}) {
  278. Slic3r::GUI::_3DScene::select_volume($self->{canvas}, $itemData->{volume_id});
  279. }
  280. $self->{btn_delete}->Enable;
  281. $self->{btn_split}->Enable;
  282. $self->{btn_move_up}->Enable if $itemData->{volume_id} > 0;
  283. $self->{btn_move_down}->Enable if $itemData->{volume_id} + 1 < $self->{model_object}->volumes_count;
  284. # attach volume config to settings panel
  285. my $volume = $self->{model_object}->volumes->[ $itemData->{volume_id} ];
  286. if (! $volume->model_part) {
  287. $self->{optgroup_movers}->enable;
  288. if ($volume->support_enforcer || $volume->support_blocker) {
  289. $support = 1;
  290. $type = $volume->support_enforcer ?
  291. Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_SUPPORT_ENFORCER :
  292. Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_SUPPORT_BLOCKER;
  293. } else {
  294. $type = Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_MODIFIER;
  295. }
  296. } else {
  297. $type = Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_PART;
  298. $self->{optgroup_movers}->disable;
  299. }
  300. $config = $volume->config;
  301. $self->{staticbox}->SetLabel('Part Settings');
  302. # get default values
  303. @opt_keys = $support ? () : @{Slic3r::Config::PrintRegion->new->get_keys};
  304. } elsif ($itemData->{type} eq 'object') {
  305. # select nothing in 3D preview
  306. # attach object config to settings panel
  307. $self->{optgroup_movers}->disable;
  308. $self->{staticbox}->SetLabel('Object Settings');
  309. @opt_keys = (map @{$_->get_keys}, Slic3r::Config::PrintObject->new, Slic3r::Config::PrintRegion->new);
  310. $config = $self->{model_object}->config;
  311. }
  312. # get default values
  313. my $default_config = Slic3r::Config::new_from_defaults_keys(\@opt_keys);
  314. # decide which settings will be shown by default
  315. if ($itemData->{type} eq 'object') {
  316. $config->set_ifndef('wipe_into_objects', 0);
  317. $config->set_ifndef('wipe_into_infill', 0);
  318. }
  319. # append default extruder
  320. if (! $support) {
  321. push @opt_keys, 'extruder';
  322. $default_config->set('extruder', 0);
  323. $config->set_ifndef('extruder', 0);
  324. }
  325. $self->{settings_panel}->set_type($type);
  326. $self->{settings_panel}->set_default_config($default_config);
  327. $self->{settings_panel}->set_config($config);
  328. $self->{settings_panel}->set_opt_keys(\@opt_keys);
  329. # disable minus icon to remove the settings
  330. my $fixed_options =
  331. ($itemData->{type} eq 'object') ? [qw(extruder), qw(wipe_into_infill), qw(wipe_into_objects)] :
  332. $support ? [] : [qw(extruder)];
  333. $self->{settings_panel}->set_fixed_options($fixed_options);
  334. $self->{settings_panel}->enable;
  335. }
  336. Slic3r::GUI::_3DScene::render($self->{canvas}) if $self->{canvas};
  337. }
  338. sub set_part_type
  339. {
  340. my ($self, $part_type) = @_;
  341. if (my $itemData = $self->get_selection) {
  342. if ($itemData->{type} eq 'volume') {
  343. my $volume = $self->{model_object}->volumes->[ $itemData->{volume_id} ];
  344. if ($part_type == Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_MODIFIER ||
  345. $part_type == Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_PART) {
  346. $volume->set_modifier($part_type == Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_MODIFIER);
  347. } elsif ($part_type == Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_SUPPORT_ENFORCER) {
  348. $volume->set_support_enforcer;
  349. } elsif ($part_type == Slic3r::GUI::Plater::OverrideSettingsPanel->TYPE_SUPPORT_BLOCKER) {
  350. $volume->set_support_blocker;
  351. }
  352. # We want the icon of the selected item to be changed as well.
  353. $self->reload_tree($itemData->{volume_id});
  354. }
  355. }
  356. }
  357. sub on_btn_load {
  358. my ($self, $is_modifier) = @_;
  359. my @input_files = wxTheApp->open_model($self);
  360. foreach my $input_file (@input_files) {
  361. my $model = eval { Slic3r::Model->read_from_file($input_file) };
  362. if ($@) {
  363. Slic3r::GUI::show_error($self, $@);
  364. next;
  365. }
  366. foreach my $object (@{$model->objects}) {
  367. my $delta_x = 0.0;
  368. my $delta_y = 0.0;
  369. my $delta_z = 0.0;
  370. if (($self->{model_object}->origin_translation->x != 0.0) || ($self->{model_object}->origin_translation->y != 0.0) || ($self->{model_object}->origin_translation->z != 0.0)) {
  371. $object->center_around_origin;
  372. $delta_x = $self->{model_object}->origin_translation->x - $object->origin_translation->x;
  373. $delta_y = $self->{model_object}->origin_translation->y - $object->origin_translation->y;
  374. $delta_z = $self->{model_object}->origin_translation->z - $object->origin_translation->z;
  375. }
  376. foreach my $volume (@{$object->volumes}) {
  377. my $new_volume = $self->{model_object}->add_volume($volume);
  378. $new_volume->set_modifier($is_modifier);
  379. $new_volume->set_name(basename($input_file));
  380. # apply the same translation we applied to the object
  381. if (($delta_x != 0.0) || ($delta_y != 0.0) || ($delta_z != 0.0)) {
  382. $new_volume->mesh->translate($delta_x, $delta_y, $delta_z);
  383. $new_volume->convex_hull->translate($delta_x, $delta_y, $delta_z);
  384. }
  385. # set a default extruder value, since user can't add it manually
  386. $new_volume->config->set_ifndef('extruder', 0);
  387. $self->{parts_changed} = 1;
  388. }
  389. }
  390. }
  391. $self->{model_object}->center_around_origin if $self->{parts_changed};
  392. $self->_parts_changed;
  393. }
  394. sub on_btn_lambda {
  395. my ($self, $is_modifier) = @_;
  396. my $dlg = Slic3r::GUI::Plater::LambdaObjectDialog->new($self);
  397. if ($dlg->ShowModal() == wxID_CANCEL) {
  398. return;
  399. }
  400. my $params = $dlg->ObjectParameter;
  401. my $type = "".$params->{"type"};
  402. my $name = "lambda-".$params->{"type"};
  403. my $mesh;
  404. if ($type eq "box") {
  405. $mesh = Slic3r::TriangleMesh::cube($params->{"dim"}[0], $params->{"dim"}[1], $params->{"dim"}[2]);
  406. } elsif ($type eq "cylinder") {
  407. $mesh = Slic3r::TriangleMesh::cylinder($params->{"cyl_r"}, $params->{"cyl_h"});
  408. } elsif ($type eq "sphere") {
  409. $mesh = Slic3r::TriangleMesh::sphere($params->{"sph_rho"});
  410. } elsif ($type eq "slab") {
  411. $mesh = Slic3r::TriangleMesh::cube($self->{model_object}->bounding_box->size->x*1.5, $self->{model_object}->bounding_box->size->y*1.5, $params->{"slab_h"});
  412. # box sets the base coordinate at 0,0, move to center of plate and move it up to initial_z
  413. $mesh->translate(-$self->{model_object}->bounding_box->size->x*1.5/2.0, -$self->{model_object}->bounding_box->size->y*1.5/2.0, $params->{"slab_z"});
  414. } else {
  415. return;
  416. }
  417. $mesh->repair;
  418. my $new_volume = $self->{model_object}->add_volume(mesh => $mesh);
  419. $new_volume->set_modifier($is_modifier);
  420. $new_volume->set_name($name);
  421. # set a default extruder value, since user can't add it manually
  422. $new_volume->config->set_ifndef('extruder', 0);
  423. $self->{parts_changed} = 1;
  424. $self->_parts_changed;
  425. }
  426. sub on_tree_key_down {
  427. my ($self, $event) = @_;
  428. my $keycode = $event->GetKeyCode;
  429. # Wx >= 0.9911
  430. if (defined(&Wx::TreeEvent::GetKeyEvent)) {
  431. if ($event->GetKeyEvent->GetModifiers & wxMOD_CONTROL) {
  432. if ($keycode == WXK_UP) {
  433. $event->Skip;
  434. $self->on_btn_move_up;
  435. } elsif ($keycode == WXK_DOWN) {
  436. $event->Skip;
  437. $self->on_btn_move_down;
  438. }
  439. } elsif ($keycode == WXK_DELETE) {
  440. $self->on_btn_delete;
  441. }
  442. }
  443. }
  444. sub on_btn_move_up {
  445. my ($self) = @_;
  446. my $itemData = $self->get_selection;
  447. if ($itemData && $itemData->{type} eq 'volume') {
  448. my $volume_id = $itemData->{volume_id};
  449. if ($self->{model_object}->move_volume_up($volume_id)) {
  450. Slic3r::GUI::_3DScene::move_volume_up($self->{canvas}, $volume_id);
  451. $self->{parts_changed} = 1;
  452. $self->reload_tree($volume_id - 1);
  453. }
  454. }
  455. }
  456. sub on_btn_move_down {
  457. my ($self) = @_;
  458. my $itemData = $self->get_selection;
  459. if ($itemData && $itemData->{type} eq 'volume') {
  460. my $volume_id = $itemData->{volume_id};
  461. if ($self->{model_object}->move_volume_down($volume_id)) {
  462. Slic3r::GUI::_3DScene::move_volume_down($self->{canvas}, $volume_id);
  463. $self->{parts_changed} = 1;
  464. $self->reload_tree($volume_id + 1);
  465. }
  466. }
  467. }
  468. sub on_btn_delete {
  469. my ($self) = @_;
  470. my $itemData = $self->get_selection;
  471. if ($itemData && $itemData->{type} eq 'volume') {
  472. my $volume = $self->{model_object}->volumes->[$itemData->{volume_id}];
  473. # if user is deleting the last solid part, throw error
  474. if (!$volume->modifier && scalar(grep !$_->modifier, @{$self->{model_object}->volumes}) == 1) {
  475. Slic3r::GUI::show_error($self, "You can't delete the last solid part from this object.");
  476. return;
  477. }
  478. $self->{model_object}->delete_volume($itemData->{volume_id});
  479. $self->{parts_changed} = 1;
  480. }
  481. $self->{model_object}->center_around_origin if $self->{parts_changed};
  482. $self->_parts_changed;
  483. }
  484. sub on_btn_split {
  485. my ($self) = @_;
  486. my $itemData = $self->get_selection;
  487. if ($itemData && $itemData->{type} eq 'volume') {
  488. my $volume = $self->{model_object}->volumes->[$itemData->{volume_id}];
  489. my $nozzle_dmrs = $self->GetParent->GetParent->GetParent->{config}->get('nozzle_diameter');
  490. $self->{parts_changed} = 1 if $volume->split(scalar(@$nozzle_dmrs)) > 1;
  491. }
  492. $self->_parts_changed;
  493. }
  494. sub _parts_changed {
  495. my ($self) = @_;
  496. $self->reload_tree;
  497. if ($self->{canvas}) {
  498. Slic3r::GUI::_3DScene::reset_volumes($self->{canvas});
  499. Slic3r::GUI::_3DScene::load_model_object($self->{canvas}, $self->{model_object}, 0, [0]);
  500. Slic3r::GUI::_3DScene::zoom_to_volumes($self->{canvas});
  501. Slic3r::GUI::_3DScene::update_volumes_colors_by_extruder($self->{canvas});
  502. Slic3r::GUI::_3DScene::render($self->{canvas});
  503. }
  504. }
  505. sub CanClose {
  506. my $self = shift;
  507. return 1; # skip validation for now
  508. # validate options before allowing user to dismiss the dialog
  509. # the validate method only works on full configs so we have
  510. # to merge our settings with the default ones
  511. my $config = $self->GetParent->GetParent->GetParent->GetParent->GetParent->config->clone;
  512. eval {
  513. $config->apply($self->model_object->config);
  514. $config->validate;
  515. };
  516. return ! Slic3r::GUI::catch_error($self);
  517. }
  518. sub Destroy {
  519. my ($self) = @_;
  520. $self->{canvas}->Destroy if ($self->{canvas});
  521. }
  522. sub PartsChanged {
  523. my ($self) = @_;
  524. return $self->{parts_changed};
  525. }
  526. sub PartSettingsChanged {
  527. my ($self) = @_;
  528. return $self->{part_settings_changed};
  529. }
  530. sub _update_canvas {
  531. my ($self) = @_;
  532. if ($self->{canvas}) {
  533. Slic3r::GUI::_3DScene::reset_volumes($self->{canvas});
  534. Slic3r::GUI::_3DScene::load_model_object($self->{canvas}, $self->{model_object}, 0, [0]);
  535. # restore selection, if any
  536. if (my $itemData = $self->get_selection) {
  537. if ($itemData->{type} eq 'volume') {
  538. Slic3r::GUI::_3DScene::select_volume($self->{canvas}, $itemData->{volume_id});
  539. }
  540. }
  541. Slic3r::GUI::_3DScene::update_volumes_colors_by_extruder($self->{canvas});
  542. Slic3r::GUI::_3DScene::render($self->{canvas});
  543. }
  544. }
  545. sub _update {
  546. my ($self) = @_;
  547. my ($m_x, $m_y, $m_z) = ($self->{move_options}{x}, $self->{move_options}{y}, $self->{move_options}{z});
  548. my ($l_x, $l_y, $l_z) = ($self->{last_coords}{x}, $self->{last_coords}{y}, $self->{last_coords}{z});
  549. my $itemData = $self->get_selection;
  550. if ($itemData && $itemData->{type} eq 'volume') {
  551. my $d = Slic3r::Pointf3->new($m_x - $l_x, $m_y - $l_y, $m_z - $l_z);
  552. my $volume = $self->{model_object}->volumes->[$itemData->{volume_id}];
  553. $volume->mesh->translate(@{$d});
  554. $self->{last_coords}{x} = $m_x;
  555. $self->{last_coords}{y} = $m_y;
  556. $self->{last_coords}{z} = $m_z;
  557. }
  558. $self->{parts_changed} = 1;
  559. my @objects = ();
  560. push @objects, $self->{model_object};
  561. Slic3r::GUI::_3DScene::reset_volumes($self->{canvas});
  562. Slic3r::GUI::_3DScene::load_model_object($self->{canvas}, $_, 0, [0]) for @objects;
  563. Slic3r::GUI::_3DScene::update_volumes_colors_by_extruder($self->{canvas});
  564. Slic3r::GUI::_3DScene::render($self->{canvas});
  565. }
  566. 1;