Message Boards Message Boards

The "Mathy" Arts of Coding Postcards

enter image description here

And so, the holidays are upon us once more and celebrations are on order. Wolfram Language fans enjoy fun recreation and arts, because beautiful things can be made with beautiful code, concise and elegant. I wanted to find a few gems from the past to honor the holidays and the traditions our users have. Surprisingly those few gems combined into one Christmas postcard you can see above.

The story of this postcard begins six years ago, when our sister community Mathematica Stack Exchange sprang up a question about simulating a snow fall with Wolfram Language. One of Wolfram most creative users, @Simon Woods gave a wonderful answer that was very popular. Then about five years ago I have run into a viral Reddit discussion dubbed

$$t * sin (t) \approx Christmas tree$$

which showcased a beautiful minimalistic Christmas tree built with simple $t * sin (t)$ function and Java Script. I recreated the concept with Wolfram Language and our another wonderful user @Silvia Hao ornamented it with festoon lamps. An idea came to me to combine them, because a Christmas Tree sparkling lights in a snowfall is the icon of winter holidays. But beware a few subtle tricks ;-) In depth those discussed at the original references I gave. Below are slightly changed code and a few comments.

The Tree

Our Christmas Tree is indeed spun with $t * sin (t)$. But in 3D rather than 2D. This is basically a conical spiral whose amplitude increases, a 2D circle dragged along 3-rd axis like this:

enter image description here

but only with increasing radius. Density of lights and their motion is one subtlety to take care of with math. Another subtlety is increasing 3D depth perception by slightly dimming the lights that are further from the observer. This function defines the mathematics of the tree:

 PD = .5;
 s[t_, f_] := t^.6 - f
 dt[cl_, ps_, sg_, hf_, dp_, f_, flag_] :=
  Module[{sv, basePt},
         {PointSize[ps],
          sv = s[t, f];
          Hue[cl (1 + Sin[.02 t])/2, 1, .8 + sg .2 Sin[hf sv]],
          basePt = {-sg s[t, f] Sin[sv], -sg s[t, f] Cos[sv], dp + sv};
          Point[basePt],
         If[flag,
            {Hue[cl (1 + Sin[.1 t])/2, 1, .8 + sg .2 Sin[hf sv]], PointSize[RandomReal[.01]],
             Point[basePt + 1/2 RotationTransform[20 sv, {-Cos[sv], Sin[sv], 0}][{Sin[sv], Cos[sv], 0}]]},
            {}]
        }]

and this code uses the function to build 228 frames of the animated tree:

treeFrames = ParallelTable[
                       Graphics3D[Table[{
                                         dt[1, .01, -1, 1, 0, f, True], 
                                         dt[.45, .01, 1, 1, 0, f, True],
                                         dt[1, .005, -1, 4, .2, f, False], 
                                         dt[.45, .005, 1, 4, .2, f, False]},
                                        {t, 0, 200, PD}],
                                  ViewPoint -> Left, BoxRatios -> {1, 1, 1.3}, 
                                  ViewVertical -> {0, 0, -1}, Boxed -> False,
                                  ViewCenter -> {{0.5, 0.5, 0.5}, {0.5, 0.55}},
                                  PlotRange -> {{-20, 20}, {-20, 20}, {0, 20}}, 
                                  Background -> Black,ImageSize->350],
                       {f, 0, 1, 1/199.}];

Let's check a single frame of The Tree:

First[treeFrames]

enter image description here

The Snow

This function below builds a single random snowflake. They are of course six-fold symmetric polygons.

