exercises/solutions/pgftree.tex
author Eugen Sawin <sawine@me73.com>
Sat, 09 Jun 2012 22:15:16 +0200
changeset 14 fdfb70dec1a8
permissions -rw-r--r--
Added first version of sol05.
sawine@11
     1
%    pgftree.tex
sawine@11
     2
%    Version 1.1, 25 Dec 2009
sawine@11
     3
sawine@11
     4
%    Copyright 2009 by David Chiang
sawine@11
     5
sawine@11
     6
%    This program is free software; you can redistribute it and/or modify
sawine@11
     7
%    it under the terms of the GNU General Public License as published by
sawine@11
     8
%    the Free Software Foundation; either version 2 of the License, or
sawine@11
     9
%    (at your option) any later version.
sawine@11
    10
sawine@11
    11
%    This program is distributed in the hope that it will be useful,
sawine@11
    12
%    but WITHOUT ANY WARRANTY; without even the implied warranty of
sawine@11
    13
%    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
sawine@11
    14
%    GNU General Public License for more details.
sawine@11
    15
sawine@11
    16
%    You should have received a copy of the GNU General Public License along
sawine@11
    17
%    with this program; if not, write to the Free Software Foundation, Inc.,
sawine@11
    18
%    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
sawine@11
    19
sawine@11
    20
% New in version 1.1:
sawine@11
    21
% - major restructuring to not do arbitrary nesting of subpicture environments
sawine@11
    22
% - sideways trees
sawine@11
    23
sawine@11
    24
% To do:
sawine@11
    25
% - trees with all leaves at same level
sawine@11
    26
% - if \nodename does not exist as desired, wrap inside a rectangle node
sawine@11
    27
% - don't use pgfsubpic internals
sawine@11
    28
sawine@11
    29
\newdimen\levelsep \levelsep=30pt
sawine@11
    30
\newdimen\subtreesep \subtreesep=2pt
sawine@11
    31
sawine@11
    32
\def\leveldirection{down}
sawine@11
    33
\def\siblingdirection{right}
sawine@11
    34
sawine@11
    35
% definitions of growing directions
sawine@11
    36
\def\pgftree@levelshift{\csname pgftree@levelshift@\leveldirection\endcsname}
sawine@11
    37
\def\pgftree@parentanchor{\csname pgftree@parentanchor@\leveldirection\endcsname}
sawine@11
    38
\def\pgftree@childanchor{\csname pgftree@childanchor@\leveldirection\endcsname}
sawine@11
    39
% these assume that the current subpicture is the child
sawine@11
    40
\def\pgftree@presiblingshift{\csname pgftree@presiblingshift@\siblingdirection\endcsname}
sawine@11
    41
\def\pgftree@postsiblingshift{\csname pgftree@postsiblingshift@\siblingdirection\endcsname}
sawine@11
    42
sawine@11
    43
\def\pgftree@levelshift@down{\pgfpoint{0}{-\levelsep}}
sawine@11
    44
\def\pgftree@parentanchor@down{south}
sawine@11
    45
\def\pgftree@childanchor@down{north}
sawine@11
    46
sawine@11
    47
\def\pgftree@levelshift@up{\pgfpoint{0}{\levelsep}}
sawine@11
    48
\def\pgftree@parentanchor@up{north}
sawine@11
    49
\def\pgftree@childanchor@up{south}
sawine@11
    50
sawine@11
    51
\def\pgftree@levelshift@right{\pgfpoint{\levelsep}{0}}
sawine@11
    52
\def\pgftree@parentanchor@right{east}
sawine@11
    53
\def\pgftree@childanchor@right{west}
sawine@11
    54
sawine@11
    55
\def\pgftree@levelshift@left{\pgfpoint{-\levelsep}{0}}
sawine@11
    56
\def\pgftree@parentanchor@left{west}
sawine@11
    57
\def\pgftree@childanchor@left{east}
sawine@11
    58
sawine@11
    59
\def\pgftree@presiblingshift@right{\pgf@process{\pgf@x-\pgf@subpicminx \advance\pgf@x\subtreesep \pgf@y 0pt}}
sawine@11
    60
