
Por que o código tikz a seguir coloca a ponta da seta onde está?
\documentclass[border=5pt, tikz]{standalone}
\usetikzlibrary{arrows.meta, bending}
\begin{document}
\begin{tikzpicture}
\draw[-Latex, double] (0:0.5) arc (0:180:0.5);
\end{tikzpicture}
\end{document}
Obtenho o comportamento esperado se remover a linha dupla, remover a bending
biblioteca ou usar a latex
ponta de seta padrão em vez da Latex
ponta de seta de arrows.meta
.
De alguma forma, a combinação dessas três características parece falhar. Existe alguma maneira de fazer com que os três trabalhem juntos?
Responder1
dr.
Redefina a seguinte macro adicionando duas novas linhas:
\documentclass[border=5pt,tikz]{standalone}
\usetikzlibrary{arrows.meta,bending}
\begin{document}
\makeatletter
\def\pgf@draw@curved#1#2{%
% Prepare:
{%
\pgf@xc\pgf@xb % <--- adding new line
\pgfarrows@getid{#1}{#2}%
\pgf@xb\pgf@xc % <--- adding new line
% Do shift:
\expandafter\expandafter\expandafter\pgf@arrow@drawer@rigid@shift\csname pgf@ar@ends@\pgf@arrow@id\endcsname%
\expandafter\let\expandafter\pgf@arrow@bending@mode\csname pgf@ar@bending@mode@#1\endcsname%
\ifx\pgf@arrow@bending@mode\pgfutil@empty\let\pgf@arrow@flex@mode\pgf@arrow@mode@is@flex\fi%
% do swap:
{%
\csname pgf@ar@saves@\pgf@arrow@id\endcsname%
\ifcase\pgf@arrow@flex@mode\relax%
\expandafter\expandafter\expandafter\pgf@arrow@drawer@rigid\csname pgf@ar@visual@\pgf@arrow@id\endcsname% like flex
\or%
\expandafter\expandafter\expandafter\pgf@arrow@drawer@rigid\csname pgf@ar@visual@\pgf@arrow@id\endcsname%
\or%
\expandafter\expandafter\expandafter\pgf@arrow@drawer@rigid\csname pgf@ar@ends@\pgf@arrow@id\endcsname%
\or%
\pgf@arrow@drawer@bend%
\fi%
% hull points inside the above
}%
\expandafter}%
% Transform to next tip:
\expandafter\pgf@xb\the\pgf@xb%
}
\def\pgf@arrow@drawer@rigid@shift#1#2#3{% tip end, back end, line end, sep
% Let xa be the actual back end of the current arrow plus the back end:
\advance\pgf@xb by#2%
\pgf@xa\pgf@xb%
% Update the xb:
\pgf@x#1%
\advance\pgf@x by\pgfarrowsep%
\advance\pgf@xb by-\pgf@x%
}
\def\pgf@arrow@drawer#1#2{%
% Prepare:
{%
\pgfarrows@getid{#1}{#2}%
% Do shift:
\expandafter\expandafter\expandafter\pgf@arrow@drawer@shift\csname pgf@ar@ends@\pgf@arrow@id\endcsname%
% Do slant:
\ifdim\pgfarrows@slant pt=0pt%
\else%
\pgftransformxslant{\pgfarrows@slant}%
\fi%
% do swap:
\ifpgfarrowswap%
\pgftransformyscale{-1}%
\fi%
{%
\csname pgf@ar@saves@\pgf@arrow@id\endcsname%
\pgfscope%
\pgf@arrows@color@setup%
\pgflowlevelsynccm\csname pgf@ar@cache@\pgf@arrow@id\endcsname%
\endpgfscope%
\pgf@arrows@rigid@hull%
}%
\expandafter}%
% Transform to next tip:
\expandafter\pgftransformxshift\expandafter{\the\pgf@xc}%
}
\def\test#1{\tikz\draw[double,-{#1}](1,0)..controls(1,1)and(0,1)..(0,0);}
\test{Rectangle[length=1,width=5,black]
Rectangle[length=2,width=4,black!80]
Rectangle[length=3,width=3,black!60]
Rectangle[length=4,width=2,black!40]
Rectangle[length=10,width=1,black!20]}
\test{latex[]}
\test{Straight Barb}
\test{Hooks}
\test{Arc Barb}
\test{Tee Barb}
\test{Classical TikZ Rightarrow}
\test{Computer Modern Rightarrow}
\test{Implies}
\test{Latex}
\test{Stealth}
\test{Kite}
\test{Square}
\test{Circle}
\test{Round Cap}
\test{Butt Cap}
\test{Triangle Cap}
\test{Fast Triangle}
\test{Fast Round}
\test{Rays}
\end{document}
História mais longa
A dimensão \pgf@xb
é usada para lembrar as posições das pontas das setas. Mas, por algum motivo, ele foi substituído por algo no arquivo \pgfarrows@getid
.
Portanto, uma solução alternativa é usar \pgf@xc
para lembrar \pgf@xb
e reparar \pgf@xb
depois \pgfarrows@getid
. Pode-se usar qualquer registro de dimensão que desejar. No entanto, é melhor declarar uma nova dimensão a longo prazo.
História ainda mais longa
Por que bending
biblioteca
Sem bending
biblioteca o xshift é feito fora do grupo onde \pgfarrows@getid
estraga tudo. Ou seja, o sobrescrito \pgf@xb
morre quando o grupo termina, então o correto \pgf@xb
é usado. (veja \pgf@arrow@drawer
. Esta é a versão original de \pgf@draw@curved
.)
Porém, como bending
a biblioteca apresenta três modos que requerem tratamentos diferentes de xshift, o xshift é feito no \ifcase
que fica no grupo onde \pgfarrows@getid
estraga tudo.
Por que algumas pontas de flecha são imunes?
Algumas pontas de seta são duplicatas daquelas que usamos no modo matemático. Estas setas são independentes de PGF/TikParâmetros de Z. Resumindo: latex
um exemplo.
Para pontas de seta como Latex
, \pgfarrows@getid
envolve alguns cálculos em PGF/TikParâmetros de Z. Para ser mais específico, eles precisam ser executados \pgfarrowslinewidthdependent
.
Por que double
?
Porque quando double
é usado, o cálculo de \pgfarrowslinewidthdependent
fica ainda mais complexo e \pgf@xb
é sobrescrito!
Responder2
Quando você carrega a bending
biblioteca, o método padrão para desenhar pontas de seta é alterado de quick
para flex
. flex
é mais caro computacionalmente que quick
, mas menos caro que bend
, e geralmente dá melhores resultados. Mas não sempre. Você pode restaurar o quick
padrão especificando-o explicitamente:
\documentclass[border=5pt, tikz]{standalone}
\usetikzlibrary{arrows.meta, bending}
\begin{document}
\begin{tikzpicture}
\draw[-Latex, double] (0:0.5) arc (0:180:0.5);
\scoped[yshift=10mm]{\draw[-{Latex[quick]}, double] (0:0.5) arc (0:180:0.5);}
\end{tikzpicture}
\end{document}
No entanto, nenhum dos três métodos dá grandes resultados – quick
apenas acontece com o melhor dos males disponíveis.