Extend medial axis endpoints to fill the entire length. Includes fix for a minor memory leak caused by usage of old signature for Line::point_at()
This commit is contained in:
parent
52de292a48
commit
ed8a2f7330
@ -148,6 +148,12 @@ ExPolygon::medial_axis(double max_width, double min_width, Polylines* polylines)
|
|||||||
// compute the Voronoi diagram
|
// compute the Voronoi diagram
|
||||||
ma.build(polylines);
|
ma.build(polylines);
|
||||||
|
|
||||||
|
// extend initial and final segments of each polyline (they will be clipped)
|
||||||
|
for (Polylines::iterator polyline = polylines->begin(); polyline != polylines->end(); ++polyline) {
|
||||||
|
polyline->extend_start(max_width);
|
||||||
|
polyline->extend_end(max_width);
|
||||||
|
}
|
||||||
|
|
||||||
// clip segments to our expolygon area
|
// clip segments to our expolygon area
|
||||||
intersection(*polylines, *this, *polylines);
|
intersection(*polylines, *this, *polylines);
|
||||||
}
|
}
|
||||||
|
@ -62,15 +62,22 @@ Line::midpoint() const
|
|||||||
return new Point ((this->a.x + this->b.x) / 2.0, (this->a.y + this->b.y) / 2.0);
|
return new Point ((this->a.x + this->b.x) / 2.0, (this->a.y + this->b.y) / 2.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
Point*
|
void
|
||||||
Line::point_at(double distance) const
|
Line::point_at(double distance, Point* point) const
|
||||||
{
|
{
|
||||||
double len = this->length();
|
double len = this->length();
|
||||||
Point* p = new Point(this->a);
|
*point = this->a;
|
||||||
if (this->a.x != this->b.x)
|
if (this->a.x != this->b.x)
|
||||||
p->x = this->a.x + (this->b.x - this->a.x) * distance / len;
|
point->x = this->a.x + (this->b.x - this->a.x) * distance / len;
|
||||||
if (this->a.y != this->b.y)
|
if (this->a.y != this->b.y)
|
||||||
p->y = this->a.y + (this->b.y - this->a.y) * distance / len;
|
point->y = this->a.y + (this->b.y - this->a.y) * distance / len;
|
||||||
|
}
|
||||||
|
|
||||||
|
Point
|
||||||
|
Line::point_at(double distance) const
|
||||||
|
{
|
||||||
|
Point p;
|
||||||
|
this->point_at(distance, &p);
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,7 +25,8 @@ class Line
|
|||||||
void reverse();
|
void reverse();
|
||||||
double length() const;
|
double length() const;
|
||||||
Point* midpoint() const;
|
Point* midpoint() const;
|
||||||
Point* point_at(double distance) const;
|
void point_at(double distance, Point* point) const;
|
||||||
|
Point point_at(double distance) const;
|
||||||
bool coincides_with(const Line* line) const;
|
bool coincides_with(const Line* line) const;
|
||||||
double distance_to(const Point* point) const;
|
double distance_to(const Point* point) const;
|
||||||
double atan2_() const;
|
double atan2_() const;
|
||||||
|
@ -43,7 +43,7 @@ Polyline::clip_end(double distance)
|
|||||||
}
|
}
|
||||||
|
|
||||||
Line segment(last_point, *this->last_point());
|
Line segment(last_point, *this->last_point());
|
||||||
this->points.push_back(*segment.point_at(distance));
|
this->points.push_back(segment.point_at(distance));
|
||||||
distance = 0;
|
distance = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -57,6 +57,23 @@ Polyline::clip_start(double distance)
|
|||||||
if (this->points.size() >= 2) this->reverse();
|
if (this->points.size() >= 2) this->reverse();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Polyline::extend_end(double distance)
|
||||||
|
{
|
||||||
|
// relocate last point by extending the last segment by the specified length
|
||||||
|
Line line(this->points[ this->points.size()-2 ], this->points.back());
|
||||||
|
this->points.pop_back();
|
||||||
|
this->points.push_back(line.point_at(line.length() + distance));
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
Polyline::extend_start(double distance)
|
||||||
|
{
|
||||||
|
// relocate first point by extending the first segment by the specified length
|
||||||
|
Line line(this->points[1], this->points.front());
|
||||||
|
this->points[0] = line.point_at(line.length() + distance);
|
||||||
|
}
|
||||||
|
|
||||||
/* this method returns a collection of points picked on the polygon contour
|
/* this method returns a collection of points picked on the polygon contour
|
||||||
so that they are evenly spaced according to the input distance */
|
so that they are evenly spaced according to the input distance */
|
||||||
Points
|
Points
|
||||||
@ -79,7 +96,7 @@ Polyline::equally_spaced_points(double distance) const
|
|||||||
|
|
||||||
double take = segment_length - (len - distance); // how much we take of this segment
|
double take = segment_length - (len - distance); // how much we take of this segment
|
||||||
Line segment(*(it-1), *it);
|
Line segment(*(it-1), *it);
|
||||||
pts.push_back(*segment.point_at(take));
|
pts.push_back(segment.point_at(take));
|
||||||
it--;
|
it--;
|
||||||
len = -take;
|
len = -take;
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,8 @@ class Polyline : public MultiPoint {
|
|||||||
Lines lines() const;
|
Lines lines() const;
|
||||||
void clip_end(double distance);
|
void clip_end(double distance);
|
||||||
void clip_start(double distance);
|
void clip_start(double distance);
|
||||||
|
void extend_end(double distance);
|
||||||
|
void extend_start(double distance);
|
||||||
Points equally_spaced_points(double distance) const;
|
Points equally_spaced_points(double distance) const;
|
||||||
void simplify(double tolerance);
|
void simplify(double tolerance);
|
||||||
|
|
||||||
|
@ -4,7 +4,7 @@ use strict;
|
|||||||
use warnings;
|
use warnings;
|
||||||
|
|
||||||
use Slic3r::XS;
|
use Slic3r::XS;
|
||||||
use Test::More tests => 7;
|
use Test::More tests => 10;
|
||||||
|
|
||||||
my $points = [
|
my $points = [
|
||||||
[100, 100],
|
[100, 100],
|
||||||
@ -42,4 +42,13 @@ is_deeply $polyline->pp, [ @$points, @$points ], 'append_polyline';
|
|||||||
is_deeply $polyline->pp, [ [0, 0], [50, 50], [125, -25], [150, 50] ], 'Douglas-Peucker';
|
is_deeply $polyline->pp, [ [0, 0], [50, 50], [125, -25], [150, 50] ], 'Douglas-Peucker';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
my $polyline = Slic3r::Polyline->new(@$points);
|
||||||
|
is $polyline->length, 100*2, 'length';
|
||||||
|
$polyline->extend_end(50);
|
||||||
|
is $polyline->length, 100*2 + 50, 'extend_end';
|
||||||
|
$polyline->extend_start(50);
|
||||||
|
is $polyline->length, 100*2 + 50 + 50, 'extend_start';
|
||||||
|
}
|
||||||
|
|
||||||
__END__
|
__END__
|
||||||
|
@ -26,7 +26,7 @@
|
|||||||
Point* midpoint()
|
Point* midpoint()
|
||||||
%code{% const char* CLASS = "Slic3r::Point"; RETVAL = THIS->midpoint(); %};
|
%code{% const char* CLASS = "Slic3r::Point"; RETVAL = THIS->midpoint(); %};
|
||||||
Point* point_at(double distance)
|
Point* point_at(double distance)
|
||||||
%code{% const char* CLASS = "Slic3r::Point"; RETVAL = THIS->point_at(distance); %};
|
%code{% const char* CLASS = "Slic3r::Point"; RETVAL = new Point(THIS->point_at(distance)); %};
|
||||||
Polyline* as_polyline()
|
Polyline* as_polyline()
|
||||||
%code{% const char* CLASS = "Slic3r::Polyline"; RETVAL = new Polyline(*THIS); %};
|
%code{% const char* CLASS = "Slic3r::Polyline"; RETVAL = new Polyline(*THIS); %};
|
||||||
%{
|
%{
|
||||||
|
@ -29,6 +29,8 @@
|
|||||||
bool is_valid();
|
bool is_valid();
|
||||||
void clip_end(double distance);
|
void clip_end(double distance);
|
||||||
void clip_start(double distance);
|
void clip_start(double distance);
|
||||||
|
void extend_end(double distance);
|
||||||
|
void extend_start(double distance);
|
||||||
void simplify(double tolerance);
|
void simplify(double tolerance);
|
||||||
%{
|
%{
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user