Я пишу документ (написанный в формате plainTeX) о жизни выдающихся математиков и физиков в истории человечества.
Теперь у меня есть проблема, которую я хотел бы решить. Мне нужно уметьвизуализироватьих продолжительность жизни (чтобы читатель мог видеть, кто был чьим современником/современником). К сожалению, я не настолько искусен в создании макросов TeX. Может кто-нибудь подсказать, как это сделать?
В частности, я предполагаю, что это будет работать следующим образом:
\person{name}{surname}{year_of_birth}{year_of_death}
\person{name}{surname}{year_of_birth}{year_of_death}
...
\renderpeople
что произведет (например, с примерами Баха и Моцарта)
Bach J.S. |----------------|
Mozart A. |-----------------|
...
|----------|-----------|----------|-----------|---------|
1750 1800 1850 1900 1950 2000
В основном, как реализовать изменение масштаба строк, чтобы они поместились?
решение1
Быстрый хак в простом TeX с использованием vanilla TeX. Сначала собираются данные о людях и вычисляются некоторые значения (минимальный год, максимальный год, ширина имен), затем собранные данные о людях устанавливаются и, наконец, печатается вся временная шкала.
\newdimen\NameWidth
\NameWidth=0pt
\newcount\MinYear
\MinYear=100000
\newcount\MaxYear
\MaxYear=-100000
\newtoks\PersonData
\PersonData={}
\newdimen\ScaleWidth
\newdimen\ScaleUnit
\newcount\TempCount
\newdimen\TickMarkWidth
\TickMarkWidth=.4pt
\newdimen\TickMarkHeight
\TickMarkHeight=8pt
\newdimen\LineWidth
\LineWidth=.4pt
\newdimen\LastYearCorr
\def\person#1#2#3#4{%
\setbox0=\hbox{\makeperson{#1}{#2}\personsep}%
\ifdim\wd0>\NameWidth
\NameWidth=\wd0 %
\fi
\ifnum#3<\MinYear
\MinYear=#3\relax
\fi
\ifnum#4>\MaxYear
\MaxYear=#4\relax
\fi
\ifnum#3>#4\relax
\errmessage{#2 #1 has negative live span, born in #3 and died in #4}%
\fi
\PersonData=\expandafter{\the\PersonData
\DoPerson{#1}{#2}{#3}{#4}%
}%
}
\def\makeperson#1#2{#1, #2}%
\def\personsep{ }
\def\renderpeople{%
\par
\divide\MinYear by 50\relax
\multiply\MinYear by 50\relax
\advance\MaxYear by 49\relax
\divide\MaxYear by 50\relax
\multiply\MaxYear by 50\relax
\ScaleWidth=\hsize
\advance\ScaleWidth by -\NameWidth
\setbox0=\hbox{$\textstyle\the\MaxYear$}%
\LastYearCorr=.5\wd0 %
\advance\ScaleWidth by -\LastYearCorr
\ScaleUnit=\ScaleWidth
\TempCount=\MaxYear
\advance\TempCount by -\MinYear
\divide\ScaleUnit by \TempCount
\ScaleWidth=\ScaleUnit
\multiply\ScaleWidth by \TempCount
\noindent
\the\PersonData
\ScaleLine
\par
}
\def\DoPerson#1#2#3#4{%
\hbox to \hsize{%
\hbox to \NameWidth{%
\makeperson{#1}{#2}%
\hfill
\personsep
}%
\hfill
\hbox to \ScaleWidth{%
\kern#3\ScaleUnit
\kern-\MinYear\ScaleUnit
\SetTickMark
\TempCount=#4\relax
\advance\TempCount by -#3\relax
\SetLine\TempCount
\SetTickMark
\hfill
}%
\kern\LastYearCorr
}%
\hskip0pt\relax
}
\def\ScaleLine{%
\hbox to \hsize{%
\hbox to \NameWidth{\hfill}%
\hfill
\hbox to \ScaleWidth{%
\TempCount=\MinYear
\SetTickYearMark
\loop
\ifnum\TempCount<\MaxYear
\SetLine{50}%
\advance\TempCount by 50 %
\SetTickYearMark
\repeat
}%
\kern\LastYearCorr
}%
\hskip0pt\relax
}
\def\SetTickYearMark{%
\hbox to 0pt{%
\hss
$\mathsurround=0pt\relax
\mathop{\vcenter{%
\hrule width \TickMarkWidth
height .5\TickMarkHeight
depth .5\TickMarkHeight
}}\limits_{\textstyle\the\TempCount}%
$%
\hss
}%
}
\def\SetTickMark{%
\hbox to 0pt{%
\hss
$\mathsurround=0pt\vcenter{%
\hrule width \TickMarkWidth
height .5\TickMarkHeight
depth .5\TickMarkHeight
}$%
\hss
}%
}
\def\SetLine#1{%
$\mathsurround=0pt\vcenter{%
\hrule width#1\ScaleUnit
height.5\LineWidth
depth.5\LineWidth
}$%
}
\person{Pachelbel}{J.}{1653}{1706}
\person{Bach}{J. S.}{1685}{1750}
\person{Mozart}{W. A.}{1756}{1791}
\person{Euler}{L.}{1707}{1783}
\renderpeople
\bye
Замечания:
Я бы предпочел e-TeX,
\dimexpr
и\numexpr
если они доступны, они более удобны. Однако деления отличаются, потому что они\...expr
округляют результат.Мощный вариант
pgf/TikZ
также доступен в простом TeX.
решение2
Вот похожее решение TeX:
\input pst-grad
\input pstricks-add
\newbox\TBox
\psset{gradbegin=white,gradend=lightgray}
\catcode`\@11\relax
\def\BoxText{\@ifnextchar[\BoxText@i{\BoxText@i[3cm]}}
\def\BoxText@i[#1]#2#3#4{%
\emergencystretch=3em
\setbox\TBox\vbox{\hsize #1 #4\par}
\rput[t](#2){\psframebox[fillstyle=gradient]{\leavevmode\copy\TBox\relax}}%
\pnode(#2|0,0){A}\pnode(#2){B}
\ncline{->}{A}{B}%
\uput*{0.5cm}[-90](A){#3}%
}
\catcode`\@12\relax
\pspicture(0,0.25)(2.5,-10)
\psset{xunit=5}
\psaxes[yAxis=false,Ox=1518]{|->}(0,0)(2.5,-10)%
\BoxText[4cm]{1.1123,-5.5}{10.2.}{%
Hier muss jetzt irgendetwas hinkommen, was, weiss ich auch nicht genau}
\BoxText{.767,-2}{15.10.}{%
Hier muss jetzt irgendetwas hinkommen, was, weiss ich auch nicht genau}
\endpspicture
\pspicture(0,0.25)(2.5,-10)
\psset{xunit=5cm}
\psaxes[yAxis=false,Ox=1518]{|->}(0,0)(2.5,-10)%
\BoxText{.787,-5}{17.10.}{%
Hier muss jetzt irgendetwas hinkommen, was, weiss ich auch nicht genau}
\pnode(A){A0}
\BoxText{.767,-2}{15.10.}{%
Hier muss jetzt irgendetwas hinkommen, was, weiss ich auch nicht genau}
\uput*{1.2cm}[-90](A0){17.10.}%
\endpspicture
\bye