Prisma (psm) is a modern C++20 library for color space conversions with
support for professional graphics and photography workflows.
Prisma supports conversion between the following color spaces, each implemented as an independent module:
- sRGB - Core reference color space
- Adobe RGB (psm::adobe_rgb) - Professional print color space
- DISPLAY-P3 (psm::display_p3) - Wide gamut color space for modern displays
- oRGB (psm::orgb) - Perceptually uniform color space with three channels:- L - Luminance
- Cyb - Chromatic Yellow-blue
- Crg - Chromatic Red-Green Adjusting these channels allows fine control over warm-cool tones in images
 
- ProPhotoRGB (psm::pro_photo_rgb) - Wide gamut color space for photography
- Accurate Color Space Conversion: All conversions maintain color accuracy through proper intermediate transformations
- Channel Adjustment (psm::adjust_channels): Allows percentage-based adjustment of individual color channels
- C++20 Compatibility: Built with modern C++20 features
- Modular Architecture: Each color space is implemented as a separate module that can be included or excluded as needed
- Simple API: Single header inclusion (#include "psm/psm.hpp") provides access to all available color spaces
- Container Flexibility: Works with any container that satisfies
std::ranges::contiguous_range
- Command-line Tool: Includes a CLI utility for image processing
- GUI Demo Tool: Interactive desktop application for real-time color space exploration
The library is compatible with various container types including:
- std::vector
- std::array
- std::span
- Custom allocator containers
- Const containers
Any container that satisfies the std::ranges::contiguous_range concept is
supported.
- Additional channels (like alpha) are not supported during conversion
The library uses exceptions for error handling:
- std::invalid_argumentis thrown if input buffer size is not a multiple of 3
- std::invalid_argumentis thrown if output buffer size doesn't match input buffer size
Follow the instructions in the INSTALL file to build and integrate Prisma into your project.
Prisma has the following dependencies:
- 
Core Library Dependency: - Eigen3 - Used for matrix operations in color space conversions
 
- 
Optional Dependencies: - GoogleTest - Used for testing only
- STB - Used for the CLI tool only
- GLFW - Used for the GUI tool only
- OpenGL - Used for the GUI tool only
- ImGui - Used for the GUI tool only
- GLEW - Used for the GUI tool only
- Native File Dialog - Used for the GUI tool only
 
When using vcpkg, these dependencies are automatically managed. The optional components (tests and CLI tool) can be disabled during the build process if you don't need them. See the INSTALL file for build configuration options.
Here's a simple example showing color space conversion from sRGB to Adobe RGB, channel adjustments, and conversion back to sRGB for display:
#include "psm/psm.hpp"  // Single header for all color space functionality
int main() {
    // Input data in BGR format
    const std::vector<unsigned char> input_image = {
        0,   0,   255,  // Red pixel (B=0, G=0, R=255)
        0,   255, 0,    // Green pixel (B=0, G=255, R=0)
        255, 0,   0     // Blue pixel (B=255, G=0, R=0)
    };
    // Create output buffer
    std::vector<unsigned char> adobe_rgb_image(input_image.size());
    std::vector<unsigned char> orgb_image(input_image.size());
    std::vector<unsigned char> output_image(input_image.size());
    // Convert between any supported color spaces
    psm::Convert<psm::sRGB, psm::AdobeRGB>(input_image, adobe_rgb_image);
    // Adjust channels in any color space (AdobeRGB in this case)
    // Adjust the individual channels
    psm::Percent adjustments{20, -10, 0};  // +20% blue, -10% green, 0% red
    psm::AdjustChannels(adobe_rgb_image, adjustments);
    // Convert through oRGB and back to sRGB
    psm::Convert<psm::AdobeRGB, psm::oRGB>(adobe_rgb_image, orgb_image);
    psm::Convert<psm::oRGB, psm::sRGB>(orgb_image, output_image);
}Below are visual examples of the same image converted to different color spaces. Note how each color space captures different color characteristics:
 Left to
right: Original sRGB, Adobe RGB, Display-P3, and ProPhoto RGB conversions
 Left to
right: Original sRGB, Adobe RGB, Display-P3, and ProPhoto RGB conversions
The gamut differences are particularly noticeable in saturated colors, where wider gamut spaces like ProPhoto RGB preserve details in highly saturated regions.
The following GIF demonstrates the effect of adjusting different channels in the oRGB color space:
 Adjusting Cyb
(yellow-blue) and Crg (red-green) channels in oRGB space
 Adjusting Cyb
(yellow-blue) and Crg (red-green) channels in oRGB space
This visualization shows how oRGB's perceptually uniform channels allow for precise control over color temperature and tonal balance without affecting luminance.
Prisma includes a command-line tool (psm_cli) for image processing, color
space conversion, and channel adjustment:
- Convert images between any supported color space
- Adjust RGB channels by percentage values
- Process and save images in JPEG format
When adjusting channels, the effect varies by color space:
- In RGB-based spaces (sRGB, AdobeRGB, DisplayP3, ProPhotoRGB), values adjust the Red, Green, and Blue channels
- In oRGB space, channel adjustments work on:
- Channel 0: Luminance - Overall brightness
- Channel 1: Cyb - Yellow-blue axis (positive values add yellow tones, negative values add blue tones)
- Channel 2: Crg - Red-green axis (positive values add red tones, negative values add green tones)
 
This makes oRGB particularly useful for adjusting the warm-cool balance of an image while preserving perceptual uniformity.
# Convert an image from sRGB to AdobeRGB
psm_cli -i input.jpg -o output.jpg -f sRGB -t AdobeRGB
# Convert an image and adjust channels (increase red by 10%, leave green unchanged, decrease blue by 5%)
psm_cli -i input.jpg -o output.jpg -f sRGB -t DisplayP3 -a 10,0,-5
# Help
psm_cli --help-i, --input FILE       Input image file
-o, --output FILE      Output image file
-f, --from COLORSPACE  Source color space (sRGB, AdobeRGB, DisplayP3, oRGB, ProPhotoRGB)
-t, --to COLORSPACE    Target color space (sRGB, AdobeRGB, DisplayP3, oRGB, ProPhotoRGB)
-a, --adjust R,G,B     Adjust channels by percent (e.g., 10,5,-5)
-h, --help             Show this help message
Prisma includes an interactive GUI demo tool (psm_gui) that provides a visual interface for exploring color space conversions and image processing:
- Real-time Image Processing: Load images and see instant color space conversions
- Interactive Controls: Dual-axis sliders with color space-specific adjustments
- Live Preview: Real-time image preview with pixel inspection and magnifying glass
- Multiple Color Spaces: Switch between sRGB, AdobeRGB, DisplayP3, and oRGB
- Pixel Inspection: Hover over any pixel to see its RGB values
- Modern Interface: Clean, responsive UI built with ImGui
Each color space provides unique slider controls optimized for its characteristics:
- sRGB: Luminance and Contrast adjustments
- AdobeRGB: Saturation and Magenta-Green balance
- DisplayP3: Contrast and Red-Orange balance
- oRGB: Chromaticity and Temperature adjustments
# Run the GUI tool
psm_guiThe GUI tool provides an intuitive way to:
- Explore different color spaces and their visual characteristics
- Experiment with channel adjustments in real-time
- Understand how color space conversions affect image appearance
- Test the PSM library's capabilities interactively
Add Prisma (psm) to your CMake project:
find_package(psm REQUIRED)
# Link core functionality
target_link_libraries(<your_target> PRIVATE psm::psm)
# Optionally link specific color space modules if you dont want the whole library
target_link_libraries(<your_target> PRIVATE
    psm::orgb          # Link oRGB support
    psm::adobe_rgb     # Link Adobe RGB support
)Note: Regardless of which modules you link against, you always include the same
header (psm/psm.hpp). The library automatically enables/disables color spaces
based on what modules were built and linked.
Prisma is licensed under the MIT License. See LICENSE for details.