Onda singola in linea tikzcd

Jan 10 2021

Sto cercando di creare un formato per le frecce in tikzcd che è effettivamente una tilde allungata:

Ho disegnato questo esempio usando l'opzione 1 in questa risposta , ma ho dovuto calcolare manualmente la lunghezza del segmento corretta per il serpente.

C'è un modo per calcolarlo automaticamente? Dì ad esempio:

arrows={decorate, decoration={snake, segment length=DISTANCE/2, amplitude=0.5mm}}

In caso negativo, esiste un approccio alternativo?

Qualsiasi aiuto sarebbe molto apprezzato.

Risposte

2 Dave Jan 11 2021 at 04:03

Ho una soluzione parziale basata su questa risposta .

Ho copiato la loro macro di supporto nella sua interezza, insieme alla maggior parte della prima metà della loro definizione di stile. L'unico codice di cui sono responsabile è quello che si trova sotto il commento "Disegna l'onda":

\makeatletter

% This helper macro finds the start and endpoints of a line between the source and target nodes and stores them in \sourcecoordinate and \targetcoordinate.
% #1 -- source node
% #2 -- target node
\def\findedgesourcetarget#1#2{
\let\sourcecoordinate\pgfutil@empty
\ifx\tikzcd@startanchor\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{\noexpand\pgfpointanchor{#1}{\expandafter\@gobble\tikzcd@startanchor}} % If it has an anchor, use that
    \let\sourcecoordinate\tempa
\fi
\ifx\tikzcd@endanchor\pgfutil@empty % check that the target doesn't have a specified anchor
    \def\tempb{\pgfpointshapeborder{#2}{\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{\noexpand\pgfpointanchor{#2}{\expandafter\@gobble\tikzcd@endanchor}}% If it has a specified anchor, use that
\fi
\let\targetcoordinate\tempb
\ifx\sourcecoordinate\pgfutil@empty%
    \def\sourcecoordinate{\pgfpointshapeborder{#1}{\tempb}}%
\fi
}

\tikzset{wave/.style = {
-,
to path={\pgfextra{
        \findedgesourcetarget{\tikzcd@ar@start}{\tikzcd@ar@target} % find endpoints
    % Rotate coordinate system so that line goes in x direction
    \ifx\tikzcd@startanchor\pgfutil@empty
        \def\tikzcd@startanchor{.center}
    \fi
    \ifx\tikzcd@endanchor\pgfutil@empty
        \def\tikzcd@endanchor{.center}
    \fi
    \pgfmathanglebetweenpoints{\pgfpointanchor{\tikzcd@ar@start}{\expandafter\@gobble\tikzcd@startanchor}}{\pgfpointanchor{\tikzcd@ar@target}{\expandafter\@gobble\tikzcd@endanchor}}
    \pgftransformrotate{\pgfmathresult}
    % Draw the wave
    \newdimen\xdiff
    \pgfextractx{\xdiff}{\pgfpointdiff{\sourcecoordinate}{\targetcoordinate}}
    \newdimen\ydiff
    \pgfextracty{\ydiff}{\pgfpointdiff{\sourcecoordinate}{\targetcoordinate}}
    \newdimen\finalDist
    \pgfmathparse{abs(veclen(\xdiff,\ydiff))*0.85}
    \pgfmathsetlength\finalDist{\pgfmathresult pt}

    \pgfmathsetlength\pgfdecorationsegmentlength{\finalDist}

    \pgfmathparse{0.038*\finalDist+0.6}
    \pgfmathsetlength\pgfdecorationsegmentamplitude{\pgfmathresult pt}

    \pgfsetarrows{->}

    \pgfpathmoveto{\sourcecoordinate}
    \pgfpathsnaketo{snake}{\targetcoordinate}

    \pgfusepath{stroke}
}}}}

\makeatother

Utilizzando la macro helper, recupero i punti da unire e calcolo la distanza tra loro. Quindi lo uso per definire l'ampiezza e la lunghezza del segmento del serpente.

Le costanti che uso in queste definizioni sembrano arbitrarie, le ho trovate attraverso prove e miglioramenti, ma lavorerò su un approccio più rigoroso.


Poi

\begin{center}
    Original method:
    \begin{tikzcd}[arrows={decorate, decoration={snake,segment length=7.3mm, amplitude=0.5mm}}]
        A \arrow[r,"",decorate=true] & B
    \end{tikzcd}
\end{center}
\hspace{2cm}
\begin{center}
    Custom style
    \begin{tikzcd}
        A \arrow[r, wave] & B & & \\
        A \arrow[rr, wave] & & B & \\
        A \arrow[rrr, wave] & & & B
    \end{tikzcd}
\end{center}


Il mio problema principale attualmente è che le etichette sopra le frecce non funzionano più. Proverò a risolvere questo problema e ad aggiornare questa risposta se posso.

Se qualcuno ha un approccio più semplice che evita tali problemi, fatemelo sapere!