3-D L-Systems

Three iterations of generic 3D L-system:

generic L-system

3D Hilbert (space-filling) curve:

3-D Hilbert curve, blocky version

Line version of the same thing:

3-D Hilbert curve, line version

Extruded 2D Hilbert curve:

2-D Hilbert curve, blocky version

The ancient code, such as it is:

Off[General::spell1, General::spell];

(* the 3D rotation matrices *)

RotMatPsi[angle_] :=
    {{ Cos[angle], Sin[angle], 0},
     {-Sin[angle], Cos[angle], 0},
     {0,           0,          1}};

RotMatPsiII[angle_] :=
   {{Cos[angle], -Sin[angle], 0},
    {Sin[angle],  Cos[angle], 0},
    {0,           0,          1}};

RotMatTheta[angle_] :=
    {{ Cos[angle], 0, Sin[angle]},
     { 0,          1, 0         },
     {-Sin[angle], 0, Cos[angle]}};

RotMatThetaII[angle_] :=
   {{Cos[angle], 0, -Sin[angle]},
    {0,          1,  0         },
    {Sin[angle], 0,  Cos[angle]}};

RotMatPhi[angle_] :=
   {{1, 0,          0         },
    {0, Cos[angle], Sin[angle]},
    {0,-Sin[angle], Cos[angle]}};

RotMatPhiII[angle_] :=
   {{1, 0,           0         },
    {0, Cos[angle], -Sin[angle]},
    {0, Sin[angle],  Cos[angle]}};

On[General::spell1, General::spell];

(* make the string: starting with 'axiom', use StringReplace the
    specified number of times *)

LSystem[axiom_, rules_List, n_Integer,
   Ldelta_:{N[90 Degree], N[90 Degree], N[90 Degree]}] :=
      Nest[StringReplace[#, rules]& , axiom, n];

(* carry out the forward and backward moves and the various
    3D rotations by updating the global location 'Lpos' and
    direction matrix 'Ldir'. *)

Lmove[z_String, Ldelta_] :=

  Switch[z,
    "F", Lpos += First[Transpose[Ldir]],
    "B", Lpos -= First[Transpose[Ldir]],
    "+", Ldir = Ldir . RotMatPsi[Ldelta[[1]]];,
    "-", Ldir = Ldir . RotMatPsiII[Ldelta[[1]]];,
    "^", Ldir = Ldir . RotMatTheta[Ldelta[[2]]];,
    "&", Ldir = Ldir . RotMatThetaII[Ldelta[[2]]];,
    "<", Ldir = Ldir . RotMatPhi[Ldelta[[3]]];,
    ">", Ldir = Ldir . RotMatPhiII[Ldelta[[3]]];,
     _ , Null];

(* initialize the position 'Lpos' and the direction matrix 'Ldir';
    create the Line graphics primitive represented by the L-system by
    mapping 'Lmove' over the characters in the L-string, deleting all the
    Nulls; then show the Graphics3D object *)

LShow3D[lstring_String,
  Ldelta_:{N[90 Degree], N[90 Degree], N[90 Degree]}] :=

   (Lpos = {0., 0., 0.};
    Ldir = N[IdentityMatrix[3]];
    Show[
     Graphics3D[
       Line[
          Prepend[
             DeleteCases[
               (Lmove[#, Ldelta]&) /@ Characters[lstring],
             Null],
          {0, 0, 0}]]],
       AspectRatio -> Automatic]);

(* same as above, plus a list of colors for each segment contained in
    'templist' -- unfortunately, 'templist' isn't really 'temp', but
    stays in memory as a global variable; so sue me *)

LShowColor3D[lstring_String,
    Ldelta_:{N[90 Degree], N[90 Degree], N[90 Degree]},
    opts___Rule] :=

   (Lpos = {0., 0., 0.};
    Ldir = N[IdentityMatrix[3]];
    templist =
      Line /@
       Partition[
         Prepend[
           DeleteCases[
            (Lmove[#, Ldelta]&) /@ Characters[lstring],
           Null],
         0, 0, 0}],
       2, 1];
    ncol = N[Length[templist]];
    huelist = Table[Hue[k/ncol], {k, 1., ncol}];
    Show[Graphics3D[
      N[Flatten[Transpose[{huelist, templist}]]]],
     AspectRatio -> Automatic, opts]);

(* create just the list of 3D corners, supposing such a thing desirable *)

LCorners3D[lstring_String,
    Ldelta_:{N[90 Degree], N[90 Degree], N[90 Degree]}] :=

   (Lpos = {0., 0., 0.};
    Ldir = N[IdentityMatrix[3]];
    Prepend[DeleteCases[ (Lmove[#, Ldelta]&) /@
        Characters[lstring], Null], {0, 0, 0}]);

(* code for the colored-line version of the Hilbert curve *)

LShowColor3D[
  LSystem["X",
    {"X" -> "^<XF^<XFX-F^>>XFX&F+>>XFX-F>X->"}, 4],
     Pi/2.0{1,1,1}, Boxed->False]

Hilbert curve axiom and production rule by Stan Wagon, Mathematica in Action (Chapter 6), W. H. Freeman and Co., 1991. His code is miles better than the code here, so buy the book.

Designed and rendered using Mathematica versions 2.2 and 3.0 for the Apple Macintosh.

© 1996–2025 by Robert Dickau

[ home ] || [ 97???? ]

www.robertdickau.com/lsys3d.html