Обратная цветовая карта в matplotlib

254

Я хотел бы знать, как просто изменить порядок цветов данной карты цветов, чтобы использовать ее с plot_surface.

Мермос
источник

Ответы:

467

Стандартные карты цветов также имеют обратную версию. У них одинаковые имена с _rприкрепленным до конца. ( Документация здесь. )

ptomato
источник
Это не работает с "amfhot": "ValueError: Colormap amfhot_r не распознан". Я полагаю, что "hot_r" должно быть достаточно.
Shockburner
Аналогично, «ValueError: Colormap red_r не распознается».
Алекс Виллисон
19

В matplotlib карта цветов не является списком, но содержит список своих цветов как colormap.colors. И модуль matplotlib.colorsпредоставляет функцию ListedColormap()для создания цветовой карты из списка. Таким образом, вы можете повернуть любую цветовую карту, выполнив

colormap_r = ListedColormap(colormap.colors[::-1])
жилль
источник
7
+1. Тем не менее, это не приведет к обратному изменению любой цветовой карты. Только ListedColormaps (то есть дискретный, а не интерполированный) имеет colorsатрибут. Реверс LinearSegmentedColormapsнемного сложнее. (Вам нужно перевернуть каждый пункт в _segmentdataдиктанте.)
Джо Кингтон
3
Что касается реверса LinearSegmentedColormaps, я просто сделал это для некоторых цветовых карт. Вот записная книжка IPython об этом.
kwinkunks
@kwinkunks Я думаю, что функция в твоей записной книжке неправильная, см. ответ ниже
Mattijn
14

Решение довольно простое. Предположим, вы хотите использовать цветовую схему «осень». Стандартная версия:

cmap = matplotlib.cm.autumn

Чтобы изменить цветовую гамму цветовой карты, используйте функцию get_cmap () и добавьте '_r' к заголовку цветовой карты следующим образом:

cmap_reversed = matplotlib.cm.get_cmap('autumn_r')
Джм М
источник
Не могли бы вы предоставить ссылку на документацию, откуда вы взяли .autumn?
Xitcod13
Это может сломаться позже ... matplotlib.org/3.1.1/gallery/color/colormap_reference.html , но я уверен, что любой заинтересованный сможет найти это по запросу в любом случае.
Jlanger
13

Поскольку LinearSegmentedColormapsсловарь основан на словаре красного, зеленого и синего, необходимо поменять местами каждый элемент:

import matplotlib.pyplot as plt
import matplotlib as mpl
def reverse_colourmap(cmap, name = 'my_cmap_r'):
    """
    In: 
    cmap, name 
    Out:
    my_cmap_r

    Explanation:
    t[0] goes from 0 to 1
    row i:   x  y0  y1 -> t[0] t[1] t[2]
                   /
                  /
    row i+1: x  y0  y1 -> t[n] t[1] t[2]

    so the inverse should do the same:
    row i+1: x  y1  y0 -> 1-t[0] t[2] t[1]
                   /
                  /
    row i:   x  y1  y0 -> 1-t[n] t[2] t[1]
    """        
    reverse = []
    k = []   

    for key in cmap._segmentdata:    
        k.append(key)
        channel = cmap._segmentdata[key]
        data = []

        for t in channel:                    
            data.append((1-t[0],t[2],t[1]))            
        reverse.append(sorted(data))    

    LinearL = dict(zip(k,reverse))
    my_cmap_r = mpl.colors.LinearSegmentedColormap(name, LinearL) 
    return my_cmap_r

Посмотрите, что это работает:

my_cmap        
<matplotlib.colors.LinearSegmentedColormap at 0xd5a0518>

my_cmap_r = reverse_colourmap(my_cmap)

fig = plt.figure(figsize=(8, 2))
ax1 = fig.add_axes([0.05, 0.80, 0.9, 0.15])
ax2 = fig.add_axes([0.05, 0.475, 0.9, 0.15])
norm = mpl.colors.Normalize(vmin=0, vmax=1)
cb1 = mpl.colorbar.ColorbarBase(ax1, cmap = my_cmap, norm=norm,orientation='horizontal')
cb2 = mpl.colorbar.ColorbarBase(ax2, cmap = my_cmap_r, norm=norm, orientation='horizontal')

введите описание изображения здесь

РЕДАКТИРОВАТЬ


Я не получаю комментарий пользователя 3445587. Он отлично работает на карте цветов радуги:

cmap = mpl.cm.jet
cmap_r = reverse_colourmap(cmap)

fig = plt.figure(figsize=(8, 2))
ax1 = fig.add_axes([0.05, 0.80, 0.9, 0.15])
ax2 = fig.add_axes([0.05, 0.475, 0.9, 0.15])
norm = mpl.colors.Normalize(vmin=0, vmax=1)
cb1 = mpl.colorbar.ColorbarBase(ax1, cmap = cmap, norm=norm,orientation='horizontal')
cb2 = mpl.colorbar.ColorbarBase(ax2, cmap = cmap_r, norm=norm, orientation='horizontal')

введите описание изображения здесь

Но это особенно хорошо работает для пользовательских объявленных цветовых карт, так как для пользовательских объявленных цветовых карт нет значения _rпо умолчанию . Следующий пример взят из http://matplotlib.org/examples/pylab_examples/custom_cmap.html :

