doc fixes

This commit is contained in:
Adrian Mariano 2021-11-06 10:42:45 -04:00
parent 924421fa22
commit 4315b8c0b6

View file

@ -1,9 +1,11 @@
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////
// LibFile: regions.scad // LibFile: regions.scad
// This file provides 2D boolean geometry operations on paths, where you can // This file provides 2D boolean set operations on polygons, where you can
// compute the intersection or union of the shape defined by point lists, producing // compute, for example, the intersection or union of the shape defined by point lists, producing
// a new point list. Of course, boolean operations may produce shapes with multiple // a new point list. Of course, such operations may produce shapes with multiple
// components. To handle that, we use "regions" which are defined as lists of paths. // components. To handle that, we use "regions" which are defined by lists of polygons.
// In addition to set operations, you can calculate offsets, determine whether a point is in a
// region and you can decompose a region into parts.
// Includes: // Includes:
// include <BOSL2/std.scad> // include <BOSL2/std.scad>
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////
@ -16,30 +18,32 @@
// Section: Regions // Section: Regions
// A region is a list of polygons meeting these conditions: // A region is a list of polygons meeting these conditions:
// . // .
// - Every polygon on the list is simpel, meaning it does not intersect itself // - Every polygon on the list is simple, meaning it does not intersect itself
// - Two polygons on the list do not cross each other // - Two polygons on the list do not cross each other
// - A vertex of one polygon never meets the edge of another one except at a vertex // - A vertex of one polygon never meets the edge of another one except at a vertex
// . // .
// Note that this means vertex-vertex touching between two polygons is acceptable // Note that this means vertex-vertex touching between two polygons is acceptable
// to define a region. Note, however, that regions with vertex-vertex contact usually // to define a region. Note, however, that regions with vertex-vertex contact usually
// cannot be rendered with CGAL. // cannot be rendered with CGAL. See {{is_valid_region()}} for examples of valid regions and
// lists of polygons that are not regions. Note that {{is_region_simple()}} will identify
// regions with no polygon intersections at all, which should render successfully witih CGAL.
// . // .
// The actual geometry of the region is defined by XORing together // The actual geometry of the region is defined by XORing together
// all of the polygons on the list. This may sound obscure, but it simply means that nested // all of the polygons in the list. This may sound obscure, but it simply means that nested
// boundaries make rings in the obvious fashion, and non-nested shapes simply union together. // boundaries make rings in the obvious fashion, and non-nested shapes simply union together.
// Checking that a list of paths is a valid region, meaning that it satisfies all of the conditions // Checking that a list of polygons is a valid region, meaning that it satisfies all of the conditions
// above, can be a time consuming test, so it is not done automatically. It is your responsibility to ensure that your regions are // above, can be a time consuming test, so it is not done automatically. It is your responsibility to ensure that your regions are
// compliant. You can construct regions by making a list of polygons, or by using // compliant. You can construct regions by making a suitable list of polygons, or by using
// boolean function operations such as union() or difference(), which all acccept paths, as // set operation function such as union() or difference(), which all acccept polygons, as
// well as regions, as their inputs. And if you must you can clean up an ill-formed region using make_region(), // well as regions, as their inputs. And if you must you can clean up an ill-formed region using make_region(),
// which will break up self-intersecting paths and paths that cross each other. // which will break up self-intersecting polygons and polygons that cross each other.
// Function: is_region() // Function: is_region()
// Usage: // Usage:
// is_region(x); // is_region(x);
// Description: // Description:
// Returns true if the given item looks like a region. A region is a list of non-crossing simple paths. This test just checks // Returns true if the given item looks like a region. A region is a list of non-crossing simple polygons. This test just checks
// that the argument is a list whose first entry is a path. // that the argument is a list whose first entry is a path.
function is_region(x) = is_list(x) && is_path(x.x); function is_region(x) = is_list(x) && is_path(x.x);
@ -48,16 +52,16 @@ function is_region(x) = is_list(x) && is_path(x.x);
// Usage: // Usage:
// bool = is_valid_region(region, [eps]); // bool = is_valid_region(region, [eps]);
// Description: // Description:
// Returns true if the input is a valid region, meaning that it is a list of simple paths whose segments do not cross each other. // Returns true if the input is a valid region, meaning that it is a list of simple polygons whose segments do not cross each other.
// This test can be time consuming with regions that contain many points. // This test can be time consuming with regions that contain many points.
// It differs from `is_region()` which simply checks that the object appears to be a list of paths // It differs from `is_region()` which simply checks that the object is a list whose first entry is a path
// because it searches all the region paths for any self-intersections or intersections with each other. // because it searches all the list polygons for any self-intersections or intersections with each other.
// Will also return true if given a single simple path. Use {{make_region()}} to convert sets of self-intersecting polygons into // Will also return true if given a single simple polygon. Use {{make_region()}} to convert sets of self-intersecting polygons into
// a region. // a region.
// Arguments: // Arguments:
// region = region to check // region = region to check
// eps = tolerance for geometric comparisons. Default: `EPSILON` = 1e-9 // eps = tolerance for geometric comparisons. Default: `EPSILON` = 1e-9
// Example(2D,NoAxes): In all of the examples each path in the region appears in a different color. Two non-intersecting squares make a valid region. // Example(2D,NoAxes): In all of the examples each polygon in the region appears in a different color. Two non-intersecting squares make a valid region.
// region = [square(10), right(11,square(8))]; // region = [square(10), right(11,square(8))];
// rainbow(region)stroke($item, width=.2,closed=true); // rainbow(region)stroke($item, width=.2,closed=true);
// back(11)text(is_valid_region(region) ? "region" : "non-region", size=2); // back(11)text(is_valid_region(region) ? "region" : "non-region", size=2);
@ -73,23 +77,27 @@ function is_region(x) = is_list(x) && is_path(x.x);
// object = [square(10), move([8,8], square(8))]; // object = [square(10), move([8,8], square(8))];
// rainbow(object)stroke($item, width=.2,closed=true); // rainbow(object)stroke($item, width=.2,closed=true);
// back(17)text(is_valid_region(object) ? "region" : "non-region", size=2); // back(17)text(is_valid_region(object) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): Not a region due to a self-intersecting (non-simple) hourglass path // Example(2D,NoAxes): A union is one way to fix the above example and get a region. (Note that union is run here on two simple polygons, which are valid regions themselves and hence acceptable inputs to union.
// region = union([square(10), move([8,8], square(8))]);
// rainbow(region)stroke($item, width=.25,closed=true);
// back(12)text(is_valid_region(region) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): Not a region due to a self-intersecting (non-simple) hourglass polygon
// object = [move([-2,-2],square(14)), [[0,0],[10,0],[0,10],[10,10]]]; // object = [move([-2,-2],square(14)), [[0,0],[10,0],[0,10],[10,10]]];
// rainbow(object)stroke($item, width=.2,closed=true); // rainbow(object)stroke($item, width=.2,closed=true);
// move([-1.5,13])text(is_valid_region(object) ? "region" : "non-region", size=2); // move([-1.5,13])text(is_valid_region(object) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): Breaking hourglass in half fixes it. Now it's a region: // Example(2D,NoAxes): Breaking hourglass in half fixes it. Now it's a region:
// region = [move([-2,-2],square(14)), [[0,0],[10,0],[5,5]], [[5,5],[0,10],[10,10]]]; // region = [move([-2,-2],square(14)), [[0,0],[10,0],[5,5]], [[5,5],[0,10],[10,10]]];
// rainbow(region)stroke($item, width=.2,closed=true); // rainbow(region)stroke($item, width=.2,closed=true);
// Example(2D,NoAxes): A single path corner touches an edge, so not a region: // Example(2D,NoAxes): A single polygon corner touches an edge, so not a region:
// object = [[[-10,0], [-10,10], [20,10], [20,-20], [-10,-20], // object = [[[-10,0], [-10,10], [20,10], [20,-20], [-10,-20],
// [-10,-10], [0,0], [10,-10], [10,0]]]; // [-10,-10], [0,0], [10,-10], [10,0]]];
// rainbow(object)stroke($item, width=.3,closed=true); // rainbow(object)stroke($item, width=.3,closed=true);
// move([-4,12])text(is_valid_region(object) ? "region" : "non-region", size=3); // move([-4,12])text(is_valid_region(object) ? "region" : "non-region", size=3);
// Example(2D,NoAxes): Corners touch in the same path, so the path is not simple and the object is not a region. // Example(2D,NoAxes): Corners touch in the same polygon, so the polygon is not simple and the object is not a region.
// object = [[[0,0],[10,0],[10,10],[-10,10],[-10,0],[0,0],[-5,5],[5,5]]]; // object = [[[0,0],[10,0],[10,10],[-10,10],[-10,0],[0,0],[-5,5],[5,5]]];
// rainbow(object)stroke($item, width=.3,closed=true); // rainbow(object)stroke($item, width=.3,closed=true);
// move([-10,12])text(is_valid_region(object) ? "region" : "non-region", size=3); // move([-10,12])text(is_valid_region(object) ? "region" : "non-region", size=3);
// Example(2D,NoAxes): The shape above as a valid region with two paths: // Example(2D,NoAxes): The shape above as a valid region with two polygons:
// region = [ [[0,0],[10,0],[10,10],[-10,10],[-10,0]], // region = [ [[0,0],[10,0],[10,10],[-10,10],[-10,0]],
// [[0,0],[5,5],[-5,5]] ]; // [[0,0],[5,5],[-5,5]] ];
// rainbow(region)stroke($item, width=.3,closed=true); // rainbow(region)stroke($item, width=.3,closed=true);
@ -98,10 +106,6 @@ function is_region(x) = is_list(x) && is_path(x.x);
// region = [square(10), move([10,10], square(8))]; // region = [square(10), move([10,10], square(8))];
// rainbow(region)stroke($item, width=.25,closed=true); // rainbow(region)stroke($item, width=.25,closed=true);
// back(12)text(is_valid_region(region) ? "region" : "non-region", size=2); // back(12)text(is_valid_region(region) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): A union is one way to fix the above example and get a region. (Note that union is run here on two simple paths, which are valid regions themselves and hence acceptable inputs to union.
// region = union([square(10), move([8,8], square(8))]);
// rainbow(region)stroke($item, width=.25,closed=true);
// back(12)text(is_valid_region(region) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): These two squares share part of an edge, hence not a region // Example(2D,NoAxes): These two squares share part of an edge, hence not a region
// object = [square(10), move([10,2], square(7))]; // object = [square(10), move([10,2], square(7))];
// stroke(object[0], width=0.2,closed=true); // stroke(object[0], width=0.2,closed=true);
@ -121,11 +125,11 @@ function is_region(x) = is_list(x) && is_path(x.x);
// object = [square(10), [[10,0],[0,10],[8,13],[13,8]]]; // object = [square(10), [[10,0],[0,10],[8,13],[13,8]]];
// rainbow(object)stroke($item, width=.2,closed=true); // rainbow(object)stroke($item, width=.2,closed=true);
// back(14)text(is_valid_region(object) ? "region" : "non-region", size=2); // back(14)text(is_valid_region(object) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): One path touches another in the middle of an edge // Example(2D,NoAxes): One polygon touches another in the middle of an edge
// object = [square(10), [[10,5],[15,0],[15,10]]]; // object = [square(10), [[10,5],[15,0],[15,10]]];
// rainbow(object)stroke($item, width=.2,closed=true); // rainbow(object)stroke($item, width=.2,closed=true);
// back(11)text(is_valid_region(object) ? "region" : "non-region", size=2); // back(11)text(is_valid_region(object) ? "region" : "non-region", size=2);
// Example(2D,NoAxes): The path touches the side, but the side has a vertex at the contact point so this is a region // Example(2D,NoAxes): The polygon touches the side, but the side has a vertex at the contact point so this is a region
// poly1 = [ each square(30,center=true), [15,0]]; // poly1 = [ each square(30,center=true), [15,0]];
// poly2 = right(10,circle(5,$fn=4)); // poly2 = right(10,circle(5,$fn=4));
// poly3 = left(0,circle(5,$fn=4)); // poly3 = left(0,circle(5,$fn=4));
@ -134,7 +138,7 @@ function is_region(x) = is_list(x) && is_path(x.x);
// rainbow(region)stroke($item, width=.25,closed=true); // rainbow(region)stroke($item, width=.25,closed=true);
// move([-5,16.5])text(is_valid_region(region) ? "region" : "non-region", size=3); // move([-5,16.5])text(is_valid_region(region) ? "region" : "non-region", size=3);
// color("black")move_copies(region[0]) circle(r=.4); // color("black")move_copies(region[0]) circle(r=.4);
// Example(2D,NoAxes): The path touches the side, but not at a vertex so this is not a region // Example(2D,NoAxes): The polygon touches the side, but not at a vertex so this is not a region
// poly1 = fwd(4,[ each square(30,center=true), [15,0]]); // poly1 = fwd(4,[ each square(30,center=true), [15,0]]);
// poly2 = right(10,circle(5,$fn=4)); // poly2 = right(10,circle(5,$fn=4));
// poly3 = left(0,circle(5,$fn=4)); // poly3 = left(0,circle(5,$fn=4));
@ -143,13 +147,13 @@ function is_region(x) = is_list(x) && is_path(x.x);
// rainbow(object)stroke($item, width=.25,closed=true); // rainbow(object)stroke($item, width=.25,closed=true);
// move([-9,12.5])text(is_valid_region(object) ? "region" : "non-region", size=3); // move([-9,12.5])text(is_valid_region(object) ? "region" : "non-region", size=3);
// color("black")move_copies(object[0]) circle(r=.4); // color("black")move_copies(object[0]) circle(r=.4);
// Example(2D,NoAxes): The inner path touches the middle of the edges, so not a region // Example(2D,NoAxes): The inner polygon touches the middle of the edges, so not a region
// poly1 = square(20,center=true); // poly1 = square(20,center=true);
// poly2 = circle(10,$fn=8); // poly2 = circle(10,$fn=8);
// object=[poly1,poly2]; // object=[poly1,poly2];
// rainbow(object)stroke($item, width=.25,closed=true); // rainbow(object)stroke($item, width=.25,closed=true);
// move([-10,11.4])text(is_valid_region(object) ? "region" : "non-region", size=3); // move([-10,11.4])text(is_valid_region(object) ? "region" : "non-region", size=3);
// Example(2D,NoAxes): The above shape made into a region using difference(): // Example(2D,NoAxes): The above shape made into a region using {{difference()}} now has four components that touch at corners
// poly1 = square(20,center=true); // poly1 = square(20,center=true);
// poly2 = circle(10,$fn=8); // poly2 = circle(10,$fn=8);
// region = difference(poly1,poly2); // region = difference(poly1,poly2);
@ -212,7 +216,15 @@ function _polygon_crosses_region(region, poly, eps=EPSILON) =
// Arguments: // Arguments:
// region = region to check // region = region to check
// eps = tolerance for geometric comparisons. Default: `EPSILON` = 1e-9 // eps = tolerance for geometric comparisons. Default: `EPSILON` = 1e-9
function is_simple_region(region, eps=EPSILON) = // Example(2D,NoAxes): Corner contact means it's not simple
// region = [move([-2,-2],square(14)), [[0,0],[10,0],[5,5]], [[5,5],[0,10],[10,10]]];
// rainbow(region)stroke($item, width=.2,closed=true);
// move([-1,13])text(is_region_simple(region) ? "simple" : "not-simple", size=2);
// Example(2D,NoAxes): Moving apart the triangles makes it simple:
// region = [move([-2,-2],square(14)), [[0,0],[10,0],[5,4.5]], [[5,5.5],[0,10],[10,10]]];
// rainbow(region)stroke($item, width=.2,closed=true);
// move([1,13])text(is_region_simple(region) ? "simple" : "not-simple", size=2);
function is_region_simple(region, eps=EPSILON) =
let(region=force_region(region)) let(region=force_region(region))
assert(is_region(region), "Input is not a region") assert(is_region(region), "Input is not a region")
[for(p=region) if (!is_path_simple(p,closed=true,eps=eps)) 1] == [] [for(p=region) if (!is_path_simple(p,closed=true,eps=eps)) 1] == []
@ -224,32 +236,41 @@ function is_simple_region(region, eps=EPSILON) =
// Function: make_region() // Function: make_region()
// Usage: // Usage:
// r_fixed = make_region(r, [nonzero], [eps]); // region = make_region(polys, [nonzero], [eps]);
// Description: // Description:
// Takes a malformed input region that contains self-intersecting polygons or polygons // Takes a list of polygons that may intersect themselves or cross each other
// that cross each other and converts it into a properly defined region without // and converts it into a properly defined region without
// these defects. // these defects.
// Arguments: // Arguments:
// r = region to sanitize // polys = list of polygons to use
// nonzero = set to true to use nonzero rule for polygon membership. Default: false // nonzero = set to true to use nonzero rule for polygon membership. Default: false
// eps = Epsilon for geometric comparisons. Default: `EPSILON` (1e-9) // eps = Epsilon for geometric comparisons. Default: `EPSILON` (1e-9)
// Examples: // Example(2D,NoAxes): The pentagram is self-intersecting, so it is not a region. Here it becomes five triangles:
// // pentagram = turtle(["move",100,"left",144], repeat=4);
function make_region(r,nonzero=false,eps=EPSILON) = // region = make_region(pentagram);
let(r=force_region(r)) // rainbow(region)stroke($item, width=1,closed=true);
assert(is_region(r), "Input is not a region") // Examples(2D,NoAxes): Alternatively with the nonzero option you can get the perimeter:
// pentagram = turtle(["move",100,"left",144], repeat=4);
// region = make_region(pentagram,nonzero=true);
// stroke(region, width=1,closed=true);
// Example(2D,NoAxes): To crossing squares become two L-shaped components
// region = make_region([square(10), move([5,5],square(8))]);
// rainbow(region)stroke($item, width=.3,closed=true);
function make_region(polys,nonzero=false,eps=EPSILON) =
let(polys=force_region(polys))
assert(is_region(polys), "Input is not a region")
exclusive_or( exclusive_or(
[for(poly=r) each polygon_parts(poly,nonzero,eps)], [for(poly=polys) each polygon_parts(poly,nonzero,eps)],
eps=eps); eps=eps);
// Function: force_region() // Function: force_region()
// Usage: // Usage:
// region = force_region(path) // region = force_region(poly)
// Description: // Description:
// If the input is a path then return it as a region. Otherwise return it unaltered. // If the input is a polygon then return it as a region. Otherwise return it unaltered.
function force_region(path) = is_path(path) ? [path] : path; // Arguments:
// poly = polygon to turn into a region
function force_region(poly) = is_path(poly) ? [poly] : poly;
// Section: Turning a region into geometry // Section: Turning a region into geometry
@ -258,12 +279,14 @@ function force_region(path) = is_path(path) ? [path] : path;
// Usage: // Usage:
// region(r); // region(r);
// Description: // Description:
// Creates 2D polygons for the given region. The region given is a list of closed 2D paths. // Creates the 2D polygons described by the given region or list of polygons. This module works on
// Each path will be effectively exclusive-ORed from all other paths in the region, so if a // arbitrary lists of polygons that cross each other and hence do not define a valid region. The
// path is inside another path, it will be effectively subtracted from it. // displayed result is the exclusive-or of the polygons listed in the input.
// Example(2D): // Arguments:
// r = region to create as geometry
// Example(2D): Displaying a region
// region([circle(d=50), square(25,center=true)]); // region([circle(d=50), square(25,center=true)]);
// Example(2D): // Example(2D): Displaying a list of polygons that intersect each other, which is not a region
// rgn = concat( // rgn = concat(
// [for (d=[50:-10:10]) circle(d=d-5)], // [for (d=[50:-10:10]) circle(d=d-5)],
// [square([60,10], center=true)] // [square([60,10], center=true)]
@ -283,7 +306,7 @@ module region(r)
// Section: Gometrical calculations with region // Section: Gometrical calculations with regions
// Function: point_in_region() // Function: point_in_region()
// Usage: // Usage:
@ -295,15 +318,24 @@ module region(r)
// Returns 1 if the point lies inside the region. // Returns 1 if the point lies inside the region.
// Arguments: // Arguments:
// point = The point to test. // point = The point to test.
// region = The region to test against. Given as a list of polygon paths. // region = The region to test against, as a list of polygon paths.
// eps = Acceptable variance. Default: `EPSILON` (1e-9) // eps = Acceptable variance. Default: `EPSILON` (1e-9)
function point_in_region(point, region, eps=EPSILON, _i=0, _cnt=0) = // Example(2D,NoAxes):
_i >= len(region) ? ((_cnt%2==1)? 1 : -1) // region = [for(i=[2:8]) hexagon(r=i)];
// for(x=[-4:4],y=[-4:4]) color(point_in_region
function point_in_region(point, region, eps=EPSILON) =
let(region=force_region(region))
assert(is_region(region), "Region given to point_in_region is not a region")
assert(is_vector(point,2), "Point must be a 2D point in point_in_region")
_point_in_region(point, region, eps);
function pointin_region(point, region, eps=EPSILON, i=0, cnt=0) =
i >= len(region) ? ((cnt%2==1)? 1 : -1)
: let( : let(
pip = point_in_polygon(point, region[_i], eps=eps) pip = point_in_polygon(point, region[i], eps=eps)
) )
pip==0? 0 pip == 0 ? 0
: point_in_region(point, region, eps=eps, _i=_i+1, _cnt = _cnt + (pip>0? 1 : 0)); : _point_in_region(point, region, eps=eps, i=i+1, cnt = cnt + (pip>0? 1 : 0));
// Function: region_area() // Function: region_area()
@ -312,6 +344,10 @@ function point_in_region(point, region, eps=EPSILON, _i=0, _cnt=0) =
// Description: // Description:
// Computes the area of the specified valid region. (If the region is invalid and has self intersections // Computes the area of the specified valid region. (If the region is invalid and has self intersections
// the result is meaningless.) // the result is meaningless.)
// Arguments:
// region = region whose area to compute
// Examples:
// area = region_area([square(10), right(20,square(8))]); // Returns 164
function region_area(region) = function region_area(region) =
assert(is_region(region), "Input must be a region") assert(is_region(region), "Input must be a region")
let( let(
@ -437,9 +473,13 @@ function _region_region_intersections(region1, region2, closed1=true,closed2=tru
// Splits region1 at the places where polygons in region1 touches each other at corners and at locations // Splits region1 at the places where polygons in region1 touches each other at corners and at locations
// where region1 intersections region2. Split region2 similarly with respect to region1. // where region1 intersections region2. Split region2 similarly with respect to region1.
// The return is a pair of results of the form [split1, split2] where split1=[frags1,frags2,...] // The return is a pair of results of the form [split1, split2] where split1=[frags1,frags2,...]
// and frags1 is a list of path pieces (in order) from the first path of the region. // and frags1 is a list of paths that when placed end to end (in the given order), give the first polygon of region1.
// You can pass a single path in for either region, but the output will be a singleton list, as ify // Each path in the list is either entirely inside or entirely outside region2.
// you passed in a singleton region. // Then frags2 is the decomposition of the second polygon into path pieces, and so on. Finally split2 is
// the same list, but for the polygons in region2.
// You can pass a single polygon in for either region, but the output will be a singleton list, as if
// you passed in a singleton region. If you set the closed parameters to false then the region components
// will be treated as open paths instead of polygons.
// Arguments: // Arguments:
// region1 = first region // region1 = first region
// region2 = second region // region2 = second region
@ -543,9 +583,9 @@ function region_parts(region) =
// Usage: // Usage:
// linear_sweep(region, height, [center], [slices], [twist], [scale], [style], [convexity]); // linear_sweep(region, height, [center], [slices], [twist], [scale], [style], [convexity]);
// Description: // Description:
// If called as a module, creates a polyhedron that is the linear extrusion of the given 2D region or path. // If called as a module, creates a polyhedron that is the linear extrusion of the given 2D region or polygon.
// If called as a function, returns a VNF that can be used to generate a polyhedron of the linear extrusion // If called as a function, returns a VNF that can be used to generate a polyhedron of the linear extrusion
// of the given 2D region or path. The benefit of using this, over using `linear_extrude region(rgn)` is // of the given 2D region or polygon. The benefit of using this, over using `linear_extrude region(rgn)` is
// that it supports `anchor`, `spin`, `orient` and attachments. You can also make more refined // that it supports `anchor`, `spin`, `orient` and attachments. You can also make more refined
// twisted extrusions by using `maxseg` to subsample flat faces. // twisted extrusions by using `maxseg` to subsample flat faces.
// Note that the center option centers vertically using the named anchor "zcenter" whereas // Note that the center option centers vertically using the named anchor "zcenter" whereas
@ -553,7 +593,7 @@ function region_parts(region) =
// the shape's centroid, or other centerpoint you specify. The centerpoint can be "centroid", "mean", "box" or // the shape's centroid, or other centerpoint you specify. The centerpoint can be "centroid", "mean", "box" or
// a custom point location. // a custom point location.
// Arguments: // Arguments:
// region = The 2D [Region](regions.scad) or path that is to be extruded. // region = The 2D [Region](regions.scad) or polygon that is to be extruded.
// height = The height to extrude the region. Default: 1 // height = The height to extrude the region. Default: 1
// center = If true, the created polyhedron will be vertically centered. If false, it will be extruded upwards from the XY plane. Default: `false` // center = If true, the created polyhedron will be vertically centered. If false, it will be extruded upwards from the XY plane. Default: `false`
// slices = The number of slices to divide the shape into along the Z axis, to allow refinement of detail, especially when working with a twist. Default: `twist/5` // slices = The number of slices to divide the shape into along the Z axis, to allow refinement of detail, especially when working with a twist. Default: `twist/5`
@ -656,7 +696,7 @@ function linear_sweep(region, height=1, center, twist=0, scale=1, slices,
// Section: Offsets and Boolean 2D Geometry // Section: Offset and 2D Boolean Set Operations
function _offset_chamfer(center, points, delta) = function _offset_chamfer(center, points, delta) =
@ -773,10 +813,11 @@ function _point_dist(path,pathseg_unit,pathseg_len,pt) =
// offsetpath = offset(path, [r|delta], [chamfer], [closed], [check_valid], [quality]) // offsetpath = offset(path, [r|delta], [chamfer], [closed], [check_valid], [quality])
// path_faces = offset(path, return_faces=true, [r|delta], [chamfer], [closed], [check_valid], [quality], [firstface_index], [flip_faces]) // path_faces = offset(path, return_faces=true, [r|delta], [chamfer], [closed], [check_valid], [quality], [firstface_index], [flip_faces])
// Description: // Description:
// Takes a 2D input path and returns a path offset by the specified amount. As with the built-in // Takes a 2D input path, polygon or region and returns a path offset by the specified amount. As with the built-in
// offset() module, you can use `r` to specify rounded offset and `delta` to specify offset with // offset() module, you can use `r` to specify rounded offset and `delta` to specify offset with
// corners. If you used `delta` you can set `chamfer` to true to get chamfers. // corners. If you used `delta` you can set `chamfer` to true to get chamfers.
// Positive offsets shift the path to the left (relative to the direction of the path). Note // For paths and polygons positive offsets make the polygons larger. For paths,
// positive offsets shift the path to the left, relative to the direction of the path. Note
// that the path must not include any 180 degree turns, where the path reverses direction. // that the path must not include any 180 degree turns, where the path reverses direction.
// . // .
// When offsets shrink the path, segments cross and become invalid. By default `offset()` checks // When offsets shrink the path, segments cross and become invalid. By default `offset()` checks
@ -800,7 +841,7 @@ function _point_dist(path,pathseg_unit,pathseg_len,pt) =
// r = offset radius. Distance to offset. Will round over corners. // r = offset radius. Distance to offset. Will round over corners.
// delta = offset distance. Distance to offset with pointed corners. // delta = offset distance. Distance to offset with pointed corners.
// chamfer = chamfer corners when you specify `delta`. Default: false // chamfer = chamfer corners when you specify `delta`. Default: false
// closed = path is a closed curve. Default: False. // closed = if true path is treate as a polygon. Default: False.
// check_valid = perform segment validity check. Default: True. // check_valid = perform segment validity check. Default: True.
// quality = validity check quality parameter, a small integer. Default: 1. // quality = validity check quality parameter, a small integer. Default: 1.
// return_faces = return face list. Default: False. // return_faces = return face list. Default: False.
@ -1050,11 +1091,11 @@ function _list_three(a,b,c) =
// region = union(REGION1,REGION2); // region = union(REGION1,REGION2);
// region = union(REGION1,REGION2,REGION3); // region = union(REGION1,REGION2,REGION3);
// Description: // Description:
// When called as a function and given a list of regions, where each region is a list of closed // When called as a function and given a list of regions or 2D polygons,
// 2D paths, returns the boolean union of all given regions. Result is a single region. // returns the union of all given regions and polygons. Result is a single region.
// When called as the built-in module, makes the boolean union of the given children. // When called as the built-in module, makes the union of the given children.
// Arguments: // Arguments:
// regions = List of regions to union. Each region is a list of closed paths. // regions = List of regions to union.
// Example(2D): // Example(2D):
// shape1 = move([-8,-8,0], p=circle(d=50)); // shape1 = move([-8,-8,0], p=circle(d=50));
// shape2 = move([ 8, 8,0], p=circle(d=50)); // shape2 = move([ 8, 8,0], p=circle(d=50));
@ -1080,12 +1121,12 @@ function union(regions=[],b=undef,c=undef,eps=EPSILON) =
// region = difference(REGION1,REGION2); // region = difference(REGION1,REGION2);
// region = difference(REGION1,REGION2,REGION3); // region = difference(REGION1,REGION2,REGION3);
// Description: // Description:
// When called as a function, and given a list of regions, where each region is a list of closed // When called as a function, and given a list of regions or 2D polygons,
// 2D paths, takes the first region and differences away all other regions from it. The resulting // takes the first region or polygon and differences away all other regions/polygons from it. The resulting
// region is returned. // region is returned.
// When called as the built-in module, makes the boolean difference of the given children. // When called as the built-in module, makes the set difference of the given children.
// Arguments: // Arguments:
// regions = List of regions to difference. Each region is a list of closed paths. // regions = List of regions or polygons to difference.
// Example(2D): // Example(2D):
// shape1 = move([-8,-8,0], p=circle(d=50)); // shape1 = move([-8,-8,0], p=circle(d=50));
// shape2 = move([ 8, 8,0], p=circle(d=50)); // shape2 = move([ 8, 8,0], p=circle(d=50));
@ -1112,11 +1153,11 @@ function difference(regions=[],b=undef,c=undef,eps=EPSILON) =
// region = intersection(REGION1,REGION2); // region = intersection(REGION1,REGION2);
// region = intersection(REGION1,REGION2,REGION3); // region = intersection(REGION1,REGION2,REGION3);
// Description: // Description:
// When called as a function, and given a list of regions, where each region is a list of closed // When called as a function, and given a list of regions or polygons returns the
// 2D paths, returns the boolean intersection of all given regions. Result is a single region. // intersection of all given regions. Result is a single region.
// When called as the built-in module, makes the boolean intersection of all the given children. // When called as the built-in module, makes the intersection of all the given children.
// Arguments: // Arguments:
// regions = List of regions to intersection. Each region is a list of closed paths. // regions = List of regions to intersect.
// Example(2D): // Example(2D):
// shape1 = move([-8,-8,0], p=circle(d=50)); // shape1 = move([-8,-8,0], p=circle(d=50));
// shape2 = move([ 8, 8,0], p=circle(d=50)); // shape2 = move([ 8, 8,0], p=circle(d=50));
@ -1143,12 +1184,13 @@ function intersection(regions=[],b=undef,c=undef,eps=EPSILON) =
// region = exclusive_or(REGION1,REGION2); // region = exclusive_or(REGION1,REGION2);
// region = exclusive_or(REGION1,REGION2,REGION3); // region = exclusive_or(REGION1,REGION2,REGION3);
// Description: // Description:
// When called as a function and given a list of regions, where each region is a list of closed // When called as a function and given a list of regions or 2D polygons,
// 2D paths, returns the boolean exclusive_or of all given regions. Result is a single region. // returns the exclusive_or of all given regions. Result is a single region.
// When called as a module, performs a boolean exclusive-or of up to 10 children. Note that the // When called as a module, performs a boolean exclusive-or of up to 10 children. Note that when
// xor operator tends to produce shapes that meet at corners, which do not render in CGAL. // the input regions cross each other the exclusive-or operator will produce shapes that
// meet at corners (non-simple regions), which do not render in CGAL.
// Arguments: // Arguments:
// regions = List of regions to exclusive_or. Each region is a list of closed paths. // regions = List of regions or polygons to exclusive_or
// Example(2D): As Function. A linear_sweep of this shape fails to render in CGAL. // Example(2D): As Function. A linear_sweep of this shape fails to render in CGAL.
// shape1 = move([-8,-8,0], p=circle(d=50)); // shape1 = move([-8,-8,0], p=circle(d=50));
// shape2 = move([ 8, 8,0], p=circle(d=50)); // shape2 = move([ 8, 8,0], p=circle(d=50));