1 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 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102
|
%
% \iffalse
% psfrag.dtx Copyright (C) 1996 Craig Barratt, Michael C. Grant,
% and David Carlisle.
% All rights are reserved.
%
% This system is distributed in the hope that it will be
% useful, but WITHOUT ANY WARRANTY; without even the
% implied warranty of MERCHANTABILITY or FITNESS FOR A
% PARTICULAR PURPOSE. Don't come complaining to us if you
% modify this file and it doesn't work! If this file is
% modified by anyone but the authors, those changes and
% their authors must be explicitly stated HERE.
%
%<package>\NeedsTeXFormat{LaTeX2e}[1995/12/01]
%<package>\ProvidesPackage{psfrag}[1998/04/11 v3.04 PSfrag (MCG)]
% \fi
%
% \iffalse
%<*driver>
\documentclass{ltxdoc}
\usepackage{graphicx,psfrag}
\begin{document}
\DocInput{psfrag.dtx}
\end{document}
%</driver>
% \fi
%
% \GetFileInfo{psfrag.sty}
% \RecordChanges
%
% \let\pkg\textsf
% \let\fname\texttt
% \let\pscom\texttt
% \newcommand{\pfg}{\textsf{PSfrag}}
% \newcommand{\bsl}{\protect\bslash}
% \newcommand{\ie}{\emph{i.e.}}
% \newcommand{\eg}{\emph{e.g.}}
% \newcommand{\etc}{\emph{etc.}}
% \newcommand{\netaddress}[1]{\texttt{#1}}
% \marginparsep 0pt
%
% \title{\fname{psfrag.sty} and \fname{psfrag.pro}%^^A
% \thanks{This file has version number \fileversion,
% last revised \filedate.}}
%
% \author{Michael Grant, David Carlisle, and Craig Barratt \\
% \netaddress{psfrag@rascals.stanford.edu}}
% \date{\filedate}
% \maketitle
% \PrintChanges
%
% \changes{3.04}{1998/04/11}{Made some (theoretically) back-compatible
% changes to \fname{psfrag.pro} which should make it more friendly
% to EPS and page-manipulation packages like \pkg{psnup}. Fixed
% orientation problem for DVIPSone. Added support for EPS files
% as \pfg\ replacements.}
% \changes{3.03}{1997/01/21}{Fixed some typographical errors that
% broke the \pscom{awidthshow} command. Improved the scanner to
% handle multi-line \cs{tex} commands. Fixed a bug in the
% \pscom{replace} command which would break some legal EPS files.}
% \changes{3.02}{1996/12/06}{Continued to improve the transformation
% code to make it more robust to rotations and odd scaling.
% Improved the range of strings accepted as PSfrag tags.}
% \changes{3.01}{1996/11/26}{Expanded the PSfrag dictionary
% for Level 1 printers. Fixed bugs that produce incorrect output
% for rotated figures. Output should now print properly when
% combined with page-manipulation programs like \pkg{psnup}.}
% \changes{3.0}{1996/10/31}{Major rewrite. Elimination of the
% preprocessing step, deprecation of the \pscom{tex} command. I've
% wiped most of the source-level change entries as a result.}
% \changes{2.0}{1995/04/04}{Bug fixes}
% \changes{2.0}{1995/03/14}{Bug fixes. Better support for \LaTeX\ 2.09
% \pfg\ files, including both naming schemes. Improved
% documentation and separated it from this file.}
% \changes{1.99}{1995/02/14}{Taken over from Craig, completely
% overhauled for \LaTeXe. Completely changed the placement code;
% baseline alignment added. Integrated \pscom{psfrag} commands into
% the PostScript file. \textsf{graphics} package used for
% portability across multiple DVI-to-PostScript packages.
% Improved ps2frag.ps file provides improved (correct) support
% for ashow, widthshow, awidthshow, and kshow.}
% \changes{1.1}{1992/10/20}{Released version 1.1, adding support for
% ashow, widthshow, awidthshow, and kshow.}
% \changes{1.0}{1992/06/01}{Released version 1.0.}
%
% \section{\fname{psfrag.sty}}
%
% \begin{macrocode}
%<*package>
% \end{macrocode}
%
% \subsection{Intialization}
%
% \begin{macrocode}
\newif\ifpfg@compat
\newif\ifpfg@prepass
\newif\ifpfg@debug \pfg@debugfalse
\if@compatibility
\pfg@compattrue\pfg@prepasstrue
\else
\pfg@compatfalse\pfg@prepassfalse
\fi
\def\psfragscanon{\pfg@prepasstrue}
\def\psfragscanoff{\pfg@prepassfalse}
\def\psfragdebugon{\pfg@debugtrue}
\def\psfragdebugoff{\pfg@debugfalse}
% \end{macrocode}
%
% \begin{macrocode}
\DeclareOption{scanall}{\pfg@prepasstrue}
\DeclareOption{2emode}{\pfg@compatfalse\pfg@prepassfalse}
\DeclareOption{209mode}{\pfg@compattrue\pfg@prepasstrue}
\DeclareOption{debugshow}{\pfg@debugtrue
\PassOptionsToPackage\CurrentOption{graphics}}
\DeclareOption*{\PassOptionsToPackage\CurrentOption{graphics}}
\ProcessOptions
% \end{macrocode}
%
% \fname{psfrag.sty} requires the \pkg{graphics} package and the
% support file \fname{psfrag.pro}. (\fname{epsf.sty} can still be
% used,but \pkg{graphics} package is still used for internal
% purposes.) To load \fname{psfrag.pro} in a portable way, we provide
% and use the \cs{Gin@PS@file@header} command, which the
% \pkg{graphics} will eventually provide for us.
% \changes{3.01}{1996/11/25}{Added the \cs{AtBeginDocument} command
% to save some important global transformation information.}
% \changes{3.02}{1996/12/03}{Deleted this \cs{AtBeginDocument} info;
% see \cs{pfg@orient}.}
% \begin{macrocode}
\RequirePackage{graphics}
\providecommand\Gin@PS@file@header[1]{\AtBeginDvi{\special{header=#1}}}
\Gin@PS@file@header{psfrag.pro}
% \end{macrocode}
%
% \begin{macro}{\pfg@orient}
% In order to properly handle all cases of rotation, scaling,
% \etc\ that \pfg\ may encounter, it is necessary that we know a bit
% about the transformations that the DVI to PS driver uses itself.
% Specifically, we need to know what transformations, if any, the
% DVI to PS driver appends to PostScript's \pscom{defaultmatrix}
% before it starts to typeset text. At the moment, all of the drivers
% which work with \pfg\ reverse the orientation of the $y$ axis;
% new drivers may not. But until we can find a portable way to make
% this determination in PostScript itself, we have to supply that
% information here.
% \changes{3.02}{1996/12/03}{Created. Added \pkg{dvips} and
% \pkg{textures} as ``$y$-reversers.''}
% \changes{3.04}{1998/04/11}{\pkg{dvipsone} now uses the same
% orientation as \pkg{dvips}, so we added a line for it here.}
% \begin{macrocode}
\def\pfg@orient{1}
\def\@tempa{dvipsone.def}\ifx\Gin@driver\@tempa\def\pfg@orient{-1}\fi%
\def\@tempa{dvips.def}\ifx\Gin@driver\@tempa\def\pfg@orient{-1}\fi%
\def\@tempa{textures.def}\ifx\Gin@driver\@tempa\def\pfg@orient{-1}\fi%
% \end{macrocode}
% \end{macro}
%
% \subsection{Interface to the \pkg{graphics}
% and \pkg{epsf} packages}
%
% \begin{macro}{\pfg@epsfbox}
% The original definition of \cs{epsfbox} from \pkg{epsf.sty} or \pkg{graphics}.
% \begin{macro}{\pfg@Ginclude@eps}
% The original definition of \cs{Ginclude@eps} from \pkg{graphics},
% the key routine inside the \cs{includegraphics} routine for EPS files.
%
% Below we redefine these two commands to add \pfg\ functionality.
% \begin{macrocode}
\let\pfg@epsfbox\epsfbox
\let\pfg@Ginclude@eps\Ginclude@eps
% \end{macrocode}
% \end{macro}\end{macro}
%
% \begin{macro}{\pfg@pcount}
% Counts how many fragments are active in the current figure.
% \begin{macro}{\pfg@scount}
% Counts how many replacements strings are active in the current
% figure.
%
% Since it is possible for a single string to accept multiple
% replacements, \cs{pfg@scount} $\le$ \cs{pfg@pcount}.
% \begin{macrocode}
\newcount\pfg@pcount \pfg@pcount=\z@
\newcount\pfg@scount \pfg@scount=\z@
% \end{macrocode}
% \end{macro}\end{macro}
%
% \begin{macro}{\epsfbox}
% Here we intercept calls to \cs{epsfbox} and wrap them with the
% \pfg\ processing. The first thing we do is revert \cs{epsfbox}
% and \cs{Ginclude@eps} back to their original definitions, to
% prevent recursion---which could occur if some of the replacements
% are themselves EPS files.
% \changes{3.04}{1998/04/11}
% {Improved the recursion protection by reverting
% \cs{epsfbox} and \cs{Ginclude@eps} back to their original
% definitions within the block. Deleted \cs{ifpfg@epsf}, which
% is no longer needed in this new system.}
% \begin{macrocode}
\def\epsfbox#1{%
\begingroup
\let\Ginclude@eps\pfg@Ginclude@eps
\let\epsfbox\pfg@epsfbox
\ifpfg@prepass\pfg@scan{#1}\fi
\ifnum\pfg@scount>\z@\pfg@hidestart\fi
\pfg@epsfbox{#1}%
\ifnum\pfg@scount>\z@\pfg@hideend\fi
\endgroup}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\Ginclude@eps}
% We need to override this function from the \pkg{graphics}
% package in a similar fashion to \cs{epsfbox}. As we did for
% \cs{epsfbox}, we revert \cs{epsfbox} and \cs{Ginclude@eps}
% back to their original definitions in order to prevent recursion.
% \changes{3.01}{1996/11/26}
% {Added the \cs{pfg@hideinit} functionality.}
% \changes{3.02}{1996/12/01}
% {Removed \cs{pfg@hideinit}.}
% \changes{3.04}{1998/04/11}
% {Improved the recursion protection by reverting
% \cs{epsfbox} and \cs{Ginclude@eps} back to their original
% definitions within the block.}
% \begin{macrocode}
\def\Ginclude@eps#1{%
\begingroup
\let\Ginclude@eps\pfg@Ginclude@eps
\let\epsfbox\pfg@epsfbox
\ifpfg@prepass\pfg@scan{#1}\fi
\ifnum\pfg@scount>\z@\pfg@hidestart\fi
\pfg@Ginclude@eps{#1}%
\ifnum\pfg@scount>\z@\pfg@hideend\fi
\endgroup}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{psfrags}
% Defines a dummy environment to delimit the scope of \pfg\
% replacements.
% \begin{macrocode}
\newenvironment{psfrags}{\ignorespaces}{\global\@ignoretrue}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\psfragspecial}
% This macro is no longer supported.
% \begin{macrocode}
\ifpfg@compat
\def\psfragspecial#1#2#3#4#5#6{%
\PackageError{PSfrag}%
{\string\psfragspecial no longer implemented.}{}}
\fi
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\psfrag}
% \marg{tag}\oarg{posn}\oarg{psposn}\oarg{scl}\oarg{rot}\\
% The main macro \cs{psfrag} must be defined specially in order
% to handle its odd combination of optional and required arguments.
% In addition, catcode tricks (thanks to David Carlisle and others)
% allow special characters to be used inside the tag text. It also
% replaces spaces with their octal equivalents to prevent drivers
% like \pkg{dvips} from breaking one in the middle.
% \changes{3.01}{1996/11/26}{Now changes every space in the tag
% text by its octal code. Some DVI to PS drivers were breaking the
% strings onto multiple lines; this should prevent that from happening.}
% \changes{3.01}{1996/11/26}{Removed octal code in favour of convert
% routine in the prologue}
% \changes{3.03}{1997/1/6}{Added another ``special'' character.}
% \begin{macrocode}
\newif\ifpfg@star
\def\psfrag{\@ifstar{\pfg@startrue\@psfraga}{\pfg@starfalse\@psfraga}}
% \end{macrocode}
%
% \begin{macrocode}
\def\@psfraga{\begingroup
\@makeother\"\@makeother\*\@makeother\!\@makeother\~%
\@makeother\:\@makeother\\\@makeother\%\@makeother\#%
\@makeother\ \@psfragb}
% \end{macrocode}
%
% \begin{macrocode}
\ifpfg@compat
\def\@psfragb#1{\xdef\@gtempa{#1}\endgroup
\@ifnextchar[{\@psfragc{\@gtempa}}%
{\@psfrag{\@gtempa}[bl][bl][1][0]}}
\def\@psfragc#1[#2]{\@ifnextchar [{\@psfragd{#1}[#2]}%
{\@psfrag{#1}[#2][bl][1][0]}}
% \end{macrocode}
%
% \begin{macrocode}
\else
\def\@psfragb#1{\xdef\@gtempa{#1}\endgroup
\@ifnextchar [{\@psfragc{\@gtempa}}%
{\@psfrag{\@gtempa}[Bl][Bl][1][0]}}
\def\@psfragc#1[#2]{\@ifnextchar [{\@psfragd{#1}[#2]}%
{\@psfrag{#1}[#2][Bl][1][0]}}
\fi
% \end{macrocode}
%
% \begin{macrocode}
\def\@psfragd#1[#2][#3]{\@ifnextchar [{\@psfrage{#1}[#2][#3]}%
{\@psfrag{#1}[#2][#3][1][0]}}
\def\@psfrage#1[#2][#3][#4]{\@ifnextchar [{\@psfrag{#1}[#2][#3][#4]}%
{\@psfrag{#1}[#2][#3][#4][0]}}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\@psfrag}\marg{tag}\\
% Now that we have all of the arguments, we need to check to check to
% see if the string has been used yet; if not, increment the string
% count and create a place for the new string. Then pass the string
% number on down the line to \cs{@@psfrag}.
% \begin{macrocode}
\def\pfg@newstring#1.#2{%
\@nameedef{pfg@using@#2}{#1}%
\pfg@starfalse\advance\pfg@scount\@ne}
% \end{macrocode}
%
% \begin{macrocode}
\def\@psfrag#1{%
\@ifundefined{pfg@using@#1}%
{\expandafter\pfg@newstring\the\pfg@scount.{#1}}%
{}%
\edef\@tempa{\@nameuse{pfg@using@#1}}%
\expandafter\@@psfrag\@tempa.{#1}}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\@@psfrag}\marg{tag-key}.\marg{tag}\\
% \cs{@@psfrag} passes along the current \emph{replacement} number to
% to \cs{@@@psfrag}, which is doing all of the dirty work.
% \begin{macrocode}
\def\@@psfrag{\expandafter\@@@psfrag\the\pfg@pcount.}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\@@@psfrag}
% \marg{repl-key}.\marg{tag-key}.\marg{tag}\oarg{posn}%
% \oarg{psposn}\oarg{scl}\oarg{rot}\oarg{repl}
% \begin{macro}{\pfg@align}\marg{tag-key}.
% \begin{macro}{\pfg@place}\marg{repl-key}.\\
% \cs{@@@psfrag}, armed with numeric keys for the tag string and the
% replacement, creates a \cs{pfg@place} procedure call for this
% fragment and appends the control point, scale, and rotation
% information to the proper tag's list, found in
% \cs{pfg@align\emph{<tag-key>}.}
% \changes{3.01}{1996/11/26}{Added some extra spaces in
% \cs{pfg@align@} to encourage more benevlolent line-breaking
% by DVI to PS drivers.}
% \begin{macrocode}
\def\@nameedef#1#2{\expandafter\edef\csname #1\endcsname{#2}}
% \end{macrocode}
%
% \begin{macrocode}
\def\@@@psfrag#1.#2.#3[#4][#5][#6][#7]#8{%
\advance\pfg@pcount\@ne
\ifpfg@star\else\@nameedef{pfg@align@#2}{(#3)[}\fi
\@nameedef{pfg@align@#2}{\@nameuse{pfg@align@#2}[#1(#5)#6 #7]}%
\@namedef{pfg@place@#1}{\pfg@@place{#1}{#4}{#8}}\ignorespaces}
% \end{macrocode}
%
% \begin{macrocode}
\def\pfg@align#1.{\@nameuse{pfg@align@#1}}
\def\pfg@place#1.{\@nameuse{pfg@place@#1}}
% \end{macrocode}
% \end{macro}\end{macro}\end{macro}
%
% \subsection{Fragment processing}
%
% \begin{macro}{\pfg@hidestart}
% This procedure provides PostScript with a list of the strings that
% \pfg\ would like to replace, and the control points, scales, and
% rotations of each one. The PostScript code in \fname{psfrag.pro}
% will use this information to construct transformation matrices for
% each replacement to move them into proper position.
% \changes{3.01}{1996/11/26}{Moved the \pkg{xdvi} stub to
% \cs{pfg@hideinit}}.
% \changes{3.02}{1996/12/01}{Moved it back.}
% \begin{macrocode}
\def\pfg@hidestart{%
\def\pfg@{}\count@=\z@\loop\ifnum\count@<\pfg@scount
\toks@=\expandafter{\expandafter\pfg@align\the\count@.}%
\edef\pfg@{\pfg@\the\toks@]}%
\advance\count@ by\@ne\repeat
\Gin@PS@raw{/PSfrag where{pop\pfg@\the\pfg@scount\space
\ifpfg@debug1\else0\fi\space\pfg@orient/Begin
PSfrag}{userdict /PSfrag{pop}put}ifelse}}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\pfg@hideend}
% Now that the EPS file has been loaded, and (therefore) all of the
% replacement positions have been determined, move each replacement
% to its proper position by calling the \cs{pfg@place} procedure for
% each one. Except for the calls to \cs{Gin@PS@raw}, it looks like
% we're building a vertical list of the replacements: that's the idea,
% because we want it to look decent when viewed by a DVI viewer.
% \begin{macrocode}
\def\pfg@hideend{%
\Gin@PS@raw{/End PSfrag}%
\vbox to\z@{\vss
\Gin@PS@raw{/Hide PSfrag}%
\hbox to\z@{\hss \raisebox{\depth}{%
\underline{PSfrag replacements}}}%
\Gin@PS@raw{/Unhide PSfrag}%
\count@=\z@\loop\ifnum\count@<\pfg@pcount
\expandafter\pfg@place\the\count@.%
\advance\count@ by\@ne\repeat}}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\pfg@@place}
% \marg{repl-key}\marg{posn}\marg{replacement}\\
% The following macro does the dirty work. Its job is to insert the
% appropriate code so that the \TeX\ replacement is transformed to
% its requested location; and, if desired, repeated for as many times
% as the PostScript tag text appeared in the figure.
%
% It performs this magic by first surrounding the \TeX-produced code
% in curly braces, so we can defer its execution while we figure out
% where to place it. Then, we call the \pscom{Place} routine in
% \fname{psfrag.pro}, which takes \marg{repl-key}
% and provides the proper coordinate transformations.
%
% We're allowed to reassign these dimension registers because we are
% always inside a group when we use them. This technique is taken from
% \fname{graphics.sty}...
% \begin{macrocode}
\let\pfg@dp\leftmargini\let\pfg@wd\leftmarginii
\let\pfg@dx\leftmarginiii\let\pfg@dy\leftmarginiv
% \end{macrocode}
% We need to stuff the curly braces into macros.
% \begin{macrocode}
\edef\pfg@bchar{ \string{ }
\edef\pfg@echar{ \string} }
\def\pfg@@place#1#2#3{\begingroup
% \end{macrocode}
% Typeset the replacement into a box and determine its dimensions.
% \begin{macrocode}
\sbox\z@{#3}%
\dimen@=\ht\z@ \advance\dimen@ by\dp\z@%
% \end{macrocode}
% Determine the desired alignment of the \TeX\ box, and the glue
% that will be needed in order to achieve this alignment.
% \begin{macrocode}
\pfg@wd=0.5\wd\z@\pfg@dp=0.5\dimen@
\pfg@dx=\pfg@wd\pfg@dy=\pfg@dp
\@tfor\@tempa:=#2\do{%
\if l\@tempa \pfg@dx=\z@ \pfg@wd=\wd\z@ \else
\if r\@tempa \pfg@dx=\wd\z@ \pfg@wd=\z@ \else
\if b\@tempa \pfg@dy=\z@ \pfg@dp=\z@ \else
\if t\@tempa \pfg@dy=\dimen@\pfg@dp=\dimen@\else
\if B\@tempa \pfg@dy=\dp\z@ \pfg@dp=\dp\z@ \fi\fi\fi\fi\fi}%
% \end{macrocode}
% Create a zero-area box with the desired \pfg\ replacement
% text typeset inside of it (and aligned with the glue). This
% is actually the box that will be moved, rotated, and/or scaled.
% \begin{macrocode}
\setbox\z@\hbox to\z@{\hskip-\pfg@dx\box\z@\hss}%
\setbox\z@\vbox to\z@{\vss\box\z@\vskip-\pfg@dy}%
% \end{macrocode}
% Typeset the box. The inner, zero-size box is the one we place
% with the PostScript code, and the outer, full-height, right-\
% justified box is the one that we use to stack up the preview
% list.
% \begin{macrocode}
\vbox to\dimen@{\vss\hbox to\z@{\hss
\Gin@PS@raw{\pfg@bchar}\box\z@
\Gin@PS@raw{\pfg@echar#1/Place PSfrag}%
\hskip\pfg@wd}\vskip\pfg@dp}%
\endgroup}
% \end{macrocode}
% \end{macro}
%
% \begin{macrocode}
%</package>
%<*filepro>
% \end{macrocode}
%
% \section{The PostScript library, \fname{psfrag.pro}}
% This code must perform two functions: 1) determine and save the
% alignment point, rotation, and scaling for each piece of PostScript
% text that is going to serve as a \pfg\ tag; and 2) use that
% information to properly transform the coordinate system for each
% \TeX\ replacement.
%
% The only two symbols that we define globally are \pscom{PSfragLib},
% the library of support routines, \pscom{PSfragDict}, the dictionary
% containing the ``magic'' \pscom{show} commands,
% and \pscom{PSfrag}, the gateway from
% the global namespace to \pscom{PSfragLib}. This technique minimizes
% the possibility that our procedure names will conflict with those
% defined by the PostScript driver or the figure itself.
% \changes{3.02}{1996/12/03}{Added code to insure that \pscom{PSfragLib}
% and \pscom{PSfrag} are stored in \pscom{userdict}. Some cruel DVI to
% PostScript drivers try and force these into their own dictionary. Also
% moved \pscom{PSfragDict} out of \pscom{PSfragLib}.}
% \begin{macrocode}
userdict begin
/PSfragLib 90 dict def
/PSfragDict 6 dict def
/PSfrag { PSfragLib begin load exec end } bind def
end
% \end{macrocode}
%
% \begin{macro}{PSfragLib}
% \pscom{PSfragLib} is the dictionary with the meaty stuff. We're also
% defining some useful abbreviations. Of note is the \pscom{OE} command
% (``outside exec''), which executes some code after removing
% \pscom{PSfragLib} from the stack (to insure that there are no
% collisions with \pscom{PSfragLib}'s internal names.
% \changes{3.01}{1996/11/26}{Fixed \pscom{islev2}.}
% \changes{3.02}{1996/12/03}{Added \pscom{readonly} to \pscom{BD} to
% help insure that this dictionary isn't being trampled on.}
% \changes{3.02}{1996/12/03}{Added some abbreviations.}
% \changes{3.04}{1998/04/11}{Added the default definition of \pscom{S} to
% avert potential problems if it is defined elsewhere. Thanks again to
% J. Scott Berg for this fix.}
% \begin{macrocode}
PSfragLib begin
/RO /readonly load def
/CP /currentpoint load def
/CM /currentmatrix load def
/B { bind RO def } bind def
/X { exch def } B
/MD { { X } forall } B
/OE { end exec PSfragLib begin } B
/S false def
/tstr 8 string def
/islev2 { languagelevel } stopped { false } { 2 ge } ifelse def
% \end{macrocode}
% Allocate some space for matrices that we will use frequently.
% \pscom{sM} is the matrix we encounter at the time this library
% is defined; this should theoretically be \pscom{initmatrix}, unless
% we are in an EPS/psnup situation.
% \pscom{srcFM} contains a matrix which describes the difference
% between PostScript's \pscom{defaultmatrix} and the transformation
% defined by the DVI-to-PS driver's ``default matrix.'' This information
% is crucial to the proper placing of the \pfg\ replacements.
% \changes{3.02}{1996/12/01}{Changed the names to be a tiny bit
% more descriptive, and added a couple.}
% \changes{3.02}{1996/12/03}{Added the \pscom{readonly} commands.}
% \changes{3.04}{1998/04/11}{Added \pscom{sM}, which stores the
% \pscom{currentmatrix} found at the time \pscom{PSfragLib} is defined.
% See the \pscom{replace} command for its use. Thanks to J. Scott Berg.}
% \begin{macrocode}
[ /sM /tM /srcM /dstM /dM /idM /srcFM /dstFM ] { matrix def } forall
sM currentmatrix RO pop
dM defaultmatrix RO idM invertmatrix RO pop
srcFM identmatrix pop
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{Hide}
% \begin{macro}{Unhide}
% Surrounding a (fairly) arbitrary piece of PostScript code with calls
% to \pscom{Hide} and \pscom{Unhide} should render it invisible.
% \pfg\ uses these calls to hide unused replacements. Note that we are
% assuming that the DVI-to-PS driver isn't relying on any of the
% information that will be wiped out by the \pscom{grestore}, except
% for the \pscom{currentpoint}.
% \begin{macrocode}
/Hide { gsave { CP } stopped not newpath clip { moveto } if } B
/Unhide { { CP } stopped not grestore { moveto } if } B
% \end{macrocode}
% \end{macro}\end{macro}
%
% \begin{macro}{setrepl}
% \begin{macro}{getrepl}
% These macros are defined differently for Level 1 and Level 2
% PostScript. \pscom{setrepl} accepts an array of N elements,
% followed by the length N, and store them in a global variable (or
% the equivalent) for later use. \pscom{getrepl} restores that exact
% information to the stack. This is used to store the replacement
% information in a persistent fashion.
% \begin{macrocode}
/setrepl islev2 {{ /glob currentglobal def true setglobal array astore
globaldict exch /PSfrags exch put glob setglobal }}
{{ array astore /PSfrags X }} ifelse B
/getrepl islev2 {{ globaldict /PSfrags get aload length }}
{{ PSfrags aload length }} ifelse B
% \end{macrocode}
% \end{macro}\end{macro}
%
% \begin{macro}{convert}
% This routine takes a string and replaces every character whose
% ASCII code is less than 32 with a space. This makes the replacement
% dictionary robust to multiple-line tags, etc.
% \begin{macrocode}
/convert {
/src X src length string
/c 0 def src length {
dup c src c get dup 32 lt { pop 32 } if put /c c 1 add def
} repeat
} B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{Begin}
% \pfg\ calls this routine once per figure, once it has placed
% on the stack all of each tag's alignment, rotation, and scaling
% information. This information is collected into a dictionary:
% each tag has an entry which is an array of [ repl-key, (psposn),
% scl, rot ] quads. Then it saves the current transformation
% information in a form ready for \pscom{/replace} to use.
% \changes{3.02}{1996/12/03}{\pscom{PSfragDict} now goes at the
% \emph{bottom} of the read-write portion of the dictionary stack,
% to insure that it does not conflict with dictionary manipulation
% by the EPS file or the DVI to PS driver.}
% \changes{3.02}{1996/12/03}{Changes to account for transformation
% rearrangement; see \pscom{/replace}.}
% \begin{macrocode}
/Begin {
/saver save def
srcFM exch 3 exch put
0 ne /debugMode X 0 setrepl
dup /S exch dict def { S 3 1 roll exch convert exch put } repeat
srcM CM dup invertmatrix pop
mark { currentdict { end } stopped { pop exit } if } loop
PSfragDict counttomark { begin } repeat pop
} B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{End}
% End converts the alignment information that has been collected
% into a dictionary: each repl-key that was encountered in
% \pscom{Begin} has an entry which is an array of
% transformation matrices.
% \changes{3.02}{1996/12/03}{Changes to the dictionary placement
% as discussed in \pscom{/Begin}.}
% \begin{macrocode}
/End {
mark { currentdict end dup PSfragDict eq { pop exit } if } loop
counttomark { begin } repeat pop
getrepl saver restore
7 idiv dup /S exch dict def {
6 array astore /mtrx X tstr cvs /K X
S K [ S K known { S K get aload pop } if mtrx ] put
} repeat
} B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{Place}
% This macro is called once for each \pfg\ replacement. It checks to
% see if the replacement's tag was actually encountered in the PS
% figure; if not, it surrounds the code in a ``clip all'' context and
% executes it. This makes it invisible, as requested, but still allows
% it to run so that any side effects that the code may have will take
% place.
%
% If the replacement's tag \emph{was} encountered, however, it
% retrieves the transformation matrix that was calculated for it; it
% prepends this calculation to the current (\TeX) transformation matrix
% (some tricky math to wade through there!), and then draws the
% replacement.
%
% If there are multiple copies of the replacement to be laid down,
% this will do it, surrounding all but the last replacement in a
% \pscom{save}-\pscom{restore} pair. This \pscom{save}-\pscom{restore}
% pair may slow things down, but theoretically it is necessary,
% because the procedure might modify other variables in the PostScript
% namespace. We need to be sure that those modifications do not
% ``accumulate''.
%
% For multiple replacements to work, the code that \pfg\ surrounds with
% curly braces must leave the stack the same way it found it. So far,
% we have confirmed this fact for \pkg{dvips} and \pkg{DVIPSONE}.
% \changes{3.02}{1996/12/03}{Changes to account for transformation
% rearrangement; see \pscom{/replace}.}
% \begin{macrocode}
/Place {
tstr cvs /K X
S K known {
bind /proc X tM CM pop
CP /cY X /cX X
0 0 transform idtransform neg /aY X neg /aX X
S K get dup length /maxiter X
/iter 1 def {
iter maxiter ne { /saver save def } if
tM setmatrix aX aY translate
[ exch aload pop idtransform ] concat
cX neg cY neg translate cX cY moveto
/proc load OE
iter maxiter ne { saver restore /iter iter 1 add def } if
} forall
/noXY { CP /cY X /cX X } stopped def
tM setmatrix noXY { newpath } { cX cY moveto } ifelse
} {
Hide OE Unhide
} ifelse
} B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{normalize}
% This function accepts three numbers, $x$, $y$, and $s$, and returns
% $sx/\sqrt{x^2+y^2}$ and $sy/\sqrt{x^2+y^2}$.
% \begin{macrocode}
/normalize {
2 index dup mul 2 index dup mul add sqrt div
dup 4 -1 roll exch mul 3 1 roll mul
} B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{replace}
% This routine is called for every string in the string list. It
% must determine the translation, scaling, and rotation necessary to
% move a piece of \TeX code to the proper orientation. It stores in
% \pscom{replArray}, for each alignment/scale/rotation combination, a
% transformation matrix that achieves this.
% \begin{macrocode}
/replace {
aload pop MD
% \end{macrocode}
% Trace out the text and determine its bounding box. We need to
% temporarily revert to an identity transformation matrix, so that
% we can be sure to get the tightest bounding box possible. If the
% text is rotated, the bounding box that PostScript produces with the
% \pscom{pathbbox} command is not optimal.
%
% We also need to save the position of the point following the text
% placement, and move to that position when we've finished, to fool
% the PS interpreter into thinking we actually drew that text.
% \changes{3.01}{1996/11/22}{Modified this transformation to put the
% beginning of the text at the origin, and forced its width to be
% $\sim5$ in. Some figures were using coordinate systems that were
% so ``large'' that they would break PostScript with
% ``limitcheck'' errors here.}
% \changes{3.01}{1996/12/03}{The added transformation is no longer
% ``slipped under'' the CTM, but now it is more properly appended.
% this should allow the replacements to move properly if the entire
% figure is rotated or transformed. This required changes throughout
% the code, but primarily here.}
% \changes{3.03}{1997/01/06}{Added a \pscom{gsave}/\pscom{grestore}
% pair around the bounding-box calculations. I thought I was being
% clever by avoiding it; but it wasn't infallible, so I was wrong.}
% \changes{3.04}{1998/04/11}{Changed \pscom{initmatrix} to \pscom{sM}
% \pscom{setmatrix}, to be more EPS-friendly. Thanks to J. Scott Berg.}
% \begin{macrocode}
CP /bY X /lX X gsave sM setmatrix
str stringwidth abs exch abs add dup 0 eq
{ pop } { 360 exch div dup scale } ifelse
lX neg bY neg translate newpath lX bY moveto
str { /ch X ( ) dup 0 ch put false charpath ch Kproc } forall
flattenpath pathbbox [ /uY /uX /lY /lX ] MD
CP grestore moveto
% \end{macrocode}
% If the FontMatrix shows that the font is draw in the opposite X or
% opposite Y direction, then the X or Y coordinates of the bounding
% box, respectively, need to be swapped. This actually occurs quite
% often. Here we also determine the center of the box.
% \begin{macrocode}
currentfont /FontMatrix get dstFM copy dup
0 get 0 lt { uX lX /uX X /lX X } if
3 get 0 lt { uY lY /uY X /lY X } if
/cX uX lX add 0.5 mul def
/cY uY lY add 0.5 mul def
% \end{macrocode}
% If debug mode has been enabled, we draw the bounding box, the
% baseline, and the center lines of the text.
% \begin{macrocode}
debugMode { gsave 0 setgray 1 setlinewidth
lX lY moveto lX uY lineto uX uY lineto uX lY lineto closepath
lX bY moveto uX bY lineto lX cY moveto uX cY lineto
cX lY moveto cX uY lineto stroke
grestore } if
% \end{macrocode}
% Add each replacement's alignment to \pscom{replArray}.
% \changes{3.02}{1996/12/01}{Now saves the current CTM here.}
% \begin{macrocode}
dstFM dup invertmatrix dstM CM srcM
2 { dstM concatmatrix } repeat pop
getrepl /temp X
S str convert get {
% \end{macrocode}
% Retrieve the replacement information
% \begin{macrocode}
aload pop [ /rot /scl /loc /K ] MD
% \end{macrocode}
% Determine the PostScript alignment point.
% \begin{macrocode}
/aX cX def /aY cY def
loc {
dup 66 eq { /aY bY def } { % B
dup 98 eq { /aY lY def } { % b
dup 108 eq { /aX lX def } { % l
dup 114 eq { /aX uX def } { % r
dup 116 eq { /aY uY def } % t
if } ifelse } ifelse } ifelse } ifelse pop
} forall
% \end{macrocode}
% Store the replacement key, and the transformation information, in
% \pscom{replArray}. These transformations are appended to the CTM
% encountered when \pscom{Place} is run. Indeed, this and
% \pscom{Place} are the two most difficult pieces of code to
% understand.
% \changes{3.03}{1996/12/03}{The added transformation is no longer
% ``slipped under'' the CTM, but now it is more properly appended.
% this should allow the replacements to move properly if the entire
% figure is rotated or transformed. This required changes throughout
% the code, but primarily here.}
% \begin{macrocode}
K srcFM rot tM rotate dstM
2 { tM concatmatrix } repeat aload pop pop pop
2 { scl normalize 4 2 roll } repeat
aX aY transform
/temp temp 7 add def
} forall
temp setrepl
} B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{rs}
% \begin{macro}{rks}
% \begin{macro}{ras}
% \begin{macro}{rws}
% \begin{macro}{raws}
% These macros intercept calls to their PostScript
% counterparts and execute \pscom{replace} in their stead if the
% argument string matches one in the replacement list.
% \changes{3.03}{1996/12/17}{Corrected typo for \pscom{raws}.}
% \begin{macrocode}
/Rif {
S 3 index convert known { pop replace } { exch pop OE } ifelse
} B
/XA { bind [ /Kproc /str } B /XC { ] 2 array astore def } B
/xs { pop } XA XC
/xks { /kern load OE } XA /kern XC
/xas { pop ax ay rmoveto } XA /ay /ax XC
/xws { c eq { cx cy rmoveto } if } XA /c /cy /cx XC
/xaws { ax ay rmoveto c eq { cx cy rmoveto } if }
XA /ay /ax /c /cy /cx XC
/raws { xaws { awidthshow } Rif } B
/rws { xws { widthshow } Rif } B
/rks { xks { kshow } Rif } B
/ras { xas { ashow } Rif } B
/rs { xs { show } Rif } B
% \end{macrocode}
% \end{macro}\end{macro}\end{macro}\end{macro}\end{macro}
%
% \begin{macro}{restore}
% In Level 1 PostScript, this is necessary because global variables
% do not exist (and we need one!). Every call to \pscom{restore}
% wipes out any variables that were modified since the previous call
% to \pscom{save}. Unfortunately, this is not guaranteed to work,
% although it does work for \pkg{dvips}.
%
% For Level 2, we can use global variables so this is not necessary.
% We make this distinction \emph{at print time}, so a file that does
% not seem to work on a Level 1 printer for which it was originally
% intended should still work on a Level 2 printer without
% modification.
% \changes{3.04}{1998/04/11}{Added a fix recommended by J. Scott Berg.}
% \begin{macrocode}
/rrs { getrepl dup 2 add -1 roll //restore exec setrepl } B
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{Dict}
% This dictionary is added to the stack just before the figure, so
% that its definitions of the \pscom{show} operators can substitute
% for the originals.
% \changes{3.02}{1996/12/03}{Made \pscom{Dict} read-only to help
% track down problems.}
% \changes{3.03}{1996/12/17}{Corrected typo in \pscom{awidthshow}.}
% \begin{macrocode}
PSfragDict begin
islev2 not { /restore { /rrs PSfrag } B } if
/show { /rs PSfrag } B
/kshow { /rks PSfrag } B
/ashow { /ras PSfrag } B
/widthshow { /rws PSfrag } B
/awidthshow { /raws PSfrag } B
end PSfragDict RO pop
end
% \end{macrocode}
% \end{macro}
%
% \begin{macrocode}
%</filepro>
% \end{macrocode}
%
% \begin{macrocode}
%<*package>
% \end{macrocode}
%
% \section{The scanner}
% This code implements the \TeX-based scanner that performs the work
% that the \pfg\ Perl script used to perform. This is
% very similar to the code used
% to scan for bounding boxes in \fname{graphics.sty}.
%
% We need to read the string with two sets of catcodes: as `verbatim
% text' for using in the first argument of |\psfrag|, and as normal
% \TeX\ commands for using in the replacemnt text argument.
% The only way to do this sensibly in standard \TeX\ is to write
% the string to a file and then read it back. So while reading the
% EPS file, the character |\| will be `active'.
% Then the argument can be duplicated
% before writing this temp file, once with |\| expanding to |\\| (two
% catcode 12 tokens) and once to |\| (one catcode 12 token). When this
% is read back, these tokens will be re-catcoded by the |psfrag|
% command.
% \begin{macrocode}
\newwrite\pfg@temp
% \end{macrocode}
%
% \begin{macro}{\pfg@scan}
% The main command. |#1| is the name of the file. It might be better
% to use the bounding box (`|.bb|') file name so
% that compressed files could work by you copying any lines with
% |(\\tex|\ldots strings in them to the |bb| file before compressing.
% However, if you are
% doing that, you may as well just move the |\psfrag| commands to the
% main \TeX\ file. So currently compression is not supported by this
% system.
% \begin{macrocode}
\def\pfg@scan#1{\begingroup
% \end{macrocode}
% First the standard making `safe' of illegal characters.
% \begin{macrocode}
\@tempcnta\z@
\loop
\ifnum\@tempcnta<\@xxxii
\catcode\@tempcnta12 \advance\@tempcnta\@ne
\repeat
% \end{macrocode}
%
% Now normalise some catcodes we need. Most things are treated verbatim,
% but |{}| have their normal catcodes so |\read| will read multi-line
% |\\tex| expresions; and, as discussed above, |\| is made active.
% The end of a line is treated as space. Originally this meant that
% multi-line |\tex| commands did not work, but with v3.02, the PS side
% of things normalises white space to `space', as well so everyone is in
% agreement.
% \begin{macrocode}
\let\do\@makeother\dospecials\catcode`\ 10 %
\catcode`\{=1\catcode`\}=2\catcode127=12 %
\catcode`\\=\active\catcode\endlinechar5 %
% \end{macrocode}
%
% Open the scratch file. Don't complain if it already exists as that
% is probably just a previous graphic, or a previous run.
% \fname{pfgguide.tex} warns that any existing file of this name will
% be zapped, so: you've been warned.
% \begin{macrocode}
\immediate\openout\pfg@temp=\jobname.pfg %
% \end{macrocode}
%
% If the graphic file is not there, complain, else start reading it
% line by line. Look at each line with |\pfg@find|. Unlike the looking
% for |%%BoundingBox| where you can stop once you've found it, here
% you need to go to the end of the file looking for all the occurrences
% of |\tex| so if you give it megabytes of data, this may take a while.
% \begin{macrocode}
\immediate\openin\@inputcheck=#1 %
\ifeof\@inputcheck
\PackageWarning{psfrag}{Could not scan #1...}\endgroup
\else
\message{<psfrag:#1>}%
\@tempswatrue
\loop
\ifeof\@inputcheck
\@tempswafalse
\else
\read\@inputcheck to\@tempa
\expandafter\pfg@find\@tempa{\@nil}%
\fi
\if@tempswa
\repeat
\closein\@inputcheck
\fi
\immediate\closeout\pfg@temp
\endgroup
\@input{\jobname.pfg}}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\pfg@find}
% \changes{3.03}{1997/01/07}
% {Find strings even if nested inside brace groups}
% This command first looks for an explicit brace group |{ }|.
% (one definitely exists as |{\@nil}| is added at the end as
% a `marker').
% \begin{macrocode}
\long\def\pfg@find#1#{\pfg@finda{#1}}
% \end{macrocode}
% \end{macro}
%
% To enable an `active' copy or |\| to be accessed easily, use a special
% lowercase table while defining these macros.
% \begin{macrocode}
\begingroup
\lccode`\~=`\\
\lowercase{\endgroup
% \end{macrocode}
%
% \begin{macro}{\pfg@finda}
% If the group found contained the |\@nil| marker stop, else
% start looking for |(\\tex| in the tokens before the group.
% \begin{macrocode}
\long\def\pfg@finda#1#2{%
\def\@tempa{#2}%
\ifx\@tempa\@nnil
\else
\pfg@findb#1(~~tex(~~tex%
\fi}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\pfg@findb}
% Having found a brace group, look for the string |\\tex| in the
% preceding tokens. If there is such, discard any tokens before that
% then take all tokens after |\\tex| (up to the brace already found)
% to be the optional arguments. In that case write out a suitable
% call to |\psfrag|, and then continue after the brace group.
% Otherwise if no |\\tex| is found, Add the contents of the brace group
% (without the braces) back in front of the list before restarting the
% search. In that way any strings inside brace groups will be found.
%
% As |{}| have their normal catcodes, multiline |\tex| commands work
% as long as any line breaks occur inside the main |{ }| argument, not
% between the optional arguments (normally there is no space at all
% there, so no possibility of a break). This is because \TeX\ will
% read more than a line if necessary to ensure that braces balance.
%
% The arguments are delimited by |(\\tex| sequences:\\
% |#1| Is always discarded; the tokens before any |(\\tex| string.\\
% |#2| contains any optional arguments (complete with |[]| brackets).\\
% |#3| Will be the `dummy tokens' placed at the end, unless
% no |(\\tex| is in the string, in which case |#3| will be empty.
% This argument is |\fi| delimited which allows for tail recursion
% in a slightly sneaky way.
% \begin{macrocode}
\long\def\pfg@findb#1(~~tex#2(~~tex#3\fi{%
\fi
\ifx\box#3\box
\else
% \end{macrocode}
% Since |\| is the PS escape character in strings, as well as the \TeX\
% escape character, the following looks a bit weird, but it's probably
% right. A |\| will appear in the PS string as |\\|, so\ldots
%
% First set |\| to be |\string| (recall |\| is active, and can be written
% as |~| due to the |\lowercase| above).
% This means that
% |\\foo| expands to |\foo| (one catcode 12 |\| coming from |\string\|
% and then a catcode 11 |foo|). This expansion is frozen by an |\edef|
% for use in the second argument. Then redefine |\| to be |\relax|
% so it is a non-expandable active token, which writes as itself.
%
% Now write a call to |\psfrag| to the temp file.
% The `user' string |\foo| which was the literal PS |\\foo| thus ends
% up as:\\
% |\psfrag{|\ldots |\\foo| \ldots |}{| \ldots |\foo| |}%|
%
% When this file is read back |\psfrag| will read the first argument
% verbatim, and the second argument with normal catcodes so finally
% |\foo| gets to be a single token instead of 4, and is executed
% as part of the replacement text.
%
% Finally reset |\@tempa| (which contatined the contents of the brace group
% to empty, as we've done with that.
% \begin{macrocode}
{\let~\string
\edef\@tempb{\@tempa}%
\let~\relax
\immediate\write\pfg@temp{%
\string\psfrag\space
{\string\\tex#2{\@tempa}}#2\@percentchar
^^J\@spaces\@spaces{\@tempb}\@percentchar}}%
\let\@tempa\@empty
\fi
% \end{macrocode}
%
% Now start looking for the next brace group to test. First tip
% the contents of |\@tempa| in front of the tokens not yet seen, so that
% the contents of a brace group will be searched if they were not
% used as the argument to |\tex|.
% \begin{macrocode}
\expandafter\pfg@find\@tempa}
% \end{macrocode}
% \end{macro}
%
%
% Finally close the call to |\lowercase|.
% \begin{macrocode}
}
% \end{macrocode}
%
% \begin{macrocode}
%</package>
% \end{macrocode}
%
%
|