cdict1 = {'red':   ((0.0, 0.0, 0.0),
                   (0.5, 0.0, 0.1),
                   (1.0, 1.0, 1.0)),

         'green': ((0.0, 0.0, 0.0),
                   (1.0, 0.0, 0.0)),

         'blue':  ((0.0, 0.0, 1.0),
                   (0.5, 0.1, 0.0),
                   (1.0, 0.0, 0.0))
         }

blue_red1 = mpl.colors.LinearSegmentedColormap('BlueRed1', cdict1)
blue_red1_r = reverse_colourmap(blue_red1)

fig = plt.figure(figsize=(8, 2))
ax1 = fig.add_axes([0.05, 0.80, 0.9, 0.15])
ax2 = fig.add_axes([0.05, 0.475, 0.9, 0.15])

norm = mpl.colors.Normalize(vmin=0, vmax=1)
cb1 = mpl.colorbar.ColorbarBase(ax1, cmap = blue_red1, norm=norm,orientation='horizontal')
cb2 = mpl.colorbar.ColorbarBase(ax2, cmap = blue_red1_r, norm=norm, orientation='horizontal')

введите описание изображения здесь

Mattijn
источник
Этот пример не является полным в том смысле, что сегментные данные не должны быть в списках, поэтому они не обязательно обратимы (например, стандартная карта цветов радуги). Я думаю, что в принципе все LinearSegmentedColormaps в princible должны быть обратимыми с использованием лямбда-функции, как в карте цветов радуги?
за границей
@ user3445587 Я добавляю еще несколько примеров, но я думаю, что это прекрасно работает на стандартной
цветовой карте
Поскольку это было слишком долго, я добавил новый ответ, который должен работать для всех видов LinearSegmentData. Проблема в том, что для радуги _segmentdata реализован по-другому. Так что ваш код - по крайней мере, на моей машине - не работает с картой цветов радуги.
за рубежом
12

Начиная с Matplotlib 2.0, есть reversed()метод для ListedColormapи LinearSegmentedColorMapобъектов, так что вы можете просто сделать

cmap_reversed = cmap.reversed()

Вот документация.

Дэвид Стэнсби
источник
1

Существует два типа LinearSegmentedColormaps. В некоторых _segmentdata дается явно, например, для jet:

>>> cm.jet._segmentdata
{'blue': ((0.0, 0.5, 0.5), (0.11, 1, 1), (0.34, 1, 1), (0.65, 0, 0), (1, 0, 0)), 'red': ((0.0, 0, 0), (0.35, 0, 0), (0.66, 1, 1), (0.89, 1, 1), (1, 0.5, 0.5)), 'green': ((0.0, 0, 0), (0.125, 0, 0), (0.375, 1, 1), (0.64, 1, 1), (0.91, 0, 0), (1, 0, 0))}

Для радуги _segmentdata задается следующим образом:

>>> cm.rainbow._segmentdata
{'blue': <function <lambda> at 0x7fac32ac2b70>, 'red': <function <lambda> at 0x7fac32ac7840>, 'green': <function <lambda> at 0x7fac32ac2d08>}

Мы можем найти функции в источнике matplotlib, где они заданы как

_rainbow_data = {
        'red': gfunc[33],   # 33: lambda x: np.abs(2 * x - 0.5),
        'green': gfunc[13], # 13: lambda x: np.sin(x * np.pi),
        'blue': gfunc[10],  # 10: lambda x: np.cos(x * np.pi / 2)
}

Все, что вы хотите, уже сделано в matplotlib, просто вызовите cm.revcmap, который переворачивает оба типа сегмента данных, поэтому

cm.revcmap(cm.rainbow._segmentdata)

должен сделать работу - вы можете просто создать новую LinearSegmentData из этого. В revcmap реверсирование на основе функций SegmentData выполняется с

def _reverser(f):
    def freversed(x):
        return f(1 - x)
    return freversed

в то время как другие списки, как обычно, меняются местами

valnew = [(1.0 - x, y1, y0) for x, y0, y1 in reversed(val)] 

Так что на самом деле все, что вы хотите, это

def reverse_colourmap(cmap, name = 'my_cmap_r'):
     return mpl.colors.LinearSegmentedColormap(name, cm.revcmap(cmap._segmentdata)) 
за рубежом
источник
1

Нет встроенного способа (пока) реверсирования произвольных цветовых карт, но одно простое решение состоит в том, чтобы на самом деле не модифицировать цветную полосу, а создать инвертирующий объект Normalize:

from matplotlib.colors import Normalize

class InvertedNormalize(Normalize):
    def __call__(self, *args, **kwargs):
        return 1 - super(InvertedNormalize, self).__call__(*args, **kwargs)

Затем вы можете использовать это с plot_surfaceдругими функциями построения графиков Matplotlib, выполнив, например,

inverted_norm = InvertedNormalize(vmin=10, vmax=100)
ax.plot_surface(..., cmap=<your colormap>, norm=inverted_norm)

Это будет работать с любой картой цветов Matplotlib.

astrofrog
источник
Есть сейчас! matplotlib.org/api/_as_gen/…
Дэвид