Extruder.pm 3.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. package Slic3r::Extruder;
  2. use Moo;
  3. use Slic3r::Geometry qw(PI scale);
  4. use constant OPTIONS => [qw(
  5. extruder_offset
  6. nozzle_diameter filament_diameter extrusion_multiplier temperature first_layer_temperature
  7. retract_length retract_lift retract_speed retract_restart_extra retract_before_travel
  8. retract_layer_change retract_length_toolchange retract_restart_extra_toolchange wipe
  9. )];
  10. has 'id' => (is => 'rw', required => 1);
  11. has $_ => (is => 'ro', required => 1) for @{&OPTIONS};
  12. has 'config'=> (is => 'ro', required => 1);
  13. has 'bridge_flow' => (is => 'lazy');
  14. has 'E' => (is => 'rw', default => sub {0} );
  15. has 'absolute_E' => (is => 'rw', default => sub {0} );
  16. has 'retracted' => (is => 'rw', default => sub {0} );
  17. has 'restart_extra' => (is => 'rw', default => sub {0} );
  18. has 'e_per_mm3' => (is => 'lazy');
  19. has 'retract_speed_mm_min' => (is => 'lazy');
  20. has 'scaled_wipe_distance' => (is => 'lazy'); # scaled mm
  21. has '_mm3_per_mm_cache' => (is => 'ro', default => sub {{}});
  22. sub _build_bridge_flow {
  23. my $self = shift;
  24. return Slic3r::Flow::Bridge->new(nozzle_diameter => $self->nozzle_diameter);
  25. }
  26. sub _build_e_per_mm3 {
  27. my $self = shift;
  28. return $self->extrusion_multiplier * (4 / (($self->filament_diameter ** 2) * PI));
  29. }
  30. sub _build_retract_speed_mm_min {
  31. my $self = shift;
  32. return $self->retract_speed * 60;
  33. }
  34. sub _build_scaled_wipe_distance {
  35. my $self = shift;
  36. # reduce feedrate a bit; travel speed is often too high to move on existing material
  37. # too fast = ripping of existing material; too slow = short wipe path, thus more blob
  38. return scale($self->retract_length / $self->retract_speed * $Slic3r::Config->travel_speed * 0.8);
  39. }
  40. sub extrude {
  41. my ($self, $E) = @_;
  42. $self->E(0) if $self->config->use_relative_e_distances;
  43. $self->absolute_E($self->absolute_E + $E);
  44. return $self->E($self->E + $E);
  45. }
  46. sub extruded_volume {
  47. my ($self) = @_;
  48. return $self->absolute_E * ($self->filament_diameter**2) * PI/4;
  49. }
  50. sub make_flow {
  51. my $self = shift;
  52. return Slic3r::Flow->new(nozzle_diameter => $self->nozzle_diameter, @_);
  53. }
  54. sub mm3_per_mm {
  55. my $self = shift;
  56. my ($s, $h) = @_;
  57. my $cache_key = "${s}_${h}";
  58. if (!exists $self->_mm3_per_mm_cache->{$cache_key}) {
  59. my $w_threshold = $h + $self->nozzle_diameter;
  60. my $s_threshold = $w_threshold - &Slic3r::OVERLAP_FACTOR * ($w_threshold - ($w_threshold - $h * (1 - PI/4)));
  61. if ($s >= $s_threshold) {
  62. # rectangle with semicircles at the ends
  63. my $w = $s + &Slic3r::OVERLAP_FACTOR * $h * (1 - PI/4);
  64. $self->_mm3_per_mm_cache->{$cache_key} = $w * $h + ($h**2) / 4 * (PI - 4);
  65. } else {
  66. # rectangle with shrunk semicircles at the ends
  67. my $w = ($s + $self->nozzle_diameter * &Slic3r::OVERLAP_FACTOR * (PI/4 - 1)) / (1 + &Slic3r::OVERLAP_FACTOR * (PI/4 - 1));
  68. $self->_mm3_per_mm_cache->{$cache_key} = $self->nozzle_diameter * $h * (1 - PI/4) + $h * $w * PI/4;
  69. }
  70. }
  71. return $self->_mm3_per_mm_cache->{$cache_key};
  72. }
  73. sub e_per_mm {
  74. my $self = shift;
  75. my ($s, $h) = @_;
  76. return $self->mm3_per_mm($s, $h) * $self->e_per_mm3;
  77. }
  78. 1;