UdiB
I want to draw a tikzpicture with nodes positioned relative to text width of the document. To achieve it I am setting a bounding box:
```tex
\documentclass{article}
\synctex=0
\usepackage{tikz,lipsum}
\usetikzlibrary{positioning}
\begin{document}
\lipsum[1]
\vskip 10pt
\hrule
\noindent
\begin{tikzpicture}
\path (0,0) (\textwidth,0);% set bounding box
\node(n1) [draw=black,outer sep=0pt,anchor=north] at (current bounding box.west) {node 1};
\node (n2) [draw=black,outer sep=0pt,anchor=east] at (current bounding box.east) {node 2};
\filldraw (n2.north east) circle (1pt);
\end{tikzpicture}
\end{document}
```
I need help with placing `node 2` flush right with the text area.
data:image/s3,"s3://crabby-images/b0e17/b0e17c2f2f39c454234823f1552e186a960c2c02" alt="Screen Shot 2025-01-24 at 09.01.33.png"
Without any node,
```
\noindent
\begin{tikzpicture}
\path (0,0) (\textwidth,0);% set bounding box
\filldraw (current bounding box.north east) circle (1pt);
\end{tikzpicture}
```
I get the small circle flush right with the text area.
Once I add `node 1`,
```
\noindent
\begin{tikzpicture}
\path (0,0) (\textwidth,0);% set bounding box
\node(n1) [draw=black,outer sep=0pt,anchor=north] at (current bounding box.west) {node 1};
\filldraw (current bounding box.north east) circle (1pt);
\end{tikzpicture}
```
The bounding box extends for some reason to the left of the text area.
It seems to me that the bounding box changes its size whenever I add more content into it. Is that true? Explanation about it would be very welcome.
Top Answer
frougon
As far as I know, the `(0,0)` coordinate has nothing special in Ti*k*Z. Typically, you can use it to place things relatively to one another. When you draw the `n1` node, Ti*k*Z chooses to place `(0,0)` somewhere that does not follow your expectations. This can be shown by replacing your `\path` with a `\draw` statement (I added a `\smallskip` to make the horizontal rules easy to distinguish):
```
\documentclass{article}
\usepackage{tikz,lipsum}
\begin{document}
\lipsum[1]
\vskip 10pt
\hrule
\smallskip\noindent
\begin{tikzpicture}
\draw[thick, green] (0,0) -- (\textwidth,0);
\node (n1) [draw=black,outer sep=0pt,anchor=north] at (current bounding box.west) {node 1};
\node (n2) [draw=black,outer sep=0pt,anchor=east] at (current bounding box.east) {node 2};
\filldraw (n2.north east) circle (1pt);
\end{tikzpicture}
\end{document}
```
data:image/s3,"s3://crabby-images/c5fc3/c5fc3b9639a0a45e6564e6a2a11416dc228005a9" alt="image.png"
One can improve things by:
- capturing the bounding box created by drawing the rule that lives in the `tikzpicture`;
- using the `(0,0)` coordinate (or any other one) for positioning the left box relatively to the rule.
```
\documentclass{article}
\usepackage{tikz,lipsum}
\begin{document}
\lipsum[1]
\vskip 10pt
\hrule
\smallskip
\noindent
\begin{tikzpicture}
\begin{scope}[local bounding box=bb]
\draw[thick, green] (0,0) -- (\textwidth,0);
\end{scope}
\node[draw, anchor=west] at (0,0) {node 1};
\node (n2) [draw, anchor=east] at (bb.east) {node 2};
\filldraw (n2.north east) circle[radius=1pt];
\end{tikzpicture}
\end{document}
```
data:image/s3,"s3://crabby-images/1e32d/1e32d1ea6cfc7bb3ccddad9706620019222eb5cd" alt="image.png"
However, this is imperfect: there are overfull boxes due to rule widths, line caps and, obviously, the little disc. My recommendation would be to use the `tikzpagenodes` Ti*k*Z library at least for the horizontal positioning, along with `remember picture` and `overlay` options to:
* allow Ti*k*Z coordinates to be written to and read back from the .aux file;
* avoid taking up space on the page (thus, no overfull box here).
Here, I use `\tikzmark` to obtain the vertical position immediately below your `\hrule`:
```
\documentclass{article}
\usepackage{tikz,lipsum,tikzpagenodes}
\usetikzlibrary{tikzmark}
\begin{document}
\lipsum[1]
\vskip 10pt
\hrule
\vbox to 0pt {\hbox{\tikzmark{belowRule}}\vss}
\begin{tikzpicture}[remember picture,overlay]
\node[draw, anchor=north west]
at (current page text area.west |- {pic cs:belowRule}) {node 1};
\node[draw, anchor=north east]
at (current page text area.east |- {pic cs:belowRule}) {node 2};
\end{tikzpicture}
\end{document}
```
data:image/s3,"s3://crabby-images/97cca/97cca9ac054f65c515e0f521e84965c1648aa833" alt="image.png"
If you want some vertical shift below the rule, you can use for instance:
```
\node[draw, anchor=north west]
at ([yshift=-1ex]current page text area.west |- {pic cs:belowRule}) {node 1};
\node[draw, anchor=north east]
at ([yshift=-1ex]current page text area.east |- {pic cs:belowRule}) {node 2};
```
data:image/s3,"s3://crabby-images/a1a94/a1a944def929f39dd076742f1a276a28f53ade6f" alt="image.png"
# Suggested reading
Regarding how Ti*k*Z places a `tikzpicture` relatively to the surrounding TeX material, I suggest you to read in the [Ti*k*Z & PGF manual](https://mirrors.ctan.org/graphics/pgf/base/doc/pgfmanual.pdf) §12.2.1, the documentation of the `/tikz/baseline` option and the preceding text starting with “At the end of the environment, PGF tries to make a good guess at the size of a bounding box (...)”).
(Things are different when the `overlay` option is used, as the `tikzpicture` is supposed to take no space at all for TeX's typesetting algorithm in this case.)
Note: running `texdoc pgfmanual` from a command prompt is likely to bring up the Ti*k*Z & PGF manual without downloading anything if your TeX distribution installed it.
Answer #2
samcarter
- If you anchor the first node `north`, half of it will "stick out" to the left side. This will make your tikz drawing wider by half the width of the node. You can see what's happening more clearly with `\node(n1) [draw=black,outer sep=0pt,anchor=north,overlay] at (current bounding box.west) {node 1};`:
data:image/s3,"s3://crabby-images/8d9b8/8d9b828cc21cad3b59e5af864f9fa53c86840bb7" alt="Screenshot 2025-01-24 at 10.12.01.png"
- TikZ normally adds line caps to lines, this means that a line of `\textwidth` is actually a tiny bit longer than the text width.
### Solution 1
```
\documentclass{article}
\usepackage{tikz}
\usepackage{lipsum}
\begin{document}
\lipsum[1]
\vskip 10pt
\hrule
\noindent
\begin{tikzpicture}
\path (0,0) (\textwidth-1.4pt,0);% set bounding box
\node[draw=black,outer sep=0pt,anchor=west] (n1) at (current bounding box.west) {node 1};
\node[draw=black,outer sep=0pt,anchor=east] (n2) at (current bounding box.east) {node 2};
\end{tikzpicture}%
\end{document}
```
data:image/s3,"s3://crabby-images/019b0/019b061c41e2c6b2c74259a862a69d2285351f14" alt="document-1.png"
### Solution 2
In case you just need to position the nodes at the width of the text and not necessarily at the exact position where the text block ends, you could use the `tikzpagenodes` package:
```
\documentclass{article}
\usepackage{lipsum}
\usepackage{tikzpagenodes}
\begin{document}
\begin{tikzpicture}[overlay,remember picture]
\node[draw=black,outer sep=0pt,anchor=north west] (n1) at (current page text area.south west) {node 1};
\node[draw=black,outer sep=0pt,anchor=north east] (n2) at (current page text area.south east) {node 2};
\end{tikzpicture}%
\lipsum
\end{document}
```
data:image/s3,"s3://crabby-images/20160/201603f7c952a29450fe6b70242e06d654865602" alt="document-1.png"