\immediate\write18{tex rulercompass.dtx} \documentclass{ltxdoc} \usepackage{alphalph} \usepackage{fancyvrb} \usepackage{listings} \usepackage{tikz} \usepackage{hyperref} \usetikzlibrary{rulercompass} \pgfmathsetmacro\ptra{2}%{.5 + .25*rand} \pgfmathsetmacro\ptrb{2.7}%{.5 + .25*rand} \pgfmathsetmacro\ptaa{30}%180*rand} \pgfmathsetmacro\ptab{70}%180*rand} \lstloadlanguages{[LaTeX]TeX} \lstset{breakatwhitespace=true,breaklines=true,language=TeX} \EnableCrossrefs \CodelineIndex \RecordChanges \def\figurename{Example} \newenvironment{example} {\VerbatimEnvironment \begin{VerbatimOut}{example.out}} {\end{VerbatimOut} \begin{center} \setlength{\parindent}{0pt} \fbox{\begin{minipage}{.9\linewidth} \lstset{breakatwhitespace=true,breaklines=true,language=TeX,basicstyle=\small} \lstinputlisting[]{example.out} \end{minipage}} \fbox{\begin{minipage}{.9\linewidth} \centering \input{example.out} \end{minipage}} \end{center} } \newcommand\tikzname{\textsf{Ti\textit{k}Z}} \title{The \textsf{rulercompass} \tikzname\ Library} \author{Andrew Stacey} \date{10th December 2013} \begin{document} \maketitle \section{Introduction} The \textsf{rulercompass} \tikzname\ library provides some commands and styles for drawing straight-{}edge\footnote{The name \textsf{straightedgecompass} was just a bit too long.} and compass diagrams in \tikzname. With this, one can draw diagrams such as that in Example~\ref{fig:bisect}. \begin{figure} \centering \begin{example} \begin{tikzpicture}[ stop jumping, constrain ] \path (0,0) node[name=0,ruler compass/point=red,label={0}]; \path (0) ++(30:2) node[ruler compass/point=red,label={a}]; \ruler{0}{a} \compass{0}{a} \compass{a}{0} \point{c0a}{ca0}{1} \point{c0a}{ca0}{2} \ruler{b}{c} \end{tikzpicture} \end{example} \caption{Constructing the perpendicular bisector between two points} \label{fig:bisect} \end{figure} \section{Straight-{}Edge and Compass Diagrams} To understand the commands of this package, it is necessary to know a bit about the type of diagram involved. A straight-{}edge and compass diagram consists of a family of points in the plane, a family of straight lines, and a family of circles. The usual rules for constructing these families is to start with a given set of points and apply the following rules: \begin{enumerate} \item Construct a straight line passing through any two points. \item Construct a circle centred at one point which passes through another point. \item Place a point at an intersection of two lines, two circles, or a circle and a line. \end{enumerate} The usual goal of a straight-{}edge and compass diagram is to place a point at a particular position given a certain family of initial points. Some examples are: % \begin{enumerate} \item Given three points in the plane, say \(0, a, b\), view \(a\) and \(b\) as vectors from \(0\) and construct their sum, \(a + b\). \item Given four points in the plane, say \(0, 1, a, b\), view \(a\) and \(b\) as complex numbers with \(0\) and \(1\) interpreted in the obvious way, and construct their product, \(a b\). \item Given two points in the plane, say \(0\) and \(1\), construct a regular hexagon with \(0\) and \(1\) as adjacent vertices. \item Given two points in the plane, say \(0\) and \(1\), construct a regular \(17\)--{}sided polygon with \(0\) and \(1\) as adjacent vertices. \end{enumerate} % Often, a diagram is completed by ``inking in'' some segments of the lines and circles. When drawing a straight-{}edge and compass diagram we need to be able to carry out the three constructions. Whilst none correspond to basic \tikzname\ path operations, they are all possible with the use of some additional libraries, namely the \Verb+intersections+ and \Verb+calc+ libraries. These are therefore loaded automatically. \section{Usage} This package comes as a \tikzname\ library so can be loaded in the same way as other \tikzname\ libraries: % \begin{verbatim} \usetikzlibrary{rulercompass} \end{verbatim} % This loads in a family of commands and \tikzname\ styles for drawing these diagrams. \subsection{Naming Schemes} The various constructions refer to points or paths already constructed (or given at the start). Therefore, it is necessary to have a naming scheme. Points are labelled automatically using an internal \LaTeX\ counter, \Verb+pointlabels+. The format name can be changed in the same way as the format of any \LaTeX\ counter, by redefining \Verb+\thepointlabels+. However, because it is sometimes needed to be able to go back from the label to the value of the counter, it is better to change it by one of the \tikzname\ styles provided. As a default, \Verb+rulercompass+ uses lowercase letters. Thus the first point will be \Verb+a+, the second \Verb+b+, and so on. Points are actually coordinates so these can be used as such within the \Verb+tikzpicture+. Since paths are defined in terms of points, the naming scheme for paths is a triplet of tokens or token lists where the first distinguishes between lines and circles and the other two are the two points used to construct the path. \begin{itemize} \item \Verb+rab+ is a line (\Verb+r+ for ``ruler'') through the points \Verb+a+ and \Verb+b+. Note that although \Verb+rab+ and \Verb+rba+ are the same line, the two names cannot be used interchangeably. The order of points must be the same as that used in the construction of the line. \item \Verb+c{ab}v+ is a circle (\Verb+c+ for ``compass'') with centre \Verb+ab+ passing through \Verb+v+. Here there is no risk of confusion since \Verb+cab+ and \Verb+cba+ are distinct circles. \end{itemize} \subsection{Primary Commands} There are three primary commands available: \begin{itemize} \item \Verb+\point[style]{}{}{}+ This marks an intersection with a point, using the next label. Some pairs of paths have two intersection points and the \Verb++ argument is used to choose which one to mark. \item \Verb+\ruler[style]{}{}+ This constructs the line through the two points. \item \Verb+\compass[style]{}{}+ This constructs the circle centred at the first point which passes through the second point. \end{itemize} If the \Verb+rulercompass+ library detects that \Verb+beamer+ has been loaded, then all of the above are made overlay-{}aware. In addition, the \Verb+\point+ command does a little optimisation: as intersections can be expensive to compute (if there are a lot of them), then it tries to figure out if this particular intersection has already been computed in a previous version of this diagram. Example~\ref{fig:bisect} contains an example of each of these commands. \subsection{Styles} The primary commands all take an optional argument to which arbitrary \tikzname\ styles can be passed. Each command internally expands to a \Verb+\draw+ (for \Verb+\ruler+ and \Verb+\compass+) or a \Verb+node+ on a \Verb+\path+ (for \Verb+\point+) with the optional argument inserted in a suitable place. If this does not provide enough opportunity for control, there are style equivalents of the commands that can be used directly: \begin{itemize} \item \Verb+ruler compass/point=+ This marks the current point. Inside the \Verb+\point+ command, this is invoked inside a \Verb+node+ after the appropriate intersection has been computed. The \Verb++ is optional. If the node has not already been named (via \Verb+name=+) then this gives it the next name as determined by the \Verb+pointlabels+ counter. \item \Verb+ruler compass/ruler={}{}+ This inserts the line through the points into the current path. \item \Verb+ruler compass/compass={}{}+ This inserts the compass with centre the first point which passes through the second point into the current path. \end{itemize} The initial two commands in Example~\ref{fig:bisect} use the style format for greater flexibility. In the first, the \Verb+name=0+ key overrides the automatic naming of the point meaning that this point is known as \Verb+0+ instead of \Verb+a+. In both the first and the second, a colour is specified. There are a variety of styles that can be used to change how the various constructions are rendered: \begin{itemize} \item \Verb+ruler compass/every point+ \item \Verb+ruler compass/every construction path+ Encompasses both \Verb+ruler+ and \Verb+compass+. \item \Verb+ruler compass/every ruler+ \item \Verb+ruler compass/every compass+ \item \Verb+ruler compass/construction in use+ and \Verb+ruler compass/construction not in use+ When \Verb+beamer+ is detected, every time a path is used to compute a point then the current frame is noted. This means that on the second (and subsequent) runs, it is possible to style paths differently depending on whether or not they are still in use. \item \Verb+ruler compass/every aux point+ When a path is computed, then the two points that are used in its construction are designated as ``auxiliary points'' and are given an extra style. This is mainly of use when stepping through a construction in \Verb+beamer+. \item \Verb+ruler compass/draft label+ If in draft mode (determined by the key \Verb+ruler compass/draft mode+), then constructed points are labelled. This can be used to style the label. \end{itemize} Some other miscellaneous keys are the following. \begin{itemize} \item \Verb+ruler compass/draft mode=+ This controls a switch to display the labels above the constructed points. \item \Verb+ruler compass/ruler length=+ The default line extends \(20\mathrm{cm}\) beyond each point. This changes that. \item \Verb+ruler compass/point labels=+ This changes the style of the point labels. For the last two, the \Verb+alphalph+ package must be loaded. If this library detects that the \Verb+alphalph+ package has already been loaded then the default is \Verb+alphalph+, otherwise the default is \Verb+alph+. It is better to use this than to redefine \Verb+\thepointlabels+. \end{itemize} \section{Relative Labels} The system for labelling points has rudimentary support for arithmetic. For this, it is important that \TeX\ know how to reverse the labelling system to get a number back from a label and this is why the key \Verb+ruler compass/point labels+ should be used to change the labelling format. The operations allowed are addition and subtraction of a number from a label, together with the use of a period to designate the last marked point. This makes it easier to work on a drawing in parts without having to know exactly how many points are constructed in each part, or to define auxiliary macros to construct some number of extra points. \begin{figure} \begin{example} \begin{tikzpicture}[ stop jumping, constrain ] \path (0,0) node[name=0,ruler compass/point=red,label={0}]; \path (0) ++(30:2) node[ruler compass/point=red,label={a}]; \path (0) ++(70:2.7) node[ruler compass/point=red,label={b}]; \ruler{0}{.} \ruler{0}{a} \compass{0}{.-1} \point{r0b}{c0a}{1} \compass{0}{a+1} \point{r0a}{c0{a+1}}{1} \end{tikzpicture} \end{example} \caption{Demonstrating relative point labels.} \label{fig:ptlabel} \end{figure} In Example~\ref{fig:ptlabel}, in the line \Verb+\ruler{0}{.}+ then the second point is \Verb+b+. In the line \Verb+\compass{0}{.-1}+ then the second point is \Verb+a+. In the last line, the second path is the circle with centre \Verb+0+ which goes through \Verb+b+. \section{Bounding Boxes} There are two considerations here relating to bounding boxes. The first is the ``infinite'' lines. Even though these are not actually infinite, they are constructed to be very long (so that any intersection is almost certain to actually exist). This usually means that they dwarf the actual region of interest. To ensure that the diagram does not get too big, the lines are actually drawn without affecting the bounding box (via the \Verb+overlay+ key), but that simply means that they can bleed into the surrounding text. Clearly a \Verb+\clip+ would be useful here, but it would also be nice not to have to compute it by hand. The second consideration comes when using \Verb+beamer+ and doing a step-{}by-{}step construction. Each piece of the construction adds a new element and thus, potentially, changes the bounding box. This could lead to the picture jumping around on the page. Inserting a \Verb+\path+ of a suitable size would solve this, but again it would be nice not to have to compute this by hand. Both of these have the same solution: compute the effective bounding box at the end of the construction and use that information at the start. This involves saving information to the \Verb+aux+ file and so requires at least two runs. In the first situation, we clip the picture to its eventual bounding box (possibly slightly enlarged). In the second, we add a rectangle of the eventual bounding box at the end (adding it at the end means that we can first compute the bounding box without it to ensure that our saved bounding box is always up to date). Once we give ourselves the technology to save the bounding box, we can also think of other ways to use it. One more is given here: the ability to automatically resize a \Verb+tikzpicture+ to fit inside a given rectangle (preserving the aspect ratio). In addition, a pseudo-{}node \Verb+enclosing box+ is defined which can be used like the \Verb+current bounding box+ pseudo-{}node. The original code for saving and reusing the bounding box comes from the \href{http://tex.stackexchange.com}{TeX-SX} site question \href{http://tex.stackexchange.com/q/18704/86}{How can I fix jumping TikZ pictures in beamer?}. This code provides the following new keys which are designed to be used in the optional argument to the \Verb+tikzpicture+ environment: \begin{itemize} \item \Verb+stop jumping+ This installs the code to save the bounding box and also to insert the rectangle at the end of the picture. \item \Verb+max size={}{}+ This compares the (saved) bounding box to the given rectangle and then scales down (only down) to ensure that it fits inside the given rectangle. \item \Verb+enclosing box/offset=+ This adds a little extra room around the bounding box. This has an effect in two places: on the computation in the \Verb+max size+ it ensures that the extra room fits inside the given rectangle, and on the size of the \Verb+enclosing box+ pseudo-{}node. \item \Verb+constrain+ If the \Verb+enclosing box+ pseudo-{}node is used directly, it should almost always be used with the \Verb+overlay+ key because otherwise there is potential for the bounding box size to explode. This makes it somewhat tricky to use with the \Verb+\clip+ path constructor as that does its own stuff with the computation of the bounding box. For this reason, the key \Verb+constrain+ is provided which does a ``safe'' clip to the enclosing box. If this is to be used somewhere other than at the start of the picture, there is the command \Verb+\constrain+. \end{itemize} \section{Drawing Segments} Drawing a line segment between two points is already adequately provided for by \tikzname. Drawing an arc is not, so this library provides a key \Verb+centred arc to={}{}+ which draws an arc from the current point to the second specified point which is centred at the first specified point. The key \Verb+arc flip+ can be used to switch which part of the circle is drawn. These keys are for the \Verb+arc+ path constructor, as in Example~\ref{ex:arcs}. \begin{figure} \begin{example} \begin{tikzpicture} \path (0,0) node[name=0,ruler compass/point=red,label={0}]; \path (0) ++(30:2) node[ruler compass/point=red,label={a}]; \path (0) ++(70:2.7) node[ruler compass/point=red,label={b}]; \compass{0}{a} \compass{a}{b} \point{c0a}{cab}{1} \point{c0a}{cab}{2} \draw[ultra thick,red] (c) arc[centred arc to={a}{d}] arc[arc flip,centred arc to={0}{c}]; \end{tikzpicture} \end{example} \caption{Drawing an arc.} \label{ex:arcs} \end{figure} \section{Miscellaneous Extras} As a little extra (because I found them useful when using this library), this library also provides the code used in the answers to \href{http://tex.stackexchange.com/a/20426/86}{``Z-level in \tikzname''} and \href{http://tex.stackexchange.com/a/43946/86}{How can I force \tikzname\ pin angle?}. The first provides a per-{}path and per-{}node layering system, the second helps with label positioning. The extra keys are: \begin{itemize} \item \Verb+on layer=+ This puts the current path on a particular layer. \item \Verb+node on layer=+ This is the same, but for nodes. \item \Verb+reset label anchor=+ This turns off \tikzname's automatic label anchoring to make it possible to specify a particular anchor to use. \end{itemize} \section{Beamer Examples} The following show some examples using \Verb+beamer+ to step through. As these are ``full screen'', the page is used as the bounding box and so the \Verb+constrain+ key is not needed. \subsection{Addition of Points} \lstinputlisting{rc_example_add.tex} The final picture looks something like the following. \begin{center} \begin{tikzpicture}[ stop jumping, constrain % max size={\textwidth}{\textheight}, ] % Mark two points: one at the centre of the page and one a random offset. These define our unit of length for constructions \path (0,0) node[name=0,ruler compass/point=red,label={0}]; \path (0) ++(\ptaa:\ptra) node[ruler compass/point=red,label={a}]; \path (0) ++(\ptab:\ptrb) node[ruler compass/point=red,label={b}]; \ruler{0}{b} \ruler{0}{a} \compass{0}{a} \point{r0b}{c0a}{2} \compass{0}{b} \point{r0a}{c0b}{2} \compass{a}{0} \point{ca0}{c0a}{1} \point{ca0}{c0a}{2} \ruler{e}{f} \point[label={[above]\(a/2\)}]{r0a}{ref}{1} \compass{g}{d} \point{r0a}{cgd}{1} \compass{a}{h} \compass{b}{0} \point{cb0}{c0b}{1} \point{cb0}{c0b}{2} \ruler{i}{j} \point[label={[above]\(b/2\)}]{r0b}{rij}{1} \compass{k}{c} \point{r0b}{ckc}{1} \compass{b}{l} \point[fill=red,label={[above]\(a+b\)}]{cbl}{cah}{1} \end{tikzpicture}% \end{center} \subsection{Multiplication of Points} \lstinputlisting{rc_example_mult.tex} A comparison of the \Verb+beamer+ version with the following picture will show the wisdom of a step-{}by-{}step approach to these drawings. \begin{center} \begin{tikzpicture}[ stop jumping, constrain ] % Mark two points: one at the centre of the page and one a random offset. These define our unit of length for constructions \path (0,0) node[name=0,ruler compass/point=red,label={0}]; \path (1,0) node[name=1,ruler compass/point=red,label={1}]; \path (0) ++(\ptaa:\ptra) node[ruler compass/point=red,label={a}]; \path (0) ++(\ptab:\ptrb) node[ruler compass/point=red,label={b}]; \ruler{0}{1} \ruler{0}{b} \compass{0}{1} \point{c01}{r0b}{1} \compass{1}{a} \point{c1a}{r01}{1} \compass{0}{d} \point{c0d}{r0b}{1} \point{c0d}{r0b}{2} \compass{0}{a} \compass{c}{e} \point{c0a}{cce}{2} \point{c0a}{r0b}{2} \ruler{0}{g} \compass{0}{b} \compass{b}{0} \point{c0b}{cb0}{1} \point{c0b}{cb0}{2} \ruler{i}{j} \point{r0b}{rij}{1} \compass{k}{c} \point{r0b}{ckc}{1} \compass{0}{l} \compass{l}{0} \point{c0l}{cl0}{1} \point{c0l}{cl0}{2} \ruler{m}{n} \point{r0b}{rmn}{1} \compass{o}{f} \point{cof}{r0b}{1} \compass{o}{h} \point{coh}{r0b}{1} \compass{l}{q} \compass{b}{p} \point{clq}{cbp}{2} \ruler{b}{r} \point[fill=red,label={[above]\(ab\)}]{rbr}{r0g}{1} \end{tikzpicture}% \end{center} \end{document}