mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-01 09:49:45 +00:00
Added repeat to turtle. Added centroid to geometry.scad.
This commit is contained in:
parent
81fd73588f
commit
a71868431b
2 changed files with 47 additions and 11 deletions
|
@ -385,6 +385,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);
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
include<BOSL2/std.scad>
|
||||||
//////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////
|
||||||
// LibFile: shapes2d.scad
|
// LibFile: shapes2d.scad
|
||||||
// Common useful 2D shapes.
|
// Common useful 2D shapes.
|
||||||
|
@ -568,11 +569,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 +583,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 +620,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 +643,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,
|
||||||
|
|
Loading…
Reference in a new issue