1
use Test::More tests => 14;
7
use lib "$FindBin::Bin/../lib";
10
use List::Util qw(first sum);
12
use Slic3r::Geometry qw(scale epsilon deg2rad rad2deg PI);
17
my ($bridge_size, $rotate, $expected_angle, $tolerance) = @_;
19
my ($x, $y) = @$bridge_size;
20
my $lower = Slic3r::ExPolygon->new(
21
Slic3r::Polygon->new_scale([-2,-2], [$x+2,-2], [$x+2,$y+2], [-2,$y+2]),
22
Slic3r::Polygon->new_scale([0,0], [0,$y], [$x,$y], [$x,0]),
24
$lower->translate(scale 20, scale 20); # avoid negative coordinates for easier SVG preview
25
$lower->rotate(deg2rad($rotate), [$x/2,$y/2]);
26
my $bridge = $lower->[1]->clone;
28
$bridge = Slic3r::ExPolygon->new($bridge);
30
ok check_angle([$lower], $bridge, $expected_angle, $tolerance), 'correct bridge angle for O-shaped overhang';
33
$test->([20,10], 0, 90);
34
$test->([10,20], 0, 0);
35
$test->([20,10], 45, 135, 20);
36
$test->([20,10], 135, 45, 20);
40
my $bridge = Slic3r::ExPolygon->new(
41
Slic3r::Polygon->new_scale([0,0], [20,0], [20,10], [0,10]),
44
Slic3r::ExPolygon->new(
45
Slic3r::Polygon->new_scale([-2,0], [0,0], [0,10], [-2,10]),
48
$_->translate(scale 20, scale 20) for $bridge, @$lower; # avoid negative coordinates for easier SVG preview
50
$lower->[1] = $lower->[0]->clone;
51
$lower->[1]->translate(scale 22, 0);
53
ok check_angle($lower, $bridge, 0), 'correct bridge angle for two-sided bridge';
57
my $bridge = Slic3r::ExPolygon->new(
58
Slic3r::Polygon->new_scale([0,0], [20,0], [10,10], [0,10]),
61
Slic3r::ExPolygon->new(
62
Slic3r::Polygon->new_scale([0,0], [0,10], [10,10], [10,12], [-2,12], [-2,-2], [22,-2], [22,0]),
65
$_->translate(scale 20, scale 20) for $bridge, @$lower; # avoid negative coordinates for easier SVG preview
67
ok check_angle($lower, $bridge, 135), 'correct bridge angle for C-shaped overhang';
71
my $bridge = Slic3r::ExPolygon->new(
72
Slic3r::Polygon->new_scale([10,10],[20,10],[20,20], [10,20]),
75
Slic3r::ExPolygon->new(
76
Slic3r::Polygon->new_scale([10,10],[10,20],[20,20],[20,30],[0,30],[0,10]),
79
$_->translate(scale 20, scale 20) for $bridge, @$lower; # avoid negative coordinates for easier SVG preview
81
ok check_angle($lower, $bridge, 45, undef, $bridge->area/2), 'correct bridge angle for square overhang with L-shaped anchors';
85
my ($lower, $bridge, $expected, $tolerance, $expected_coverage) = @_;
87
$expected_coverage //= -1;
88
$expected_coverage = $bridge->area if $expected_coverage == -1;
90
my $bd = Slic3r::Layer::BridgeDetector->new(
92
lower_slices => $lower,
93
extrusion_width => scale 0.5,
96
$tolerance //= rad2deg($bd->resolution) + epsilon;
97
my $result = $bd->detect_angle;
98
my $coverage = $bd->coverage;
99
is sum(map $_->area, @$coverage), $expected_coverage, 'correct coverage area';
101
# our epsilon is equal to the steps used by the bridge detection algorithm
102
###use XXX; YYY [ rad2deg($result), $expected ];
103
# returned value must be non-negative, check for that too
104
my $delta=rad2deg($result) - $expected;
105
$delta-=180 if $delta>=180 - epsilon;
106
return defined $result && $result>=0 && abs($delta) < $tolerance;