123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275 |
- package Slic3r::GUI::Controller::ManualControlDialog;
- use strict;
- use warnings;
- use utf8;
- use Scalar::Util qw(looks_like_number);
- use Slic3r::Geometry qw(PI X Y unscale);
- use Wx qw(:dialog :id :misc :sizer :choicebook :button :bitmap :textctrl
- wxBORDER_NONE wxTAB_TRAVERSAL);
- use Wx::Event qw(EVT_CLOSE EVT_BUTTON EVT_TEXT_ENTER);
- use base qw(Wx::Dialog Class::Accessor);
- __PACKAGE__->mk_accessors(qw(sender writer config2 x_homed y_homed));
- sub new {
- my ($class, $parent, $config, $sender, $manual_control_config) = @_;
-
- my $self = $class->SUPER::new($parent, -1, "Manual Control", wxDefaultPosition,
- [500,400], wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER);
- $self->sender($sender);
- $self->writer(Slic3r::GCode::Writer->new);
- $self->writer->config->apply_dynamic($config);
-
- $self->config2($manual_control_config);
-
- my $bed_sizer = Wx::FlexGridSizer->new(2, 3, 1, 1);
- $bed_sizer->AddGrowableCol(1, 1);
- $bed_sizer->AddGrowableRow(0, 1);
-
- my $move_button = sub {
- my ($sizer, $label, $icon, $bold, $pos, $handler) = @_;
-
- my $btn = Wx::Button->new($self, -1, $label, wxDefaultPosition, wxDefaultSize,
- wxBU_LEFT | wxBU_EXACTFIT);
- $btn->SetFont($bold ? $Slic3r::GUI::small_bold_font : $Slic3r::GUI::small_font);
- if ($Slic3r::GUI::have_button_icons) {
- $btn->SetBitmap(Wx::Bitmap->new($Slic3r::var->("$icon.png"), wxBITMAP_TYPE_PNG));
- $btn->SetBitmapPosition($pos);
- }
- EVT_BUTTON($self, $btn, $handler);
- $sizer->Add($btn, 1, wxEXPAND | wxALL, 0);
- };
-
- # Y buttons
- {
- my $sizer = Wx::BoxSizer->new(wxVERTICAL);
- for my $d (qw(+10 +1 +0.1)) {
- $move_button->($sizer, $d, 'arrow_up', 0, wxLEFT, sub { $self->rel_move('Y', $d) });
- }
- $move_button->($sizer, 'Y', 'house', 1, wxLEFT, sub { $self->home('Y') });
- for my $d (qw(-0.1 -1 -10)) {
- $move_button->($sizer, $d, 'arrow_down', 0, wxLEFT, sub { $self->rel_move('Y', $d) });
- };
- $bed_sizer->Add($sizer, 1, wxEXPAND, 0);
- }
-
- # Bed canvas
- {
- my $bed_shape = $config->bed_shape;
- $self->{canvas} = my $canvas = Slic3r::GUI::2DBed->new($self, $bed_shape);
- $canvas->interactive(1);
- $canvas->on_move(sub {
- my ($pos) = @_;
-
- if (!($self->x_homed && $self->y_homed)) {
- Slic3r::GUI::show_error($self, "Please home both X and Y before moving.");
- return ;
- }
-
- # delete any pending commands to get a smoother movement
- $self->sender->purge_queue(1);
- $self->abs_xy_move($pos);
- });
- $bed_sizer->Add($canvas, 0, wxEXPAND | wxRIGHT, 3);
- }
-
- # Z buttons
- {
- my $sizer = Wx::BoxSizer->new(wxVERTICAL);
- for my $d (qw(+10 +1 +0.1)) {
- $move_button->($sizer, $d, 'arrow_up', 0, wxLEFT, sub { $self->rel_move('Z', $d) });
- }
- $move_button->($sizer, 'Z', 'house', 1, wxLEFT, sub { $self->home('Z') });
- for my $d (qw(-0.1 -1 -10)) {
- $move_button->($sizer, $d, 'arrow_down', 0, wxLEFT, sub { $self->rel_move('Z', $d) });
- };
- $bed_sizer->Add($sizer, 1, wxEXPAND, 0);
- }
-
- # XYZ home button
- $move_button->($bed_sizer, 'XYZ', 'house', 1, wxTOP, sub { $self->home(undef) });
-
- # X buttons
- {
- my $sizer = Wx::BoxSizer->new(wxHORIZONTAL);
- for my $d (qw(-10 -1 -0.1)) {
- $move_button->($sizer, $d, 'arrow_left', 0, wxTOP, sub { $self->rel_move('X', $d) });
- }
- $move_button->($sizer, 'X', 'house', 1, wxTOP, sub { $self->home('X') });
- for my $d (qw(+0.1 +1 +10)) {
- $move_button->($sizer, $d, 'arrow_right', 0, wxTOP, sub { $self->rel_move('X', $d) });
- }
- $bed_sizer->Add($sizer, 1, wxEXPAND, 0);
- }
-
- my $optgroup = Slic3r::GUI::OptionsGroup->new(
- parent => $self,
- title => 'Manual motion settings',
- on_change => sub {
- my ($opt_id, $value) = @_;
- $self->config2->{$opt_id} = $value;
- },
- );
- {
- my $line = Slic3r::GUI::OptionsGroup::Line->new(
- label => 'Speed (mm/s)',
- );
- $line->append_option(Slic3r::GUI::OptionsGroup::Option->new(
- opt_id => 'xy_travel_speed',
- type => 'f',
- label => 'X/Y',
- tooltip => '',
- default => $self->config2->{xy_travel_speed},
- ));
- $line->append_option(Slic3r::GUI::OptionsGroup::Option->new(
- opt_id => 'z_travel_speed',
- type => 'f',
- label => 'Z',
- tooltip => '',
- default => $self->config2->{z_travel_speed},
- ));
- $optgroup->append_line($line);
- }
- my $left_sizer = Wx::BoxSizer->new(wxVERTICAL);
- $left_sizer->Add($bed_sizer, 1, wxEXPAND | wxALL, 10);
- $left_sizer->Add($optgroup->sizer, 0, wxEXPAND | wxALL, 10);
-
- my $right_sizer = Wx::BoxSizer->new(wxVERTICAL);
- {
- my $optgroup = Slic3r::GUI::OptionsGroup->new(
- parent => $self,
- title => 'Temperature',
- on_change => sub {
- my ($opt_id, $value) = @_;
- $self->config2->{$opt_id} = $value;
- },
- );
- $right_sizer->Add($optgroup->sizer, 0, wxEXPAND | wxALL, 10);
- {
- my $line = $optgroup->create_single_option_line(Slic3r::GUI::OptionsGroup::Option->new(
- opt_id => 'temperature',
- type => 's',
- label => 'Extruder',
- default => '',
- sidetext => '°C',
- default => $self->config2->{temperature},
- ));
- $line->append_button("Set", "tick.png", sub {
- if (!looks_like_number($self->config2->{temperature})) {
- Slic3r::GUI::show_error($self, "Invalid temperature.");
- return;
- }
- my $cmd = $self->writer->set_temperature($self->config2->{temperature});
- $self->GetParent->append_to_log(">> $cmd\n");
- $self->sender->send($cmd, 1);
- });
- $optgroup->append_line($line);
- }
- {
- my $line = $optgroup->create_single_option_line(Slic3r::GUI::OptionsGroup::Option->new(
- opt_id => 'bed_temperature',
- type => 's',
- label => 'Bed',
- default => '',
- sidetext => '°C',
- default => $self->config2->{bed_temperature},
- ));
- $line->append_button("Set", "tick.png", sub {
- if (!looks_like_number($self->config2->{bed_temperature})) {
- Slic3r::GUI::show_error($self, "Invalid bed temperature.");
- return;
- }
- my $cmd = $self->writer->set_bed_temperature($self->config2->{bed_temperature});
- $self->GetParent->append_to_log(">> $cmd\n");
- $self->sender->send($cmd, 1);
- });
- $optgroup->append_line($line);
- }
- }
-
- {
- my $box = Wx::StaticBox->new($self, -1, "Send manual command");
- my $sbsizer = Wx::StaticBoxSizer->new($box, wxVERTICAL);
- $right_sizer->Add($sbsizer, 1, wxEXPAND | wxALL, 10);
-
- my $cmd_sizer = Wx::BoxSizer->new(wxHORIZONTAL);
- my $cmd_textctrl = Wx::TextCtrl->new($self, -1, '', wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER);
- $cmd_sizer->Add($cmd_textctrl, 1, wxEXPAND, 0);
-
- my $btn = Wx::Button->new($self, -1,
- "Send", wxDefaultPosition, wxDefaultSize, wxBU_LEFT | wxBU_EXACTFIT);
- $btn->SetFont($Slic3r::GUI::small_font);
- if ($Slic3r::GUI::have_button_icons) {
- $btn->SetBitmap(Wx::Bitmap->new($Slic3r::var->("cog_go.png"), wxBITMAP_TYPE_PNG));
- }
- $cmd_sizer->Add($btn, 0, wxEXPAND | wxLEFT, 5);
-
- my $do_send = sub {
- my $cmd = $cmd_textctrl->GetValue;
- return if $cmd eq '';
- $self->GetParent->append_to_log(">> $cmd\n");
- $self->sender->send($cmd, 1);
- $cmd_textctrl->SetValue('');
- };
- EVT_BUTTON($self, $btn, $do_send);
- EVT_TEXT_ENTER($self, $cmd_textctrl, $do_send);
-
- $sbsizer->Add($cmd_sizer, 0, wxEXPAND | wxTOP, 5);
- }
-
- my $main_sizer = Wx::BoxSizer->new(wxHORIZONTAL);
- $main_sizer->Add($left_sizer, 1, wxEXPAND | wxRIGHT, 10);
- $main_sizer->Add($right_sizer, 0, wxEXPAND, 0);
-
- $self->SetSizer($main_sizer);
- $self->SetMinSize($self->GetSize);
- $main_sizer->SetSizeHints($self);
- $self->Layout;
-
- # needed to actually free memory
- EVT_CLOSE($self, sub {
- $self->EndModal(wxID_OK);
- $self->Destroy;
- });
-
- return $self;
- }
- sub abs_xy_move {
- my ($self, $pos) = @_;
-
- $self->sender->send("G90", 1); # set absolute positioning
- $self->sender->send(sprintf("G1 X%.1f Y%.1f F%d", @$pos, $self->config2->{xy_travel_speed}*60), 1);
- $self->{canvas}->set_pos($pos);
- }
- sub rel_move {
- my ($self, $axis, $distance) = @_;
-
- my $speed = ($axis eq 'Z') ? $self->config2->{z_travel_speed} : $self->config2->{xy_travel_speed};
- $self->sender->send("G91", 1); # set relative positioning
- $self->sender->send(sprintf("G1 %s%.1f F%d", $axis, $distance, $speed*60), 1);
- $self->sender->send("G90", 1); # set absolute positioning
-
- if (my $pos = $self->{canvas}->pos) {
- if ($axis eq 'X') {
- $pos->translate($distance, 0);
- } elsif ($axis eq 'Y') {
- $pos->translate(0, $distance);
- }
- $self->{canvas}->set_pos($pos);
- }
- }
- sub home {
- my ($self, $axis) = @_;
-
- $axis //= '';
- $self->sender->send(sprintf("G28 %s", $axis), 1);
- $self->{canvas}->set_pos(undef);
- $self->x_homed(1) if $axis eq 'X';
- $self->y_homed(1) if $axis eq 'Y';
- }
- 1;
|