同じ行のノードに基づいて tikz ノードの幅を設定する方法

同じ行のノードに基づいて tikz ノードの幅を設定する方法

次のような tikzpicture を作成したいと思います。

ここに画像の説明を入力してください

現時点で私が持っているのは次のものです:

\documentclass{article}
\usepackage[showframe]{geometry}
\usepackage{tikz}
\usetikzlibrary{positioning,backgrounds}
\begin{document}
\noindent\begin{tikzpicture}[background rectangle/.style={fill=gray!25}, show background rectangle,every node/.style={inner sep=0pt,outer sep=0pt,draw}]

\begin{scope}[local bounding box=scope1]
\path[use as bounding box] (0,0) rectangle (.5\textwidth,0);
\filldraw (0,0) circle (2pt); 
\node [text width=0.5\textwidth, align=left,anchor=north west](n1) {scope 1, node 1};
\node [below = 0pt of n1.south west,anchor=north west] (n2) {scope 1, node 2};
\node [below left = 0pt and 0pt of n2.north east,anchor=north west,align=left,text width=5.1cm] (n3) {scope 1, node 3 that includes longer text to be broken into lines};
\end{scope}

\begin{scope}[shift={(scope1.north east)},anchor=north west]
\node [text width=0.5\textwidth, align=left](n1) {scope 2, node 1};
\node [below = 0pt of n1.south west,anchor=north west] (n2) {scope 2, node 2};
\node [below left = 0pt and 0pt of n2.north east,anchor=north west,align=left,text width=5.1cm] (n3) {scope 2, node 3 which also includes longer text to be broken into lines};
\end{scope}

\end{tikzpicture}
\end{document}

ここに画像の説明を入力してください

次の 2 つの点について助けが必要です。

  1. ノード の幅を定義してn3、 と の合計幅が の幅 ( ) と同じになるようにするにはn2どうすれn3ばよいでしょうか。現時点では、見た目に問題がないため 5.1cm に設定していますが、ノード 2 のコンテンツを変更すると、ノード 3 の寸法も変更されるはずです。最初のスコープ用に作成した境界ボックスをどのように活用すればよいでしょうか。n1.5\textwidth

  2. 2 つのスコープの間にセパレーターを挿入するにはどうすればよいですか (最初の画像で黄色で描かれているようなもの)。2 つのスコープの幅が を超えて拡張されるのはなぜですかtextwidth。 ですが、左右に灰色の余白がありますouter sep=0pt

答え1

背景の四角形にはパディング(と呼ばれる)があるため、実際のコンテンツよりも灰色で表示されます。inner frame [xy]septight backgroundこれを変更するには、 を使用します。
さらに、線の幅 (ノードだけでなく、外側の区切りに関係なく、すべてのパスの幅) は、画像の最終的な境界ボックスに影響し、画像の最終的な配置も少しずれてしまいます (hbox がいっぱいになることもあります)。
元の設定では、 を使用して境界ボックスの水平方向の測定値を上書きできます。以下の私のコードでは、画像を原点を中心に作成するため、trim left=0pt, trim right=\textwidthを使用しています。±.5\textwidth

他のノードに関連するサイズを持つノードを作成するために、fitテキストの幅/高さ/深さを設定することで機能するライブラリがありますが、ノードに実際にテキストを持たせたい場合は適していません。

私のext.positioning-plusライブラリはfitライブラリを使用して同様のことを行いますが、 を使用して新しいノードのサイズを制御しますminimum widthminimum heightこれは、ノード内にテキストを配置するのに適していますが、ノードに収まるよりも長いテキストの場合は少し作業が必要です。

text width betweenここでは、ライブラリを使用してcalc2つの任意の点間の距離を測定する非常に基本的なキーを使用しています。バツtext width次元を計算し、それを使ってノードの面積を設定します。inner xsepここで2回減算する必要があります(最終ノードが実際に面積をカバーするようにするためです)。内部のxsep)。これは基本的に固定幅最小幅の代わりに。(係数 2 は他の形状では異なります。)

ノードのサイズを同期するには、ext.node-families図書館私のtikz-extパッケージ

設定されているすべてのノードはnode family={height=row1}同じを共有しますminimum height。これにより、各ノード内のテキストが垂直方向に中央揃えされます。

