dr.

dr.

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}

ponta de seta mal posicionada

Obtenho o comportamento esperado se remover a linha dupla, remover a bendingbiblioteca ou usar a latexponta de seta padrão em vez da Latexponta de seta de arrows.meta.

sem linha duplasem flexãoponta de flecha de látex

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@xcpara lembrar \pgf@xbe reparar \pgf@xbdepois \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 bendingbiblioteca

Sem bendingbiblioteca o xshift é feito fora do grupo onde \pgfarrows@getidestraga tudo. Ou seja, o sobrescrito \pgf@xbmorre 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 bendinga biblioteca apresenta três modos que requerem tratamentos diferentes de xshift, o xshift é feito no \ifcaseque fica no grupo onde \pgfarrows@getidestraga 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: latexum exemplo.

Para pontas de seta como Latex, \pgfarrows@getidenvolve 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 \pgfarrowslinewidthdependentfica ainda mais complexo e \pgf@xbé sobrescrito!

Responder2

Quando você carrega a bendingbiblioteca, o método padrão para desenhar pontas de seta é alterado de quickpara 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 quickpadrã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}

pontas de seta

No entanto, nenhum dos três métodos dá grandes resultados – quickapenas acontece com o melhor dos males disponíveis.

informação relacionada