Czy ktoś wie - w Matplotlib - jak wytworzyć dobrze wyglądającą wypełnioną linię konturu z półprzezroczystymi kolorami? Jeśli contourf()
przepuszcza się colormap z półprzezroczystych kolorach, produkuje małe luki pomiędzy wypełnionych obszarach:Matplotlib: wypełniony kontur z przezroczystymi kolorami
Według docs, to nie jest błąd (”contourf()
[...] nie narysuj krawędzie wielokąta "). Aby narysować krawędzie, sugeruje się "dodać kontury linii z połączeniami do contour()
". Ale to nie wygląda dobrze zarówno jako krawędzie stają się zbyt nieprzezroczyste:
Możesz poeksperymentować z linewidth
argument contour()
, ale to nie pomaga. Jakieś pomysły?
Oto kod, który reprodukuje problemu (używam API obiektowego, ale wynik jest taki sam z pyplot
):
import matplotlib
import numpy as np
from matplotlib.figure import Figure
from matplotlib.backends.backend_agg import FigureCanvasAgg
# generate some data
shape = (100, 100)
x_rng = np.linspace(-1, 1, shape[1])
y_rng = np.linspace(-1, 1, shape[0])
x, y = np.meshgrid(x_rng, y_rng)
z = np.sqrt(x**2 + y**2)
# create figure
width_inch, height_inch = 5, 5 # results in 500x500px with dpi=100
fig = Figure()
fig.set_size_inches((width_inch, height_inch))
FigureCanvasAgg(fig)
ax = fig.add_axes([0., 0., 1., 1.])
ax.set_axis_off()
# define some colors with alpha < 1
alpha = 0.9
colors = [
(0.1, 0.1, 0.5, alpha), # dark blue
(0.0, 0.7, 0.3, alpha), # green
(0.9, 0.2, 0.7, alpha), # pink
(0.0, 0.0, 0.0, alpha), # black
(0.1, 0.7, 0.7, alpha), # light blue
]
cmap = matplotlib.colors.ListedColormap(colors)
levels = np.array(np.linspace(0, z.max(), len(colors)))
norm = matplotlib.colors.BoundaryNorm(levels, ncolors=cmap.N)
# contourf plot produces small gaps between filled areas
cnt = ax.contourf(x, y, z, levels, cmap=cmap, norm=norm,
antialiased=True, linecolor='none')
# this fills the gaps, but it makes them too opaque
# ax.contour(x, y, z, levels, cmap=cmap, norm=norm,
# antialiased=True)
# the same is true for this trick:
# for c in cnt.collections:
# c.set_edgecolor("face")
filename = "/tmp/contourf.png"
fig.savefig(filename, dpi=100, transparent=True, format="png")
PS: Taka sama fabuła wygląda dobrze z backend SVG.
PPS: pcolormesh()
ma podobny problem:
ax.pcolormesh(x, y, z, cmap=cmap, norm=norm,
edgecolor="face", antialiased=True)
Uwaga: '' antialiased = True'' wydaje się mieć efekt (ale wyłączenie go również nie daje dobrego wyniku). – weatherfrog