%%
%% Package: spectralsequences v1.3.3 2023-01-28
%% Author: Hood Chatham
%% Email:
[email protected]
%% Date: 2023-01-28
%% License: Latex Project Public License
%%
%% File: sseqdrawing.code.tex
%%
%% Defines the macros that draw the features of the spectral sequence (at least, those that aren't drawn by tikz).
%% Everything here has to be super optimized, because like 90% of the execution time is in these functions.
%% In particular, tikz is a horrible performance hole that must be avoided at all costs. For example, the code
%% the code to produce the axis ticks used to use the tikz \node command, and in the file example_KF3n.tex drawing
%% the axes ticks was taking about 1/3 of the compile time (a little under 1s out of a 3.2)
%%
% Only for integer valued coordinates...
\def\sseq@qpointxy#1#2{\pgfqpointxy{\numexpr#1+\sseq@xoffset\relax}{\numexpr#2+\sseq@yoffset\relax}}
\def\sseq@transform@xymirror{
\pgfsetxvec{\pgfqpoint{1cm}{0cm}}
\pgfsetyvec{\pgfqpoint{0cm}{1cm}}
\pgftransformcm{0}{1}{1}{0}{\pgfpointxy{\sseq@xoffset*\sseq@xscale }{\sseq@yoffset*\sseq@yscale}} % Reflect x and y axes,
\pgftransformshift{\pgfpointxy{-\sseq@xoffset*\sseq@yscale}{-\sseq@yoffset*\sseq@xscale}} % but we need to conjugate by a shift of (xoffset,yoffset)
\pgfsetyvec{\pgfqpoint{0cm}{\sseq@xscale cm}}
\pgfsetxvec{\pgfqpoint{\sseq@yscale cm}{0cm}}
}
\def\sseq@setlayoutparameter#1#2{\@xp\let\csname sseq@#1\@xp\endcsname\csname sseq@#1@#2\endcsname}
\def\sseq@setlayoutparameters{
\sseq@setlayoutparameter{yaxisorigin}{x\sseq@xaxistype}
\sseq@setlayoutparameter{drawyaxis}{x\sseq@xaxistype}
\sseq@setlayoutparameter{yaxisstartoffset}{x\sseq@xaxistype}
\sseq@setlayoutparameter{yaxisendoffset}{x\sseq@xaxistype}
\sseq@setlayoutparameter{bottomgridpadding}{x\sseq@xaxistype}
\sseq@setlayoutparameter{topgridpadding}{x\sseq@xaxistype}
\sseq@setlayoutparameter{bottomclippadding}{x\sseq@xaxistype}
\sseq@setlayoutparameter{topclippadding}{x\sseq@xaxistype}
\sseq@setlayoutparameter{xlabelposition}{x\sseq@xaxistype}
\sseq@setlayoutparameter{totalverticalmargin}{x\sseq@xaxistype}
%
\sseq@setlayoutparameter{xaxisorigin}{y\sseq@yaxistype}
\sseq@setlayoutparameter{drawxaxis}{y\sseq@yaxistype}
\sseq@setlayoutparameter{xaxisstartoffset}{y\sseq@yaxistype}
\sseq@setlayoutparameter{xaxisendoffset}{y\sseq@yaxistype}
\sseq@setlayoutparameter{leftgridpadding}{y\sseq@yaxistype}
\sseq@setlayoutparameter{rightgridpadding}{y\sseq@yaxistype}
\sseq@setlayoutparameter{leftclippadding}{y\sseq@yaxistype}
\sseq@setlayoutparameter{rightclippadding}{y\sseq@yaxistype}
\sseq@setlayoutparameter{ylabelposition}{y\sseq@yaxistype}
\sseq@setlayoutparameter{totalhorizontalmargin}{y\sseq@yaxistype}
}
% The appropriate one of these is chosen by the "axis type" keys in sseqkeys.
% Used to calculate available area in range check
\def\sseq@totalhorizontalmargin@yborder{(\sseq@xaxis@tail + \sseq@yaxisgap + \sseq@xaxis@end@extend + \sseq@clip@padding@right)}
\def\sseq@totalhorizontalmargin@yframe{(\sseq@ylabelgap + \sseq@maxylabelwidth + \sseq@yaxisgap * 2)}
\def\sseq@totalhorizontalmargin@ycenter{(\sseq@xaxis@start@extend + \sseq@clip@padding@left + \sseq@xaxis@end@extend + \sseq@clip@padding@right)}
\def\sseq@totalhorizontalmargin@ynone{(\sseq@xaxis@start@extend + \sseq@clip@padding@left + \sseq@xaxis@end@extend + \sseq@clip@padding@right)}
\def\sseq@totalverticalmargin@xborder{(\sseq@yaxis@tail + \sseq@xaxisgap + \sseq@yaxis@end@extend + \sseq@clip@padding@top)}
\def\sseq@totalverticalmargin@xframe{(\sseq@xlabelgap + \sseq@xlabelheight + \sseq@xaxisgap * 2)}
\def\sseq@totalverticalmargin@xcenter{(\sseq@yaxis@start@extend + \sseq@clip@padding@bottom + \sseq@yaxis@end@extend + \sseq@clip@padding@top)}
\def\sseq@totalverticalmargin@xnone{(\sseq@yaxis@start@extend + \sseq@clip@padding@bottom + \sseq@yaxis@end@extend + \sseq@clip@padding@top)}
% Used to determine axis and ticks placement
\def\sseq@xaxisorigin@yborder{\sseq@xmin}
\def\sseq@xaxisorigin@yframe{\sseq@xmin}
\def\sseq@xaxisorigin@ycenter{\sseq@xaxisorigin@center@}
\def\sseq@yaxisorigin@xborder{\sseq@ymin}
\def\sseq@yaxisorigin@xframe{\sseq@ymin}
\def\sseq@yaxisorigin@xcenter{\sseq@yaxisorigin@center@}
% Used to determine clipping and grid boundaries
\def\sseq@leftgridpadding@yborder{\dimexpr\sseq@yaxisgap-\sseq@yclip@axisgap\relax}
\def\sseq@rightgridpadding@yborder{\sseq@xaxis@end@extend}
\def\sseq@leftgridpadding@ycenter{\sseq@xaxis@start@extend}
\def\sseq@rightgridpadding@ycenter{\sseq@xaxis@end@extend}
\def\sseq@leftgridpadding@yframe{\dimexpr\sseq@yaxisgap-\sseq@yclip@axisgap\relax}
\def\sseq@rightgridpadding@yframe{5cm} % Just make it big enough to push off the end of the clip
\def\sseq@leftgridpadding@ynone{\sseq@yaxisgap}
\def\sseq@rightgridpadding@ynone{\sseq@yaxisgap}
\def\sseq@bottomgridpadding@xborder{\dimexpr\sseq@xaxisgap-\sseq@xclip@axisgap\relax}
\def\sseq@topgridpadding@xborder{\sseq@yaxis@end@extend}
\def\sseq@bottomgridpadding@xcenter{\sseq@yaxis@start@extend}
\def\sseq@topgridpadding@xcenter{\sseq@yaxis@end@extend}
\def\sseq@bottomgridpadding@xframe{\dimexpr\sseq@xaxisgap-\sseq@xclip@axisgap\relax}
\def\sseq@topgridpadding@xframe{5cm}
\def\sseq@bottomgridpadding@xnone{\sseq@xaxisgap}
\def\sseq@topgridpadding@xnone{\sseq@xaxisgap}
\def\sseq@leftclippadding@yborder{\dimexpr-\sseq@yaxisgap+\sseq@yclip@axisgap\relax}
\def\sseq@rightclippadding@yborder{\dimexpr\sseq@xaxis@end@extend+\sseq@clip@padding@right\relax}
\def\sseq@leftclippadding@ycenter{\dimexpr-\sseq@xaxis@start@extend-\sseq@clip@padding@left\relax}
\def\sseq@rightclippadding@ycenter{\dimexpr\sseq@xaxis@end@extend+\sseq@clip@padding@right\relax}
\def\sseq@leftclippadding@yframe{\dimexpr-\sseq@yaxisgap+\sseq@yclip@axisgap\relax}
\def\sseq@rightclippadding@yframe{\dimexpr\sseq@yaxisgap-\sseq@yclip@axisgap\relax}
\def\sseq@leftclippadding@ynone{\dimexpr-\sseq@yaxisgap-\sseq@clip@padding@left\relax}
\def\sseq@rightclippadding@ynone{\dimexpr\sseq@yaxisgap+\sseq@clip@padding@right\relax}
\def\sseq@bottomclippadding@xborder{\dimexpr-\sseq@xaxisgap+\sseq@xclip@axisgap\relax}
\def\sseq@topclippadding@xborder{\dimexpr\sseq@yaxis@end@extend+\sseq@clip@padding@top\relax}
\def\sseq@bottomclippadding@xcenter{\dimexpr-\sseq@yaxis@start@extend-\sseq@clip@padding@bottom\relax}
\def\sseq@topclippadding@xcenter{\dimexpr\sseq@yaxis@end@extend+\sseq@clip@padding@top\relax}
\def\sseq@bottomclippadding@xframe{\dimexpr-\sseq@xaxisgap+\sseq@xclip@axisgap\relax}
\def\sseq@topclippadding@xframe{\dimexpr\sseq@xaxisgap-\sseq@xclip@axisgap\relax}
\def\sseq@bottomclippadding@xnone{\dimexpr-\sseq@xaxisgap-\sseq@clip@padding@bottom\relax}
\def\sseq@topclippadding@xnone{\dimexpr\sseq@xaxisgap+\sseq@clip@padding@top\relax}
% Used to figure out how much further beyond (min -- max) to draw axes
\def\sseq@xaxisstartoffset@yborder{\dimexpr\sseq@xaxis@tail+\sseq@yaxisgap\relax}
\def\sseq@xaxisendoffset@yborder{\sseq@xaxis@end@extend}
\def\sseq@xaxisstartoffset@ycenter{\sseq@xaxis@start@extend}
\def\sseq@xaxisendoffset@ycenter{\sseq@xaxis@end@extend}
\def\sseq@xaxisstartoffset@yframe{\sseq@yaxisgap}
\def\sseq@xaxisendoffset@yframe{\sseq@yaxisgap}
\def\sseq@yaxisstartoffset@xborder{\dimexpr\sseq@yaxis@tail+\sseq@xaxisgap\relax}
\def\sseq@yaxisendoffset@xborder{\sseq@yaxis@end@extend}
\def\sseq@yaxisstartoffset@xcenter{\sseq@yaxis@start@extend}
\def\sseq@yaxisendoffset@xcenter{\sseq@yaxis@end@extend}
\def\sseq@yaxisstartoffset@xframe{\sseq@xaxisgap}
\def\sseq@yaxisendoffset@xframe{\sseq@xaxisgap}
\def\sseq@xlabelposition@xborder{(0,-\sseq@xaxisstartoffset)}
\def\sseq@xlabelposition@xcenter{(-\sseq@yaxisgap,-\sseq@xaxisstartoffset-5pt)}
\def\sseq@xlabelposition@xframe{(0,-\sseq@xaxisgap-\sseq@xlabelgap-10pt)}
\def\sseq@xlabelposition@xnone{(0,0)}
\def\sseq@ylabelposition@yborder{(0,\sseq@yaxisstartoffset)}
\def\sseq@ylabelposition@ycenter{(-\sseq@xaxisgap,\sseq@yaxisstartoffset+5pt)}
\def\sseq@ylabelposition@yframe{(0,\sseq@yaxisgap+\sseq@ylabelgap+10pt)}
\def\sseq@ylabelposition@ynone{(0,0)}
%%%
%%% Draw axes and clip
%%%
\def\sseq@handlexaxis{
\bgroup
\sseq@drawxaxis
\ifsseq@drawxaxisticks
\sseq@drawxticks
\fi
\egroup
}
\def\sseq@handleyaxis{
\bgroup
\sseq@drawyaxis
\ifsseq@drawyaxisticks
\sseq@drawyticks
\fi
\egroup
}
% Draws an x axis. Use with \sseq@transform@xymirror to draw y axis.
% #1 -- horizontal start offset
% #2 -- horizontal end offset
% #3 -- vertical offset
% #4 -- start value
% #5 -- end value
% #6 -- vertical value
\def\sseq@drawaxis@generic#1#2#3#4#5#6{
\bgroup
\pgftransformshift{\pgfqpoint{#2}{#3}}
\pgfpathmoveto{\sseq@qpointxy{#5}{#6}}
\egroup
%
\bgroup
\pgftransformshift{\pgfqpoint{#1}{#3}}
\pgfpathlineto{\sseq@qpointxy{#4}{#6}}
\egroup
}
\def\sseq@drawxaxis@ynone{
\sseq@drawxaxisticksfalse
}
\def\sseq@drawyaxis@xnone{
\sseq@drawyaxisticksfalse
}
\def\sseq@drawxaxis@yborder{
\sseq@drawaxis@generic{-\sseq@xaxisstartoffset}{\sseq@xaxisendoffset}{-\sseq@xaxisgap}{\sseq@xmin}{\sseq@xmax}{\sseq@yaxisorigin}
\pgfusepath{stroke}
}
\let\sseq@drawxaxis@ycenter\sseq@drawxaxis@yborder
\def\sseq@drawxaxis@yframe{
\sseq@drawaxis@generic{-\sseq@xaxisstartoffset}{\sseq@xaxisendoffset}{-\sseq@xaxisgap}{\sseq@xmin}{\sseq@xmax}{\sseq@ymin}
\sseq@drawaxis@generic{-\sseq@xaxisstartoffset}{\sseq@xaxisendoffset}{ \sseq@xaxisgap}{\sseq@xmin}{\sseq@xmax}{\sseq@ymax}
\pgfusepath{stroke}
}
\def\sseq@drawyaxis@xborder{
\bgroup
\sseq@transform@xymirror
\sseq@drawaxis@generic{-\sseq@yaxisstartoffset}{\sseq@yaxisendoffset}{-\sseq@yaxisgap}{\sseq@ymin}{\sseq@ymax}{\sseq@xaxisorigin}
\pgfusepath{stroke}%
\egroup
}
\let\sseq@drawyaxis@xcenter\sseq@drawyaxis@xborder
\def\sseq@drawyaxis@xframe{
\bgroup
\sseq@transform@xymirror
\sseq@drawaxis@generic{-\sseq@yaxisstartoffset}{\sseq@yaxisendoffset}{-\sseq@yaxisgap}{\sseq@ymin}{\sseq@ymax}{\sseq@xmin}
\sseq@drawaxis@generic{-\sseq@yaxisstartoffset}{\sseq@yaxisendoffset}{ \sseq@yaxisgap}{\sseq@ymin}{\sseq@ymax}{\sseq@xmax}
\pgfusepath{stroke}%
\egroup
}
\ExplSyntaxOn
\def\sseq@intdivceiling#1#2{%
\ifnum#1>\z@ % \int_div_truncate:nn rounds towards 0. We want \int_div_ceiling
\int_div_truncate:nn{#1+#2-1}{#2} % if positive, use ceil(p/q)=floor((p+q-1)/q) for p,q integers
\else
\int_div_truncate:nn{#1}{#2} % if we are negative, towards 0 is ceiling
\fi
}
\def\sseq@intdivfloor#1#2{%
\ifnum#1<\z@ % \int_div_truncate:nn rounds towards 0. We want \int_div_floor
\int_div_truncate:nn{#1-#2+1}{#2} % if we are negative, use floor(p/q) = ceil(p-q+1/q) for p,q integers
\else
\int_div_truncate:nn{#1}{#2} % if positive, towards 0 is floor
\fi
}
\ExplSyntaxOff
% #1 -- xmin
% #2 -- xmax
% #3 -- step
% #4 -- offset
% #5 -- ymin
% #6 -- xaxisgap
% #7 -- code
\def\sseq@tickloop@generic#1#2#3#4#5#6#7{
\sseq@tempcount=\numexpr % min
\sseq@intdivceiling{#1}{#3} * #3
-
\sseq@intdivceiling{#4}{#3} * #3 + #4
\relax
\ifnum\sseq@tempcount<#1\relax
\advance\sseq@tempcount#3\relax
\fi
\sseq@tempcountb=\numexpr#2+1\relax % max
\loop
\bgroup
\pgftransformshift{\sseq@qpointxy{\sseq@tempcount}{#5}}%
\pgftransformshift{\pgfqpoint{0pt}{#6}}
#7
\egroup
\advance\sseq@tempcount#3\relax
\ifnum\sseq@tempcount<\sseq@tempcountb\repeat
}
\def\sseq@drawxticks{%
\sseq@tickloop@generic{\sseq@xmin}{\sseq@xmax}{\sseq@xtickstep}{\sseq@xtickstepoffset}{\sseq@yaxisorigin}{-\sseq@xaxisgap}{
\pgftransformshift{\pgfqpoint{0pt}{-\sseq@xlabelgap}}
\pgftransformresetnontranslations
%\@xp\tikzset\@xp{\sseq@xtickstyle}
%\pgftext{\tikz@options\tikz@textfont\hbox{$\sseq@xtickfn{\the\sseq@tempx}$}}%
\@xp\node\@xp[\sseq@xtickstyle]{\hbox{$\sseq@xtickfn{\the\sseq@tempcount}$}};
}
\ifnum\sseq@xmajortickstep>\z@
\sseq@tickloop@generic{\sseq@xmin}{\sseq@xmax}{\sseq@xmajortickstep}{\sseq@xtickstepoffset}{\sseq@yaxisorigin}{-\sseq@xaxisgap}{
\pgfpathmoveto{\pgfpointorigin}
\pgfpathlineto{\pgfpoint{0}{-0.4*\sseq@xaxisgap}}
}
\fi
\ifnum\sseq@xminortickstep>\z@
\sseq@tickloop@generic{\sseq@xmin}{\sseq@xmax}{\sseq@xminortickstep}{\sseq@xtickstepoffset}{\sseq@yaxisorigin}{-\sseq@xaxisgap}{
\pgfpathmoveto{\pgfpointorigin}
\pgfpathlineto{\pgfpoint{0}{-0.2*\sseq@xaxisgap}}
}
\fi
\pgfusepath{stroke}
}
\def\sseq@drawyticks{%
\sseq@transform@xymirror
\sseq@tickloop@generic{\sseq@ymin}{\sseq@ymax}{\sseq@ytickstep}{\sseq@ytickstepoffset}{\sseq@xaxisorigin}{-\sseq@yaxisgap}{
\pgftransformshift{\pgfqpoint{0pt}{-\sseq@ylabelgap}}
\pgftransformresetnontranslations
%\@xp\tikzset\@xp{\sseq@ytickstyle}
%\pgftext{\tikz@options\tikz@textfont\hbox{$\sseq@ytickfn{\the\sseq@tempx}$}}%
\@xp\node\@xp[\sseq@ytickstyle]{\hbox{$\sseq@ytickfn{\the\sseq@tempcount}$}};
}
\ifnum\sseq@ymajortickstep>\z@
\sseq@tickloop@generic{\sseq@ymin}{\sseq@ymax}{\sseq@ymajortickstep}{\sseq@ytickstepoffset}{\sseq@xaxisorigin}{-\sseq@yaxisgap}{
\pgfpathmoveto{\pgfpointorigin}
\pgfpathlineto{\pgfpoint{0}{-0.4*\sseq@yaxisgap}}
}
\fi
\ifnum\sseq@yminortickstep>\z@
\sseq@tickloop@generic{\sseq@ymin}{\sseq@ymax}{\sseq@yminortickstep}{\sseq@ytickstepoffset}{\sseq@xaxisorigin}{-\sseq@yaxisgap}{
\pgfpathmoveto{\pgfpointorigin}
\pgfpathlineto{\pgfpoint{0}{-0.2*\sseq@yaxisgap}}
}
\fi
\pgfusepath{stroke}
}
\def\sseq@setupclip{
%\clip(\
[email protected],\
[email protected]) rectangle (\sseq@xmax+0.5,\sseq@ymax+0.5);%
\ifsseq@clip
\ifx\sseq@customclip\pgfutil@empty
\bgroup
\def\xmin{\pgftransformshift{\pgfqpoint{\sseq@leftclippadding}{0pt}}}
\def\ymin{\pgftransformshift{\pgfqpoint{0pt}{\sseq@bottomclippadding}}}
\def\xmax{\pgftransformshift{\pgfqpoint{\sseq@rightclippadding}{0pt}}}
\def\ymax{\pgftransformshift{\pgfqpoint{0pt}{\sseq@topclippadding}}}
\bgroup
\xmin\ymin
\pgfpathmoveto{\pgfqpointxy{\numexpr\sseq@xmin+\sseq@xoffset\relax}{\numexpr\sseq@ymin+\sseq@yoffset\relax}}
\egroup
\bgroup
\xmin\ymax
\pgfpathlineto{\pgfqpointxy{\numexpr\sseq@xmin+\sseq@xoffset\relax}{\numexpr\sseq@ymax+\sseq@yoffset\relax}}
\egroup
\bgroup
\xmax\ymax
\pgfpathlineto{\pgfqpointxy{\numexpr\sseq@xmax+\sseq@xoffset\relax}{\numexpr\sseq@ymax+\sseq@yoffset\relax}}
\egroup
\bgroup
\xmax\ymin
\pgfpathlineto{\pgfqpointxy{\numexpr\sseq@xmax+\sseq@xoffset\relax}{\numexpr\sseq@ymin+\sseq@yoffset\relax}}
\egroup
\egroup
\pgfpathclose
\pgfgetpath\sseq@theclippath % This stores the clipping so I can apply it later
\pgfusepath{discard}% This has to be after the egroup or else the clipping gets screwed up
%
\bgroup
\pgfpathmoveto{\pgfqpointxy{\numexpr\sseq@xmin+\sseq@xoffset\relax}{\numexpr\sseq@ymin+\sseq@yoffset\relax}}
\pgfpathlineto{\pgfqpointxy{\numexpr\sseq@xmin+\sseq@xoffset\relax}{\numexpr\sseq@ymax+\sseq@yoffset\relax}}
\pgfpathlineto{\pgfqpointxy{\numexpr\sseq@xmax+\sseq@xoffset\relax}{\numexpr\sseq@ymax+\sseq@yoffset\relax}}
\pgfpathlineto{\pgfqpointxy{\numexpr\sseq@xmax+\sseq@xoffset\relax}{\numexpr\sseq@ymin+\sseq@yoffset\relax}}
\pgfpathclose
\egroup
\pgfgetpath\sseq@therangepath % Only for deciding whether to draw "tricky edges"
\pgfusepath{discard}
\else
\def\sseq@temp{\path[name path=temp]}
\@xptwo\sseq@temp\@xp\@gobble\sseq@customclip
\pgfgetpath\sseq@theclippath
\let\sseq@theclippath\tikz@intersect@path@name@temp
\fi
\else
\let\sseq@theclippath\relax
\fi
}
\def\sseq@useclip{\ifx\sseq@theclippath\relax\else\pgfsetpath\sseq@theclippath\pgfusepath{clip}\fi}
% sets \pgf@xa and \pgf@ya equal to the coordinates of (1,1), sets \pgf@xb and \pgf@xb equal to (x grid step, y grid step)
\def\sseq@grid@setstepandgridstep{
\pgf@process{\pgfqpointxy{1}{1}}
\pgf@xa=\pgf@x
\pgf@ya=\pgf@y
\pgf@process{\pgfqpointxy{\sseq@xgridstep}{\sseq@ygridstep}}
\pgf@xb=\pgf@x
\pgf@yb=\pgf@y
}
% #1 -- macro to store result in
% #2 -- "x" or "y" as appropriate
% #3 -- length (with units)
% #4 -- fraction of gridstep
% If #2 is x, then it either sets #1 to #3/(x scale) or xgridstep*#4, whichever is smaller
% The output is intended for use with \pgfpointxy.
\def\sseq@grid@atmostgridstep#1#2#3#4{
\ifdim#3<\dimexpr\csname pgf@#2b\endcsname*#4\relax
\pgfmathparse{#3/\csname pgf@#2a\endcsname}
\let#1\pgfmathresult
\else
\edef#1{\csname sseq@#2gridstep\endcsname*#4}
\fi
}
% This grid was a huge pain in the ass to get right...
\def\sseq@grid@chess{
\bgroup
\pgfscope
% Because of complicated aliasing issues that arise when misusing the \pgfgrid command in this way,
% it's more convenient to add a clip than to actually stop things in the right place.
\clip (\
[email protected],\
[email protected])
rectangle (\sseq@xmax+0.49,\sseq@ymax+0.49);
% \clip (\sseq@xmin,\sseq@ymin) ++ (-0.49, -0.49)
% rectangle ([shift={(0.5cm*\sseq@yscale,0.5*\sseq@xscale)}]
% \sseq@xmax,\sseq@ymax);
%
\sseq@useclip
\pgfsetcolor{\sseq@gridcolor}
\pgfsetstrokeopacity{0.3} % We don't set low opacity for the other grids, but this one is visible even if it is very faint.
\sseq@grid@setstepandgridstep
%
% Use ifdim with cm to compare real scalars b/c tex is dumb.
% okay, now we need to be careful in order to avoid overflows. The hack we are using requires that the xvec and the yvec are equal
% because we can't apply these via lowlevelsynccm. However, xvec and yvec are the things that prevent overflows.
% What we need to do is set them both equal to the smaller value. Now the scale factor determines the size of the squares,
% so we mix in the grid steps. We also need to scale the larger scaled coordinate by the ratio of the two scales.
\ifdim\sseq@xscale cm<\sseq@yscale cm\relax
% x is smaller so set both x and y vecs to \sseq@xscale cm.
\pgfsetxvec{\pgfpoint{\sseq@xscale cm}{0cm}}
\pgfsetyvec{\pgfpoint{0cm}{\sseq@xscale cm}}
\def\sseq@xadjustscale{\sseq@xgridstep}
\def\sseq@yadjustscale{\sseq@yscale/\sseq@xscale*\sseq@ygridstep}
\def\sseq@stepscale{\sseq@xscale}
\else
\pgfsetxvec{\pgfpoint{\sseq@yscale cm}{0cm}}
\pgfsetyvec{\pgfpoint{0cm}{\sseq@yscale cm}}
\def\sseq@xadjustscale{(\sseq@xscale/\sseq@yscale*\sseq@xgridstep)}
\def\sseq@yadjustscale{\sseq@ygridstep}
\def\sseq@stepscale{\sseq@yscale}
\fi
\pgftransformxscale{\sseq@xadjustscale}
\pgftransformyscale{\sseq@yadjustscale}
% Now we have to put in some ridiculously complicated shift in order to get the bottom corner in the right place.
% I found this formula mostly by trial and error.
% The basic idea that we should be making something divisible by twice the grid step (which is the actual period of our grid)
% is reasonable enough, but what the heck are these other two terms for? I don't know either, thank god it works.
\pgftransformshift{\pgfpointxy{
floor((\sseq@xgridstep-1)/2)/\sseq@xgridstep
+ \ifodd\sseq@xgridstep\space 0 \else 0.5/\sseq@xgridstep\fi
+ mod(1+\sseq@xmin/\sseq@xgridstep,2*\sseq@xgridstep)
}{
floor((\sseq@ygridstep-1)/2)/\sseq@ygridstep
+ \ifodd\sseq@ygridstep\space 0\else 0.5/\sseq@ygridstep\fi
+ mod(1+\sseq@ymin/\sseq@ygridstep,2*\sseq@ygridstep)
}
}
% We use lowlevelsynccm to make the scale adjust the line width. Normally scaling doesn't affect line width, text size, etc, which is sensible.
% When you apply \pgflowlevelsynccm, pgf loses track of the coordinate matrix and it gets indiscriminately applied to everything.
% This is good for us, because we're doing this absurd thing where we make a checkerboard by drawing a really really fat line.
\pgflowlevelsynccm
\pgfsetlinewidth{1cm*\sseq@stepscale}
\pgfsetdash{{1cm*\sseq@stepscale}{1cm*\sseq@stepscale}}{1cm*\sseq@stepscale}
%
% Naturally some trial and error occurred here too.
% Note the huge multiples of \sseq@xgridstep I added in -- they probably aren't necessary, but I don't understand what's going on so whatever. They make sure that the pattern always extends past the edge of the clipping region. Otherwise the gird occasionally stops short for some reason.
% The basic idea is that things need to be divisible by twice the grid step. The part I added in the shift above
% was the remainder when you divide \sseq@xmin by twice the grid step, now we need to add in the multiple of twice the grid step.
% The -0.5 is ensuring that the checkerboard boundaries lie at half-integer coordinates (I think).
\pgfpathgrid[stepx= 2cm*\sseq@stepscale,stepy=2cm*\sseq@stepscale]
{ \pgfpointxy
{ -0.5 + \sseq@intdivfloor{\numexpr\sseq@xmin - \sseq@xgridstep\relax}{2*\sseq@xgridstep}*2*\sseq@xgridstep + \sseq@xoffset -50*\sseq@xgridstep
}
{ -0.5 + \sseq@intdivfloor{\numexpr\sseq@ymin - \sseq@ygridstep\relax}{2*\sseq@ygridstep}*2*\sseq@ygridstep + \sseq@yoffset -50*\sseq@ygridstep
}
}{ \pgfpointxy
{ \sseq@xmax + \sseq@xoffset + 50*\sseq@xgridstep }
{ \sseq@ymax + \sseq@yoffset + 50*\sseq@ygridstep }
}
\pgfusepath{stroke}
\endpgfscope
\egroup
}
\def\sseq@grid@crossword{
\bgroup
\pgfscope
\sseq@useclip
\pgfsetcolor{\sseq@gridcolor}
\pgfsetlinewidth{\the\sseq@gridstrokethickness}
\sseq@grid@setstepandgridstep
%
% We don't want the grid to end with an extra line, so extend it by one half the step distance or
% \sseq@---gridpadding, whichever is shorter.
\sseq@grid@atmostgridstep\sseq@xminpadding{x}{\sseq@leftgridpadding}{1/2}
\sseq@grid@atmostgridstep\sseq@xmaxpadding{x}{\sseq@rightgridpadding}{1/2}
\sseq@grid@atmostgridstep\sseq@yminpadding{y}{\sseq@bottomgridpadding}{1/2}
\sseq@grid@atmostgridstep\sseq@ymaxpadding{y}{\sseq@topgridpadding}{1/2}
%
\pgftransformshift{\pgfpointxy{-\sseq@xgridstep/2}{-\sseq@ygridstep/2}}
\pgfpathgrid[stepx=\pgf@xb,stepy=\pgf@yb]
{ \pgfpointxy
{ \sseq@xmin - \sseq@xminpadding + 0.01 + \sseq@xoffset + \sseq@xgridstep/2 }
{ \sseq@ymin - \sseq@yminpadding + 0.01 + \sseq@yoffset + \sseq@ygridstep/2 } }
{ \pgfpointxy
{ \sseq@xmax + \sseq@xmaxpadding - 0.01 + \sseq@xoffset + \sseq@xgridstep/2 }
{ \sseq@ymax + \sseq@ymaxpadding - 0.01 + \sseq@yoffset + \sseq@ygridstep/2 } }
\pgfusepath{stroke}
\endpgfscope
\egroup
}
\def\sseq@grid@go{
\bgroup
\pgfscope
\sseq@useclip
\pgfsetcolor{\sseq@gridcolor}
\pgfsetlinewidth{\the\sseq@gridstrokethickness}
\sseq@grid@setstepandgridstep
%
% We don't want the grid to end with an extra line, so extend it by one step distance or
% \sseq@---gridpadding, whichever is shorter.
\sseq@grid@atmostgridstep\sseq@xminpadding{x}{\sseq@leftgridpadding}{1}
\sseq@grid@atmostgridstep\sseq@xmaxpadding{x}{\sseq@rightgridpadding}{1}
\sseq@grid@atmostgridstep\sseq@yminpadding{y}{\sseq@bottomgridpadding}{1}
\sseq@grid@atmostgridstep\sseq@ymaxpadding{y}{\sseq@topgridpadding}{1}
%
\pgfpathgrid[stepx=\pgf@xb,stepy=\pgf@yb]
{ \pgfpointxy
{ \sseq@xmin - \sseq@xminpadding + 0.01 + \sseq@xoffset }
{ \sseq@ymin - \sseq@yminpadding + 0.01 + \sseq@yoffset } }
{ \pgfpointxy
{ \sseq@xmax + \sseq@xmaxpadding - 0.01 + \sseq@xoffset }
{ \sseq@ymax + \sseq@ymaxpadding - 0.01 + \sseq@yoffset }
}
\pgfusepath{stroke}
\endpgfscope
\egroup
}
\def\sseq@grid@none{}
\def\sseq@grid@dots{
\bgroup
\pgfscope
\pgfgettransform\sseq@savetransform
\pgfgettransformentries{\sseq@a}{\sseq@b}{\sseq@c}{\sseq@d}{\sseq@u}{\sseq@v}
\pgfsettransform\sseq@savetransform
\sseq@useclip
\pgftransformshift{\pgfqpoint{-1.5cm}{-0.5cm}}
\pgfsetdash{{1pt}{\sseq@a*1cm-1pt}}{0.5cm+.5pt}
\pgfsetlinewidth{1pt}
\sseq@tempy=\sseq@ymin\relax
\advance\sseq@tempy\m@ne
\loop
\advance\sseq@tempy\@ne
\pgfpathmoveto{\pgfpointxy{\sseq@xmin + 0.5/\sseq@a}{\the\sseq@tempy}}
\pgfpathlineto{\pgfpointxy{\sseq@xmax + 1.01 }{\the\sseq@tempy}}
\ifnum\sseq@tempy<\sseq@ymax\repeat
%\pgfpathgrid[stepx=1cm,stepy=1cm]{\pgfpoint{-0.5cm}{-0.5cm}}{\pgfpoint{\xmax cm-0.5cm}{\ymax cm-0.5cm}}
\pgfusepath{stroke}
\endpgfscope
\egroup
}
%%%
%%% Draw Classes
%%%
%%% Class offsets
\SseqNewClassPattern{standard}{
(0,0);
(-0.13,0)(0.13,0);
(-0.2,0)(0,0)(0.2,0);
(-0.13,-0.13)(0.13,-0.13)(-0.13,0.13)(0.13,0.13);
(-0.16,-0.16)(0.16,-0.16)(-0.16,0.16)(0.16,0.16)(0,0);
(-0.13,-0.2)(-0.13,0)(-0.13,0.2)(0.13,-0.2)(0.13,0)(0.13,0.2);
}
\sseqnewclasspattern{linear}{
(0,0);
(-0.13,0)(0.13,0);
(-0.2,0)(0,0)(0.2,0);
(-0.3,0)(-0.1,0)(0.1,0)(0.3,0);
(-0.4,0)(-0.2,0)(0,0)(0.2,0)(0.4,0);
(-0.5,0)(-0.3,0)(-0.1,0)(0.1,0)(0.3,0)(0.5,0);
}
\def\sseq@offset#1#2{
\sseq@eval{\@nx\pgftransformshift{
\@nx\pgfqpointxy
{ \csname sseq@\sseq@classpattern xoffset#1/#2\endcsname }
{ \csname sseq@\sseq@classpattern yoffset#1/#2\endcsname }
}}
}
\def\sseq@tooltip@wrapper#1#2{%
\edef\temp{\detokenize\@xpthree{#2}}%
\edef\temp{\@xp\sseqtooltip@replaceslashes\@xp{\temp}}%
\sseq@eval{\@nx\pdftooltip{\unexpanded{#1}}{\temp}}%
}
\bgroup\lccode`\!=`\\\lowercase{\egroup
\def\sseqtooltip@replaceslashes#1{\sseqtooltip@replaceslashes@#1!\sseq@nil}
\def\sseqtooltip@replaceslashes@#1!#2{%
#1%
\ifx\sseq@nil#2\@xp\@gobble\else
\@nx\@nx\@nx\textbackslash
\@xp\sseqtooltip@replaceslashes@
\fi#2%
}
}
% #1 -- "class."
% (#2,#3,#4) -- the class name x,y,n
% #5 -- generation
\def\sseq@class@getparts#1(#2,#3,#4)[#5].{
\sseq@seterrorannotation@drawing{#1}{#2}{#3}{#4}{#5}
\def\sseq@thisnodename{sseq{#2,#3,#4}}
\def\sseq@thispos{(#2,#3)}
\edef\sseq@thisposnodeindex{\sseq@obj{class.(#2,#3,#4).n}}
\edef\sseq@thispostotalnodes{\sseq@obj{partcoord.(#2,#3).numnodes}}
\def\sseq@thisclassnum{#5}
\sseq@tempx=#2\relax
\sseq@tempy=#3\relax
}
% #1 -- the name of the class object given as (x,y,n)[gen]
% Mandatory fields:
% partcoord.(x,y).numnodes
% (x,y,n).num -- number of generations of this class
% (x,y,n).n -- node n of numnodes (needed to allow node position to differ from creation order).
% class.page -- the death page of the class
% Optional fields:
% class.options -- Options for this class.
% class.nodetext -- The internal text of the node
% class.nodetext.options -- options (color, font) to change the nodetext
% class.labelnodes -- a list of external labels
% class.name & class.showname -- If there is a name, use it as a label in some way
% class.tooltip -- Make a tooltip. Fails if the user didn't load package with tooltip option
% class.needstikz -- Use tikz to draw this node
% Someday I should document this horrible mess of code here
\newif\ifsseq@permanentcycle
\newcount\sseq@totalclassesdrawn
\AtEndDocument{\message{Total classes: \the\sseq@totalclassesdrawn}}
\def\sseq@class@drawnode#1{%
\global\advance\sseq@totalclassesdrawn\@ne
\begingroup
\sseq@class@getparts#1.
% defines:
% \sseq@thisnodename -- name we should give the pdfnode
% \sseq@thispos -- (x,y)
% \sseq@thispostotalnodes -- total number of nodes at (x,y)
% \sseq@thisposnodeindex -- index of this node
\sseq@needstikzfalse
% Apply first pass styles? TODO: What is this for again?
\sseq@options@firstpassmode
\sseq@thesseqstyle
\sseq@theclassstyle
\ifnum\sseq@obj{#1.page}=\sseq@infinitycount
\sseq@thepermanentcyclestyle
\else
\sseq@thetransientcyclestyle
\ifsseq@thispage
\sseq@thethispagecyclestyle
\fi
\fi
\the\sseq@scope@toks
\sseq@obj{#1.needstikz}
%
\sseq@outofrangetrue\relax % Mysterious that we need this \relax here...
% If we are in range, we'll draw the node, if not we just mark the coordinate.
\ifnum\sseq@tempx<\sseq@xmaxpp\relax\ifnum\sseq@tempx>\sseq@xminmm\relax\ifnum\sseq@tempy<\sseq@ymaxpp\relax\ifnum\sseq@tempy>\sseq@yminmm\relax
\sseq@outofrangefalse
\pgfscope
% Finish options setup
\let\tikz@options\pgfutil@empty
\let\tikz@alias=\pgfutil@empty
\def\pgfkeysdefaultpath{/sseqpages/class/}%
\sseq@options@secondpassmode
\sseq@thesseqstyle
\sseq@theclassstyle
\ifnum\sseq@obj{#1.page}=\sseq@infinitycount
\sseq@permanentcycletrue % This is to communicate with family style code...
\sseq@thepermanentcyclestyle
\else
\sseq@permanentcyclefalse
\sseq@thetransientcyclestyle
\ifsseq@thispage
\sseq@thethispagecyclestyle
\fi
\fi
\def\sseq@collections@featuretype{class}
\the\sseq@scope@toks
\sseq@obj{#1.options}
% Set up node position coordinate transform
\pgftransformshift{\pgfqpointxy{\numexpr\sseq@tempx +\sseq@xoffset-\sseq@x\relax}{\numexpr\sseq@tempy + \sseq@yoffset-\sseq@y\relax}}
\iftikz@fullytransformed\pgfgettransform{\savetransform}\fi
\pgftransformresetnontranslations
\sseq@globalrotatetransform
% Now the origin is at (x,y). Set up class placement offset.
\sseq@classplacementtransform
\sseq@obj@ifdef{#1.offset}{\sseq@obj{#1.offset}}{%
\sseq@offset{\sseq@thisposnodeindex}{\sseq@thispostotalnodes}%
}%
\iftikz@fullytransformed\pgfsettransform{\savetransform}\else\pgftransformresetnontranslations\ifsseq@rotatelabels\sseq@globalrotatetransform\fi\fi
%
\tikz@options
% the value of \sseq@class@showname comes from styles. If there was a local option with showname, it's stored in #1.showname.
% local value takes priority.
\let\sseq@classlabelnodes\pgfutil@empty
\edef\sseq@classnodetextoptions{\sseq@obj@ifdef{#1.nodetext.options}{\@xptwo\@nx\sseq@obj{#1.nodetext.options}}{}}
\edef\sseq@classnodetext{\sseq@obj@ifdef{#1.nodetext}{\@xptwo\@nx\sseq@obj{#1.nodetext}}{}}
\sseq@obj@ifdef{#1.showname}{\sseq@lettoobj\sseq@class@showname{#1.showname}}{}
\ifcsname sseq@class@showname\endcsname
\sseq@obj@ifdef{#1.name}{
\sseq@eval{\@nx\sseq@handleclassquotes@inner{\sseq@obj{#1.name}}{\sseq@class@showname}}
}{}
\fi
% Okay now we're ready to make the node
\ifsseq@needstikz
% The options code above put mode information into \tikz@mode which gets wiped by tikz
\let\sseq@mode\tikz@mode
\sseq@eval{%
\@nx\node[%
every text node part/.code/.expand once={\sseq@globalrotatetransform\sseq@classnodetextoptions{}},%
/utils/exec={\let\@nx\tikz@mode\@nx\sseq@mode}] % Set mode based on outer options
(\sseq@thisnodename) {\unexpanded\@xp{\sseq@classnodetext}};%
}%
\else
\tikz@node@textfont
\sseq@setnodetext{\sseq@classnodetext}{\sseq@classnodetextoptions}%
\let\tikz@fig@name\sseq@thisnodename
\pgfmultipartnode{\tikz@shape}{\tikz@anchor}{\tikz@fig@name}{\sseq@drawnode}%
\tikz@alias % makes extra names for this shape
\fi
\sseq@obj{#1.labelnodes}%
\sseq@classlabelnodes % classlabelnodes comes from show name (I guess it can't put them into #1.labelnodes? TODO: why?)
\sseq@obj@ifdef{#1.tooltip}{\sseq@dotooltip{#1}}{}%
\endpgfscope
\fi\fi\fi\fi
% If the node is out of range, to save time we don't draw anything. However, structlines etc may depend on the out of range node,
% so we mark the coordinate.
\ifsseq@outofrange
\sseq@eval{\@nx\pgftransformshift{\@nx\pgfqpointxy{\numexpr\sseq@tempx+\sseq@xoffset-\sseq@x\relax}{\numexpr\sseq@tempy+\sseq@yoffset-\sseq@y\relax}}}%
\pgftransformresetnontranslations
\sseq@globalrotatetransform
\sseq@classplacementtransform
\sseq@offset{\sseq@thisposnodeindex}{\sseq@thispostotalnodes}%
\pgfcoordinate{\sseq@thisnodename}{\pgfpointorigin}%
\fi
\endgroup
}
\def\sseq@dotooltip#1{%
\pgfpointanchor{\sseq@thisnodename}{west}%
\pgf@xa=\pgf@x
\pgfpointanchor{\sseq@thisnodename}{south}%
\pgf@ya=\pgf@y
%
\pgf@process{\pgfpointdiff{\pgfpointtransformed{\pgfpointanchor{\sseq@thisnodename}{west}}}{\pgfpointtransformed{\pgfpointanchor{\sseq@thisnodename}{east}}}}%
\pgf@xb=\pgf@x
\pgf@process{\pgfpointdiff{\pgfpointtransformed{\pgfpointanchor{\sseq@thisnodename}{south}}}{\pgfpointtransformed{\pgfpointanchor{\sseq@thisnodename}{north}}}}%
\pgf@yb=\pgf@y
%
\setbox\tikz@tempbox=\hbox{%
\pgfinterruptpicture
\sseqtooltip{\rule{\pgf@xb}{0pt}\rule{0pt}{\pgf@yb}}{\sseq@obj{#1.tooltip}}%
\endpgfinterruptpicture
}
{%
\pgftransformshift{\pgfqpoint{\pgf@xa}{\pgf@ya}}%
\pgfapproximatenonlineartransformation%
\pgfqboxsynced{\tikz@tempbox}%
}%
}
% #1 -- label text
% #2 -- options
% Make the node textbox.
\def\sseq@setnodetext#1#2{%
\sseq@ifempty{#1}{% more often than not, the node is empty...
\setbox\pgfnodeparttextbox=\hbox{}%
}{%
\setbox\pgfnodeparttextbox=\hbox{%
\pgfscope%
#2
\tikzset{every text node part/.try}%
\ifx\tikz@textopacity\pgfutil@empty%
\else%
\pgfsetfillopacity{\tikz@textopacity}%
\pgfsetstrokeopacity{\tikz@textopacity}%
\fi%
\pgfinterruptpicture
\ifx\tikz@text@width\pgfutil@empty%
\tikz@textfont%
\else%
\begingroup%
\pgfmathsetlength{\pgf@x}{\tikz@text@width}%
\pgfutil@minipage[t]{\pgf@x}\leavevmode\hbox{}%
\tikz@textfont%
\tikz@text@action%
\fi%
\ifx\tikz@textcolor\pgfutil@empty%
\else%
\pgfutil@colorlet{.}{\tikz@textcolor}%
\fi%
\pgfsetcolor{.}%
\tikz@atbegin@node%
#1%
\tikz@atend@node%
\ifx\tikz@text@width\pgfutil@empty%
\else%
\pgfutil@endminipage%
\endgroup%
\fi%
\endpgfinterruptpicture
\endpgfscope%
}%
\ifx\tikz@text@width\pgfutil@empty%
\else%
\pgfmathsetlength{\pgf@x}{\tikz@text@width}%
\wd\pgfnodeparttextbox=\pgf@x%
\fi%
\ifx\tikz@text@height\pgfutil@empty%
\else%
\pgfmathsetlength{\pgf@x}{\tikz@text@height}%
\ht\pgfnodeparttextbox=\pgf@x%
\fi%
\ifx\tikz@text@depth\pgfutil@empty%
\else%
\pgfmathsetlength{\pgf@x}{\tikz@text@depth}%
\dp\pgfnodeparttextbox=\pgf@x%
\fi%
}
}
% A small part of the tikz main loop that has been paired down as much as possible for efficiency.
\def\sseq@drawnode{%
\pgfutil@tempdima=\pgflinewidth%
{%
\tikz@mode%
\iftikz@mode@draw%
\iftikz@mode@double%
% Change line width
\begingroup%
\pgfsys@beginscope%
\tikz@double@setup%
\fi%
\fi%
%
% Step 10: Do stroke/fill as needed
%
\sseq@eval{\noexpand\pgfusepath{%
\iftikz@mode@fill fill,\fi%
\iftikz@mode@draw draw,\fi%
}}%
%
% Step 11: Double stroke, if necessary
%
\iftikz@mode@draw%
\iftikz@mode@double%
\pgfsys@endscope%
\endgroup%
\fi%
\fi
}%
\global\pgflinewidth=\pgfutil@tempdima%
}
%%% Labels
% #1 -- label text
% #2 -- options
\def\sseq@drawlabel#1#2{%
\bgroup\pgfscope
\def\tikz@mode{}%
\let\sseq@tikz@transform@save\tikz@transform
\pgfkeyssetvalue{/pgf/inner xsep}{2pt}%
\pgfkeyssetvalue{/pgf/inner ysep}{2pt}%
\def\tikz@shape{rectangle}
\let\tikz@transform\empty % The next line was set up to fix the classlabelstyle glitch (what does this mean?)
\sseq@options@secondpassmode
\sseq@thesseqstyle\sseq@thelabelstyle\sseq@theclasslabelstyle#2
\tikz@options
\pgftransformreset
\pgftransformshift{\tikz@node@at}%
\tikz@lib@pos@call
\tikz@transform
\tikz@mode
\let\tikz@transform\sseq@tikz@transform@save
\sseq@setnodetext{\sseq@labeltextfn{#1}}{}%
\pgfmultipartnode{\tikz@shape}{\tikz@anchor}{label}{\sseq@drawnode}%
\ifsseq@pin
\def\sseq@pinoptions{}%
\let\tikz@options\empty
\let\tikz@mode\empty
\sseq@thepinstyle
#2
\sseq@pinoptions
\tikz@options
\tikz@mode
\sseq@drawedge@findsourcetarget{\tikz@fig@name}{}{label}{}%
\pgfpathmoveto{\sseq@sourcecoord}%
\pgfpathlineto{\sseq@targetcoord}%
\sseq@eval{\noexpand\pgfusepath{%
draw,
\iftikz@mode@fill fill,\fi
\iftikz@mode@draw draw,\fi
}}%
\fi
\endpgfscope\egroup
}
%%%
%%% Drawing edges
%%%
\def\sseq@ifinrange(#1){\sseq@ifinrange@#1,\sseq@nil}
\def\sseq@ifinrange@#1,#2,#3\sseq@nil{%
\sseq@tempx=#1\relax\sseq@tempy=#2\relax
\sseq@outofrangetrue
\ifnum\sseq@tempx<\sseq@xmaxpp\relax\ifnum\sseq@tempx>\sseq@xminmm\relax\ifnum\sseq@tempy<\sseq@ymaxpp\relax\ifnum\sseq@tempy>\sseq@yminmm\relax
\sseq@outofrangefalse
\fi\fi\fi\fi
\ifsseq@outofrange
\@xp\pgfutil@secondoftwo
\else
\@xp\pgfutil@firstoftwo
\fi
}
% #1 -- source node
% #2 -- source anchor
% #3 -- target node
% #4 -- target anchor
% Calculate actual start and end of the edge (node borders), return the results stored in \sseq@sourcecoord, \sseq@targetcoord
\def\sseq@drawedge@findsourcetarget#1#2#3#4{
\edef\sseq@edgesourceanchor{#2}
\edef\sseq@edgetargetanchor{#4}
\let\tempaf\pgfutil@empty
\ifx\sseq@edgesourceanchor\pgfutil@empty % Check that the source doesn't have a specified anchor
\def\tempa{\pgfpointanchor{#1}{center}}% if so, start by taking the center of that coordinate
\else
\edef\tempa{\@nx\pgfpointanchor{#1}{\sseq@edgesourceanchor}} % If it has an anchor, use that
\let\tempaf\tempa
\fi
\ifx\sseq@edgetargetanchor\pgfutil@empty % check that the target doesn't have a specified anchor
\def\tempb{\pgfpointshapeborder{#3}{\tempa}}% if so, our end point is the point on the boundary of node b that is in the direction of our initial start coordinate
\else
\edef\tempb{\@nx\pgfpointanchor{#3}{\sseq@edgetargetanchor}}% If it has a specified anchor, use that
\fi
\let\tempbf\tempb
\ifx\tempaf\pgfutil@empty%
\def\tempaf{\pgfpointshapeborder{#1}{\tempb}}%
\fi
\let\sseq@sourcecoord\tempaf
\let\sseq@targetcoord\tempbf
}
\def\sseq@fullcoord@to@partialcoord(#1){\sseq@fullcoord@to@partialcoord@#1,\@nil}
\def\sseq@fullcoord@to@partialcoord@#1,#2,#3\@nil{{#1cm}{#2cm}}
% #1 -- source (full)
% #2 -- target (full)
% #3 -- which type of edge ("differential", "structline", or "extension")
% #4 -- options
\def\sseq@drawedge(#1)(#2)#3#4{%
\begingroup\pgfscope
% If either class is part of a family we aren't drawing, don't draw the edge either.
\expandafter\ifx\csname pgf@sh@pi@sseq{#1}\endcsname\pgfpictureid\else
\@xp\sseq@break
\fi
\expandafter\ifx\csname pgf@sh@pi@sseq{#2}\endcsname\pgfpictureid\else
\@xp\sseq@break
\fi
\def\sseq@edgetype{#3}
\let\sseq@collections@featuretype\sseq@edgetype
\let\sseq@edgesourceanchor\pgfutil@empty
\let\sseq@edgetargetanchor\pgfutil@empty
%
\sseq@needstikzfalse
\def\pgfkeysdefaultpath{/sseqpages/#3/}%
\sseq@options@bothpassmode
\sseq@thesseqstyle
\sseq@theedgestyle\csname sseq@the#3style\endcsname
\ifsseq@thispage
\csname sseq@thethispage#3style\endcsname
\fi
\the\sseq@scope@toks
#4%
\csname sseq@collections@#3@hook\endcsname
\pgftransformshift{\pgfqpointxy{-\the\sseq@x}{-\the\sseq@y}}%
\ifsseq@needstikz
\pgfnodealias{tempa}{sseq{#1}}
\pgfnodealias{tempb}{sseq{#2}}
\def\sseq@sourcecoord{\pgfpointanchor{sseq{#1}}{center}}
\def\sseq@targetcoord{\pgfpointanchor{sseq{#2}}{center}}
\else
% puts results into \sseq@sourcecoord and \sseq@targetcoord
\sseq@drawedge@findsourcetarget{sseq{#1}}{\sseq@edgesourceanchor}{sseq{#2}}{\sseq@edgetargetanchor}
\pgfcoordinate{tempa}{\sseq@sourcecoord}%
\pgfcoordinate{tempb}{\sseq@targetcoord}%
\fi
%
\tikz@options
\tikz@mode
\def\temparrowstartspec{}%
\def\temparrowendspec{}%
\pgftransformreset
\sseq@outofrangefalse
\sseq@ifinrange(#1){}{
\edef\temparrowstartspec{\@nx\pgfsetarrowsstart{\csname sseq@runoffarrow@start@#3@spec\endcsname}}
\sseq@outofrangetrue
}%
\sseq@ifinrange(#2){}{
\edef\temparrowendspec{\@nx\pgfsetarrowsend{\csname sseq@runoffarrow@end@#3@spec\endcsname}}
\sseq@outofrangetrue
}
\ifsseq@outofrange
\sseq@handleoffpageedge{#1}{#2}%
\fi
\ifsseq@drawedge
% TODO: should some sort of transformation manipulation be here? Maybe allow user to specify preference?
% Don't draw dots on very short segments
\pgfpointdiff{\sseq@targetcoord}{\sseq@sourcecoord}
\pgfmathveclen{\pgf@x}{\pgf@y}%
\@xp\pgfmathint\@xp{\pgfmathresult}%
\ifnum\pgfmathresult<10\relax%%17? % TODO: Fix this predicate
\tikzset{every text node part/.append code={\pgfsetcolor{white}}}% I wonder why this is here...
\ifx\temparrowstartspec\pgfutil@empty
\else
\def\temparrowstartspec{\pgfsetarrowsstart{}}%
\fi
\ifx\temparrowendspec\pgfutil@empty
\else
\def\temparrowendspec{\pgfsetarrowsend{}}%
\fi
\fi
\ifsseq@needstikz
\draw[/sseqpages,%
/utils/exec={%
\sseq@thesseqstyle\sseq@theedgestyle
\csname sseq@the#3style\endcsname
\ifsseq@thispage
\csname sseq@thethispage#3style\endcsname
\fi
\the\sseq@scope@toks
\temparrowstartspec\temparrowendspec #4%
}%
] (tempa) to (tempb);%
\else
\temparrowstartspec
\temparrowendspec
\pgfpathmoveto{\pgfpointanchor{tempa}{center}}%
\pgfpathlineto{\pgfpointanchor{tempb}{center}}%
\sseq@eval{\noexpand\pgfusepath{%
\iftikz@mode@fill fill,\fi
\iftikz@mode@draw draw,\fi
}}%
\fi
\fi
\sseq@breakpoint
\endpgfscope\endgroup
}
% TODO: this macro is super expensive. Make it faster
\def\sseq@handleoffpageedge#1#2{
\pgfpathmoveto{\sseq@sourcecoord}%
\pgfpathlineto{\sseq@targetcoord}%
\pgfgetpath\thispath
\pgfusepath{discard}%
\pgfintersectionsortbysecondpath
\pgfintersectionofpaths{\pgfsetpath\sseq@theclippath}{\pgfsetpath\thispath}%
\ifcase\pgfintersectionsolutions\relax
% No intersections, but one or both endpoints may be out of range but still in clipping region due to scaling. Add ellipses as appropriate.
\sseq@ifinrange(#1){% If the first endpoint is in range, the second must be out of range b/c sseq@outofrange is true.
%\edef\temparrowendspec{\@nx\pgfsetarrowsend{\csname sseq@runoffarrow@end@#3@spec\endcsname}}
}{%
\sseq@ifinrange(#2){}{\sseq@drawedge@handletrickyedge}% uh-oh, both ends are out of range
}%
\or
\sseq@ifinrange(#1){% If the startpoint is in range, the intersection must be the end.
\def\sseq@targetcoord{\pgfpointintersectionsolution{1}}
\pgfcoordinate{tempb}{\sseq@targetcoord}
}{%
\sseq@ifinrange(#2){% If the startpoint is out of range and the endpoint is in range, the intersection must be the start
\def\sseq@sourcecoord{\pgfpointintersectionsolution{1}}%
\pgfcoordinate{tempa}{\sseq@sourcecoord}%
}{\sseq@drawedge@handletrickyedge}% Uh-oh, both ends are out of range.
}
\or% an orphan
\ifsseq@draworphanedges
\sseq@drawedge@handleorphan
\else
\sseq@drawedgefalse % Don't draw "orphaned edges"
\fi
\else
\sseq@error{clip-not-convex}%
\sseq@breakfifi
\fi
}
\def\sseq@drawedge@handletrickyedge{%
\ifsseq@draworphanedges
\pgfintersectionsortbysecondpath
\pgfintersectionofpaths{\pgfsetpath\sseq@therangepath}{\pgfsetpath\thispath}%
\ifnum\pgfintersectionsolutions=\z@
\sseq@drawedgefalse % don't draw orphan edges that never intersect actual range
\else% Now we have to make a line through tempa and tempb long enough so that it intersects the original clip area twice.
\pgfmathanglebetweenpoints{\pgfpointanchor{tempa}{center}}{\pgfpointanchor{tempb}{center}}%
\edef\tempangle{\pgfmathresult}
\pgfpathmoveto{\pgfpointadd{\pgfpointanchor{tempa}{center}}{\pgfpointpolar{\tempangle}{100cm}}}% a really long line
\pgfpathlineto{\pgfpointadd{\pgfpointanchor{tempa}{center}}{\pgfpointpolar{\tempangle}{-100cm}}}%
\pgfgetpath\thispath
\pgfusepath{discard}
\pgfintersectionofpaths{\pgfsetpath\sseq@theclippath}{\pgfsetpath\thispath}
\sseq@drawedge@handleorphan
\fi
\else
\sseq@drawedgefalse
\fi
}
\def\sseq@drawedge@handleorphan{%
\def\sseq@sourcecoord{\pgfpointintersectionsolution{2}}%
\def\sseq@targetcoord{\pgfpointintersectionsolution{1}}%
\edef\temparrowstartspec{\@nx\pgfsetarrowsstart{\csname sseq@runoffarrow@start@\sseq@edgetype @spec\endcsname}}%
\edef\temparrowendspec{\@nx\pgfsetarrowsend{\csname sseq@runoffarrow@end@\sseq@edgetype @spec\endcsname}}%
\pgfcoordinate{tempa}{\sseq@sourcecoord}%
\pgfcoordinate{tempb}{\sseq@targetcoord}%
}
% #1 -- uid
% #2 -- first coordinate
% #3 -- second coordinate
\def\sseq@circleclass@draw#1#2#3{
\begingroup
% If either class is part of a family we aren't drawing, don't draw the fit either.
\expandafter\ifx\csname pgf@sh@pi@sseq{#2}\endcsname\pgfpictureid\else
\@xp\sseq@break
\fi
\expandafter\ifx\csname pgf@sh@pi@sseq{#3}\endcsname\pgfpictureid\else
\@xp\sseq@break
\fi
\pgfmathanglebetweenpoints{\pgfpointanchor{sseq{#2}}{center}}{\pgfpointanchor{sseq{#3}}{center}}
\let\tempangle\pgfmathresult
\let\tikz@lib@fit@scan@handle\sseq@fit@tikz@lib@fit@scan@handle % install fit modifications.
\let\tikz@calc@anchor\sseq@fit@tikz@calc@anchor
\sseq@tempiftrue
\sseq@options@secondpassmode
\node[
rotate fit=\tempangle,
/utils/exec={\sseq@thesseqstyle\sseq@thecircleclassstyle\the\sseq@scope@toks\sseq@savedoptioncode
\sseq@obj{#1.options}
\sseq@obj{#1.fitnodes}
\@xp\pgfkeysalso\@xp{\romannumeral0\sseq@obj{#1.tikzprimoptions}}
}
] {};
\sseq@breakpoint
\endgroup
}
% Modifies tikz commands \tikz@lib@fit@scan@handle from \pgf\frontendlayer\tikz\libraries\tikzlibraryfit.code.tex line 81 and
% \tikz@calc@anchor from \pgf\frontendlayer\tikz\tikz.code.tex line 5164
% make it so that fit silently ignores nodes that are not defined.
% This is copied with modification from \pgf\frontendlayer\tikz\libraries\tikzlibraryfit.code.tex line 81
\def\sseq@fit@tikz@lib@fit@scan@handle#1{%
\ifsseq@tempif % this has been set in the following macro to be true if there is a node with the given name. If it's not true, ignore this.
\iftikz@shapeborder%
% Ok, fit all four external anchors, if they exist
\tikz@lib@fit@adjust{\pgfpointanchor{\tikz@shapeborder@name}{west}}%
\tikz@lib@fit@adjust{\pgfpointanchor{\tikz@shapeborder@name}{east}}%
\tikz@lib@fit@adjust{\pgfpointanchor{\tikz@shapeborder@name}{north}}%
\tikz@lib@fit@adjust{\pgfpointanchor{\tikz@shapeborder@name}{south}}%
\else%
\tikz@lib@fit@adjust{#1}%
\fi%
\fi
\sseq@tempiftrue
\tikz@lib@fit@scan%
}
%\def\tikz@calc@anchor#1.#2\tikz@stop{%
% \ifcsname pgf@sh@ns@\tikz@pp@name {#1}\endcsname
% \pgfpointanchor {\tikz@pp@name {#1}}{#2}%
% \else
% \pgfpointanchor {#1}{#2}%
% \fi
%}
%l.8 \show\tikz@calc@anchor
% This is copied with modification from \pgf\frontendlayer\tikz\tikz.code.tex line 5164
\def\sseq@fit@tikz@calc@anchor#1.#2\tikz@stop{%
\ifcsname pgf@sh@ns@\tikz@pp@name {#1}\endcsname
\pgfpointanchor {\tikz@pp@name {#1}}{#2}%
\else
\ifcsname pgf@sh@ns@#1\endcsname
\pgfpointanchor {#1}{#2}%
\else
\sseq@tempiffalse
\fi
\fi
}
%\def\sseq@fit@tikz@calc@anchor#1.#2\tikz@stop{%
% \pgfutil@ifundefined{pgf@sh@ns@#1}{\sseq@tempiffalse}{%If the node doesn't exist, don't throw an error but record that we should skip it
% \pgfpointanchor{\tikz@pp@name{#1}}{#2}%
% }%s
%}
%%
%% Patch tikz coords
%%
\def\sseq@patchtikzcoords{
\let\sseq@tikz@scan@one@point@noshift\sseq@tikz@scan@one@point@noshift@active
\tikzoption{shift}{\sseq@tikzshift{##1}}
\let\tikz@@@parse@regular\sseq@tikz@@@parse@regular
\let\tikz@to@curve@path\sseq@tikz@to@curve@path
\let\tikz@@@to@compute@relative\sseq@tikz@@@to@compute@relative
\let\tikz@grid\sseq@tikz@grid
\let\tikz@scan@handle@options\sseq@tikz@scan@handle@options
\let\tikz@@@parse@polar\sseq@tikz@@@parse@polar
\let\tikz@scan@relative\sseq@tikz@scan@relative
}
\newif\ifsseq@draw@addoffset % control whether to offset coordinates by (\sseq@xoffset,\sseq@yoffset).
\sseq@draw@addoffsettrue
% Some of the stuff in tikzlibrarycalc will probably be broken, hopefully not too much
\let\sseq@tikz@scan@one@point@noshift\tikz@scan@one@point
\let\sseq@tikz@@@parse@regular@save\tikz@@@parse@regular
\def\sseq@tikz@scan@relative#1+{%
\global\sseq@draw@addoffsetfalse
%\global\let\tikz@@@parse@regular\sseq@tikz@@@parse@regular@save
\pgfutil@ifnextchar+{%
\tikz@scan@plusplus{\global\sseq@draw@addoffsettrue#1}%
}{%
\tikz@scan@oneplus{\global\sseq@draw@addoffsettrue#1}%
}%
}
\def\sseq@tikz@scan@one@point@noshift@active#1{%
\global\sseq@draw@addoffsetfalse
\def\sseq@scanonepoint@cmd{\global\sseq@draw@addoffsettrue#1}
\tikz@scan@one@point\sseq@scanonepoint@cmd%
}
% Probably there are more places that shouldn't have shifts inserted.
\def\sseq@tikzshift#1{\tikz@addtransform{\sseq@tikz@scan@one@point@noshift\pgftransformshift#1\relax}}
\def\sseq@tikz@to@curve@path{%
[every curve to]
\pgfextra{
%\sseq@draw@addoffsetfalse
\let\tikz@@@parse@regular\sseq@tikz@@@parse@regular@save % I added this to prevent repeated offsets from screwing us up
\let\sseq@tikz@scan@one@point@noshift\tikz@scan@one@point
\iftikz@to@relative\tikz@to@compute@relative\else\tikz@to@compute\fi
}
\tikz@computed@path
\tikztonodes%
}
\let\sseq@tikz@@@to@compute@relative\tikz@@@to@compute@relative
\patchcmd\sseq@tikz@@@to@compute@relative{%
\let\tikz@second@point=\tikz@toto
}{%
\pgf@process{\pgfpointadd{\tikz@toto}{\pgfqpointxy{\sseq@xoffset}{\sseq@yoffset}}}%
\edef\tikz@toto{\@nx\pgfpoint{\the\pgf@x}{\the\pgf@y}}%
\let\tikz@second@point=\tikz@toto
}{}{\error}
% \tikz@parse@splitxyz: we should set up an error to make this unreachable?
\let\sseq@tikz@grid\tikz@grid % line 3158
\let\sseq@tikz@scan@handle@options\tikz@scan@handle@options % 4959
\let\sseq@tikz@@@parse@polar\tikz@@@parse@polar % 5063
\def\sseq@tikz@@@parse@regular#1#2#3){%
\pgfutil@in@,{#3}%
\ifpgfutil@in@%
\tikz@parse@splitxyz{#1}{#2}#3,% Perhaps put an error here? We probably don't handle 3d coords correctly.
\else%
\tikz@checkunit{#2}%
\iftikz@isdimension%
\def\temp@xcoord{(#2)/1cm}%
\else
\def\temp@xcoord{#2}%
\fi
\tikz@checkunit{#3}%
\iftikz@isdimension
\def\temp@ycoord{(#3)/1cm}%
\else
\def\temp@ycoord{#3}%
\fi
\ifsseq@draw@addoffset
\edef\temp@xcoord{\unexpanded\@xp{\temp@xcoord}+\unexpanded\@xp{\sseq@xoffset}}%
\edef\temp@ycoord{\unexpanded\@xp{\temp@ycoord}+\unexpanded\@xp{\sseq@yoffset}}%
\fi
\edef\@next{\unexpanded{#1}{\@nx\pgfpointxy{\unexpanded\@xp{\temp@xcoord}}{\unexpanded\@xp{\temp@ycoord}}}}%
\fi%
\@next%
}
%% New shapes and arrows
%% These use lots of keys with spaces so it's convenient to turn off ExplSyntax.
% Stolen from:
https://tex.stackexchange.com/a/24621
\pgfqkeys{/pgf}{
ellipse ratio/.code={\pgfkeyssetvalue{/pgf/ellipse ratio}{#1}},
ellipse ratio/.initial=1
}
\pgfdeclareshape{newellipse}
{
\inheritsavedanchors[from=ellipse]
\inheritanchorborder[from=ellipse]
\savedanchor\radius{%
%
% Caculate ``height radius''
%
\pgf@y=.5\ht\pgfnodeparttextbox%
\advance\pgf@y by.5\dp\pgfnodeparttextbox%
\pgfmathsetlength\pgf@yb{\pgfkeysvalueof{/pgf/inner ysep}}%
\advance\pgf@y by\pgf@yb%
%
% Caculate ``width radius''
%
\pgf@x=.5\wd\pgfnodeparttextbox%
\pgfmathsetlength\pgf@xb{\pgfkeysvalueof{/pgf/inner xsep}}%
\advance\pgf@x by\pgf@xb%
%
% Adjust
%
\pgfkeysgetvalue{/pgf/ellipse ratio}{\ratioscale}
\pgfmathsetmacro\widthfactor{sqrt(\ratioscale^2+1)/\ratioscale}
\pgfmathsetmacro\heightfactor{sqrt(\ratioscale^2+1)}
\pgf@x=\widthfactor\pgf@x%
\pgf@y=\heightfactor\pgf@y%
%
% Adjust height, if necessary
%
\pgfmathsetlength\pgf@yc{\pgfkeysvalueof{/pgf/minimum height}}%
\ifdim\pgf@y<.5\pgf@yc%
\pgf@y=.5\pgf@yc%
\fi%
%
% Adjust width, if necessary
%
\pgfmathsetlength\pgf@xc{\pgfkeysvalueof{/pgf/minimum width}}%
\ifdim\pgf@x<.5\pgf@xc%
\pgf@x=.5\pgf@xc%
\fi%
%
% Add outer sep
%
\pgfmathsetlength{\pgf@xb}{\pgfkeysvalueof{/pgf/outer xsep}}%
\pgfmathsetlength{\pgf@yb}{\pgfkeysvalueof{/pgf/outer ysep}}%
\advance\pgf@x by\pgf@xb%
\advance\pgf@y by\pgf@yb%
}
\inheritanchor[from=ellipse]{center}
\inheritanchor[from=ellipse]{mid}
\inheritanchor[from=ellipse]{base}
\inheritanchor[from=ellipse]{north}
\inheritanchor[from=ellipse]{south}
\inheritanchor[from=ellipse]{west}
\inheritanchor[from=ellipse]{mid west}
\inheritanchor[from=ellipse]{base west}
\inheritanchor[from=ellipse]{north west}
\inheritanchor[from=ellipse]{south west}
\inheritanchor[from=ellipse]{east}
\inheritanchor[from=ellipse]{mid east}
\inheritanchor[from=ellipse]{base east}
\inheritanchor[from=ellipse]{north east}
\inheritanchor[from=ellipse]{south east}
\inheritbackgroundpath[from=ellipse]
}
%%
%%
%% n concentric circles
%%
\tikzset{circlen/.code={\def\circlen@n{#1}\pgfkeysalso{shape=circlen@shape}}}
\pgfdeclareshape{circlen@shape}
{
\savedanchor\centerpoint{%
\pgf@x=.5\wd\pgfnodeparttextbox%
\pgf@y=.5\ht\pgfnodeparttextbox%
\advance\pgf@y by-.5\dp\pgfnodeparttextbox%
}
\saveddimen\radius{%
%
% Caculate ``height radius''
%
\pgf@ya=.5\ht\pgfnodeparttextbox%
\advance\pgf@ya by.5\dp\pgfnodeparttextbox%
\pgfmathsetlength\pgf@yb{\pgfkeysvalueof{/pgf/inner ysep}}%
\advance\pgf@ya by\pgf@yb%
%
% Caculate ``width radius''
%
\pgf@xa=.5\wd\pgfnodeparttextbox%
\pgfmathsetlength\pgf@xb{\pgfkeysvalueof{/pgf/inner xsep}}%
\advance\pgf@xa by\pgf@xb%
%
% Calculate length of radius vector:
%
\pgf@process{\pgfpointnormalised{\pgfqpoint{\pgf@xa}{\pgf@ya}}}%
\ifdim\pgf@x>\pgf@y%
\c@pgf@counta=\pgf@x%
\ifnum\c@pgf@counta=\z@%
\else%
\divide\c@pgf@counta by 255\relax%
\pgf@xa=16\pgf@xa\relax%
\divide\pgf@xa by\c@pgf@counta%
\pgf@xa=16\pgf@xa\relax%
\fi%
\else%
\c@pgf@counta=\pgf@y%
\ifnum\c@pgf@counta=\z@%
\else%
\divide\c@pgf@counta by 255\relax%
\pgf@ya=16\pgf@ya\relax%
\divide\pgf@ya by\c@pgf@counta%
\pgf@xa=16\pgf@ya\relax%
\fi%
\fi%
\pgf@x=\pgf@xa%
%
% If necessary, adjust radius so that the size requirements are
% met:
%
\pgfmathsetlength{\pgf@xb}{\pgfkeysvalueof{/pgf/minimum width}}%
\pgfmathsetlength{\pgf@yb}{\pgfkeysvalueof{/pgf/minimum height}}%
\ifdim\pgf@x<.5\pgf@xb%
\pgf@x=.5\pgf@xb%
\fi%
\ifdim\pgf@x<.5\pgf@yb%
\pgf@x=.5\pgf@yb%
\fi%
%
% Now, add larger of outer sepearations.
%
\pgfmathsetlength{\pgf@xb}{\pgfkeysvalueof{/pgf/outer xsep}}%
\pgfmathsetlength{\pgf@yb}{\pgfkeysvalueof{/pgf/outer ysep}}%
\ifdim\pgf@xb<\pgf@yb%
\advance\pgf@x by\pgf@yb%
\else%
\advance\pgf@x by\pgf@xb%
\fi%
\pgf@xb=2pt
\multiply\pgf@xb\circlen@n
\advance\pgf@x\pgf@xb
\advance\pgf@x-2pt\relax
}
%
% Anchors
%
\anchor{center}{\centerpoint}
\anchor{mid}{\centerpoint\pgfmathsetlength\pgf@y{.5ex}}
\anchor{base}{\centerpoint\pgf@y=0pt}
\anchor{north}{\centerpoint\advance\pgf@y by\radius}
\anchor{south}{\centerpoint\advance\pgf@y by-\radius}
\anchor{west}{\centerpoint\advance\pgf@x by-\radius}
\anchor{east}{\centerpoint\advance\pgf@x by\radius}
\anchor{mid west}{\centerpoint\advance\pgf@x by-\radius\pgfmathsetlength\pgf@y{.5ex}}
\anchor{mid east}{\centerpoint\advance\pgf@x by\radius\pgfmathsetlength\pgf@y{.5ex}}
\anchor{base west}{\centerpoint\advance\pgf@x by-\radius\pgf@y=0pt}
\anchor{base east}{\centerpoint\advance\pgf@x by\radius\pgf@y=0pt}
\anchor{north west}{
\centerpoint
\pgf@xa=\radius
\advance\pgf@x by-0.707107\pgf@xa
\advance\pgf@y by0.707107\pgf@xa
}
\anchor{south west}{
\centerpoint
\pgf@xa=\radius
\advance\pgf@x by-0.707107\pgf@xa
\advance\pgf@y by-0.707107\pgf@xa
}
\anchor{north east}{
\centerpoint
\pgf@xa=\radius
\advance\pgf@x by0.707107\pgf@xa
\advance\pgf@y by0.707107\pgf@xa
}
\anchor{south east}{
\centerpoint
\pgf@xa=\radius
\advance\pgf@x by0.707107\pgf@xa
\advance\pgf@y by-0.707107\pgf@xa
}
\anchorborder{
\pgf@xa=\pgf@x%
\pgf@ya=\pgf@y%
\edef\pgf@marshal{%
\noexpand\pgfpointborderellipse
{\noexpand\pgfqpoint{\the\pgf@xa}{\the\pgf@ya}}
{\noexpand\pgfqpoint{\radius}{\radius}}%
}%
\pgf@marshal%
\pgf@xa=\pgf@x%
\pgf@ya=\pgf@y%
\centerpoint%
\advance\pgf@x by\pgf@xa%
\advance\pgf@y by\pgf@ya%
}
%
% Background path
%
\behindbackgroundpath{
\pgfutil@tempdima=\radius%
\pgfmathsetlength{\pgf@xb}{\pgfkeysvalueof{/pgf/outer xsep}}%
\pgfmathsetlength{\pgf@yb}{\pgfkeysvalueof{/pgf/outer ysep}}%
\ifdim\pgf@xb<\pgf@yb%
\advance\pgfutil@tempdima by-\pgf@yb%
\else%
\advance\pgfutil@tempdima by-\pgf@xb%
\fi%
\sseq@tempcount=\@ne
\loop
\pgfpathcircle{\centerpoint}{\pgfutil@tempdima}%
\advance\pgfutil@tempdima-2pt\relax
\advance\sseq@tempcount\@ne
\ifnum\sseq@tempcount<\circlen@n \repeat
\tikz@mode
\sseq@eval{\noexpand\pgfusepath{
\iftikz@mode@draw draw,\fi
}}
}
\backgroundpath{%
\pgfutil@tempdima=\radius%
\pgfmathsetlength{\pgf@xb}{\pgfkeysvalueof{/pgf/outer xsep}}%
\pgfmathsetlength{\pgf@yb}{\pgfkeysvalueof{/pgf/outer ysep}}%
\ifdim\pgf@xb<\pgf@yb%
\advance\pgfutil@tempdima by-\pgf@yb%
\else%
\advance\pgfutil@tempdima by-\pgf@xb%
\fi%
\advance\pgfutil@tempdima2pt\relax
\pgfutil@tempdimb=-2pt\relax
\multiply\pgfutil@tempdimb\circlen@n
\advance\pgfutil@tempdima\pgfutil@tempdimb\relax
\pgfpathcircle{\centerpoint}{\pgfutil@tempdima}
}
}
% For out of bounds edges:
\pgfdeclarearrow{
name = ...,
parameters = { \the\pgfarrowlength\the\pgflinewidth},
setup code = {
% The different end values:
\pgfarrowssetlineend{-\pgfarrowlength}
\pgfarrowssetbackend{-0.6\pgfarrowlength}
% The hull
\pgfarrowshullpoint{-\pgfarrowlength}{0pt}
\pgfarrowshullpoint{\pgfarrowlength}{0pt}
% Saves: Only the length:
\pgfarrowssavethe\pgfarrowlength
\pgfarrowssavethe\pgflinewidth
},
drawing code = {
\pgfpathcircle{\pgfpoint{-0.7\pgfarrowlength}{0pt}}{1.5\pgflinewidth}
\pgfpathcircle{\pgfpoint{-0.4\pgfarrowlength}{0pt}}{1.5\pgflinewidth}
\pgfpathcircle{\pgfpoint{-0.1\pgfarrowlength}{0pt}}{1.5\pgflinewidth}
\pgfpathclose
\pgfusepathqfill
},
defaults = { length = 0.3cm }
}