はじめに
複数の条件(例:Before / After)と複数の系列(例:Group A / Group B)が交差するようなプロットにおいて、legend()
だけでは情報を整理して見せるのが難しい場合があります。特に、行列のように分類軸を視覚的に整理したい場合には、工夫が必要です。
本記事では、matplotlib
において ax.text()
などを活用し、凡例をマトリクス(行列)状に美しく表示する方法を紹介します。論文図やプレゼン資料など、見た目にもこだわりたいケースを想定しています。
本記事のコードはGoogle Colabこちらからも閲覧できます。
基本的な凡例の表示方法
まずは、label
引数と legend()
を使った基本的な凡例表示の例です。
import matplotlib.pyplot as plt
import numpy as np
# データ準備
x = np.linspace(0, 10, 100)
y1_wo = np.sin(x)
y1_w = np.sin(x) * 1.2
y2_wo = np.sin(x + 1)
y2_w = np.sin(x + 1) * 1.2
# プロット
fig, ax = plt.subplots()
ax.plot(x, y1_wo, ':', label='Condition A / w/o Corrected', color='teal')
ax.plot(x, y1_w, '-', label='Condition A / w/ Corrected', color='cyan')
ax.plot(x, y2_wo, ':', label='Condition B / w/o Corrected', color='purple')
ax.plot(x, y2_w, '-', label='Condition B / w/ Corrected', color='magenta')
# 凡例
ax.legend(loc='upper right')
ax.set_title("Basic Legend Example")
plt.tight_layout()
plt.show()
このように凡例に情報をすべて詰め込むと、やや煩雑な印象になりがちです。
行列のように整理された凡例の表示
マトリックスで凡例を表示することで視覚的にわかりやすい図を作成することができます。
方法1: 空白ラベルを活用したシンプルな方法
import matplotlib.pyplot as plt
import numpy as np
# --- プロットの準備 ---
fig, ax = plt.subplots()
# ダミーデータ
x = np.linspace(0, 10, 100)
y1_wo = np.sin(x)
y1_w = np.sin(x) * 1.2
y2_wo = np.sin(x + 1)
y2_w = np.sin(x + 1) * 1.2
# プロット
# w/o Correction(点線)の2系列(labelに空白を入れることで凡例に残す)
ax.plot(x, y1_wo, ':', color='teal', alpha=0.8, label=' ')
ax.plot(x, y2_wo, ':', color='purple', alpha=0.8, label=' ')
# w/ Correction(実線)の2系列に名前を付ける
ax.plot(x, y1_w, '-', color='cyan', alpha=0.8, label='Condition A')
ax.plot(x, y2_w, '-', color='magenta', alpha=0.8, label='Condition B')
# タイトルのように空白入りで書いてカラム名っぽく見せる
ax.legend(
ncol=2,
columnspacing=-0.2,
loc='upper right',
title="w/o w/ Corrected",
handlelength=2.5,
)
ax.set_title("Custom Grouped Legend Example")
plt.tight_layout()
plt.show()
ポイント
-
ax.plot(label=' ')
のように空白を入れること凡例に残しつつ、ラベルを表示をしない設定にすることができます。 - この方法では、
ax.legend()
に力技で、適当な空白を入れて対応する凡例の位置に揃うようにラベルを入れています。より柔軟な列の凡例を表示する方法を次の節で示します。
方法2: 空白ラベルと後入れのtext()を活用した方法
方法1では列ラベルが思うような位置に配置できない場合があります。そこで、ax.text()
で任意の位置にテキストを表示する自由度の高い方法を紹介します。
import matplotlib.pyplot as plt
import numpy as np
# --- プロットの準備 ---
fig, ax = plt.subplots()
# ダミーデータ
x = np.linspace(0, 10, 100)
y1_wo = np.sin(x)
y1_w = np.sin(x) * 1.2
y2_wo = np.sin(x + 1)
y2_w = np.sin(x + 1) * 1.2
# w/o Correction(点線)の2系列(labelに空白を入れることで凡例に残す)
ax.plot(x, y1_wo, ':', color='teal', alpha=0.8, label=' ')
ax.plot(x, y2_wo, ':', color='purple', alpha=0.8, label=' ')
# w/ Correction(実線)の2系列に名前を付ける
ax.plot(x, y1_w, '-', color='cyan', alpha=0.8, label='Condition A')
ax.plot(x, y2_w, '-', color='magenta', alpha=0.8, label='Condition B')
# 凡例表示(タイトルに余白を入れることで凡例のカラムスペースを確保)
ax.legend(
ncol=2,
columnspacing=-0.2,
loc='upper right',
title=" ",
handlelength=2.5,
)
# 凡例の上にカラム名表示(transform=ax.transAxes でグラフ描画内の0〜1で位置を範囲)
ax.text(0.69, 0.965, 'w/o w/', color='k',
ha='left', va='top', transform=ax.transAxes, zorder=10, fontsize="medium")
ax.set_title("Custom Grouped Legend Example with ax.text")
plt.tight_layout()
plt.show()
ポイント
-
legend(title=' ')
に空白を入れて列の文字が入るスペースを確保しています。 -
ax.text(transform=ax.transAxes)
のtransform
オプションを使うことで、x
およびy
の位置をグラフ描画領域内の 0〜1 の相対座標で指定できるようになります。
これを指定しない場合、テキスト位置は実際のデータ座標に基づくため、グラフのスケールを変更するたびに再調整が必要になります。そのため、筆者は 汎用性と再利用性を重視して、このtransform
オプションをよく活用しています。 - 基本的に
legend()
の文字サイズのデフォルト値はfontsize="medium"
です。
ax.text(fontsize="medium")
と指定することで、凡例とテキストの文字サイズに統一感が生まれ、全体として美しい表示が可能になります。
方法3: HandlerTuple
を使って1行に複数線をまとめる
ここではもう少し踏み込んだ方法として、HandlerTuple
を使って凡例を1行に複数の線でまとめるテクニックを紹介します。以下の記事を参考にしています。
方法1と方法2の弱点は、プロットの順番に依存して凡例が並んでしまう点です。一方この方法では、HandlerTuple
を使って グループを明示的に指定できるため、より柔軟かつ視覚的にわかりやすい凡例配置が可能になります。
とはいえ少し複雑なので、筆者は状況に応じて方法1〜3を使い分けています。
import matplotlib.pyplot as plt
from matplotlib.legend_handler import HandlerTuple
import numpy as np
# --- プロットの準備 ---
fig, ax = plt.subplots()
# ダミーデータ
x = np.linspace(0, 10, 100)
y1_wo = np.sin(x)
y1_w = np.sin(x) * 1.2
y2_wo = np.sin(x + 1)
y2_w = np.sin(x + 1) * 1.2
# プロット
# w/o Correction(点線)の2系列(labelに空白を入れることで凡例に残す)
p1_wo, = ax.plot(x, y1_wo, ':', color='teal', alpha=0.8, label='Condition A (w/o)')
p2_wo, = ax.plot(x, y2_wo, ':', color='purple', alpha=0.8, label='Condition B (w/o)')
# w/ Correction(実線)の2系列に名前を付ける
p1_w, = ax.plot(x, y1_w, '-', color='cyan', alpha=0.8, label='Condition A (w/)')
p2_w, = ax.plot(x, y2_w, '-', color='magenta', alpha=0.8, label='Condition (w/)')
# 凡例をまとめる
p1 = (p1_wo, p1_w)
p2 = (p2_wo, p2_w)
# HandlerTupleで完結に表示
ax.legend(
[p1, p2],
['Condition A', 'Condition B'],
handler_map={tuple: HandlerTuple(ndivide=None)},
loc='upper right',
title="w/o w/ Corrected",
handlelength=5
)
ax.set_title("Custom Grouped Legend Example")
plt.tight_layout()
plt.show()
やや上級ですが、1行に複数の線を表示したい場合にこの方法も活用できます。
補足:レイアウト調整Tips
ここでは、各方法で使えるオプションとその効果を整理した比較表をまとめ、実際の凡例を表示して視覚的に整理していきます。
凡例オプションの比較表(方法1〜3)
オプション | 方法1・2(標準legend) | 方法3(HandlerTuple使用) | 説明 |
---|---|---|---|
handlelength |
◯ 効く | ◯ 効く | 線の長さ(グループの横幅)を調整 |
columnspacing |
◯ 効く | × 効かない(無視される) | 列間のスペース |
HandlerTuple(pad) |
× 使用不可 | ◯ グループ内の表示スペースを調整 |
HandlerTuple(pad=1.0) など |
handletextpad |
◯ 効く | ◯ 効く | 線と文字の間隔 |
方法1・2では、columnspacing
を使って凡例内の線(列)の間隔を調整します。一方、方法3の HandlerTuple()
を使う場合は pad
オプションで調整する点が大きな違いです。
そのほかにも、オプションごとに効果の出方が異なる部分があるため、以下の凡例付き出力図に興味のある方はぜひチェックしてみてください。
出力図付き:オプションごとの効果まとめ
まとめ
本記事では、matplotlib における複数条件・複数系列の凡例をマトリクス状に整理して表示するテクニックを紹介しました。以下の3つの方法を紹介しました。
- 空白ラベル+凡例タイトルの工夫
-
ax.text()
による手動カラム名追加 -
HandlerTuple
を使った1行に2系列まとめる方法
図の見た目にこだわりたい論文・発表資料の作成において、ぜひ参考にしてみてください。
関連記事