mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-11-02 12:41:18 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			149 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
package Slic3r::Fill::Rectilinear;
 | 
						|
use Moo;
 | 
						|
 | 
						|
extends 'Slic3r::Fill::Base';
 | 
						|
with qw(Slic3r::Fill::WithDirection);
 | 
						|
 | 
						|
has '_min_spacing'          => (is => 'rw');
 | 
						|
has '_line_spacing'         => (is => 'rw');
 | 
						|
has '_diagonal_distance'    => (is => 'rw');
 | 
						|
has '_line_oscillation'     => (is => 'rw');
 | 
						|
 | 
						|
use Slic3r::Geometry qw(scale unscale scaled_epsilon);
 | 
						|
use Slic3r::Geometry::Clipper qw(intersection_pl);
 | 
						|
 | 
						|
sub fill_surface {
 | 
						|
    my $self = shift;
 | 
						|
    my ($surface, %params) = @_;
 | 
						|
    
 | 
						|
    # rotate polygons so that we can work with vertical lines here
 | 
						|
    my $expolygon = $surface->expolygon->clone;
 | 
						|
    my $rotate_vector = $self->infill_direction($surface);
 | 
						|
    $self->rotate_points($expolygon, $rotate_vector);
 | 
						|
    
 | 
						|
    $self->_min_spacing(scale $self->spacing);
 | 
						|
    $self->_line_spacing($self->_min_spacing / $params{density});
 | 
						|
    $self->_diagonal_distance($self->_line_spacing * 2);
 | 
						|
    $self->_line_oscillation($self->_line_spacing - $self->_min_spacing);  # only for Line infill
 | 
						|
    my $bounding_box = $expolygon->bounding_box;
 | 
						|
    
 | 
						|
    # define flow spacing according to requested density
 | 
						|
    if ($params{density} == 1 && !$params{dont_adjust}) {
 | 
						|
        $self->_line_spacing($self->adjust_solid_spacing(
 | 
						|
            width       => $bounding_box->size->x,
 | 
						|
            distance    => $self->_line_spacing,
 | 
						|
        ));
 | 
						|
        $self->spacing(unscale $self->_line_spacing);
 | 
						|
    } else {
 | 
						|
        # extend bounding box so that our pattern will be aligned with other layers
 | 
						|
        $bounding_box->merge_point(Slic3r::Point->new(
 | 
						|
            $bounding_box->x_min - ($bounding_box->x_min % $self->_line_spacing),
 | 
						|
            $bounding_box->y_min - ($bounding_box->y_min % $self->_line_spacing),
 | 
						|
        ));
 | 
						|
    }
 | 
						|
    
 | 
						|
    # generate the basic pattern
 | 
						|
    my $x_max = $bounding_box->x_max + scaled_epsilon;
 | 
						|
    my @vertical_lines  = ();
 | 
						|
    for (my $x = $bounding_box->x_min; $x <= $x_max; $x += $self->_line_spacing) {
 | 
						|
        push @vertical_lines, $self->_line($#vertical_lines, $x, $bounding_box->y_min, $bounding_box->y_max);
 | 
						|
    }
 | 
						|
    
 | 
						|
    # clip paths against a slightly larger expolygon, so that the first and last paths
 | 
						|
    # are kept even if the expolygon has vertical sides
 | 
						|
    # the minimum offset for preventing edge lines from being clipped is scaled_epsilon;
 | 
						|
    # however we use a larger offset to support expolygons with slightly skewed sides and 
 | 
						|
    # not perfectly straight
 | 
						|
    my @polylines = @{intersection_pl(\@vertical_lines, $expolygon->offset(+scale 0.02))};
 | 
						|
    
 | 
						|
    my $extra = $self->_min_spacing * &Slic3r::INFILL_OVERLAP_OVER_SPACING;
 | 
						|
    foreach my $polyline (@polylines) {
 | 
						|
        my ($first_point, $last_point) = @$polyline[0,-1];
 | 
						|
        if ($first_point->y > $last_point->y) { #>
 | 
						|
            ($first_point, $last_point) = ($last_point, $first_point);
 | 
						|
        }
 | 
						|
        $first_point->set_y($first_point->y - $extra);  #--
 | 
						|
        $last_point->set_y($last_point->y + $extra);    #++
 | 
						|
    }
 | 
						|
    
 | 
						|
    # connect lines
 | 
						|
    unless ($params{dont_connect} || !@polylines) {  # prevent calling leftmost_point() on empty collections
 | 
						|
        # offset the expolygon by max(min_spacing/2, extra)
 | 
						|
        my ($expolygon_off) = @{$expolygon->offset_ex($self->_min_spacing/2)};
 | 
						|
        my $collection = Slic3r::Polyline::Collection->new(@polylines);
 | 
						|
        @polylines = ();
 | 
						|
        
 | 
						|
        foreach my $polyline (@{$collection->chained_path_from($collection->leftmost_point, 0)}) {
 | 
						|
            if (@polylines) {
 | 
						|
                my $first_point = $polyline->first_point;
 | 
						|
                my $last_point = $polylines[-1]->last_point;
 | 
						|
                my @distance = map abs($first_point->$_ - $last_point->$_), qw(x y);
 | 
						|
                
 | 
						|
                # TODO: we should also check that both points are on a fill_boundary to avoid 
 | 
						|
                # connecting paths on the boundaries of internal regions
 | 
						|
                if ($self->_can_connect(@distance) && $expolygon_off->contains_line(Slic3r::Line->new($last_point, $first_point))) {
 | 
						|
                    $polylines[-1]->append_polyline($polyline);
 | 
						|
                    next;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            
 | 
						|
            # make a clone before $collection goes out of scope
 | 
						|
            push @polylines, $polyline->clone;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    # paths must be rotated back
 | 
						|
    $self->rotate_points_back(\@polylines, $rotate_vector);
 | 
						|
    
 | 
						|
    return @polylines;
 | 
						|
}
 | 
						|
 | 
						|
sub _line {
 | 
						|
    my ($self, $i, $x, $y_min, $y_max) = @_;
 | 
						|
    
 | 
						|
    return Slic3r::Polyline->new(
 | 
						|
        [$x, $y_min],
 | 
						|
        [$x, $y_max],
 | 
						|
    );
 | 
						|
}
 | 
						|
 | 
						|
sub _can_connect {
 | 
						|
    my ($self, $dist_X, $dist_Y) = @_;
 | 
						|
    
 | 
						|
    return $dist_X <= $self->_diagonal_distance
 | 
						|
        && $dist_Y <= $self->_diagonal_distance;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
package Slic3r::Fill::Line;
 | 
						|
use Moo;
 | 
						|
extends 'Slic3r::Fill::Rectilinear';
 | 
						|
 | 
						|
use Slic3r::Geometry qw(scaled_epsilon);
 | 
						|
 | 
						|
sub _line {
 | 
						|
    my ($self, $i, $x, $y_min, $y_max) = @_;
 | 
						|
    
 | 
						|
    if ($i % 2) {
 | 
						|
        return Slic3r::Polyline->new(
 | 
						|
            [$x - $self->_line_oscillation, $y_min],
 | 
						|
            [$x + $self->_line_oscillation, $y_max],
 | 
						|
        );
 | 
						|
    } else {
 | 
						|
        return Slic3r::Polyline->new(
 | 
						|
            [$x, $y_min],
 | 
						|
            [$x, $y_max],
 | 
						|
        );
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
sub _can_connect {
 | 
						|
    my ($self, $dist_X, $dist_Y) = @_;
 | 
						|
    
 | 
						|
    my $TOLERANCE = 10 * scaled_epsilon;
 | 
						|
    return ($dist_X >= ($self->_line_spacing - $self->_line_oscillation) - $TOLERANCE)
 | 
						|
        && ($dist_X <= ($self->_line_spacing + $self->_line_oscillation) + $TOLERANCE)
 | 
						|
        && $dist_Y <= $self->_diagonal_distance;
 | 
						|
}
 | 
						|
 | 
						|
1;
 |