For the holidays, I wanted to follow up to the question about making fireworks with tikz (How can we display fireworks?):
How can we make fireworks with only text?
I have made a crude attempt below. For some reason I can't manage to put each firework burst into a box so I can manipulate it easily. Also strange things start to happen if I ask for more than 12 rotations (see right column).
Please help me improve this example or suggest something better.
Please, no tikz, pstricks, Metapost, or picture mode! Text only, but any flavor of TeX is welcome.
\documentclass{article}
\usepackage{rotating, color}
\newcounter{degrees}
\newcounter{degreeIncrement}
% More than 360 repetitions will max out TeX's memory
% Print error message and end document if more than 360 reps are requested
\newcommand{\textwheelErrorTest}[1]{%
\ifnum#1 > 360
\typeout{%
ERROR: You asked for #1 repetitions in
\noexpand\textwheel, but the maximum is 360.
I am stopping before TeX exceeds its memory capacity.%
}%
\def\erroraction{\end{document}}
\else
\def\erroraction{\relax}
\fi
\erroraction
}
% For calculating length of string for kerning
\newlength{\strlen}
% #1 text to repeat, #2 color of text, #3 number of times to repeat it
\newcommand{\textwheel}[3]{%
\textwheelErrorTest{#3}
%
\setcounter{degreeIncrement}{360}
\setcounter{degrees}{0}
\divide\value{degreeIncrement} by #3
\settowidth{\strlen}{#1}
%
\loop
\rlap{%
\turnbox{\value{degrees}}{\color{#2}\quad#1}%
}\kern -0.5\strlen%
%
\addtocounter{degrees}{\value{degreeIncrement}}
\ifnum\value{degrees} < 360
\repeat%
}
\begin{document}
\pagecolor{black}
\begin{minipage}{0.45\linewidth}
%
\textwheel{\TeX}{yellow}{6}
\vspace{10\baselineskip}
\textwheel{\LaTeX}{cyan}{12}
\vspace{10\baselineskip}
\textwheel{Xe\TeX}{red}{8}
\vspace{10\baselineskip}
\textwheel{Lua\TeX}{green}{5}
\vspace{15\baselineskip}
\textwheel{Con\TeX t}{magenta}{4}
%
\end{minipage}%
\hfill%
\begin{minipage}{0.45\linewidth}
%
\textwheel{\LaTeX}{white}{20}
\vspace{10\baselineskip}
\textwheel{\LaTeX}{green}{45}
\vspace{15\baselineskip}
\textwheel{\LaTeX}{yellow}{90}
\vspace{25\baselineskip}
\textwheel{\LaTeX}{red}{180}
%
\end{minipage}
\end{document}
Best Answer
2016 Christmas Edition
I experimented here with trying to produce a geometric figure near the firework core, by using simple glyphs like
+
and-
. When duplicated with rotation and shift, I think the effect is very nice way to highlight the event.2015 CHRISTMAS EDITION
Here I introduce my first triple-burst firework, with "For unto us" as the inner burst, "a child is born" as the second burst, and "Lord!" as the outer burst. The code can be found in the MWE later in this post. I wish a blessed Christmas season to all.
2015 NEW YEAR'S EDITION
2014 EDITION:
Merry Christmas!
The syntax is
where
reps
is the number of repetitions, andfade_length
is the length over which the fade from the inner to outer color occurs (upon a recent EDIT, it can now be smaller or larger than thestring
length and work appropriately). If set to 0pt, it sets thefade_length
to the length ofstring
.See end of post for a description of the PROGRAM LOGIC.
EDIT for nested textwheels (I feel like Ptolemy with epicycles), which can be formulated as such:
where one
\textwheel
is saved into a box (\ST
) and then placed at the end of the subsequent\textwheeel
.EDITED to remove stray spaces. EDITED to remove
\strut
bias which produced a small amount of ovalization. EDITED to minimize roundoff error by working in thousandths of degrees. RE-EDITED (based on comment suggestion) to reintroduce optional level of\strut
bias, which is now controlled by two parameters:\biasfrac
and\biaslength
. Symmetry is retained when\biaslength=0pt
(the default).In the second figure, the inner firework says "Merry Christmas" 28 times. And at the end of each "Merry Christmas", there is a
\tiny
explosion that says "Hallelujah!" 13 times.P.S. I just coincidentally realized that 28x13 'Hallelujah!'s make 364... one for every day of the year. God bless!
Here is a zoom on the prior figure:
This show two cases with only the
fade_length
altered (from 1.5cm to 3cm):And finally, an example of
\strut
bias asymmetry introduced with a nonzero\biaslength
:Note the difference from the image above, especially around the center of the "firework". Additionally, the firework takes on a slight eccentricity.
PROGRAM LOGIC
The OP has asked for a description of the logic. The OP used the
\turnbox
macro, which is described in the rotating.sty documentation as "a macro version of the rotate environment." I was not familiar with that macro, so I stuck with what I knew, which is the\rotatebox
macro of graphicx.sty, using the[origin=l]
option to rotate around the left edge of the box.I knew I wanted to lay (via lapping) an array of
\rotatebox
es concentrically, and the OP had laid out logic for such a loop, which I used directly. The many edits to my post indicate that many corrections had to be made along the way. One issue, for rotations between 90 and 270, is that the "tail" of the box cannot push leftward past the x=0 position, and so the left-side origin of the box gets pushed rightward. The remedy for that was to use\llap
between 90 and 270, and\rlap
elsewhere on the circle. The\ifnum\value{degrees}...
branch logic inside\textwheel
addresses that point.The next issue is that the vertical height of the rotated box also limits the leftward push of the box. This introduced a discontinuity at 90 and 270 degrees of rotation. This is what I call
\strut
-bias, since differing vertical heights would introduce differing levels of discontinuity. My initial correction was the\kern
introduced between 90 and 270 degrees of rotation. But even the kern-correction left a slightly ovalized result.Later, I realized that
\smash
ing the text to be rotated would take care of the problem making the result beautifully symmetrical. One user actually missed the\strut
-bias, however, and so I reintroduced it as a controllable parameter through\biaslength
(the total height of the reintroduced strut) and\biasfrac
(that fraction of\biaslength
applied below the baseline).One of my early ideas for improving the look over the OP's question was the use of continuous color transition of the firework. I had recently answered the question at How can I 'fade out' text after a certain length? in which I made text fade from black to white. I grabbed that
\FadeAfter
logic and had to tweak it a little to make it work from color1 to color2. It essentially grabs the argument letter by letter, measures the cumulative length to that point, and uses that cumulative length (relative to a specified fadeout length) to decide the color of that letter. Whereas in the original logic, I turned off the fade after reaching the "fade length" since it was white, here I just continued processing the string using color2.Another issue that arose is that TeX's
\divide
command truncates the result to an integer. This macro was used to divide 360 by the number of repetitions in order to get the angular increment. The bad news is that it may give the wrong result (I found that asking for 28 repetitions gave me actually 30). Also, all the roundoff errors became visually obvious between the final box and the 0-degree box. My solution was to perform the calculation in thousandths of degrees (thus the loop to 360000), so that the roundoff error would be smaller. Even so, the loop is set to only lay boxes up to 359 degrees to avoid two nearly overlaid boxes (at 0 and 359.xx) degrees.Perhaps the best addition I made to the OP's proposal was the introduction of nested boxes. For this I used a
\savestack
from mystackengine
package (though a\savebox
/\usebox
would work equally well. It is quite straightforward. One just saves a small\textwheel
in a box, and then uses that box as part of the core\textwheel
, laying it at the end of the string. Since the two wheels can have different color combinations, the possibilities are endless.The
\textwheel
s, because of the laps and smashes, have zero height and width. A non-zero\biaslength
may actually change that slightly, I'm not sure, which is why I\smash
ed the nested\textwheel
s. Thus, I use a series of\vspace
es and\hspace
es to lay the\textwheel
s around the page. It makes it easy to overlap two independent\textwheel
s if that is desired.