Added enumerate() and sortidx()

This commit is contained in:
Revar Desmera 2019-04-04 20:27:01 -07:00
parent 3d4b7e9ebb
commit 670c47ca86

View file

@ -110,10 +110,10 @@ function posmod(x,m) = (x%m+m)%m;
// m = Modulo value. // m = Modulo value.
// step = Step by this amount. // step = Step by this amount.
// Examples: // Examples:
// echo(modrange(90,270,360, step=45)); // Outputs [90,135,180,225,270] // modrange(90,270,360, step=45); // Outputs [90,135,180,225,270]
// echo(modrange(270,90,360, step=45)); // Outputs [270,315,0,45,90] // modrange(270,90,360, step=45); // Outputs [270,315,0,45,90]
// echo(modrange(90,270,360, step=-45)); // Outputs [90,45,0,315,270] // modrange(90,270,360, step=-45); // Outputs [90,45,0,315,270]
// echo(modrange(270,90,360, step=-45)); // Outputs [270,225,180,135,90] // modrange(270,90,360, step=-45); // Outputs [270,225,180,135,90]
function modrange(x, y, m, step=1) = function modrange(x, y, m, step=1) =
let( let(
a = posmod(x, m), a = posmod(x, m),
@ -373,6 +373,7 @@ function count_true(l, nmax=undef, i=0, cnt=0) =
// Section: List/Array Operations // Section: List/Array Operations
// Function: cdr() // Function: cdr()
// Status: DEPRECATED, use `slice(list,1,-1)` instead. // Status: DEPRECATED, use `slice(list,1,-1)` instead.
// Description: Returns all but the first item of a given array. // Description: Returns all but the first item of a given array.
@ -381,7 +382,6 @@ function count_true(l, nmax=undef, i=0, cnt=0) =
function cdr(list) = len(list)<=1? [] : [for (i=[1:len(list)-1]) list[i]]; function cdr(list) = len(list)<=1? [] : [for (i=[1:len(list)-1]) list[i]];
// Function: replist() // Function: replist()
// Usage: // Usage:
// replist(val, n) // replist(val, n)
@ -416,7 +416,6 @@ function replist(val, n, i=0) =
function in_list(x,l,idx=undef) = search([x], l, num_returns_per_match=1, index_col_num=idx) != [[]]; function in_list(x,l,idx=undef) = search([x], l, num_returns_per_match=1, index_col_num=idx) != [[]];
// Function: slice() // Function: slice()
// Description: // Description:
// Returns a slice of a list. The first item is index 0. // Returns a slice of a list. The first item is index 0.
@ -437,7 +436,6 @@ function slice(arr,st,end) = let(
) (s==e)? [] : [for (i=[s:e-1]) if (e>s) arr[i]]; ) (s==e)? [] : [for (i=[s:e-1]) if (e>s) arr[i]];
// Function: wrap_range() // Function: wrap_range()
// Status: DEPRECATED, use `select()` instead. // Status: DEPRECATED, use `select()` instead.
// Description: // Description:
@ -494,7 +492,6 @@ function select(list, start, end=undef) =
); );
// Function: reverse() // Function: reverse()
// Description: Reverses a list/array. // Description: Reverses a list/array.
// Arguments: // Arguments:
@ -601,6 +598,24 @@ function array_trim(v, maxlen) = maxlen<1? [] : [for (i=[0:min(len(v),maxlen)-1]
function array_fit(v, length, fill) = let(l=len(v)) (l==length)? v : (l>length)? array_trim(v,length) : array_pad(v,length,fill); function array_fit(v, length, fill) = let(l=len(v)) (l==length)? v : (l>length)? array_trim(v,length) : array_pad(v,length,fill);
// Function: enumerate()
// Description:
// Returns a list, with each item of the given list `l` numbered in a sublist.
// Something like: `[[0,l[0]], [1,l[1]], [2,l[2]], ...]`
// Arguments:
// l = List to enumerate.
// idx = If given, enumerates just the given subindex items of `l`.
// Example:
// enumerate(["a","b","c"]); // Returns: [[0,"a"], [1,"b"], [2,"c"]]
// enumerate([[88,"a"],[76,"b"],[21,"c"]], idx=1); // Returns: [[0,"a"], [1,"b"], [2,"c"]]
// enumerate([["cat","a",12],["dog","b",10],["log","c",14]], idx=[1:2]); // Returns: [[0,"a",12], [1,"b",10], [2,"c",14]]
function enumerate(l,idx=undef) =
(l==[])? [] :
(idx==undef)?
[for (i=[0:len(l)-1]) [i,l[i]]] :
[for (i=[0:len(l)-1]) concat([i], [for (j=idx) l[i][j]])];
// Function: array_zip() // Function: array_zip()
// Usage: // Usage:
// array_zip(v1, v2, v3, [fit], [fill]); // array_zip(v1, v2, v3, [fit], [fill]);
@ -694,6 +709,37 @@ function sort(arr, idx=undef) =
concat(sort(lesser,idx), equal, sort(greater,idx)); concat(sort(lesser,idx), equal, sort(greater,idx));
// Function: sortidx()
// Description:
// Given a list, calculates the sort order of the list, and returns
// a list of indexes into the original list in that sorted order.
// If you iterate the returned list in order, and use the list items
// to index into the original list, you will be iterating the original
// values in sorted order.
// Example:
// lst = ["d","b","e","c"];
// idxs = sortidx(lst); // Returns: [1,3,0,2]
// ordered = [for (i=idxs) lst[i]]; // Returns: ["b", "c", "d", "e"]
// Example:
// lst = [
// ["foo", 88, [0,0,1], false],
// ["bar", 90, [0,1,0], true],
// ["baz", 89, [1,0,0], false],
// ["qux", 23, [1,1,1], true]
// ];
// idxs1 = sortidx(lst, idx=1); // Returns: [3,0,2,1]
// idxs2 = sortidx(lst, idx=0); // Returns: [1,2,0,3]
// idxs3 = sortidx(lst, idx=[1,3]); // Returns: [3,0,2,1]
function sortidx(l, idx=undef) =
let(ll=enumerate(l,idx=idx))
array_subindex(
sort(ll, idx=(
idx==undef? 1 :
(ll==[])? 1 :
[1:len(ll[0])-1]
)),
0);
// Function: unique() // Function: unique()
// Usage: // Usage:
@ -965,7 +1011,6 @@ function rotate_points3d(pts, v=0, cp=[0,0,0], axis=undef, from=undef, to=undef,
ang = vector_angle(from, to), ang = vector_angle(from, to),
axis = vector_axis(from, to) axis = vector_axis(from, to)
) )
echo(axis=axis, ang=ang, v=v)
matrix4_rot_by_axis(axis, ang) * matrix4_rot_by_axis(from, v) matrix4_rot_by_axis(axis, ang) * matrix4_rot_by_axis(from, v)
) : is_def(axis)? ( ) : is_def(axis)? (
matrix4_rot_by_axis(axis, v) matrix4_rot_by_axis(axis, v)