mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-21 03:49:38 +00:00
Merge pull request #580 from RonaldoCMP/master
introduce convex collision and distance
This commit is contained in:
commit
7a3720a812
10 changed files with 476 additions and 162 deletions
98
arrays.scad
98
arrays.scad
|
@ -59,14 +59,17 @@ function _same_type(a,b, depth) =
|
||||||
// Returns a portion of a list, wrapping around past the beginning, if end<start.
|
// Returns a portion of a list, wrapping around past the beginning, if end<start.
|
||||||
// The first item is index 0. Negative indexes are counted back from the end.
|
// The first item is index 0. Negative indexes are counted back from the end.
|
||||||
// The last item is -1. If only the `start` index is given, returns just the value
|
// The last item is -1. If only the `start` index is given, returns just the value
|
||||||
// at that position.
|
// at that position when `start` is a number or the selected list of entries when `start` is
|
||||||
|
// a list of indices or a range.
|
||||||
// Usage:
|
// Usage:
|
||||||
// item = select(list,start);
|
// item = select(list,start);
|
||||||
|
// item = select(list,[s:d:e]);
|
||||||
|
// item = select(list,[i0,i1...,ik]);
|
||||||
// list = select(list,start,end);
|
// list = select(list,start,end);
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// list = The list to get the portion of.
|
// list = The list to get the portion of.
|
||||||
// start = The index of the first item.
|
// start = Either the index of the first item or an index range or a list of indices.
|
||||||
// end = The index of the last item.
|
// end = The index of the last item when `start` is a number. When `start` is a list or a range, `end` should not be given.
|
||||||
// See Also: slice(), subindex(), last()
|
// See Also: slice(), subindex(), last()
|
||||||
// Example:
|
// Example:
|
||||||
// l = [3,4,5,6,7,8,9];
|
// l = [3,4,5,6,7,8,9];
|
||||||
|
@ -78,7 +81,7 @@ function _same_type(a,b, depth) =
|
||||||
// f = select(l, 4); // Returns 7
|
// f = select(l, 4); // Returns 7
|
||||||
// g = select(l, -2); // Returns 8
|
// g = select(l, -2); // Returns 8
|
||||||
// h = select(l, [1:3]); // Returns [4,5,6]
|
// h = select(l, [1:3]); // Returns [4,5,6]
|
||||||
// i = select(l, [1,3]); // Returns [4,6]
|
// i = select(l, [3,1]); // Returns [6,4]
|
||||||
function select(list, start, end) =
|
function select(list, start, end) =
|
||||||
assert( is_list(list) || is_string(list), "Invalid list.")
|
assert( is_list(list) || is_string(list), "Invalid list.")
|
||||||
let(l=len(list))
|
let(l=len(list))
|
||||||
|
@ -89,7 +92,7 @@ function select(list, start, end) =
|
||||||
? list[ (start%l+l)%l ]
|
? list[ (start%l+l)%l ]
|
||||||
: assert( is_list(start) || is_range(start), "Invalid start parameter")
|
: assert( is_list(start) || is_range(start), "Invalid start parameter")
|
||||||
[for (i=start) list[ (i%l+l)%l ] ]
|
[for (i=start) list[ (i%l+l)%l ] ]
|
||||||
: assert(is_finite(start), "Invalid start parameter.")
|
: assert(is_finite(start), "When `end` is given, `start` parameter should be a number.")
|
||||||
assert(is_finite(end), "Invalid end parameter.")
|
assert(is_finite(end), "Invalid end parameter.")
|
||||||
let( s = (start%l+l)%l, e = (end%l+l)%l )
|
let( s = (start%l+l)%l, e = (end%l+l)%l )
|
||||||
(s <= e)
|
(s <= e)
|
||||||
|
@ -946,6 +949,31 @@ function shuffle(list,seed) =
|
||||||
)
|
)
|
||||||
concat(shuffle(left), shuffle(right));
|
concat(shuffle(left), shuffle(right));
|
||||||
|
|
||||||
|
// idx should be an index of the arrays l[i]
|
||||||
|
function _group_sort_by_index(l,idx) =
|
||||||
|
len(l) == 0 ? [] :
|
||||||
|
len(l) == 1 ? [l] :
|
||||||
|
let( pivot = l[floor(len(l)/2)][idx],
|
||||||
|
equal = [ for(li=l) if( li[idx]==pivot) li ] ,
|
||||||
|
lesser = [ for(li=l) if( li[idx]< pivot) li ] ,
|
||||||
|
greater = [ for(li=l) if( li[idx]> pivot) li ]
|
||||||
|
)
|
||||||
|
concat( _group_sort_by_index(lesser,idx),
|
||||||
|
[equal],
|
||||||
|
_group_sort_by_index(greater,idx) ) ;
|
||||||
|
|
||||||
|
function _group_sort(l) =
|
||||||
|
len(l) == 0 ? [] :
|
||||||
|
len(l) == 1 ? [l] :
|
||||||
|
let( pivot = l[floor(len(l)/2)],
|
||||||
|
equal = [ for(li=l) if( li==pivot) li ] ,
|
||||||
|
lesser = [ for(li=l) if( li< pivot) li ] ,
|
||||||
|
greater = [ for(li=l) if( li> pivot) li ]
|
||||||
|
)
|
||||||
|
concat( _group_sort(lesser),
|
||||||
|
[equal],
|
||||||
|
_group_sort(greater) ) ;
|
||||||
|
|
||||||
|
|
||||||
// Sort a vector of scalar values with the native comparison operator
|
// Sort a vector of scalar values with the native comparison operator
|
||||||
// all elements should have the same type.
|
// all elements should have the same type.
|
||||||
|
@ -1007,7 +1035,7 @@ function _sort_general(arr, idx=undef, indexed=false) =
|
||||||
|
|
||||||
// lexical sort using compare_vals()
|
// lexical sort using compare_vals()
|
||||||
function _lexical_sort(arr) =
|
function _lexical_sort(arr) =
|
||||||
arr==[] ? [] : len(arr)==1? arr :
|
len(arr)<=1? arr :
|
||||||
let( pivot = arr[floor(len(arr)/2)] )
|
let( pivot = arr[floor(len(arr)/2)] )
|
||||||
let(
|
let(
|
||||||
lesser = [ for (entry=arr) if (compare_vals(entry, pivot) <0 ) entry ],
|
lesser = [ for (entry=arr) if (compare_vals(entry, pivot) <0 ) entry ],
|
||||||
|
@ -1034,7 +1062,7 @@ function _indexed_sort(arrind) =
|
||||||
// Usage:
|
// Usage:
|
||||||
// slist = sort(list, <idx>);
|
// slist = sort(list, <idx>);
|
||||||
// Topics: List Handling
|
// Topics: List Handling
|
||||||
// See Also: shuffle(), sortidx(), unique(), unique_count()
|
// See Also: shuffle(), sortidx(), unique(), unique_count(), group_sort()
|
||||||
// Description:
|
// Description:
|
||||||
// Sorts the given list in lexicographic order. If the input is a homogeneous simple list or a homogeneous
|
// Sorts the given list in lexicographic order. If the input is a homogeneous simple list or a homogeneous
|
||||||
// list of vectors (see function is_homogeneous), the sorting method uses the native comparison operator and is faster.
|
// list of vectors (see function is_homogeneous), the sorting method uses the native comparison operator and is faster.
|
||||||
|
@ -1075,7 +1103,7 @@ function sort(list, idx=undef) =
|
||||||
// Usage:
|
// Usage:
|
||||||
// idxlist = sortidx(list, <idx>);
|
// idxlist = sortidx(list, <idx>);
|
||||||
// Topics: List Handling
|
// Topics: List Handling
|
||||||
// See Also: shuffle(), sort(), unique(), unique_count()
|
// See Also: shuffle(), sort(), group_sort(), unique(), unique_count()
|
||||||
// Description:
|
// Description:
|
||||||
// Given a list, sort it as function `sort()`, and returns
|
// Given a list, sort it as function `sort()`, and returns
|
||||||
// a list of indexes into the original list in that sorted order.
|
// a list of indexes into the original list in that sorted order.
|
||||||
|
@ -1123,27 +1151,70 @@ function sortidx(list, idx=undef) =
|
||||||
: _sort_general(list,idx,indexed=true);
|
: _sort_general(list,idx,indexed=true);
|
||||||
|
|
||||||
|
|
||||||
|
// Function: group_sort()
|
||||||
|
// Usage:
|
||||||
|
// ulist = group_sort(list);
|
||||||
|
// Topics: List Handling
|
||||||
|
// See Also: shuffle(), sort(), sortidx(), unique(), unique_count()
|
||||||
|
// Description:
|
||||||
|
// Given a list of values, returns the sorted list with all repeated items grouped in a list.
|
||||||
|
// When the list entries are themselves lists, the sorting may be done based on the `idx` entry
|
||||||
|
// of those entries, that should be numbers.
|
||||||
|
// The result is always a list of lists.
|
||||||
|
// Arguments:
|
||||||
|
// list = The list to sort.
|
||||||
|
// idx = If given, do the comparison based just on the specified index. Default: zero.
|
||||||
|
// Example:
|
||||||
|
// sorted = group_sort([5,2,8,3,1,3,8,7,5]); // Returns: [[1],[2],[3,3],[5,5],[7],[8,8]]
|
||||||
|
// sorted2 = group_sort([[5,"a"],[2,"b"], [5,"c"], [3,"d"], [2,"e"] ], idx=0); // Returns: [[[2,"b"],[2,"e"]], [[5,"a"],[5,"c"]], [[3,"d"]] ]
|
||||||
|
function group_sort(list, idx) =
|
||||||
|
assert(is_list(list), "Input should be a list." )
|
||||||
|
assert(is_undef(idx) || (is_finite(idx) && idx>=0) , "Invalid index." )
|
||||||
|
len(list)<=1 ? [list] :
|
||||||
|
is_vector(list)? _group_sort(list) :
|
||||||
|
let( idx = is_undef(idx) ? 0 : idx )
|
||||||
|
assert( [for(entry=list) if(!is_list(entry) || len(entry)<idx || !is_num(entry[idx]) ) 1]==[],
|
||||||
|
"Some entry of the list is a list shorter than `idx` or the indexed entry of it is not a number." )
|
||||||
|
_group_sort_by_index(list,idx);
|
||||||
|
|
||||||
|
|
||||||
// Function: unique()
|
// Function: unique()
|
||||||
// Usage:
|
// Usage:
|
||||||
// ulist = unique(list);
|
// ulist = unique(list);
|
||||||
// Topics: List Handling
|
// Topics: List Handling
|
||||||
// See Also: shuffle(), sort(), sortidx(), unique_count()
|
// See Also: shuffle(), sort(), sortidx(), unique_count()
|
||||||
// Description:
|
// Description:
|
||||||
// Returns a sorted list with all repeated items removed.
|
// Given a string or a list returns the sorted string or the sorted list with all repeated items removed.
|
||||||
|
// The sorting order of non homogeneous lists is the function `sort` order.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// list = The list to uniquify.
|
// list = The list to uniquify.
|
||||||
// Example:
|
// Example:
|
||||||
// sorted = unique([5,2,8,3,1,3,8,7,5]); // Returns: [1,2,3,5,7,8]
|
// sorted = unique([5,2,8,3,1,3,8,7,5]); // Returns: [1,2,3,5,7,8]
|
||||||
|
// sorted = unique("axdbxxc"); // Returns: "abcdx"
|
||||||
|
// sorted = unique([true,2,"xba",[1,0],true,[0,0],3,"a",[0,0],2]); // Returns: [true,2,3,"a","xba",[0,0],[1,0]]
|
||||||
function unique(list) =
|
function unique(list) =
|
||||||
assert(is_list(list)||is_string(list), "Invalid input." )
|
assert(is_list(list)||is_string(list), "Invalid input." )
|
||||||
is_string(list)? str_join(unique([for (x = list) x])) :
|
is_string(list)? str_join(unique([for (x = list) x])) :
|
||||||
len(list)<=1? list :
|
len(list)<=1? list :
|
||||||
let( sorted = sort(list))
|
is_homogeneous(list,1) && ! is_list(list[0])
|
||||||
|
? _unique_sort(list)
|
||||||
|
: let( sorted = sort(list))
|
||||||
[ for (i=[0:1:len(sorted)-1])
|
[ for (i=[0:1:len(sorted)-1])
|
||||||
if (i==0 || (sorted[i] != sorted[i-1]))
|
if (i==0 || (sorted[i] != sorted[i-1]))
|
||||||
sorted[i]
|
sorted[i]
|
||||||
];
|
];
|
||||||
|
|
||||||
|
function _unique_sort(l) =
|
||||||
|
len(l) <= 1 ? l :
|
||||||
|
let( pivot = l[floor(len(l)/2)],
|
||||||
|
equal = [ for(li=l) if( li==pivot) li ] ,
|
||||||
|
lesser = [ for(li=l) if( li<pivot ) li ] ,
|
||||||
|
greater = [ for(li=l) if( li>pivot) li ]
|
||||||
|
)
|
||||||
|
concat( _unique_sort(lesser),
|
||||||
|
equal[0],
|
||||||
|
_unique_sort(greater) ) ;
|
||||||
|
|
||||||
|
|
||||||
// Function: unique_count()
|
// Function: unique_count()
|
||||||
// Usage:
|
// Usage:
|
||||||
|
@ -1160,7 +1231,10 @@ function unique(list) =
|
||||||
function unique_count(list) =
|
function unique_count(list) =
|
||||||
assert(is_list(list) || is_string(list), "Invalid input." )
|
assert(is_list(list) || is_string(list), "Invalid input." )
|
||||||
list == [] ? [[],[]] :
|
list == [] ? [[],[]] :
|
||||||
let( list=sort(list) )
|
is_homogeneous(list,1) && ! is_list(list[0])
|
||||||
|
? let( sorted = _group_sort(list) )
|
||||||
|
[ [for(s=sorted) s[0] ], [for(s=sorted) len(s) ] ]
|
||||||
|
: let( list=sort(list) )
|
||||||
let( ind = [0, for(i=[1:1:len(list)-1]) if (list[i]!=list[i-1]) i] )
|
let( ind = [0, for(i=[1:1:len(list)-1]) if (list[i]!=list[i-1]) i] )
|
||||||
[ select(list,ind), deltas( concat(ind,[len(list)]) ) ];
|
[ select(list,ind), deltas( concat(ind,[len(list)]) ) ];
|
||||||
|
|
||||||
|
@ -1692,7 +1766,7 @@ function submatrix_set(M,A,m=0,n=0) =
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// v = The list of items to group.
|
// v = The list of items to group.
|
||||||
// cnt = The number of items to put in each grouping. Default:2
|
// cnt = The number of items to put in each grouping. Default:2
|
||||||
// dflt = The default value to fill in with is the list is not a multiple of `cnt` items long. Default: 0
|
// dflt = The default value to fill in with if the list is not a multiple of `cnt` items long. Default: 0
|
||||||
// Example:
|
// Example:
|
||||||
// v = [1,2,3,4,5,6];
|
// v = [1,2,3,4,5,6];
|
||||||
// a = array_group(v,2) returns [[1,2], [3,4], [5,6]]
|
// a = array_group(v,2) returns [[1,2], [3,4], [5,6]]
|
||||||
|
|
|
@ -237,7 +237,7 @@ function project_plane(plane,p) =
|
||||||
let(plane = plane_from_points(plane))
|
let(plane = plane_from_points(plane))
|
||||||
assert(is_def(plane), "Point list is not coplanar")
|
assert(is_def(plane), "Point list is not coplanar")
|
||||||
project_plane(plane)
|
project_plane(plane)
|
||||||
: assert(is_def(p), str("Invalid plane specification",plane))
|
: assert(is_def(p), str("Invalid plane specification: ",plane))
|
||||||
is_vnf(p) ? [project_plane(plane,p[0]), p[1]]
|
is_vnf(p) ? [project_plane(plane,p[0]), p[1]]
|
||||||
: is_list(p) && is_list(p[0]) && is_vector(p[0][0],3) ? // bezier patch or region
|
: is_list(p) && is_list(p[0]) && is_vector(p[0][0],3) ? // bezier patch or region
|
||||||
[for(plist=p) project_plane(plane,plist)]
|
[for(plist=p) project_plane(plane,plist)]
|
||||||
|
|
|
@ -34,7 +34,7 @@ module move_copies(a=[[0,0,0]])
|
||||||
assert(is_list(a));
|
assert(is_list(a));
|
||||||
for ($idx = idx(a)) {
|
for ($idx = idx(a)) {
|
||||||
$pos = a[$idx];
|
$pos = a[$idx];
|
||||||
assert(is_vector($pos));
|
assert(is_vector($pos),"move_copies offsets should be a 2d or 3d vector.");
|
||||||
translate($pos) children();
|
translate($pos) children();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
398
geometry.scad
398
geometry.scad
|
@ -19,15 +19,8 @@
|
||||||
// edge = Array of two points forming the line segment to test against.
|
// edge = Array of two points forming the line segment to test against.
|
||||||
// eps = Tolerance in geometric comparisons. Default: `EPSILON` (1e-9)
|
// eps = Tolerance in geometric comparisons. Default: `EPSILON` (1e-9)
|
||||||
function point_on_segment2d(point, edge, eps=EPSILON) =
|
function point_on_segment2d(point, edge, eps=EPSILON) =
|
||||||
assert( is_vector(point,2), "Invalid point." )
|
|
||||||
assert( is_finite(eps) && (eps>=0), "The tolerance should be a non-negative value." )
|
assert( is_finite(eps) && (eps>=0), "The tolerance should be a non-negative value." )
|
||||||
assert( _valid_line(edge,2,eps=eps), "Invalid segment." )
|
point_segment_distance(point, edge)<eps;
|
||||||
let( dp = point-edge[0],
|
|
||||||
de = edge[1]-edge[0],
|
|
||||||
ne = norm(de) )
|
|
||||||
( dp*de >= -eps*ne )
|
|
||||||
&& ( (dp-de)*de <= eps*ne ) // point projects on the segment
|
|
||||||
&& _dist2line(point-edge[0],unit(de))<eps; // point is on the line
|
|
||||||
|
|
||||||
|
|
||||||
//Internal - distance from point `d` to the line passing through the origin with unit direction n
|
//Internal - distance from point `d` to the line passing through the origin with unit direction n
|
||||||
|
@ -44,7 +37,7 @@ function _point_above_below_segment(point, edge) =
|
||||||
//Internal
|
//Internal
|
||||||
function _valid_line(line,dim,eps=EPSILON) =
|
function _valid_line(line,dim,eps=EPSILON) =
|
||||||
is_matrix(line,2,dim)
|
is_matrix(line,2,dim)
|
||||||
&& ! approx(norm(line[1]-line[0]), 0, eps);
|
&& norm(line[1]-line[0])>eps*max(norm(line[1]),norm(line[0]));
|
||||||
|
|
||||||
//Internal
|
//Internal
|
||||||
function _valid_plane(p, eps=EPSILON) = is_vector(p,4) && ! approx(norm(p),0,eps);
|
function _valid_plane(p, eps=EPSILON) = is_vector(p,4) && ! approx(norm(p),0,eps);
|
||||||
|
@ -87,7 +80,7 @@ function collinear(a, b, c, eps=EPSILON) =
|
||||||
|
|
||||||
// Function: point_line_distance()
|
// Function: point_line_distance()
|
||||||
// Usage:
|
// Usage:
|
||||||
// point_line_distance(pt, line);
|
// point_line_distance(line, pt);
|
||||||
// Description:
|
// Description:
|
||||||
// Finds the perpendicular distance of a point `pt` from the line `line`.
|
// Finds the perpendicular distance of a point `pt` from the line `line`.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
|
@ -113,6 +106,8 @@ function point_line_distance(pt, line) =
|
||||||
// dist = point_segment_distance([3,8], [[-10,0], [10,0]]); // Returns: 8
|
// dist = point_segment_distance([3,8], [[-10,0], [10,0]]); // Returns: 8
|
||||||
// dist2 = point_segment_distance([14,3], [[-10,0], [10,0]]); // Returns: 5
|
// dist2 = point_segment_distance([14,3], [[-10,0], [10,0]]); // Returns: 5
|
||||||
function point_segment_distance(pt, seg) =
|
function point_segment_distance(pt, seg) =
|
||||||
|
assert( is_matrix(concat([pt],seg),3),
|
||||||
|
"Input should be a point and a valid segment with the dimension equal to the point." )
|
||||||
norm(seg[0]-seg[1]) < EPSILON ? norm(pt-seg[0]) :
|
norm(seg[0]-seg[1]) < EPSILON ? norm(pt-seg[0]) :
|
||||||
norm(pt-segment_closest_point(seg,pt));
|
norm(pt-segment_closest_point(seg,pt));
|
||||||
|
|
||||||
|
@ -129,34 +124,9 @@ function point_segment_distance(pt, seg) =
|
||||||
// dist = segment_distance([[-14,3], [-15,9]], [[-10,0], [10,0]]); // Returns: 5
|
// dist = segment_distance([[-14,3], [-15,9]], [[-10,0], [10,0]]); // Returns: 5
|
||||||
// dist2 = segment_distance([[-5,5], [5,-5]], [[-10,3], [10,-3]]); // Returns: 0
|
// dist2 = segment_distance([[-5,5], [5,-5]], [[-10,3], [10,-3]]); // Returns: 0
|
||||||
function segment_distance(seg1, seg2) =
|
function segment_distance(seg1, seg2) =
|
||||||
let(
|
assert( is_matrix(concat(seg1,seg2),4),
|
||||||
dseg1 = seg1[1]-seg1[0],
|
"Inputs should be two valid segments." )
|
||||||
dseg2 = seg2[1]-seg2[0],
|
convex_distance(seg1,seg2);
|
||||||
A = [ [dseg1*dseg1, -dseg1*dseg2],
|
|
||||||
[-dseg2*dseg1, dseg2*dseg2] ],
|
|
||||||
b = -[ dseg1, -dseg2 ]*(seg1[0]-seg2[0]),
|
|
||||||
uv = linear_solve(A,b)
|
|
||||||
)
|
|
||||||
!uv ?
|
|
||||||
norm(dseg1)<EPSILON
|
|
||||||
? norm(dseg2)<EPSILON
|
|
||||||
? norm(seg1[0]-seg2[0])
|
|
||||||
: point_segment_distance(seg1[0],seg2)
|
|
||||||
: point_segment_distance(seg2[0],seg1) :
|
|
||||||
uv[0]>=0 && uv[0]<=1 ?
|
|
||||||
let( p1 = seg1[0] + uv[0]*dseg1 )
|
|
||||||
uv[1]>=0 && uv[1]<=1
|
|
||||||
? norm(p1 - (seg2[0] + uv[1]*dseg2) )
|
|
||||||
: min(norm(p1-seg2[0]), norm(p1-seg2[1])) :
|
|
||||||
uv[1]>=0 && uv[1]<=1
|
|
||||||
? let( p2 = seg2[0] + uv[1]*dseg2 )
|
|
||||||
min(norm(p2-seg1[0]), norm(p2-seg1[1]))
|
|
||||||
: min(
|
|
||||||
norm(seg1[0]-seg2[0]),
|
|
||||||
norm(seg1[0]-seg2[1]),
|
|
||||||
norm(seg1[1]-seg2[0]),
|
|
||||||
norm(seg1[1]-seg2[1])
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
// Function: line_normal()
|
// Function: line_normal()
|
||||||
|
@ -494,17 +464,9 @@ function ray_closest_point(ray,pt) =
|
||||||
// color("blue") translate(pt) sphere(r=1,$fn=12);
|
// color("blue") translate(pt) sphere(r=1,$fn=12);
|
||||||
// color("red") translate(p2) sphere(r=1,$fn=12);
|
// color("red") translate(p2) sphere(r=1,$fn=12);
|
||||||
function segment_closest_point(seg,pt) =
|
function segment_closest_point(seg,pt) =
|
||||||
assert(_valid_line(seg), "Invalid segment." )
|
assert( is_matrix(concat([pt],seg),3) ,
|
||||||
assert(len(pt)==len(seg[0]), "Incompatible dimensions." )
|
"Invalid point or segment or incompatible dimensions." )
|
||||||
approx(seg[0],seg[1])? seg[0] :
|
pt + _closest_s1([seg[0]-pt, seg[1]-pt])[0];
|
||||||
let(
|
|
||||||
seglen = norm(seg[1]-seg[0]),
|
|
||||||
segvec = (seg[1]-seg[0])/seglen,
|
|
||||||
projection = (pt-seg[0]) * segvec
|
|
||||||
)
|
|
||||||
projection<=0 ? seg[0] :
|
|
||||||
projection>=seglen ? seg[1] :
|
|
||||||
seg[0] + projection*segvec;
|
|
||||||
|
|
||||||
|
|
||||||
// Function: line_from_points()
|
// Function: line_from_points()
|
||||||
|
@ -512,7 +474,7 @@ function segment_closest_point(seg,pt) =
|
||||||
// line_from_points(points, [fast], [eps]);
|
// line_from_points(points, [fast], [eps]);
|
||||||
// Description:
|
// Description:
|
||||||
// Given a list of 2 or more collinear points, returns a line containing them.
|
// Given a list of 2 or more collinear points, returns a line containing them.
|
||||||
// If `fast` is false and the points are coincident, then `undef` is returned.
|
// If `fast` is false and the points are coincident or non-collinear, then `undef` is returned.
|
||||||
// if `fast` is true, then the collinearity test is skipped and a line passing through 2 distinct arbitrary points is returned.
|
// if `fast` is true, then the collinearity test is skipped and a line passing through 2 distinct arbitrary points is returned.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// points = The list of points to find the line through.
|
// points = The list of points to find the line through.
|
||||||
|
@ -522,7 +484,7 @@ function line_from_points(points, fast=false, eps=EPSILON) =
|
||||||
assert( is_path(points,dim=undef), "Improper point list." )
|
assert( is_path(points,dim=undef), "Improper point list." )
|
||||||
assert( is_finite(eps) && (eps>=0), "The tolerance should be a non-negative value." )
|
assert( is_finite(eps) && (eps>=0), "The tolerance should be a non-negative value." )
|
||||||
let( pb = furthest_point(points[0],points) )
|
let( pb = furthest_point(points[0],points) )
|
||||||
approx(norm(points[pb]-points[0]),0) ? undef :
|
norm(points[pb]-points[0])<eps*max(norm(points[pb]),norm(points[0])) ? undef :
|
||||||
fast || collinear(points) ? [points[pb], points[0]] : undef;
|
fast || collinear(points) ? [points[pb], points[0]] : undef;
|
||||||
|
|
||||||
|
|
||||||
|
@ -1171,7 +1133,7 @@ function _general_plane_line_intersection(plane, line, eps=EPSILON) =
|
||||||
// Description:
|
// Description:
|
||||||
// Returns a new representation [A,B,C,D] of `plane` where norm([A,B,C]) is equal to one.
|
// Returns a new representation [A,B,C,D] of `plane` where norm([A,B,C]) is equal to one.
|
||||||
function normalize_plane(plane) =
|
function normalize_plane(plane) =
|
||||||
assert( _valid_plane(plane), "Invalid plane." )
|
assert( _valid_plane(plane), str("Invalid plane. ",plane ) )
|
||||||
plane/norm(point3d(plane));
|
plane/norm(point3d(plane));
|
||||||
|
|
||||||
|
|
||||||
|
@ -1179,12 +1141,12 @@ function normalize_plane(plane) =
|
||||||
// Usage:
|
// Usage:
|
||||||
// angle = plane_line_angle(plane,line);
|
// angle = plane_line_angle(plane,line);
|
||||||
// Description:
|
// Description:
|
||||||
// Compute the angle between a plane [A, B, C, D] and a line, specified as a pair of points [p1,p2].
|
// Compute the angle between a plane [A, B, C, D] and a 3d line, specified as a pair of 3d points [p1,p2].
|
||||||
// The resulting angle is signed, with the sign positive if the vector p2-p1 lies on
|
// The resulting angle is signed, with the sign positive if the vector p2-p1 lies on
|
||||||
// the same side of the plane as the plane's normal vector.
|
// the same side of the plane as the plane's normal vector.
|
||||||
function plane_line_angle(plane, line) =
|
function plane_line_angle(plane, line) =
|
||||||
assert( _valid_plane(plane), "Invalid plane." )
|
assert( _valid_plane(plane), "Invalid plane." )
|
||||||
assert( _valid_line(line), "Invalid line." )
|
assert( _valid_line(line,dim=3), "Invalid 3d line." )
|
||||||
let(
|
let(
|
||||||
linedir = unit(line[1]-line[0]),
|
linedir = unit(line[1]-line[0]),
|
||||||
normal = plane_normal(plane),
|
normal = plane_normal(plane),
|
||||||
|
@ -1209,7 +1171,7 @@ function plane_line_angle(plane, line) =
|
||||||
// eps = Tolerance in geometric comparisons. Default: `EPSILON` (1e-9)
|
// eps = Tolerance in geometric comparisons. Default: `EPSILON` (1e-9)
|
||||||
function plane_line_intersection(plane, line, bounded=false, eps=EPSILON) =
|
function plane_line_intersection(plane, line, bounded=false, eps=EPSILON) =
|
||||||
assert( is_finite(eps) && eps>=0, "The tolerance should be a positive number." )
|
assert( is_finite(eps) && eps>=0, "The tolerance should be a positive number." )
|
||||||
assert(_valid_plane(plane,eps=eps) && _valid_line(line,dim=3,eps=eps), "Invalid plane and/or line.")
|
assert(_valid_plane(plane,eps=eps) && _valid_line(line,dim=3,eps=eps), "Invalid plane and/or 3d line.")
|
||||||
assert(is_bool(bounded) || is_bool_list(bounded,2), "Invalid bound condition.")
|
assert(is_bool(bounded) || is_bool_list(bounded,2), "Invalid bound condition.")
|
||||||
let(
|
let(
|
||||||
bounded = is_list(bounded)? bounded : [bounded, bounded],
|
bounded = is_list(bounded)? bounded : [bounded, bounded],
|
||||||
|
@ -1240,7 +1202,7 @@ function polygon_line_intersection(poly, line, bounded=false, eps=EPSILON) =
|
||||||
assert( is_finite(eps) && eps>=0, "The tolerance should be a positive number." )
|
assert( is_finite(eps) && eps>=0, "The tolerance should be a positive number." )
|
||||||
assert(is_path(poly,dim=3), "Invalid polygon." )
|
assert(is_path(poly,dim=3), "Invalid polygon." )
|
||||||
assert(!is_list(bounded) || len(bounded)==2, "Invalid bound condition(s).")
|
assert(!is_list(bounded) || len(bounded)==2, "Invalid bound condition(s).")
|
||||||
assert(_valid_line(line,dim=3,eps=eps), "Invalid line." )
|
assert(_valid_line(line,dim=3,eps=eps), "Invalid 3D line." )
|
||||||
let(
|
let(
|
||||||
bounded = is_list(bounded)? bounded : [bounded, bounded],
|
bounded = is_list(bounded)? bounded : [bounded, bounded],
|
||||||
poly = deduplicate(poly),
|
poly = deduplicate(poly),
|
||||||
|
@ -1694,7 +1656,7 @@ function circle_circle_tangents(c1,r1,c2,r2,d1,d2) =
|
||||||
// eps = epsilon used for identifying the case with one solution. Default: 1e-9
|
// eps = epsilon used for identifying the case with one solution. Default: 1e-9
|
||||||
function circle_line_intersection(c,r,d,line,bounded=false,eps=EPSILON) =
|
function circle_line_intersection(c,r,d,line,bounded=false,eps=EPSILON) =
|
||||||
let(r=get_radius(r=r,d=d,dflt=undef))
|
let(r=get_radius(r=r,d=d,dflt=undef))
|
||||||
assert(_valid_line(line,2), "Input 'line' is not a valid 2d line.")
|
assert(_valid_line(line,2), "Invalid 2d line.")
|
||||||
assert(is_vector(c,2), "Circle center must be a 2-vector")
|
assert(is_vector(c,2), "Circle center must be a 2-vector")
|
||||||
assert(is_num(r) && r>0, "Radius must be positive")
|
assert(is_num(r) && r>0, "Radius must be positive")
|
||||||
assert(is_bool(bounded) || is_bool_list(bounded,2), "Invalid bound condition")
|
assert(is_bool(bounded) || is_bool_list(bounded,2), "Invalid bound condition")
|
||||||
|
@ -1738,7 +1700,7 @@ function noncollinear_triple(points,error=true,eps=EPSILON) =
|
||||||
pb = points[b],
|
pb = points[b],
|
||||||
nrm = norm(pa-pb)
|
nrm = norm(pa-pb)
|
||||||
)
|
)
|
||||||
approx(nrm, 0)
|
nrm <= eps*max(norm(pa),norm(pb))
|
||||||
? assert(!error, "Cannot find three noncollinear points in pointlist.")
|
? assert(!error, "Cannot find three noncollinear points in pointlist.")
|
||||||
[]
|
[]
|
||||||
: let(
|
: let(
|
||||||
|
@ -1761,12 +1723,13 @@ function noncollinear_triple(points,error=true,eps=EPSILON) =
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// pts = List of points.
|
// pts = List of points.
|
||||||
function pointlist_bounds(pts) =
|
function pointlist_bounds(pts) =
|
||||||
assert(is_matrix(pts) && len(pts)>0 && len(pts[0])>0 , "Invalid pointlist." )
|
assert(is_path(pts,dim=undef,fast=true) , "Invalid pointlist." )
|
||||||
let(ptsT = transpose(pts))
|
let(
|
||||||
[
|
select = ident(len(pts[0])),
|
||||||
[for(row=ptsT) min(row)],
|
spread = [for(i=[0:len(pts[0])-1])
|
||||||
[for(row=ptsT) max(row)]
|
let( spreadi = pts*select[i] )
|
||||||
];
|
[min(spreadi), max(spreadi)] ] )
|
||||||
|
transpose(spread);
|
||||||
|
|
||||||
|
|
||||||
// Function: closest_point()
|
// Function: closest_point()
|
||||||
|
@ -1805,7 +1768,7 @@ function furthest_point(pt, points) =
|
||||||
// area = polygon_area(poly);
|
// area = polygon_area(poly);
|
||||||
// Description:
|
// Description:
|
||||||
// Given a 2D or 3D planar polygon, returns the area of that polygon.
|
// Given a 2D or 3D planar polygon, returns the area of that polygon.
|
||||||
// If the polygon is self-crossing, the results are undefined. For non-planar 3D polygon the result is [].
|
// If the polygon is self-crossing, the results are undefined. For non-planar 3D polygon the result is `undef`.
|
||||||
// When `signed` is true, a signed area is returned; a positive area indicates a clockwise polygon.
|
// When `signed` is true, a signed area is returned; a positive area indicates a clockwise polygon.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// poly = Polygon to compute the area of.
|
// poly = Polygon to compute the area of.
|
||||||
|
@ -1817,53 +1780,17 @@ function polygon_area(poly, signed=false) =
|
||||||
? let( total = sum([for(i=[1:1:len(poly)-2]) cross(poly[i]-poly[0],poly[i+1]-poly[0]) ])/2 )
|
? let( total = sum([for(i=[1:1:len(poly)-2]) cross(poly[i]-poly[0],poly[i+1]-poly[0]) ])/2 )
|
||||||
signed ? total : abs(total)
|
signed ? total : abs(total)
|
||||||
: let( plane = plane_from_polygon(poly) )
|
: let( plane = plane_from_polygon(poly) )
|
||||||
plane==[]? [] :
|
plane==[]? undef :
|
||||||
let(
|
let(
|
||||||
n = plane_normal(plane),
|
n = plane_normal(plane),
|
||||||
total = sum([
|
total =
|
||||||
for(i=[1:1:len(poly)-2])
|
sum([ for(i=[1:1:len(poly)-2])
|
||||||
let(
|
cross(poly[i]-poly[0], poly[i+1]-poly[0])
|
||||||
v1 = poly[i] - poly[0],
|
|
||||||
v2 = poly[i+1] - poly[0]
|
|
||||||
)
|
|
||||||
cross(v1,v2)
|
|
||||||
]) * n/2
|
]) * n/2
|
||||||
)
|
)
|
||||||
signed ? total : abs(total);
|
signed ? total : abs(total);
|
||||||
|
|
||||||
|
|
||||||
// Function: is_convex_polygon()
|
|
||||||
// Usage:
|
|
||||||
// is_convex_polygon(poly);
|
|
||||||
// Description:
|
|
||||||
// Returns true if the given 2D or 3D polygon is convex.
|
|
||||||
// The result is meaningless if the polygon is not simple (self-intersecting) or non coplanar.
|
|
||||||
// If the points are collinear an error is generated.
|
|
||||||
// Arguments:
|
|
||||||
// poly = Polygon to check.
|
|
||||||
// eps = Tolerance for the collinearity test. Default: EPSILON.
|
|
||||||
// Example:
|
|
||||||
// is_convex_polygon(circle(d=50)); // Returns: true
|
|
||||||
// is_convex_polygon(rot([50,120,30], p=path3d(circle(1,$fn=50)))); // Returns: true
|
|
||||||
// Example:
|
|
||||||
// spiral = [for (i=[0:36]) let(a=-i*10) (10+i)*[cos(a),sin(a)]];
|
|
||||||
// is_convex_polygon(spiral); // Returns: false
|
|
||||||
function is_convex_polygon(poly,eps=EPSILON) =
|
|
||||||
assert(is_path(poly), "The input should be a 2D or 3D polygon." )
|
|
||||||
let( lp = len(poly),
|
|
||||||
p0 = poly[0] )
|
|
||||||
assert( lp>=3 , "A polygon must have at least 3 points" )
|
|
||||||
let( crosses = [for(i=[0:1:lp-1]) cross(poly[(i+1)%lp]-poly[i], poly[(i+2)%lp]-poly[(i+1)%lp]) ] )
|
|
||||||
len(p0)==2
|
|
||||||
? assert( !approx(sqrt(max(max(crosses),-min(crosses))),eps), "The points are collinear" )
|
|
||||||
min(crosses) >=0 || max(crosses)<=0
|
|
||||||
: let( prod = crosses*sum(crosses),
|
|
||||||
minc = min(prod),
|
|
||||||
maxc = max(prod) )
|
|
||||||
assert( !approx(sqrt(max(maxc,-minc)),eps), "The points are collinear" )
|
|
||||||
minc>=0 || maxc<=0;
|
|
||||||
|
|
||||||
|
|
||||||
// Function: polygon_shift()
|
// Function: polygon_shift()
|
||||||
// Usage:
|
// Usage:
|
||||||
// polygon_shift(poly, i);
|
// polygon_shift(poly, i);
|
||||||
|
@ -2030,9 +1957,9 @@ function centroid(poly, eps=EPSILON) =
|
||||||
// Returns -1 if the point is outside the polygon.
|
// Returns -1 if the point is outside the polygon.
|
||||||
// Returns 0 if the point is on the boundary.
|
// Returns 0 if the point is on the boundary.
|
||||||
// Returns 1 if the point lies in the interior.
|
// Returns 1 if the point lies in the interior.
|
||||||
// The polygon does not need to be simple: it can have self-intersections.
|
// The polygon does not need to be simple: it may have self-intersections.
|
||||||
// But the polygon cannot have holes (it must be simply connected).
|
// But the polygon cannot have holes (it must be simply connected).
|
||||||
// Rounding error may give mixed results for points on or near the boundary.
|
// Rounding errors may give mixed results for points on or near the boundary.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// point = The 2D point to check position of.
|
// point = The 2D point to check position of.
|
||||||
// poly = The list of 2D path points forming the perimeter of the polygon.
|
// poly = The list of 2D path points forming the perimeter of the polygon.
|
||||||
|
@ -2125,7 +2052,7 @@ function ccw_polygon(poly) =
|
||||||
// poly = The list of the path points for the perimeter of the polygon.
|
// poly = The list of the path points for the perimeter of the polygon.
|
||||||
function reverse_polygon(poly) =
|
function reverse_polygon(poly) =
|
||||||
assert(is_path(poly), "Input should be a polygon")
|
assert(is_path(poly), "Input should be a polygon")
|
||||||
let(lp=len(poly)) [for (i=idx(poly)) poly[(lp-i)%lp]];
|
[poly[0], for(i=[len(poly)-1:-1:1]) poly[i] ];
|
||||||
|
|
||||||
|
|
||||||
// Function: polygon_normal()
|
// Function: polygon_normal()
|
||||||
|
@ -2133,7 +2060,7 @@ function reverse_polygon(poly) =
|
||||||
// n = polygon_normal(poly);
|
// n = polygon_normal(poly);
|
||||||
// Description:
|
// Description:
|
||||||
// Given a 3D planar polygon, returns a unit-length normal vector for the
|
// Given a 3D planar polygon, returns a unit-length normal vector for the
|
||||||
// clockwise orientation of the polygon. If the polygon points are collinear, returns [].
|
// clockwise orientation of the polygon. If the polygon points are collinear, returns `undef`.
|
||||||
// It doesn't check for coplanarity.
|
// It doesn't check for coplanarity.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// poly = The list of 3D path points for the perimeter of the polygon.
|
// poly = The list of 3D path points for the perimeter of the polygon.
|
||||||
|
@ -2141,7 +2068,7 @@ function polygon_normal(poly) =
|
||||||
assert(is_path(poly,dim=3), "Invalid 3D polygon." )
|
assert(is_path(poly,dim=3), "Invalid 3D polygon." )
|
||||||
len(poly)==3 ? point3d(plane3pt(poly[0],poly[1],poly[2])) :
|
len(poly)==3 ? point3d(plane3pt(poly[0],poly[1],poly[2])) :
|
||||||
let( triple = sort(noncollinear_triple(poly,error=false)) )
|
let( triple = sort(noncollinear_triple(poly,error=false)) )
|
||||||
triple==[] ? [] :
|
triple==[] ? undef :
|
||||||
point3d(plane3pt(poly[triple[0]],poly[triple[1]],poly[triple[2]])) ;
|
point3d(plane3pt(poly[triple[0]],poly[triple[1]],poly[triple[2]])) ;
|
||||||
|
|
||||||
|
|
||||||
|
@ -2295,4 +2222,255 @@ function split_polygons_at_each_z(polys, zs, _i=0) =
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Section: Convex Sets
|
||||||
|
|
||||||
|
|
||||||
|
// Function: is_convex_polygon()
|
||||||
|
// Usage:
|
||||||
|
// is_convex_polygon(poly);
|
||||||
|
// Description:
|
||||||
|
// Returns true if the given 2D or 3D polygon is convex.
|
||||||
|
// The result is meaningless if the polygon is not simple (self-intersecting) or non coplanar.
|
||||||
|
// If the points are collinear an error is generated.
|
||||||
|
// Arguments:
|
||||||
|
// poly = Polygon to check.
|
||||||
|
// eps = Tolerance for the collinearity test. Default: EPSILON.
|
||||||
|
// Example:
|
||||||
|
// is_convex_polygon(circle(d=50)); // Returns: true
|
||||||
|
// is_convex_polygon(rot([50,120,30], p=path3d(circle(1,$fn=50)))); // Returns: true
|
||||||
|
// Example:
|
||||||
|
// spiral = [for (i=[0:36]) let(a=-i*10) (10+i)*[cos(a),sin(a)]];
|
||||||
|
// is_convex_polygon(spiral); // Returns: false
|
||||||
|
function is_convex_polygon(poly,eps=EPSILON) =
|
||||||
|
assert(is_path(poly), "The input should be a 2D or 3D polygon." )
|
||||||
|
let( lp = len(poly),
|
||||||
|
p0 = poly[0] )
|
||||||
|
assert( lp>=3 , "A polygon must have at least 3 points" )
|
||||||
|
let( crosses = [for(i=[0:1:lp-1]) cross(poly[(i+1)%lp]-poly[i], poly[(i+2)%lp]-poly[(i+1)%lp]) ] )
|
||||||
|
len(p0)==2
|
||||||
|
? assert( !approx(sqrt(max(max(crosses),-min(crosses))),eps), "The points are collinear" )
|
||||||
|
min(crosses) >=0 || max(crosses)<=0
|
||||||
|
: let( prod = crosses*sum(crosses),
|
||||||
|
minc = min(prod),
|
||||||
|
maxc = max(prod) )
|
||||||
|
assert( !approx(sqrt(max(maxc,-minc)),eps), "The points are collinear" )
|
||||||
|
minc>=0 || maxc<=0;
|
||||||
|
|
||||||
|
|
||||||
|
// Function: convex_distance()
|
||||||
|
// Usage:
|
||||||
|
// convex_distance(pts1, pts2,<eps=>);
|
||||||
|
// See also:
|
||||||
|
// convex_collision
|
||||||
|
// Description:
|
||||||
|
// Returns the smallest distance between a point in convex hull of `points1`
|
||||||
|
// and a point in the convex hull of `points2`. All the points in the lists
|
||||||
|
// should have the same dimension, either 2D or 3D.
|
||||||
|
// A zero result means the hulls intercept whithin a tolerance `eps`.
|
||||||
|
// Arguments:
|
||||||
|
// points1 - first list of 2d or 3d points.
|
||||||
|
// points2 - second list of 2d or 3d points.
|
||||||
|
// eps - tolerance in distance evaluations. Default: EPSILON.
|
||||||
|
// Example(2D):
|
||||||
|
// pts1 = move([-3,0], p=square(3,center=true));
|
||||||
|
// pts2 = rot(a=45, p=square(2,center=true));
|
||||||
|
// pts3 = [ [2,0], [1,2],[3,2], [3,-2], [1,-2] ];
|
||||||
|
// polygon(pts1);
|
||||||
|
// polygon(pts2);
|
||||||
|
// polygon(pts3);
|
||||||
|
// echo(convex_distance(pts1,pts2)); // Returns: 0.0857864
|
||||||
|
// echo(convex_distance(pts2,pts3)); // Returns: 0
|
||||||
|
// Example(3D):
|
||||||
|
// sphr1 = sphere(2,$fn=10);
|
||||||
|
// sphr2 = move([4,0,0], p=sphr1);
|
||||||
|
// sphr3 = move([4.5,0,0], p=sphr1);
|
||||||
|
// vnf_polyhedron(sphr1);
|
||||||
|
// vnf_polyhedron(sphr2);
|
||||||
|
// echo(convex_distance(sphr1[0], sphr2[0])); // Returns: 0
|
||||||
|
// echo(convex_distance(sphr1[0], sphr3[0])); // Returns: 0.5
|
||||||
|
function convex_distance(points1, points2, eps=EPSILON) =
|
||||||
|
assert(is_matrix(points1) && is_matrix(points2,undef,len(points1[0])),
|
||||||
|
"The input list should be a consistent non empty list of points of same dimension.")
|
||||||
|
assert(len(points1[0])==2 || len(points1[0])==3 ,
|
||||||
|
"The input points should be 2d or 3d points.")
|
||||||
|
let( d = points1[0]-points2[0] )
|
||||||
|
norm(d)<eps ? 0 :
|
||||||
|
let( v = _support_diff(points1,points2,-d) )
|
||||||
|
norm(_GJK_distance(points1, points2, eps, 0, v, [v]));
|
||||||
|
|
||||||
|
|
||||||
|
// Finds the vector difference between the hulls of the two pointsets by the GJK algorithm
|
||||||
|
// Based on:
|
||||||
|
// http://www.dtecta.com/papers/jgt98convex.pdf
|
||||||
|
function _GJK_distance(points1, points2, eps=EPSILON, lbd, d, simplex=[]) =
|
||||||
|
let( nrd = norm(d) ) // distance upper bound
|
||||||
|
nrd<eps ? d :
|
||||||
|
let(
|
||||||
|
v = _support_diff(points1,points2,-d),
|
||||||
|
lbd = max(lbd, d*v/nrd), // distance lower bound
|
||||||
|
close = (nrd-lbd <= eps*nrd)
|
||||||
|
)
|
||||||
|
// v already in the simplex is a degenerence due to numerical errors
|
||||||
|
// and may produce a non-stopping loop
|
||||||
|
close || [for(nv=norm(v), s=simplex) if(norm(s-v)<=eps*nv) 1]!=[] ? d :
|
||||||
|
let( newsplx = _closest_simplex(concat(simplex,[v]),eps) )
|
||||||
|
_GJK_distance(points1, points2, eps, lbd, newsplx[0], newsplx[1]);
|
||||||
|
|
||||||
|
|
||||||
|
// Function: convex_collision()
|
||||||
|
// Usage:
|
||||||
|
// convex_collision(pts1, pts2,<eps=>);
|
||||||
|
// See also:
|
||||||
|
// convex_distance
|
||||||
|
// Description:
|
||||||
|
// Returns `true` if the convex hull of `points1` intercepts the convex hull of `points2`
|
||||||
|
// otherwise, `false`.
|
||||||
|
// All the points in the lists should have the same dimension, either 2D or 3D.
|
||||||
|
// This function is tipically faster than `convex_distance` to find a non-collision.
|
||||||
|
// Arguments:
|
||||||
|
// points1 - first list of 2d or 3d points.
|
||||||
|
// points2 - second list of 2d or 3d points.
|
||||||
|
// eps - tolerance for the intersection tests. Default: EPSILON.
|
||||||
|
// Example(2D):
|
||||||
|
// pts1 = move([-3,0], p=square(3,center=true));
|
||||||
|
// pts2 = rot(a=45, p=square(2,center=true));
|
||||||
|
// pts3 = [ [2,0], [1,2],[3,2], [3,-2], [1,-2] ];
|
||||||
|
// polygon(pts1);
|
||||||
|
// polygon(pts2);
|
||||||
|
// polygon(pts3);
|
||||||
|
// echo(convex_collision(pts1,pts2)); // Returns: false
|
||||||
|
// echo(convex_collision(pts2,pts3)); // Returns: true
|
||||||
|
// Example(3D):
|
||||||
|
// sphr1 = sphere(2,$fn=10);
|
||||||
|
// sphr2 = move([4,0,0], p=sphr1);
|
||||||
|
// sphr3 = move([4.5,0,0], p=sphr1);
|
||||||
|
// vnf_polyhedron(sphr1);
|
||||||
|
// vnf_polyhedron(sphr2);
|
||||||
|
// echo(convex_collision(sphr1[0], sphr2[0])); // Returns: true
|
||||||
|
// echo(convex_collision(sphr1[0], sphr3[0])); // Returns: false
|
||||||
|
//
|
||||||
|
function convex_collision(points1, points2, eps=EPSILON) =
|
||||||
|
assert(is_matrix(points1) && is_matrix(points2,undef,len(points1[0])),
|
||||||
|
"The input list should be a consistent non empty list of points of same dimension.")
|
||||||
|
assert(len(points1[0])==2 || len(points1[0])==3 ,
|
||||||
|
"The input points should be 2d or 3d points.")
|
||||||
|
let( d = points1[0]-points2[0] )
|
||||||
|
norm(d)<eps ? true :
|
||||||
|
let( v = _support_diff(points1,points2,-d) )
|
||||||
|
_GJK_collide(points1, points2, v, [v], eps);
|
||||||
|
|
||||||
|
|
||||||
|
// Based on the GJK collision algorithms found in:
|
||||||
|
// http://uu.diva-portal.org/smash/get/diva2/FFULLTEXT01.pdf
|
||||||
|
// or
|
||||||
|
// http://www.dtecta.com/papers/jgt98convex.pdf
|
||||||
|
function _GJK_collide(points1, points2, d, simplex, eps=EPSILON) =
|
||||||
|
norm(d) < eps ? true : // does collide
|
||||||
|
let( v = _support_diff(points1,points2,-d) )
|
||||||
|
v*d > eps ? false : // no collision
|
||||||
|
let( newsplx = _closest_simplex(concat(simplex,[v]),eps) )
|
||||||
|
_GJK_collide(points1, points2, newsplx[0], newsplx[1], eps);
|
||||||
|
|
||||||
|
|
||||||
|
// given a simplex s, returns a pair:
|
||||||
|
// - the point of the s closest to the origin
|
||||||
|
// - the smallest sub-simplex of s that contains that point
|
||||||
|
function _closest_simplex(s,eps=EPSILON) =
|
||||||
|
assert(len(s)>=2 && len(s)<=4, "Internal error.")
|
||||||
|
len(s)==2 ? _closest_s1(s,eps) :
|
||||||
|
len(s)==3 ? _closest_s2(s,eps)
|
||||||
|
: _closest_s3(s,eps);
|
||||||
|
|
||||||
|
|
||||||
|
// find the closest to a 1-simplex
|
||||||
|
// Based on: http://uu.diva-portal.org/smash/get/diva2/FFULLTEXT01.pdf
|
||||||
|
function _closest_s1(s,eps=EPSILON) =
|
||||||
|
norm(s[1]-s[0])<eps*(norm(s[0])+norm(s[1]))/2 ? [ s[0], [s[0]] ] :
|
||||||
|
let(
|
||||||
|
c = s[1]-s[0],
|
||||||
|
t = -s[0]*c/(c*c)
|
||||||
|
)
|
||||||
|
t<0 ? [ s[0], [s[0]] ] :
|
||||||
|
t>1 ? [ s[1], [s[1]] ] :
|
||||||
|
[ s[0]+t*c, s ];
|
||||||
|
|
||||||
|
|
||||||
|
// find the closest to a 2-simplex
|
||||||
|
// Based on: http://uu.diva-portal.org/smash/get/diva2/FFULLTEXT01.pdf
|
||||||
|
function _closest_s2(s,eps=EPSILON) =
|
||||||
|
let(
|
||||||
|
dim = len(s[0]),
|
||||||
|
a = dim==3 ? s[0]: [ each s[0], 0] ,
|
||||||
|
b = dim==3 ? s[1]: [ each s[1], 0] ,
|
||||||
|
c = dim==3 ? s[2]: [ each s[2], 0] ,
|
||||||
|
ab = norm(a-b),
|
||||||
|
bc = norm(b-c),
|
||||||
|
ca = norm(c-a),
|
||||||
|
nr = cross(b-a,c-a)
|
||||||
|
)
|
||||||
|
norm(nr) <= eps*max(ab,bc,ca) // degenerate case
|
||||||
|
? let( i = max_index([ab, bc, ca]) )
|
||||||
|
_closest_s1([s[i],s[(i+1)%3]],eps)
|
||||||
|
// considering that s[2] was the last inserted vertex in s,
|
||||||
|
// the only possible outcomes are :
|
||||||
|
// s, [s[0],s[2]] and [s[1],s[2]]
|
||||||
|
: let(
|
||||||
|
class = (cross(nr,a-b)*a<0 ? 1 : 0 )
|
||||||
|
+ (cross(nr,c-a)*a<0 ? 2 : 0 )
|
||||||
|
+ (cross(nr,b-c)*b<0 ? 4 : 0 )
|
||||||
|
)
|
||||||
|
assert( class!=1, "Internal error" )
|
||||||
|
class==0 ? [ nr*(nr*a)/(nr*nr), s] : // origin projects (or is) on the tri
|
||||||
|
// class==1 ? _closest_s1([s[0],s[1]]) :
|
||||||
|
class==2 ? _closest_s1([s[0],s[2]],eps) :
|
||||||
|
class==4 ? _closest_s1([s[1],s[2]],eps) :
|
||||||
|
// class==3 ? a*(a-b)> 0 ? _closest_s1([s[0],s[1]]) : _closest_s1([s[0],s[2]]) :
|
||||||
|
class==3 ? _closest_s1([s[0],s[2]],eps) :
|
||||||
|
// class==5 ? b*(b-c)<=0 ? _closest_s1([s[0],s[1]]) : _closest_s1([s[1],s[2]]) :
|
||||||
|
class==5 ? _closest_s1([s[1],s[2]],eps) :
|
||||||
|
c*(c-a)>0 ? _closest_s1([s[0],s[2]],eps) : _closest_s1([s[1],s[2]],eps);
|
||||||
|
|
||||||
|
|
||||||
|
// find the closest to a 3-simplex
|
||||||
|
// it seems that degenerate 3-simplices are correctly manage without extra code
|
||||||
|
function _closest_s3(s,eps=EPSILON) =
|
||||||
|
assert( len(s[0])==3 && len(s)==4, "Internal error." )
|
||||||
|
let( nr = cross(s[1]-s[0],s[2]-s[0]),
|
||||||
|
sz = [ norm(s[1]-s[0]), norm(s[1]-s[2]), norm(s[2]-s[0]) ] )
|
||||||
|
norm(nr)<eps*max(sz)
|
||||||
|
? let( i = max_index(sz) )
|
||||||
|
_closest_s2([ s[i], s[(i+1)%3], s[3] ], eps) // degenerate case
|
||||||
|
// considering that s[3] was the last inserted vertex in s,
|
||||||
|
// the only possible outcomes will be:
|
||||||
|
// s or some of the 3 triangles of s containing s[3]
|
||||||
|
: let(
|
||||||
|
tris = [ [s[0], s[1], s[3]],
|
||||||
|
[s[1], s[2], s[3]],
|
||||||
|
[s[2], s[0], s[3]] ],
|
||||||
|
cntr = sum(s)/4,
|
||||||
|
// indicator of the tris facing the origin
|
||||||
|
facing = [for(i=[0:2])
|
||||||
|
let( nrm = _tri_normal(tris[i]) )
|
||||||
|
if( ((nrm*(s[i]-cntr))>0)==(nrm*s[i]<0) ) i ]
|
||||||
|
)
|
||||||
|
len(facing)==0 ? [ [0,0,0], s ] : // origin is inside the simplex
|
||||||
|
len(facing)==1 ? _closest_s2(tris[facing[0]], eps) :
|
||||||
|
let( // look for the origin-facing tri closest to the origin
|
||||||
|
closest = [for(i=facing) _closest_s2(tris[i], eps) ],
|
||||||
|
dist = [for(cl=closest) norm(cl[0]) ],
|
||||||
|
nearest = min_index(dist)
|
||||||
|
)
|
||||||
|
closest[nearest];
|
||||||
|
|
||||||
|
|
||||||
|
function _tri_normal(tri) = cross(tri[1]-tri[0],tri[2]-tri[0]);
|
||||||
|
|
||||||
|
|
||||||
|
function _support_diff(p1,p2,d) =
|
||||||
|
let( p1d = p1*d, p2d = p2*d )
|
||||||
|
p1[search(max(p1d),p1d,1)[0]] - p2[search(min(p2d),p2d,1)[0]];
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|
||||||
|
|
14
hull.scad
14
hull.scad
|
@ -202,17 +202,15 @@ function hull3d_faces(points) =
|
||||||
|
|
||||||
|
|
||||||
// Adds the remaining points one by one to the convex hull
|
// Adds the remaining points one by one to the convex hull
|
||||||
function _hull3d_iterative(points, triangles, planes, remaining, _i=0) =
|
function _hull3d_iterative(points, triangles, planes, remaining, _i=0) = //let( EPSILON=1e-12 )
|
||||||
_i >= len(remaining) ? triangles :
|
_i >= len(remaining) ? triangles :
|
||||||
let (
|
let (
|
||||||
// pick a point
|
// pick a point
|
||||||
i = remaining[_i],
|
i = remaining[_i],
|
||||||
// evaluate the triangle plane equations at point i
|
// evaluate the triangle plane equations at point i
|
||||||
// xx=[for(i=[0:len(planes)-1],p=[planes[i]]) if(len(p)!=4) echo(i=i,len(p))0 ],//echo([each points[i], -1]),
|
|
||||||
// planeq_val = [for(p=planes) p*[each points[i], -1]],
|
|
||||||
planeq_val = planes*[each points[i], -1],
|
planeq_val = planes*[each points[i], -1],
|
||||||
// find the triangles that are in conflict with the point (point not inside)
|
// find the triangles that are in conflict with the point (point not inside)
|
||||||
conflicts = [ for (i = [0:1:len(planes)-1]) if (planeq_val[i]>EPSILON) i ],
|
conflicts = [for (i = [0:1:len(planeq_val)-1]) if (planeq_val[i]>EPSILON) i ],
|
||||||
// collect the halfedges of all triangles that are in conflict
|
// collect the halfedges of all triangles that are in conflict
|
||||||
halfedges = [
|
halfedges = [
|
||||||
for(c = conflicts, i = [0:2])
|
for(c = conflicts, i = [0:2])
|
||||||
|
@ -222,18 +220,15 @@ function _hull3d_iterative(points, triangles, planes, remaining, _i=0) =
|
||||||
horizon = _remove_internal_edges(halfedges),
|
horizon = _remove_internal_edges(halfedges),
|
||||||
// generate new triangles connecting point i to each horizon halfedge vertices
|
// generate new triangles connecting point i to each horizon halfedge vertices
|
||||||
tri2add = [ for (h = horizon) concat(h,i) ],
|
tri2add = [ for (h = horizon) concat(h,i) ],
|
||||||
// w=[for(t=tri2add) if(collinear(points[t[0]],points[t[1]],points[t[2]])) echo(t)],
|
|
||||||
// add tria2add and remove conflict triangles
|
// add tria2add and remove conflict triangles
|
||||||
new_triangles =
|
new_triangles =
|
||||||
concat( tri2add,
|
concat( tri2add,
|
||||||
[ for (i = [0:1:len(planes)-1]) if (planeq_val[i]<=EPSILON) triangles[i] ]
|
[ for (i = [0:1:len(planes)-1]) if (planeq_val[i]<=EPSILON) triangles[i] ]
|
||||||
),
|
),
|
||||||
// y=[for(t=tri2add) if([]==plane3pt_indexed(points,t[0],t[1],t[2])) echo(tri2add=t,pts=[for(ti=t) points[ti]])],
|
|
||||||
// add the plane equations of new added triangles and remove the plane equations of the conflict ones
|
// add the plane equations of new added triangles and remove the plane equations of the conflict ones
|
||||||
new_planes =
|
new_planes =
|
||||||
concat( [ for (t = tri2add) plane3pt_indexed(points, t[0], t[1], t[2]) ],
|
[ for (t = tri2add) plane3pt_indexed(points, t[0], t[1], t[2]) ,
|
||||||
[ for (i = [0:1:len(planes)-1]) if (planeq_val[i]<=EPSILON) planes[i] ]
|
for (i = [0:1:len(planes)-1]) if (planeq_val[i]<=EPSILON) planes[i] ]
|
||||||
)
|
|
||||||
) _hull3d_iterative(
|
) _hull3d_iterative(
|
||||||
points,
|
points,
|
||||||
new_triangles,
|
new_triangles,
|
||||||
|
@ -249,7 +244,6 @@ function _remove_internal_edges(halfedges) = [
|
||||||
h
|
h
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
||||||
function _find_first_noncoplanar(plane, points, i=0) =
|
function _find_first_noncoplanar(plane, points, i=0) =
|
||||||
(i >= len(points) || !points_on_plane([points[i]],plane))? i :
|
(i >= len(points) || !points_on_plane([points[i]],plane))? i :
|
||||||
_find_first_noncoplanar(plane, points, i+1);
|
_find_first_noncoplanar(plane, points, i+1);
|
||||||
|
|
38
math.scad
38
math.scad
|
@ -640,17 +640,19 @@ function sum_of_sines(a, sines) =
|
||||||
// Usage:
|
// Usage:
|
||||||
// delts = deltas(v);
|
// delts = deltas(v);
|
||||||
// Description:
|
// Description:
|
||||||
// Returns a list with the deltas of adjacent entries in the given list.
|
// Returns a list with the deltas of adjacent entries in the given list, optionally wrapping back to the front.
|
||||||
// The list should be a consistent list of numeric components (numbers, vectors, matrix, etc).
|
// The list should be a consistent list of numeric components (numbers, vectors, matrix, etc).
|
||||||
// Given [a,b,c,d], returns [b-a,c-b,d-c].
|
// Given [a,b,c,d], returns [b-a,c-b,d-c].
|
||||||
|
//
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// v = The list to get the deltas of.
|
// v = The list to get the deltas of.
|
||||||
|
// wrap = If true, wrap back to the start from the end. ie: return the difference between the last and first items as the last delta. Default: false
|
||||||
// Example:
|
// Example:
|
||||||
// deltas([2,5,9,17]); // returns [3,4,8].
|
// deltas([2,5,9,17]); // returns [3,4,8].
|
||||||
// deltas([[1,2,3], [3,6,8], [4,8,11]]); // returns [[2,4,5], [1,2,3]]
|
// deltas([[1,2,3], [3,6,8], [4,8,11]]); // returns [[2,4,5], [1,2,3]]
|
||||||
function deltas(v) =
|
function deltas(v, wrap=false) =
|
||||||
assert( is_consistent(v) && len(v)>1 , "Inconsistent list or with length<=1.")
|
assert( is_consistent(v) && len(v)>1 , "Inconsistent list or with length<=1.")
|
||||||
[for (p=pair(v)) p[1]-p[0]] ;
|
[for (p=pair(v,wrap)) p[1]-p[0]] ;
|
||||||
|
|
||||||
|
|
||||||
// Function: product()
|
// Function: product()
|
||||||
|
@ -771,21 +773,31 @@ function _med3(a,b,c) =
|
||||||
// Usage:
|
// Usage:
|
||||||
// x = convolve(p,q);
|
// x = convolve(p,q);
|
||||||
// Description:
|
// Description:
|
||||||
// Given two vectors, finds the convolution of them.
|
// Given two vectors, or one vector and a path or
|
||||||
// The length of the returned vector is len(p)+len(q)-1 .
|
// two paths of the same dimension, finds the convolution of them.
|
||||||
|
// If both parameter are vectors, returns the vector convolution.
|
||||||
|
// If one parameter is a vector and the other a path,
|
||||||
|
// convolves using products by scalars and returns a path.
|
||||||
|
// If both parameters are paths, convolve using scalar products
|
||||||
|
// and returns a vector.
|
||||||
|
// The returned vector or path has length len(p)+len(q)-1.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// p = The first vector.
|
// p = The first vector or path.
|
||||||
// q = The second vector.
|
// q = The second vector or path.
|
||||||
// Example:
|
// Example:
|
||||||
// a = convolve([1,1],[1,2,1]); // Returns: [1,3,3,1]
|
// a = convolve([1,1],[1,2,1]); // Returns: [1,3,3,1]
|
||||||
// b = convolve([1,2,3],[1,2,1])); // Returns: [1,4,8,8,3]
|
// b = convolve([1,2,3],[1,2,1])); // Returns: [1,4,8,8,3]
|
||||||
|
// c = convolve([[1,1],[2,2],[3,1]],[1,2,1])); // Returns: [[1,1],[4,4],[8,6],[8,4],[3,1]]
|
||||||
|
// d = convolve([[1,1],[2,2],[3,1]],[[1,2],[2,1]])); // Returns: [3,9,11,7]
|
||||||
function convolve(p,q) =
|
function convolve(p,q) =
|
||||||
p==[] || q==[] ? [] :
|
p==[] || q==[] ? [] :
|
||||||
assert( is_vector(p) && is_vector(q), "The inputs should be vectors.")
|
assert( (is_vector(p) || is_matrix(p))
|
||||||
|
&& ( is_vector(q) || (is_matrix(q) && ( !is_vector(p[0]) || (len(p[0])==len(q[0])) ) ) ) ,
|
||||||
|
"The inputs should be vectors or paths all of the same dimension.")
|
||||||
let( n = len(p),
|
let( n = len(p),
|
||||||
m = len(q))
|
m = len(q))
|
||||||
[for(i=[0:n+m-2], k1 = max(0,i-n+1), k2 = min(i,m-1) )
|
[for(i=[0:n+m-2], k1 = max(0,i-n+1), k2 = min(i,m-1) )
|
||||||
[for(j=[k1:k2]) p[i-j] ] * [for(j=[k1:k2]) q[j] ]
|
sum([for(j=[k1:k2]) p[i-j]*q[j] ])
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
||||||
|
@ -1694,7 +1706,7 @@ function polynomial(p,z,k,total) =
|
||||||
// x = polymult(p,q)
|
// x = polymult(p,q)
|
||||||
// x = polymult([p1,p2,p3,...])
|
// x = polymult([p1,p2,p3,...])
|
||||||
// Description:
|
// Description:
|
||||||
// Given a list of polynomials represented as real coefficient lists, with the highest degree coefficient first,
|
// Given a list of polynomials represented as real algebraic coefficient lists, with the highest degree coefficient first,
|
||||||
// computes the coefficient list of the product polynomial.
|
// computes the coefficient list of the product polynomial.
|
||||||
function poly_mult(p,q) =
|
function poly_mult(p,q) =
|
||||||
is_undef(q) ?
|
is_undef(q) ?
|
||||||
|
@ -1714,8 +1726,8 @@ function poly_mult(p,q) =
|
||||||
// Description:
|
// Description:
|
||||||
// Computes division of the numerator polynomial by the denominator polynomial and returns
|
// Computes division of the numerator polynomial by the denominator polynomial and returns
|
||||||
// a list of two polynomials, [quotient, remainder]. If the division has no remainder then
|
// a list of two polynomials, [quotient, remainder]. If the division has no remainder then
|
||||||
// the zero polynomial [] is returned for the remainder. Similarly if the quotient is zero
|
// the zero polynomial [0] is returned for the remainder. Similarly if the quotient is zero
|
||||||
// the returned quotient will be [].
|
// the returned quotient will be [0].
|
||||||
function poly_div(n,d) =
|
function poly_div(n,d) =
|
||||||
assert( is_vector(n) && is_vector(d) , "Invalid polynomials." )
|
assert( is_vector(n) && is_vector(d) , "Invalid polynomials." )
|
||||||
let( d = _poly_trim(d),
|
let( d = _poly_trim(d),
|
||||||
|
@ -1740,7 +1752,7 @@ function _poly_div(n,d,q) =
|
||||||
/// _poly_trim(p,[eps])
|
/// _poly_trim(p,[eps])
|
||||||
/// Description:
|
/// Description:
|
||||||
/// Removes leading zero terms of a polynomial. By default zeros must be exact,
|
/// Removes leading zero terms of a polynomial. By default zeros must be exact,
|
||||||
/// or give epsilon for approximate zeros.
|
/// or give epsilon for approximate zeros. Returns [0] for a zero polynomial.
|
||||||
function _poly_trim(p,eps=0) =
|
function _poly_trim(p,eps=0) =
|
||||||
let( nz = [for(i=[0:1:len(p)-1]) if ( !approx(p[i],0,eps)) i])
|
let( nz = [for(i=[0:1:len(p)-1]) if ( !approx(p[i],0,eps)) i])
|
||||||
len(nz)==0 ? [0] : list_tail(p,nz[0]);
|
len(nz)==0 ? [0] : list_tail(p,nz[0]);
|
||||||
|
|
|
@ -997,7 +997,7 @@ module jittered_poly(path, dist=1/512) {
|
||||||
|
|
||||||
// Module: extrude_from_to()
|
// Module: extrude_from_to()
|
||||||
// Description:
|
// Description:
|
||||||
// Extrudes a 2D shape between the points pt1 and pt2. Takes as children a set of 2D shapes to extrude.
|
// Extrudes a 2D shape between the 3d points pt1 and pt2. Takes as children a set of 2D shapes to extrude.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
// pt1 = starting point of extrusion.
|
// pt1 = starting point of extrusion.
|
||||||
// pt2 = ending point of extrusion.
|
// pt2 = ending point of extrusion.
|
||||||
|
@ -1010,6 +1010,7 @@ module jittered_poly(path, dist=1/512) {
|
||||||
// xcopies(3) circle(3, $fn=32);
|
// xcopies(3) circle(3, $fn=32);
|
||||||
// }
|
// }
|
||||||
module extrude_from_to(pt1, pt2, convexity, twist, scale, slices) {
|
module extrude_from_to(pt1, pt2, convexity, twist, scale, slices) {
|
||||||
|
assert( is_path([pt1,pt2],3), "The points should be 3d points");
|
||||||
rtp = xyz_to_spherical(pt2-pt1);
|
rtp = xyz_to_spherical(pt2-pt1);
|
||||||
translate(pt1) {
|
translate(pt1) {
|
||||||
rotate([0, rtp[2], rtp[1]]) {
|
rotate([0, rtp[2], rtp[1]]) {
|
||||||
|
|
|
@ -358,11 +358,21 @@ module test_sortidx() {
|
||||||
}
|
}
|
||||||
test_sortidx();
|
test_sortidx();
|
||||||
|
|
||||||
|
module test_group_sort() {
|
||||||
|
assert_equal(group_sort([]), [[]]);
|
||||||
|
assert_equal(group_sort([8]), [[8]]);
|
||||||
|
assert_equal(group_sort([7,3,9,4,3,1,8]), [[1], [3, 3], [4], [7], [8], [9]]);
|
||||||
|
assert_equal(group_sort([[5,"a"],[2,"b"], [5,"c"], [3,"d"], [2,"e"] ], idx=0), [[[2, "b"], [2, "e"]], [[3, "d"]], [[5, "a"], [5, "c"]]]);
|
||||||
|
assert_equal(group_sort([["a",5],["b",6], ["c",1], ["d",2], ["e",6] ], idx=1), [[["c", 1]], [["d", 2]], [["a", 5]], [["b", 6], ["e", 6]]] );
|
||||||
|
}
|
||||||
|
test_group_sort();
|
||||||
|
|
||||||
|
|
||||||
module test_unique() {
|
module test_unique() {
|
||||||
assert(unique([]) == []);
|
assert_equal(unique([]), []);
|
||||||
assert(unique([8]) == [8]);
|
assert_equal(unique([8]), [8]);
|
||||||
assert(unique([7,3,9,4,3,1,8]) == [1,3,4,7,8,9]);
|
assert_equal(unique([7,3,9,4,3,1,8]), [1,3,4,7,8,9]);
|
||||||
|
assert_equal(unique(["A","B","R","A","C","A","D","A","B","R","A"]), ["A", "B", "C", "D", "R"]);
|
||||||
}
|
}
|
||||||
test_unique();
|
test_unique();
|
||||||
|
|
||||||
|
|
|
@ -90,7 +90,8 @@ test_cleanup_path();
|
||||||
test_simplify_path();
|
test_simplify_path();
|
||||||
test_simplify_path_indexed();
|
test_simplify_path_indexed();
|
||||||
test_is_region();
|
test_is_region();
|
||||||
|
test_convex_distance();
|
||||||
|
test_convex_collision();
|
||||||
|
|
||||||
// to be used when there are two alternative symmetrical outcomes
|
// to be used when there are two alternative symmetrical outcomes
|
||||||
// from a function like a plane output; v must be a vector
|
// from a function like a plane output; v must be a vector
|
||||||
|
@ -1075,6 +1076,44 @@ module test_is_region() {
|
||||||
}
|
}
|
||||||
*test_is_region();
|
*test_is_region();
|
||||||
|
|
||||||
|
module test_convex_distance() {
|
||||||
|
// 2D
|
||||||
|
c1 = circle(10,$fn=24);
|
||||||
|
c2 = move([15,0], p=c1);
|
||||||
|
assert(convex_distance(c1, c2)==0);
|
||||||
|
c3 = move([22,0],c1);
|
||||||
|
assert_approx(convex_distance(c1, c3),2);
|
||||||
|
// 3D
|
||||||
|
s1 = sphere(10,$fn=4);
|
||||||
|
s2 = move([15,0], p=s1);
|
||||||
|
assert_approx(convex_distance(s1[0], s2[0]), 0.857864376269);
|
||||||
|
s3 = move([25.3,0],s1);
|
||||||
|
assert_approx(convex_distance(s1[0], s3[0]), 11.1578643763);
|
||||||
|
s4 = move([30,25],s1);
|
||||||
|
assert_approx(convex_distance(s1[0], s4[0]), 28.8908729653);
|
||||||
|
s5 = move([10*sqrt(2),0],s1);
|
||||||
|
assert_approx(convex_distance(s1[0], s5[0]), 0);
|
||||||
|
}
|
||||||
|
*test_convex_distance();
|
||||||
|
|
||||||
|
module test_convex_collision() {
|
||||||
|
// 2D
|
||||||
|
c1 = circle(10,$fn=24);
|
||||||
|
c2 = move([15,0], p=c1);
|
||||||
|
assert(convex_collision(c1, c2));
|
||||||
|
c3 = move([22,0],c1);
|
||||||
|
assert(!convex_collision(c1, c3));
|
||||||
|
// 3D
|
||||||
|
s1 = sphere(10,$fn=4);
|
||||||
|
s2 = move([15,0], p=s1);
|
||||||
|
assert(!convex_collision(s1[0], s2[0]));
|
||||||
|
s3 = move([25.3,0],s1);
|
||||||
|
assert(!convex_collision(s1[0], s3[0]));
|
||||||
|
s4 = move([5,0],s1);
|
||||||
|
assert(convex_collision(s1[0], s4[0]));
|
||||||
|
s5 = move([10*sqrt(2),0],s1);
|
||||||
|
assert(convex_collision(s1[0], s5[0]));
|
||||||
|
}
|
||||||
|
*test_convex_distance();
|
||||||
|
|
||||||
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|
||||||
|
|
|
@ -546,7 +546,9 @@ test_sum_of_sines();
|
||||||
|
|
||||||
module test_deltas() {
|
module test_deltas() {
|
||||||
assert_equal(deltas([2,5,9,17]), [3,4,8]);
|
assert_equal(deltas([2,5,9,17]), [3,4,8]);
|
||||||
|
assert_equal(deltas([2,5,9,17],wrap=true), [3,4,8,-15]);
|
||||||
assert_equal(deltas([[1,2,3], [3,6,8], [4,8,11]]), [[2,4,5], [1,2,3]]);
|
assert_equal(deltas([[1,2,3], [3,6,8], [4,8,11]]), [[2,4,5], [1,2,3]]);
|
||||||
|
assert_equal(deltas([[1,2,3], [3,6,8], [4,8,11]],wrap=true), [[2,4,5], [1,2,3], [-3,-6,-8]]);
|
||||||
}
|
}
|
||||||
test_deltas();
|
test_deltas();
|
||||||
|
|
||||||
|
@ -582,6 +584,10 @@ module test_convolve() {
|
||||||
assert_equal(convolve([1,1],[]), []);
|
assert_equal(convolve([1,1],[]), []);
|
||||||
assert_equal(convolve([1,1],[1,2,1]), [1,3,3,1]);
|
assert_equal(convolve([1,1],[1,2,1]), [1,3,3,1]);
|
||||||
assert_equal(convolve([1,2,3],[1,2,1]), [1,4,8,8,3]);
|
assert_equal(convolve([1,2,3],[1,2,1]), [1,4,8,8,3]);
|
||||||
|
assert_equal(convolve([1,2,3],[[1],[2],[1]]), [[1], [4], [8], [8], [3]]);
|
||||||
|
assert_equal(convolve([[1],[2],[3]],[[1],[2],[1]]), [1,4,8,8,3]);
|
||||||
|
assert_equal(convolve([[1,0],[2,1],[3,2]],[[1,0],[2,1],[1,2]]), [1,4,9,12,7]);
|
||||||
|
assert_equal(convolve([1,2,3],[[1,0],[2,1],[1,2]]), [[1,0],[4,1],[8,4],[8,7],[3,6]]);
|
||||||
}
|
}
|
||||||
test_convolve();
|
test_convolve();
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue