mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-01 09:49:45 +00:00
Added texture support to linear_sweep()
This commit is contained in:
parent
3fe1ae77ee
commit
709c7a152f
2 changed files with 318 additions and 274 deletions
62
skin.scad
62
skin.scad
|
@ -521,6 +521,13 @@ function skin(profiles, slices, refine=1, method="direct", sampling, caps, close
|
||||||
// shift = The amount to shift the top of the shape, in the X and Y directions, relative to the position of the bottom. Default: [0,0]
|
// shift = The amount to shift the top of the shape, in the X and Y directions, relative to the position of the bottom. Default: [0,0]
|
||||||
// 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`
|
||||||
// maxseg = If given, then any long segments of the region will be subdivided to be shorter than this length. This can refine twisting flat faces a lot. Default: `undef` (no subsampling)
|
// maxseg = If given, then any long segments of the region will be subdivided to be shorter than this length. This can refine twisting flat faces a lot. Default: `undef` (no subsampling)
|
||||||
|
// texture = A texture name string, or a rectangular array of scalar height values (0.0 to 1.0), or a VNF tile that defines the texture to apply to vertical surfaces. See {{texture()}} for what named textures are supported.
|
||||||
|
// tex_size = An optional 2D target size for the textures. Actual texture sizes will be scaled somewhat to evenly fit the available surface. Default: `[5,5]`
|
||||||
|
// tex_counts = If given instead of tex_size, gives the tile repetition counts for textures over the surface length and height.
|
||||||
|
// tex_inset = If numeric, lowers the texture into the surface by that amount, before the tscale multiplier is applied. If `true`, insets by exactly `1`. Default: `false`
|
||||||
|
// tex_rot = If true, rotates the texture 90º.
|
||||||
|
// tex_scale = Scaling multiplier for the texture depth.
|
||||||
|
// tex_samples = Minimum number of "bend points" to have in VNF texture tiles. Default: 8
|
||||||
// style = The style to use when triangulating the surface of the object. Valid values are `"default"`, `"alt"`, or `"quincunx"`.
|
// style = The style to use when triangulating the surface of the object. Valid values are `"default"`, `"alt"`, or `"quincunx"`.
|
||||||
// convexity = Max number of surfaces any single ray could pass through. Module use only.
|
// convexity = Max number of surfaces any single ray could pass through. Module use only.
|
||||||
// cp = Centerpoint for determining intersection anchors or centering the shape. Determines the base of the anchor vector. Can be "centroid", "mean", "box" or a 3D point. Default: `"centroid"`
|
// cp = Centerpoint for determining intersection anchors or centering the shape. Determines the base of the anchor vector. Can be "centroid", "mean", "box" or a 3D point. Default: `"centroid"`
|
||||||
|
@ -569,10 +576,14 @@ function skin(profiles, slices, refine=1, method="direct", sampling, caps, close
|
||||||
// orgn = difference(mrgn,rgn3);
|
// orgn = difference(mrgn,rgn3);
|
||||||
// linear_sweep(orgn,height=20,convexity=16)
|
// linear_sweep(orgn,height=20,convexity=16)
|
||||||
// show_anchors();
|
// show_anchors();
|
||||||
|
|
||||||
module linear_sweep(
|
module linear_sweep(
|
||||||
region, height, center,
|
region, height, center,
|
||||||
twist=0, scale=1, shift=[0,0],
|
twist=0, scale=1, shift=[0,0],
|
||||||
slices, maxseg, style="default", convexity,
|
slices, maxseg, style="default", convexity,
|
||||||
|
texture, tex_size=[5,5], tex_counts,
|
||||||
|
tex_inset=false, tex_rot=false,
|
||||||
|
tex_scale=1, tex_samples,
|
||||||
cp, atype="hull", h,
|
cp, atype="hull", h,
|
||||||
anchor, spin=0, orient=UP
|
anchor, spin=0, orient=UP
|
||||||
) {
|
) {
|
||||||
|
@ -585,7 +596,15 @@ module linear_sweep(
|
||||||
vnf = linear_sweep(
|
vnf = linear_sweep(
|
||||||
region, height=h, style=style,
|
region, height=h, style=style,
|
||||||
twist=twist, scale=scale, shift=shift,
|
twist=twist, scale=scale, shift=shift,
|
||||||
slices=slices, maxseg=maxseg,
|
texture=texture,
|
||||||
|
tex_size=tex_size,
|
||||||
|
tex_counts=tex_counts,
|
||||||
|
tex_inset=tex_inset,
|
||||||
|
tex_rot=tex_rot,
|
||||||
|
tex_scale=tex_scale,
|
||||||
|
tex_samples=tex_samples,
|
||||||
|
slices=slices,
|
||||||
|
maxseg=maxseg,
|
||||||
anchor="origin"
|
anchor="origin"
|
||||||
);
|
);
|
||||||
anchors = [
|
anchors = [
|
||||||
|
@ -607,6 +626,9 @@ function linear_sweep(
|
||||||
twist=0, scale=1, shift=[0,0],
|
twist=0, scale=1, shift=[0,0],
|
||||||
slices, maxseg, style="default",
|
slices, maxseg, style="default",
|
||||||
cp, atype="hull", h,
|
cp, atype="hull", h,
|
||||||
|
texture, tex_size=[5,5], tex_counts,
|
||||||
|
tex_inset=false, tex_rot=false,
|
||||||
|
tex_scale=1, tex_samples,
|
||||||
anchor, spin=0, orient=UP
|
anchor, spin=0, orient=UP
|
||||||
) =
|
) =
|
||||||
let( region = force_region(region) )
|
let( region = force_region(region) )
|
||||||
|
@ -614,7 +636,18 @@ function linear_sweep(
|
||||||
assert(is_num(scale) || is_vector(scale))
|
assert(is_num(scale) || is_vector(scale))
|
||||||
assert(is_vector(shift, 2), str(shift))
|
assert(is_vector(shift, 2), str(shift))
|
||||||
let(
|
let(
|
||||||
h = first_defined([h, height, 1]),
|
h = first_defined([h, height, 1])
|
||||||
|
)
|
||||||
|
!is_undef(texture)? textured_linear_sweep(
|
||||||
|
region, h=h,
|
||||||
|
texture=texture, tex_size=tex_size,
|
||||||
|
counts=tex_counts, inset=tex_inset,
|
||||||
|
rot=tex_rot, tscale=tex_scale,
|
||||||
|
twist=twist, scale=scale, shift=shift,
|
||||||
|
style=style, samples=tex_samples,
|
||||||
|
anchor=anchor, spin=spin, orient=orient
|
||||||
|
) :
|
||||||
|
let(
|
||||||
anchor = center==true? "origin" :
|
anchor = center==true? "origin" :
|
||||||
center == false? "original_base" :
|
center == false? "original_base" :
|
||||||
default(anchor, "original_base"),
|
default(anchor, "original_base"),
|
||||||
|
@ -2760,11 +2793,11 @@ function texture(tex, n, inset, gap, roughness) =
|
||||||
// vnf_polyhedron(vnf, convexity=10);
|
// vnf_polyhedron(vnf, convexity=10);
|
||||||
|
|
||||||
function textured_linear_sweep(
|
function textured_linear_sweep(
|
||||||
region, texture,
|
region, texture, tex_size=[5,5],
|
||||||
tex_size=[5,5], h, counts,
|
h, counts, inset=false, rot=false,
|
||||||
inset=false, rot=false, tscale=1,
|
tscale=1, twist, scale, shift,
|
||||||
twist, scale, shift,
|
style="min_edge", l,
|
||||||
style="min_edge", l, samples,
|
height, length, samples,
|
||||||
anchor=CENTER, spin=0, orient=UP
|
anchor=CENTER, spin=0, orient=UP
|
||||||
) =
|
) =
|
||||||
assert(is_path(region,[2]) || is_region(region))
|
assert(is_path(region,[2]) || is_region(region))
|
||||||
|
@ -2780,7 +2813,7 @@ function textured_linear_sweep(
|
||||||
rot==180? reverse([for (row=tex) reverse(row)]) :
|
rot==180? reverse([for (row=tex) reverse(row)]) :
|
||||||
rot==270? [for (row=transpose(tex)) reverse(row)] :
|
rot==270? [for (row=transpose(tex)) reverse(row)] :
|
||||||
reverse(transpose(tex)),
|
reverse(transpose(tex)),
|
||||||
h = first_defined([h, l, 1]),
|
h = first_defined([h, l, height, length, 1]),
|
||||||
inset = is_num(inset)? inset : inset? 1 : 0,
|
inset = is_num(inset)? inset : inset? 1 : 0,
|
||||||
twist = default(twist, 0),
|
twist = default(twist, 0),
|
||||||
shift = default(shift, [0,0]),
|
shift = default(shift, [0,0]),
|
||||||
|
@ -2824,7 +2857,8 @@ function textured_linear_sweep(
|
||||||
) [for (i = [0:1:rlen]) [i/rlen, row[i%rlen]]],
|
) [for (i = [0:1:rlen]) [i/rlen, row[i%rlen]]],
|
||||||
tmat = scale(scale) * zrot(twist) * up(h/2),
|
tmat = scale(scale) * zrot(twist) * up(h/2),
|
||||||
pre_skew_vnf = vnf_join([
|
pre_skew_vnf = vnf_join([
|
||||||
for (rgn = regions) let(
|
/*for (rgn = regions)*/ let(
|
||||||
|
rgn = last(regions),
|
||||||
walls_vnf = vnf_join([
|
walls_vnf = vnf_join([
|
||||||
for (path = rgn) let(
|
for (path = rgn) let(
|
||||||
path = reverse(path),
|
path = reverse(path),
|
||||||
|
@ -2922,8 +2956,8 @@ function textured_linear_sweep(
|
||||||
obases = resample_path(path, n=counts.x * samples, closed=true),
|
obases = resample_path(path, n=counts.x * samples, closed=true),
|
||||||
onorms = path_normals(obases, closed=true),
|
onorms = path_normals(obases, closed=true),
|
||||||
bases = close_path(obases),
|
bases = close_path(obases),
|
||||||
norms = close_path(onorms)
|
norms = close_path(onorms),
|
||||||
) [
|
nupath = [
|
||||||
for (j = [0:1:counts.x-1], vert = tpath) let(
|
for (j = [0:1:counts.x-1], vert = tpath) let(
|
||||||
part = (j + vert.x) * samples,
|
part = (j + vert.x) * samples,
|
||||||
u = floor(part),
|
u = floor(part),
|
||||||
|
@ -2934,6 +2968,7 @@ function textured_linear_sweep(
|
||||||
xy = base + norm * texh
|
xy = base + norm * texh
|
||||||
) xy
|
) xy
|
||||||
]
|
]
|
||||||
|
) nupath
|
||||||
],
|
],
|
||||||
bot_vnf = vnf_from_region(brgn, down(h/2), reverse=true),
|
bot_vnf = vnf_from_region(brgn, down(h/2), reverse=true),
|
||||||
top_vnf = vnf_from_region(brgn, tmat, reverse=false)
|
top_vnf = vnf_from_region(brgn, tmat, reverse=false)
|
||||||
|
@ -2954,11 +2989,12 @@ module textured_linear_sweep(
|
||||||
path, texture, tex_size=[5,5], h,
|
path, texture, tex_size=[5,5], h,
|
||||||
inset=false, rot=false, tscale=1,
|
inset=false, rot=false, tscale=1,
|
||||||
twist, scale, shift, samples,
|
twist, scale, shift, samples,
|
||||||
style="min_edge", l, counts,
|
style="min_edge", l,
|
||||||
|
height, length, counts,
|
||||||
anchor=CENTER, spin=0, orient=UP,
|
anchor=CENTER, spin=0, orient=UP,
|
||||||
convexity=10
|
convexity=10
|
||||||
) {
|
) {
|
||||||
h = first_defined([h, l]);
|
h = first_defined([h, l, height, length, 1]);
|
||||||
vnf = textured_linear_sweep(
|
vnf = textured_linear_sweep(
|
||||||
path, texture, h=h,
|
path, texture, h=h,
|
||||||
tex_size=tex_size, counts=counts,
|
tex_size=tex_size, counts=counts,
|
||||||
|
|
24
vnf.scad
24
vnf.scad
|
@ -178,23 +178,28 @@ function vnf_vertex_array(
|
||||||
let(
|
let(
|
||||||
d42=norm(pts[i4]-pts[i2]),
|
d42=norm(pts[i4]-pts[i2]),
|
||||||
d13=norm(pts[i1]-pts[i3]),
|
d13=norm(pts[i1]-pts[i3]),
|
||||||
shortedge = d42<d13+EPSILON ? [[i1,i4,i2],[i2,i4,i3]]
|
shortedge = d42<d13+EPSILON
|
||||||
|
? [[i1,i4,i2],[i2,i4,i3]]
|
||||||
: [[i1,i3,i2],[i1,i4,i3]]
|
: [[i1,i3,i2],[i1,i4,i3]]
|
||||||
)
|
)
|
||||||
shortedge
|
shortedge
|
||||||
: style=="convex"?
|
: style=="convex"?
|
||||||
let( // Find normal for 3 of the points. Is the other point above or below?
|
let( // Find normal for 3 of the points. Is the other point above or below?
|
||||||
n = (reverse?-1:1)*cross(pts[i2]-pts[i1],pts[i3]-pts[i1]),
|
n = (reverse?-1:1)*cross(pts[i2]-pts[i1],pts[i3]-pts[i1]),
|
||||||
convexfaces = n==0 ? [[i1,i4,i3]]
|
convexfaces = n==0
|
||||||
: n*pts[i4] > n*pts[i1] ? [[i1,i4,i2],[i2,i4,i3]]
|
? [[i1,i4,i3]]
|
||||||
|
: n*pts[i4] > n*pts[i1]
|
||||||
|
? [[i1,i4,i2],[i2,i4,i3]]
|
||||||
: [[i1,i3,i2],[i1,i4,i3]]
|
: [[i1,i3,i2],[i1,i4,i3]]
|
||||||
)
|
)
|
||||||
convexfaces
|
convexfaces
|
||||||
: style=="concave"?
|
: style=="concave"?
|
||||||
let( // Find normal for 3 of the points. Is the other point above or below?
|
let( // Find normal for 3 of the points. Is the other point above or below?
|
||||||
n = (reverse?-1:1)*cross(pts[i2]-pts[i1],pts[i3]-pts[i1]),
|
n = (reverse?-1:1)*cross(pts[i2]-pts[i1],pts[i3]-pts[i1]),
|
||||||
concavefaces = n==0 ? [[i1,i4,i3]]
|
concavefaces = n==0
|
||||||
: n*pts[i4] <= n*pts[i1] ? [[i1,i4,i2],[i2,i4,i3]]
|
? [[i1,i4,i3]]
|
||||||
|
: n*pts[i4] <= n*pts[i1]
|
||||||
|
? [[i1,i4,i2],[i2,i4,i3]]
|
||||||
: [[i1,i3,i2],[i1,i4,i3]]
|
: [[i1,i3,i2],[i1,i4,i3]]
|
||||||
)
|
)
|
||||||
concavefaces
|
concavefaces
|
||||||
|
@ -573,10 +578,14 @@ function _bridge(pt, outer,eps) =
|
||||||
// vnf_wireframe(vnf,width=.25);
|
// vnf_wireframe(vnf,width=.25);
|
||||||
function vnf_from_region(region, transform, reverse=false) =
|
function vnf_from_region(region, transform, reverse=false) =
|
||||||
let (
|
let (
|
||||||
|
region = [for (path = region) deduplicate(path, closed=true)],
|
||||||
regions = region_parts(force_region(region)),
|
regions = region_parts(force_region(region)),
|
||||||
vnfs =
|
vnfs =
|
||||||
[ for (rgn = regions)
|
[
|
||||||
let( cleaved = path3d(_cleave_connected_region(rgn)) )
|
for (rgn = regions)
|
||||||
|
let(
|
||||||
|
cleaved = path3d(_cleave_connected_region(rgn))
|
||||||
|
)
|
||||||
assert( cleaved, "The region is invalid")
|
assert( cleaved, "The region is invalid")
|
||||||
let(
|
let(
|
||||||
face = is_undef(transform)? cleaved : apply(transform,cleaved),
|
face = is_undef(transform)? cleaved : apply(transform,cleaved),
|
||||||
|
@ -1226,7 +1235,6 @@ function vnf_bend(vnf,r,d,axis="Z") =
|
||||||
axis=="X"? [p.x, p.z*sin(a), p.z*cos(a)] :
|
axis=="X"? [p.x, p.z*sin(a), p.z*cos(a)] :
|
||||||
axis=="Y"? [p.z*sin(a), p.y, p.z*cos(a)] :
|
axis=="Y"? [p.z*sin(a), p.y, p.z*cos(a)] :
|
||||||
[p.y*sin(a), p.y*cos(a), p.z]]
|
[p.y*sin(a), p.y*cos(a), p.z]]
|
||||||
|
|
||||||
) [new_vert,sliced[1]];
|
) [new_vert,sliced[1]];
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue