mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-30 04:02:52 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			105 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			105 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| package Slic3r::ExtrusionLoop;
 | |
| use Moo;
 | |
| 
 | |
| use Slic3r::Geometry qw(same_point);
 | |
| 
 | |
| # the underlying Slic3r::Polygon objects holds the geometry
 | |
| has 'polygon' => (
 | |
|     is          => 'rw',
 | |
|     required    => 1,
 | |
|     handles     => [qw(is_printable nearest_point_index_to reverse)],
 | |
| );
 | |
| 
 | |
| has 'flow_spacing' => (is => 'rw');
 | |
| 
 | |
| # see EXTR_ROLE_* constants in ExtrusionPath.pm
 | |
| has 'role'         => (is => 'rw', required => 1);
 | |
| 
 | |
| use constant PACK_FMT => 'fca*';
 | |
| 
 | |
| # class or object method
 | |
| sub pack {
 | |
|     my $self = shift;
 | |
|     my %args = @_;
 | |
|     
 | |
|     if (ref $self) {
 | |
|         %args = map { $_ => $self->$_ } qw(flow_spacing role polygon);
 | |
|     }
 | |
|     
 | |
|     my $o = \ pack PACK_FMT,
 | |
|         $args{flow_spacing} || -1,
 | |
|         $args{role}         // (die "Missing mandatory attribute 'role'"), #/
 | |
|         $args{polygon}->serialize;
 | |
|     
 | |
|     bless $o, 'Slic3r::ExtrusionLoop::Packed';
 | |
|     return $o;
 | |
| }
 | |
| 
 | |
| sub split_at_index {
 | |
|     my $self = shift;
 | |
|     my ($index) = @_;
 | |
| 
 | |
|     my @new_points = ();
 | |
|     push @new_points, @{$self->polygon}[$index .. $#{$self->polygon}];
 | |
|     push @new_points, @{$self->polygon}[0 .. $index];
 | |
|     
 | |
|     return Slic3r::ExtrusionPath->new(
 | |
|         polyline    => Slic3r::Polyline->new(\@new_points),
 | |
|         role        => $self->role,
 | |
|         flow_spacing => $self->flow_spacing,
 | |
|     );
 | |
| }
 | |
| 
 | |
| sub split_at {
 | |
|     my $self = shift;
 | |
|     my ($point) = @_;
 | |
|     
 | |
|     $point = Slic3r::Point->new($point);
 | |
|     
 | |
|     # find index of point
 | |
|     my $i = -1;
 | |
|     for (my $n = 0; $n <= $#{$self->polygon}; $n++) {
 | |
|         if (same_point($point, $self->polygon->[$n])) {
 | |
|             $i = $n;
 | |
|             last;
 | |
|         }
 | |
|     }
 | |
|     die "Point not found" if $i == -1;
 | |
|     
 | |
|     return $self->split_at_index($i);
 | |
| }
 | |
| 
 | |
| sub split_at_first_point {
 | |
|     my $self = shift;
 | |
|     return $self->split_at_index(0);
 | |
| }
 | |
| 
 | |
| # although a loop doesn't have endpoints, this method is provided to allow
 | |
| # ExtrusionLoop objects to be added to an ExtrusionPath::Collection and
 | |
| # sorted by the ->shortest_path() method
 | |
| sub endpoints {
 | |
|     my $self = shift;
 | |
|     return ($self->polygon->[0], $self->polygon->[-1]);
 | |
| }
 | |
| 
 | |
| # provided for ExtrusionPath::Collection->shortest_path()
 | |
| sub points {
 | |
|     my $self = shift;
 | |
|     return $self->polygon;
 | |
| }
 | |
| 
 | |
| package Slic3r::ExtrusionLoop::Packed;
 | |
| sub unpack {
 | |
|     my $self = shift;
 | |
|     
 | |
|     my ($flow_spacing, $role, $polygon_s)
 | |
|         = unpack Slic3r::ExtrusionLoop::PACK_FMT, $$self;
 | |
|     
 | |
|     return Slic3r::ExtrusionLoop->new(
 | |
|         flow_spacing    => ($flow_spacing == -1) ? undef : $flow_spacing,
 | |
|         role            => $role,
 | |
|         polygon         => Slic3r::Polygon->deserialize($polygon_s),
 | |
|     );
 | |
| }
 | |
| 
 | |
| 1;
 | 
