pens

Adjust text color to be readable on light and dark backgrounds of user interfaces

By

in

Most modern user interfaces are supporting different color schemes for day and night: the so called light and dark modes. Selecting a text color for each of those modes is not a big deal and it’s the way to go when designing the user interface.

In some cases, the text color is driven by the displayed contents. In the example below, the tint color is matched to the color of the drink. The global tint color of this app is totally different, but this color adjustment gives a very nice effect. But as you might already see, there is a small problem when it comes to very light or very dark colors: each color either has a good readability on light or dark backgrounds. Some colors might fit to both, but that’s not always the case. In the example below, the light yellow is still visible, but when it comes to small icons or small text, the details are lost.

screenshot of the "mixable" app with light and dark mode

To overcome this issue, a simple solution is to select two colors for each recipe so that each mode has a different one. That’s fine, but it might totally change the effect of this colored pages.

How can we calculate a suitable color?

Some time ago, there was an article about Black or white text on a colour background? In this one, I described different algorithms to calculate the best text color (black or white) for a colored background. But now, we need the opposite: a colored text that has a good readability on white (light) or black (dark) backgrounds.

When we look at HSL and HSV/HSB color models, we already have a value for ‘lightness’ or ‘brightness’. The idea is to find a color that matches a given hue and saturation and that has a brightness which is readable on light and dark background. For this, we can use different algorithms. Very good results could be achieved with a ‘Weighted W3C Formula‘. This formula take into consideration that the human eye perceives some of the primary colors darker than others.

f(x)=r0.299+g0.587+b0.114f'(x) = r \cdot 0.299 + g \cdot 0.587 + b \cdot 0.114

Each color that is located at the border between the black and white overlay is suitable for light and dark backgrounds.

spektrum for a weighted distribution based on w3c

Adjusting the HSV color value

Step 1: convert the given color to HSV/HSB

Step 2: keep hue and saturation constant, adjust the brightness (make the color lighter or darker)

Step 3: convert the HSV/HSB value back to the required color format

Implementation in PHP

A simple calculation for a given RGB color is shown below. The classes used in this snippet are available on GitHub. The code checks the initial brightness of the color and lightens or darkens the values until the ‘border’ calculated by the ‘Weighted W3C Formula’ is reached. This is the case for the value 127, the total range of the brightness is 0 to 255.

PHP
$hsv = Convert::rgb2hsv($rgb);

$step = 0.01;
$brightness = Calculate::weightedW3C($rgb);
if ($brightness < 127) {
    while ($brightness < 127 && $hsv[2] >= 0 && $hsv[2] <= 1) {
        $hsv[2] += $step;
        $brightness = Calculate::weightedW3C(Convert::hsv2rgb($hsv));
    }
} else {
    while ($brightness > 127 && $hsv[2] >= 0 && $hsv[2] <= 1) {
        $hsv[2] -= $step;
        $brightness = Calculate::weightedW3C(Convert::hsv2rgb($hsv));
    }
}

return Convert::hsv2rgb($hsv);

Some examples

But how does this result look for different colors? Let’s start with some dark colors. Those are fine for a light background, but they become unreadable on a dark one. The top colors show the input color (before) and the color below shows the output of the calculation above (after).

colors on a light and a dark background
Color #632300 adjusted to be readable on light and dark background
colors on a light and a dark background
Color #454545 adjusted to be readable on light and dark background

And now let’s look at some light colors which are fine for dark backgrounds, but they are totally unreadable on light backgrounds.

colors on a light and a dark background
Color #73FEFF adjusted to be readable on light and dark background
colors on a light and a dark background
Color #F0C696 adjusted to be readable on light and dark background

The last color is similar to the example at the beginning and as you can see, the optimized color has a much better readability. This could be achieved for both light and dark colors.

Adjusting the HSL color value

Step 1: convert the given color to HSL

Step 2: keep hue and saturation constant, adjust the lightness (make the color lighter or darker)

Step 3: convert the HSL value back to the required color format

Implementation in PHP

A simple calculation for a given RGB color is shown below. The classes used in this snippet are also available on GitHub. The code checks the initial brightness of the color and lightens or darkens the values until the ‘border’ calculated by the ‘Weighted W3C Formula’ is reached. This is the case for the value 127, the total range of the brightness is 0 to 255.

