ObjectPartsPanel.pm 21 KB

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