青いノードと黄色のセパレーター (これもノード) は両方とも、同じ垂直スペースにまたがります。

コード (ノード ファミリ)

\documentclass{article}
\usepackage[showframe]{geometry}
\usepackage{tikz, amsmath}
\makeatletter % https://tex.stackexchange.com/a/656319/16595
\tikzset{parse let/.code={\def\tikz@cc@stop@let in{}\tikz@let@command et #1in}}
\makeatother
\usetikzlibrary{calc, positioning, backgrounds, ext.node-families}
\tikzset{
  text width between/.style args={#1 and #2}{
    parse let={\p@=($(#2)-(#1)$)},
    text width/.expanded={abs(\x@)-2*(\noexpand\pgfkeysvalueof{/pgf/inner xsep}}}}
\begin{document}
\noindent
\begin{tikzpicture}[
  background rectangle/.style={fill=gray!25},
  show background rectangle, tight background,
  trim left=-.5\textwidth, trim right=.5\textwidth,
  every node/.style={outer sep=+0pt},
  Yellow/.style={fill=yellow, node family={height=row1}},
  Blue/.style  ={fill=blue,   node family={height=row1}, text=white},
  Red/.style   ={fill=red,   align=center},
  Green/.style ={fill=green, align=center},
  node distance=+0pt
]
\node[Yellow] (sep) {};
\node[Blue, left=of sep,  text width between={-.5\textwidth,0 and sep.west}]
  (Blue-left) {scope 1, node 1 $\displaystyle e^x = \cfrac{1}{1 - \cfrac{x}{1 + x -
                               \cfrac{\frac{1}{2}x}{1 + \frac{1}{2}x - \ddots}}}$};
\node[Blue, right=of sep, text width between={ .5\textwidth,0 and sep.east}]
  (Blue-right) {scope 1, node 1};

\node[Red, below right=of Blue-left.south  west] (Red-left)  {scope 1,\\ node 2};
\node[Red, below right=of Blue-right.south west] (Red-right) {scope 1,\\ node 2};

\node[Green, below left=of Blue-left.south east,
     text width between=Red-left.east and Blue-left.east]
  (Red-left) {scope 1, node 3 that includes longer text to be broken into lines};

\node[Green, below left=of Blue-right.south east,
     text width between=Red-right.east and Blue-right.east]
  (Red-left) {scope 2, node 3 which also includes longer text to be broken into lines
    \dots\ and here's an extra linre};
\end{tikzpicture}
\end{document}

コード (ノード ファミリなし)

\documentclass{article}
\usepackage[showframe]{geometry}
\usepackage{tikz, amsmath}
\makeatletter % https://tex.stackexchange.com/a/656319/16595
\tikzset{parse let/.code={\def\tikz@cc@stop@let in{}\tikz@let@command et #1in}}
\makeatother
\usetikzlibrary{calc, positioning, backgrounds}
\tikzset{
  text width between/.style args={#1 and #2}{
    parse let={\p@=($(#2)-(#1)$)},
    text width/.expanded={abs(\x@)-2*(\noexpand\pgfkeysvalueof{/pgf/inner xsep}}},
  minimum height of three nodes/.style n args={3}{
    parse let={\p1=($(#1.north)-(#1.south)$),
               \p2=($(#2.north)-(#2.south)$),
               \p3=($(#3.north)-(#3.south)$),
               \n@={max(\y1,\y2,\y3)}},
    minimum height/.expanded={\n@}}}
\DeclareDocumentCommand{\tikzthreenodessameheight}{O{} m O{} m O{} m}{
  \node[#1,alias=@1,overlay,path only,outer ysep=+0pt]{\phantom{#2}};
  \node[#3,alias=@2,overlay,path only,outer ysep=+0pt]{\phantom{#4}};
  \node[#5,alias=@3,overlay,path only,outer ysep=+0pt]{\phantom{#6}};
  \node[#1,minimum height of three nodes={@1}{@2}{@3}] {#2};
  \node[#3,minimum height of three nodes={@1}{@2}{@3}] {#4};
  \node[#5,minimum height of three nodes={@1}{@2}{@3}] {#6};}
\begin{document}
\noindent
\begin{tikzpicture}[
  background rectangle/.style={fill=gray!25},
  show background rectangle, tight background,
  trim left=-.5\textwidth, trim right=.5\textwidth,
  every node/.style={outer sep=+0pt},
  Yellow/.style={fill=yellow},
  Blue/.style  ={fill=blue,   text=white},
  Red/.style   ={fill=red,   align=center},
  Green/.style ={fill=green, align=center},
  node distance=+0pt
]
\tikzthreenodessameheight
  [Yellow, name=sep]{}
  [Blue, left=of sep, name=Blue-left,
   text width between={-.5\textwidth,0 and sep.west}]
  {scope 1, node 1 $\displaystyle e^x = \cfrac{1}{1 - \cfrac{x}{1 + x -
                          \cfrac{\frac{1}{2}x}{1 + \frac{1}{2}x - \ddots}}}$}
  [Blue, right=of sep, text width between={ .5\textwidth,0 and sep.east}, name=Blue-right]
  {scope 1, node 1};

\node[Red, below right=of Blue-left.south  west] (Red-left)  {scope 1,\\ node 2};
\node[Red, below right=of Blue-right.south west] (Red-right) {scope 1,\\ node 2};

\node[Green, below left=of Blue-left.south east,
     text width between=Red-left.east and Blue-left.east]
  (Red-left) {scope 1, node 3 that includes longer text to be broken into lines};

\node[Green, below left=of Blue-right.south east,
     text width between=Red-right.east and Blue-right.east]
  (Red-left) {scope 2, node 3 which also includes longer text to be broken into lines
    \dots\ and here's an extra linre};
\end{tikzpicture}
\end{document}

出力

ここに画像の説明を入力してください

答え2

デフォルトではtcbraster、(またはtcbitemize) fromtcolorbox , we can leave the package to do all the work for us. Theraster use\texwidth` を使用すると、すべてのラスター列の幅は同じになりますが、手動で調整することもできます。

\documentclass{article}
\usepackage[showframe]{geometry}
\usepackage[most]{tcolorbox}

\begin{document}

\begin{tcbitemize}[raster every box/.style={sharp corners,  
    fontupper=\sffamily, colupper=white, boxrule=0pt, halign=center},
    raster left skip=0pt, raster right skip=0pt, 
    raster before skip=0pt, raster row skip=0pt, 
    raster after skip=0pt, raster valign=top]
    \tcbitem[colback=blue!90!black] scope 1, node 1
    \tcbitem[colback=blue!90!black] scope 2, node 1
    \tcbitem[blankest, raster column skip=0pt]
        \begin{tcbitemize}[raster force size=false]
            \tcbitem[colback=red!90!black, width=.4\linewidth] scope 1, node 2
            \tcbitem[colback=green!90!black, width=.6\linewidth] scope 1, node 3 that includes longer text to be broken into lines
        \end{tcbitemize}
    \tcbitem[blankest, raster column skip=0pt]
        \begin{tcbitemize}[raster force size=false]
            \tcbitem[colback=red!90!black, width=.4\linewidth] scope 2, node 2
            \tcbitem[colback=green!90!black, width=.6\linewidth] scope 2, node 3 which also includes longer text to be broken into lines
        \end{tcbitemize}
\end{tcbitemize}
\end{document}

ここに画像の説明を入力してください

答え3

ここに画像の説明を入力してください

当初のコードに近づけるようにしました。ただし、考慮されていない問題が 1 つあります。右上のノードの高さが左上のノードの高さよりも大きい場合はどうなるかということです。解決策としては、構造を逆にして右上のノードを参照にすることが考えられます。

コード

\documentclass{article}
\usepackage[showframe]{geometry}
\usepackage{lipsum}
\usepackage{tikz}
\usetikzlibrary{calc, positioning}

\begin{document}

\lipsum[1]
\bigskip

\tikzset{%
  T/.style={blue!50!black, fill=blue!50!black, text=white,
    minimum width=0.48\textwidth, text width=0.48\textwidth-1ex
  },
  BW/.style={green!50!black, fill=green!70!black, text=black,
    minimum width=0.17\textwidth, text width=0.17\textwidth-1ex
  },
  BE/.style={red!75!black, fill=red!95!black, text=black,
    minimum width=0.30\textwidth, text width=0.30\textwidth-1ex,
  }
}
\noindent\begin{tikzpicture}[%
  every node/.style={draw, inner sep=1ex, outer sep=0pt, align=center,
  minimum height=7ex}
  ]
  \node[T, anchor=north west] at (0, 0)
  (nTW) {\bfseries \lipsum[2]};
  \node[BW, below=0pt of nTW.south west, anchor=north west]
  (n2) {scope 1, node 2};
  \node[BE, below left=0pt and 0pt of n2.north east, anchor=north west]
  (n3) {scope 1, node 3 that includes longer text to be broken into lines};

  \path ($(nTW.south) -(nTW.north)$);
  \pgfgetlastxy{\newW}{\newH}

  \node[T, right=.02\textwidth of nTW.north east,
  anchor=north west, minimum height={-\newH}]
  (nTE) {\bfseries scope 2, node 1};
  \node[BW, below=0pt of nTE.south west, anchor=north west]
  (n2E) {scope 2, node 2};
  \node[BE, below left=0pt and 0pt of n2E.north east, anchor=north west]
  (n3E) {scope 2, node 3 that includes longer text to be broken
    into lines; can be longer than the West corresponding node};

  \fill[yellow!90!red] (nTW.north east) rectangle (nTE.south west);
\end{tikzpicture}
\end{document}

答え4

代替案。私のコードは技術的ではありません (しかし、エレガントさも劣るかもしれません)。

黄色のセパレーターの幅 (\yellowGap私のコードでは ) と、ノード 1 の幅のパーセンテージ ( percentAmount、0 から 100 までの整数ですが、列が非常に狭くなるのを避けるために 0 または 100 に近すぎる値を選択しない方がよいでしょう) を調整して、ノード 2 の幅を取得できます。

最後に、ノード内の内側の分離の長さを調整できます\innerXSep

\documentclass[a4paper,10pt]{article}
\usepackage[showframe]{geometry}
\usepackage[x11names]{xcolor}
\usepackage{tikz}
\usetikzlibrary{calc}
\usepackage{calc}

\newlength{\yellowGap}
\setlength{\yellowGap}{3mm}
\newcounter{percentAmount}
\setcounter{percentAmount}{40}% =40% of the width of the first node. Value needs to be integer
\newlength{\innerXSep}
\setlength{\innerXSep}{.3333em}% default value

\newlength{\mainNodeWidth}
\setlength{\mainNodeWidth}{(\textwidth-\yellowGap)/2}
\newlength{\secondNodeWidth}
\setlength{\secondNodeWidth}{\mainNodeWidth*\value{percentAmount}/100}
\newlength{\thirdNodeWidth}
\setlength{\thirdNodeWidth}{\mainNodeWidth-\secondNodeWidth}

\begin{document}

\noindent\begin{tikzpicture}[outer sep=0pt,inner xsep=\innerXSep,text=white]
    \node[text width=\mainNodeWidth-2\innerXSep,fill=DodgerBlue3,align=center]  (mainLeft) {Scope 1, Node 1};
    \node[text width=\secondNodeWidth-2\innerXSep,fill=Firebrick2,align=center,anchor=north west] (secondLeft) at (mainLeft.south west) {Scope 1,\\Node 2};
    \node[text width=\thirdNodeWidth-2\innerXSep,fill=Green3,align=center,anchor=north west] (thirdLeft) at (secondLeft.north east) {Scope 1, Node 3\\ that includes longer text to be broken into lines};
    \fill[DarkGoldenrod1] (mainLeft.north east) rectangle ($(mainLeft.south east)+(\yellowGap,0)$);
    \node[text width=\mainNodeWidth-2\innerXSep,fill=DodgerBlue3,align=center,anchor=north west]    (mainRight) at ($(mainLeft.north east)+(\yellowGap,0)$) {Scope 2, Node 1};
    \node[text width=\secondNodeWidth-2\innerXSep,fill=Firebrick2,align=center,anchor=north west] (secondRight) at (mainRight.south west) {Scope 2,\\Node 2};
    \node[text width=\thirdNodeWidth-2\innerXSep,fill=Green3,align=center,anchor=north west] at (secondRight.north east) {Scope 2, Node 3\\ which also includes longer text to be broken into lines};
\end{tikzpicture}
\end{document}

ここに画像の説明を入力してください

関連情報