mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-01 09:49:45 +00:00
path_text fixes in progress, doc fix for plane3pt
This commit is contained in:
parent
a012a7cc62
commit
537a888a28
2 changed files with 57 additions and 56 deletions
|
@ -442,6 +442,7 @@ function is_coplanar(points, eps=EPSILON) =
|
||||||
// Function: plane3pt()
|
// Function: plane3pt()
|
||||||
// Usage:
|
// Usage:
|
||||||
// plane = plane3pt(p1, p2, p3);
|
// plane = plane3pt(p1, p2, p3);
|
||||||
|
// plane = plane3pt([p1, p2, p3]);
|
||||||
// Topics: Geometry, Planes
|
// Topics: Geometry, Planes
|
||||||
// Description:
|
// Description:
|
||||||
// Generates the normalized cartesian equation of a plane from three 3d points.
|
// Generates the normalized cartesian equation of a plane from three 3d points.
|
||||||
|
|
108
shapes3d.scad
108
shapes3d.scad
|
@ -27,8 +27,8 @@ use <builtins.scad>
|
||||||
// vnf = cube(size, ...);
|
// vnf = cube(size, ...);
|
||||||
// See Also: cuboid(), prismoid()
|
// See Also: cuboid(), prismoid()
|
||||||
// Description:
|
// Description:
|
||||||
// Creates a 3D cubic object with support for anchoring and attachments.
|
// Creates a 3D cubic object.
|
||||||
// This can be used as a drop-in replacement for the built-in `cube()` module.
|
// This module extends the built-in cube()` module by providing support for attachments and a function form.
|
||||||
// When called as a function, returns a [VNF](vnf.scad) for a cube.
|
// When called as a function, returns a [VNF](vnf.scad) for a cube.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// size = The size of the cube.
|
// size = The size of the cube.
|
||||||
|
@ -1063,7 +1063,7 @@ function wedge(size=[1,1,1], center, anchor, spin=0, orient=UP) =
|
||||||
|
|
||||||
// Function&Module: cylinder()
|
// Function&Module: cylinder()
|
||||||
// Topics: Shapes (3D), Attachable, VNF Generators
|
// Topics: Shapes (3D), Attachable, VNF Generators
|
||||||
// Usage: As Module (as in Native OpenSCAD)
|
// Usage: As Module (as in native OpenSCAD)
|
||||||
// cylinder(h, r=/d=, [center=]);
|
// cylinder(h, r=/d=, [center=]);
|
||||||
// cylinder(h, r1/d1=, r2/d2=, [center=]);
|
// cylinder(h, r1/d1=, r2/d2=, [center=]);
|
||||||
// Usage: With BOSL2 anchoring and attachment extensions
|
// Usage: With BOSL2 anchoring and attachment extensions
|
||||||
|
@ -1074,14 +1074,14 @@ function wedge(size=[1,1,1], center, anchor, spin=0, orient=UP) =
|
||||||
// vnf = cylinder(h, r1/d1=, r2/d2=, ...);
|
// vnf = cylinder(h, r1/d1=, r2/d2=, ...);
|
||||||
// See Also: cyl()
|
// See Also: cyl()
|
||||||
// Description:
|
// Description:
|
||||||
// Creates a 3D cylinder or conic object with support for anchoring and attachments.
|
// Creates a 3D cylinder or conic object.
|
||||||
// This modules extends the built-in `cylinder()` module by adding support for attachment.
|
// This modules extends the built-in `cylinder()` module by adding support for attachment and by adding a function version.
|
||||||
// When called as a function, returns a [VNF](vnf.scad) for a cylinder.
|
// When called as a function, returns a [VNF](vnf.scad) for a cylinder.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// l / h = The height of the cylinder.
|
// h = The height of the cylinder.
|
||||||
// r1 = The bottom radius of the cylinder. (Before orientation.)
|
// r1 = The bottom radius of the cylinder. (Before orientation.)
|
||||||
// r2 = The top radius of the cylinder. (Before orientation.)
|
// r2 = The top radius of the cylinder. (Before orientation.)
|
||||||
// center = If given, overrides `anchor`. A true value sets `anchor=CENTER`, false sets `anchor=BOTTOM`.
|
// center = If given, overrides `anchor`. A true value sets `anchor=CENTER`, false sets `anchor=BOTTOM`. Default: false
|
||||||
// ---
|
// ---
|
||||||
// d1 = The bottom diameter of the cylinder. (Before orientation.)
|
// d1 = The bottom diameter of the cylinder. (Before orientation.)
|
||||||
// d2 = The top diameter of the cylinder. (Before orientation.)
|
// d2 = The top diameter of the cylinder. (Before orientation.)
|
||||||
|
@ -1112,24 +1112,24 @@ function wedge(size=[1,1,1], center, anchor, spin=0, orient=UP) =
|
||||||
// cylinder(h=30, d1=25, d2=10) show_anchors();
|
// cylinder(h=30, d1=25, d2=10) show_anchors();
|
||||||
// }
|
// }
|
||||||
|
|
||||||
module cylinder(h, r1, r2, center, l, r, d, d1, d2, anchor, spin=0, orient=UP)
|
module cylinder(h, r1, r2, center, r, d, d1, d2, anchor, spin=0, orient=UP)
|
||||||
{
|
{
|
||||||
anchor = get_anchor(anchor, center, BOTTOM, BOTTOM);
|
anchor = get_anchor(anchor, center, BOTTOM, BOTTOM);
|
||||||
r1 = get_radius(r1=r1, r=r, d1=d1, d=d, dflt=1);
|
r1 = get_radius(r1=r1, r=r, d1=d1, d=d, dflt=1);
|
||||||
r2 = get_radius(r1=r2, r=r, d1=d2, d=d, dflt=1);
|
r2 = get_radius(r1=r2, r=r, d1=d2, d=d, dflt=1);
|
||||||
l = first_defined([h, l, 1]);
|
h = default(h,1);
|
||||||
attachable(anchor,spin,orient, r1=r1, r2=r2, l=l) {
|
attachable(anchor,spin,orient, r1=r1, r2=r2, l=h) {
|
||||||
_cylinder(h=l, r1=r1, r2=r2, center=true);
|
_cylinder(h=h, r1=r1, r2=r2, center=true);
|
||||||
children();
|
children();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function cylinder(h, r1, r2, center, l, r, d, d1, d2, anchor, spin=0, orient=UP) =
|
function cylinder(h, r1, r2, center, r, d, d1, d2, anchor, spin=0, orient=UP) =
|
||||||
let(
|
let(
|
||||||
anchor = get_anchor(anchor, center, BOTTOM, BOTTOM),
|
anchor = get_anchor(anchor, center, BOTTOM, BOTTOM),
|
||||||
r1 = get_radius(r1=r1, r=r, d1=d1, d=d, dflt=1),
|
r1 = get_radius(r1=r1, r=r, d1=d1, d=d, dflt=1),
|
||||||
r2 = get_radius(r1=r2, r=r, d1=d2, d=d, dflt=1),
|
r2 = get_radius(r1=r2, r=r, d1=d2, d=d, dflt=1),
|
||||||
l = first_defined([h, l, 1]),
|
l = default(h,1),
|
||||||
sides = segs(max(r1,r2)),
|
sides = segs(max(r1,r2)),
|
||||||
verts = [
|
verts = [
|
||||||
for (i=[0:1:sides-1]) let(a=360*(1-i/sides)) [r1*cos(a),r1*sin(a),-l/2],
|
for (i=[0:1:sides-1]) let(a=360*(1-i/sides)) [r1*cos(a),r1*sin(a),-l/2],
|
||||||
|
@ -1977,21 +1977,21 @@ function pie_slice(
|
||||||
|
|
||||||
// Function&Module: sphere()
|
// Function&Module: sphere()
|
||||||
// Topics: Shapes (3D), Attachable, VNF Generators
|
// Topics: Shapes (3D), Attachable, VNF Generators
|
||||||
// Usage: As Module
|
// Usage: As Module (native OpenSCAD)
|
||||||
// sphere(r|d=, [circum=], [style=], ...) [ATTACHMENTS];
|
// sphere(r|d=);
|
||||||
// Usage: As Function
|
// Usage: Using BOSL2 attachments extensions
|
||||||
// vnf = sphere(r|d=, [circum=], [style=], ...);
|
// sphere(r|d=, [anchor=], [spin=], [orient=]) [ATTACHMENTS];
|
||||||
|
// Usage: As Function (BOSL2 extension)
|
||||||
|
// vnf = sphere(r|d=, [anchor=], [spin=], [orient=]) [ATTACHMENTS];
|
||||||
// See Also: spheroid()
|
// See Also: spheroid()
|
||||||
// Description:
|
// Description:
|
||||||
// Creates a sphere object, with support for anchoring and attachments.
|
// Creates a sphere object.
|
||||||
// This is a drop-in replacement for the built-in `sphere()` module.
|
// This module extends the built-in `sphere()` module by providing support for BOSL2 anchoring and attachments, and a function form.
|
||||||
// When called as a function, returns a [VNF](vnf.scad) for a sphere.
|
// When called as a function, returns a [VNF](vnf.scad) for a sphere.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// r = Radius of the sphere.
|
// r = Radius of the sphere.
|
||||||
// ---
|
// ---
|
||||||
// d = Diameter of the sphere.
|
// d = Diameter of the sphere.
|
||||||
// circum = If true, the sphere is made large enough to circumscribe the sphere of the ideal side. Otherwise inscribes. Default: false (inscribes)
|
|
||||||
// style = The style of the sphere's construction. One of "orig", "aligned", "stagger", "octa", or "icosa". Default: "orig"
|
|
||||||
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
|
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
|
||||||
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
|
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
|
||||||
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
|
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
|
||||||
|
@ -1999,16 +1999,6 @@ function pie_slice(
|
||||||
// sphere(r=50);
|
// sphere(r=50);
|
||||||
// Example: By Diameter
|
// Example: By Diameter
|
||||||
// sphere(d=100);
|
// sphere(d=100);
|
||||||
// Example: style="orig"
|
|
||||||
// sphere(d=100, style="orig", $fn=10);
|
|
||||||
// Example: style="aligned"
|
|
||||||
// sphere(d=100, style="aligned", $fn=10);
|
|
||||||
// Example: style="stagger"
|
|
||||||
// sphere(d=100, style="stagger", $fn=10);
|
|
||||||
// Example: style="icosa"
|
|
||||||
// sphere(d=100, style="icosa", $fn=10);
|
|
||||||
// // In "icosa" style, $fn is quantized
|
|
||||||
// // to the nearest multiple of 5.
|
|
||||||
// Example: Anchoring
|
// Example: Anchoring
|
||||||
// sphere(d=100, anchor=FRONT);
|
// sphere(d=100, anchor=FRONT);
|
||||||
// Example: Spin
|
// Example: Spin
|
||||||
|
@ -2021,23 +2011,16 @@ function pie_slice(
|
||||||
// vnf = sphere(d=100, style="icosa");
|
// vnf = sphere(d=100, style="icosa");
|
||||||
// vnf_polyhedron(vnf);
|
// vnf_polyhedron(vnf);
|
||||||
|
|
||||||
module sphere(r, d, circum=false, style="orig", anchor=CENTER, spin=0, orient=UP) {
|
module sphere(r, d, anchor=CENTER, spin=0, orient=UP) {
|
||||||
r = get_radius(r=r, d=d, dflt=1);
|
r = get_radius(r=r, d=d, dflt=1);
|
||||||
if (!circum && style=="orig" && is_num(r)) {
|
|
||||||
attachable(anchor,spin,orient, r=r) {
|
attachable(anchor,spin,orient, r=r) {
|
||||||
_sphere(r=r);
|
_sphere(r=r);
|
||||||
children();
|
children();
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
spheroid(
|
|
||||||
r=r, circum=circum, style=style,
|
|
||||||
anchor=anchor, spin=spin, orient=orient
|
|
||||||
) children();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function sphere(r, d, circum=false, style="orig", anchor=CENTER, spin=0, orient=UP) =
|
function sphere(r, d, anchor=CENTER, spin=0, orient=UP) =
|
||||||
spheroid(r=r, d=d, circum=circum, style=style, anchor=anchor, spin=spin, orient=orient);
|
spheroid(r=r, d=d, anchor=anchor, spin=spin, orient=orient);
|
||||||
|
|
||||||
|
|
||||||
// Function&Module: spheroid()
|
// Function&Module: spheroid()
|
||||||
|
@ -2920,7 +2903,8 @@ function _cut_interp(pathcut, path, data) =
|
||||||
// top = direction or list of directions pointing toward the top of the text
|
// top = direction or list of directions pointing toward the top of the text
|
||||||
// reverse = reverse the letters if true. Not allowed for 2D path. Default: false
|
// reverse = reverse the letters if true. Not allowed for 2D path. Default: false
|
||||||
// textmetrics = if set to true and lettersize is not given then use the experimental textmetrics feature. You must be running a dev snapshot that includes this feature and have the feature turned on in your preferences. Default: false
|
// textmetrics = if set to true and lettersize is not given then use the experimental textmetrics feature. You must be running a dev snapshot that includes this feature and have the feature turned on in your preferences. Default: false
|
||||||
// kern = scalar or array giving size adjustments for each letter. Default: 0
|
// valign = align text to the path using "top", "bottom", "center" or "baseline". This only works with textmetrics enabled. Default: "baseline"
|
||||||
|
// kern = scalar or array giving spacing adjusments between each letter. If it's an array it should have one less entry than the text string. Default: 0
|
||||||
// Example(3D,NoScales): The examples use Courier, a monospaced font. The width is 1/1.2 times the specified size for this font. This text could wrap around a cylinder.
|
// Example(3D,NoScales): The examples use Courier, a monospaced font. The width is 1/1.2 times the specified size for this font. This text could wrap around a cylinder.
|
||||||
// path = path3d(arc(100, r=25, angle=[245, 370]));
|
// path = path3d(arc(100, r=25, angle=[245, 370]));
|
||||||
// color("red")stroke(path, width=.3);
|
// color("red")stroke(path, width=.3);
|
||||||
|
@ -2983,7 +2967,7 @@ function _cut_interp(pathcut, path, data) =
|
||||||
// kern = [1,1.2,1,1,.3,-.2,1,0,.8,1,1.1,1];
|
// kern = [1,1.2,1,1,.3,-.2,1,0,.8,1,1.1,1];
|
||||||
// path_text(path, "Example text", font="Courier", size=5, lettersize = 5/1.2, kern=kern, normal=UP);
|
// path_text(path, "Example text", font="Courier", size=5, lettersize = 5/1.2, kern=kern, normal=UP);
|
||||||
|
|
||||||
module path_text(path, text, font, size, thickness, lettersize, offset=0, reverse=false, normal, top, center=false, textmetrics=false, kern=0, height,h)
|
module path_text(path, text, font, size, thickness, lettersize, offset=0, reverse=false, normal, top, center=false, textmetrics=false, kern=0, height,h,spacing=1, valign="baseline")
|
||||||
{
|
{
|
||||||
no_children($children);
|
no_children($children);
|
||||||
dummy2=assert(is_path(path,[2,3]),"Must supply a 2d or 3d path")
|
dummy2=assert(is_path(path,[2,3]),"Must supply a 2d or 3d path")
|
||||||
|
@ -3008,27 +2992,38 @@ module path_text(path, text, font, size, thickness, lettersize, offset=0, revers
|
||||||
: is_def(top) ? top
|
: is_def(top) ? top
|
||||||
: undef;
|
: undef;
|
||||||
|
|
||||||
kern = force_list(kern, len(text));
|
kern = force_list(kern, len(text)-1);
|
||||||
dummy3 = assert(is_list(kern) && len(kern)==len(text), "kern must be a scalar or list whose length is len(text)");
|
dummy3 = assert(is_list(kern) && len(kern)==len(text)-1, "kern must be a scalar or list whose length is len(text)-1");
|
||||||
|
|
||||||
lsize = kern + (
|
lsize = is_def(lettersize) ? force_list(lettersize, len(text))
|
||||||
is_def(lettersize) ? force_list(lettersize, len(text))
|
|
||||||
: textmetrics ? [for(letter=text) let(t=textmetrics(letter, font=font, size=size)) t.advance[0]]
|
: textmetrics ? [for(letter=text) let(t=textmetrics(letter, font=font, size=size)) t.advance[0]]
|
||||||
: assert(false, "textmetrics disabled: Must specify letter size")
|
: assert(false, "textmetrics disabled: Must specify letter size");
|
||||||
);
|
lcenter = spacing * convolve(lsize,[1,1]/2)+[0,each kern,0] ;
|
||||||
textlength = sum(lsize);
|
textlength = spacing*sum(lsize)+sum(kern);
|
||||||
|
|
||||||
|
ascent = !textmetrics ? undef
|
||||||
|
: textmetrics(text, font=font, size=size).ascent;
|
||||||
|
descent = !textmetrics ? undef
|
||||||
|
: textmetrics(text, font=font, size=size).descent;
|
||||||
|
|
||||||
|
vadjustment = !textmetrics ? assert(valign=="baseline","valign requires textmetrics support") 0
|
||||||
|
: valign=="baseline" ? 0
|
||||||
|
: valign=="top" ? -ascent
|
||||||
|
: valign=="bottom" ? descent
|
||||||
|
: valign=="center" ? (descent-ascent)/2
|
||||||
|
: assert(false,"Invalid valign value");
|
||||||
|
|
||||||
dummy1 = assert(textlength<=path_length(path),"Path is too short for the text");
|
dummy1 = assert(textlength<=path_length(path),"Path is too short for the text");
|
||||||
|
|
||||||
start = center ? (path_length(path) - textlength)/2 : 0;
|
start = center ? (path_length(path) - textlength)/2 : 0;
|
||||||
|
|
||||||
pts = path_cut_points(path, add_scalar([0, each cumsum(lsize)],start+lsize[0]/2), direction=true);
|
pts = path_cut_points(path, add_scalar([ each cumsum(lcenter)],start+0*lsize[0]/2), direction=true);
|
||||||
|
|
||||||
|
|
||||||
usernorm = is_def(normal);
|
usernorm = is_def(normal);
|
||||||
usetop = is_def(top);
|
usetop = is_def(top);
|
||||||
|
|
||||||
normpts = is_undef(normal) ? (reverse?1:-1)*column(pts,3) : _cut_interp(pts,path, normal);
|
normpts = is_undef(normal) ? (reverse?1:-1)*column(pts,3) : _cut_interp(pts,path, normal);
|
||||||
toppts = is_undef(top) ? undef : _cut_interp(pts,path,top);
|
toppts = is_undef(top) ? undef : _cut_interp(pts,path,top);
|
||||||
|
|
||||||
for (i = idx(text)) {
|
for (i = idx(text)) {
|
||||||
tangent = pts[i][2];
|
tangent = pts[i][2];
|
||||||
checks =
|
checks =
|
||||||
|
@ -3047,9 +3042,14 @@ module path_text(path, text, font, size, thickness, lettersize, offset=0, revers
|
||||||
y=usetop ? toppts[i] : undef
|
y=usetop ? toppts[i] : undef
|
||||||
) up(offset-thickness/2) {
|
) up(offset-thickness/2) {
|
||||||
linear_extrude(height=thickness)
|
linear_extrude(height=thickness)
|
||||||
left(lsize[0]/2)
|
back(vadjustment)
|
||||||
|
{
|
||||||
|
//stroke([[0,0],[0,14]], width=.3, endcap2="arrow");
|
||||||
|
|
||||||
|
left(lsize[i]/2)
|
||||||
text(text[i], font=font, size=size);
|
text(text[i], font=font, size=size);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
frame_map(
|
frame_map(
|
||||||
x=point3d(tangent-adjustment),
|
x=point3d(tangent-adjustment),
|
||||||
|
|
Loading…
Reference in a new issue