\def\pgftree@postsiblingshift@right{\pgf@process{\pgf@x\pgf@subpicmaxx \pgf@y 0pt}}
sawine@11
    61
sawine@11
    62
\def\pgftree@presiblingshift@left{\pgf@process{\pgf@x-\pgf@subpicmaxx \advance\pgf@x-\subtreesep \pgf@y 0pt}}
sawine@11
    63
\def\pgftree@postsiblingshift@left{\pgf@process{\pgf@x\pgf@subpicminx \pgf@y 0pt}}
sawine@11
    64
sawine@11
    65
\def\pgftree@presiblingshift@up{\pgf@process{\pgf@x 0pt \pgf@y-\pgf@subpicminy \advance\pgf@y\subtreesep}}
sawine@11
    66
\def\pgftree@postsiblingshift@up{\pgf@process{\pgf@x 0pt \pgf@y\pgf@subpicmaxy}}
sawine@11
    67
sawine@11
    68
\def\pgftree@presiblingshift@down{\pgf@process{\pgf@x 0pt \pgf@y-\pgf@subpicmaxy \advance\pgf@y-\subtreesep}}
sawine@11
    69
\def\pgftree@postsiblingshift@down{\pgf@process{\pgf@x 0pt \pgf@y\pgf@subpicminy}}
sawine@11
    70
sawine@11
    71
% for convenience if you are using \pgftree directly
sawine@11
    72
\def\drawnode#1{\pgfnode{rectangle}{base}{#1}{\nodename}{\pgfusepath{discard}}}
sawine@11
    73
\def\drawedge{%
sawine@11
    74
\pgfpathmoveto{\pgfpointanchor{\parentnodename}{\pgftree@parentanchor}}%
sawine@11
    75
\pgfpathlineto{\pgfpointanchor{\nodename}{\pgftree@childanchor}}%
sawine@11
    76
\pgfusepath{stroke}}
sawine@11
    77
sawine@11
    78
% local variables that we need to assign to inside a \pgfforeach
sawine@11
    79
\newdimen\pgftree@childx
sawine@11
    80
\newdimen\pgftree@savechildx
sawine@11
    81
\newdimen\pgftree@childy
sawine@11
    82
\newdimen\pgftree@savechildy
sawine@11
    83
\newcount\pgftree@childi
sawine@11
    84
\newcount\pgftree@savechildi
sawine@11
    85
sawine@11
    86
%%% \pgftree{subtree}
sawine@11
    87
sawine@11
    88
\def\pgftree#1{%
sawine@11
    89
\def\nodename{r}%
sawine@11
    90
#1%
sawine@11
    91
\pgfplacesubpicture
sawine@11
    92
}
sawine@11
    93
sawine@11
    94
%%% \pgfsubtree{root}{subtrees}
sawine@11
    95
% The first argument draws the root node using PGF/TikZ commands.
sawine@11
    96
% The node must be named \nodename.
sawine@11
    97
sawine@11
    98
% The second argument is an even-length sequence of tokens.
sawine@11
    99
% Token 2n-1 in the sequence draws the nth edge. It should draw an edge from \parentnodename to \nodename.
sawine@11
   100
% Token 2n in the sequence draws the nth subtree. Its root must be named \nodename.
sawine@11
   101
sawine@11
   102
\pgfnewsubpicture{children}
sawine@11
   103
\newdimen\pgftree@lastchildx
sawine@11
   104
\newdimen\pgftree@lastchildy
sawine@11
   105
sawine@11
   106
\def\pgfsubtree#1#2{%
sawine@11
   107
\let\parentnodename\nodename
sawine@11
   108
\pgftree@savechildx=\pgftree@childx
sawine@11
   109
\pgftree@savechildy=\pgftree@childy
sawine@11
   110
\pgftree@savechildi=\pgftree@childi
sawine@11
   111
