Improve docs for path_sweep

make echo_matrix handle non-numeric entries gracefully and add
separator arg.
This commit is contained in:
Adrian Mariano 2022-02-21 20:45:57 -05:00
parent c280513253
commit 77a78bf8bc
3 changed files with 57 additions and 30 deletions

View file

@ -64,25 +64,36 @@ function is_matrix_symmetric(A,eps=1e-12) =
// Function&Module: echo_matrix() // Function&Module: echo_matrix()
// Usage: // Usage:
// echo_matrix(M, [description=], [sig=], [eps=]); // echo_matrix(M, [description], [sig], [sep], [eps]);
// dummy = echo_matrix(M, [description=], [sig=], [eps=]), // dummy = echo_matrix(M, [description], [sig], [sep], [eps]),
// Description: // Description:
// Display a numerical matrix in a readable columnar format with `sig` significant // Display a numerical matrix in a readable columnar format with `sig` significant
// digits. Values smaller than eps display as zero. If you give a description // digits. Values smaller than eps display as zero. If you give a description
// it is displayed at the top. // it is displayed at the top. You can change the space between columns by
function echo_matrix(M,description,sig=4,eps=1e-9) = // setting `sep` to a number of spaces, which will use wide figure spaces the same
// width as digits, or you can set it to any string to separate the columns.
// Values that are NaN or INF will display as "nan" and "inf". Values which are
// otherwise non-numerica display as two dashes. Note that this includes lists, so
// a 3D array will display as a list of dashes.
// Arguments:
// M = matrix to display, which should be numerical
// description = optional text to print before the matrix
// sig = number of digits to display. Default: 4
// sep = number of spaces between columns or a text string to separate columns. Default: 1
// eps = numbers smaller than this display as zero. Default: 1e-9
function echo_matrix(M,description,sig=4,sep,eps=1e-9) =
let( let(
horiz_line = chr(8213), horiz_line = chr(8213),
matstr = _format_matrix(M,sig=sig,eps=eps), matstr = _format_matrix(M,sig=sig,sep=sep,eps=eps),
separator = str_join(repeat(horiz_line,10)), separator = str_join(repeat(horiz_line,10)),
dummy=echo(str(separator,is_def(description) ? str(" ",description) : "")) dummy=echo(str(separator,is_def(description) ? str(" ",description) : ""))
[for(row=matstr) echo(row)] [for(row=matstr) echo(row)]
) )
echo(separator); echo(separator);
module echo_matrix(M,description,sig=4,eps=1e-9) module echo_matrix(M,description,sig=4,sep,eps=1e-9)
{ {
dummy = echo_matrix(M,description,sig,eps); dummy = echo_matrix(M,description,sig,sep,eps);
} }

View file

@ -33,8 +33,8 @@
// The profiles can be specified either as a list of 3d curves or they can be specified as // The profiles can be specified either as a list of 3d curves or they can be specified as
// 2d curves with heights given in the `z` parameter. It is your responsibility to ensure // 2d curves with heights given in the `z` parameter. It is your responsibility to ensure
// that the resulting polyhedron is free from self-intersections, which would make it invalid // that the resulting polyhedron is free from self-intersections, which would make it invalid
// and can result in cryptic CGAL errors upon rendering, even though the polyhedron appears // and can result in cryptic CGAL errors upon rendering with a second object present, even though the polyhedron appears
// OK during preview. // OK during preview or when rendered by itself.
// . // .
// For this operation to be well-defined, the profiles must all have the same vertex count and // For this operation to be well-defined, the profiles must all have the same vertex count and
// we must assume that profiles are aligned so that vertex `i` links to vertex `i` on all polygons. // we must assume that profiles are aligned so that vertex `i` links to vertex `i` on all polygons.
@ -730,45 +730,53 @@ module spiral_sweep(poly, h, r, turns=1, higbee, center, r1, r2, d, d1, d2, higb
// Function&Module: path_sweep() // Function&Module: path_sweep()
// Usage: As module // Usage: As module
// path_sweep(shape, path, [method], [normal=], [closed=], [twist=], [twist_by_length=], [symmetry=], [last_normal=], [tangent=], [relaxed=], [caps=], [style=], [convexity=], [anchor=], [cp=], [spin=], [orient=], [atype=]) {attachments}; // path_sweep(shape, path, [method], [normal=], [closed=], [twist=], [twist_by_length=], [symmetry=], [last_normal=], [tangent=], [uniform=], [relaxed=], [caps=], [style=], [convexity=], [anchor=], [cp=], [spin=], [orient=], [atype=]) {attachments};
// Usage: As function // Usage: As function
// vnf = path_sweep(shape, path, [method], [normal=], [closed=], [twist=], [twist_by_length=], [symmetry=], [last_normal=], [tangent=], [relaxed=], [caps=], [style=], [transforms=], [anchor=], [cp=], [spin=], [orient=], [atype=]) {attachments}; // vnf = path_sweep(shape, path, [method], [normal=], [closed=], [twist=], [twist_by_length=], [symmetry=], [last_normal=], [tangent=], [uniform=], [relaxed=], [caps=], [style=], [transforms=], [anchor=], [cp=], [spin=], [orient=], [atype=]) {attachments};
// Description: // Description:
// Takes as input `shape`, a 2D polygon path (list of points), and `path`, a 2d or 3d path (also a list of points) // Takes as input `shape`, a 2D polygon path (list of points), and `path`, a 2d or 3d path (also a list of points)
// and constructs a polyhedron by sweeping the shape along the path. When run as a module returns the polyhedron geometry. // and constructs a polyhedron by sweeping the shape along the path. When run as a module returns the polyhedron geometry.
// When run as a function returns a VNF by default or if you set `transforms=true` then it returns a list of transformations suitable as input to `sweep`. // When run as a function returns a VNF by default or if you set `transforms=true` then it returns a list of transformations suitable as input to `sweep`.
// . // .
// Figure(3D,Big,VPR=[70,0,345],VPD=20,VPT=[5.5,10.8,-2.7],NoScales): This example shows how the shape, in this case the triangle defined by `[[0, 0], [0, 1], [1, 0]]`, appears as the cross section of the swept polyhedron. The blue line shows the path. The normal vector to the shape points upwards, in the Z direction. // The sweeping process places one copy of the shape for each point in the path. The origin in `shape` is translated to
// tri= [[0, 0], [0, 1], [1, 0]]; // the point in `path`. The normal vector of the shape, which points in the Z direction, is aligned with the tangent
// vector for the path, so this process is constructing a shape whose normal cross sections are equal to your specified shape.
// If you do not supply a list of tangent vectors then an approximate tangent vector is computed
// based on the path points you supply using {{path_tangents()}}.
// .
// Figure(3D,Big,VPR=[70,0,345],VPD=20,VPT=[5.5,10.8,-2.7],NoScales): This example shows how the shape, in this case the quadrilateral defined by `[[0, 0], [0, 1], [0.25, 1], [1, 0]]`, appears as the cross section of the swept polyhedron. The blue line shows the path. The normal vector to the shape is shown in black; it is based at the origin and points upwards in the Z direction. The sweep aligns this normal vector with the blue path tangent, which in this case, flips the shape around. Note that for a 2D path like this one, the Y direction in the shape is mapped to the Z direction in the sweep.
// tri= [[0, 0], [0, 1], [.25,1], [1, 0]];
// path = arc(r=5,N=81,angle=[-20,65]); // path = arc(r=5,N=81,angle=[-20,65]);
// % path_sweep(tri,path); // % path_sweep(tri,path);
// T = path_sweep(tri,path,transforms=true); // T = path_sweep(tri,path,transforms=true);
// color("red")for(i=[0:20:80]) stroke(apply(T[i],path3d(tri)),width=.1,closed=true); // color("red")for(i=[0:20:80]) stroke(apply(T[i],path3d(tri)),width=.1,closed=true);
// color("blue")stroke(path3d(arc(r=5,N=101,angle=[-20,80])),width=.1,endcap2="arrow2"); // color("blue")stroke(path3d(arc(r=5,N=101,angle=[-20,80])),width=.1,endcap2="arrow2");
// color("red")stroke([path3d(tri)],width=.1); // color("red")stroke([path3d(tri)],width=.1);
// stroke(move(centroid(tri),[CENTER,UP]), width=.07,endcap2="arrow2",color="black"); // stroke([CENTER,UP], width=.07,endcap2="arrow2",color="black");
// . // .
// In the figure you can see that the swept polyhedron, shown in transparent gray, has the triangle as its cross // In the figure you can see that the swept polyhedron, shown in transparent gray, has the quadrilateral as its cross
// section. The triangle is positioned perpendicular to the path, which is shown in blue, so that the normal // section. The quadrilateral is positioned perpendicular to the path, which is shown in blue, so that the normal
// vector for the triangle is parallel to the tangent vector for the path. The origin for the shape is the point // vector for the quadrilateral is parallel to the tangent vector for the path. The origin for the shape is the point
// which follows the path. For a 2D path, the Y axis of the shape is mapped to the Z axis and in this case, // which follows the path. For a 2D path, the Y axis of the shape is mapped to the Z axis and in this case,
// pointing the triangle's normal vector (in black) along the tangent line of // pointing the quadrilateral's normal vector (in black) along the tangent line of
// the path, which is going in the direction of the blue arrow, requires that the triangle be "turned around". If we // the path, which is going in the direction of the blue arrow, requires that the quadrilateral be "turned around". If we
// reverse the order of points in the path we get a different result: // reverse the order of points in the path we get a different result:
// Figure(3D,Big,VPR=[70,0,20],VPD=20,VPT=[1.25,9.25,-2.65],NoScales): The same sweep operation with the path traveling in the opposite direction. // Figure(3D,Big,VPR=[70,0,20],VPD=20,VPT=[1.25,9.25,-2.65],NoScales): The same sweep operation with the path traveling in the opposite direction. Note that in order to line up the normal correctly, the shape is reversed, so the resulting sweep looks quite different.
// tri= [[0, 0], [0, 1], [1, 0]]; // tri= [[0, 0], [0, 1], [.25,1], [1, 0]];
// path = reverse(arc(r=5,N=81,angle=[-20,65])); // path = reverse(arc(r=5,N=81,angle=[-20,65]));
// % path_sweep(tri,path); // % path_sweep(tri,path);
// T = path_sweep(tri,path,transforms=true); // T = path_sweep(tri,path,transforms=true);
// color("red")for(i=[0:20:80]) stroke(apply(T[i],path3d(tri)),width=.1,closed=true); // color("red")for(i=[0:20:80]) stroke(apply(T[i],path3d(tri)),width=.1,closed=true);
// color("blue")stroke(reverse(path3d(arc(r=5,N=101,angle=[-20-15,65]))),width=.1,endcap2="arrow2"); // color("blue")stroke(reverse(path3d(arc(r=5,N=101,angle=[-20-15,65]))),width=.1,endcap2="arrow2");
// color("red")stroke([path3d(tri)],width=.1);
// stroke([CENTER,UP], width=.07,endcap2="arrow2",color="black");
// Continues: // Continues:
// If your shape is too large for the curves in the path you can create a situation where the shapes cross each // If your shape is too large for the curves in the path you can create a situation where the shapes cross each
// other. This results in an invalid polyhedron, which may appear OK when previewed, but will give rise // other. This results in an invalid polyhedron, which may appear OK when previewed or rendered alone, but will give rise
// to cryptic CGAL errors when rendered with a second object in your model. You may be able to use {{path_sweep2d()}} // to cryptic CGAL errors when rendered with a second object in your model. You may be able to use {{path_sweep2d()}}
// to produce a valid model in cases like this. You can debug models like this using the `profiles=true` option which will show all // to produce a valid model in cases like this. You can debug models like this using the `profiles=true` option which will show all
// the cross sections in your polyhedron. If any of them intersect, the polyhedron will be invalid. // the cross sections in your polyhedron. If any of them intersect, the polyhedron will be invalid.
// Figure(3D,Big,VPR=[47,0,325],VPD=23,VPT=[6.8,4,-3.8],NoScales): We have scaled the path to an ellipse and enlarged the triangle, and it is now sometimes bigger than the local radius of the path, leading to an invalid polyhedron. // Figure(3D,Big,VPR=[47,0,325],VPD=23,VPT=[6.8,4,-3.8],NoScales): We have scaled the path to an ellipse and show a large triangle as the shpae. The triangle is sometimes bigger than the local radius of the path, leading to an invalid polyhedron, which you can identify because the red lines cross in the middle.
// tri= scale([4.5,2.5],[[0, 0], [0, 1], [1, 0]]); // tri= scale([4.5,2.5],[[0, 0], [0, 1], [1, 0]]);
// path = xscale(1.5,arc(r=5,N=81,angle=[-70,70])); // path = xscale(1.5,arc(r=5,N=81,angle=[-70,70]));
// % path_sweep(tri,path); // % path_sweep(tri,path);
@ -781,14 +789,15 @@ module spiral_sweep(poly, h, r, turns=1, higbee, center, r1, r2, d, d1, d2, higb
// this ambiguity by aligning the Y axis in the shape to the Z axis in the swept polyhedron. We can force the // this ambiguity by aligning the Y axis in the shape to the Z axis in the swept polyhedron. We can force the
// shape to twist with the `twist` parameter and get a result like this: // shape to twist with the `twist` parameter and get a result like this:
// Figure(3D,Big,VPR=[66,0,14],VPD=20,VPT=[3.4,4.5,-0.8]): The shape twists as we sweep. Note that it still aligns the origin in the shape with the path, and still aligns the normal vector with the path tangent vector. // Figure(3D,Big,VPR=[66,0,14],VPD=20,VPT=[3.4,4.5,-0.8]): The shape twists as we sweep. Note that it still aligns the origin in the shape with the path, and still aligns the normal vector with the path tangent vector.
// tri= [[0, 0], [0, 1], [1, 0]]; // tri= [[0, 0], [0, 1], [.25,1],[1, 0]];
// path = arc(r=5,N=81,angle=[-20,65]); // path = arc(r=5,N=81,angle=[-20,65]);
// % path_sweep(tri,path,twist=-60); // % path_sweep(tri,path,twist=-60);
// T = path_sweep(tri,path,transforms=true,twist=-60); // T = path_sweep(tri,path,transforms=true,twist=-60);
// color("red")for(i=[0:20:80]) stroke(apply(T[i],path3d(tri)),width=.1,closed=true); // color("red")for(i=[0:20:80]) stroke(apply(T[i],path3d(tri)),width=.1,closed=true);
// color("blue")stroke(path3d(arc(r=5,N=101,angle=[-20,80])),width=.1,endcap2="arrow2"); // color("blue")stroke(path3d(arc(r=5,N=101,angle=[-20,80])),width=.1,endcap2="arrow2");
// Continues: // Continues:
// When the path is full three-dimensional, things can become more complex. You may find that the shape rotates // When the path is full three-dimensional, things can become more complex. It is no longer possible to use a simple
// alignment rule like the one we use in 2D. You may find that the shape rotates
// unexpectedly around its axis as it traverses the path. The `method` parameter allows you to specify how the shapes // unexpectedly around its axis as it traverses the path. The `method` parameter allows you to specify how the shapes
// are aligned, resulting in different twist in the resulting polyhedron. You can choose from three different methods // are aligned, resulting in different twist in the resulting polyhedron. You can choose from three different methods
// for selecting the rotation of your shape. None of these methods will produce good, or even valid, results on all // for selecting the rotation of your shape. None of these methods will produce good, or even valid, results on all
@ -812,7 +821,7 @@ module spiral_sweep(poly, h, r, turns=1, higbee, center, r1, r2, d, d1, d2, higb
// . // .
// The "natural" method works by computing the Frenet frame at each point on the path. This is defined by the tangent to the curve and // The "natural" method works by computing the Frenet frame at each point on the path. This is defined by the tangent to the curve and
// the normal which lies in the plane defined by the curve at each point. This normal points in the direction of curvature of the curve. // the normal which lies in the plane defined by the curve at each point. This normal points in the direction of curvature of the curve.
// The result is a very well behaved set of sections without any unexpected twisting---as long as the curvature never falls to zero. At a // The result is a very well behaved set of shape positions without any unexpected twisting---as long as the curvature never falls to zero. At a
// point of zero curvature (a flat point), the curve does not define a plane and the natural normal is not defined. Furthermore, even if // point of zero curvature (a flat point), the curve does not define a plane and the natural normal is not defined. Furthermore, even if
// you skip over this troublesome point so the normal is defined, it can change direction abruptly when the curvature is zero, leading to // you skip over this troublesome point so the normal is defined, it can change direction abruptly when the curvature is zero, leading to
// a nasty twist and an invalid model. A simple example is a circular arc joined to another arc that curves the other direction. Note // a nasty twist and an invalid model. A simple example is a circular arc joined to another arc that curves the other direction. Note
@ -830,6 +839,9 @@ module spiral_sweep(poly, h, r, turns=1, higbee, center, r1, r2, d, d1, d2, higb
// For any method you can use the `twist` argument to add the specified number of degrees of twist into the model. // For any method you can use the `twist` argument to add the specified number of degrees of twist into the model.
// If the model is closed then the twist must be a multiple of 360/symmetry. The twist is normally spread uniformly along your shape // If the model is closed then the twist must be a multiple of 360/symmetry. The twist is normally spread uniformly along your shape
// based on the path length. If you set `twist_by_length` to false then the twist will be uniform based on the point count of your path. // based on the path length. If you set `twist_by_length` to false then the twist will be uniform based on the point count of your path.
// Twisted shapes will produce twisted faces, so if you want them to look good you should use lots of points on your path and also
// lots of points on the shape. If your shape is a simple polygon, use {{subdivide_path()}} or {{subdivide_long_segments()}} to
// increase the number of points.
// Arguments: // Arguments:
// shape = A 2D polygon path or region describing the shape to be swept. // shape = A 2D polygon path or region describing the shape to be swept.
// path = 2D or 3D path giving the path to sweep over // path = 2D or 3D path giving the path to sweep over

View file

@ -568,7 +568,7 @@ function format_float(f,sig=12) =
/// Function: _format_matrix() /// Function: _format_matrix()
/// Usage: /// Usage:
/// _format_matrix(M, [sig], [eps]) /// _format_matrix(M, [sig], [sep], [eps])
/// Description: /// Description:
/// Convert a numerical matrix into a matrix of strings where every column /// Convert a numerical matrix into a matrix of strings where every column
/// is the same width so it will display in neat columns when printed. /// is the same width so it will display in neat columns when printed.
@ -577,18 +577,22 @@ function format_float(f,sig=12) =
/// Arguments: /// Arguments:
/// M = numerical matrix to convert /// M = numerical matrix to convert
/// sig = significant digits to display. Default: 4 /// sig = significant digits to display. Default: 4
// sep = number of spaces between columns or a text string to separate columns. Default: 1
/// eps = values smaller than this are shown as zero. Default: 1e-9 /// eps = values smaller than this are shown as zero. Default: 1e-9
function _format_matrix(M, sig=4, eps=1e-9) = function _format_matrix(M, sig=4, sep=1, eps=1e-9) =
let( let(
columngap = 1,
figure_dash = chr(8210), figure_dash = chr(8210),
space_punc = chr(8200), space_punc = chr(8200),
space_figure = chr(8199), space_figure = chr(8199),
sep = is_num(sep) && sep>=0 ? str_join(repeat(space_figure,sep))
: is_string(sep) ? sep
: assert(false,"Invalid separator: must be a string or positive integer giving number of spaces"),
strarr= strarr=
[for(row=M) [for(row=M)
[for(entry=row) [for(entry=row)
let( let(
text = is_undef(entry) ? "und" text = is_undef(entry) ? "und"
: !is_num(entry) ? str_join(repeat(figure_dash,2))
: abs(entry) < eps ? "0" // Replace hyphens with figure dashes : abs(entry) < eps ? "0" // Replace hyphens with figure dashes
: str_replace_char(format_float(entry, sig),"-",figure_dash), : str_replace_char(format_float(entry, sig),"-",figure_dash),
have_dot = is_def(str_find(text, ".")) have_dot = is_def(str_find(text, "."))
@ -610,7 +614,7 @@ function _format_matrix(M, sig=4, eps=1e-9) =
let( let(
extra = ends_with(row[i],"inf") ? 1 : 0 extra = ends_with(row[i],"inf") ? 1 : 0
) )
str_pad(row[i],maxlen[i]+extra+(i==0?0:columngap),space_figure,left=true)])] str_pad(row[i],maxlen[i]+extra,space_figure,left=true)],sep=sep)]
) )
padded; padded;