mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-19 19:09:36 +00:00
Merge pull request #383 from adrianVmariano/master
Fixed bug with collinear points in round_corners. Allow use of "joint" option with circle method in round_corners.
This commit is contained in:
commit
08c2923550
1 changed files with 32 additions and 20 deletions
|
@ -24,25 +24,29 @@ include <structs.scad>
|
||||||
// tactile "bump" where the curvature changes from flat to circular.
|
// tactile "bump" where the curvature changes from flat to circular.
|
||||||
// See https://hackernoon.com/apples-icons-have-that-shape-for-a-very-good-reason-720d4e7c8a14
|
// See https://hackernoon.com/apples-icons-have-that-shape-for-a-very-good-reason-720d4e7c8a14
|
||||||
// .
|
// .
|
||||||
// You select the type of rounding using the `method` option, which should be `"smooth"` to
|
// You select the type of rounding using the `method` parameter, which should be `"smooth"` to
|
||||||
// get continuous curvature rounding, `"circle"` to get circular rounding, or `"chamfer"` to get chamfers. The default is circle
|
// get continuous curvature rounding, `"circle"` to get circular rounding, or `"chamfer"` to get chamfers. The default is circle
|
||||||
// rounding. Each method has two options you can use to specify the amount of rounding.
|
// rounding. Each method accepts multiple options to specify the amount of rounding.
|
||||||
// All of the rounding methods accept the cut option. This mode specifies the distance from the unrounded corner to the rounded tip, so how
|
// .
|
||||||
|
// The `cut` parameter specifies the distance from the unrounded corner to the rounded tip, so how
|
||||||
// much of the corner to "cut" off. This can be easier to understand than setting a circular radius, which can be
|
// much of the corner to "cut" off. This can be easier to understand than setting a circular radius, which can be
|
||||||
// unexpectedly extreme when the corner is very sharp. It also allows a systematic specification of
|
// unexpectedly extreme when the corner is very sharp. It also allows a systematic specification of
|
||||||
// corner treatments that are the same size for all three methods.
|
// corner treatments that are the same size for all three methods.
|
||||||
// .
|
// .
|
||||||
// For circular rounding you can also use the `radius` parameter, which sets a circular rounding
|
// The `joint` parameter specifies the distance
|
||||||
// radius. For chamfers and smooth rounding you can specify the `joint` parameter, which specifies the distance
|
|
||||||
// away from the corner along the path where the roundover or chamfer should start. The figure below shows
|
// away from the corner along the path where the roundover or chamfer should start. The figure below shows
|
||||||
// the cut and joint distances for a given roundover.
|
// the cut and joint distances for a given roundover. This parameter is good for ensuring that your roundover will
|
||||||
|
// fit on the polygon, since you can easily tell whether adjacent corner treatments will interfere.
|
||||||
|
// .
|
||||||
|
// For circular rounding you can also use the `radius` parameter, which sets a circular rounding
|
||||||
|
// radius.
|
||||||
// .
|
// .
|
||||||
// The `"smooth"` method rounding also has a parameter that specifies how smooth the curvature match
|
// The `"smooth"` method rounding also has a parameter that specifies how smooth the curvature match
|
||||||
// is. This parameter, `k`, ranges from 0 to 1, with a default of 0.5. Larger values give a more
|
// is. This parameter, `k`, ranges from 0 to 1, with a default of 0.5. Larger values give a more
|
||||||
// abrupt transition and smaller ones a more gradual transition. If you set the value much higher
|
// abrupt transition and smaller ones a more gradual transition. If you set the value much higher
|
||||||
// than 0.8 the curvature changes abruptly enough that though it is theoretically continuous, it may
|
// than 0.8 the curvature changes abruptly enough that though it is theoretically continuous, it may
|
||||||
// not be continuous in practice. If you set it very small then the transition is so gradual that
|
// not be continuous in practice. If you set it very small then the transition is so gradual that
|
||||||
// the length of the roundover may be extremely long.
|
// the length of the roundover may be extremely long.
|
||||||
// .
|
// .
|
||||||
// If you select curves that are too large to fit the function will fail with an error. You can set `verbose=true` to
|
// If you select curves that are too large to fit the function will fail with an error. You can set `verbose=true` to
|
||||||
// get a message showing a list of scale factors you can apply to your rounding parameters so that the
|
// get a message showing a list of scale factors you can apply to your rounding parameters so that the
|
||||||
|
@ -54,6 +58,9 @@ include <structs.scad>
|
||||||
// of the curve are not rounded. In this case you can specify a full list of points anyway, and the endpoint values are ignored,
|
// of the curve are not rounded. In this case you can specify a full list of points anyway, and the endpoint values are ignored,
|
||||||
// or you can specify a list that has length len(path)-2, omitting the two dummy values.
|
// or you can specify a list that has length len(path)-2, omitting the two dummy values.
|
||||||
// .
|
// .
|
||||||
|
// If your input path includes collinear points you must use a cut or radius value of zero for those "corners". You can
|
||||||
|
// choose a nonzero joint parameter, which will cause extra points to be inserted.
|
||||||
|
// .
|
||||||
// Examples:
|
// Examples:
|
||||||
// * `method="circle", radius=2`:
|
// * `method="circle", radius=2`:
|
||||||
// Rounds every point with circular, radius 2 roundover
|
// Rounds every point with circular, radius 2 roundover
|
||||||
|
@ -211,7 +218,6 @@ function round_corners(path, method="circle", radius, cut, joint, k, closed=true
|
||||||
assert(k_ok,method=="smooth" ? str("Input k must be a number or list with length ",len(path), closed?"":str(" or ",len(path)-2)) :
|
assert(k_ok,method=="smooth" ? str("Input k must be a number or list with length ",len(path), closed?"":str(" or ",len(path)-2)) :
|
||||||
"Input k is only allowed with method=\"smooth\"")
|
"Input k is only allowed with method=\"smooth\"")
|
||||||
assert(method=="circle" || measure!="radius", "radius parameter allowed only with method=\"circle\"")
|
assert(method=="circle" || measure!="radius", "radius parameter allowed only with method=\"circle\"")
|
||||||
assert(method!="circle" || measure!="joint", "joint parameter not allowed with method=\"circle\"")
|
|
||||||
let(
|
let(
|
||||||
parm = is_num(size) ? repeat(size, len(path)) :
|
parm = is_num(size) ? repeat(size, len(path)) :
|
||||||
len(size)<len(path) ? [0, each size, 0] :
|
len(size)<len(path) ? [0, each size, 0] :
|
||||||
|
@ -236,20 +242,23 @@ function round_corners(path, method="circle", radius, cut, joint, k, closed=true
|
||||||
angle = vector_angle(select(path,i-1,i+1))/2
|
angle = vector_angle(select(path,i-1,i+1))/2
|
||||||
)
|
)
|
||||||
(!closed && (i==0 || i==len(path)-1)) ? [0] : // Force zeros at ends for non-closed
|
(!closed && (i==0 || i==len(path)-1)) ? [0] : // Force zeros at ends for non-closed
|
||||||
|
parm[i]==0 ? [0] : // If no rounding requested then don't try to compute parameters
|
||||||
(method=="chamfer" && measure=="joint")? [parm[i]] :
|
(method=="chamfer" && measure=="joint")? [parm[i]] :
|
||||||
(method=="chamfer" && measure=="cut") ? [parm[i]/cos(angle)] :
|
(method=="chamfer" && measure=="cut") ? [parm[i]/cos(angle)] :
|
||||||
(method=="smooth" && measure=="joint") ? [parm[i],k[i]] :
|
(method=="smooth" && measure=="joint") ? [parm[i],k[i]] :
|
||||||
(method=="smooth" && measure=="cut") ? [8*parm[i]/cos(angle)/(1+4*k[i]),k[i]] :
|
(method=="smooth" && measure=="cut") ? [8*parm[i]/cos(angle)/(1+4*k[i]),k[i]] :
|
||||||
(method=="circle" && measure=="radius")? [parm[i]/tan(angle), parm[i]] :
|
(method=="circle" && measure=="radius")? [parm[i]/tan(angle), parm[i]] :
|
||||||
let( circ_radius = parm[i] / (1/sin(angle) - 1))
|
(method=="circle" && measure=="joint") ? [parm[i], parm[i]*tan(angle)] :
|
||||||
[circ_radius/tan(angle), circ_radius],
|
/*(method=="circle" && measure=="cut")*/ approx(angle,90) ? [INF] :
|
||||||
|
let( circ_radius = parm[i] / (1/sin(angle) - 1))
|
||||||
|
[circ_radius/tan(angle), circ_radius],
|
||||||
],
|
],
|
||||||
lengths = [for(i=[0:1:len(path)]) norm(select(path,i)-select(path,i-1))],
|
lengths = [for(i=[0:1:len(path)]) norm(select(path,i)-select(path,i-1))],
|
||||||
scalefactors = [
|
scalefactors = [
|
||||||
for(i=[0:1:len(path)-1])
|
for(i=[0:1:len(path)-1])
|
||||||
min(
|
min(
|
||||||
lengths[i]/sum(subindex(select(dk,i-1,i),0)),
|
lengths[i]/(select(dk,i-1)[0]+dk[i][0]),
|
||||||
lengths[i+1]/sum(subindex(select(dk,i,i+1),0))
|
lengths[i+1]/(dk[i][0]+select(dk,i+1)[0])
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
dummy = verbose ? echo("Roundover scale factors:",scalefactors) : 0
|
dummy = verbose ? echo("Roundover scale factors:",scalefactors) : 0
|
||||||
|
@ -320,14 +329,17 @@ function _chamfcorner(points, parm) =
|
||||||
|
|
||||||
function _circlecorner(points, parm) =
|
function _circlecorner(points, parm) =
|
||||||
let(
|
let(
|
||||||
angle = vector_angle(points)/2,
|
angle = vector_angle(points)/2,
|
||||||
d = parm[0],
|
d = parm[0],
|
||||||
r = parm[1],
|
r = parm[1],
|
||||||
prev = unit(points[0]-points[1]),
|
prev = unit(points[0]-points[1]),
|
||||||
next = unit(points[2]-points[1]),
|
next = unit(points[2]-points[1])
|
||||||
center = r/sin(angle) * unit(prev+next)+points[1],
|
)
|
||||||
start = points[1]+prev*d,
|
approx(angle,90) ? [points[1]+prev*d, points[1]+next*d] :
|
||||||
end = points[1]+next*d
|
let(
|
||||||
|
center = r/sin(angle) * unit(prev+next)+points[1],
|
||||||
|
start = points[1]+prev*d,
|
||||||
|
end = points[1]+next*d
|
||||||
) // 90-angle is half the angle of the circular arc
|
) // 90-angle is half the angle of the circular arc
|
||||||
arc(max(3,ceil((90-angle)/180*segs(r))), cp=center, points=[start,end]);
|
arc(max(3,ceil((90-angle)/180*segs(r))), cp=center, points=[start,end]);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue