reorder functions in array.scad

list_{de,in}creasing -> is_{de,in}creasing, add strict option to both
This commit is contained in:
Adrian Mariano 2021-10-16 23:01:52 -04:00
parent d6576da79e
commit f7b08f1b9d
9 changed files with 341 additions and 284 deletions

View file

@ -53,36 +53,34 @@ function _same_type(a,b, depth) =
&& []==[for(i=idx(a)) if( ! _same_type(a[i],b[i],depth-1) ) 0] ); && []==[for(i=idx(a)) if( ! _same_type(a[i],b[i],depth-1) ) 0] );
// Function: min_length()
// Function: list_shortest()
// Usage: // Usage:
// llen = list_shortest(array); // llen = min_length(array);
// Topics: List Handling // Topics: List Handling
// See Also: list_longest() // See Also: max_length()
// Description: // Description:
// Returns the length of the shortest sublist in a list of lists. // Returns the length of the shortest sublist in a list of lists.
// Arguments: // Arguments:
// array = A list of lists. // array = A list of lists.
// Example: // Example:
// slen = list_shortest([[3,4,5],[6,7,8,9]]); // Returns: 3 // slen = min_length([[3,4,5],[6,7,8,9]]); // Returns: 3
function list_shortest(array) = function min_length(array) =
assert(is_list(array), "Invalid input." ) assert(is_list(array), "Invalid input." )
min([for (v = array) len(v)]); min([for (v = array) len(v)]);
// Function: list_longest() // Function: max_length()
// Usage: // Usage:
// llen = list_longest(array); // llen = max_length(array);
// Topics: List Handling // Topics: List Handling
// See Also: list_shortest() // See Also: min_length()
// Description: // Description:
// Returns the length of the longest sublist in a list of lists. // Returns the length of the longest sublist in a list of lists.
// Arguments: // Arguments:
// array = A list of lists. // array = A list of lists.
// Example: // Example:
// llen = list_longest([[3,4,5],[6,7,8,9]]); // Returns: 4 // llen = max_length([[3,4,5],[6,7,8,9]]); // Returns: 4
function list_longest(array) = function max_length(array) =
assert(is_list(array), "Invalid input." ) assert(is_list(array), "Invalid input." )
max([for (v = array) len(v)]); max([for (v = array) len(v)]);
@ -111,71 +109,6 @@ function in_list(val,list,idx) =
: val==list[s][idx]; : val==list[s][idx];
// Function: find_first_match()
// Topics: List Handling
// See Also: in_list()
// Usage:
// idx = find_first_match(val, list, [start=], [eps=]);
// indices = find_first_match(val, list, all=true, [start=], [eps=]);
// Description:
// Finds the first item in `list` that matches `val`, returning the index.
// Arguments:
// val = The value to search for. If given a function literal of signature `function (x)`, uses that function to check list items. Returns true for a match.
// list = The list to search through.
// ---
// start = The index to start searching from.
// all = If true, returns a list of all matching item indices.
// eps = The maximum allowed floating point rounding error for numeric comparisons.
function find_first_match(val, list, start=0, all=false, eps=EPSILON) =
all? [
for (i=[start:1:len(list)-1])
if (
(!is_func(val) && approx(val, list[i], eps=eps)) ||
(is_func(val) && val(list[i]))
) i
] :
__find_first_match(val, list, eps=eps, i=start);
function __find_first_match(val, list, eps, i=0) =
i >= len(list)? undef :
(
(!is_func(val) && approx(val, list[i], eps=eps)) ||
(is_func(val) && val(list[i]))
)? i : __find_first_match(val, list, eps=eps, i=i+1);
// Function: list_increasing()
// Usage:
// bool = list_increasing(list);
// Topics: List Handling
// See Also: max_index(), min_index(), list_decreasing()
// Description:
// Returns true if the list is (non-strictly) increasing
// Example:
// a = list_increasing([1,2,3,4]); // Returns: true
// b = list_increasing([1,3,2,4]); // Returns: false
// c = list_increasing([4,3,2,1]); // Returns: false
function list_increasing(list) =
assert(is_list(list)||is_string(list))
len([for (p=pair(list)) if(p.x>p.y) true])==0;
// Function: list_decreasing()
// Usage:
// bool = list_decreasing(list);
// Topics: List Handling
// See Also: max_index(), min_index(), list_increasing()
// Description:
// Returns true if the list is (non-strictly) decreasing
// Example:
// a = list_decreasing([1,2,3,4]); // Returns: false
// b = list_decreasing([4,2,3,1]); // Returns: false
// c = list_decreasing([4,3,2,1]); // Returns: true
function list_decreasing(list) =
assert(is_list(list)||is_string(list))
len([for (p=pair(list)) if(p.x<p.y) true])==0;
// Function: add_scalar() // Function: add_scalar()
// Usage: // Usage:
// v = add_scalar(v, s); // v = add_scalar(v, s);
@ -194,6 +127,110 @@ function add_scalar(v,s) =
// Section: Operations using approx()
// Function: deduplicate()
// Usage:
// list = deduplicate(list, [close], [eps]);
// Topics: List Handling
// See Also: deduplicate_indexed()
// Description:
// Removes consecutive duplicate items in a list.
// When `eps` is zero, the comparison between consecutive items is exact.
// Otherwise, when all list items and subitems are numbers, the comparison is within the tolerance `eps`.
// This is different from `unique()` in that the list is *not* sorted.
// Arguments:
// list = The list to deduplicate.
// closed = If true, drops trailing items if they match the first list item.
// eps = The maximum tolerance between items.
// Example:
// a = deduplicate([8,3,4,4,4,8,2,3,3,8,8]); // Returns: [8,3,4,8,2,3,8]
// b = deduplicate(closed=true, [8,3,4,4,4,8,2,3,3,8,8]); // Returns: [8,3,4,8,2,3]
// c = deduplicate("Hello"); // Returns: "Helo"
// d = deduplicate([[3,4],[7,2],[7,1.99],[1,4]],eps=0.1); // Returns: [[3,4],[7,2],[1,4]]
// e = deduplicate([[7,undef],[7,undef],[1,4],[1,4+1e-12]],eps=0); // Returns: [[7,undef],[1,4],[1,4+1e-12]]
function deduplicate(list, closed=false, eps=EPSILON) =
assert(is_list(list)||is_string(list))
let(
l = len(list),
end = l-(closed?0:1)
)
is_string(list) ? str_join([for (i=[0:1:l-1]) if (i==end || list[i] != list[(i+1)%l]) list[i]]) :
eps==0 ? [for (i=[0:1:l-1]) if (i==end || list[i] != list[(i+1)%l]) list[i]] :
[for (i=[0:1:l-1]) if (i==end || !approx(list[i], list[(i+1)%l], eps)) list[i]];
// Function: deduplicate_indexed()
// Usage:
// new_idxs = deduplicate_indexed(list, indices, [closed], [eps]);
// Topics: List Handling
// See Also: deduplicate()
// Description:
// Given a list, and indices into it, removes consecutive indices that
// index to the same values in the list.
// Arguments:
// list = The list that the indices index into.
// indices = The list of indices to deduplicate.
// closed = If true, drops trailing indices if what they index matches what the first index indexes.
// eps = The maximum difference to allow between numbers or vectors.
// Example:
// a = deduplicate_indexed([8,6,4,6,3], [1,4,3,1,2,2,0,1]); // Returns: [1,4,3,2,0,1]
// b = deduplicate_indexed([8,6,4,6,3], [1,4,3,1,2,2,0,1], closed=true); // Returns: [1,4,3,2,0]
// c = deduplicate_indexed([[7,undef],[7,undef],[1,4],[1,4],[1,4+1e-12]],eps=0); // Returns: [0,2,4]
function deduplicate_indexed(list, indices, closed=false, eps=EPSILON) =
assert(is_list(list)||is_string(list), "Improper list or string.")
indices==[]? [] :
assert(is_vector(indices), "Indices must be a list of numbers.")
let(
ll = len(list),
l = len(indices),
end = l-(closed?0:1)
) [
for (i = [0:1:l-1]) let(
idx1 = indices[i],
idx2 = indices[(i+1)%l],
a = assert(idx1>=0,"Bad index.")
assert(idx1<len(list),"Bad index in indices.")
list[idx1],
b = assert(idx2>=0,"Bad index.")
assert(idx2<len(list),"Bad index in indices.")
list[idx2],
eq = (a == b)? true :
(a*0 != b*0) || (eps==0)? false :
is_num(a) || is_vector(a) ? approx(a, b, eps=eps)
: false
)
if (i==end || !eq) indices[i]
];
// Function: find_approx()
// Topics: List Handling
// See Also: in_list()
// Usage:
// idx = find_approx(val, list, [start=], [eps=]);
// indices = find_approx(val, list, all=true, [start=], [eps=]);
// Description:
// Finds the first item in `list` that matches `val`, returning the index.
// Arguments:
// val = The value to search for. If given a function literal of signature `function (x)`, uses that function to check list items. Returns true for a match.
// list = The list to search through.
// ---
// start = The index to start searching from.
// all = If true, returns a list of all matching item indices.
// eps = The maximum allowed floating point rounding error for numeric comparisons.
function find_approx(val, list, start=0, all=false, eps=EPSILON) =
all ? [for (i=[start:1:len(list)-1]) if (approx(val, list[i], eps=eps)) i]
: __find_approx(val, list, eps=eps, i=start);
function __find_approx(val, list, eps, i=0) =
i >= len(list)? undef :
approx(val, list[i], eps=eps)
? i
: __find_approx(val, list, eps=eps, i=i+1);
// Section: List Indexing // Section: List Indexing
@ -373,49 +410,6 @@ function bselect(array,index) =
// Section: List Construction // Section: List Construction
// Function: list()
// Topics: List Handling, Type Conversion
// Usage:
// list = list(l)
// Description:
// Expands a range into a full list. If given a list, returns it verbatim.
// If given a string, explodes it into a list of single letters.
// Arguments:
// l = The value to expand.
// See Also: scalar_vec3(), force_list()
// Example:
// l1 = list([3:2:9]); // Returns: [3,5,7,9]
// l2 = list([3,4,5]); // Returns: [3,4,5]
// l3 = list("Foo"); // Returns: ["F","o","o"]
// l4 = list(23); // Returns: [23]
function list(l) = is_list(l)? l : [for (x=l) x];
// Function: force_list()
// Usage:
// list = force_list(value, [n], [fill]);
// Topics: List Handling
// See Also: scalar_vec3()
// Description:
// Coerces non-list values into a list. Makes it easy to treat a scalar input
// consistently as a singleton list, as well as list inputs.
// - If `value` is a list, then that list is returned verbatim.
// - If `value` is not a list, and `fill` is not given, then a list of `n` copies of `value` will be returned.
// - If `value` is not a list, and `fill` is given, then a list `n` items long will be returned where `value` will be the first item, and the rest will contain the value of `fill`.
// Arguments:
// value = The value or list to coerce into a list.
// n = The number of items in the coerced list. Default: 1
// fill = The value to pad the coerced list with, after the firt value. Default: undef (pad with copies of `value`)
// Example:
// x = force_list([3,4,5]); // Returns: [3,4,5]
// y = force_list(5); // Returns: [5]
// z = force_list(7, n=3); // Returns: [7,7,7]
// w = force_list(4, n=3, fill=1); // Returns: [4,1,1]
function force_list(value, n=1, fill) =
is_list(value) ? value :
is_undef(fill)? [for (i=[1:1:n]) value] : [value, for (i=[2:1:n]) fill];
// Function: repeat() // Function: repeat()
// Usage: // Usage:
// list = repeat(val, n); // list = repeat(val, n);
@ -493,6 +487,52 @@ function list_bset(indexset, valuelist, dflt=0) =
// Function: list()
// Topics: List Handling, Type Conversion
// Usage:
// list = list(l)
// Description:
// Expands a range into a full list. If given a list, returns it verbatim.
// If given a string, explodes it into a list of single letters.
// Arguments:
// l = The value to expand.
// See Also: scalar_vec3(), force_list()
// Example:
// l1 = list([3:2:9]); // Returns: [3,5,7,9]
// l2 = list([3,4,5]); // Returns: [3,4,5]
// l3 = list("Foo"); // Returns: ["F","o","o"]
// l4 = list(23); // Returns: [23]
function list(l) = is_list(l)? l : [for (x=l) x];
// Function: force_list()
// Usage:
// list = force_list(value, [n], [fill]);
// Topics: List Handling
// See Also: scalar_vec3()
// Description:
// Coerces non-list values into a list. Makes it easy to treat a scalar input
// consistently as a singleton list, as well as list inputs.
// - If `value` is a list, then that list is returned verbatim.
// - If `value` is not a list, and `fill` is not given, then a list of `n` copies of `value` will be returned.
// - If `value` is not a list, and `fill` is given, then a list `n` items long will be returned where `value` will be the first item, and the rest will contain the value of `fill`.
// Arguments:
// value = The value or list to coerce into a list.
// n = The number of items in the coerced list. Default: 1
// fill = The value to pad the coerced list with, after the firt value. Default: undef (pad with copies of `value`)
// Example:
// x = force_list([3,4,5]); // Returns: [3,4,5]
// y = force_list(5); // Returns: [5]
// z = force_list(7, n=3); // Returns: [7,7,7]
// w = force_list(4, n=3, fill=1); // Returns: [4,1,1]
function force_list(value, n=1, fill) =
is_list(value) ? value :
is_undef(fill)? [for (i=[1:1:n]) value] : [value, for (i=[2:1:n]) fill];
// Section: List Modification // Section: List Modification
@ -549,79 +589,38 @@ function list_rotate(list,n=1) =
is_string(list)? str_join(elems) : elems; is_string(list)? str_join(elems) : elems;
// Function: deduplicate()
// Usage:
// list = deduplicate(list, [close], [eps]);
// Topics: List Handling
// See Also: deduplicate_indexed()
// Description:
// Removes consecutive duplicate items in a list.
// When `eps` is zero, the comparison between consecutive items is exact.
// Otherwise, when all list items and subitems are numbers, the comparison is within the tolerance `eps`.
// This is different from `unique()` in that the list is *not* sorted.
// Arguments:
// list = The list to deduplicate.
// closed = If true, drops trailing items if they match the first list item.
// eps = The maximum tolerance between items.
// Example:
// a = deduplicate([8,3,4,4,4,8,2,3,3,8,8]); // Returns: [8,3,4,8,2,3,8]
// b = deduplicate(closed=true, [8,3,4,4,4,8,2,3,3,8,8]); // Returns: [8,3,4,8,2,3]
// c = deduplicate("Hello"); // Returns: "Helo"
// d = deduplicate([[3,4],[7,2],[7,1.99],[1,4]],eps=0.1); // Returns: [[3,4],[7,2],[1,4]]
// e = deduplicate([[7,undef],[7,undef],[1,4],[1,4+1e-12]],eps=0); // Returns: [[7,undef],[1,4],[1,4+1e-12]]
function deduplicate(list, closed=false, eps=EPSILON) =
assert(is_list(list)||is_string(list))
let(
l = len(list),
end = l-(closed?0:1)
)
is_string(list) ? str_join([for (i=[0:1:l-1]) if (i==end || list[i] != list[(i+1)%l]) list[i]]) :
eps==0 ? [for (i=[0:1:l-1]) if (i==end || list[i] != list[(i+1)%l]) list[i]] :
[for (i=[0:1:l-1]) if (i==end || !approx(list[i], list[(i+1)%l], eps)) list[i]];
// Function: shuffle()
// Function: deduplicate_indexed()
// Usage: // Usage:
// new_idxs = deduplicate_indexed(list, indices, [closed], [eps]); // shuffled = shuffle(list, [seed]);
// Topics: List Handling // Topics: List Handling
// See Also: deduplicate() // See Also: sort(), sortidx(), unique(), unique_count()
// Description: // Description:
// Given a list, and indices into it, removes consecutive indices that // Shuffles the input list into random order.
// index to the same values in the list. // If given a string, shuffles the characters within the string.
// If you give a numeric seed value then the permutation
// will be repeatable.
// Arguments: // Arguments:
// list = The list that the indices index into. // list = The list to shuffle.
// indices = The list of indices to deduplicate. // seed = Optional random number seed for the shuffling.
// closed = If true, drops trailing indices if what they index matches what the first index indexes.
// eps = The maximum difference to allow between numbers or vectors.
// Example: // Example:
// a = deduplicate_indexed([8,6,4,6,3], [1,4,3,1,2,2,0,1]); // Returns: [1,4,3,2,0,1] // // Spades Hearts Diamonds Clubs
// b = deduplicate_indexed([8,6,4,6,3], [1,4,3,1,2,2,0,1], closed=true); // Returns: [1,4,3,2,0] // suits = ["\u2660", "\u2661", "\u2662", "\u2663"];
// c = deduplicate_indexed([[7,undef],[7,undef],[1,4],[1,4],[1,4+1e-12]],eps=0); // Returns: [0,2,4] // ranks = [2,3,4,5,6,7,8,9,10,"J","Q","K","A"];
function deduplicate_indexed(list, indices, closed=false, eps=EPSILON) = // cards = [for (suit=suits, rank=ranks) str(rank,suit)];
assert(is_list(list)||is_string(list), "Improper list or string.") // deck = shuffle(cards);
indices==[]? [] : function shuffle(list,seed) =
assert(is_vector(indices), "Indices must be a list of numbers.") assert(is_list(list)||is_string(list), "Invalid input." )
is_string(list)? str_join(shuffle([for (x = list) x],seed=seed)) :
len(list)<=1 ? list :
let( let(
ll = len(list), rval = is_num(seed) ? rands(0,1,len(list),seed_value=seed)
l = len(indices), : rands(0,1,len(list)),
end = l-(closed?0:1) left = [for (i=[0:len(list)-1]) if (rval[i]< 0.5) list[i]],
) [ right = [for (i=[0:len(list)-1]) if (rval[i]>=0.5) list[i]]
for (i = [0:1:l-1]) let(
idx1 = indices[i],
idx2 = indices[(i+1)%l],
a = assert(idx1>=0,"Bad index.")
assert(idx1<len(list),"Bad index in indices.")
list[idx1],
b = assert(idx2>=0,"Bad index.")
assert(idx2<len(list),"Bad index in indices.")
list[idx2],
eq = (a == b)? true :
(a*0 != b*0) || (eps==0)? false :
is_num(a) || is_vector(a) ? approx(a, b, eps=eps)
: false
) )
if (i==end || !eq) indices[i] concat(shuffle(left), shuffle(right));
];
// Function: repeat_entries() // Function: repeat_entries()
@ -875,7 +874,7 @@ function list_fit(array, length, fill) =
: list_pad(array,length,fill); : list_pad(array,length,fill);
// Section: List Shuffling and Sorting // Section: Sorting
// returns true for valid index specifications idx in the interval [imin, imax) // returns true for valid index specifications idx in the interval [imin, imax)
@ -893,38 +892,6 @@ function _valid_idx(idx,imin,imax) =
&& ( is_undef(imin) || (idx[1]>0 && idx[0]>=imin ) || (idx[1]<0 && idx[0]<=imax ) ) && ( is_undef(imin) || (idx[1]>0 && idx[0]>=imin ) || (idx[1]<0 && idx[0]<=imax ) )
&& ( is_undef(imax) || (idx[1]>0 && idx[2]<=imax ) || (idx[1]<0 && idx[2]>=imin ) ) ); && ( is_undef(imax) || (idx[1]>0 && idx[2]<=imax ) || (idx[1]<0 && idx[2]>=imin ) ) );
// Function: shuffle()
// Usage:
// shuffled = shuffle(list, [seed]);
// Topics: List Handling
// See Also: sort(), sortidx(), unique(), unique_count()
// Description:
// Shuffles the input list into random order.
// If given a string, shuffles the characters within the string.
// If you give a numeric seed value then the permutation
// will be repeatable.
// Arguments:
// list = The list to shuffle.
// seed = Optional random number seed for the shuffling.
// Example:
// // Spades Hearts Diamonds Clubs
// suits = ["\u2660", "\u2661", "\u2662", "\u2663"];
// ranks = [2,3,4,5,6,7,8,9,10,"J","Q","K","A"];
// cards = [for (suit=suits, rank=ranks) str(rank,suit)];
// deck = shuffle(cards);
function shuffle(list,seed) =
assert(is_list(list)||is_string(list), "Invalid input." )
is_string(list)? str_join(shuffle([for (x = list) x],seed=seed)) :
len(list)<=1 ? list :
let(
rval = is_num(seed) ? rands(0,1,len(list),seed_value=seed)
: rands(0,1,len(list)),
left = [for (i=[0:len(list)-1]) if (rval[i]< 0.5) list[i]],
right = [for (i=[0:len(list)-1]) if (rval[i]>=0.5) list[i]]
)
concat(shuffle(left), shuffle(right));
// idx should be an index of the arrays l[i] // idx should be an index of the arrays l[i]
function _group_sort_by_index(l,idx) = function _group_sort_by_index(l,idx) =
len(l) == 0 ? [] : len(l) == 0 ? [] :
@ -1134,31 +1101,51 @@ function sortidx(list, idx=undef) =
: _sort_general(list,idx,indexed=true); : _sort_general(list,idx,indexed=true);
// Function: group_sort()
// Function: is_increasing()
// Usage: // Usage:
// ulist = group_sort(list); // bool = is_increasing(list);
// Topics: List Handling // Topics: List Handling
// See Also: shuffle(), sort(), sortidx(), unique(), unique_count() // See Also: max_index(), min_index(), is_decreasing()
// Description: // Description:
// Given a list of values, returns the sorted list with all repeated items grouped in a list. // Returns true if the list is (non-strictly) increasing, or strictly increasing if strict is set to true.
// When the list entries are themselves lists, the sorting may be done based on the `idx` entry // The list can be a list of any items that OpenSCAD can compare, or it can be a string which will be
// of those entries, that should be numbers. // evaluated character by character.
// The result is always a list of lists.
// Arguments: // Arguments:
// list = The list to sort. // list = list (or string) to check
// idx = If given, do the comparison based just on the specified index. Default: zero. // strict = set to true to test that list is strictly increasing
// Example: // Example:
// sorted = group_sort([5,2,8,3,1,3,8,7,5]); // Returns: [[1],[2],[3,3],[5,5],[7],[8,8]] // a = is_increasing([1,2,3,4]); // Returns: true
// 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"]] ] // b = is_increasing([1,3,2,4]); // Returns: false
function group_sort(list, idx) = // c = is_increasing([1,3,3,4]); // Returns: true
assert(is_list(list), "Input should be a list." ) // d = is_increasing([1,3,3,4],strict=true); // Returns: false
assert(is_undef(idx) || (is_finite(idx) && idx>=0) , "Invalid index." ) // e = is_increasing([4,3,2,1]); // Returns: false
len(list)<=1 ? [list] : function is_increasing(list,strict=false) =
is_vector(list)? _group_sort(list) : assert(is_list(list)||is_string(list))
let( idx = is_undef(idx) ? 0 : idx ) strict ? len([for (p=pair(list)) if(p.x>=p.y) true])==0
assert( [for(entry=list) if(!is_list(entry) || len(entry)<idx || !is_num(entry[idx]) ) 1]==[], : len([for (p=pair(list)) if(p.x>p.y) true])==0;
"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: is_decreasing()
// Usage:
// bool = is_decreasing(list);
// Topics: List Handling
// See Also: max_index(), min_index(), is_increasing()
// Description:
// Returns true if the list is (non-strictly) decreasing, or strictly decreasing if strict is set to true.
// The list can be a list of any items that OpenSCAD can compare, or it can be a string which will be
// evaluated character by character.
// Arguments:
// list = list (or string) to check
// strict = set to true to test that list is strictly decreasing
// Example:
// a = is_decreasing([1,2,3,4]); // Returns: false
// b = is_decreasing([4,2,3,1]); // Returns: false
// c = is_decreasing([4,3,2,1]); // Returns: true
function is_decreasing(list,strict=false) =
assert(is_list(list)||is_string(list))
strict ? len([for (p=pair(list)) if(p.x<=p.y) true])==0
: len([for (p=pair(list)) if(p.x<p.y) true])==0;
// Function: unique() // Function: unique()
@ -1228,6 +1215,35 @@ function unique_count(list) =
[ select(list,ind), deltas( concat(ind,[len(list)]) ) ]; [ select(list,ind), deltas( concat(ind,[len(list)]) ) ];
// 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: group_data() // Function: group_data()
// Usage: // Usage:
// groupings = group_data(groups, values); // groupings = group_data(groups, values);
@ -1453,7 +1469,7 @@ function permutations(l,n=2) =
// // ECHO: [7,3] // // ECHO: [7,3]
function zip(a,b,c) = function zip(a,b,c) =
b!=undef? zip([a,b,if (c!=undef) c]) : b!=undef? zip([a,b,if (c!=undef) c]) :
let(n = list_shortest(a)) let(n = min_length(a))
[for (i=[0:1:n-1]) [for (x=a) x[i]]]; [for (i=[0:1:n-1]) [for (x=a) x[i]]];
@ -1483,7 +1499,7 @@ function zip(a,b,c) =
// // ECHO: [6,88]] // // ECHO: [6,88]]
function zip_long(a,b,c,fill) = function zip_long(a,b,c,fill) =
b!=undef? zip_long([a,b,if (c!=undef) c],fill=fill) : b!=undef? zip_long([a,b,if (c!=undef) c],fill=fill) :
let(n = list_longest(a)) let(n = max_length(a))
[for (i=[0:1:n-1]) [for (x=a) i<len(x)? x[i] : fill]]; [for (i=[0:1:n-1]) [for (x=a) i<len(x)? x[i] : fill]];
@ -1680,8 +1696,8 @@ function hstack(M1, M2, M3) =
(M2!=undef)? hstack([M1,M2]) : (M2!=undef)? hstack([M1,M2]) :
assert(all([for(v=M1) is_list(v)]), "One of the inputs to hstack is not a list") assert(all([for(v=M1) is_list(v)]), "One of the inputs to hstack is not a list")
let( let(
minlen = list_shortest(M1), minlen = min_length(M1),
maxlen = list_longest(M1) maxlen = max_length(M1)
) )
assert(minlen==maxlen, "Input vectors to hstack must have the same length") assert(minlen==maxlen, "Input vectors to hstack must have the same length")
[for(row=[0:1:minlen-1]) [for(row=[0:1:minlen-1])

View file

@ -1997,7 +1997,7 @@ function are_polygons_equal(poly1, poly2, eps=EPSILON) =
l1 = len(poly1), l1 = len(poly1),
l2 = len(poly2) l2 = len(poly2)
) l1 != l2 ? false : ) l1 != l2 ? false :
let( maybes = find_first_match(poly1[0], poly2, eps=eps, all=true) ) let( maybes = find_approx(poly1[0], poly2, eps=eps, all=true) )
maybes == []? false : maybes == []? false :
[for (i=maybes) if (_are_polygons_equal(poly1, poly2, eps, i)) 1] != []; [for (i=maybes) if (_are_polygons_equal(poly1, poly2, eps, i)) 1] != [];

View file

@ -787,7 +787,7 @@ function _path_cut_points(path, dists, closed=false, direction=false) =
assert(long_enough,len(path)<2 ? "Two points needed to define a path" : "Closed path must include three points") assert(long_enough,len(path)<2 ? "Two points needed to define a path" : "Closed path must include three points")
is_num(dists) ? _path_cut_points(path, [dists],closed, direction)[0] : is_num(dists) ? _path_cut_points(path, [dists],closed, direction)[0] :
assert(is_vector(dists)) assert(is_vector(dists))
assert(list_increasing(dists), "Cut distances must be an increasing list") assert(is_increasing(dists), "Cut distances must be an increasing list")
let(cuts = _path_cut_points_recurse(path,dists,closed)) let(cuts = _path_cut_points_recurse(path,dists,closed))
!direction !direction
? cuts ? cuts

View file

@ -413,7 +413,7 @@ function skin(profiles, slices, refine=1, method="direct", sampling, caps, close
slices = is_list(slices) ? slices : repeat(slices, profcount), slices = is_list(slices) ? slices : repeat(slices, profcount),
refineOK = [for(i=idx(refine)) if (refine[i]<=0 || !is_integer(refine[i])) i], refineOK = [for(i=idx(refine)) if (refine[i]<=0 || !is_integer(refine[i])) i],
slicesOK = [for(i=idx(slices)) if (!is_integer(slices[i]) || slices[i]<0) i], slicesOK = [for(i=idx(slices)) if (!is_integer(slices[i]) || slices[i]<0) i],
maxsize = list_longest(profiles), maxsize = max_length(profiles),
methodok = is_list(method) || in_list(method, legal_methods), methodok = is_list(method) || in_list(method, legal_methods),
methodlistok = is_list(method) ? [for(i=idx(method)) if (!in_list(method[i], legal_methods)) i] : [], methodlistok = is_list(method) ? [for(i=idx(method)) if (!in_list(method[i], legal_methods)) i] : [],
method = is_string(method) ? repeat(method, profcount) : method, method = is_string(method) ? repeat(method, profcount) : method,
@ -1156,7 +1156,7 @@ module sweep(shape, transforms, closed=false, caps, style="min_edge", convexity=
// closed = the first and last profile are connected. Default: false // closed = the first and last profile are connected. Default: false
function subdivide_and_slice(profiles, slices, numpoints, method="length", closed=false) = function subdivide_and_slice(profiles, slices, numpoints, method="length", closed=false) =
let( let(
maxsize = list_longest(profiles), maxsize = max_length(profiles),
numpoints = is_undef(numpoints) ? maxsize : numpoints = is_undef(numpoints) ? maxsize :
numpoints == "lcm" ? lcmlist([for(p=profiles) len(p)]) : numpoints == "lcm" ? lcmlist([for(p=profiles) len(p)]) :
is_num(numpoints) ? round(numpoints) : undef is_num(numpoints) ? round(numpoints) : undef

View file

@ -91,35 +91,53 @@ module test_in_list() {
test_in_list(); test_in_list();
module test_min_index() { module test_is_increasing() {
assert(min_index([5,3,9,6,2,7,8,2,1])==8); assert(is_increasing([1,2,3,4]) == true);
assert(min_index([5,3,9,6,2,7,8,2,7],all=true)==[4,7]); assert(is_increasing([1,2,2,2]) == true);
assert(is_increasing([1,3,2,4]) == false);
assert(is_increasing([4,3,2,1]) == false);
assert(is_increasing([1,2,3,4],strict=true) == true);
assert(is_increasing([1,2,2,2],strict=true) == false);
assert(is_increasing([1,3,2,4],strict=true) == false);
assert(is_increasing([4,3,2,1],strict=true) == false);
assert(is_increasing(["AB","BC","DF"]) == true);
assert(is_increasing(["AB","DC","CF"]) == false);
assert(is_increasing([[1,2],[1,4],[2,3],[2,2]])==false);
assert(is_increasing([[1,2],[1,4],[2,3],[2,3]])==true);
assert(is_increasing([[1,2],[1,4],[2,3],[2,3]],strict=true)==false);
assert(is_increasing("ABCFZ")==true);
assert(is_increasing("ZYWRA")==false);
} }
test_min_index(); test_is_increasing();
module test_max_index() { module test_is_decreasing() {
assert(max_index([5,3,9,6,2,7,8,9,1])==2); assert(is_decreasing([1,2,3,4]) == false);
assert(max_index([5,3,9,6,2,7,8,9,7],all=true)==[2,7]); assert(is_decreasing([4,2,3,1]) == false);
assert(is_decreasing([4,2,2,1]) == true);
assert(is_decreasing([4,3,2,1]) == true);
assert(is_decreasing([1,2,3,4],strict=true) == false);
assert(is_decreasing([4,2,3,1],strict=true) == false);
assert(is_decreasing([4,2,2,1],strict=true) == false);
assert(is_decreasing([4,3,2,1],strict=true) == true);
assert(is_decreasing(reverse(["AB","BC","DF"])) == true);
assert(is_decreasing(reverse(["AB","DC","CF"])) == false);
assert(is_decreasing(reverse([[1,2],[1,4],[2,3],[2,2]]))==false);
assert(is_decreasing(reverse([[1,2],[1,4],[2,3],[2,3]]))==true);
assert(is_decreasing(reverse([[1,2],[1,4],[2,3],[2,3]]),strict=true)==false);
assert(is_decreasing("ABCFZ")==false);
assert(is_decreasing("ZYWRA")==true);
} }
test_max_index(); test_is_decreasing();
module test_list_increasing() { module test_find_approx() {
assert(list_increasing([1,2,3,4]) == true); assert(find_approx(1, [2,3,1.05,4,1,2,.99], eps=.1)==2);
assert(list_increasing([1,3,2,4]) == false); assert(find_approx(1, [2,3,1.05,4,1,2,.99], all=true, eps=.1)==[2,4,6]);
assert(list_increasing([4,3,2,1]) == false);
} }
test_list_increasing(); test_find_approx();
module test_list_decreasing() {
assert(list_decreasing([1,2,3,4]) == false);
assert(list_decreasing([4,2,3,1]) == false);
assert(list_decreasing([4,3,2,1]) == true);
}
test_list_decreasing();
// Section: Basic List Generation // Section: Basic List Generation
@ -244,16 +262,16 @@ module test_list_bset() {
test_list_bset(); test_list_bset();
module test_list_shortest() { module test_min_length() {
assert(list_shortest(["foobar", "bazquxx", "abcd"]) == 4); assert(min_length(["foobar", "bazquxx", "abcd"]) == 4);
} }
test_list_shortest(); test_min_length();
module test_list_longest() { module test_max_length() {
assert(list_longest(["foobar", "bazquxx", "abcd"]) == 7); assert(max_length(["foobar", "bazquxx", "abcd"]) == 7);
} }
test_list_longest(); test_max_length();
module test_list_pad() { module test_list_pad() {

View file

@ -1223,7 +1223,7 @@ module test_qr_factor() {
is_ut(qr[1]) is_ut(qr[1])
&& approx(qr[0]*transpose(qr[0]), ident(len(qr[0]))) && approx(qr[0]*transpose(qr[0]), ident(len(qr[0])))
&& approx(qr[0]*qr[1]*transpose(qr[2]),M) && approx(qr[0]*qr[1]*transpose(qr[2]),M)
&& list_decreasing([for(i=[0:1:min(len(qr[1]),len(qr[1][0]))-1]) abs(qr[1][i][i])]); && is_decreasing([for(i=[0:1:min(len(qr[1]),len(qr[1][0]))-1]) abs(qr[1][i][i])]);
M = [[1,2,9,4,5], M = [[1,2,9,4,5],

View file

@ -208,6 +208,23 @@ module test_vector_nearest(){
test_vector_nearest(); test_vector_nearest();
module test_min_index() {
assert(min_index([5,3,9,6,2,7,8,2,1])==8);
assert(min_index([5,3,9,6,2,7,8,2,7],all=true)==[4,7]);
}
test_min_index();
module test_max_index() {
assert(max_index([5,3,9,6,2,7,8,9,1])==2);
assert(max_index([5,3,9,6,2,7,8,9,7],all=true)==[2,7]);
}
test_max_index();
module test_pointlist_bounds() { module test_pointlist_bounds() {
pts = [ pts = [
[-53,27,12], [-53,27,12],

View file

@ -268,7 +268,7 @@ function vector_axis(v1,v2=undef,v3=undef) =
// idx = min_index(vals); // idx = min_index(vals);
// idxlist = min_index(vals, all=true); // idxlist = min_index(vals, all=true);
// Topics: List Handling // Topics: List Handling
// See Also: max_index(), list_increasing(), list_decreasing() // See Also: max_index(), is_increasing(), is_decreasing()
// Description: // Description:
// Returns the index of the first occurrence of the minimum value in the given list. // Returns the index of the first occurrence of the minimum value in the given list.
// If `all` is true then returns a list of all indices where the minimum value occurs. // If `all` is true then returns a list of all indices where the minimum value occurs.
@ -288,7 +288,7 @@ function min_index(vals, all=false) =
// idx = max_index(vals); // idx = max_index(vals);
// idxlist = max_index(vals, all=true); // idxlist = max_index(vals, all=true);
// Topics: List Handling // Topics: List Handling
// See Also: min_index(), list_increasing(), list_decreasing() // See Also: min_index(), is_increasing(), is_decreasing()
// Description: // Description:
// Returns the index of the first occurrence of the maximum value in the given list. // Returns the index of the first occurrence of the maximum value in the given list.
// If `all` is true then returns a list of all indices where the maximum value occurs. // If `all` is true then returns a list of all indices where the maximum value occurs.

View file

@ -407,7 +407,7 @@ function _cleave_connected_region(region) =
// Usage: // Usage:
// vnf = vnf_from_region(region, [transform], [reverse], [vnf]); // vnf = vnf_from_region(region, [transform], [reverse], [vnf]);
// Description: // Description:
// Given a (two-dimensional) region, applies the given transformation matrix to it and makes a triangulated VNF of // Given a (two-dimensional) region, applies the given transformation matrix to it and makes a (three-dimensional) triangulated VNF of
// faces for that region, reversed if desired. // faces for that region, reversed if desired.
// Arguments: // Arguments:
// region = The region to conver to a vnf. // region = The region to conver to a vnf.
@ -529,7 +529,9 @@ function _link_indicator(l,imin,imax) =
// vnf2 = vnf_triangulate(vnf); // vnf2 = vnf_triangulate(vnf);
// Description: // Description:
// Triangulates faces in the VNF that have more than 3 vertices. // Triangulates faces in the VNF that have more than 3 vertices.
// Example: // Arguments:
// vnf = vnf to triangulate
// Example(3D):
// include <BOSL2/polyhedra.scad> // include <BOSL2/polyhedra.scad>
// vnf = zrot(33,regular_polyhedron_info("vnf", "dodecahedron", side=12)); // vnf = zrot(33,regular_polyhedron_info("vnf", "dodecahedron", side=12));
// vnf_polyhedron(vnf); // vnf_polyhedron(vnf);
@ -549,8 +551,12 @@ function vnf_triangulate(vnf) =
// sliced = vnf_slice(vnf, dir, cuts); // sliced = vnf_slice(vnf, dir, cuts);
// Description: // Description:
// Slice the faces of a VNF along a specified axis direction at a given list // Slice the faces of a VNF along a specified axis direction at a given list
// of cut points. You can use this to refine the faces of a VNF before applying // of cut points. The cut points can appear in any order. You can use this to refine the faces of a VNF before applying
// a nonlinear transformation to its vertex set. // a nonlinear transformation to its vertex set.
// Arguments:
// vnf = vnf to slice
// dir = normal direction to the slices, either "X", "Y" or "Z"
// cuts = X, Y or Z values where cuts occur
// Example(3D): // Example(3D):
// include <BOSL2/polyhedra.scad> // include <BOSL2/polyhedra.scad>
// vnf = regular_polyhedron_info("vnf", "dodecahedron", side=12); // vnf = regular_polyhedron_info("vnf", "dodecahedron", side=12);