Skip to main content

OpenVoiceOS's multilingual text color parsing and formatting library

Project description

OVOS Color Parser

:warning: this package is a work in progress

What does this have to do with voice?

  • "change the lamp color to moss green"
  • "make it darker"
  • "more saturated"
  • "a bit more yellowish"
  • "perfect"

NOTE: physicists are huge nerds, so they might say something like "change the lamp wave lenght to X nanometers", this is a terrible way to talk about color and innacurate but we also added basic support for this

Extracting a color from text

The parser will do it's best to parse "color modifiers"

from ovos_color_parser import color_from_description

names = [
    "Bright, vibrant green",
    "Pale pink",
    "Muted, warm gray",
    "Dark, cool blue",

]
for n in names:
    c = color_from_description(n)
    print(c.hex_str)
    print(c)

img_7.png

Color names are ambiguous, the same name sometimes refers to multiple colors. When a color is matched by the parser it "averages all matched colors"

from ovos_color_parser import color_from_description

color = color_from_description("Red")
print(color.hex_str)  #D21B1B
print(color) 
# sRGBColor(r=210, g=27, b=27, name='Red', description='Red')

img_9.png

We can tell the parser to always return a known/named color with cast_to_palette=True, but this might not always return what you expect

from ovos_color_parser import color_from_description

color = color_from_description("Red", cast_to_palette=True)
print(color.hex_str)  #CE202B
print(color)
# sRGBColor(r=206, g=32, b=43, name='Fire engine red', description='Red')

img_8.png

Beware of impossible colors

Some colors are impossible, but that doesn't stop text from describing them

"Reddish-green" doesn’t make much sense as a description, unless you mean yellow or orange, which you don’t, because you would have said “yellow” or “orange”. The same applies to "Yellowish–blue"

the Colour of Magic or the King Colour, was the eighth colour of the Discworld spectrum. Only visible to wizards and cats. It is described in "The Colour of Magic" as the colour of imagination and is a fluorescent greenish yellow-purple. The only time non-wizards can see it is when they close their eyes; the bursts of color are octarine.

Why is this color impossible?

Fluorescent greenish-yellow and purple are essentially opposite colors on the color wheel, with wavelengths that can’t coexist in a single light wave in the visible spectrum. Here’s why:

  1. Color Wavelengths and Light: Greenish-yellow light falls in a wavelength range of about 560–590 nanometers, while purple is not a pure spectral color but a combination of blue (around 450–495 nm) and red (around 620–750 nm). Human eyes perceive purple as a combination of these two ends of the spectrum.
  2. Color Opponency Theory: The human visual system relies on color opponency, where certain pairs of colors (like red-green and blue-yellow) are processed in opposing channels. Because of this, our brains can’t interpret colors that simultaneously activate both ends of an opponent channel. This is why we don’t perceive colors like reddish-green or yellowish-blue—our brains are simply wired to cancel out those combinations.
  3. Perceptual Limits: Fluorescent colors are especially intense because they emit light in a narrow, concentrated wavelength range, making them appear very saturated and bright. Attempting to mix fluorescent greenish-yellow with purple not only challenges the physiology of the eye but would also result in a muted brown or gray tone, as the colors cancel each other out.

In short, fluorescent greenish-yellow and purple light can’t coexist in a way our eyes can interpret as a single, stable color because of the biological limits of human color perception.

from ovos_color_parser import color_from_description

# look! an impossible color
color = color_from_description("fluorescent greenish-yellow purple")
color.name = "Octarine"
print(color.hex_str) #76B11D
print(color)
# sRGBColor(r=118, g=177, b=29, name='Octarine', description='fluorescent greenish-yellow purple')

the parser will gladly output something... it just might not make sense

in this case the parser focused on "greenish-yellow"

img_10.png

but it could have focused on "purple"

img_12.png

Comparing color objects

compare color distances (smaller is better)

from ovos_color_parser import color_distance, color_from_description

color_a = color_from_description("green")
color_b = color_from_description("purple")
print(color_distance(color_a, color_b))
# 64.97192890677195

color_a = color_from_description("green")
color_b = color_from_description("yellow")
print(color_distance(color_a, color_b))
# 44.557493285361

color_a = color_from_description("yellow")
color_b = color_from_description("purple")
print(color_distance(color_a, color_b))
# 78.08287998809946

match a color object to a list of colors

from ovos_color_parser import sRGBAColor, sRGBAColorPalette, closest_color

# https://en.wikipedia.org/wiki/Blue-green
BlueGreenPalette = sRGBAColorPalette(colors=[
  sRGBAColor(r=0, g=128, b=128, name="Blue-green"),
  sRGBAColor(r=0, g=255, b=255, name="Cyan (Aqua)", description="Brilliant bluish green"),
  sRGBAColor(r=64, g=224, b=208, name="Turquoise", description="Brilliant bluish green"),
  sRGBAColor(r=17, g=100, b=180, name="Green-blue", description="Strong blue"),
  sRGBAColor(r=57, g=55, b=223, name="Bondi blue"),
  sRGBAColor(r=0, g=165, b=156, name="Blue green (Munsell)", description="Brilliant bluish green"),
  sRGBAColor(r=0, g=123, b=167, name="Cerulean", description="Strong greenish blue"),
  sRGBAColor(r=0, g=63, b=255, name="Cerulean (RGB)", description="Vivid blue"),
  sRGBAColor(r=0, g=128, b=128, name="Teal", description="Moderate bluish green"),
])

print(closest_color(sRGBAColor(r=0, g=0, b=255, name="Blue"),
                    BlueGreenPalette.colors))
# sRGBColor(r=0, g=63, b=255, name='Cerulean (RGB)', description='Vivid blue')
print(closest_color(sRGBAColor(r=0, g=255, b=0, name="Green"),
                    BlueGreenPalette.colors))
# sRGBColor(r=64, g=224, b=208, name='Turquoise', description='Brilliant bluish green')

Color Theory

Color and Language

Different languages are different names and numbers of colors! color is not universal!

For example, in many languages the colors described in English as "blue" and "green" are colexified, i.e., expressed using a single umbrella term. To render this ambiguous notion in English, linguists use the blend word grue, from green and blue

The wikipedia pages for Linguistic relativity and the color naming debate , [Blue/Green distinction](Blue–green distinction in language) and Color terms offer a good introduction to this fascinating topic

Colors in language follow a specific evolutionary pattern. This pattern is as follows:

  1. All languages contain terms for black and white.
  2. If a language contains three terms, then it contains a term for red.
  3. If a language contains four terms, then it contains a term for either green or yellow (but not both).
  4. If a language contains five terms, then it contains terms for both green and yellow.
  5. If a language contains six terms, then it contains a term for blue.
  6. If a language contains seven terms, then it contains a term for brown.
  7. If a language contains eight or more terms, then it contains terms for purple, pink, orange or gray.
Color Terms per Language

In the Bassa language, there are two terms for classifying colors; ziza (white, yellow, orange, and red) and hui ( black, violet, blue, and green)

In the Bambara language, there are three color terms: dyema (white, beige), blema (reddish, brownish), and fima ( dark green, indigo, and black).

The Ovahimba use four color names: zuzu stands for dark shades of blue, red, green, and purple; vapa is white and some shades of yellow; buru is some shades of green and blue; and dambu is some other shades of green, red, and brown.

img.png

English has 11 basic color terms: black, white, red, green, yellow, blue, brown, orange, pink, purple, and gray; other languages have between 2 and 12. All other colors are considered by most speakers of that language to be variants of these basic color terms

Italian, Russian and Hebrew have twelve basic color terms, each distinguishing blue and light blue. A Russian will make the same red/pink and orange/brown distinctions, but will also make a further distinction between синий (sinii) and голубой (goluboi), which English speakers would call dark and light blue. To Russian speakers, sinii and goluboi are as separate as red and pink, or orange and brown.

Color Spaces

RGB uses additive color mixing, because it describes what kind of light needs to be emitted to produce a given color. RGB stores individual values for red, green and blue. RGBA is RGB with an additional channel, alpha, to indicate transparency. Common color spaces based on the RGB model include sRGB, Adobe RGB, ProPhoto RGB, scRGB, and CIE RGB.

img_3.png

CMYK uses subtractive color mixing used in the printing process, because it describes what kind of inks need to be applied so the light reflected from the substrate and through the inks produces a given color. One starts with a white substrate (canvas, page, etc.), and uses ink to subtract color from white to create an image. CMYK stores ink values for cyan, magenta, yellow and black. There are many CMYK color spaces for different sets of inks, substrates, and press characteristics (which change the dot gain or transfer function for each ink and thus change the appearance).

YIQ was formerly used in NTSC (North America, Japan and elsewhere) television broadcasts for historical reasons. This system stores a luma value roughly analogous to (and sometimes incorrectly identified as)[9][10] luminance, along with two chroma values as approximate representations of the relative amounts of blue and red in the color. It is similar to the YUV scheme used in most video capture systems[11] and in PAL (Australia, Europe, except France, which uses SECAM) television, except that the YIQ color space is rotated 33° with respect to the YUV color space and the color axes are swapped. The YDbDr scheme used by SECAM television is rotated in another way.

YPbPr is a scaled version of YUV. It is most commonly seen in its digital form, YCbCr, used widely in video and image compression schemes such as MPEG and JPEG.

xvYCC is a new international digital video color space standard published by the IEC (IEC 61966-2-4). It is based on the ITU BT.601 and BT.709 standards but extends the gamut beyond the R/G/B primaries specified in those standards.

HSV (hue, saturation, value), also known as HSB (hue, saturation, brightness) is often used by artists because it is often more natural to think about a color in terms of hue and saturation than in terms of additive or subtractive color components. HSV is a transformation of an RGB color space, and its components and colorimetry are relative to the RGB color space from which it was derived.

img_2.png

HSL (hue, saturation, lightness/luminance), also known as HLS or HSI (hue, saturation, intensity) is quite similar to HSV, with "lightness" replacing "brightness". The difference is that the brightness of a pure color is equal to the brightness of white, while the lightness of a pure color is equal to the lightness of a medium gray.

Conversion Errors

A color in one absolute color space can be converted into another absolute color space, and back again, in general; however, some color spaces may have gamut limitations, and converting colors that lie outside that gamut will not produce correct results. There are also likely to be rounding errors, especially if the popular range of only 256 distinct values per component (8-bit color) is used.

One part of the definition of an absolute color space is the viewing conditions. The same color, viewed under different natural or artificial lighting conditions, will look different. Those involved professionally with color matching may use viewing rooms, lit by standardized lighting.

Occasionally, there are precise rules for converting between non-absolute color spaces. For example, HSL and HSV spaces are defined as mappings of RGB. Both are non-absolute, but the conversion between them should maintain the same color. However, in general, converting between two non-absolute color spaces (for example, RGB to CMYK) or between absolute and non-absolute color spaces (for example, RGB to Lab*) is almost a meaningless concept.

Color Names

A color term (or color name) is a word or phrase that refers to a specific color. The color term may refer to human perception of that color (which is affected by visual context), or to an underlying physical property (such as a specific wavelength of visible light).

There are also numerical systems of color specification, referred to as color spaces.

Not all colors have a name, think about a random combination of RGB values, we don't have names for every single hue!

img_1.png

Color lists

We expect all computers to represent a color term as the same Hex value, but does this happen in practice? Who names the colors?

Some standards have been proposed over the years to clearly define colors as a specific number we can represent in a computer,

  • X11 colors - In computing, on the X Window System, X11 color names are represented in a simple text file, which maps certain strings to RGB color values. It was traditionally shipped with every X11 installation, hence the name. The web colors list is descended from it but differs for certain color names.
  • Web colors standard - Web colors are colors used in displaying web pages on the World Wide Web; they can be described by way of three methods: a color may be specified as an RGB triplet, in hexadecimal format (a hex triplet) or according to its common English name in some case
  • Crayola colors - Since 1903, Crayola has created over 200 distinct colors for crayons, which often correlate to physical pigments.
  • RAL colors - Used mainly in Europe, RAL colors are a standard color matching system administered by the German organization RAL gGmbH.
  • Traditional colors of Japan - The traditional colors of Japan are a collection of colors traditionally used in Japanese art, literature, textiles such as kimono, and other Japanese arts and crafts.
  • XKCD color list - The 954 most common RGB monitor colors, as defined by several hundred thousand participants in the xkcd color name survey.

Nowadays there are many online projects that attempt to "name every color", anyone can go in there and name a hex value whatever they want, we are not considering these color names as they have no widespread adoption and are essentiallly a joke.

Color Temperature

As a block of metal heats, its emitted light changes color from red to blue, with each color corresponding to a specific temperature in Kelvin, known as the “Color Temperature.”

For colors based on black-body theory, blue occurs at higher temperatures, whereas red occurs at lower temperatures. This is the opposite of the cultural associations attributed to colors, in which "red" is "hot", and "blue" is "cold".

food for thought: Why are there no green stars?

img_14.png

Warmer colors (2700K–3000K) create a welcoming, relaxed atmosphere ideal for residential, hospitality, and lounge spaces, while cooler colors (4000K and above) provide a clean, focused environment suited for commercial, industrial, and some modern residential areas like kitchens.

img_15.png

Language support

When describing color in natural language to approximate it in RGB, there are several keywords that can convey its properties effectively

# Parse complex color descriptions
color = color_from_description("very bright, slightly warm muted blue")
Hue
  • Description: Hue refers to the basic color family, such as red, blue, green, or yellow.
  • Translation to RGB:
    • The hue determines which of the primary RGB channels (red, green, or blue) will be most prominent. For example, “red” means a strong red channel with low green and blue, while “blue” means a high blue channel with low red and green.
    • Hues like "yellow" indicate both red and green channels are high with blue low, while "purple" combines red and blue with little green.
Saturation (Intensity)
  • Description: Saturation, or chroma, is how pure or intense the color is. Terms like “vibrant,” “dull,” or “washed out” refer to saturation.
  • Translation to RGB:
    • High saturation (vibrant): Increase the difference between the dominant channel(s) and others. For example, making the red channel much higher than green and blue for a vibrant red.
    • Low saturation (dull): Reduce the contrast between channels, creating a blend closer to grayscale. For instance, balancing red, green, and blue channels to similar values lowers saturation.
Brightness (Value or Lightness)
  • Description: Brightness refers to how light or dark the color appears. Words like “bright,” “dim,” “dark,” or “pale” are often used.
  • Translation to RGB:
    • High brightness (bright): Increase the values across all channels.
    • Low brightness (dark): Decrease values across channels while maintaining the hue's relative balance.
Temperature (Warmth/Coolness)
  • Description: Color temperature reflects whether a color feels warm or cool. Terms like "warm red," "cool green," or "cold blue" apply here.
  • Translation to RGB:
    • Warm colors: Increase red or red and green channels.
    • Cool colors: Increase blue or decrease red.
Opacity/Transparency
  • Description: Opacity doesn’t affect RGB but is relevant for color perception, especially in design. Terms like “translucent,” “opaque,” or “sheer” describe it.
  • Translation to RGB:
    • Opacity affects the alpha channel (RGBA) rather than RGB values.

This approach, while interpretative, offers a structured way to translate natural language color descriptions into RGB approximations.

Color Keywords

To categorize adjectives and keywords that describe color in ways that translate into RGB or color space adjustments the parser uses a .json file per language

Example JSON structure for English color keywords:

{
  "saturation": {
    "high": ["vibrant", "rich", "bold", "deep"],
    "low": ["dull", "muted", "washed-out", "faded"]
  },
  "brightness": {
    "high": ["bright", "light", "pale", "glowing"],
    "low": ["dim", "dark", "shadowy", "faint"]
  }
}

Color name lists in each language are also used to determine the hue.

English has a word list of almost ~6000 color name mappings

Below are some examples of non-color-name keywords that define other qualities of a color

Saturation (Intensity)
  • Very High Saturation: For colors that are extremely intense or vivid.
    • Keywords: “neon,” “saturated,” “intense,” “brilliant,” “flamboyant”
  • High Saturation: These adjectives indicate vibrant or intense colors where the hue is pronounced.
    • Keywords: “vibrant,” “rich,” “bold,” “deep,” “vivid,” “intense,” “pure,” “electric”
  • Low Saturation: These adjectives imply a muted or washed-out appearance, often making the color appear closer to grayscale.
    • Keywords: “dull,” “muted,” “washed-out,” “faded,” “soft,” “pale,” “subdued,” “pastel”
  • Very Low Saturation: For colors that are very desaturated, nearing grayscale.
    • Keywords: “drab,” “grayed,” “washed-out,” “faded,” “subdued”
Brightness (Lightness/Value)
  • Very High Brightness: Extremely bright colors, often implying high lightness or near-whiteness.
    • Keywords: “blinding,” “radiant,” “glowing,” “white,” “light-filled”
  • High Brightness: Bright colors, often indicating a lighter shade or close to white.
    • Keywords: “bright,” “light,” “pale,” “glowing,” “luminous,” “brilliant,” “clear,” “radiant”
  • Low Brightness: These terms describe darker or dimmer shades, closer to black.
    • Keywords: “dim,” “dark,” “shadowy,” “faint,” “gloomy,” “subdued,” “deep,” “midnight”
  • Very Low Brightness: Colors that are nearly black or very dark.
    • Keywords: “pitch-dark,” “black,” “shadowed,” “deep,” “ink-like”
Temperature (Warmth)
  • Very High Temperature (Very Warm): Intense warm colors, strongly leaning toward red, orange, or intense yellow.
    • Keywords: “fiery,” “lava-like,” “burning,” “blazing”
  • High Temperature (Warm Colors): Warmer colors suggest a shift towards red or yellow tones, giving the color a warmer feel.
    • Keywords: “warm,” “hot,” “fiery,” “sunny,” “toasty,” “scorching,” “amber,” “reddish”
  • Low Temperature (Cool Colors): Cooler colors involve blue or green tones, giving the color a cooler or icy appearance.
    • Keywords: “cool,” “cold,” “chilly,” “icy,” “frosty,” “crisp,” “bluish,” “aqua”
  • Very Low Temperature (Very Cool): Extremely cool tones, verging on cold, icy blues or greens.
    • Keywords: “icy,” “arctic,” “frigid,” “wintry,” “glacial”
Opacity/Transparency
  • Very High Opacity: Extremely solid or dense colors.
    • Keywords: “impenetrable,” “opaque,” “thick”
  • High Opacity: Describes solid colors without transparency.
    • Keywords: “opaque,” “solid,” “dense,” “thick,” “cloudy,” “impenetrable,” “strong”
  • Low Opacity: Indicates transparency or translucency, where the background may show through.
    • Keywords: “transparent,” “translucent,” “sheer,” “see-through,” “misty,” “delicate,” “airy”
  • Very Low Opacity: Highly transparent or barely visible colors.
    • Keywords: “ethereal,” “ghostly,” “barely-there,” “translucent”

Related Projects

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

ovos-color-parser-0.0.1.tar.gz (28.6 kB view details)

Uploaded Source

Built Distribution

ovos_color_parser-0.0.1-py3-none-any.whl (231.7 kB view details)

Uploaded Python 3

File details

Details for the file ovos-color-parser-0.0.1.tar.gz.

File metadata

  • Download URL: ovos-color-parser-0.0.1.tar.gz
  • Upload date:
  • Size: 28.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.9.20

File hashes

Hashes for ovos-color-parser-0.0.1.tar.gz
Algorithm Hash digest
SHA256 f669153bc9aa0f8813cf6e5901ab5e426b249a1d891dfecb204c890e49cda7a6
MD5 39809c5738d65b178510cf74949effa3
BLAKE2b-256 a3e8154397087eb2f5e1ff3d36e8ed25756d0fc2fe9b3e47a922720b32f5f4e5

See more details on using hashes here.

File details

Details for the file ovos_color_parser-0.0.1-py3-none-any.whl.

File metadata

File hashes

Hashes for ovos_color_parser-0.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 18ba32fd11efea2ed51c9bcbcdcf7296aa9c8c4383e8296921526be5d9cef7fd
MD5 8f2066a635ff44981cbc5b55c5d5d5be
BLAKE2b-256 4fbf30522c6a13f460d023da840394a29a5d8c52114a5b8a51967102e0c97a2e

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page