PHP
$hsl = Convert::rgb2hsl($rgb);

$step = 0.01;
$brightness = Calculate::weightedW3C($rgb);
if ($brightness < 127) {
    while ($brightness < 127 && $hsl[2] >= 0 && $hsl[2] <= 255) {
        $hsl[2] += $step;
        $brightness = Calculate::weightedW3C(Convert::hsl2rgb($hsl));
    }
} else {
    while ($brightness > 127 && $hsl[2] >= 0 && $hsl[2] <= 255) {
        $hsl[2] -= $step;
        $brightness = Calculate::weightedW3C(Convert::hsl2rgb($hsl));
    }
}

return Convert::hsl2rgb($hsl);

Some examples

The top colors show the input color (before) and the color below shows the output of the calculation above (after). Let’s check for the same colors that have been used in the HSV code above.

HSL
Color #632300 adjusted to be readable on light and dark background
Color #454545 adjusted to be readable on light and dark background
Color #73FEFF adjusted to be readable on light and dark background
Color #F0C696 adjusted to be readable on light and dark background

Again, the optimized colors have a much better readability for both light and dark backgrounds.

Conclusion

For both solutions using the HSV or the HSL color value, the optimized colors have a much better readability. This is possible for both light and dark colors. Compared to the HSV solution, the HSL value results in some more “colorful” colors. But the readability is similar. Based on the brightness calculation using the ‘Weighted W3C Formula’, both solutions result in adjusted color values with a good readable brightness.

The code example shown above are written in PHP. An adoption should be easily possible for any other coding or scripting language.

The algorithm mentioned in this post is also available on GitHub:
https://github.com/mixable/color-utils. This package is usable with composer:

composer require mixable/color-utils

Using those classes, optimized colors can be calculated by using the following code. Since version 1.0.5, the library supports both solutions using the HSV or the HSL color value:

PHP
use Mixable\Color\Calculate;
// ...

// Calculate readable color using _HSL_ value
$optimizedColor = Calculate::readableColorForLightAndDarkBackground('#ffcc00');

// Calculate readable color using _HSV_ value
$optimizedColor = Calculate::readableColorForLightAndDarkBackgroundHsv('#ffcc00');


Comments

4 responses to “Adjust text color to be readable on light and dark backgrounds of user interfaces”

  1. I was trying to reproduce your results. Within a margin on error I think I did it, but for the last two, #73feff and #f0c696, I got different results: #00b6b7 and #c0731b, respectively. They seem to have conserved the S_L value of the originals, though, which in yours is different. Also, I used the HSL formulas from Wikipedia (with some corrections for Python) instead of HSV.

    1. Mathias Lipowski Avatar
      Mathias Lipowski

      Thank you for your comment. Great to see that you almost got the same results. So you completely implemented this in Python?

      But your results are not that bad! The colors seem to be more “colorful”. I don’t know where the differences come from. I double checked the formulas I used in this class, but I didn’t see any errors so far. I will also check the HSL solution and see if I can reproduce your results.

      1. Oops, I thought I’d get an email when you replied.
        Yes, I typed a little code in the iOS Sage Math app, which uses Python 3.
        I’ve also been checking what it does to the most saturated colors, and it seems the results for pure green (#00ff00) and magenta (#ff00ff) are still a little too bright (#00da00 and #ff25ff, respectively). Yellow (#ffff00) seems to get too dark in turn (#909000). Most other colors I’ve tried seem to do fine.

      2. Mathias Lipowski Avatar
        Mathias Lipowski

        Thank you for letting me know! I double-checked the color calculations and I could verify that the differences come from the HSL vs. the HSV color values. My examples used the HSV value – and you mentioned the HSL values. When changing to HSL, the results are similar to yours. I think the small differences that are still there come from rounding errors. I updated the post and added the solution for using the HSL value (and I also added the color images).

        So it’s no error, these are just two different solutions for this problem. The results are similar and according to the ‘Weighted W3C Formula’, the color brightness of all the optimized colors is the same (approx. 127).

        So thank you for pointing me to this. I think the HSL solution results in more “colorful” values. I like this!

Leave a Reply

Your email address will not be published. Required fields are marked *