Colormaps#
The cmap.Colormap
type represents a colormap.
Also known as a LUT (look-up table), a colormap is a mapping from a scalar value to a color. Colormaps are used in a variety of contexts, including image processing, data visualization, and scientific visualization. The cmap
library provides a number of built-in colormaps, including all of the colormaps in matplotlib, napari, vispy, and more.
from cmap import Colormap
# argument can be any "ColormapLike". See rules below
cm = Colormap('viridis')
ColormapLike
objects#
The following objects can be interpreted as a colormap, and used as the first argument to the cmap.Colormap
constructor; cmap
refers to these objects collectively as "ColormapLike
". Briefly, valid arguments are of type:
- ⬇️
str
- ⬇️
Iterable[ColorLike | tuple[float, ColorLike]]
(seeColorLike
) - ⬇️
numpy.ndarray
- ⬇️
dict
- ⬇️
Callable[[ArrayLike], ArrayLike]
str
#
-
A
string
containing a recognized colormap name.Colormap('viridis')
Colormap('batlow')
Matplotlib names ✅
Any valid matplotlib colormap key that could be used in
matplotlib.colormaps[...]
is also a validcmap
colormap name. -
A string containing a recognized colormap name suffixed with
"_r"
to reverse the colormap:Colormap('viridis_r')
Colormap('batlow_r')
Iterable[ColorLike | tuple]
#
-
An
Iterable
ofColorLike
objects:Colormap(['blue', 'yellow', 'red'])
Colormap([(0, 0, 1.), "#FF0", "rgb(255, 0, 0)"])
Note
In the case of an iterable of colors, all colors are assumed to be equally spaced along the colormap.
-
An
Iterable
ofColorLike
ORtuple[float, ColorLike]
objects, where thefloat
represents the position of the color along the colormap from 0-1 (aka the "color stop"):Colormap([(0, 'blue'), (0.8, 'yellow'), (1, 'red')])
If omitted, the first and last color stops are assumed to be at 0 and 1, respectively.
Colormap(['blue', (0.8, 'yellow'), 'red'])
If the first or last color stop is not at 0 or 1, the first/last color is repeated at the 0th or 1 position, respectively.
-
Colormap([(0.4, 'blue'), (0.8, 'yellow'), 'red'])
(same as
['blue', (0.4, 'blue'), (0.8, 'yellow'), 'red']
)
If internal stops are partially provided, the missing values are assumed to be equally spaced between any provided neighboring positions (or 0 and 1, if none are provided). They are NOT placed at their global
index / (len(colors) - 1)
.-
Colormap(['blue', 'green', (0.8, 'yellow'), 'red'])
(same as
['blue', (0.4, 'green'), (0.8, 'yellow'), 'red']
)
numpy.ndarray
#
A numpy.ndarray
, in one of the following formats:
-
an
(N, 3)
array ofN
RGB colors equally spaced along the colormap:Colormap(np.array([[0, 0, 1.], [1., 1., 0], [1., 0, 0]]))
-
an
(N, 4)
array ofN
RGBA colors equally spaced along the colormap:Colormap(np.array([[0, 0, 1., 1.], [1., 1., 0, 0.7], [1., 0, 0, 0.3]]))
-
an
(N, 5)
array of color stops, where the first column is the position of the color stop and the remaining 4 columns are the RGBA colors:Colormap( np.array([ [0.0, 0.0, 0.0, 1.0, 1.0], [0.8, 1.0, 1.0, 0.0, 1.0], [1.0, 1.0, 0.0, 0.0, 1.0] ]) )
dict
#
-
A
{position -> color}
dict
of color stops, where the keys are the positions of the color stops and the values are the colors:Colormap({0: 'blue', 0.5: 'yellow', 1: 'red'})
-
A matplotlib-style
segmentdata
dict, with keys"red"
,"green"
,"blue"
, and (optionally)"alpha"
and values that are either:-
an (N, 3) array-like
[[x, y0, y1], ...]
wherex
is the color stop position, andy0
andy1
are the values of the color on either side of the stop position.cdict = {'red': [[0.0, 0.0, 0.0], [0.5, 1.0, 1.0], [1.0, 1.0, 1.0]], 'green': [[0.0, 0.0, 0.0], [0.25, 0.0, 0.0], [0.75, 1.0, 1.0], [1.0, 1.0, 1.0]], 'blue': [[0.0, 0.0, 0.0], [0.5, 0.0, 0.0], [1.0, 1.0, 1.0]]} Colormap(cdict)
-
a callable that accepts an array of 0-1 values and returns another array of 0-1 values:
Colormap({"red": lambda x: x, "green": lambda x: x**2, "blue": lambda x: x**0.5})
-
Callable
#
- A single
Callable
, object which must accept an array of values and return an(N, 3)
or(N, 4)
array of colorsColormap(lambda x: np.stack([x, np.sin(x*10), np.cos(x*10)], axis=1))
Usage#
Using qualitative colormaps#
Consider a ColorBrewer qualitative colormap from the default catalog with eight color stops
c = Colormap("colorbrewer:set1_8")
c.color_stops
As with all colormaps, the color stop positions are in [0, 1]
ColorStops(
(0.0, Color((0.8941, 0.102, 0.1098))),
(0.14285714285714285, Color((0.2157, 0.4941, 0.7216))),
(0.2857142857142857, Color((0.302, 0.6863, 0.2902))),
(0.42857142857142855, Color((0.5961, 0.3059, 0.6392))),
(0.5714285714285714, Color((1.0, 0.498, 0.0))),
(0.7142857142857142, Color((1.0, 1.0, 0.2))),
(0.8571428571428571, Color((0.651, 0.3373, 0.1569))),
(1.0, Color((0.9686, 0.5059, 0.749)))
)
so a floating point value in [0, 1] can be used to map to a color
c(0.2)
which will use nearest neighbor interpolation by default to return the second color exactly
Color((0.2157, 0.4941, 0.7216))
even though the position is in between the second and third color stops.
However, qualitative colormaps are often used to map integer valued or categorical values to colors. The behavior of calling a Colormap
depends on the type of the input. Calling a Colormap
with an integer
c(1)
indexes directly into the colormap's LUT
Color((0.2157, 0.4941, 0.7216))
which is often a more natural operation.
Cycling through colors#
When using qualitative colormaps to map integer values, sometimes the input domain may be larger than the number of colors in the colormap.
By default, values less than zero
c(-1)
map to the first color
Color((0.8941, 0.102, 0.1098))
and values greater than the number of colors
c(9)
map to the last color
Color((0.9686, 0.5059, 0.749))
This behavior can be customized by providing the under
and over
colors when initializing a Colormap
. Instead, sometimes it is preferable for the mapping to cycle through the color stops.
There is currently no built-in way to do this when calling the Colormap
, but it can be done by using the modulo operator on the input value with Colormap.num_colors
c(9 % c.num_colors)
which now maps to the first color
Color((0.8941, 0.102, 0.1098))
This also works well when using an array as input
c(np.arange(16 % c.num_colors))
which returns the cycled RGBA color values in an array output
array([[0.89411765, 0.10196078, 0.10980392, 1. ],
[0.21568627, 0.49411765, 0.72156863, 1. ],
[0.30196078, 0.68627451, 0.29019608, 1. ],
[0.59607843, 0.30588235, 0.63921569, 1. ],
[1. , 0.49803922, 0. , 1. ],
[1. , 1. , 0.2 , 1. ],
[0.65098039, 0.3372549 , 0.15686275, 1. ],
[0.96862745, 0.50588235, 0.74901961, 1. ],
[0.89411765, 0.10196078, 0.10980392, 1. ],
[0.21568627, 0.49411765, 0.72156863, 1. ],
[0.30196078, 0.68627451, 0.29019608, 1. ],
[0.59607843, 0.30588235, 0.63921569, 1. ],
[1. , 0.49803922, 0. , 1. ],
[1. , 1. , 0.2 , 1. ],
[0.65098039, 0.3372549 , 0.15686275, 1. ],
[0.96862745, 0.50588235, 0.74901961, 1. ]])
The behavior of calling a Colormap
with an array depends on its dtype
. With a floating point dtype
, it expects the values to be [0, 1], so the equivalent call to the above is
c(np.linspace(0, 2, 16, endpoint=False) % 1)
which returns the same output array values.
Immutability#
All colormaps are immutable and cannot be modified after instantiation.
Usage with external visualization libraries#
A primary motivation of cmap
is to make it easy to use colormaps in external visualization libraries. To that end, cmap.Colormap
provides to_<libname>()
methods for a number of libraries:
Tip
Some of these methods take additional arguments, see Colormap API for details.
-
Colormap("viridis").to_mpl() # or to_matplotlib()
Returns an instance of
matplotlib.colors.Colormap
. -
Colormap("viridis").to_napari()
Returns an instance of
napari.utils.colormaps.colormap.Colormap
. -
Colormap("viridis").to_vispy()
Returns an instance of
vispy.color.colormap.Colormap
. -
Colormap("viridis").to_pygfx()
Returns an instance of
pygfx.Texture
. -
Colormap("viridis").to_plotly()
Returns a list of tuples, where each tuple is a color stop:
[ [0.0, 'rgb(68, 1, 84)'], [0.00392156862745098, 'rgb(68, 2, 86)'], [0.00784313725490196, 'rgb(69, 4, 87)'], [0.011764705882352941, 'rgb(69, 5, 89)'], ...
-
Colormap("viridis").to_bokeh()
Returns an instance of
bokeh.models.mappers.LinearColorMapper
-
Colormap("viridis").to_altair()
Returns a list of hexadecimal color strings.
-
Colormap("viridis").to_pyqtgraph()
Returns an instance of
pyqtgraph.ColorMap
Usage with pydantic#
Colormap
can be used as a field type in pydantic. models.
from pydantic import BaseModel
from cmap import Colormap
class Foo(BaseModel):
colormap: Colormap
foo = Foo(colormap='viridis') # or any other ColormapLike
foo.colormap
Serialization in pydantic
Unfortunately, serialization with the json
module is not easily pluggable, so if you want to serialize a pydantic model with a Colormap
field to JSON, add the following encoder to your model:
class Foo(BaseModel):
colormap: Colormap
class Config:
json_encoders = {Colormap: Colormap.as_dict}
#example
Foo(colormap=['red', 'green']).json()
results in
{"colormap":
{"name": "custom colormap",
"identifier": "custom_colormap",
"category": null,
"color_stops": [[0.0, [1.0, 0.0, 0.0, 1]], [1.0, [0.0, 0.5019607843137255, 0.0, 1]]]
}
}
Serialization in psygnal.EventedModel
Colormap
supports serialization in psygnal.EventedModel
out of the box. The json_encoders = {Colormap: Colormap.as_dict}
line in the Config
class mentioned above is not necessary.
Rich repr#
If you use rich
pretty printing, Colormap
objects have a nice repr that shows the color in the terminal
from rich import pretty
pretty.install()