有时候,我们需要对文中的内容做进一步的解释;有时候,我们会想在文章边注的区域内,给特定的内容加上一个俏皮话(如下图)。
这篇文章,我们将用 TikZ 实现这个效果。
TikZ 的知识
我们知道,在 tikzpicture
环境中,类似上面的效果,很容易就能实现。
tikzpicture_sample.tex1 2 3 4 5 6 7 8 9 10 11
| \documentclass{article} \usepackage{tikz} \usetikzlibrary{positioning} \tikzset{>=stealth} \begin{document} \begin{tikzpicture}[node distance = 1.5cm] \node (test) {I'm a soldier!}; \node (testDesc) [above right = of test] {Yes, you are!}; \draw [->,thick] (testDesc) to [in = 60, out = -120] (test); \end{tikzpicture} \end{document}
|
在这里,我们引入了 tikz
宏包,以及它的 positioning
库,用来绘制和定位 node
s。在 tikzpicture
中,我们建立了两个 node
: test
和 testDesc
,后者的位于前者的右上方。最后,我们用绘制了从 testDesc
到 test
的曲线箭头,其样式由之前的 \tikzset{>=stealth}
指定。
这是 TikZ 的基本应用,不必多言。然而我们的需求,是将位于正文中的文字(它应该是一个 node
)和正文外 tikzpicture
中的 node
连起来。如果能解决这一点,那么我们就能将未知问题转换为已知问题。
我们从 TeX 的执行过程和 TikZ 出发,思考一下,为了解决这个问题,需要如何操作。首先,我们需要记录 test
和 testDesc
的位置。由于我们不可能将这个位置信息直接写入输出的 PDF 文件中,所以我们需要将它写入辅助文件中。这意味着,为了正确实现我们需要的效果,我们至少应该编译两次源文件。其次,对于连接 test
和 testDesc
的箭头来说,它的边界(bounding box)需要特别处理——如果按照正常的方式处理,那么箭头和正文部分就不能重叠。
所幸,TikZ 已经为我们做好了这些工作。我们需要它提供的 remember picture
和 overlay
连个选项。它们的作用是:
remember picture
: 将位置信息写入辅助文件,供后续使用;
overlay
: 不计算边界,允许与其它内容重叠。
实际实现看看
首先,我们来实现 test
的部分。这部分比较通用,本质上就是用 TikZ 给几个单词打上 node
标记的过程。于是我们可以定义一个命令
1 2 3
| \newcommand{\tikzmark}[3][] {\tikz[remember picture, baseline] \node [anchor=base,#1](#2) {#3};}
|
注意,这里我们用了 remember picture
选项,确保 \tikzmark
的位置会被保存下来。之后,在写注释的时候,就可以引用 \tikzmark
的位置了。这里有一个简单的实现
tikz_comment_sample.tex1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| \documentclass{article} \usepackage{tikz} \usetikzlibrary{positioning} \tikzset{>=stealth}
\newcommand{\tikzmark}[3][] {\tikz[remember picture, baseline] \node [anchor=base,#1](#2) {#3};}
\usepackage{mwe} \begin{document} \blindtext \tikzmark{test}{I'm a soldier!} \blindtext
\begin{tikzpicture}[overlay, remember picture, node distance = 1.5cm] \node (testDesc) [above left = of test, xshift = -1cm] {Yes, you are!}; \draw [->,thick] (testDesc) to [in = 120, out = -60] (test); \end{tikzpicture} \end{document}
|
在写注释的时候,我们给 tikzpicture
环境加上了 overlay
选项。这是因为从 testDesc
到 test
的箭头应该可以与其它正文重叠。这段代码的效果,就是文章开头的那个样子。
还能用在数学公式里?
是的,\tikzmark
也可以写在数学公式里。
comment_to_equation.tex1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| \documentclass{article} \usepackage{amsmath} \usepackage{tikz} \usetikzlibrary{positioning} \tikzset{>=stealth}
\newcommand{\tikzmark}[3][] {\tikz[remember picture, baseline] \node [anchor=base,#1](#2) {#3};}
\begin{document} \[ \mathcal{A} = (\tikzmark{identity}{\texttt{I}} -\tikzmark[red]{G}{\texttt{G}} \tikzmark[blue]{L}{\texttt{L}} - \tikzmark[purple]{C}{\texttt{C }}) \] \begin{tikzpicture}[overlay, remember picture,node distance =1.5cm] \node (identitydescr) [below left=of identity ]{words}; \draw[,->,thick] (identitydescr) to [in=-90,out=90] (identity); \node[red] (Gdescr) [below =of G]{other words}; \draw[red,->,thick] (Gdescr) to [in=-90,out=90] (G); \node[blue,xshift=1cm] (Ldescr) [above right =of L]{some words}; \draw[blue,->,thick] (Ldescr) to [in=45,out=-90] (L.north); \node[purple] (Cdescr) [below right =of C]{more words}; \draw[purple,->,thick] (Cdescr) to [in=-90,out=90] (C.south); \end{tikzpicture} \end{document}
|
你可以试着编译上面的代码,将得到以下效果
期待你能做出更多有趣的效果。:)