% Build subpicture with all the children and their subtrees
sawine@11
   112
{\pgftree@childx=0pt%
sawine@11
   113
\pgftree@childy=0pt%
sawine@11
   114
\pgftree@childi=0%
sawine@11
   115
\process@children #2}%
sawine@11
   116
\begin{pgfsubpicture}%
sawine@11
   117
% Create node
sawine@11
   118
#1%
sawine@11
   119
\ifnum\pgftree@childi>0%
sawine@11
   120
% Place children
sawine@11
   121
% move down \levelsep
sawine@11
   122
{\pgftransformshift{\pgftree@levelshift}%
sawine@11
   123
% center so that parent is midway between origins of first and last children
sawine@11
   124
\pgftransformshift{\pgfpointscale{-0.5}{\pgfqpoint{\the\pgftree@childx}{\the\pgftree@childy}}}%
sawine@11
   125
\pgfplacesubpicture}%
sawine@11
   126
% Draw the edges
sawine@11
   127
{\pgftree@childi=0%
sawine@11
   128
\process@edges #2}%
sawine@11
   129
\fi
sawine@11
   130
\end{pgfsubpicture}%
sawine@11
   131
\global\pgftree@childi=\pgftree@savechildi
sawine@11
   132
\global\pgftree@childx=\pgftree@savechildx
sawine@11
   133
\global\pgftree@childy=\pgftree@savechildy
sawine@11
   134
}
sawine@11
   135
sawine@11
   136
\def\process@children{%
sawine@11
   137
\pgfutil@ifnextchar\egroup
sawine@11
   138
{% No more children, step back to origin of last child
sawine@11
   139
\global\pgftree@childx\pgftree@lastchildx
sawine@11
   140
\global\pgftree@childy\pgftree@lastchildy
sawine@11
   141
\global\pgftree@childi\pgftree@childi
sawine@11
   142
\ifnum\pgftree@childi>0%
sawine@11
   143
\pgfrestoresubpicture{children}% pass children back to caller
sawine@11
   144
\fi
sawine@11
   145
}%
sawine@11
   146
{\@process@children}%
sawine@11
   147
}
sawine@11
   148
\def\@process@children#1#2{% #1 is the edge, #2 is the child
sawine@11
   149
% Build the current child
sawine@11
   150
{\edef\nodename{\parentnodename-\the\pgftree@childi}%
sawine@11
   151
#2}%
sawine@11
   152
\begin{pgfsubpicture}%
sawine@11
   153
% Place current child
sawine@11
   154
\ifnum\pgftree@childi>0% the first child is always at 0
sawine@11
   155
\pgftree@presiblingshift \global\advance\pgftree@childx\pgf@x \global\advance\pgftree@childy\pgf@y
sawine@11
   156
\fi
sawine@11
   157
{\pgftransformshift{\pgfqpoint{\the\pgftree@childx}{\the\pgftree@childy}}%
sawine@11
   158
\pgfplacesubpicture}%
sawine@11
   159
\global\pgftree@lastchildx\pgftree@childx
sawine@11
   160
\global\pgftree@lastchildy\pgftree@childy
sawine@11
   161
\pgftree@postsiblingshift \global\advance\pgftree@childx\pgf@x \global\advance\pgftree@childy\pgf@y
sawine@11
   162
% Save the augmented row of children back into "children"
sawine@11
   163
\end{pgfsubpicture}
sawine@11
   164
\ifnum\pgftree@childi>0%
sawine@11
   165
\pgfmergesubpicture{children}
sawine@11
   166
\else
sawine@11
   167
\pgfsavesubpicture{children}%
sawine@11
   168
\fi
sawine@11
   169
\advance\pgftree@childi by 1%
sawine@11
   170
\process@children
sawine@11
   171
}
sawine@11
   172
sawine@11
   173
\def\process@edges{%
sawine@11
   174
\pgfutil@ifnextchar\egroup
sawine@11
   175
{}%
sawine@11
   176
{\@process@edges}%
sawine@11
   177
}
sawine@11
   178
\def\@process@edges#1#2{%
sawine@11
   179
\edef\nodename{\parentnodename-\the\pgftree@childi}%
sawine@11
   180
#1%
sawine@11
   181
\advance\pgftree@childi by 1%
sawine@11
   182
\process@edges
sawine@11
   183
}
sawine@11
   184
sawine@11
   185
\def\subtreeof#1{%
sawine@11
   186
% the subpicture which contains a node also contains exactly its subtree
sawine@11
   187
\csname pgf@sh@pi@#1\endcsname
sawine@11
   188
}