Merge pull request #74 from adrianVmariano/master

Added repeat to turtle.  Added centroid to geometry.scad.
This commit is contained in:
Revar Desmera 2019-07-15 19:11:26 -07:00 committed by GitHub
commit aee7e4d9bd
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 46 additions and 11 deletions

View file

@ -430,6 +430,18 @@ function polygon_area(vertices) =
0.5*sum([for(i=[0:len(vertices)-1]) det2(select(vertices,i,i+1))]); 0.5*sum([for(i=[0:len(vertices)-1]) det2(select(vertices,i,i+1))]);
// Function: centroid()
// Usage:
// centroid(vertices)
// Description:
// Given a simple polygon, returns the coordinates of the polygon's centroid.
// If the polygon is self-intersecting, the results are undefined.
function centroid(vertices) =
sum([for(i=[0:len(vertices)-1])
let(segment=select(vertices,i,i+1))
det2(segment)*sum(segment)]) / 6 / polygon_area(vertices);
// Function: assemble_path_fragments() // Function: assemble_path_fragments()
// Usage: // Usage:
// assemble_path_fragments(subpaths); // assemble_path_fragments(subpaths);

View file

@ -568,11 +568,13 @@ module supershape(step=0.5,m1=4,m2=undef,n1,n2=undef,n3=undef,a=1,b=undef, r=und
// - "length", length: Change the turtle move distance to `length` // - "length", length: Change the turtle move distance to `length`
// - "scale", factor: Multiply turtle move distance by `factor` // - "scale", factor: Multiply turtle move distance by `factor`
// - "addlength", length: Add `length` to the turtle move distance // - "addlength", length: Add `length` to the turtle move distance
// - "repeat", count, comands: Repeats a list of commands `count` times.
// //
// Arguments: // Arguments:
// commands = list of turtle commands // commands = list of turtle commands
// state = starting turtle state (from previous call) or starting point. Default: start at the origin // state = starting turtle state (from previous call) or starting point. Default: start at the origin
// full_state = if true return the full turtle state for continuing the path in subsequent turtle calls. Default: false // full_state = if true return the full turtle state for continuing the path in subsequent turtle calls. Default: false
// repeat = number of times to repeat the command list. Default: 1
// //
// Example(2D): Simple rectangle // Example(2D): Simple rectangle
// path = turtle(["xmove",3, "ymove", "xmove",-3, "ymove",-1]); // path = turtle(["xmove",3, "ymove", "xmove",-3, "ymove",-1]);
@ -580,8 +582,14 @@ module supershape(step=0.5,m1=4,m2=undef,n1,n2=undef,n3=undef,a=1,b=undef, r=und
// Example(2D): Pentagon // Example(2D): Pentagon
// path=turtle(["angle",360/5,"move","turn","move","turn","move","turn","move"]); // path=turtle(["angle",360/5,"move","turn","move","turn","move","turn","move"]);
// stroke(path,width=.1,closed=true); // stroke(path,width=.1,closed=true);
// Example(2D): Pentagon using the repeat argument
// path=turtle(["move","turn",360/5],repeat=5);
// stroke(path,width=.1,closed=true);
// Example(2D): Pentagon using the repeat turtle command, setting the turn angle
// path=turtle(["angle",360/5,"repeat",5,["move","turn"]]);
// stroke(path,width=.1,closed=true);
// Example(2D): Pentagram // Example(2D): Pentagram
// path = turtle(flatten(replist(["move","left",144],10))); // path = turtle(["move","left",144], repeat=10);
// stroke(path,width=.05); // stroke(path,width=.05);
// Example(2D): Sawtooth path // Example(2D): Sawtooth path
// path = turtle([ // path = turtle([
@ -611,16 +619,16 @@ module supershape(step=0.5,m1=4,m2=undef,n1,n2=undef,n3=undef,a=1,b=undef, r=und
// ]); // ]);
// stroke(path, width=.1); // stroke(path, width=.1);
// Example(2DMed): square spiral // Example(2DMed): square spiral
// path = turtle(flatten(replist(["move","left","addlength",1],50))); // path = turtle(["move","left","addlength",1],repeat=50);
// stroke(path,width=.2); // stroke(path,width=.2);
// Example(2DMed): pentagonal spiral // Example(2DMed): pentagonal spiral
// path = turtle(concat(["angle",360/5],flatten(replist(["move","left","addlength",1],50)))); // path = turtle(["move","left",360/5,"addlength",1],repeat=50);
// stroke(path,width=.2); // stroke(path,width=.2);
// Example(2DMed): yet another spiral // Example(2DMed): yet another spiral, without using `repeat`
// path = turtle(concat(["angle",71],flatten(replist(["move","left","addlength",1],50)))); // path = turtle(concat(["angle",71],flatten(replist(["move","left","addlength",1],50))));
// stroke(path,width=.2); // stroke(path,width=.2);
// Example(2DMed): The previous spiral grows linearly and eventually intersects itself. This one grows geometrically and does not. // Example(2DMed): The previous spiral grows linearly and eventually intersects itself. This one grows geometrically and does not.
// path = turtle(concat(["angle",71],flatten(replist(["move","left","scale",1.05],50)))); // path = turtle(["move","left",71,"scale",1.05],repeat=50);
// stroke(path,width=.05); // stroke(path,width=.05);
// Example(2D): Koch Snowflake // Example(2D): Koch Snowflake
// function koch_unit(depth) = // function koch_unit(depth) =
@ -634,24 +642,39 @@ module supershape(step=0.5,m1=4,m2=undef,n1,n2=undef,n3=undef,a=1,b=undef, r=und
// ["right"], // ["right"],
// koch_unit(depth-1) // koch_unit(depth-1)
// ); // );
// koch=concat(["angle",60],flatten(replist(concat(koch_unit(3),["left","left"]),3))); // koch=concat(["angle",60,"repeat",3],[concat(koch_unit(3),["left","left"])]);
// polygon(turtle(koch)); // polygon(turtle(koch));
function turtle(commands, state=[[[0,0]],[1,0],90], full_state=false) = function turtle(commands, state=[[[0,0]],[1,0],90], full_state=false, repeat=1) =
let( state = is_vector(state) ? [[state],[1,0],90] : state ) let( state = is_vector(state) ? [[state],[1,0],90] : state )
_turtle(commands,state,full_state); repeat == 1 ? _turtle(commands,state,full_state)
: _turtle_repeat(commands, state, full_state, repeat);
function _turtle_repeat(commands, state, full_state, repeat) =
repeat==1 ? _turtle(commands,state,full_state)
: _turtle_repeat(commands, _turtle(commands, state, true), full_state, repeat-1);
function _turtle_command_len(commands, index) =
commands[index] == "repeat" ? 3 : // Repeat command requires 2 args
is_string(commands[index+1]) ? 1 // If 2nd item is a string it's must be a new command
: 2; // Otherwise we have command and arg
function _turtle(commands, state, full_state, index=0) = function _turtle(commands, state, full_state, index=0) =
index < len(commands) ? index < len(commands) ?
_turtle(commands, _turtle(commands,
turtle_command(commands[index],commands[index+1],state,index), _turtle_command(commands[index],commands[index+1],commands[index+2],state,index),
full_state, full_state,
index+(!is_string(commands[index+1])?2:1) index+_turtle_command_len(commands,index)
) : ) :
( full_state ? state : state[0] ); ( full_state ? state : state[0] );
// Turtle state: state = [path, step_vector, default angle] // Turtle state: state = [path, step_vector, default angle]
function turtle_command(command, parm, state, index) = function _turtle_command(command, parm, parm2, state, index) =
command == "repeat" ? assert(is_num(parm),str("\"repeat\" command requires a numeric parameter at index ",index))
assert(is_list(parm2),str("\"repeat\" command requires a command list parameter at index ",index))
_turtle_repeat(parm2, state, true, parm)
:
let( let(
path = 0, path = 0,
step=1, step=1,