flake := Module[{arm},
   arm = Accumulate[{{0, 0.1}}~Join~RandomReal[{-1, 1}, {5, 2}]];
   arm = arm . Transpose@RotationMatrix[{arm[[-1]], {0, 1}}];
   arm = arm~Join~Rest@Reverse[arm . {{-1, 0}, {0, 1}}];
   Polygon[Flatten[arm . RotationMatrix[# \[Pi]/3] & /@ Range[6], 1]]];

Let's see a few random shapes, they are fun in black on white ;-)

Multicolumn[Table[Graphics[flake, ImageSize -> 50], 100], 10]

enter image description here

Now it's time to build the snowfield which has a few tricks. To simulate 3D perception 2 things need to be observed:

  1. Real further snowflakes appear smaller

  2. Real further snowflakes have slower perceived angular speeds

The 2nd observation is taken care of by the size_ variable below.

snowfield[flakesize_, size_, num_] := 
  Module[{x = 100/flakesize}, 
   ImageData@
    Image[Graphics[{White,Opacity[.8],
       Table[Translate[
         Rotate[flake, RandomReal[{0, \[Pi]/6}]], {RandomReal[{0, x}],
           RandomReal[{0, x}]}], {num}]}, Background -> Black, 
      PlotRange -> {{0, x}, {0, x}}], ImageSize -> {size, size}]];

and by 3 different sizes given here:

size=720;
r=snowfield@@@{{.9,size,250},{1.2,size,30},{1.6,size,10}};

Thumbnail /@ 
 snowfieldTMP @@@ {{.9, size, 250}, {1.2, size, 30}, {1.6, size, 10}}

enter image description here

So we sort of have 3 different fields of vision reproaching the observer. The 1st observation is simulated with different speed with which different fields of vision are rotated, the closer one being the fastest. This simulates rotation of the fields of vision and builds the frames for the snowfall:

snowFrames=ParallelTable[Image[Total[(RotateRight[r[[#]],Round@k#]&/@{1,2,3})[[All, ;;size]]]],{k,0,719,3.6}];

snowFrames // Length
200

snowFrames[[1]] // Thumbnail

enter image description here

The Postcard

ImageCompose is an easy way to join to visuals:

ImageCompose[ColorConvert[snowFrames[[1]], "Grayscale"], 
 Show[treeFrames[[1]], ImageSize -> 600]]

enter image description here

I also would like to add a header with hint how tree was constructed:

header = 
 Rasterize[
  Framed[Style["Christmas Tree \[TildeTilde] t*sin(t)", 55, 
    GrayLevel[.9], FontFamily -> "Optima"], Background -> Black], 
  RasterSize -> 720]

enter image description here

Here is the final frame for the animation:

finalFrames=
Parallelize[MapThread[
    ImageAssemble[{{header,ImageCompose[ColorConvert[#1,"Grayscale"],Show[#2,ImageSize->600]]}}//Transpose]&,
{snowFrames,treeFrames}]];

finalFrames[[1]]

enter image description here

and this exports the frames to the GIF you see at the top of the post:

SetDirectory[NotebookDirectory[]]
Export["xmas.gif", finalFrames,RasterSize->550,"DisplayDurations"->.05]

I hope you had fun. Feel free to share your own crafts. Happy holidays!

Attachments:
POSTED BY: Vitaliy Kaurov
4 Replies

This post has been updated.

POSTED BY: EDITORIAL BOARD

Hi Vitaly,

Great work! Unfortunately, the code will throw errors in Version 12.2 when executing the finalFrames = ... cell:

ImageAdd::bdchan : The number of channels must be equal or one of them must be a single-channel image and the other a multichannel image.

It looks like the slots have to be swapped as the error will not arise for the following code:

finalFrames = Parallelize @ MapThread[
    ImageAdd[ SetAlphaChannel[ #2, 1 ], #1 ]&,
    { treeFrames, snowFrames }
];

Unfortunately, the result then is not as neat as the one you have posted here (i.e. the tree will flicker and the animation does is not looking as smooth). Since image processing is not my sweet spot, maybe you can have a quick look and fix your beautiful example? It's simply too nice to not work out.

Thanks, Guido

Dear @Guido, only 4 years later, updated. Happy holidays.

POSTED BY: Vitaliy Kaurov

Thank you for your beautiful Xmas card Vitaliy!

Reply to this discussion
Community posts can be styled and formatted using the Markdown syntax.
Reply Preview
Attachments
Remove
or Discard

Group Abstract Group Abstract