/palette

A Rust library for linear color calculations and conversion

Primary LanguageRustApache License 2.0Apache-2.0

palette

Build Status Crates.io

A Rust library that makes linear color calculations and conversion easy and accessible for anyone. It provides both precision tools that lets you work in exactly the color space you want to, as well as a general color type that abstracts away some of the technical details.

Online Documentation

Released

Master branch.

Cargo.toml Entries

Add the following lines to your Cargo.toml file:

[dependencies]
palette = "0.4"

Features

These features are enabled by default:

  • "named" - Enables color constants, located in the named module.
  • "named_from_str" - Enables the named::from_str, which maps name string to colors. This requires the standard library.
  • "std" - Enables use of the standard library.

These features are disabled by default:

  • "serializing" - Enables color serializing and deserializing using serde.

Without the standard library

Here is an example Cargo.toml entry for using palette on #![no_std]:

[dependencies.palette]
version = "0.4"
default-features = false

It's Never "Just RGB"

Colors in, for example, images are often "gamma corrected" or stored in sRGB format as a compression method and to prevent banding. This is also a bit of a legacy from the ages of the CRT monitors, where the output from the electron gun was nonlinear. The problem is that these formats don't represent the actual intensities, and the compression has to be reverted to make sure that any operations on the colors are accurate. This library uses a completely linear work flow, and comes with the tools for transitioning between linear and non-linear RGB.

Adding to that, there are more than one kind of non-linear RGB. Ironically enough, this turns RGB into one of the most complex color spaces.

What It Can Do

Palette provides tools for both color manipulation and conversion between color spaces. These are some highlights.

Color Spaces

RGB is probably the most widely known color space, but it's not the only one. You have probably used a color picker with a rainbow wheel and a brightness slider. That may have been an HSV or an HSL color picker, where the color is encoded as hue, saturation and brightness/lightness. There's also a group of color spaces that are designed to be perceptually uniform, meaning that the perceptual change is equal to the numerical change.

Selecting the proper color space can have a big impact on how the resulting image looks (as illustrated by some of the programs in examples), and Palette makes the conversion between them as easy as a call to from or into.

This example takes an sRGB color, converts it to CIE L*C*h°, shifts its hue by 180° and converts it back to RGB:

extern crate palette;
use palette::{Srgb, LinSrgb, Lch, Hue};

let lch_color: Lch = Srgb::new(0.8, 0.2, 0.1).into();
let new_color = LinSrgb::from(lch_color.shift_hue(180.0));

This results in the following two colors:

Hue Shift Comparison

Manipulation

Palette comes with a number of color manipulation tools, that are implemented as traits. These includes lighten/darken, saturate/desaturate and hue shift. These traits are only implemented on types where they are meaningful, which means that you can't shift the hue of an RGB color without converting it to a color space where it makes sense.

This may seem limiting, but the point is to avoid inconsistent behavior due to arbitrary defaults, such as saturating a gray color to red when there is no available hue information. The abstract Color type does still support every operation, for when this is less important.

The following example shows how the Color type is used to make a lighter and a desaturated version of the original.

extern crate palette;
use palette::{Saturate, Shade, Srgb, Lch};

let color = Srgb::new(0.8, 0.2, 0.1).into_linear();
let lighter = color.lighten(0.1);
let desaturated = Lch::from(color).desaturate(0.5);

This results in the following three colors:

Manipulation Comparison

Gradients

There is also a linear gradient type which makes it easy to interpolate between a series of colors. This gradient can be used in any color space and it can be used to make color sequence iterators.

The following example shows two gradients between the same two endpoints, but one is in RGB and the other in is HSV space.

extern crate palette;
use palette::{LinSrgb, Hsv, Gradient};

let grad1 = Gradient::new(vec![
    LinSrgb::new(1.0, 0.1, 0.1),
    LinSrgb::new(0.1, 1.0, 1.0)
]);

let grad2 = Gradient::new(vec![
    Hsv::from(LinSrgb::new(1.0, 0.1, 0.1)),
    Hsv::from(LinSrgb::new(0.1, 1.0, 1.0))
]);

The RGB gradient goes through gray, while the HSV gradients changes only the hue:

Gradient Comparison

What It Isn't

This library is only meant for color manipulation and conversion. It's not a fully features image manipulation library. It will only handle colors, and not whole images. There are features that are meant to work as bridges between Palette and other graphical libraries, but the main features are limited to only focus on single pixel operations, to keep the scope at a manageable size.

Using palette in an embedded environment

Palette supports #![no_std] environments by disabling the "std" feature. However, there are some things that are unavailable without the standard library:

  • Gradients are unavailable, because they depend heavily on Vectors
  • The "named_from_str" feature requires the standard library as well
  • Serialization using serde is unavailable

It uses libm to provide the floating-point operations that are typically in std.

Contributing

All sorts of contributions are welcome, no matter how huge or tiny, so take a look at CONTRIBUTING.md for guidelines, if you are interested.

License

Licensed under either of

at your option.

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.